Files
billai/web/src/routes/review/+page.svelte
clz 9d409d6a93 feat(analysis): 增强图表交互功能
- 分类支出排行: 饼图支持点击类别切换显示/隐藏,百分比动态重新计算
- 每日支出趋势: 图例支持点击切换类别显示,隐藏类别不参与堆叠计算
- Dialog列表: 添加列排序功能(时间/商家/描述/金额)
- Dialog列表: 添加分页功能,每页10条(分类)/8条(每日)
- 饼图hover效果: 扇形放大、阴影增强、中心显示详情
2026-01-08 02:55:54 +08:00

241 lines
8.7 KiB
Svelte

<script lang="ts">
import { page } from '$app/stores';
import { onMount } from 'svelte';
import { getReviewRecords, type ReviewRecord, type ReviewData } from '$lib/api';
import * as Card from '$lib/components/ui/card';
import { Button } from '$lib/components/ui/button';
import { Badge } from '$lib/components/ui/badge';
import { Input } from '$lib/components/ui/input';
import * as Table from '$lib/components/ui/table';
import Search from '@lucide/svelte/icons/search';
import Loader2 from '@lucide/svelte/icons/loader-2';
import AlertCircle from '@lucide/svelte/icons/alert-circle';
import CheckCircle from '@lucide/svelte/icons/check-circle';
import AlertTriangle from '@lucide/svelte/icons/alert-triangle';
import Clock from '@lucide/svelte/icons/clock';
import PartyPopper from '@lucide/svelte/icons/party-popper';
import FileText from '@lucide/svelte/icons/file-text';
let fileName = $state('');
let isLoading = $state(false);
let errorMessage = $state('');
let reviewData: ReviewData | null = $state(null);
let filterLevel = $state<'all' | 'HIGH' | 'LOW'>('all');
// 从 URL 获取文件名
onMount(() => {
const urlFileName = $page.url.searchParams.get('file');
if (urlFileName) {
fileName = urlFileName;
loadReviewData();
}
});
async function loadReviewData() {
if (!fileName) return;
isLoading = true;
errorMessage = '';
try {
const result = await getReviewRecords(fileName);
if (result.result && result.data) {
reviewData = result.data;
} else {
errorMessage = result.message || '获取数据失败';
}
} catch (err) {
errorMessage = err instanceof Error ? err.message : '网络错误';
} finally {
isLoading = false;
}
}
function handleSearch() {
loadReviewData();
}
// 过滤后的记录
let filteredRecords = $derived(
reviewData?.records.filter(r =>
filterLevel === 'all' || r.review_level === filterLevel
) || []
);
</script>
<svelte:head>
<title>复核记录 - BillAI</title>
</svelte:head>
<div class="space-y-6">
<!-- 页面标题 -->
<div>
<h1 class="text-2xl font-bold tracking-tight">复核记录</h1>
<p class="text-muted-foreground">系统无法确定分类的交易记录,需要人工复核</p>
</div>
<!-- 搜索栏 -->
<div class="flex gap-3">
<div class="relative flex-1">
<Search class="absolute left-3 top-1/2 h-4 w-4 -translate-y-1/2 text-muted-foreground" />
<Input
type="text"
placeholder="输入文件名..."
class="pl-10"
bind:value={fileName}
onkeydown={(e) => e.key === 'Enter' && handleSearch()}
/>
</div>
<Button onclick={handleSearch} disabled={isLoading}>
{#if isLoading}
<Loader2 class="mr-2 h-4 w-4 animate-spin" />
查询中
{:else}
<Search class="mr-2 h-4 w-4" />
查询
{/if}
</Button>
</div>
<!-- 错误提示 -->
{#if errorMessage}
<div class="flex items-center gap-2 rounded-lg border border-destructive/50 bg-destructive/10 p-3 text-sm text-destructive">
<AlertCircle class="h-4 w-4" />
{errorMessage}
</div>
{/if}
{#if reviewData}
<!-- 统计卡片 -->
<div class="grid gap-4 md:grid-cols-3">
<Card.Root>
<Card.Header class="flex flex-row items-center justify-between space-y-0 pb-2">
<Card.Title class="text-sm font-medium">待复核总数</Card.Title>
<Clock class="h-4 w-4 text-muted-foreground" />
</Card.Header>
<Card.Content>
<div class="text-2xl font-bold">{reviewData.total}</div>
<p class="text-xs text-muted-foreground">需要人工确认的记录</p>
</Card.Content>
</Card.Root>
<Card.Root class="border-red-200 dark:border-red-900">
<Card.Header class="flex flex-row items-center justify-between space-y-0 pb-2">
<Card.Title class="text-sm font-medium">高优先级</Card.Title>
<AlertTriangle class="h-4 w-4 text-red-500" />
</Card.Header>
<Card.Content>
<div class="text-2xl font-bold text-red-600 dark:text-red-400">{reviewData.high}</div>
<p class="text-xs text-muted-foreground">无法确定分类</p>
</Card.Content>
</Card.Root>
<Card.Root class="border-yellow-200 dark:border-yellow-900">
<Card.Header class="flex flex-row items-center justify-between space-y-0 pb-2">
<Card.Title class="text-sm font-medium">低优先级</Card.Title>
<AlertCircle class="h-4 w-4 text-yellow-500" />
</Card.Header>
<Card.Content>
<div class="text-2xl font-bold text-yellow-600 dark:text-yellow-400">{reviewData.low}</div>
<p class="text-xs text-muted-foreground">分类可能有变更</p>
</Card.Content>
</Card.Root>
</div>
<!-- 过滤器和表格 -->
<Card.Root>
<Card.Header>
<div class="flex items-center justify-between">
<Card.Title>记录列表</Card.Title>
<div class="flex gap-2">
<Button
variant={filterLevel === 'all' ? 'default' : 'outline'}
size="sm"
onclick={() => filterLevel = 'all'}
>
全部 ({reviewData.total})
</Button>
<Button
variant={filterLevel === 'HIGH' ? 'destructive' : 'outline'}
size="sm"
onclick={() => filterLevel = 'HIGH'}
>
高优先级 ({reviewData.high})
</Button>
<Button
variant={filterLevel === 'LOW' ? 'secondary' : 'outline'}
size="sm"
onclick={() => filterLevel = 'LOW'}
>
低优先级 ({reviewData.low})
</Button>
</div>
</div>
</Card.Header>
<Card.Content>
{#if filteredRecords.length > 0}
<div class="rounded-md border">
<Table.Root>
<Table.Header>
<Table.Row>
<Table.Head class="w-[160px]">时间</Table.Head>
<Table.Head>分类</Table.Head>
<Table.Head>交易对方</Table.Head>
<Table.Head>商品说明</Table.Head>
<Table.Head>收/支</Table.Head>
<Table.Head class="text-right">金额</Table.Head>
<Table.Head class="w-[80px]">等级</Table.Head>
</Table.Row>
</Table.Header>
<Table.Body>
{#each filteredRecords as record}
<Table.Row>
<Table.Cell class="text-muted-foreground text-sm">
{record.time}
</Table.Cell>
<Table.Cell>
<Badge variant="secondary">{record.category}</Badge>
</Table.Cell>
<Table.Cell class="max-w-[200px] truncate" title={record.merchant}>
{record.merchant}
</Table.Cell>
<Table.Cell class="max-w-[200px] truncate text-muted-foreground" title={record.description}>
{record.description || '-'}
</Table.Cell>
<Table.Cell>
<span class={record.income_expense === '支出' ? 'text-red-500' : 'text-green-500'}>
{record.income_expense}
</span>
</Table.Cell>
<Table.Cell class="text-right font-mono font-medium">
¥{record.amount}
</Table.Cell>
<Table.Cell>
<Badge variant={record.review_level === 'HIGH' ? 'destructive' : 'outline'}>
{record.review_level}
</Badge>
</Table.Cell>
</Table.Row>
{/each}
</Table.Body>
</Table.Root>
</div>
{:else}
<div class="flex flex-col items-center justify-center py-12 text-center">
<PartyPopper class="h-12 w-12 text-muted-foreground mb-4" />
<p class="text-muted-foreground">没有需要复核的记录</p>
</div>
{/if}
</Card.Content>
</Card.Root>
{:else if !isLoading && !errorMessage}
<Card.Root>
<Card.Content class="flex flex-col items-center justify-center py-16">
<FileText class="h-16 w-16 text-muted-foreground mb-4" />
<p class="text-lg font-medium">输入文件名查询复核记录</p>
<p class="text-sm text-muted-foreground">上传账单后可在此查看需要复核的交易</p>
</Card.Content>
</Card.Root>
{/if}
</div>