feat:【mall 商城】交易统计、商品统计【antd】15%:product-rank-card.vue 完善
This commit is contained in:
@@ -81,7 +81,7 @@ onMounted(() => {
|
||||
:value-format="rangePickerProps.valueFormat"
|
||||
:placeholder="rangePickerProps.placeholder"
|
||||
:presets="rangePickerProps.presets"
|
||||
class="!w-[240px]"
|
||||
class="!w-[235px]"
|
||||
@change="handleDateRangeChange"
|
||||
/>
|
||||
<slot></slot>
|
||||
|
||||
@@ -1,8 +1,8 @@
|
||||
<script lang="ts" setup>
|
||||
import { DocAlert, Page } from '@vben/common-ui';
|
||||
|
||||
import ProductRankCard from './modules/product-rank-card.vue';
|
||||
import ProductSummaryCard from './modules/product-summary-card.vue';
|
||||
import ProductRankCard from './modules/rank-card.vue';
|
||||
|
||||
/** 商品统计 */
|
||||
defineOptions({ name: 'ProductStatistics' });
|
||||
|
||||
@@ -1,231 +0,0 @@
|
||||
<script lang="ts" setup>
|
||||
import type { Dayjs } from 'dayjs';
|
||||
|
||||
import type { MallProductStatisticsApi } from '#/api/mall/statistics/product';
|
||||
|
||||
import { onMounted, reactive, ref } from 'vue';
|
||||
|
||||
import { formatToFraction } from '@vben/utils';
|
||||
|
||||
import { Card, Image, Pagination, Table } from 'ant-design-vue';
|
||||
|
||||
import * as ProductStatisticsApi from '#/api/mall/statistics/product';
|
||||
import ShortcutDateRangePicker from '#/components/shortcut-date-range-picker/shortcut-date-range-picker.vue';
|
||||
|
||||
/** 商品排行 */
|
||||
defineOptions({ name: 'ProductRankCard' });
|
||||
|
||||
const loading = ref(false); // 列表的加载中
|
||||
const total = ref(0); // 列表的总页数
|
||||
const list = ref<MallProductStatisticsApi.ProductStatistics[]>([]); // 列表的数据
|
||||
const shortcutDateRangePicker = ref();
|
||||
|
||||
// 查询参数
|
||||
const queryParams = reactive({
|
||||
pageNo: 1,
|
||||
pageSize: 10,
|
||||
times: undefined as string[] | undefined,
|
||||
sortingFields: [] as any[],
|
||||
});
|
||||
|
||||
/** 格式化:访客-支付转化率 */
|
||||
const formatConvertRate = (value: number) => {
|
||||
return `${value || 0}%`;
|
||||
};
|
||||
|
||||
/** 格式化金额(分转元)*/
|
||||
const formatPrice = (value: number) => {
|
||||
return formatToFraction(value / 100, 2);
|
||||
};
|
||||
|
||||
/** 处理排序变化 */
|
||||
const handleSortChange = (sorter: any) => {
|
||||
queryParams.sortingFields =
|
||||
sorter && sorter.field
|
||||
? [
|
||||
{
|
||||
field: sorter.field,
|
||||
order: sorter.order === 'ascend' ? 'ASC' : 'DESC',
|
||||
},
|
||||
]
|
||||
: [];
|
||||
getSpuList();
|
||||
};
|
||||
|
||||
/** 处理日期范围变化 */
|
||||
const handleDateRangeChange = (times?: [Dayjs, Dayjs]) => {
|
||||
queryParams.times = times
|
||||
? [
|
||||
times[0].format('YYYY-MM-DD HH:mm:ss'),
|
||||
times[1].format('YYYY-MM-DD HH:mm:ss'),
|
||||
]
|
||||
: undefined;
|
||||
getSpuList();
|
||||
};
|
||||
|
||||
/** 查询商品列表 */
|
||||
const getSpuList = async () => {
|
||||
loading.value = true;
|
||||
try {
|
||||
const data =
|
||||
await ProductStatisticsApi.getProductStatisticsRankPage(queryParams);
|
||||
list.value = data.list;
|
||||
total.value = data.total;
|
||||
} finally {
|
||||
loading.value = false;
|
||||
}
|
||||
};
|
||||
|
||||
/** 分页变化 */
|
||||
const handlePaginationChange = (page: number, pageSize: number) => {
|
||||
queryParams.pageNo = page;
|
||||
queryParams.pageSize = pageSize;
|
||||
getSpuList();
|
||||
};
|
||||
|
||||
/** 表格列配置 */
|
||||
const columns = [
|
||||
{
|
||||
title: '商品 ID',
|
||||
dataIndex: 'spuId',
|
||||
key: 'spuId',
|
||||
width: 100,
|
||||
},
|
||||
{
|
||||
title: '商品图片',
|
||||
dataIndex: 'spuPicUrl',
|
||||
key: 'spuPicUrl',
|
||||
width: 100,
|
||||
},
|
||||
{
|
||||
title: '商品名称',
|
||||
dataIndex: 'spuName',
|
||||
key: 'spuName',
|
||||
ellipsis: true,
|
||||
width: 200,
|
||||
},
|
||||
{
|
||||
title: '浏览量',
|
||||
dataIndex: 'browseCount',
|
||||
key: 'browseCount',
|
||||
sorter: true,
|
||||
width: 100,
|
||||
},
|
||||
{
|
||||
title: '访客数',
|
||||
dataIndex: 'browseUserCount',
|
||||
key: 'browseUserCount',
|
||||
sorter: true,
|
||||
width: 100,
|
||||
},
|
||||
{
|
||||
title: '加购件数',
|
||||
dataIndex: 'cartCount',
|
||||
key: 'cartCount',
|
||||
sorter: true,
|
||||
width: 110,
|
||||
},
|
||||
{
|
||||
title: '下单件数',
|
||||
dataIndex: 'orderCount',
|
||||
key: 'orderCount',
|
||||
sorter: true,
|
||||
width: 110,
|
||||
},
|
||||
{
|
||||
title: '支付件数',
|
||||
dataIndex: 'orderPayCount',
|
||||
key: 'orderPayCount',
|
||||
sorter: true,
|
||||
width: 110,
|
||||
},
|
||||
{
|
||||
title: '支付金额',
|
||||
dataIndex: 'orderPayPrice',
|
||||
key: 'orderPayPrice',
|
||||
sorter: true,
|
||||
width: 110,
|
||||
},
|
||||
{
|
||||
title: '收藏数',
|
||||
dataIndex: 'favoriteCount',
|
||||
key: 'favoriteCount',
|
||||
sorter: true,
|
||||
width: 100,
|
||||
},
|
||||
{
|
||||
title: '访客-支付转化率(%)',
|
||||
dataIndex: 'browseConvertPercent',
|
||||
key: 'browseConvertPercent',
|
||||
sorter: true,
|
||||
width: 180,
|
||||
},
|
||||
];
|
||||
|
||||
/** 初始化 */
|
||||
onMounted(async () => {
|
||||
await getSpuList();
|
||||
});
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<Card :bordered="false" title="商品排行" class="h-full">
|
||||
<template #extra>
|
||||
<!-- 查询条件 -->
|
||||
<ShortcutDateRangePicker
|
||||
ref="shortcutDateRangePicker"
|
||||
@change="handleDateRangeChange"
|
||||
/>
|
||||
</template>
|
||||
|
||||
<!-- 排行列表 -->
|
||||
<Table
|
||||
:columns="columns"
|
||||
:data-source="list"
|
||||
:loading="loading"
|
||||
:pagination="false"
|
||||
@change="handleSortChange"
|
||||
row-key="spuId"
|
||||
>
|
||||
<template #bodyCell="{ column, record }">
|
||||
<!-- 商品图片 -->
|
||||
<template v-if="column.key === 'spuPicUrl'">
|
||||
<Image
|
||||
:src="record.spuPicUrl"
|
||||
:width="30"
|
||||
:height="30"
|
||||
:preview="true"
|
||||
fallback=""
|
||||
/>
|
||||
</template>
|
||||
|
||||
<!-- 商品名称 -->
|
||||
<template v-else-if="column.key === 'spuName'">
|
||||
<span class="break-words">{{ record.spuName }}</span>
|
||||
</template>
|
||||
|
||||
<!-- 支付金额 -->
|
||||
<template v-else-if="column.key === 'orderPayPrice'">
|
||||
¥{{ formatPrice(record.orderPayPrice) }}
|
||||
</template>
|
||||
|
||||
<!-- 访客-支付转化率 -->
|
||||
<template v-else-if="column.key === 'browseConvertPercent'">
|
||||
{{ formatConvertRate(record.browseConvertPercent) }}
|
||||
</template>
|
||||
</template>
|
||||
</Table>
|
||||
|
||||
<!-- 分页 -->
|
||||
<div class="mt-4 flex justify-center">
|
||||
<Pagination
|
||||
:current="queryParams.pageNo"
|
||||
:page-size="queryParams.pageSize"
|
||||
:total="total"
|
||||
show-size-changer
|
||||
show-quick-jumper
|
||||
@change="handlePaginationChange"
|
||||
/>
|
||||
</div>
|
||||
</Card>
|
||||
</template>
|
||||
@@ -0,0 +1,135 @@
|
||||
<script lang="ts" setup>
|
||||
import type { Dayjs } from 'dayjs';
|
||||
|
||||
import type { VxeTableGridOptions } from '#/adapter/vxe-table';
|
||||
|
||||
import { ref } from 'vue';
|
||||
|
||||
import { buildSortingField } from '@vben/request';
|
||||
import { formatDateTime } from '@vben/utils';
|
||||
|
||||
import { Card } from 'ant-design-vue';
|
||||
|
||||
import { useVbenVxeGrid } from '#/adapter/vxe-table';
|
||||
import * as ProductStatisticsApi from '#/api/mall/statistics/product';
|
||||
import ShortcutDateRangePicker from '#/components/shortcut-date-range-picker/shortcut-date-range-picker.vue';
|
||||
|
||||
/** 商品排行 */
|
||||
defineOptions({ name: 'ProductRankCard' });
|
||||
|
||||
const searchTimes = ref<string[]>([]);
|
||||
|
||||
/** 处理日期范围变化 */
|
||||
const handleDateRangeChange = (times?: [Dayjs, Dayjs]) => {
|
||||
if (times?.length !== 2) {
|
||||
return;
|
||||
}
|
||||
searchTimes.value = [
|
||||
formatDateTime(times[0]) as string,
|
||||
formatDateTime(times[1]) as string,
|
||||
];
|
||||
gridApi.query();
|
||||
};
|
||||
|
||||
const columns: VxeTableGridOptions['columns'] = [
|
||||
{ field: 'spuId', title: '商品 ID', minWidth: 100 },
|
||||
{
|
||||
field: 'picUrl',
|
||||
title: '商品图片',
|
||||
minWidth: 100,
|
||||
cellRender: { name: 'CellImage' },
|
||||
},
|
||||
{
|
||||
field: 'name',
|
||||
title: '商品名称',
|
||||
minWidth: 200,
|
||||
},
|
||||
{
|
||||
field: 'browseCount',
|
||||
title: '浏览量',
|
||||
minWidth: 100,
|
||||
sortable: true,
|
||||
},
|
||||
{
|
||||
field: 'browseUserCount',
|
||||
title: '访客数',
|
||||
minWidth: 100,
|
||||
sortable: true,
|
||||
},
|
||||
{
|
||||
field: 'cartCount',
|
||||
title: '加购件数',
|
||||
minWidth: 110,
|
||||
sortable: true,
|
||||
},
|
||||
{
|
||||
field: 'orderCount',
|
||||
title: '下单件数',
|
||||
minWidth: 110,
|
||||
sortable: true,
|
||||
},
|
||||
{
|
||||
field: 'orderPayCount',
|
||||
title: '支付件数',
|
||||
minWidth: 110,
|
||||
sortable: true,
|
||||
},
|
||||
{
|
||||
field: 'orderPayPrice',
|
||||
title: '支付金额(元)',
|
||||
minWidth: 120,
|
||||
formatter: 'formatFenToYuanAmount',
|
||||
sortable: true,
|
||||
},
|
||||
{
|
||||
field: 'favoriteCount',
|
||||
title: '收藏数',
|
||||
minWidth: 100,
|
||||
sortable: true,
|
||||
},
|
||||
{
|
||||
field: 'browseConvertPercent',
|
||||
title: '访客-支付转化率(%)',
|
||||
minWidth: 160,
|
||||
sortable: true,
|
||||
formatter: ({ cellValue }) => `${cellValue || 0}%`,
|
||||
},
|
||||
];
|
||||
|
||||
const [Grid, gridApi] = useVbenVxeGrid({
|
||||
gridOptions: {
|
||||
columns,
|
||||
height: 400,
|
||||
keepSource: true,
|
||||
proxyConfig: {
|
||||
ajax: {
|
||||
query: async ({ page, sorts }) => {
|
||||
return await ProductStatisticsApi.getProductStatisticsRankPage({
|
||||
pageNo: page.currentPage,
|
||||
pageSize: page.pageSize,
|
||||
times: searchTimes.value.length > 0 ? searchTimes.value : undefined,
|
||||
...buildSortingField(sorts),
|
||||
});
|
||||
},
|
||||
},
|
||||
sort: true,
|
||||
},
|
||||
sortConfig: {
|
||||
remote: true,
|
||||
multiple: false,
|
||||
},
|
||||
toolbarConfig: {
|
||||
refresh: false,
|
||||
},
|
||||
} as VxeTableGridOptions,
|
||||
});
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<Card :bordered="false" title="商品排行">
|
||||
<template #extra>
|
||||
<ShortcutDateRangePicker @change="handleDateRangeChange" />
|
||||
</template>
|
||||
<Grid />
|
||||
</Card>
|
||||
</template>
|
||||
@@ -102,7 +102,7 @@ onMounted(() => {
|
||||
:start-placeholder="rangePickerProps.startPlaceholder"
|
||||
:end-placeholder="rangePickerProps.endPlaceholder"
|
||||
:default-time="rangePickerProps.defaultTime as any"
|
||||
class="!w-[360px]"
|
||||
class="!w-[215px]"
|
||||
@change="handleDateRangeChange"
|
||||
/>
|
||||
<slot></slot>
|
||||
|
||||
@@ -23,19 +23,4 @@ export * from './window';
|
||||
export { default as cloneDeep } from 'lodash.clonedeep';
|
||||
export { default as get } from 'lodash.get';
|
||||
export { default as isEqual } from 'lodash.isequal';
|
||||
export { default as set } from 'lodash.set';
|
||||
|
||||
/**
|
||||
* 构建排序字段
|
||||
* @param prop 字段名称
|
||||
* @param order 顺序
|
||||
*/
|
||||
export const buildSortingField = ({
|
||||
prop,
|
||||
order,
|
||||
}: {
|
||||
order: 'ascending' | 'descending';
|
||||
prop: string;
|
||||
}) => {
|
||||
return { field: prop, order: order === 'ascending' ? 'asc' : 'desc' };
|
||||
};
|
||||
export { default as set } from 'lodash.set';
|
||||
@@ -162,4 +162,21 @@ class RequestClient {
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 构建排序字段,处理 vxe 排序条件
|
||||
*
|
||||
* add by 芋艿
|
||||
*/
|
||||
export const buildSortingField = (sorts: any[]) => {
|
||||
if (!sorts || sorts.length === 0) {
|
||||
return {};
|
||||
}
|
||||
const result: Record<string, any> = {};
|
||||
sorts.forEach((sort: any, index: number) => {
|
||||
result[`sortingFields[${index}].field`] = sort.field;
|
||||
result[`sortingFields[${index}].order`] = sort.order;
|
||||
});
|
||||
return result;
|
||||
};
|
||||
|
||||
export { RequestClient };
|
||||
|
||||
Reference in New Issue
Block a user