6f415428d3
* refactor(web): centralize data table implementation - route all TanStack table setup through a shared data-table hook to remove repeated state and row model wiring. - move table rendering, static table wrappers, empty states, and primitive exports behind the data-table module. - update feature tables and configuration editors to share the same table UX while preserving their existing workflows. * refactor(web): trim data table public API - remove unused data-table exports and dead static table helper types. - keep internal table header, skeleton, empty state, and faceted filter helpers private to the data-table module. - route feature imports through the data-table barrel to avoid subpath coupling. * refactor(web): unify table rendering components - centralize static table headers, bodies, empty states, and shared class names behind the data-table package. - migrate settings, pricing, channel, key, subscription, and model tables to the shared table APIs. - remove data-table exports for low-level table primitives so feature code uses one supported abstraction. * perf(web): keep list tables fixed within page content - make shared data table pages fill available height and scroll row data inside the table body. - add a fixed content layout mode so selected list pages avoid page-level scrolling. - apply the fixed table behavior to keys, logs, channels, models, users, redemptions, and subscriptions. * perf(web): refine table pagination controls - show total row counts instead of redundant page range text. - tighten visible page buttons so pagination fits constrained table widths. - align pagination controls and tune text hierarchy for clearer scanning. * perf(web): stabilize model pricing table columns - keep model pricing columns at fixed widths so headers do not collapse in narrow layouts. - truncate long model names and pricing summaries within their cells instead of squeezing adjacent columns. * refactor(web): simplify data table rendering internals - split table body rendering into focused helpers for loading, empty, and row states. - extract static table row and cell class resolution to reduce branching in the main component. - reuse a single pagination page-size option list to avoid duplicated constants. * perf(pricing): reduce dynamic pricing table render work - reuse dynamic pricing field metadata instead of rebuilding it inside table columns. - precompute formatted dynamic prices per tier and group to avoid repeated entry mapping for each cell. - simplify select option construction in related dialogs while preserving the same choices. * refactor(web): streamline pricing table rendering - reuse translated endpoint select options between trigger data and menu items. - precompute dynamic pricing maps per group so table cells only resolve formatted values. - add local dynamic pricing type aliases to keep helper signatures readable. * refactor(web): merge pricing table imports * refactor(web): merge upstream ratio table imports * refactor(web): merge channel selector table imports * refactor(web): simplify tiered pricing select items * refactor(web): reuse model ratio row state * refactor(web): rely on table view row defaults * refactor(web): reuse pagination state values * refactor(web): hoist pagination size select items * refactor(web): clarify static table body rows * refactor(web): extract table page pagination rendering * fix(web): remove direct hast type dependency - rely on Shiki transformer contextual typing for line nodes. - allow frontend typecheck to pass without an undeclared hast package. * refactor(web): trim data table hook return API - return only the TanStack table instance from useDataTable. - keep internal state handling private because callers do not consume it directly. * refactor(web): keep static table empty row private - stop exporting the internal StaticDataTableEmptyRow helper. - keep the public static table API focused on the table component and column type. * refactor(web): hide data table view props from barrel * refactor(web): remove stale long text lint override * fix(web): keep pinned table columns opaque - apply pinned column background classes after custom column classes. - use an opaque hover background so scrolled content cannot show through fixed cells. * refactor(data-table): organize shared table components - group table primitives, page composition, toolbar controls, static tables, and hooks by responsibility. - split shared view types, row rendering, header rendering, and pinned-column styling out of the main table view. - keep the public data-table barrel stable while documenting the new ownership boundaries. * fix(web): stabilize split table column sizing - derive default colgroup widths from visible columns when split headers or header sizing are enabled. - apply a fixed table layout with computed minimum width so header and body columns stay aligned. - keep split-header containers from leaking horizontal overflow and avoid extra pinned-column borders. * fix(web): set stable table utility column widths - assign fixed widths to selection columns so shared colgroup sizing keeps checkbox cells compact. - size id columns in redemption and user tables to keep split headers aligned with body rows. * fix(web): align model metadata icon cells - render compact provider avatars in the metadata icon column instead of wide wordmarks. - position icons in a fixed-size wrapper so they line up with the existing icon header alignment. * fix(status-badge): hide status dot by default * fix(web): prevent user invite info overlap - give the invite info and created-at columns explicit widths so table sizing reserves enough space. - allow invite badges to wrap within the cell instead of spilling into adjacent columns. * perf(data-table): cache pinned column class resolution - reuse the pinned column lookup while table props stay stable to reduce repeated per-render work. - share the resolved column class handler across unified and split-header table layouts. - localize page-number screen reader labels so pagination remains accessible in every locale. * refactor(data-table): tighten static table modes - make StaticDataTable distinguish data-driven and children-only usage through explicit prop shapes. - remove unsupported columns-without-data fallback after confirming no repository callers rely on it. - default manual table modes away from unused local row models to reduce repeated table work. * fix(data-table): make pinned edit column opaque - use an opaque muted background for the active action column so sticky cells do not reveal scrolled content underneath. * fix(data-table): prevent narrow column overlap - apply stable header sizing to remaining desktop data table pages so constrained layouts scroll instead of compressing cells. - add explicit widths for key, quota, badge, and timestamp columns that contain fixed-format content. - constrain masked values and timestamp cells with truncation to keep content inside its assigned column. * fix(table): align table cell content with headers - remove extra inline padding from masked table text buttons so values start at the cell edge. - tag status badges and offset leading badges inside table cells to match header text alignment. * fix(table): prevent admin list column overflow - widen redemption and subscription table columns so masked codes, timestamps, and localized headers fit. - localize subscription ID headers and add Received amount translations across supported locales. * fix(provider-badge): unify provider icon spacing - add a shared provider badge component for icon and status label layout. - reuse it in channel type and model vendor columns so OpenAI icons align consistently.
233 lines
7.5 KiB
TypeScript
Vendored
233 lines
7.5 KiB
TypeScript
Vendored
/*
|
|
Copyright (C) 2023-2026 QuantumNous
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU Affero General Public License as
|
|
published by the Free Software Foundation, either version 3 of the
|
|
License, or (at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU Affero General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Affero General Public License
|
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
For commercial licensing, please contact support@quantumnous.com
|
|
*/
|
|
import { useState, useEffect, useRef } from 'react'
|
|
import { type Table } from '@tanstack/react-table'
|
|
import { X } from 'lucide-react'
|
|
import { useTranslation } from 'react-i18next'
|
|
import { cn } from '@/lib/utils'
|
|
import { Badge } from '@/components/ui/badge'
|
|
import { Button } from '@/components/ui/button'
|
|
import { Separator } from '@/components/ui/separator'
|
|
import {
|
|
Tooltip,
|
|
TooltipContent,
|
|
TooltipTrigger,
|
|
} from '@/components/ui/tooltip'
|
|
|
|
type DataTableBulkActionsProps<TData> = {
|
|
table: Table<TData>
|
|
entityName: string
|
|
children: React.ReactNode
|
|
}
|
|
|
|
/**
|
|
* A modular toolbar for displaying bulk actions when table rows are selected.
|
|
*
|
|
* @template TData The type of data in the table.
|
|
* @param {object} props The component props.
|
|
* @param {Table<TData>} props.table The react-table instance.
|
|
* @param {string} props.entityName The name of the entity being acted upon (e.g., "task", "user").
|
|
* @param {React.ReactNode} props.children The action buttons to be rendered inside the toolbar.
|
|
* @returns {React.ReactNode | null} The rendered component or null if no rows are selected.
|
|
*/
|
|
export function DataTableBulkActions<TData>({
|
|
table,
|
|
entityName,
|
|
children,
|
|
}: DataTableBulkActionsProps<TData>): React.ReactNode | null {
|
|
const { t } = useTranslation()
|
|
const selectedRows = table.getFilteredSelectedRowModel().rows
|
|
const selectedCount = selectedRows.length
|
|
const toolbarRef = useRef<HTMLDivElement>(null)
|
|
const [announcement, setAnnouncement] = useState('')
|
|
|
|
// Announce selection changes to screen readers
|
|
useEffect(() => {
|
|
if (selectedCount > 0) {
|
|
const message = `${selectedCount} ${entityName}${selectedCount > 1 ? 's' : ''} selected. Bulk actions toolbar is available.`
|
|
// eslint-disable-next-line react-hooks/set-state-in-effect
|
|
setAnnouncement(message)
|
|
|
|
// Clear announcement after a delay
|
|
const timer = setTimeout(() => setAnnouncement(''), 3000)
|
|
return () => clearTimeout(timer)
|
|
}
|
|
}, [selectedCount, entityName])
|
|
|
|
const handleClearSelection = () => {
|
|
table.resetRowSelection()
|
|
}
|
|
|
|
const handleKeyDown = (event: React.KeyboardEvent) => {
|
|
const buttons = toolbarRef.current?.querySelectorAll('button')
|
|
if (!buttons) return
|
|
|
|
const currentIndex = Array.from(buttons).findIndex(
|
|
(button) => button === document.activeElement
|
|
)
|
|
|
|
switch (event.key) {
|
|
case 'ArrowRight': {
|
|
event.preventDefault()
|
|
const nextIndex = (currentIndex + 1) % buttons.length
|
|
buttons[nextIndex]?.focus()
|
|
break
|
|
}
|
|
case 'ArrowLeft': {
|
|
event.preventDefault()
|
|
const prevIndex =
|
|
currentIndex === 0 ? buttons.length - 1 : currentIndex - 1
|
|
buttons[prevIndex]?.focus()
|
|
break
|
|
}
|
|
case 'Home':
|
|
event.preventDefault()
|
|
buttons[0]?.focus()
|
|
break
|
|
case 'End':
|
|
event.preventDefault()
|
|
buttons[buttons.length - 1]?.focus()
|
|
break
|
|
case 'Escape': {
|
|
// Check if the Escape key came from a dropdown trigger or content
|
|
// We can't check dropdown state because the menu closes before our handler runs.
|
|
const target = event.target as HTMLElement
|
|
const activeElement = document.activeElement as HTMLElement
|
|
|
|
// Check if the event target or currently focused element is a dropdown trigger
|
|
const isFromDropdownTrigger =
|
|
target?.getAttribute('data-slot') === 'dropdown-menu-trigger' ||
|
|
activeElement?.getAttribute('data-slot') ===
|
|
'dropdown-menu-trigger' ||
|
|
target?.closest('[data-slot="dropdown-menu-trigger"]') ||
|
|
activeElement?.closest('[data-slot="dropdown-menu-trigger"]')
|
|
|
|
// Check if the focused element is inside dropdown content (which is portaled)
|
|
const isFromDropdownContent =
|
|
activeElement?.closest('[data-slot="dropdown-menu-content"]') ||
|
|
target?.closest('[data-slot="dropdown-menu-content"]')
|
|
|
|
if (isFromDropdownTrigger || isFromDropdownContent) {
|
|
// Escape was meant for the dropdown - don't clear selection
|
|
return
|
|
}
|
|
|
|
// Escape was meant for the toolbar - clear selection
|
|
event.preventDefault()
|
|
handleClearSelection()
|
|
break
|
|
}
|
|
}
|
|
}
|
|
|
|
if (selectedCount === 0) {
|
|
return null
|
|
}
|
|
|
|
return (
|
|
<>
|
|
{/* Live region for screen reader announcements */}
|
|
<div
|
|
aria-live='polite'
|
|
aria-atomic='true'
|
|
className='sr-only'
|
|
role='status'
|
|
>
|
|
{announcement}
|
|
</div>
|
|
|
|
<div
|
|
ref={toolbarRef}
|
|
role='toolbar'
|
|
aria-label={`Bulk actions for ${selectedCount} selected ${entityName}${selectedCount > 1 ? 's' : ''}`}
|
|
aria-describedby='bulk-actions-description'
|
|
tabIndex={-1}
|
|
onKeyDown={handleKeyDown}
|
|
className={cn(
|
|
'fixed bottom-6 left-1/2 z-50 -translate-x-1/2 rounded-xl',
|
|
'transition-all delay-100 duration-300 ease-out hover:scale-105',
|
|
'focus-visible:ring-ring/50 focus-visible:ring-2 focus-visible:outline-none'
|
|
)}
|
|
>
|
|
<div
|
|
className={cn(
|
|
'p-2 shadow-xl',
|
|
'rounded-xl border',
|
|
'bg-background/95 supports-[backdrop-filter]:bg-background/60 backdrop-blur-lg',
|
|
'flex items-center gap-x-2'
|
|
)}
|
|
>
|
|
<Tooltip>
|
|
<TooltipTrigger
|
|
render={
|
|
<Button
|
|
variant='outline'
|
|
size='icon'
|
|
onClick={handleClearSelection}
|
|
className='size-6'
|
|
aria-label={t('Clear selection')}
|
|
title={t('Clear selection (Escape)')}
|
|
/>
|
|
}
|
|
>
|
|
<X />
|
|
<span className='sr-only'>{t('Clear selection')}</span>
|
|
</TooltipTrigger>
|
|
<TooltipContent>
|
|
<p>{t('Clear selection (Escape)')}</p>
|
|
</TooltipContent>
|
|
</Tooltip>
|
|
|
|
<Separator
|
|
className='h-5'
|
|
orientation='vertical'
|
|
aria-hidden='true'
|
|
/>
|
|
|
|
<div
|
|
className='flex items-center gap-x-1 text-sm'
|
|
id='bulk-actions-description'
|
|
>
|
|
<Badge
|
|
variant='default'
|
|
className='min-w-8 rounded-lg'
|
|
aria-label={`${selectedCount} selected`}
|
|
>
|
|
{selectedCount}
|
|
</Badge>{' '}
|
|
<span className='hidden sm:inline'>
|
|
{entityName}
|
|
{selectedCount > 1 ? 's' : ''}
|
|
</span>{' '}
|
|
{t('selected')}
|
|
</div>
|
|
|
|
<Separator
|
|
className='h-5'
|
|
orientation='vertical'
|
|
aria-hidden='true'
|
|
/>
|
|
|
|
{children}
|
|
</div>
|
|
</div>
|
|
</>
|
|
)
|
|
}
|