226 lines
6.5 KiB
Vue
226 lines
6.5 KiB
Vue
<script lang="ts" setup>
|
||
import type { EchartsUIType } from '@vben/plugins/echarts';
|
||
|
||
import { onMounted, reactive, ref } from 'vue';
|
||
|
||
import { AnalysisChartCard } from '@vben/common-ui';
|
||
import { EchartsUI, useEcharts } from '@vben/plugins/echarts';
|
||
import { fenToYuan, formatDate } from '@vben/utils';
|
||
|
||
import dayjs, { Dayjs } from 'dayjs';
|
||
|
||
import * as TradeStatisticsApi from '#/api/mall/statistics/trade';
|
||
import { TimeRangeTypeEnum } from '#/utils/constants';
|
||
|
||
/** 交易量趋势 */
|
||
defineOptions({ name: 'TradeTrendCard' });
|
||
|
||
const timeRangeType = ref(TimeRangeTypeEnum.DAY30); // 日期快捷选择按钮, 默认30天
|
||
const loading = ref(true); // 加载中
|
||
const chartRef = ref<EchartsUIType>();
|
||
const { renderEcharts } = useEcharts(chartRef);
|
||
// 时间范围 Map
|
||
const timeRange = new Map()
|
||
.set(TimeRangeTypeEnum.DAY30, {
|
||
name: '30天',
|
||
series: [
|
||
{ name: '订单金额', type: 'bar', smooth: true, data: [] },
|
||
{ name: '订单数量', type: 'line', smooth: true, data: [] },
|
||
],
|
||
})
|
||
.set(TimeRangeTypeEnum.WEEK, {
|
||
name: '周',
|
||
series: [
|
||
{ name: '上周金额', type: 'bar', smooth: true, data: [] },
|
||
{ name: '本周金额', type: 'bar', smooth: true, data: [] },
|
||
{ name: '上周数量', type: 'line', smooth: true, data: [] },
|
||
{ name: '本周数量', type: 'line', smooth: true, data: [] },
|
||
],
|
||
})
|
||
.set(TimeRangeTypeEnum.MONTH, {
|
||
name: '月',
|
||
series: [
|
||
{ name: '上月金额', type: 'bar', smooth: true, data: [] },
|
||
{ name: '本月金额', type: 'bar', smooth: true, data: [] },
|
||
{ name: '上月数量', type: 'line', smooth: true, data: [] },
|
||
{ name: '本月数量', type: 'line', smooth: true, data: [] },
|
||
],
|
||
})
|
||
.set(TimeRangeTypeEnum.YEAR, {
|
||
name: '年',
|
||
series: [
|
||
{ name: '去年金额', type: 'bar', smooth: true, data: [] },
|
||
{ name: '今年金额', type: 'bar', smooth: true, data: [] },
|
||
{ name: '去年数量', type: 'line', smooth: true, data: [] },
|
||
{ name: '今年数量', type: 'line', smooth: true, data: [] },
|
||
],
|
||
});
|
||
/** 图表配置 */
|
||
const eChartOptions = reactive({
|
||
grid: {
|
||
left: 20,
|
||
right: 20,
|
||
bottom: 20,
|
||
top: 80,
|
||
containLabel: true,
|
||
},
|
||
legend: {
|
||
top: 50,
|
||
data: [] as string[],
|
||
},
|
||
series: [] as any[],
|
||
toolbox: {
|
||
feature: {
|
||
// 数据区域缩放
|
||
dataZoom: {
|
||
yAxisIndex: false, // Y轴不缩放
|
||
},
|
||
brush: {
|
||
type: ['lineX', 'clear'], // 区域缩放按钮、还原按钮
|
||
},
|
||
saveAsImage: { show: true, name: '订单量趋势' }, // 保存为图片
|
||
},
|
||
},
|
||
tooltip: {
|
||
trigger: 'axis',
|
||
axisPointer: {
|
||
type: 'cross',
|
||
},
|
||
padding: [5, 10],
|
||
},
|
||
xAxis: {
|
||
type: 'category' as const,
|
||
inverse: true,
|
||
boundaryGap: false,
|
||
axisTick: {
|
||
show: false,
|
||
},
|
||
data: [] as string[],
|
||
axisLabel: {
|
||
formatter: (date: string) => {
|
||
switch (timeRangeType.value) {
|
||
case TimeRangeTypeEnum.DAY30: {
|
||
return formatDate(date, 'MM-DD');
|
||
}
|
||
case TimeRangeTypeEnum.MONTH: {
|
||
return formatDate(date, 'D');
|
||
}
|
||
case TimeRangeTypeEnum.WEEK: {
|
||
let weekDay = formatDate(date, 'ddd');
|
||
if (weekDay === '0') weekDay = '日';
|
||
return `周${weekDay}`;
|
||
}
|
||
case TimeRangeTypeEnum.YEAR: {
|
||
return `${formatDate(date, 'M')}月`;
|
||
}
|
||
default: {
|
||
return date;
|
||
}
|
||
}
|
||
},
|
||
},
|
||
},
|
||
yAxis: {
|
||
axisTick: {
|
||
show: false,
|
||
},
|
||
},
|
||
});
|
||
|
||
/** 时间范围类型单选按钮选中 */
|
||
const handleTimeRangeTypeChange = async () => {
|
||
// 设置时间范围
|
||
let beginTime: Dayjs;
|
||
let endTime: Dayjs;
|
||
switch (timeRangeType.value) {
|
||
case TimeRangeTypeEnum.MONTH: {
|
||
beginTime = dayjs().startOf('month');
|
||
endTime = dayjs().endOf('month');
|
||
break;
|
||
}
|
||
case TimeRangeTypeEnum.WEEK: {
|
||
beginTime = dayjs().startOf('week');
|
||
endTime = dayjs().endOf('week');
|
||
break;
|
||
}
|
||
case TimeRangeTypeEnum.YEAR: {
|
||
beginTime = dayjs().startOf('year');
|
||
endTime = dayjs().endOf('year');
|
||
break;
|
||
}
|
||
default: {
|
||
beginTime = dayjs().subtract(30, 'day').startOf('d');
|
||
endTime = dayjs().endOf('d');
|
||
break;
|
||
}
|
||
}
|
||
// 发送时间范围选中事件
|
||
await getOrderCountTrendComparison(beginTime, endTime);
|
||
};
|
||
|
||
/** 查询订单数量趋势对照数据 */
|
||
const getOrderCountTrendComparison = async (
|
||
beginTime: dayjs.ConfigType,
|
||
endTime: dayjs.ConfigType,
|
||
) => {
|
||
loading.value = true;
|
||
// 查询数据
|
||
const list = await TradeStatisticsApi.getOrderCountTrendComparison(
|
||
timeRangeType.value,
|
||
dayjs(beginTime).toDate(),
|
||
dayjs(endTime).toDate(),
|
||
);
|
||
// 处理数据
|
||
const dates: string[] = [];
|
||
const series = [...timeRange.get(timeRangeType.value).series];
|
||
for (const item of list) {
|
||
dates.push(item.value.date);
|
||
if (series.length === 2) {
|
||
series[0].data.push(fenToYuan(item?.value?.orderPayPrice || 0)); // 当前金额
|
||
series[1].data.push(item?.value?.orderPayCount || 0); // 当前数量
|
||
} else {
|
||
series[0].data.push(fenToYuan(item?.reference?.orderPayPrice || 0)); // 对照金额
|
||
series[1].data.push(fenToYuan(item?.value?.orderPayPrice || 0)); // 当前金额
|
||
series[2].data.push(item?.reference?.orderPayCount || 0); // 对照数量
|
||
series[3].data.push(item?.value?.orderPayCount || 0); // 当前数量
|
||
}
|
||
}
|
||
eChartOptions.xAxis!.data = dates;
|
||
eChartOptions.series = series;
|
||
// legend在4个切换到2个的时候,还是显示成4个,需要手动配置一下
|
||
eChartOptions.legend.data = series.map((item) => item.name);
|
||
loading.value = false;
|
||
};
|
||
|
||
/** 初始化 */
|
||
onMounted(async () => {
|
||
await handleTimeRangeTypeChange();
|
||
renderEcharts(eChartOptions as any);
|
||
});
|
||
</script>
|
||
<template>
|
||
<AnalysisChartCard title="交易量趋势">
|
||
<template #header-suffix>
|
||
<div class="flex flex-row items-center justify-between">
|
||
<!-- 查询条件 -->
|
||
<div class="flex flex-row items-center gap-2">
|
||
<el-radio-group
|
||
v-model="timeRangeType"
|
||
@change="handleTimeRangeTypeChange"
|
||
>
|
||
<el-radio-button
|
||
v-for="[key, value] in timeRange.entries()"
|
||
:key="key"
|
||
:value="key"
|
||
>
|
||
{{ value.name }}
|
||
</el-radio-button>
|
||
</el-radio-group>
|
||
</div>
|
||
</div>
|
||
</template>
|
||
<!-- 折线图 -->
|
||
<EchartsUI ref="chartRef" />
|
||
</AnalysisChartCard>
|
||
</template>
|