newPieStatisticsCharts.vue 15 KB

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