feat:【mall 商城】商城首页的迁移【antd】20%:初始化

This commit is contained in:
YunaiV
2025-10-19 10:09:33 +08:00
parent ad6ba25b3e
commit 95fffb8af0
6 changed files with 57 additions and 57 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,4 +1,4 @@
import type { MallDataComparisonResp } from './common'; import type { DataComparisonRespVO } from './common';
import { formatDate } from '@vben/utils'; import { formatDate } from '@vben/utils';
@@ -23,7 +23,7 @@ export namespace MallMemberStatisticsApi {
orderUserCount: number; // 下单用户数 orderUserCount: number; // 下单用户数
payUserCount: number; // 支付用户数 payUserCount: number; // 支付用户数
atv: number; // 平均客单价 atv: number; // 平均客单价
comparison: MallDataComparisonResp<AnalyseComparison>; // 对照数据 comparison: DataComparisonRespVO<AnalyseComparison>; // 对照数据
} }
/** 会员地区统计 Response */ /** 会员地区统计 Response */
@@ -57,7 +57,7 @@ export namespace MallMemberStatisticsApi {
} }
/** 会员数量统计 Response */ /** 会员数量统计 Response */
export interface Count { export interface MemberCountRespVO {
visitUserCount: string; // 用户访问量 visitUserCount: string; // 用户访问量
registerUserCount: number; // 注册用户数量 registerUserCount: number; // 注册用户数量
} }
@@ -112,7 +112,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');
} }

View File

@@ -1,6 +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 { requestClient } from '#/api/request'; import { requestClient } from '#/api/request';
@@ -43,7 +43,7 @@ export namespace MallProductStatisticsApi {
/** 获得商品统计分析 */ /** 获得商品统计分析 */
export function getProductStatisticsAnalyse(params: PageParam) { export function getProductStatisticsAnalyse(params: PageParam) {
return requestClient.get< return requestClient.get<
MallDataComparisonResp<MallProductStatisticsApi.ProductStatistics> DataComparisonRespVO<MallProductStatisticsApi.ProductStatistics>
>('/statistics/product/analyse', { params }); >('/statistics/product/analyse', { params });
} }

View File

@@ -1,4 +1,4 @@
import type { MallDataComparisonResp } from './common'; import type { DataComparisonRespVO } from './common';
import { formatDate } from '@vben/utils'; import { formatDate } from '@vben/utils';
@@ -43,7 +43,7 @@ export namespace MallTradeStatisticsApi {
} }
/** 交易订单统计 Response */ /** 交易订单统计 Response */
export interface TradeOrderSummary { export interface TradeOrderSummaryRespVO {
/** 支付订单商品数 */ /** 支付订单商品数 */
orderPayCount?: number; orderPayCount?: number;
/** 总支付金额,单位:分 */ /** 总支付金额,单位:分 */
@@ -71,7 +71,7 @@ const formatDateParam = (params: MallTradeStatisticsApi.TradeTrendReq) => {
/** 查询交易统计 */ /** 查询交易统计 */
export function getTradeStatisticsSummary() { export function getTradeStatisticsSummary() {
return requestClient.get< return requestClient.get<
MallDataComparisonResp<MallTradeStatisticsApi.TradeSummary> DataComparisonRespVO<MallTradeStatisticsApi.TradeSummary>
>('/statistics/trade/summary'); >('/statistics/trade/summary');
} }
@@ -80,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) });
} }
@@ -113,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');
} }
@@ -124,7 +124,7 @@ 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,

View File

