Files
EmailBill/Web/src/views/StatisticsView.vue

1162 lines
29 KiB
Vue
Raw Normal View History

2025-12-26 17:13:57 +08:00
<template>
2025-12-27 21:15:26 +08:00
<div class="page-container-flex">
2025-12-26 17:13:57 +08:00
<!-- 顶部导航栏 -->
<van-nav-bar title="账单统计" placeholder>
<template #right>
<van-icon name="chat-o" size="20" @click="goToAnalysis" />
</template>
</van-nav-bar>
2025-12-27 20:57:15 +08:00
<!-- 月份选择器 固定区域 -->
2025-12-27 21:15:26 +08:00
<div class="sticky-header">
2025-12-27 20:57:15 +08:00
<van-button
icon="arrow-left"
plain
size="small"
@click="changeMonth(-1)"
/>
2025-12-27 21:15:26 +08:00
<div class="sticky-header-text" @click="showMonthPicker = true">
2025-12-27 20:57:15 +08:00
{{ currentYear }}{{ currentMonth }}
<van-icon name="arrow-down" />
</div>
<van-button
icon="arrow"
plain
size="small"
@click="changeMonth(1)"
:disabled="isCurrentMonth"
/>
</div>
2025-12-26 17:13:57 +08:00
<!-- 下拉刷新 -->
<van-pull-refresh v-model="refreshing" @refresh="onRefresh">
<!-- 加载中 -->
<van-loading v-if="loading" vertical style="padding: 100px 0">
加载统计数据中...
</van-loading>
<!-- 统计内容 -->
<div v-else class="statistics-content">
<!-- 月度概览卡片 -->
<div class="overview-card">
2025-12-27 22:05:50 +08:00
<div class="overview-item clickable" @click="goToTypeOverviewBills(0)">
2025-12-26 17:13:57 +08:00
<div class="label">总支出</div>
<div class="value expense">¥{{ formatMoney(monthlyData.totalExpense) }}</div>
<div class="sub-text">{{ monthlyData.expenseCount }}</div>
</div>
<div class="divider"></div>
2025-12-27 22:05:50 +08:00
<div class="overview-item clickable" @click="goToTypeOverviewBills(1)">
2025-12-26 17:13:57 +08:00
<div class="label">总收入</div>
<div class="value income">¥{{ formatMoney(monthlyData.totalIncome) }}</div>
<div class="sub-text">{{ monthlyData.incomeCount }}</div>
</div>
<div class="divider"></div>
2025-12-27 22:05:50 +08:00
<div class="overview-item clickable" @click="goToTypeOverviewBills(null)">
2025-12-26 17:13:57 +08:00
<div class="label">结余</div>
<div class="value" :class="monthlyData.balance >= 0 ? 'income' : 'expense'">
{{ monthlyData.balance >= 0 ? '' : '-' }}¥{{ formatMoney(Math.abs(monthlyData.balance)) }}
</div>
<div class="sub-text">{{ monthlyData.totalCount }}笔交易</div>
</div>
</div>
<!-- 分类统计 -->
2025-12-26 17:29:17 +08:00
<div class="common-card">
<div class="card-header">
<h3 class="card-title">支出分类统计</h3>
2025-12-26 17:13:57 +08:00
<van-tag type="primary" size="medium">{{ expenseCategories.length }}</van-tag>
</div>
<!-- 环形图区域 -->
<div class="chart-container" v-if="expenseCategories.length > 0">
<div class="ring-chart">
<svg viewBox="0 0 200 200" class="ring-svg">
<circle
v-for="(segment, index) in chartSegments"
:key="index"
cx="100"
cy="100"
r="70"
fill="none"
:stroke="segment.color"
:stroke-width="35"
:stroke-dasharray="`${segment.length} ${circumference - segment.length}`"
:stroke-dashoffset="-segment.offset"
transform="rotate(-90 100 100)"
class="ring-segment"
/>
</svg>
<div class="ring-center">
<div class="center-value">¥{{ formatMoney(monthlyData.totalExpense) }}</div>
<div class="center-label">总支出</div>
</div>
</div>
</div>
<!-- 分类列表 -->
<div class="category-list" v-if="expenseCategories.length > 0">
<div
v-for="(category) in expenseCategories"
:key="category.classify"
2025-12-26 17:56:08 +08:00
class="category-item clickable"
@click="goToCategoryBills(category.classify, 0)"
2025-12-26 17:13:57 +08:00
>
<div class="category-info">
<div class="category-color" :style="{ backgroundColor: category.color }"></div>
2025-12-26 17:56:08 +08:00
<div class="category-name-with-count">
<span class="category-name">{{ category.classify || '未分类' }}</span>
<span class="category-count">{{ category.count }}</span>
</div>
2025-12-26 17:13:57 +08:00
</div>
<div class="category-stats">
<div class="category-amount">¥{{ formatMoney(category.amount) }}</div>
<div class="category-percent">{{ category.percent }}%</div>
</div>
2025-12-26 17:56:08 +08:00
<van-icon name="arrow" class="category-arrow" />
2025-12-26 17:13:57 +08:00
</div>
</div>
<van-empty
v-else
description="本月暂无支出记录"
image="search"
/>
</div>
<!-- 收入分类统计 -->
2025-12-26 17:29:17 +08:00
<div class="common-card" v-if="incomeCategories.length > 0">
<div class="card-header">
<h3 class="card-title">收入分类统计</h3>
2025-12-26 17:13:57 +08:00
<van-tag type="success" size="medium">{{ incomeCategories.length }}</van-tag>
</div>
<div class="category-list">
<div
v-for="category in incomeCategories"
:key="category.classify"
2025-12-26 17:56:08 +08:00
class="category-item clickable"
@click="goToCategoryBills(category.classify, 1)"
2025-12-26 17:13:57 +08:00
>
<div class="category-info">
<div class="category-color income-color"></div>
2025-12-26 17:56:08 +08:00
<div class="category-name-with-count">
<span class="category-name">{{ category.classify || '未分类' }}</span>
<span class="category-count">{{ category.count }}</span>
</div>
2025-12-26 17:13:57 +08:00
</div>
<div class="category-stats">
<div class="category-amount income-text">¥{{ formatMoney(category.amount) }}</div>
<div class="category-percent">{{ category.percent }}%</div>
</div>
2025-12-26 17:56:08 +08:00
<van-icon name="arrow" class="category-arrow" />
2025-12-26 17:13:57 +08:00
</div>
</div>
</div>
2025-12-27 22:34:19 +08:00
<!-- 不计收支分类统计 -->
<div class="common-card" v-if="noneCategories.length > 0">
<div class="card-header">
<h3 class="card-title">不计收支分类统计</h3>
<van-tag type="info" size="medium">{{ noneCategories.length }}</van-tag>
</div>
<div class="category-list">
<div
v-for="category in noneCategories"
:key="category.classify"
class="category-item clickable"
@click="goToCategoryBills(category.classify, 2)"
>
<div class="category-info">
<div class="category-color none-color"></div>
<div class="category-name-with-count">
<span class="category-name">{{ category.classify || '未分类' }}</span>
<span class="category-count">{{ category.count }}</span>
</div>
</div>
<div class="category-stats">
<div class="category-amount none-text">¥{{ formatMoney(category.amount) }}</div>
<div class="category-percent">{{ category.percent }}%</div>
</div>
<van-icon name="arrow" class="category-arrow" />
</div>
</div>
</div>
2025-12-26 17:13:57 +08:00
<!-- 趋势统计 -->
2025-12-26 17:29:17 +08:00
<div class="common-card">
<div class="card-header">
<h3 class="card-title">近6个月趋势</h3>
2025-12-26 17:13:57 +08:00
</div>
<div class="trend-chart">
<div class="trend-bars">
<div
v-for="item in trendData"
:key="item.month"
class="trend-bar-group"
>
<div class="bar-container">
<div
class="bar expense-bar"
:style="{ height: getBarHeight(item.expense, maxTrendValue) }"
>
<div class="bar-value" v-if="item.expense > 0">
{{ formatShortMoney(item.expense) }}
</div>
</div>
<div
class="bar income-bar"
:style="{ height: getBarHeight(item.income, maxTrendValue) }"
>
<div class="bar-value" v-if="item.income > 0">
{{ formatShortMoney(item.income) }}
</div>
</div>
</div>
<div class="bar-label">{{ item.label }}</div>
</div>
</div>
<div class="trend-legend">
<div class="legend-item">
<div class="legend-color expense-color"></div>
<span>支出</span>
</div>
<div class="legend-item">
<div class="legend-color income-color"></div>
<span>收入</span>
</div>
</div>
</div>
</div>
<!-- 其他统计 -->
2025-12-26 17:29:17 +08:00
<div class="common-card">
<div class="card-header">
<h3 class="card-title">其他统计</h3>
2025-12-26 17:13:57 +08:00
</div>
<div class="other-stats">
<div class="stat-item">
<div class="stat-label">日均支出</div>
<div class="stat-value">¥{{ formatMoney(dailyAverage.expense) }}</div>
</div>
<div class="stat-item">
<div class="stat-label">日均收入</div>
<div class="stat-value income-text">¥{{ formatMoney(dailyAverage.income) }}</div>
</div>
<div class="stat-item">
<div class="stat-label">最大单笔支出</div>
<div class="stat-value">¥{{ formatMoney(monthlyData.maxExpense) }}</div>
</div>
<div class="stat-item">
<div class="stat-label">最大单笔收入</div>
<div class="stat-value income-text">¥{{ formatMoney(monthlyData.maxIncome) }}</div>
</div>
</div>
</div>
<!-- 底部安全距离 -->
<div style="height: calc(50px + env(safe-area-inset-bottom, 0px))"></div>
</div>
</van-pull-refresh>
<!-- 月份选择器 -->
<van-popup v-model:show="showMonthPicker" position="bottom" round>
<van-date-picker
v-model="selectedDate"
title="选择月份"
:min-date="minDate"
:max-date="maxDate"
:columns-type="['year', 'month']"
@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} 笔交易` : ''"
height="85%"
2025-12-26 17:56:08 +08:00
>
2025-12-30 17:02:30 +08:00
<template #header-actions>
<SmartClassifyButton
ref="smartClassifyButtonRef"
v-if="isUnclassified"
:transactions="categoryBills"
:onBeforeClassify="beforeSmartClassify"
@save="onSmartClassifySave"
2025-12-30 18:49:46 +08:00
@notify-doned-transaction-id="handleNotifiedTransactionId"
2025-12-30 17:02:30 +08:00
/>
</template>
<TransactionList
:transactions="categoryBills"
:loading="billListLoading"
:finished="billListFinished"
:show-delete="false"
@load="loadCategoryBills"
@click="viewBillDetail"
/>
</PopupContainer>
2025-12-26 17:56:08 +08:00
<!-- 交易详情编辑组件 -->
<TransactionDetail
v-model:show="detailVisible"
:transaction="currentTransaction"
@save="onBillSave"
/>
2025-12-26 17:13:57 +08:00
</div>
</template>
<script setup>
2025-12-30 18:49:46 +08:00
import { ref, computed, onMounted, onActivated, nextTick } from 'vue'
2025-12-26 17:13:57 +08:00
import { showToast } from 'vant'
import { useRouter } from 'vue-router'
import { getMonthlyStatistics, getCategoryStatistics, getTrendStatistics } from '@/api/statistics'
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'
2025-12-26 17:13:57 +08:00
const router = useRouter()
// 响应式数据
const loading = ref(true)
const refreshing = ref(false)
const showMonthPicker = ref(false)
const currentYear = ref(new Date().getFullYear())
const currentMonth = ref(new Date().getMonth() + 1)
const selectedDate = ref([new Date().getFullYear().toString(), (new Date().getMonth() + 1).toString().padStart(2, '0')])
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)
2025-12-29 21:17:18 +08:00
let 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,
totalCount: 0,
maxExpense: 0,
maxIncome: 0
})
// 分类数据
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
// 趋势数据
const trendData = ref([])
// 日期范围
const minDate = new Date(2020, 0, 1)
const maxDate = new Date()
// 颜色配置
const colors = [
'#FF6B6B', '#4ECDC4', '#45B7D1', '#FFA07A', '#98D8C8',
'#F7DC6F', '#BB8FCE', '#85C1E2', '#F8B88B', '#AAB7B8',
'#FF8ED4', '#67E6DC', '#FFAB73', '#C9B1FF', '#7BDFF2'
]
// 计算环形图数据
const circumference = computed(() => 2 * Math.PI * 70)
const chartSegments = computed(() => {
let offset = 0
return expenseCategories.value.map((category) => {
const percent = category.percent / 100
const length = circumference.value * percent
const segment = {
color: category.color,
length,
offset
}
offset += length
return segment
})
})
// 日均统计
const dailyAverage = computed(() => {
const daysInMonth = new Date(currentYear.value, currentMonth.value, 0).getDate()
return {
expense: monthlyData.value.totalExpense / daysInMonth,
income: monthlyData.value.totalIncome / daysInMonth
}
})
// 趋势图最大值
const maxTrendValue = computed(() => {
const allValues = trendData.value.flatMap(item => [item.expense, item.income])
return Math.max(...allValues, 1)
})
// 是否是当前月
const isCurrentMonth = computed(() => {
const now = new Date()
return currentYear.value === now.getFullYear() && currentMonth.value === now.getMonth() + 1
})
2025-12-29 21:17:18 +08:00
// 是否为未分类账单
const isUnclassified = computed(() => {
return selectedClassify.value === '未分类' || selectedClassify.value === ''
})
2025-12-26 17:13:57 +08:00
// 格式化金额
const formatMoney = (value) => {
2025-12-27 22:44:28 +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
}
// 格式化短金额k为单位
const formatShortMoney = (value) => {
if (!value) return '0'
if (value >= 10000) {
return (value / 10000).toFixed(1) + 'w'
}
if (value >= 1000) {
return (value / 1000).toFixed(1) + 'k'
}
return value.toFixed(0)
}
// 获取柱状图高度
const getBarHeight = (value, maxValue) => {
if (!value || !maxValue) return '0%'
const percent = (value / maxValue) * 100
return Math.max(percent, 5) + '%' // 最小5%以便显示
}
// 切换月份
const changeMonth = (offset) => {
let newMonth = currentMonth.value + offset
let newYear = currentYear.value
if (newMonth > 12) {
newMonth = 1
newYear++
} else if (newMonth < 1) {
newMonth = 12
newYear--
}
// 不能超过当前月份
const now = new Date()
const targetDate = new Date(newYear, newMonth - 1)
if (targetDate > now) {
return
}
currentYear.value = newYear
currentMonth.value = newMonth
fetchStatistics()
}
// 确认月份选择
const onMonthConfirm = ({ selectedValues }) => {
currentYear.value = parseInt(selectedValues[0])
currentMonth.value = parseInt(selectedValues[1])
showMonthPicker.value = false
fetchStatistics()
}
// 下拉刷新
const onRefresh = async () => {
await fetchStatistics()
refreshing.value = false
}
// 获取统计数据
const fetchStatistics = async () => {
loading.value = true
try {
await Promise.all([
fetchMonthlyData(),
fetchCategoryData(),
fetchTrendData()
])
} catch (error) {
console.error('获取统计数据失败:', error)
showToast('获取统计数据失败')
} finally {
loading.value = false
}
}
// 获取月度数据
const fetchMonthlyData = async () => {
try {
const response = await getMonthlyStatistics({
year: currentYear.value,
month: currentMonth.value
})
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,
month: currentMonth.value,
type: 0 // 支出
})
if (expenseResponse.success && expenseResponse.data) {
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,
color: colors[index % colors.length]
}))
}
// 获取收入分类
const incomeResponse = await getCategoryStatistics({
year: currentYear.value,
month: currentMonth.value,
type: 1 // 收入
})
if (incomeResponse.success && incomeResponse.data) {
incomeCategories.value = incomeResponse.data.map(item => ({
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,
month: currentMonth.value,
type: 2 // 不计收支
})
if (noneResponse.success && noneResponse.data) {
noneCategories.value = noneResponse.data.map(item => ({
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('获取分类数据失败')
}
}
// 获取趋势数据
const fetchTrendData = async () => {
try {
// 计算开始年月当前月往前推5个月
let startYear = currentYear.value
let startMonth = currentMonth.value - 5
if (startMonth <= 0) {
startMonth += 12
startYear--
}
const response = await getTrendStatistics({
startYear,
startMonth,
monthCount: 6
})
if (response.success && response.data) {
trendData.value = response.data.map(item => ({
year: item.year,
month: item.month,
label: `${item.month}`,
expense: item.expense,
income: item.income
}))
}
} catch (error) {
console.error('获取趋势数据失败:', error)
showToast('获取趋势数据失败')
}
}
// 跳转到智能分析页面
const goToAnalysis = () => {
router.push('/bill-analysis')
}
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 ? '支出' : '收入'}`
// 重置分页状态
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
billListVisible.value = true
2025-12-27 22:05:50 +08:00
// 打开弹窗后加载数据
loadCategoryBills()
}
// 打开总支出/总收入的所有账单列表
const goToTypeOverviewBills = (type) => {
selectedClassify.value = null
selectedType.value = type
selectedCategoryTitle.value = `${type === 0 ? '总支出' : '总收入'} - 明细`
// 重置分页状态
categoryBills.value = []
billPageIndex.value = 1
billListFinished.value = false
billListVisible.value = true
// 打开弹窗后加载数据
loadCategoryBills()
2025-12-26 17:56:08 +08:00
}
2025-12-29 21:17:18 +08:00
const smartClassifyButtonRef = 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) => {
2025-12-26 17:56:08 +08:00
if (billListLoading.value || billListFinished.value) return
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,
2025-12-27 22:05:50 +08:00
month: currentMonth.value,
sortByAmount: true
2025-12-26 17:56:08 +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
}
const response = await getTransactionList(params)
if (response.success) {
const newList = response.data || []
categoryBills.value = [...categoryBills.value, ...newList]
2025-12-27 22:05:50 +08:00
categoryBillsTotal.value = response.total
2025-12-26 17:56:08 +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('获取详情失败')
}
}
// 账单保存后的回调
const onBillSave = async () => {
// 刷新统计数据
await fetchStatistics()
// 刷新账单列表
categoryBills.value = []
2025-12-27 22:05:50 +08:00
billPageIndex.value = 1
2025-12-26 17:56:08 +08:00
billListFinished.value = false
await loadCategoryBills()
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
// 刷新统计数据
await fetchStatistics()
showToast('智能分类已保存')
}
2025-12-30 18:49:46 +08:00
const handleNotifiedTransactionId = async (transactionId) => {
console.log('收到已处理交易ID通知:', transactionId)
// 滚动到指定的交易项
const index = categoryBills.value.findIndex(item => item.id === transactionId)
if (index !== -1) {
// 等待 DOM 更新
await nextTick()
const listElement = document.querySelector('.transaction-list')
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()
})
2025-12-26 17:56:08 +08:00
// 页面激活时刷新数据(从其他页面返回时)
onActivated(() => {
fetchStatistics()
})
2025-12-26 17:13:57 +08:00
</script>
<style scoped>
2025-12-27 21:15:26 +08:00
.statistics-content {
padding: 16px 0 0 0;
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;
}
2025-12-26 17:13:57 +08:00
/* 月度概览卡片 */
.overview-card {
display: flex;
justify-content: space-around;
align-items: center;
2025-12-26 17:29:17 +08:00
background: #ffffff;
2025-12-26 17:13:57 +08:00
margin: 0 12px 16px;
padding: 24px 12px;
border-radius: 16px;
box-shadow: 0 2px 12px rgba(0, 0, 0, 0.08);
2025-12-26 17:29:17 +08:00
border: 1px solid #ebedf0;
}
@media (prefers-color-scheme: dark) {
.overview-card {
background: #1f1f1f;
border-color: #2c2c2c;
}
2025-12-26 17:13:57 +08:00
}
.overview-item {
flex: 1;
text-align: center;
}
2025-12-27 22:05:50 +08:00
.overview-item.clickable {
cursor: pointer;
transition: background-color 0.2s;
2025-12-27 22:44:28 +08:00
padding: 0;
2025-12-27 22:05:50 +08:00
border-radius: 8px;
}
.overview-item.clickable:active {
background-color: #f0f0f0;
}
@media (prefers-color-scheme: dark) {
.overview-item.clickable:active {
background-color: #2c2c2c;
}
}
2025-12-26 17:13:57 +08:00
.overview-item .label {
font-size: 13px;
color: var(--van-text-color-2);
margin-bottom: 8px;
}
.overview-item .value {
font-size: 20px;
font-weight: 700;
margin-bottom: 4px;
}
.overview-item .sub-text {
font-size: 12px;
color: var(--van-text-color-3);
}
.divider {
width: 1px;
height: 40px;
background: var(--van-border-color);
}
.expense {
color: #ff6b6b;
}
.income {
color: #51cf66;
}
/* 环形图 */
.chart-container {
padding: 20px;
}
.ring-chart {
position: relative;
width: 200px;
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;
2025-12-26 17:29:17 +08:00
border-bottom: 1px solid #ebedf0;
2025-12-26 17:56:08 +08:00
transition: background-color 0.2s;
gap: 12px;
2025-12-26 17:29:17 +08:00
}
@media (prefers-color-scheme: dark) {
.category-item {
border-bottom: 1px solid #2c2c2c;
}
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 {
background-color: #f7f8fa;
}
@media (prefers-color-scheme: dark) {
.category-item.clickable:active {
background-color: #141414;
}
}
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
}
.category-amount {
font-size: 15px;
font-weight: 600;
color: var(--van-text-color);
}
.category-percent {
font-size: 12px;
color: var(--van-text-color-3);
2025-12-26 17:29:17 +08:00
background: #f7f8fa;
2025-12-26 17:13:57 +08:00
padding: 2px 8px;
border-radius: 10px;
}
2025-12-26 17:29:17 +08:00
@media (prefers-color-scheme: dark) {
.category-percent {
background: #141414;
}
}
2025-12-26 17:13:57 +08:00
.income-color {
background-color: #51cf66;
}
.income-text {
color: #51cf66;
}
2025-12-27 22:34:19 +08:00
/* 不计收支颜色 */
.none-color {
background-color: #909399;
}
.none-text {
color: #909399;
}
2025-12-26 17:13:57 +08:00
.expense-color {
background-color: #ff6b6b;
}
/* 趋势图 */
.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;
justify-content: center;
gap: 24px;
padding-top: 12px;
2025-12-26 17:29:17 +08:00
border-top: 1px solid #ebedf0;
}
@media (prefers-color-scheme: dark) {
.trend-legend {
border-top: 1px solid #2c2c2c;
}
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;
}
/* 其他统计 */
.other-stats {
display: grid;
grid-template-columns: repeat(2, 1fr);
gap: 16px;
padding: 16px;
}
.stat-item {
2025-12-26 17:29:17 +08:00
background: #f7f8fa;
2025-12-26 17:13:57 +08:00
padding: 16px;
border-radius: 12px;
text-align: center;
}
2025-12-26 17:29:17 +08:00
@media (prefers-color-scheme: dark) {
.stat-item {
background: #141414;
}
}
2025-12-26 17:13:57 +08:00
.stat-label {
font-size: 13px;
color: var(--van-text-color-2);
margin-bottom: 8px;
}
.stat-value {
font-size: 18px;
font-weight: 700;
color: var(--van-text-color);
}
2025-12-26 18:03:52 +08:00
/* 设置页面容器背景色 */
:deep(.van-nav-bar) {
background: transparent !important;
}
2025-12-26 17:13:57 +08:00
</style>