Data Display
stable
Data Table Advanced
Data Table Advanced: Feature-rich table with sorting, filtering, and large dataset support.
Preview
Team overview
| Team | Owner | Status | |
|---|---|---|---|
| Platform | Ava | Stable | |
| Growth | Leo | Review | |
| Ops | Mina | Blocked |
0 of 3 row(s) selected.
Page 1 of 1
Import
tsx
import { DataTableAdvanced } from "@/components/ui/data-table-advanced";bash
npx sui add data-table-advancedSource
From components/ui/data-table-advanced.tsx
tsx
import React, { useState, useRef } from "react";
import {
useReactTable,
getCoreRowModel,
getPaginationRowModel,
getSortedRowModel,
getFilteredRowModel,
flexRender,
ColumnDef,
SortingState,
VisibilityState,
FilterFn,
} from "@tanstack/react-table";
import { useVirtualizer } from "@tanstack/react-virtual";
import {
Table,
TableBody,
TableCell,
TableHead,
TableHeader,
TableRow,
} from "./table";
import { Input } from "./input";
import { Button } from "./button";
import { Checkbox } from "./checkbox";
import {
ChevronDown,
ChevronLeft,
ChevronRight,
Search,
SlidersHorizontal,
ArrowUpDown,
Download,
MoreHorizontal,
Settings2,
Pencil,
Trash2,
Plus,
RefreshCw,
} from "lucide-react";
import {
DropdownMenu,
DropdownMenuCheckboxItem,
DropdownMenuContent,
DropdownMenuItem,
DropdownMenuLabel,
DropdownMenuSeparator,
DropdownMenuTrigger,
} from "./dropdown-menu";
// Fuzzy text filter for global search
const fuzzyFilter: FilterFn<any> = (row, columnId, value, addMeta) => {
const itemValue = row.getValue(columnId);
if (itemValue == null) return false;
return String(itemValue).toLowerCase().includes(String(value).toLowerCase());
};
export interface DataTableAdvancedProps<TData, TValue> {
columns: ColumnDef<TData, TValue>[];
data: TData[];
searchPlaceholder?: string;
enableVirtualization?: boolean;
enableCRUD?: boolean;
onAdd?: () => void;
onEdit?: (row: TData) => void;
onDelete?: (row: TData) => void;
onRefresh?: () => void;
title?: string;
}
export function DataTableAdvanced<TData, TValue>({
columns,
data,
searchPlaceholder = "Search all columns...",
enableVirtualization = false,
enableCRUD = false,
onAdd,
onEdit,
onDelete,
onRefresh,
title,
}: DataTableAdvancedProps<TData, TValue>) {
const [sorting, setSorting] = useState<SortingState>([]);
const [globalFilter, setGlobalFilter] = useState("");
const [columnVisibility, setColumnVisibility] = useState<VisibilityState>({});
const [rowSelection, setRowSelection] = useState({});
const crudColumn: ColumnDef<TData, any> = {
id: "actions",
header: () => <span className="text-xs text-muted-foreground">Actions</span>,
cell: ({ row }) => (
<div className="flex items-center gap-1">
{onEdit && (
<Button
variant="ghost"
size="sm"
className="h-7 w-7 p-0 text-muted-foreground hover:text-foreground"
onClick={() => onEdit(row.original)}
>
<Pencil className="h-3.5 w-3.5" />
</Button>
)}
{onDelete && (
<Button
variant="ghost"
size="sm"
className="h-7 w-7 p-0 text-muted-foreground hover:text-destructive"
onClick={() => onDelete(row.original)}
>
<Trash2 className="h-3.5 w-3.5" />
</Button>
)}
</div>
),
enableSorting: false,
enableHiding: false,
};
const finalColumns = React.useMemo<ColumnDef<TData, any>[]>(() => [
{
id: "select",
header: ({ table }) => (
<Checkbox
checked={
table.getIsAllPageRowsSelected()
? true
: table.getIsSomePageRowsSelected()
? "indeterminate"
: false
}
onCheckedChange={(value) => table.toggleAllPageRowsSelected(!!value)}
aria-label="Select all"
className="translate-y-[2px]"
/>
),
cell: ({ row }) => (
<Checkbox
checked={row.getIsSelected()}
onCheckedChange={(value) => row.toggleSelected(!!value)}
aria-label="Select row"
className="translate-y-[2px]"
/>
),
enableSorting: false,
enableHiding: false,
},
...columns,
...(enableCRUD && (onEdit || onDelete) ? [crudColumn] : []),
], [columns, enableCRUD, onEdit, onDelete]);
const table = useReactTable({
data,
columns: finalColumns,
state: {
sorting,
globalFilter,
columnVisibility,
rowSelection,
},
enableRowSelection: true,
globalFilterFn: fuzzyFilter,
onSortingChange: setSorting,
onGlobalFilterChange: setGlobalFilter,
onColumnVisibilityChange: setColumnVisibility,
onRowSelectionChange: setRowSelection,
getCoreRowModel: getCoreRowModel(),
getPaginationRowModel: getPaginationRowModel(),
getSortedRowModel: getSortedRowModel(),
getFilteredRowModel: getFilteredRowModel(),
});
// Virtualization setup
const tableContainerRef = useRef<HTMLDivElement>(null);
const { rows } = table.getRowModel();
const rowVirtualizer = useVirtualizer({
count: rows.length,
getScrollElement: () => tableContainerRef.current,
estimateSize: () => 53, // rough estimate of row height
overscan: 10,
});
return (
<div className="space-y-4">
{/* Table title + CRUD primary actions */}
{(title || onAdd || onRefresh) && (
<div className="flex items-center justify-between">
{title && <h3 className="font-semibold text-sm">{title}</h3>}
<div className="flex items-center gap-2 ml-auto">
{onRefresh && (
<Button variant="outline" size="sm" className="h-8 gap-1.5" onClick={onRefresh}>
<RefreshCw className="h-3.5 w-3.5" /> Refresh
</Button>
)}
{onAdd && (
<Button size="sm" className="h-8 gap-1.5" onClick={onAdd}>
<Plus className="h-3.5 w-3.5" /> Add Row
</Button>
)}
</div>
</div>
)}
{/* Top Bar: Search, Column Toggle, Export */}
<div className="flex flex-col sm:flex-row items-center justify-between gap-4">
<div className="relative w-full sm:max-w-sm">
<Search className="absolute left-3 top-1/2 -translate-y-1/2 h-4 w-4 text-muted-foreground" />
<Input
placeholder={searchPlaceholder}
value={globalFilter ?? ""}
onChange={(e) => setGlobalFilter(e.target.value)}
className="pl-10 h-10 w-full rounded-lg"
/>
</div>
<div className="flex items-center gap-2 w-full sm:w-auto">
<DropdownMenu>
<DropdownMenuTrigger asChild>
<Button variant="outline" size="sm" className="h-10 px-3 lg:px-4">
<Settings2 className="mr-2 h-4 w-4" />
View
</Button>
</DropdownMenuTrigger>
<DropdownMenuContent align="end" className="w-[150px]">
<DropdownMenuLabel>Toggle columns</DropdownMenuLabel>
<DropdownMenuSeparator />
{table
.getAllColumns()
.filter((column) => column.getCanHide())
.map((column) => {
return (
<DropdownMenuCheckboxItem
key={column.id}
className="capitalize"
checked={column.getIsVisible()}
onCheckedChange={(value) => column.toggleVisibility(!!value)}
>
{column.id}
</DropdownMenuCheckboxItem>
);
})}
</DropdownMenuContent>
</DropdownMenu>
<Button variant="outline" size="sm" className="h-10 px-3 lg:px-4">
<Download className="mr-2 h-4 w-4" /> Export
</Button>
</div>
</div>
{/* Table Area */}
<div
ref={tableContainerRef}
className={`rounded-xl border bg-background overflow-auto ${enableVirtualization ? "max-h-[500px]" : ""}`}
>
<Table>
<TableHeader className="sticky top-0 z-10 bg-muted/50 backdrop-blur-md">
{table.getHeaderGroups().map((headerGroup) => (
<TableRow key={headerGroup.id}>
{headerGroup.headers.map((header) => {
return (
<TableHead key={header.id} className="font-bold whitespace-nowrap">
{header.isPlaceholder
? null
: flexRender(
header.column.columnDef.header,
header.getContext()
)}
</TableHead>
);
})}
</TableRow>
))}
</TableHeader>
<TableBody>
{enableVirtualization ? (
// Virtualized Body
<>
{rowVirtualizer.getVirtualItems().length > 0 ? (
rowVirtualizer.getVirtualItems().map((virtualRow) => {
const row = rows[virtualRow.index];
return (
<TableRow
key={row.id}
data-state={row.getIsSelected() && "selected"}
className="hover:bg-muted/5 transition-colors absolute w-full"
style={{
transform: `translateY(${virtualRow.start}px)`,
}}
>
{row.getVisibleCells().map((cell) => (
<TableCell key={cell.id} className="whitespace-nowrap">
{flexRender(cell.column.columnDef.cell, cell.getContext())}
</TableCell>
))}
</TableRow>
);
})
) : (
<TableRow>
<TableCell colSpan={columns.length} className="h-24 text-center">
No results found.
</TableCell>
</TableRow>
)}
</>
) : (
// Standard Body
<>
{table.getRowModel().rows?.length ? (
table.getRowModel().rows.map((row) => (
<TableRow
key={row.id}
data-state={row.getIsSelected() && "selected"}
className="hover:bg-muted/5 transition-colors"
>
{row.getVisibleCells().map((cell) => (
<TableCell key={cell.id} className="whitespace-nowrap">
{flexRender(cell.column.columnDef.cell, cell.getContext())}
</TableCell>
))}
</TableRow>
))
) : (
<TableRow>
<TableCell colSpan={columns.length} className="h-24 text-center">
No results found.
</TableCell>
</TableRow>
)}
</>
)}
</TableBody>
</Table>
</div>
{/* Pagination & Selected Rows indicator */}
<div className="flex flex-col sm:flex-row items-center justify-between px-2 gap-4">
<div className="flex-1 text-sm text-muted-foreground">
{table.getFilteredSelectedRowModel().rows.length} of{" "}
{table.getFilteredRowModel().rows.length} row(s) selected.
</div>
<div className="flex items-center space-x-2">
<Button
variant="outline"
size="sm"
onClick={() => table.previousPage()}
disabled={!table.getCanPreviousPage()}
>
<ChevronLeft className="h-4 w-4 mr-1" /> Previous
</Button>
<span className="text-sm font-medium">
Page {table.getState().pagination.pageIndex + 1} of{" "}
{table.getPageCount() || 1}
</span>
<Button
variant="outline"
size="sm"
onClick={() => table.nextPage()}
disabled={!table.getCanNextPage()}
>
Next <ChevronRight className="h-4 w-4 ml-1" />
</Button>
</div>
</div>
</div>
);
}
Component Info
Slug
data-table-advancedCategory
Data Display
Status
stable
Quick Actions