mirror of
https://github.com/ruvnet/RuVector.git
synced 2026-05-29 19:33:34 +00:00
Major additions: - Complete Next.js studio application with 1600+ components - Docker support (Dockerfile.combined, docker-compose.yml) - GCP deployment documentation and benchmarks - SQL benchmark scripts for performance testing - Sentry integration for monitoring - Comprehensive test suite and mocks Studio features: - Dashboard and admin interfaces - Data visualization components - Authentication and user management - API integration with RuVector backend - Static data and public assets 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
123 lines
3.6 KiB
TypeScript
123 lines
3.6 KiB
TypeScript
import { ChevronDown, ChevronUp, X } from 'lucide-react'
|
|
import { ReactNode, useCallback, useEffect, useMemo } from 'react'
|
|
|
|
import { Button, cn, Separator, Skeleton, Tooltip, TooltipContent, TooltipTrigger } from 'ui'
|
|
import { Kbd } from './primitives/Kbd'
|
|
import { useDataTable } from './providers/DataTableProvider'
|
|
|
|
export interface DataTableSheetDetailsProps {
|
|
title?: ReactNode
|
|
titleClassName?: string
|
|
children?: ReactNode
|
|
}
|
|
|
|
export function DataTableSheetDetails({
|
|
title,
|
|
titleClassName,
|
|
children,
|
|
}: DataTableSheetDetailsProps) {
|
|
const { table, rowSelection, isLoading } = useDataTable()
|
|
|
|
const selectedRowKey = Object.keys(rowSelection)?.[0]
|
|
|
|
const selectedRow = useMemo(() => {
|
|
if (isLoading && !selectedRowKey) return
|
|
return table.getRowModel().rows.find((row) => row.id === selectedRowKey)
|
|
}, [selectedRowKey, isLoading, table])
|
|
|
|
const index = table.getRowModel().rows.findIndex((row) => row.id === selectedRow?.id)
|
|
|
|
const nextId = useMemo(() => table.getRowModel().rows[index + 1]?.id, [index, table])
|
|
|
|
const prevId = useMemo(() => table.getRowModel().rows[index - 1]?.id, [index, table])
|
|
|
|
const onPrev = useCallback(() => {
|
|
if (prevId) table.setRowSelection({ [prevId]: true })
|
|
}, [prevId, table])
|
|
|
|
const onNext = useCallback(() => {
|
|
if (nextId) table.setRowSelection({ [nextId]: true })
|
|
}, [nextId, table])
|
|
|
|
useEffect(() => {
|
|
const down = (e: KeyboardEvent) => {
|
|
if (!selectedRowKey) return
|
|
|
|
// REMINDER: prevent dropdown navigation inside of sheet to change row selection
|
|
const activeElement = document.activeElement
|
|
const isMenuActive = activeElement?.closest('[role="menu"]')
|
|
|
|
if (isMenuActive) return
|
|
|
|
if (e.key === 'ArrowUp') {
|
|
e.preventDefault()
|
|
onPrev()
|
|
}
|
|
if (e.key === 'ArrowDown') {
|
|
e.preventDefault()
|
|
onNext()
|
|
}
|
|
}
|
|
|
|
document.addEventListener('keydown', down)
|
|
return () => document.removeEventListener('keydown', down)
|
|
}, [selectedRowKey, onNext, onPrev])
|
|
|
|
return (
|
|
<div className="relative bg-sidebar">
|
|
<div className="flex items-center justify-between gap-2 pl-5 pr-2 py-1">
|
|
<h5 className={cn(titleClassName, 'truncate text-left')}>
|
|
{isLoading && !selectedRowKey ? <Skeleton className="h-7 w-36" /> : title}
|
|
</h5>
|
|
<div className="flex h-7 items-center gap-1">
|
|
<Tooltip>
|
|
<TooltipTrigger asChild>
|
|
<Button
|
|
size="tiny"
|
|
type="text"
|
|
disabled={!prevId}
|
|
onClick={onPrev}
|
|
className="px-1"
|
|
icon={<ChevronUp />}
|
|
/>
|
|
</TooltipTrigger>
|
|
<TooltipContent>
|
|
<p>
|
|
Navigate <Kbd>↑</Kbd>
|
|
</p>
|
|
</TooltipContent>
|
|
</Tooltip>
|
|
|
|
<Tooltip>
|
|
<TooltipTrigger asChild>
|
|
<Button
|
|
size="tiny"
|
|
type="text"
|
|
disabled={!nextId}
|
|
onClick={onNext}
|
|
className="px-1"
|
|
icon={<ChevronDown />}
|
|
/>
|
|
</TooltipTrigger>
|
|
<TooltipContent>
|
|
<p>
|
|
Navigate <Kbd>↓</Kbd>
|
|
</p>
|
|
</TooltipContent>
|
|
</Tooltip>
|
|
<Separator orientation="vertical" className="mx-1" />
|
|
|
|
<Button
|
|
size="tiny"
|
|
type="text"
|
|
onClick={() => table.resetRowSelection()}
|
|
className="px-1"
|
|
icon={<X />}
|
|
/>
|
|
</div>
|
|
</div>
|
|
<Separator />
|
|
{children}
|
|
</div>
|
|
)
|
|
}
|