|
@@ -0,0 +1,604 @@
|
|
|
+<template>
|
|
|
+ <div class="search-form">
|
|
|
+ <!-- 站点选择和时间筛选 -->
|
|
|
+ <a-row class="r1" :gutter="8">
|
|
|
+ <a-col :xl="7" :xxl="6">
|
|
|
+ <div class="choose-site">
|
|
|
+ <span class="t1">站点:</span>
|
|
|
+ <select-site @set-site-info="changeSite" select-width="100%" />
|
|
|
+ </div>
|
|
|
+ </a-col>
|
|
|
+ <a-col :xl="8" :xxl="6">
|
|
|
+ <div class="choose-site">
|
|
|
+ <span class="t1">统计时间:</span>
|
|
|
+ <a-range-picker @change="onChangeDatePicker" :disabledDate="disabledDate" :value="rangeDate" style="width: 70%" />
|
|
|
+ </div>
|
|
|
+ </a-col>
|
|
|
+ <a-col :xl="9" :xxl="12">
|
|
|
+ <a-button-group class="time-btn-group">
|
|
|
+ <a-button :class="queryParam.dateType == '' ? 'active' : ''" @click="setTime('')">全部时间</a-button>
|
|
|
+ <a-button :class="queryParam.dateType == 'thirtyDay' ? 'active' : ''" @click="setTime('thirtyDay')">近30天</a-button>
|
|
|
+ <a-button :class="queryParam.dateType == 'sevenDay' ? 'active' : ''" @click="setTime('sevenDay')">近一周</a-button>
|
|
|
+ <a-button :class="queryParam.dateType == 'yesterday' ? 'active' : ''" @click="setTime('yesterday')">昨日</a-button>
|
|
|
+ <a-button :class="queryParam.dateType == 'today' ? 'active' : ''" @click="setTime('today')">今日</a-button>
|
|
|
+ </a-button-group>
|
|
|
+ </a-col>
|
|
|
+ </a-row>
|
|
|
+ </div>
|
|
|
+ <a-spin :spinning="loading" tip="加载中...">
|
|
|
+ <a-row class="r2">
|
|
|
+ <a-col :span="6">
|
|
|
+ <p class="t1">账户名称</p>
|
|
|
+ <p class="t3">{{ 111 }}</p>
|
|
|
+ </a-col>
|
|
|
+ <a-col :span="6">
|
|
|
+ <p class="t1">账户余额</p>
|
|
|
+ <p class="t3">{{ 222 }}</p>
|
|
|
+ </a-col>
|
|
|
+ <a-col :span="6">
|
|
|
+ <p class="t1">昨日花费</p>
|
|
|
+ <p class="t3">{{ 333 }}</p>
|
|
|
+ </a-col>
|
|
|
+ <a-col :span="6">
|
|
|
+ <p class="t1">账户优化分数</p>
|
|
|
+ <p class="t3">{{ 444 }}</p>
|
|
|
+ </a-col>
|
|
|
+ </a-row>
|
|
|
+ <a-row>
|
|
|
+ <a-col :span="24">
|
|
|
+ <a-card style="margin: 10px" title="核心数据">
|
|
|
+ <a-row class="r5" :gutter="8">
|
|
|
+ <a-row class="r5-1">
|
|
|
+ <a-col :span="24">
|
|
|
+ <div class="fl">
|
|
|
+ <a-button-group>
|
|
|
+ <a-button :class="{ active: activeChart === 'impression' }" @click="switchChart('impression')">展示</a-button>
|
|
|
+ <a-button :class="{ active: activeChart === 'clicks' }" @click="switchChart('clicks')">点击</a-button>
|
|
|
+ <a-button :class="{ active: activeChart === 'ctr' }" @click="switchChart('ctr')">点击率</a-button>
|
|
|
+ <a-button :class="{ active: activeChart === 'conversion' }" @click="switchChart('conversion')">转化数</a-button>
|
|
|
+ <a-button :class="{ active: activeChart === 'cost' }" @click="switchChart('cost')">花费</a-button>
|
|
|
+ </a-button-group>
|
|
|
+ </div>
|
|
|
+ <line-chart :chartType="activeChart" />
|
|
|
+ </a-col>
|
|
|
+ </a-row>
|
|
|
+ </a-row>
|
|
|
+ </a-card>
|
|
|
+ </a-col>
|
|
|
+ </a-row>
|
|
|
+ <a-row>
|
|
|
+ <a-col :span="24">
|
|
|
+ <a-card style="margin: 10px" title="广告系列">
|
|
|
+ <a-table
|
|
|
+ :columns="columns"
|
|
|
+ :data-source="tableData"
|
|
|
+ :loading="loading"
|
|
|
+ :pagination="false"
|
|
|
+ style="width: 100%"
|
|
|
+ />
|
|
|
+ </a-card>
|
|
|
+ </a-col>
|
|
|
+ </a-row>
|
|
|
+ <a-row :gutter="8">
|
|
|
+ <a-col :span="12">
|
|
|
+ <a-card style="margin: 10px" title="TOP关键词">
|
|
|
+ <a-table
|
|
|
+ :columns="keywordColumns"
|
|
|
+ :data-source="keywordData"
|
|
|
+ :loading="loading"
|
|
|
+ :pagination="false"
|
|
|
+ style="width: 100%"
|
|
|
+ />
|
|
|
+ </a-card>
|
|
|
+ </a-col>
|
|
|
+ <a-col :span="12">
|
|
|
+ <a-card style="margin: 10px" title="TOP展示位">
|
|
|
+ <a-table
|
|
|
+ :columns="positionColumns"
|
|
|
+ :data-source="positionData"
|
|
|
+ :loading="loading"
|
|
|
+ :pagination="false"
|
|
|
+ style="width: 100%"
|
|
|
+ />
|
|
|
+ </a-card>
|
|
|
+ </a-col>
|
|
|
+ </a-row>
|
|
|
+ <a-row>
|
|
|
+ <a-col :span="24">
|
|
|
+ <a-card style="margin: 10px" title="TOP国家/地区">
|
|
|
+ <a-row class="r5">
|
|
|
+ <a-col :span="18">
|
|
|
+ <map-adweb v-if="countryMapData.length > 0" :dataSource="countryMapData"
|
|
|
+ height="400"></map-adweb>
|
|
|
+ <a-empty v-else style="margin-top: 50px;">
|
|
|
+ </a-empty>
|
|
|
+ </a-col>
|
|
|
+
|
|
|
+ <a-col :span="6">
|
|
|
+ <a-table
|
|
|
+ :rowKey="(record,index)=>{return index}"
|
|
|
+ class="chartTable"
|
|
|
+ :scroll="{ y: 500 }"
|
|
|
+ :pagination=false
|
|
|
+ :columns="chartDetailDataCol"
|
|
|
+ :data-source="chartDetailData"
|
|
|
+ :showHeader="false">
|
|
|
+ <template #bodyCell="{ column, record }">
|
|
|
+ <template v-if="column.key === 'flagSlot' ">
|
|
|
+ <span class="img-box">
|
|
|
+ <span :class="'flag-icon flag-icon-'+record.countryCode"></span>
|
|
|
+ </span>
|
|
|
+ </template>
|
|
|
+ <template v-if="column.key === 'numSlot' ">
|
|
|
+ {{ record.enquires }} | {{ record.enquiresProportion }}
|
|
|
+ </template>
|
|
|
+ </template>
|
|
|
+ </a-table>
|
|
|
+ </a-col>
|
|
|
+ </a-row>
|
|
|
+ </a-card>
|
|
|
+ </a-col>
|
|
|
+ </a-row>
|
|
|
+</a-spin>
|
|
|
+
|
|
|
+</template>
|
|
|
+
|
|
|
+<script setup lang="ts" name="marketing-googleads">
|
|
|
+import { getAction } from '@/api/manage/manage';
|
|
|
+import { ref, reactive } from 'vue';
|
|
|
+import dayjs from 'dayjs';
|
|
|
+import selectSite from "@/components/Adweb/selectSite.vue";
|
|
|
+import LineChart from './charts/Line.vue';
|
|
|
+import MapAdweb from '@/components/chart/mapAdweb.vue';
|
|
|
+
|
|
|
+const rangeDate = ref([]);
|
|
|
+const queryParam = reactive<any>({});
|
|
|
+queryParam.limit = 10;
|
|
|
+queryParam.siteCode = localStorage.getItem("siteCode");
|
|
|
+const loading = ref(false);
|
|
|
+const activeChart = ref('impression');
|
|
|
+const chartDetailData = ref([]);
|
|
|
+const countryMapData = ref([]);
|
|
|
+
|
|
|
+//访客数地域分布
|
|
|
+const getCountryMapData = async () => {
|
|
|
+ try {
|
|
|
+ const res = await getAction("/dmp-data/country/stats", queryParam);
|
|
|
+ if (res.code === 200) {
|
|
|
+ chartDetailData.value = res.result;
|
|
|
+ countryMapData.value = chartDetailData.value.map(entry => ({
|
|
|
+ name: entry.countryName,
|
|
|
+ value: entry.totalUsers
|
|
|
+ }));
|
|
|
+ console.log("countryMapData", countryMapData.value);
|
|
|
+ }
|
|
|
+ } catch (error) {
|
|
|
+ console.error(error);
|
|
|
+ }
|
|
|
+};
|
|
|
+
|
|
|
+const changeSite = (selectedSiteInfo: any) => {
|
|
|
+ queryParam.siteCode = selectedSiteInfo.code;
|
|
|
+ localStorage.setItem("siteCode", queryParam.siteCode);
|
|
|
+ reloadData();
|
|
|
+}
|
|
|
+
|
|
|
+const onChangeDatePicker = (date, dateString) => {
|
|
|
+ if (dateString.length > 0) {
|
|
|
+ console.log("rangeDate:", rangeDate.value);
|
|
|
+ rangeDate.value = date;
|
|
|
+ console.log("date:", date);
|
|
|
+ queryParam.start = dateString[0];
|
|
|
+ queryParam.end = dateString[1];
|
|
|
+ queryParam.dateType = undefined;
|
|
|
+ reloadData();
|
|
|
+ }
|
|
|
+};
|
|
|
+
|
|
|
+const setTime = (time) => {
|
|
|
+ queryParam.dateType = time;
|
|
|
+ queryParam.start = "";
|
|
|
+ queryParam.end = "";
|
|
|
+
|
|
|
+ if (time == "") {
|
|
|
+ rangeDate.value = undefined;
|
|
|
+ } else if (time == "sevenDay") {
|
|
|
+ rangeDate.value = [dayjs().add(-7, 'd'), dayjs().add(-1, 'd')];
|
|
|
+ } else if (time == "thirtyDay") {
|
|
|
+ rangeDate.value = [dayjs().add(-30, 'd'), dayjs().add(-1, 'd')];
|
|
|
+ } else if (time == "yesterday") {
|
|
|
+ rangeDate.value = [dayjs().add(-1, 'd'), dayjs().add(-1, 'd')];
|
|
|
+ } else if (time == "today") {
|
|
|
+ rangeDate.value = [dayjs(), dayjs()];
|
|
|
+ }
|
|
|
+
|
|
|
+ reloadData();
|
|
|
+};
|
|
|
+
|
|
|
+//日期选择只能今天之前
|
|
|
+const disabledDate = (current) => {
|
|
|
+ return current && current > dayjs();
|
|
|
+}
|
|
|
+
|
|
|
+//重新刷新页面数据
|
|
|
+const reloadData = () => {
|
|
|
+// loading.value = true;
|
|
|
+ getCountryMapData();
|
|
|
+}
|
|
|
+
|
|
|
+const switchChart = (type: string) => {
|
|
|
+ activeChart.value = type;
|
|
|
+};
|
|
|
+
|
|
|
+const columns = ref([
|
|
|
+ {
|
|
|
+ title: '广告系列名称',
|
|
|
+ dataIndex: 'name',
|
|
|
+ key: 'name',
|
|
|
+ },
|
|
|
+ {
|
|
|
+ title: '状态',
|
|
|
+ dataIndex: 'status',
|
|
|
+ key: 'status',
|
|
|
+ },
|
|
|
+ {
|
|
|
+ title: '展示次数',
|
|
|
+ dataIndex: 'impressions',
|
|
|
+ key: 'impressions',
|
|
|
+ },
|
|
|
+ {
|
|
|
+ title: '点击次数',
|
|
|
+ dataIndex: 'clicks',
|
|
|
+ key: 'clicks',
|
|
|
+ },
|
|
|
+ {
|
|
|
+ title: '点击率',
|
|
|
+ dataIndex: 'ctr',
|
|
|
+ key: 'ctr',
|
|
|
+ },
|
|
|
+ {
|
|
|
+ title: '花费',
|
|
|
+ dataIndex: 'cost',
|
|
|
+ key: 'cost',
|
|
|
+ },
|
|
|
+]);
|
|
|
+
|
|
|
+const tableData = ref([]);
|
|
|
+
|
|
|
+const keywordColumns = ref([
|
|
|
+ {
|
|
|
+ title: '关键词',
|
|
|
+ dataIndex: 'keyword',
|
|
|
+ key: 'keyword',
|
|
|
+ },
|
|
|
+ {
|
|
|
+ title: '展示次数',
|
|
|
+ dataIndex: 'impressions',
|
|
|
+ key: 'impressions',
|
|
|
+ },
|
|
|
+ {
|
|
|
+ title: '点击次数',
|
|
|
+ dataIndex: 'clicks',
|
|
|
+ key: 'clicks',
|
|
|
+ },
|
|
|
+ {
|
|
|
+ title: '点击率',
|
|
|
+ dataIndex: 'ctr',
|
|
|
+ key: 'ctr',
|
|
|
+ },
|
|
|
+ {
|
|
|
+ title: '花费',
|
|
|
+ dataIndex: 'cost',
|
|
|
+ key: 'cost',
|
|
|
+ },
|
|
|
+]);
|
|
|
+
|
|
|
+const positionColumns = ref([
|
|
|
+ {
|
|
|
+ title: '展示位置',
|
|
|
+ dataIndex: 'position',
|
|
|
+ key: 'position',
|
|
|
+ },
|
|
|
+ {
|
|
|
+ title: '展示次数',
|
|
|
+ dataIndex: 'impressions',
|
|
|
+ key: 'impressions',
|
|
|
+ },
|
|
|
+ {
|
|
|
+ title: '点击次数',
|
|
|
+ dataIndex: 'clicks',
|
|
|
+ key: 'clicks',
|
|
|
+ },
|
|
|
+ {
|
|
|
+ title: '点击率',
|
|
|
+ dataIndex: 'ctr',
|
|
|
+ key: 'ctr',
|
|
|
+ },
|
|
|
+ {
|
|
|
+ title: '花费',
|
|
|
+ dataIndex: 'cost',
|
|
|
+ key: 'cost',
|
|
|
+ },
|
|
|
+]);
|
|
|
+
|
|
|
+const keywordData = ref([]);
|
|
|
+const positionData = ref([]);
|
|
|
+</script>
|
|
|
+
|
|
|
+<style scoped lang="less">
|
|
|
+.r1 {
|
|
|
+ margin: 20px;
|
|
|
+
|
|
|
+ .choose-site {
|
|
|
+ display: flex;
|
|
|
+ }
|
|
|
+
|
|
|
+ .t1 {
|
|
|
+ font-size: 18px;
|
|
|
+ font-weight: 400;
|
|
|
+ letter-spacing: 0px;
|
|
|
+ line-height: 32px;
|
|
|
+ margin-left: 10px;
|
|
|
+ }
|
|
|
+
|
|
|
+ .ant-form-item {
|
|
|
+ flex: 1;
|
|
|
+ }
|
|
|
+
|
|
|
+ .ant-calendar-picker {
|
|
|
+ margin-right: 20px;
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ /deep/ .ant-btn {
|
|
|
+ background: transparent;
|
|
|
+ margin-right: 10px;
|
|
|
+ padding: 4px 15px;
|
|
|
+ border: 1px solid #d9d9d9;
|
|
|
+ border-radius: 4px;
|
|
|
+ transition: all 0.3s;
|
|
|
+
|
|
|
+ &:hover {
|
|
|
+ color: @primary-color;
|
|
|
+ border-color: @primary-color;
|
|
|
+ }
|
|
|
+
|
|
|
+ &.active {
|
|
|
+ color: @primary-color;
|
|
|
+ background: #e6f7ff;
|
|
|
+ border-color: @primary-color;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ .time-btn-group {
|
|
|
+ /deep/ .ant-btn {
|
|
|
+ background: #fff;
|
|
|
+ padding: 4px 15px;
|
|
|
+ border: 1px solid #d9d9d9;
|
|
|
+ transition: all 0.3s;
|
|
|
+ margin-right: 0;
|
|
|
+
|
|
|
+ &:first-child {
|
|
|
+ border-top-left-radius: 4px;
|
|
|
+ border-bottom-left-radius: 4px;
|
|
|
+ }
|
|
|
+
|
|
|
+ &:last-child {
|
|
|
+ border-top-right-radius: 4px;
|
|
|
+ border-bottom-right-radius: 4px;
|
|
|
+ }
|
|
|
+
|
|
|
+ &:not(:first-child) {
|
|
|
+ margin-left: -1px;
|
|
|
+ }
|
|
|
+
|
|
|
+ &:hover {
|
|
|
+ color: @primary-color;
|
|
|
+ border-color: @primary-color;
|
|
|
+ position: relative;
|
|
|
+ z-index: 1;
|
|
|
+ background: #fff;
|
|
|
+ }
|
|
|
+
|
|
|
+ &.active {
|
|
|
+ color: @primary-color;
|
|
|
+ background: #e6f7ff;
|
|
|
+ border-color: @primary-color;
|
|
|
+ position: relative;
|
|
|
+ z-index: 2;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+}
|
|
|
+.r2 {
|
|
|
+ background: #fff;
|
|
|
+ padding: 30px 20px;
|
|
|
+ margin: 10px;
|
|
|
+
|
|
|
+ .ant-col:not(:last-child) {
|
|
|
+ border-right: 1px solid #e6e6e6;
|
|
|
+ }
|
|
|
+
|
|
|
+ p {
|
|
|
+ margin: 0;
|
|
|
+ text-align: center;
|
|
|
+
|
|
|
+ &.t1 {
|
|
|
+ color: #333;
|
|
|
+ margin-bottom: 15px;
|
|
|
+
|
|
|
+ img {
|
|
|
+ margin-right: 10px;
|
|
|
+ width: 15px;
|
|
|
+ margin-top: -5px;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ &.t2 {
|
|
|
+ color: @primary-color;
|
|
|
+ font-size: 30px;
|
|
|
+ font-weight: 500;
|
|
|
+ line-height: 1;
|
|
|
+ padding-left: 25px;
|
|
|
+ }
|
|
|
+
|
|
|
+ &.t3 {
|
|
|
+ font-size: 32px;
|
|
|
+ font-weight: 700;
|
|
|
+ letter-spacing: 0px;
|
|
|
+ line-height: 38px;
|
|
|
+ color: rgba(13, 62, 122, 1);
|
|
|
+ }
|
|
|
+ }
|
|
|
+}
|
|
|
+.r5 {
|
|
|
+ background: #fff;
|
|
|
+ padding: 10px;
|
|
|
+ border-radius: 10px;
|
|
|
+ margin: 0 !important;
|
|
|
+
|
|
|
+ .wrap {
|
|
|
+ box-shadow: 0px 2px 4px 0px @primary-color;
|
|
|
+ padding: 15px;
|
|
|
+ border-radius: 10px;
|
|
|
+ overflow: hidden;
|
|
|
+ background: #fff;
|
|
|
+ transition: all .3s;
|
|
|
+
|
|
|
+ &.blue {
|
|
|
+ box-shadow: 0px 2px 4px 0px @primary-color;
|
|
|
+ }
|
|
|
+
|
|
|
+ &.effect:hover {
|
|
|
+ box-shadow: none;
|
|
|
+ background: rgb(241, 248, 255);
|
|
|
+ }
|
|
|
+
|
|
|
+ img {
|
|
|
+ width: 15px;
|
|
|
+ }
|
|
|
+
|
|
|
+ .fr {
|
|
|
+ float: right;
|
|
|
+ width: calc(100% - 15px);
|
|
|
+ text-align: center;
|
|
|
+
|
|
|
+ p:last-child {
|
|
|
+ font-size: 30px;
|
|
|
+ text-align: center;
|
|
|
+ margin-top: 10px;
|
|
|
+
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ /deep/ .ant-table-thead > tr > th {
|
|
|
+ background: rgb(241, 248, 255);
|
|
|
+ border: none;
|
|
|
+ color: #000;
|
|
|
+ padding: 10px;
|
|
|
+ }
|
|
|
+
|
|
|
+ /deep/ .ant-table-tbody .ant-table-row td {
|
|
|
+ padding: 10px;
|
|
|
+ color: #000;
|
|
|
+ }
|
|
|
+
|
|
|
+ .r5-1 {
|
|
|
+ display: inline-block;
|
|
|
+ width: 100%;
|
|
|
+ margin-top: 30px;
|
|
|
+
|
|
|
+ .fl {
|
|
|
+ float: left;
|
|
|
+ position: relative;
|
|
|
+
|
|
|
+ .ant-btn {
|
|
|
+ border-radius: 0;
|
|
|
+ border: none;
|
|
|
+ margin-right: 10px;
|
|
|
+ }
|
|
|
+
|
|
|
+ .ant-btn-group {
|
|
|
+ .ant-btn {
|
|
|
+ background: #f5f5f5;
|
|
|
+
|
|
|
+ &:hover {
|
|
|
+ background: #fff;
|
|
|
+ }
|
|
|
+
|
|
|
+ &.active {
|
|
|
+ color: @primary-color;
|
|
|
+ background: #e6f7ff;
|
|
|
+ border-color: @primary-color;
|
|
|
+ z-index: 2;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ .fr {
|
|
|
+ float: right;
|
|
|
+ line-height: 2;
|
|
|
+
|
|
|
+ span {
|
|
|
+ margin-right: 30px;
|
|
|
+
|
|
|
+ i {
|
|
|
+ display: inline-block;
|
|
|
+ width: 25px;
|
|
|
+ height: 3px;
|
|
|
+ background: #544BEB;
|
|
|
+ position: relative;
|
|
|
+ top: -4px;
|
|
|
+ margin-right: 20px;
|
|
|
+ }
|
|
|
+
|
|
|
+ &:last-child i {
|
|
|
+ background: #F0B358;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ .box {
|
|
|
+ border-radius: 10px;
|
|
|
+ text-align: center;
|
|
|
+ min-height: 180px;
|
|
|
+ display: flex;
|
|
|
+ flex-direction: column;
|
|
|
+ justify-content: center;
|
|
|
+
|
|
|
+ p {
|
|
|
+ color: #fff;
|
|
|
+
|
|
|
+ img {
|
|
|
+ width: 19px;
|
|
|
+ margin: -5px 10px 0 0;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ .num {
|
|
|
+ font-size: 30px;
|
|
|
+ margin-bottom: 10px;
|
|
|
+ }
|
|
|
+
|
|
|
+ &.b1 {
|
|
|
+ background: rgb(233, 107, 95);
|
|
|
+ }
|
|
|
+
|
|
|
+ &.b2 {
|
|
|
+ background: rgb(88, 204, 168);
|
|
|
+ }
|
|
|
+
|
|
|
+ &.b3 {
|
|
|
+ background: rgb(124, 152, 252);
|
|
|
+ }
|
|
|
+
|
|
|
+ &.b4 {
|
|
|
+ background: #F0B358;
|
|
|
+ }
|
|
|
+ }
|
|
|
+}
|
|
|
+</style>
|
|
|
+
|