BoatKkjk.vue 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476
  1. <template>
  2. <div class="boat-info" :class="{'tab-bgxx': currentTab==='bgxx', 'tab-gzxx': currentTab==='gzxx'}">
  3. <div class="bi-header">
  4. <span class="bih-title">{{ props.boatInfo.shipNameCn || '未查到信息' }}</span>
  5. <div class="bih-btns">
  6. <span :class="{'highlight': cameraType==='down'}" @click="toggleCamera('down')">船身</span>
  7. <span :class="{'highlight': cameraType==='in'}" @click="toggleCamera('in')">船舱</span>
  8. <span :class="{'highlight': cameraType==='follow'}" @click="toggleCamera('follow')">船尾</span>
  9. </div>
  10. </div>
  11. <div class="dialog-tab bi-tab bottom-divider">
  12. <div :class="{'highlight': currentTab==='cbxx'}" @click="changeTab('cbxx')"><span>船舶信息</span></div>
  13. <div :class="{'highlight': currentTab==='gzxx'}" @click="changeTab('gzxx')"><span>感知信息</span></div>
  14. <div :class="{'highlight': currentTab==='bgxx'}" @click="changeTab('bgxx')"><span>报告信息</span></div>
  15. </div>
  16. <ul class="boat-fields cf-cbxx bottom-divider" v-if="currentTab==='cbxx'">
  17. <li><span>MMSI:</span><span>{{ props.boatInfo.mmsi }}</span></li>
  18. <li><span>船舶类型:</span><span>{{ props.boatInfo.shipTypeName || '-' }}</span></li>
  19. <li><span>船舶宽度:</span><span>{{ props.boatInfo.shipBreadth || '- ' }}米</span></li>
  20. <li><span>船舶长度:</span><span>{{ props.boatInfo.shipLength || '- ' }}米</span></li>
  21. <li><span>船舶总吨:</span><span>{{ props.boatInfo.shipGrossTon || '-' }}</span></li>
  22. <li><span>满载吃水:</span><span>{{ props.boatInfo.loadedDraft || '-' }}</span></li>
  23. <li style="margin-bottom: 0;"><span>空载吃水:</span><span>{{ props.boatInfo.noLoadedDraft || '-' }}</span></li>
  24. </ul>
  25. <ul class="boat-fields cf-bgxx bottom-divider" v-if="currentTab==='bgxx'">
  26. <li><span>停靠泊位:</span><span>--</span></li>
  27. <li><span>报告时间:</span><span>--</span></li>
  28. <li><span>预抵时间:</span><span>--</span></li>
  29. <li><span>当前港口:</span><span>--</span></li>
  30. <li><span>上一港口:</span><span>--</span></li>
  31. <li><span>下一港口:</span><span>--</span></li>
  32. <li><span>申报人员:</span><span>--</span></li>
  33. <li><span>申报电话:</span><span>--</span></li>
  34. </ul>
  35. <div class="content-gzxx bottom-divider" v-if="currentTab==='gzxx'">
  36. <div class="sjrh-wrapper bottom-divider">
  37. <span>数据融合:</span>
  38. <span class="sjrh-btn" :class="{'disabled': !sjrhData.video}">视频</span>
  39. <span class="sjrh-btn" :class="{'disabled': !sjrhData.laser}">激光</span>
  40. <span class="sjrh-btn" :class="{'disabled': !sjrhData.AIS}">AIS</span>
  41. <span class="sjrh-btn" :class="{'disabled': !sjrhData.RFID}">RFID</span>
  42. </div>
  43. <ul class="gzxx-fields boat-fields">
  44. <li><span>航向:</span><span>{{ props.boatInfo.inOrOut || '-' }}</span></li>
  45. <li><span>航速:</span><span>{{ props.boatInfo.speed || '- ' }}(公里/时)</span></li>
  46. <li><span>载重:</span><span>{{ props.boatInfo.load || '-' }}</span></li>
  47. <li><span>船长:</span><span>{{ props.boatInfo.length || '-' }}</span></li>
  48. <li><span>船宽:</span><span>{{ props.boatInfo.width || '-' }}</span></li>
  49. <li><span>船高:</span><span>{{ props.boatInfo.height || '-' }}</span></li>
  50. </ul>
  51. <ul class="gzxx-warning">
  52. <li v-for="item in warningData.value" :class="{'isOn': item.isWarning}"><i></i><span>{{ item.name }}</span></li>
  53. </ul>
  54. </div>
  55. <div class="boat-playback">
  56. <div class="cb-row1">
  57. <span :class="{'highlight': bottomType==='playback'}" @click="toggleBottom('playback')">轨迹回放</span>
  58. <span :class="{'highlight': bottomType==='pic'}" @click="toggleBottom('pic')">过船图片</span>
  59. </div>
  60. <div class="cb-time">
  61. <div>
  62. <i class="cbt-icon"></i>
  63. <el-date-picker
  64. v-model="timeRange.time1"
  65. type="datetime"
  66. placeholder="请选择"
  67. size="small"
  68. popper-class="date-popper"
  69. class="date-picker-custom1"
  70. value-format="YYYY-MM-DD HH:mm:ss"
  71. />
  72. </div>
  73. <div>
  74. <i class="cbt-icon"></i>
  75. <el-date-picker
  76. v-model="timeRange.time2"
  77. type="datetime"
  78. placeholder="请选择"
  79. size="small"
  80. popper-class="date-popper"
  81. class="date-picker-custom1"
  82. value-format="YYYY-MM-DD HH:mm:ss"
  83. />
  84. </div>
  85. </div>
  86. <div class="cb-play">
  87. <span class="speed" :class="{'selected': playSpeed===1.5}" @click="ChangePlaySpeed(1.5)">x1.5</span>
  88. <span class="speed" :class="{'selected': playSpeed===2}" @click="ChangePlaySpeed(2)">x2.0</span>
  89. <span class="speed" :class="{'selected': playSpeed===3}" @click="ChangePlaySpeed(3)">x3.0</span>
  90. <span class="play-btn" :class="{'btn-disabled': playState===-1}" v-show="playState<1" @click="track_play">播放</span>
  91. <span class="play-btn" v-show="playState===1||playState===2" @click="track_stop">停止</span>
  92. <span class="play-btn" v-show="playState===1" @click="track_pause">暂停</span>
  93. <span class="play-btn" v-show="playState===2" @click="track_resume">继续</span>
  94. </div>
  95. </div>
  96. <div class="pic-carousel-wrapper" v-show="picsShow">
  97. <i class="pc-switch-left" @click="pcSwitch('left')"></i>
  98. <i class="pc-switch-right" @click="pcSwitch('right')"></i>
  99. <i class="pc-close" @click="picsShow=false;bottomType='playback'"></i>
  100. <el-carousel
  101. type="card"
  102. class="pic-carousel"
  103. arrow="never"
  104. :autoplay="false"
  105. ref="pic_carousel"
  106. >
  107. <el-carousel-item v-for="pic, index in props.boatInfo.photoList" :key="pic">
  108. <img :src="pic" :alt="'过船图片-'+(index+1)" class="pics">
  109. </el-carousel-item>
  110. </el-carousel>
  111. </div>
  112. </div>
  113. </template>
  114. <script>
  115. export default {
  116. name: 'BoatKkjk',
  117. }
  118. </script>
  119. <script setup>
  120. import { reactive, ref, watch, onBeforeUnmount, nextTick } from 'vue'
  121. import { ElDatePicker,ElMessage } from 'element-plus'
  122. import 'element-plus/es/components/message/style/css'
  123. import 'element-plus/es/components/date-picker/style/css'
  124. import { ElCarousel, ElCarouselItem } from 'element-plus'
  125. import 'element-plus/es/components/carousel/style/css'
  126. import 'element-plus/es/components/carousel-item/style/css'
  127. import bus from '@/utils/bus'
  128. import { ueCallBoatGuiji, ueCallSetBoatDriveSpeed, ueCallBoatDrive, ueCallBoatCloseDrive, ueCallBoatStop, ueCallBoatContinue, ueCallChangeTrackType } from '@/utils/UIInteractions'
  129. import { useDateFormat } from '@vueuse/core'
  130. const props = defineProps(['boat-info'])
  131. const cameraType = ref('follow')
  132. function toggleCamera(type) {
  133. cameraType.value = type
  134. ueCallChangeTrackType(type)
  135. }
  136. const sjrhData = reactive({
  137. video: false,
  138. laser: false,
  139. AIS: false,
  140. RFID: false,
  141. })
  142. watch(()=>props.boatInfo, (val) => {
  143. nextTick(() => {
  144. warningData.value.forEach(i => {
  145. if(val.warningList.length===0) {
  146. i.isWarning = false
  147. } else {
  148. val.warningList.forEach(j => {
  149. if(i.fullName===j.warningType) {
  150. i.isWarning = true
  151. return
  152. } else {
  153. i.isWarning = false
  154. }
  155. })
  156. }
  157. })
  158. if(!val.isFinish) { return }
  159. val.isFinish.split('').forEach((i,index) => {
  160. switch (index) {
  161. case 0:
  162. sjrhData.AIS = +i
  163. break
  164. case 1:
  165. sjrhData.video = +i
  166. break
  167. case 3:
  168. sjrhData.RFID = +i
  169. }
  170. })
  171. timeRange.time1 = useDateFormat((new Date).getTime() - 30 * 60 * 1000, 'YYYY-MM-DD HH:mm:00').value
  172. timeRange.time2 = useDateFormat((new Date).getTime(), 'YYYY-MM-DD HH:mm:00').value
  173. })
  174. },{immediate:true})
  175. const currentTab = ref('cbxx')
  176. const warningData = reactive({value:[
  177. { name: 'AIS', fullName: 'AIS未开启', isWarning: false },
  178. { name: '报港', fullName: '申报预警', isWarning: false },
  179. { name: '超载', fullName: '超载预警', isWarning: false },
  180. { name: '重点', fullName: '重点监控船舶', isWarning: false },
  181. // { name: '超限', fullName: '超限预警', isWarning: false },
  182. { name: '证书', fullName: '船舶证书预警', isWarning: false },
  183. // { name: '救生衣', fullName: '未穿救生衣', isWarning: false },
  184. { name: '自定义', fullName: '自定义预警', isWarning: false },
  185. ]})
  186. const timeRange = reactive({
  187. time1: '',
  188. time2: ''
  189. })
  190. bus.on('ueRec_boatGuiji', (data) => {
  191. if(data.isOk=='true') {
  192. playState.value=0
  193. }
  194. else {
  195. ElMessage({type: 'warning',message:'此船舶该时段未查到轨迹点'})
  196. }
  197. })
  198. onBeforeUnmount(() => {
  199. bus.off('ueRec_boatGuiji')
  200. bus.off('ueRec_boatFockClear')
  201. ueCallBoatCloseDrive()
  202. })
  203. const bottomType = ref('playback')
  204. const picsShow = ref(false)
  205. const pic_carousel = ref(null)
  206. function pcSwitch(type) {
  207. if(type==='left') {
  208. pic_carousel.value.prev()
  209. } else if(type==='right') {
  210. pic_carousel.value.next()
  211. }
  212. }
  213. function toggleBottom(type) {
  214. if(type==='pic') {
  215. picsShow.value = true
  216. // let len = document.getElementsByClassName('pics').length
  217. // for(let i=0; i<len; i++) {
  218. // document.getElementsByClassName('pics')[i].addEventListener('wheel', (e) => {
  219. // console.log(e)
  220. // })
  221. // }
  222. }
  223. bottomType.value = type
  224. }
  225. const playSpeed = ref(1)
  226. const playState = ref(0) /* 0--未开始/已结束; 1--播放中; 2--已暂停; -1--禁用状态 */
  227. function ChangePlaySpeed(s) {
  228. playSpeed.value = s===playSpeed.value? 1: s
  229. ueCallSetBoatDriveSpeed(playSpeed.value)
  230. }
  231. function track_play() {
  232. if(timeRange.time1&&timeRange.time2) {
  233. playState.value=-1
  234. ueCallBoatGuiji(timeRange.time1, timeRange.time2)
  235. let timer = setInterval(() => {
  236. if(playState.value===-1) { return }
  237. ueCallBoatDrive()
  238. playState.value = 1
  239. clearInterval(timer)
  240. }, 300);
  241. }
  242. }
  243. function track_stop() {
  244. playState.value = 0
  245. ueCallBoatCloseDrive()
  246. }
  247. function track_pause() {
  248. playState.value = 2
  249. ueCallBoatStop()
  250. }
  251. function track_resume() {
  252. playState.value = 1
  253. ueCallBoatContinue()
  254. }
  255. const emit = defineEmits(['closeBoatInfo'])
  256. function changeTab(name) {
  257. currentTab.value = name
  258. }
  259. </script>
  260. <style lang="scss" scoped>
  261. .boat-info {
  262. box-sizing: border-box;
  263. width: 429px;
  264. height: 374px;
  265. background: url('@/assets/imgs/page_kkjk/bi-bg-1.png') no-repeat;
  266. background-size: contain;
  267. .dialog-close {
  268. top: 8px;
  269. }
  270. &.tab-bgxx {
  271. height: 408px;
  272. background: url('@/assets/imgs/page_kkjk/bi-bg-2.png') no-repeat;
  273. background-size: contain;
  274. }
  275. &.tab-gzxx {
  276. height: 494px;
  277. background: url('@/assets/imgs/page_kkjk/bi-bg-3.png') no-repeat;
  278. background-size: contain;
  279. }
  280. .bi-tab {
  281. padding-right: 20px;
  282. margin: 10px 0 20px;
  283. &>div.highlight::after {
  284. display: none;
  285. }
  286. }
  287. .cf-cbxx {
  288. margin: 15px 0 16px;
  289. li {
  290. margin-bottom: 10px;
  291. }
  292. }
  293. .cf-bgxx {
  294. margin-bottom: 22px;
  295. }
  296. .content-gzxx {
  297. margin-bottom: 28px;
  298. .sjrh-wrapper {
  299. display: flex;
  300. justify-content: space-between;
  301. align-items: center;
  302. box-sizing: border-box;
  303. padding: 0 40px 0 30px;
  304. margin-bottom: 25px;
  305. &>span:first-child {
  306. font-size: 14px;
  307. color: #D9E6FF;
  308. text-shadow: 0px 3px 5px rgba(0,38,76,0.4);
  309. }
  310. .sjrh-btn {
  311. display: block;
  312. width: 71px;
  313. height: 30px;
  314. background: url('@/assets/imgs/page_kkjk/bi-sjrh-btn1.png');
  315. background-size: 89px 37px;
  316. background-position: center;
  317. cursor: pointer;
  318. line-height: 30px;
  319. font-size: 14px;
  320. color: #eee;
  321. &.disabled {
  322. filter: brightness(0.5);
  323. }
  324. }
  325. }
  326. .gzxx-fields {
  327. &>li>span:first-child {
  328. width: 50px;
  329. }
  330. }
  331. .gzxx-warning {
  332. box-sizing: border-box;
  333. padding: 0 20px 0 10px;
  334. display: flex;
  335. justify-content: center;
  336. align-items: center;
  337. margin-top: 6px;
  338. &>li {
  339. display: flex;
  340. flex-direction: column;
  341. align-items: center;
  342. &.isOn {
  343. &>i {
  344. background: url('@/assets/imgs/page_kkjk/bi-warning-on.png');
  345. background-size: contain;
  346. }
  347. &>span {
  348. color: rgba($color: #fff, $alpha: 0.9);
  349. }
  350. }
  351. &>i {
  352. display: block;
  353. width: 20px;
  354. height: 20px;
  355. background: url('@/assets/imgs/page_kkjk/bi-warning-off.png');
  356. background-size: contain;
  357. margin-top: 8px;
  358. }
  359. &>span {
  360. font-size: 12px;
  361. line-height: 12px;
  362. color: rgba($color: #fff, $alpha: 0.3);
  363. }
  364. width: 51px;
  365. height: 51px;
  366. background: url('@/assets/imgs/page_kkjk/bi-warning-bg.png');
  367. background-size: contain;
  368. }
  369. }
  370. }
  371. .pic-carousel-wrapper {
  372. position: fixed;
  373. top: 0;
  374. left: 0;
  375. width: 100vw;
  376. height: 100vh;
  377. background-color: rgba($color: #000000, $alpha: 0.7);
  378. z-index: 999;
  379. :deep(.pic-carousel) {
  380. position: absolute;
  381. top: 20%;
  382. left: 50%;
  383. transform: translateX(-50%);
  384. width: 85%;
  385. height: 55vh;
  386. .el-carousel__indicators {
  387. display: none;
  388. }
  389. .el-carousel__container {
  390. height: 100%;
  391. }
  392. .el-carousel__item {
  393. box-sizing: border-box;
  394. border: 1px solid #4DA6FF;
  395. &:not(.is-active) {
  396. filter: brightness(60%);
  397. }
  398. &>img {
  399. width: 100%;
  400. height: 100%;
  401. }
  402. }
  403. }
  404. .pc-switch-right,.pc-switch-left {
  405. display: block;
  406. position: absolute;
  407. top: 46%;
  408. width: 35px;
  409. height: 59px;
  410. cursor: pointer;
  411. }
  412. .pc-switch-left {
  413. left: 4%;
  414. background: url('@/assets/imgs/navi/sxdx-arrow-left.png');
  415. background-size: contain;
  416. }
  417. .pc-switch-right {
  418. right: 4%;
  419. background: url('@/assets/imgs/navi/sxdx-arrow-right.png');
  420. background-size: contain;
  421. }
  422. .pc-close {
  423. display: block;
  424. position: absolute;
  425. bottom: 17%;
  426. left: calc(50% - 12px);
  427. width: 24px;
  428. height: 24px;
  429. cursor: pointer;
  430. background: url('@/assets/imgs/page_kkjk/bi-pics-close.png');
  431. background-size: contain;
  432. &:hover {
  433. transform: scale(1.1);
  434. }
  435. }
  436. }
  437. }
  438. </style>