2026-05-26 09:15:14 +08:00
|
|
|
|
<template>
|
|
|
|
|
|
<div class="budget-trend-chart">
|
2026-05-26 20:07:56 +08:00
|
|
|
|
<Bar :data="chartData" :options="chartOptions" />
|
2026-05-26 09:15:14 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
|
|
|
|
|
|
<script setup>
|
|
|
|
|
|
import { computed } from 'vue'
|
2026-05-26 20:07:56 +08:00
|
|
|
|
import { Bar } from 'vue-chartjs'
|
2026-05-26 09:15:14 +08:00
|
|
|
|
import {
|
|
|
|
|
|
Chart as ChartJS,
|
2026-05-26 20:07:56 +08:00
|
|
|
|
BarElement,
|
2026-05-26 09:15:14 +08:00
|
|
|
|
CategoryScale,
|
|
|
|
|
|
Legend,
|
|
|
|
|
|
LinearScale,
|
|
|
|
|
|
Tooltip
|
|
|
|
|
|
} from 'chart.js'
|
|
|
|
|
|
import { useAnimationProgress } from '../../composables/useAnimationProgress.js'
|
|
|
|
|
|
|
2026-05-26 20:07:56 +08:00
|
|
|
|
ChartJS.register(CategoryScale, LinearScale, BarElement, Tooltip, Legend)
|
2026-05-26 09:15:14 +08:00
|
|
|
|
|
|
|
|
|
|
const props = defineProps({
|
|
|
|
|
|
labels: { type: Array, required: true },
|
|
|
|
|
|
budget: { type: Array, required: true },
|
2026-05-26 20:07:56 +08:00
|
|
|
|
used: { type: Array, required: true },
|
|
|
|
|
|
occupied: { type: Array, default: () => [] },
|
|
|
|
|
|
available: { type: Array, default: () => [] }
|
2026-05-26 09:15:14 +08:00
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
const progress = useAnimationProgress([
|
|
|
|
|
|
() => props.labels,
|
|
|
|
|
|
() => props.budget,
|
2026-05-26 20:07:56 +08:00
|
|
|
|
() => props.used,
|
|
|
|
|
|
() => props.occupied,
|
|
|
|
|
|
() => props.available
|
2026-05-26 09:15:14 +08:00
|
|
|
|
], 1000)
|
|
|
|
|
|
|
2026-05-26 20:07:56 +08:00
|
|
|
|
const currency = (value) =>
|
|
|
|
|
|
Number(value || 0).toLocaleString('zh-CN', {
|
|
|
|
|
|
minimumFractionDigits: 2,
|
|
|
|
|
|
maximumFractionDigits: 2
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
const percent = (value, total) => {
|
|
|
|
|
|
const denominator = Number(total || 0)
|
|
|
|
|
|
if (!denominator) return 0
|
|
|
|
|
|
return Number(((Number(value || 0) / denominator) * 100).toFixed(2))
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const percentSeries = (series) =>
|
|
|
|
|
|
props.budget.map((total, index) => percent(series[index], total))
|
|
|
|
|
|
|
2026-05-26 09:15:14 +08:00
|
|
|
|
const scaleSeries = (series) =>
|
|
|
|
|
|
series.map((value) => Number((Number(value || 0) * progress.value).toFixed(2)))
|
|
|
|
|
|
|
2026-05-26 20:07:56 +08:00
|
|
|
|
const usedPercent = computed(() => percentSeries(props.used))
|
|
|
|
|
|
const occupiedPercent = computed(() => percentSeries(props.occupied))
|
|
|
|
|
|
const availablePercent = computed(() =>
|
|
|
|
|
|
props.budget.map((total, index) => {
|
|
|
|
|
|
const usedValue = Number(props.used[index] || 0)
|
|
|
|
|
|
const occupiedValue = Number(props.occupied[index] || 0)
|
|
|
|
|
|
return percent(Math.max(Number(total || 0) - usedValue - occupiedValue, 0), total)
|
|
|
|
|
|
})
|
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
const yAxisMax = computed(() => {
|
|
|
|
|
|
const maxUsage = Math.max(
|
|
|
|
|
|
100,
|
|
|
|
|
|
...usedPercent.value.map((value, index) => value + Number(occupiedPercent.value[index] || 0))
|
|
|
|
|
|
)
|
|
|
|
|
|
return Math.ceil(maxUsage / 20) * 20
|
|
|
|
|
|
})
|
|
|
|
|
|
|
2026-05-26 09:15:14 +08:00
|
|
|
|
const chartData = computed(() => ({
|
|
|
|
|
|
labels: props.labels,
|
|
|
|
|
|
datasets: [
|
|
|
|
|
|
{
|
2026-05-26 20:07:56 +08:00
|
|
|
|
label: '已使用',
|
|
|
|
|
|
data: scaleSeries(usedPercent.value),
|
|
|
|
|
|
backgroundColor: '#13a66b',
|
|
|
|
|
|
borderRadius: 5,
|
|
|
|
|
|
borderSkipped: false,
|
|
|
|
|
|
stack: 'budgetUsage',
|
|
|
|
|
|
amounts: props.used
|
2026-05-26 09:15:14 +08:00
|
|
|
|
},
|
|
|
|
|
|
{
|
2026-05-26 20:07:56 +08:00
|
|
|
|
label: '已占用',
|
|
|
|
|
|
data: scaleSeries(occupiedPercent.value),
|
|
|
|
|
|
backgroundColor: '#f59e0b',
|
|
|
|
|
|
borderRadius: 5,
|
|
|
|
|
|
borderSkipped: false,
|
|
|
|
|
|
stack: 'budgetUsage',
|
|
|
|
|
|
amounts: props.occupied
|
|
|
|
|
|
},
|
|
|
|
|
|
{
|
|
|
|
|
|
label: '剩余可用',
|
|
|
|
|
|
data: scaleSeries(availablePercent.value),
|
|
|
|
|
|
backgroundColor: '#e5edf3',
|
|
|
|
|
|
borderRadius: 5,
|
|
|
|
|
|
borderSkipped: false,
|
|
|
|
|
|
stack: 'budgetUsage',
|
|
|
|
|
|
amounts: props.available
|
2026-05-26 09:15:14 +08:00
|
|
|
|
}
|
|
|
|
|
|
]
|
|
|
|
|
|
}))
|
|
|
|
|
|
|
2026-05-26 20:07:56 +08:00
|
|
|
|
const chartOptions = computed(() => ({
|
2026-05-26 09:15:14 +08:00
|
|
|
|
responsive: true,
|
|
|
|
|
|
maintainAspectRatio: false,
|
|
|
|
|
|
interaction: {
|
|
|
|
|
|
mode: 'index',
|
|
|
|
|
|
intersect: false
|
|
|
|
|
|
},
|
|
|
|
|
|
animation: {
|
|
|
|
|
|
duration: 760,
|
|
|
|
|
|
easing: 'easeOutQuart'
|
|
|
|
|
|
},
|
|
|
|
|
|
plugins: {
|
|
|
|
|
|
legend: {
|
|
|
|
|
|
display: false
|
|
|
|
|
|
},
|
|
|
|
|
|
tooltip: {
|
|
|
|
|
|
backgroundColor: '#ffffff',
|
|
|
|
|
|
borderColor: '#e2e8f0',
|
|
|
|
|
|
borderWidth: 1,
|
|
|
|
|
|
bodyColor: '#475569',
|
|
|
|
|
|
titleColor: '#0f172a',
|
|
|
|
|
|
padding: 12,
|
|
|
|
|
|
displayColors: true,
|
|
|
|
|
|
callbacks: {
|
|
|
|
|
|
label(context) {
|
|
|
|
|
|
const value = Number(context.parsed.y || 0)
|
2026-05-26 20:07:56 +08:00
|
|
|
|
const amount = Number(context.dataset.amounts?.[context.dataIndex] || 0)
|
|
|
|
|
|
return `${context.dataset.label}: ${value.toFixed(2)}%(¥${currency(amount)})`
|
|
|
|
|
|
},
|
|
|
|
|
|
afterBody(items) {
|
|
|
|
|
|
const index = items[0]?.dataIndex ?? 0
|
|
|
|
|
|
return `预算总额: ¥${currency(props.budget[index])}`
|
2026-05-26 09:15:14 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
},
|
|
|
|
|
|
scales: {
|
|
|
|
|
|
x: {
|
|
|
|
|
|
grid: { display: false },
|
|
|
|
|
|
ticks: {
|
|
|
|
|
|
color: '#64748b',
|
|
|
|
|
|
font: { size: 12 }
|
|
|
|
|
|
},
|
|
|
|
|
|
border: { display: false }
|
|
|
|
|
|
},
|
|
|
|
|
|
y: {
|
|
|
|
|
|
beginAtZero: true,
|
2026-05-26 20:07:56 +08:00
|
|
|
|
max: yAxisMax.value,
|
|
|
|
|
|
stacked: true,
|
2026-05-26 09:15:14 +08:00
|
|
|
|
grid: {
|
|
|
|
|
|
color: '#edf2f7',
|
|
|
|
|
|
drawTicks: false
|
|
|
|
|
|
},
|
|
|
|
|
|
border: { display: false },
|
|
|
|
|
|
ticks: {
|
|
|
|
|
|
color: '#64748b',
|
|
|
|
|
|
font: { size: 12 },
|
2026-05-26 20:07:56 +08:00
|
|
|
|
stepSize: 20,
|
2026-05-26 09:15:14 +08:00
|
|
|
|
callback(value) {
|
2026-05-26 20:07:56 +08:00
|
|
|
|
return `${Number(value)}%`
|
2026-05-26 09:15:14 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2026-05-26 20:07:56 +08:00
|
|
|
|
},
|
|
|
|
|
|
datasets: {
|
|
|
|
|
|
bar: {
|
|
|
|
|
|
categoryPercentage: 0.58,
|
|
|
|
|
|
barPercentage: 0.72
|
|
|
|
|
|
}
|
2026-05-26 09:15:14 +08:00
|
|
|
|
}
|
2026-05-26 20:07:56 +08:00
|
|
|
|
}))
|
2026-05-26 09:15:14 +08:00
|
|
|
|
</script>
|
|
|
|
|
|
|
|
|
|
|
|
<style scoped>
|
|
|
|
|
|
.budget-trend-chart {
|
|
|
|
|
|
position: relative;
|
|
|
|
|
|
width: 100%;
|
|
|
|
|
|
height: 220px;
|
|
|
|
|
|
}
|
|
|
|
|
|
</style>
|