index.vue 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329
  1. <template>
  2. <div class="waybill">
  3. <div class="waybill-info">
  4. <div class="waybill-info-title">运单基本信息</div>
  5. <div class="waybill-info-content flex">
  6. <div
  7. v-for="(item, index) in waybillInfoItems"
  8. :key="index"
  9. class="waybill-info-item"
  10. >
  11. <div>{{ item.label }}:</div>
  12. <el-tooltip
  13. :disabled="!computedWaybillInfo(item) || !item.showOverflowTooltip"
  14. :content="computedWaybillInfo(item)"
  15. effect="light"
  16. >
  17. <div>{{ computedWaybillInfo(item) }}</div>
  18. </el-tooltip>
  19. </div>
  20. </div>
  21. </div>
  22. <div class="waybill-header flex">
  23. <div class="waybill-header-title flex-wrap">
  24. <div class="manageTitle">运单跟踪信息</div>
  25. <div class="status">正常</div>
  26. </div>
  27. <div class="waybill-header-operate flex-wrap">
  28. <Search @clear="clear" @search="search" />
  29. <el-button
  30. class="button-sqaure"
  31. :icon="Download"
  32. color="#ac014d"
  33. @click="downloadHandler"
  34. />
  35. <ColumnSet
  36. class="button-sqaure"
  37. :table-columns="tableColumns"
  38. @checked-submit="columnChecked"
  39. />
  40. </div>
  41. </div>
  42. <div
  43. v-show="trackAirlines.length"
  44. :style="{
  45. maxHeight: trackAirlines.length > 1 ? `${208 * 2 + 8}px` : '208px',
  46. }"
  47. class="waybill-track"
  48. >
  49. <el-scrollbar always>
  50. <div class="waybill-track-warpper">
  51. <div
  52. v-for="trackAirline in trackAirlines"
  53. :key="trackAirline.flightNO"
  54. class="waybill-track-row"
  55. >
  56. <div
  57. v-for="(trackAirport, index) in trackAirline.airports"
  58. :key="trackAirport.airport"
  59. class="waybill-track-box"
  60. :style="trackBoxStyle(trackAirline.airports, index)"
  61. >
  62. <div class="title flex-wrap">
  63. <span v-if="index === 0" class="title-span"
  64. >航班号:{{ trackAirline.flightNO }}</span
  65. >
  66. <span class="title-span"
  67. >{{ trackAirport.isDeparture ? '出港' : '进港' }}:{{
  68. trackAirport.airport
  69. }}</span
  70. >
  71. <span class="title-span"
  72. >日期:{{ trackAirline.flightDate }}</span
  73. >
  74. </div>
  75. <Steps :steps="trackAirport.trackSteps" />
  76. </div>
  77. </div>
  78. </div>
  79. </el-scrollbar>
  80. </div>
  81. <div class="goods-list">
  82. <SimpleTable
  83. ref="tableRef"
  84. :data="tableData"
  85. :columns="filteredColumns"
  86. scrollbar-always-on
  87. :row-class-name="rowClass"
  88. :cell-class-name="cellClass"
  89. @cell-click="cellClickHandler"
  90. :column-props="{ formatter, minWidth: 75 }"
  91. />
  92. </div>
  93. </div>
  94. </template>
  95. <script setup lang="ts">
  96. import { Download } from '@element-plus/icons-vue'
  97. import Search from '@/components/search/index.vue'
  98. import Steps from '@/components/steps/index.vue'
  99. import ColumnSet from '@/components/ColumnSet/index.vue'
  100. import SimpleTable from '@/components/SimpleTable/index.vue'
  101. import { useTrackData } from '../../hooks/useTrackData'
  102. import { useTable } from '../../hooks/useTable'
  103. import { useTableColumnSet } from '@/hooks/useTableColumnSet'
  104. import { useTableExport } from '../../hooks/useTableExport'
  105. import { useTableStyle } from '../../hooks/useTableStyle'
  106. import { useTableCellClick } from '../../hooks/useTableCellClick'
  107. import { useWaybillInfo } from './useWaybillInfo'
  108. import { CommonData } from '~/common'
  109. import { useLoop } from '@/hooks/useLoop'
  110. const props = defineProps({
  111. name: {
  112. type: String,
  113. required: true,
  114. },
  115. })
  116. const route = useRoute()
  117. const { flightDate, waybillNO } = route.query
  118. const dataContent = [flightDate, waybillNO] as string[]
  119. const {
  120. waybillInfoItems,
  121. waybillInfo,
  122. computedWaybillInfo,
  123. getWaybillInfo,
  124. } = useWaybillInfo(props.name, dataContent)
  125. const { tableColumns, tableData: trackData, getTableData } = useTable(
  126. `${props.name}Goods`,
  127. dataContent
  128. )
  129. useLoop([getWaybillInfo, getTableData], 'waybill')
  130. const { trackAirlines, trackBoxStyle } = useTrackData(props.name, trackData)
  131. const tableData = computed(() => {
  132. const mergedTableData = trackData.value.reduce(
  133. (data: CommonData[], current) => {
  134. const sameRow = data.find(row =>
  135. ['stockCode', 'flightNO', 'flightDate', 'cargoSN'].every(
  136. key => row[key] === current[key]
  137. )
  138. )
  139. const {
  140. stockCode,
  141. flightNO,
  142. flightDate,
  143. departureAirport, // 装载机场
  144. arriveAirport, // 卸载机场
  145. ULDNO,
  146. cargoSN,
  147. pullMark,
  148. returnMark,
  149. // transMark,
  150. exceptionCustomsMark,
  151. execPosition, // 读取位置
  152. nodeCode, // 节点名称
  153. ConsignmentItemPackagingQuantityQuantity, // 跟踪节点件数
  154. execResult,
  155. execTime,
  156. } = current
  157. const nodeValue = `${execPosition ?? ''}\n${
  158. execResult ? '通过' : '未通过'
  159. }\n${execTime ?? ''}`
  160. if (sameRow) {
  161. sameRow[String(nodeCode)] = nodeValue
  162. ;[
  163. 'ULDNO',
  164. // 'cargoSN'
  165. ].forEach(key => {
  166. const oldValue = sameRow[key]
  167. const currentValue = current[key]
  168. if (typeof currentValue === 'string' && currentValue !== oldValue) {
  169. if (typeof oldValue === 'string') {
  170. sameRow[key] = [
  171. ...new Set([
  172. ...oldValue.split(','),
  173. ...currentValue.split(','),
  174. ]),
  175. ].join(',')
  176. } else {
  177. sameRow[key] = currentValue
  178. }
  179. }
  180. })
  181. ;['pullMark', 'returnMark'].forEach(key => {
  182. sameRow[key] = sameRow[key] ?? current[key]
  183. })
  184. } else {
  185. data.push({
  186. ...current,
  187. [String(nodeCode)]: nodeValue,
  188. })
  189. }
  190. return data
  191. },
  192. []
  193. )
  194. return mergedTableData.reduce((data: CommonData[], current) => {
  195. const { cargoSN } = current
  196. if (typeof cargoSN === 'string') {
  197. const splitedRows = cargoSN.split(',').map(splitedCargoSN => ({
  198. ...current,
  199. cargoSN: splitedCargoSN,
  200. }))
  201. data.push(...splitedRows)
  202. }
  203. return data
  204. }, [])
  205. })
  206. const formatter = (row, column, cellValue, index) => {
  207. const value = String(cellValue ?? '').split('\n')
  208. if (value[2]) {
  209. value[2] = value[2].split('T')[1].slice(0, -3)
  210. }
  211. return value.join('\n')
  212. }
  213. const search = (text: string) => {
  214. console.log(text)
  215. }
  216. const clear = () => {}
  217. const tableRef = ref<InstanceType<typeof SimpleTable> | null>(null)
  218. const { exportToExcel } = useTableExport()
  219. const downloadHandler = () => {
  220. const table = tableRef.value!.table!
  221. exportToExcel({ table })
  222. }
  223. const { filteredColumns, columnChecked } = useTableColumnSet(tableColumns)
  224. const { rowClass, cellClass } = useTableStyle(`${props.name}Goods`)
  225. const { cellClickHandler } = useTableCellClick(`${props.name}Goods`)
  226. </script>
  227. <style lang="scss" scoped>
  228. .waybill {
  229. height: 100%;
  230. display: flex;
  231. flex-direction: column;
  232. &-info {
  233. height: 144px;
  234. background: #410425;
  235. padding: 24px 30px;
  236. color: #ffffff;
  237. &-title {
  238. font-size: 18px;
  239. font-family: Microsoft YaHei;
  240. font-weight: bold;
  241. margin-bottom: 40px;
  242. }
  243. &-item {
  244. display: flex;
  245. .el-tooltip__trigger {
  246. max-width: 300px;
  247. overflow: hidden;
  248. white-space: nowrap;
  249. text-overflow: ellipsis;
  250. }
  251. }
  252. }
  253. &-header :deep {
  254. margin: 24px 0;
  255. line-height: 32px;
  256. .status {
  257. font-size: 16px;
  258. font-family: Microsoft YaHei;
  259. font-weight: bold;
  260. color: #519f6b;
  261. }
  262. .el-button {
  263. background-color: #d5327b;
  264. border: none;
  265. }
  266. .button-sqaure {
  267. width: 30px;
  268. height: 30px;
  269. border-radius: 4px;
  270. font-size: 16px;
  271. margin-left: 24px;
  272. &:first-of-type {
  273. margin-left: 35px;
  274. }
  275. }
  276. }
  277. &-track {
  278. height: 0;
  279. flex: 1.5;
  280. &-row {
  281. height: 208px;
  282. display: flex;
  283. &:not(:last-child) {
  284. margin-bottom: 8px;
  285. }
  286. }
  287. &-box {
  288. background: #ffffff;
  289. padding: 24px 24px 12px 24px;
  290. &:not(:last-child) {
  291. margin-right: 8px;
  292. }
  293. .title {
  294. font-size: 16px;
  295. font-family: Microsoft YaHei;
  296. font-weight: bold;
  297. color: #101116;
  298. margin-bottom: 10px;
  299. &-span:not(:last-of-type) {
  300. margin-right: 8px;
  301. }
  302. }
  303. }
  304. }
  305. .goods-list :deep {
  306. margin-top: 16px;
  307. height: 0;
  308. flex: 1;
  309. .el-table__body .el-table__cell .cell {
  310. padding: 4px 8px;
  311. }
  312. }
  313. }
  314. </style>