newPieStatisticsCharts.vue 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647
  1. <template>
  2. <div class="statstics-wrapper">
  3. <div
  4. ref="headerWrapper"
  5. class="statstics-header"
  6. >
  7. <StatisticsHeader
  8. :title="`${chartsTitle}统计`"
  9. :custom-items="customFormItems"
  10. @getFormData="getFormData"
  11. @export="exportHandler"
  12. />
  13. </div>
  14. <div class="statstics-content">
  15. <div
  16. id="chart"
  17. class="statistics-chart"
  18. :style="{ height: chartHeight }"
  19. />
  20. </div>
  21. </div>
  22. </template>
  23. <script>
  24. import StatisticsHeader from './newStatisticsHeader.vue'
  25. import { Query } from '@/api/webApi'
  26. import { mapGetters } from 'vuex'
  27. import * as XLSX from 'xlsx'
  28. import XLSX_STYLE from 'xlsx-style'
  29. import FileSaver from 'file-saver'
  30. export default {
  31. name: 'NewPieStatisticsCharts',
  32. components: { StatisticsHeader },
  33. props: {
  34. chartsTitle: {
  35. type: String,
  36. required: true,
  37. },
  38. querySettings: {
  39. type: Object,
  40. required: true,
  41. },
  42. categories: {
  43. type: Array,
  44. required: true,
  45. },
  46. customFormItems: {
  47. type: Array,
  48. default: () => [],
  49. },
  50. pieTitle: {
  51. type: String,
  52. default: '总件数',
  53. },
  54. },
  55. data() {
  56. return {
  57. myChart: null,
  58. debounceTime: 300,
  59. chartHeight: '70vh',
  60. hasChartData: false,
  61. filters: [],
  62. tableData: [],
  63. params: [],
  64. totalCount: [{ value: 0 }],
  65. categoryDatas: [],
  66. categoryKey: 'specialnum',
  67. baseKey: 'special',
  68. options: {
  69. backgroundColor: '#ffffff',
  70. tooltip: {
  71. trigger: 'item',
  72. },
  73. title: {
  74. text: '',
  75. // 副标题
  76. subtext: '0',
  77. // 主副标题间距
  78. itemGap: 24,
  79. x: 'left',
  80. y: 'center',
  81. left: '30%',
  82. top: '40%',
  83. textAlign: 'center',
  84. // 主标题样式
  85. textStyle: {
  86. fontSize: '48',
  87. color: '#ffffff',
  88. fontWeight: 'bold',
  89. fontFamily: 'Microsoft YaHei',
  90. },
  91. // 副标题样式
  92. subtextStyle: {
  93. fontSize: '80',
  94. color: '#ffffff',
  95. fontWeight: 'bold',
  96. },
  97. },
  98. legend: {
  99. show: true,
  100. left: '60%',
  101. x: 'right',
  102. y: 'center',
  103. icon: 'rect',
  104. itemWidth: 20,
  105. itemHeight: 20,
  106. formatter: name => this.legendFormatter(name),
  107. textStyle: {
  108. backgroundColor: 'transparent',
  109. lineHeight: 0,
  110. rich: {
  111. chartsTitle: {
  112. width: 200,
  113. lineHeight: 100,
  114. fontSize: 32,
  115. fontFamily: 'Microsoft YaHei',
  116. fontWeight: 'bold',
  117. color: '#101116',
  118. padding: [0, 1000, 0, -20],
  119. },
  120. name: {
  121. fontSize: 20,
  122. fontFamily: 'Microsoft YaHei',
  123. fontWeight: 'bold',
  124. color: '#101116',
  125. lineHeight: 100,
  126. },
  127. label: {
  128. fontSize: 16,
  129. fontFamily: 'Microsoft YaHei',
  130. color: '#101116',
  131. },
  132. value: {
  133. width: 96,
  134. fontSize: 16,
  135. fontFamily: 'Helvetica',
  136. fontWeight: 'bold',
  137. color: '#101116',
  138. },
  139. ratio: {
  140. width: 80,
  141. fontSize: 16,
  142. fontFamily: 'Helvetica',
  143. fontWeight: 'bold',
  144. color: '#101116',
  145. },
  146. wrap: {
  147. padding: [0, 40, 0, 0],
  148. },
  149. },
  150. },
  151. selected: {
  152. [this.chartsTitle]: false,
  153. },
  154. },
  155. series: [
  156. {
  157. name: '',
  158. type: 'pie',
  159. left: '30%',
  160. width: 560,
  161. height: 560,
  162. center: [0, '60%'],
  163. radius: ['60%', '90%'],
  164. avoidLabelOverlap: false,
  165. label: {
  166. show: false,
  167. position: 'center',
  168. },
  169. emphasis: {
  170. label: {
  171. show: false,
  172. fontSize: '40',
  173. fontWeight: 'bold',
  174. },
  175. },
  176. labelLine: {
  177. show: false,
  178. },
  179. data: [],
  180. },
  181. {
  182. name: '总数',
  183. type: 'pie',
  184. left: '30%',
  185. width: 560,
  186. height: 560,
  187. center: [0, '60%'],
  188. radius: ['0%', '50%'],
  189. avoidLabelOverlap: false,
  190. itemStyle: {
  191. normal: {
  192. color: '#101116',
  193. },
  194. },
  195. label: {
  196. show: false,
  197. position: 'center',
  198. },
  199. // 自定义中心内容的话需要把这个关闭
  200. emphasis: {
  201. label: {
  202. show: false,
  203. },
  204. },
  205. labelLine: {
  206. show: false,
  207. },
  208. data: [],
  209. },
  210. ],
  211. },
  212. }
  213. },
  214. computed: {
  215. ...mapGetters(['sidebar']),
  216. seriesKey() {
  217. const filterMap = {
  218. 全部: '',
  219. 进港: '_in',
  220. 离港: '_out',
  221. }
  222. return `${this.baseKey}${this.filters
  223. .map(filter => filterMap[filter])
  224. .join('')}`
  225. },
  226. },
  227. watch: {
  228. pieTitle: {
  229. handler(val) {
  230. this.options.title.text = val
  231. },
  232. immediate: true,
  233. },
  234. // 监听数据变化 重绘图形
  235. options: {
  236. handler(obj) {
  237. this.myChart.setOption(obj)
  238. this.resizeHandler()
  239. },
  240. deep: true,
  241. },
  242. categories: {
  243. handler(arr) {
  244. this.categoryDatas = arr.map(categoryName => ({
  245. name: categoryName,
  246. value: 0,
  247. }))
  248. this.categoryDatas.unshift({
  249. name: this.chartsTitle,
  250. value: null,
  251. })
  252. },
  253. deep: true,
  254. immediate: true,
  255. },
  256. querySettings: {
  257. handler({ categoryKey, seriesKey }) {
  258. if (seriesKey) {
  259. this.baseKey = seriesKey
  260. }
  261. if (categoryKey) {
  262. this.categoryKey = categoryKey
  263. }
  264. },
  265. deep: true,
  266. immediate: true,
  267. },
  268. 'sidebar.expand'() {
  269. this.setChartHeight()
  270. },
  271. },
  272. mounted() {
  273. this.setChartHeight()
  274. this.myChart = this.$echarts.init(document.getElementById('chart'))
  275. this.options.series[0].data = this.categoryDatas
  276. this.options.legend.data = this.categoryDatas.map(({ name }, index) => {
  277. if (index === 0) {
  278. return {
  279. name,
  280. icon: 'none',
  281. }
  282. } else {
  283. return {
  284. name,
  285. }
  286. }
  287. })
  288. this.options.series[1].data = this.totalCount
  289. this.myChart.setOption(this.options)
  290. this.myChart.on('legendselectchanged', ({ name }) => {
  291. if (name === this.chartsTitle) {
  292. this.myChart.dispatchAction({
  293. type: 'legendUnSelect',
  294. name,
  295. })
  296. }
  297. })
  298. // 监听页面缩放
  299. this.debouncedChartHeightSetter = this._.debounce(
  300. this.setChartHeight,
  301. this.debounceTime
  302. )
  303. window.addEventListener('resize', this.debouncedChartHeightSetter)
  304. },
  305. beforeDestroy() {
  306. // 销毁实例和移除监听
  307. window.removeEventListener('resize', this.debouncedChartHeightSetter)
  308. if (this.myChart) {
  309. this.myChart.dispose()
  310. this.myChart = null
  311. }
  312. },
  313. methods: {
  314. legendFormatter(name) {
  315. const index = this.categoryDatas.findIndex(
  316. category => category.name === name
  317. )
  318. if (index === 0) {
  319. return `{chartsTitle|${name}}`
  320. } else {
  321. const value = this.categoryDatas[index].value
  322. const ratio =
  323. value && this.totalCount.value
  324. ? ((value / this.totalCount.value) * 100).toFixed(2) + '%'
  325. : '0%'
  326. const richString = `{name|${name}}\n{label|数量:}{value|${value}}{label|占比:}{ratio|${ratio}}`
  327. return index % 2 ? richString + '{wrap| }' : richString
  328. }
  329. },
  330. resetDatas() {
  331. this.hasChartData = false
  332. this.categoryDatas.forEach(category => {
  333. category && (category.value = 0)
  334. })
  335. this.options.title.subtext = '0'
  336. this.options.series[1].data[0].value = 0
  337. },
  338. getFormData(formData) {
  339. this.resetDatas()
  340. let serviceId
  341. let dataContentList = []
  342. const dataContent = {
  343. IATA: 'CA',
  344. td: formData.interval,
  345. fd1: formData.dateTime[0],
  346. fd2: formData.dateTime[1],
  347. }
  348. switch (formData.range) {
  349. case '全部':
  350. serviceId = SERVICE_ID.flightClassificationAll
  351. dataContentList = [dataContent]
  352. break
  353. case '基地分公司':
  354. serviceId = SERVICE_ID.flightClassificationAll
  355. dataContentList = [
  356. {
  357. ...dataContent,
  358. IATA: formData.area,
  359. },
  360. ]
  361. break
  362. case '航线':
  363. serviceId = SERVICE_ID.flightClassificationByAirline
  364. dataContentList =
  365. formData.airline instanceof Array
  366. ? formData.airline.map(airline => ({
  367. ...dataContent,
  368. air_line: airline,
  369. }))
  370. : [
  371. {
  372. ...dataContent,
  373. air_line: formData.airline,
  374. },
  375. ]
  376. break
  377. case '航站':
  378. serviceId = SERVICE_ID.flightClassificationByAirport
  379. dataContentList =
  380. formData.airport instanceof Array
  381. ? formData.airport.map(airport => ({
  382. ...dataContent,
  383. airport,
  384. }))
  385. : [
  386. {
  387. ...dataContent,
  388. airport: formData.airport,
  389. },
  390. ]
  391. break
  392. case '航站楼':
  393. serviceId = SERVICE_ID.flightClassificationByTerminal
  394. dataContentList = [
  395. {
  396. ...dataContent,
  397. airport: formData.terminal.split('-')[0],
  398. terminal: formData.terminal.split('-')[1],
  399. },
  400. ]
  401. break
  402. default:
  403. return
  404. }
  405. this.filters = this.querySettings.filters || []
  406. this.filters.push(formData.inOrOut)
  407. const rangeMap = {
  408. 航线: 'airline',
  409. 基地分公司: 'area',
  410. 航站: 'airport',
  411. 航站楼: 'terminal',
  412. }
  413. this.params = [
  414. formData.interval,
  415. formData.range,
  416. formData[rangeMap[formData.range]],
  417. formData.inOrOut,
  418. formData.dateTime[0],
  419. formData.dateTime[1],
  420. ...this.filters,
  421. ]
  422. this.getMultipleChartsData(serviceId, dataContentList)
  423. },
  424. async getMultipleChartsData(serviceId, dataContentList) {
  425. try {
  426. const listValuesArray = await Promise.all(
  427. dataContentList.map(dataContent =>
  428. this.getChartsData(serviceId, dataContent)
  429. )
  430. )
  431. const listValues = listValuesArray.reduce(
  432. (preValues, currentValues) => {
  433. currentValues.forEach(value => {
  434. const preValue = preValues.find(
  435. preValue =>
  436. preValue.fd === value.fd &&
  437. preValue[this.categoryKey] === value[this.categoryKey]
  438. )
  439. if (preValue) {
  440. preValue[this.seriesKey] += value[this.seriesKey]
  441. } else {
  442. preValues.push({
  443. fd: value.fd,
  444. [this.categoryKey]: value[this.categoryKey],
  445. [this.seriesKey]: value[this.seriesKey],
  446. })
  447. }
  448. })
  449. return preValues
  450. },
  451. []
  452. )
  453. this.setChartsData(listValues)
  454. this.tableData = listValuesArray
  455. .map(list =>
  456. this._.sortBy(list, 'fd', o =>
  457. this.categories.indexOf(o.flight_attr)
  458. )
  459. )
  460. .flat()
  461. } catch (error) {
  462. this.$message.error(error.message)
  463. }
  464. },
  465. async getChartsData(serviceId, dataContent) {
  466. try {
  467. const { code, returnData, message } = await Query({
  468. serviceId,
  469. dataContent,
  470. event: '0',
  471. })
  472. if (String(code) === '0') {
  473. return returnData
  474. } else {
  475. return Promise.reject(message || '失败')
  476. }
  477. } catch (error) {
  478. return Promise.reject(error.message || '失败')
  479. }
  480. },
  481. setChartsData(listValues) {
  482. if (listValues.length === 0) {
  483. this.$message.info('未查询到对应数据')
  484. return
  485. }
  486. let totalCount = 0
  487. listValues.forEach(element => {
  488. this.categoryDatas.forEach(category => {
  489. if (element[this.categoryKey]?.includes(category.name)) {
  490. category.value += element[this.seriesKey]
  491. }
  492. })
  493. totalCount += element[this.seriesKey]
  494. })
  495. this.options.title.subtext = totalCount.toString()
  496. this.totalCount.value = totalCount
  497. this.hasChartData = true
  498. },
  499. setChartHeight() {
  500. const topBarHeight = 80
  501. const headerBlankHeight = 24
  502. const tabsWrapperHeight = 62
  503. const headerHeight = this.$refs['headerWrapper'].offsetHeight
  504. const footerBlankHeight = 24
  505. this.chartHeight = `calc(100vh - ${
  506. topBarHeight +
  507. headerBlankHeight +
  508. tabsWrapperHeight +
  509. headerHeight +
  510. footerBlankHeight
  511. }px)`
  512. this.$nextTick(() => {
  513. this.resizeHandler()
  514. })
  515. },
  516. resizeHandler() {
  517. if (this.myChart) {
  518. this.myChart.resize()
  519. }
  520. },
  521. exportHandler() {
  522. if (!this.hasChartData) {
  523. this.$message.warning('请查询后再进行导出')
  524. return
  525. }
  526. // const myCanvas = this.myChart._dom.querySelectorAll('canvas')[0]
  527. // const image = myCanvas.toDataURL('image/png')
  528. // const $a = document.createElement('a')
  529. // $a.setAttribute('href', image)
  530. // $a.setAttribute('download', `${this.chartsTitle}统计.png`)
  531. // $a.click()
  532. const xlsxDatas = [['时间', '位置', '分类', '数量', '占比']]
  533. xlsxDatas.push(
  534. ...this.tableData.map(element => [
  535. element['fd'],
  536. element['location'],
  537. element[this.categoryKey],
  538. element[this.seriesKey],
  539. ((element[this.seriesKey] / this.totalCount.value) * 100).toFixed(2) +
  540. '%',
  541. ])
  542. )
  543. xlsxDatas.push(['合计', '', '', this.totalCount.value, ''])
  544. // 计算列宽
  545. const columnWidths = []
  546. xlsxDatas.forEach((row, rowIndex) => {
  547. // 计算每一列宽度,考虑换行
  548. row.forEach((cell, columnIndex) => {
  549. const cellWidth = Math.max(
  550. ...cell
  551. .toString()
  552. .split('\n')
  553. .map(cellRow =>
  554. cellRow.split('').reduce((pre, curr) => {
  555. const letterSize = curr.charCodeAt(0) > 255 ? 2 : 1
  556. return pre + letterSize
  557. }, 0)
  558. )
  559. )
  560. if (
  561. (!columnWidths[columnIndex] && cellWidth > 0) ||
  562. cellWidth > columnWidths[columnIndex]
  563. ) {
  564. columnWidths[columnIndex] = cellWidth
  565. }
  566. })
  567. })
  568. // 生成表格
  569. const sheet = XLSX.utils.aoa_to_sheet(xlsxDatas)
  570. // 添加列宽度
  571. sheet['!cols'] = columnWidths.map(width => ({
  572. wch: width + 2,
  573. }))
  574. // 样式
  575. const borderStyle = {
  576. style: 'medium',
  577. color: {
  578. rgb: 'FFFFFF',
  579. },
  580. }
  581. const reg = /^[A-Z]+([\d]+$)/
  582. for (const key in sheet) {
  583. const match = reg.test(key)
  584. if (match) {
  585. const rowIndex = reg.exec(key)[1]
  586. let cellStyle = {
  587. alignment: {
  588. horizontal: 'center',
  589. vertical: 'center',
  590. wrapText: true,
  591. },
  592. }
  593. if (Number(rowIndex) === 1) {
  594. cellStyle = {
  595. ...cellStyle,
  596. border: {
  597. top: borderStyle,
  598. right: borderStyle,
  599. bottom: borderStyle,
  600. left: borderStyle,
  601. },
  602. font: {
  603. color: {
  604. rgb: 'FFFFFF',
  605. },
  606. },
  607. fill: {
  608. fgColor: {
  609. rgb: '3366FF',
  610. },
  611. },
  612. }
  613. } else {
  614. cellStyle.alignment.horizontal = 'left'
  615. }
  616. sheet[key].s = cellStyle
  617. }
  618. }
  619. // 表格数据转换
  620. const workBook = XLSX.utils.book_new()
  621. XLSX.utils.book_append_sheet(workBook, sheet, this.chartsTitle)
  622. const tableWrite = XLSX_STYLE.write(workBook, {
  623. bookType: 'xlsx',
  624. bookSST: true,
  625. type: 'buffer',
  626. cellStyles: true,
  627. })
  628. // 下载表格
  629. const fileName = `${this.chartsTitle}统计-${this.params.join('-')}.xlsx`
  630. FileSaver.saveAs(
  631. new Blob([tableWrite], { type: 'application/octet-stream' }),
  632. fileName
  633. )
  634. },
  635. },
  636. }
  637. </script>
  638. <style lang="scss" scoped>
  639. .statistics-chart {
  640. width: 100%;
  641. }
  642. </style>