197 lines
4.8 KiB
Vue
197 lines
4.8 KiB
Vue
![]() |
<template>
|
|||
|
<view class="app-container">
|
|||
|
<HeaderBar title="统计数据" text-align="center" enable-back />
|
|||
|
|
|||
|
<!-- 时间选择器 -->
|
|||
|
<view class="time-selector">
|
|||
|
<view class="preset-times">
|
|||
|
<view
|
|||
|
v-for="(item, index) in presetTimes"
|
|||
|
:key="index"
|
|||
|
:class="['preset-item', { active: currentTimeType === item.type }]"
|
|||
|
@tap="handlePresetTimeSelect(item.type)"
|
|||
|
>
|
|||
|
{{ item.label }}
|
|||
|
</view>
|
|||
|
</view>
|
|||
|
<view class="custom-time">
|
|||
|
<uni-datetime-picker
|
|||
|
v-model="dateRange"
|
|||
|
type="daterange"
|
|||
|
@change="handleDateRangeChange"
|
|||
|
/>
|
|||
|
</view>
|
|||
|
</view>
|
|||
|
|
|||
|
<!-- 数据统计卡片 -->
|
|||
|
<view class="stat-cards">
|
|||
|
<view class="stat-card" v-for="(item, index) in statData" :key="index">
|
|||
|
<view class="stat-title">{{ item.title }}</view>
|
|||
|
<view class="stat-value">{{ item.value }}</view>
|
|||
|
</view>
|
|||
|
</view>
|
|||
|
</view>
|
|||
|
</template>
|
|||
|
|
|||
|
<script>
|
|||
|
import HeaderBar from '@/components/HeaderBar.vue'
|
|||
|
import { mchGetTaskStatistics } from '@/api/mch/dashboard'
|
|||
|
|
|||
|
export default {
|
|||
|
components: {
|
|||
|
HeaderBar
|
|||
|
},
|
|||
|
data() {
|
|||
|
return {
|
|||
|
currentTimeType: 'today',
|
|||
|
dateRange: [],
|
|||
|
presetTimes: [
|
|||
|
{ type: 'today', label: '今日' },
|
|||
|
{ type: 'yesterday', label: '昨日' },
|
|||
|
{ type: 'last7days', label: '近7天' },
|
|||
|
{ type: 'last30days', label: '近30天' }
|
|||
|
],
|
|||
|
statData: [
|
|||
|
{ title: '接单数量', value: 0 },
|
|||
|
{ title: '完成数量', value: 0 },
|
|||
|
{ title: '取消数量', value: 0 },
|
|||
|
{ title: '售后数量', value: 0 }
|
|||
|
]
|
|||
|
}
|
|||
|
},
|
|||
|
created() {
|
|||
|
this.initDateRange()
|
|||
|
this.fetchStatData()
|
|||
|
},
|
|||
|
methods: {
|
|||
|
initDateRange() {
|
|||
|
const today = new Date()
|
|||
|
this.dateRange = [today, today]
|
|||
|
},
|
|||
|
handlePresetTimeSelect(type) {
|
|||
|
this.currentTimeType = type
|
|||
|
const today = new Date()
|
|||
|
let startDate = new Date()
|
|||
|
|
|||
|
switch(type) {
|
|||
|
case 'today':
|
|||
|
startDate = today
|
|||
|
break
|
|||
|
case 'yesterday':
|
|||
|
startDate = new Date(today.setDate(today.getDate() - 1))
|
|||
|
break
|
|||
|
case 'last7days':
|
|||
|
startDate = new Date(today.setDate(today.getDate() - 6))
|
|||
|
break
|
|||
|
case 'last30days':
|
|||
|
startDate = new Date(today.setDate(today.getDate() - 29))
|
|||
|
break
|
|||
|
}
|
|||
|
|
|||
|
this.dateRange = [startDate, new Date()]
|
|||
|
this.fetchStatData()
|
|||
|
},
|
|||
|
handleDateRangeChange(e) {
|
|||
|
this.currentTimeType = 'custom'
|
|||
|
if (Array.isArray(e)) {
|
|||
|
this.dateRange = e
|
|||
|
this.fetchStatData()
|
|||
|
}
|
|||
|
},
|
|||
|
fetchStatData() {
|
|||
|
const formatDate = (dateStr) => {
|
|||
|
// 如果已经是YYYY-MM-DD格式,直接返回
|
|||
|
if (typeof dateStr === 'string' && dateStr.match(/^\d{4}-\d{2}-\d{2}$/)) {
|
|||
|
return dateStr
|
|||
|
}
|
|||
|
// 如果是时间戳或日期字符串,转换为Date对象
|
|||
|
const date = new Date(dateStr)
|
|||
|
const year = date.getFullYear()
|
|||
|
const month = String(date.getMonth() + 1).padStart(2, '0')
|
|||
|
const day = String(date.getDate()).padStart(2, '0')
|
|||
|
return `${year}-${month}-${day}`
|
|||
|
}
|
|||
|
|
|||
|
const formattedDateRange = this.dateRange.map(date => formatDate(date))
|
|||
|
|
|||
|
this.$modal.loading("加载中");
|
|||
|
mchGetTaskStatistics({
|
|||
|
mchId: this.$store.state.user.mchId,
|
|||
|
dateRange: formattedDateRange
|
|||
|
}).then(res => {
|
|||
|
let data = res.data;
|
|||
|
this.statData = [
|
|||
|
{ title: '接单数量', value: data.receiveCount },
|
|||
|
{ title: '完成数量', value: data.completeCount },
|
|||
|
{ title: '取消数量', value: data.cancelCount },
|
|||
|
{ title: '售后数量', value: data.afterSaleCount }
|
|||
|
]
|
|||
|
this.$modal.closeLoading();
|
|||
|
})
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
</script>
|
|||
|
|
|||
|
<style lang="scss" scoped>
|
|||
|
.app-container {
|
|||
|
padding: 0 20rpx 20rpx;
|
|||
|
}
|
|||
|
|
|||
|
.time-selector {
|
|||
|
margin: 20rpx 0;
|
|||
|
|
|||
|
.preset-times {
|
|||
|
display: flex;
|
|||
|
justify-content: space-between;
|
|||
|
margin-bottom: 20rpx;
|
|||
|
gap: 10rpx;
|
|||
|
|
|||
|
.preset-item {
|
|||
|
padding: 10rpx 20rpx;
|
|||
|
background-color: #ffffff;
|
|||
|
border-radius: 8rpx;
|
|||
|
font-size: 28rpx;
|
|||
|
flex: 1;
|
|||
|
text-align: center;
|
|||
|
|
|||
|
&.active {
|
|||
|
background-color: #007AFF;
|
|||
|
color: #fff;
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
.custom-time {
|
|||
|
background-color: #fff;
|
|||
|
border-radius: 8rpx;
|
|||
|
padding: 20rpx;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
.stat-cards {
|
|||
|
display: grid;
|
|||
|
grid-template-columns: repeat(2, 1fr);
|
|||
|
gap: 20rpx;
|
|||
|
margin-top: 30rpx;
|
|||
|
|
|||
|
.stat-card {
|
|||
|
background-color: #fff;
|
|||
|
border-radius: 12rpx;
|
|||
|
padding: 30rpx;
|
|||
|
box-shadow: 0 2rpx 12rpx rgba(0, 0, 0, 0.1);
|
|||
|
|
|||
|
.stat-title {
|
|||
|
font-size: 28rpx;
|
|||
|
color: #666;
|
|||
|
margin-bottom: 10rpx;
|
|||
|
}
|
|||
|
|
|||
|
.stat-value {
|
|||
|
font-size: 40rpx;
|
|||
|
color: #333;
|
|||
|
font-weight: bold;
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
</style>
|