adStruct.vue 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514
  1. <template>
  2. <div v-loading="loading">
  3. <el-row :gutter="5">
  4. <el-col :span="9">
  5. <div>
  6. <TextSelector v-model="modelValue" :options="pieOptions" @change="changePie" style="margin-top: 5px"/>
  7. </div>
  8. <div ref="pie" style="height: 400px;"></div>
  9. </el-col>
  10. <el-col :span="15">
  11. <div style="margin-left: 40%">
  12. <span style="background: #3a83f7; width: 18px; height: 10px; margin-top: 8px; display: inline-block; border-radius: 3px;"></span>
  13. <TextSelector v-model="barModelValue1" :options="barOptions1" @change="changeBarOne" style="margin-top: 5px; margin-left: 8px;"/>
  14. <span style="background: #f19a37; width: 18px; height: 10px; margin-top: 8px; margin-left: 20px; display: inline-block; border-radius: 3px;"></span>
  15. <TextSelector v-model="barModelValue2" :options="barOptions2" @change="changeBarTwo" style="margin-top: 5px; margin-left: 8px;"/>
  16. </div>
  17. <div ref="bar" style="height: 400px;"></div>
  18. </el-col>
  19. </el-row>
  20. </div>
  21. </template>
  22. <script setup>
  23. import { onMounted, ref, inject, watch, nextTick } from "vue"
  24. import * as echarts from "echarts"
  25. import TextSelector from '/@/components/TextSelector/index.vue'
  26. import { getAdStructureData } from "/@/views/adManage/sp/campaigns/api"
  27. let pieChart = ref()
  28. let barChart = ref()
  29. const pie = ref()
  30. const bar = ref()
  31. const loading = ref(true)
  32. let dateRange = inject('dateRange')
  33. console.log('dateRange', dateRange.value)
  34. watch(
  35. dateRange,
  36. () => {
  37. console.log('-------watch dateRange', dateRange.value)
  38. }
  39. )
  40. // 下拉框相关
  41. const pieOptions = [
  42. {
  43. value: 'Spend',
  44. label: '花费',
  45. },
  46. {
  47. value: 'TotalSales',
  48. label: '销售额',
  49. },
  50. {
  51. value: 'TotalPurchases',
  52. label: '订单数',
  53. },
  54. {
  55. value: 'TotalUnitOrdered',
  56. label: '销量',
  57. },
  58. {
  59. value: 'Impression',
  60. label: '曝光量',
  61. },
  62. {
  63. value: 'Click',
  64. label: '点击量',
  65. },
  66. ]
  67. const metricMap = {
  68. 'Spend': '花费',
  69. 'TotalSales': '销售额',
  70. 'TotalPurchases': '订单数',
  71. 'TotalUnitOrdered': '销量',
  72. 'Impression': '曝光量',
  73. 'Click': '点击量',
  74. }
  75. let modelValue = ref(pieOptions[0].value)
  76. const barOptions1 = [
  77. {
  78. value: 'ACOS',
  79. label: 'ACOS',
  80. },
  81. {
  82. value: 'ROAS',
  83. label: 'ROAS',
  84. },
  85. {
  86. value: 'Spend',
  87. label: '花费',
  88. units: '$',
  89. },
  90. {
  91. value: 'TotalSales',
  92. label: '销售额',
  93. },
  94. {
  95. value: 'TotalPurchases',
  96. label: '订单数',
  97. },
  98. {
  99. value: 'TotalUnitOrdered',
  100. label: '销量',
  101. },
  102. {
  103. value: 'CPC',
  104. label: '点击成本'
  105. },
  106. {
  107. value: 'CPA',
  108. label: '订单成本'
  109. },
  110. {
  111. value: 'Impression',
  112. label: '曝光量',
  113. },
  114. {
  115. value: 'Click',
  116. label: '点击量',
  117. },
  118. {
  119. value: 'qwe',
  120. label: '点击率'
  121. },
  122. {
  123. value: '转化率',
  124. label: '转化率'
  125. },
  126. {
  127. value: 'TotalSalesSameSKU',
  128. label: '推广商品销售额'
  129. },
  130. {
  131. value: 'TotalSalesOtherSKU',
  132. label: '其他商品销售额'
  133. },
  134. {
  135. value: 'TotalPurchasesSameSKU',
  136. label: '推广商品订单数'
  137. },
  138. {
  139. value: 'TotalPurchasesOtherSKU',
  140. label: '其他商品订单数'
  141. },
  142. {
  143. value: 'TotalUnitOrderedSameSKU',
  144. label: '推广商品销量'
  145. },
  146. {
  147. value: 'TotalUnitOrderedOtherSKU',
  148. label: '其他商品销量'
  149. },
  150. {
  151. value: 'TopOfSearchImpressionShare',
  152. label: '搜索结果顶部展示份额'
  153. },
  154. ]
  155. let barModelValue1 = ref(barOptions1[0].value)
  156. const barOptions2 = [
  157. {
  158. value: 'ACOS',
  159. label: 'ACOS',
  160. },
  161. {
  162. value: 'ROAS',
  163. label: 'ROAS',
  164. },
  165. {
  166. value: 'Spend',
  167. label: '花费',
  168. units: '$',
  169. },
  170. {
  171. value: 'TotalSales',
  172. label: '销售额',
  173. },
  174. {
  175. value: 'TotalPurchases',
  176. label: '订单数',
  177. },
  178. {
  179. value: 'TotalUnitOrdered',
  180. label: '销量',
  181. },
  182. {
  183. value: 'CPC',
  184. label: '点击成本'
  185. },
  186. {
  187. value: 'CPA',
  188. label: '订单成本'
  189. },
  190. {
  191. value: 'Impression',
  192. label: '曝光量',
  193. },
  194. {
  195. value: 'Click',
  196. label: '点击量',
  197. },
  198. {
  199. value: 'qwe',
  200. label: '点击率'
  201. },
  202. {
  203. value: '转化率',
  204. label: '转化率'
  205. },
  206. {
  207. value: 'TotalSalesSameSKU',
  208. label: '推广商品销售额'
  209. },
  210. {
  211. value: 'TotalSalesOtherSKU',
  212. label: '其他商品销售额'
  213. },
  214. {
  215. value: 'TotalPurchasesSameSKU',
  216. label: '推广商品订单数'
  217. },
  218. {
  219. value: 'TotalPurchasesOtherSKU',
  220. label: '其他商品订单数'
  221. },
  222. {
  223. value: 'TotalUnitOrderedSameSKU',
  224. label: '推广商品销量'
  225. },
  226. {
  227. value: 'TotalUnitOrderedOtherSKU',
  228. label: '其他商品销量'
  229. },
  230. {
  231. value: 'TopOfSearchImpressionShare',
  232. label: '搜索结果顶部展示份额'
  233. },
  234. ]
  235. let barModelValue2 = ref(barOptions2[2].value)
  236. onMounted(async () => {
  237. barChart = echarts.init(bar.value)
  238. pieChart = echarts.init(pie.value)
  239. window.addEventListener('resize', resizeChart) // 监听窗口大小变化,调整图表大小
  240. setTimeout(() => {
  241. resizeChart()
  242. }, 0)
  243. await initPieBarData()
  244. initChart()
  245. })
  246. // 获取总数据
  247. let allData = null
  248. async function setAdStructureData() {
  249. allData = await getAdStructureData({ start: dateRange.value[0], end: dateRange.value[1], profile: '3006125408623189' })
  250. console.log('allData.data', allData.data)
  251. return allData.data
  252. }
  253. // 饼图总数据和柱状图总数据
  254. let pieData = null
  255. let barData = null
  256. let pieBarData = null
  257. // 柱状图初始数据
  258. let ACOSList
  259. let SpendList
  260. let ClassificationList
  261. let classificationMapList
  262. async function initPieBarData() {
  263. pieBarData = await setAdStructureData()
  264. pieData = [
  265. { value: pieBarData.pie_data[0].Spend, name: '自动' },
  266. { value: pieBarData.pie_data[1].Spend, name: '手动' },
  267. ]
  268. barData = pieBarData.line_data
  269. // 柱状图初始化数据
  270. ACOSList = barData.map(item => item.ACOS)
  271. SpendList = barData.map(item => item.Spend)
  272. // 将x轴映射为中文
  273. ClassificationList = barData.map(item => item.Classification)
  274. const classificationMap = {
  275. 'BROAD': '关键词-广泛',
  276. 'category': '品类',
  277. 'EXACT': '关键词-精准',
  278. 'asin': '商品',
  279. 'PHRASE': '关键词-词组',
  280. 'close-match': '紧密匹配',
  281. 'loose-match': '广泛匹配',
  282. 'substitutes': '同类商品',
  283. 'complements': '关联商品'
  284. }
  285. classificationMapList = ClassificationList.map(item => classificationMap[item])
  286. loading.value = false
  287. }
  288. // 重置图像
  289. let flag = ref()
  290. let barFlag = ref()
  291. let barFlag2 = ref()
  292. let option
  293. let option2
  294. function changePie(newValue) {
  295. modelValue.value = newValue
  296. flag.value = modelValue.value
  297. option2.series[0].data = [
  298. { value: pieBarData.pie_data[0][flag.value], name: '自动' },
  299. { value: pieBarData.pie_data[1][flag.value], name: '手动' },
  300. ]
  301. // pieChart.clear() // 清除当前饼图的配置
  302. // option2.series[0].data = newPieData // 确保更新的是数组中的正确位置
  303. // pieChart.setOption(option2, true) // 使用 true 强制合并选项
  304. pieChart.setOption(option2)
  305. }
  306. function changeBarOne(newValue) {
  307. barModelValue1.value = newValue
  308. barFlag.value = barModelValue1.value
  309. const barValues = []
  310. try {
  311. for (let i = 0; i < barData.length; i++) {
  312. const value = barData[i][barFlag.value]
  313. barValues.push(value)
  314. }
  315. option.series[0].data = barValues
  316. barChart.setOption(option)
  317. } catch (error) {
  318. console.log(error)
  319. }
  320. }
  321. function changeBarTwo(newValue) {
  322. barModelValue2.value = newValue
  323. barFlag2.value = barModelValue2.value
  324. const barValues = []
  325. try {
  326. for (let i = 0; i < barData.length; i++) {
  327. const value = barData[i][barFlag2.value]
  328. barValues.push(value)
  329. }
  330. option.series[1].data = barValues
  331. barChart.setOption(option)
  332. } catch (error) {
  333. console.log(error)
  334. }
  335. }
  336. // 初始化图表
  337. function initChart() {
  338. // 柱状图配置
  339. option = {
  340. tooltip: {
  341. trigger: 'axis',
  342. axisPointer: {
  343. type: 'shadow',
  344. label: {
  345. backgroundColor: '#6a7985'
  346. }
  347. }
  348. },
  349. // legend: {data: ['数据1', '数据2'],},
  350. toolbox: {
  351. feature: {
  352. saveAsImage: { yAxisIndex: 'none' }
  353. }
  354. },
  355. grid: {
  356. top: 55, right: 60, bottom: 55, left: 55,
  357. },
  358. xAxis: [
  359. {
  360. type: 'category',
  361. boundaryGap: true,
  362. data: classificationMapList,
  363. axisLabel: {
  364. rotate: -30, // 将标签旋转
  365. fontSize: 13
  366. },
  367. },
  368. ],
  369. yAxis: [
  370. {
  371. type: 'value',
  372. // name: '数据1',
  373. // axisLabel: {
  374. // formatter: '{value} %'
  375. // },
  376. axisLine: {
  377. show: true,
  378. lineStyle: {
  379. color: '#3a83f7' // 第一个 Y 轴的颜色
  380. }
  381. }
  382. },
  383. {
  384. type: 'value',
  385. // name: '数据2',
  386. splitLine: {
  387. show: false
  388. },
  389. // axisLabel: {
  390. // formatter: '{value} 单位2'
  391. // },
  392. axisLine: {
  393. show: true,
  394. lineStyle: {
  395. color: '#f19a37' // 第一个 Y 轴的颜色
  396. }
  397. }
  398. }
  399. ],
  400. series: [
  401. {
  402. // name: '数据1',
  403. type: 'bar',
  404. barWidth: '15%',
  405. data: ACOSList,
  406. yAxisIndex: 0,
  407. itemStyle: {
  408. color: new echarts.graphic.LinearGradient(0, 0, 0, 1, [
  409. { offset: 0, color: '#3a83f7' },
  410. { offset: 1, color: 'rgb(111, 209, 206)' },
  411. ]),
  412. // 柱状图圆角
  413. borderRadius: [6, 6, 6, 6],
  414. },
  415. },
  416. {
  417. // name: '数据2',
  418. type: 'bar',
  419. barWidth: '15%',
  420. data: SpendList,
  421. yAxisIndex: 1,
  422. itemStyle: {
  423. color: new echarts.graphic.LinearGradient(0, 0, 0, 1, [
  424. { offset: 0, color: '#f19a37' },
  425. { offset: 1, color: 'rgb(234,207,135)' },
  426. ]),
  427. // 柱状图圆角
  428. borderRadius: [6, 6, 6, 6],
  429. },
  430. },
  431. ],
  432. }
  433. barChart.setOption(option)
  434. // 饼图配置
  435. option2 = {
  436. tooltip: {
  437. show: false,
  438. trigger: 'item',
  439. },
  440. series: [
  441. {
  442. // name: modelValue.value,
  443. type: 'pie',
  444. radius: ['20%', '45%'],
  445. avoidLabelOverlap: false,
  446. itemStyle: {
  447. // borderRadius: 10,
  448. borderWidth: 1, // 设置边框的宽度
  449. borderColor: '#fff', // 将边框颜色设置为白色或图表背景颜色
  450. },
  451. emphasis: {
  452. label: {
  453. show: true,
  454. // fontSize: 40,
  455. fontWeight: 'bold',
  456. }
  457. },
  458. label: {
  459. show: true,
  460. position: 'outside', // 标签显示在外侧
  461. // formatter: `{b}\n{b|${metricMap[modelValue.value]}:{c}}\n{d}%`, // 标签文本格式器
  462. formatter: (params) => {
  463. return params.name + '\n' + '{b|' + metricMap[modelValue.value] + ':}' + '{b|' + params.data.value + '}' + '\n' + params.percent + '%'
  464. },
  465. rich: {
  466. b: {
  467. color: '#4C5058',
  468. fontSize: 15,
  469. fontWeight: 'bold',
  470. lineHeight: 33
  471. },
  472. }
  473. },
  474. labelLine: {
  475. normal: {
  476. show: true
  477. }
  478. },
  479. data: pieData
  480. }
  481. ]
  482. }
  483. pieChart.setOption(option2)
  484. resizeChart()
  485. }
  486. function resizeChart() {
  487. barChart.resize()
  488. pieChart.resize()
  489. }
  490. </script>
  491. <style scoped>
  492. </style>