newPieStatisticsCharts.vue 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589
  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. // console.log(listValues)
  390. this.setChartsData(this._.sortBy(listValues, 'fdt'))
  391. } catch (error) {
  392. this.$message.error(error.message)
  393. }
  394. this.loading = false
  395. },
  396. async getChartsData(serviceId, params) {
  397. try {
  398. const {
  399. code,
  400. returnData: listValues,
  401. message,
  402. } = await TempQuery({
  403. serviceId,
  404. dataContent: params,
  405. })
  406. if (String(code) === '0') {
  407. return listValues.map(obj => ({
  408. ...obj,
  409. location: params.air_line || params.airport || '',
  410. }))
  411. } else {
  412. throw new Error(message)
  413. }
  414. } catch (error) {
  415. return Promise.reject(error.message || '失败')
  416. }
  417. },
  418. setChartsData(listValues) {
  419. if (listValues.length === 0) {
  420. this.$message.info('未查询到对应数据')
  421. return
  422. }
  423. let totalCount = 0
  424. listValues.forEach(element => {
  425. this.categoryDatas.forEach(category => {
  426. if (element[category.key]) {
  427. category.value += element[category.key]
  428. totalCount += element[category.key]
  429. }
  430. })
  431. })
  432. this.options.title.subtext = totalCount.toString()
  433. this.totalCount.value = totalCount
  434. this.hasChartData = true
  435. },
  436. setChartHeight() {
  437. const topBarHeight = 80
  438. const headerBlankHeight = 24
  439. const tabsWrapperHeight = 62
  440. const headerHeight = this.$refs['headerWrapper'].offsetHeight
  441. const footerBlankHeight = 24
  442. this.chartHeight = `calc(100vh - ${
  443. topBarHeight +
  444. headerBlankHeight +
  445. tabsWrapperHeight +
  446. headerHeight +
  447. footerBlankHeight
  448. }px)`
  449. this.$nextTick(() => {
  450. this.resizeHandler()
  451. })
  452. },
  453. resizeHandler() {
  454. if (this.myChart) {
  455. this.myChart.resize()
  456. }
  457. },
  458. exportHandler() {
  459. if (!this.hasChartData) {
  460. this.$message.warning('请查询后再进行导出')
  461. return
  462. }
  463. // const myCanvas = this.myChart._dom.querySelectorAll('canvas')[0]
  464. // const image = myCanvas.toDataURL('image/png')
  465. // const $a = document.createElement('a')
  466. // $a.setAttribute('href', image)
  467. // $a.setAttribute('download', `${this.chartsTitle}统计.png`)
  468. // $a.click()
  469. const xlsxDatas = [['时间', '位置', '分类', '数量', '占比']]
  470. xlsxDatas.push(
  471. ...this.tableData
  472. .map(element =>
  473. this.categoryDatas.map(category => [
  474. element['fdt'],
  475. element['location'],
  476. category.name,
  477. element[category.key],
  478. `${(
  479. (element[category.key] / this.totalCount.value) *
  480. 100
  481. ).toFixed(2)}%`,
  482. ])
  483. )
  484. .flat(1)
  485. )
  486. xlsxDatas.push(['合计', '', '', this.totalCount.value, ''])
  487. // 计算列宽
  488. const columnWidths = []
  489. xlsxDatas.forEach((row, rowIndex) => {
  490. // 计算每一列宽度,考虑换行
  491. row.forEach((cell, columnIndex) => {
  492. const cellWidth = Math.max(
  493. ...cell
  494. .toString()
  495. .split('\n')
  496. .map(cellRow =>
  497. cellRow.split('').reduce((pre, curr) => {
  498. const letterSize = curr.charCodeAt(0) > 255 ? 2 : 1
  499. return pre + letterSize
  500. }, 0)
  501. )
  502. )
  503. if (
  504. (!columnWidths[columnIndex] && cellWidth > 0) ||
  505. cellWidth > columnWidths[columnIndex]
  506. ) {
  507. columnWidths[columnIndex] = cellWidth
  508. }
  509. })
  510. })
  511. // 生成表格
  512. const sheet = XLSX.utils.aoa_to_sheet(xlsxDatas)
  513. // 添加列宽度
  514. sheet['!cols'] = columnWidths.map(width => ({
  515. wch: width + 2,
  516. }))
  517. // 样式
  518. const borderStyle = {
  519. style: 'medium',
  520. color: {
  521. rgb: 'FFFFFF',
  522. },
  523. }
  524. const reg = /^[A-Z]+([\d]+$)/
  525. for (const key in sheet) {
  526. const match = reg.test(key)
  527. if (match) {
  528. const rowIndex = reg.exec(key)[1]
  529. let cellStyle = {
  530. alignment: {
  531. horizontal: 'center',
  532. vertical: 'center',
  533. wrapText: true,
  534. },
  535. }
  536. if (Number(rowIndex) === 1) {
  537. cellStyle = {
  538. ...cellStyle,
  539. border: {
  540. top: borderStyle,
  541. right: borderStyle,
  542. bottom: borderStyle,
  543. left: borderStyle,
  544. },
  545. font: {
  546. color: {
  547. rgb: 'FFFFFF',
  548. },
  549. },
  550. fill: {
  551. fgColor: {
  552. rgb: '3366FF',
  553. },
  554. },
  555. }
  556. } else {
  557. cellStyle.alignment.horizontal = 'left'
  558. }
  559. sheet[key].s = cellStyle
  560. }
  561. }
  562. // 表格数据转换
  563. const workBook = XLSX.utils.book_new()
  564. XLSX.utils.book_append_sheet(workBook, sheet, this.chartsTitle)
  565. const tableWrite = XLSX_STYLE.write(workBook, {
  566. bookType: 'xlsx',
  567. bookSST: true,
  568. type: 'buffer',
  569. cellStyles: true,
  570. })
  571. // 下载表格
  572. const fileName = `${this.chartsTitle}统计-${this.params.join('-')}.xlsx`
  573. FileSaver.saveAs(
  574. new Blob([tableWrite], { type: 'application/octet-stream' }),
  575. fileName
  576. )
  577. },
  578. },
  579. }
  580. </script>
  581. <style lang="scss" scoped>
  582. .statistics-chart {
  583. width: 100%;
  584. }
  585. </style>