googleads.vue 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962
  1. <template>
  2. <div class="googlleads">
  3. <div class="search-form">
  4. <!-- 站点选择和时间筛选 -->
  5. <a-row class="r1 search-form-container" :gutter="8">
  6. <a-col :xl="7" :xxl="6">
  7. <div class="choose-site">
  8. <span class="t1">站点:</span>
  9. <select-site @set-site-info="changeSite" select-width="100%" />
  10. </div>
  11. </a-col>
  12. <a-col :xl="8" :xxl="6">
  13. <div class="choose-site">
  14. <span class="t1">统计时间:</span>
  15. <a-range-picker @change="onChangeDatePicker" :disabledDate="disabledDate" :value="rangeDate" style="width: 70%" />
  16. </div>
  17. </a-col>
  18. <a-col :xl="9" :xxl="12">
  19. <a-button-group class="time-btn-group">
  20. <a-button :class="queryParam.dateType == '' ? 'active' : ''" @click="setTime('')">全部时间</a-button>
  21. <a-button :class="queryParam.dateType == 'thirtyDay' ? 'active' : ''" @click="setTime('thirtyDay')">近30天</a-button>
  22. <a-button :class="queryParam.dateType == 'sevenDay' ? 'active' : ''" @click="setTime('sevenDay')">近一周</a-button>
  23. <a-button :class="queryParam.dateType == 'yesterday' ? 'active' : ''" @click="setTime('yesterday')">昨日</a-button>
  24. <a-button :class="queryParam.dateType == 'today' ? 'active' : ''" @click="setTime('today')">今日</a-button>
  25. </a-button-group>
  26. </a-col>
  27. </a-row>
  28. </div>
  29. <a-spin :spinning="loading" tip="加载中...">
  30. <a-row>
  31. <a-col :span="16">
  32. <a-row class="r2">
  33. <a-col :span="24">
  34. <p class="t3 company-name">{{ customerStats.descriptiveName }}</p>
  35. </a-col>
  36. </a-row>
  37. <a-row class="r2">
  38. <a-col :span="8">
  39. <p class="t1">账户余额</p>
  40. <p class="t3">
  41. <span class="value">{{ customerStats.balance }}</span>
  42. <span class="currency">{{ customerStats.currency }}</span>
  43. </p>
  44. </a-col>
  45. <a-col :span="8">
  46. <p class="t1">总花费</p>
  47. <p class="t3">
  48. <span class="value">{{ customerStats.cost }}</span>
  49. <span class="currency">{{ customerStats.currency }}</span>
  50. </p>
  51. </a-col>
  52. <a-col :span="8">
  53. <p class="t1">转化数</p>
  54. <p class="t3">{{ customerStats.conversions }}</p>
  55. </a-col>
  56. </a-row>
  57. </a-col>
  58. <a-col :span="8">
  59. <a-row class="r2" style="height: 195px">
  60. <a-col :span="24">
  61. <p class="t1">优化得分</p>
  62. <div class="score-circle" style="padding-bottom: 10px">
  63. <a-progress
  64. type="circle"
  65. :percent="customerStats.optiScore.toFixed(2)"
  66. :width="80"
  67. :stroke-color="{
  68. '0%': '#FFB800',
  69. '100%': '#FFC53D',
  70. }"
  71. />
  72. </div>
  73. </a-col>
  74. </a-row>
  75. </a-col>
  76. </a-row>
  77. <a-row>
  78. <a-col :span="24">
  79. <a-card style="margin: 10px" title="核心数据">
  80. <a-row class="r5" :gutter="8">
  81. <a-row class="r5-1">
  82. <a-col :span="24">
  83. <template v-if="dailyStats.values && dailyStats.values.length > 0">
  84. <div class="fl">
  85. <a-button-group>
  86. <a-button :class="{ active: activeChart === 'impression' }" @click="switchChart('impression')">展示</a-button>
  87. <a-button :class="{ active: activeChart === 'clicks' }" @click="switchChart('clicks')">点击</a-button>
  88. <a-button :class="{ active: activeChart === 'ctr' }" @click="switchChart('ctr')">点击率(%)</a-button>
  89. <a-button :class="{ active: activeChart === 'conversion' }" @click="switchChart('conversion')">转化数</a-button>
  90. <a-button :class="{ active: activeChart === 'cost' }" @click="switchChart('cost')">花费</a-button>
  91. </a-button-group>
  92. </div>
  93. <line-chart :chartType="activeChart" :dailyStats="dailyStats" />
  94. </template>
  95. <template v-else>
  96. <a-empty />
  97. </template>
  98. </a-col>
  99. </a-row>
  100. </a-row>
  101. </a-card>
  102. </a-col>
  103. </a-row>
  104. <a-row>
  105. <a-col :span="24">
  106. <a-card style="margin: 10px" title="广告系列">
  107. <a-table
  108. :columns="campaignColumns"
  109. :data-source="tableData"
  110. :loading="loading"
  111. :pagination="{
  112. pageSize: 10,
  113. showSizeChanger: false,
  114. showQuickJumper: true,
  115. showTotal: (total) => `共 ${total} 条`,
  116. }"
  117. bordered
  118. :row-class-name="(_record, index) => (index % 2 === 1 ? 'table-striped' : '')"
  119. style="width: 100%"
  120. />
  121. </a-card>
  122. </a-col>
  123. </a-row>
  124. <a-row :gutter="8">
  125. <a-col :span="12">
  126. <a-card style="margin: 10px" title="TOP关键词">
  127. <a-table
  128. :columns="keywordColumns"
  129. :data-source="keywordData"
  130. :loading="loading"
  131. :pagination="{
  132. pageSize: 10,
  133. showSizeChanger: false,
  134. showQuickJumper: true,
  135. showTotal: (total) => `共 ${total} 条`,
  136. }"
  137. bordered
  138. :row-class-name="(_record, index) => (index % 2 === 1 ? 'table-striped' : '')"
  139. style="width: 100%"
  140. />
  141. </a-card>
  142. </a-col>
  143. <a-col :span="12">
  144. <a-card style="margin: 10px" title="TOP展示位">
  145. <a-table
  146. :columns="positionColumns"
  147. :data-source="positionData"
  148. :loading="loading"
  149. :pagination="{
  150. pageSize: 10,
  151. showSizeChanger: false,
  152. showQuickJumper: true,
  153. showTotal: (total) => `共 ${total} 条`,
  154. }"
  155. bordered
  156. :row-class-name="(_record, index) => (index % 2 === 1 ? 'table-striped' : '')"
  157. style="width: 100%"
  158. />
  159. </a-card>
  160. </a-col>
  161. </a-row>
  162. <a-row>
  163. <a-col :span="24">
  164. <a-card style="margin: 10px" title="TOP国家/地区">
  165. <a-row class="r5">
  166. <a-col :span="18">
  167. <map-adweb v-if="countryMapData.length > 0" :dataSource="countryMapData" height="400" />
  168. <a-empty v-else style="margin-top: 50px" />
  169. </a-col>
  170. <a-col :span="6">
  171. <a-table
  172. :rowKey="
  173. (record, index) => {
  174. return index;
  175. }
  176. "
  177. class="chartTable"
  178. :scroll="{ y: 500 }"
  179. :pagination="false"
  180. :columns="chartDetailDataCol"
  181. :data-source="chartDetailData"
  182. :showHeader="true"
  183. >
  184. <template #bodyCell="{ column, record }">
  185. <template v-if="column.key === 'flagSlot'">
  186. <span class="img-box">
  187. <span :class="'flag-icon flag-icon-' + record.countryCode"></span>
  188. </span>
  189. </template>
  190. <template v-if="column.key === 'numSlot'"> {{ record.impressions }} | {{ record.clicks }} </template>
  191. </template>
  192. </a-table>
  193. </a-col>
  194. </a-row>
  195. </a-card>
  196. </a-col>
  197. </a-row>
  198. </a-spin>
  199. </div>
  200. </template>
  201. <script setup lang="ts" name="marketing-googleads">
  202. import { ref, reactive } from 'vue';
  203. import dayjs from 'dayjs';
  204. import selectSite from '@/components/Adweb/selectSite.vue';
  205. import LineChart from './charts/Line.vue';
  206. import MapAdweb from '@/components/chart/mapAdweb.vue';
  207. import {
  208. getGoogleAdsCustomerStats,
  209. getGoogleAdsDailyStats,
  210. getGoogleAdsCampaignStats,
  211. getGoogleAdsKeywordStats,
  212. getGoogleAdsPlacementStats,
  213. getGoogleAdsCountryStats,
  214. } from './googleads.api';
  215. import 'flag-icon-css/css/flag-icons.css';
  216. const rangeDate = ref([]);
  217. const queryParam = reactive<any>({});
  218. queryParam.limit = 10;
  219. queryParam.siteCode = localStorage.getItem('siteCode');
  220. const loading = ref(false);
  221. const activeChart = ref('impression');
  222. const chartDetailData = ref([]);
  223. const countryMapData = ref([]);
  224. const chartDetailDataCol = ref([
  225. {
  226. title: '',
  227. key: 'flagSlot',
  228. width: '30px',
  229. },
  230. {
  231. title: '国家/地区名称',
  232. dataIndex: 'countryName',
  233. key: 'countryName',
  234. },
  235. {
  236. title: '展示数 | 点击数',
  237. key: 'numSlot',
  238. align: 'right',
  239. },
  240. ]);
  241. const changeSite = (selectedSiteInfo: any) => {
  242. queryParam.siteCode = selectedSiteInfo.code;
  243. localStorage.setItem('siteCode', queryParam.siteCode);
  244. reloadData();
  245. };
  246. const onChangeDatePicker = (date, dateString) => {
  247. if (dateString.length > 0) {
  248. console.log('rangeDate:', rangeDate.value);
  249. rangeDate.value = date;
  250. console.log('date:', date);
  251. queryParam.start = dateString[0];
  252. queryParam.end = dateString[1];
  253. queryParam.dateType = undefined;
  254. reloadData();
  255. }
  256. };
  257. const setTime = (time) => {
  258. queryParam.dateType = time;
  259. queryParam.start = '';
  260. queryParam.end = '';
  261. if (time == '') {
  262. rangeDate.value = undefined;
  263. } else if (time == 'sevenDay') {
  264. rangeDate.value = [dayjs().add(-7, 'd'), dayjs().add(-1, 'd')];
  265. } else if (time == 'thirtyDay') {
  266. rangeDate.value = [dayjs().add(-30, 'd'), dayjs().add(-1, 'd')];
  267. } else if (time == 'yesterday') {
  268. rangeDate.value = [dayjs().add(-1, 'd'), dayjs().add(-1, 'd')];
  269. } else if (time == 'today') {
  270. rangeDate.value = [dayjs(), dayjs()];
  271. }
  272. reloadData();
  273. };
  274. //日期选择只能今天之前
  275. const disabledDate = (current) => {
  276. return current && current > dayjs();
  277. };
  278. //重新刷新页面数据
  279. const reloadData = () => {
  280. loading.value = true;
  281. getCustomerStats();
  282. getDailyStats();
  283. getCampaignStats();
  284. getKeywordStats();
  285. getPlacementStats();
  286. getCountryStats();
  287. loading.value = false;
  288. };
  289. // 存储所有图表数据
  290. const allChartData = ref({
  291. dates: [] as string[],
  292. impression: [] as number[],
  293. clicks: [] as number[],
  294. ctr: [] as number[],
  295. conversion: [] as number[],
  296. cost: [] as number[],
  297. });
  298. // 修改 getDailyStats 函数
  299. const getDailyStats = async () => {
  300. try {
  301. loading.value = true;
  302. const res = await getGoogleAdsDailyStats(queryParam);
  303. if (res) {
  304. // 按日期排序并保存原始数据
  305. const sortedData = res.sort((a, b) => new Date(a.date).getTime() - new Date(b.date).getTime());
  306. // 保存所有数据
  307. allChartData.value = {
  308. dates: sortedData.map((item) => item.date),
  309. impression: sortedData.map((item) => Number(item.impressions) || 0),
  310. clicks: sortedData.map((item) => Number(item.clicks) || 0),
  311. ctr: sortedData.map((item) => (Number(item.ctr) * 100).toFixed(2) || 0.0),
  312. conversion: sortedData.map((item) => Number(item.conversions) || 0),
  313. cost: sortedData.map((item) => Number(item.cost) || 0),
  314. };
  315. // 设置初始显示数据
  316. dailyStats.value = {
  317. dates: allChartData.value.dates,
  318. values: allChartData.value[activeChart.value] || [],
  319. };
  320. }
  321. } catch (error) {
  322. console.error('获取每日统计数据失败:', error);
  323. } finally {
  324. loading.value = false;
  325. }
  326. };
  327. // 简化切换图表类型的函数
  328. const switchChart = (type: string) => {
  329. activeChart.value = type;
  330. // 直接使用已有数据更新图表
  331. dailyStats.value = {
  332. dates: allChartData.value.dates,
  333. values: allChartData.value[type] || [],
  334. };
  335. };
  336. const campaignColumns = ref([
  337. {
  338. title: '广告系列名称',
  339. dataIndex: 'name',
  340. key: 'name',
  341. },
  342. {
  343. title: '状态',
  344. dataIndex: 'status',
  345. key: 'status',
  346. },
  347. {
  348. title: '类型',
  349. dataIndex: 'advertisingChannelType',
  350. key: 'advertisingChannelType',
  351. },
  352. {
  353. title: '展示数',
  354. dataIndex: 'impressions',
  355. key: 'impressions',
  356. },
  357. {
  358. title: '点击数',
  359. dataIndex: 'clicks',
  360. key: 'clicks',
  361. },
  362. {
  363. title: '点击率(%)',
  364. dataIndex: 'ctr',
  365. key: 'ctr',
  366. customRender: ({ text }) => {
  367. return text ? (Number(text) * 100).toFixed(2) + '%' : '0.00%';
  368. },
  369. },
  370. {
  371. title: 'CPC',
  372. dataIndex: 'averageCpc',
  373. key: 'cpc',
  374. slots: { customRender: 'cpc' },
  375. tooltip: '单次点击费用',
  376. customRender: ({ text }) => `${text.toFixed(2)}`,
  377. },
  378. {
  379. title: '转化数',
  380. dataIndex: 'conversions',
  381. key: 'conversions',
  382. customRender: ({ text }) => `${text.toFixed(2)}`,
  383. },
  384. {
  385. title: '花费',
  386. dataIndex: 'cost',
  387. key: 'cost',
  388. customRender: ({ text }) => `${text.toFixed(2)}`,
  389. },
  390. ]);
  391. const tableData = ref([]);
  392. const keywordColumns = ref([
  393. {
  394. title: '关键词',
  395. dataIndex: 'keyword',
  396. key: 'keyword',
  397. },
  398. {
  399. title: '展示次数',
  400. dataIndex: 'impressions',
  401. key: 'impressions',
  402. },
  403. {
  404. title: '点击次数',
  405. dataIndex: 'clicks',
  406. key: 'clicks',
  407. },
  408. {
  409. title: '花费',
  410. dataIndex: 'cost',
  411. key: 'cost',
  412. customRender: ({ text }) => `${text.toFixed(2)}`,
  413. },
  414. ]);
  415. const positionColumns = ref([
  416. {
  417. title: '展示位置',
  418. dataIndex: 'placement',
  419. key: 'placement',
  420. },
  421. {
  422. title: '类型',
  423. dataIndex: 'type',
  424. key: 'type',
  425. },
  426. {
  427. title: '展示次数',
  428. dataIndex: 'impressions',
  429. key: 'impressions',
  430. },
  431. {
  432. title: '点击次数',
  433. dataIndex: 'clicks',
  434. key: 'clicks',
  435. },
  436. {
  437. title: '花费',
  438. dataIndex: 'cost',
  439. key: 'cost',
  440. customRender: ({ text }) => `${text.toFixed(2)}`,
  441. },
  442. ]);
  443. const keywordData = ref([]);
  444. const positionData = ref([]);
  445. // 修改响应式数据的类型定义
  446. const customerStats = ref({
  447. customerId: '-',
  448. descriptiveName: '-',
  449. currency: '',
  450. balance: '-',
  451. cost: '-',
  452. conversions: '-',
  453. optiScore: 0,
  454. });
  455. // 添加格式化数字的函数
  456. const formatNumber = (num: string | number) => {
  457. if (!num) return '-';
  458. // 处理字符串类型的数字
  459. const numStr = typeof num === 'string' ? num : num.toString();
  460. // 分离整数和小数部分
  461. const parts = numStr.split('.');
  462. // 对整数部分添加千位分隔符
  463. parts[0] = parts[0].replace(/\B(?=(\d{3})+(?!\d))/g, ',');
  464. // 重新组合整数和小数部分
  465. return parts.join('.');
  466. };
  467. // 获取客户统计数据
  468. const getCustomerStats = async () => {
  469. try {
  470. loading.value = true;
  471. const res = await getGoogleAdsCustomerStats(queryParam);
  472. const stats = res;
  473. // // 处理余额和花费的分割
  474. // const balanceParts = (stats.balance ?? '-').toString().match(/^([\d,.]+)\s*(.*)$/);
  475. // const costParts = (stats.cost ?? '-').toString().match(/^([\d,.]+)\s*(.*)$/);
  476. customerStats.value = {
  477. customerId: stats.customerId ?? '-',
  478. descriptiveName: stats.descriptiveName ?? '-',
  479. currency: stats.currencyCode ?? '',
  480. balance: stats.balance ? formatNumber(stats.balance) : '0',
  481. cost: stats.cost ? formatNumber(stats.cost) : '-',
  482. conversions: stats.conversions ? formatNumber(stats.conversions) : '-',
  483. optiScore: stats.optiScore ? stats.optiScore * 100 : 0,
  484. };
  485. } catch (error) {
  486. console.error('获取客户统计数据失败:', error);
  487. } finally {
  488. loading.value = false;
  489. }
  490. };
  491. const dailyStats = ref({
  492. dates: [] as string[],
  493. values: [] as number[],
  494. });
  495. const getCampaignStats = async () => {
  496. try {
  497. loading.value = true;
  498. const res = await getGoogleAdsCampaignStats(queryParam);
  499. if (res) {
  500. tableData.value = res.map((item) => ({
  501. name: item.name,
  502. status: item.status,
  503. advertisingChannelType: item.advertisingChannelType,
  504. biddingStrategyType: item.biddingStrategyType,
  505. budget: item.budget,
  506. impressions: item.impressions,
  507. clicks: item.clicks,
  508. ctr: item.ctr,
  509. averageCpc: item.averageCpc,
  510. averageCpm: item.averageCpm,
  511. conversions: item.conversions,
  512. cost: item.cost,
  513. }));
  514. }
  515. } catch (error) {
  516. console.error('获取广告系列数据失败:', error);
  517. } finally {
  518. loading.value = false;
  519. }
  520. };
  521. const getKeywordStats = async () => {
  522. try {
  523. loading.value = true;
  524. const res = await getGoogleAdsKeywordStats(queryParam);
  525. if (res) {
  526. keywordData.value = res.map((item) => ({
  527. keyword: item.keyword,
  528. impressions: item.impressions,
  529. clicks: item.clicks,
  530. ctr: item.ctr,
  531. cost: item.cost,
  532. }));
  533. }
  534. } catch (error) {
  535. console.error('获取关键词统计数据失败:', error);
  536. } finally {
  537. loading.value = false;
  538. }
  539. };
  540. const getPlacementStats = async () => {
  541. try {
  542. loading.value = true;
  543. const res = await getGoogleAdsPlacementStats(queryParam);
  544. if (res) {
  545. positionData.value = res.map((item) => ({
  546. placement: item.placement,
  547. type: item.type,
  548. impressions: item.impressions,
  549. clicks: item.clicks,
  550. ctr: item.ctr,
  551. cost: item.cost,
  552. }));
  553. }
  554. } catch (error) {
  555. console.error('获取展示位置统计数据失败:', error);
  556. } finally {
  557. loading.value = false;
  558. }
  559. };
  560. const getCountryStats = async () => {
  561. try {
  562. loading.value = true;
  563. const res = await getGoogleAdsCountryStats(queryParam);
  564. if (res) {
  565. // 处理地图数据
  566. countryMapData.value = res.map((item) => ({
  567. name: item.countryName,
  568. value: item.impressions,
  569. }));
  570. // 处理表格数据
  571. chartDetailData.value = res.map((item) => ({
  572. countryCode: item.countryCode?.toLowerCase(),
  573. countryName: item.countryName,
  574. clicks: item.clicks,
  575. impressions: item.impressions,
  576. }));
  577. }
  578. } catch (error) {
  579. console.error('获取国家统计数据失败:', error);
  580. } finally {
  581. loading.value = false;
  582. }
  583. };
  584. // onMounted(() => {
  585. // getCustomerStats();
  586. // getDailyStats();
  587. // getCampaignStats();
  588. // getKeywordStats();
  589. // getPlacementStats();
  590. // getCountryStats();
  591. // });
  592. </script>
  593. <style scoped lang="less">
  594. .googlleads {
  595. padding-top: 72px;
  596. .search-form {
  597. position: fixed;
  598. width: 100%;
  599. z-index: 999;
  600. top: 110px;
  601. background-color: #f5f5f5;
  602. //background-color: var(--header-bg-color) !important;
  603. .search-form-container {
  604. align-items: center;
  605. }
  606. }
  607. }
  608. .r1 {
  609. margin: 20px;
  610. .choose-site {
  611. display: flex;
  612. align-items: center;
  613. :deep(.ant-form-item) {
  614. margin-bottom: 0;
  615. }
  616. }
  617. .t1 {
  618. font-size: 18px;
  619. font-weight: 400;
  620. letter-spacing: 0px;
  621. line-height: 32px;
  622. margin-left: 10px;
  623. }
  624. .ant-form-item {
  625. flex: 1;
  626. }
  627. .ant-calendar-picker {
  628. margin-right: 20px;
  629. }
  630. /deep/ .ant-btn {
  631. background: transparent;
  632. margin-right: 10px;
  633. padding: 4px 15px;
  634. border: 1px solid #d9d9d9;
  635. border-radius: 4px;
  636. transition: all 0.3s;
  637. &:hover {
  638. color: @primary-color;
  639. border-color: @primary-color;
  640. }
  641. &.active {
  642. color: @primary-color;
  643. background: #e6f7ff;
  644. border-color: @primary-color;
  645. }
  646. }
  647. .time-btn-group {
  648. /deep/ .ant-btn {
  649. background: #fff;
  650. padding: 4px 15px;
  651. border: 1px solid #d9d9d9;
  652. transition: all 0.3s;
  653. margin-right: 0;
  654. &:first-child {
  655. border-top-left-radius: 4px;
  656. border-bottom-left-radius: 4px;
  657. }
  658. &:last-child {
  659. border-top-right-radius: 4px;
  660. border-bottom-right-radius: 4px;
  661. }
  662. &:not(:first-child) {
  663. margin-left: -1px;
  664. }
  665. &:hover {
  666. color: @primary-color;
  667. border-color: @primary-color;
  668. position: relative;
  669. z-index: 1;
  670. background: #fff;
  671. }
  672. &.active {
  673. color: @primary-color;
  674. background: #e6f7ff;
  675. border-color: @primary-color;
  676. position: relative;
  677. z-index: 2;
  678. }
  679. }
  680. }
  681. }
  682. .r2 {
  683. background: #fff;
  684. padding: 20px;
  685. margin: 15px 10px 10px;
  686. .ant-col:not(:last-child) {
  687. border-right: 1px solid #e6e6e6;
  688. }
  689. p {
  690. margin: 0;
  691. text-align: center;
  692. &.t1 {
  693. color: #333;
  694. margin-bottom: 15px;
  695. img {
  696. margin-right: 10px;
  697. width: 15px;
  698. margin-top: -5px;
  699. }
  700. }
  701. &.t2 {
  702. color: @primary-color;
  703. font-size: 30px;
  704. font-weight: 500;
  705. line-height: 1;
  706. padding-left: 25px;
  707. }
  708. &.t3 {
  709. font-size: 32px;
  710. font-weight: 700;
  711. letter-spacing: 0px;
  712. line-height: 38px;
  713. color: rgba(13, 62, 122, 1);
  714. .value {
  715. font-size: 32px;
  716. }
  717. .currency {
  718. font-size: 16px;
  719. margin-left: 4px;
  720. }
  721. &.company-name {
  722. font-size: 18px;
  723. text-align: left;
  724. line-height: 1;
  725. font-weight: 600;
  726. }
  727. }
  728. }
  729. }
  730. .r5 {
  731. background: #fff;
  732. padding: 10px;
  733. border-radius: 10px;
  734. margin: 0 !important;
  735. .wrap {
  736. box-shadow: 0px 2px 4px 0px @primary-color;
  737. padding: 15px;
  738. border-radius: 10px;
  739. overflow: hidden;
  740. background: #fff;
  741. transition: all 0.3s;
  742. &.blue {
  743. box-shadow: 0px 2px 4px 0px @primary-color;
  744. }
  745. &.effect:hover {
  746. box-shadow: none;
  747. background: rgb(241, 248, 255);
  748. }
  749. img {
  750. width: 15px;
  751. }
  752. .fr {
  753. float: right;
  754. width: calc(100% - 15px);
  755. text-align: center;
  756. p:last-child {
  757. font-size: 30px;
  758. text-align: center;
  759. margin-top: 10px;
  760. }
  761. }
  762. }
  763. /deep/ .ant-table-thead > tr > th {
  764. background: rgb(241, 248, 255);
  765. border: none;
  766. color: #000;
  767. padding: 10px;
  768. }
  769. /deep/ .ant-table-tbody .ant-table-row td {
  770. padding: 10px;
  771. color: #000;
  772. }
  773. .r5-1 {
  774. display: inline-block;
  775. width: 100%;
  776. margin-top: 30px;
  777. .fl {
  778. float: left;
  779. position: relative;
  780. .ant-btn {
  781. border-radius: 0;
  782. border: none;
  783. margin-right: 10px;
  784. }
  785. .ant-btn-group {
  786. .ant-btn {
  787. background: #f5f5f5;
  788. &:hover {
  789. background: #fff;
  790. }
  791. &.active {
  792. color: @primary-color;
  793. background: #e6f7ff;
  794. border-color: @primary-color;
  795. z-index: 2;
  796. }
  797. }
  798. }
  799. }
  800. .fr {
  801. float: right;
  802. line-height: 2;
  803. span {
  804. margin-right: 30px;
  805. i {
  806. display: inline-block;
  807. width: 25px;
  808. height: 3px;
  809. background: #544beb;
  810. position: relative;
  811. top: -4px;
  812. margin-right: 20px;
  813. }
  814. &:last-child i {
  815. background: #f0b358;
  816. }
  817. }
  818. }
  819. }
  820. .box {
  821. border-radius: 10px;
  822. text-align: center;
  823. min-height: 180px;
  824. display: flex;
  825. flex-direction: column;
  826. justify-content: center;
  827. p {
  828. color: #fff;
  829. img {
  830. width: 19px;
  831. margin: -5px 10px 0 0;
  832. }
  833. }
  834. .num {
  835. font-size: 30px;
  836. margin-bottom: 10px;
  837. }
  838. &.b1 {
  839. background: rgb(233, 107, 95);
  840. }
  841. &.b2 {
  842. background: rgb(88, 204, 168);
  843. }
  844. &.b3 {
  845. background: rgb(124, 152, 252);
  846. }
  847. &.b4 {
  848. background: #f0b358;
  849. }
  850. }
  851. }
  852. .score-circle {
  853. text-align: center;
  854. margin-top: 20px;
  855. }
  856. // 添加表格样式
  857. :deep(.table-striped) {
  858. background-color: #fafafa;
  859. }
  860. // 修改表格头部样式
  861. :deep(.ant-table-thead > tr > th) {
  862. background: #e6f7ff !important;
  863. font-weight: 700; // 字重改为700
  864. color: rgba(13, 62, 122, 1) !important; // 字体颜色调整
  865. }
  866. :deep(.ant-table-tbody > tr > td) {
  867. border-bottom: 1px solid #f0f0f0;
  868. }
  869. </style>