voyageDetail.vue 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630
  1. <template>
  2. <div class="line-container-p24">
  3. <i class="el-icon-arrow-left"></i>
  4. <div
  5. class="dib go-back ml8 pointer"
  6. @click="router.replace('/voyage/voyageList')"
  7. >
  8. 返回航次列表
  9. </div>
  10. </div>
  11. <div class="container-title">航次信息</div>
  12. <div class="line-container-p24">
  13. <div class="line">
  14. <div class="info-line">
  15. <div class="info-line-title">航次名称</div>
  16. <el-input
  17. class="info-line-text"
  18. v-model="voyage.voyageName"
  19. disabled
  20. ></el-input>
  21. </div>
  22. <div class="info-line">
  23. <div class="info-line-title">货主</div>
  24. <el-input
  25. class="info-line-text"
  26. v-model="voyage.cargoOwnerName"
  27. disabled
  28. ></el-input>
  29. </div>
  30. </div>
  31. <!-- <div class="line">
  32. <div class="info-line">
  33. <div class="info-line-title">船东</div>
  34. <el-input
  35. class="info-line-text"
  36. v-model="voyage.shipOwnerName"
  37. disabled
  38. ></el-input>
  39. </div>
  40. <div class="info-line">
  41. <div class="info-line-title">船东手机号</div>
  42. <el-input
  43. class="info-line-text"
  44. v-model="voyage.shipOwnerPhone"
  45. disabled
  46. ></el-input>
  47. </div>
  48. </div> -->
  49. <div class="line">
  50. <div class="info-line">
  51. <div class="info-line-title">船舶</div>
  52. <el-input
  53. class="info-line-text"
  54. v-model="voyage.shipName"
  55. disabled
  56. ></el-input>
  57. </div>
  58. <div class="info-line">
  59. <div class="info-line-title">MMSI</div>
  60. <el-input
  61. class="info-line-text"
  62. v-model="voyage.shipMmsi"
  63. disabled
  64. ></el-input>
  65. </div>
  66. </div>
  67. <div id="map-container" class="map-container"></div>
  68. <div class="line" style="margin-top: 30px">
  69. <div class="info-line">
  70. <div class="info-line-title">开始时间</div>
  71. <el-input
  72. class="info-line-text"
  73. v-model="voyage.startTime"
  74. disabled
  75. ></el-input>
  76. </div>
  77. <div class="info-line">
  78. <div class="info-line-title">结束时间</div>
  79. <el-input
  80. class="info-line-text"
  81. v-model="voyage.endTime"
  82. disabled
  83. ></el-input>
  84. </div>
  85. </div>
  86. <div class="line">
  87. <div class="info-line">
  88. <div class="info-line-title">装货港</div>
  89. <el-input
  90. class="info-line-text"
  91. v-model="voyage.loadPort"
  92. disabled
  93. ></el-input>
  94. </div>
  95. <div class="info-line">
  96. <div class="info-line-title">卸货港</div>
  97. <el-input
  98. class="info-line-text"
  99. v-model="voyage.dischargeProt"
  100. disabled
  101. ></el-input>
  102. </div>
  103. </div>
  104. <div class="line">
  105. <div class="info-line">
  106. <div class="info-line-title">货种</div>
  107. <el-input
  108. class="info-line-text"
  109. v-model="voyage.cargo"
  110. disabled
  111. ></el-input>
  112. </div>
  113. <div class="info-line">
  114. <div class="info-line-title">吨位</div>
  115. <el-input
  116. class="info-line-text"
  117. v-model="voyage.tons"
  118. disabled
  119. ></el-input>
  120. </div>
  121. </div>
  122. </div>
  123. <div class="container-title">航次信息</div>
  124. <div class="line-container-p24">
  125. <div class="line">
  126. <div class="info-line">
  127. <div class="info-line-title">运输状态</div>
  128. <el-select
  129. v-model="voyage.transStatus"
  130. placeholder="Select"
  131. class="info-line-text"
  132. :disabled="disabledStatus"
  133. >
  134. <el-option
  135. v-for="item in options"
  136. :key="item.value"
  137. :label="item.label"
  138. :value="item.value"
  139. ></el-option>
  140. </el-select>
  141. </div>
  142. <div class="info-line">
  143. <div class="info-line-title">实际卸货吨位</div>
  144. <el-input
  145. class="info-line-text"
  146. v-model="voyage.actualDischargeTons"
  147. :disabled="disabledStatus"
  148. ></el-input>
  149. </div>
  150. </div>
  151. <div class="line">
  152. <div class="info-line">
  153. <div class="info-line-title">装货开始时间</div>
  154. <el-date-picker
  155. class="info-line-text"
  156. v-model="voyage.loadStartTime"
  157. type="datetime"
  158. format="YYYY/MM/DD HH:mm:ss"
  159. value-format="YYYY/MM/DD HH:mm:ss"
  160. placeholder="装货开始时间"
  161. :disabled="disabledStatus"
  162. ></el-date-picker>
  163. </div>
  164. <div class="info-line">
  165. <div class="info-line-title">装货结束时间</div>
  166. <el-date-picker
  167. class="info-line-text"
  168. v-model="voyage.loadEndTime"
  169. type="datetime"
  170. format="YYYY/MM/DD HH:mm:ss"
  171. value-format="YYYY/MM/DD HH:mm:ss"
  172. placeholder="装货开始时间"
  173. :disabled="disabledStatus"
  174. ></el-date-picker>
  175. </div>
  176. </div>
  177. <div class="line">
  178. <div class="info-line">
  179. <div class="info-line-title">卸货开始时间</div>
  180. <el-date-picker
  181. class="info-line-text"
  182. v-model="voyage.dischargeStartTime"
  183. type="datetime"
  184. format="YYYY/MM/DD HH:mm:ss"
  185. value-format="YYYY/MM/DD HH:mm:ss"
  186. placeholder="装货开始时间"
  187. :disabled="disabledStatus"
  188. ></el-date-picker>
  189. </div>
  190. <div class="info-line">
  191. <div class="info-line-title">卸货结束时间</div>
  192. <el-date-picker
  193. class="info-line-text"
  194. v-model="voyage.dischargeEndTime"
  195. type="datetime"
  196. format="YYYY/MM/DD HH:mm:ss"
  197. value-format="YYYY/MM/DD HH:mm:ss"
  198. placeholder="装货开始时间"
  199. :disabled="disabledStatus"
  200. ></el-date-picker>
  201. </div>
  202. </div>
  203. <div class="line">
  204. <div class="info-line">
  205. <div class="info-line-title">备注</div>
  206. <el-input
  207. class="info-line-textarea"
  208. v-model="voyage.remark"
  209. autosize
  210. type="textarea"
  211. :disabled="disabledStatus"
  212. ></el-input>
  213. </div>
  214. </div>
  215. <div class="media-content df ffw">
  216. <div class="pic-container">
  217. <div v-for="(item, index) in media" :key="item" class="pic-main">
  218. <div :class="index % 2 == 0 ? 'box' : 'box bottom-box'">
  219. <div class="card-note">
  220. {{ item.shipName }} 拍摄于
  221. <br />
  222. {{ item.createTime }}
  223. </div>
  224. <div class="media-box" style="position: relative">
  225. <el-image
  226. v-if="item.mediaType == 1"
  227. style="width: 100%; height: 100%"
  228. fit="contain"
  229. :src="item.downloadUrl"
  230. :preview-src-list="previewSrcList"
  231. ></el-image>
  232. <video
  233. style="width: 100%; height: 100%"
  234. v-else
  235. :src="item.downloadUrl"
  236. ></video>
  237. <img
  238. @click="openVideoModal(item.downloadUrl)"
  239. v-if="item.mediaType == 2"
  240. src="../../assets/icon-player.png"
  241. style="
  242. object-fit: contain;
  243. width: 40px;
  244. height: 40px;
  245. position: absolute;
  246. top: calc(50% - 20px);
  247. left: calc(50% - 20px);
  248. background: #fff;
  249. border-radius: 50%;
  250. "
  251. alt=""
  252. />
  253. </div>
  254. <div class="checkbox-group df aic jcsa">
  255. <el-checkbox
  256. @change="auditMedia(item.id, 1, index, item.mediaType)"
  257. :model-value="item.audit == 1"
  258. label="通过"
  259. ></el-checkbox>
  260. <el-checkbox
  261. @change="auditMedia(item.id, 2, index, item.mediaType)"
  262. :model-value="item.audit == 2"
  263. label="未通过"
  264. ></el-checkbox>
  265. </div>
  266. </div>
  267. <div :class="index % 2 == 0 ? 's-line' : 's-line top210px'"></div>
  268. <div class="point"></div>
  269. <div class="l-line" v-if="index + 1 != media.length"></div>
  270. </div>
  271. <el-dialog
  272. v-model="videoModal"
  273. title="视频审核"
  274. width="20%"
  275. :before-close="videoClose"
  276. >
  277. <video
  278. autoplay
  279. controls
  280. style="width: 100%; height: 100%"
  281. :src="currentUrl"
  282. ></video>
  283. </el-dialog>
  284. </div>
  285. </div>
  286. <div class="df aic jcfe mt20" v-if="voyage.voyageStatus == 1">
  287. <el-button v-if="disabledStatus" type="primary" @click="changeVoyageInfo">
  288. 修改航次
  289. </el-button>
  290. <div v-else>
  291. <div>
  292. <el-button @click="cancelVoyageChange">取消修改</el-button>
  293. <el-button type="primary" @click="submitVoyageChange">
  294. 提交修改
  295. </el-button>
  296. </div>
  297. </div>
  298. <el-button
  299. v-if="voyage.dischargeEndTime && disabledStatus"
  300. type="primary"
  301. @click="finishVoyage"
  302. >
  303. 完成航次
  304. </el-button>
  305. </div>
  306. </div>
  307. </template>
  308. <script>
  309. import { onMounted, reactive, ref, toRefs } from "_vue@3.2.20@vue";
  310. import api from "../../apis/fetch";
  311. import { useRoute } from "vue-router";
  312. import _ from "lodash";
  313. import router from "../../router";
  314. import store from "../../store";
  315. import { ElNotification } from "element-plus";
  316. import PicTimelineVue from "../../components/PicTimeline.vue";
  317. export default {
  318. components: {
  319. PicTimelineVue,
  320. },
  321. setup() {
  322. const route = useRoute();
  323. let map = ref();
  324. let voyage = ref({});
  325. let media = ref();
  326. let coordinates = ref();
  327. let previewSrcList = ref([]);
  328. async function getVoyageDetail() {
  329. let res = await api.getVoyageDetail({
  330. type: 2,
  331. voyageId: route.query.id,
  332. });
  333. coordinates.value = res.data.result.coordinates;
  334. voyage.value = res.data.result.voyage;
  335. media.value = [
  336. ...res.data.result.medias,
  337. ...res.data.result.medias,
  338. ...res.data.result.medias,
  339. ...res.data.result.medias,
  340. ...res.data.result.medias,
  341. ];
  342. for (let i of media.value) {
  343. previewSrcList.value.push(i.downloadUrl);
  344. }
  345. setShipMarker();
  346. }
  347. function initMap() {
  348. map.value = new AMap.Map("map-container", {
  349. zoom: 11, //级别
  350. center: [121.524761, 31.228721], //中心点坐标
  351. });
  352. }
  353. function setShipMarker(longitude = 121.524761, latitude = 31.228721) {
  354. var marker = new AMap.Marker({
  355. position: new AMap.LngLat(longitude, latitude),
  356. // offset: new AMap.Pixel(-10, -10),
  357. size: new AMap.Size(80, 80),
  358. icon: "https://hhd-pat-1255802371.cos.ap-shanghai.myqcloud.com/frontend/ship-red-icon.png", // 添加 Icon 图标 URL
  359. title: "北京",
  360. });
  361. map.value.add(marker);
  362. }
  363. let disabledStatus = ref(true);
  364. let updateCache = {};
  365. function changeVoyageInfo() {
  366. updateCache = _.cloneDeep(voyage.value);
  367. disabledStatus.value = false;
  368. }
  369. function cancelVoyageChange() {
  370. voyage.value = updateCache;
  371. disabledStatus.value = true;
  372. }
  373. async function submitVoyageChange() {
  374. let {
  375. id,
  376. transStatus,
  377. loadStartTime,
  378. loadEndTime,
  379. dischargeStartTime,
  380. dischargeEndTime,
  381. actualDischargeTons,
  382. remark,
  383. } = voyage.value;
  384. let res = await api.updateVoyage({
  385. id,
  386. transStatus,
  387. loadStartTime,
  388. loadEndTime,
  389. dischargeStartTime,
  390. dischargeEndTime,
  391. actualDischargeTons,
  392. remark,
  393. });
  394. if (res.data.status == 0) {
  395. ElNotification({
  396. type: "success",
  397. title: res.data.msg,
  398. });
  399. disabledStatus.value = true;
  400. } else {
  401. ElNotification({
  402. type: "error",
  403. title: res.data.msg,
  404. });
  405. console.log(res);
  406. }
  407. }
  408. let options = ref([
  409. { value: 0, label: "请选择" },
  410. {
  411. value: 1,
  412. label: "航行",
  413. },
  414. {
  415. value: 2,
  416. label: "停泊",
  417. },
  418. {
  419. value: 3,
  420. label: "装货",
  421. },
  422. {
  423. value: 4,
  424. label: "运输中",
  425. },
  426. {
  427. value: 5,
  428. label: "卸货",
  429. },
  430. ]);
  431. async function finishVoyage() {
  432. if (!voyage.value.dischargeEndTime) return;
  433. let res = await api.finishVoyage({
  434. voyageId: route.query.id,
  435. });
  436. if (res.data.status == 0) {
  437. voyage.value.voyageStatus = 2;
  438. ElNotification({
  439. type: "success",
  440. title: res.data.msg,
  441. });
  442. } else {
  443. ElNotification({
  444. type: "error",
  445. title: res.data.msg,
  446. });
  447. console.log(res);
  448. }
  449. }
  450. let currentUrl = ref("");
  451. let videoModal = ref(false);
  452. function openVideoModal(url) {
  453. currentUrl.value = url;
  454. videoModal.value = true;
  455. }
  456. async function auditMedia(mediaId, a, index, mediaType) {
  457. console.log(mediaId, a, index, mediaType);
  458. let res = await api.auditMedia({
  459. mediaId,
  460. audit: a,
  461. });
  462. if (res.data.status == 0) {
  463. media.value[index].audit = a;
  464. }
  465. console.log(res);
  466. }
  467. onMounted(() => {
  468. initMap();
  469. getVoyageDetail();
  470. });
  471. return {
  472. options,
  473. voyage,
  474. coordinates,
  475. media,
  476. disabledStatus,
  477. changeVoyageInfo,
  478. cancelVoyageChange,
  479. submitVoyageChange,
  480. finishVoyage,
  481. openVideoModal,
  482. previewSrcList,
  483. router,
  484. auditMedia,
  485. currentUrl,
  486. videoModal,
  487. };
  488. },
  489. };
  490. </script>
  491. <style scoped>
  492. .map-container {
  493. width: 100%;
  494. height: 500px;
  495. }
  496. .card-note {
  497. height: 30px;
  498. font-size: 12px;
  499. font-family: PingFangSC-Regular, PingFang SC;
  500. font-weight: 400;
  501. color: #777777;
  502. }
  503. .media-box {
  504. width: 200px;
  505. height: 200px;
  506. margin-top: 20px;
  507. }
  508. .checkbox-group {
  509. width: 200px;
  510. height: 50px;
  511. margin-top: 20px;
  512. }
  513. .media-content {
  514. width: 100%;
  515. height: 600px;
  516. background: #f7f7f7;
  517. border-radius: 2px;
  518. }
  519. .pic-container {
  520. width: 100%;
  521. height: 100%;
  522. box-sizing: border-box;
  523. display: flex;
  524. padding: 20px;
  525. overflow-x: scroll;
  526. overflow-y: hidden;
  527. white-space: nowrap;
  528. }
  529. .pic-main {
  530. position: relative;
  531. width: 120px;
  532. }
  533. .box {
  534. position: absolute;
  535. height: 240px;
  536. width: var(--box-width);
  537. border: 1px solid #dddddd;
  538. transition: all 0.5s;
  539. background: #fff;
  540. z-index: 10;
  541. }
  542. .s-line {
  543. position: absolute;
  544. left: 100px;
  545. top: 242px;
  546. height: 20px;
  547. border-left: 2px dashed;
  548. box-sizing: border-box;
  549. border-color: #97caf6;
  550. }
  551. .point {
  552. position: relative;
  553. left: 93px;
  554. top: 258px;
  555. width: 16px;
  556. height: 16px;
  557. background-image: url(../../images/blue-circle.png);
  558. }
  559. .l-line {
  560. position: relative;
  561. bottom: 30px;
  562. left: 111px;
  563. top: 249px;
  564. height: 3px;
  565. width: 100px;
  566. background-color: #0094fe;
  567. }
  568. .bottom-box {
  569. top: 290px;
  570. }
  571. .top210px {
  572. top: 270px;
  573. }
  574. .box:hover {
  575. transform: scale(1.2);
  576. }
  577. .media-box {
  578. width: 80px;
  579. height: 80px;
  580. margin-top: 10px;
  581. }
  582. .card-note {
  583. height: 30px;
  584. font-size: 12px;
  585. font-family: PingFangSC-Regular, PingFang SC;
  586. font-weight: 400;
  587. color: #777777;
  588. padding: 10px 20px;
  589. }
  590. .media-box {
  591. width: 100%;
  592. height: 100px;
  593. margin-top: 20px;
  594. }
  595. .checkbox-group {
  596. width: 180px;
  597. height: 50px;
  598. margin-top: 20px;
  599. }
  600. .el-checkbox {
  601. margin: 0;
  602. }
  603. </style>