map-style.ts 8.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247
  1. import * as style from 'ol/style'
  2. import * as geom from 'ol/geom'
  3. import * as extent from "ol/extent";
  4. import MapWarningIconRed from "@/assets/images/web/map-warning-icon-red.svg";
  5. import MapWarningIconYellow from "@/assets/images/web/map-warning-icon-yellow.svg";
  6. import MapWarningIconBlue from "@/assets/images/web/map-warning-icon-blue.svg";
  7. import AIS from './ship-type/AIS.svg'
  8. import AIS_BEIDOU from './ship-type/AIS_BEIDOU.svg'
  9. import AIS_BEIDOU_RADAR from './ship-type/AIS_BEIDOU_RADAR.svg'
  10. import AIS_RADAR from './ship-type/AIS_RADAR.svg'
  11. import BEIDOU from './ship-type/BEIDOU.svg'
  12. import BEIDOU_RADAR from './ship-type/BEIDOU_RADAR.svg'
  13. import OTHER from './ship-type/OTHER.svg'
  14. import RADAR from './ship-type/RADAR.svg'
  15. import FIXED from './ship-type/FIXED.svg'
  16. const shipIcon = {
  17. AIS, AIS_BEIDOU, AIS_BEIDOU_RADAR, AIS_RADAR, BEIDOU, BEIDOU_RADAR, OTHER, RADAR, FIXED
  18. }
  19. const speedStyle = (lineRadius, rotation) => {
  20. return new style.Style({
  21. image: new style.RegularShape({
  22. stroke: new style.Stroke({
  23. color: '#000000',
  24. width: 1
  25. }),
  26. displacement: [0, lineRadius],
  27. radius: lineRadius,
  28. rotation,
  29. points: 2
  30. })
  31. })
  32. }
  33. const headingStyle = (lineRadius, headAngle, rotation) => {
  34. const headRadius = 6
  35. return new style.Style({
  36. image: new style.RegularShape({
  37. stroke: new style.Stroke({
  38. color: '#ee1919',
  39. width: 1
  40. }),
  41. displacement: [
  42. headRadius * Math.sin(headAngle / 180 * Math.PI),
  43. (lineRadius * 2) + headRadius * Math.cos(headAngle / 180 * Math.PI)
  44. ], //这是偏移量
  45. radius: headRadius,
  46. rotation,
  47. angle: Math.PI / (180 / headAngle), //这是角度计算出的地图上显示的角度
  48. points: 2
  49. })
  50. })
  51. }
  52. const ShipNormalStyle = ({course = 0, speed = 0, head = null, color = '#01f200', mergeType = 'OTHER'}) => {
  53. const _style: any = []
  54. const allRotation = Math.PI / (180 / course)
  55. let lineRadius = 0
  56. if (speed > 0) {
  57. if (speed <= 5) {
  58. lineRadius = 2
  59. } else if (speed <= 10) {
  60. lineRadius = 6
  61. } else if (speed <= 20) {
  62. lineRadius = 18
  63. } else if (speed > 20) {
  64. lineRadius = 30
  65. }
  66. }
  67. // 船速
  68. _style.push(speedStyle(lineRadius, allRotation))
  69. if (head) {
  70. const headAngle = head + (360 - course)
  71. // 船首向
  72. _style.push(headingStyle(lineRadius, headAngle, allRotation))
  73. }
  74. _style.push(new style.Style({
  75. image: new style.Icon({
  76. color: color,
  77. src: shipIcon[mergeType],
  78. rotation: allRotation,
  79. scale: 1
  80. }),
  81. }))
  82. return _style
  83. }
  84. const trackStyle = (feature, resolution, map, color, callback) => {
  85. const geometry = feature.getGeometry();
  86. const length = geometry.getLength();//获取线段长度
  87. const radio = (200 * resolution) / length;
  88. const dradio = 1;//投影坐标系,如3857等,在EPSG:4326下可以设置dradio=10000
  89. const _style = [
  90. new style.Style({
  91. stroke: new style.Stroke({
  92. color: color,
  93. width: 2,
  94. })
  95. })
  96. ];
  97. const radius = 10
  98. const longRadius = radius * Math.SQRT2;
  99. const flag = map.getView().getZoom() < map.getView().getMaxZoom() - 1
  100. const E = 0.00000001
  101. const judgeIs = (p1, p2, p3) => {
  102. const k1 = (ps, pe) => {
  103. return (pe[1] - ps[1]) / (pe[0] - ps[0])
  104. }
  105. const k2 = (ps, pe) => {
  106. return (pe[0] - ps[0]) / (pe[1] - ps[0])
  107. }
  108. const a = (ps, pe) => {
  109. return Math.abs(k1(p1, ps) - k1(p1, pe)) <= E && Math.abs(k1(p1, ps) - k1(p1, pe)) >= -E
  110. }
  111. const d = (ps, pe) => {
  112. return Math.abs(k2(p1, ps) - k2(p1, pe)) <= E && Math.abs(k2(p1, ps) - k2(p1, pe)) >= -E
  113. }
  114. const s = (p) => {
  115. return p3[0] === p[0] && p3[1] === p[1]
  116. }
  117. return s(p1) || s(p2) || a(p2, p3) || d(p2, p3)
  118. }
  119. for (let i = 0; i <= 1; i += radio) {
  120. const arrowLocation = geometry.getCoordinateAt(i);
  121. if (extent.containsCoordinate(map.getView().calculateExtent(map.getSize()), arrowLocation)) {
  122. geometry.forEachSegment((start, end) => {
  123. if (!judgeIs(start, end, arrowLocation)) {
  124. return
  125. }
  126. let rotation = 0;
  127. const dx = end[0] - start[0];
  128. const dy = end[1] - start[1];
  129. rotation = Math.atan2(dy, dx);
  130. const pushStyle = (position) => {
  131. _style.push(new style.Style({
  132. geometry: new geom.Point(position),
  133. image: new style.RegularShape({
  134. stroke: new style.Stroke({
  135. color,
  136. width: 2,
  137. lineDash: [
  138. longRadius - (4 * (radius / 10)),
  139. longRadius + (5.5 * (radius / 10)),
  140. longRadius,
  141. 0
  142. ]
  143. }),
  144. radius: radius / Math.SQRT2,
  145. rotation: -rotation,
  146. angle: Math.PI / (180 / 90),
  147. points: 4
  148. })
  149. }));
  150. }
  151. if (flag) {
  152. const dx1 = end[0] - arrowLocation[0];
  153. const dy1 = end[1] - arrowLocation[1];
  154. const dx2 = arrowLocation[0] - start[0];
  155. const dy2 = arrowLocation[1] - start[1];
  156. if (dx1 != dx2 && dy1 != dy2) {
  157. if (Math.abs(dradio * dx1 * dy2 - dradio * dx2 * dy1) < 0.001) {
  158. pushStyle(arrowLocation)
  159. }
  160. }
  161. } else {
  162. if (Math.sqrt(Math.pow(start[0] - end[0], 2) + Math.pow(start[1] - end[1], 2)) > resolution * 100) {
  163. pushStyle([(start[0] + end[0]) / 2, (start[1] + end[1]) / 2])
  164. }
  165. }
  166. });
  167. }
  168. }
  169. const trackPointList = feature.get('trackPointList')
  170. const pList: any = []
  171. let trackPointListIndex = 0
  172. let lC = 0
  173. pList.push(trackPointList[0])
  174. if (flag) {
  175. geometry.forEachSegment((start, end) => {
  176. trackPointListIndex++
  177. const l = new geom.LineString([start,end])
  178. lC += l.getLength()
  179. if (extent.containsCoordinate(map.getView().calculateExtent(map.getSize()), end) && lC > 200 * resolution) {
  180. pList.push(trackPointList[trackPointListIndex])
  181. lC = 0
  182. }
  183. });
  184. } else {
  185. geometry.forEachSegment((start, end) => {
  186. trackPointListIndex++
  187. const l = new geom.LineString([start,end])
  188. if (extent.containsCoordinate(map.getView().calculateExtent(map.getSize()), end)) {
  189. pList.push(trackPointList[trackPointListIndex])
  190. }
  191. });
  192. }
  193. const arr: any = []
  194. const num = 100
  195. const step = Math.ceil(_style.length / num)
  196. for (let i = 0; i < _style.length; i += step) {
  197. arr.push(_style[i])
  198. }
  199. return callback([..._style, ...arr], pList)
  200. }
  201. const trackPointNormalStyle = (color) => {
  202. const _style: any = []
  203. _style.push(new style.Style({
  204. image: new style.Circle({
  205. radius: 3,
  206. fill: new style.Fill({
  207. color: color,
  208. }),
  209. stroke: new style.Stroke({
  210. color: '#ffffff',
  211. width: 1,
  212. }),
  213. })
  214. }))
  215. return _style
  216. }
  217. const warningStyle = (level) => {
  218. let icon = MapWarningIconBlue;
  219. switch (warningStyle) {
  220. case 1: {icon = MapWarningIconRed}
  221. break;
  222. case 2: {icon = MapWarningIconYellow}
  223. break;
  224. case 3: {icon = MapWarningIconBlue}
  225. break;
  226. }
  227. return new style.Style({
  228. image: new style.Icon({
  229. src: icon,
  230. scale: 2,
  231. anchor: [0.5, 0],
  232. anchorOrigin: 'bottom-left',
  233. anchorXUnits: 'fraction', // 定位点的单位
  234. anchorYUnits: 'fraction',
  235. }),
  236. })
  237. }
  238. export default {
  239. ShipNormalStyle,
  240. trackStyle,
  241. trackPointNormalStyle,
  242. warningStyle,
  243. }