trafficAnalysis.vue 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801
  1. <template>
  2. <div class="search-form">
  3. <!-- 站点选择和时间筛选 -->
  4. <a-row class="r1">
  5. <a-col :span="6">
  6. <div class="choose-site">
  7. <span class="t1">站点:</span>
  8. <select-site @set-site-info="changeSite" selectWidth="300px" />
  9. </div>
  10. </a-col>
  11. <a-col :span="8">
  12. <span class="t1">统计时间:</span>
  13. <a-range-picker @change="onChangeDatePciker" :disabledDate="disabledDate" :value="rangeDate"
  14. style="width:70%" />
  15. </a-col>
  16. <a-col :span="10">
  17. <a-button :class="queryParam.dateType == '' ? 'active' : ''" @click="setTime('')">全部时间
  18. </a-button>
  19. <a-button :class="queryParam.dateType == 'thirtyDay' ? 'active' : ''"
  20. @click="setTime('thirtyDay')">近30天
  21. </a-button>
  22. <a-button :class="queryParam.dateType == 'sevenDay' ? 'active' : ''"
  23. @click="setTime('sevenDay')">近一周
  24. </a-button>
  25. <a-button :class="queryParam.dateType == 'yesterday' ? 'active' : ''"
  26. @click="setTime('yesterday')">昨日
  27. </a-button>
  28. <a-button :class="queryParam.dateType == 'today' ? 'active' : ''" @click="setTime('today')">
  29. 今日
  30. </a-button>
  31. </a-col>
  32. </a-row>
  33. </div>
  34. <a-spin :spinning="loading" tip="加载中...">
  35. <a-row class="r2">
  36. <a-col :span="6">
  37. <p class="t1"><img src="@/assets/trafficAnalysis/uvicon.svg" />访客数(UV)</p>
  38. <p class="t3">{{ flowIndexNums.uv }}</p>
  39. </a-col>
  40. <a-col :span="6">
  41. <p class="t1"><img src="@/assets/trafficAnalysis/pvicon.svg" />浏览量(PV)</p>
  42. <p class="t3">{{ flowIndexNums.pv }}</p>
  43. </a-col>
  44. <a-col :span="6">
  45. <p class="t1"><img src="@/assets/trafficAnalysis/sessions.svg" />会话数</p>
  46. <p class="t3">{{ flowIndexNums.sessions }}</p>
  47. </a-col>
  48. <a-col :span="6">
  49. <p class="t1"><img src="@/assets/trafficAnalysis/enquirycounticon.svg" />询盘数</p>
  50. <router-link
  51. :to="{ path: '/inquiry/list', query: {dateType: queryParam.dateType, start: queryParam.start, end: queryParam.end} }">
  52. <p class="t2">{{ flowIndexNums.enquiry }}</p>
  53. </router-link>
  54. </a-col>
  55. </a-row>
  56. <a-row>
  57. <a-col :span="24">
  58. <a-card style="margin: 10px" title="核心数据">
  59. <a-row class="r5" :gutter="[20,20]">
  60. <a-row class="r5-1">
  61. <a-col :span="24">
  62. <div class="fr" v-if="coreDataChart.x.length > 0">
  63. <span><i style="background: #544BEB;"></i>访客数(UV)</span>
  64. <span><i style="background: #F0B358;"></i>浏览量(PV)</span>
  65. <span><i style="background: #58CCA8;"></i>询盘数</span>
  66. </div>
  67. <area-chart v-if="coreDataChart.x.length > 0"
  68. :dataSource="coreDataChart"></area-chart>
  69. <a-empty v-else style="float: right;width: 100%;margin-top: 110px;"></a-empty>
  70. </a-col>
  71. </a-row>
  72. <a-col style="width: 20%;">
  73. <div class="wrap">
  74. <img src="@/assets/trafficAnalysis/dailyVisitCount.svg" />
  75. <div class="fr">
  76. <p>日均访问量</p>
  77. <p style="font-size: 25px;">{{ statistics.averageVisit }}</p>
  78. </div>
  79. </div>
  80. </a-col>
  81. <a-col style="width: 20%;">
  82. <div class="wrap">
  83. <img src="@/assets/trafficAnalysis/avgVisitTime.svg" />
  84. <div class="fr">
  85. <p>平均访问时长</p>
  86. <p style="font-size: 25px;">{{ statistics.averageVisitDuration }}</p>
  87. </div>
  88. </div>
  89. </a-col>
  90. <a-col style="width: 20%;">
  91. <div class="wrap">
  92. <img src="@/assets/trafficAnalysis/avgVisitPage.svg" />
  93. <div class="fr">
  94. <p>访客平均访问页面数</p>
  95. <p style="font-size: 25px;">{{ statistics.averageVisitPage }}</p>
  96. </div>
  97. </div>
  98. </a-col>
  99. <a-col style="width: 20%;">
  100. <div class="wrap">
  101. <img src="@/assets/trafficAnalysis/tiaochu.svg" />
  102. <div class="fr">
  103. <p>跳出率</p>
  104. <p style="font-size: 25px;">{{ statistics.bounceRate }}</p>
  105. </div>
  106. </div>
  107. </a-col>
  108. <a-col style="width: 20%;">
  109. <div class="wrap">
  110. <img src="@/assets/trafficAnalysis/uvTransfer.svg" />
  111. <div class="fr">
  112. <p>UV到询盘转化率</p>
  113. <p style="font-size: 25px;">{{ statistics.conversionRate }}</p>
  114. </div>
  115. </div>
  116. </a-col>
  117. </a-row>
  118. </a-card>
  119. </a-col>
  120. <a-col :span="24">
  121. <a-card style="margin: 10px" title="访客数地域分布">
  122. <a-row class="r5">
  123. <a-col :span="18">
  124. <map-adweb v-if="countryMapData.length > 0" :dataSource="countryMapData"
  125. height="400"></map-adweb>
  126. <a-empty v-else style="margin-top: 50px;">
  127. </a-empty>
  128. </a-col>
  129. <a-col :span="6">
  130. <a-table
  131. :rowKey="(record,index)=>{return index}"
  132. class="chartTable"
  133. :scroll="{ y: 500 }"
  134. :pagination=false
  135. :columns="chartDetailDataCol"
  136. :data-source="chartDetailData"
  137. :showHeader="false">
  138. <template #bodyCell="{ column, record }">
  139. <template v-if="column.key === 'flagSlot' ">
  140. <span class="img-box">
  141. <span :class="'flag-icon flag-icon-'+record.countryCode"></span>
  142. </span>
  143. </template>
  144. <template v-if="column.key === 'numSlot' ">
  145. {{ record.totalUsers }} | {{ record.totalUsersProportion }}
  146. </template>
  147. </template>
  148. </a-table>
  149. </a-col>
  150. </a-row>
  151. </a-card>
  152. </a-col>
  153. <a-col :span="24">
  154. <a-card style="margin: 10px" title="来源媒介">
  155. <a-row class="r5" :gutter="[20,20]">
  156. <a-col :span="24">
  157. <a-table
  158. :columns="mediaListColumns"
  159. :data-source="mediaDatasource"
  160. size="middle"
  161. rowKey="type"
  162. :pagination="false">
  163. <div style="padding: 10px;" slot="filterDropdown">
  164. affiliate:通过联属营销计划点击链接的用户<br />
  165. cpc:(每次点击费用的缩写)点击付费广告的用户<br />
  166. organic:点击搜索引擎中的链接的用户<br />
  167. referral:点击网站上的链接(例如,视频说明中的链接)的用户<br />
  168. (none):直接流量
  169. </div>
  170. <a-icon slot="filterIcon" type='question-circle'
  171. :style="{ fontSize:'16px',color: '#108ee9' }" />
  172. <template #bodyCell="{ column, record, index, text }">
  173. <template v-if="column.key === 'typeSlotFirst' ">
  174. {{ record.type.split("/")[0] }}
  175. </template>
  176. <template v-if="column.key === 'typeSlotLast' ">
  177. <a-popover>
  178. <template slot="content">
  179. <template v-if="record.type.split('/')[1] === ' affiliate'">
  180. 通过联属营销计划点击链接的用户
  181. </template>
  182. <template v-if="record.type.split('/')[1] === ' cpc'">
  183. (每次点击费用的缩写)点击付费广告的用户
  184. </template>
  185. <template v-if="record.type.split('/')[1] === ' organic'">
  186. 点击搜索引擎中的链接的用户
  187. </template>
  188. <template v-if="record.type.split('/')[1] === ' referral'">
  189. 点击网站上的链接(例如,视频说明中的链接)的用户
  190. </template>
  191. <template v-if="record.type.split('/')[1] === ' (none)'">
  192. 直接流量
  193. </template>
  194. </template>
  195. {{ record.type.split("/")[1] }}
  196. </a-popover>
  197. </template>
  198. <template v-if="column.key === 'avgSessionDurationSlot' ">
  199. <span style="margin-left: 30px;">{{ record.avgSessionDuration }} s</span>
  200. </template>
  201. </template>
  202. </a-table>
  203. </a-col>
  204. </a-row>
  205. </a-card>
  206. </a-col>
  207. <a-col :span="24">
  208. <a-card style="margin: 10px" title="最多访问TOP10">
  209. <a-row class="r5" :gutter="[20,20]">
  210. <a-col :span="24">
  211. <a-table
  212. :columns="mostAccessColumns"
  213. :data-source="mostAccessDatasource"
  214. size="middle"
  215. rowKey="type"
  216. :pagination="false">
  217. <template #bodyCell="{ column, record, index, text }">
  218. <template v-if="column.key ==='pagePathSlot' ">
  219. <a-popover>
  220. <template slot="content">
  221. {{ text }}
  222. </template>
  223. <a :href="text" target="_blank">
  224. <div
  225. style="width: 700px; white-space: nowrap; overflow: hidden; text-overflow: ellipsis">
  226. {{ text }}
  227. </div>
  228. </a>
  229. </a-popover>
  230. </template>
  231. <template v-if="column.key ==='centerSlot' ">
  232. <span style="margin-left: 20px;">{{ text }}</span>
  233. </template>
  234. <template v-if="column.key ==='avgTimeOnPageSlot' ">
  235. <span style="margin-left: 30px;">{{ text }} s</span>
  236. </template>
  237. </template>
  238. </a-table>
  239. </a-col>
  240. </a-row>
  241. </a-card>
  242. </a-col>
  243. </a-row>
  244. </a-spin>
  245. </template>
  246. <script lang="ts" name="data-trafficAnalysis" setup>
  247. import selectSite from "@/components/Adweb/selectSite.vue";
  248. import areaChart from "./chart/areaChart.vue";
  249. import { onMounted, reactive, ref, watch } from "vue";
  250. import { getAction } from "@/api/manage/manage";
  251. import moment from "moment";
  252. import MapAdweb from "@/components/chart/mapAdweb.vue";
  253. import 'flag-icon-css/css/flag-icons.css'
  254. const queryParam = reactive<any>({});
  255. queryParam.limit = 10;
  256. queryParam.siteCode = localStorage.getItem("siteCode");
  257. const loading = ref(false);
  258. const chartDetailDataCol = ref([
  259. {
  260. title: "国旗",
  261. align: "center",
  262. key: "flagSlot",
  263. width: 30,
  264. scopedSlots: { customRender: "flagSlot" }
  265. },
  266. {
  267. title: "国家",
  268. align: "left",
  269. dataIndex: "countryName",
  270. customRender: function(text, record) {
  271. return text === null ? record.country : text;
  272. }
  273. },
  274. {
  275. title: "数量",
  276. align: "right",
  277. key: "numSlot",
  278. scopedSlots: { customRender: "numSlot" }
  279. }
  280. ]);
  281. // 来源媒介列表
  282. const mediaListColumns = ref([
  283. {
  284. title: "来源",
  285. key: "typeSlotFirst",
  286. scopedSlots: {
  287. customRender: "typeSlotFirst"
  288. }
  289. },
  290. {
  291. title: "媒介",
  292. key: "typeSlotLast",
  293. scopedSlots: {
  294. filterDropdown: "filterDropdown",
  295. filterIcon: "filterIcon",
  296. customRender: "typeSlotLast"
  297. }
  298. },
  299. {
  300. title: "访客数(UV)",
  301. dataIndex: "totalUsers",
  302. },
  303. {
  304. title: "占比",
  305. dataIndex: "totalUsersProportion"
  306. },
  307. {
  308. title: "新访客数",
  309. dataIndex: "newUsers",
  310. },
  311. {
  312. title: "新客占比",
  313. dataIndex: "newUsersRatio"
  314. },
  315. {
  316. title: "浏览量(PV)",
  317. dataIndex: "pageViews",
  318. },
  319. {
  320. title: "平均访问页面数",
  321. dataIndex: "pageViewsPerSession",
  322. },
  323. {
  324. title: "会话数",
  325. dataIndex: "sessions",
  326. },
  327. {
  328. title: "平均会话时长",
  329. key: "avgSessionDurationSlot",
  330. sortDirections: ['descend', 'ascend'],
  331. sorter: (a, b) => a.avgSessionDuration - b.avgSessionDuration,
  332. scopedSlots: {
  333. customRender: 'avgSessionDurationSlot',
  334. }
  335. }
  336. ]);
  337. // 最多访问TOP10列表
  338. const mostAccessColumns = ref([
  339. {
  340. title: "来源",
  341. dataIndex: "pagePath",
  342. scopedSlots: {
  343. customRender: "pagePathSlot"
  344. }
  345. },
  346. {
  347. title: "浏览量(PV)",
  348. dataIndex: "pageViews",
  349. defaultSortOrder: "descend",
  350. sorter: (a, b) => a.pageViews - b.pageViews,
  351. width: 160,
  352. scopedSlots: {
  353. customRender: "centerSlot"
  354. }
  355. },
  356. {
  357. title: "浏览量占比",
  358. dataIndex: "pvProportion",
  359. width: 160,
  360. scopedSlots: {
  361. customRender: "centerSlot"
  362. }
  363. }
  364. // {
  365. // title: '平均页面停留时间',
  366. // dataIndex: 'avgTimeOnPage',
  367. // sortDirections: ['descend', 'ascend'],
  368. // width: 160,
  369. // scopedSlots: {
  370. // customRender: 'avgTimeOnPageSlot',
  371. // }
  372. // },
  373. ]);
  374. function changeSite(selectedSiteInfo: any) {
  375. queryParam.siteCode = selectedSiteInfo.code;
  376. localStorage.setItem("siteCode", queryParam.siteCode);
  377. reloadData();
  378. }
  379. //重新刷新页面数据
  380. function reloadData() {
  381. loading.value = true;
  382. getFlowIndexNumber();
  383. getCountryMapData();
  384. getMediaList();
  385. getMostAccessList();
  386. }
  387. const flowIndexNums = ref({
  388. uv: 0,
  389. pv: 0,
  390. sessions: 0,
  391. enquiry: 0
  392. });
  393. const coreDataChart = ref({
  394. x: [],
  395. uv: [],
  396. pv: [],
  397. enquiry: []
  398. });
  399. const statistics = ref({
  400. averageVisit: 0,
  401. averageVisitDuration: 0,
  402. averageVisitPage: 0,
  403. bounceRate: "0%",
  404. conversionRate: "0%"
  405. });
  406. //访客量、浏览量、询盘数量、折线图以及统计
  407. const getFlowIndexNumber = async () => {
  408. try {
  409. const res = await getAction("/dmp-data/site-overview/stats", queryParam);
  410. if (!res.result) {
  411. flowIndexNums.value = {
  412. uv: 0,
  413. pv: 0,
  414. sessions: 0,
  415. enquiry: 0
  416. };
  417. coreDataChart.value = {
  418. x: [],
  419. uv: [],
  420. pv: [],
  421. enquiry: []
  422. };
  423. statistics.value = {
  424. averageVisit: 0,
  425. averageVisitDuration: 0,
  426. averageVisitPage: 0,
  427. bounceRate: "0%",
  428. conversionRate: "0%"
  429. };
  430. loading.value = false;
  431. return;
  432. }
  433. flowIndexNums.value.uv = res.result.totalUsers;
  434. flowIndexNums.value.pv = res.result.pageViews;
  435. flowIndexNums.value.sessions = res.result.sessions;
  436. flowIndexNums.value.enquiry = res.result.enquires;
  437. const r = res.result.dailyStats;
  438. const x = [], pv = [], uv = [], enquiry = [];
  439. if (r != null && r.length > 0) {
  440. for (let item of r) {
  441. x.push(item.date);
  442. pv.push(item.pageViews);
  443. uv.push(item.totalUsers);
  444. enquiry.push(item.enquires);
  445. }
  446. }
  447. coreDataChart.value.x = x;
  448. coreDataChart.value.pv = pv;
  449. coreDataChart.value.uv = uv;
  450. coreDataChart.value.enquiry = enquiry;
  451. statistics.value.averageVisit = res.result.dailyTotalUsers;
  452. statistics.value.averageVisitDuration = res.result.avgTimeOnPage;
  453. statistics.value.averageVisitPage = res.result.pageViewsPerSession;
  454. statistics.value.bounceRate = res.result.bounceRate;
  455. statistics.value.conversionRate = res.result.enquiryConversionRate;
  456. loading.value = false;
  457. } catch (error) {
  458. console.error(error);
  459. }
  460. };
  461. const chartDetailData = ref([]);
  462. const countryMapData = ref([]);
  463. //访客数地域分布
  464. const getCountryMapData = async () => {
  465. try {
  466. const res = await getAction("/dmp-data/country/stats", queryParam);
  467. if (res.code === 200) {
  468. chartDetailData.value = res.result;
  469. countryMapData.value = chartDetailData.value.map(entry => ({
  470. name: entry.countryName,
  471. value: entry.totalUsers
  472. }));
  473. console.log("countryMapData", countryMapData.value);
  474. }
  475. } catch (error) {
  476. console.error(error);
  477. }
  478. };
  479. const mediaDatasource = ref([]);
  480. //来源媒介列表、最多访问top10列表
  481. const getMediaList = async () => {
  482. try {
  483. const res = await getAction("/dmp-data/source-medium/stats", queryParam);
  484. if (res.code == 200) {
  485. mediaDatasource.value = res.result;
  486. } else {
  487. mediaDatasource.value = [];
  488. }
  489. } catch (error) {
  490. console.error(error);
  491. }
  492. };
  493. const mostAccessDatasource = ref([]);
  494. //
  495. const getMostAccessList = async () => {
  496. try {
  497. const res = await getAction("/dmp-data/page-path/stats", queryParam);
  498. if (res.code == 200) {
  499. mostAccessDatasource.value = res.result;
  500. } else {
  501. mostAccessDatasource.value = [];
  502. }
  503. } catch (error) {
  504. console.error(error);
  505. }
  506. };
  507. const rangeDate = ref([]);
  508. const onChangeDatePciker = (date, dateString) => {
  509. if (dateString.length > 0) {
  510. rangeDate.value = date;
  511. queryParam.value.start = dateString[0];
  512. queryParam.value.end = dateString[1];
  513. queryParam.value.dateType = undefined;
  514. reloadData();
  515. }
  516. };
  517. //日期选择只能今天之前
  518. function disabledDate(current) {
  519. return current && current > moment().subtract(0, "days");
  520. }
  521. const setTime = (time) => {
  522. queryParam.dateType = time;
  523. queryParam.start = "";
  524. queryParam.end = "";
  525. if (time == "") {
  526. rangeDate.value = undefined;
  527. } else if (time == "sevenDay") {
  528. rangeDate.value = [moment().subtract(7, "days"), moment().subtract(1, "days")];
  529. } else if (time == "thirtyDay") {
  530. rangeDate.value = [moment().subtract(30, "days"), moment().subtract(1, "days")];
  531. } else if (time == "yesterday") {
  532. rangeDate.value = [moment().subtract(1, "days"), moment().subtract(1, "days")];
  533. } else if (time == "today") {
  534. rangeDate.value = [moment(), moment()];
  535. }
  536. reloadData();
  537. };
  538. </script>
  539. <style lang="less" scoped>
  540. .self-pop {
  541. .ant-popover-inner-content {
  542. background: rgb(245, 243, 254);
  543. p {
  544. font-size: 13px;
  545. }
  546. }
  547. .ant-popover-arrow {
  548. border-color: rgb(245, 243, 254) !important;
  549. }
  550. }
  551. .img-box {
  552. width: 22px;
  553. height: 15px;
  554. display: flex;
  555. justify-content: center;
  556. align-items: center;
  557. img {
  558. width: 100%;
  559. height: 100%;
  560. }
  561. }
  562. .ant-alert {
  563. /deep/ .ant-btn {
  564. border-radius: 0;
  565. margin-left: 10px;
  566. }
  567. }
  568. .theme-color {
  569. color: @primary-color;
  570. }
  571. .r1 {
  572. margin: 20px;
  573. .choose-site {
  574. display: flex;
  575. }
  576. .t1 {
  577. font-size: 18px;
  578. }
  579. .ant-form-item {
  580. flex: 1;
  581. }
  582. .ant-calendar-picker {
  583. margin-right: 20px;
  584. }
  585. /deep/ .ant-btn {
  586. background: transparent;
  587. margin-right: 10px;
  588. &.active {
  589. color: @primary-color;
  590. }
  591. }
  592. }
  593. .r2 {
  594. background: #fff;
  595. padding: 30px 20px;
  596. margin: 10px;
  597. .ant-col:not(:last-child) {
  598. border-right: 1px solid #e6e6e6;
  599. }
  600. p {
  601. margin: 0;
  602. text-align: center;
  603. &.t1 {
  604. color: #333;
  605. margin-bottom: 15px;
  606. img {
  607. margin-right: 10px;
  608. width: 15px;
  609. margin-top: -5px;
  610. }
  611. }
  612. &.t2 {
  613. color: @primary-color;
  614. font-size: 30px;
  615. font-weight: 500;
  616. line-height: 1;
  617. padding-left: 25px;
  618. }
  619. &.t3 {
  620. font-size: 30px;
  621. font-weight: 500;
  622. line-height: 1;
  623. padding-left: 25px;
  624. }
  625. }
  626. }
  627. .r5 {
  628. background: #fff;
  629. padding: 10px;
  630. border-radius: 10px;
  631. margin: 0 !important;
  632. .wrap {
  633. box-shadow: 0px 2px 4px 0px @primary-color;
  634. padding: 15px;
  635. border-radius: 10px;
  636. overflow: hidden;
  637. background: #fff;
  638. transition: all .3s;
  639. &.blue {
  640. box-shadow: 0px 2px 4px 0px @primary-color;
  641. }
  642. &.effect:hover {
  643. box-shadow: none;
  644. background: rgb(241, 248, 255);
  645. }
  646. img {
  647. width: 15px;
  648. }
  649. .fr {
  650. float: right;
  651. width: calc(100% - 15px);
  652. text-align: center;
  653. p:last-child {
  654. font-size: 30px;
  655. text-align: center;
  656. margin-top: 10px;
  657. }
  658. }
  659. }
  660. /deep/ .ant-table-thead > tr > th {
  661. background: rgb(241, 248, 255);
  662. border: none;
  663. color: #000;
  664. padding: 10px;
  665. }
  666. /deep/ .ant-table-tbody .ant-table-row td {
  667. padding: 10px;
  668. color: #000;
  669. }
  670. .r5-1 {
  671. display: inline-block;
  672. width: 100%;
  673. margin-top: 30px;
  674. .fl {
  675. float: left;
  676. position: relative;
  677. .ant-btn {
  678. border-radius: 0;
  679. border: none;
  680. margin-right: 10px;
  681. }
  682. }
  683. .fr {
  684. float: right;
  685. line-height: 2;
  686. span {
  687. margin-right: 30px;
  688. i {
  689. display: inline-block;
  690. width: 25px;
  691. height: 3px;
  692. background: #544BEB;
  693. position: relative;
  694. top: -4px;
  695. margin-right: 20px;
  696. }
  697. &:last-child i {
  698. background: #F0B358;
  699. }
  700. }
  701. }
  702. }
  703. .box {
  704. border-radius: 10px;
  705. text-align: center;
  706. min-height: 180px;
  707. display: flex;
  708. flex-direction: column;
  709. justify-content: center;
  710. p {
  711. color: #fff;
  712. img {
  713. width: 19px;
  714. margin: -5px 10px 0 0;
  715. }
  716. }
  717. .num {
  718. font-size: 30px;
  719. margin-bottom: 10px;
  720. }
  721. &.b1 {
  722. background: rgb(233, 107, 95);
  723. }
  724. &.b2 {
  725. background: rgb(88, 204, 168);
  726. }
  727. &.b3 {
  728. background: rgb(124, 152, 252);
  729. }
  730. &.b4 {
  731. background: #F0B358;
  732. }
  733. }
  734. }
  735. </style>