ruvector/studio/components/interfaces/Auth/Users/UsersGridComponents.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

111 lines
3.3 KiB
TypeScript

import { ChevronDown, SortAsc, SortDesc } from 'lucide-react'
import { useEffect, useRef, useState } from 'react'
import { useRowSelection } from 'react-data-grid'
import {
Button,
DropdownMenu,
DropdownMenuContent,
DropdownMenuItem,
DropdownMenuTrigger,
} from 'ui'
// [Joshen] Currently not being used, refer to Users.utils -> formatUserColumns for more info
export const SelectHeaderCell = ({
selectedUsers,
allRowsSelected,
}: {
selectedUsers: Set<any>
allRowsSelected: boolean
}) => {
const inputRef = useRef<HTMLInputElement>(null)
const [isRowSelected, onRowSelectionChange] = useRowSelection()
const isIndeterminate = selectedUsers.size > 0 && !allRowsSelected
useEffect(() => {
if (inputRef.current) inputRef.current.indeterminate = isIndeterminate
}, [isIndeterminate])
return (
<div className="px-4 flex items-center sb-grid">
<div className="sb-grid-select-cell__header">
<input
ref={inputRef}
type="checkbox"
aria-label="Select-all"
className="sb-grid-select-cell__header__input"
disabled={false}
checked={isRowSelected}
onChange={(e) => onRowSelectionChange({ type: 'HEADER', checked: e.target.checked })}
/>
</div>
</div>
)
}
export const HeaderCell = ({
col,
specificFilterColumn,
setSortByValue,
}: {
col: any
specificFilterColumn: string
setSortByValue: (value: string) => void
}) => {
const ref = useRef<number>(0)
const [open, setOpen] = useState(false)
useEffect(() => {
ref.current = Number(new Date())
}, [open])
return (
<div className="flex items-center justify-between font-normal text-xs w-full">
<div className="flex items-center gap-x-2">
<p className="!text-foreground">{col.name}</p>
</div>
{specificFilterColumn === 'freeform' && ['created_at', 'email', 'phone'].includes(col.id) && (
<DropdownMenu
open={open}
onOpenChange={(val) => {
// [Joshen] This is a temp hack between the DropdownMenu and react data grid
// as the header cell is selectable, which takes the focus away from the dropdown menu
// causing it to close immediately.
if (val === false && Number(new Date()) - ref.current > 100) setOpen(val)
}}
>
<DropdownMenuTrigger asChild>
<Button
type="text"
icon={<ChevronDown />}
className="p-0 h-5 w-5"
onClick={() => setOpen(!open)}
/>
</DropdownMenuTrigger>
<DropdownMenuContent className="w-36">
<DropdownMenuItem
className="flex items-center gap-x-2"
onClick={() => {
setOpen(false)
setSortByValue(`${col.id}:desc`)
}}
>
<SortDesc size={14} />
Sort descending
</DropdownMenuItem>
<DropdownMenuItem
className="flex items-center gap-x-2"
onClick={() => {
setOpen(false)
setSortByValue(`${col.id}:asc`)
}}
>
<SortAsc size={14} />
Sort ascending
</DropdownMenuItem>
</DropdownMenuContent>
</DropdownMenu>
)}
</div>
)
}