@@ -1,13 +1,17 @@
<script lang="ts" setup> <script lang="ts" setup>
import { onMounted, ref } from 'vue'; import type { DataComparisonRespVO } from '#/api/mall/statistics/common';
import type { MallMemberStatisticsApi } from '#/api/mall/statistics/member';
import type { MallTradeStatisticsApi } from '#/api/mall/statistics/trade';
import { Col, Row } from 'ant-design-vue'; import { onMounted, ref } from 'vue';
import { DocAlert, Page } from '@vben/common-ui'; import { DocAlert, Page } from '@vben/common-ui';
import { fenToYuan } from '@vben/utils'; import { fenToYuan } from '@vben/utils';
import * as MemberStatisticsApi from '#/api/mall/statistics/member'; import { Col, Row } from 'ant-design-vue';
import * as TradeStatisticsApi from '#/api/mall/statistics/trade';
import { getUserCountComparison } from '#/api/mall/statistics/member';
import { getOrderComparison } from '#/api/mall/statistics/trade';
import ComparisonCard from './modules/comparison-card.vue'; import ComparisonCard from './modules/comparison-card.vue';
import MemberFunnelCard from './modules/member-funnel-card.vue'; import MemberFunnelCard from './modules/member-funnel-card.vue';
@@ -21,23 +25,25 @@ import TradeTrendCard from './modules/trade-trend-card.vue';
defineOptions({ name: 'MallHome' }); defineOptions({ name: 'MallHome' });
const loading = ref(true); // 加载中 const loading = ref(true); // 加载中
const orderComparison = ref<any>(); // 交易对照数据 const orderComparison =
const userComparison = ref<any>(); // 用户对照数据 ref<DataComparisonRespVO<MallTradeStatisticsApi.TradeOrderSummaryRespVO>>(); // 交易对照数据
const userComparison =
ref<DataComparisonRespVO<MallMemberStatisticsApi.MemberCountRespVO>>(); // 用户对照数据
/** 查询交易对照卡片数据 */ /** 查询交易对照卡片数据 */
const getOrderComparison = async () => { async function loadOrderComparison() {
orderComparison.value = await TradeStatisticsApi.getOrderComparison(); orderComparison.value = await getOrderComparison();
}; }
/** 查询会员用户数量对照卡片数据 */ /** 查询会员用户数量对照卡片数据 */
const getUserCountComparison = async () => { async function loadUserCountComparison() {
userComparison.value = await MemberStatisticsApi.getUserCountComparison(); userComparison.value = await getUserCountComparison();
}; }
/** 初始化 */ /** 初始化 */
onMounted(async () => { onMounted(async () => {
loading.value = true; loading.value = true;
await Promise.all([getOrderComparison(), getUserCountComparison()]); await Promise.all([loadOrderComparison(), loadUserCountComparison()]);
loading.value = false; loading.value = false;
}); });
</script> </script>
@@ -61,7 +67,9 @@ onMounted(async () => {
prefix="¥" prefix="¥"
:decimals="2" :decimals="2"
:value="fenToYuan(orderComparison?.value?.orderPayPrice || 0)" :value="fenToYuan(orderComparison?.value?.orderPayPrice || 0)"
:reference="fenToYuan(orderComparison?.reference?.orderPayPrice || 0)" :reference="
fenToYuan(orderComparison?.reference?.orderPayPrice || 0)
"
/> />
</Col> </Col>
<Col :md="6" :sm="12" :xs="24"> <Col :md="6" :sm="12" :xs="24">
@@ -89,7 +97,6 @@ onMounted(async () => {
/> />
</Col> </Col>
</Row> </Row>
<!-- 快捷入口和运营数据 --> <!-- 快捷入口和运营数据 -->
<Row :gutter="16"> <Row :gutter="16">
<Col :md="12" :xs="24"> <Col :md="12" :xs="24">
@@ -99,7 +106,6 @@ onMounted(async () => {
<OperationDataCard /> <OperationDataCard />
</Col> </Col>
</Row> </Row>
<!-- 会员概览和会员终端 --> <!-- 会员概览和会员终端 -->
<Row :gutter="16"> <Row :gutter="16">
<Col :md="18" :sm="24" :xs="24"> <Col :md="18" :sm="24" :xs="24">
@@ -109,10 +115,8 @@ onMounted(async () => {
<MemberTerminalCard /> <MemberTerminalCard />
</Col> </Col>
</Row> </Row>
<!-- 交易量趋势 --> <!-- 交易量趋势 -->
<TradeTrendCard /> <TradeTrendCard />
<!-- 会员统计 --> <!-- 会员统计 -->
<MemberStatisticsCard /> <MemberStatisticsCard />
</div> </div>

View File

@@ -1,31 +1,32 @@
<script lang="ts" setup> <script lang="ts" setup>
import { computed } from 'vue'; import { computed } from 'vue';
import { Card, Tag } from 'ant-design-vue';
import { IconifyIcon } from '@vben/icons'; import { IconifyIcon } from '@vben/icons';
import { erpCalculatePercentage } from '@vben/utils'; import { Card, Tag } from 'ant-design-vue';
/** 交易对照卡片 */ /** 交易对照卡片 */
defineOptions({ name: 'ComparisonCard' }); defineOptions({ name: 'ComparisonCard' });
interface Props {
title: string;
tag?: string;
prefix?: string;
value: number | string;
reference: number | string;
decimals?: number;
}
const props = withDefaults(defineProps<Props>(), { const props = withDefaults(defineProps<Props>(), {
title: '',
tag: '', tag: '',
prefix: '', prefix: '',
value: 0,
reference: 0,
decimals: 0, decimals: 0,
}); });
// 计算环比 interface Props {
title?: string;
tag?: string;
prefix?: string;
value?: number | string;
reference?: number | string;
decimals?: number;
}
/** 计算环比百分比 */
const percent = computed(() => { const percent = computed(() => {
const refValue = Number(props.reference); const refValue = Number(props.reference);
const curValue = Number(props.value); const curValue = Number(props.value);
@@ -33,12 +34,13 @@ const percent = computed(() => {
return ((curValue - refValue) / refValue) * 100; return ((curValue - refValue) / refValue) * 100;
}); });
// 格式化数值 /** 格式化今日数据 */
const formattedValue = computed(() => { const formattedValue = computed(() => {
const numValue = Number(props.value); const numValue = Number(props.value);
return numValue.toFixed(props.decimals); return numValue.toFixed(props.decimals);
}); });
/** 格式化昨日数据 */
const formattedReference = computed(() => { const formattedReference = computed(() => {
const numValue = Number(props.reference); const numValue = Number(props.reference);
return numValue.toFixed(props.decimals); return numValue.toFixed(props.decimals);
@@ -46,17 +48,18 @@ const formattedReference = computed(() => {
</script> </script>
<template> <template>
<Card :bordered="false" class="comparison-card"> <Card :bordered="false" class="h-full">
<div class="flex flex-col gap-2"> <div class="flex flex-col gap-2">
<div class="flex items-center justify-between text-gray-500"> <div class="flex items-center justify-between text-gray-500">
<span>{{ title }}</span> <span>{{ title }}</span>
<Tag v-if="tag">{{ tag }}</Tag> <Tag v-if="tag">{{ tag }}</Tag>
</div> </div>
<div class="flex items-baseline justify-between"> <div class="flex items-baseline justify-between">
<div class="text-3xl font-semibold"> <div class="text-3xl">{{ prefix }}{{ formattedValue }}</div>
{{ prefix }}{{ formattedValue }} <span
</div> :class="percent > 0 ? 'text-red-500' : 'text-green-500'"
<span :class="percent > 0 ? 'text-red-500' : 'text-green-500'"> class="flex items-center gap-0.5"
>
{{ Math.abs(percent).toFixed(2) }}% {{ Math.abs(percent).toFixed(2) }}%
<IconifyIcon <IconifyIcon
:icon="percent > 0 ? 'ep:caret-top' : 'ep:caret-bottom'" :icon="percent > 0 ? 'ep:caret-top' : 'ep:caret-bottom'"
@@ -73,10 +76,3 @@ const formattedReference = computed(() => {
</div> </div>
</Card> </Card>
</template> </template>
<style lang="less" scoped>
.comparison-card {
height: 100%;
}
</style>