new-api/web/default/src/components/empty-state.tsx
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

69 lines
2.0 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 type { ReactNode } from 'react'
import { Database, type LucideIcon } from 'lucide-react'
import { useTranslation } from 'react-i18next'
import { cn } from '@/lib/utils'
import {
Empty,
EmptyContent,
EmptyDescription,
EmptyHeader,
EmptyMedia,
EmptyTitle,
} from '@/components/ui/empty'
import { FadeIn } from '@/components/page-transition'
interface EmptyStateProps {
icon?: LucideIcon
title?: string
description?: string
action?: ReactNode
className?: string
bordered?: boolean
}
export function EmptyState(props: EmptyStateProps) {
const { t } = useTranslation()
const Icon = props.icon ?? Database
return (
<FadeIn>
<Empty
className={cn(
'min-h-[300px]',
props.bordered && 'border',
props.className
)}
>
<EmptyHeader>
<EmptyMedia variant='icon'>
<Icon className='size-6' />
</EmptyMedia>
<EmptyTitle>{props.title ?? t('No Data')}</EmptyTitle>
{props.description != null && (
<EmptyDescription>{props.description}</EmptyDescription>
)}
</EmptyHeader>
{props.action != null && <EmptyContent>{props.action}</EmptyContent>}
</Empty>
</FadeIn>
)
}