import { ColumnDef, ColumnFiltersState, getFilteredRowModel, SortingState, getSortedRowModel, flexRender, VisibilityState, getCoreRowModel, useReactTable, Row, Table as TableType, } from "@tanstack/react-table" import { TableBody, TableCell, TableHead, TableHeader, TableRow } from "@/components/ui/table" import { Button } from "@/components/ui/button" import { DropdownMenu, DropdownMenuCheckboxItem, DropdownMenuContent, DropdownMenuItem, DropdownMenuLabel, DropdownMenuRadioGroup, DropdownMenuRadioItem, DropdownMenuSeparator, DropdownMenuTrigger, } from "@/components/ui/dropdown-menu" import { SystemRecord } from "@/types" import { ArrowUpDownIcon, LayoutGridIcon, LayoutListIcon, ArrowDownIcon, ArrowUpIcon, Settings2Icon, EyeIcon, FilterIcon, } from "lucide-react" import { memo, useEffect, useMemo, useRef, useState } from "react" import { $pausedSystems, $downSystems, $upSystems, $systems } from "@/lib/stores" import { useStore } from "@nanostores/react" import { cn, runOnce, useBrowserStorage } from "@/lib/utils" import { $router, Link } from "../router" import { useLingui, Trans } from "@lingui/react/macro" import { Card, CardContent, CardDescription, CardHeader, CardTitle } from "../ui/card" import { Input } from "@/components/ui/input" import { getPagePath } from "@nanostores/router" import SystemsTableColumns, { ActionsButton, IndicatorDot } from "./systems-table-columns" import AlertButton from "../alerts/alert-button" import { SystemStatus } from "@/lib/enums" import { useVirtualizer, VirtualItem } from "@tanstack/react-virtual" type ViewMode = "table" | "grid" type StatusFilter = "all" | SystemRecord["status"] const preloadSystemDetail = runOnce(() => import("@/components/routes/system.tsx")) export default function SystemsTable() { const data = useStore($systems) const downSystems = $downSystems.get() const upSystems = $upSystems.get() const pausedSystems = $pausedSystems.get() const { i18n, t } = useLingui() const [filter, setFilter] = useState() const [statusFilter, setStatusFilter] = useState("all") const [sorting, setSorting] = useBrowserStorage( "sortMode", [{ id: "system", desc: false }], sessionStorage ) const [columnFilters, setColumnFilters] = useState([]) const [columnVisibility, setColumnVisibility] = useBrowserStorage("cols", {}) const locale = i18n.locale // Filter data based on status filter const filteredData = useMemo(() => { if (statusFilter === "all") { return data } if (statusFilter === SystemStatus.Up) { return Object.values(upSystems) ?? [] } if (statusFilter === SystemStatus.Down) { return Object.values(downSystems) ?? [] } return Object.values(pausedSystems) ?? [] }, [data, statusFilter]) const [viewMode, setViewMode] = useBrowserStorage( "viewMode", // show grid view on mobile if there are less than 200 systems (looks better but table is more efficient) window.innerWidth < 1024 && filteredData.length < 200 ? "grid" : "table" ) useEffect(() => { if (filter !== undefined) { table.getColumn("system")?.setFilterValue(filter) } }, [filter]) const columnDefs = useMemo(() => SystemsTableColumns(viewMode), [viewMode]) const table = useReactTable({ data: filteredData, columns: columnDefs, getCoreRowModel: getCoreRowModel(), onSortingChange: setSorting, getSortedRowModel: getSortedRowModel(), onColumnFiltersChange: setColumnFilters, getFilteredRowModel: getFilteredRowModel(), onColumnVisibilityChange: setColumnVisibility, state: { sorting, columnFilters, columnVisibility, }, defaultColumn: { invertSorting: true, sortUndefined: "last", minSize: 0, size: 900, maxSize: 900, }, }) const rows = table.getRowModel().rows const columns = table.getAllColumns() const visibleColumns = table.getVisibleLeafColumns() const [upSystemsLength, downSystemsLength, pausedSystemsLength] = useMemo(() => { return [Object.values(upSystems).length, Object.values(downSystems).length, Object.values(pausedSystems).length] }, [upSystems, downSystems, pausedSystems]) // TODO: hiding temp then gpu messes up table headers const CardHead = useMemo(() => { return (
All Systems Click on a system to view more information.
setFilter(e.target.value)} className="px-4" />
Layout setViewMode(view as ViewMode)} > e.preventDefault()} className="gap-2"> Table e.preventDefault()} className="gap-2"> Grid
Status setStatusFilter(value as StatusFilter)} > e.preventDefault()}> All Systems e.preventDefault()}> Up ({upSystemsLength}) e.preventDefault()}> Down ({downSystemsLength}) e.preventDefault()}> Paused ({pausedSystemsLength})
Sort By
{columns.map((column) => { if (!column.getCanSort()) return null let Icon = // if current sort column, show sort direction if (sorting[0]?.id === column.id) { if (sorting[0]?.desc) { Icon = } else { Icon = } } return ( { e.preventDefault() setSorting([{ id: column.id, desc: sorting[0]?.id === column.id && !sorting[0]?.desc }]) }} key={column.id} > {Icon} {/* @ts-ignore */} {column.columnDef.name()} ) })}
Visible Fields
{columns .filter((column) => column.getCanHide()) .map((column) => { return ( e.preventDefault()} checked={column.getIsVisible()} onCheckedChange={(value) => column.toggleVisibility(!!value)} > {/* @ts-ignore */} {column.columnDef.name()} ) })}
) }, [ visibleColumns.length, sorting, viewMode, locale, statusFilter, upSystemsLength, downSystemsLength, pausedSystemsLength, ]) return ( {CardHead}
{viewMode === "table" ? ( // table layout
) : ( // grid layout
{rows?.length ? ( rows.map((row) => { return }) ) : (
No systems found.
)}
)}
) } const AllSystemsTable = memo(function ({ table, rows, colLength, }: { table: TableType rows: Row[] colLength: number }) { // The virtualizer will need a reference to the scrollable container element const scrollRef = useRef(null) const virtualizer = useVirtualizer({ count: rows.length, estimateSize: () => (rows.length > 10 ? 56 : 60), getScrollElement: () => scrollRef.current, overscan: 5, }) const virtualRows = virtualizer.getVirtualItems() const paddingTop = Math.max(0, virtualRows[0]?.start ?? 0 - virtualizer.options.scrollMargin) const paddingBottom = Math.max(0, virtualizer.getTotalSize() - (virtualRows[virtualRows.length - 1]?.end ?? 0)) return (
2) && "min-h-50" )} ref={scrollRef} > {/* add header height to table size */}
{rows.length ? ( virtualRows.map((virtualRow) => { const row = rows[virtualRow.index] as Row return ( ) }) ) : ( No systems found. )}
) }) function SystemsTableHead({ table, colLength }: { table: TableType; colLength: number }) { const { i18n } = useLingui() return useMemo(() => { return ( {table.getHeaderGroups().map((headerGroup) => ( {headerGroup.headers.map((header) => { return ( {flexRender(header.column.columnDef.header, header.getContext())} ) })} ))} ) }, [i18n.locale, colLength]) } const SystemTableRow = memo(function ({ row, virtualRow, colLength, }: { row: Row virtualRow: VirtualItem length: number colLength: number }) { const system = row.original const { t } = useLingui() return useMemo(() => { return ( {row.getVisibleCells().map((cell) => ( {flexRender(cell.column.columnDef.cell, cell.getContext())} ))} ) }, [system, system.status, colLength, t]) }) const SystemCard = memo( ({ row, table, colLength }: { row: Row; table: TableType; colLength: number }) => { const system = row.original const { t } = useLingui() return useMemo(() => { return (
{system.name}
{table.getColumn("actions")?.getIsVisible() && (
)}
{table.getAllColumns().map((column) => { if (!column.getIsVisible() || column.id === "system" || column.id === "actions") return null const cell = row.getAllCells().find((cell) => cell.column.id === column.id) if (!cell) return null // @ts-ignore const { Icon, name } = column.columnDef as ColumnDef return ( <>
{column.id === "lastSeen" ? ( ) : ( Icon && )}
{name()}:
{flexRender(cell.column.columnDef.cell, cell.getContext())}
) })}
{row.original.name}
) }, [system, colLength, t]) } )