Refresh the overview page with an actionable Get Started guide, live API request details, real usage sparklines, and OpenAI-inspired dashboard panels. Add collapsible setup state, role-aware quick actions, and localized copy so returning users can focus on platform health.
97 lines
2.4 KiB
TypeScript
Vendored
97 lines
2.4 KiB
TypeScript
Vendored
import { type ReactNode } from 'react'
|
|
import { useTranslation } from 'react-i18next'
|
|
import { cn } from '@/lib/utils'
|
|
import { Skeleton } from '@/components/ui/skeleton'
|
|
|
|
interface PanelWrapperProps {
|
|
title: ReactNode
|
|
description?: ReactNode
|
|
loading?: boolean
|
|
empty?: boolean
|
|
emptyMessage?: string
|
|
height?: string
|
|
className?: string
|
|
contentClassName?: string
|
|
headerActions?: ReactNode
|
|
children?: ReactNode
|
|
}
|
|
|
|
function PanelHeader(props: {
|
|
title: ReactNode
|
|
description?: ReactNode
|
|
actions?: ReactNode
|
|
}) {
|
|
const heading = (
|
|
<div className='flex flex-col gap-1'>
|
|
<div className='text-sm font-semibold'>{props.title}</div>
|
|
{props.description != null && (
|
|
<div className='text-muted-foreground text-xs'>{props.description}</div>
|
|
)}
|
|
</div>
|
|
)
|
|
|
|
return (
|
|
<div className='border-b px-4 py-3 sm:px-5'>
|
|
{props.actions != null ? (
|
|
<div className='flex items-start justify-between gap-2'>
|
|
{heading}
|
|
{props.actions}
|
|
</div>
|
|
) : (
|
|
heading
|
|
)}
|
|
</div>
|
|
)
|
|
}
|
|
|
|
export function PanelWrapper(props: PanelWrapperProps) {
|
|
const { t } = useTranslation()
|
|
const resolvedEmptyMessage = props.emptyMessage ?? t('No data available')
|
|
const height = props.height ?? 'h-64'
|
|
const frameClassName = cn(
|
|
'overflow-hidden rounded-2xl border bg-card shadow-xs',
|
|
props.className
|
|
)
|
|
|
|
if (props.loading) {
|
|
return (
|
|
<div className={frameClassName}>
|
|
<PanelHeader title={props.title} description={props.description} />
|
|
<div className={cn('p-4 sm:p-5', props.contentClassName)}>
|
|
<Skeleton className={`w-full ${height}`} />
|
|
</div>
|
|
</div>
|
|
)
|
|
}
|
|
|
|
if (props.empty) {
|
|
return (
|
|
<div className={frameClassName}>
|
|
<PanelHeader title={props.title} description={props.description} />
|
|
<div
|
|
className={cn(
|
|
'text-muted-foreground flex items-center justify-center px-4 text-sm',
|
|
height,
|
|
props.contentClassName
|
|
)}
|
|
>
|
|
{resolvedEmptyMessage}
|
|
</div>
|
|
</div>
|
|
)
|
|
}
|
|
|
|
return (
|
|
<div className={frameClassName}>
|
|
<PanelHeader
|
|
title={props.title}
|
|
description={props.description}
|
|
actions={props.headerActions}
|
|
/>
|
|
<div className={cn('p-4 sm:p-5', props.contentClassName)}>
|
|
{props.children}
|
|
</div>
|
|
</div>
|
|
)
|
|
}
|