ol-map.vue 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280
  1. <template>
  2. <div class="easy-map-ol">
  3. <div class="map" ref="ref_easyMapOl"/>
  4. <div class="easy-map_ol-info_tips">
  5. <div class="easy-map_ol-info-scaleLine" ref="ref_easyMap_scaleLine"></div>
  6. <div class="easy-map_ol-info-info">
  7. <div @click="controlMousePosition.format = !controlMousePosition.format">经度:{{controlMousePosition.format ? controlMousePosition.formatLongitude : controlMousePosition.longitude}}</div>
  8. <div @click="controlMousePosition.format = !controlMousePosition.format">纬度:{{controlMousePosition.format ? controlMousePosition.formatLatitude : controlMousePosition.latitude}}</div>
  9. <div>层级:
  10. <div class="easy-map_ol-info-zoom">
  11. <div class="easy-map_ol-info-zoom-button easy-map_ol-info-zoom-button_min __hover" @click="zoomChange(false)">-</div>
  12. <div class="easy-map_ol-info-zoom-num">{{Math.floor(interactionZoom)}}</div>
  13. <div class="easy-map_ol-info-zoom-button easy-map_ol-info-zoom-button_max __hover" @click="zoomChange(true)">+</div>
  14. </div>
  15. </div>
  16. </div>
  17. </div>
  18. </div>
  19. </template>
  20. <script lang="ts">
  21. import {
  22. defineComponent,
  23. onMounted,
  24. ref,
  25. toRefs,
  26. reactive,
  27. watch,
  28. getCurrentInstance,
  29. ComponentInternalInstance,
  30. computed,
  31. nextTick, onActivated,
  32. } from "vue";
  33. import { useStore } from "vuex";
  34. import * as ol from 'ol'
  35. import * as style from 'ol/style'
  36. import * as layer from 'ol/layer'
  37. import * as source from 'ol/source'
  38. import * as geom from 'ol/geom'
  39. import * as proj from 'ol/proj'
  40. import * as interaction from 'ol/interaction'
  41. import * as coordinate from 'ol/coordinate'
  42. import * as control from 'ol/control'
  43. import InitMapInfoClass from "./initMapInfo";
  44. export default defineComponent({
  45. name: "",
  46. components: {
  47. },
  48. props: {
  49. baseMapLayers: { default: () => [] },
  50. baseMapView: {},
  51. },
  52. setup(props, { emit }) {
  53. const store = useStore();
  54. const that = (getCurrentInstance() as ComponentInternalInstance).appContext.config.globalProperties;
  55. const state = reactive({
  56. controlMousePosition: {
  57. longitude: <any>null,
  58. latitude: <any>null,
  59. format: true,
  60. formatLongitude: <any>null,
  61. formatLatitude: <any>null,
  62. },
  63. interactionZoom: props.baseMapView.zoom,
  64. realBaseLayers: null,
  65. });
  66. const easyMapOl = ref()
  67. const ref_easyMapOl = ref()
  68. const ref_easyMap_olMousePosition = ref()
  69. const ref_easyMap_olZoom = ref()
  70. const ref_easyMap_scaleLine = ref()
  71. const initMap = () => {
  72. state.realBaseLayers = props.baseMapLayers.map(v => InitMapInfoClass.isInternet ? InitMapInfoClass.initInternet(v) : InitMapInfoClass.initBaseLayer(v))
  73. easyMapOl.value = new ol.Map({
  74. target: ref_easyMapOl.value,
  75. layers: [
  76. new layer.Group({
  77. _easyMapOl_layerGroupType: 'base',
  78. layers: state.realBaseLayers,
  79. zIndex: 1
  80. }),
  81. ],
  82. view: new ol.View(props.baseMapView),
  83. controls: control.defaults({
  84. attribution: false,
  85. rotate: false,
  86. zoom: false,
  87. }).extend([
  88. new control.MousePosition({
  89. target: ref_easyMap_olMousePosition.value,
  90. coordinateFormat: (e) => {
  91. state.controlMousePosition.longitude = e[0].toFixed(6)
  92. state.controlMousePosition.latitude = e[1].toFixed(6)
  93. const f = coordinate.toStringHDMS(e, 0).split(' ')
  94. state.controlMousePosition.formatLatitude = `${f[0]} ${f[1]} ${f[2]} ${f[3]}`
  95. state.controlMousePosition.formatLongitude = `${f[4]} ${f[5]} ${f[6]} ${f[7]}`
  96. return null
  97. },
  98. placeholder: ''
  99. }),
  100. new control.Zoom({
  101. target: ref_easyMap_olZoom.value,
  102. }),
  103. new control.ScaleLine({
  104. target: ref_easyMap_scaleLine.value,
  105. bar: true
  106. })
  107. ]),
  108. interactions: interaction.defaults({
  109. doubleClickZoom: false
  110. })
  111. })
  112. easyMapOl.value.getView().on('change:resolution', e => {
  113. state.interactionZoom = e.target.getZoom()
  114. emit('olZoomChange', state.interactionZoom)
  115. })
  116. const defaultBaseLayer = state.realBaseLayers.filter(v => v.getVisible())[0]
  117. setLayerView(defaultBaseLayer)
  118. emit('olMapLoad', easyMapOl.value)
  119. easyMapOl.value.on('contextmenu', e => {
  120. window.event.returnValue = false
  121. if (window?.event?.shiftKey) {
  122. const str = document.createElement('input')
  123. str.setAttribute('value', `POINT(${e.coordinate[0]} ${e.coordinate[1]})`)
  124. document.body.appendChild(str)
  125. str.select()
  126. document.execCommand('copy')
  127. document.body.removeChild(str)
  128. }
  129. })
  130. }
  131. const zoomChange = (flag) => {
  132. state.interactionZoom = flag ? state.interactionZoom + 1 : state.interactionZoom - 1
  133. if (state.interactionZoom > easyMapOl.value.getView().getMaxZoom()) {
  134. state.interactionZoom = easyMapOl.value.getView().getMaxZoom()
  135. } else if (state.interactionZoom < easyMapOl.value.getView().getMinZoom()) {
  136. state.interactionZoom = easyMapOl.value.getView().getMinZoom()
  137. }
  138. easyMapOl.value.getView().setZoom(state.interactionZoom)
  139. }
  140. const baseLayersMap = computed(() => {
  141. const map = new Map()
  142. easyMapOl.value?.getLayers().getArray().filter(v => v.get('_easyMapOl_layerGroupType') === 'base')[0].getLayers().getArray().forEach(v => {
  143. map.set(v.get('_easyMapOl_layerName'), v)
  144. })
  145. return map
  146. })
  147. const switchBaseLayer = (layerName) => {
  148. baseLayersMap.value.forEach((v, k) => {
  149. if (layerName === k) {
  150. setLayerView(v)
  151. v.setVisible(true)
  152. } else {
  153. v.setVisible(false)
  154. }
  155. })
  156. }
  157. const setLayerView = (_layer) => {
  158. easyMapOl.value.getView().setMaxZoom(_layer.get('_maxZoom'))
  159. easyMapOl.value.getView().setMinZoom(_layer.get('_minZoom'))
  160. }
  161. const refresh = () => {
  162. easyMapOl.value.updateSize();
  163. easyMapOl.value.render()
  164. }
  165. onMounted(() => {
  166. nextTick(() => {
  167. initMap()
  168. setTimeout(() => {
  169. refresh()
  170. })
  171. })
  172. })
  173. return {
  174. ...toRefs(state),
  175. ref_easyMapOl,
  176. ref_easyMap_olMousePosition,
  177. ref_easyMap_olZoom,
  178. ref_easyMap_scaleLine,
  179. easyMapOl,
  180. zoomChange,
  181. switchBaseLayer,
  182. baseLayersMap,
  183. refresh,
  184. }
  185. },
  186. });
  187. </script>
  188. <style scoped lang="scss">
  189. .easy-map-ol {
  190. width: 100%;
  191. height: 100%;
  192. position: relative;
  193. .map {
  194. width: 100%;
  195. height: 100%;
  196. background-color: #bfdbf3;
  197. }
  198. ::v-deep(.ol-zoom) {
  199. display: none;
  200. }
  201. ::v-deep(.ol-scale-bar-inner) {
  202. position: absolute;
  203. bottom: 1%;
  204. left: 1%;
  205. &>div>div:nth-child(2) {
  206. .ol-scale-singlebar {
  207. border-left: 2px solid #807A7A;
  208. }
  209. }
  210. &>div>div:nth-child(5) {
  211. .ol-scale-singlebar {
  212. border-right: 2px solid #807A7A;
  213. }
  214. }
  215. .ol-scale-step-text {
  216. padding-bottom: 20px;
  217. font-size: 12px;
  218. transform: scale(0.8);
  219. position: absolute;
  220. bottom: -5px;
  221. font-size: 12px;
  222. z-index: 11;
  223. color: #000000;
  224. width: max-content;
  225. text-shadow: -2px 0 #ffffff,
  226. 0 2px #ffffff,
  227. 2px 0 #ffffff,
  228. 0 -2px #ffffff;
  229. }
  230. .ol-scale-singlebar {
  231. border: 0;
  232. background-color: transparent !important;
  233. border-bottom: 2px solid #807A7A;
  234. height: 10px;
  235. }
  236. .ol-scale-step-marker {
  237. display: none;
  238. }
  239. }
  240. .easy-map_ol-info_tips {
  241. position: absolute;
  242. z-index: 1;
  243. right: 10px;
  244. bottom: 10px;
  245. display: flex;
  246. .easy-map_ol-info-info {
  247. background-color: rgba(255, 255, 255, 0.7);
  248. border-radius: 8px;
  249. color: #333333;
  250. font-size: 12px;
  251. padding: 6px;
  252. >div {
  253. display: flex;
  254. align-items: center;
  255. }
  256. .easy-map_ol-info-zoom {
  257. display: flex;
  258. align-items: center;
  259. .easy-map_ol-info-zoom-num {
  260. line-height: 1;
  261. padding: 1px 10px;
  262. border: 1px solid rgba(51, 51, 51, 0.51);
  263. }
  264. .easy-map_ol-info-zoom-button {
  265. padding: 0 3px;
  266. &.easy-map_ol-info-zoom-button_max {
  267. color: #356de7;
  268. }
  269. }
  270. }
  271. }
  272. .easy-map_ol-info-scaleLine {
  273. width: 180px;
  274. }
  275. }
  276. }
  277. </style>