2026-01-28 17:00:58 +08:00
|
|
|
|
<template>
|
2025-12-27 21:15:26 +08:00
|
|
|
|
<div class="page-container-flex">
|
2025-12-26 17:13:57 +08:00
|
|
|
|
<!-- 顶部导航栏 -->
|
2026-01-16 11:15:44 +08:00
|
|
|
|
<van-nav-bar
|
|
|
|
|
|
placeholder
|
|
|
|
|
|
>
|
2026-01-16 15:56:53 +08:00
|
|
|
|
<template #title>
|
|
|
|
|
|
<div
|
|
|
|
|
|
class="nav-date-picker"
|
|
|
|
|
|
@click="showMonthPicker = true"
|
|
|
|
|
|
>
|
2026-01-28 17:00:58 +08:00
|
|
|
|
<span>{{ currentMonth === 0 ? `${currentYear}年` : `${currentYear}年${currentMonth}月` }}</span>
|
2026-01-16 15:56:53 +08:00
|
|
|
|
<van-icon name="arrow-down" />
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</template>
|
2025-12-26 17:13:57 +08:00
|
|
|
|
<template #right>
|
2026-02-09 19:25:51 +08:00
|
|
|
|
<div class="nav-right-buttons">
|
|
|
|
|
|
<van-icon
|
|
|
|
|
|
name="upgrade"
|
|
|
|
|
|
size="18"
|
|
|
|
|
|
style="margin-right: 12px;"
|
|
|
|
|
|
@click="goToStatisticsV2"
|
|
|
|
|
|
/>
|
|
|
|
|
|
<van-icon
|
|
|
|
|
|
name="chat-o"
|
|
|
|
|
|
size="20"
|
|
|
|
|
|
@click="goToAnalysis"
|
|
|
|
|
|
/>
|
|
|
|
|
|
</div>
|
2025-12-26 17:13:57 +08:00
|
|
|
|
</template>
|
|
|
|
|
|
</van-nav-bar>
|
|
|
|
|
|
|
|
|
|
|
|
<!-- 下拉刷新 -->
|
2026-01-16 11:15:44 +08:00
|
|
|
|
<van-pull-refresh
|
|
|
|
|
|
v-model="refreshing"
|
|
|
|
|
|
@refresh="onRefresh"
|
|
|
|
|
|
>
|
2026-01-11 16:33:55 +08:00
|
|
|
|
<!-- 初始加载中 -->
|
2026-01-16 11:15:44 +08:00
|
|
|
|
<van-loading
|
|
|
|
|
|
v-if="loading && firstLoading"
|
|
|
|
|
|
vertical
|
|
|
|
|
|
style="padding: 100px 0"
|
|
|
|
|
|
>
|
2025-12-26 17:13:57 +08:00
|
|
|
|
加载统计数据中...
|
|
|
|
|
|
</van-loading>
|
|
|
|
|
|
|
2026-01-16 15:56:53 +08:00
|
|
|
|
<!-- 统计内容(可滚动部分) -->
|
2026-01-16 11:15:44 +08:00
|
|
|
|
<div
|
|
|
|
|
|
v-if="!firstLoading"
|
2026-01-16 15:56:53 +08:00
|
|
|
|
class="statistics-content"
|
2026-01-16 11:15:44 +08:00
|
|
|
|
>
|
2026-01-16 15:56:53 +08:00
|
|
|
|
<div>
|
2026-01-21 16:09:38 +08:00
|
|
|
|
<!-- 余额卡片 -->
|
|
|
|
|
|
<div
|
|
|
|
|
|
class="common-card"
|
|
|
|
|
|
style="margin-top: 12px;"
|
|
|
|
|
|
>
|
|
|
|
|
|
<div
|
|
|
|
|
|
class="card-header"
|
|
|
|
|
|
style="padding-bottom: 0;"
|
|
|
|
|
|
>
|
|
|
|
|
|
<h3 class="card-title">
|
|
|
|
|
|
余额
|
|
|
|
|
|
</h3>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<!-- 余额金额 -->
|
|
|
|
|
|
<div class="balance-amount">
|
|
|
|
|
|
<span
|
|
|
|
|
|
class="balance-value"
|
|
|
|
|
|
:class="{ 'balance-positive': displayBalance >= 0, 'balance-negative': displayBalance < 0 }"
|
|
|
|
|
|
>
|
|
|
|
|
|
¥{{ formatMoney(displayBalance) }}
|
|
|
|
|
|
</span>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
2026-02-01 10:27:04 +08:00
|
|
|
|
<!-- 余额变化图表(融合收支趋势) -->
|
2026-01-21 16:09:38 +08:00
|
|
|
|
<div
|
|
|
|
|
|
class="balance-chart"
|
2026-02-01 10:27:04 +08:00
|
|
|
|
style="height: 190px; padding: 0"
|
2026-01-21 16:09:38 +08:00
|
|
|
|
>
|
|
|
|
|
|
<div
|
|
|
|
|
|
ref="balanceChartRef"
|
|
|
|
|
|
style="width: 100%; height: 100%"
|
|
|
|
|
|
/>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
2026-01-16 15:56:53 +08:00
|
|
|
|
<!-- 分类统计 -->
|
|
|
|
|
|
<div
|
|
|
|
|
|
class="common-card"
|
|
|
|
|
|
style="padding-bottom: 10px;"
|
|
|
|
|
|
>
|
|
|
|
|
|
<div class="card-header">
|
|
|
|
|
|
<h3 class="card-title">
|
|
|
|
|
|
支出分类
|
|
|
|
|
|
</h3>
|
|
|
|
|
|
<van-tag
|
|
|
|
|
|
type="primary"
|
|
|
|
|
|
size="medium"
|
2026-01-16 11:15:44 +08:00
|
|
|
|
>
|
2026-01-16 15:56:53 +08:00
|
|
|
|
{{ expenseCategoriesView.length }}类
|
|
|
|
|
|
</van-tag>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<!-- 环形图区域 -->
|
|
|
|
|
|
<div
|
|
|
|
|
|
v-if="expenseCategoriesView.length > 0"
|
|
|
|
|
|
class="chart-container"
|
|
|
|
|
|
>
|
|
|
|
|
|
<div class="ring-chart">
|
|
|
|
|
|
<div
|
|
|
|
|
|
ref="pieChartRef"
|
|
|
|
|
|
style="width: 100%; height: 100%"
|
|
|
|
|
|
/>
|
2026-01-11 16:33:55 +08:00
|
|
|
|
</div>
|
2026-01-16 15:56:53 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
<!-- 分类列表 -->
|
|
|
|
|
|
<div
|
|
|
|
|
|
class="category-list"
|
|
|
|
|
|
>
|
2026-01-16 11:15:44 +08:00
|
|
|
|
<div
|
2026-01-16 15:56:53 +08:00
|
|
|
|
v-for="category in expenseCategoriesSimpView"
|
|
|
|
|
|
:key="category.classify"
|
|
|
|
|
|
class="category-item clickable"
|
|
|
|
|
|
@click="goToCategoryBills(category.classify, 0)"
|
2026-01-16 11:15:44 +08:00
|
|
|
|
>
|
2026-01-16 15:56:53 +08:00
|
|
|
|
<div class="category-info">
|
|
|
|
|
|
<div
|
|
|
|
|
|
class="category-color"
|
|
|
|
|
|
:style="{ backgroundColor: category.color }"
|
|
|
|
|
|
/>
|
|
|
|
|
|
<div class="category-name-with-count">
|
|
|
|
|
|
<span class="category-name">{{ category.classify || '未分类' }}</span>
|
|
|
|
|
|
<span class="category-count">{{ category.count }}笔</span>
|
|
|
|
|
|
</div>
|
2026-01-16 11:15:44 +08:00
|
|
|
|
</div>
|
2026-01-16 15:56:53 +08:00
|
|
|
|
<div class="category-stats">
|
|
|
|
|
|
<div class="category-amount">
|
|
|
|
|
|
¥{{ formatMoney(category.amount) }}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="category-percent">
|
|
|
|
|
|
{{ category.percent }}%
|
|
|
|
|
|
</div>
|
2026-01-11 16:33:55 +08:00
|
|
|
|
</div>
|
2026-01-16 15:56:53 +08:00
|
|
|
|
<van-icon
|
|
|
|
|
|
name="arrow"
|
|
|
|
|
|
class="category-arrow"
|
|
|
|
|
|
/>
|
2026-01-11 16:33:55 +08:00
|
|
|
|
</div>
|
2026-01-16 15:56:53 +08:00
|
|
|
|
<!-- 展开/收起按钮 -->
|
2026-01-16 11:15:44 +08:00
|
|
|
|
<div
|
2026-01-16 15:56:53 +08:00
|
|
|
|
v-if="expenseCategoriesView.length > 1"
|
|
|
|
|
|
class="expand-toggle"
|
|
|
|
|
|
@click="showAllExpense = !showAllExpense"
|
2026-01-16 11:15:44 +08:00
|
|
|
|
>
|
2026-01-16 15:56:53 +08:00
|
|
|
|
<van-icon :name="showAllExpense ? 'arrow-up' : 'arrow-down'" />
|
2026-01-11 16:33:55 +08:00
|
|
|
|
</div>
|
2025-12-26 17:13:57 +08:00
|
|
|
|
</div>
|
2026-01-16 15:56:53 +08:00
|
|
|
|
<van-empty
|
|
|
|
|
|
v-if="!expenseCategoriesView || !expenseCategoriesView.length"
|
|
|
|
|
|
description="本月暂无支出记录"
|
|
|
|
|
|
image="search"
|
|
|
|
|
|
/>
|
2025-12-26 17:13:57 +08:00
|
|
|
|
</div>
|
2026-01-16 15:56:53 +08:00
|
|
|
|
<!-- 收支和不计收支并列显示 -->
|
|
|
|
|
|
<div
|
|
|
|
|
|
v-if="incomeCategoriesView.length > 0 || noneCategoriesView.length > 0"
|
|
|
|
|
|
class="side-by-side-cards"
|
|
|
|
|
|
>
|
|
|
|
|
|
<!-- 收入分类统计 -->
|
2026-01-16 12:08:11 +08:00
|
|
|
|
<div
|
2026-01-16 15:56:53 +08:00
|
|
|
|
v-if="incomeCategoriesView.length > 0"
|
|
|
|
|
|
class="common-card half-card"
|
2026-01-16 12:08:11 +08:00
|
|
|
|
>
|
2026-01-15 22:10:57 +08:00
|
|
|
|
<div class="card-header">
|
2026-01-16 11:15:44 +08:00
|
|
|
|
<h3 class="card-title">
|
2026-01-16 15:56:53 +08:00
|
|
|
|
收入
|
2026-01-20 19:56:29 +08:00
|
|
|
|
<span
|
|
|
|
|
|
class="income-text"
|
|
|
|
|
|
style="font-size: 13px; margin-left: 4px"
|
|
|
|
|
|
>
|
|
|
|
|
|
¥{{ formatMoney(monthlyData.totalIncome) }}
|
|
|
|
|
|
</span>
|
2026-01-16 11:15:44 +08:00
|
|
|
|
</h3>
|
|
|
|
|
|
<van-tag
|
2026-01-16 15:56:53 +08:00
|
|
|
|
type="success"
|
2026-01-16 11:15:44 +08:00
|
|
|
|
size="medium"
|
|
|
|
|
|
>
|
2026-01-16 15:56:53 +08:00
|
|
|
|
{{ incomeCategoriesView.length }}
|
2026-01-16 11:15:44 +08:00
|
|
|
|
</van-tag>
|
2026-01-15 22:10:57 +08:00
|
|
|
|
</div>
|
2026-01-16 15:56:53 +08:00
|
|
|
|
|
|
|
|
|
|
<div class="category-list">
|
2026-01-16 11:15:44 +08:00
|
|
|
|
<div
|
2026-01-16 15:56:53 +08:00
|
|
|
|
v-for="category in incomeCategoriesView"
|
2026-01-16 12:08:11 +08:00
|
|
|
|
:key="category.classify"
|
2026-01-16 11:15:44 +08:00
|
|
|
|
class="category-item clickable"
|
2026-01-16 15:56:53 +08:00
|
|
|
|
@click="goToCategoryBills(category.classify, 1)"
|
2026-01-16 11:15:44 +08:00
|
|
|
|
>
|
|
|
|
|
|
<div class="category-info">
|
2026-01-16 15:56:53 +08:00
|
|
|
|
<div class="category-color income-color" />
|
|
|
|
|
|
<span class="category-name text-ellipsis">{{ category.classify || '未分类' }}</span>
|
2026-01-16 11:15:44 +08:00
|
|
|
|
</div>
|
2026-01-16 15:56:53 +08:00
|
|
|
|
<div class="category-amount income-text">
|
|
|
|
|
|
¥{{ formatMoney(category.amount) }}
|
2026-01-16 11:15:44 +08:00
|
|
|
|
</div>
|
2026-01-16 12:08:11 +08:00
|
|
|
|
</div>
|
2025-12-26 17:13:57 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
2026-01-16 15:56:53 +08:00
|
|
|
|
|
|
|
|
|
|
<!-- 不计收支分类统计 -->
|
2026-01-16 11:15:44 +08:00
|
|
|
|
<div
|
2026-01-16 15:56:53 +08:00
|
|
|
|
v-if="noneCategoriesView.length > 0"
|
|
|
|
|
|
class="common-card half-card"
|
2026-01-16 11:15:44 +08:00
|
|
|
|
>
|
2026-01-15 22:10:57 +08:00
|
|
|
|
<div class="card-header">
|
2026-01-16 11:15:44 +08:00
|
|
|
|
<h3 class="card-title">
|
2026-01-16 15:56:53 +08:00
|
|
|
|
不计收支
|
2026-01-16 11:15:44 +08:00
|
|
|
|
</h3>
|
2026-01-16 15:56:53 +08:00
|
|
|
|
<van-tag
|
|
|
|
|
|
type="warning"
|
|
|
|
|
|
size="medium"
|
|
|
|
|
|
>
|
|
|
|
|
|
{{ noneCategoriesView.length }}
|
|
|
|
|
|
</van-tag>
|
2025-12-26 17:13:57 +08:00
|
|
|
|
</div>
|
2026-01-16 11:15:44 +08:00
|
|
|
|
|
2026-01-16 15:56:53 +08:00
|
|
|
|
<div class="category-list">
|
|
|
|
|
|
<div
|
|
|
|
|
|
v-for="category in noneCategoriesView"
|
|
|
|
|
|
:key="category.classify"
|
|
|
|
|
|
class="category-item clickable"
|
|
|
|
|
|
@click="goToCategoryBills(category.classify, 2)"
|
|
|
|
|
|
>
|
|
|
|
|
|
<div class="category-info">
|
|
|
|
|
|
<div class="category-color none-color" />
|
|
|
|
|
|
<span class="category-name text-ellipsis">{{ category.classify || '未分类' }}</span>
|
2026-01-16 11:15:44 +08:00
|
|
|
|
</div>
|
2026-01-16 15:56:53 +08:00
|
|
|
|
<div class="category-amount none-text">
|
|
|
|
|
|
¥{{ formatMoney(category.amount) }}
|
2026-01-16 11:15:44 +08:00
|
|
|
|
</div>
|
2026-01-15 22:10:57 +08:00
|
|
|
|
</div>
|
2025-12-26 17:13:57 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
2026-01-11 16:33:55 +08:00
|
|
|
|
</div>
|
2026-01-16 15:56:53 +08:00
|
|
|
|
<!-- 底部安全距离 -->
|
|
|
|
|
|
<div style="height: calc(50px + env(safe-area-inset-bottom, 0px))" />
|
|
|
|
|
|
</div>
|
2025-12-26 17:13:57 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</van-pull-refresh>
|
|
|
|
|
|
|
2026-01-28 17:00:58 +08:00
|
|
|
|
<!-- 日期选择器 -->
|
2026-01-16 11:15:44 +08:00
|
|
|
|
<van-popup
|
|
|
|
|
|
v-model:show="showMonthPicker"
|
|
|
|
|
|
position="bottom"
|
|
|
|
|
|
round
|
|
|
|
|
|
teleport="body"
|
|
|
|
|
|
>
|
2026-01-28 17:00:58 +08:00
|
|
|
|
<div class="date-picker-header">
|
|
|
|
|
|
<van-tabs
|
|
|
|
|
|
v-model:active="dateSelectionMode"
|
|
|
|
|
|
line-width="20px"
|
|
|
|
|
|
:ellipsis="false"
|
|
|
|
|
|
>
|
2026-01-28 19:32:11 +08:00
|
|
|
|
<van-tab
|
|
|
|
|
|
title="按月"
|
|
|
|
|
|
name="month"
|
|
|
|
|
|
/>
|
|
|
|
|
|
<van-tab
|
|
|
|
|
|
title="按年"
|
|
|
|
|
|
name="year"
|
|
|
|
|
|
/>
|
2026-01-28 17:00:58 +08:00
|
|
|
|
</van-tabs>
|
|
|
|
|
|
</div>
|
2025-12-26 17:13:57 +08:00
|
|
|
|
<van-date-picker
|
|
|
|
|
|
v-model="selectedDate"
|
2026-01-28 17:00:58 +08:00
|
|
|
|
:title="dateSelectionMode === 'year' ? '选择年份' : '选择月份'"
|
2025-12-26 17:13:57 +08:00
|
|
|
|
:min-date="minDate"
|
|
|
|
|
|
:max-date="maxDate"
|
2026-01-28 17:00:58 +08:00
|
|
|
|
:columns-type="dateSelectionMode === 'year' ? ['year'] : ['year', 'month']"
|
2025-12-26 17:13:57 +08:00
|
|
|
|
@confirm="onMonthConfirm"
|
|
|
|
|
|
@cancel="showMonthPicker = false"
|
|
|
|
|
|
/>
|
|
|
|
|
|
</van-popup>
|
2025-12-26 17:56:08 +08:00
|
|
|
|
|
|
|
|
|
|
<!-- 分类账单列表弹出层 -->
|
2025-12-30 17:02:30 +08:00
|
|
|
|
<PopupContainer
|
|
|
|
|
|
v-model="billListVisible"
|
|
|
|
|
|
:title="selectedCategoryTitle"
|
|
|
|
|
|
:subtitle="categoryBillsTotal ? `共 ${categoryBillsTotal} 笔交易` : ''"
|
2026-01-11 11:21:13 +08:00
|
|
|
|
height="75%"
|
2025-12-26 17:56:08 +08:00
|
|
|
|
>
|
2025-12-30 17:02:30 +08:00
|
|
|
|
<template #header-actions>
|
|
|
|
|
|
<SmartClassifyButton
|
|
|
|
|
|
v-if="isUnclassified"
|
2026-01-07 14:33:30 +08:00
|
|
|
|
ref="smartClassifyButtonRef"
|
2025-12-30 17:02:30 +08:00
|
|
|
|
:transactions="categoryBills"
|
2026-01-07 14:33:30 +08:00
|
|
|
|
:on-before-classify="beforeSmartClassify"
|
2025-12-30 17:02:30 +08:00
|
|
|
|
@save="onSmartClassifySave"
|
2025-12-30 18:49:46 +08:00
|
|
|
|
@notify-doned-transaction-id="handleNotifiedTransactionId"
|
2025-12-30 17:02:30 +08:00
|
|
|
|
/>
|
|
|
|
|
|
</template>
|
2026-01-16 11:15:44 +08:00
|
|
|
|
|
2025-12-30 17:02:30 +08:00
|
|
|
|
<TransactionList
|
2026-01-10 12:22:37 +08:00
|
|
|
|
ref="transactionListRef"
|
2025-12-30 17:02:30 +08:00
|
|
|
|
:transactions="categoryBills"
|
|
|
|
|
|
:loading="billListLoading"
|
|
|
|
|
|
:finished="billListFinished"
|
2025-12-31 11:49:25 +08:00
|
|
|
|
:show-delete="true"
|
2025-12-30 17:02:30 +08:00
|
|
|
|
@load="loadCategoryBills"
|
|
|
|
|
|
@click="viewBillDetail"
|
2026-01-01 11:58:21 +08:00
|
|
|
|
@delete="handleCategoryBillsDelete"
|
2025-12-30 17:02:30 +08:00
|
|
|
|
/>
|
|
|
|
|
|
</PopupContainer>
|
2025-12-26 17:56:08 +08:00
|
|
|
|
|
|
|
|
|
|
<!-- 交易详情编辑组件 -->
|
|
|
|
|
|
<TransactionDetail
|
|
|
|
|
|
v-model:show="detailVisible"
|
|
|
|
|
|
:transaction="currentTransaction"
|
|
|
|
|
|
@save="onBillSave"
|
|
|
|
|
|
/>
|
2026-02-09 19:25:51 +08:00
|
|
|
|
|
|
|
|
|
|
<!-- 液态玻璃底部导航栏 -->
|
|
|
|
|
|
<GlassBottomNav
|
|
|
|
|
|
v-model="activeTab"
|
|
|
|
|
|
@tab-click="handleTabClick"
|
|
|
|
|
|
/>
|
2025-12-26 17:13:57 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
|
|
|
|
|
|
<script setup>
|
2026-01-16 11:15:44 +08:00
|
|
|
|
import { ref, computed, onMounted, onActivated, nextTick, watch } from 'vue'
|
2026-01-01 11:58:21 +08:00
|
|
|
|
import { onBeforeUnmount } from 'vue'
|
2025-12-26 17:13:57 +08:00
|
|
|
|
import { showToast } from 'vant'
|
|
|
|
|
|
import { useRouter } from 'vue-router'
|
2026-01-21 16:09:38 +08:00
|
|
|
|
import { getMonthlyStatistics, getCategoryStatistics, getDailyStatistics, getBalanceStatistics } from '@/api/statistics'
|
2026-01-15 22:10:57 +08:00
|
|
|
|
import * as echarts from 'echarts'
|
2025-12-26 17:56:08 +08:00
|
|
|
|
import { getTransactionList, getTransactionDetail } from '@/api/transactionRecord'
|
|
|
|
|
|
import TransactionList from '@/components/TransactionList.vue'
|
|
|
|
|
|
import TransactionDetail from '@/components/TransactionDetail.vue'
|
2025-12-29 21:17:18 +08:00
|
|
|
|
import SmartClassifyButton from '@/components/SmartClassifyButton.vue'
|
2025-12-30 17:02:30 +08:00
|
|
|
|
import PopupContainer from '@/components/PopupContainer.vue'
|
2026-02-09 19:25:51 +08:00
|
|
|
|
import GlassBottomNav from '@/components/GlassBottomNav.vue'
|
2026-01-16 15:56:53 +08:00
|
|
|
|
import { getCssVar } from '@/utils/theme'
|
2025-12-26 17:13:57 +08:00
|
|
|
|
|
|
|
|
|
|
const router = useRouter()
|
|
|
|
|
|
|
2026-02-09 19:25:51 +08:00
|
|
|
|
// 底部导航栏
|
|
|
|
|
|
const activeTab = ref('statistics')
|
|
|
|
|
|
const handleTabClick = (item, index) => {
|
|
|
|
|
|
console.log('Tab clicked:', item.name, index)
|
|
|
|
|
|
// 导航逻辑已在组件内部处理
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-26 17:13:57 +08:00
|
|
|
|
// 响应式数据
|
|
|
|
|
|
const loading = ref(true)
|
2026-01-11 16:33:55 +08:00
|
|
|
|
const firstLoading = ref(true)
|
2025-12-26 17:13:57 +08:00
|
|
|
|
const refreshing = ref(false)
|
|
|
|
|
|
const showMonthPicker = ref(false)
|
2026-01-11 16:33:55 +08:00
|
|
|
|
const showAllExpense = ref(false)
|
2026-01-28 17:00:58 +08:00
|
|
|
|
const dateSelectionMode = ref('month')
|
2025-12-26 17:13:57 +08:00
|
|
|
|
const currentYear = ref(new Date().getFullYear())
|
|
|
|
|
|
const currentMonth = ref(new Date().getMonth() + 1)
|
2026-01-16 11:15:44 +08:00
|
|
|
|
const selectedDate = ref([
|
|
|
|
|
|
new Date().getFullYear().toString(),
|
|
|
|
|
|
(new Date().getMonth() + 1).toString().padStart(2, '0')
|
|
|
|
|
|
])
|
2025-12-26 17:13:57 +08:00
|
|
|
|
|
2025-12-26 17:56:08 +08:00
|
|
|
|
const billListVisible = ref(false)
|
|
|
|
|
|
const billListLoading = ref(false)
|
|
|
|
|
|
const billListFinished = ref(false)
|
|
|
|
|
|
const categoryBills = ref([])
|
2025-12-27 22:05:50 +08:00
|
|
|
|
const categoryBillsTotal = ref(0)
|
2025-12-26 17:56:08 +08:00
|
|
|
|
const selectedCategoryTitle = ref('')
|
|
|
|
|
|
const selectedClassify = ref('')
|
|
|
|
|
|
const selectedType = ref(null)
|
2025-12-27 22:05:50 +08:00
|
|
|
|
const billPageIndex = ref(1)
|
2026-01-16 11:15:44 +08:00
|
|
|
|
const billPageSize = 20
|
2025-12-26 17:56:08 +08:00
|
|
|
|
|
|
|
|
|
|
// 详情编辑相关
|
|
|
|
|
|
const detailVisible = ref(false)
|
|
|
|
|
|
const currentTransaction = ref(null)
|
|
|
|
|
|
|
2025-12-26 17:13:57 +08:00
|
|
|
|
// 月度数据
|
|
|
|
|
|
const monthlyData = ref({
|
|
|
|
|
|
totalExpense: 0,
|
|
|
|
|
|
totalIncome: 0,
|
|
|
|
|
|
balance: 0,
|
|
|
|
|
|
expenseCount: 0,
|
|
|
|
|
|
incomeCount: 0,
|
2026-01-16 15:56:53 +08:00
|
|
|
|
totalCount: 0
|
2025-12-26 17:13:57 +08:00
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
// 分类数据
|
|
|
|
|
|
const expenseCategories = ref([])
|
|
|
|
|
|
const incomeCategories = ref([])
|
2025-12-27 22:34:19 +08:00
|
|
|
|
const noneCategories = ref([])
|
2025-12-26 17:13:57 +08:00
|
|
|
|
|
2026-01-12 14:46:11 +08:00
|
|
|
|
const expenseCategoriesSimpView = computed(() => {
|
|
|
|
|
|
const list = expenseCategoriesView.value
|
2026-01-16 11:15:44 +08:00
|
|
|
|
|
2026-01-16 12:08:11 +08:00
|
|
|
|
if (showAllExpense.value) {
|
2026-01-16 11:15:44 +08:00
|
|
|
|
return list
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-16 12:08:11 +08:00
|
|
|
|
// 只展示未分类
|
|
|
|
|
|
const unclassified = list.filter((c) => c.classify === '未分类' || !c.classify)
|
|
|
|
|
|
if (unclassified.length > 0) {
|
|
|
|
|
|
return [...unclassified]
|
|
|
|
|
|
}
|
|
|
|
|
|
return []
|
2026-01-04 18:24:39 +08:00
|
|
|
|
})
|
|
|
|
|
|
|
2026-01-12 14:46:11 +08:00
|
|
|
|
const expenseCategoriesView = computed(() => {
|
|
|
|
|
|
const list = [...expenseCategories.value]
|
2026-01-16 11:15:44 +08:00
|
|
|
|
const unclassifiedIndex = list.findIndex((c) => !c.classify)
|
2026-01-12 14:46:11 +08:00
|
|
|
|
if (unclassifiedIndex !== -1) {
|
|
|
|
|
|
const [unclassified] = list.splice(unclassifiedIndex, 1)
|
|
|
|
|
|
list.unshift(unclassified)
|
|
|
|
|
|
}
|
2026-01-16 11:15:44 +08:00
|
|
|
|
|
2026-01-12 14:46:11 +08:00
|
|
|
|
return list
|
|
|
|
|
|
})
|
|
|
|
|
|
|
2026-01-04 18:24:39 +08:00
|
|
|
|
const incomeCategoriesView = computed(() => {
|
2026-01-11 16:33:55 +08:00
|
|
|
|
const list = [...incomeCategories.value]
|
2026-01-16 11:15:44 +08:00
|
|
|
|
const unclassifiedIndex = list.findIndex((c) => !c.classify)
|
2026-01-04 18:24:39 +08:00
|
|
|
|
if (unclassifiedIndex !== -1) {
|
2026-01-11 16:33:55 +08:00
|
|
|
|
const [unclassified] = list.splice(unclassifiedIndex, 1)
|
|
|
|
|
|
list.unshift(unclassified)
|
2026-01-04 18:24:39 +08:00
|
|
|
|
}
|
2026-01-16 11:15:44 +08:00
|
|
|
|
|
2026-01-16 12:08:11 +08:00
|
|
|
|
return list
|
|
|
|
|
|
})
|
|
|
|
|
|
|
2026-01-04 18:24:39 +08:00
|
|
|
|
const noneCategoriesView = computed(() => {
|
2026-01-11 16:33:55 +08:00
|
|
|
|
const list = [...noneCategories.value]
|
2026-01-16 11:15:44 +08:00
|
|
|
|
const unclassifiedIndex = list.findIndex((c) => !c.classify)
|
2026-01-04 18:24:39 +08:00
|
|
|
|
if (unclassifiedIndex !== -1) {
|
2026-01-11 16:33:55 +08:00
|
|
|
|
const [unclassified] = list.splice(unclassifiedIndex, 1)
|
|
|
|
|
|
list.unshift(unclassified)
|
2026-01-04 18:24:39 +08:00
|
|
|
|
}
|
2026-01-16 11:15:44 +08:00
|
|
|
|
|
2026-01-16 12:08:11 +08:00
|
|
|
|
return list
|
2026-01-04 18:24:39 +08:00
|
|
|
|
})
|
|
|
|
|
|
|
2025-12-26 17:13:57 +08:00
|
|
|
|
// 趋势数据
|
2026-01-15 22:10:57 +08:00
|
|
|
|
const dailyData = ref([])
|
2026-01-21 16:09:38 +08:00
|
|
|
|
// 余额数据(独立)
|
|
|
|
|
|
const balanceData = ref([])
|
2026-01-15 22:10:57 +08:00
|
|
|
|
const pieChartRef = ref(null)
|
2026-01-21 16:09:38 +08:00
|
|
|
|
const balanceChartRef = ref(null)
|
2026-01-15 22:10:57 +08:00
|
|
|
|
let pieChartInstance = null
|
2026-01-21 16:09:38 +08:00
|
|
|
|
let balanceChartInstance = null
|
2025-12-26 17:13:57 +08:00
|
|
|
|
|
|
|
|
|
|
// 日期范围
|
|
|
|
|
|
const minDate = new Date(2020, 0, 1)
|
|
|
|
|
|
const maxDate = new Date()
|
|
|
|
|
|
|
|
|
|
|
|
// 颜色配置
|
2026-01-16 15:56:53 +08:00
|
|
|
|
const getChartColors = () => [
|
|
|
|
|
|
getCssVar('--chart-color-1'),
|
|
|
|
|
|
getCssVar('--chart-color-2'),
|
|
|
|
|
|
getCssVar('--chart-color-3'),
|
|
|
|
|
|
getCssVar('--chart-color-4'),
|
|
|
|
|
|
getCssVar('--chart-color-5'),
|
|
|
|
|
|
getCssVar('--chart-color-6'),
|
|
|
|
|
|
getCssVar('--chart-color-7'),
|
|
|
|
|
|
getCssVar('--chart-color-8'),
|
|
|
|
|
|
getCssVar('--chart-color-9'),
|
|
|
|
|
|
getCssVar('--chart-color-10'),
|
|
|
|
|
|
getCssVar('--chart-color-11'),
|
|
|
|
|
|
getCssVar('--chart-color-12'),
|
|
|
|
|
|
getCssVar('--chart-color-13'),
|
|
|
|
|
|
getCssVar('--chart-color-14'),
|
|
|
|
|
|
getCssVar('--chart-color-15')
|
2025-12-26 17:13:57 +08:00
|
|
|
|
]
|
|
|
|
|
|
|
2025-12-29 21:17:18 +08:00
|
|
|
|
// 是否为未分类账单
|
|
|
|
|
|
const isUnclassified = computed(() => {
|
|
|
|
|
|
return selectedClassify.value === '未分类' || selectedClassify.value === ''
|
|
|
|
|
|
})
|
|
|
|
|
|
|
2026-01-21 16:09:38 +08:00
|
|
|
|
// 当月累积余额
|
|
|
|
|
|
const currentMonthBalance = computed(() => {
|
|
|
|
|
|
if (balanceData.value.length === 0) {
|
|
|
|
|
|
return 0
|
|
|
|
|
|
}
|
|
|
|
|
|
// 获取最后一天的累积余额
|
|
|
|
|
|
return balanceData.value[balanceData.value.length - 1].cumulativeBalance || 0
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
// 显示的动画余额
|
|
|
|
|
|
const displayBalance = ref(0)
|
|
|
|
|
|
|
|
|
|
|
|
// 监听余额变化,执行动画
|
|
|
|
|
|
watch(currentMonthBalance, (newVal, oldVal) => {
|
|
|
|
|
|
if (oldVal === undefined) {
|
|
|
|
|
|
// 初始加载,直接设置,不需要动画
|
|
|
|
|
|
displayBalance.value = newVal
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 数字跳动动画
|
|
|
|
|
|
const duration = 800 // 动画持续时间(毫秒)
|
|
|
|
|
|
const startValue = oldVal
|
|
|
|
|
|
const endValue = newVal
|
|
|
|
|
|
const startTime = Date.now()
|
|
|
|
|
|
|
|
|
|
|
|
const animate = () => {
|
|
|
|
|
|
const elapsed = Date.now() - startTime
|
|
|
|
|
|
const progress = Math.min(elapsed / duration, 1)
|
|
|
|
|
|
|
|
|
|
|
|
// 使用缓动函数(easeOutQuad)
|
|
|
|
|
|
const easeProgress = 1 - (1 - progress) * (1 - progress)
|
|
|
|
|
|
displayBalance.value = Math.round(startValue + (endValue - startValue) * easeProgress)
|
|
|
|
|
|
|
|
|
|
|
|
if (progress < 1) {
|
|
|
|
|
|
requestAnimationFrame(animate)
|
|
|
|
|
|
} else {
|
|
|
|
|
|
displayBalance.value = endValue
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
animate()
|
|
|
|
|
|
})
|
|
|
|
|
|
|
2025-12-26 17:13:57 +08:00
|
|
|
|
// 格式化金额
|
|
|
|
|
|
const formatMoney = (value) => {
|
2026-01-16 11:15:44 +08:00
|
|
|
|
if (!value && value !== 0) {
|
|
|
|
|
|
return '0'
|
|
|
|
|
|
}
|
|
|
|
|
|
return Number(value)
|
|
|
|
|
|
.toFixed(0)
|
|
|
|
|
|
.replace(/\B(?=(\d{3})+(?!\d))/g, ',')
|
2025-12-26 17:13:57 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-16 15:56:53 +08:00
|
|
|
|
// 下拉刷新
|
|
|
|
|
|
const onRefresh = async () => {
|
|
|
|
|
|
await fetchStatistics(false)
|
|
|
|
|
|
refreshing.value = false
|
2025-12-26 17:13:57 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 确认月份选择
|
|
|
|
|
|
const onMonthConfirm = ({ selectedValues }) => {
|
2026-01-11 16:33:55 +08:00
|
|
|
|
const newYear = parseInt(selectedValues[0])
|
2026-01-28 17:00:58 +08:00
|
|
|
|
const newMonth = dateSelectionMode.value === 'year' ? 0 : parseInt(selectedValues[1])
|
2026-01-16 11:15:44 +08:00
|
|
|
|
|
2026-01-11 16:33:55 +08:00
|
|
|
|
currentYear.value = newYear
|
|
|
|
|
|
currentMonth.value = newMonth
|
2025-12-26 17:13:57 +08:00
|
|
|
|
showMonthPicker.value = false
|
2026-01-16 11:15:44 +08:00
|
|
|
|
|
2026-01-11 16:33:55 +08:00
|
|
|
|
// 重置展开状态
|
|
|
|
|
|
showAllExpense.value = false
|
2026-01-16 11:15:44 +08:00
|
|
|
|
|
2025-12-26 17:13:57 +08:00
|
|
|
|
fetchStatistics()
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 获取统计数据
|
2026-01-11 16:33:55 +08:00
|
|
|
|
const fetchStatistics = async (showLoading = true) => {
|
|
|
|
|
|
if (showLoading && firstLoading.value) {
|
|
|
|
|
|
loading.value = true
|
|
|
|
|
|
}
|
2026-01-16 11:15:44 +08:00
|
|
|
|
|
2025-12-26 17:13:57 +08:00
|
|
|
|
try {
|
2026-01-21 16:09:38 +08:00
|
|
|
|
await Promise.all([fetchMonthlyData(), fetchCategoryData(), fetchDailyData(), fetchBalanceData()])
|
2025-12-26 17:13:57 +08:00
|
|
|
|
} catch (error) {
|
|
|
|
|
|
console.error('获取统计数据失败:', error)
|
|
|
|
|
|
showToast('获取统计数据失败')
|
|
|
|
|
|
} finally {
|
|
|
|
|
|
loading.value = false
|
2026-01-11 16:33:55 +08:00
|
|
|
|
firstLoading.value = false
|
2026-01-15 22:10:57 +08:00
|
|
|
|
// DOM 更新后渲染图表
|
|
|
|
|
|
nextTick(() => {
|
|
|
|
|
|
renderPieChart()
|
2026-01-21 16:09:38 +08:00
|
|
|
|
renderBalanceChart()
|
2026-01-15 22:10:57 +08:00
|
|
|
|
})
|
2025-12-26 17:13:57 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 获取月度数据
|
|
|
|
|
|
const fetchMonthlyData = async () => {
|
|
|
|
|
|
try {
|
|
|
|
|
|
const response = await getMonthlyStatistics({
|
|
|
|
|
|
year: currentYear.value,
|
2026-01-28 17:00:58 +08:00
|
|
|
|
month: currentMonth.value || 0
|
2025-12-26 17:13:57 +08:00
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
if (response.success && response.data) {
|
|
|
|
|
|
monthlyData.value = response.data
|
|
|
|
|
|
}
|
|
|
|
|
|
} catch (error) {
|
|
|
|
|
|
console.error('获取月度数据失败:', error)
|
|
|
|
|
|
showToast('获取月度数据失败')
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 获取分类数据
|
|
|
|
|
|
const fetchCategoryData = async () => {
|
|
|
|
|
|
try {
|
|
|
|
|
|
// 获取支出分类
|
|
|
|
|
|
const expenseResponse = await getCategoryStatistics({
|
|
|
|
|
|
year: currentYear.value,
|
2026-01-28 17:00:58 +08:00
|
|
|
|
month: currentMonth.value || 0,
|
2025-12-26 17:13:57 +08:00
|
|
|
|
type: 0 // 支出
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
if (expenseResponse.success && expenseResponse.data) {
|
2026-01-16 15:56:53 +08:00
|
|
|
|
const currentColors = getChartColors()
|
2025-12-26 17:13:57 +08:00
|
|
|
|
expenseCategories.value = expenseResponse.data.map((item, index) => ({
|
|
|
|
|
|
classify: item.classify,
|
|
|
|
|
|
amount: item.amount,
|
2025-12-26 17:56:08 +08:00
|
|
|
|
count: item.count,
|
2025-12-26 17:13:57 +08:00
|
|
|
|
percent: item.percent,
|
2026-01-16 15:56:53 +08:00
|
|
|
|
color: currentColors[index % currentColors.length]
|
2025-12-26 17:13:57 +08:00
|
|
|
|
}))
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 获取收入分类
|
|
|
|
|
|
const incomeResponse = await getCategoryStatistics({
|
|
|
|
|
|
year: currentYear.value,
|
2026-01-28 17:00:58 +08:00
|
|
|
|
month: currentMonth.value || 0,
|
2025-12-26 17:13:57 +08:00
|
|
|
|
type: 1 // 收入
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
if (incomeResponse.success && incomeResponse.data) {
|
2026-01-16 11:15:44 +08:00
|
|
|
|
incomeCategories.value = incomeResponse.data.map((item) => ({
|
2025-12-26 17:13:57 +08:00
|
|
|
|
classify: item.classify,
|
|
|
|
|
|
amount: item.amount,
|
2025-12-26 17:56:08 +08:00
|
|
|
|
count: item.count,
|
2025-12-26 17:13:57 +08:00
|
|
|
|
percent: item.percent
|
|
|
|
|
|
}))
|
|
|
|
|
|
}
|
2025-12-27 22:34:19 +08:00
|
|
|
|
|
|
|
|
|
|
// 获取不计收支分类
|
|
|
|
|
|
const noneResponse = await getCategoryStatistics({
|
|
|
|
|
|
year: currentYear.value,
|
2026-01-28 17:00:58 +08:00
|
|
|
|
month: currentMonth.value || 0,
|
2025-12-27 22:34:19 +08:00
|
|
|
|
type: 2 // 不计收支
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
if (noneResponse.success && noneResponse.data) {
|
2026-01-16 11:15:44 +08:00
|
|
|
|
noneCategories.value = noneResponse.data.map((item) => ({
|
2025-12-27 22:34:19 +08:00
|
|
|
|
classify: item.classify,
|
|
|
|
|
|
amount: item.amount,
|
|
|
|
|
|
count: item.count,
|
|
|
|
|
|
percent: item.percent
|
|
|
|
|
|
}))
|
|
|
|
|
|
}
|
2025-12-26 17:13:57 +08:00
|
|
|
|
} catch (error) {
|
|
|
|
|
|
console.error('获取分类数据失败:', error)
|
|
|
|
|
|
showToast('获取分类数据失败')
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-15 22:10:57 +08:00
|
|
|
|
// 获取每日统计数据并渲染图表
|
|
|
|
|
|
const fetchDailyData = async () => {
|
2025-12-26 17:13:57 +08:00
|
|
|
|
try {
|
2026-01-15 22:10:57 +08:00
|
|
|
|
const response = await getDailyStatistics({
|
|
|
|
|
|
year: currentYear.value,
|
2026-01-28 17:00:58 +08:00
|
|
|
|
month: currentMonth.value || 0
|
2025-12-26 17:13:57 +08:00
|
|
|
|
})
|
2026-01-16 11:15:44 +08:00
|
|
|
|
|
2025-12-26 17:13:57 +08:00
|
|
|
|
if (response.success && response.data) {
|
2026-01-15 22:10:57 +08:00
|
|
|
|
dailyData.value = response.data
|
|
|
|
|
|
// 如果不是首次加载(即DOM已存在),直接渲染
|
|
|
|
|
|
if (!firstLoading.value) {
|
|
|
|
|
|
nextTick(() => {
|
2026-02-01 10:27:04 +08:00
|
|
|
|
renderBalanceChart()
|
2026-01-15 22:10:57 +08:00
|
|
|
|
})
|
|
|
|
|
|
}
|
2025-12-26 17:13:57 +08:00
|
|
|
|
}
|
|
|
|
|
|
} catch (error) {
|
2026-01-15 22:10:57 +08:00
|
|
|
|
console.error('获取每日统计数据失败:', error)
|
|
|
|
|
|
showToast('获取每日统计数据失败')
|
2025-12-26 17:13:57 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-21 16:09:38 +08:00
|
|
|
|
// 获取余额统计数据(独立接口)
|
|
|
|
|
|
const fetchBalanceData = async () => {
|
|
|
|
|
|
try {
|
|
|
|
|
|
const response = await getBalanceStatistics({
|
|
|
|
|
|
year: currentYear.value,
|
2026-01-28 17:00:58 +08:00
|
|
|
|
month: currentMonth.value || 0
|
2026-01-21 16:09:38 +08:00
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
if (response.success && response.data) {
|
|
|
|
|
|
balanceData.value = response.data
|
2026-02-01 10:27:04 +08:00
|
|
|
|
// 如果不是首次加载,重新渲染余额图表
|
|
|
|
|
|
if (!firstLoading.value) {
|
|
|
|
|
|
nextTick(() => {
|
|
|
|
|
|
renderBalanceChart()
|
|
|
|
|
|
})
|
|
|
|
|
|
}
|
2026-01-21 16:09:38 +08:00
|
|
|
|
}
|
|
|
|
|
|
} catch (error) {
|
|
|
|
|
|
console.error('获取余额统计数据失败:', error)
|
|
|
|
|
|
showToast('获取余额统计数据失败')
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-15 22:10:57 +08:00
|
|
|
|
const renderPieChart = () => {
|
2026-01-16 11:15:44 +08:00
|
|
|
|
if (!pieChartRef.value) {
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
if (expenseCategoriesView.value.length === 0) {
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 尝试获取DOM上的现有实例
|
|
|
|
|
|
const existingInstance = echarts.getInstanceByDom(pieChartRef.value)
|
|
|
|
|
|
|
|
|
|
|
|
if (pieChartInstance && pieChartInstance !== existingInstance) {
|
|
|
|
|
|
if (!pieChartInstance.isDisposed()) {
|
|
|
|
|
|
pieChartInstance.dispose()
|
|
|
|
|
|
}
|
|
|
|
|
|
pieChartInstance = null
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (pieChartInstance && pieChartInstance.getDom() !== pieChartRef.value) {
|
|
|
|
|
|
pieChartInstance.dispose()
|
|
|
|
|
|
pieChartInstance = null
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (!pieChartInstance && existingInstance) {
|
|
|
|
|
|
pieChartInstance = existingInstance
|
|
|
|
|
|
}
|
2026-01-15 22:10:57 +08:00
|
|
|
|
|
|
|
|
|
|
if (!pieChartInstance) {
|
|
|
|
|
|
pieChartInstance = echarts.init(pieChartRef.value)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 使用 Top N + Other 的数据逻辑,确保图表不会太拥挤
|
|
|
|
|
|
const list = [...expenseCategoriesView.value]
|
|
|
|
|
|
let chartData = []
|
2026-01-16 11:15:44 +08:00
|
|
|
|
|
2026-01-15 22:10:57 +08:00
|
|
|
|
// 按照金额排序
|
|
|
|
|
|
list.sort((a, b) => b.amount - a.amount)
|
2026-01-16 11:15:44 +08:00
|
|
|
|
|
2026-01-16 15:56:53 +08:00
|
|
|
|
const MAX_SLICES = 8 // 最大显示扇区数,其余合并为"其他"
|
|
|
|
|
|
const currentColors = getChartColors()
|
|
|
|
|
|
|
|
|
|
|
|
if (list.length > MAX_SLICES) {
|
|
|
|
|
|
const topList = list.slice(0, MAX_SLICES - 1)
|
|
|
|
|
|
const otherList = list.slice(MAX_SLICES - 1)
|
|
|
|
|
|
const otherAmount = otherList.reduce((sum, item) => sum + item.amount, 0)
|
|
|
|
|
|
|
|
|
|
|
|
chartData = topList.map((item, index) => ({
|
|
|
|
|
|
value: item.amount,
|
|
|
|
|
|
name: item.classify || '未分类',
|
|
|
|
|
|
itemStyle: { color: currentColors[index % currentColors.length] }
|
|
|
|
|
|
}))
|
|
|
|
|
|
|
|
|
|
|
|
chartData.push({
|
|
|
|
|
|
value: otherAmount,
|
|
|
|
|
|
name: '其他',
|
|
|
|
|
|
itemStyle: { color: getCssVar('--van-gray-6') } // 使用灰色表示其他
|
|
|
|
|
|
})
|
|
|
|
|
|
} else {
|
2026-01-15 22:10:57 +08:00
|
|
|
|
chartData = list.map((item, index) => ({
|
|
|
|
|
|
value: item.amount,
|
|
|
|
|
|
name: item.classify || '未分类',
|
2026-01-16 15:56:53 +08:00
|
|
|
|
itemStyle: { color: currentColors[index % currentColors.length] }
|
2026-01-15 22:10:57 +08:00
|
|
|
|
}))
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const option = {
|
|
|
|
|
|
title: {
|
|
|
|
|
|
text: '¥' + formatMoney(monthlyData.value.totalExpense),
|
|
|
|
|
|
subtext: '总支出',
|
|
|
|
|
|
left: 'center',
|
|
|
|
|
|
top: 'center',
|
|
|
|
|
|
textStyle: {
|
2026-01-16 15:56:53 +08:00
|
|
|
|
color: getCssVar('--chart-text-muted'), // 适配深色模式
|
2026-01-15 22:10:57 +08:00
|
|
|
|
fontSize: 20,
|
|
|
|
|
|
fontWeight: 'bold'
|
|
|
|
|
|
},
|
|
|
|
|
|
subtextStyle: {
|
2026-01-16 15:56:53 +08:00
|
|
|
|
color: getCssVar('--chart-text-muted'),
|
2026-01-15 22:10:57 +08:00
|
|
|
|
fontSize: 13
|
|
|
|
|
|
}
|
|
|
|
|
|
},
|
|
|
|
|
|
tooltip: {
|
|
|
|
|
|
trigger: 'item',
|
2026-01-28 19:32:11 +08:00
|
|
|
|
formatter: (params) => {
|
|
|
|
|
|
return `${params.name}: ¥${formatMoney(params.value)} (${params.percent.toFixed(1)}%)`
|
|
|
|
|
|
}
|
2026-01-15 22:10:57 +08:00
|
|
|
|
},
|
|
|
|
|
|
series: [
|
|
|
|
|
|
{
|
|
|
|
|
|
name: '支出分类',
|
|
|
|
|
|
type: 'pie',
|
|
|
|
|
|
radius: ['50%', '80%'],
|
|
|
|
|
|
avoidLabelOverlap: true,
|
|
|
|
|
|
minAngle: 5, // 最小扇区角度,防止扇区太小看不见
|
|
|
|
|
|
itemStyle: {
|
|
|
|
|
|
borderRadius: 5,
|
2026-01-16 15:56:53 +08:00
|
|
|
|
borderColor: getCssVar('--van-background-2'),
|
2026-01-15 22:10:57 +08:00
|
|
|
|
borderWidth: 2
|
|
|
|
|
|
},
|
|
|
|
|
|
label: {
|
|
|
|
|
|
show: true,
|
|
|
|
|
|
position: 'outside',
|
|
|
|
|
|
formatter: '{b}',
|
2026-01-16 15:56:53 +08:00
|
|
|
|
color: getCssVar('--van-text-color-2') // 适配深色模式
|
|
|
|
|
|
},
|
|
|
|
|
|
labelLayout: {
|
|
|
|
|
|
hideOverlap: true // 隐藏重叠标签
|
2026-01-15 22:10:57 +08:00
|
|
|
|
},
|
|
|
|
|
|
labelLine: {
|
|
|
|
|
|
show: true,
|
|
|
|
|
|
lineStyle: {
|
2026-01-16 15:56:53 +08:00
|
|
|
|
color: getCssVar('--van-text-color-3')
|
|
|
|
|
|
},
|
|
|
|
|
|
smooth: 0.2,
|
|
|
|
|
|
length: 10,
|
|
|
|
|
|
length2: 20
|
2026-01-15 22:10:57 +08:00
|
|
|
|
},
|
|
|
|
|
|
data: chartData
|
|
|
|
|
|
}
|
|
|
|
|
|
]
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
pieChartInstance.setOption(option)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-01 10:27:04 +08:00
|
|
|
|
// 渲染余额变化图表(融合支出、收入、余额三条线)
|
2026-01-21 16:09:38 +08:00
|
|
|
|
const renderBalanceChart = () => {
|
|
|
|
|
|
if (!balanceChartRef.value) {
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
2026-02-01 10:27:04 +08:00
|
|
|
|
if (balanceData.value.length === 0 && dailyData.value.length === 0) {
|
2026-01-21 16:09:38 +08:00
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 尝试获取DOM上的现有实例
|
|
|
|
|
|
const existingInstance = echarts.getInstanceByDom(balanceChartRef.value)
|
|
|
|
|
|
|
|
|
|
|
|
if (balanceChartInstance && balanceChartInstance !== existingInstance) {
|
|
|
|
|
|
if (!balanceChartInstance.isDisposed()) {
|
|
|
|
|
|
balanceChartInstance.dispose()
|
|
|
|
|
|
}
|
|
|
|
|
|
balanceChartInstance = null
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (balanceChartInstance && balanceChartInstance.getDom() !== balanceChartRef.value) {
|
|
|
|
|
|
balanceChartInstance.dispose()
|
|
|
|
|
|
balanceChartInstance = null
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (!balanceChartInstance && existingInstance) {
|
|
|
|
|
|
balanceChartInstance = existingInstance
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (!balanceChartInstance) {
|
|
|
|
|
|
balanceChartInstance = echarts.init(balanceChartRef.value)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-01 10:27:04 +08:00
|
|
|
|
// 判断是年度统计还是月度统计
|
|
|
|
|
|
const isYearlyView = currentMonth.value === 0
|
|
|
|
|
|
let dates, expenses, incomes, balances
|
|
|
|
|
|
|
|
|
|
|
|
if (isYearlyView) {
|
|
|
|
|
|
// 按年统计:按月聚合数据
|
|
|
|
|
|
const monthlyMap = new Map()
|
|
|
|
|
|
const balanceMonthlyMap = new Map()
|
|
|
|
|
|
|
|
|
|
|
|
// 聚合 dailyData 按月
|
|
|
|
|
|
dailyData.value.forEach((item) => {
|
|
|
|
|
|
const date = new Date(item.date)
|
|
|
|
|
|
const month = date.getMonth() + 1 // 1-12
|
|
|
|
|
|
if (!monthlyMap.has(month)) {
|
|
|
|
|
|
monthlyMap.set(month, { expense: 0, income: 0 })
|
|
|
|
|
|
}
|
|
|
|
|
|
const data = monthlyMap.get(month)
|
|
|
|
|
|
data.expense += item.expense
|
|
|
|
|
|
data.income += item.income
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
// 聚合 balanceData 按月(取每月最后一天的余额)
|
|
|
|
|
|
balanceData.value.forEach((item) => {
|
|
|
|
|
|
const date = new Date(item.date)
|
|
|
|
|
|
const month = date.getMonth() + 1
|
|
|
|
|
|
const day = date.getDate()
|
|
|
|
|
|
|
|
|
|
|
|
if (!balanceMonthlyMap.has(month) || day > balanceMonthlyMap.get(month).day) {
|
|
|
|
|
|
balanceMonthlyMap.set(month, { balance: item.cumulativeBalance, day })
|
|
|
|
|
|
}
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
// 构建12个月的完整数据
|
|
|
|
|
|
const now = new Date()
|
|
|
|
|
|
const currentMonthNum = now.getFullYear() === currentYear.value ? now.getMonth() + 1 : 12
|
|
|
|
|
|
|
|
|
|
|
|
dates = []
|
|
|
|
|
|
const monthlyExpenses = []
|
|
|
|
|
|
const monthlyIncomes = []
|
|
|
|
|
|
const monthlyBalances = []
|
|
|
|
|
|
|
|
|
|
|
|
let accumulatedExpense = 0
|
|
|
|
|
|
let accumulatedIncome = 0
|
|
|
|
|
|
|
|
|
|
|
|
for (let m = 1; m <= currentMonthNum; m++) {
|
|
|
|
|
|
dates.push(`${m}月`)
|
|
|
|
|
|
|
|
|
|
|
|
const data = monthlyMap.get(m) || { expense: 0, income: 0 }
|
|
|
|
|
|
accumulatedExpense += data.expense
|
|
|
|
|
|
accumulatedIncome += data.income
|
|
|
|
|
|
|
|
|
|
|
|
monthlyExpenses.push(accumulatedExpense)
|
|
|
|
|
|
monthlyIncomes.push(accumulatedIncome)
|
|
|
|
|
|
|
|
|
|
|
|
const balanceData = balanceMonthlyMap.get(m)
|
|
|
|
|
|
monthlyBalances.push(balanceData ? balanceData.balance : 0)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
expenses = monthlyExpenses
|
|
|
|
|
|
incomes = monthlyIncomes
|
|
|
|
|
|
balances = monthlyBalances
|
2026-01-21 16:09:38 +08:00
|
|
|
|
|
2026-02-01 10:27:04 +08:00
|
|
|
|
} else {
|
|
|
|
|
|
// 按月统计:按日显示
|
|
|
|
|
|
const now = new Date()
|
|
|
|
|
|
let daysInMonth
|
|
|
|
|
|
|
|
|
|
|
|
if (currentYear.value === now.getFullYear() && currentMonth.value === now.getMonth() + 1) {
|
|
|
|
|
|
daysInMonth = now.getDate()
|
|
|
|
|
|
} else {
|
|
|
|
|
|
daysInMonth = new Date(currentYear.value, currentMonth.value, 0).getDate()
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const fullData = []
|
|
|
|
|
|
const dataMap = new Map()
|
|
|
|
|
|
dailyData.value.forEach((item) => {
|
|
|
|
|
|
const day = new Date(item.date).getDate()
|
|
|
|
|
|
dataMap.set(day, item)
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
// 创建余额映射
|
|
|
|
|
|
const balanceMap = new Map()
|
|
|
|
|
|
if (balanceData.value && balanceData.value.length > 0) {
|
|
|
|
|
|
balanceData.value.forEach((item) => {
|
|
|
|
|
|
const day = new Date(item.date).getDate()
|
|
|
|
|
|
balanceMap.set(day, item.cumulativeBalance)
|
|
|
|
|
|
})
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
for (let i = 1; i <= daysInMonth; i++) {
|
|
|
|
|
|
const item = dataMap.get(i)
|
|
|
|
|
|
if (item) {
|
|
|
|
|
|
fullData.push(item)
|
|
|
|
|
|
} else {
|
|
|
|
|
|
fullData.push({
|
|
|
|
|
|
date: `${currentYear.value}-${String(currentMonth.value).padStart(2, '0')}-${String(i).padStart(2, '0')}`,
|
|
|
|
|
|
count: 0,
|
|
|
|
|
|
expense: 0,
|
|
|
|
|
|
income: 0,
|
|
|
|
|
|
balance: 0
|
|
|
|
|
|
})
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
dates = fullData.map((item) => {
|
|
|
|
|
|
const date = new Date(item.date)
|
|
|
|
|
|
return `${date.getDate()}日`
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
// 计算累计支出和收入
|
|
|
|
|
|
let accumulatedExpense = 0
|
|
|
|
|
|
let accumulatedIncome = 0
|
|
|
|
|
|
|
|
|
|
|
|
expenses = fullData.map((item) => {
|
|
|
|
|
|
accumulatedExpense += item.expense
|
|
|
|
|
|
return accumulatedExpense
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
incomes = fullData.map((item) => {
|
|
|
|
|
|
accumulatedIncome += item.income
|
|
|
|
|
|
return accumulatedIncome
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
// 使用余额接口数据
|
|
|
|
|
|
balances = fullData.map((item, index) => {
|
|
|
|
|
|
const day = index + 1
|
|
|
|
|
|
return balanceMap.get(day) || 0
|
|
|
|
|
|
})
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const legendData = [
|
|
|
|
|
|
{ name: '支出', value: '¥' + formatMoney(expenses[expenses.length - 1]) },
|
|
|
|
|
|
{ name: '收入', value: '¥' + formatMoney(incomes[incomes.length - 1]) },
|
|
|
|
|
|
{ name: '余额', value: '¥' + formatMoney(balances[balances.length - 1]) }
|
|
|
|
|
|
]
|
2026-01-21 16:09:38 +08:00
|
|
|
|
|
|
|
|
|
|
const option = {
|
|
|
|
|
|
tooltip: {
|
|
|
|
|
|
trigger: 'axis',
|
|
|
|
|
|
formatter: function (params) {
|
2026-02-01 10:27:04 +08:00
|
|
|
|
let result = params[0].name + '<br/>'
|
|
|
|
|
|
params.forEach((param) => {
|
|
|
|
|
|
result += param.marker + param.seriesName + ': ¥' + formatMoney(param.value) + '<br/>'
|
|
|
|
|
|
})
|
|
|
|
|
|
return result
|
|
|
|
|
|
}
|
|
|
|
|
|
},
|
|
|
|
|
|
legend: {
|
|
|
|
|
|
data: legendData.map((item) => item.name),
|
|
|
|
|
|
bottom: 0,
|
|
|
|
|
|
textStyle: {
|
|
|
|
|
|
color: getCssVar('--chart-text-muted')
|
|
|
|
|
|
},
|
|
|
|
|
|
formatter: function (name) {
|
|
|
|
|
|
const item = legendData.find((d) => d.name === name)
|
|
|
|
|
|
return item ? `${name} ${item.value}` : name
|
2026-01-21 16:09:38 +08:00
|
|
|
|
}
|
|
|
|
|
|
},
|
|
|
|
|
|
grid: {
|
|
|
|
|
|
left: '3%',
|
|
|
|
|
|
right: '4%',
|
2026-02-01 10:27:04 +08:00
|
|
|
|
bottom: '15%',
|
2026-01-21 16:09:38 +08:00
|
|
|
|
top: '5%',
|
|
|
|
|
|
containLabel: true
|
|
|
|
|
|
},
|
|
|
|
|
|
xAxis: {
|
|
|
|
|
|
type: 'category',
|
|
|
|
|
|
boundaryGap: false,
|
|
|
|
|
|
data: dates,
|
|
|
|
|
|
axisLabel: {
|
|
|
|
|
|
color: getCssVar('--chart-text-muted'),
|
|
|
|
|
|
fontSize: 11
|
|
|
|
|
|
}
|
|
|
|
|
|
},
|
|
|
|
|
|
yAxis: {
|
|
|
|
|
|
type: 'value',
|
|
|
|
|
|
splitNumber: 4,
|
|
|
|
|
|
axisLabel: {
|
|
|
|
|
|
color: getCssVar('--chart-text-muted'),
|
|
|
|
|
|
fontSize: 11,
|
|
|
|
|
|
formatter: (value) => {
|
|
|
|
|
|
return value / 1000 + 'k'
|
|
|
|
|
|
}
|
|
|
|
|
|
},
|
|
|
|
|
|
splitLine: {
|
|
|
|
|
|
lineStyle: {
|
|
|
|
|
|
type: 'dashed',
|
|
|
|
|
|
color: getCssVar('--van-border-color')
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
},
|
|
|
|
|
|
series: [
|
2026-02-01 10:27:04 +08:00
|
|
|
|
{
|
|
|
|
|
|
name: '支出',
|
|
|
|
|
|
type: 'line',
|
|
|
|
|
|
data: expenses,
|
|
|
|
|
|
itemStyle: { color: '#ff6b6b' },
|
|
|
|
|
|
showSymbol: false,
|
|
|
|
|
|
smooth: true,
|
|
|
|
|
|
lineStyle: { width: 2 }
|
|
|
|
|
|
},
|
|
|
|
|
|
{
|
|
|
|
|
|
name: '收入',
|
|
|
|
|
|
type: 'line',
|
|
|
|
|
|
data: incomes,
|
|
|
|
|
|
itemStyle: { color: '#51cf66' },
|
|
|
|
|
|
showSymbol: false,
|
|
|
|
|
|
smooth: true,
|
|
|
|
|
|
lineStyle: { width: 2 }
|
|
|
|
|
|
},
|
2026-01-21 16:09:38 +08:00
|
|
|
|
{
|
|
|
|
|
|
name: '余额',
|
|
|
|
|
|
type: 'line',
|
|
|
|
|
|
data: balances,
|
2026-02-01 10:27:04 +08:00
|
|
|
|
itemStyle: { color: '#4c9cf1' },
|
2026-01-21 16:09:38 +08:00
|
|
|
|
showSymbol: false,
|
|
|
|
|
|
smooth: true,
|
2026-02-01 10:27:04 +08:00
|
|
|
|
lineStyle: { width: 2 }
|
2026-01-21 16:09:38 +08:00
|
|
|
|
}
|
|
|
|
|
|
]
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
balanceChartInstance.setOption(option)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-26 17:13:57 +08:00
|
|
|
|
// 跳转到智能分析页面
|
|
|
|
|
|
const goToAnalysis = () => {
|
|
|
|
|
|
router.push('/bill-analysis')
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-09 19:25:51 +08:00
|
|
|
|
// 切换到统计V2页面
|
|
|
|
|
|
const goToStatisticsV2 = () => {
|
|
|
|
|
|
router.push('/statistics-v2')
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-26 17:56:08 +08:00
|
|
|
|
// 打开分类账单列表
|
|
|
|
|
|
const goToCategoryBills = (classify, type) => {
|
2025-12-27 22:45:23 +08:00
|
|
|
|
selectedClassify.value = classify || '未分类' // TODO 如果是未分类的 添加智能分类按钮
|
2025-12-26 17:56:08 +08:00
|
|
|
|
selectedType.value = type
|
|
|
|
|
|
selectedCategoryTitle.value = `${classify || '未分类'} - ${type === 0 ? '支出' : '收入'}`
|
2026-01-16 11:15:44 +08:00
|
|
|
|
|
2025-12-26 17:56:08 +08:00
|
|
|
|
// 重置分页状态
|
|
|
|
|
|
categoryBills.value = []
|
2025-12-27 22:05:50 +08:00
|
|
|
|
categoryBillsTotal.value = 0
|
|
|
|
|
|
billPageIndex.value = 1
|
2025-12-26 17:56:08 +08:00
|
|
|
|
billListFinished.value = false
|
2026-01-16 11:15:44 +08:00
|
|
|
|
|
2025-12-26 17:56:08 +08:00
|
|
|
|
billListVisible.value = true
|
2025-12-27 22:05:50 +08:00
|
|
|
|
// 打开弹窗后加载数据
|
|
|
|
|
|
loadCategoryBills()
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-29 21:17:18 +08:00
|
|
|
|
const smartClassifyButtonRef = ref(null)
|
2026-01-10 12:22:37 +08:00
|
|
|
|
const transactionListRef = ref(null)
|
2025-12-26 17:56:08 +08:00
|
|
|
|
// 加载分类账单数据
|
2025-12-29 21:17:18 +08:00
|
|
|
|
const loadCategoryBills = async (customIndex = null, customSize = null) => {
|
2026-01-16 11:15:44 +08:00
|
|
|
|
if (billListLoading.value || billListFinished.value) {
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-26 17:56:08 +08:00
|
|
|
|
billListLoading.value = true
|
|
|
|
|
|
try {
|
|
|
|
|
|
const params = {
|
2025-12-29 21:17:18 +08:00
|
|
|
|
pageIndex: customIndex || billPageIndex.value,
|
|
|
|
|
|
pageSize: customSize || billPageSize,
|
2025-12-26 17:56:08 +08:00
|
|
|
|
type: selectedType.value,
|
|
|
|
|
|
year: currentYear.value,
|
2026-01-28 17:00:58 +08:00
|
|
|
|
month: currentMonth.value || 0,
|
2025-12-27 22:05:50 +08:00
|
|
|
|
sortByAmount: true
|
2025-12-26 17:56:08 +08:00
|
|
|
|
}
|
2026-01-16 11:15:44 +08:00
|
|
|
|
|
2025-12-27 22:05:50 +08:00
|
|
|
|
// 仅当选择了分类时才添加classify参数
|
|
|
|
|
|
if (selectedClassify.value !== null) {
|
|
|
|
|
|
params.classify = selectedClassify.value
|
2025-12-26 17:56:08 +08:00
|
|
|
|
}
|
2026-01-16 11:15:44 +08:00
|
|
|
|
|
2025-12-26 17:56:08 +08:00
|
|
|
|
const response = await getTransactionList(params)
|
2026-01-16 11:15:44 +08:00
|
|
|
|
|
2025-12-26 17:56:08 +08:00
|
|
|
|
if (response.success) {
|
|
|
|
|
|
const newList = response.data || []
|
|
|
|
|
|
categoryBills.value = [...categoryBills.value, ...newList]
|
2025-12-27 22:05:50 +08:00
|
|
|
|
categoryBillsTotal.value = response.total
|
2026-01-16 11:15:44 +08:00
|
|
|
|
|
2025-12-27 22:05:50 +08:00
|
|
|
|
if (newList.length === 0 || newList.length < billPageSize) {
|
2025-12-26 17:56:08 +08:00
|
|
|
|
billListFinished.value = true
|
2025-12-27 22:05:50 +08:00
|
|
|
|
} else {
|
|
|
|
|
|
billListFinished.value = false
|
|
|
|
|
|
billPageIndex.value++
|
2025-12-26 17:56:08 +08:00
|
|
|
|
}
|
2025-12-29 21:17:18 +08:00
|
|
|
|
|
2025-12-30 18:49:46 +08:00
|
|
|
|
smartClassifyButtonRef.value?.reset()
|
2025-12-26 17:56:08 +08:00
|
|
|
|
} else {
|
|
|
|
|
|
showToast(response.message || '加载账单失败')
|
|
|
|
|
|
billListFinished.value = true
|
|
|
|
|
|
}
|
|
|
|
|
|
} catch (error) {
|
|
|
|
|
|
console.error('加载分类账单失败:', error)
|
|
|
|
|
|
showToast('加载账单失败')
|
|
|
|
|
|
billListFinished.value = true
|
|
|
|
|
|
} finally {
|
|
|
|
|
|
billListLoading.value = false
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 查看账单详情
|
|
|
|
|
|
const viewBillDetail = async (transaction) => {
|
|
|
|
|
|
try {
|
|
|
|
|
|
const response = await getTransactionDetail(transaction.id)
|
|
|
|
|
|
if (response.success) {
|
|
|
|
|
|
currentTransaction.value = response.data
|
|
|
|
|
|
detailVisible.value = true
|
|
|
|
|
|
} else {
|
|
|
|
|
|
showToast(response.message || '获取详情失败')
|
|
|
|
|
|
}
|
|
|
|
|
|
} catch (error) {
|
|
|
|
|
|
console.error('获取详情出错:', error)
|
|
|
|
|
|
showToast('获取详情失败')
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-01 11:58:21 +08:00
|
|
|
|
const handleCategoryBillsDelete = (deletedId) => {
|
2026-01-16 11:15:44 +08:00
|
|
|
|
categoryBills.value = categoryBills.value.filter((t) => t.id !== deletedId)
|
2026-01-01 11:58:21 +08:00
|
|
|
|
categoryBillsTotal.value--
|
|
|
|
|
|
|
|
|
|
|
|
// 被删除后刷新统计数据和账单列表
|
|
|
|
|
|
fetchStatistics()
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-26 17:56:08 +08:00
|
|
|
|
// 账单保存后的回调
|
2026-01-10 12:22:37 +08:00
|
|
|
|
const onBillSave = async (updatedTransaction) => {
|
2025-12-26 17:56:08 +08:00
|
|
|
|
// 刷新统计数据
|
|
|
|
|
|
await fetchStatistics()
|
2026-01-16 11:15:44 +08:00
|
|
|
|
|
2026-01-10 12:22:37 +08:00
|
|
|
|
// 只刷新列表中指定的账单项
|
2026-01-16 11:15:44 +08:00
|
|
|
|
const item = categoryBills.value.find((t) => t.id === updatedTransaction.id)
|
|
|
|
|
|
if (!item) {
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
2026-01-10 12:22:37 +08:00
|
|
|
|
|
|
|
|
|
|
// 如果分类发生了变化
|
2026-01-16 11:15:44 +08:00
|
|
|
|
if (item.classify !== updatedTransaction.classify) {
|
2026-01-10 12:22:37 +08:00
|
|
|
|
// 从列表中移除该项
|
2026-01-16 11:15:44 +08:00
|
|
|
|
categoryBills.value = categoryBills.value.filter((t) => t.id !== updatedTransaction.id)
|
2026-01-10 12:22:37 +08:00
|
|
|
|
categoryBillsTotal.value--
|
|
|
|
|
|
// 通知智能分类按钮组件移除指定项
|
|
|
|
|
|
smartClassifyButtonRef.value?.removeClassifiedTransaction(updatedTransaction.id)
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Object.assign(item, updatedTransaction)
|
|
|
|
|
|
|
2025-12-26 17:56:08 +08:00
|
|
|
|
showToast('保存成功')
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-29 21:17:18 +08:00
|
|
|
|
const beforeSmartClassify = async () => {
|
|
|
|
|
|
showToast({
|
|
|
|
|
|
message: '加载完整账单列表,请稍候...',
|
|
|
|
|
|
duration: 0,
|
|
|
|
|
|
forbidClick: true
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
await loadCategoryBills(1, categoryBillsTotal.value || 1000)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 智能分类保存后的回调
|
|
|
|
|
|
const onSmartClassifySave = async () => {
|
|
|
|
|
|
// 关闭账单列表弹窗
|
|
|
|
|
|
billListVisible.value = false
|
2026-01-16 11:15:44 +08:00
|
|
|
|
|
2025-12-29 21:17:18 +08:00
|
|
|
|
// 刷新统计数据
|
|
|
|
|
|
await fetchStatistics()
|
2026-01-16 11:15:44 +08:00
|
|
|
|
try {
|
|
|
|
|
|
window.dispatchEvent(
|
|
|
|
|
|
new CustomEvent('transactions-changed', {
|
|
|
|
|
|
detail: { reason: selectedClassify.value, type: selectedType.value }
|
|
|
|
|
|
})
|
|
|
|
|
|
)
|
|
|
|
|
|
} catch (e) {
|
2026-01-03 11:26:50 +08:00
|
|
|
|
console.error('触发 transactions-changed 事件失败:', e)
|
|
|
|
|
|
}
|
2026-01-16 11:15:44 +08:00
|
|
|
|
|
2025-12-29 21:17:18 +08:00
|
|
|
|
showToast('智能分类已保存')
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-30 18:49:46 +08:00
|
|
|
|
const handleNotifiedTransactionId = async (transactionId) => {
|
2026-01-16 15:56:53 +08:00
|
|
|
|
console.info('收到已处理交易ID通知:', transactionId)
|
2025-12-30 18:49:46 +08:00
|
|
|
|
// 滚动到指定的交易项
|
2026-01-16 11:15:44 +08:00
|
|
|
|
const index = categoryBills.value.findIndex((item) => String(item.id) === String(transactionId))
|
2025-12-30 18:49:46 +08:00
|
|
|
|
if (index !== -1) {
|
|
|
|
|
|
// 等待 DOM 更新
|
|
|
|
|
|
await nextTick()
|
2026-01-16 11:15:44 +08:00
|
|
|
|
|
2026-01-10 12:22:37 +08:00
|
|
|
|
// 允许一丁点延迟让浏览器响应渲染
|
2026-01-16 11:15:44 +08:00
|
|
|
|
await new Promise((resolve) => setTimeout(resolve, 0))
|
|
|
|
|
|
|
2026-01-10 12:22:37 +08:00
|
|
|
|
const listElement = transactionListRef.value?.$el
|
2025-12-30 18:49:46 +08:00
|
|
|
|
if (listElement) {
|
|
|
|
|
|
const items = listElement.querySelectorAll('.transaction-item')
|
|
|
|
|
|
const itemElement = items[index]
|
|
|
|
|
|
if (itemElement) {
|
|
|
|
|
|
itemElement.scrollIntoView({ behavior: 'smooth', block: 'center' })
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-26 17:13:57 +08:00
|
|
|
|
// 初始化
|
|
|
|
|
|
onMounted(() => {
|
|
|
|
|
|
fetchStatistics()
|
2026-01-15 22:10:57 +08:00
|
|
|
|
window.addEventListener('resize', handleResize)
|
2025-12-26 17:13:57 +08:00
|
|
|
|
})
|
2025-12-26 17:56:08 +08:00
|
|
|
|
|
2026-01-15 22:10:57 +08:00
|
|
|
|
const handleResize = () => {
|
|
|
|
|
|
pieChartInstance && pieChartInstance.resize()
|
2026-01-21 16:09:38 +08:00
|
|
|
|
balanceChartInstance && balanceChartInstance.resize()
|
2026-01-15 22:10:57 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-16 11:15:44 +08:00
|
|
|
|
// 监听DOM引用变化,确保在月份切换DOM重建后重新渲染图表
|
|
|
|
|
|
watch(pieChartRef, (newVal) => {
|
|
|
|
|
|
if (newVal) {
|
|
|
|
|
|
setTimeout(() => {
|
|
|
|
|
|
renderPieChart()
|
|
|
|
|
|
pieChartInstance && pieChartInstance.resize()
|
|
|
|
|
|
}, 50)
|
|
|
|
|
|
}
|
|
|
|
|
|
})
|
|
|
|
|
|
|
2026-01-21 16:09:38 +08:00
|
|
|
|
watch(balanceChartRef, (newVal) => {
|
|
|
|
|
|
if (newVal) {
|
|
|
|
|
|
setTimeout(() => {
|
|
|
|
|
|
renderBalanceChart()
|
|
|
|
|
|
balanceChartInstance && balanceChartInstance.resize()
|
|
|
|
|
|
}, 50)
|
|
|
|
|
|
}
|
|
|
|
|
|
})
|
|
|
|
|
|
|
2025-12-26 17:56:08 +08:00
|
|
|
|
// 页面激活时刷新数据(从其他页面返回时)
|
|
|
|
|
|
onActivated(() => {
|
|
|
|
|
|
fetchStatistics()
|
|
|
|
|
|
})
|
2026-01-01 11:58:21 +08:00
|
|
|
|
|
|
|
|
|
|
// 全局监听交易删除事件,确保统计数据一致
|
2026-01-03 11:26:50 +08:00
|
|
|
|
const onGlobalTransactionDeleted = () => {
|
2026-01-01 11:58:21 +08:00
|
|
|
|
// e.detail contains transaction id
|
|
|
|
|
|
fetchStatistics()
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-16 11:15:44 +08:00
|
|
|
|
window.addEventListener &&
|
|
|
|
|
|
window.addEventListener('transaction-deleted', onGlobalTransactionDeleted)
|
2026-01-01 11:58:21 +08:00
|
|
|
|
|
|
|
|
|
|
onBeforeUnmount(() => {
|
2026-01-16 11:15:44 +08:00
|
|
|
|
window.removeEventListener &&
|
|
|
|
|
|
window.removeEventListener('transaction-deleted', onGlobalTransactionDeleted)
|
2026-01-15 22:10:57 +08:00
|
|
|
|
window.removeEventListener('resize', handleResize)
|
|
|
|
|
|
pieChartInstance && pieChartInstance.dispose()
|
2026-01-21 16:09:38 +08:00
|
|
|
|
balanceChartInstance && balanceChartInstance.dispose()
|
2026-01-01 11:58:21 +08:00
|
|
|
|
})
|
|
|
|
|
|
|
2026-01-03 11:26:50 +08:00
|
|
|
|
const onGlobalTransactionsChanged = () => {
|
2026-01-01 11:58:21 +08:00
|
|
|
|
fetchStatistics()
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-16 11:15:44 +08:00
|
|
|
|
window.addEventListener &&
|
|
|
|
|
|
window.addEventListener('transactions-changed', onGlobalTransactionsChanged)
|
2026-01-01 11:58:21 +08:00
|
|
|
|
|
|
|
|
|
|
onBeforeUnmount(() => {
|
2026-01-16 11:15:44 +08:00
|
|
|
|
window.removeEventListener &&
|
|
|
|
|
|
window.removeEventListener('transactions-changed', onGlobalTransactionsChanged)
|
2026-01-01 11:58:21 +08:00
|
|
|
|
})
|
2026-01-28 17:00:58 +08:00
|
|
|
|
|
|
|
|
|
|
// 监听日期选择模式变化,更新selectedDate数组
|
|
|
|
|
|
watch(dateSelectionMode, (newMode) => {
|
|
|
|
|
|
if (newMode === 'year') {
|
|
|
|
|
|
// 切换到年份模式:只保留年份
|
|
|
|
|
|
selectedDate.value = [currentYear.value.toString()]
|
|
|
|
|
|
} else {
|
|
|
|
|
|
// 切换到月份模式:添加当前月份
|
|
|
|
|
|
selectedDate.value = [
|
|
|
|
|
|
currentYear.value.toString(),
|
|
|
|
|
|
(currentMonth.value || new Date().getMonth() + 1).toString().padStart(2, '0')
|
|
|
|
|
|
]
|
|
|
|
|
|
}
|
|
|
|
|
|
})
|
2025-12-26 17:13:57 +08:00
|
|
|
|
</script>
|
|
|
|
|
|
|
|
|
|
|
|
<style scoped>
|
2026-01-16 11:15:44 +08:00
|
|
|
|
.page-container-flex {
|
2026-01-11 16:50:18 +08:00
|
|
|
|
background: transparent !important;
|
2026-01-11 16:44:32 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-27 21:15:26 +08:00
|
|
|
|
.statistics-content {
|
2026-01-11 16:33:55 +08:00
|
|
|
|
padding: 0;
|
|
|
|
|
|
overflow-x: hidden; /* 防止滑动动画出现横向滚动条 */
|
2025-12-27 20:57:15 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-27 21:15:26 +08:00
|
|
|
|
:deep(.van-pull-refresh) {
|
2025-12-27 20:57:15 +08:00
|
|
|
|
flex: 1;
|
|
|
|
|
|
overflow-y: auto;
|
|
|
|
|
|
-webkit-overflow-scrolling: touch;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-16 15:56:53 +08:00
|
|
|
|
/* 导航栏日期选择器 */
|
|
|
|
|
|
.nav-date-picker {
|
2026-01-11 16:33:55 +08:00
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
2026-01-16 15:56:53 +08:00
|
|
|
|
gap: 4px;
|
2025-12-27 22:05:50 +08:00
|
|
|
|
cursor: pointer;
|
2026-01-16 15:56:53 +08:00
|
|
|
|
color: var(--van-text-color);
|
2025-12-26 17:13:57 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-09 19:25:51 +08:00
|
|
|
|
/* 导航栏右侧按钮组 */
|
|
|
|
|
|
.nav-right-buttons {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-21 16:09:38 +08:00
|
|
|
|
/* 余额卡片 */
|
|
|
|
|
|
.balance-amount {
|
|
|
|
|
|
text-align: center;
|
|
|
|
|
|
padding: 9px 0 8px 0;
|
|
|
|
|
|
font-size: 32px;
|
|
|
|
|
|
font-weight: bold;
|
|
|
|
|
|
color: var(--van-text-color);
|
|
|
|
|
|
font-family: -apple-system, BlinkMacSystemFont, "SF Pro Display", "SF Pro Text",
|
|
|
|
|
|
"Segoe UI", Roboto, "Helvetica Neue", Arial, sans-serif;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.balance-value {
|
|
|
|
|
|
color: var(--chart-color-13);
|
|
|
|
|
|
transition: color 0.3s ease;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.balance-value.balance-positive {
|
|
|
|
|
|
color: var(--van-success-color);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.balance-value.balance-negative {
|
|
|
|
|
|
color: var(--van-danger-color);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.balance-chart {
|
|
|
|
|
|
border-radius: 8px;
|
|
|
|
|
|
padding: 10px 0;
|
|
|
|
|
|
margin: 0 -12px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-26 17:13:57 +08:00
|
|
|
|
/* 环形图 */
|
|
|
|
|
|
.chart-container {
|
2026-01-16 17:52:40 +08:00
|
|
|
|
padding: 0;
|
2025-12-26 17:13:57 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.ring-chart {
|
|
|
|
|
|
position: relative;
|
2026-01-15 22:10:57 +08:00
|
|
|
|
width: 100%;
|
2025-12-26 17:13:57 +08:00
|
|
|
|
height: 200px;
|
|
|
|
|
|
margin: 0 auto;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.ring-svg {
|
|
|
|
|
|
width: 100%;
|
|
|
|
|
|
height: 100%;
|
|
|
|
|
|
transform: scale(1);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.ring-segment {
|
|
|
|
|
|
transition: all 0.3s ease;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.ring-center {
|
|
|
|
|
|
position: absolute;
|
|
|
|
|
|
top: 50%;
|
|
|
|
|
|
left: 50%;
|
|
|
|
|
|
transform: translate(-50%, -50%);
|
|
|
|
|
|
text-align: center;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.center-value {
|
|
|
|
|
|
font-size: 20px;
|
|
|
|
|
|
font-weight: 700;
|
|
|
|
|
|
color: var(--van-text-color);
|
|
|
|
|
|
margin-bottom: 4px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.center-label {
|
|
|
|
|
|
font-size: 13px;
|
|
|
|
|
|
color: var(--van-text-color-2);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* 分类列表 */
|
|
|
|
|
|
.category-list {
|
2025-12-26 17:29:17 +08:00
|
|
|
|
padding: 0;
|
2025-12-26 17:13:57 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.category-item {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
padding: 12px 0;
|
2026-01-13 17:00:44 +08:00
|
|
|
|
border-bottom: 1px solid var(--van-border-color);
|
2025-12-26 17:56:08 +08:00
|
|
|
|
transition: background-color 0.2s;
|
|
|
|
|
|
gap: 12px;
|
2025-12-26 17:29:17 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-26 17:13:57 +08:00
|
|
|
|
.category-item:last-child {
|
|
|
|
|
|
border-bottom: none;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-26 17:56:08 +08:00
|
|
|
|
.category-item.clickable {
|
|
|
|
|
|
cursor: pointer;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.category-item.clickable:active {
|
2026-01-13 17:00:44 +08:00
|
|
|
|
background-color: var(--van-background);
|
2025-12-26 17:56:08 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-26 17:13:57 +08:00
|
|
|
|
.category-info {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
gap: 12px;
|
2025-12-26 17:56:08 +08:00
|
|
|
|
flex: 1;
|
|
|
|
|
|
min-width: 0;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.category-name-with-count {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
gap: 8px;
|
2025-12-26 17:13:57 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.category-color {
|
|
|
|
|
|
width: 12px;
|
|
|
|
|
|
height: 12px;
|
|
|
|
|
|
border-radius: 50%;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.category-name {
|
|
|
|
|
|
font-size: 14px;
|
|
|
|
|
|
color: var(--van-text-color);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-26 17:56:08 +08:00
|
|
|
|
.category-count {
|
|
|
|
|
|
font-size: 12px;
|
|
|
|
|
|
color: var(--van-text-color-3);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-26 17:13:57 +08:00
|
|
|
|
.category-stats {
|
|
|
|
|
|
display: flex;
|
2025-12-26 17:56:08 +08:00
|
|
|
|
align-items: center;
|
|
|
|
|
|
gap: 8px;
|
|
|
|
|
|
flex-shrink: 0;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.category-arrow {
|
|
|
|
|
|
margin-left: 8px;
|
|
|
|
|
|
color: var(--van-text-color-3);
|
|
|
|
|
|
font-size: 16px;
|
|
|
|
|
|
flex-shrink: 0;
|
2025-12-26 17:13:57 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-16 12:08:11 +08:00
|
|
|
|
.expand-toggle {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
justify-content: center;
|
|
|
|
|
|
align-items: center;
|
2026-01-16 17:52:40 +08:00
|
|
|
|
padding-top: 0;
|
2026-01-16 12:08:11 +08:00
|
|
|
|
color: var(--van-text-color-3);
|
|
|
|
|
|
font-size: 20px;
|
|
|
|
|
|
cursor: pointer;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.expand-toggle:active {
|
|
|
|
|
|
opacity: 0.7;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-26 17:13:57 +08:00
|
|
|
|
.category-amount {
|
|
|
|
|
|
font-size: 15px;
|
|
|
|
|
|
font-weight: 600;
|
|
|
|
|
|
color: var(--van-text-color);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.category-percent {
|
|
|
|
|
|
font-size: 12px;
|
|
|
|
|
|
color: var(--van-text-color-3);
|
2026-01-13 17:00:44 +08:00
|
|
|
|
background: var(--van-background);
|
2025-12-26 17:13:57 +08:00
|
|
|
|
padding: 2px 8px;
|
|
|
|
|
|
border-radius: 10px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.income-color {
|
2026-01-13 17:00:44 +08:00
|
|
|
|
background-color: var(--van-success-color);
|
2025-12-26 17:13:57 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.income-text {
|
2026-01-13 17:00:44 +08:00
|
|
|
|
color: var(--van-success-color);
|
2025-12-26 17:13:57 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-27 22:34:19 +08:00
|
|
|
|
/* 不计收支颜色 */
|
|
|
|
|
|
.none-color {
|
2026-01-13 17:00:44 +08:00
|
|
|
|
background-color: var(--van-gray-6);
|
2025-12-27 22:34:19 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.none-text {
|
2026-01-13 17:00:44 +08:00
|
|
|
|
color: var(--van-gray-6);
|
2025-12-27 22:34:19 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-26 17:13:57 +08:00
|
|
|
|
.expense-color {
|
2026-01-13 17:00:44 +08:00
|
|
|
|
background-color: var(--van-danger-color);
|
2025-12-26 17:13:57 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* 趋势图 */
|
|
|
|
|
|
.trend-chart {
|
|
|
|
|
|
padding: 20px 16px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.trend-bars {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
justify-content: space-between;
|
|
|
|
|
|
align-items: flex-end;
|
|
|
|
|
|
height: 180px;
|
|
|
|
|
|
margin-bottom: 16px;
|
|
|
|
|
|
padding: 0 4px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.trend-bar-group {
|
|
|
|
|
|
flex: 1;
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
flex-direction: column;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
gap: 8px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.bar-container {
|
|
|
|
|
|
width: 100%;
|
|
|
|
|
|
height: 150px;
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
justify-content: center;
|
|
|
|
|
|
align-items: flex-end;
|
|
|
|
|
|
gap: 4px;
|
|
|
|
|
|
padding: 0 2px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.bar {
|
|
|
|
|
|
flex: 1;
|
|
|
|
|
|
max-width: 20px;
|
|
|
|
|
|
min-height: 4px;
|
|
|
|
|
|
border-radius: 4px 4px 0 0;
|
|
|
|
|
|
position: relative;
|
|
|
|
|
|
transition: all 0.3s ease;
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: flex-start;
|
|
|
|
|
|
justify-content: center;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.expense-bar {
|
|
|
|
|
|
background: linear-gradient(180deg, #ff6b6b 0%, #ff8787 100%);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.income-bar {
|
|
|
|
|
|
background: linear-gradient(180deg, #51cf66 0%, #69db7c 100%);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.bar-value {
|
|
|
|
|
|
font-size: 10px;
|
|
|
|
|
|
color: var(--van-text-color-2);
|
|
|
|
|
|
white-space: nowrap;
|
|
|
|
|
|
margin-top: -18px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.bar-label {
|
|
|
|
|
|
font-size: 11px;
|
|
|
|
|
|
color: var(--van-text-color-3);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.trend-legend {
|
|
|
|
|
|
display: flex;
|
2026-01-13 17:00:44 +08:00
|
|
|
|
justify-content: space-center;
|
2025-12-26 17:13:57 +08:00
|
|
|
|
gap: 24px;
|
|
|
|
|
|
padding-top: 12px;
|
2026-01-13 17:00:44 +08:00
|
|
|
|
border-top: 1px solid var(--van-border-color);
|
2025-12-26 17:13:57 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.legend-item {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
gap: 6px;
|
|
|
|
|
|
font-size: 13px;
|
|
|
|
|
|
color: var(--van-text-color-2);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.legend-color {
|
|
|
|
|
|
width: 12px;
|
|
|
|
|
|
height: 12px;
|
|
|
|
|
|
border-radius: 2px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-26 18:03:52 +08:00
|
|
|
|
/* 设置页面容器背景色 */
|
|
|
|
|
|
:deep(.van-nav-bar) {
|
|
|
|
|
|
background: transparent !important;
|
|
|
|
|
|
}
|
2026-01-16 12:27:00 +08:00
|
|
|
|
|
|
|
|
|
|
/* 并列显示卡片 */
|
|
|
|
|
|
.side-by-side-cards {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
gap: 12px;
|
|
|
|
|
|
margin: 0 12px 16px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.side-by-side-cards .common-card {
|
|
|
|
|
|
margin: 0;
|
|
|
|
|
|
flex: 1;
|
|
|
|
|
|
min-width: 0; /* 允许内部元素缩小 */
|
|
|
|
|
|
padding: 12px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-21 16:09:38 +08:00
|
|
|
|
.card-header {
|
|
|
|
|
|
margin-bottom: 0;
|
2026-01-16 12:27:00 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.text-ellipsis {
|
|
|
|
|
|
overflow: hidden;
|
|
|
|
|
|
text-overflow: ellipsis;
|
|
|
|
|
|
white-space: nowrap;
|
|
|
|
|
|
font-size: 13px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-28 17:00:58 +08:00
|
|
|
|
/* 日期选择器头部 */
|
|
|
|
|
|
.date-picker-header {
|
|
|
|
|
|
padding: 12px 16px 0;
|
|
|
|
|
|
background: var(--van-background-2);
|
|
|
|
|
|
border-bottom: 1px solid var(--van-border-color);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.date-picker-header :deep(.van-tabs) {
|
|
|
|
|
|
background: transparent;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.date-picker-header :deep(.van-tabs__nav) {
|
|
|
|
|
|
background: transparent;
|
|
|
|
|
|
padding-bottom: 0;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.date-picker-header :deep(.van-tab) {
|
|
|
|
|
|
font-size: 15px;
|
|
|
|
|
|
font-weight: 500;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-13 19:11:37 +08:00
|
|
|
|
</style>
|