Files
chaos-api/web/default/src/features/redemption-codes/components/data-table-bulk-actions.tsx
T
t0ng7u d146e45e2f ⚖️ chore(web/default): add reusable copyright header tooling
Add a Bun script to apply and normalize AGPL copyright headers across the default frontend source files.

The script keeps headers idempotent, upgrades existing headers to the 2023-2026 QuantumNous range, and is exposed through `bun run copyright` for future maintenance.
2026-05-09 11:35:07 +08:00

138 lines
4.3 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, useMemo } from 'react'
import { type Table } from '@tanstack/react-table'
import { Trash2 } from 'lucide-react'
import { useTranslation } from 'react-i18next'
import { toast } from 'sonner'
import { Button } from '@/components/ui/button'
import {
Tooltip,
TooltipContent,
TooltipTrigger,
} from '@/components/ui/tooltip'
import { ConfirmDialog } from '@/components/confirm-dialog'
import { CopyButton } from '@/components/copy-button'
import { DataTableBulkActions as BulkActionsToolbar } from '@/components/data-table'
import { deleteInvalidRedemptions } from '../api'
import { type Redemption } from '../types'
import { useRedemptions } from './redemptions-provider'
type DataTableBulkActionsProps<TData> = {
table: Table<TData>
}
export function DataTableBulkActions<TData>({
table,
}: DataTableBulkActionsProps<TData>) {
const { t } = useTranslation()
const { triggerRefresh } = useRedemptions()
const [showDeleteInvalidConfirm, setShowDeleteInvalidConfirm] =
useState(false)
const [isDeleting, setIsDeleting] = useState(false)
const selectedRows = table.getFilteredSelectedRowModel().rows
const contentToCopy = useMemo(() => {
const selectedCodes = selectedRows.map((row) => {
const redemption = row.original as Redemption
return `${redemption.name}\t${redemption.key}`
})
return selectedCodes.join('\n')
}, [selectedRows])
const handleDeleteInvalid = async () => {
setIsDeleting(true)
try {
const result = await deleteInvalidRedemptions()
if (result.success) {
const count = result.data || 0
toast.success(
t('Successfully deleted {{count}} invalid redemption codes', {
count,
})
)
table.resetRowSelection()
triggerRefresh()
setShowDeleteInvalidConfirm(false)
}
} finally {
setIsDeleting(false)
}
}
return (
<>
<BulkActionsToolbar table={table} entityName={t('redemption code')}>
<CopyButton
value={contentToCopy}
variant='outline'
size='icon'
className='size-8'
tooltip={t('Copy selected codes')}
successTooltip={t('Codes copied!')}
aria-label={t('Copy selected codes')}
/>
<Tooltip>
<TooltipTrigger
render={
<Button
variant='destructive'
size='icon'
onClick={() => setShowDeleteInvalidConfirm(true)}
className='size-8'
aria-label={t('Delete invalid redemption codes')}
title={t('Delete invalid redemption codes')}
/>
}
>
<Trash2 />
<span className='sr-only'>{t('Delete invalid codes')}</span>
</TooltipTrigger>
<TooltipContent>
<p>{t('Delete invalid codes (used/disabled/expired)')}</p>
</TooltipContent>
</Tooltip>
</BulkActionsToolbar>
<ConfirmDialog
destructive
open={showDeleteInvalidConfirm}
onOpenChange={setShowDeleteInvalidConfirm}
handleConfirm={handleDeleteInvalid}
isLoading={isDeleting}
className='max-w-md'
title={t('Delete Invalid Redemption Codes?')}
desc={
<>
{t('This will delete all')} <strong>{t('used')}</strong>,{' '}
<strong>{t('disabled')}</strong>
{t(', and')} <strong>{t('expired')}</strong>{' '}
{t('redemption codes.')}
<br />
{t('This action cannot be undone.')}
</>
}
confirmText={t('Delete Invalid')}
/>
</>
)
}