import { type Column, type ColumnDef, flexRender, getCoreRowModel, getExpandedRowModel, useReactTable, } from "@tanstack/react-table" import { Loader } from "lucide-react" import { type CSSProperties, useCallback, useMemo } from "react" import { Pagination, type PaginationProps } from "@/components/ui/pagination" import { TableBody, TableCell, TableHead, TableHeader, Table as TableRoot, TableRow, } from "@/components/ui/table" import { cn } from "@/lib/utils" export type DataTableProps = { data: T[] status: "load" | "done" | "fail" columns: ColumnDef[] pagination?: PaginationProps classNames?: { root?: string headRow?: string dataRow?: string } } export function DataTable>( props: DataTableProps, ) { const table = useReactTable({ data: props.data, columns: props.columns, getCoreRowModel: getCoreRowModel(), manualPagination: true, rowCount: props.pagination?.total, state: { pagination: props.pagination && { pageIndex: props.pagination.page, pageSize: props.pagination.size, }, columnFilters: [], }, initialState: { columnPinning: { left: props.columns .map(column => column.meta?.pin === "left" ? column.id || column.accessorKey : undefined, ) .filter(Boolean), right: props.columns .map(column => column.meta?.pin === "right" ? column.id || column.accessorKey : undefined, ) .filter(Boolean), }, }, }) const pinStyle = (column: Column) => { const pinned = column.getIsPinned() if (!pinned) return {} return { position: pinned ? ("sticky" as const) : undefined, backgroundColor: "white", zIndex: 1, ...{ left: { left: column.getStart(pinned), boxShadow: "inset 1px 0 var(--border)", }, right: { right: column.getAfter(pinned), boxShadow: "inset 1px 0 var(--border)", }, }[pinned], } as CSSProperties } return (
{/* 数据表 */}
{table.getHeaderGroups().map(group => ( {group.headers.map(header => ( {header.isPlaceholder ? null : flexRender( header.column.columnDef.header, header.getContext(), )} ))} ))} {props.status === "fail" ? ( 加载失败 ) : !props.data?.length ? ( 暂无数据 ) : ( table.getRowModel().rows.map(row => ( {row.getVisibleCells().map(cell => ( {flexRender( cell.column.columnDef.cell, cell.getContext(), )} ))} )) )} {props.status === "load" && (
加载中
)}
{/* 分页器 */} {props.pagination && }
) }