ruvector/studio/components/ui/DataTable/DataTableSheetDetails.tsx
rUv 814f595995 feat(studio): Add complete RuVector Studio application
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>
2025-12-06 23:04:48 +00:00

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>
)
}