advancedHome.vue 39 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366
  1. <template>
  2. <div class="advance">
  3. <div class="advance__head flex">
  4. <div class="flex-wrap interfaceLog_head_time">
  5. <div class="manageTitle">高级查询</div>
  6. <!-- <div class="interfaceLog_head_time_start mr10">
  7. <el-date-picker
  8. v-model="flightDate[0]"
  9. class="input-shadow"
  10. size="small"
  11. type="date"
  12. value-format="yyyy-MM-dd"
  13. placeholder="选择开始日期时间"
  14. @change="startDateChangeHandler"
  15. />
  16. </div>
  17. <div class="interfaceLog_head_time_end">
  18. <el-date-picker
  19. v-model="flightDate[1]"
  20. class="input-shadow"
  21. size="small"
  22. type="date"
  23. value-format="yyyy-MM-dd"
  24. placeholder="选择结束日期时间"
  25. @change="endDateChangeHandler"
  26. />
  27. </div> -->
  28. <el-date-picker
  29. v-model="flightDate"
  30. size="small"
  31. type="daterange"
  32. value-format="yyyy-MM-dd"
  33. start-placeholder="开始日期"
  34. end-placeholder="结束日期"
  35. :picker-options="dateRangePickerOptions"
  36. :clearable="false"
  37. />
  38. </div>
  39. <Search
  40. ref="search"
  41. class="advanced-search"
  42. :is-title="false"
  43. :is-slot="true"
  44. :search-tooltip="'请输入航班号(示例:CA1234)或行李牌号(示例:1234567890)'"
  45. @getSearchData="getSearchData"
  46. @clearSearchData="clearSearchData"
  47. >
  48. <div class="flex-wrap">
  49. <button
  50. class="btnAn"
  51. @click="dialogShow"
  52. >
  53. 高级查询
  54. </button>
  55. <button
  56. class="btnAn"
  57. @click="exportHandler('table', '高级查询结果')"
  58. >
  59. 导出
  60. </button>
  61. <button
  62. class="btnAn"
  63. @click="toNewAdvance"
  64. >
  65. 切换
  66. </button>
  67. <!-- <div
  68. class="setting"
  69. @click="show"
  70. /> -->
  71. </div>
  72. </Search>
  73. </div>
  74. <!--表格-->
  75. <div
  76. v-loading="loading"
  77. element-loading-text="拼命加载中"
  78. element-loading-spinner="el-icon-loading"
  79. element-loading-background="rgba(0, 0, 0, 0.8)"
  80. class="advance__table"
  81. >
  82. <el-table
  83. ref="table"
  84. v-el-table-infinite-scroll="load"
  85. max-height="100%"
  86. class="table"
  87. :data="dealedTableData"
  88. border
  89. stripe
  90. fit
  91. height="calc(100vh - 158px)"
  92. style="width: 100%"
  93. show-summary
  94. :summary-method="summaryRow(dealedTableData.length)"
  95. :header-cell-class-name="headerCellClass"
  96. :row-class-name="tableRowClassName"
  97. :cell-class-name="cellClass"
  98. :span-method="objectSpanMethod"
  99. @cell-click="cellClickHandler"
  100. >
  101. <el-table-column
  102. v-for="col in tableCols"
  103. :key="col.prop"
  104. :prop="col.prop"
  105. :label="col.label"
  106. :width="col.width"
  107. :fixed="col.fixed"
  108. :formatter="tableFormat"
  109. >
  110. <template #header>
  111. <el-tooltip
  112. :content="col.desc || col.label"
  113. placement="top"
  114. >
  115. <TableHeaderCell
  116. :label="col.label"
  117. :filter-options="tableDataFilters[col.prop]"
  118. :filter-values.sync="filterValues[col.prop]"
  119. :sortable="col.sortable"
  120. :sort-rule.sync="tableDataSortRules[col.prop]"
  121. />
  122. </el-tooltip>
  123. </template>
  124. </el-table-column>
  125. </el-table>
  126. </div>
  127. <!--列设置-->
  128. <!-- <Dialog
  129. :flag="dialogFlag"
  130. class="dialog-check-group"
  131. >
  132. <div class="dialog-wrapper">
  133. <div class="title">列设置</div>
  134. <div class="content">
  135. <el-tree
  136. :data="tableCols"
  137. :class="colsCheckClass"
  138. show-checkbox
  139. node-key="index"
  140. :default-expand-all="true"
  141. :props="{
  142. label: 'label',
  143. children: 'children'
  144. }"
  145. :default-checked-keys="checkedKeysTemp"
  146. @check="handleCheck"
  147. />
  148. </div>
  149. <div class="foot right t30">
  150. <el-button
  151. size="medium"
  152. class="r24"
  153. type="primary"
  154. @click="onCheck"
  155. >确定</el-button>
  156. <el-button
  157. size="medium"
  158. @click="hide"
  159. >取消</el-button>
  160. </div>
  161. </div>
  162. </Dialog> -->
  163. <!--高级查询-->
  164. <Dialog
  165. width="900px"
  166. :flag="gjFlag"
  167. >
  168. <div
  169. ref="dialog"
  170. class="advanced-dialog"
  171. :tabindex="0"
  172. @keyup.enter="advancedSubmitHandler(false)"
  173. @keyup.self.esc="gjFlag = false"
  174. >
  175. <div class="title">高级查询</div>
  176. <div class="content">
  177. <el-form
  178. ref="form"
  179. :model="form"
  180. :rules="rules"
  181. label-width="100px"
  182. >
  183. <el-row :gutter="20">
  184. <el-col
  185. v-for="item in formItems"
  186. :key="item.prop"
  187. :span="item.span || 8"
  188. >
  189. <el-form-item
  190. :label="item.label"
  191. :prop="item.prop"
  192. >
  193. <template v-if="item.prop === 'flightDate'">
  194. <el-date-picker
  195. v-model="flightDate"
  196. size="small"
  197. type="daterange"
  198. value-format="yyyy-MM-dd"
  199. start-placeholder="开始日期"
  200. end-placeholder="结束日期"
  201. :picker-options="dateRangePickerOptions"
  202. :clearable="false"
  203. @keyup.esc.native="dialogFocus"
  204. />
  205. </template>
  206. <template v-else-if="item.prop === 'status'">
  207. <el-select
  208. v-model="form.status"
  209. size="small"
  210. filterable
  211. default-first-option
  212. clearable
  213. @keyup.esc.native="dialogFocus"
  214. >
  215. <el-option
  216. v-for="(option, index) in statusList"
  217. :key="index"
  218. :label="option.statusName"
  219. :value="option.statusCode"
  220. />
  221. </el-select>
  222. </template>
  223. <template v-else-if="item.prop === 'specialType'">
  224. <el-select
  225. v-model="form.specialType"
  226. size="small"
  227. class="multiple-select"
  228. multiple
  229. collapse-tags
  230. filterable
  231. allow-create
  232. default-first-option
  233. clearable
  234. @keyup.esc.native="dialogFocus"
  235. >
  236. <el-option
  237. v-for="option in baggageTypeList"
  238. :key="option.specialType"
  239. :label="option.specialType"
  240. :value="option.specialType"
  241. />
  242. </el-select>
  243. </template>
  244. <template v-else-if="item.prop === 'unLoad'">
  245. <el-select
  246. v-model="form.unLoad"
  247. size="small"
  248. clearable
  249. @keyup.esc.native="dialogFocus"
  250. >
  251. <el-option
  252. label="已翻减"
  253. :value="1"
  254. />
  255. <el-option
  256. label="待翻减"
  257. :value="0"
  258. />
  259. </el-select>
  260. </template>
  261. <template v-else-if="['checkIn', 'active', 'transferIn', 'canceled'].includes(item.prop)">
  262. <el-select
  263. v-model="form[item.prop]"
  264. size="small"
  265. clearable
  266. @keyup.esc.native="dialogFocus"
  267. >
  268. <el-option
  269. label="是"
  270. :value="1"
  271. />
  272. <el-option
  273. label="否"
  274. :value="0"
  275. />
  276. </el-select>
  277. </template>
  278. <template v-else-if="item.prop === 'noBSM'">
  279. <el-select
  280. v-model="form.noBSM"
  281. size="small"
  282. clearable
  283. @keyup.esc.native="dialogFocus"
  284. >
  285. <el-option
  286. label="是"
  287. :value="0"
  288. />
  289. <el-option
  290. label="否"
  291. :value="1"
  292. />
  293. </el-select>
  294. </template>
  295. <template v-else-if="item.prop === 'loadType'">
  296. <el-select
  297. v-model="form.loadType"
  298. size="small"
  299. clearable
  300. @keyup.esc.native="dialogFocus"
  301. >
  302. <el-option
  303. label="容器"
  304. :value="0"
  305. />
  306. <el-option
  307. label="散装"
  308. :value="1"
  309. />
  310. </el-select>
  311. </template>
  312. <template v-else>
  313. <el-popover
  314. v-model="item.hintVisible"
  315. placement="right"
  316. trigger="manual"
  317. >
  318. <span>{{ item.hintText }}</span>
  319. <el-input
  320. :ref="'input-' + item.prop"
  321. slot="reference"
  322. v-model="form[item.prop]"
  323. size="small"
  324. @focus="item.hintVisible = true"
  325. @blur="item.hintVisible = false"
  326. @keyup.esc.native="dialogFocus"
  327. />
  328. </el-popover>
  329. </template>
  330. </el-form-item>
  331. </el-col>
  332. </el-row>
  333. </el-form>
  334. </div>
  335. <div class="foot right t30">
  336. <el-button
  337. size="medium"
  338. class="r24"
  339. type="primary"
  340. @click="advancedSubmitHandler(false)"
  341. >确定</el-button
  342. >
  343. <el-button
  344. size="medium"
  345. @click="dialogHide"
  346. >取消</el-button
  347. >
  348. </div>
  349. </div>
  350. </Dialog>
  351. </div>
  352. </template>
  353. <script>
  354. import Search from '@/components/SearchWithTooltip'
  355. import Dialog from '@/layout/components/Dialog'
  356. import { parseTime } from '@/utils/index'
  357. import { Query, myQuery } from '@/api/dataIntegration'
  358. import { mapGetters } from 'vuex'
  359. import TableHeaderCell from '@/components/TableHeaderCell'
  360. import { setTableFilters, throttledExportToExcel } from '@/utils/table'
  361. export default {
  362. name: 'AdvancedHome',
  363. components: { Search, Dialog, TableHeaderCell },
  364. data() {
  365. return {
  366. loading: false,
  367. colDialogFlag: false,
  368. gjFlag: false,
  369. tableData: [],
  370. page: -1,
  371. noMore: false,
  372. checkList: [],
  373. tableCols: [
  374. {
  375. prop: 'flightNO',
  376. label: '航班号',
  377. desc: '指航班编号',
  378. fixed: 'left',
  379. filterable: true,
  380. sortable: true
  381. },
  382. {
  383. prop: 'flightDate',
  384. label: '航班日期',
  385. desc: '指航班计划起飞日期(机票上的静态数据)',
  386. fixed: 'left',
  387. width: 110,
  388. filterable: true,
  389. sortable: true
  390. },
  391. {
  392. prop: 'departureTime',
  393. label: '起飞时间',
  394. desc: '指航班预计起飞时间,动态数据,仅显示最新结果',
  395. width: 150
  396. },
  397. {
  398. prop: 'sourceAirport',
  399. label: '起飞站',
  400. desc: '指航班执飞航段的起飞航站,以航站英文三字码显示',
  401. filterable: true,
  402. sortable: true
  403. },
  404. {
  405. prop: 'targetAirport',
  406. label: '目的地',
  407. desc: '指航班执飞航段的目的航站,以航站英文三字码显示',
  408. filterable: true,
  409. sortable: true
  410. },
  411. {
  412. prop: 'passengerNameUpcase',
  413. label: '旅客姓名',
  414. desc: '指旅客姓名的拼音大写',
  415. width: 150,
  416. filterable: true,
  417. sortable: true
  418. },
  419. {
  420. prop: 'bagSN',
  421. label: '行李牌号',
  422. desc: '指行李的10位数字行李牌号码',
  423. width: 110,
  424. filterable: true,
  425. sortable: true
  426. },
  427. {
  428. prop: 'specialType',
  429. label: '特殊行李类型',
  430. desc: '指有别于普通托运行李的特殊行李分类,包括(装笼动物、机组行李、易碎行李、VIP行李等),参考BSM报文.E项说明',
  431. width: 115,
  432. filterable: true,
  433. sortable: true
  434. },
  435. {
  436. prop: 'deleted',
  437. label: '删除',
  438. desc: '指旅客是否取消值机托运,根据BSM报文状态是否有DEL判断,已删除的行李记录为斜体灰色字体',
  439. filterable: true,
  440. sortable: true
  441. },
  442. {
  443. prop: 'activated',
  444. label: '激活',
  445. desc: '指托运行李是否被激活,参照BSM报文.S项说明',
  446. filterable: true,
  447. sortable: true
  448. },
  449. {
  450. prop: 'bagWeight',
  451. label: '重量',
  452. desc: '指托运行李的重量,参照BSM报文.W项说明'
  453. },
  454. {
  455. prop: 'latestStatus',
  456. label: '最新状态',
  457. desc: '指托运行李的当前查询时间所在的节点状态',
  458. width: 110,
  459. filterable: true,
  460. sortable: true
  461. },
  462. {
  463. prop: 'bagLocation',
  464. label: '最新位置',
  465. desc: '指托运行李的当前查询时间所在的节点状态的识读位置代号',
  466. width: 110,
  467. filterable: true,
  468. sortable: true
  469. },
  470. {
  471. prop: 'U_Device_ID',
  472. label: '容器编号',
  473. desc: '指集装器ID信息',
  474. width: 110,
  475. filterable: true,
  476. sortable: true
  477. },
  478. {
  479. prop: 'preFlightNO',
  480. label: '中转进航班',
  481. desc: '指有中转行李转出的进港航班号',
  482. width: 110,
  483. filterable: true,
  484. sortable: true
  485. },
  486. {
  487. prop: 'transferFlightNO',
  488. label: '中转出航班',
  489. desc: '指有中转行李转入的离港航班号',
  490. width: 110,
  491. filterable: true,
  492. sortable: true
  493. }
  494. ],
  495. flightDate: [parseTime(new Date(), '{y}-{m}-{d}'), parseTime(new Date(), '{y}-{m}-{d}')],
  496. dateRangePickerOptions: {
  497. onPick: this.dateRangePickHandler,
  498. disabledDate: this.dateRangeDisabled
  499. },
  500. form: {
  501. flightNO: '',
  502. destination: '',
  503. departureStation: '',
  504. baggageNO: '',
  505. specialType: [],
  506. loadType: '',
  507. U_Device_ID: '',
  508. passengerName: '',
  509. PNR: '',
  510. checkInSequence: '',
  511. transferArrival: '',
  512. transferDeparture: '',
  513. unLoad: '',
  514. checkIn: '',
  515. active: '',
  516. transferIn: '',
  517. canceled: '',
  518. status: '',
  519. noBSM: ''
  520. },
  521. formItems: [
  522. {
  523. prop: 'flightDate',
  524. label: '航班日期',
  525. span: 16
  526. },
  527. {
  528. prop: 'status',
  529. label: '当前状态'
  530. },
  531. {
  532. prop: 'flightNO',
  533. label: '航班号',
  534. hintText: '示例:CA1234',
  535. hintVisible: false
  536. },
  537. {
  538. prop: 'departureStation',
  539. label: '起飞站',
  540. hintText: '示例:PEK',
  541. hintVisible: false
  542. },
  543. {
  544. prop: 'destination',
  545. label: '目的地',
  546. hintText: '示例:CTU',
  547. hintVisible: false
  548. },
  549. {
  550. prop: 'baggageNO',
  551. label: '行李牌号',
  552. hintText: '示例:1234567890 或 CA123456',
  553. hintVisible: false
  554. },
  555. {
  556. prop: 'specialType',
  557. label: '特殊行李类型'
  558. },
  559. {
  560. prop: 'loadType',
  561. label: '装载类型'
  562. },
  563. {
  564. prop: 'U_Device_ID',
  565. label: '容器编号',
  566. hintText: '示例:AKE12345CA',
  567. hintVisible: false
  568. },
  569. {
  570. prop: 'passengerName',
  571. label: '旅客姓名',
  572. hintText: '示例:ZHAOWEI',
  573. hintVisible: false
  574. },
  575. {
  576. prop: 'PNR',
  577. label: 'PNR',
  578. hintText: '示例:PZR25X',
  579. hintVisible: false
  580. },
  581. {
  582. prop: 'checkInSequence',
  583. label: '值机序号',
  584. hintText: '示例:001',
  585. hintVisible: false
  586. },
  587. {
  588. prop: 'transferArrival',
  589. label: '中转进航班',
  590. hintText: '示例:CA1234',
  591. hintVisible: false
  592. },
  593. {
  594. prop: 'transferDeparture',
  595. label: '中转出航班',
  596. hintText: '示例:CA1234',
  597. hintVisible: false
  598. },
  599. {
  600. prop: 'unLoad',
  601. label: '翻减状态'
  602. },
  603. // {
  604. // prop: 'checkIn',
  605. // label: '已值机'
  606. // },
  607. {
  608. prop: 'active',
  609. label: '已激活'
  610. },
  611. {
  612. prop: 'transferIn',
  613. label: '中转行李'
  614. },
  615. {
  616. prop: 'canceled',
  617. label: '已取消'
  618. },
  619. {
  620. prop: 'noBSM',
  621. label: '无BSM'
  622. }
  623. ],
  624. statusList: [
  625. {
  626. statusName: '值机',
  627. statusCode: '值机'
  628. },
  629. {
  630. statusName: '安检',
  631. statusCode: '安检'
  632. },
  633. {
  634. statusName: '分拣',
  635. statusCode: '分拣'
  636. },
  637. {
  638. statusName: '装车',
  639. statusCode: '装车'
  640. },
  641. {
  642. statusName: '装机',
  643. statusCode: '装机'
  644. },
  645. {
  646. statusName: '到达',
  647. statusCode: '到达'
  648. },
  649. {
  650. statusName: '卸机',
  651. statusCode: '卸机'
  652. }
  653. // {
  654. // statusName: '已中转',
  655. // statusCode: '已中转'
  656. // }
  657. ],
  658. baggageTypeList: [],
  659. dataContent: [],
  660. rules: {
  661. // 机器信息表单验证
  662. // flightNO: [
  663. // { required: true, message: "请输入有效航班号", trigger: "blur" },
  664. // ],
  665. },
  666. tableDataFilters: {},
  667. filterValues: {},
  668. tableDataSortRules: {},
  669. spanArr: [],
  670. contactDot: 0,
  671. scrollTop: 0,
  672. dom: null
  673. }
  674. },
  675. computed: {
  676. ...mapGetters(['clickedCells', 'queryForm']),
  677. dealedTableData() {
  678. const filtered = this.tableData.filter(item => {
  679. let flag = true
  680. Object.entries(this.filterValues).forEach(([key, arr]) => {
  681. if (arr.length && !arr.includes(String(item[key]))) {
  682. flag = false
  683. }
  684. })
  685. return flag
  686. })
  687. const sortRules = Object.entries(this.tableDataSortRules).reduce(
  688. (pre, [key, value]) => {
  689. if (value) {
  690. pre[0].push(key)
  691. value = value === 'ascending' ? 'asc' : 'desc'
  692. pre[1].push(value)
  693. }
  694. return pre
  695. },
  696. [[], []]
  697. )
  698. return this._.orderBy(filtered, sortRules[0], sortRules[1])
  699. }
  700. },
  701. watch: {
  702. $route: {
  703. handler({ path, query }) {
  704. if (path === '/advance') {
  705. this.queryChangeHandler(query)
  706. }
  707. },
  708. deep: true
  709. },
  710. flightDate: {
  711. handler(val) {
  712. if (val === null) {
  713. this.flightDate = ['', '']
  714. }
  715. },
  716. deep: true
  717. },
  718. dealedTableData: {
  719. handler(val) {
  720. this.spanArr = []
  721. let contactDot = this.contactDot
  722. val.forEach((item, index, arr) => {
  723. if (index === 0) {
  724. this.spanArr.push(1)
  725. } else {
  726. if (
  727. item['flightNO'] === arr[index - 1]['flightNO'] &&
  728. item['flightDate'] === arr[index - 1]['flightDate'] &&
  729. item['passengerNameUpcase'] === arr[index - 1]['passengerNameUpcase'] &&
  730. item['checkInNO'] === arr[index - 1]['checkInNO'] &&
  731. item['bagWeight'] === arr[index - 1]['bagWeight']
  732. ) {
  733. this.spanArr[contactDot] += 1
  734. this.spanArr.push(0)
  735. } else {
  736. this.spanArr.push(1)
  737. contactDot = index
  738. }
  739. }
  740. })
  741. },
  742. deep: true
  743. }
  744. },
  745. created() {
  746. Object.values(this.tableCols).forEach(({ prop, filterable, sortable }) => {
  747. if (filterable) {
  748. this.$set(this.tableDataFilters, prop, [])
  749. this.$set(this.filterValues, prop, [])
  750. }
  751. if (sortable) {
  752. this.$set(this.tableDataSortRules, prop, '')
  753. }
  754. })
  755. },
  756. mounted() {
  757. this.baggageTypeQuery()
  758. this.queryChangeHandler(this.$route.query)
  759. const that = this
  760. this.dom = this.$refs.table.bodyWrapper
  761. this.dom.addEventListener('scroll', () => {
  762. that.scrollTop = this.dom.scrollTop
  763. })
  764. },
  765. activated() {
  766. this.dom.scrollTop = this.scrollTop
  767. // else if (this.queryForm) {
  768. // Object.keys(this.form).forEach(key => {
  769. // this.form[key] = this.queryForm[key]
  770. // })
  771. // this.flightDate = this.queryForm.flightDate
  772. // this.advancedSubmitHandler()
  773. // }
  774. },
  775. updated() {
  776. // table数据更新
  777. this.$nextTick(() => {
  778. this.$refs.table.doLayout()
  779. })
  780. },
  781. deactivated() {
  782. this.loading = false
  783. },
  784. beforeDestroy() {
  785. this.loading = false
  786. //
  787. // if (this.$route.matched.filter(item => item.name && item.meta.title).length > 1) {
  788. // this.$store.dispatch('app/setQueryForm', {
  789. // ...this.form,
  790. // flightDate: this.flightDate
  791. // })
  792. // } else {
  793. // this.$store.dispatch('app/setQueryForm', null)
  794. // }
  795. },
  796. methods: {
  797. load() {
  798. if (this.noMore || this.loading || this.page < 0) {
  799. return
  800. }
  801. this.advancedQuery(this.dataContent)
  802. },
  803. resetTable() {
  804. this.page = 0
  805. this.noMore = false
  806. this.tableData = []
  807. },
  808. dialogShow() {
  809. this.gjFlag = true
  810. this.$nextTick(() => {
  811. this.dialogFocus()
  812. })
  813. },
  814. dialogHide() {
  815. this.gjFlag = false
  816. },
  817. toNewAdvance() {
  818. this.$router.push('/advance/advanceNew')
  819. },
  820. dialogFocus() {
  821. this.$refs['dialog'].focus()
  822. },
  823. queryChangeHandler(query) {
  824. let queryFlag = false
  825. const { startDate, endDate, singleJump } = query
  826. const queryEntries = Object.entries(query)
  827. if (queryEntries.length) {
  828. this.clearForm()
  829. }
  830. queryEntries.forEach(([key, value]) => {
  831. if (!['startDate', 'endDate', 'singleJump'].includes(key) && (value ?? '') !== '') {
  832. queryFlag = true
  833. this.form[key] = ['unLoad', 'checkIn', 'active', 'transferIn', 'canceled', 'noBSM'].includes(key)
  834. ? Number(value)
  835. : value
  836. }
  837. })
  838. if (startDate) {
  839. this.$set(this.flightDate, 0, startDate)
  840. }
  841. if (endDate) {
  842. this.$set(this.flightDate, 1, endDate)
  843. }
  844. if (queryFlag) {
  845. this.advancedSubmitHandler(singleJump)
  846. }
  847. },
  848. // startDateChangeHandler(val) {
  849. // this.flightDate[0] = val ?? ''
  850. // if (!val || !this.flightDate[1]) {
  851. // return
  852. // }
  853. // const startDate = new Date(val)
  854. // const endDate = new Date(this.flightDate[1])
  855. // if (startDate > endDate) {
  856. // this.flightDate.splice(1, 1, '')
  857. // this.$message.info('结束时间不能早于开始时间,请重新选择')
  858. // } else if (endDate - startDate > 2 * 24 * 60 * 60 * 1000) {
  859. // this.flightDate.splice(1, 1, '')
  860. // this.$message.info('时间跨度不能超过三天,请重新选择')
  861. // }
  862. // },
  863. // endDateChangeHandler(val) {
  864. // this.flightDate[1] = val ?? ''
  865. // if (!val || !this.flightDate[0]) {
  866. // return
  867. // }
  868. // const startDate = new Date(this.flightDate[0])
  869. // const endDate = new Date(val)
  870. // if (startDate > endDate) {
  871. // this.flightDate.splice(0, 1, '')
  872. // this.$message.info('开始时间不能晚于结束时间,请重新选择')
  873. // } else if (endDate - startDate > 2 * 24 * 60 * 60 * 1000) {
  874. // this.flightDate.splice(0, 1, '')
  875. // this.$message.info('时间跨度不能超过三天,请重新选择')
  876. // }
  877. // },
  878. dateRangePickHandler({ maxDate, minDate }) {
  879. if (!maxDate) {
  880. this.pickedDate = minDate
  881. } else {
  882. this.pickedDate = null
  883. }
  884. },
  885. dateRangeDisabled(date) {
  886. return this.pickedDate ? Math.abs(date - this.pickedDate) > 2 * 24 * 60 * 60 * 1000 : false
  887. },
  888. objectSpanMethod({ row, column, rowIndex, columnIndex }) {
  889. if (['passengerNameUpcase', 'bagWeight'].includes(column.property)) {
  890. const _row = this.spanArr[rowIndex]
  891. const _col = _row > 0 ? 1 : 0
  892. return {
  893. rowspan: _row,
  894. colspan: _col
  895. }
  896. }
  897. },
  898. // 给表头单元格加上 ascending 或 descending 使用 element 自带的排序箭头变色
  899. headerCellClass({ row, column, rowIndex, columnIndex }) {
  900. const classes = []
  901. const rule = this.tableDataSortRules[column.property]
  902. if (rule) {
  903. classes.push(rule)
  904. }
  905. return classes.join(' ')
  906. },
  907. tableRowClassName({ row, rowIndex }) {
  908. const classes = []
  909. if (row.deleted === 'DEL') {
  910. classes.push('bgl-deleted')
  911. }
  912. return classes.join(' ')
  913. },
  914. cellClass({ row, column, rowIndex, columnIndex }) {
  915. const classes = []
  916. if (
  917. ['flightNO', 'passengerNameUpcase', 'bagSN', 'U_Device_ID', 'preFlightNO', 'transferFlightNO'].includes(
  918. column.property
  919. ) &&
  920. row[column.property] &&
  921. row[column.property] !== 'FBULK'
  922. ) {
  923. classes.push('cell-click')
  924. if (
  925. this.clickedCells.some(
  926. cell =>
  927. cell.pageName === 'advance' &&
  928. Object.entries(cell.row).every(([key, value]) => row[key] === value) &&
  929. cell.columnProp === column.property
  930. )
  931. ) {
  932. classes.push('cell-clicked')
  933. }
  934. }
  935. return classes.join(' ')
  936. },
  937. cellClickHandler(row, column, cell, event) {
  938. if (
  939. ['flightNO', 'passengerNameUpcase', 'bagSN', 'U_Device_ID', 'preFlightNO', 'transferFlightNO'].includes(
  940. column.property
  941. ) &&
  942. row[column.property] &&
  943. row[column.property] !== 'FBULK'
  944. ) {
  945. this.$store.dispatch('keepAlive/addClickedCell', {
  946. row,
  947. columnProp: column.property,
  948. pageName: 'advance'
  949. })
  950. switch (column.property) {
  951. case 'flightNO':
  952. this.$router.push({
  953. path: '/advance/flightView',
  954. query: {
  955. flightNO: row.flightNO,
  956. flightDate: row.flightDate
  957. }
  958. })
  959. break
  960. case 'passengerNameUpcase':
  961. this.$store.dispatch('app/setPassengerQueryParams', {
  962. flightNO: row.flightNO,
  963. flightDate: row.flightDate,
  964. passengerName: row.passengerNameUpcase
  965. })
  966. this.$store.dispatch('app/togglePassengerDialogFlag', true)
  967. break
  968. case 'bagSN':
  969. this.$router.push({
  970. path: '/advance/baggageView',
  971. query: {
  972. flightNO: row.flightNO,
  973. flightDate: row.flightDate,
  974. bagSN: row.bagSN
  975. }
  976. })
  977. break
  978. case 'U_Device_ID':
  979. this.$router.push({
  980. path: '/advance/containerView',
  981. query: {
  982. flightNO: row.flightNO,
  983. flightDate: row.flightDate,
  984. departureAirport: row.sourceAirport,
  985. landingAirport: row.targetAirport,
  986. containerID: row.U_Device_ID
  987. }
  988. })
  989. break
  990. case 'transferFlightNO':
  991. this.$router.push({
  992. path: '/advance/flightView',
  993. query: {
  994. flightNO: row.transferFlightNO,
  995. flightDate: row.transferFlightDate
  996. }
  997. })
  998. break
  999. case 'preFlightNO':
  1000. this.$router.push({
  1001. path: '/advance/flightView',
  1002. query: {
  1003. flightNO: row.preFlightNO,
  1004. flightDate: row.preFlightDate
  1005. }
  1006. })
  1007. break
  1008. default:
  1009. break
  1010. }
  1011. }
  1012. },
  1013. tableFormat(row, column, cellValue) {
  1014. switch (column.property) {
  1015. case 'departureTime':
  1016. return (cellValue ?? '').replace('T', ' ')
  1017. case 'deleted':
  1018. return cellValue === 'DEL' ? cellValue : ''
  1019. case 'activated':
  1020. return cellValue === 1 ? '激活' : '未激活'
  1021. default:
  1022. return cellValue ?? ''
  1023. }
  1024. },
  1025. // 查询
  1026. getSearchData(val) {
  1027. if (this.flightDate[0] === '' || this.flightDate[1] === '' || val === '') {
  1028. this.$message.error('请先输入完整查询信息')
  1029. return
  1030. }
  1031. this.clearForm()
  1032. // 点击搜索后清除跳转携带的查询信息
  1033. this.$route.query && this.$router.replace(this.$route.path)
  1034. // let searchData = {dataContent:[this.time[0],this.time[1],val]}
  1035. const az = /^[a-zA-Z]+$/
  1036. const azNum = /^(?![0-9]+$)(?![a-zA-Z]+$)[0-9A-Za-z]*$/
  1037. // const top2 = /^[a-zA-Z]{2}\w*$/
  1038. const top2 = /^([a-zA-Z][0-9])|([0-9][a-zA-Z])|([a-zA-Z]{2})/
  1039. const bagNum = /^[0-9]{10}$/
  1040. const bagNo = /^[a-zA-Z]{2}[0-9]{6}$/
  1041. // const bagNumCA = /^CA[0-9]{6}$/
  1042. const bagNumShort = /^[0-9a-zA-Z][a-zA-Z][0-9]{6}$/
  1043. // 纯字母则为旅客姓名
  1044. if (az.test(val)) {
  1045. this.form['passengerName'] = val
  1046. this.advancedSubmitHandler(true)
  1047. } else if (azNum.test(val) && top2.test(val) && val.length < 8) {
  1048. // 字母加数字且前两位为字母则为航班号
  1049. this.form['flightNO'] = val
  1050. this.advancedSubmitHandler(true)
  1051. } else if (
  1052. bagNum.test(val) ||
  1053. bagNo.test(val) ||
  1054. (bagNumShort.test(val) && Object.keys(AIRPORT_ID).includes(val.slice(0, 2).toUpperCase()))
  1055. ) {
  1056. // 纯数字且位数等于10则为行李牌号
  1057. this.form['baggageNO'] = val
  1058. this.advancedSubmitHandler(true)
  1059. } else {
  1060. this.$message.error('请先输入有效查询信息如航班号、旅客姓名首字母、行李牌号')
  1061. }
  1062. },
  1063. // 清除查询
  1064. clearSearchData() {
  1065. this.clearForm()
  1066. this.resetTable()
  1067. },
  1068. // 清除表单
  1069. clearForm() {
  1070. Object.keys(this.form).forEach(key => {
  1071. this.form[key] = ['specialType'].includes(key) ? [] : ''
  1072. })
  1073. },
  1074. // 高级查询-确定
  1075. advancedSubmitHandler(singleJump) {
  1076. function digitFormat(num) {
  1077. if (num) {
  1078. num = '000' + num
  1079. return num.slice(-3)
  1080. } else {
  1081. return ''
  1082. }
  1083. }
  1084. function baggageFormat(baggageNO) {
  1085. // const bagNumCA = /^CA[0-9]{6}$/
  1086. const bagNumShort = /^[0-9a-zA-Z][a-zA-Z][0-9]{6}$/
  1087. if (bagNumShort.test(baggageNO) && Object.keys(AIRPORT_ID).includes(baggageNO.slice(0, 2).toUpperCase())) {
  1088. return Array.from(
  1089. { length: 10 },
  1090. (_, i) => `${i}${AIRPORT_ID[baggageNO.slice(0, 2).toUpperCase()]}${baggageNO.slice(2)}`
  1091. ).join()
  1092. }
  1093. return baggageNO || null
  1094. }
  1095. this.resetTable()
  1096. /* 参数顺序
  1097. 【航班开始日期,航班结束日期,航班号,航班号,行李牌号,行李牌号,起飞站,起飞站,目的站,目的站,特殊行李类型,特殊行李类型,旅客姓名大写拼音,旅客姓名大写拼音,
  1098. PNR,PNR,值机号,值机号,中转进航班,中转进航班,中转出航班,中转出航班,容器编号,容器编号,
  1099. 是否已翻减(null/OFF/其他),是否已翻减(null/OFF/其他),是否值机(null/0/1),是否值机(null/0/1),
  1100. 是否激活(null/0/1),是否激活(null/0/1),是否中转(null/0/1),是否中转(null/0/1),是否取消行李(null/0/1),是否取消行李(null/0/1)】 */
  1101. this.dataContent = []
  1102. if (!this.flightDate?.length) {
  1103. this.$message.error('请先选择要查询的时间')
  1104. return
  1105. }
  1106. if (Object.values(this.form).every(value => value === '')) {
  1107. this.$message.error('请先输入查询信息')
  1108. return
  1109. }
  1110. this.dataContent.push(...this.flightDate)
  1111. const {
  1112. flightNO,
  1113. baggageNO,
  1114. departureStation,
  1115. destination,
  1116. specialType,
  1117. passengerName,
  1118. PNR,
  1119. checkInSequence,
  1120. transferArrival,
  1121. transferDeparture,
  1122. loadType,
  1123. U_Device_ID,
  1124. unLoad,
  1125. checkIn,
  1126. active,
  1127. transferIn,
  1128. canceled,
  1129. status,
  1130. noBSM
  1131. } = this.form
  1132. this.setDataContent(
  1133. flightNO,
  1134. baggageFormat(baggageNO),
  1135. departureStation,
  1136. destination,
  1137. specialType.map(v => v.replaceAll(',', '/')).join(),
  1138. passengerName,
  1139. PNR,
  1140. digitFormat(checkInSequence),
  1141. transferArrival,
  1142. transferDeparture,
  1143. U_Device_ID,
  1144. unLoad,
  1145. checkIn,
  1146. active,
  1147. transferIn,
  1148. canceled,
  1149. status,
  1150. noBSM,
  1151. loadType
  1152. )
  1153. this.advancedQuery(this.dataContent, singleJump)
  1154. this.gjFlag = false
  1155. },
  1156. setDataContent(...dataContent) {
  1157. dataContent.forEach(target => {
  1158. target = typeof target === 'string' ? target.trim() : target
  1159. if ((target ?? '') !== '') {
  1160. this.dataContent.push(target, target)
  1161. } else {
  1162. this.dataContent.push(null, null)
  1163. }
  1164. })
  1165. },
  1166. // 数据查询
  1167. async advancedQuery(dataContent, singleJump) {
  1168. this.loading = true
  1169. try {
  1170. const {
  1171. code,
  1172. returnData: { listValues: result, needPage }
  1173. } = await Query({
  1174. id: DATACONTENT_ID.advancedQueryId,
  1175. needPage: ++this.page,
  1176. dataContent
  1177. })
  1178. if (Number(code) !== 0) {
  1179. this.page--
  1180. this.loading = false
  1181. this.$message.error('获取数据失败')
  1182. return
  1183. }
  1184. if (needPage === this.page && result.length) {
  1185. if (singleJump) {
  1186. if (result.length === 1) {
  1187. this.$router.push({
  1188. path: '/advance/baggageView',
  1189. query: {
  1190. bagSN: result[0].bagSN,
  1191. flightNO: result[0].flightNO,
  1192. flightDate: result[0].flightDate
  1193. }
  1194. })
  1195. } else {
  1196. const onlyFlight = result.reduce((pre, curr) => {
  1197. if (
  1198. pre === null ||
  1199. (curr.flightNO &&
  1200. curr.flightDate &&
  1201. curr.flightNO === pre.flightNO &&
  1202. curr.flightDate === pre.flightDate)
  1203. ) {
  1204. return {
  1205. flightNO: curr.flightNO,
  1206. flightDate: curr.flightDate
  1207. }
  1208. } else {
  1209. return {}
  1210. }
  1211. }, null)
  1212. if (onlyFlight.flightNO) {
  1213. this.$router.push({
  1214. path: '/advance/flightView',
  1215. query: onlyFlight
  1216. })
  1217. }
  1218. }
  1219. }
  1220. this.tableData = [...this.tableData, ...result]
  1221. setTableFilters(this.tableData, this.tableDataFilters)
  1222. } else {
  1223. if (this.page === 1) {
  1224. this.$message.info('未查询到匹配结果')
  1225. }
  1226. this.page--
  1227. this.noMore = true
  1228. }
  1229. } catch (error) {
  1230. this.page--
  1231. }
  1232. this.loading = false
  1233. },
  1234. // 特殊行李类型下拉选项查询
  1235. async baggageTypeQuery() {
  1236. try {
  1237. const result = await myQuery(DATACONTENT_ID.baggageTypeId)
  1238. this.baggageTypeList = result
  1239. } catch (error) {
  1240. this.$message.error('失败')
  1241. }
  1242. },
  1243. // 统计行数
  1244. summaryRow(num) {
  1245. return function () {
  1246. return ['合计', `共${num}件`]
  1247. }
  1248. },
  1249. exportHandler(refName, tableName) {
  1250. if (!this.tableData.length) {
  1251. this.$message.info('无数据')
  1252. return
  1253. }
  1254. const table = this.$refs[refName].$el.cloneNode(true)
  1255. const fileName = `${tableName}.xlsx`
  1256. throttledExportToExcel(table, tableName, fileName)
  1257. }
  1258. }
  1259. }
  1260. </script>
  1261. <style
  1262. lang="scss"
  1263. scoped
  1264. >
  1265. .advance {
  1266. padding: 8px;
  1267. &__head {
  1268. line-height: 32px;
  1269. margin-top: 8px;
  1270. margin-bottom: 16px;
  1271. .btnAn:not(:last-child) {
  1272. margin-right: 12px;
  1273. }
  1274. .setting {
  1275. height: 32px;
  1276. width: 32px;
  1277. cursor: pointer;
  1278. background-size: 100% 100%;
  1279. background: url('../../../assets/baggage/ic_setting.png') no-repeat;
  1280. margin-left: 12px;
  1281. position: relative;
  1282. top: 2px;
  1283. }
  1284. .mr10 {
  1285. margin-right: 10px;
  1286. }
  1287. ::v-deep .interfaceLog_head_time {
  1288. .el-input__prefix {
  1289. left: 10px;
  1290. color: #101116;
  1291. }
  1292. .el-input--prefix .el-input__inner {
  1293. padding-left: 50px;
  1294. }
  1295. }
  1296. }
  1297. }
  1298. .advance__table {
  1299. width: 100%;
  1300. ::v-deep .table {
  1301. width: 100%;
  1302. .cell {
  1303. padding: 0;
  1304. text-align: center;
  1305. font-size: 14px;
  1306. font-family: Helvetica, 'Microsoft YaHei';
  1307. letter-spacing: 0;
  1308. }
  1309. .cell-click {
  1310. cursor: pointer;
  1311. color: #2d7cff;
  1312. &.cell-clicked {
  1313. color: purple;
  1314. }
  1315. }
  1316. .el-table__header-wrapper,
  1317. .el-table__fixed-header-wrapper {
  1318. .cell {
  1319. font-weight: bold;
  1320. color: #101116;
  1321. }
  1322. .has-gutter {
  1323. tr {
  1324. .bgl-huang {
  1325. background: #fcf0b1;
  1326. }
  1327. }
  1328. }
  1329. }
  1330. .el-table__body-wrapper,
  1331. .el-table__fixed-body-wrapper {
  1332. tr.bgl-deleted {
  1333. background: #d2d6df;
  1334. td {
  1335. background: #d2d6df;
  1336. font-style: italic;
  1337. }
  1338. }
  1339. }
  1340. }
  1341. }
  1342. .el-table tbody tr:hover > td {
  1343. background-color: red !important;
  1344. }
  1345. .el-table__body tr.hover-row.current-row > td,
  1346. .el-table__body tr.hover-row.el-table__row--striped.current-row > td,
  1347. .el-table__body tr.hover-row.el-table__row--striped > td,
  1348. .el-table__body tr.hover-row > td {
  1349. background-color: red;
  1350. }
  1351. .advanced-dialog::v-deep .el-form > .el-row > .el-col {
  1352. &:nth-child(1) .el-date-editor {
  1353. width: 100%;
  1354. }
  1355. .multiple-select .el-select__tags > span > .el-tag:first-child > .el-select__tags-text {
  1356. width: 35px;
  1357. }
  1358. }
  1359. </style>