|
@@ -1,6 +1,36 @@
|
|
|
<template>
|
|
|
<div class="main">
|
|
|
- <div id="map"></div>
|
|
|
+ <EasyMapComponent
|
|
|
+ class="map"
|
|
|
+ layout="info"
|
|
|
+ @easyMapLoad="mapLoad"
|
|
|
+ />
|
|
|
+ <div class="filter">
|
|
|
+ <CusForm labelWidth="70px" ref="ref_form">
|
|
|
+ <CusFormColumn
|
|
|
+ :span="24"
|
|
|
+ required
|
|
|
+ label="时间"
|
|
|
+ link="datetime"
|
|
|
+ type="datetimerange"
|
|
|
+ v-model:param="queryForm.timeArea"/>
|
|
|
+ <CusFormColumn
|
|
|
+ :span="24"
|
|
|
+ required
|
|
|
+ label="数量"
|
|
|
+ link="number"
|
|
|
+ v-model:param="queryForm.total"/>
|
|
|
+ <el-button type="primary" @click="onSearch">查询</el-button>
|
|
|
+ <el-button type="primary" @click="mapFunc.toLocation({position: [109.6915958479584, 19.111636735969318], zoom: 9})">定位</el-button>
|
|
|
+ </CusForm>
|
|
|
+ </div>
|
|
|
+ <PlayBarCom
|
|
|
+ class="play-bar"
|
|
|
+ :timeArea="playBar.timeArea"
|
|
|
+ :cachePro="playBar.cachePro"
|
|
|
+ :total="playBar.total"
|
|
|
+ @refresh="onPlayRefresh"
|
|
|
+ />
|
|
|
</div>
|
|
|
</template>
|
|
|
|
|
@@ -20,15 +50,31 @@ import {
|
|
|
import {useStore} from 'vuex'
|
|
|
import {useRouter, useRoute} from 'vue-router'
|
|
|
import {ElMessage, ElMessageBox} from "element-plus";
|
|
|
-import Map from 'ol/Map.js';
|
|
|
-import View from 'ol/View.js';
|
|
|
-import TileLayer from 'ol/layer/Tile.js';
|
|
|
-import OSM from 'ol/source/OSM.js';
|
|
|
-import * as ol from 'ol'
|
|
|
+import PlayBarCom from "./play-bar.vue";
|
|
|
+import * as layer from "ol/layer";
|
|
|
+import WebGLVectorLayerRenderer from "ol/renderer/webgl/VectorLayer";
|
|
|
+import * as format from "ol/format";
|
|
|
+import * as source from "ol/source";
|
|
|
+import ShipImg from './AIS.svg'
|
|
|
+
|
|
|
+class WebGLLineLayer extends layer.Layer {
|
|
|
+ createRenderer() {
|
|
|
+ return new WebGLVectorLayerRenderer(this, {
|
|
|
+ disableHitDetection: false,
|
|
|
+ style: {
|
|
|
+ 'stroke-color': ['*', ['get', 'lineColor'], [220, 220, 220]],
|
|
|
+ 'stroke-width': 2,
|
|
|
+ // 'stroke-line-dash': ['get', 'lineDasharray'],
|
|
|
+ },
|
|
|
+ });
|
|
|
+ }
|
|
|
+}
|
|
|
|
|
|
export default defineComponent({
|
|
|
name: '',
|
|
|
- components: {},
|
|
|
+ components: {
|
|
|
+ PlayBarCom
|
|
|
+ },
|
|
|
props: {},
|
|
|
setup(props, {emit}) {
|
|
|
const store = useStore();
|
|
@@ -37,27 +83,138 @@ export default defineComponent({
|
|
|
const that = (getCurrentInstance() as ComponentInternalInstance).appContext.config.globalProperties
|
|
|
const state = reactive({
|
|
|
map: <any>null,
|
|
|
- popupHover: <any>null
|
|
|
+ mapFunc: <any>null,
|
|
|
+ queryForm: {
|
|
|
+ timeArea: [
|
|
|
+ '2024-03-18 00:00:00',
|
|
|
+ '2024-03-19 00:00:00',
|
|
|
+ ],
|
|
|
+ total: 10
|
|
|
+ },
|
|
|
+ playBar: {
|
|
|
+ timeArea: [],
|
|
|
+ total: 0,
|
|
|
+ cache: 0,
|
|
|
+ cachePro: 0,
|
|
|
+ },
|
|
|
+ shipData: <any>{},
|
|
|
+ webglPointLayer: <any>null,
|
|
|
+ webglLineLayer: <any>null,
|
|
|
})
|
|
|
+ const ref_form = ref()
|
|
|
+ const mapLoad = (map, func) => {
|
|
|
+ state.map = map
|
|
|
+ state.mapFunc = func
|
|
|
+ }
|
|
|
const initMap = () => {
|
|
|
- const map = new ol.Map({
|
|
|
- view: new View({
|
|
|
- center: [0, 0],
|
|
|
- zoom: 1,
|
|
|
- }),
|
|
|
- layers: [
|
|
|
- new TileLayer({
|
|
|
- source: new OSM(),
|
|
|
- }),
|
|
|
- ],
|
|
|
- target: 'map',
|
|
|
+ }
|
|
|
+ const onSearch = () => {
|
|
|
+ ref_form.value.submit().then(() => {
|
|
|
+ state.playBar.timeArea = JSON.parse(JSON.stringify(state.queryForm.timeArea))
|
|
|
+ state.playBar.cache = 0
|
|
|
+ state.playBar.total = JSON.parse(JSON.stringify(state.queryForm.total))
|
|
|
+ initWS()
|
|
|
+ }).catch((e) => {
|
|
|
+ ElMessage({
|
|
|
+ message: e[0].message,
|
|
|
+ grouping: true,
|
|
|
+ type: 'warning',
|
|
|
+ })
|
|
|
+ })
|
|
|
+ }
|
|
|
+ const initWS = () => {
|
|
|
+ const ws = new WebSocket(`ws://${location.host}/${store.state.app.apiProxy.shipPlaybackWSApi}`)
|
|
|
+ ws.onopen = (e) => {
|
|
|
+ const str = {
|
|
|
+ total: state.playBar.total,
|
|
|
+ timeArea: state.playBar.timeArea
|
|
|
+ }
|
|
|
+ ws.send(JSON.stringify(str))
|
|
|
+ }
|
|
|
+ ws.onmessage = (e) => {
|
|
|
+ try {
|
|
|
+ const json = JSON.parse(e.data)
|
|
|
+ state.playBar.cache++
|
|
|
+ const areaH = Math.ceil((new Date(state.playBar.timeArea[1]).getTime() - new Date(state.playBar.timeArea[0]).getTime()) / (1000 * 60 * 60))
|
|
|
+ state.playBar.cachePro = state.playBar.cache / areaH
|
|
|
+ setShipData(json)
|
|
|
+ } catch (e) {
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ const setShipData = (arr) => {
|
|
|
+ arr.forEach(v => {
|
|
|
+ if (state.shipData[v.id]) {
|
|
|
+ state.shipData[v.id].points.push(v)
|
|
|
+ } else {
|
|
|
+ state.shipData[v.id] = {
|
|
|
+ points: [v],
|
|
|
+ config: {
|
|
|
+ color: `rgb(${that.$util.randomNum(0, 255)}, ${that.$util.randomNum(0, 255)}, ${that.$util.randomNum(0, 255)})`
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ })
|
|
|
+ }
|
|
|
+ const onPlayRefresh = ({time, flag}) => {
|
|
|
+ const pointFeats: any = []
|
|
|
+ const lineFeats: any = []
|
|
|
+ Object.entries(state.shipData).forEach(([id, value]: any) => {
|
|
|
+ const lines: any = []
|
|
|
+ let real: any = null
|
|
|
+ for (let i = 0; i < value.points.length; i++) {
|
|
|
+ if (new Date(value.points[i].time).getTime() > time) {
|
|
|
+ break
|
|
|
+ }
|
|
|
+ lines.push(value.points[i].wkt)
|
|
|
+ real = value.points[i]
|
|
|
+ }
|
|
|
+ const pf: any = new format.WKT().readFeature(real.wkt)
|
|
|
+ pf.set('course', real.cogs)
|
|
|
+ pointFeats.push(pf)
|
|
|
+ const lf: any = new format.WKT().readFeature(that.$easyMap.formatPosition.wptTwl(lines))
|
|
|
+ lf.set('lineColor', value.config.color)
|
|
|
+ lineFeats.push(lf)
|
|
|
});
|
|
|
+ // 线
|
|
|
+ if (state.webglLineLayer) {
|
|
|
+ state.map.removeLayer(state.webglLineLayer)
|
|
|
+ state.webglLineLayer.dispose()
|
|
|
+ }
|
|
|
+ state.webglLineLayer = new WebGLLineLayer({
|
|
|
+ zIndex: 20,
|
|
|
+ source: new source.Vector({
|
|
|
+ features: lineFeats
|
|
|
+ }),
|
|
|
+ })
|
|
|
+ state.map.addLayer(state.webglLineLayer)
|
|
|
+ // 点
|
|
|
+ if (state.webglPointLayer) {
|
|
|
+ state.map.removeLayer(state.webglPointLayer)
|
|
|
+ state.webglPointLayer.dispose()
|
|
|
+ }
|
|
|
+ state.webglPointLayer = new layer.WebGLPoints({
|
|
|
+ zIndex: 30,
|
|
|
+ source: new source.Vector({
|
|
|
+ features: pointFeats
|
|
|
+ }),
|
|
|
+ style: {
|
|
|
+ "icon-src": ShipImg,
|
|
|
+ "icon-color": '#095217',
|
|
|
+ "icon-rotation": ['get', 'course']
|
|
|
+ }
|
|
|
+ })
|
|
|
+ state.map.addLayer(state.webglPointLayer)
|
|
|
}
|
|
|
onMounted(() => {
|
|
|
initMap()
|
|
|
})
|
|
|
return {
|
|
|
...toRefs(state),
|
|
|
+ mapLoad,
|
|
|
+ onSearch,
|
|
|
+ ref_form,
|
|
|
+ onPlayRefresh
|
|
|
}
|
|
|
},
|
|
|
})
|
|
@@ -68,8 +225,23 @@ export default defineComponent({
|
|
|
width: 100%;
|
|
|
height: 100vh;
|
|
|
display: flex;
|
|
|
- >div {
|
|
|
- flex: 1;
|
|
|
+ position: relative;
|
|
|
+ justify-content: center;
|
|
|
+ .play-bar {
|
|
|
+ position: absolute;
|
|
|
+ z-index: 2;
|
|
|
+ bottom: 20px;
|
|
|
+ left: 20px;
|
|
|
+ width: calc(100% - 400px);
|
|
|
+ }
|
|
|
+ .filter {
|
|
|
+ position: absolute;
|
|
|
+ z-index: 2;
|
|
|
+ top: 0;
|
|
|
+ left: 0;
|
|
|
+ width: 450px;
|
|
|
+ background-color: #FFFFFF;
|
|
|
+ padding: 10px;
|
|
|
}
|
|
|
}
|
|
|
</style>
|