diff --git a/packages/web-extension/src/background/index.ts b/packages/web-extension/src/background/index.ts index 5fec1a1c9d..ed299a2007 100644 --- a/packages/web-extension/src/background/index.ts +++ b/packages/web-extension/src/background/index.ts @@ -19,7 +19,7 @@ import type { SyncData, } from '~/types'; import { isFirefox } from '~/utils'; -import { saveSession } from '~/utils/storage'; +import { addSession } from '~/utils/storage'; void (async () => { // assign default value to settings of this extension @@ -105,7 +105,7 @@ void (async () => { // ignore error })) ?? 'new session'; const newSession = generateSession(title); - await saveSession(newSession, events).catch((e) => { + await addSession(newSession, events).catch((e) => { recorderStatus.errorMessage = (e as { message: string }).message; void Browser.storage.local.set({ [LocalDataKey.recorderStatus]: recorderStatus, diff --git a/packages/web-extension/src/pages/SessionList.tsx b/packages/web-extension/src/pages/SessionList.tsx index e3ec6bd545..03c9c552ed 100644 --- a/packages/web-extension/src/pages/SessionList.tsx +++ b/packages/web-extension/src/pages/SessionList.tsx @@ -1,24 +1,29 @@ import { useEffect, useMemo, useRef, useState } from 'react'; import { nanoid } from 'nanoid'; import { + Box, + Button, chakra, + Checkbox, + Divider, + Editable, + EditableInput, + EditablePreview, + Flex, + IconButton, + Input, + Select, + Spacer, Table, - Thead, + TableContainer, Tbody, - Tr, - Th, Td, Text, - TableContainer, - Flex, - Checkbox, - Button, - Spacer, - IconButton, - Select, + Th, + Thead, + Tr, + useEditableControls, useToast, - Input, - Divider, } from '@chakra-ui/react'; import { createColumnHelper, @@ -30,6 +35,7 @@ import { type PaginationState, } from '@tanstack/react-table'; import { VscTriangleDown, VscTriangleUp } from 'react-icons/vsc'; +import { FiEdit3 as EditIcon } from 'react-icons/fi'; import { useNavigate } from 'react-router-dom'; import type { eventWithTime } from 'rrweb'; import { type Session, EventName } from '~/types'; @@ -38,7 +44,8 @@ import { deleteSessions, getAllSessions, downloadSessions, - saveSession, + addSession, + updateSession, } from '~/utils/storage'; import { FiChevronLeft, @@ -110,7 +117,58 @@ export function SessionList() { ), }), columnHelper.accessor((row) => row.name, { - cell: (info) => info.getValue(), + cell: (info) => { + const [isHovered, setIsHovered] = useState(false); + function EditableControls() { + const { isEditing, getEditButtonProps } = useEditableControls(); + return ( + isHovered && + !isEditing && ( + e.stopPropagation()} + > + } + variant="ghost" + {...getEditButtonProps()} + /> + + ) + ); + } + + return ( + setIsHovered(true)} + onMouseLeave={() => setIsHovered(false)} + alignItems="center" + position="relative" + > + { + const newSession = { ...info.row.original, name: nextValue }; + setSessions( + sessions.map((s) => + s.id === newSession.id ? newSession : s, + ), + ); + void updateSession(newSession); + }} + > + + + e.stopPropagation()} /> + + + ); + }, header: 'Name', }), columnHelper.accessor((row) => row.createTimestamp, { @@ -124,7 +182,7 @@ export function SessionList() { header: 'RRWEB Version', }), ], - [], + [sessions], ); const table = useReactTable({ columns, @@ -169,7 +227,7 @@ export function SessionList() { }; const id = nanoid(); data.session.id = id; - await saveSession(data.session, data.events); + await addSession(data.session, data.events); toast({ title: 'Session imported', description: 'The session was successfully imported.', diff --git a/packages/web-extension/src/utils/storage.ts b/packages/web-extension/src/utils/storage.ts index 766ba86fba..f18d7b6b5f 100644 --- a/packages/web-extension/src/utils/storage.ts +++ b/packages/web-extension/src/utils/storage.ts @@ -44,13 +44,25 @@ export async function getSessionStore() { }); } -export async function saveSession(session: Session, events: eventWithTime[]) { +export async function addSession(session: Session, events: eventWithTime[]) { const eventStore = await getEventStore(); await eventStore.put(EventStoreName, { id: session.id, events }); const store = await getSessionStore(); await store.add(SessionStoreName, session); } +export async function updateSession( + session: Session, + events?: eventWithTime[], +) { + const eventStore = await getEventStore(); + if (events) { + await eventStore.put(EventStoreName, { id: session.id, events }); + } + const store = await getSessionStore(); + await store.put(SessionStoreName, session); +} + export async function getSession(id: string) { const store = await getSessionStore(); return store.get(SessionStoreName, id) as Promise;