2024-07-29 07:45:59 -07:00
|
|
|
import React, { useCallback, useEffect, useState } from 'react';
|
|
|
|
|
import type { ConversationTagsResponse, TConversationTag } from 'librechat-data-provider';
|
2024-07-29 19:25:36 -04:00
|
|
|
import { Table, TableHeader, TableBody, TableRow, TableCell, Input, Button } from '~/components/ui';
|
2024-07-29 07:45:59 -07:00
|
|
|
import { BookmarkContext, useBookmarkContext } from '~/Providers/BookmarkContext';
|
|
|
|
|
import BookmarkTableRow from './BookmarkTableRow';
|
|
|
|
|
import { useLocalize } from '~/hooks';
|
|
|
|
|
|
2024-08-22 17:09:05 -04:00
|
|
|
const removeDuplicates = (bookmarks: TConversationTag[]) => {
|
|
|
|
|
const seen = new Set();
|
|
|
|
|
return bookmarks.filter((bookmark) => {
|
|
|
|
|
const duplicate = seen.has(bookmark._id);
|
|
|
|
|
seen.add(bookmark._id);
|
|
|
|
|
return !duplicate;
|
|
|
|
|
});
|
|
|
|
|
};
|
|
|
|
|
|
2024-07-29 07:45:59 -07:00
|
|
|
const BookmarkTable = () => {
|
|
|
|
|
const localize = useLocalize();
|
|
|
|
|
const [rows, setRows] = useState<ConversationTagsResponse>([]);
|
2024-07-29 19:25:36 -04:00
|
|
|
const [pageIndex, setPageIndex] = useState(0);
|
|
|
|
|
const [searchQuery, setSearchQuery] = useState('');
|
|
|
|
|
const pageSize = 10;
|
2024-07-29 07:45:59 -07:00
|
|
|
|
2024-08-22 17:09:05 -04:00
|
|
|
const { bookmarks = [] } = useBookmarkContext();
|
2024-07-29 07:45:59 -07:00
|
|
|
useEffect(() => {
|
2024-08-22 17:09:05 -04:00
|
|
|
const _bookmarks = removeDuplicates(bookmarks).sort((a, b) => a.position - b.position);
|
|
|
|
|
setRows(_bookmarks);
|
2024-07-29 07:45:59 -07:00
|
|
|
}, [bookmarks]);
|
|
|
|
|
|
|
|
|
|
const moveRow = useCallback((dragIndex: number, hoverIndex: number) => {
|
|
|
|
|
setRows((prevTags: TConversationTag[]) => {
|
|
|
|
|
const updatedRows = [...prevTags];
|
|
|
|
|
const [movedRow] = updatedRows.splice(dragIndex, 1);
|
|
|
|
|
updatedRows.splice(hoverIndex, 0, movedRow);
|
2024-08-08 21:25:10 -04:00
|
|
|
return updatedRows.map((row, index) => ({ ...row, position: index }));
|
2024-07-29 07:45:59 -07:00
|
|
|
});
|
|
|
|
|
}, []);
|
|
|
|
|
|
2024-08-08 21:25:10 -04:00
|
|
|
const renderRow = useCallback(
|
|
|
|
|
(row: TConversationTag) => {
|
2024-08-22 17:09:05 -04:00
|
|
|
return <BookmarkTableRow key={row._id} moveRow={moveRow} row={row} position={row.position} />;
|
2024-08-08 21:25:10 -04:00
|
|
|
},
|
|
|
|
|
[moveRow],
|
|
|
|
|
);
|
2024-07-29 07:45:59 -07:00
|
|
|
|
2024-08-22 17:09:05 -04:00
|
|
|
const filteredRows = rows.filter(
|
|
|
|
|
(row) => row.tag && row.tag.toLowerCase().includes(searchQuery.toLowerCase()),
|
2024-07-29 19:25:36 -04:00
|
|
|
);
|
|
|
|
|
|
|
|
|
|
const currentRows = filteredRows.slice(pageIndex * pageSize, (pageIndex + 1) * pageSize);
|
2024-07-29 07:45:59 -07:00
|
|
|
return (
|
|
|
|
|
<BookmarkContext.Provider value={{ bookmarks }}>
|
2024-12-23 11:14:40 +01:00
|
|
|
<div className=" mt-2 space-y-2">
|
|
|
|
|
<div className="flex items-center gap-4">
|
|
|
|
|
<Input
|
|
|
|
|
aria-label={localize('com_ui_bookmarks_filter')}
|
|
|
|
|
placeholder={localize('com_ui_bookmarks_filter')}
|
|
|
|
|
value={searchQuery}
|
|
|
|
|
onChange={(e) => setSearchQuery(e.target.value)}
|
|
|
|
|
/>
|
|
|
|
|
</div>
|
|
|
|
|
<div className="overflow-y-auto rounded-md border border-border-light">
|
|
|
|
|
<Table className="table-fixed border-separate border-spacing-0">
|
|
|
|
|
<TableHeader>
|
|
|
|
|
<TableRow>
|
|
|
|
|
<TableCell className="w-full bg-header-primary px-3 py-3.5 pl-6">
|
|
|
|
|
<div>{localize('com_ui_bookmarks_title')}</div>
|
|
|
|
|
</TableCell>
|
|
|
|
|
<TableCell className="w-full bg-header-primary px-3 py-3.5 sm:pl-6">
|
|
|
|
|
<div>{localize('com_ui_bookmarks_count')}</div>
|
|
|
|
|
</TableCell>
|
|
|
|
|
</TableRow>
|
|
|
|
|
</TableHeader>
|
|
|
|
|
<TableBody>{currentRows.map((row) => renderRow(row))}</TableBody>
|
|
|
|
|
</Table>
|
2024-07-29 19:25:36 -04:00
|
|
|
</div>
|
2024-12-23 11:14:40 +01:00
|
|
|
<div className="flex items-center justify-between py-4">
|
|
|
|
|
<div className="pl-1 text-text-secondary">
|
|
|
|
|
{localize('com_ui_page')} {pageIndex + 1} {localize('com_ui_of')}{' '}
|
|
|
|
|
{Math.ceil(filteredRows.length / pageSize)}
|
|
|
|
|
</div>
|
|
|
|
|
<div className="flex items-center space-x-2">
|
|
|
|
|
<Button
|
|
|
|
|
variant="outline"
|
|
|
|
|
size="sm"
|
|
|
|
|
onClick={() => setPageIndex((prev) => Math.max(prev - 1, 0))}
|
|
|
|
|
disabled={pageIndex === 0}
|
|
|
|
|
>
|
|
|
|
|
{localize('com_ui_prev')}
|
|
|
|
|
</Button>
|
|
|
|
|
<Button
|
|
|
|
|
variant="outline"
|
|
|
|
|
size="sm"
|
|
|
|
|
onClick={() =>
|
|
|
|
|
setPageIndex((prev) =>
|
|
|
|
|
(prev + 1) * pageSize < filteredRows.length ? prev + 1 : prev,
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
disabled={(pageIndex + 1) * pageSize >= filteredRows.length}
|
|
|
|
|
>
|
|
|
|
|
{localize('com_ui_next')}
|
|
|
|
|
</Button>
|
|
|
|
|
</div>
|
2024-07-29 19:25:36 -04:00
|
|
|
</div>
|
2024-07-29 07:45:59 -07:00
|
|
|
</div>
|
|
|
|
|
</BookmarkContext.Provider>
|
|
|
|
|
);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
export default BookmarkTable;
|