commonBarStatisticsCharts.vue 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603
  1. <template>
  2. <div
  3. v-loading="loading"
  4. element-loading-text="拼命加载中"
  5. element-loading-spinner="el-icon-loading"
  6. element-loading-background="rgba(0, 0, 0, 0.8)"
  7. class="statstics-wrapper"
  8. >
  9. <div
  10. ref="headerWrapper"
  11. class="statstics-header"
  12. >
  13. <StatisticsHeader
  14. :title="`${chartsTitle}统计`"
  15. :custom-items="customFormItems"
  16. :items="formItems"
  17. :data.sync="formData"
  18. @getFormData="getFormData"
  19. @export="exportHandler"
  20. />
  21. </div>
  22. <div class="statstics-content">
  23. <div
  24. id="chart"
  25. class="statistics-chart"
  26. :style="{ height: chartHeight }"
  27. />
  28. </div>
  29. </div>
  30. </template>
  31. <script>
  32. import StatisticsHeader from './statisticsHeader.vue'
  33. import { TempQuery } from '@/api/temp'
  34. import { mapGetters } from 'vuex'
  35. import * as XLSX from 'xlsx'
  36. import XLSX_STYLE from 'xlsx-style'
  37. import FileSaver from 'file-saver'
  38. export default {
  39. name: 'CommonBarStatisticsCharts',
  40. components: { StatisticsHeader },
  41. props: {
  42. chartsTitle: {
  43. type: String,
  44. required: true,
  45. },
  46. querySettings: {
  47. type: Object,
  48. required: true,
  49. },
  50. customFormItems: {
  51. type: Array,
  52. default: () => [],
  53. },
  54. formItems: {
  55. type: Array,
  56. },
  57. formData: {
  58. type: Object,
  59. required: true,
  60. },
  61. },
  62. data() {
  63. return {
  64. loading: false,
  65. myChart: null,
  66. debounceTime: 300,
  67. chartHeight: '70vh',
  68. hasChartData: false,
  69. seriesKey: 'seriesData',
  70. xAxisKey: 'flight_date',
  71. filters: [],
  72. tableData: [],
  73. params: [],
  74. options: {
  75. backgroundColor: '#fff',
  76. tooltip: {
  77. trigger: 'axis',
  78. axisPointer: {
  79. type: 'cross',
  80. crossStyle: {
  81. color: '#999',
  82. },
  83. },
  84. },
  85. legend: {
  86. top: '5%',
  87. right: '5%',
  88. icon: 'rect',
  89. height: 14,
  90. itemWidth: 14,
  91. itemHeight: 14,
  92. itemGap: 30,
  93. data: [
  94. this.chartsTitle.replace('量', '数量'),
  95. // `${this.chartsTitle}量同比`,
  96. `${this.chartsTitle}环比`,
  97. ],
  98. textStyle: {
  99. fontFamily: 'Helvetica, "Microsoft YaHei"',
  100. color: '#101116',
  101. },
  102. },
  103. grid: {
  104. top: '15%',
  105. left: '5%',
  106. right: '5%',
  107. bottom: '5%',
  108. },
  109. xAxis: {
  110. data: [],
  111. axisLine: {
  112. show: true,
  113. lineStyle: {
  114. color: '#000000',
  115. },
  116. },
  117. axisTick: {
  118. show: false, // 隐藏X轴刻度
  119. },
  120. axisLabel: {
  121. fontFamily: 'Helvetica, "Microsoft YaHei"',
  122. color: '#101116',
  123. },
  124. axisPointer: {
  125. type: 'shadow',
  126. },
  127. },
  128. yAxis: [
  129. {
  130. min: 0,
  131. max: 60000,
  132. splitLine: {
  133. lineStyle: {
  134. type: 'dashed',
  135. color: '#B0B3C3',
  136. opacity: 0.5,
  137. },
  138. },
  139. axisPointer: {
  140. label: {
  141. formatter: ({ value }) => value.toFixed(),
  142. },
  143. },
  144. axisLabel: {
  145. fontFamily: 'Helvetica, "Microsoft YaHei"',
  146. color: '#101116',
  147. },
  148. },
  149. {
  150. min: -0.3,
  151. max: 0.5,
  152. axisLabel: {
  153. formatter: value => (value * 100).toFixed(2) + '%',
  154. fontFamily: 'Helvetica, "Microsoft YaHei"',
  155. color: '#101116',
  156. },
  157. axisPointer: {
  158. label: {
  159. formatter: ({ value }) => (value * 100).toFixed(2) + '%',
  160. },
  161. },
  162. splitLine: {
  163. show: false,
  164. },
  165. },
  166. ],
  167. series: [
  168. {
  169. name: this.chartsTitle.replace('量', '数量'),
  170. type: 'bar',
  171. z: 2,
  172. itemStyle: {
  173. color: '#6682B5',
  174. },
  175. barWidth: 40,
  176. label: {
  177. show: true,
  178. position: 'top',
  179. },
  180. data: [],
  181. },
  182. {
  183. name: `${this.chartsTitle}同比`,
  184. type: 'line',
  185. z: 4,
  186. yAxisIndex: 1,
  187. symbol: 'circle',
  188. itemStyle: {
  189. color: '#F2B849',
  190. borderColor: '#ffffff',
  191. borderWidth: 4,
  192. },
  193. lineStyle: {
  194. width: 4,
  195. color: '#F2B849',
  196. },
  197. symbolSize: 32,
  198. tooltip: {
  199. valueFormatter: value => (value * 100).toFixed(2) + '%',
  200. },
  201. data: [],
  202. },
  203. {
  204. name: `${this.chartsTitle}环比`,
  205. type: 'line',
  206. z: 3,
  207. yAxisIndex: 1,
  208. symbol: 'circle',
  209. itemStyle: {
  210. color: '#E33D3D',
  211. borderColor: '#ffffff',
  212. borderWidth: 4,
  213. },
  214. lineStyle: {
  215. width: 4,
  216. color: '#E33D3D',
  217. },
  218. symbolSize: 32,
  219. tooltip: {
  220. valueFormatter: value => (value * 100).toFixed(2) + '%',
  221. },
  222. data: [],
  223. },
  224. ],
  225. },
  226. admin: {},
  227. }
  228. },
  229. computed: {
  230. ...mapGetters(['sidebar']),
  231. },
  232. watch: {
  233. // 监听数据变化 重绘图形
  234. options: {
  235. handler(obj) {
  236. this.myChart.setOption(obj)
  237. this.resizeHandler()
  238. },
  239. deep: true,
  240. },
  241. 'sidebar.expand'() {
  242. this.setChartHeight()
  243. },
  244. querySettings: {
  245. handler({ seriesKey, filters, xAxisKey }) {
  246. if (seriesKey) {
  247. this.seriesKey = seriesKey
  248. }
  249. if (xAxisKey) {
  250. this.xAxisKey = xAxisKey
  251. }
  252. if (filters?.length) {
  253. this.filters = filters
  254. }
  255. },
  256. deep: true,
  257. immediate: true,
  258. },
  259. },
  260. mounted() {
  261. this.setChartHeight()
  262. this.myChart = this.$echarts.init(document.getElementById('chart'))
  263. this.myChart.setOption(this.options)
  264. // 监听页面缩放
  265. this.debouncedChartHeightSetter = this._.debounce(
  266. this.setChartHeight,
  267. this.debounceTime
  268. )
  269. window.addEventListener('resize', this.debouncedChartHeightSetter)
  270. },
  271. beforeDestroy() {
  272. // 销毁实例和移除监听
  273. window.removeEventListener('resize', this.debouncedChartHeightSetter)
  274. if (this.myChart) {
  275. this.myChart.dispose()
  276. this.myChart = null
  277. }
  278. },
  279. methods: {
  280. resetDatas() {
  281. this.hasChartData = false
  282. this.options.yAxis[0].max = 60000
  283. this.options.xAxis.data = []
  284. this.options.series[0].data = []
  285. this.options.series[2].data = []
  286. this.options.yAxis[1].min = -0.3
  287. this.options.yAxis[1].max = 0.5
  288. },
  289. getFormData(formData) {
  290. this.resetDatas()
  291. let params = {}
  292. params = JSON.parse(JSON.stringify(formData))
  293. params.fd1 = formData.dateTime[0]
  294. params.fd2 = formData.dateTime[1]
  295. delete params.dateTime
  296. this.admin = JSON.parse(JSON.stringify(params))
  297. delete params.timedim
  298. this.getSingleChartsData(this.querySettings.serviceId, params)
  299. // if (formData.passengerType.length) {
  300. // this.filters = [
  301. // {
  302. // key: formData.passengerType[0],
  303. // value: formData.passengerType[1]
  304. // }
  305. // ]
  306. // }
  307. // this.params = [...params, ...this.filters.map(({ value }) => value)]
  308. // if (params[2] instanceof Array) {
  309. // const paramsList = params[2].map(param => [...params.slice(0, 2), param, ...params.slice(3)])
  310. // this.getMultipleChartsData(id, paramsList)
  311. // } else {
  312. // this.getSingleChartsData(id, params)
  313. // }
  314. },
  315. async getMultipleChartsData(id, paramsList) {
  316. this.loading = true
  317. try {
  318. const listValuesArray = await Promise.all(
  319. paramsList.map(params => this.getChartsData(id, params))
  320. )
  321. const listValues = listValuesArray.reduce(
  322. (preValues, currentValues) => {
  323. currentValues.forEach(value => {
  324. const preValue = preValues.find(
  325. preValue => preValue.A === value.A
  326. )
  327. if (preValue) {
  328. preValue[this.seriesKey] += value[this.seriesKey]
  329. } else {
  330. preValues.push({
  331. A: value.A,
  332. [this.seriesKey]: value[this.seriesKey],
  333. })
  334. }
  335. })
  336. return preValues
  337. },
  338. []
  339. )
  340. this.setChartsData(this._.sortBy(listValues, 'A'))
  341. } catch (error) {
  342. this.$message.error(error.message)
  343. }
  344. this.loading = false
  345. },
  346. async getSingleChartsData(id, params) {
  347. this.loading = true
  348. try {
  349. const listValues = await this.getChartsData(id, params)
  350. this.setChartsData(listValues)
  351. } catch (error) {
  352. this.$message.error(error.message)
  353. }
  354. this.loading = false
  355. },
  356. async getChartsData(serviceId, params) {
  357. try {
  358. const { code, returnData, message } = await TempQuery({
  359. serviceId,
  360. dataContent: [params],
  361. })
  362. if (Number(code) === 0) {
  363. return returnData.listValues || returnData
  364. } else {
  365. return Promise.reject(message || '失败')
  366. }
  367. } catch (error) {
  368. return Promise.reject(error.message || '失败')
  369. }
  370. },
  371. setChartsData(listValues) {
  372. const xAxisData = []
  373. const yAxisData = [0]
  374. const seriesDatas = []
  375. let filteredList = []
  376. if (listValues && listValues.length) {
  377. filteredList = listValues.filter(element =>
  378. this.filters.every((key, value) => {
  379. if (key && value && element[key] !== value) {
  380. return false
  381. } else {
  382. return true
  383. }
  384. })
  385. )
  386. }
  387. if (filteredList.length === 0) {
  388. this.$message.info('未查询到对应数据')
  389. return
  390. }
  391. // console.log(this.admin)
  392. if (this.admin.io === '进港') {
  393. this.seriesKey = 'in_num'
  394. } else if (this.admin.io === '离港') {
  395. this.seriesKey = 'out_num'
  396. } else if (this.admin.io === '中转') {
  397. this.seriesKey = 'trans_num'
  398. } else if (this.admin.timedim === '正常') {
  399. this.seriesKey = 'bag_num'
  400. } else if (this.admin.timedim === '异常') {
  401. this.seriesKey = 'exception_num'
  402. }
  403. for (let i = 0; i < filteredList.length; i++) {
  404. xAxisData.push(filteredList[i][this.xAxisKey])
  405. seriesDatas.push(filteredList[i][this.seriesKey])
  406. if (i > 0) {
  407. if (filteredList[i - 1][this.seriesKey] > 0) {
  408. yAxisData.push(
  409. (filteredList[i][this.seriesKey] -
  410. filteredList[i - 1][this.seriesKey]) /
  411. filteredList[i - 1][this.seriesKey]
  412. )
  413. } else {
  414. yAxisData.push(0)
  415. }
  416. }
  417. }
  418. let max = Math.max(...seriesDatas)
  419. max = Math.ceil(max / 10) * 10
  420. this.options.yAxis[0].max = max
  421. this.options.xAxis.data = xAxisData
  422. this.options.series[0].data = seriesDatas
  423. this.options.series[2].data = yAxisData
  424. this.options.yAxis[1].min = (Math.min(...yAxisData) - 0.1).toFixed(2)
  425. this.options.yAxis[1].max = (Math.max(...yAxisData) + 0.1).toFixed(2)
  426. this.tableData = [xAxisData, seriesDatas, yAxisData]
  427. this.hasChartData = true
  428. },
  429. setChartHeight() {
  430. const topBarHeight = 80
  431. const headerBlankHeight = 24
  432. const tabsWrapperHeight = 62
  433. const headerHeight = this.$refs['headerWrapper'].offsetHeight
  434. const footerBlankHeight = 24
  435. this.chartHeight = `calc(100vh - ${
  436. topBarHeight +
  437. headerBlankHeight +
  438. tabsWrapperHeight +
  439. headerHeight +
  440. footerBlankHeight
  441. }px)`
  442. this.$nextTick(() => {
  443. this.resizeHandler()
  444. })
  445. },
  446. resizeHandler() {
  447. if (this.myChart) {
  448. this.myChart.resize()
  449. }
  450. },
  451. exportHandler() {
  452. if (!this.hasChartData) {
  453. this.$message.warning('请查询后再进行导出')
  454. return
  455. }
  456. // const myCanvas = this.myChart._dom.querySelectorAll('canvas')[0]
  457. // const image = myCanvas.toDataURL('image/png')
  458. // const $a = document.createElement('a')
  459. // $a.setAttribute('href', image)
  460. // $a.setAttribute('download', `${this.chartsTitle}统计.png`)
  461. // $a.click()
  462. // 生成表格数据
  463. const xlsxDatas = [
  464. [
  465. '时间',
  466. this.chartsTitle.replace('量', '数量'),
  467. `${this.chartsTitle}环比`,
  468. ],
  469. ]
  470. const transposition = this.tableData[0].map((col, colIndex) => {
  471. return this.tableData.map((row, rowIndex) => {
  472. return rowIndex === 2
  473. ? (row[colIndex] * 100).toFixed(2) + '%'
  474. : row[colIndex]
  475. })
  476. })
  477. xlsxDatas.push(...transposition)
  478. // 添加合计行
  479. if (xlsxDatas.length > 2) {
  480. const summaryRow = ['合计']
  481. const colNum = xlsxDatas[0].length
  482. for (let colIndex = 1; colIndex < colNum; colIndex++) {
  483. summaryRow[colIndex] = xlsxDatas.reduce(
  484. (pre, currentRow, rowIndex) => {
  485. if (colIndex === 1) {
  486. if (rowIndex === 0) {
  487. return 0
  488. } else {
  489. return pre + currentRow[colIndex]
  490. }
  491. } else {
  492. return pre
  493. }
  494. },
  495. ''
  496. )
  497. }
  498. xlsxDatas.push(summaryRow)
  499. }
  500. // 计算列宽
  501. const columnWidths = []
  502. xlsxDatas.forEach((row, rowIndex) => {
  503. // 计算每一列宽度,考虑换行
  504. row.forEach((cell, columnIndex) => {
  505. const cellWidth = Math.max(
  506. ...cell
  507. .toString()
  508. .split('\n')
  509. .map(cellRow =>
  510. cellRow.split('').reduce((pre, curr) => {
  511. const letterSize = curr.charCodeAt(0) > 255 ? 2 : 1
  512. return pre + letterSize
  513. }, 0)
  514. )
  515. )
  516. if (
  517. (!columnWidths[columnIndex] && cellWidth > 0) ||
  518. cellWidth > columnWidths[columnIndex]
  519. ) {
  520. columnWidths[columnIndex] = cellWidth
  521. }
  522. })
  523. })
  524. // 生成表格
  525. const sheet = XLSX.utils.aoa_to_sheet(xlsxDatas)
  526. // 添加列宽度
  527. sheet['!cols'] = columnWidths.map(width => ({
  528. wch: width + 2,
  529. }))
  530. // 样式
  531. const borderStyle = {
  532. style: 'medium',
  533. color: {
  534. rgb: 'FFFFFF',
  535. },
  536. }
  537. const reg = /^[A-Z]+([\d]+$)/
  538. for (const key in sheet) {
  539. const match = reg.test(key)
  540. if (match) {
  541. const rowIndex = reg.exec(key)[1]
  542. let cellStyle = {
  543. alignment: {
  544. horizontal: 'center',
  545. vertical: 'center',
  546. wrapText: true,
  547. },
  548. }
  549. if (Number(rowIndex) === 1) {
  550. cellStyle = {
  551. ...cellStyle,
  552. border: {
  553. top: borderStyle,
  554. right: borderStyle,
  555. bottom: borderStyle,
  556. left: borderStyle,
  557. },
  558. font: {
  559. color: {
  560. rgb: 'FFFFFF',
  561. },
  562. },
  563. fill: {
  564. fgColor: {
  565. rgb: '3366FF',
  566. },
  567. },
  568. }
  569. } else {
  570. cellStyle.alignment.horizontal = 'left'
  571. }
  572. sheet[key].s = cellStyle
  573. }
  574. }
  575. // 表格数据转换
  576. const workBook = XLSX.utils.book_new()
  577. XLSX.utils.book_append_sheet(workBook, sheet, this.chartsTitle)
  578. const tableWrite = XLSX_STYLE.write(workBook, {
  579. bookType: 'xlsx',
  580. bookSST: true,
  581. type: 'buffer',
  582. cellStyles: true,
  583. })
  584. // 下载表格
  585. const fileName = `${this.chartsTitle}统计-${this.params.join('-')}.xlsx`
  586. FileSaver.saveAs(
  587. new Blob([tableWrite], { type: 'application/octet-stream' }),
  588. fileName
  589. )
  590. },
  591. },
  592. }
  593. </script>
  594. <style lang="scss" scoped>
  595. .statistics-chart {
  596. width: 100%;
  597. }
  598. </style>