1522 lines
42 KiB
Vue
1522 lines
42 KiB
Vue
<template>
|
||
<div class="app-container" style="padding-bottom: 100px">
|
||
<!-- <div class="card_wrap">
|
||
<div class="card">
|
||
<div class="header">
|
||
<div class="card_title">总销售额</div>
|
||
<el-tooltip effect="dark" content="订单支付金额" placement="top">
|
||
<i class="icon el-icon-warning-outline"></i>
|
||
</el-tooltip>
|
||
</div>
|
||
<div class="number">{{ topData.totalSales || 0 }}</div>
|
||
<div class="row">平均每单{{ topData.averageSales || 0 }}</div>
|
||
<div class="row">今日销售额{{ topData.totalSalesToday || 0 }}</div>
|
||
</div>
|
||
<div class="card">
|
||
<div class="header">
|
||
<div class="card_title">支付笔数</div>
|
||
</div>
|
||
<div class="number">{{ topData.paymentsNumber }}</div>
|
||
<div class="row" ref="cardPayChart" style="padding-bottom: 2px;"></div>
|
||
<div class="row">
|
||
今日支付笔数{{ topData.paymentsNumberToday || 0 }}
|
||
</div>
|
||
</div>
|
||
<div class="card">
|
||
<div class="header">
|
||
<div class="card_title">访问量</div>
|
||
</div>
|
||
<div class="number">{{ topData.totalVisits }}</div>
|
||
<div class="row" ref="cardCountChart" style="padding-bottom: 2px;"></div>
|
||
<div class="row">
|
||
<div class="dot"></div>
|
||
今日访问 {{ topData.totalVisitsToday || 0 }}
|
||
</div>
|
||
</div>
|
||
<div class="card">
|
||
<div class="header">
|
||
<div class="card_title">用户数</div>
|
||
</div>
|
||
<div class="number">{{ topData.totalUser }}</div>
|
||
<div class="row" ref="cardUserChart" style="padding-bottom: 2px;"></div>
|
||
<div class="row">
|
||
今日新增 {{ topData.userToday || 0 }}
|
||
<i class="icon el-icon-caret-top"></i>
|
||
</div>
|
||
</div>
|
||
</div> -->
|
||
<div class="h_card_wrap">
|
||
<div class="status_wrap">
|
||
<div class="left">
|
||
<div class="dot"></div>
|
||
<span>营业</span>
|
||
</div>
|
||
<div class="time_wrap">
|
||
<el-radio-group v-model="timeValue" @change="timeChange">
|
||
<el-radio-button value="0">今天</el-radio-button>
|
||
<el-radio-button value="-1">昨天</el-radio-button>
|
||
<el-radio-button value="-7">最近7天</el-radio-button>
|
||
<el-radio-button value="-30">最近30天</el-radio-button>
|
||
<el-radio-button value="week">本周</el-radio-button>
|
||
<el-radio-button value="month">本月</el-radio-button>
|
||
<el-radio-button value="custom">自定义</el-radio-button>
|
||
</el-radio-group>
|
||
<el-date-picker v-model="query.createdAt" type="daterange" range-separator="至" start-placeholder="开始日期"
|
||
end-placeholder="结束日期" value-format="YYYY-MM-DD " v-if="timeValue == 'custom'"
|
||
@change="summarytrade"></el-date-picker>
|
||
</div>
|
||
</div>
|
||
<div class="content" v-loading="tradeLoading">
|
||
<div class="top">
|
||
<div class="item earnings">
|
||
<div class="num_wrap">
|
||
<div class="num">{{ formatDecimal(yingyeShiShou || 0) }}</div>
|
||
<div class="tips">
|
||
营业实收(元)
|
||
<el-tooltip popper-class="popper" effect="light" placement="bottom">
|
||
<template #content>
|
||
<div class="tips_row">
|
||
<div class="item">
|
||
<div class="left">
|
||
<img class="icon" src="@/assets/images/data/wx.png" />
|
||
<span>微信小程序</span>
|
||
</div>
|
||
<span class="num">{{ trade.wechatPayAmount }}</span>
|
||
</div>
|
||
<div class="item">
|
||
<div class="left">
|
||
<img class="icon" src="@/assets/images/data/ali.png" />
|
||
<span>支付宝小程序</span>
|
||
</div>
|
||
<span class="num">{{ trade.aliPayAmount }}</span>
|
||
</div>
|
||
<div class="item">
|
||
<div class="left">
|
||
<img class="icon" src="@/assets/images/data/scan.png" />
|
||
<span>主扫收款</span>
|
||
</div>
|
||
<span class="num">{{ trade.scanPayAmount }}</span>
|
||
</div>
|
||
<div class="item">
|
||
<div class="left">
|
||
<img class="icon" src="@/assets/images/data/cash.png" />
|
||
<span>现金</span>
|
||
</div>
|
||
<span class="num">{{ trade.cashPayAmount }}</span>
|
||
</div>
|
||
<div class="item">
|
||
<div class="left">
|
||
<img class="icon" src="@/assets/images/data/cash.png" />
|
||
<span>充值</span>
|
||
</div>
|
||
<span class="num">{{ trade.rechargeAmount }}</span>
|
||
</div>
|
||
<div class="item">
|
||
<div class="left">
|
||
<img class="icon" src="@/assets/images/data/cash.png" />
|
||
<span>挂账</span>
|
||
</div>
|
||
<span class="num">{{ trade.creditPayAmount }}</span>
|
||
</div>
|
||
</div>
|
||
</template>
|
||
<el-icon>
|
||
<QuestionFilled />
|
||
</el-icon>
|
||
</el-tooltip>
|
||
</div>
|
||
</div>
|
||
<div class="line_wrap">
|
||
<div class="line_item">
|
||
<div class="line_item_top">
|
||
<div>销售金额(元)</div>
|
||
<!-- <div class="t">{{ formatDecimal(tradeSale.totalSaleAmount || 0) }}</div> -->
|
||
</div>
|
||
<div class="line_gropress">
|
||
<div class="gropress l" :style="{
|
||
width: `${trade.saleAmount
|
||
? (trade.saleAmount / (trade.saleAmount * 1 + trade.refundAmount * 1)) *
|
||
100
|
||
: 0
|
||
}%`,
|
||
}"></div>
|
||
<div class="gropress r" :style="{
|
||
width: `${trade.refundAmount
|
||
? (trade.refundAmount / (trade.saleAmount * 1 + trade.refundAmount * 1)) *
|
||
100
|
||
: 0
|
||
}%`,
|
||
}"></div>
|
||
</div>
|
||
<div class="line_btm">
|
||
<el-icon class="icon el-icon-caret-right" />
|
||
<div class="info">
|
||
<span class="l_t">收:{{ formatDecimal(trade.saleAmount || 0) }}</span>
|
||
<span class="l_r">退:{{ formatDecimal(trade.refundAmount || 0) }}</span>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<div class="line_item">
|
||
<div class="line_item_top">
|
||
<div>充值金额(元)</div>
|
||
<!-- <div class="t">{{ formatDecimal(tradeSale.totalVipAmount || 0) }}</div> -->
|
||
</div>
|
||
<div class="line_gropress">
|
||
<div class="gropress l" :style="{
|
||
width: `${trade.rechargeAmount
|
||
? (trade.rechargeAmount /
|
||
(trade.memberPayAmount + trade.rechargeRefundAmount * 1)) *
|
||
100
|
||
: 0
|
||
}%`,
|
||
}"></div>
|
||
<div class="gropress r" :style="{
|
||
width: `${trade.rechargeRefundAmount
|
||
? (trade.rechargeRefundAmount /
|
||
(trade.memberPayAmount + trade.rechargeRefundAmount * 1)) *
|
||
100
|
||
: 0
|
||
}%`,
|
||
}"></div>
|
||
</div>
|
||
<div class="line_btm">
|
||
<el-icon class="icon el-icon-caret-right" />
|
||
<div class="info">
|
||
<span class="l_t">收:{{ formatDecimal(trade.rechargeAmount || 0) }}</span>
|
||
<span class="l_r">
|
||
退:{{ formatDecimal(trade.rechargeRefundAmount || 0) }}
|
||
</span>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<div class="item data">
|
||
<div class="data_item">
|
||
<div class="num_wrap">
|
||
<div class="num">{{ formatDecimal(trade.memberPayAmount || 0) }}</div>
|
||
<div class="tips">会员消费(元)</div>
|
||
</div>
|
||
</div>
|
||
<div class="data_item_right">
|
||
<div class="t">
|
||
<span>新增会员数</span>
|
||
<span class="n">{{ trade.newMemberCoun || 0 }}(人)</span>
|
||
</div>
|
||
<div class="t">
|
||
<span>会员消费笔数</span>
|
||
<span class="n">{{ trade.memberPayCount || 0 }}</span>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<div class="btm">
|
||
<div class="item item1">
|
||
<div class="title">客单价</div>
|
||
<div class="icon_wrap">
|
||
<img class="img" src="@/assets/images/data_home_item1_icon.png" />
|
||
<div class="t">{{ formatDecimal(trade.customerUnitPrice || 0) }}</div>
|
||
</div>
|
||
</div>
|
||
<div class="item item2">
|
||
<div class="title">
|
||
翻台率
|
||
<el-tooltip effect="dark" content="翻台率=(客单数-桌台数)/桌台数*100%" placement="top">
|
||
<el-icon>
|
||
<QuestionFilled />
|
||
</el-icon>
|
||
</el-tooltip>
|
||
</div>
|
||
<div class="icon_wrap">
|
||
<img class="img" src="@/assets/images/data_home_item2_icon.png" />
|
||
<div class="t">{{ trade.tableTurnoverRate }}</div>
|
||
</div>
|
||
</div>
|
||
<div class="item item3">
|
||
<div class="title">优惠金额</div>
|
||
<div class="icon_wrap">
|
||
<img class="img" src="@/assets/images/data_home_item3_icon.png" />
|
||
<div class="t">{{ formatDecimal(trade.discountAmount || 0) }}</div>
|
||
</div>
|
||
</div>
|
||
<div class="item item4">
|
||
<div class="title">优惠笔数</div>
|
||
<div class="icon_wrap">
|
||
<img class="img" src="@/assets/images/data_home_item4_icon.png" />
|
||
<div class="t">{{ trade.discountCount || 0 }}</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<!-- 销售额 -->
|
||
<div class="chart_wrap" style="display: flex">
|
||
<div class="item">
|
||
<div class="header">
|
||
<div class="tab_wrap">
|
||
<div class="item" :class="{ active: lineChartType == 0 }" @click="lineChartTypeChange(0)">
|
||
销售趋势
|
||
</div>
|
||
<div class="item" :class="{ active: lineChartType == 1 }" @click="lineChartTypeChange(1)">
|
||
支付占比
|
||
</div>
|
||
</div>
|
||
<el-radio-group v-model="saleActive" @change="lineChartTypeChange(lineChartType)">
|
||
<el-radio-button value="7">近7天</el-radio-button>
|
||
<el-radio-button value="30">30天</el-radio-button>
|
||
</el-radio-group>
|
||
</div>
|
||
<div class="chart" ref="saleChart" v-loading="saleLoading" style="height: 350px" v-show="lineChartType == 0">
|
||
</div>
|
||
<div class="chart" ref="payChart" v-loading="payChartLoading" style="height: 350px" v-show="lineChartType == 1">
|
||
</div>
|
||
</div>
|
||
<!-- 商品销售排行 -->
|
||
<div class="item" style="margin-left: 20px">
|
||
<div class="header">
|
||
<div class="tab_wrap">
|
||
<div class="item active">商品销售排行</div>
|
||
</div>
|
||
<el-radio-group v-model="saleTableActive" @change="rankChange">
|
||
<!-- <el-radio-button label="1">今天</el-radio-button> -->
|
||
<el-radio-button value="7">近7天</el-radio-button>
|
||
<el-radio-button value="30">30天</el-radio-button>
|
||
</el-radio-group>
|
||
</div>
|
||
<div class="sale_data">
|
||
<!-- <div class="card">
|
||
<div class="sale_data_header">
|
||
<div class="card_title">销售数量</div>
|
||
</div>
|
||
<div class="number">{{ productCount }}</div>
|
||
<div class="product_chart_wrap" ref="productCountChart"></div>
|
||
</div> -->
|
||
<!-- <div class="card">
|
||
<div class="sale_data_header">
|
||
<div class="card_title">销售金额</div>
|
||
</div>
|
||
<div class="number">¥{{ productSum }}</div>
|
||
<div class="product_chart_wrap" ref="productSumChart"></div>
|
||
</div> -->
|
||
</div>
|
||
<div class="table w-full">
|
||
<el-table :data="saleTable" v-loading="saleTableLoading" style="width: 100%">
|
||
<!-- <el-table-column label="排名" prop="productId"></el-table-column> -->
|
||
<el-table-column label="商品名称" prop="productName"></el-table-column>
|
||
<el-table-column label="数量" prop="number"></el-table-column>
|
||
<el-table-column label="金额" prop="amount"></el-table-column>
|
||
</el-table>
|
||
<div class="head-container" style="padding-top: 20px; display: flex; justify-content: flex-end">
|
||
<el-pagination :total="saleTableTotal" :page-size="saleTableSize" :current-page="saleTablePage"
|
||
@current-change="paginationChange" layout="total, prev, pager, next, jumper"></el-pagination>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<!-- <div class="chart_wrap" style="display: flex;">
|
||
<div class="item" style="margin-left: 20px;">
|
||
<div class="header">
|
||
<div class="tab_wrap">
|
||
<div class="item active">支付占比类型</div>
|
||
</div>
|
||
<el-radio-group v-model="payChartDay" @change="datePayType">
|
||
<el-radio-button label="7">近7天</el-radio-button>
|
||
<el-radio-button label="30">30天</el-radio-button>
|
||
</el-radio-group>
|
||
</div>
|
||
<div style="height: 400px;margin-top: 30px;" ref="payChart" v-loading="payChartLoading"></div>
|
||
</div>
|
||
</div> -->
|
||
</div>
|
||
</template>
|
||
|
||
<script>
|
||
import dataSummaryApi from "@/api/order/data-summary";
|
||
import dayjs from "dayjs";
|
||
import * as echarts from "echarts";
|
||
import { debounce, formatDecimal } from "@/utils/tools";
|
||
export default {
|
||
name: "home",
|
||
data() {
|
||
return {
|
||
payCount: [
|
||
{
|
||
icon: "https://cashier-oss.oss-cn-beijing.aliyuncs.com/static/wx.png",
|
||
isAmount: "1",
|
||
payAmount: 0,
|
||
payType: "微信小程序",
|
||
saveAmount: null,
|
||
},
|
||
{
|
||
icon: "https://cashier-oss.oss-cn-beijing.aliyuncs.com/static/ali.png",
|
||
isAmount: "1",
|
||
payAmount: 0,
|
||
payType: "支付宝小程序",
|
||
saveAmount: null,
|
||
},
|
||
{
|
||
icon: "https://cashier-oss.oss-cn-beijing.aliyuncs.com/static/scan.png",
|
||
isAmount: "1",
|
||
payAmount: 0,
|
||
payType: "主扫收款",
|
||
saveAmount: null,
|
||
},
|
||
{
|
||
icon: "https://cashier-oss.oss-cn-beijing.aliyuncs.com/static/bscan.png",
|
||
isAmount: "1",
|
||
payAmount: 0,
|
||
payType: "收款码收款",
|
||
saveAmount: null,
|
||
},
|
||
{
|
||
icon: "https://cashier-oss.oss-cn-beijing.aliyuncs.com/static/cash.png",
|
||
isAmount: "1",
|
||
payAmount: 0,
|
||
payType: "现金",
|
||
saveAmount: null,
|
||
},
|
||
{
|
||
icon: "https://cashier-oss.oss-cn-beijing.aliyuncs.com/static/vipIn.png",
|
||
isAmount: "1",
|
||
payAmount: 0,
|
||
payType: "充值",
|
||
saveAmount: null,
|
||
},
|
||
],
|
||
trade: {},
|
||
formatDecimal,
|
||
topData: "",
|
||
saleTab: "sale",
|
||
saleActive: "7",
|
||
cardPayChart: null,
|
||
cardCountChart: null,
|
||
cardUserChart: null,
|
||
saleLoading: false,
|
||
saleChart: null,
|
||
payChartDay: "7",
|
||
payChartLoading: false,
|
||
payChart: null,
|
||
chartType: 1,
|
||
productCount: 0,
|
||
productSum: 0,
|
||
saleTableActive: "7",
|
||
saleTable: [],
|
||
saleTableLoading: false,
|
||
saleTablePage: 1,
|
||
saleTableTotal: 0,
|
||
saleTableSize: 5,
|
||
__resizeHandler: null,
|
||
productCountChart: null,
|
||
productSumChart: null,
|
||
lineChartType: 0,
|
||
timeValue: "0",
|
||
query: {
|
||
createdAt: "",
|
||
},
|
||
tradeLoading: false,
|
||
tradeSale: {
|
||
payCount: [],
|
||
},
|
||
tradeVip: "",
|
||
tradeCount: "",
|
||
};
|
||
},
|
||
computed: {
|
||
yingyeShiShou() {
|
||
if (!this.trade) {
|
||
return 0;
|
||
}
|
||
return (
|
||
this.trade.wechatPayAmount * 1 +
|
||
this.trade.aliPayAmount * 1 +
|
||
this.trade.aliPayAmount * 1 +
|
||
this.trade.memberPayAmount * 1 +
|
||
this.trade.scanPayAmount * 1 +
|
||
this.trade.cashPayAmount * 1 +
|
||
this.trade.rechargeAmount * 1
|
||
);
|
||
},
|
||
},
|
||
mounted() {
|
||
// 增加首页提示是否账号30天过期
|
||
let date = dayjs(localStorage.getItem("expireDate")).diff(dayjs().format("YYYY-MM-DD"), "day");
|
||
if (date <= 30 && date >= 0) {
|
||
this.$alert(
|
||
`店铺账号有限期至${localStorage.getItem("expireDate")},店铺账号到期剩余${date}天!`,
|
||
"提示"
|
||
);
|
||
}
|
||
// this.summaryGet();
|
||
this.dateAmount();
|
||
this.dateProduct();
|
||
// this.summaryDateGet();
|
||
this.timeChange("0");
|
||
|
||
this.__resizeHandler = debounce(() => {
|
||
if (this.saleChart) {
|
||
this.saleChart.resize();
|
||
}
|
||
if (this.payChart) {
|
||
this.payChart.resize();
|
||
}
|
||
if (this.cardPayChart) {
|
||
this.cardPayChart.resize();
|
||
}
|
||
if (this.cardUserChart) {
|
||
this.cardUserChart.resize();
|
||
}
|
||
if (this.productCountChart) {
|
||
this.productCountChart.resize();
|
||
}
|
||
// if (this.productSumChart) {
|
||
// this.productSumChart.resize();
|
||
// }
|
||
}, 100);
|
||
window.addEventListener("resize", this.__resizeHandler);
|
||
// this.initCardUserChart();
|
||
},
|
||
methods: {
|
||
// 切换时间
|
||
timeChange(e) {
|
||
const format = ["YYYY-MM-DD 00:00:00", "YYYY-MM-DD 23:59:59"];
|
||
switch (e) {
|
||
case "":
|
||
// 全部
|
||
this.query.createdAt = [];
|
||
break;
|
||
case "0":
|
||
// 今天
|
||
this.query.createdAt = [dayjs().format(format[0]), dayjs().format(format[1])];
|
||
break;
|
||
case "-1":
|
||
// 昨天
|
||
this.query.createdAt = [
|
||
dayjs().add(-1, "d").format(format[0]),
|
||
dayjs().add(-1, "d").format(format[1]),
|
||
];
|
||
break;
|
||
case "-7":
|
||
// 最近7天
|
||
this.query.createdAt = [
|
||
dayjs().add(-7, "d").format(format[0]),
|
||
dayjs().format(format[1]),
|
||
];
|
||
break;
|
||
case "-30":
|
||
// 最近7天
|
||
this.query.createdAt = [
|
||
dayjs().add(-30, "d").format(format[0]),
|
||
dayjs().format(format[1]),
|
||
];
|
||
break;
|
||
case "week":
|
||
// 本周
|
||
this.query.createdAt = [
|
||
dayjs().startOf("week").format(format[0]),
|
||
dayjs().endOf("week").format(format[1]),
|
||
];
|
||
break;
|
||
case "month":
|
||
// 本周
|
||
this.query.createdAt = [
|
||
dayjs().startOf("month").format(format[0]),
|
||
dayjs().endOf("month").format(format[1]),
|
||
];
|
||
break;
|
||
case "custom":
|
||
// 自定义
|
||
this.query.createdAt = [];
|
||
break;
|
||
default:
|
||
break;
|
||
}
|
||
if (e != "custom") {
|
||
this.summarytrade();
|
||
}
|
||
},
|
||
// 获取营业板块数据
|
||
async summarytrade() {
|
||
try {
|
||
this.tradeLoading = true;
|
||
const res = await dataSummaryApi.trade({
|
||
beginDate: this.query.createdAt[0],
|
||
endDate: this.query.createdAt[1],
|
||
});
|
||
this.trade = res;
|
||
this.tradeLoading = false;
|
||
this.tradeSale = res.sale;
|
||
this.tradeVip = res.vip;
|
||
this.tradeCount = res.count;
|
||
} catch (error) {
|
||
console.log(error);
|
||
}
|
||
},
|
||
lineChartTypeChange(i) {
|
||
this.lineChartType = i;
|
||
if (i == 0) {
|
||
this.dateAmount();
|
||
} else {
|
||
this.datePayType();
|
||
}
|
||
},
|
||
// 初始化支付笔数柱状图
|
||
initCardPayChart(time = [], data = []) {
|
||
this.cardPayChart = echarts.init(this.$refs.cardPayChart);
|
||
this.cardPayChart.setOption({
|
||
tooltip: {
|
||
trigger: "axis",
|
||
},
|
||
grid: {
|
||
x: 0,
|
||
y: 0,
|
||
x2: 0,
|
||
y2: 0,
|
||
},
|
||
xAxis: [
|
||
{
|
||
type: "category",
|
||
data: time,
|
||
show: false, // 不显示坐标轴线、坐标轴刻度线和坐标轴上的文字
|
||
axisTick: {
|
||
show: false, // 不显示坐标轴刻度线
|
||
},
|
||
axisLine: {
|
||
show: false, // 不显示坐标轴线
|
||
},
|
||
axisLabel: {
|
||
show: false, // 不显示坐标轴上的文字
|
||
},
|
||
splitLine: {
|
||
show: false, // 不显示网格线
|
||
},
|
||
},
|
||
],
|
||
color: "#409eff",
|
||
yAxis: [
|
||
{
|
||
type: "value",
|
||
show: false, // 不显示坐标轴线、坐标轴刻度线和坐标轴上的文字
|
||
axisTick: {
|
||
show: false, // 不显示坐标轴刻度线
|
||
},
|
||
axisLine: {
|
||
show: false, // 不显示坐标轴线
|
||
},
|
||
axisLabel: {
|
||
show: false, // 不显示坐标轴上的文字
|
||
},
|
||
splitLine: {
|
||
show: false, // 不显示网格线
|
||
},
|
||
},
|
||
],
|
||
series: [
|
||
{
|
||
data: data,
|
||
type: "bar",
|
||
barWidth: "30%",
|
||
},
|
||
],
|
||
});
|
||
},
|
||
// 初始化访问量柱状图
|
||
initCardCountChart(time = [], data = []) {
|
||
this.cardCountChart = echarts.init(this.$refs.cardCountChart);
|
||
this.cardCountChart.setOption({
|
||
tooltip: {
|
||
trigger: "axis",
|
||
},
|
||
grid: {
|
||
x: 0,
|
||
y: 0,
|
||
x2: 0,
|
||
y2: 0,
|
||
},
|
||
xAxis: [
|
||
{
|
||
type: "category",
|
||
data: time,
|
||
show: false, // 不显示坐标轴线、坐标轴刻度线和坐标轴上的文字
|
||
axisTick: {
|
||
show: false, // 不显示坐标轴刻度线
|
||
},
|
||
axisLine: {
|
||
show: false, // 不显示坐标轴线
|
||
},
|
||
axisLabel: {
|
||
show: false, // 不显示坐标轴上的文字
|
||
},
|
||
splitLine: {
|
||
show: false, // 不显示网格线
|
||
},
|
||
},
|
||
],
|
||
color: "#409eff",
|
||
yAxis: [
|
||
{
|
||
type: "value",
|
||
show: false, // 不显示坐标轴线、坐标轴刻度线和坐标轴上的文字
|
||
axisTick: {
|
||
show: false, // 不显示坐标轴刻度线
|
||
},
|
||
axisLine: {
|
||
show: false, // 不显示坐标轴线
|
||
},
|
||
axisLabel: {
|
||
show: false, // 不显示坐标轴上的文字
|
||
},
|
||
splitLine: {
|
||
show: false, // 不显示网格线
|
||
},
|
||
},
|
||
],
|
||
series: [
|
||
{
|
||
data: data,
|
||
type: "bar",
|
||
barWidth: "30%",
|
||
},
|
||
],
|
||
});
|
||
},
|
||
// 初始化用户数折线图
|
||
initCardUserChart(time = [], data = []) {
|
||
this.cardUserChart = echarts.init(this.$refs.cardUserChart);
|
||
this.cardUserChart.setOption({
|
||
tooltip: {
|
||
trigger: "axis",
|
||
},
|
||
grid: {
|
||
x: 0,
|
||
y: 10,
|
||
x2: 0,
|
||
y2: 2,
|
||
},
|
||
xAxis: [
|
||
{
|
||
type: "category",
|
||
data: time,
|
||
show: false, // 不显示坐标轴线、坐标轴刻度线和坐标轴上的文字
|
||
axisTick: {
|
||
show: false, // 不显示坐标轴刻度线
|
||
},
|
||
axisLine: {
|
||
show: false, // 不显示坐标轴线
|
||
},
|
||
axisLabel: {
|
||
show: false, // 不显示坐标轴上的文字
|
||
},
|
||
splitLine: {
|
||
show: false, // 不显示网格线
|
||
},
|
||
},
|
||
],
|
||
color: "#409eff",
|
||
yAxis: [
|
||
{
|
||
type: "value",
|
||
show: false, // 不显示坐标轴线、坐标轴刻度线和坐标轴上的文字
|
||
axisTick: {
|
||
show: false, // 不显示坐标轴刻度线
|
||
},
|
||
axisLine: {
|
||
show: false, // 不显示坐标轴线
|
||
},
|
||
axisLabel: {
|
||
show: false, // 不显示坐标轴上的文字
|
||
},
|
||
splitLine: {
|
||
show: false, // 不显示网格线
|
||
},
|
||
},
|
||
],
|
||
series: [
|
||
{
|
||
data: data,
|
||
type: "line",
|
||
symbol: "none",
|
||
},
|
||
],
|
||
});
|
||
},
|
||
// 初始化销售额图标
|
||
initSaleChart(time, data) {
|
||
this.saleChart = null;
|
||
this.saleChart = echarts.init(this.$refs.saleChart);
|
||
this.saleChart.setOption({
|
||
title: {
|
||
text: "销售趋势",
|
||
x: "center",
|
||
},
|
||
tooltip: {
|
||
trigger: "axis",
|
||
},
|
||
xAxis: [
|
||
{
|
||
type: "category",
|
||
data: time,
|
||
axisTick: {
|
||
alignWithLabel: true,
|
||
},
|
||
axisLine: {
|
||
lineStyle: {
|
||
color: "#999",
|
||
},
|
||
},
|
||
axisLabel: {
|
||
rotate: time.length <= 7 ? 0 : 45,
|
||
interval: 0,
|
||
fontSize: "9",
|
||
},
|
||
},
|
||
],
|
||
color: ["#409eff", "#FFC83A", "#F98B26"],
|
||
yAxis: [
|
||
{
|
||
type: "value",
|
||
axisLine: {
|
||
lineStyle: {
|
||
color: "#999",
|
||
},
|
||
},
|
||
splitLine: {
|
||
lineStyle: {
|
||
type: "dashed",
|
||
color: "#ececec",
|
||
},
|
||
},
|
||
},
|
||
],
|
||
series: [
|
||
{
|
||
name: "订单金额",
|
||
type: "bar",
|
||
barWidth: time.length <= 7 ? "50%" : "30%",
|
||
data: data.map((item) => item.orderAmount),
|
||
},
|
||
// {
|
||
// name: '实收金额',
|
||
// type: "bar",
|
||
// barWidth: time.length <= 7 ? "30%" : "20%",
|
||
// data: data.map(item => item.actualAmount),
|
||
// },
|
||
// {
|
||
// name: '优惠金额',
|
||
// type: "bar",
|
||
// barWidth: time.length <= 7 ? "30%" : "20%",
|
||
// data: data.map(item => item.discountAmount),
|
||
// }
|
||
],
|
||
});
|
||
},
|
||
// 初始化销售额图表
|
||
initPayChart(data) {
|
||
this.payChart = echarts.init(this.$refs.payChart);
|
||
this.payChart.setOption({
|
||
tooltip: {
|
||
trigger: "item",
|
||
formatter: (params) => {
|
||
return `${params.name}:${params.value}笔`;
|
||
},
|
||
},
|
||
legend: {
|
||
top: "5%",
|
||
left: "center",
|
||
},
|
||
color: [
|
||
"#409eff",
|
||
"#91cc75",
|
||
"#fac858",
|
||
"#ee6666",
|
||
"#73c0de",
|
||
"#3ba272",
|
||
"#fc8452",
|
||
"#9a60b4",
|
||
"#ea7ccc",
|
||
],
|
||
series: [
|
||
{
|
||
type: "pie",
|
||
radius: ["40%", "70%"],
|
||
avoidLabelOverlap: false,
|
||
label: {
|
||
show: false,
|
||
position: "center",
|
||
},
|
||
emphasis: {
|
||
label: {
|
||
show: true,
|
||
fontSize: 20,
|
||
},
|
||
},
|
||
labelLine: {
|
||
show: false,
|
||
},
|
||
data: data,
|
||
},
|
||
],
|
||
});
|
||
},
|
||
// 获取销售额柱状图数据
|
||
async dateAmount() {
|
||
try {
|
||
this.saleLoading = true;
|
||
const res = await dataSummaryApi.dateAmount({ day: this.saleActive });
|
||
const data = res.total.map((item) => {
|
||
return {
|
||
orderAmount: item.orderAmount,
|
||
// actualAmount: item.actualAmount,
|
||
// discountAmount: item.discountAmount
|
||
};
|
||
});
|
||
const time = res.total.map((item) => item.tradeDay);
|
||
this.initSaleChart(time, data);
|
||
setTimeout(() => {
|
||
this.saleLoading = false;
|
||
}, 300);
|
||
} catch (error) {
|
||
console.log(error);
|
||
}
|
||
},
|
||
paginationChange(e) {
|
||
this.saleTablePage = e;
|
||
this.dateProduct();
|
||
},
|
||
// 获取销售额排行表格数据
|
||
async dateProduct() {
|
||
try {
|
||
this.saleTableLoading = true;
|
||
const res = await dataSummaryApi.productSaleDate({
|
||
day: this.saleTableActive,
|
||
page: this.saleTablePage,
|
||
size: this.saleTableSize,
|
||
});
|
||
this.saleTable = res.records;
|
||
this.saleTableTotal = res.totalRow * 1;
|
||
this.summaryDateGet(res.countList);
|
||
setTimeout(() => {
|
||
this.saleTableLoading = false;
|
||
}, 300);
|
||
} catch (error) {
|
||
console.log(error);
|
||
}
|
||
},
|
||
// 支付类型占比 饼图
|
||
async datePayType() {
|
||
try {
|
||
this.payChartLoading = true;
|
||
const res = await dataSummaryApi.datePayType({ day: this.saleActive });
|
||
const data = res.countPayType.map((item) => {
|
||
return {
|
||
value: item.count,
|
||
name: item.payType,
|
||
};
|
||
});
|
||
setTimeout(() => {
|
||
this.payChartLoading = false;
|
||
}, 300);
|
||
this.initPayChart(data);
|
||
} catch (error) {
|
||
console.log(error);
|
||
}
|
||
},
|
||
// 汇总数据
|
||
async summaryGet() {
|
||
try {
|
||
const res1 = await summaryGet();
|
||
const res2 = await summaryTodayGet();
|
||
this.topData = {
|
||
...res1,
|
||
...res2,
|
||
};
|
||
let payTime = res1.countDateList.map((item) => item.tradeDay);
|
||
let payData = res1.countDateList.map((item) => item.count);
|
||
|
||
let countTime = res1.visitsCountList.map((item) => item.tradeDay);
|
||
let countData = res1.visitsCountList.map((item) => item.count);
|
||
|
||
this.initCardPayChart(payTime, payData);
|
||
this.initCardCountChart(countTime, countData);
|
||
console.log(this.topData);
|
||
} catch (error) {
|
||
console.log(error);
|
||
}
|
||
},
|
||
rankChange() {
|
||
this.dateProduct();
|
||
// this.summaryDateGet();
|
||
},
|
||
// 初始化销售图标
|
||
initProduceChart(p1, p2) {
|
||
this.productCountChart = echarts.init(this.$refs.productCountChart);
|
||
// this.productSumChart = echarts.init(this.$refs.productSumChart);
|
||
|
||
this.productCountChart.setOption({
|
||
tooltip: {
|
||
trigger: "axis",
|
||
},
|
||
grid: {
|
||
x: 0,
|
||
y: 0,
|
||
x2: 0,
|
||
y2: 0,
|
||
},
|
||
xAxis: [
|
||
{
|
||
boundaryGap: false,
|
||
type: "category",
|
||
data: p1[0],
|
||
show: false, // 不显示坐标轴线、坐标轴刻度线和坐标轴上的文字
|
||
axisTick: {
|
||
show: false, // 不显示坐标轴刻度线
|
||
},
|
||
axisLine: {
|
||
show: false, // 不显示坐标轴线
|
||
},
|
||
axisLabel: {
|
||
show: false, // 不显示坐标轴上的文字
|
||
},
|
||
splitLine: {
|
||
show: false, // 不显示网格线
|
||
},
|
||
},
|
||
],
|
||
color: "#409eff",
|
||
yAxis: [
|
||
{
|
||
type: "value",
|
||
show: false, // 不显示坐标轴线、坐标轴刻度线和坐标轴上的文字
|
||
axisTick: {
|
||
show: false, // 不显示坐标轴刻度线
|
||
},
|
||
axisLine: {
|
||
show: false, // 不显示坐标轴线
|
||
},
|
||
axisLabel: {
|
||
show: false, // 不显示坐标轴上的文字
|
||
},
|
||
splitLine: {
|
||
show: false, // 不显示网格线
|
||
},
|
||
},
|
||
],
|
||
series: [
|
||
{
|
||
data: p1[1],
|
||
type: "line",
|
||
symbol: "none",
|
||
smooth: true,
|
||
areaStyle: {
|
||
color: new echarts.graphic.LinearGradient(0, 0, 0, 1, [
|
||
{
|
||
offset: 0,
|
||
color: "#409eff", // 渐变颜色
|
||
},
|
||
{
|
||
offset: 1,
|
||
color: "#409eff", // 渐变颜色
|
||
},
|
||
]),
|
||
},
|
||
},
|
||
],
|
||
});
|
||
|
||
// this.productSumChart.setOption({
|
||
// tooltip: {
|
||
// trigger: "axis"
|
||
// },
|
||
// grid: {
|
||
// x: 0,
|
||
// y: 0,
|
||
// x2: 0,
|
||
// y2: 0
|
||
// },
|
||
// xAxis: [
|
||
// {
|
||
// boundaryGap: false,
|
||
// type: "category",
|
||
// data: p2[0],
|
||
// show: false, // 不显示坐标轴线、坐标轴刻度线和坐标轴上的文字
|
||
// axisTick: {
|
||
// show: false // 不显示坐标轴刻度线
|
||
// },
|
||
// axisLine: {
|
||
// show: false // 不显示坐标轴线
|
||
// },
|
||
// axisLabel: {
|
||
// show: false // 不显示坐标轴上的文字
|
||
// },
|
||
// splitLine: {
|
||
// show: false // 不显示网格线
|
||
// }
|
||
// }
|
||
// ],
|
||
// color: "#409eff",
|
||
// yAxis: [
|
||
// {
|
||
// type: "value",
|
||
// show: false, // 不显示坐标轴线、坐标轴刻度线和坐标轴上的文字
|
||
// axisTick: {
|
||
// show: false // 不显示坐标轴刻度线
|
||
// },
|
||
// axisLine: {
|
||
// show: false // 不显示坐标轴线
|
||
// },
|
||
// axisLabel: {
|
||
// show: false // 不显示坐标轴上的文字
|
||
// },
|
||
// splitLine: {
|
||
// show: false // 不显示网格线
|
||
// }
|
||
// }
|
||
// ],
|
||
// series: [
|
||
// {
|
||
// data: p2[1],
|
||
// type: "line",
|
||
// symbol: "none",
|
||
// smooth: true,
|
||
// areaStyle: {
|
||
// color: new echarts.graphic.LinearGradient(0, 0, 0, 1, [
|
||
// {
|
||
// offset: 0,
|
||
// color: "#409eff" // 渐变颜色
|
||
// },
|
||
// {
|
||
// offset: 1,
|
||
// color: "#409eff" // 渐变颜色
|
||
// }
|
||
// ])
|
||
// }
|
||
// }
|
||
// ]
|
||
// });
|
||
},
|
||
// 日期汇总数据
|
||
async summaryDateGet(res) {
|
||
try {
|
||
console.log(res);
|
||
// this.initProduceChart(p1, p2);
|
||
} catch (error) {
|
||
console.log(error);
|
||
}
|
||
},
|
||
},
|
||
};
|
||
</script>
|
||
|
||
<style>
|
||
.popper {
|
||
border: 1px solid #d9d9d9 !important;
|
||
}
|
||
</style>
|
||
<style scoped lang="scss">
|
||
.tips_row {
|
||
width: 300px;
|
||
padding: 0 10px;
|
||
|
||
.item {
|
||
height: 40px;
|
||
display: flex;
|
||
align-items: center;
|
||
color: #666;
|
||
justify-content: space-between;
|
||
|
||
&:not(:last-child) {
|
||
border-bottom: 1px solid #d9d9d9;
|
||
}
|
||
|
||
.left {
|
||
display: flex;
|
||
align-items: center;
|
||
|
||
.icon {
|
||
font-size: 20px;
|
||
margin-right: 10px;
|
||
}
|
||
}
|
||
|
||
.num {
|
||
color: #1890ff;
|
||
}
|
||
}
|
||
}
|
||
|
||
.app-container {
|
||
padding: 20px;
|
||
background-color: #f5f5f5;
|
||
}
|
||
|
||
.h_card_wrap {
|
||
.status_wrap {
|
||
display: flex;
|
||
align-items: center;
|
||
justify-content: space-between;
|
||
|
||
.left {
|
||
display: flex;
|
||
align-items: center;
|
||
|
||
.dot {
|
||
$size: 6px;
|
||
width: $size;
|
||
height: $size;
|
||
border-radius: 50%;
|
||
background-color: #1890ff;
|
||
margin-right: 10px;
|
||
}
|
||
}
|
||
|
||
.time {
|
||
display: flex;
|
||
align-items: center;
|
||
}
|
||
}
|
||
|
||
.content {
|
||
background-color: #fff;
|
||
margin-top: 15px;
|
||
padding: 20px;
|
||
|
||
.top {
|
||
display: flex;
|
||
|
||
.item {
|
||
flex: 1;
|
||
|
||
&.earnings {
|
||
display: flex;
|
||
}
|
||
|
||
&.data {
|
||
margin-left: 40px;
|
||
background: url("../../assets/images/data_home_bg1.png") no-repeat center center / cover;
|
||
display: flex;
|
||
align-items: center;
|
||
}
|
||
|
||
.data_item {
|
||
flex: 1;
|
||
display: flex;
|
||
align-items: center;
|
||
justify-content: center;
|
||
}
|
||
|
||
.data_item_right {
|
||
flex: 1;
|
||
display: flex;
|
||
flex-direction: column;
|
||
|
||
.t {
|
||
width: 200px;
|
||
color: #333;
|
||
display: flex;
|
||
|
||
&:not(:first-child) {
|
||
margin-top: 20px;
|
||
}
|
||
}
|
||
|
||
.n {
|
||
color: #1890ff;
|
||
margin-left: 20px;
|
||
}
|
||
}
|
||
|
||
.num_wrap {
|
||
display: flex;
|
||
flex-direction: column;
|
||
align-items: center;
|
||
justify-content: center;
|
||
|
||
.num {
|
||
font-size: 30px;
|
||
color: #1890ff;
|
||
}
|
||
|
||
.tips {
|
||
font-size: 12px;
|
||
margin-top: 10px;
|
||
}
|
||
}
|
||
|
||
.line_wrap {
|
||
flex: 1;
|
||
padding-left: 20px;
|
||
|
||
.line_item {
|
||
&:not(:first-child) {
|
||
margin-top: 20px;
|
||
}
|
||
|
||
.line_item_top {
|
||
display: flex;
|
||
justify-content: space-between;
|
||
font-size: 12px;
|
||
|
||
.t {
|
||
color: #999;
|
||
}
|
||
}
|
||
|
||
.line_gropress {
|
||
display: flex;
|
||
margin-top: 6px;
|
||
background-color: #f6f6f6;
|
||
|
||
.gropress {
|
||
height: 5px;
|
||
|
||
&.l {
|
||
background-color: #1890ff;
|
||
}
|
||
|
||
&.r {
|
||
background-color: #ff3f3f;
|
||
}
|
||
}
|
||
}
|
||
|
||
.line_btm {
|
||
display: flex;
|
||
justify-content: space-between;
|
||
margin-top: 5px;
|
||
|
||
.icon {
|
||
color: #a2a2a2;
|
||
position: relative;
|
||
left: -5px;
|
||
font-size: 12px;
|
||
}
|
||
|
||
.info {
|
||
font-size: 12px;
|
||
|
||
.l_t {
|
||
color: #1890ff;
|
||
}
|
||
|
||
.l_r {
|
||
color: #ff3f3f;
|
||
margin-left: 10px;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
.btm {
|
||
display: flex;
|
||
justify-content: space-between;
|
||
margin-top: 20px;
|
||
|
||
.item {
|
||
width: 200px;
|
||
height: 83px;
|
||
background-repeat: no-repeat;
|
||
background-position: center center;
|
||
background-size: 100% 100%;
|
||
display: flex;
|
||
flex-direction: column;
|
||
justify-content: center;
|
||
padding-left: 18px;
|
||
|
||
&.item1 {
|
||
background-image: url("../../assets/images/data_home_item1.png");
|
||
}
|
||
|
||
&.item2 {
|
||
background-image: url("../../assets/images/data_home_item2.png");
|
||
}
|
||
|
||
&.item3 {
|
||
background-image: url("../../assets/images/data_home_item3.png");
|
||
}
|
||
|
||
&.item4 {
|
||
background-image: url("../../assets/images/data_home_item4.png");
|
||
}
|
||
|
||
.title {
|
||
display: flex;
|
||
align-items: center;
|
||
color: #333;
|
||
font-size: 14px;
|
||
|
||
.icon {
|
||
font-size: 12px;
|
||
color: #999;
|
||
margin-left: 6px;
|
||
}
|
||
}
|
||
|
||
.icon_wrap {
|
||
display: flex;
|
||
align-items: center;
|
||
margin-top: 10px;
|
||
|
||
.img {
|
||
width: 36px;
|
||
height: 36px;
|
||
}
|
||
|
||
.t {
|
||
font-size: 24px;
|
||
margin-left: 10px;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
.card_wrap {
|
||
display: flex;
|
||
flex-wrap: wrap;
|
||
gap: 20px;
|
||
|
||
.card {
|
||
flex: 1;
|
||
background-color: #fff;
|
||
border-radius: 2px;
|
||
padding: 0 20px;
|
||
|
||
.header {
|
||
display: flex;
|
||
justify-content: space-between;
|
||
color: #999;
|
||
padding-top: 20px;
|
||
|
||
.card_title {
|
||
font-size: 14px;
|
||
flex: 1;
|
||
}
|
||
}
|
||
|
||
.number {
|
||
padding: 20px 0 10px 0;
|
||
font-size: 24px;
|
||
height: 60px;
|
||
}
|
||
|
||
.row {
|
||
height: 50px;
|
||
color: #555;
|
||
font-size: 14px;
|
||
display: flex;
|
||
align-items: center;
|
||
|
||
&:not(:last-child) {
|
||
border-bottom: 1px solid #ececec;
|
||
}
|
||
|
||
.icon {
|
||
color: rgb(255, 85, 85);
|
||
margin-left: 4px;
|
||
}
|
||
|
||
.dot {
|
||
$size: 6px;
|
||
width: $size;
|
||
height: $size;
|
||
border-radius: 50%;
|
||
background-color: #1890ff;
|
||
margin-right: 6px;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
.chart_wrap {
|
||
margin-top: 20px;
|
||
|
||
.sale_data {
|
||
display: flex;
|
||
|
||
.card {
|
||
flex: 1;
|
||
background-color: #fff;
|
||
border-radius: 2px;
|
||
padding: 0 20px;
|
||
|
||
.sale_data_header {
|
||
display: flex;
|
||
justify-content: space-between;
|
||
color: #999;
|
||
padding-top: 20px;
|
||
|
||
.card_title {
|
||
font-size: 14px;
|
||
flex: 1;
|
||
}
|
||
}
|
||
|
||
.number {
|
||
padding-top: 10px;
|
||
font-size: 24px;
|
||
height: 60px;
|
||
}
|
||
|
||
.product_chart_wrap {
|
||
height: 50px;
|
||
}
|
||
}
|
||
}
|
||
|
||
.item {
|
||
flex: 1;
|
||
background-color: #fff;
|
||
border-radius: 2px;
|
||
|
||
.header {
|
||
display: flex;
|
||
align-items: center;
|
||
justify-content: space-between;
|
||
border-bottom: 1px solid #ececec;
|
||
padding: 0 20px;
|
||
|
||
.tab_wrap {
|
||
display: flex;
|
||
$color: #1890ff;
|
||
|
||
.item {
|
||
padding: 0 10px;
|
||
height: 60px;
|
||
display: flex;
|
||
align-items: center;
|
||
justify-content: center;
|
||
font-size: 14px;
|
||
margin-right: 20px;
|
||
|
||
&.active {
|
||
position: relative;
|
||
color: $color;
|
||
|
||
&::after {
|
||
content: "";
|
||
width: 100%;
|
||
border-bottom: 2px solid $color;
|
||
position: absolute;
|
||
bottom: 0;
|
||
left: 0;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
.chart {
|
||
padding: 20px 0;
|
||
height: 300px;
|
||
}
|
||
|
||
.table {
|
||
padding: 20px;
|
||
}
|
||
}
|
||
}
|
||
</style>
|