feat:【mall 商城】商城首页的迁移【ele】100%:评审完成

This commit is contained in:
YunaiV
2025-10-19 16:59:47 +08:00
parent 73208aa304
commit bd9d8376c2
7 changed files with 181 additions and 131 deletions

View File

@@ -1,5 +1,5 @@
/** 数据对照 Response */ /** 数据对照 Response */
export interface MallDataComparisonResp<T> { export interface DataComparisonRespVO<T> {
value: T; value: T;
reference: T; reference: T;
} }

View File

@@ -1,73 +1,73 @@
import type { MallDataComparisonResp } from './common'; import type { Dayjs } from 'dayjs';
import { formatDate2 } from '@vben/utils'; import type { DataComparisonRespVO } from './common';
import { formatDateTime } from '@vben/utils';
import { requestClient } from '#/api/request'; import { requestClient } from '#/api/request';
export namespace MallMemberStatisticsApi { export namespace MallMemberStatisticsApi {
/** 会员分析 Request */ /** 会员分析 Request */
export interface AnalyseReq { export interface MemberAnalyseReqVO {
times: Date[]; times: Date[] | Dayjs[]; // 时间范围
} }
/** 会员分析对照数据 Response */ /** 会员分析对照数据 Response */
export interface AnalyseComparison { export interface AnalyseComparison {
registerUserCount: number; registerUserCount: number; // 注册用户数
visitUserCount: number; visitUserCount: number; // 访问用户数
rechargeUserCount: number; rechargeUserCount: number; // 充值用户数
} }
/** 会员分析 Response */ /** 会员分析 Response */
export interface Analyse { export interface Analyse {
visitUserCount: number; visitUserCount: number; // 访问用户数
orderUserCount: number; orderUserCount: number; // 下单用户数
payUserCount: number; payUserCount: number; // 支付用户数
atv: number; atv: number; // 平均客单价
comparison: MallDataComparisonResp<AnalyseComparison>; comparison: DataComparisonRespVO<AnalyseComparison>; // 对照数据
} }
/** 会员地区统计 Response */ /** 会员地区统计 Response */
export interface AreaStatistics { export interface AreaStatistics {
areaId: number; areaId: number; // 地区ID
areaName: string; areaName: string; // 地区名称
userCount: number; userCount: number; // 用户数
orderCreateUserCount: number; orderCreateUserCount: number; // 下单用户数
orderPayUserCount: number; orderPayUserCount: number; // 支付用户数
orderPayPrice: number; orderPayPrice: number; // 支付金额
} }
/** 会员性别统计 Response */ /** 会员性别统计 Response */
export interface SexStatistics { export interface SexStatistics {
sex: number; sex: number; // 性别
userCount: number; userCount: number; // 用户数
} }
/** 会员统计 Response */ /** 会员统计 Response */
export interface Summary { export interface Summary {
userCount: number; userCount: number; // 用户数
rechargeUserCount: number; rechargeUserCount: number; // 充值用户数
rechargePrice: number; rechargePrice: number; // 充值金额
expensePrice: number; expensePrice: number; // 消费金额
} }
/** 会员终端统计 Response */ /** 会员终端统计 Response */
export interface TerminalStatistics { export interface TerminalStatistics {
terminal: number; terminal: number; // 终端
userCount: number; userCount: number; // 用户数
} }
/** 会员数量统计 Response */ /** 会员数量统计 Response */
export interface Count { export interface MemberCountRespVO {
/** 用户访问量 */ visitUserCount: string; // 用户访问量
visitUserCount: string; registerUserCount: number; // 注册用户数量
/** 注册用户数量 */
registerUserCount: number;
} }
/** 会员注册数量 Response */ /** 会员注册数量 Response */
export interface RegisterCount { export interface RegisterCount {
date: string; date: string; // 日期
count: number; count: number; // 数量
} }
} }
@@ -79,14 +79,16 @@ export function getMemberSummary() {
} }
/** 查询会员分析数据 */ /** 查询会员分析数据 */
export function getMemberAnalyse(params: MallMemberStatisticsApi.AnalyseReq) { export function getMemberAnalyse(
params: MallMemberStatisticsApi.MemberAnalyseReqVO,
) {
return requestClient.get<MallMemberStatisticsApi.Analyse>( return requestClient.get<MallMemberStatisticsApi.Analyse>(
'/statistics/member/analyse', '/statistics/member/analyse',
{ {
params: { params: {
times: [ times: [
formatDate2(params.times[0] || new Date()), formatDateTime(params.times[0]),
formatDate2(params.times[1] || new Date()), formatDateTime(params.times[1]),
], ],
}, },
}, },
@@ -117,7 +119,7 @@ export function getMemberTerminalStatisticsList() {
/** 获得用户数量量对照 */ /** 获得用户数量量对照 */
export function getUserCountComparison() { export function getUserCountComparison() {
return requestClient.get< return requestClient.get<
MallDataComparisonResp<MallMemberStatisticsApi.Count> DataComparisonRespVO<MallMemberStatisticsApi.MemberCountRespVO>
>('/statistics/member/user-count-comparison'); >('/statistics/member/user-count-comparison');
} }
@@ -127,7 +129,7 @@ export function getMemberRegisterCountList(beginTime: Date, endTime: Date) {
'/statistics/member/register-count-list', '/statistics/member/register-count-list',
{ {
params: { params: {
times: [formatDate2(beginTime), formatDate2(endTime)], times: [formatDateTime(beginTime), formatDateTime(endTime)],
}, },
}, },
); );

View File

@@ -1,8 +1,6 @@
import type { PageParam, PageResult } from '@vben/request'; import type { PageParam, PageResult } from '@vben/request';
import type { MallDataComparisonResp } from './common'; import type { DataComparisonRespVO } from './common';
import { formatDate2 } from '@vben/utils';
import { requestClient } from '#/api/request'; import { requestClient } from '#/api/request';
@@ -40,58 +38,26 @@ export namespace MallProductStatisticsApi {
/** 浏览转化率 */ /** 浏览转化率 */
browseConvertPercent: number; browseConvertPercent: number;
} }
/** 会员分析 Request */
export interface ProductStatisticsReq {
times: Date[];
}
} }
/** 获得商品统计分析 */ /** 获得商品统计分析 */
export function getProductStatisticsAnalyse( export function getProductStatisticsAnalyse(params: PageParam) {
params: MallProductStatisticsApi.ProductStatisticsReq,
) {
return requestClient.get< return requestClient.get<
MallDataComparisonResp<MallProductStatisticsApi.ProductStatistics> DataComparisonRespVO<MallProductStatisticsApi.ProductStatistics>
>('/statistics/product/analyse', { >('/statistics/product/analyse', { params });
params: {
times: [
formatDate2(params.times[0] || new Date()),
formatDate2(params.times[1] || new Date()),
],
},
});
} }
/** 获得商品状况明细 */ /** 获得商品状况明细 */
export function getProductStatisticsList( export function getProductStatisticsList(params: PageParam) {
params: MallProductStatisticsApi.ProductStatisticsReq,
) {
return requestClient.get<MallProductStatisticsApi.ProductStatistics[]>( return requestClient.get<MallProductStatisticsApi.ProductStatistics[]>(
'/statistics/product/list', '/statistics/product/list',
{ { params },
params: {
times: [
formatDate2(params.times[0] || new Date()),
formatDate2(params.times[1] || new Date()),
],
},
},
); );
} }
/** 导出获得商品状况明细 Excel */ /** 导出获得商品状况明细 Excel */
export function exportProductStatisticsExcel( export function exportProductStatisticsExcel(params: PageParam) {
params: MallProductStatisticsApi.ProductStatisticsReq, return requestClient.download('/statistics/product/export-excel', { params });
) {
return requestClient.download('/statistics/product/export-excel', {
params: {
times: [
formatDate2(params.times[0] || new Date()),
formatDate2(params.times[1] || new Date()),
],
},
});
} }
/** 获得商品排行榜分页 */ /** 获得商品排行榜分页 */

View File

@@ -1,6 +1,6 @@
import type { MallDataComparisonResp } from './common'; import type { DataComparisonRespVO } from './common';
import { formatDate2 } from '@vben/utils'; import { formatDate, formatDateTime } from '@vben/utils';
import { requestClient } from '#/api/request'; import { requestClient } from '#/api/request';
@@ -15,7 +15,7 @@ export namespace MallTradeStatisticsApi {
/** 交易状况 Request */ /** 交易状况 Request */
export interface TradeTrendReq { export interface TradeTrendReq {
times: Date[]; times: [Date, Date];
} }
/** 交易状况统计 Response */ /** 交易状况统计 Response */
@@ -43,7 +43,7 @@ export namespace MallTradeStatisticsApi {
} }
/** 交易订单统计 Response */ /** 交易订单统计 Response */
export interface TradeOrderSummary { export interface TradeOrderSummaryRespVO {
/** 支付订单商品数 */ /** 支付订单商品数 */
orderPayCount?: number; orderPayCount?: number;
/** 总支付金额,单位:分 */ /** 总支付金额,单位:分 */
@@ -64,17 +64,14 @@ export namespace MallTradeStatisticsApi {
/** 时间参数需要格式化, 确保接口能识别 */ /** 时间参数需要格式化, 确保接口能识别 */
const formatDateParam = (params: MallTradeStatisticsApi.TradeTrendReq) => { const formatDateParam = (params: MallTradeStatisticsApi.TradeTrendReq) => {
return { return {
times: [ times: [formatDate(params.times[0]), formatDate(params.times[1])],
formatDate2(params.times[0] || new Date()), } as MallTradeStatisticsApi.TradeTrendReq;
formatDate2(params.times[1] || new Date()),
],
};
}; };
/** 查询交易统计 */ /** 查询交易统计 */
export function getTradeStatisticsSummary() { export function getTradeStatisticsSummary() {
return requestClient.get< return requestClient.get<
MallDataComparisonResp<MallTradeStatisticsApi.TradeSummary> DataComparisonRespVO<MallTradeStatisticsApi.TradeSummary>
>('/statistics/trade/summary'); >('/statistics/trade/summary');
} }
@@ -83,7 +80,7 @@ export function getTradeStatisticsAnalyse(
params: MallTradeStatisticsApi.TradeTrendReq, params: MallTradeStatisticsApi.TradeTrendReq,
) { ) {
return requestClient.get< return requestClient.get<
MallDataComparisonResp<MallTradeStatisticsApi.TradeTrendSummary> DataComparisonRespVO<MallTradeStatisticsApi.TradeTrendSummary>
>('/statistics/trade/analyse', { params: formatDateParam(params) }); >('/statistics/trade/analyse', { params: formatDateParam(params) });
} }
@@ -116,7 +113,7 @@ export function getOrderCount() {
/** 获得交易订单数量对照 */ /** 获得交易订单数量对照 */
export function getOrderComparison() { export function getOrderComparison() {
return requestClient.get< return requestClient.get<
MallDataComparisonResp<MallTradeStatisticsApi.TradeOrderSummary> DataComparisonRespVO<MallTradeStatisticsApi.TradeOrderSummaryRespVO>
>('/statistics/trade/order-comparison'); >('/statistics/trade/order-comparison');
} }
@@ -127,12 +124,12 @@ export function getOrderCountTrendComparison(
endTime: Date, endTime: Date,
) { ) {
return requestClient.get< return requestClient.get<
MallDataComparisonResp<MallTradeStatisticsApi.TradeOrderTrend>[] DataComparisonRespVO<MallTradeStatisticsApi.TradeOrderTrend>[]
>('/statistics/trade/order-count-trend', { >('/statistics/trade/order-count-trend', {
params: { params: {
type, type,
beginTime: formatDate2(beginTime), beginTime: formatDateTime(beginTime),
endTime: formatDate2(endTime), endTime: formatDateTime(endTime),
}, },
}); });
} }

View File

@@ -0,0 +1 @@
export { default as ShortcutDateRangePicker } from './shortcut-date-range-picker.vue';

View File

@@ -0,0 +1,110 @@
<script lang="ts" setup>
import type { Dayjs } from 'dayjs';
import { onMounted, ref } from 'vue';
import dayjs from 'dayjs';
import { ElDatePicker, ElRadio, ElRadioGroup } from 'element-plus';
import { getRangePickerDefaultProps } from '#/utils/rangePickerProps';
/** 快捷日期范围选择组件 */
defineOptions({ name: 'ShortcutDateRangePicker' });
const emits = defineEmits<{
change: [times: [Dayjs, Dayjs]];
}>();
const times = ref<[Dayjs, Dayjs]>(); // 日期范围
const rangePickerProps = getRangePickerDefaultProps();
const timeRangeOptions = [
{
label: '昨天',
value: () => [
dayjs().subtract(1, 'day').startOf('day'),
dayjs().subtract(1, 'day').endOf('day'),
],
},
{
label: '最近 7 天',
value: () => [
dayjs().subtract(7, 'day').startOf('day'),
dayjs().endOf('day'),
],
},
{
label: '最近 30 天',
value: () => [
dayjs().subtract(30, 'day').startOf('day'),
dayjs().endOf('day'),
],
},
];
const timeRangeType = ref(timeRangeOptions[1]!.label); // 默认选中"最近 7 天"
/** 设置时间范围 */
function setTimes() {
// 根据选中的选项设置时间范围
const selectedOption = timeRangeOptions.find(
(option) => option.label === timeRangeType.value,
);
if (selectedOption) {
times.value = selectedOption.value() as [Dayjs, Dayjs];
}
}
/** 快捷日期单选按钮选中 */
async function handleShortcutDaysChange() {
// 设置时间范围
setTimes();
// 触发时间范围选中事件
emitDateRangePicker();
}
/** 日期范围改变 */
function handleDateRangeChange() {
emitDateRangePicker();
}
/** 触发时间范围选中事件 */
function emitDateRangePicker() {
if (times.value && times.value.length === 2) {
emits('change', times.value);
}
}
/** 初始化 */
onMounted(() => {
handleShortcutDaysChange();
});
</script>
<template>
<div class="flex items-center gap-2">
<ElRadioGroup v-model="timeRangeType" @change="handleShortcutDaysChange">
<ElRadio
v-for="option in timeRangeOptions"
:key="option.label"
:value="option.label"
>
{{ option.label }}
</ElRadio>
</ElRadioGroup>
<ElDatePicker
v-model="times as any"
type="daterange"
:shortcuts="rangePickerProps.shortcuts"
:format="rangePickerProps.format"
:value-format="rangePickerProps.valueFormat"
:start-placeholder="rangePickerProps.startPlaceholder"
:end-placeholder="rangePickerProps.endPlaceholder"
:default-time="rangePickerProps.defaultTime as any"
class="!w-[360px]"
@change="handleDateRangeChange"
/>
<slot></slot>
</div>
</template>

View File

@@ -1,38 +1,30 @@
<script lang="ts" setup> <script lang="ts" setup>
import type { Dayjs } from 'dayjs'; import type { Dayjs } from 'dayjs';
import { onMounted, ref } from 'vue'; import { ref } from 'vue';
import { fenToYuan } from '@vben/utils'; import { fenToYuan } from '@vben/utils';
import { ElCard, ElDatePicker } from 'element-plus'; import { ElCard } from 'element-plus';
import dayjs from 'dayjs';
import * as MemberStatisticsApi from '#/api/mall/statistics/member'; import * as MemberStatisticsApi from '#/api/mall/statistics/member';
import { getRangePickerDefaultProps } from '#/utils/rangePickerProps'; import { ShortcutDateRangePicker } from '#/components/shortcut-date-range-picker';
/** 会员概览卡片 */ /** 会员概览卡片 */
defineOptions({ name: 'MemberFunnelCard' }); defineOptions({ name: 'MemberFunnelCard' });
const loading = ref(false); const loading = ref(false);
const analyseData = ref<any>(); const analyseData = ref<any>();
const dateRange = ref<[Dayjs, Dayjs]>([
dayjs().subtract(7, 'day').startOf('day'),
dayjs().endOf('day'),
]);
const rangePickerProps = getRangePickerDefaultProps();
/** 查询会员概览数据列表 */ /** 查询会员概览数据列表 */
async function loadData(times?: [Dayjs, Dayjs]) { async function loadData(times: [Dayjs, Dayjs]) {
const timesToUse = times || dateRange.value; if (!times || times.length !== 2) {
if (!timesToUse || timesToUse.length !== 2) {
return; return;
} }
loading.value = true; loading.value = true;
try { try {
analyseData.value = await MemberStatisticsApi.getMemberAnalyse({ analyseData.value = await MemberStatisticsApi.getMemberAnalyse({
times: timesToUse, times,
}); });
} finally { } finally {
loading.value = false; loading.value = false;
@@ -40,10 +32,8 @@ async function loadData(times?: [Dayjs, Dayjs]) {
} }
/** 时间范围改变 */ /** 时间范围改变 */
const handleDateRangeChange = () => { const handleTimeRangeChange = (times: [Dayjs, Dayjs]) => {
if (dateRange.value && dateRange.value.length === 2) { loadData(times);
loadData(dateRange.value);
}
}; };
/** 计算环比增长率 */ /** 计算环比增长率 */
@@ -53,11 +43,6 @@ const calculateRelativeRate = (value?: number, reference?: number) => {
} }
return (((value || 0) - reference) / reference) * 100; return (((value || 0) - reference) / reference) * 100;
}; };
/** 初始化 */
onMounted(() => {
loadData();
});
</script> </script>
<template> <template>
@@ -65,18 +50,7 @@ onMounted(() => {
<template #header> <template #header>
<div class="flex items-center justify-between"> <div class="flex items-center justify-between">
<span>会员概览</span> <span>会员概览</span>
<ElDatePicker <ShortcutDateRangePicker @change="handleTimeRangeChange" />
v-model="dateRange"
type="datetimerange"
:shortcuts="rangePickerProps.shortcuts"
:format="rangePickerProps.format"
:value-format="rangePickerProps.valueFormat"
:start-placeholder="rangePickerProps.startPlaceholder"
:end-placeholder="rangePickerProps.endPlaceholder"
:default-time="rangePickerProps.defaultTime"
class="!w-[280px]"
@change="handleDateRangeChange"
/>
</div> </div>
</template> </template>
<div v-loading="loading" class="min-w-[900px] py-4"> <div v-loading="loading" class="min-w-[900px] py-4">