CzRger 1 mēnesi atpakaļ
vecāks
revīzija
3c8d44812a

+ 4 - 3
package.json

@@ -10,12 +10,15 @@
     "serve": "vite --host 0.0.0.0"
   },
   "dependencies": {
+    "@turf/turf": "^7.1.0",
     "@types/node": "^20.14.12",
     "axios": "^1.7.2",
+    "dayjs": "^1.11.13",
     "default-passive-events": "^2.0.0",
     "echarts": "^5.5.1",
     "element-plus": "^2.8.8",
     "fast-glob": "^3.3.2",
+    "ol": "^9.2.2",
     "pinia": "^2.1.7",
     "rollup-plugin-visualizer": "^5.12.0",
     "sass": "^1.77.8",
@@ -26,9 +29,7 @@
     "vite-plugin-svg-icons": "^2.0.1",
     "vite-plugin-top-level-await": "^1.4.2",
     "vue": "^3.4.31",
-    "vue-router": "^4.4.0",
-    "ol": "^9.2.4",
-    "@turf/turf": "^7.1.0"
+    "vue-router": "^4.4.0"
   },
   "devDependencies": {
     "@vitejs/plugin-vue": "^5.0.5",

+ 1 - 1
src/views/web/index.vue

@@ -69,7 +69,7 @@
       <archiveCom v-model:show="state.tools.showArchive"/>
       <warningCom v-model:show="state.tools.showWarning" :map="state.map"/>
       <exampleCom v-if="state.mapFunc" v-model:show="state.tools.showExample" :mapHeight="state.mapFunc.mapHeight" :mapWidth="state.mapFunc.mapWidth"/>
-      <trackCom v-model:show="state.tools.showTrack"/>
+      <trackCom v-model:show="state.tools.showTrack" :map="state.map" :mapFunc="state.mapFunc"/>
       <searchCom v-model:show="state.tools.showSearch"/>
       <trackPointCom ref="ref_trackPoint"/>
       <trackArchiveCom/>

+ 164 - 0
src/views/web/track/analysis-chart.vue

@@ -0,0 +1,164 @@
+<template>
+  <div class="chart-main" ref="ref_main">
+    <div class="chart-ref" ref="ref_chart"/>
+  </div>
+</template>
+
+<script setup lang="ts">
+import {computed, getCurrentInstance, nextTick, onMounted, reactive, ref, watch} from "vue";
+import {useShipMapStore} from "@/stores";
+import * as echarts from "echarts";
+import dayjs from "dayjs";
+
+const emit = defineEmits(['toPosition'])
+const ShipMapStore = useShipMapStore()
+const {proxy} = getCurrentInstance()
+const props = defineProps({
+  data: {
+    required: true,
+    default: () => []
+  },
+  tab: {
+    default: 'speed'
+  },
+})
+const state: any = reactive({
+  resizeObserver: <any>null,
+  chart: <any>null
+})
+const ref_chart = ref();
+const ref_main = ref();
+const initChart = () => {
+  echarts.dispose(ref_chart.value)
+  const chart = echarts.init(ref_chart.value);
+  if (props.data?.length > 0) {
+    let name
+    let key = ''
+    if (props.tab === 'speed') {
+      name = '航速'
+      key = ShipMapStore.trackKeys.speed
+    } else {
+      name = '航向'
+      key = ShipMapStore.trackKeys.course
+    }
+    const d = JSON.parse(JSON.stringify(props.data))
+    const option = {
+      tooltip: {
+        trigger: 'axis'
+      },
+      grid:{
+        top: 16,
+        right: '3%',
+        bottom: 48,
+        left: 20,
+        containLabel: true
+      },
+      dataZoom: [
+        {
+          type: 'inside',
+          start: 0,
+          end: 100
+        },
+        {
+          type: 'slider',
+          start: 0,
+          end: 100,
+          backgroundColor: '#1b40d6',
+          height: 20,
+          left: 16,
+          right: 16,
+          bottom: 20,
+          textStyle: {
+            color: '#ffffff'
+          }
+        }
+      ],
+      xAxis: {
+        axisLine: {
+          lineStyle: {
+            color: 'rgba(255, 255,255,.2)'
+          },
+        },
+        axisTick:{show: false},
+        splitLine: {
+          show: true,
+          lineStyle: {
+            color: 'rgba(255, 255,255,.2)'
+          }
+        },
+        axisLabel: {
+          color: '#ccc'
+        },
+        data: d.map(v => {
+          return dayjs(v[ShipMapStore.trackKeys.mergeTime]).format("MM/DD HH:mm:ss")
+        }),
+      },
+      yAxis: {
+        axisLine: {
+          show: true,
+          lineStyle: {
+            color: 'rgba(255, 255,255,.2)'
+          },
+        },
+        splitLine: {
+          show: true,
+          lineStyle: {
+            color: 'rgba(255, 255,255,.2)'
+          }
+        },
+        axisLabel: {
+          color: '#ccc'
+        },
+      },
+      series: [
+        {
+          name: name,
+          type: "line",
+          data: d.map(v => ({...v, value: Number(v[key]).toFixed(2)})),
+          lineStyle: {
+            width: 3,
+            color: '#7feffc'
+          },
+          itemStyle: {
+            color: '#65d0a9'
+          }
+        },
+      ],
+    };
+    chart.setOption(option);
+    chart.on('click', (params) => {
+      emit('toPosition', [params.data[ShipMapStore.trackKeys.lon], params.data[ShipMapStore.trackKeys.lat]])
+    })
+    state.resizeObserver = new ResizeObserver((entries) => {
+      for (const entry of entries) {
+        chart && chart?.resize()
+      }
+    })
+    state.resizeObserver.observe(ref_main.value);
+  }
+  return chart
+}
+watch(() => [props.data, props.tab], () => {
+  state.chart = initChart()
+})
+onMounted(() => {
+  nextTick(() => {
+    state.chart = initChart()
+  })
+  return () => {
+    state.resizeObserver?.unobserve(ref_main?.value)
+    state.resizeObserver?.disconnect()
+  }
+})
+</script>
+
+<style scoped lang="scss">
+.chart-main {
+  width: 100%;
+  height: 100%;
+  .chart-ref {
+    width: 100%;
+    height: 100%;
+  }
+}
+</style>

+ 130 - 0
src/views/web/track/analysis.vue

@@ -0,0 +1,130 @@
+<template>
+  <DragWindow
+    v-if="show"
+    @onClose="$emit('update:show', false)"
+    v-model:layout="state.layout"
+    expend
+    close
+  >
+    <template #title>
+      <div class="track-analysis-title __hover" :class="{active: state.tab === 'speed'}" @click="state.tab = 'speed'">航速分析</div>
+      <div class="track-analysis-title __hover" :class="{active: state.tab === 'course'}" @click="state.tab = 'course'">航向分析</div>
+    </template>
+    <div class="track-analysis-com">
+      <div class="tac-head">
+        <template v-if="state.tab === 'speed'">
+          <div class="tac-head-item">航速(节)</div>
+          <div class="tac-head-item">平均速度{{ trackCpt.avaSpeed }}节</div>
+        </template>
+        <template v-else-if="state.tab === 'course'">
+          <div class="tac-head-item">航向(°)</div>
+        </template>
+        <div class="tac-head-item">航行时间:{{ trackCpt.time }}</div>
+        <div class="tac-head-item special">点击航迹点定位</div>
+      </div>
+      <div class="chart">
+        <LineChart :data="state.points" :tab="state.tab" @toPosition="toPosition"/>
+      </div>
+    </div>
+  </DragWindow>
+</template>
+
+<script setup lang="ts">
+import {computed, getCurrentInstance, reactive} from "vue";
+import DragWindow from "@/views/web/components/drag-window.vue";
+import {comTimeByArea} from "@/utils/util";
+import {useShipMapStore} from "@/stores";
+import LineChart from './analysis-chart.vue'
+
+const ShipMapStore = useShipMapStore()
+const {proxy} = getCurrentInstance()
+const props = defineProps({
+  show: {},
+  mapFunc: {},
+  map: {},
+})
+const state: any = reactive({
+  layout: {
+    width: 430,
+    left: 85,
+    top: 110
+  },
+  points: [],
+  tab: 'speed'
+})
+const init = (layout, points) => {
+  state.tab = 'speed'
+  state.layout = layout
+  state.points = points
+}
+const trackCpt = computed(() => {
+  const arr = state.points
+  const time = comTimeByArea(arr[0][ShipMapStore.trackKeys.mergeTime], arr[arr.length - 1][ShipMapStore.trackKeys.mergeTime], true)
+  let allSpeed = 0
+  arr.forEach(v => {
+    allSpeed += Number(v[ShipMapStore.trackKeys.speed])
+  })
+  const avaSpeed = (allSpeed / arr.length).toFixed(2)
+  return {
+    time, avaSpeed
+  }
+})
+const toPosition = (coor) => {
+  props.mapFunc.toLocation({
+    position: coor,
+    zoom: props.map.getView().getMaxZoom()
+  })
+}
+defineExpose({
+  init,
+})
+</script>
+
+<style lang="scss" scoped>
+.track-analysis-title {
+  color: rgba(255, 255, 255, 0.7);
+  &:not(:last-child) {
+    margin-right: 20px;
+  }
+  &.active {
+    color: rgba(255, 255, 255, 1);
+    position: relative;
+    &:after {
+      content: '';
+      position: absolute;
+      left: 0;
+      bottom: -8px;
+      width: 100%;
+      height: 2px;
+      background-color: #87b0ed;
+    }
+  }
+}
+.track-analysis-com {
+  height: 264px;
+  display: flex;
+  flex-direction: column;
+  .tac-head {
+    display: flex;
+    justify-content: space-between;
+    align-items: center;
+    font-size: 12px;
+    color: #FFFFFF;
+    padding: 10px 10px 0 10px;
+    .special {
+      display: flex;
+      align-items: center;
+      color: #47f4ff;
+      .export-button {
+        border: 1px solid #21A0F1;
+        border-radius: 4px;
+        padding: 1px 4px;
+        margin-left: 4px;
+      }
+    }
+  }
+  .chart {
+    flex: 1;
+  }
+}
+</style>