144 lines
4.1 KiB
TypeScript
144 lines
4.1 KiB
TypeScript
'use client';
|
|
|
|
import { ResponsiveContainer, LineChart, Line, XAxis, YAxis, Tooltip, CartesianGrid } from 'recharts';
|
|
import type { Locale } from '@/lib/locale';
|
|
|
|
interface DailyData {
|
|
date: string;
|
|
amount: number;
|
|
count: number;
|
|
}
|
|
|
|
interface DailyChartProps {
|
|
data: DailyData[];
|
|
dark?: boolean;
|
|
locale?: Locale;
|
|
}
|
|
|
|
function formatDate(dateStr: string) {
|
|
const [, m, d] = dateStr.split('-');
|
|
return `${m}/${d}`;
|
|
}
|
|
|
|
function formatAmount(value: number) {
|
|
if (value >= 10000) return `¥${(value / 10000).toFixed(1)}w`;
|
|
if (value >= 1000) return `¥${(value / 1000).toFixed(1)}k`;
|
|
return `¥${value}`;
|
|
}
|
|
|
|
interface TooltipPayload {
|
|
value: number;
|
|
dataKey: string;
|
|
}
|
|
|
|
function CustomTooltip({
|
|
active,
|
|
payload,
|
|
label,
|
|
dark,
|
|
currency,
|
|
amountLabel,
|
|
countLabel,
|
|
}: {
|
|
active?: boolean;
|
|
payload?: TooltipPayload[];
|
|
label?: string;
|
|
dark?: boolean;
|
|
currency: string;
|
|
amountLabel: string;
|
|
countLabel: string;
|
|
}) {
|
|
if (!active || !payload?.length) return null;
|
|
return (
|
|
<div
|
|
className={[
|
|
'rounded-lg border px-3 py-2 text-sm shadow-lg',
|
|
dark ? 'border-slate-600 bg-slate-800 text-slate-200' : 'border-slate-200 bg-white text-slate-800',
|
|
].join(' ')}
|
|
>
|
|
<p className={['mb-1 text-xs', dark ? 'text-slate-400' : 'text-slate-500'].join(' ')}>{label}</p>
|
|
{payload.map((p) => (
|
|
<p key={p.dataKey}>
|
|
{p.dataKey === 'amount' ? amountLabel : countLabel}:{' '}
|
|
{p.dataKey === 'amount' ? `${currency}${p.value.toLocaleString()}` : p.value}
|
|
</p>
|
|
))}
|
|
</div>
|
|
);
|
|
}
|
|
|
|
export default function DailyChart({ data, dark, locale = 'zh' }: DailyChartProps) {
|
|
const currency = locale === 'en' ? '$' : '¥';
|
|
const chartTitle = locale === 'en' ? 'Daily Recharge Trend' : '每日充值趋势';
|
|
const emptyText = locale === 'en' ? 'No data' : '暂无数据';
|
|
const amountLabel = locale === 'en' ? 'Amount' : '金额';
|
|
const countLabel = locale === 'en' ? 'Orders' : '笔数';
|
|
const tickInterval = data.length > 30 ? Math.ceil(data.length / 12) - 1 : 0;
|
|
if (data.length === 0) {
|
|
return (
|
|
<div
|
|
className={[
|
|
'rounded-xl border p-6',
|
|
dark ? 'border-slate-700 bg-slate-800/60' : 'border-slate-200 bg-white shadow-sm',
|
|
].join(' ')}
|
|
>
|
|
<h3 className={['mb-4 text-sm font-semibold', dark ? 'text-slate-200' : 'text-slate-800'].join(' ')}>
|
|
{chartTitle}
|
|
</h3>
|
|
<p className={['text-center text-sm py-16', dark ? 'text-slate-500' : 'text-gray-400'].join(' ')}>
|
|
{emptyText}
|
|
</p>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
const axisColor = dark ? '#64748b' : '#94a3b8';
|
|
const gridColor = dark ? '#334155' : '#e2e8f0';
|
|
|
|
return (
|
|
<div
|
|
className={[
|
|
'rounded-xl border p-6',
|
|
dark ? 'border-slate-700 bg-slate-800/60' : 'border-slate-200 bg-white shadow-sm',
|
|
].join(' ')}
|
|
>
|
|
<h3 className={['mb-4 text-sm font-semibold', dark ? 'text-slate-200' : 'text-slate-800'].join(' ')}>
|
|
{chartTitle}
|
|
</h3>
|
|
<ResponsiveContainer width="100%" height={320}>
|
|
<LineChart data={data} margin={{ top: 5, right: 20, bottom: 5, left: 10 }}>
|
|
<CartesianGrid stroke={gridColor} strokeDasharray="3 3" />
|
|
<XAxis
|
|
dataKey="date"
|
|
tickFormatter={formatDate}
|
|
tick={{ fill: axisColor, fontSize: 12 }}
|
|
axisLine={{ stroke: gridColor }}
|
|
tickLine={false}
|
|
interval={tickInterval}
|
|
/>
|
|
<YAxis
|
|
tickFormatter={formatAmount}
|
|
tick={{ fill: axisColor, fontSize: 12 }}
|
|
axisLine={{ stroke: gridColor }}
|
|
tickLine={false}
|
|
width={60}
|
|
/>
|
|
<Tooltip
|
|
content={
|
|
<CustomTooltip dark={dark} currency={currency} amountLabel={amountLabel} countLabel={countLabel} />
|
|
}
|
|
/>
|
|
<Line
|
|
type="monotone"
|
|
dataKey="amount"
|
|
stroke={dark ? '#818cf8' : '#4f46e5'}
|
|
strokeWidth={2}
|
|
dot={{ r: 3, fill: dark ? '#818cf8' : '#4f46e5' }}
|
|
activeDot={{ r: 5 }}
|
|
/>
|
|
</LineChart>
|
|
</ResponsiveContainer>
|
|
</div>
|
|
);
|
|
}
|