index.vue 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534
  1. <template>
  2. <div class="init-speed-track" v-loading="loading" element-loading-background="rgba(0, 0, 0, 0.5)">
  3. <EasyMapComponent
  4. class="map"
  5. :showBaseSwitch="true"
  6. @easyMapLoad="mapLoad"
  7. />
  8. <div class="track">
  9. <el-card shadow="always">
  10. <template #header>
  11. <div class="card-header">
  12. <span>轨迹列表</span>
  13. <el-button-group>
  14. <template v-for="[key, value] in SourceMap">
  15. <el-button :color="value.color" size="small" @click="drawTrack(key)" style="color: white">{{value.label}}</el-button>
  16. </template>
  17. </el-button-group>
  18. </div>
  19. </template>
  20. <div class="track-line">
  21. <template v-for="(item, index) in trackList">
  22. <div class="line">
  23. <div class="label" :style="`color: ${SourceMap.get(item.type).color};`">{{SourceMap.get(item.type).label}}</div>
  24. <el-tooltip :enterable="false" placement="top" content="隐藏" v-if="item.show">
  25. <img class="__hover" src="./ship-track-visible.svg" @click="handleShow(false, item)"/>
  26. </el-tooltip>
  27. <el-tooltip :enterable="false" placement="top" content="显示" v-else>
  28. <img class="__hover" src="./ship-track-invisible.svg" @click="handleShow(true, item)"/>
  29. </el-tooltip>
  30. </div>
  31. </template>
  32. </div>
  33. </el-card>
  34. <el-card shadow="always">
  35. <template #header>
  36. <div class="card-header">
  37. <span>轨迹点列表</span>
  38. <el-button v-if="trackPointList.length > 0" type="primary" size="small" @click="onSubmit" style="color: white">保存</el-button>
  39. </div>
  40. </template>
  41. <div class="track-point">
  42. <template v-for="item in formParams">
  43. <div class="item">
  44. <span>{{item.label}}:</span><el-input v-model="form[item.value]" clearable/>
  45. </div>
  46. </template>
  47. <template v-for="(item, index) in trackPointList">
  48. <div class="point">
  49. <div class="position">
  50. <span>{{item.position[0]}}</span><br/>
  51. <span>{{item.position[1]}}</span>
  52. </div>
  53. <div class="speed">
  54. <el-input-number v-model="item.speed" :precision="2" :step="0.1" :max="100" :min="0" @focus="onPointFocus(trackPointList[index - 1], item, trackPointList[index + 1])"/>
  55. </div>
  56. </div>
  57. </template>
  58. </div>
  59. </el-card>
  60. </div>
  61. </div>
  62. </template>
  63. <script lang="ts">
  64. import {
  65. defineComponent,
  66. ref,
  67. nextTick,
  68. onMounted,
  69. watch,
  70. computed,
  71. ComponentInternalInstance,
  72. reactive,
  73. toRefs,
  74. getCurrentInstance
  75. } from 'vue'
  76. import {useStore} from 'vuex'
  77. import * as source from "ol/source";
  78. import * as layer from "ol/layer";
  79. import * as style from "ol/style";
  80. import * as ol from "ol";
  81. import * as sphere from "ol/sphere";
  82. import * as interaction from "ol/interaction";
  83. import {createBox} from "ol/interaction/Draw";
  84. import {unByKey} from "ol/Observable";
  85. import { Geometry } from 'ol/geom';
  86. import { EventsKey } from 'ol/events';
  87. import { Coordinate } from 'ol/coordinate';
  88. import TrackStyle from './track-style'
  89. import axios from "axios";
  90. import {ElMessage} from "element-plus";
  91. export default defineComponent({
  92. name: 'App',
  93. components: {},
  94. setup() {
  95. const store = useStore()
  96. const that = (getCurrentInstance() as ComponentInternalInstance).appContext.config.globalProperties
  97. const SourceMap = new Map(window.cusConfig.trackSource)
  98. const formParams = window.cusConfig.trackParams
  99. const state = reactive({
  100. map: <any>null,
  101. mapFunc: null,
  102. trackPointList: [],
  103. trackList: <any>[],
  104. formTrackPointStartCount: 0,
  105. formTrackPointEndCount: 0,
  106. formTrackPointList: [],
  107. initTrackPointStartCount: 0,
  108. initTrackPointEndCount: 0,
  109. initTrackPointList: [],
  110. loading: false,
  111. form: {}
  112. });
  113. formParams.forEach(v => {
  114. state.form[v.value] = v.init
  115. })
  116. const mapLoad = (map: null, func: null) => {
  117. state.map = map
  118. state.mapFunc = func
  119. }
  120. const startDraw = (cb: { (evt: any): void; (arg0: { feature: { getGeometry: () => any; }; }): void; }) => {
  121. let measureTooltipElement: HTMLDivElement | null;
  122. let helpTooltipElement: HTMLDivElement | null;
  123. const realLayer = state.map.getLayers().getArray().filter((v: { get: (arg0: string) => string; }) => v.get('layerName') === 'measureLayer')
  124. let sketch: { getGeometry: () => { (): any; new(): any; on: { (arg0: string, arg1: (evt: any) => void): any; new(): any; }; }; } | null;
  125. let helpTooltip: ol.Overlay;
  126. let measureTooltip: ol.Overlay;
  127. let continueMsg = '双击结束标绘';
  128. const createMeasureTooltip = () => {
  129. const id = 'measureTooltipElementId'
  130. if (measureTooltipElement) {
  131. state.map.removeOverlay(state.map.getOverlayById(id))
  132. measureTooltipElement.parentNode?.removeChild(measureTooltipElement);
  133. }
  134. measureTooltipElement = document.createElement('div');
  135. measureTooltipElement.className = 'tooltip tooltip-measure';
  136. measureTooltip = new ol.Overlay({
  137. id,
  138. element: measureTooltipElement,
  139. offset: [0, -15],
  140. positioning: 'bottom-center'
  141. });
  142. state.map.addOverlay(measureTooltip);
  143. }
  144. const createHelpTooltip = () => {
  145. const id = 'helpTooltipElementId'
  146. if (helpTooltipElement) {
  147. state.map.removeOverlay(state.map.getOverlayById(id))
  148. helpTooltipElement.parentNode?.removeChild(helpTooltipElement);
  149. }
  150. helpTooltipElement = document.createElement('div');
  151. helpTooltipElement.className = 'tooltip hidden';
  152. helpTooltip = new ol.Overlay({
  153. id,
  154. element: helpTooltipElement,
  155. offset: [15, 0],
  156. positioning: 'center-left'
  157. });
  158. state.map.addOverlay(helpTooltip);
  159. }
  160. const formatLength = (line: Geometry) => {
  161. // 获取投影坐标系
  162. const sourceProj = state.map.getView().getProjection();
  163. // ol/sphere里有getLength()和getArea()用来测量距离和区域面积,默认的投影坐标系是EPSG:3857, 其中有个options的参数,可以设置投影坐标系
  164. const length = sphere.getLength(line, {projection: sourceProj});
  165. // const length = getLength(line);
  166. let output;
  167. if (length > 100) {
  168. const km = Math.round((length / 1000) * 100) / 100;
  169. output = `${km} 千米 <br>${parseFloat(String(km * 0.53995)).toFixed(2)} 海里`;
  170. } else {
  171. output = `${Math.round(length * 100) / 100} m`;
  172. }
  173. return output;
  174. };
  175. const addInteraction = () => {
  176. const id = 'drawName'
  177. const draw = new interaction.Draw({
  178. type: 'LineString', //几何图形类型
  179. style: new style.Style({
  180. fill: new style.Fill({
  181. color: "rgba(255, 255, 255, 0.2)",
  182. }),
  183. stroke: new style.Stroke({
  184. color: "#f3584a",
  185. width: 2,
  186. }),
  187. image: new style.Circle({
  188. radius: 5,
  189. stroke: new style.Stroke({
  190. color: "rgba(0, 0, 0, 0.7)",
  191. }),
  192. fill: new style.Fill({
  193. color: "rgba(255, 255, 255, 0.2)",
  194. }),
  195. }),
  196. }),
  197. });
  198. draw.set(id, id)
  199. createMeasureTooltip(); //创建测量工具提示框
  200. createHelpTooltip(); //创建帮助提示框
  201. state.map.addInteraction(draw);
  202. let listener: EventsKey | EventsKey[];
  203. //绑定交互绘制工具开始绘制的事件
  204. const drawstartHandle = (evt: { feature: { getGeometry: () => { (): any; new(): any; on: { (arg0: string, arg1: (evt: any) => void): any; new(): any; }; }; } | null; coordinate: any; }) => {
  205. sketch = evt.feature; //绘制的要素
  206. let tooltipCoord = evt.coordinate;// 绘制的坐标
  207. //绑定change事件,根据绘制几何类型得到测量长度值或面积值,并将其设置到测量工具提示框中显示
  208. listener = sketch?.getGeometry().on('change', function (evt) {
  209. const geom = evt.target
  210. let output;
  211. output = formatLength(geom);//长度值
  212. tooltipCoord = geom.getLastCoordinate();//坐标
  213. if (measureTooltipElement) measureTooltipElement.innerHTML = output;//将测量值设置到测量工具提示框中显示
  214. measureTooltip.setPosition(tooltipCoord);//设置测量工具提示框的显示位置
  215. });
  216. }
  217. draw.on('drawstart', drawstartHandle);
  218. //绑定交互绘制工具结束绘制的事件
  219. const copy = (value: string) => {
  220. const str = document.createElement('input')
  221. str.setAttribute('value', value)
  222. document.body.appendChild(str)
  223. str.select()
  224. document.execCommand('copy')
  225. document.body.removeChild(str)
  226. }
  227. const drawendHandle = (evt: { feature: { getGeometry: () => any; }; }) => {
  228. state.map.removeInteraction(state.map.getInteractions().getArray().filter((v: { get: (arg0: string) => string; }) => v.get(id) === id)[0]);
  229. sketch = null; //置空当前绘制的要素对象
  230. measureTooltipElement?.parentNode?.removeChild(measureTooltipElement);
  231. measureTooltipElement = null; //置空测量工具提示框对象
  232. helpTooltipElement?.parentNode?.removeChild(helpTooltipElement);
  233. helpTooltipElement = null; //置空测量工具提示框对象
  234. unByKey(listener);
  235. draw.un('drawstart', drawstartHandle);
  236. draw.un('drawend', drawendHandle);
  237. state.map.removeInteraction(state.map.getInteractions().getArray().filter((v: { get: (arg0: string) => string; }) => v.get(id) === id)[0]);
  238. state.map.un('pointermove', pointerMoveHandler)
  239. cb(evt)
  240. }
  241. draw.on('drawend', drawendHandle);
  242. }
  243. addInteraction(); //调用加载绘制交互控件方法,添加绘图进行测量
  244. const pointerMoveHandler = (evt: { dragging: any; coordinate: Coordinate | undefined; }) => {
  245. if (evt.dragging) {
  246. return;
  247. }
  248. let helpMsg = '单击开始标绘';//当前默认提示信息
  249. //判断绘制几何类型设置相应的帮助提示信息
  250. if (sketch) {
  251. const geom = sketch.getGeometry()
  252. helpMsg = continueMsg;
  253. // if (geom.getType() === 'Polygon') {
  254. // helpMsg = continueMsg; //绘制多边形时提示相应内容
  255. // } else if (geom.getType() === 'LineString') {
  256. // helpMsg = continueMsg; //绘制线时提示相应内容
  257. // }
  258. }
  259. if (helpTooltipElement)helpTooltipElement.innerHTML = helpMsg; //将提示信息设置到对话框中显示
  260. helpTooltip.setPosition(evt.coordinate);//设置帮助提示框的位置
  261. helpTooltipElement?.classList.remove('hidden');//移除帮助提示框的隐藏样式进行显示
  262. };
  263. state.map.on('pointermove', pointerMoveHandler); //地图容器绑定鼠标移动事件,动态显示帮助提示框内容
  264. //地图绑定鼠标移出事件,鼠标移出时为帮助提示框设置隐藏样式
  265. // state.map.getViewport().on('mouseout', () => {
  266. // helpTooltipElement?.addClass('hidden');
  267. // });
  268. }
  269. const drawTrack = (trackSource: string) => {
  270. startDraw((evt) => {
  271. const geom = evt.feature.getGeometry()
  272. const pMap = new Map()
  273. state.trackPointList = geom.getCoordinates().map((v: any) => {
  274. const obj = {
  275. source: trackSource,
  276. position: v,
  277. speed: 0
  278. }
  279. pMap.set(`${v[0]}-${v[1]}`, obj)
  280. return obj
  281. })
  282. that.$easyMap.initShape({
  283. map: state.map,
  284. layerName: "form-track-point-line",
  285. layerZIndex: 9,
  286. list: [
  287. {
  288. easyMapParams: {
  289. id: new Date().getTime(),
  290. position: that.$easyMap.formatPosition.wptTwl(state.trackPointList.map(v => that.$easyMap.formatPosition.cptTwpt(v.position))),
  291. normalStyle: (f: any, r: any) => TrackStyle.trackLineStyle(f, r, state.map, SourceMap.get(trackSource)?.color, pMap, (s, p) => {
  292. state.formTrackPointStartCount++
  293. setTimeout(() => {
  294. state.formTrackPointEndCount++
  295. state.formTrackPointList.push(...p)
  296. if (state.formTrackPointStartCount === state.formTrackPointEndCount) {
  297. that.$easyMap.initShape({
  298. map: state.map,
  299. layerName: 'form-track-point',
  300. layerZIndex: 10,
  301. list: state.formTrackPointList.map((v, i) => {
  302. return {
  303. easyMapParams: {
  304. id: `form-track-point-${i}`,
  305. position: that.$easyMap.formatPosition.cptTwpt(v.position),
  306. normalStyle: TrackStyle.trackPointStyle(SourceMap.get(v.source).color, v.speed)
  307. }
  308. }
  309. })
  310. })
  311. state.formTrackPointStartCount = 0
  312. state.formTrackPointEndCount = 0
  313. state.formTrackPointList = []
  314. }
  315. }, 10)
  316. return s
  317. }),
  318. }
  319. }
  320. ]
  321. });
  322. })
  323. }
  324. const onPointFocus = (p1: any, p2: any, p3: any) => {
  325. that.$easyMap.getShapeView(state.map, [p1?.position, p2.position, p3?.position].filter(v => v))
  326. const radius = 25
  327. const longRadius = radius * Math.SQRT2
  328. that.$easyMap.initShape({
  329. map: state.map,
  330. layerName: "focus",
  331. layerZIndex: 20,
  332. list: [
  333. {
  334. easyMapParams: {
  335. id: 'focus',
  336. position: that.$easyMap.formatPosition.cptTwpt(p2.position),
  337. normalStyle: [new style.Style({ //图层样式
  338. image: new style.RegularShape({
  339. stroke: new style.Stroke({
  340. color: '#9F2EFF',
  341. width: 2,
  342. lineDash: [
  343. (longRadius * 3) / 10,
  344. (longRadius * 4) / 10,
  345. (longRadius * 3) / 10,
  346. 0
  347. ]
  348. }),
  349. radius1: radius,
  350. rotation: Math.PI / (180 / 45),
  351. points: 4
  352. })
  353. })]
  354. }
  355. }
  356. ]
  357. });
  358. }
  359. const onSubmit = () => {
  360. const obj = {
  361. type: state.trackPointList[0].source,
  362. lines: state.trackPointList.map(v => {
  363. return {
  364. lon: v.position[0],
  365. lat: v.position[1],
  366. speed: v.speed
  367. }
  368. })
  369. }
  370. Object.assign(obj, state.form)
  371. const result = JSON.parse(JSON.stringify(obj))
  372. console.log(result)
  373. state.loading = true
  374. axios.post("/init-speed-track-api/hujie-track-server/mock", [result], {
  375. contentType: "application/json"
  376. }).then(res => {
  377. console.log(res)
  378. if (res.status === 200 && res.data) {
  379. ElMessage.success('添加成功!')
  380. state.trackList.push(Object.assign(result, {show: false, ID: new Date().getTime()}))
  381. state.trackPointList = []
  382. that.$easyMap.initShape({
  383. map: state.map,
  384. layerName: "form-track-point-line",
  385. layerZIndex: 9,
  386. list: []
  387. });
  388. that.$easyMap.initShape({
  389. map: state.map,
  390. layerName: "form-track-point",
  391. layerZIndex: 10,
  392. list: []
  393. });
  394. } else {
  395. ElMessage.error('添加失败!')
  396. }
  397. state.loading = false
  398. })
  399. }
  400. const trackShowListCom = computed(() => {
  401. return state.trackList.filter((v: { show: any; }) => v.show)
  402. })
  403. const initTrack = () => {
  404. that.$easyMap.initShape({
  405. map: state.map,
  406. layerName: "track-point-line",
  407. layerZIndex: 7,
  408. list: []
  409. });
  410. that.$easyMap.initShape({
  411. map: state.map,
  412. layerName: "track-point",
  413. layerZIndex: 8,
  414. list: []
  415. });
  416. that.$easyMap.initShape({
  417. map: state.map,
  418. layerName: "track-point-line",
  419. layerZIndex: 7,
  420. list: trackShowListCom.value.map((v: any) => {
  421. const pMap = new Map()
  422. v.lines.forEach((p: {
  423. speed: any; type: any; lon: any; lat: any;
  424. }) => {
  425. const obj = {
  426. source: v.type,
  427. position: [p.lon, p.lat],
  428. speed: p.speed
  429. }
  430. pMap.set(`${p.lon}-${p.lat}`, obj)
  431. })
  432. return {
  433. easyMapParams: {
  434. id: v.ID,
  435. position: that.$easyMap.formatPosition.wptTwl(v.lines.map((c: { lon: any; lat: any; }) => that.$easyMap.formatPosition.cptTwpt([c.lon, c.lat]))),
  436. normalStyle: (f: any, r: any) => TrackStyle.trackLineStyle(f, r, state.map, SourceMap.get(v.type)?.color, pMap, (s, p) => {
  437. state.initTrackPointStartCount++
  438. setTimeout(() => {
  439. state.initTrackPointEndCount++
  440. state.initTrackPointList.push(...p)
  441. if (state.initTrackPointStartCount === state.initTrackPointEndCount) {
  442. that.$easyMap.initShape({
  443. map: state.map,
  444. layerName: 'track-point',
  445. layerZIndex: 8,
  446. list: state.initTrackPointList.map((v, i) => {
  447. return {
  448. easyMapParams: {
  449. id: `init-track-point-${v.ID}-${i}`,
  450. position: that.$easyMap.formatPosition.cptTwpt(v.position),
  451. normalStyle: TrackStyle.trackPointStyle(SourceMap.get(v.source).color, v.speed)
  452. }
  453. }
  454. })
  455. })
  456. state.initTrackPointStartCount = 0
  457. state.initTrackPointEndCount = 0
  458. state.initTrackPointList = []
  459. }
  460. }, 10)
  461. return s
  462. }),
  463. }
  464. }
  465. })
  466. });
  467. }
  468. const handleShow = (show: any, item: any) => {
  469. item.show = show
  470. initTrack()
  471. }
  472. return {
  473. ...toRefs(state),
  474. mapLoad,
  475. drawTrack,
  476. onPointFocus,
  477. SourceMap,
  478. onSubmit,
  479. handleShow,
  480. formParams
  481. }
  482. }
  483. })
  484. </script>
  485. <style lang="scss" scoped>
  486. .init-speed-track {
  487. width: 100%;
  488. height: 100%;
  489. position: relative;
  490. .map {
  491. width: 100%;
  492. height: 100vh;
  493. }
  494. .track {
  495. position: absolute;
  496. z-index: 20;
  497. top: 0;
  498. left: 0;
  499. .track-line {
  500. overflow-y: auto;
  501. max-height: 180px;
  502. .line {
  503. height: 20px;
  504. display: flex;
  505. align-items: center;
  506. .label {
  507. width: 90px;
  508. }
  509. }
  510. }
  511. .track-point {
  512. overflow-y: auto;
  513. max-height: 600px;
  514. .item {
  515. display: flex;
  516. align-items: center;
  517. white-space: nowrap;
  518. }
  519. .point {
  520. display: flex;
  521. .position {
  522. font-size: 12px;
  523. }
  524. border-bottom: 1px solid black;
  525. }
  526. }
  527. }
  528. }
  529. </style>