mirror of
https://github.com/MODSetter/SurfSense.git
synced 2026-06-08 20:25:19 +02:00
Merge pull request #427 from MODSetter/dev
feat: fixed chat Documents table
This commit is contained in:
commit
70808eb08b
2 changed files with 301 additions and 182 deletions
|
|
@ -25,7 +25,7 @@ import {
|
||||||
SelectValue,
|
SelectValue,
|
||||||
} from "@/components/ui/select";
|
} from "@/components/ui/select";
|
||||||
import { getConnectorIcon } from "@/contracts/enums/connectorIcons";
|
import { getConnectorIcon } from "@/contracts/enums/connectorIcons";
|
||||||
import { type Document, useDocuments } from "@/hooks/use-documents";
|
import type { Document } from "@/hooks/use-documents";
|
||||||
import { useLLMConfigs, useLLMPreferences } from "@/hooks/use-llm-configs";
|
import { useLLMConfigs, useLLMPreferences } from "@/hooks/use-llm-configs";
|
||||||
import { useSearchSourceConnectors } from "@/hooks/use-search-source-connectors";
|
import { useSearchSourceConnectors } from "@/hooks/use-search-source-connectors";
|
||||||
|
|
||||||
|
|
@ -40,20 +40,9 @@ const DocumentSelector = React.memo(
|
||||||
const { search_space_id } = useParams();
|
const { search_space_id } = useParams();
|
||||||
const [isOpen, setIsOpen] = useState(false);
|
const [isOpen, setIsOpen] = useState(false);
|
||||||
|
|
||||||
const { documents, loading, isLoaded, fetchDocuments } = useDocuments(Number(search_space_id), {
|
const handleOpenChange = useCallback((open: boolean) => {
|
||||||
lazy: true,
|
setIsOpen(open);
|
||||||
pageSize: -1, // Fetch all documents with large page size
|
}, []);
|
||||||
});
|
|
||||||
|
|
||||||
const handleOpenChange = useCallback(
|
|
||||||
(open: boolean) => {
|
|
||||||
setIsOpen(open);
|
|
||||||
if (open && !isLoaded) {
|
|
||||||
fetchDocuments();
|
|
||||||
}
|
|
||||||
},
|
|
||||||
[fetchDocuments, isLoaded]
|
|
||||||
);
|
|
||||||
|
|
||||||
const handleSelectionChange = useCallback(
|
const handleSelectionChange = useCallback(
|
||||||
(documents: Document[]) => {
|
(documents: Document[]) => {
|
||||||
|
|
@ -91,21 +80,12 @@ const DocumentSelector = React.memo(
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div className="flex-1 min-h-0 p-4 md:p-6">
|
<div className="flex-1 min-h-0 p-4 md:p-6">
|
||||||
{loading ? (
|
<DocumentsDataTable
|
||||||
<div className="flex items-center justify-center h-full">
|
searchSpaceId={Number(search_space_id)}
|
||||||
<div className="text-center space-y-2">
|
onSelectionChange={handleSelectionChange}
|
||||||
<div className="animate-spin h-8 w-8 border-2 border-primary border-t-transparent rounded-full mx-auto" />
|
onDone={handleDone}
|
||||||
<p className="text-sm text-muted-foreground">Loading documents...</p>
|
initialSelectedDocuments={selectedDocuments}
|
||||||
</div>
|
/>
|
||||||
</div>
|
|
||||||
) : isLoaded ? (
|
|
||||||
<DocumentsDataTable
|
|
||||||
documents={documents}
|
|
||||||
onSelectionChange={handleSelectionChange}
|
|
||||||
onDone={handleDone}
|
|
||||||
initialSelectedDocuments={selectedDocuments}
|
|
||||||
/>
|
|
||||||
) : null}
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</DialogContent>
|
</DialogContent>
|
||||||
|
|
|
||||||
|
|
@ -2,21 +2,18 @@
|
||||||
|
|
||||||
import {
|
import {
|
||||||
type ColumnDef,
|
type ColumnDef,
|
||||||
type ColumnFiltersState,
|
|
||||||
flexRender,
|
flexRender,
|
||||||
getCoreRowModel,
|
getCoreRowModel,
|
||||||
getFilteredRowModel,
|
|
||||||
getPaginationRowModel,
|
|
||||||
getSortedRowModel,
|
|
||||||
type SortingState,
|
type SortingState,
|
||||||
useReactTable,
|
useReactTable,
|
||||||
type VisibilityState,
|
|
||||||
} from "@tanstack/react-table";
|
} from "@tanstack/react-table";
|
||||||
import { ArrowUpDown, Calendar, FileText, Search } from "lucide-react";
|
import { ArrowUpDown, Calendar, FileText, Filter, Search } from "lucide-react";
|
||||||
import { useEffect, useMemo, useState } from "react";
|
import { useCallback, useEffect, useMemo, useRef, useState } from "react";
|
||||||
import { Button } from "@/components/ui/button";
|
import { Button } from "@/components/ui/button";
|
||||||
import { Checkbox } from "@/components/ui/checkbox";
|
import { Checkbox } from "@/components/ui/checkbox";
|
||||||
import { Input } from "@/components/ui/input";
|
import { Input } from "@/components/ui/input";
|
||||||
|
import { Label } from "@/components/ui/label";
|
||||||
|
import { Popover, PopoverContent, PopoverTrigger } from "@/components/ui/popover";
|
||||||
import {
|
import {
|
||||||
Select,
|
Select,
|
||||||
SelectContent,
|
SelectContent,
|
||||||
|
|
@ -32,26 +29,24 @@ import {
|
||||||
TableHeader,
|
TableHeader,
|
||||||
TableRow,
|
TableRow,
|
||||||
} from "@/components/ui/table";
|
} from "@/components/ui/table";
|
||||||
import { EnumConnectorName } from "@/contracts/enums/connector";
|
|
||||||
import { getConnectorIcon } from "@/contracts/enums/connectorIcons";
|
import { getConnectorIcon } from "@/contracts/enums/connectorIcons";
|
||||||
import type { Document, DocumentType } from "@/hooks/use-documents";
|
import { type Document, type DocumentType, useDocuments } from "@/hooks/use-documents";
|
||||||
|
|
||||||
interface DocumentsDataTableProps {
|
interface DocumentsDataTableProps {
|
||||||
documents: Document[];
|
searchSpaceId: number;
|
||||||
onSelectionChange: (documents: Document[]) => void;
|
onSelectionChange: (documents: Document[]) => void;
|
||||||
onDone: () => void;
|
onDone: () => void;
|
||||||
initialSelectedDocuments?: Document[];
|
initialSelectedDocuments?: Document[];
|
||||||
}
|
}
|
||||||
|
|
||||||
// Combine EnumConnectorName with additional document types
|
function useDebounced<T>(value: T, delay = 300) {
|
||||||
const DOCUMENT_TYPES: (string | "ALL")[] = [
|
const [debounced, setDebounced] = useState(value);
|
||||||
"ALL",
|
useEffect(() => {
|
||||||
"FILE",
|
const t = setTimeout(() => setDebounced(value), delay);
|
||||||
"EXTENSION",
|
return () => clearTimeout(t);
|
||||||
"CRAWLED_URL",
|
}, [value, delay]);
|
||||||
"YOUTUBE_VIDEO",
|
return debounced;
|
||||||
...Object.values(EnumConnectorName),
|
}
|
||||||
];
|
|
||||||
|
|
||||||
const columns: ColumnDef<Document>[] = [
|
const columns: ColumnDef<Document>[] = [
|
||||||
{
|
{
|
||||||
|
|
@ -177,93 +172,193 @@ const columns: ColumnDef<Document>[] = [
|
||||||
];
|
];
|
||||||
|
|
||||||
export function DocumentsDataTable({
|
export function DocumentsDataTable({
|
||||||
documents,
|
searchSpaceId,
|
||||||
onSelectionChange,
|
onSelectionChange,
|
||||||
onDone,
|
onDone,
|
||||||
initialSelectedDocuments = [],
|
initialSelectedDocuments = [],
|
||||||
}: DocumentsDataTableProps) {
|
}: DocumentsDataTableProps) {
|
||||||
const [sorting, setSorting] = useState<SortingState>([]);
|
const [sorting, setSorting] = useState<SortingState>([]);
|
||||||
const [columnFilters, setColumnFilters] = useState<ColumnFiltersState>([]);
|
const [search, setSearch] = useState("");
|
||||||
const [columnVisibility, setColumnVisibility] = useState<VisibilityState>({});
|
const debouncedSearch = useDebounced(search, 300);
|
||||||
const [documentTypeFilter, setDocumentTypeFilter] = useState<string | "ALL">("ALL");
|
const [documentTypeFilter, setDocumentTypeFilter] = useState<string[]>([]);
|
||||||
|
const [pageIndex, setPageIndex] = useState(0);
|
||||||
|
const [pageSize, setPageSize] = useState(10);
|
||||||
|
const [typeCounts, setTypeCounts] = useState<Record<string, number>>({});
|
||||||
|
|
||||||
|
// Use server-side pagination, search, and filtering
|
||||||
|
const { documents, total, loading, fetchDocuments, searchDocuments, getDocumentTypeCounts } =
|
||||||
|
useDocuments(searchSpaceId, {
|
||||||
|
page: pageIndex,
|
||||||
|
pageSize: pageSize,
|
||||||
|
});
|
||||||
|
|
||||||
|
// Fetch document type counts on mount
|
||||||
|
useEffect(() => {
|
||||||
|
if (searchSpaceId && getDocumentTypeCounts) {
|
||||||
|
getDocumentTypeCounts().then(setTypeCounts);
|
||||||
|
}
|
||||||
|
}, [searchSpaceId, getDocumentTypeCounts]);
|
||||||
|
|
||||||
|
// Refetch when pagination changes or when search/filters change
|
||||||
|
useEffect(() => {
|
||||||
|
if (searchSpaceId) {
|
||||||
|
if (debouncedSearch.trim()) {
|
||||||
|
searchDocuments?.(
|
||||||
|
debouncedSearch,
|
||||||
|
pageIndex,
|
||||||
|
pageSize,
|
||||||
|
documentTypeFilter.length > 0 ? documentTypeFilter : undefined
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
fetchDocuments?.(
|
||||||
|
pageIndex,
|
||||||
|
pageSize,
|
||||||
|
documentTypeFilter.length > 0 ? documentTypeFilter : undefined
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, [
|
||||||
|
pageIndex,
|
||||||
|
pageSize,
|
||||||
|
debouncedSearch,
|
||||||
|
documentTypeFilter,
|
||||||
|
searchSpaceId,
|
||||||
|
fetchDocuments,
|
||||||
|
searchDocuments,
|
||||||
|
]);
|
||||||
|
|
||||||
// Memoize initial row selection to prevent infinite loops
|
// Memoize initial row selection to prevent infinite loops
|
||||||
const initialRowSelection = useMemo(() => {
|
const initialRowSelection = useMemo(() => {
|
||||||
if (!documents.length || !initialSelectedDocuments.length) return {};
|
if (!initialSelectedDocuments.length) return {};
|
||||||
|
|
||||||
const selection: Record<string, boolean> = {};
|
const selection: Record<string, boolean> = {};
|
||||||
initialSelectedDocuments.forEach((selectedDoc) => {
|
initialSelectedDocuments.forEach((selectedDoc) => {
|
||||||
selection[selectedDoc.id] = true;
|
selection[selectedDoc.id] = true;
|
||||||
});
|
});
|
||||||
return selection;
|
return selection;
|
||||||
}, [documents, initialSelectedDocuments]);
|
}, [initialSelectedDocuments]);
|
||||||
|
|
||||||
const [rowSelection, setRowSelection] = useState<Record<string, boolean>>({});
|
const [rowSelection, setRowSelection] = useState<Record<string, boolean>>(
|
||||||
|
() => initialRowSelection
|
||||||
|
);
|
||||||
|
|
||||||
// Only update row selection when initialRowSelection actually changes and is not empty
|
// Maintain a separate state for actually selected documents (across all pages)
|
||||||
|
const [selectedDocumentsMap, setSelectedDocumentsMap] = useState<Map<number, Document>>(() => {
|
||||||
|
const map = new Map<number, Document>();
|
||||||
|
initialSelectedDocuments.forEach((doc) => map.set(doc.id, doc));
|
||||||
|
return map;
|
||||||
|
});
|
||||||
|
|
||||||
|
// Track the last notified selection to avoid redundant parent calls
|
||||||
|
const lastNotifiedSelection = useRef<string>("");
|
||||||
|
|
||||||
|
// Update row selection only when initialSelectedDocuments changes (not rowSelection itself)
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const hasChanges = JSON.stringify(rowSelection) !== JSON.stringify(initialRowSelection);
|
const initialKeys = Object.keys(initialRowSelection);
|
||||||
if (hasChanges && Object.keys(initialRowSelection).length > 0) {
|
if (initialKeys.length === 0) return;
|
||||||
setRowSelection(initialRowSelection);
|
|
||||||
}
|
|
||||||
}, [initialRowSelection]);
|
|
||||||
|
|
||||||
// Initialize row selection on mount
|
const currentKeys = Object.keys(rowSelection);
|
||||||
|
// Quick length check before expensive comparison
|
||||||
|
if (currentKeys.length === initialKeys.length) {
|
||||||
|
// Check if all keys match (order doesn't matter for Sets)
|
||||||
|
const hasAllKeys = initialKeys.every((key) => rowSelection[key]);
|
||||||
|
if (hasAllKeys) return;
|
||||||
|
}
|
||||||
|
|
||||||
|
setRowSelection(initialRowSelection);
|
||||||
|
}, [initialRowSelection]); // Remove rowSelection from dependencies to prevent loop
|
||||||
|
|
||||||
|
// Update the selected documents map when row selection changes
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (Object.keys(rowSelection).length === 0 && Object.keys(initialRowSelection).length > 0) {
|
if (!documents || documents.length === 0) return;
|
||||||
setRowSelection(initialRowSelection);
|
|
||||||
}
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
const filteredDocuments = useMemo(() => {
|
setSelectedDocumentsMap((prev) => {
|
||||||
if (documentTypeFilter === "ALL") return documents;
|
const newMap = new Map(prev);
|
||||||
return documents.filter((doc) => doc.document_type === documentTypeFilter);
|
let hasChanges = false;
|
||||||
}, [documents, documentTypeFilter]);
|
|
||||||
|
// Process only current page documents
|
||||||
|
for (const doc of documents) {
|
||||||
|
const docId = doc.id;
|
||||||
|
const isSelected = rowSelection[docId.toString()];
|
||||||
|
const wasInMap = newMap.has(docId);
|
||||||
|
|
||||||
|
if (isSelected && !wasInMap) {
|
||||||
|
newMap.set(docId, doc);
|
||||||
|
hasChanges = true;
|
||||||
|
} else if (!isSelected && wasInMap) {
|
||||||
|
newMap.delete(docId);
|
||||||
|
hasChanges = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Return same reference if no changes to avoid unnecessary re-renders
|
||||||
|
return hasChanges ? newMap : prev;
|
||||||
|
});
|
||||||
|
}, [rowSelection, documents]);
|
||||||
|
|
||||||
|
// Memoize selected documents array
|
||||||
|
const selectedDocumentsArray = useMemo(() => {
|
||||||
|
return Array.from(selectedDocumentsMap.values());
|
||||||
|
}, [selectedDocumentsMap]);
|
||||||
|
|
||||||
|
// Notify parent of selection changes only when content actually changes
|
||||||
|
useEffect(() => {
|
||||||
|
// Create a stable string representation for comparison
|
||||||
|
const selectionKey = selectedDocumentsArray
|
||||||
|
.map((d) => d.id)
|
||||||
|
.sort()
|
||||||
|
.join(",");
|
||||||
|
|
||||||
|
// Skip if selection hasn't actually changed
|
||||||
|
if (selectionKey === lastNotifiedSelection.current) return;
|
||||||
|
|
||||||
|
lastNotifiedSelection.current = selectionKey;
|
||||||
|
onSelectionChange(selectedDocumentsArray);
|
||||||
|
}, [selectedDocumentsArray, onSelectionChange]);
|
||||||
|
|
||||||
const table = useReactTable({
|
const table = useReactTable({
|
||||||
data: filteredDocuments,
|
data: documents || [],
|
||||||
columns,
|
columns,
|
||||||
getRowId: (row) => row.id.toString(),
|
getRowId: (row) => row.id.toString(),
|
||||||
onSortingChange: setSorting,
|
onSortingChange: setSorting,
|
||||||
onColumnFiltersChange: setColumnFilters,
|
|
||||||
getCoreRowModel: getCoreRowModel(),
|
getCoreRowModel: getCoreRowModel(),
|
||||||
getPaginationRowModel: getPaginationRowModel(),
|
|
||||||
getSortedRowModel: getSortedRowModel(),
|
|
||||||
getFilteredRowModel: getFilteredRowModel(),
|
|
||||||
onColumnVisibilityChange: setColumnVisibility,
|
|
||||||
onRowSelectionChange: setRowSelection,
|
onRowSelectionChange: setRowSelection,
|
||||||
initialState: { pagination: { pageSize: 10 } },
|
manualPagination: true,
|
||||||
state: { sorting, columnFilters, columnVisibility, rowSelection },
|
pageCount: Math.ceil(total / pageSize),
|
||||||
|
state: { sorting, rowSelection, pagination: { pageIndex, pageSize } },
|
||||||
});
|
});
|
||||||
|
|
||||||
useEffect(() => {
|
const handleClearAll = useCallback(() => {
|
||||||
const selectedRows = table.getFilteredSelectedRowModel().rows;
|
setRowSelection({});
|
||||||
const selectedDocuments = selectedRows.map((row) => row.original);
|
setSelectedDocumentsMap(new Map());
|
||||||
onSelectionChange(selectedDocuments);
|
}, []);
|
||||||
}, [rowSelection, onSelectionChange, table]);
|
|
||||||
|
|
||||||
const handleClearAll = () => setRowSelection({});
|
const handleSelectPage = useCallback(() => {
|
||||||
|
|
||||||
const handleSelectPage = () => {
|
|
||||||
const currentPageRows = table.getRowModel().rows;
|
const currentPageRows = table.getRowModel().rows;
|
||||||
const newSelection = { ...rowSelection };
|
const newSelection = { ...rowSelection };
|
||||||
currentPageRows.forEach((row) => {
|
currentPageRows.forEach((row) => {
|
||||||
newSelection[row.id] = true;
|
newSelection[row.id] = true;
|
||||||
});
|
});
|
||||||
setRowSelection(newSelection);
|
setRowSelection(newSelection);
|
||||||
};
|
}, [table, rowSelection]);
|
||||||
|
|
||||||
const handleSelectAllFiltered = () => {
|
const handleToggleType = useCallback((type: string, checked: boolean) => {
|
||||||
const allFilteredRows = table.getFilteredRowModel().rows;
|
setDocumentTypeFilter((prev) => {
|
||||||
const newSelection: Record<string, boolean> = {};
|
if (checked) {
|
||||||
allFilteredRows.forEach((row) => {
|
return [...prev, type];
|
||||||
newSelection[row.id] = true;
|
}
|
||||||
|
return prev.filter((t) => t !== type);
|
||||||
});
|
});
|
||||||
setRowSelection(newSelection);
|
setPageIndex(0); // Reset to first page when filter changes
|
||||||
};
|
}, []);
|
||||||
|
|
||||||
const selectedCount = table.getFilteredSelectedRowModel().rows.length;
|
const selectedCount = selectedDocumentsMap.size;
|
||||||
const totalFiltered = table.getFilteredRowModel().rows.length;
|
|
||||||
|
// Get available document types from type counts (memoized)
|
||||||
|
const availableTypes = useMemo(() => {
|
||||||
|
const types = Object.keys(typeCounts);
|
||||||
|
return types.length > 0 ? types.sort() : [];
|
||||||
|
}, [typeCounts]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="flex flex-col h-full space-y-3 md:space-y-4">
|
<div className="flex flex-col h-full space-y-3 md:space-y-4">
|
||||||
|
|
@ -275,33 +370,70 @@ export function DocumentsDataTable({
|
||||||
<Search className="absolute left-3 top-1/2 transform -translate-y-1/2 h-4 w-4 text-muted-foreground" />
|
<Search className="absolute left-3 top-1/2 transform -translate-y-1/2 h-4 w-4 text-muted-foreground" />
|
||||||
<Input
|
<Input
|
||||||
placeholder="Search documents..."
|
placeholder="Search documents..."
|
||||||
value={(table.getColumn("title")?.getFilterValue() as string) ?? ""}
|
value={search}
|
||||||
onChange={(event) => table.getColumn("title")?.setFilterValue(event.target.value)}
|
onChange={(event) => {
|
||||||
|
setSearch(event.target.value);
|
||||||
|
setPageIndex(0); // Reset to first page on search
|
||||||
|
}}
|
||||||
className="pl-10 text-sm"
|
className="pl-10 text-sm"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
<Select
|
<Popover>
|
||||||
value={documentTypeFilter}
|
<PopoverTrigger asChild>
|
||||||
onValueChange={(value) => setDocumentTypeFilter(value as string | "ALL")}
|
<Button variant="outline" className="w-full sm:w-auto">
|
||||||
>
|
<Filter className="mr-2 h-4 w-4 opacity-60" />
|
||||||
<SelectTrigger className="w-full sm:w-[180px]">
|
Type
|
||||||
<SelectValue />
|
{documentTypeFilter.length > 0 && (
|
||||||
</SelectTrigger>
|
<span className="ml-2 inline-flex h-5 items-center rounded border border-border bg-background px-1.5 text-[0.625rem] font-medium text-muted-foreground/70">
|
||||||
<SelectContent>
|
{documentTypeFilter.length}
|
||||||
{DOCUMENT_TYPES.map((type) => (
|
</span>
|
||||||
<SelectItem key={type} value={type}>
|
)}
|
||||||
{type === "ALL" ? "All Types" : type.replace(/_/g, " ")}
|
</Button>
|
||||||
</SelectItem>
|
</PopoverTrigger>
|
||||||
))}
|
<PopoverContent className="w-64 p-3" align="start">
|
||||||
</SelectContent>
|
<div className="space-y-3">
|
||||||
</Select>
|
<div className="text-xs font-medium text-muted-foreground">Filter by Type</div>
|
||||||
|
<div className="space-y-2 max-h-[300px] overflow-y-auto">
|
||||||
|
{availableTypes.map((type) => (
|
||||||
|
<div key={type} className="flex items-center gap-2">
|
||||||
|
<Checkbox
|
||||||
|
id={`type-${type}`}
|
||||||
|
checked={documentTypeFilter.includes(type)}
|
||||||
|
onCheckedChange={(checked) => handleToggleType(type, !!checked)}
|
||||||
|
/>
|
||||||
|
<Label
|
||||||
|
htmlFor={`type-${type}`}
|
||||||
|
className="flex grow justify-between gap-2 font-normal text-sm cursor-pointer"
|
||||||
|
>
|
||||||
|
<span>{type.replace(/_/g, " ")}</span>
|
||||||
|
<span className="text-xs text-muted-foreground">{typeCounts[type]}</span>
|
||||||
|
</Label>
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
{documentTypeFilter.length > 0 && (
|
||||||
|
<Button
|
||||||
|
variant="ghost"
|
||||||
|
size="sm"
|
||||||
|
className="w-full text-xs"
|
||||||
|
onClick={() => {
|
||||||
|
setDocumentTypeFilter([]);
|
||||||
|
setPageIndex(0);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Clear Filters
|
||||||
|
</Button>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</PopoverContent>
|
||||||
|
</Popover>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* Action Controls Row */}
|
{/* Action Controls Row */}
|
||||||
<div className="flex flex-col sm:flex-row sm:items-center sm:justify-between gap-3">
|
<div className="flex flex-col sm:flex-row sm:items-center sm:justify-between gap-3">
|
||||||
<div className="flex flex-col sm:flex-row sm:items-center gap-2">
|
<div className="flex flex-col sm:flex-row sm:items-center gap-2">
|
||||||
<span className="text-sm text-muted-foreground whitespace-nowrap">
|
<span className="text-sm text-muted-foreground whitespace-nowrap">
|
||||||
{selectedCount} of {totalFiltered} selected
|
{selectedCount} selected {loading && "· Loading..."}
|
||||||
</span>
|
</span>
|
||||||
<div className="hidden sm:block h-4 w-px bg-border mx-2" />
|
<div className="hidden sm:block h-4 w-px bg-border mx-2" />
|
||||||
<div className="flex items-center gap-2 flex-wrap">
|
<div className="flex items-center gap-2 flex-wrap">
|
||||||
|
|
@ -319,25 +451,28 @@ export function DocumentsDataTable({
|
||||||
size="sm"
|
size="sm"
|
||||||
onClick={handleSelectPage}
|
onClick={handleSelectPage}
|
||||||
className="text-xs sm:text-sm"
|
className="text-xs sm:text-sm"
|
||||||
|
disabled={loading}
|
||||||
>
|
>
|
||||||
Select Page
|
Select Page
|
||||||
</Button>
|
</Button>
|
||||||
<Button
|
<Select
|
||||||
variant="ghost"
|
value={pageSize.toString()}
|
||||||
size="sm"
|
onValueChange={(v) => {
|
||||||
onClick={handleSelectAllFiltered}
|
setPageSize(Number(v));
|
||||||
className="text-xs sm:text-sm hidden sm:inline-flex"
|
setPageIndex(0);
|
||||||
|
}}
|
||||||
>
|
>
|
||||||
Select All Filtered
|
<SelectTrigger className="w-[100px] h-8 text-xs">
|
||||||
</Button>
|
<SelectValue>{pageSize} per page</SelectValue>
|
||||||
<Button
|
</SelectTrigger>
|
||||||
variant="ghost"
|
<SelectContent>
|
||||||
size="sm"
|
{[10, 25, 50, 100].map((size) => (
|
||||||
onClick={handleSelectAllFiltered}
|
<SelectItem key={size} value={size.toString()}>
|
||||||
className="text-xs sm:hidden"
|
{size} per page
|
||||||
>
|
</SelectItem>
|
||||||
Select All
|
))}
|
||||||
</Button>
|
</SelectContent>
|
||||||
|
</Select>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<Button
|
<Button
|
||||||
|
|
@ -353,82 +488,86 @@ export function DocumentsDataTable({
|
||||||
{/* Table Container */}
|
{/* Table Container */}
|
||||||
<div className="border rounded-lg flex-1 min-h-0 overflow-hidden bg-background">
|
<div className="border rounded-lg flex-1 min-h-0 overflow-hidden bg-background">
|
||||||
<div className="overflow-auto h-full">
|
<div className="overflow-auto h-full">
|
||||||
<Table>
|
{loading ? (
|
||||||
<TableHeader className="sticky top-0 bg-background/95 backdrop-blur supports-[backdrop-filter]:bg-background/60 z-10">
|
<div className="flex items-center justify-center h-full">
|
||||||
{table.getHeaderGroups().map((headerGroup) => (
|
<div className="text-center space-y-2">
|
||||||
<TableRow key={headerGroup.id} className="border-b">
|
<div className="animate-spin h-8 w-8 border-2 border-primary border-t-transparent rounded-full mx-auto" />
|
||||||
{headerGroup.headers.map((header) => (
|
<p className="text-sm text-muted-foreground">Loading documents...</p>
|
||||||
<TableHead key={header.id} className="h-12 text-xs sm:text-sm">
|
</div>
|
||||||
{header.isPlaceholder
|
</div>
|
||||||
? null
|
) : (
|
||||||
: flexRender(header.column.columnDef.header, header.getContext())}
|
<Table>
|
||||||
</TableHead>
|
<TableHeader className="sticky top-0 bg-background/95 backdrop-blur supports-[backdrop-filter]:bg-background/60 z-10">
|
||||||
))}
|
{table.getHeaderGroups().map((headerGroup) => (
|
||||||
</TableRow>
|
<TableRow key={headerGroup.id} className="border-b">
|
||||||
))}
|
{headerGroup.headers.map((header) => (
|
||||||
</TableHeader>
|
<TableHead key={header.id} className="h-12 text-xs sm:text-sm">
|
||||||
<TableBody>
|
{header.isPlaceholder
|
||||||
{table.getRowModel().rows?.length ? (
|
? null
|
||||||
table.getRowModel().rows.map((row) => (
|
: flexRender(header.column.columnDef.header, header.getContext())}
|
||||||
<TableRow
|
</TableHead>
|
||||||
key={row.id}
|
|
||||||
data-state={row.getIsSelected() && "selected"}
|
|
||||||
className="hover:bg-muted/30"
|
|
||||||
>
|
|
||||||
{row.getVisibleCells().map((cell) => (
|
|
||||||
<TableCell key={cell.id} className="py-3 text-xs sm:text-sm">
|
|
||||||
{flexRender(cell.column.columnDef.cell, cell.getContext())}
|
|
||||||
</TableCell>
|
|
||||||
))}
|
))}
|
||||||
</TableRow>
|
</TableRow>
|
||||||
))
|
))}
|
||||||
) : (
|
</TableHeader>
|
||||||
<TableRow>
|
<TableBody>
|
||||||
<TableCell
|
{table.getRowModel().rows?.length ? (
|
||||||
colSpan={columns.length}
|
table.getRowModel().rows.map((row) => (
|
||||||
className="h-32 text-center text-muted-foreground text-sm"
|
<TableRow
|
||||||
>
|
key={row.id}
|
||||||
No documents found.
|
data-state={row.getIsSelected() && "selected"}
|
||||||
</TableCell>
|
className="hover:bg-muted/30"
|
||||||
</TableRow>
|
>
|
||||||
)}
|
{row.getVisibleCells().map((cell) => (
|
||||||
</TableBody>
|
<TableCell key={cell.id} className="py-3 text-xs sm:text-sm">
|
||||||
</Table>
|
{flexRender(cell.column.columnDef.cell, cell.getContext())}
|
||||||
|
</TableCell>
|
||||||
|
))}
|
||||||
|
</TableRow>
|
||||||
|
))
|
||||||
|
) : (
|
||||||
|
<TableRow>
|
||||||
|
<TableCell
|
||||||
|
colSpan={columns.length}
|
||||||
|
className="h-32 text-center text-muted-foreground text-sm"
|
||||||
|
>
|
||||||
|
No documents found.
|
||||||
|
</TableCell>
|
||||||
|
</TableRow>
|
||||||
|
)}
|
||||||
|
</TableBody>
|
||||||
|
</Table>
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* Footer Pagination */}
|
{/* Footer Pagination */}
|
||||||
<div className="flex flex-col sm:flex-row sm:items-center sm:justify-between gap-3 text-xs sm:text-sm text-muted-foreground border-t pt-3 md:pt-4 flex-shrink-0">
|
<div className="flex flex-col sm:flex-row sm:items-center sm:justify-between gap-3 text-xs sm:text-sm text-muted-foreground border-t pt-3 md:pt-4 flex-shrink-0">
|
||||||
<div className="text-center sm:text-left">
|
<div className="text-center sm:text-left">
|
||||||
Showing {table.getState().pagination.pageIndex * table.getState().pagination.pageSize + 1}{" "}
|
Showing {pageIndex * pageSize + 1} to {Math.min((pageIndex + 1) * pageSize, total)} of{" "}
|
||||||
to{" "}
|
{total} documents
|
||||||
{Math.min(
|
|
||||||
(table.getState().pagination.pageIndex + 1) * table.getState().pagination.pageSize,
|
|
||||||
table.getFilteredRowModel().rows.length
|
|
||||||
)}{" "}
|
|
||||||
of {table.getFilteredRowModel().rows.length} documents
|
|
||||||
</div>
|
</div>
|
||||||
<div className="flex items-center justify-center sm:justify-end space-x-2">
|
<div className="flex items-center justify-center sm:justify-end space-x-2">
|
||||||
<Button
|
<Button
|
||||||
variant="outline"
|
variant="outline"
|
||||||
size="sm"
|
size="sm"
|
||||||
onClick={() => table.previousPage()}
|
onClick={() => setPageIndex((p) => Math.max(0, p - 1))}
|
||||||
disabled={!table.getCanPreviousPage()}
|
disabled={pageIndex === 0 || loading}
|
||||||
className="text-xs sm:text-sm"
|
className="text-xs sm:text-sm"
|
||||||
>
|
>
|
||||||
Previous
|
Previous
|
||||||
</Button>
|
</Button>
|
||||||
<div className="flex items-center space-x-1 text-xs sm:text-sm">
|
<div className="flex items-center space-x-1 text-xs sm:text-sm">
|
||||||
<span>Page</span>
|
<span>Page</span>
|
||||||
<strong>{table.getState().pagination.pageIndex + 1}</strong>
|
<strong>{pageIndex + 1}</strong>
|
||||||
<span>of</span>
|
<span>of</span>
|
||||||
<strong>{table.getPageCount()}</strong>
|
<strong>{Math.ceil(total / pageSize)}</strong>
|
||||||
</div>
|
</div>
|
||||||
<Button
|
<Button
|
||||||
variant="outline"
|
variant="outline"
|
||||||
size="sm"
|
size="sm"
|
||||||
onClick={() => table.nextPage()}
|
onClick={() => setPageIndex((p) => p + 1)}
|
||||||
disabled={!table.getCanNextPage()}
|
disabled={pageIndex >= Math.ceil(total / pageSize) - 1 || loading}
|
||||||
className="text-xs sm:text-sm"
|
className="text-xs sm:text-sm"
|
||||||
>
|
>
|
||||||
Next
|
Next
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue