index.vue 28 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043
  1. <!--
  2. * @Author: your name
  3. * @Date: 2022-01-17 10:39:22
  4. * @LastEditTime: 2022-08-12 19:14:09
  5. * @LastEditors: your name
  6. * @Description: 行李视图
  7. -->
  8. <template>
  9. <div class="baggage-view">
  10. <div
  11. ref="basicInfo"
  12. class="part1"
  13. >
  14. <div class="title">
  15. <span>行李基本信息</span>
  16. <el-radio-group
  17. v-model="infoBtn"
  18. class="radioBtn"
  19. size="mini"
  20. fill="#FFFFFF"
  21. text-color="#28344D"
  22. >
  23. <el-radio-button
  24. v-for="item in infoRadios"
  25. :key="item"
  26. :label="item"
  27. />
  28. </el-radio-group>
  29. </div>
  30. <div class="part1_info">
  31. <el-row :gutter="12">
  32. <el-col
  33. v-for="(item, index) in baggageBasicInfoCols"
  34. :key="index"
  35. :xl="[0, 2, 4].includes(index % 7) ? 4 : 3"
  36. :sm="6"
  37. :xs="6"
  38. >
  39. <span class="label">{{ item.label }}:</span><span
  40. class="content"
  41. :title="formattedBaggageInfo(item.prop)"
  42. >{{ formattedBaggageInfo(item.prop) }}</span>
  43. </el-col>
  44. </el-row>
  45. </div>
  46. </div>
  47. <div
  48. v-show="infoBtn === infoRadios[0]"
  49. class="part2"
  50. >
  51. <div class="part2_info">
  52. <div class="title">行李跟踪信息</div>
  53. <div class="type normal">
  54. {{ baggageBasicInfo.BagStatus }}
  55. </div>
  56. <div class="airline">
  57. <el-select
  58. v-model="selectedAirline"
  59. size="mini"
  60. class="airline-select"
  61. >
  62. <el-option
  63. v-for="airline in airlineList"
  64. :key="airline.value"
  65. :value="airline.value"
  66. :label="airline.label"
  67. />
  68. </el-select>
  69. </div>
  70. <div class="baggage-track-chart">
  71. <div class="step-line">
  72. <div
  73. v-for="(line, index) in 6"
  74. :key="index"
  75. :class="['step-line-segment', { 'step-line-active': activeStepLine(index) }]"
  76. />
  77. </div>
  78. <div
  79. v-for="(item, index) in stepNodes"
  80. :key="index"
  81. :class="{ 'step-item': true, 'active-item': item.processingTime }"
  82. >
  83. <div class="step-circle">
  84. <span class="step-name">{{ item.nodeName }}</span>
  85. </div>
  86. <div class="step-info">
  87. <!-- <div class="step-status">{{ item.status }}</div> -->
  88. <span class="step-time">{{ item.processingTime }}</span>
  89. <!-- <div class="step-location">{{ item.location }}</div> -->
  90. </div>
  91. </div>
  92. </div>
  93. </div>
  94. <div class="btns">
  95. <img
  96. class="btn-square btn-shadow"
  97. src="../../../../assets/baggage/ic_export.png"
  98. title="导出"
  99. @click="exportHandler('table', '行李节点列表')"
  100. >
  101. <img
  102. class="btn-square btn-shadow"
  103. src="../../../../assets/baggage/ic_setting.png"
  104. title="列设置"
  105. @click="show"
  106. >
  107. </div>
  108. </div>
  109. <div
  110. v-show="infoBtn == infoRadios[0]"
  111. class="part3"
  112. :style="{
  113. 'height': `calc(100vh - 80px - ${basicInfoHeight}px - 176px - 3 * 8px - 44px)`
  114. }"
  115. >
  116. <el-table
  117. ref="table"
  118. :data="baggageTableData"
  119. height="100%"
  120. size="mini"
  121. border
  122. fit
  123. :header-cell-class-name="headerCellClass"
  124. :header-cell-style="{ color: '#101116' }"
  125. :cell-class-name="cellClass"
  126. :span-method="tableSpanMethod"
  127. @cell-click="cellClickHandler"
  128. >
  129. <el-table-column
  130. v-for="item in tableColsCopy"
  131. :key="item.index"
  132. :prop="item.prop"
  133. :label="item.name"
  134. :align="item.align || 'center'"
  135. :width="item.width"
  136. show-overflow-tooltip
  137. >
  138. <template slot="header">
  139. <div class="cell-content">{{ item.name }}</div>
  140. </template>
  141. <template slot-scope="scope">
  142. <div class="cell-content">{{ scope.row[item.prop] }}</div>
  143. </template>
  144. </el-table-column>
  145. </el-table>
  146. </div>
  147. <div
  148. v-show="infoBtn === infoRadios[1]"
  149. class="part4"
  150. :style="{
  151. 'height': `calc(100vh - 80px - ${basicInfoHeight}px - 2 * 8px - 44px)`
  152. }"
  153. >
  154. <header class="head">
  155. <div class="title">行李跟踪信息</div>
  156. <div class="btns">
  157. <img
  158. class="btn-square btn-shadow"
  159. src="../../../../assets/baggage/ic_export.png"
  160. title="导出"
  161. @click="exportMessageToExcel"
  162. >
  163. </div>
  164. </header>
  165. <main class="main">
  166. <template v-if="messageList.length">
  167. <el-row
  168. :gutter="24"
  169. type="flex"
  170. >
  171. <el-col
  172. v-for="(message, index) in messageList"
  173. :key="index"
  174. :span="6"
  175. >
  176. <div class="card">
  177. <div class="message-date">{{ message.date }}</div>
  178. <div class="message-content">
  179. {{ message.dataContent.replaceAll(/[\r\n]{2,}/g, '\n').replaceAll('\\', '') }}
  180. </div>
  181. </div>
  182. </el-col>
  183. </el-row>
  184. </template>
  185. <template v-else>
  186. <el-empty
  187. :image-size="1"
  188. description="暂无数据"
  189. />
  190. </template>
  191. </main>
  192. </div>
  193. <!--列设置-->
  194. <Dialog
  195. :flag="dialogFlag"
  196. class="dialog-check-group"
  197. >
  198. <div class="dialog-wrapper">
  199. <div class="title">列设置</div>
  200. <div class="content">
  201. <el-tree
  202. :data="tableCols"
  203. :class="colsCheckClass"
  204. show-checkbox
  205. node-key="index"
  206. :default-expand-all="true"
  207. :props="{
  208. label: 'name',
  209. children: 'children',
  210. }"
  211. :default-checked-keys="checkedKeysTemp"
  212. @check="handleCheck"
  213. />
  214. </div>
  215. <div class="foot right t30">
  216. <el-button
  217. size="medium"
  218. class="r24"
  219. type="primary"
  220. @click="onCheck('baggageTableData')"
  221. >确定</el-button>
  222. <el-button
  223. size="medium"
  224. @click="hide"
  225. >取消</el-button>
  226. </div>
  227. </div>
  228. </Dialog>
  229. </div>
  230. </template>
  231. <script>
  232. import Dialog from '@/layout/components/Dialog/index.vue'
  233. import { myQuery } from '@/api/dataIntegration'
  234. import { BaggageMessageQuery } from '@/api/flight'
  235. import tableColsMixin from '../../mixins/tableCols'
  236. import { throttledExportToExcel } from '@/utils/table'
  237. export default {
  238. name: 'BaggageView',
  239. components: {
  240. Dialog
  241. },
  242. mixins: [tableColsMixin],
  243. data() {
  244. return {
  245. queryData: {},
  246. airlineList: [],
  247. selectedAirline: '',
  248. basicInfoHeight: 0,
  249. baggageBasicInfoCols: [
  250. {
  251. label: '行李牌号',
  252. prop: 'bagNo'
  253. },
  254. {
  255. label: '航班号',
  256. prop: 'flightNO'
  257. },
  258. {
  259. label: '特殊行李类型',
  260. prop: 'specialType'
  261. },
  262. {
  263. label: 'PNR编号',
  264. prop: 'PNR'
  265. },
  266. {
  267. label: '旅客',
  268. prop: 'name'
  269. },
  270. {
  271. label: '装载序列号',
  272. prop: 'loadSequenceIndex'
  273. },
  274. {
  275. label: '总件数',
  276. prop: 'totalNumber'
  277. },
  278. {
  279. label: '总重量',
  280. prop: 'totalWeight'
  281. },
  282. {
  283. label: '尺寸',
  284. prop: 'size'
  285. },
  286. {
  287. label: '常旅客号',
  288. prop: 'frequentFlyerNumber'
  289. },
  290. {
  291. label: '常旅客级别',
  292. prop: 'frequentFlyerClass'
  293. },
  294. {
  295. label: '取消值机',
  296. prop: 'whetherToCancelTheCheckIn'
  297. },
  298. {
  299. label: '是否可装载',
  300. prop: 'isItLoadable'
  301. },
  302. {
  303. label: '是否可运输',
  304. prop: 'isItTransportable'
  305. },
  306. {
  307. label: '行李激活状态',
  308. prop: 'activeState'
  309. },
  310. {
  311. label: '无BSM状态',
  312. prop: 'noBSM'
  313. },
  314. {
  315. label: '中转进航班',
  316. prop: 'inFlightNo'
  317. },
  318. {
  319. label: '中转出航班',
  320. prop: 'transferFlightNo'
  321. },
  322. {
  323. label: '速运标记',
  324. prop: 'expressSign'
  325. },
  326. {
  327. label: '破损标记',
  328. prop: 'brokenSign'
  329. },
  330. {
  331. label: '投诉标记',
  332. prop: 'complaintSign'
  333. },
  334. {
  335. label: '赔偿标记',
  336. prop: 'compensationSign'
  337. },
  338. {
  339. label: '异常状态',
  340. prop: 'bagExcType'
  341. },
  342. {
  343. label: '企业或团队名称',
  344. prop: 'teamOrGroup'
  345. }
  346. ],
  347. baggageBasicInfo: {},
  348. dialogVisibledele: false,
  349. active: 2,
  350. infoBtn: '',
  351. infoRadios: ['跟踪信息', '跟踪报文'],
  352. messageList: [],
  353. checkList: [],
  354. stepNodes: [],
  355. tableCols: [
  356. {
  357. name: '航班号',
  358. prop: 'flightNO'
  359. },
  360. { name: '航班日期', prop: 'flightDate', width: 100 },
  361. {
  362. name: '起飞航站\n起飞时间',
  363. prop: 'departureAirport',
  364. width: 111
  365. },
  366. {
  367. name: '目的航站\n降落时间',
  368. prop: 'landingAirport',
  369. sortable: 'custom',
  370. width: 111
  371. },
  372. { name: '旅客舱位', prop: 'passengerCompartment', width: 70 },
  373. { name: '旅客座位号', prop: 'passengerSeatNumber' },
  374. { name: '值机序号', prop: 'passengerCheckInNumber', width: 70 },
  375. { name: '节点标识', prop: 'nodeCode', width: 100 },
  376. { name: '节点名称', prop: 'nodeName' },
  377. { name: '位置标识', prop: 'locationCode' },
  378. // { name: '位置码', prop: '', },
  379. { name: '位置描述', prop: 'locationRemark' },
  380. { name: '读取时间', prop: 'dealTime', width: 158 },
  381. { name: '结果', prop: 'status' },
  382. { name: '次级代码', prop: 'secondaryCode', width: 70 },
  383. { name: '操作人', prop: 'AgentCode', width: 100 },
  384. { name: '设备ID', prop: 'DeviceCode' },
  385. { name: '发往位置', prop: 'toLocation' },
  386. { name: '发往位置描述', prop: 'toLocationMark', width: 110 },
  387. { name: '装载序号', prop: 'LoadSN' },
  388. { name: '容器编号', prop: 'U_Device_ID', width: 100 },
  389. { name: '数据来源', prop: 'dataSource' }
  390. ],
  391. baggageTableData: [],
  392. spanArr: [],
  393. pos: 0,
  394. queryLoop: null,
  395. queryTrackLoop: null
  396. }
  397. },
  398. computed: {
  399. activeStepLine() {
  400. return function (index) {
  401. return this.stepNodes[index].processingTime && this.stepNodes[index + 1].processingTime
  402. }
  403. },
  404. formattedBaggageInfo() {
  405. return function (prop) {
  406. const value = this.baggageBasicInfo[prop]
  407. if ((value ?? '') === '') {
  408. return ''
  409. } else {
  410. switch (prop) {
  411. case 'transitSign':
  412. return Number(value) === 0 ? '非中转' : '中转'
  413. case 'activeState':
  414. return Number(value) === 0 ? '未激活' : '激活'
  415. case 'whetherToCancelTheCheckIn':
  416. return Number(value) === 1 ? '取消' : ''
  417. case 'isItLoadable':
  418. case 'isItTransportable':
  419. case 'noBSM':
  420. case 'expressSign':
  421. case 'brokenSign':
  422. case 'complaintSign':
  423. case 'compensationSign':
  424. return Number(value) === 1 || value === 'Y' ? '是' : '否'
  425. default:
  426. return value
  427. }
  428. }
  429. }
  430. }
  431. },
  432. watch: {
  433. $route: {
  434. handler({ path, query }) {
  435. this.infoBtn = ''
  436. if (path.includes('baggageView')) {
  437. const { flightNO, flightDate, bagSN } = query
  438. if (flightNO && flightDate && bagSN) {
  439. this.queryData = { flightNO, flightDate, bagSN }
  440. this.queryBasicInfo()
  441. this.queryAirline()
  442. this.infoBtn = this.infoRadios[0]
  443. } else {
  444. this.$router.push('/advance')
  445. }
  446. }
  447. },
  448. deep: true,
  449. immediate: true
  450. },
  451. infoBtn: {
  452. handler(val) {
  453. this.stopLoopAll()
  454. if (val === this.infoRadios[0]) {
  455. if (this.selectedAirline) {
  456. this.startQueryTrack()
  457. }
  458. this.startQueryDetails()
  459. } else if (val === this.infoRadios[1]) {
  460. this.startQueryMessage()
  461. }
  462. },
  463. immediate: true
  464. },
  465. selectedAirline() {
  466. this.stopQueryTrack()
  467. this.startQueryTrack()
  468. }
  469. },
  470. created() {
  471. this.resetStepNodes()
  472. },
  473. activated() {
  474. this.basicInfoHeight = this.$refs['basicInfo'].offsetHeight
  475. this.$refs['table']?.doLayout()
  476. },
  477. updated() {
  478. this.basicInfoHeight = this.$refs['basicInfo'].offsetHeight
  479. this.$refs['table']?.doLayout()
  480. },
  481. methods: {
  482. startQueryDetails() {
  483. this.queryDetails()
  484. this.queryLoop = setInterval(this.queryDetails.bind(this), LOOP_INTERVAL.baggageDetails)
  485. },
  486. startQueryTrack() {
  487. this.queryTrack()
  488. this.queryTrackLoop = setInterval(this.queryTrack.bind(this), LOOP_INTERVAL.baggageTrack)
  489. },
  490. startQueryMessage() {
  491. this.queryMessage()
  492. this.queryLoop = setInterval(this.queryMessage.bind(this), LOOP_INTERVAL.baggageMessage)
  493. },
  494. stopQueryTrack() {
  495. this.queryTrackLoop && clearInterval(this.queryTrackLoop)
  496. this.queryTrackLoop = null
  497. },
  498. stopLoopAll() {
  499. this.queryLoop && clearInterval(this.queryLoop)
  500. this.queryLoop = null
  501. this.stopQueryTrack()
  502. },
  503. resetStepNodes() {
  504. this.stepNodes = [
  505. {
  506. nodeCode: 'CHECKIN',
  507. nodeName: '值机',
  508. processingTime: ''
  509. },
  510. {
  511. nodeCode: 'SECURITY',
  512. nodeName: '安检',
  513. processingTime: ''
  514. },
  515. {
  516. nodeCode: 'SORT',
  517. nodeName: '分拣',
  518. processingTime: ''
  519. },
  520. {
  521. nodeCode: 'LOAD',
  522. nodeName: '装车',
  523. processingTime: ''
  524. },
  525. {
  526. nodeCode: 'INFL',
  527. nodeName: '装机',
  528. processingTime: ''
  529. },
  530. {
  531. nodeCode: 'UNLOAD',
  532. nodeName: '卸机',
  533. processingTime: ''
  534. },
  535. {
  536. nodeCode: 'ARRIVED',
  537. nodeName: '到达',
  538. processingTime: ''
  539. }
  540. ]
  541. },
  542. initTableData(tableData) {
  543. const spanArr = []
  544. let pos = 0
  545. for (let i = 0; i < tableData.length; i++) {
  546. if (i === 0) {
  547. spanArr.push(1)
  548. } else {
  549. if (
  550. tableData[i]['flightNO'] === tableData[i - 1]['flightNO'] &&
  551. tableData[i]['flightDate'] === tableData[i - 1]['flightDate'] &&
  552. tableData[i]['departureAirport'] === tableData[i - 1]['departureAirport'] &&
  553. tableData[i]['landingAirport'] === tableData[i - 1]['landingAirport']
  554. ) {
  555. spanArr[pos] += 1
  556. spanArr.push(0)
  557. } else {
  558. spanArr.push(1)
  559. pos = i
  560. }
  561. }
  562. }
  563. this.spanArr = spanArr
  564. this.pos = pos
  565. },
  566. headerCellClass({ row, column, rowIndex, columnIndex }) {
  567. if (['departureAirport', 'landingAirport'].includes(column.property)) {
  568. return 'pre-line'
  569. }
  570. },
  571. cellClass({ row, column, rowIndex, columnIndex }) {
  572. const classes = []
  573. if (column.property === 'flightNO') {
  574. classes.push('cell-click')
  575. }
  576. if (['departureAirport', 'landingAirport'].includes(column.property)) {
  577. classes.push('pre-line')
  578. }
  579. return classes.join(' ')
  580. },
  581. cellClickHandler(row, column, cell, event) {
  582. switch (column.property) {
  583. case 'flightNO':
  584. this.$router.push({
  585. path: `/${this.$route.path.split('/').slice(1, -1).join('/')}/flightView`,
  586. query: {
  587. flightNO: row.flightNO,
  588. flightDate: row.flightDate
  589. }
  590. })
  591. break
  592. default:
  593. break
  594. }
  595. },
  596. tableSpanMethod({ row, column, rowIndex, columnIndex }) {
  597. if (['flightNO', 'flightDate', 'departureAirport', 'landingAirport'].includes(column['property'])) {
  598. const _row = this.spanArr[rowIndex]
  599. const _col = _row > 0 ? 1 : 0
  600. return {
  601. rowspan: _row,
  602. colspan: _col
  603. }
  604. }
  605. },
  606. exportHandler(refName, tableName) {
  607. const table = this.$refs[refName].$el.cloneNode(true)
  608. const fileName = `${tableName}-${this.queryData.bagSN}-${this.queryData.flightNO}-${this.queryData.flightDate}.xlsx`
  609. throttledExportToExcel(table, tableName, fileName)
  610. },
  611. exportMessageToExcel() {
  612. this.$message.info('开发中')
  613. },
  614. // 行李详情基础信息
  615. queryBaggageBasicInfo(dataContent) {
  616. return myQuery(DATACONTENT_ID.baggageBasicInfo, ...dataContent)
  617. },
  618. // 行李航段
  619. queryBaggageAirline(dataContent) {
  620. return myQuery(DATACONTENT_ID.baggageAirline, ...dataContent)
  621. },
  622. // 行李详情追踪链
  623. queryBaggageTrack(dataContent) {
  624. return myQuery(DATACONTENT_ID.baggageTrack, ...dataContent)
  625. },
  626. // 行李详情表格
  627. queryBaggageDetails(dataContent) {
  628. return myQuery(DATACONTENT_ID.baggageDetails, ...dataContent)
  629. },
  630. // 原始报文
  631. async queryBaggageMessage(dataContent) {
  632. try {
  633. const { code, returnData, message } = await BaggageMessageQuery({
  634. id: DATACONTENT_ID.baggageMessage,
  635. dataContent
  636. })
  637. if (Number(code) === 0) {
  638. return returnData.listValues
  639. } else {
  640. this.$message.error(message ?? '失败')
  641. }
  642. } catch (error) {
  643. console.log('出错了', error.message || error)
  644. }
  645. },
  646. async queryBasicInfo(queryData = this.queryData) {
  647. const { flightNO, flightDate, bagSN } = queryData
  648. const dataContent = [flightNO, flightDate, bagSN]
  649. try {
  650. const baggageBasicInfo = await this.queryBaggageBasicInfo(dataContent)
  651. if (baggageBasicInfo.length) {
  652. baggageBasicInfo[0].flightNO = flightNO
  653. this.baggageBasicInfo = baggageBasicInfo[0]
  654. }
  655. } catch (error) {
  656. console.log('出错了', error.message || error)
  657. }
  658. },
  659. async queryDetails(queryData = this.queryData) {
  660. const { flightNO, flightDate, bagSN } = queryData
  661. const dataContent = [flightNO, flightDate, bagSN]
  662. try {
  663. const baggageDetails = await this.queryBaggageDetails(new Array(6).fill(dataContent).flat())
  664. this.baggageTableData = baggageDetails.map((item, index) => {
  665. if (item['dealTime']) {
  666. item['dealTime'] = item['dealTime'].replace('T', ' ')
  667. }
  668. item['departureAirport'] = `${item['departureAirport']}\n${
  669. item['departureTime'] ? item['departureTime'].replace('T', '\n') : ''
  670. }`
  671. item['landingAirport'] = `${item['landingAirport']}\n${
  672. item['landingTime'] ? item['landingTime'].replace('T', '\n') : ''
  673. }`
  674. if (item['DeviceCode'] === 'STARHUB') {
  675. item['dataSource'] = 'Manual Load'
  676. } else if (item['toLocation']?.length === 2) {
  677. item['dataSource'] = 'RFID'
  678. } else if (item['secondaryCode'] === 'R') {
  679. item['dataSource'] = 'BRS'
  680. }
  681. return item
  682. })
  683. this.initTableData(this.baggageTableData)
  684. } catch (error) {
  685. console.log('出错了', error.message || error)
  686. }
  687. },
  688. async queryAirline(queryData = this.queryData) {
  689. const { flightNO, flightDate, bagSN } = queryData
  690. const dataContent = [flightNO, flightDate, bagSN]
  691. try {
  692. const result = await this.queryBaggageAirline(dataContent)
  693. this.airlineList = result.map(({ flightNO, flightDate, departureAirport, arriveAirport, luggageSN }) => ({
  694. label: `${departureAirport}-${arriveAirport}`,
  695. value: `${flightNO},${flightDate},${luggageSN}`
  696. }))
  697. if (this.airlineList.length) {
  698. this.selectedAirline = this.airlineList[0].value
  699. } else {
  700. this.selectedAirline = ''
  701. }
  702. } catch (error) {
  703. console.log('出错了', error.message || error)
  704. }
  705. },
  706. async queryTrack() {
  707. function isSameStep(code1, code2) {
  708. const sameStepCodes = ['ARRIVED', 'TRANSFER']
  709. return sameStepCodes.includes(code1) && sameStepCodes.includes(code2)
  710. }
  711. try {
  712. const result = await this.queryBaggageTrack(this.selectedAirline.split(','))
  713. this.resetStepNodes()
  714. result.forEach(({ nodeCode, nodeName, processingTime }) => {
  715. const replaceIndex = this.stepNodes.findIndex(
  716. stepNode => stepNode.nodeCode === nodeCode || isSameStep(stepNode.nodeCode, nodeCode)
  717. )
  718. if (replaceIndex > -1) {
  719. this.stepNodes.splice(replaceIndex, 1, {
  720. nodeCode,
  721. nodeName,
  722. processingTime: processingTime.replace('T', '\n')
  723. })
  724. }
  725. })
  726. } catch (error) {
  727. console.log('出错了', error.message || error)
  728. }
  729. },
  730. async queryMessage(queryData = this.queryData) {
  731. const { flightNO, flightDate, bagSN } = queryData
  732. const dataContent = [flightNO, flightDate, bagSN]
  733. try {
  734. const result = await this.queryBaggageMessage(dataContent)
  735. const messageList = result.map(message => {
  736. return typeof message === 'string' ? JSON.parse(message) : message
  737. })
  738. this.messageList = this._.sortBy(messageList, 'ssid')
  739. } catch (error) {
  740. console.log('出错了', error.message || error)
  741. }
  742. }
  743. }
  744. }
  745. </script>
  746. <style lang="scss">
  747. .radioBtn {
  748. padding: 5px;
  749. background: #000d2a;
  750. .el-radio-button__inner {
  751. background: #000d2a;
  752. color: #fff;
  753. border: none;
  754. font-weight: bold;
  755. }
  756. .el-radio-button:first-child .el-radio-button__inner {
  757. border: none;
  758. }
  759. }
  760. </style>
  761. <style lang="scss" scoped>
  762. .baggage-view {
  763. width: 100%;
  764. height: calc(100vh - 81px);
  765. overflow: hidden;
  766. background: #dfe3ea;
  767. padding: 8px 8px 0;
  768. .part1 {
  769. width: 100%;
  770. // height: 232px;
  771. background: #041741;
  772. padding: 16px 30px;
  773. .title {
  774. font-size: 18px;
  775. font-weight: bold;
  776. color: #ffffff;
  777. width: 320px;
  778. display: flex;
  779. flex-direction: row;
  780. justify-content: space-between;
  781. align-items: center;
  782. }
  783. .part1_info {
  784. width: 100%;
  785. color: #fff;
  786. font-size: 14px;
  787. font-weight: 400;
  788. color: #ffffff;
  789. > .el-row > .el-col {
  790. height: 38px;
  791. line-height: 38px;
  792. display: flex;
  793. .label {
  794. flex-basis: 126px;
  795. text-align: right;
  796. }
  797. .content {
  798. flex: 1;
  799. margin: 0;
  800. white-space: nowrap;
  801. overflow: hidden;
  802. text-overflow: ellipsis;
  803. }
  804. }
  805. }
  806. }
  807. .part2 {
  808. margin: 8px 0;
  809. width: 100%;
  810. padding: 24px 30px 28px;
  811. background: #ffffff;
  812. display: flex;
  813. flex-direction: row;
  814. justify-content: space-between;
  815. align-items: flex-start;
  816. .part2_info {
  817. flex: 1;
  818. display: flex;
  819. flex-direction: row;
  820. justify-content: flex-start;
  821. align-items: flex-start;
  822. line-height: 42px;
  823. .title {
  824. width: 120px;
  825. font-size: 18px;
  826. font-weight: bold;
  827. color: #303133;
  828. margin-right: 20px;
  829. }
  830. .type {
  831. font-size: 18px;
  832. font-weight: bold;
  833. margin-right: 20px;
  834. .warn {
  835. color: #df3559;
  836. }
  837. .normal {
  838. color: #519f6b;
  839. }
  840. }
  841. .airline {
  842. width: 120px;
  843. margin-right: 20px;
  844. }
  845. .baggage-track-chart {
  846. flex: 1;
  847. height: 124px;
  848. max-width: 1280px;
  849. position: relative;
  850. display: flex;
  851. flex-direction: row;
  852. justify-content: space-between;
  853. width: 100%;
  854. }
  855. .step-line {
  856. width: calc(100% - 80px);
  857. height: 10px;
  858. position: absolute;
  859. top: 16px;
  860. right: 0;
  861. left: 0;
  862. margin: auto;
  863. display: flex;
  864. .step-line-segment {
  865. width: calc(100% / 6);
  866. height: 100%;
  867. background: #afb4bf;
  868. &.step-line-active {
  869. background: #2d67e3;
  870. }
  871. }
  872. }
  873. .step-item {
  874. width: 80px;
  875. height: 100%;
  876. text-align: center;
  877. font-size: 12px;
  878. display: flex;
  879. flex-direction: column;
  880. align-items: center;
  881. justify-content: flex-start;
  882. z-index: 1;
  883. font-family: Helvetica, 'Microsoft Yahei';
  884. .step-circle {
  885. width: 42px;
  886. height: 42px;
  887. border-radius: 50%;
  888. background: #aaacb2;
  889. .step-name {
  890. color: #ffffff;
  891. font-size: 14px;
  892. font-weight: bold;
  893. }
  894. }
  895. .step-info {
  896. margin-top: 15px;
  897. color: #101116;
  898. line-height: 1;
  899. .step-time {
  900. white-space: pre-line;
  901. }
  902. }
  903. &.active-item .step-circle {
  904. background: #2d67e3;
  905. }
  906. }
  907. }
  908. .btns {
  909. margin-top: 6px;
  910. }
  911. }
  912. .part3 {
  913. width: 100%;
  914. // header-80px、part1-232px、part2-128px、间隙3*8px、底部44px
  915. // height: calc(100vh - 80px - 232px - 128px - 3 * 8px - 44px);
  916. background: #ffffff;
  917. ::v-deep .el-table {
  918. width: 100%;
  919. // &.el-table--striped {
  920. // .el-table__body tr.el-table__row--striped td.el-table__cell,
  921. // .el-table__header .el-table__cell {
  922. // background: #ffffff;
  923. // }
  924. // }
  925. .el-table__cell {
  926. // background: #f0f3f7;
  927. padding: 0;
  928. &.cell-click {
  929. cursor: pointer;
  930. .cell {
  931. color: #2d7cff;
  932. }
  933. }
  934. .cell {
  935. padding: 0;
  936. word-spacing: 0;
  937. font-size: 14px;
  938. font-family: Helvetica, 'Microsoft YaHei';
  939. font-weight: 400;
  940. color: #303133;
  941. width: 100% !important;
  942. .cell-content {
  943. padding: 6px 0;
  944. }
  945. }
  946. }
  947. .el-table__body .el-table__cell .cell {
  948. padding: 6px 10px;
  949. .cell-content {
  950. display: inline;
  951. padding: 0;
  952. }
  953. }
  954. }
  955. }
  956. .part4 {
  957. width: 100%;
  958. // height: calc(100vh - 80px - 232px - 2 * 8px - 44px);
  959. .head {
  960. padding: 16px 24px 11px 30px;
  961. background: transparent;
  962. display: flex;
  963. justify-content: space-between;
  964. .title {
  965. line-height: 30px;
  966. font-size: 18px;
  967. font-weight: bold;
  968. color: #303133;
  969. }
  970. }
  971. .main {
  972. height: calc(100% - 57px);
  973. overflow-y: auto;
  974. overflow-x: hidden;
  975. ::v-deep .el-row {
  976. flex-wrap: wrap;
  977. .card {
  978. width: 100%;
  979. min-height: 440px;
  980. padding: 20px;
  981. background: #ffffff;
  982. box-shadow: 0px 3px 2px 0px rgba(0, 0, 0, 0.29);
  983. margin-bottom: 24px;
  984. > .message-date {
  985. width: 180px;
  986. height: 26px;
  987. line-height: 14px;
  988. font-size: 14px;
  989. font-family: Helvetica;
  990. color: #afb4bf;
  991. border-bottom: 1px solid #afb4bf;
  992. margin-bottom: 18px;
  993. }
  994. > .message-content {
  995. white-space: pre-line;
  996. line-height: 24px;
  997. font-size: 14px;
  998. color: #303133;
  999. word-break: break-all;
  1000. }
  1001. }
  1002. }
  1003. }
  1004. }
  1005. .btns {
  1006. height: 30px;
  1007. display: flex;
  1008. .btn-square {
  1009. margin-left: 10px;
  1010. width: 30px;
  1011. }
  1012. }
  1013. }
  1014. </style>
  1015. <style lang="scss">
  1016. .el-popover {
  1017. &.popover-dark {
  1018. background: #303133;
  1019. color: #ffffff;
  1020. border: none;
  1021. }
  1022. .pre-line {
  1023. white-space: pre-line;
  1024. }
  1025. }
  1026. .el-popper[x-placement^='top'].popover-dark .popper__arrow::after {
  1027. border-top-color: #303133;
  1028. }
  1029. .el-popper[x-placement^='right'].popover-dark .popper__arrow::after {
  1030. border-right-color: #303133;
  1031. }
  1032. .el-popper[x-placement^='bottom'].popover-dark .popper__arrow::after {
  1033. border-bottom-color: #303133;
  1034. }
  1035. .el-popper[x-placement^='left'].popover-dark .popper__arrow::after {
  1036. border-left-color: #303133;
  1037. }
  1038. </style>