feat: add useQuery implementation for document fetching in DocumentsDataTable with 3-minute stale time

This commit is contained in:
CREDO23 2025-12-08 08:41:29 +00:00
parent bccbd65333
commit 42e10bbe55
3 changed files with 77 additions and 55 deletions

View file

@ -10,6 +10,7 @@ import {
import { ArrowUpDown, Calendar, FileText, Filter, Plus, Search } from "lucide-react"; import { ArrowUpDown, Calendar, FileText, Filter, Plus, Search } from "lucide-react";
import { useRouter } from "next/navigation"; import { useRouter } from "next/navigation";
import { useCallback, useEffect, useMemo, useRef, useState } from "react"; import { useCallback, useEffect, useMemo, useRef, useState } from "react";
import { useQuery } from "@tanstack/react-query";
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";
@ -32,6 +33,9 @@ import {
} from "@/components/ui/table"; } from "@/components/ui/table";
import { getConnectorIcon } from "@/contracts/enums/connectorIcons"; import { getConnectorIcon } from "@/contracts/enums/connectorIcons";
import { type Document, type DocumentType, useDocuments } from "@/hooks/use-documents"; import { type Document, type DocumentType, useDocuments } from "@/hooks/use-documents";
import { documentsApiService } from "@/lib/apis/documents-api.service";
import { cacheKeys } from "@/lib/query-client/cache-keys";
import { DocumentTypeEnum } from "@/contracts/types/document.types";
interface DocumentsDataTableProps { interface DocumentsDataTableProps {
searchSpaceId: number; searchSpaceId: number;
@ -182,18 +186,62 @@ export function DocumentsDataTable({
const [sorting, setSorting] = useState<SortingState>([]); const [sorting, setSorting] = useState<SortingState>([]);
const [search, setSearch] = useState(""); const [search, setSearch] = useState("");
const debouncedSearch = useDebounced(search, 300); const debouncedSearch = useDebounced(search, 300);
const [documentTypeFilter, setDocumentTypeFilter] = useState<string[]>([]); const [documentTypeFilter, setDocumentTypeFilter] = useState<DocumentTypeEnum[]>([]);
const [pageIndex, setPageIndex] = useState(0); const [pageIndex, setPageIndex] = useState(0);
const [pageSize, setPageSize] = useState(10); const [pageSize, setPageSize] = useState(10);
const [typeCounts, setTypeCounts] = useState<Record<string, number>>({}); const [typeCounts, setTypeCounts] = useState<Record<string, number>>({});
const fetchQueryParams = useMemo(
() => ({
search_space_id: searchSpaceId,
page: pageIndex ,
page_size: pageSize,
...(documentTypeFilter.length > 0 && { document_types: documentTypeFilter }),
}),
[searchSpaceId, pageIndex, pageSize, documentTypeFilter, debouncedSearch]
);
const searchQueryParams = useMemo(() => {
return {
...fetchQueryParams,
title : debouncedSearch,
}
},[debouncedSearch])
// Use query for fetching documents
const {
data: documents,
isLoading: isDocumentsLoading,
} = useQuery({
queryKey: cacheKeys.documents.withQueryParams(fetchQueryParams),
queryFn: () => documentsApiService.getDocuments({ queryParams : fetchQueryParams }),
staleTime: 3 * 60 * 1000, // 3 minutes
enabled: !!searchSpaceId && !debouncedSearch.trim(),
});
// Seaching
const {
data: searchedDocuments,
isLoading: isSearchedDocumentsLoading,
} = useQuery({
queryKey: cacheKeys.documents.withQueryParams(searchQueryParams),
queryFn: () => documentsApiService.searchDocuments({ queryParams : searchQueryParams }),
staleTime: 3 * 60 * 1000, // 3 minutes
enabled: !!searchSpaceId && !!debouncedSearch.trim(),
});
// Use server-side pagination, search, and filtering // Use server-side pagination, search, and filtering
const { documents, total, loading, fetchDocuments, searchDocuments, getDocumentTypeCounts } = const { getDocumentTypeCounts } =
useDocuments(searchSpaceId, { useDocuments(searchSpaceId, {
page: pageIndex, page: pageIndex,
pageSize: pageSize, pageSize: pageSize,
}); });
// Use query data when not searching, otherwise use hook data
const actualDocuments = debouncedSearch.trim() ? searchedDocuments?.items|| [] : documents?.items|| [];
const actualTotal = debouncedSearch.trim() ? searchedDocuments?.total || 0 : documents?.total || 0;
const actualLoading = debouncedSearch.trim() ? isSearchedDocumentsLoading : isDocumentsLoading;
// Fetch document type counts on mount // Fetch document type counts on mount
useEffect(() => { useEffect(() => {
if (searchSpaceId && getDocumentTypeCounts) { if (searchSpaceId && getDocumentTypeCounts) {
@ -201,34 +249,6 @@ export function DocumentsDataTable({
} }
}, [searchSpaceId, getDocumentTypeCounts]); }, [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 (!initialSelectedDocuments.length) return {}; if (!initialSelectedDocuments.length) return {};
@ -272,14 +292,14 @@ export function DocumentsDataTable({
// Update the selected documents map when row selection changes // Update the selected documents map when row selection changes
useEffect(() => { useEffect(() => {
if (!documents || documents.length === 0) return; if (!actualDocuments || actualDocuments.length === 0) return;
setSelectedDocumentsMap((prev) => { setSelectedDocumentsMap((prev) => {
const newMap = new Map(prev); const newMap = new Map(prev);
let hasChanges = false; let hasChanges = false;
// Process only current page documents // Process only current page documents
for (const doc of documents) { for (const doc of actualDocuments) {
const docId = doc.id; const docId = doc.id;
const isSelected = rowSelection[docId.toString()]; const isSelected = rowSelection[docId.toString()];
const wasInMap = newMap.has(docId); const wasInMap = newMap.has(docId);
@ -319,14 +339,14 @@ export function DocumentsDataTable({
}, [selectedDocumentsArray, onSelectionChange]); }, [selectedDocumentsArray, onSelectionChange]);
const table = useReactTable({ const table = useReactTable({
data: documents || [], data: actualDocuments || [],
columns, columns,
getRowId: (row) => row.id.toString(), getRowId: (row) => row.id.toString(),
onSortingChange: setSorting, onSortingChange: setSorting,
getCoreRowModel: getCoreRowModel(), getCoreRowModel: getCoreRowModel(),
onRowSelectionChange: setRowSelection, onRowSelectionChange: setRowSelection,
manualPagination: true, manualPagination: true,
pageCount: Math.ceil(total / pageSize), pageCount: Math.ceil(actualTotal / pageSize),
state: { sorting, rowSelection, pagination: { pageIndex, pageSize } }, state: { sorting, rowSelection, pagination: { pageIndex, pageSize } },
}); });
@ -344,7 +364,7 @@ export function DocumentsDataTable({
setRowSelection(newSelection); setRowSelection(newSelection);
}, [table, rowSelection]); }, [table, rowSelection]);
const handleToggleType = useCallback((type: string, checked: boolean) => { const handleToggleType = useCallback((type: DocumentTypeEnum, checked: boolean) => {
setDocumentTypeFilter((prev) => { setDocumentTypeFilter((prev) => {
if (checked) { if (checked) {
return [...prev, type]; return [...prev, type];
@ -358,7 +378,7 @@ export function DocumentsDataTable({
// Get available document types from type counts (memoized) // Get available document types from type counts (memoized)
const availableTypes = useMemo(() => { const availableTypes = useMemo(() => {
const types = Object.keys(typeCounts); const types = Object.keys(typeCounts) as DocumentTypeEnum[];
return types.length > 0 ? types.sort() : []; return types.length > 0 ? types.sort() : [];
}, [typeCounts]); }, [typeCounts]);
@ -435,7 +455,7 @@ export function DocumentsDataTable({
<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} selected {loading && "· Loading..."} {selectedCount} selected {actualLoading && "· 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">
@ -453,7 +473,7 @@ 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} disabled={actualLoading}
> >
Select Page Select Page
</Button> </Button>
@ -490,7 +510,7 @@ 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">
{loading ? ( {actualLoading ? (
<div className="flex items-center justify-center h-full"> <div className="flex items-center justify-center h-full">
<div className="text-center space-y-2"> <div className="text-center space-y-2">
<div className="animate-spin h-8 w-8 border-2 border-primary border-t-transparent rounded-full mx-auto" /> <div className="animate-spin h-8 w-8 border-2 border-primary border-t-transparent rounded-full mx-auto" />
@ -561,31 +581,31 @@ export function DocumentsDataTable({
{/* 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 {pageIndex * pageSize + 1} to {Math.min((pageIndex + 1) * pageSize, total)} of{" "} Showing {pageIndex * pageSize + 1} to {Math.min((pageIndex + 1) * pageSize, actualTotal)} of{" "}
{total} documents {actualTotal} 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={() => setPageIndex((p) => Math.max(0, p - 1))} onClick={() => setPageIndex((p) => Math.max(0, p - 1))}
disabled={pageIndex === 0 || loading} disabled={pageIndex === 0 || actualLoading}
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>{pageIndex + 1}</strong> <strong>{pageIndex + 1}</strong>
<span>of</span> <span>of</span>
<strong>{Math.ceil(total / pageSize)}</strong> <strong>{Math.ceil(actualTotal / pageSize)}</strong>
</div> </div>
<Button <Button
variant="outline" variant="outline"
size="sm" size="sm"
onClick={() => setPageIndex((p) => p + 1)} onClick={() => setPageIndex((p) => p + 1)}
disabled={pageIndex >= Math.ceil(total / pageSize) - 1 || loading} disabled={pageIndex >= Math.ceil(actualTotal / pageSize) - 1 || actualLoading}
className="text-xs sm:text-sm" className="text-xs sm:text-sm"
> >
Next Next
</Button> </Button>

View file

@ -60,7 +60,7 @@ export const getDocumentsRequest = z.object({
queryParams: paginationQueryParams queryParams: paginationQueryParams
.extend({ .extend({
search_space_id: z.number().or(z.string()).optional(), search_space_id: z.number().or(z.string()).optional(),
document_type: z.array(documentTypeEnum).optional(), document_types: z.array(documentTypeEnum).optional(),
}) })
.nullish(), .nullish(),
}); });
@ -109,7 +109,7 @@ export const searchDocumentsRequest = z.object({
queryParams: paginationQueryParams queryParams: paginationQueryParams
.extend({ .extend({
search_space_id: z.number().or(z.string()).optional(), search_space_id: z.number().or(z.string()).optional(),
document_type: z.array(documentTypeEnum).optional(), document_types: z.array(documentTypeEnum).optional(),
title: z.string().optional(), title: z.string().optional(),
}) })
.nullish(), .nullish(),
@ -179,3 +179,4 @@ export type UpdateDocumentRequest = z.infer<typeof updateDocumentRequest>;
export type UpdateDocumentResponse = z.infer<typeof updateDocumentResponse>; export type UpdateDocumentResponse = z.infer<typeof updateDocumentResponse>;
export type DeleteDocumentRequest = z.infer<typeof deleteDocumentRequest>; export type DeleteDocumentRequest = z.infer<typeof deleteDocumentRequest>;
export type DeleteDocumentResponse = z.infer<typeof deleteDocumentResponse>; export type DeleteDocumentResponse = z.infer<typeof deleteDocumentResponse>;
export type DocumentTypeEnum = z.infer<typeof documentTypeEnum>

View file

@ -15,6 +15,7 @@ export const cacheKeys = {
documents: { documents: {
globalQueryParams: (queries: GetDocumentsRequest["queryParams"]) => globalQueryParams: (queries: GetDocumentsRequest["queryParams"]) =>
["documents", ...(queries ? Object.values(queries) : [])] as const, ["documents", ...(queries ? Object.values(queries) : [])] as const,
withQueryParams :(queries: GetDocumentsRequest["queryParams"]) => ["documents-with-queries", ...(queries ? Object.values(queries) : [])] as const,
document: (documentId: string) => ["document", documentId] as const, document: (documentId: string) => ["document", documentId] as const,
typeCounts: (searchSpaceId?: string) => ["documents", "type-counts", searchSpaceId] as const, typeCounts: (searchSpaceId?: string) => ["documents", "type-counts", searchSpaceId] as const,
byChunk: (chunkId: string) => ["documents", "by-chunk", chunkId] as const, byChunk: (chunkId: string) => ["documents", "by-chunk", chunkId] as const,