import { CellContext, ColumnDef, ColumnFiltersState, getFilteredRowModel, SortingState, getSortedRowModel, flexRender, VisibilityState, getCoreRowModel, useReactTable, HeaderContext, } from "@tanstack/react-table" import { Table, TableBody, TableCell, TableHead, TableHeader, TableRow } from "@/components/ui/table" import { Button, buttonVariants } from "@/components/ui/button" import { DropdownMenu, DropdownMenuCheckboxItem, DropdownMenuContent, DropdownMenuItem, DropdownMenuLabel, DropdownMenuRadioGroup, DropdownMenuRadioItem, DropdownMenuSeparator, DropdownMenuTrigger, } from "@/components/ui/dropdown-menu" import { AlertDialog, AlertDialogAction, AlertDialogCancel, AlertDialogContent, AlertDialogDescription, AlertDialogFooter, AlertDialogHeader, AlertDialogTitle, } from "@/components/ui/alert-dialog" import { SystemRecord } from "@/types" import { MoreHorizontalIcon, ArrowUpDownIcon, MemoryStickIcon, CopyIcon, PauseCircleIcon, PlayCircleIcon, Trash2Icon, WifiIcon, HardDriveIcon, ServerIcon, CpuIcon, LayoutGridIcon, LayoutListIcon, ArrowDownIcon, ArrowUpIcon, Settings2Icon, EyeIcon, PenBoxIcon, } from "lucide-react" import { memo, useEffect, useMemo, useRef, useState } from "react" import { $hubVersion, $systems, pb } from "@/lib/stores" import { useStore } from "@nanostores/react" import { cn, copyToClipboard, decimalString, isReadOnlyUser, useLocalStorage } from "@/lib/utils" import AlertsButton from "../alerts/alert-button" import { $router, Link, navigate } from "../router" import { EthernetIcon, GpuIcon, ThermometerIcon } from "../ui/icons" import { Trans, t } from "@lingui/macro" import { useLingui } from "@lingui/react" import { Card, CardContent, CardDescription, CardHeader, CardTitle } from "../ui/card" import { Input } from "../ui/input" import { ClassValue } from "clsx" import { getPagePath } from "@nanostores/router" import { SystemDialog } from "../add-system" import { Dialog } from "../ui/dialog" type ViewMode = "table" | "grid" function CellFormatter(info: CellContext) { const val = (info.getValue() as number) || 0 return (
{decimalString(val, 1)}%
) } function sortableHeader(context: HeaderContext, hideSortIcon = false) { const { column } = context return ( ) } export default function SystemsTable() { const data = useStore($systems) const hubVersion = useStore($hubVersion) const [filter, setFilter] = useState() const [sorting, setSorting] = useState([{ id: t`System`, desc: false }]) const [columnFilters, setColumnFilters] = useState([]) const [columnVisibility, setColumnVisibility] = useLocalStorage("cols", {}) const [viewMode, setViewMode] = useLocalStorage("viewMode", window.innerWidth > 1024 ? "table" : "grid") const { i18n } = useLingui() useEffect(() => { if (filter !== undefined) { table.getColumn(t`System`)?.setFilterValue(filter) } }, [filter]) const columns = useMemo(() => { return [ { // size: 200, size: 200, minSize: 0, accessorKey: "name", id: t`System`, enableHiding: false, icon: ServerIcon, cell: (info) => ( ), header: sortableHeader, }, { accessorKey: "info.cpu", id: t`CPU`, invertSorting: true, cell: CellFormatter, icon: CpuIcon, header: sortableHeader, }, { accessorKey: "info.mp", id: t`Memory`, invertSorting: true, cell: CellFormatter, icon: MemoryStickIcon, header: sortableHeader, }, { accessorKey: "info.dp", id: t`Disk`, invertSorting: true, cell: CellFormatter, icon: HardDriveIcon, header: sortableHeader, }, { accessorFn: (originalRow) => originalRow.info.g, id: "GPU", invertSorting: true, sortUndefined: -1, cell: CellFormatter, icon: GpuIcon, header: sortableHeader, }, { accessorFn: (originalRow) => originalRow.info.dt, id: t`Temp`, invertSorting: true, sortUndefined: -1, size: 50, icon: ThermometerIcon, header: sortableHeader, cell(info) { const val = info.getValue() as number if (!val) { return null } return ( {decimalString(val)} °C ) }, }, { accessorFn: (originalRow) => originalRow.info.b || 0, id: t`Net`, invertSorting: true, size: 100, icon: EthernetIcon, header: sortableHeader, cell(info) { const val = info.getValue() as number return ( {decimalString(val, val >= 100 ? 1 : 2)} MB/s ) }, }, { accessorKey: "info.v", id: t`Agent`, invertSorting: true, size: 50, icon: WifiIcon, header: sortableHeader, cell(info) { const version = info.getValue() as string if (!version || !hubVersion) { return null } const system = info.row.original return ( {info.getValue() as string} ) }, }, { id: t({ message: "Actions", comment: "Table column" }), size: 120, cell: ({ row }) => (
), }, ] as ColumnDef[] }, [hubVersion, i18n.locale]) const table = useReactTable({ data, columns, getCoreRowModel: getCoreRowModel(), onSortingChange: setSorting, getSortedRowModel: getSortedRowModel(), onColumnFiltersChange: setColumnFilters, getFilteredRowModel: getFilteredRowModel(), onColumnVisibilityChange: setColumnVisibility, state: { sorting, columnFilters, columnVisibility, }, defaultColumn: { minSize: 0, size: Number.MAX_SAFE_INTEGER, maxSize: Number.MAX_SAFE_INTEGER, }, }) return (
All Systems Updated in real time. Click on a system to view 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
Sort By
{table.getAllColumns().map((column) => { if (column.id === t`Actions` || !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} {column.id} ) })}
Visible Fields
{table .getAllColumns() .filter((column) => column.getCanHide()) .map((column) => { return ( e.preventDefault()} checked={column.getIsVisible()} onCheckedChange={(value) => column.toggleVisibility(!!value)} > {column.id} ) })}
{viewMode === "table" ? ( // table layout
{table.getHeaderGroups().map((headerGroup) => ( {headerGroup.headers.map((header) => { return ( {header.isPlaceholder ? null : flexRender(header.column.columnDef.header, header.getContext())} ) })} ))} {table.getRowModel().rows?.length ? ( table.getRowModel().rows.map((row) => ( { const target = e.target as HTMLElement if (!target.closest("[data-nolink]") && e.currentTarget.contains(target)) { navigate(getPagePath($router, "system", { name: row.original.name })) } }} > {row.getVisibleCells().map((cell) => ( 10 ? "py-2" : "py-2.5")} > {flexRender(cell.column.columnDef.cell, cell.getContext())} ))} )) ) : ( No systems found. )}
) : ( // grid layout
{table.getRowModel().rows?.length ? ( table.getRowModel().rows.map((row) => { const system = row.original const { status } = system return (
{system.name}
{table.getColumn(t`Actions`)?.getIsVisible() && (
)}
{table.getAllColumns().map((column) => { if (!column.getIsVisible() || column.id === t`System` || column.id === t`Actions`) return null const cell = row.getAllCells().find((cell) => cell.column.id === column.id) if (!cell) return null return (
{/* @ts-ignore */} {column.columnDef?.icon && ( // @ts-ignore )}
{column.id}:
{flexRender(cell.column.columnDef.cell, cell.getContext())}
) })}
{row.original.name}
) }) ) : (
No systems found.
)}
)}
) } function IndicatorDot({ system, className }: { system: SystemRecord; className?: ClassValue }) { className ||= { "bg-green-500": system.status === "up", "bg-red-500": system.status === "down", "bg-primary/40": system.status === "paused", "bg-yellow-500": system.status === "pending", } return ( ) } const ActionsButton = memo(({ system }: { system: SystemRecord }) => { const [deleteOpen, setDeleteOpen] = useState(false) const [editOpen, setEditOpen] = useState(false) let editOpened = useRef(false) const { id, status, host, name } = system return ( <> {!isReadOnlyUser() && ( { editOpened.current = true setEditOpen(true) }} > Edit )} { pb.collection("systems").update(id, { status: status === "paused" ? "pending" : "paused", }) }} > {status === "paused" ? ( <> Resume ) : ( <> Pause )} copyToClipboard(host)}> Copy host setDeleteOpen(true)}> Delete {/* edit dialog */} {editOpened.current && } {/* deletion dialog */} setDeleteOpen(open)}> Are you sure you want to delete {name}? This action cannot be undone. This will permanently delete all current records for {name} from the database. Cancel pb.collection("systems").delete(id)} > Continue ) })