index.vue 38 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043
  1. <template>
  2. <div class="dashboard">
  3. <div class="dashboard_top">
  4. <div class="dashboard_top_all">
  5. <select-box ref="checkAlls" defaultValue="全国" @selectChange="selectChange" :options="checkAlls" />
  6. </div>
  7. <div class="dashboard_top_title">
  8. <div class="dashboard_top_title_txt">行李全流程跟踪监管系统</div>
  9. <div class="shade shade_1"></div>
  10. <div class="shade shade_2"></div>
  11. <div class="shade shade_3"></div>
  12. </div>
  13. <div class="dashboard_top_times">
  14. <el-date-picker @change="timesChange" style="margin-left: 10px;margin-top: 8px;" v-model="dateTimes" size="small" type="daterange" value-format="yyyy-MM-dd" range-separator="至" start-placeholder="开始日期" end-placeholder="结束日期" />
  15. </div>
  16. </div>
  17. <div class="dashboard_bottom">
  18. <div class="dashboard_bottom_left">
  19. <dv-border-box-7 :color="colors" v-loading="airportAccessLoading" element-loading-text="拼命加载中" class="dashboard_bottom_left_box">
  20. <div class="box_title">机场接入数据量排行</div>
  21. <div class="box_content">
  22. <select-box ref="checkAccess" class="box_content_select" size="mini" defaultValue="总数据条数" @selectChange="checkAccessChange" :options="checkAccess" />
  23. <bar-charts id="airportAccessCharts" :option="airportAccess" />
  24. </div>
  25. </dv-border-box-7>
  26. <dv-border-box-7 :color="colors" v-loading="airportLuggageLoading" element-loading-text="拼命加载中" class="dashboard_bottom_left_box">
  27. <div class="box_title">机场行李运力排行</div>
  28. <div class="box_content">
  29. <select-box ref="checkLuggage" class="box_content_select" size="mini" defaultValue="总行李量" @selectChange="checkLuggageChange" :options="checkLuggage" />
  30. <bar-charts id="airportLuggageCharts" :option="airportLuggage" />
  31. </div>
  32. </dv-border-box-7>
  33. <dv-border-box-7 :color="colors" v-loading="airportNodesLoading" element-loading-text="拼命加载中" class="dashboard_bottom_left_box">
  34. <div class="box_title">机场节点覆盖率排行</div>
  35. <div class="box_content">
  36. <select-box ref="checkNodes" class="box_content_select" size="mini" defaultValue="基本节点" @selectChange="checkNodesChange" :options="checkNodes" />
  37. <bar-charts id="airportNodeCharts" :option="airportNodes" />
  38. </div>
  39. </dv-border-box-7>
  40. <dv-border-box-7 :color="colors" v-loading="airlinePlotLoading" class="dashboard_bottom_left_box thrds3">
  41. <div class="thrds3_tabs">
  42. <div v-for="(item,index) in thrdsItems" :key="index" @click="thrdsChange(index)" :class="thrdsIndex == index ? 'active' : ''" class="thrds3_tabs_list">{{item}}</div>
  43. </div>
  44. <div class="box_content">
  45. <select-box ref="airlinePlot" class="box_content_select" style="top:10px;" size="mini" :defaultValue="airlineValue" @selectChange="checkAirlineChange" :options="airlineChecks" />
  46. <bar-charts id="airlinePlotCharts" :option="airlinePlot" />
  47. </div>
  48. </dv-border-box-7>
  49. </div>
  50. <div class="dashboard_bottom_center">
  51. <dv-border-box-7 :color="colors" v-loading="mapOverviewLoading" element-loading-text="拼命加载中" class="dashboard_bottom_center_box1">
  52. <div class="box_title">机场行李跟踪系统建设总贤</div>
  53. <div class="mapOverview">
  54. <div class="mapOverview_list">
  55. <span class="mapOverview_list_icon icon1"></span>
  56. <span class="mapOverview_list_txt">已建 {{mapOverviewItem.build}}家</span>
  57. </div>
  58. <div class="mapOverview_list">
  59. <span class="mapOverview_list_icon icon2"></span>
  60. <span class="mapOverview_list_txt">在建 {{mapOverviewItem.building}}家</span>
  61. </div>
  62. <div class="mapOverview_list">
  63. <span class="mapOverview_list_icon icon3"></span>
  64. <span class="mapOverview_list_txt">未建 {{mapOverviewItem.not_build}}家</span>
  65. </div>
  66. </div>
  67. <div class="box_content">
  68. <map-charts id="mapCharts" :option="mapOverview" />
  69. </div>
  70. </dv-border-box-7>
  71. <dv-border-box-7 :color="colors" v-loading="scatterPlotLoading" element-loading-text="拼命加载中" class="dashboard_bottom_center_box2">
  72. <div class="box_title">小时行李峰值</div>
  73. <div class="box_content">
  74. <plot-charts style="height: 70%;" id="plotsCharts" :option="scatterPlot" />
  75. </div>
  76. </dv-border-box-7>
  77. </div>
  78. <div class="dashboard_bottom_right">
  79. <dv-border-box-7 :color="colors" v-loading="airportStogLoading" element-loading-text="拼命加载中" class="dashboard_bottom_left_box">
  80. <div class="box_title">航班量</div>
  81. <div class="box_content">
  82. <pie-charts id="stogCharts" :option="airportStog" />
  83. </div>
  84. </dv-border-box-7>
  85. <dv-border-box-7 :color="colors" v-loading="overallAccessLoading" element-loading-text="拼命加载中" class="dashboard_bottom_left_box">
  86. <div class="box_title">总体接入数据量</div>
  87. <div class="box_content">
  88. <pie-charts id="stogChartss" :option="overallAccess" />
  89. </div>
  90. </dv-border-box-7>
  91. <dv-border-box-7 :color="colors" v-loading="luggageStackLoading" element-loading-text="拼命加载中" class="dashboard_bottom_left_box">
  92. <div class="box_title">行李量</div>
  93. <div class="box_content">
  94. <stack-charts id="stackCharts" :option="luggageStack" />
  95. </div>
  96. </dv-border-box-7>
  97. <dv-border-box-7 style="margin-top: 40px;" :color="colors" v-loading="fligtPlotLoading" element-loading-text="拼命加载中" class="dashboard_bottom_left_box">
  98. <div class="box_title">小时航班量峰值</div>
  99. <div class="box_content">
  100. <plot-charts id="fligtCharts" :option="fligtPlot" />
  101. </div>
  102. </dv-border-box-7>
  103. </div>
  104. </div>
  105. </div>
  106. </template>
  107. <script>
  108. import selectBox from './components/boxs'
  109. import barCharts from './components/echart/commonChartsBar.vue'
  110. import mapCharts from './components/echart/commonChartsChinaMap.vue'
  111. import plotCharts from './components/echart/commonChartscok.vue'
  112. import pieCharts from './components/echart/commonChartsLine.vue'
  113. import stackCharts from './components/echart/commonChartsck.vue'
  114. import { barOption, airportName, colors, checkAlls, checkLuggage, checkNodes, checkAccess, mapOption, pieOption, scatterOption, LuggageOption } from './utils'
  115. import { Query } from '@/api/webApi'
  116. import { parseTime } from '@/utils'
  117. const defaultTime = new Date().getTime() - 24 * 60 * 60 * 1000
  118. const defaultTimeFormat = parseTime(defaultTime, '{y}-{m}-{d}')
  119. export default {
  120. name: 'DashboardSystem',
  121. components: { selectBox, barCharts, mapCharts, plotCharts, pieCharts, stackCharts },
  122. data () {
  123. return {
  124. colors: ['#92c1ef'],
  125. checkAlls: checkAlls,
  126. checkAccess: checkAccess,
  127. checkLuggage: checkLuggage,
  128. checkNodes: checkNodes,
  129. dateTimes: [defaultTimeFormat, defaultTimeFormat],
  130. thrdsItems: ['航司接入数据量排行', '航司行李运力排行', '航司节点覆盖率排行'],
  131. thrdsIndex: 0,
  132. airportAccess: _.cloneDeep(barOption),
  133. airportAccessLoading: false,
  134. airportLuggage: _.cloneDeep(barOption),
  135. airportLuggageLoading: false,
  136. airportNodes: _.cloneDeep(barOption),
  137. airportNodesLoading: false,
  138. mapOverview: _.cloneDeep(mapOption),
  139. mapOverviewLoading: false,
  140. scatterPlot: _.cloneDeep(scatterOption),
  141. scatterPlotLoading: false,
  142. airportStog: _.cloneDeep(pieOption),
  143. airportStogLoading: false,
  144. overallAccess: _.cloneDeep(pieOption),
  145. overallAccessLoading: false,
  146. luggageStack: _.cloneDeep(LuggageOption),
  147. luggageStackLoading: false,
  148. fligtPlot: _.cloneDeep(scatterOption),
  149. fligtPlotLoading: false,
  150. airlineChecks: checkAccess,
  151. airlinePlot: _.cloneDeep(barOption),
  152. airlinePlotLoading: false,
  153. mapOverviewItem: {
  154. build: 0,
  155. building: 0,
  156. not_build: 0
  157. }
  158. }
  159. },
  160. computed: {
  161. airlineValue () {
  162. if (this.thrdsIndex == 0) {
  163. return '总数据条数'
  164. } else if (this.thrdsIndex == 1) {
  165. return '总行李量'
  166. } else {
  167. return '基本节点'
  168. }
  169. }
  170. },
  171. methods: {
  172. formatChartsData (series, targets) {
  173. try {
  174. const ndatas = [...series]
  175. if (!ndatas.length) return
  176. ndatas.forEach((item, index) => {
  177. item = Object.assign(item, targets[index])
  178. })
  179. return ndatas
  180. } catch (error) {
  181. console.log(error)
  182. }
  183. },
  184. getChecksValue (name) {
  185. return this.$refs[name].value
  186. },
  187. selectChange (val) {
  188. if (!val) return
  189. this.pageInit()
  190. },
  191. //机场接入数据量排行下拉
  192. checkAccessChange (val) {
  193. if (!val) return
  194. this.setAirportAccess(val)
  195. },
  196. //机场行李运力排行下拉
  197. checkLuggageChange (val) {
  198. if (!val) return
  199. this.setAirportLuggage(val)
  200. },
  201. //机场节点覆盖率排行下拉
  202. checkNodesChange (val) {
  203. if (!val) return
  204. this.setAirportNodes(val)
  205. },
  206. //航司下拉
  207. checkAirlineChange (val) {
  208. if (!val) return
  209. const index = this.thrdsIndex
  210. if (index == 0) {
  211. this.setAirportAccess(val, true)
  212. } else if (index == 1) {
  213. this.setAirportLuggage(val, true)
  214. } else {
  215. this.setAirportNodes(val, true)
  216. }
  217. },
  218. timesChange (times) {
  219. if (!times.length) return
  220. this.pageInit()
  221. },
  222. formatBarData (arrs) {
  223. if (!arrs.length) return
  224. const ndatas = []
  225. const narrs = _.cloneDeep(arrs)
  226. narrs.forEach(item => {
  227. ndatas.push(item.data)
  228. })
  229. const newArray = ndatas.reverse()[0].map((col, i) => ndatas.map(row => row[i]))
  230. narrs.forEach((item, index) => {
  231. item.data = newArray[index]
  232. })
  233. return narrs
  234. },
  235. //机场接入数据排行
  236. async setAirportAccess (tag = '总数据条数', flag = false) {
  237. flag ? this.airlinePlotLoading = true : this.airportAccessLoading = true
  238. try {
  239. const params = flag ? { fd1: this.dateTimes[0], fd2: this.dateTimes[1], order_rules: tag || this.getChecksValue('airlinePlot') } : { fd1: this.dateTimes[0], fd2: this.dateTimes[1], area: this.getChecksValue('checkAlls') || '全国', order_rules: this.getChecksValue('checkAccess') || tag }
  240. const {
  241. code,
  242. returnData
  243. } = await Query({
  244. serviceId: flag ? SERVICE_ID.airlineDataId : SERVICE_ID.airportAccessId,
  245. dataContent: params,
  246. event: '0',
  247. })
  248. if (String(code) !== '0') {
  249. throw new Error('获取数据失败')
  250. }
  251. const ndatas = [...returnData].splice(0, 5)
  252. const ntargets = []
  253. const names = []
  254. if (ndatas.length) {
  255. ndatas.map(({ airport, aircompany, bpm_messages, bsm_messages, flight_messages, exception_messages, non_standard_messages_total }) => {
  256. const nitems = [this.formatNumberData(bpm_messages), this.formatNumberData(bsm_messages), this.formatNumberData(flight_messages), this.formatNumberData(exception_messages), this.formatNumberData(non_standard_messages_total)]
  257. names.push(airport || aircompany)
  258. ntargets.push({
  259. emphasis: {
  260. focus: 'series'
  261. },
  262. data: nitems,
  263. type: 'bar',
  264. barWidth: 10
  265. })
  266. })
  267. flag ? this.airlinePlot.yAxis.data = names?.reverse() : this.airportAccess.yAxis.data = names?.reverse()
  268. ntargets.map((item, index) => {
  269. item.name = airportName[index]
  270. item.color = colors[index]
  271. if (item.name != '不符合规范数据') item.stack = 'totals'
  272. })
  273. if (flag) {
  274. this.airlineChecks = checkAccess
  275. this.airlinePlot.title.left = '30%'
  276. this.airlinePlot.title.subtext = '单位:万条'
  277. this.airlinePlot.series = this.formatBarData(ntargets)
  278. } else {
  279. this.airportAccess.title.subtext = '单位:万条'
  280. this.airportAccess.series = this.formatBarData(ntargets)
  281. }
  282. }
  283. } catch (error) {
  284. console.log(error)
  285. }
  286. flag ? this.airlinePlotLoading = false : this.airportAccessLoading = false
  287. },
  288. //机场行李运力排行
  289. async setAirportLuggage (tag = '总行李量', flag = false) {
  290. flag ? this.airlinePlotLoading = true : this.airportLuggageLoading = true
  291. try {
  292. const parmas = flag ? { fd1: this.dateTimes[0], fd2: this.dateTimes[1], order_rules: tag || this.getChecksValue('airlinePlot') } : { fd1: this.dateTimes[0], fd2: this.dateTimes[1], area: this.getChecksValue('checkAlls') || '全国', order_rules: this.getChecksValue('checkLuggage') || tag }
  293. const {
  294. code,
  295. returnData
  296. } = await Query({
  297. serviceId: flag ? SERVICE_ID.airlineLuggageId : SERVICE_ID.airportLuggageId,
  298. dataContent: parmas,
  299. event: '0',
  300. })
  301. if (String(code) !== '0') {
  302. throw new Error('获取数据失败')
  303. }
  304. const ndatas = [...returnData].splice(0, 5)
  305. const ntargets = []
  306. const names = []
  307. const nluggages = flag ? ['总行李量', '中转行李量'] : ['离港行李量', '进港行李量', '中转行李量']
  308. if (ndatas.length) {
  309. ndatas.map(({ airport, aircompany, in_bag, out_bag, transfer_bag, total_bag }) => {
  310. const nitems = flag ? [this.formatNumberData(total_bag), this.formatNumberData(transfer_bag)] : [this.formatNumberData(out_bag), this.formatNumberData(in_bag), this.formatNumberData(transfer_bag)]
  311. names.push(airport || aircompany)
  312. ntargets.push({
  313. emphasis: {
  314. focus: 'series'
  315. },
  316. data: nitems,
  317. type: 'bar',
  318. barWidth: 10
  319. })
  320. })
  321. flag ? this.airlinePlot.yAxis.data = names?.reverse() : this.airportLuggage.yAxis.data = names?.reverse()
  322. ntargets.map((item, index) => {
  323. item.name = nluggages[index]
  324. item.color = colors[index]
  325. item.stack = 'totals'
  326. })
  327. if (flag) {
  328. this.airlineChecks = checkLuggage
  329. this.airlinePlot.title.left = '30%'
  330. this.airlinePlot.title.subtext = '单位:万件'
  331. this.airlinePlot.series = this.formatBarData(ntargets)
  332. } else {
  333. this.airportLuggage.title.subtext = '单位:万件'
  334. this.airportLuggage.series = this.formatBarData(ntargets)
  335. }
  336. }
  337. } catch (error) {
  338. console.log(error)
  339. }
  340. flag ? this.airlinePlotLoading = false : this.airportLuggageLoading = false
  341. },
  342. //机场节点覆盖率排行
  343. async setAirportNodes (tag = '基本节点', flag = false) {
  344. flag ? this.airlinePlotLoading = true : this.airportNodesLoading = true
  345. try {
  346. const params = flag ? { fd1: this.dateTimes[0], fd2: this.dateTimes[1], order_rules: tag || this.getChecksValue('airlinePlot') } : { fd1: this.dateTimes[0], fd2: this.dateTimes[1], area: this.getChecksValue('checkAlls') || '全国', order_rules: this.getChecksValue('checkNodes') || tag }
  347. const {
  348. code,
  349. returnData
  350. } = await Query({
  351. serviceId: flag ? SERVICE_ID.airlineNodesId : SERVICE_ID.airportNodesId,
  352. dataContent: params,
  353. event: '0',
  354. })
  355. if (String(code) !== '0') {
  356. throw new Error('获取数据失败')
  357. }
  358. const ndatas = [...returnData].splice(0, 5)
  359. const ntargets = []
  360. const names = []
  361. const nodes = ['标准内节点', '标准外节点', '基本节点']
  362. if (ndatas.length) {
  363. ndatas.map(({ airport, aircompany, basic_nodes, in_standard_nodes, not_in_standard_nodes }) => {
  364. const nitems = [in_standard_nodes, not_in_standard_nodes, basic_nodes]
  365. names.push(airport || aircompany)
  366. ntargets.push({
  367. emphasis: {
  368. focus: 'series'
  369. },
  370. data: nitems,
  371. type: 'bar',
  372. barWidth: 10
  373. })
  374. })
  375. flag ? this.airlinePlot.yAxis.data = names?.reverse() : this.airportNodes.yAxis.data = names?.reverse()
  376. ntargets.map((item, index) => {
  377. item.name = nodes[index]
  378. item.color = colors[index]
  379. if (item.name != '基本节点') {
  380. item.stack = 'totals'
  381. } else {
  382. item.color = '#b39ddb'
  383. }
  384. })
  385. if (flag) {
  386. this.airlineChecks = checkNodes
  387. this.airlinePlot.title.left = '30%'
  388. this.airlinePlot.title.subtext = '单位:个'
  389. this.airlinePlot.series = this.formatBarData(ntargets)
  390. } else {
  391. this.airportNodes.title.subtext = '单位:个'
  392. this.airportNodes.series = this.formatBarData(ntargets)
  393. }
  394. }
  395. } catch (error) {
  396. console.log(error)
  397. }
  398. flag ? this.airlinePlotLoading = false : this.airportNodesLoading = false
  399. },
  400. //机场行李量
  401. async setLuggageStack () {
  402. this.luggageStackLoading = true
  403. try {
  404. const {
  405. code,
  406. returnData
  407. } = await Query({
  408. serviceId: SERVICE_ID.pelLuggageId,
  409. dataContent: { fd1: this.dateTimes[0], fd2: this.dateTimes[1], area: this.getChecksValue('checkAlls') || '全国' },
  410. event: '0',
  411. })
  412. if (String(code) !== '0') {
  413. throw new Error('获取数据失败')
  414. }
  415. const ndatas = [...returnData]
  416. const ntargets = []
  417. if (ndatas.length) {
  418. this.luggageStack.xAxis.data = ['总量', '离港总量', '进港总量', '中转总量', '不正常行李量', '特殊行李量']
  419. ndatas.map(({ flightAttr, in_bag, notnormal_bag, out_bag, special_bag, total_bag, transfer_bag }) => {
  420. const nitems = [this.formatNumberData(total_bag), this.formatNumberData(out_bag), this.formatNumberData(in_bag), this.formatNumberData(transfer_bag), this.formatNumberData(notnormal_bag), this.formatNumberData(special_bag)]
  421. ntargets.push({
  422. name: flightAttr,
  423. data: nitems,
  424. type: 'bar',
  425. barWidth: 15,
  426. stack: 'total'
  427. })
  428. })
  429. this.luggageStack.title.subtext = '单位:万件'
  430. this.luggageStack.series = ntargets
  431. }
  432. } catch (error) {
  433. console.log(error)
  434. }
  435. this.luggageStackLoading = false
  436. },
  437. formatNumberData (targetNum) {
  438. return _.floor((Number(targetNum) / 10000), 4)
  439. },
  440. //机场行李跟踪系统建设总览
  441. async setMapOverview () {
  442. this.mapOverviewLoading = true
  443. try {
  444. const {
  445. code,
  446. returnData
  447. } = await Query({
  448. serviceId: SERVICE_ID.airportTotalId,
  449. dataContent: { fd1: this.dateTimes[0], fd2: this.dateTimes[1], area: this.getChecksValue('checkAlls') || '全国' },
  450. event: '0',
  451. })
  452. if (String(code) !== '0') {
  453. throw new Error('获取数据失败')
  454. }
  455. const ndatas = [...returnData]
  456. const maps = []
  457. const mapItem = new Map()
  458. const geoCoordMap = {}
  459. if (ndatas.length) {
  460. ndatas.map(item => {
  461. const nitem = _.cloneDeep(item)
  462. item.name = item.airportname
  463. item.value = item.out_bag
  464. geoCoordMap[item.airportname] = [Number(item.longitude_deg), Number(item.latitude_deg)]
  465. if (!mapItem.has(nitem.province)) {
  466. mapItem.set(nitem.province, nitem.out_bag)
  467. } else {
  468. const val = mapItem.get(nitem.province)
  469. mapItem.set(nitem.province, nitem.out_bag += val)
  470. }
  471. // const citem = {
  472. // name: item.province,
  473. // value: item.out_bag
  474. // }
  475. // maps.push(_.assign(nitem, citem))
  476. })
  477. }
  478. function convertMap (map) {
  479. const maps = []
  480. for (let [key, val] of map) {
  481. const citem = {
  482. name: key,
  483. value: val
  484. }
  485. maps.push(citem)
  486. }
  487. return maps
  488. }
  489. function convertData (data) {
  490. const res = [];
  491. for (let i = 0; i < data.length; i++) {
  492. const geoCoord = geoCoordMap[data[i].name];
  493. if (geoCoord) {
  494. res.push({
  495. name: data[i].name,
  496. value: geoCoord.concat(data[i].value)
  497. })
  498. }
  499. }
  500. return res
  501. }
  502. this.mapOverview.series[0].data = convertMap(mapItem)
  503. this.mapOverview.series[1].data = convertData(ndatas)
  504. this.mapOverview.tooltip.formatter = function (item) {
  505. const { componentSubType } = item
  506. if (componentSubType == 'scatter' && item.data) {
  507. const { name, value } = item.data
  508. const [longitude_deg, latitude_deg, val] = value
  509. const nitems = ndatas.filter(item => item.airportname == name && item.longitude_deg == longitude_deg && item.latitude_deg == latitude_deg && item.out_bag == val)
  510. if (!nitems.length) return
  511. const { airportname, airport, dom_bag, int_bag, out_bag, in_bag, transfer_bag, normal_bag, special_bag, notnormal_bag } = nitems[0]
  512. return `<div style="width:400px">
  513. <div style="font-size:18px;font-weight:600;">${airportname}</div>
  514. <div style='display:flex;margin-top:10px;'><div style='margin-right:20px'>机场三字码:${airport}</div><div>机场简称:${airportname}</div></div>
  515. <div style='display:flex;margin-top:10px;'><div style='margin-right:20px'>国内总量:${dom_bag}</div><div>国际总量:${int_bag}</div></div>
  516. <div style='display:flex;margin-top:10px;'><div style='margin-right:50px'>离港总量:${out_bag}</div><div style='margin-right:50px'>进港总量:${in_bag}</div><div>中转总量:${transfer_bag}</div></div>
  517. <div style='display:flex;margin-top:10px;'><div style='margin-right:50px'>正常行李量:${normal_bag}</div><div style='margin-right:50px'>特殊行李量:${special_bag}</div><div>异常行李量:${notnormal_bag}</div></div>
  518. </div>`
  519. }
  520. }
  521. } catch (error) {
  522. console.log(error)
  523. }
  524. this.mapOverviewLoading = false
  525. },
  526. //机场行李跟踪系统建设总览-建立情况
  527. async setMapOverviewTotals () {
  528. try {
  529. const {
  530. code,
  531. returnData
  532. } = await Query({
  533. serviceId: SERVICE_ID.pelestablishId,
  534. dataContent: {},
  535. event: '0',
  536. })
  537. if (String(code) !== '0') {
  538. throw new Error('获取数据失败')
  539. }
  540. const ndatas = [...returnData]
  541. if (ndatas.length) {
  542. this.mapOverviewItem = ndatas[0]
  543. }
  544. } catch (error) {
  545. console.log(error)
  546. }
  547. },
  548. formatPlotData (times, arrs) {
  549. const ndatas = []
  550. for (let i = 0; i < times.length; i++) {
  551. const ctime = times[i]
  552. for (let j = 0; j < arrs.length; j++) {
  553. const point = arrs[j][i]
  554. ndatas.push([ctime, j, point || 0])
  555. }
  556. }
  557. return ndatas
  558. },
  559. //小时行李峰值
  560. async setScatterPlot () {
  561. this.scatterPlotLoading = true
  562. try {
  563. const {
  564. code,
  565. returnData
  566. } = await Query({
  567. serviceId: SERVICE_ID.hourLuggageId,
  568. dataContent: { fd1: this.dateTimes[0], fd2: this.dateTimes[1], area: this.getChecksValue('checkAlls') || '全国' },
  569. event: '0',
  570. })
  571. if (String(code) !== '0') {
  572. throw new Error('获取数据失败')
  573. }
  574. const ndatas = [...returnData]
  575. const times = []
  576. const ntargets = []
  577. if (ndatas.length) {
  578. ndatas.map(({ arrival, check_in, hour_of_day, loading, on_boarding, security_check, sorting, transit }) => {
  579. const nitems = [check_in, security_check, sorting, loading, on_boarding, transit, arrival]
  580. times.push(hour_of_day)
  581. ntargets.push(nitems)
  582. })
  583. const hours = [...times]
  584. // prettier-ignore
  585. const days = [
  586. '值机', '安检', '分拣',
  587. '装车', '装机', '中转', '到达'
  588. ];
  589. // prettier-ignore
  590. const data = this.formatPlotData(times, ntargets)
  591. const title = []
  592. const singleAxis = []
  593. const series = []
  594. function formatDataItem (number) {
  595. const nnumber = Number(number)
  596. if (nnumber <= 10) {
  597. return nnumber
  598. }
  599. if (nnumber > 10 && nnumber <= 100) {
  600. return nnumber / 10
  601. }
  602. if (nnumber > 100 && nnumber <= 1000) {
  603. return nnumber / 100
  604. }
  605. if (nnumber > 1000 && nnumber <= 10000) {
  606. return nnumber / 1000
  607. }
  608. if (nnumber > 10000 && nnumber <= 100000) {
  609. return nnumber / 10000
  610. }
  611. }
  612. days.forEach(function (day, idx) {
  613. title.push({
  614. textBaseline: 'middle',
  615. top: ((idx + 0.5) * 100) / 7 + '%',
  616. left: '3%',
  617. text: day,
  618. textStyle: {
  619. color: '#666',
  620. fontSize: 14,
  621. fontWeight: 300
  622. }
  623. });
  624. singleAxis.push({
  625. left: '12%',
  626. top: 0,
  627. bottom: 120,
  628. type: 'category',
  629. boundaryGap: false,
  630. data: hours,
  631. top: (idx * 100) / 7 + 5 + '%',
  632. height: 100 / 7 - 10 + '%',
  633. axisLabel: {
  634. interval: 2
  635. },
  636. axisTick: {
  637. show: false
  638. },
  639. axisLabel: { show: false },
  640. axisLine: {
  641. show: true,
  642. lineStyle: {
  643. color: '#ddd'
  644. }
  645. },
  646. });
  647. series.push({
  648. singleAxisIndex: idx,
  649. coordinateSystem: 'singleAxis',
  650. type: 'scatter',
  651. data: [],
  652. symbolSize: function (dataItem) {
  653. return dataItem[1] * 4;
  654. }
  655. });
  656. });
  657. data.forEach(function (dataItem) {
  658. if (series[dataItem[0]]) {
  659. series[dataItem[0]].data.push([dataItem[1], formatDataItem(dataItem[2])]);
  660. }
  661. });
  662. this.scatterPlot.title = title
  663. this.scatterPlot.singleAxis = singleAxis
  664. this.scatterPlot.series = series
  665. }
  666. } catch (error) {
  667. console.log(error)
  668. }
  669. this.scatterPlotLoading = false
  670. },
  671. //小时航班量峰值
  672. async setFligtPlot () {
  673. this.fligtPlotLoading = true
  674. try {
  675. const {
  676. code,
  677. returnData
  678. } = await Query({
  679. serviceId: SERVICE_ID.hourFlightId,
  680. dataContent: { fd1: this.dateTimes[0], fd2: this.dateTimes[1], area: this.getChecksValue('checkAlls') || '全国' },
  681. event: '0',
  682. })
  683. if (String(code) !== '0') {
  684. throw new Error('获取数据失败')
  685. }
  686. const ndatas = [...returnData]
  687. const times = []
  688. const ntargets = []
  689. if (ndatas.length) {
  690. ndatas.map(({ actual_departure_flights, cancelled_flights, delayed_flights, estimated_departure_flights, flights_with_baggage, hour_of_day }) => {
  691. const nitems = [estimated_departure_flights, actual_departure_flights, delayed_flights, cancelled_flights, flights_with_baggage]
  692. times.push(hour_of_day)
  693. ntargets.push(nitems)
  694. })
  695. const hours = [...times]
  696. // prettier-ignore
  697. const days = [
  698. '预计起飞', '实际起飞', '延误',
  699. '取消', '有行李'];
  700. // prettier-ignore
  701. const data = this.formatPlotData(times, ntargets)
  702. const title = []
  703. const singleAxis = []
  704. const series = []
  705. function formatDataItem (number) {
  706. const nnumber = Number(number)
  707. if (nnumber <= 10) {
  708. return nnumber
  709. }
  710. if (nnumber > 10 && nnumber <= 100) {
  711. return nnumber / 10
  712. }
  713. if (nnumber > 100 && nnumber <= 1000) {
  714. return nnumber / 100
  715. }
  716. if (nnumber > 1000 && nnumber <= 10000) {
  717. return nnumber / 1000
  718. }
  719. if (nnumber > 10000 && nnumber <= 100000) {
  720. return nnumber / 10000
  721. }
  722. }
  723. days.forEach(function (day, idx) {
  724. title.push({
  725. textBaseline: 'middle',
  726. top: ((idx + 0.5) * 100) / 7 + '%',
  727. left: '3%',
  728. text: day,
  729. textStyle: {
  730. color: '#666',
  731. fontSize: 14,
  732. fontWeight: 300
  733. }
  734. });
  735. singleAxis.push({
  736. left: '12%',
  737. top: 0,
  738. bottom: 120,
  739. type: 'category',
  740. boundaryGap: false,
  741. data: hours,
  742. top: (idx * 100) / 7 + 5 + '%',
  743. height: 100 / 7 - 10 + '%',
  744. axisLabel: {
  745. interval: 2
  746. },
  747. axisTick: {
  748. show: false
  749. },
  750. axisLabel: { show: false },
  751. axisLine: {
  752. show: true,
  753. lineStyle: {
  754. color: '#ddd'
  755. }
  756. },
  757. });
  758. series.push({
  759. singleAxisIndex: idx,
  760. coordinateSystem: 'singleAxis',
  761. type: 'scatter',
  762. data: [],
  763. symbolSize: function (dataItem) {
  764. return dataItem[1] * 4;
  765. }
  766. });
  767. });
  768. data.forEach(function (dataItem) {
  769. if (series[dataItem[0]]) {
  770. series[dataItem[0]].data.push([dataItem[1], formatDataItem(dataItem[2])]);
  771. }
  772. });
  773. this.fligtPlot.title = title
  774. this.fligtPlot.singleAxis = singleAxis
  775. this.fligtPlot.series = series
  776. }
  777. } catch (error) {
  778. console.log(error)
  779. }
  780. this.fligtPlotLoading = false
  781. },
  782. //航班量
  783. async setAirportStog () {
  784. this.airportStogLoading = true
  785. try {
  786. const {
  787. code,
  788. returnData
  789. } = await Query({
  790. serviceId: SERVICE_ID.pelFlightId,
  791. dataContent: { fd1: this.dateTimes[0], fd2: this.dateTimes[1], area: this.getChecksValue('checkAlls') || '全国' },
  792. event: '0',
  793. })
  794. if (String(code) !== '0') {
  795. throw new Error('获取数据失败')
  796. }
  797. const ndatas = [...returnData]
  798. if (ndatas.length) {
  799. const { canfly, dlyfly, domfly, havebagfly, intfly, nohavebagfly, norfly, totalfly } = ndatas[0]
  800. this.airportStog.legend.data = ['国内航班总量', '国际航班总量', '正常航班量', '延误航班量', '取消航班量', '有行李航班量', '无行李航班量']
  801. this.airportStog.series[0].data = [{ name: '无行李航班量', value: nohavebagfly }, { name: '有行李航班量', value: havebagfly }]
  802. this.airportStog.series[0].color = ['#660066', '#993399']
  803. this.airportStog.series[1].data = [{ name: '正常航班量', value: norfly }, { name: '延误航班量', value: dlyfly }, { name: '取消航班量', value: canfly }]
  804. this.airportStog.series[1].color = ['#6666ff', '#3333ff', '#0000ff']
  805. this.airportStog.series[2].data = [{ name: '国内航班总量', value: domfly }, { name: '国际航班总量', value: intfly }]
  806. this.airportStog.series[2].color = ['#00cc33', '#33ff66']
  807. this.airportStog.series[3].data = [{ name: '航班总量', value: totalfly }]
  808. this.airportStog.series[3].color = ['#fff']
  809. }
  810. } catch (error) {
  811. console.log(error)
  812. }
  813. this.airportStogLoading = false
  814. },
  815. //总体接入数据量
  816. async setoverallAccess () {
  817. this.overallAccessLoading = true
  818. try {
  819. const {
  820. code,
  821. returnData
  822. } = await Query({
  823. serviceId: SERVICE_ID.pelTotalId,
  824. dataContent: { fd1: this.dateTimes[0], fd2: this.dateTimes[1], area: this.getChecksValue('checkAlls') || '全国' },
  825. event: '0',
  826. })
  827. if (String(code) !== '0') {
  828. throw new Error('获取数据失败')
  829. }
  830. const ndatas = [...returnData]
  831. if (ndatas.length) {
  832. const { bpm_messages, bsm_messages, flight_messages, total_messages, exception_messages, non_standard_messages_total, standard_messages_total } = ndatas[0]
  833. this.overallAccess.legend.data = ['BPM', 'BSM', '航班数据', '异常行李数据', '不符合规范数据', '符合规范数据']
  834. this.overallAccess.series[0].data = [{ name: '不符合规范数据', value: non_standard_messages_total }, { name: '符合规范数据', value: standard_messages_total }]
  835. this.overallAccess.series[0].color = ['#660066', '#993399']
  836. this.overallAccess.series[1].data = [{ name: 'BPM', value: bpm_messages }, { name: 'BSM', value: bsm_messages }, { name: '航班数据', value: flight_messages }, { name: '异常行李数据', value: exception_messages }]
  837. this.overallAccess.series[1].color = ['#0000ff', '#6666ff', '#3333ff', '#000066']
  838. this.overallAccess.series[2].data = [{ name: '', value: 0 }]
  839. this.overallAccess.series[2].color = ['#fff']
  840. this.overallAccess.series[3].data = [{ name: '数据总量', value: total_messages }]
  841. this.overallAccess.series[3].color = ['#fff']
  842. }
  843. } catch (error) {
  844. console.log(error)
  845. }
  846. this.overallAccessLoading = false
  847. },
  848. //航司数据切换
  849. thrdsChange (index) {
  850. this.thrdsIndex = index
  851. if (index == 0) {
  852. this.setAirportAccess('总条数数据', true)
  853. } else if (index == 1) {
  854. this.setAirportLuggage('总行李量', true)
  855. } else {
  856. this.setAirportNodes('基本节点', true)
  857. }
  858. },
  859. pageInit () {
  860. this.setAirportAccess()
  861. this.setAirportLuggage()
  862. this.setAirportNodes()
  863. this.setAirportAccess('总条数数据', true)
  864. this.setLuggageStack()
  865. this.setFligtPlot()
  866. this.setScatterPlot()
  867. this.setAirportStog()
  868. this.setoverallAccess()
  869. this.setMapOverview()
  870. this.setMapOverviewTotals()
  871. }
  872. },
  873. mounted () {
  874. this.pageInit()
  875. }
  876. }
  877. </script>
  878. <style lang="scss" scoped>
  879. $height: 25%;
  880. $w100: 100%;
  881. $w25: 25%;
  882. $s20: 20px;
  883. $bg: #074fb8;
  884. .dashboard {
  885. padding: $s20;
  886. height: calc(100vh - 80px);
  887. width: $w100;
  888. background-color: #fff;
  889. .mapOverview {
  890. position: absolute;
  891. left: 7%;
  892. top: 10%;
  893. font-size: 14px;
  894. color: #696969;
  895. &_list {
  896. margin-top: 5px;
  897. &_icon {
  898. display: inline-block;
  899. width: 11px;
  900. height: 11px;
  901. margin-right: 6px;
  902. }
  903. .icon1 {
  904. background-color: #4fc267;
  905. }
  906. .icon2 {
  907. background-color: #ffcc80;
  908. }
  909. .icon3 {
  910. background-color: #656565;
  911. }
  912. }
  913. }
  914. .box_title {
  915. background-color: $bg;
  916. color: #fff;
  917. font-size: 16px;
  918. font-weight: 600;
  919. clip-path: polygon(6% 0, 100% 0%, 97% 100%, -45% 160%);
  920. position: absolute;
  921. left: 2px;
  922. top: 2px;
  923. z-index: 10;
  924. line-height: 28px;
  925. padding: 0 20px;
  926. }
  927. .box_content {
  928. position: relative;
  929. width: 100%;
  930. height: 100%;
  931. &_select {
  932. position: absolute;
  933. left: 10px;
  934. top: 40px;
  935. z-index: 20;
  936. width: 90px;
  937. }
  938. }
  939. .thrds3 {
  940. margin-top: 40px;
  941. &_tabs {
  942. position: absolute;
  943. left: 0;
  944. top: -31px;
  945. line-height: 30px;
  946. z-index: 10;
  947. width: 100%;
  948. display: flex;
  949. font-size: 12px;
  950. &_list {
  951. background-color: #e2e2e2;
  952. padding: 0 8px;
  953. margin-right: 3px;
  954. border-top-left-radius: 5px;
  955. cursor: pointer;
  956. &:last-child {
  957. margin-right: 0;
  958. }
  959. }
  960. .active {
  961. background-color: #074fb8;
  962. color: #fff;
  963. }
  964. }
  965. }
  966. &_top {
  967. line-height: 50px;
  968. display: flex;
  969. justify-content: space-between;
  970. &_all {
  971. width: $w25;
  972. }
  973. &_title {
  974. color: #fff;
  975. margin: 0 $s20;
  976. width: calc(50% - 40px);
  977. display: flex;
  978. &_txt {
  979. width: 60%;
  980. margin-left: 15%;
  981. background-color: $bg;
  982. text-align: center;
  983. font-size: 22px;
  984. font-weight: 600;
  985. clip-path: polygon(6% 0, 100% 0%, 97% 100%, 0% 160%);
  986. }
  987. .shade {
  988. width: 30px;
  989. height: 50px;
  990. background-color: $bg;
  991. clip-path: polygon(60% 0, 100% 0%, 40% 100%, 0% 100%);
  992. }
  993. .shade_2 {
  994. background-color: rgba($color: $bg, $alpha: 0.7);
  995. }
  996. .shade_3 {
  997. background-color: rgba($color: $bg, $alpha: 0.3);
  998. }
  999. }
  1000. &_times {
  1001. width: $w25;
  1002. display: flex;
  1003. }
  1004. }
  1005. &_bottom {
  1006. overflow-y: scroll;
  1007. overflow-x: hidden;
  1008. height: calc(100% - 50px);
  1009. display: flex;
  1010. justify-content: space-between;
  1011. width: $w100;
  1012. &_left {
  1013. width: $w25;
  1014. &_box {
  1015. height: $height;
  1016. margin-top: $s20;
  1017. position: relative;
  1018. }
  1019. }
  1020. &_right {
  1021. width: $w25;
  1022. }
  1023. &_center {
  1024. margin: 0 $s20;
  1025. width: calc(50% - 40px);
  1026. &_box1 {
  1027. height: calc(75% + 40px);
  1028. margin-top: 20px;
  1029. }
  1030. &_box2 {
  1031. height: 25%;
  1032. margin-top: 40px;
  1033. }
  1034. }
  1035. }
  1036. }
  1037. </style>