@fleet-frontend/mower-maps 0.0.9 → 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (97) hide show
  1. package/dist/config/constants.d.ts +3 -25
  2. package/dist/config/constants.d.ts.map +1 -1
  3. package/dist/config/styles.d.ts +3 -2
  4. package/dist/config/styles.d.ts.map +1 -1
  5. package/dist/index.d.ts +2 -1
  6. package/dist/index.d.ts.map +1 -1
  7. package/dist/index.esm.js +1995 -1084
  8. package/dist/index.js +1993 -1082
  9. package/dist/processor/MapDataProcessor.d.ts +1 -1
  10. package/dist/processor/MapDataProcessor.d.ts.map +1 -1
  11. package/dist/processor/PathDataProcessor.d.ts +3 -18
  12. package/dist/processor/PathDataProcessor.d.ts.map +1 -1
  13. package/dist/processor/builder/AntennaDataBuilder.d.ts +2 -1
  14. package/dist/processor/builder/AntennaDataBuilder.d.ts.map +1 -1
  15. package/dist/processor/builder/BoundaryDataBuilder.d.ts +1 -1
  16. package/dist/processor/builder/BoundaryDataBuilder.d.ts.map +1 -1
  17. package/dist/processor/builder/ChannelDataBuilder.d.ts +1 -1
  18. package/dist/processor/builder/ChannelDataBuilder.d.ts.map +1 -1
  19. package/dist/processor/builder/ChargingPileDataBuilder.d.ts +1 -1
  20. package/dist/processor/builder/ChargingPileDataBuilder.d.ts.map +1 -1
  21. package/dist/processor/builder/ObstacleDataBuilder.d.ts +2 -1
  22. package/dist/processor/builder/ObstacleDataBuilder.d.ts.map +1 -1
  23. package/dist/processor/builder/PathDataBuilder.d.ts +6 -5
  24. package/dist/processor/builder/PathDataBuilder.d.ts.map +1 -1
  25. package/dist/processor/builder/PointDataBuilder.d.ts +2 -1
  26. package/dist/processor/builder/PointDataBuilder.d.ts.map +1 -1
  27. package/dist/processor/builder/SvgElementDataBuilder.d.ts +1 -1
  28. package/dist/processor/builder/SvgElementDataBuilder.d.ts.map +1 -1
  29. package/dist/processor/builder/VisionOffDataBuilder.d.ts +2 -1
  30. package/dist/processor/builder/VisionOffDataBuilder.d.ts.map +1 -1
  31. package/dist/processor/index.d.ts +2 -1
  32. package/dist/processor/index.d.ts.map +1 -1
  33. package/dist/render/AntennaManager.d.ts +7 -24
  34. package/dist/render/AntennaManager.d.ts.map +1 -1
  35. package/dist/render/BoundaryLabelsManager.d.ts +12 -20
  36. package/dist/render/BoundaryLabelsManager.d.ts.map +1 -1
  37. package/dist/render/ChargingPileManager.d.ts +10 -20
  38. package/dist/render/ChargingPileManager.d.ts.map +1 -1
  39. package/dist/render/MowerMapOverlay.d.ts +53 -8
  40. package/dist/render/MowerMapOverlay.d.ts.map +1 -1
  41. package/dist/render/MowerMapRenderer.d.ts.map +1 -1
  42. package/dist/render/{MowerPostionManager.d.ts → MowerPositionManager.d.ts} +16 -11
  43. package/dist/render/MowerPositionManager.d.ts.map +1 -0
  44. package/dist/render/SvgMapView.d.ts +15 -26
  45. package/dist/render/SvgMapView.d.ts.map +1 -1
  46. package/dist/render/layers/BoundaryBorderLayer.d.ts +11 -0
  47. package/dist/render/layers/BoundaryBorderLayer.d.ts.map +1 -1
  48. package/dist/render/layers/ChannelLayer.d.ts.map +1 -1
  49. package/dist/render/layers/DrawLayer.d.ts +1 -1
  50. package/dist/render/layers/DrawLayer.d.ts.map +1 -1
  51. package/dist/render/layers/PathLayer.d.ts +11 -2
  52. package/dist/render/layers/PathLayer.d.ts.map +1 -1
  53. package/dist/render/layers/SvgElementLayer.d.ts.map +1 -1
  54. package/dist/render/layers/index.d.ts +0 -1
  55. package/dist/render/layers/index.d.ts.map +1 -1
  56. package/dist/render/layers/types.d.ts +1 -36
  57. package/dist/render/layers/types.d.ts.map +1 -1
  58. package/dist/store/processMowingState.d.ts +4 -0
  59. package/dist/store/processMowingState.d.ts.map +1 -0
  60. package/dist/store/useSubBoundaryBorderStore.d.ts +9 -5
  61. package/dist/store/useSubBoundaryBorderStore.d.ts.map +1 -1
  62. package/dist/types/constants.d.ts +38 -0
  63. package/dist/types/constants.d.ts.map +1 -0
  64. package/dist/types/index.d.ts +6 -0
  65. package/dist/types/index.d.ts.map +1 -1
  66. package/dist/types/layers.d.ts +50 -0
  67. package/dist/types/layers.d.ts.map +1 -0
  68. package/dist/types/processor.d.ts +26 -0
  69. package/dist/types/processor.d.ts.map +1 -0
  70. package/dist/types/realTime.d.ts +27 -2
  71. package/dist/types/realTime.d.ts.map +1 -1
  72. package/dist/types/renderer.d.ts +28 -10
  73. package/dist/types/renderer.d.ts.map +1 -1
  74. package/dist/types/store.d.ts +22 -0
  75. package/dist/types/store.d.ts.map +1 -0
  76. package/dist/types/utils.d.ts +102 -0
  77. package/dist/types/utils.d.ts.map +1 -0
  78. package/dist/utils/boundaryUtils.d.ts +1 -29
  79. package/dist/utils/boundaryUtils.d.ts.map +1 -1
  80. package/dist/utils/common.d.ts +20 -0
  81. package/dist/utils/common.d.ts.map +1 -0
  82. package/dist/utils/coordinates.d.ts +28 -7
  83. package/dist/utils/coordinates.d.ts.map +1 -1
  84. package/dist/utils/handleRealTime.d.ts +20 -10
  85. package/dist/utils/handleRealTime.d.ts.map +1 -1
  86. package/dist/utils/mapBounds.d.ts +1 -10
  87. package/dist/utils/mapBounds.d.ts.map +1 -1
  88. package/dist/utils/math.d.ts +3 -10
  89. package/dist/utils/math.d.ts.map +1 -1
  90. package/dist/utils/mower.d.ts +5 -5
  91. package/dist/utils/mower.d.ts.map +1 -1
  92. package/dist/utils/pathSegments.d.ts +1 -15
  93. package/dist/utils/pathSegments.d.ts.map +1 -1
  94. package/dist/utils/unionFind.d.ts +49 -0
  95. package/dist/utils/unionFind.d.ts.map +1 -0
  96. package/package.json +1 -1
  97. package/dist/render/MowerPostionManager.d.ts.map +0 -1
package/dist/index.esm.js CHANGED
@@ -1,5 +1,135 @@
1
- import { jsx, jsxs } from 'react/jsx-runtime';
2
- import React, { forwardRef, useRef, useState, useMemo, useEffect, useImperativeHandle } from 'react';
1
+ import { jsxs, jsx } from 'react/jsx-runtime';
2
+ import React, { forwardRef, useState, useRef, useMemo, useCallback, useEffect, useImperativeHandle } from 'react';
3
+
4
+ /**
5
+ * 常量和枚举类型定义
6
+ */
7
+ /**
8
+ * 机器人状态枚举
9
+ */
10
+ var RobotStatus;
11
+ (function (RobotStatus) {
12
+ RobotStatus[RobotStatus["PARKED"] = 1] = "PARKED";
13
+ RobotStatus[RobotStatus["CHARGING"] = 2] = "CHARGING";
14
+ RobotStatus[RobotStatus["STANDBY"] = 3] = "STANDBY";
15
+ RobotStatus[RobotStatus["MOWING"] = 4] = "MOWING";
16
+ RobotStatus[RobotStatus["WORKING"] = 5] = "WORKING";
17
+ RobotStatus[RobotStatus["MAPPING"] = 6] = "MAPPING";
18
+ RobotStatus[RobotStatus["ERROR"] = 7] = "ERROR";
19
+ RobotStatus[RobotStatus["UPGRADING"] = 8] = "UPGRADING";
20
+ RobotStatus[RobotStatus["DISCONNECTED"] = 9] = "DISCONNECTED";
21
+ RobotStatus[RobotStatus["UNKNOWN"] = -1] = "UNKNOWN";
22
+ RobotStatus[RobotStatus["TASK_DELAY"] = 10] = "TASK_DELAY";
23
+ })(RobotStatus || (RobotStatus = {}));
24
+ /**
25
+ * RTK状态枚举
26
+ */
27
+ var RTK_STATE;
28
+ (function (RTK_STATE) {
29
+ RTK_STATE[RTK_STATE["LOW_RTK"] = 1] = "LOW_RTK";
30
+ RTK_STATE[RTK_STATE["MIDDLE_RTK"] = 2] = "MIDDLE_RTK";
31
+ RTK_STATE[RTK_STATE["HIGH_RTK"] = 3] = "HIGH_RTK";
32
+ RTK_STATE[RTK_STATE["NO_POSTURE"] = 10] = "NO_POSTURE";
33
+ RTK_STATE[RTK_STATE["OUT_OF_RANGE"] = 11] = "OUT_OF_RANGE";
34
+ RTK_STATE[RTK_STATE["OFF_LINE"] = 19] = "OFF_LINE";
35
+ })(RTK_STATE || (RTK_STATE = {}));
36
+ /**
37
+ * 实时数据类型枚举
38
+ */
39
+ var REAL_TIME_DATA_TYPE;
40
+ (function (REAL_TIME_DATA_TYPE) {
41
+ REAL_TIME_DATA_TYPE[REAL_TIME_DATA_TYPE["LOCATION"] = 1] = "LOCATION";
42
+ REAL_TIME_DATA_TYPE[REAL_TIME_DATA_TYPE["PROCESS"] = 2] = "PROCESS";
43
+ })(REAL_TIME_DATA_TYPE || (REAL_TIME_DATA_TYPE = {}));
44
+
45
+ /**
46
+ * 地图渲染相关常量配置
47
+ */
48
+ /**
49
+ * 缩放因子 - 将米转换为像素
50
+ * 与Python代码中的SVG比例一致
51
+ */
52
+ const SCALE_FACTOR = 50; // 50像素/米
53
+ /**
54
+ * 默认线宽设置
55
+ */
56
+ const DEFAULT_LINE_WIDTHS = {
57
+ OBSTACLE: 2,
58
+ CHARGING_PILE: 2,
59
+ CHANNEL: 2,
60
+ PATH: 20,
61
+ VISION_OFF_AREA: 2,
62
+ TIME_LIMIT_OBSTACLE: 1,
63
+ };
64
+ /**
65
+ * 默认透明度设置
66
+ */
67
+ const DEFAULT_OPACITIES = {
68
+ FULL: 1.0,
69
+ HIGH: 0.7,
70
+ MEDIUM: 0.6,
71
+ DOODLE: 0.8};
72
+ /**
73
+ * 默认半径设置
74
+ */
75
+ const DEFAULT_RADII = {
76
+ CHARGING_PILE: 12};
77
+ /**
78
+ * 图层等级
79
+ */
80
+ const LAYER_LEVELS = {
81
+ BOUNDARY: 2,
82
+ BOUNDARY_BORDER: 4};
83
+ /**
84
+ * 图层默认id
85
+ */
86
+ const LAYER_DEFAULT_TYPE = {
87
+ CHANNEL: 'channel',
88
+ BOUNDARY: 'boundary',
89
+ PATH: 'path',
90
+ BOUNDARY_BORDER: 'boundary_border',
91
+ OBSTACLE: 'obstacle',
92
+ CHARGING_PILE: 'charging_pile',
93
+ POINT: 'point',
94
+ SVG: 'svg',
95
+ VISION_OFF_AREA: 'vision_off_area',
96
+ ANTENNA: 'antenna',
97
+ };
98
+ const ISOLATED_BOUNDARY_SVG = `<svg width="24" height="24" viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg">
99
+ <g opacity="0.6">
100
+ <rect width="24" height="24" rx="12" fill="#1E1E1F" fill-opacity="0.5"/>
101
+ <path d="M8.20573 12.4961C6.739 11.1707 4.85775 11.2284 3.60938 11.1707C6.17744 13.0156 6.05584 15.7887 5.67404 16.9446H14.327C13.837 13.1673 15.5321 10.1289 16.4408 9.08187C15.0342 9.46118 13.5794 10.8303 13.0278 11.4674C12.8102 8.94572 13.992 5.97489 14.61 4.80469C12.5235 5.92214 11.0501 8.26056 10.4938 9.56261C9.22803 7.5947 6.83894 7.17806 5.883 6.9432C8.20573 10.2373 8.00039 11.1707 8.20573 12.4961Z" fill="white"/>
102
+ <mask id="path-3-outside-1_9822_43516" maskUnits="userSpaceOnUse" x="13.2344" y="7.20545" width="12.0208" height="12.0208" fill="black">
103
+ <rect fill="white" x="13.2344" y="7.20545" width="12.0208" height="12.0208"/>
104
+ <path d="M20.6852 11.1208C19.9042 10.3398 18.6378 10.3398 17.8568 11.1208L17.2733 11.7043C16.6707 12.3069 16.5358 13.1973 16.8631 13.9319L17.5916 13.2034C17.5657 12.9181 17.6619 12.6236 17.8803 12.4052L18.5674 11.7181C18.9579 11.3276 19.5911 11.3276 19.9816 11.7181L20.4056 12.1421C20.7959 12.5326 20.796 13.1658 20.4056 13.5563L19.7185 14.2434C19.5128 14.449 19.2393 14.5457 18.9699 14.5348L18.2235 15.2812C18.9538 15.596 19.8325 15.4565 20.429 14.86L21.0125 14.2765C21.7935 13.4955 21.7936 12.2291 21.0125 11.4481L20.6852 11.1208Z"/>
105
+ </mask>
106
+ <path d="M20.6852 11.1208C19.9042 10.3398 18.6378 10.3398 17.8568 11.1208L17.2733 11.7043C16.6707 12.3069 16.5358 13.1973 16.8631 13.9319L17.5916 13.2034C17.5657 12.9181 17.6619 12.6236 17.8803 12.4052L18.5674 11.7181C18.9579 11.3276 19.5911 11.3276 19.9816 11.7181L20.4056 12.1421C20.7959 12.5326 20.796 13.1658 20.4056 13.5563L19.7185 14.2434C19.5128 14.449 19.2393 14.5457 18.9699 14.5348L18.2235 15.2812C18.9538 15.596 19.8325 15.4565 20.429 14.86L21.0125 14.2765C21.7935 13.4955 21.7936 12.2291 21.0125 11.4481L20.6852 11.1208Z" fill="white"/>
107
+ <path d="M20.6852 11.1208L21.7458 10.0601L21.7459 10.0601L20.6852 11.1208ZM17.8568 11.1208L16.7961 10.0601L16.7962 10.0601L17.8568 11.1208ZM17.2733 11.7043L16.2126 10.6437L16.2126 10.6436L17.2733 11.7043ZM16.8631 13.9319L17.9238 14.9926L16.381 16.5354L15.493 14.5425L16.8631 13.9319ZM17.5916 13.2034L19.0855 13.0678L19.149 13.7674L18.6523 14.2641L17.5916 13.2034ZM20.4056 12.1421L21.4662 11.0814L21.4665 11.0817L20.4056 12.1421ZM18.9699 14.5348L17.9093 13.4741L18.3741 13.0093L19.0309 13.036L18.9699 14.5348ZM18.2235 15.2812L17.6298 16.6587L15.5997 15.7837L17.1628 14.2206L18.2235 15.2812ZM20.429 14.86L21.4897 15.9207L21.4897 15.9207L20.429 14.86ZM21.0125 14.2765L22.0732 15.3371L22.0732 15.3372L21.0125 14.2765ZM20.6852 11.1208L19.6246 12.1815C19.4294 11.9862 19.1127 11.9863 18.9174 12.1815L17.8568 11.1208L16.7962 10.0601C18.163 8.69337 20.379 8.6934 21.7458 10.0601L20.6852 11.1208ZM17.8568 11.1208L18.9175 12.1814L18.334 12.7649L17.2733 11.7043L16.2126 10.6436L16.7961 10.0601L17.8568 11.1208ZM17.2733 11.7043L18.334 12.7649C18.1877 12.9112 18.1488 13.1318 18.2333 13.3214L16.8631 13.9319L15.493 14.5425C14.9228 13.2629 15.1537 11.7026 16.2126 10.6437L17.2733 11.7043ZM16.8631 13.9319L15.8025 12.8713L16.531 12.1428L17.5916 13.2034L18.6523 14.2641L17.9238 14.9926L16.8631 13.9319ZM17.5916 13.2034L16.0978 13.339C16.0334 12.6302 16.2727 11.8914 16.8196 11.3445L17.8803 12.4052L18.9409 13.4658C19.051 13.3558 19.098 13.206 19.0855 13.0678L17.5916 13.2034ZM17.8803 12.4052L16.8196 11.3445L17.5067 10.6574L18.5674 11.7181L19.628 12.7788L18.9409 13.4658L17.8803 12.4052ZM18.5674 11.7181L17.5067 10.6574C18.483 9.68112 20.0659 9.68112 21.0422 10.6574L19.9816 11.7181L18.9209 12.7788C19.1162 12.974 19.4328 12.974 19.628 12.7788L18.5674 11.7181ZM19.9816 11.7181L21.0422 10.6574L21.4662 11.0814L20.4056 12.1421L19.3449 13.2027L18.9209 12.7788L19.9816 11.7181ZM20.4056 12.1421L21.4665 11.0817C22.4419 12.0577 22.4428 13.6404 21.4662 14.617L20.4056 13.5563L19.3449 12.4956C19.1493 12.6913 19.1498 13.0076 19.3446 13.2024L20.4056 12.1421ZM20.4056 13.5563L21.4662 14.617L20.7791 15.304L19.7185 14.2434L18.6578 13.1827L19.3449 12.4956L20.4056 13.5563ZM19.7185 14.2434L20.7791 15.304C20.2628 15.8204 19.5762 16.0607 18.909 16.0335L18.9699 14.5348L19.0309 13.036C18.9025 13.0308 18.7629 13.0777 18.6578 13.1827L19.7185 14.2434ZM18.9699 14.5348L20.0306 15.5954L19.2841 16.3419L18.2235 15.2812L17.1628 14.2206L17.9093 13.4741L18.9699 14.5348ZM18.2235 15.2812L18.8172 13.9037C19.0042 13.9843 19.2223 13.9455 19.3684 13.7993L20.429 14.86L21.4897 15.9207C20.4427 16.9676 18.9034 17.2077 17.6298 16.6587L18.2235 15.2812ZM20.429 14.86L19.3684 13.7994L19.9519 13.2159L21.0125 14.2765L22.0732 15.3372L21.4897 15.9207L20.429 14.86ZM21.0125 14.2765L19.9518 13.2159C20.1471 13.0206 20.1471 12.704 19.9519 12.5088L21.0125 11.4481L22.0732 10.3874C23.4401 11.7543 23.4399 13.9703 22.0732 15.3371L21.0125 14.2765ZM21.0125 11.4481L19.9519 12.5088L19.6246 12.1814L20.6852 11.1208L21.7459 10.0601L22.0732 10.3874L21.0125 11.4481Z" fill="#8E8E8F" mask="url(#path-3-outside-1_9822_43516)"/>
108
+ <mask id="path-5-outside-2_9822_43516" maskUnits="userSpaceOnUse" x="9.22265" y="10.388" width="12.7279" height="12.7279" fill="black">
109
+ <rect fill="white" x="9.22265" y="10.388" width="12.7279" height="12.7279"/>
110
+ <path d="M16.1708 14.5077C15.4353 14.1778 14.5422 14.3133 13.9383 14.9172L13.3548 15.5007C12.574 16.2818 12.5738 17.5481 13.3548 18.3291L13.6821 18.6564C14.4631 19.4374 15.7295 19.4373 16.5105 18.6564L17.094 18.0729C17.6895 17.4775 17.8299 16.6003 17.5173 15.8708L16.7743 16.6138C16.7828 16.8803 16.6863 17.1494 16.4829 17.3527L15.7952 18.0405C15.4046 18.4309 14.7714 18.431 14.3809 18.0405L13.957 17.6165C13.5667 17.226 13.5666 16.5927 13.957 16.2023L14.6447 15.5145C14.8654 15.2939 15.1639 15.1979 15.452 15.2266L16.1708 14.5077Z"/>
111
+ </mask>
112
+ <path d="M16.1708 14.5077C15.4353 14.1778 14.5422 14.3133 13.9383 14.9172L13.3548 15.5007C12.574 16.2818 12.5738 17.5481 13.3548 18.3291L13.6821 18.6564C14.4631 19.4374 15.7295 19.4373 16.5105 18.6564L17.094 18.0729C17.6895 17.4775 17.8299 16.6003 17.5173 15.8708L16.7743 16.6138C16.7828 16.8803 16.6863 17.1494 16.4829 17.3527L15.7952 18.0405C15.4046 18.4309 14.7714 18.431 14.3809 18.0405L13.957 17.6165C13.5667 17.226 13.5666 16.5927 13.957 16.2023L14.6447 15.5145C14.8654 15.2939 15.1639 15.1979 15.452 15.2266L16.1708 14.5077Z" fill="white"/>
113
+ <path d="M16.1708 14.5077L16.7847 13.1391L18.7701 14.0297L17.2315 15.5684L16.1708 14.5077ZM13.3548 15.5007L12.294 14.4402L12.2941 14.44L13.3548 15.5007ZM16.5105 18.6564L17.5712 19.7171L17.5711 19.7172L16.5105 18.6564ZM17.5173 15.8708L16.4567 14.8102L18.0236 13.2433L18.8962 15.2802L17.5173 15.8708ZM16.7743 16.6138L15.2751 16.6613L15.2545 16.0123L15.7137 15.5532L16.7743 16.6138ZM15.7952 18.0405L16.8558 19.1011L16.8556 19.1013L15.7952 18.0405ZM13.957 17.6165L12.8963 18.6772L12.8959 18.6767L13.957 17.6165ZM13.957 16.2023L12.8961 15.1418L12.8963 15.1416L13.957 16.2023ZM15.452 15.2266L16.5126 16.2872L16.0102 16.7896L15.3032 16.7192L15.452 15.2266ZM16.1708 14.5077L15.5569 15.8763C15.3688 15.792 15.1469 15.8299 14.999 15.9779L13.9383 14.9172L12.8776 13.8565C13.9375 12.7967 15.5018 12.5636 16.7847 13.1391L16.1708 14.5077ZM13.9383 14.9172L14.999 15.9779L14.4155 16.5614L13.3548 15.5007L12.2941 14.44L12.8776 13.8565L13.9383 14.9172ZM13.3548 15.5007L14.4156 16.5612C14.2202 16.7567 14.2204 17.0734 14.4155 17.2685L13.3548 18.3291L12.2941 19.3898C10.9272 18.0229 10.9277 15.8069 12.294 14.4402L13.3548 15.5007ZM13.3548 18.3291L14.4155 17.2685L14.7428 17.5958L13.6821 18.6564L12.6215 19.7171L12.2941 19.3898L13.3548 18.3291ZM13.6821 18.6564L14.7428 17.5958C14.9378 17.7908 15.2546 17.791 15.45 17.5956L16.5105 18.6564L17.5711 19.7172C16.2044 21.0836 13.9884 21.084 12.6215 19.7171L13.6821 18.6564ZM16.5105 18.6564L15.4499 17.5958L16.0334 17.0123L17.094 18.0729L18.1547 19.1336L17.5712 19.7171L16.5105 18.6564ZM17.094 18.0729L16.0334 17.0123C16.1795 16.8662 16.2185 16.6481 16.1385 16.4615L17.5173 15.8708L18.8962 15.2802C19.4413 16.5526 19.1996 18.0887 18.1547 19.1336L17.094 18.0729ZM17.5173 15.8708L18.578 16.9315L17.835 17.6745L16.7743 16.6138L15.7137 15.5532L16.4567 14.8102L17.5173 15.8708ZM16.7743 16.6138L18.2736 16.5664C18.2945 17.226 18.0544 17.9026 17.5436 18.4134L16.4829 17.3527L15.4223 16.2921C15.3182 16.3961 15.2711 16.5346 15.2751 16.6613L16.7743 16.6138ZM16.4829 17.3527L17.5436 18.4134L16.8558 19.1011L15.7952 18.0405L14.7345 16.9798L15.4223 16.2921L16.4829 17.3527ZM15.7952 18.0405L16.8556 19.1013C15.8795 20.0771 14.2967 20.0776 13.3203 19.1011L14.3809 18.0405L15.4416 16.9798C15.2461 16.7844 14.9297 16.7847 14.7347 16.9797L15.7952 18.0405ZM14.3809 18.0405L13.3203 19.1011L12.8963 18.6772L13.957 17.6165L15.0176 16.5558L15.4416 16.9798L14.3809 18.0405ZM13.957 17.6165L12.8959 18.6767C11.9207 17.7008 11.92 16.1182 12.8961 15.1418L13.957 16.2023L15.0178 17.2628C15.2133 17.0672 15.2128 16.7512 15.018 16.5563L13.957 17.6165ZM13.957 16.2023L12.8963 15.1416L13.5841 14.4539L14.6447 15.5145L15.7054 16.5752L15.0176 17.2629L13.957 16.2023ZM14.6447 15.5145L13.5841 14.4539C14.1364 13.9015 14.8848 13.6626 15.6007 13.734L15.452 15.2266L15.3032 16.7192C15.443 16.7331 15.5943 16.6862 15.7054 16.5752L14.6447 15.5145ZM15.452 15.2266L14.3913 14.1659L15.1101 13.4471L16.1708 14.5077L17.2315 15.5684L16.5126 16.2872L15.452 15.2266Z" fill="#8E8E8F" mask="url(#path-5-outside-2_9822_43516)"/>
114
+ <path d="M20.6852 11.1208C19.9042 10.3398 18.6378 10.3398 17.8568 11.1208L17.2733 11.7043C16.6707 12.3069 16.5358 13.1973 16.8631 13.9319L17.5916 13.2034C17.5657 12.9181 17.6619 12.6236 17.8803 12.4052L18.5674 11.7181C18.9579 11.3276 19.5911 11.3276 19.9816 11.7181L20.4056 12.1421C20.7959 12.5326 20.796 13.1658 20.4056 13.5563L19.7185 14.2434C19.5128 14.449 19.2393 14.5457 18.9699 14.5348L18.2235 15.2812C18.9538 15.596 19.8325 15.4565 20.429 14.86L21.0125 14.2765C21.7935 13.4955 21.7936 12.2291 21.0125 11.4481L20.6852 11.1208Z" fill="white"/>
115
+ <rect width="2.59942" height="2.97264" rx="1" transform="matrix(-0.707107 -0.707107 -0.707107 0.707107 21.1133 12.8486)" fill="#8E8E8F"/>
116
+ <rect width="1.91578" height="4.24403" transform="matrix(-0.707107 -0.707107 -0.707107 0.707107 19.6992 13.8066)" fill="#8E8E8F"/>
117
+ <rect width="2.59942" height="2.97264" rx="1" transform="matrix(-0.707107 -0.707107 -0.707107 0.707107 17.3125 16.6455)" fill="#8E8E8F"/>
118
+ </g>
119
+ </svg>`;
120
+ /**
121
+ * 遍历割草任务,下述四个字段可以在路径中唯一确定遍历的位置(当前区域、当前块、当前行、在当前行上的路程)
122
+ */
123
+ const ACTION_BLOCK_COVER = 5;
124
+ /**
125
+ * 遍历割草块转移任务(在同一区域中的一个块转移到下一个块),下述四个字段可以在路径中唯一确定转移的位置(当前区域、前置块、当前转移路径线序号、在当前线上的路程)
126
+ */
127
+ const ACTION_BLOCK_TRANSFER = 6;
128
+ /**
129
+ * 边界割草任务(割草任务内部的巡边任务)
130
+ */
131
+ const ACTION_BOUNDARY_TASK = 8;
132
+ const SVG_MAP_VIEW_ID = 'fleet-maps-svg-map-view';
3
133
 
4
134
  /**
5
135
  * SVG基础MapView
@@ -14,7 +144,6 @@ class SvgMapView {
14
144
  this.lineScale = 1; // 线条缩放系数
15
145
  // 状态标志
16
146
  this.destroyed = false;
17
- this.showScale = false;
18
147
  // 渲染系统 - 移除节流以确保用户操作的实时响应
19
148
  // 拖动功能
20
149
  this.isDragging = false;
@@ -57,6 +186,7 @@ class SvgMapView {
57
186
  this.svg.setAttribute('shape-rendering', 'geometricPrecision');
58
187
  this.svg.setAttribute('text-rendering', 'geometricPrecision');
59
188
  this.svg.setAttribute('image-rendering', 'optimizeQuality');
189
+ this.svg.setAttribute('id', SVG_MAP_VIEW_ID);
60
190
  }
61
191
  /**
62
192
  * 创建SVG组元素
@@ -79,7 +209,9 @@ class SvgMapView {
79
209
  return;
80
210
  this.layers.push(...layers);
81
211
  this.layers.sort((a, b) => a.getLevel() - b.getLevel());
82
- this.refresh();
212
+ }
213
+ getLayer(type) {
214
+ return this.layers.find((layer) => layer.getType() === type) || null;
83
215
  }
84
216
  /**
85
217
  * 添加图层
@@ -89,14 +221,12 @@ class SvgMapView {
89
221
  return;
90
222
  this.layers.push(layer);
91
223
  this.layers.sort((a, b) => a.getLevel() - b.getLevel());
92
- this.refresh();
93
224
  }
94
225
  /**
95
226
  * 移除图层
96
227
  */
97
228
  removeLayer(layer) {
98
229
  const index = this.layers.indexOf(layer);
99
- console.log('removeLayer----->', index);
100
230
  if (index !== -1) {
101
231
  this.layers.splice(index, 1);
102
232
  this.refresh();
@@ -107,15 +237,18 @@ class SvgMapView {
107
237
  * @param type 图层类型
108
238
  */
109
239
  removeLayerByType(type) {
110
- this.layers = this.layers.filter((layer) => layer.getType() !== type);
111
- this.refresh();
240
+ const layer = this.layers.find((layer) => layer.getType() === type);
241
+ if (layer) {
242
+ this.clearLayersGroup(layer);
243
+ this.layers = this.layers.filter((cLayer) => cLayer.getType() !== type);
244
+ }
112
245
  }
113
246
  // ==================== 变换系统 ====================
114
247
  /**
115
248
  * 设置自适应视图变换 - 让SVG刚好包裹住图形
116
249
  */
117
250
  fitToView(bounds) {
118
- const padding = 10; // 添加一些边距以避免内容贴边
251
+ const padding = 20; // 添加一些边距以避免内容贴边
119
252
  const boundWidth = bounds.maxX - bounds.minX;
120
253
  const boundHeight = bounds.maxY - bounds.minY;
121
254
  // 防止宽高为0的情况
@@ -138,8 +271,8 @@ class SvgMapView {
138
271
  this.viewBox = {
139
272
  x: bounds.minX - padding,
140
273
  y: bounds.minY - padding,
141
- width: boundWidth + padding,
142
- height: boundHeight + padding,
274
+ width: boundWidth + padding * 2,
275
+ height: boundHeight + padding * 2,
143
276
  };
144
277
  // 根据宽高比选择合适的preserveAspectRatio设置
145
278
  if (Math.abs(contentAspectRatio - containerAspectRatio) < 0.01) {
@@ -151,42 +284,6 @@ class SvgMapView {
151
284
  this.svg.setAttribute('preserveAspectRatio', 'xMidYMid meet');
152
285
  }
153
286
  this.updateViewBox();
154
- this.refresh();
155
- }
156
- /**
157
- * 重置变换
158
- */
159
- resetTransform() {
160
- this.scale = 1;
161
- // 重置viewBox到默认状态
162
- const containerRect = this.container.getBoundingClientRect();
163
- this.viewBox = {
164
- x: 0,
165
- y: 0,
166
- width: containerRect.width,
167
- height: containerRect.height,
168
- };
169
- this.updateViewBox();
170
- this.refresh();
171
- }
172
- /**
173
- * 设置缩放级别
174
- */
175
- setZoom(zoomLevel) {
176
- if (zoomLevel <= 0)
177
- return;
178
- const oldScale = this.scale;
179
- this.scale = zoomLevel;
180
- // 调整viewBox以实现缩放
181
- const scaleFactor = oldScale / this.scale;
182
- const centerX = this.viewBox.x + this.viewBox.width / 2;
183
- const centerY = this.viewBox.y + this.viewBox.height / 2;
184
- this.viewBox.width *= scaleFactor;
185
- this.viewBox.height *= scaleFactor;
186
- this.viewBox.x = centerX - this.viewBox.width / 2;
187
- this.viewBox.y = centerY - this.viewBox.height / 2;
188
- this.updateViewBox();
189
- this.refresh();
190
287
  }
191
288
  /**
192
289
  * 获取当前缩放级别
@@ -194,21 +291,21 @@ class SvgMapView {
194
291
  getZoom() {
195
292
  return this.scale;
196
293
  }
197
- /**
198
- * 设置线条缩放系数
199
- */
200
- setLineScale(lineScale) {
201
- if (lineScale <= 0)
202
- return;
203
- this.lineScale = lineScale;
204
- this.refresh();
205
- }
206
294
  /**
207
295
  * 获取当前线条缩放系数
208
296
  */
209
297
  getLineScale() {
210
298
  return this.lineScale;
211
299
  }
300
+ /**
301
+ * 绘制特定的图层
302
+ */
303
+ renderLayer(type) {
304
+ const layer = this.layers.find((layer) => layer.getType() === type);
305
+ // 清空图层组
306
+ this.clearLayersGroup(layer);
307
+ this.onDrawLayers(type);
308
+ }
212
309
  // ==================== 渲染系统 ====================
213
310
  /**
214
311
  * 主渲染方法
@@ -219,25 +316,112 @@ class SvgMapView {
219
316
  // 绘制所有图层
220
317
  this.onDrawLayers();
221
318
  }
319
+ /**
320
+ * 获取图层id
321
+ */
322
+ getLayerId(layer) {
323
+ return `layer-${layer.getType()}-${layer.getLevel()}`;
324
+ }
222
325
  /**
223
326
  * 清空图层组
224
327
  */
225
- clearLayersGroup() {
226
- while (this.layersGroup.firstChild) {
227
- this.layersGroup.removeChild(this.layersGroup.firstChild);
328
+ clearLayersGroup(layer) {
329
+ if (layer) {
330
+ const layerId = this.getLayerId(layer);
331
+ const layerGroup = this.layersGroup.querySelector(`#${layerId}`);
332
+ if (layerGroup) {
333
+ this.layersGroup.removeChild(layerGroup);
334
+ }
335
+ }
336
+ else {
337
+ while (this.layersGroup.firstChild) {
338
+ this.layersGroup.removeChild(this.layersGroup.firstChild);
339
+ }
228
340
  }
229
341
  }
230
342
  /**
231
- * 绘制所有图层
343
+ * 获取图层的下一个兄弟元素
344
+ * 根据图层的level来获取
232
345
  */
233
- onDrawLayers() {
234
- for (const layer of this.layers) {
235
- if (layer.isVisible()) {
236
- const layerGroup = this.createSVGGroup(`layer-${layer.getLevel()}`);
346
+ getNextSibling(layer) {
347
+ const nextLayer = this.layers.find((cLayer) => cLayer.getLevel() > layer.getLevel());
348
+ const id = `layer-${nextLayer?.getType()}-${nextLayer?.getLevel()}`;
349
+ const nextSibling = this.layersGroup.querySelector(`#${id}`);
350
+ return nextSibling;
351
+ }
352
+ /**
353
+ * 绘制图层,不传参数则默认绘制所有图层
354
+ */
355
+ onDrawLayers(type) {
356
+ if (type) {
357
+ const layer = this.layers.find((layer) => layer.getType() === type);
358
+ if (layer) {
359
+ const layerGroup = this.createSVGGroup(`layer-${layer.getType()}-${layer.getLevel()}`);
237
360
  layer.drawSVG(layerGroup, this.scale, this.lineScale);
238
- this.layersGroup.appendChild(layerGroup);
361
+ const nextSibling = this.getNextSibling(layer);
362
+ if (nextSibling) {
363
+ this.layersGroup.insertBefore(layerGroup, nextSibling);
364
+ }
365
+ else {
366
+ this.layersGroup.appendChild(layerGroup);
367
+ }
239
368
  }
240
369
  }
370
+ else {
371
+ for (const layer of this.layers) {
372
+ if (layer.isVisible()) {
373
+ const layerGroup = this.createSVGGroup(`layer-${layer.getType()}-${layer.getLevel()}`);
374
+ layer.drawSVG(layerGroup, this.scale, this.lineScale);
375
+ this.layersGroup.appendChild(layerGroup);
376
+ }
377
+ }
378
+ }
379
+ // if (type) {
380
+ // const layer = this.layers.find((layer) => layer.getType() === type);
381
+ // const layerId = this.getLayerId(layer);
382
+ // // 记录原始位置信息
383
+ // const layerGroup = this.layersGroup.querySelector(`#${layerId}`);
384
+ // let nextSibling: Node | null = null;
385
+ // if (layerGroup) {
386
+ // // 记录被删除元素的下一个兄弟元素,用于确定插入位置
387
+ // nextSibling = layerGroup.nextSibling;
388
+ // // 删除旧的图层组
389
+ // this.layersGroup.removeChild(layerGroup);
390
+ // // 从layerId解析出图层类型和层级
391
+ // // layerId格式: layer-${type}-${level}
392
+ // const layerIdParts = layerId.split('-');
393
+ // if (layerIdParts.length >= 3) {
394
+ // const layerType = layerIdParts.slice(1, -1).join('-'); // 处理类型名中可能包含连字符的情况
395
+ // const layerLevel = parseInt(layerIdParts[layerIdParts.length - 1]);
396
+ // // 查找对应的图层对象
397
+ // const targetLayer = this.layers.find(layer =>
398
+ // layer.getType() === layerType && layer.getLevel() === layerLevel
399
+ // );
400
+ // if (targetLayer && targetLayer.isVisible()) {
401
+ // // 创建新的图层组
402
+ // const newLayerGroup = this.createSVGGroup(layerId);
403
+ // // 重新绘制图层
404
+ // targetLayer.drawSVG(newLayerGroup, this.scale, this.lineScale);
405
+ // // 在原始位置插入新元素
406
+ // if (nextSibling) {
407
+ // this.layersGroup.insertBefore(newLayerGroup, nextSibling);
408
+ // } else {
409
+ // // 如果没有下一个兄弟元素,说明原来是最后一个,直接appendChild
410
+ // this.layersGroup.appendChild(newLayerGroup);
411
+ // }
412
+ // }
413
+ // }
414
+ // }
415
+ // } else {
416
+ // // 重绘所有图层
417
+ // for (const layer of this.layers) {
418
+ // if (layer.isVisible()) {
419
+ // const layerGroup = this.createSVGGroup(`layer-${layer.getType()}-${layer.getLevel()}`);
420
+ // layer.drawSVG(layerGroup, this.scale, this.lineScale);
421
+ // this.layersGroup.appendChild(layerGroup);
422
+ // }
423
+ // }
424
+ // }
241
425
  }
242
426
  /**
243
427
  * 刷新渲染
@@ -247,12 +431,6 @@ class SvgMapView {
247
431
  return;
248
432
  this.render();
249
433
  }
250
- /**
251
- * 重新初始化SVG(用于容器大小变化)
252
- */
253
- reinitializeSVG() {
254
- this.refresh();
255
- }
256
434
  // ==================== 拖拽功能 ====================
257
435
  /**
258
436
  * 设置拖拽事件处理器
@@ -377,22 +555,6 @@ class SvgMapView {
377
555
  return null;
378
556
  }
379
557
  }
380
- /**
381
- * 自动适配viewBox到实际内容
382
- */
383
- autoFitToContent() {
384
- if (this.destroyed || this.layers.length === 0)
385
- return;
386
- const bounds = this.getLayersGroupBounds();
387
- if (bounds) {
388
- this.fitToView({
389
- minX: bounds.x,
390
- minY: bounds.y,
391
- maxX: bounds.x + bounds.width,
392
- maxY: bounds.y + bounds.height,
393
- });
394
- }
395
- }
396
558
  /**
397
559
  * 获取ViewBox信息
398
560
  */
@@ -413,13 +575,6 @@ class SvgMapView {
413
575
  * 诊断SVG尺寸信息
414
576
  */
415
577
  diagnosticSizeInfo() { }
416
- /**
417
- * 设置是否显示比例尺
418
- */
419
- setShowScale(show) {
420
- this.showScale = show;
421
- this.refresh();
422
- }
423
578
  /**
424
579
  * 获取SVG元素
425
580
  */
@@ -621,8 +776,6 @@ const create = (createState) => createState ? createImpl(createState) : createIm
621
776
 
622
777
  const useSubBoundaryBorderStore = create((set, get) => ({
623
778
  subBoundaryBorder: {},
624
- // 覆盖所有数据
625
- setSubBoundaryBorder: (subBoundaryBorder) => set({ subBoundaryBorder }),
626
779
  // 追加单个数据
627
780
  addSubBoundaryBorder: (key, element) => set((state) => ({
628
781
  subBoundaryBorder: {
@@ -630,134 +783,28 @@ const useSubBoundaryBorderStore = create((set, get) => ({
630
783
  [key]: element,
631
784
  },
632
785
  })),
633
- // 追加多个数据
634
- addMultipleSubBoundaryBorders: (borders) => set((state) => ({
635
- subBoundaryBorder: {
636
- ...state.subBoundaryBorder,
637
- ...borders,
638
- },
639
- })),
640
786
  // 清空所有数据
641
787
  clearSubBoundaryBorder: () => set({ subBoundaryBorder: {} }),
788
+ // 障碍物
789
+ obstacles: {},
790
+ addObstacles: (key, element) => set((state) => ({
791
+ obstacles: {
792
+ ...state.obstacles,
793
+ [key]: element,
794
+ },
795
+ })),
796
+ clearObstacles: () => set({ obstacles: {} }),
797
+ // svg数据
798
+ svgElements: {},
799
+ addSvgElements: (key, element) => set((state) => ({
800
+ svgElements: {
801
+ ...state.svgElements,
802
+ [key]: element,
803
+ },
804
+ })),
805
+ clearSvgElements: () => set({ svgElements: {} }),
642
806
  }));
643
807
 
644
- /**
645
- * 地图渲染相关常量配置
646
- */
647
- /**
648
- * 缩放因子 - 将米转换为像素
649
- * 与Python代码中的SVG比例一致
650
- */
651
- const SCALE_FACTOR = 50; // 50像素/米
652
- /**
653
- * 默认线宽设置
654
- */
655
- const DEFAULT_LINE_WIDTHS = {
656
- OBSTACLE: 2,
657
- CHARGING_PILE: 2,
658
- CHANNEL: 2,
659
- PATH: 20,
660
- VISION_OFF_AREA: 2,
661
- TIME_LIMIT_OBSTACLE: 1,
662
- };
663
- /**
664
- * 默认透明度设置
665
- */
666
- const DEFAULT_OPACITIES = {
667
- FULL: 1.0,
668
- HIGH: 0.7,
669
- LOW: 0.4};
670
- /**
671
- * 默认半径设置
672
- */
673
- const DEFAULT_RADII = {
674
- CHARGING_PILE: 12};
675
- /**
676
- * 图层等级
677
- */
678
- const LAYER_LEVELS = {
679
- BOUNDARY: 2,
680
- BOUNDARY_BORDER: 4};
681
- /**
682
- * 图层默认id
683
- */
684
- const LAYER_DEFAULT_TYPE = {
685
- CHANNEL: 'channel',
686
- BOUNDARY: 'boundary',
687
- PATH: 'path',
688
- BOUNDARY_BORDER: 'boundary_border',
689
- OBSTACLE: 'obstacle',
690
- CHARGING_PILE: 'charging_pile',
691
- POINT: 'point',
692
- SVG: 'svg',
693
- VISION_OFF_AREA: 'vision_off_area',
694
- ANTENNA: 'antenna',
695
- };
696
- var RobotStatus;
697
- (function (RobotStatus) {
698
- RobotStatus[RobotStatus["PARKED"] = 1] = "PARKED";
699
- RobotStatus[RobotStatus["CHARGING"] = 2] = "CHARGING";
700
- RobotStatus[RobotStatus["STANDBY"] = 3] = "STANDBY";
701
- RobotStatus[RobotStatus["MOWING"] = 4] = "MOWING";
702
- RobotStatus[RobotStatus["WORKING"] = 5] = "WORKING";
703
- RobotStatus[RobotStatus["MAPPING"] = 6] = "MAPPING";
704
- RobotStatus[RobotStatus["ERROR"] = 7] = "ERROR";
705
- RobotStatus[RobotStatus["UPGRADING"] = 8] = "UPGRADING";
706
- RobotStatus[RobotStatus["DISCONNECTED"] = 9] = "DISCONNECTED";
707
- RobotStatus[RobotStatus["UNKNOWN"] = -1] = "UNKNOWN";
708
- RobotStatus[RobotStatus["TASK_DELAY"] = 10] = "TASK_DELAY";
709
- // WAITING = 'Waiting',
710
- })(RobotStatus || (RobotStatus = {}));
711
- // RTK状态
712
- var RTK_STATE;
713
- (function (RTK_STATE) {
714
- RTK_STATE[RTK_STATE["LOW_RTK"] = 1] = "LOW_RTK";
715
- RTK_STATE[RTK_STATE["MIDDLE_RTK"] = 2] = "MIDDLE_RTK";
716
- RTK_STATE[RTK_STATE["HIGH_RTK"] = 3] = "HIGH_RTK";
717
- RTK_STATE[RTK_STATE["NO_POSTURE"] = 10] = "NO_POSTURE";
718
- RTK_STATE[RTK_STATE["OUT_OF_RANGE"] = 11] = "OUT_OF_RANGE";
719
- RTK_STATE[RTK_STATE["OFF_LINE"] = 19] = "OFF_LINE";
720
- })(RTK_STATE || (RTK_STATE = {}));
721
- var REAL_TIME_DATA_TYPE;
722
- (function (REAL_TIME_DATA_TYPE) {
723
- REAL_TIME_DATA_TYPE[REAL_TIME_DATA_TYPE["LOCATION"] = 1] = "LOCATION";
724
- REAL_TIME_DATA_TYPE[REAL_TIME_DATA_TYPE["PROCESS"] = 2] = "PROCESS";
725
- })(REAL_TIME_DATA_TYPE || (REAL_TIME_DATA_TYPE = {}));
726
- const ISOLATED_BOUNDARY_SVG = `<svg width="24" height="24" viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg">
727
- <g opacity="0.6">
728
- <rect width="24" height="24" rx="12" fill="#1E1E1F" fill-opacity="0.5"/>
729
- <path d="M8.20573 12.4961C6.739 11.1707 4.85775 11.2284 3.60938 11.1707C6.17744 13.0156 6.05584 15.7887 5.67404 16.9446H14.327C13.837 13.1673 15.5321 10.1289 16.4408 9.08187C15.0342 9.46118 13.5794 10.8303 13.0278 11.4674C12.8102 8.94572 13.992 5.97489 14.61 4.80469C12.5235 5.92214 11.0501 8.26056 10.4938 9.56261C9.22803 7.5947 6.83894 7.17806 5.883 6.9432C8.20573 10.2373 8.00039 11.1707 8.20573 12.4961Z" fill="white"/>
730
- <mask id="path-3-outside-1_9822_43516" maskUnits="userSpaceOnUse" x="13.2344" y="7.20545" width="12.0208" height="12.0208" fill="black">
731
- <rect fill="white" x="13.2344" y="7.20545" width="12.0208" height="12.0208"/>
732
- <path d="M20.6852 11.1208C19.9042 10.3398 18.6378 10.3398 17.8568 11.1208L17.2733 11.7043C16.6707 12.3069 16.5358 13.1973 16.8631 13.9319L17.5916 13.2034C17.5657 12.9181 17.6619 12.6236 17.8803 12.4052L18.5674 11.7181C18.9579 11.3276 19.5911 11.3276 19.9816 11.7181L20.4056 12.1421C20.7959 12.5326 20.796 13.1658 20.4056 13.5563L19.7185 14.2434C19.5128 14.449 19.2393 14.5457 18.9699 14.5348L18.2235 15.2812C18.9538 15.596 19.8325 15.4565 20.429 14.86L21.0125 14.2765C21.7935 13.4955 21.7936 12.2291 21.0125 11.4481L20.6852 11.1208Z"/>
733
- </mask>
734
- <path d="M20.6852 11.1208C19.9042 10.3398 18.6378 10.3398 17.8568 11.1208L17.2733 11.7043C16.6707 12.3069 16.5358 13.1973 16.8631 13.9319L17.5916 13.2034C17.5657 12.9181 17.6619 12.6236 17.8803 12.4052L18.5674 11.7181C18.9579 11.3276 19.5911 11.3276 19.9816 11.7181L20.4056 12.1421C20.7959 12.5326 20.796 13.1658 20.4056 13.5563L19.7185 14.2434C19.5128 14.449 19.2393 14.5457 18.9699 14.5348L18.2235 15.2812C18.9538 15.596 19.8325 15.4565 20.429 14.86L21.0125 14.2765C21.7935 13.4955 21.7936 12.2291 21.0125 11.4481L20.6852 11.1208Z" fill="white"/>
735
- <path d="M20.6852 11.1208L21.7458 10.0601L21.7459 10.0601L20.6852 11.1208ZM17.8568 11.1208L16.7961 10.0601L16.7962 10.0601L17.8568 11.1208ZM17.2733 11.7043L16.2126 10.6437L16.2126 10.6436L17.2733 11.7043ZM16.8631 13.9319L17.9238 14.9926L16.381 16.5354L15.493 14.5425L16.8631 13.9319ZM17.5916 13.2034L19.0855 13.0678L19.149 13.7674L18.6523 14.2641L17.5916 13.2034ZM20.4056 12.1421L21.4662 11.0814L21.4665 11.0817L20.4056 12.1421ZM18.9699 14.5348L17.9093 13.4741L18.3741 13.0093L19.0309 13.036L18.9699 14.5348ZM18.2235 15.2812L17.6298 16.6587L15.5997 15.7837L17.1628 14.2206L18.2235 15.2812ZM20.429 14.86L21.4897 15.9207L21.4897 15.9207L20.429 14.86ZM21.0125 14.2765L22.0732 15.3371L22.0732 15.3372L21.0125 14.2765ZM20.6852 11.1208L19.6246 12.1815C19.4294 11.9862 19.1127 11.9863 18.9174 12.1815L17.8568 11.1208L16.7962 10.0601C18.163 8.69337 20.379 8.6934 21.7458 10.0601L20.6852 11.1208ZM17.8568 11.1208L18.9175 12.1814L18.334 12.7649L17.2733 11.7043L16.2126 10.6436L16.7961 10.0601L17.8568 11.1208ZM17.2733 11.7043L18.334 12.7649C18.1877 12.9112 18.1488 13.1318 18.2333 13.3214L16.8631 13.9319L15.493 14.5425C14.9228 13.2629 15.1537 11.7026 16.2126 10.6437L17.2733 11.7043ZM16.8631 13.9319L15.8025 12.8713L16.531 12.1428L17.5916 13.2034L18.6523 14.2641L17.9238 14.9926L16.8631 13.9319ZM17.5916 13.2034L16.0978 13.339C16.0334 12.6302 16.2727 11.8914 16.8196 11.3445L17.8803 12.4052L18.9409 13.4658C19.051 13.3558 19.098 13.206 19.0855 13.0678L17.5916 13.2034ZM17.8803 12.4052L16.8196 11.3445L17.5067 10.6574L18.5674 11.7181L19.628 12.7788L18.9409 13.4658L17.8803 12.4052ZM18.5674 11.7181L17.5067 10.6574C18.483 9.68112 20.0659 9.68112 21.0422 10.6574L19.9816 11.7181L18.9209 12.7788C19.1162 12.974 19.4328 12.974 19.628 12.7788L18.5674 11.7181ZM19.9816 11.7181L21.0422 10.6574L21.4662 11.0814L20.4056 12.1421L19.3449 13.2027L18.9209 12.7788L19.9816 11.7181ZM20.4056 12.1421L21.4665 11.0817C22.4419 12.0577 22.4428 13.6404 21.4662 14.617L20.4056 13.5563L19.3449 12.4956C19.1493 12.6913 19.1498 13.0076 19.3446 13.2024L20.4056 12.1421ZM20.4056 13.5563L21.4662 14.617L20.7791 15.304L19.7185 14.2434L18.6578 13.1827L19.3449 12.4956L20.4056 13.5563ZM19.7185 14.2434L20.7791 15.304C20.2628 15.8204 19.5762 16.0607 18.909 16.0335L18.9699 14.5348L19.0309 13.036C18.9025 13.0308 18.7629 13.0777 18.6578 13.1827L19.7185 14.2434ZM18.9699 14.5348L20.0306 15.5954L19.2841 16.3419L18.2235 15.2812L17.1628 14.2206L17.9093 13.4741L18.9699 14.5348ZM18.2235 15.2812L18.8172 13.9037C19.0042 13.9843 19.2223 13.9455 19.3684 13.7993L20.429 14.86L21.4897 15.9207C20.4427 16.9676 18.9034 17.2077 17.6298 16.6587L18.2235 15.2812ZM20.429 14.86L19.3684 13.7994L19.9519 13.2159L21.0125 14.2765L22.0732 15.3372L21.4897 15.9207L20.429 14.86ZM21.0125 14.2765L19.9518 13.2159C20.1471 13.0206 20.1471 12.704 19.9519 12.5088L21.0125 11.4481L22.0732 10.3874C23.4401 11.7543 23.4399 13.9703 22.0732 15.3371L21.0125 14.2765ZM21.0125 11.4481L19.9519 12.5088L19.6246 12.1814L20.6852 11.1208L21.7459 10.0601L22.0732 10.3874L21.0125 11.4481Z" fill="#8E8E8F" mask="url(#path-3-outside-1_9822_43516)"/>
736
- <mask id="path-5-outside-2_9822_43516" maskUnits="userSpaceOnUse" x="9.22265" y="10.388" width="12.7279" height="12.7279" fill="black">
737
- <rect fill="white" x="9.22265" y="10.388" width="12.7279" height="12.7279"/>
738
- <path d="M16.1708 14.5077C15.4353 14.1778 14.5422 14.3133 13.9383 14.9172L13.3548 15.5007C12.574 16.2818 12.5738 17.5481 13.3548 18.3291L13.6821 18.6564C14.4631 19.4374 15.7295 19.4373 16.5105 18.6564L17.094 18.0729C17.6895 17.4775 17.8299 16.6003 17.5173 15.8708L16.7743 16.6138C16.7828 16.8803 16.6863 17.1494 16.4829 17.3527L15.7952 18.0405C15.4046 18.4309 14.7714 18.431 14.3809 18.0405L13.957 17.6165C13.5667 17.226 13.5666 16.5927 13.957 16.2023L14.6447 15.5145C14.8654 15.2939 15.1639 15.1979 15.452 15.2266L16.1708 14.5077Z"/>
739
- </mask>
740
- <path d="M16.1708 14.5077C15.4353 14.1778 14.5422 14.3133 13.9383 14.9172L13.3548 15.5007C12.574 16.2818 12.5738 17.5481 13.3548 18.3291L13.6821 18.6564C14.4631 19.4374 15.7295 19.4373 16.5105 18.6564L17.094 18.0729C17.6895 17.4775 17.8299 16.6003 17.5173 15.8708L16.7743 16.6138C16.7828 16.8803 16.6863 17.1494 16.4829 17.3527L15.7952 18.0405C15.4046 18.4309 14.7714 18.431 14.3809 18.0405L13.957 17.6165C13.5667 17.226 13.5666 16.5927 13.957 16.2023L14.6447 15.5145C14.8654 15.2939 15.1639 15.1979 15.452 15.2266L16.1708 14.5077Z" fill="white"/>
741
- <path d="M16.1708 14.5077L16.7847 13.1391L18.7701 14.0297L17.2315 15.5684L16.1708 14.5077ZM13.3548 15.5007L12.294 14.4402L12.2941 14.44L13.3548 15.5007ZM16.5105 18.6564L17.5712 19.7171L17.5711 19.7172L16.5105 18.6564ZM17.5173 15.8708L16.4567 14.8102L18.0236 13.2433L18.8962 15.2802L17.5173 15.8708ZM16.7743 16.6138L15.2751 16.6613L15.2545 16.0123L15.7137 15.5532L16.7743 16.6138ZM15.7952 18.0405L16.8558 19.1011L16.8556 19.1013L15.7952 18.0405ZM13.957 17.6165L12.8963 18.6772L12.8959 18.6767L13.957 17.6165ZM13.957 16.2023L12.8961 15.1418L12.8963 15.1416L13.957 16.2023ZM15.452 15.2266L16.5126 16.2872L16.0102 16.7896L15.3032 16.7192L15.452 15.2266ZM16.1708 14.5077L15.5569 15.8763C15.3688 15.792 15.1469 15.8299 14.999 15.9779L13.9383 14.9172L12.8776 13.8565C13.9375 12.7967 15.5018 12.5636 16.7847 13.1391L16.1708 14.5077ZM13.9383 14.9172L14.999 15.9779L14.4155 16.5614L13.3548 15.5007L12.2941 14.44L12.8776 13.8565L13.9383 14.9172ZM13.3548 15.5007L14.4156 16.5612C14.2202 16.7567 14.2204 17.0734 14.4155 17.2685L13.3548 18.3291L12.2941 19.3898C10.9272 18.0229 10.9277 15.8069 12.294 14.4402L13.3548 15.5007ZM13.3548 18.3291L14.4155 17.2685L14.7428 17.5958L13.6821 18.6564L12.6215 19.7171L12.2941 19.3898L13.3548 18.3291ZM13.6821 18.6564L14.7428 17.5958C14.9378 17.7908 15.2546 17.791 15.45 17.5956L16.5105 18.6564L17.5711 19.7172C16.2044 21.0836 13.9884 21.084 12.6215 19.7171L13.6821 18.6564ZM16.5105 18.6564L15.4499 17.5958L16.0334 17.0123L17.094 18.0729L18.1547 19.1336L17.5712 19.7171L16.5105 18.6564ZM17.094 18.0729L16.0334 17.0123C16.1795 16.8662 16.2185 16.6481 16.1385 16.4615L17.5173 15.8708L18.8962 15.2802C19.4413 16.5526 19.1996 18.0887 18.1547 19.1336L17.094 18.0729ZM17.5173 15.8708L18.578 16.9315L17.835 17.6745L16.7743 16.6138L15.7137 15.5532L16.4567 14.8102L17.5173 15.8708ZM16.7743 16.6138L18.2736 16.5664C18.2945 17.226 18.0544 17.9026 17.5436 18.4134L16.4829 17.3527L15.4223 16.2921C15.3182 16.3961 15.2711 16.5346 15.2751 16.6613L16.7743 16.6138ZM16.4829 17.3527L17.5436 18.4134L16.8558 19.1011L15.7952 18.0405L14.7345 16.9798L15.4223 16.2921L16.4829 17.3527ZM15.7952 18.0405L16.8556 19.1013C15.8795 20.0771 14.2967 20.0776 13.3203 19.1011L14.3809 18.0405L15.4416 16.9798C15.2461 16.7844 14.9297 16.7847 14.7347 16.9797L15.7952 18.0405ZM14.3809 18.0405L13.3203 19.1011L12.8963 18.6772L13.957 17.6165L15.0176 16.5558L15.4416 16.9798L14.3809 18.0405ZM13.957 17.6165L12.8959 18.6767C11.9207 17.7008 11.92 16.1182 12.8961 15.1418L13.957 16.2023L15.0178 17.2628C15.2133 17.0672 15.2128 16.7512 15.018 16.5563L13.957 17.6165ZM13.957 16.2023L12.8963 15.1416L13.5841 14.4539L14.6447 15.5145L15.7054 16.5752L15.0176 17.2629L13.957 16.2023ZM14.6447 15.5145L13.5841 14.4539C14.1364 13.9015 14.8848 13.6626 15.6007 13.734L15.452 15.2266L15.3032 16.7192C15.443 16.7331 15.5943 16.6862 15.7054 16.5752L14.6447 15.5145ZM15.452 15.2266L14.3913 14.1659L15.1101 13.4471L16.1708 14.5077L17.2315 15.5684L16.5126 16.2872L15.452 15.2266Z" fill="#8E8E8F" mask="url(#path-5-outside-2_9822_43516)"/>
742
- <path d="M20.6852 11.1208C19.9042 10.3398 18.6378 10.3398 17.8568 11.1208L17.2733 11.7043C16.6707 12.3069 16.5358 13.1973 16.8631 13.9319L17.5916 13.2034C17.5657 12.9181 17.6619 12.6236 17.8803 12.4052L18.5674 11.7181C18.9579 11.3276 19.5911 11.3276 19.9816 11.7181L20.4056 12.1421C20.7959 12.5326 20.796 13.1658 20.4056 13.5563L19.7185 14.2434C19.5128 14.449 19.2393 14.5457 18.9699 14.5348L18.2235 15.2812C18.9538 15.596 19.8325 15.4565 20.429 14.86L21.0125 14.2765C21.7935 13.4955 21.7936 12.2291 21.0125 11.4481L20.6852 11.1208Z" fill="white"/>
743
- <rect width="2.59942" height="2.97264" rx="1" transform="matrix(-0.707107 -0.707107 -0.707107 0.707107 21.1133 12.8486)" fill="#8E8E8F"/>
744
- <rect width="1.91578" height="4.24403" transform="matrix(-0.707107 -0.707107 -0.707107 0.707107 19.6992 13.8066)" fill="#8E8E8F"/>
745
- <rect width="2.59942" height="2.97264" rx="1" transform="matrix(-0.707107 -0.707107 -0.707107 0.707107 17.3125 16.6455)" fill="#8E8E8F"/>
746
- </g>
747
- </svg>`;
748
- /**
749
- * 遍历割草任务,下述四个字段可以在路径中唯一确定遍历的位置(当前区域、当前块、当前行、在当前行上的路程)
750
- */
751
- const ACTION_BLOCK_COVER = 5;
752
- /**
753
- * 遍历割草块转移任务(在同一区域中的一个块转移到下一个块),下述四个字段可以在路径中唯一确定转移的位置(当前区域、前置块、当前转移路径线序号、在当前线上的路程)
754
- */
755
- const ACTION_BLOCK_TRANSFER = 6;
756
- /**
757
- * 边界割草任务(割草任务内部的巡边任务)
758
- */
759
- const ACTION_BOUNDARY_TASK = 8;
760
-
761
808
  /**
762
809
  * 路径图层
763
810
  * 专门处理路径元素的渲染
@@ -808,51 +855,90 @@ class ChannelLayer extends BaseLayer {
808
855
  }
809
856
  // �� 修改:计算包含所有分区和通道的边界框
810
857
  let minX = Infinity, minY = Infinity, maxX = -Infinity, maxY = -Infinity;
811
- // 1. 先计算所有分区的边界
812
- for (const partitionId in subBoundaryBorder) {
813
- const boundaryData = subBoundaryBorder[partitionId];
814
- if (boundaryData && boundaryData.coordinates && boundaryData.coordinates.length > 0) {
815
- for (const coord of boundaryData.coordinates) {
816
- minX = Math.min(minX, coord[0]);
817
- minY = Math.min(minY, coord[1]);
818
- maxX = Math.max(maxX, coord[0]);
819
- maxY = Math.max(maxY, coord[1]);
858
+ // 1. 先计算所有分区的边界,如果能拿到边界的svg的大小,就使用这个如果拿不到,就根据分区去计算
859
+ const svg = document.getElementById(SVG_MAP_VIEW_ID);
860
+ if (svg && svg instanceof SVGSVGElement && svg.viewBox) {
861
+ const viewBox = svg.viewBox.baseVal;
862
+ minX = viewBox.x;
863
+ minY = viewBox.y;
864
+ maxX = viewBox.x + viewBox.width;
865
+ maxY = viewBox.y + viewBox.height;
866
+ }
867
+ else {
868
+ for (const partitionId in subBoundaryBorder) {
869
+ const boundaryData = subBoundaryBorder[partitionId];
870
+ if (boundaryData && boundaryData.coordinates && boundaryData.coordinates.length > 0) {
871
+ for (const coord of boundaryData.coordinates) {
872
+ minX = Math.min(minX, coord[0]);
873
+ minY = Math.min(minY, coord[1]);
874
+ maxX = Math.max(maxX, coord[0]);
875
+ maxY = Math.max(maxY, coord[1]);
876
+ }
820
877
  }
821
878
  }
822
879
  }
823
880
  // 2. 再计算所有通道的边界
824
881
  for (const element of this.elements) {
825
- const tunnelConnection = element.originalData?.connection;
826
- if (tunnelConnection && Array.isArray(tunnelConnection)) {
827
- const clipPathId = `channel-exclude-${element.originalData?.id || Math.random().toString(36).substr(2, 9)}`;
828
- // 检查是否已存在该 clipPath
829
- const existingClipPath = defs.querySelector(`#${clipPathId}`);
830
- if (existingClipPath)
831
- continue;
832
- // 创建 clipPath
833
- const clipPath = document.createElementNS('http://www.w3.org/2000/svg', 'clipPath');
834
- clipPath.setAttribute('id', clipPathId);
835
- clipPath.setAttribute('clipPathUnits', 'userSpaceOnUse');
836
- // === 合成一个 path ===
837
- let d = `M ${minX} ${minY} L ${maxX} ${minY} L ${maxX} ${maxY} L ${minX} ${maxY} Z`;
838
- for (const partitionId of tunnelConnection) {
839
- const boundaryData = subBoundaryBorder[partitionId];
840
- if (boundaryData && boundaryData.coordinates.length >= 3) {
841
- d += ` M ${boundaryData.coordinates[0][0]} ${boundaryData.coordinates[0][1]}`;
842
- for (let i = 1; i < boundaryData.coordinates.length; i++) {
843
- d += ` L ${boundaryData.coordinates[i][0]} ${boundaryData.coordinates[i][1]}`;
844
- }
845
- d += ' Z';
882
+ // const tunnelConnection = element.originalData?.connection;
883
+ // if (tunnelConnection && Array.isArray(tunnelConnection)) {
884
+ // const clipPathId = `channel-exclude-${
885
+ // element.originalData?.id || Math.random().toString(36).substr(2, 9)
886
+ // }`;
887
+ // // 检查是否已存在该 clipPath
888
+ // const existingClipPath = defs.querySelector(`#${clipPathId}`);
889
+ // if (existingClipPath) continue;
890
+ // // 创建 clipPath
891
+ // const clipPath = document.createElementNS('http://www.w3.org/2000/svg', 'clipPath');
892
+ // clipPath.setAttribute('id', clipPathId);
893
+ // clipPath.setAttribute('clipPathUnits', 'userSpaceOnUse');
894
+ // // === 合成一个 path ===
895
+ // let d = `M ${minX} ${minY} L ${maxX} ${minY} L ${maxX} ${maxY} L ${minX} ${maxY} Z`;
896
+ // for (const partitionId of tunnelConnection) {
897
+ // const boundaryData = subBoundaryBorder[partitionId];
898
+ // if (boundaryData && boundaryData.coordinates.length >= 3) {
899
+ // d += ` M ${boundaryData.coordinates[0][0]} ${boundaryData.coordinates[0][1]}`;
900
+ // for (let i = 1; i < boundaryData.coordinates.length; i++) {
901
+ // d += ` L ${boundaryData.coordinates[i][0]} ${boundaryData.coordinates[i][1]}`;
902
+ // }
903
+ // d += ' Z';
904
+ // }
905
+ // }
906
+ // const path = document.createElementNS('http://www.w3.org/2000/svg', 'path');
907
+ // path.setAttribute('d', d);
908
+ // path.setAttribute('clip-rule', 'evenodd'); // 关键
909
+ // clipPath.appendChild(path);
910
+ // defs.appendChild(clipPath);
911
+ // clipPathIdsMap[element.originalData?.id.toString()] = clipPathId;
912
+ // } else {
913
+ const clipPathId = `channel-exclude-all-${element.originalData?.id || Math.random().toString(36).substr(2, 9)}`;
914
+ // 检查是否已存在该 clipPath
915
+ const existingClipPath = defs.querySelector(`#${clipPathId}`);
916
+ if (existingClipPath)
917
+ continue;
918
+ // 创建 clipPath
919
+ const clipPath = document.createElementNS('http://www.w3.org/2000/svg', 'clipPath');
920
+ clipPath.setAttribute('id', clipPathId);
921
+ clipPath.setAttribute('clipPathUnits', 'userSpaceOnUse');
922
+ // === 合成一个 path ===
923
+ let d = `M ${minX} ${minY} L ${maxX} ${minY} L ${maxX} ${maxY} L ${minX} ${maxY} Z`;
924
+ for (const partitionId in subBoundaryBorder) {
925
+ const boundaryData = subBoundaryBorder[partitionId];
926
+ if (boundaryData && boundaryData.coordinates.length >= 3) {
927
+ d += ` M ${boundaryData.coordinates[0][0]} ${boundaryData.coordinates[0][1]}`;
928
+ for (let i = 1; i < boundaryData.coordinates.length; i++) {
929
+ d += ` L ${boundaryData.coordinates[i][0]} ${boundaryData.coordinates[i][1]}`;
846
930
  }
931
+ d += ' Z';
847
932
  }
848
- const path = document.createElementNS('http://www.w3.org/2000/svg', 'path');
849
- path.setAttribute('d', d);
850
- path.setAttribute('clip-rule', 'evenodd'); // 关键
851
- clipPath.appendChild(path);
852
- defs.appendChild(clipPath);
853
- clipPathIdsMap[element.originalData?.id.toString()] = clipPathId;
854
933
  }
934
+ const path = document.createElementNS('http://www.w3.org/2000/svg', 'path');
935
+ path.setAttribute('d', d);
936
+ path.setAttribute('clip-rule', 'evenodd'); // 关键
937
+ clipPath.appendChild(path);
938
+ defs.appendChild(clipPath);
939
+ clipPathIdsMap[element.originalData?.id.toString()] = clipPathId;
855
940
  }
941
+ // }
856
942
  return clipPathIdsMap;
857
943
  }
858
944
  /**
@@ -916,13 +1002,14 @@ class PathLayer extends BaseLayer {
916
1002
  this.level = 3;
917
1003
  this.scale = 1;
918
1004
  this.lineScale = 1;
1005
+ this.boundaryPaths = {};
919
1006
  this.type = LAYER_DEFAULT_TYPE.PATH;
920
1007
  }
921
1008
  /**
922
1009
  * 创建所有分区并集的 clipPath
923
1010
  */
924
1011
  createUnionClipPath(svgGroup) {
925
- const { subBoundaryBorder } = useSubBoundaryBorderStore.getState();
1012
+ const { subBoundaryBorder, obstacles, svgElements } = useSubBoundaryBorderStore.getState();
926
1013
  // 确保 defs 元素存在
927
1014
  let defs = svgGroup.querySelector('defs');
928
1015
  if (!defs) {
@@ -936,7 +1023,7 @@ class PathLayer extends BaseLayer {
936
1023
  defs.removeChild(existing);
937
1024
  // 合成所有分区的 path
938
1025
  let d = '';
939
- // 合成所有边界
1026
+ // 1. 外圈(主边界,顺时针)
940
1027
  Object.values(subBoundaryBorder).forEach((item) => {
941
1028
  const bCoords = item.coordinates;
942
1029
  if (bCoords.length >= 3) {
@@ -947,6 +1034,48 @@ class PathLayer extends BaseLayer {
947
1034
  d += ' Z ';
948
1035
  }
949
1036
  });
1037
+ // 2. 内圈(禁区,逆时针)
1038
+ Object.values(obstacles).forEach((item) => {
1039
+ const bCoords = item.coordinates;
1040
+ if (bCoords.length >= 3) {
1041
+ d += `M ${bCoords[bCoords.length - 1][0]} ${bCoords[bCoords.length - 1][1]}`;
1042
+ for (let i = bCoords.length - 2; i >= 0; i--) {
1043
+ d += ` L ${bCoords[i][0]} ${bCoords[i][1]}`;
1044
+ }
1045
+ d += ' Z ';
1046
+ }
1047
+ });
1048
+ // 3. svgElements(解析 SVG 字符串并提取 path 数据)
1049
+ Object.values(svgElements).forEach((svgPath) => {
1050
+ const svgPathString = svgPath?.metadata?.svg;
1051
+ if (svgPathString && typeof svgPathString === 'string' && svgPathString.trim()) {
1052
+ // 处理转义字符
1053
+ const processedSvgString = svgPathString.replace(/\\n/g, '\n').replace(/\\"/g, '"');
1054
+ // 解析 SVG 字符串
1055
+ const parser = new DOMParser();
1056
+ const svgDoc = parser.parseFromString(processedSvgString, 'image/svg+xml');
1057
+ const svgElement = svgDoc.documentElement;
1058
+ if (svgElement.tagName === 'svg') {
1059
+ // 查找 path 元素
1060
+ const pathElement = svgElement.querySelector('path');
1061
+ if (pathElement) {
1062
+ const pathData = pathElement.getAttribute('d');
1063
+ if (pathData) {
1064
+ // 获取 SVG 元素的变换参数
1065
+ const centerCoords = svgPath.coordinates?.[0] || [0, 0];
1066
+ const center = [centerCoords[0], centerCoords[1]];
1067
+ const userScale = svgPath.metadata.scale || 1;
1068
+ const direction = svgPath.metadata?.direction || 0;
1069
+ const originalWidth = parseFloat(svgElement.getAttribute('width') || '76');
1070
+ const originalHeight = parseFloat(svgElement.getAttribute('height') || '68');
1071
+ // 应用变换到路径数据
1072
+ const transformedPathData = this.transformSvgPath(pathData, center, userScale, direction, originalWidth, originalHeight);
1073
+ d += transformedPathData + ' ';
1074
+ }
1075
+ }
1076
+ }
1077
+ }
1078
+ });
950
1079
  const path = document.createElementNS('http://www.w3.org/2000/svg', 'path');
951
1080
  path.setAttribute('d', d);
952
1081
  const clipPath = document.createElementNS('http://www.w3.org/2000/svg', 'clipPath');
@@ -971,48 +1100,132 @@ class PathLayer extends BaseLayer {
971
1100
  // 2. 创建一个组,应用 clipPath
972
1101
  const group = document.createElementNS('http://www.w3.org/2000/svg', 'g');
973
1102
  group.setAttribute('clip-path', `url(#${clipPathId})`);
974
- group.setAttribute('opacity', '0.35'); // 统一透明度,防止叠加脏乱
975
- // 3. 渲染所有路径
976
- for (const element of this.elements) {
977
- this.renderPathToGroup(group, element);
978
- }
1103
+ group.setAttribute('opacity', '0.5'); // 统一透明度,防止叠加脏乱
1104
+ // 3. 优化渲染:按样式分组并合并路径
1105
+ this.renderOptimizedPaths(group);
979
1106
  svgGroup.appendChild(group);
980
1107
  }
981
1108
  /**
982
- * 渲染单个路径到指定的组中
1109
+ * 优化渲染:按样式分组并合并路径,减少 DOM 节点数量
983
1110
  */
984
- renderPathToGroup(group, element) {
985
- const { coordinates, style } = element;
986
- if (coordinates.length < 2)
987
- return;
988
- const path = document.createElementNS('http://www.w3.org/2000/svg', 'path');
989
- // 构建路径数据
990
- let pathData = `M ${coordinates[0][0]} ${coordinates[0][1]}`;
991
- for (let i = 1; i < coordinates.length; i++) {
992
- pathData += ` L ${coordinates[i][0]} ${coordinates[i][1]}`;
993
- }
994
- path.style.mixBlendMode = 'normal';
995
- // 设置路径属性
996
- path.setAttribute('d', pathData);
997
- // 直接给fill的颜色设置透明度会导致path重叠的部分颜色叠加,所以使用fill填充实色,通过fill-opacity设置透明度
998
- path.setAttribute('fill', 'none');
999
- // path.setAttribute('fill-opacity', '0.4');
1000
- path.setAttribute('stroke', style.strokeColor || '#000000');
1001
- path.setAttribute('mix-blend-mode', 'normal');
1002
- const lineWidth = Math.max(style.lineWidth || 1, 0.5);
1003
- path.setAttribute('stroke-width', lineWidth.toString());
1004
- path.setAttribute('stroke-linecap', 'round');
1005
- path.setAttribute('stroke-linejoin', 'round');
1006
- // 注意:这里不设置 opacity,因为透明度由父组控制
1007
- // path.setAttribute('vector-effect', 'non-scaling-stroke');
1008
- if (style.lineDash && style.lineDash.length > 0) {
1009
- path.setAttribute('stroke-dasharray', style.lineDash.join(','));
1111
+ renderOptimizedPaths(group) {
1112
+ // 按样式分组存储路径数据
1113
+ const styleGroups = new Map();
1114
+ // 收集所有路径数据并按样式分组
1115
+ for (const element of this.elements) {
1116
+ // 类型断言:PathLayer 中的 elements 实际上是 PathElements 结构
1117
+ const pathElement = element;
1118
+ const { id, elements } = pathElement;
1119
+ this.boundaryPaths[id] = [];
1120
+ elements.forEach((pathElement) => {
1121
+ const { coordinates, style } = pathElement;
1122
+ if (coordinates.length < 2)
1123
+ return;
1124
+ // 生成样式键(用于分组)
1125
+ const styleKey = this.generateStyleKey(style);
1126
+ // 构建路径数据
1127
+ let pathData = `M ${coordinates[0][0]} ${coordinates[0][1]}`;
1128
+ for (let i = 1; i < coordinates.length; i++) {
1129
+ pathData += ` L ${coordinates[i][0]} ${coordinates[i][1]}`;
1130
+ }
1131
+ // 按样式分组存储
1132
+ if (!styleGroups.has(styleKey)) {
1133
+ styleGroups.set(styleKey, { pathData: [], elements: [] });
1134
+ }
1135
+ styleGroups.get(styleKey).pathData.push(pathData);
1136
+ styleGroups.get(styleKey).elements.push(pathElement);
1137
+ });
1010
1138
  }
1011
- else if (style.strokeDasharray) {
1012
- path.setAttribute('stroke-dasharray', style.strokeDasharray);
1139
+ // 为每种样式创建一个合并的 path 元素
1140
+ styleGroups.forEach((groupData) => {
1141
+ const { pathData, elements } = groupData;
1142
+ if (pathData.length === 0)
1143
+ return;
1144
+ // 使用第一个元素的样式作为该组的样式
1145
+ const firstElement = elements[0];
1146
+ const style = firstElement.style;
1147
+ // 创建合并的 path 元素
1148
+ const path = document.createElementNS('http://www.w3.org/2000/svg', 'path');
1149
+ // 合并所有路径数据
1150
+ const mergedPathData = pathData.join(' ');
1151
+ path.setAttribute('d', mergedPathData);
1152
+ // 设置样式属性
1153
+ path.setAttribute('fill', 'none');
1154
+ path.setAttribute('stroke', style.lineColor || '#000000');
1155
+ path.setAttribute('mix-blend-mode', 'normal');
1156
+ const lineWidth = Math.max(style.lineWidth || 1, 0.5);
1157
+ path.setAttribute('stroke-width', lineWidth.toString());
1158
+ path.setAttribute('stroke-linecap', 'round');
1159
+ path.setAttribute('stroke-linejoin', 'round');
1160
+ path.classList.add('vector-path');
1161
+ // 将合并的 path 添加到组中
1162
+ group.appendChild(path);
1163
+ // 保存引用到 boundaryPaths 中(保持兼容性)
1164
+ elements.forEach((element) => {
1165
+ const { id } = element;
1166
+ if (!this.boundaryPaths[id]) {
1167
+ this.boundaryPaths[id] = [];
1168
+ }
1169
+ this.boundaryPaths[id].push(path);
1170
+ });
1171
+ });
1172
+ }
1173
+ /**
1174
+ * 变换 SVG 路径数据
1175
+ */
1176
+ transformSvgPath(pathData, center, scale, direction, originalWidth, originalHeight) {
1177
+ // 解析路径数据并应用变换
1178
+ const commands = pathData.match(/[MmLlHhVvCcSsQqTtAaZz][^MmLlHhVvCcSsQqTtAaZz]*/g) || [];
1179
+ let transformedCommands = [];
1180
+ for (const command of commands) {
1181
+ const type = command[0];
1182
+ const params = command
1183
+ .slice(1)
1184
+ .trim()
1185
+ .split(/[\s,]+/)
1186
+ .filter(Boolean)
1187
+ .map(Number);
1188
+ if (type === 'Z' || type === 'z') {
1189
+ // 闭合路径,不需要变换
1190
+ transformedCommands.push(command);
1191
+ continue;
1192
+ }
1193
+ // 处理坐标参数
1194
+ let transformedParams = [];
1195
+ for (let i = 0; i < params.length; i += 2) {
1196
+ if (i + 1 < params.length) {
1197
+ let x = params[i];
1198
+ let y = params[i + 1];
1199
+ // 应用变换:先平移到中心,然后缩放、旋转,最后平移到目标位置
1200
+ // 1. 平移到原点(相对于原始尺寸的中心)
1201
+ x -= originalWidth / 2;
1202
+ y -= originalHeight / 2;
1203
+ // 2. 应用缩放
1204
+ x *= scale;
1205
+ y *= scale;
1206
+ // 3. 应用旋转
1207
+ const cos = Math.cos(-direction);
1208
+ const sin = Math.sin(-direction);
1209
+ const newX = x * cos - y * sin;
1210
+ const newY = x * sin + y * cos;
1211
+ // 4. 平移到目标位置
1212
+ x = newX + center[0];
1213
+ y = newY + center[1];
1214
+ transformedParams.push(x, y);
1215
+ }
1216
+ }
1217
+ // 重建命令
1218
+ if (transformedParams.length > 0) {
1219
+ transformedCommands.push(type + transformedParams.join(' '));
1220
+ }
1013
1221
  }
1014
- path.classList.add('vector-path');
1015
- group.appendChild(path);
1222
+ return transformedCommands.join(' ');
1223
+ }
1224
+ /**
1225
+ * 生成样式键,用于路径分组
1226
+ */
1227
+ generateStyleKey(style) {
1228
+ return `${style.lineColor || '#000000'}-${style.lineWidth || 1}-${style.opacity || 1}`;
1016
1229
  }
1017
1230
  }
1018
1231
 
@@ -1120,7 +1333,7 @@ class ObstacleLayer extends BaseLayer {
1120
1333
  }
1121
1334
  }
1122
1335
 
1123
- var chargingPileImage = "";
1336
+ var chargingPileImage = "";
1124
1337
 
1125
1338
  /**
1126
1339
  * 充电桩图层
@@ -1310,7 +1523,7 @@ class SvgElementLayer extends BaseLayer {
1310
1523
  // 在transformGroup上应用变换:平移到中心,旋转,缩放,然后居中SVG
1311
1524
  const transform = [
1312
1525
  `translate(${center[0]}, ${center[1]})`,
1313
- `rotate(${(direction * 180) / Math.PI})`,
1526
+ `rotate(${-(direction * 180) / Math.PI})`,
1314
1527
  `scale(${userScale})`,
1315
1528
  `translate(${-originalWidth / 2}, ${-originalHeight / 2})`,
1316
1529
  ].join(' ');
@@ -1339,7 +1552,6 @@ class SvgElementLayer extends BaseLayer {
1339
1552
  renderSvgPlaceholder(svgGroup, center, metadata, style) {
1340
1553
  const size = (metadata?.scale || 1) * 20;
1341
1554
  const rect = document.createElementNS('http://www.w3.org/2000/svg', 'rect');
1342
- console.log('style==', style);
1343
1555
  rect.setAttribute('x', ((center[0] - size / 2) / 50).toString());
1344
1556
  rect.setAttribute('y', ((center[1] - size / 2) / 50).toString());
1345
1557
  rect.setAttribute('width', size.toString());
@@ -1409,9 +1621,10 @@ class VisionOffLayer extends BaseLayer {
1409
1621
  */
1410
1622
  const BOUNDARY_STYLES = {
1411
1623
  lineColor: '#ffffff',
1412
- fillColor: 'rgba(239, 255, 237, 0.15)', // 更鲜艳的绿色半透明填充,增强可见性
1624
+ fillColor: 'rgba(239, 255, 237, 0.1)', // 更鲜艳的绿色半透明填充,增强可见性
1413
1625
  lineWidth: 2,
1414
1626
  opacity: DEFAULT_OPACITIES.FULL,
1627
+ mowingLineColor: 'rgba(99, 216, 174, 1)',
1415
1628
  };
1416
1629
  const VISION_OFF_AREA_STYLES = {
1417
1630
  lineColor: 'rgba(108, 167, 255, 1)',
@@ -1436,14 +1649,15 @@ const DOODLE_STYLES = {
1436
1649
  lineColor: '#ff5722',
1437
1650
  fillColor: '#ff9800', // 粉色半透明填充
1438
1651
  lineWidth: DEFAULT_LINE_WIDTHS.TIME_LIMIT_OBSTACLE,
1439
- opacity: DEFAULT_OPACITIES.HIGH,
1652
+ opacity: DEFAULT_OPACITIES.DOODLE,
1440
1653
  };
1441
1654
  const PATH_EDGE_STYLES = {
1442
1655
  lineWidth: DEFAULT_LINE_WIDTHS.PATH,
1443
- opacity: DEFAULT_OPACITIES.LOW,
1444
- edgeLineColor: 'rgba(194, 203, 212)',
1656
+ opacity: DEFAULT_OPACITIES.MEDIUM,
1657
+ edgeLineColor: 'rgba(231, 238, 246)',
1445
1658
  transLineColor: 'transparent',
1446
- mowingLineColor: 'rgba(194, 203, 212)',
1659
+ mowedLineColor: 'rgba(231, 238, 246)',
1660
+ mowingLineColor: 'rgba(123, 200, 187)',
1447
1661
  };
1448
1662
  const CHANNEL_STYLES = {
1449
1663
  lineColor: 'purple',
@@ -1477,13 +1691,13 @@ const DEFAULT_STYLES = {
1477
1691
  function convertPointsFormat(points) {
1478
1692
  if (!points || points.length === 0)
1479
1693
  return null;
1480
- return points.map(point => {
1694
+ return points.map((point) => {
1481
1695
  if (point.length >= 2) {
1482
1696
  // 对前两个元素应用缩放因子,保留其他元素
1483
1697
  return [
1484
1698
  point[0] * SCALE_FACTOR,
1485
1699
  -point[1] * SCALE_FACTOR, // Y轴翻转,与Python代码一致
1486
- ...point.slice(2) // 保留第三个及以后的元素
1700
+ ...point.slice(2), // 保留第三个及以后的元素
1487
1701
  ];
1488
1702
  }
1489
1703
  return point;
@@ -1498,7 +1712,7 @@ function convertPositionFormat(position) {
1498
1712
  return null;
1499
1713
  return {
1500
1714
  x: position[0] * SCALE_FACTOR,
1501
- y: -position[1] * SCALE_FACTOR // Y轴翻转
1715
+ y: -position[1] * SCALE_FACTOR, // Y轴翻转
1502
1716
  };
1503
1717
  }
1504
1718
  /**
@@ -1507,9 +1721,146 @@ function convertPositionFormat(position) {
1507
1721
  function convertCoordinate(x, y) {
1508
1722
  return {
1509
1723
  x: x * SCALE_FACTOR,
1510
- y: -y * SCALE_FACTOR // Y轴翻转
1724
+ y: -y * SCALE_FACTOR, // Y轴翻转
1511
1725
  };
1512
1726
  }
1727
+ /**
1728
+ * @param x x坐标
1729
+ * @param y y坐标
1730
+ * @param isAllowInBoundary 是否允许点在边界上的判断
1731
+ * @return ture-点在边界上即可视为在边界内,false-严格判断点在边界内
1732
+ */
1733
+ function isPointIn$1(x, y, pointList, isAllowInBoundary) {
1734
+ let count = 0;
1735
+ let size = pointList.length;
1736
+ let p1, p2, p3;
1737
+ for (let i = 0; i < size; i++) {
1738
+ p1 = pointList[i];
1739
+ p2 = pointList[(i + 1) % size];
1740
+ if (p1.y == null || p2.y == null || p1.x == null || p2.x == null) {
1741
+ continue;
1742
+ }
1743
+ if (p1.y === p2.y) {
1744
+ continue;
1745
+ }
1746
+ if (y > Math.min(p1.y, p2.y) && y < Math.max(p1.y, p2.y)) {
1747
+ const interX = ((y - p1.y) * (p2.x - p1.x)) / (p2.y - p1.y) + p1.x;
1748
+ if (interX >= x) {
1749
+ count++;
1750
+ }
1751
+ else if (interX == x) {
1752
+ return isAllowInBoundary;
1753
+ }
1754
+ }
1755
+ else {
1756
+ if (y == p2.y && x <= p2.x) {
1757
+ p3 = pointList[(i + 2) % size];
1758
+ if (y >= Math.min(p1.y, p3.y) && y <= Math.max(p1.y, p3.y)) {
1759
+ // 若当前点的y坐标位于 p1和p3组成的线段关于y轴的投影中,则记为该点的射线只穿过端点一次。
1760
+ ++count;
1761
+ }
1762
+ else {
1763
+ // 若当前点的y坐标不能包含在p1和p3组成的线段关于y轴的投影中,则点射线通过的两条线段组成了一个弯折的部分,
1764
+ // 此时我们记射线穿过该端点两次
1765
+ count += 2;
1766
+ }
1767
+ }
1768
+ }
1769
+ }
1770
+ return count % 2 == 1;
1771
+ }
1772
+ /**
1773
+ * 用于判断三个点的方向的辅助方法
1774
+ */
1775
+ function orientation(p, q, r) {
1776
+ const val = (q.y - p.y) * (r.x - q.x) - (q.x - p.x) * (r.y - q.y);
1777
+ if (val == 0)
1778
+ return 0; // colinear
1779
+ return val > 0 ? 1 : 2; // clock or counterclock wise
1780
+ }
1781
+ /**
1782
+ * 检查点q是否在线段pr上的辅助方法
1783
+ */
1784
+ function onSegment(p, q, r) {
1785
+ if (q.x <= Math.max(p.x, r.x) &&
1786
+ q.x >= Math.min(p.x, r.x) &&
1787
+ q.y <= Math.max(p.y, r.y) &&
1788
+ q.y >= Math.min(p.y, r.y)) {
1789
+ return true;
1790
+ }
1791
+ return false;
1792
+ }
1793
+ /**
1794
+ * 判断两条线段是否相交的方法
1795
+ */
1796
+ function doTwoLinesIntersect(p1, q1, p2, q2) {
1797
+ //处理p1和q1两个点相同的情况
1798
+ if (p1.x - q1.x == 0 && p1.y - q1.y == 0) {
1799
+ return false;
1800
+ }
1801
+ if (p2.x - q2.x == 0 && p2.y - q2.y == 0) {
1802
+ return false;
1803
+ }
1804
+ // 计算四个点的方向
1805
+ const o1 = orientation(p1, q1, p2);
1806
+ const o2 = orientation(p1, q1, q2);
1807
+ const o3 = orientation(p2, q2, p1);
1808
+ const o4 = orientation(p2, q2, q1);
1809
+ // 一般情况,如果四个方向两两不同,则线段相交
1810
+ if (o1 != o2 && o3 != o4) {
1811
+ return true;
1812
+ }
1813
+ // 特殊情况,当线段的端点在另一条线段上时
1814
+ if (o1 == 0 && onSegment(p1, q1, p2))
1815
+ return true;
1816
+ if (o2 == 0 && onSegment(p1, q1, q2))
1817
+ return true;
1818
+ if (o3 == 0 && onSegment(p2, q2, p1))
1819
+ return true;
1820
+ if (o4 == 0 && onSegment(p2, q2, q1))
1821
+ return true;
1822
+ // 如果以上情况都不满足,则线段不相交
1823
+ return false;
1824
+ }
1825
+ /**
1826
+ * 判断多点折线是否相交
1827
+ */
1828
+ function doIntersect(points1, points2) {
1829
+ if (points1 == null || points2 == null || points1.length < 3 || points2.length < 3) {
1830
+ return false;
1831
+ }
1832
+ for (let i = 0; i < points1.length - 1; i++) {
1833
+ for (let j = 0; j < points2.length - 1; j++) {
1834
+ if (doTwoLinesIntersect(points1[i], points1[i + 1], points2[j], points2[j + 1])) {
1835
+ return true;
1836
+ }
1837
+ }
1838
+ }
1839
+ return false;
1840
+ }
1841
+ /**
1842
+ * 两个图形是否完全分离,互相不包含
1843
+ */
1844
+ function isOutsideToEachOther(points1, points2) {
1845
+ // 相交关系
1846
+ if (doIntersect(points1, points2)) {
1847
+ return false;
1848
+ }
1849
+ // 点关系,判断每个图形的点都在另一个图形外部
1850
+ for (let point of points1) {
1851
+ if (isPointIn$1(point.x, point.y, points2, true)) {
1852
+ // Log.i("ycf", "isOutsideToEachOther: mapPoint1=" + mapPoint);
1853
+ return false;
1854
+ }
1855
+ }
1856
+ for (let point of points2) {
1857
+ if (isPointIn$1(point.x, point.y, points1, true)) {
1858
+ // Log.i("ycf", "isOutsideToEachOther: mapPoint2=" + mapPoint);
1859
+ return false;
1860
+ }
1861
+ }
1862
+ return true;
1863
+ }
1513
1864
 
1514
1865
  /**
1515
1866
  * 按Python逻辑创建路径段:根据连续的两点之间的关系确定线段类型
@@ -1527,11 +1878,11 @@ function createPathSegmentsByType(list) {
1527
1878
  let currentSegmentType = null;
1528
1879
  for (const currentPoint of list) {
1529
1880
  const currentCoord = {
1530
- x: currentPoint.postureX * SCALE_FACTOR,
1531
- y: -currentPoint.postureY * SCALE_FACTOR // Y轴翻转
1881
+ x: currentPoint.postureX,
1882
+ y: currentPoint.postureY
1532
1883
  };
1533
1884
  if (lastPoint !== null) {
1534
- // 判断上一个点和当前点是否需要绘制 (Python逻辑)
1885
+ // 判断上一个点和当前点是否需要绘制 (iso端逻辑)
1535
1886
  const lastShouldDraw = lastPoint.pathType === '00' || lastPoint.pathType === '01' || lastPoint.knifeRotation === '01';
1536
1887
  const currentShouldDraw = currentPoint.pathType === '00' || currentPoint.pathType === '01' || currentPoint.knifeRotation === '01';
1537
1888
  let segmentType;
@@ -1551,8 +1902,8 @@ function createPathSegmentsByType(list) {
1551
1902
  // 开始新段
1552
1903
  currentSegment = [
1553
1904
  {
1554
- x: lastPoint.postureX * SCALE_FACTOR,
1555
- y: -lastPoint.postureY * SCALE_FACTOR
1905
+ x: lastPoint.postureX,
1906
+ y: lastPoint.postureY
1556
1907
  },
1557
1908
  currentCoord
1558
1909
  ];
@@ -1772,6 +2123,136 @@ function calculateMapGpsCenter(mapData) {
1772
2123
  };
1773
2124
  }
1774
2125
 
2126
+ /**
2127
+ * 并查集(Union-Find)是一种非常高效的数据结构,用于处理动态连通性问题。
2128
+ * 它可以快速判断网络中任意两点是否连通,并能将不连通的集合合并。
2129
+ */
2130
+ class UnionFind {
2131
+ /**
2132
+ * 构造函数,n为图的节点总数
2133
+ * @param {number} n - 节点总数
2134
+ */
2135
+ constructor(n) {
2136
+ this.count = n; // 连通分量的数量
2137
+ this.parent = new Array(n); // parent[i]表示第i个元素所指向的父节点
2138
+ // 初始时,每个节点的父节点是自己
2139
+ for (let i = 0; i < n; i++) {
2140
+ this.parent[i] = i;
2141
+ }
2142
+ }
2143
+ /**
2144
+ * 查找元素p所对应的集合编号(根节点)
2145
+ * @param {number} p - 要查找的元素
2146
+ * @returns {number} 根节点的编号
2147
+ */
2148
+ find(p) {
2149
+ while (p !== this.parent[p]) {
2150
+ this.parent[p] = this.parent[this.parent[p]]; // 路径压缩
2151
+ p = this.parent[p];
2152
+ }
2153
+ return p;
2154
+ }
2155
+ /**
2156
+ * 判断元素p和元素q是否属于同一集合
2157
+ * @param {number} p - 第一个元素
2158
+ * @param {number} q - 第二个元素
2159
+ * @returns {boolean} 是否连通
2160
+ */
2161
+ isConnected(p, q) {
2162
+ return this.find(p) === this.find(q);
2163
+ }
2164
+ /**
2165
+ * 合并元素p和元素q所属的集合
2166
+ * @param {number} p - 第一个元素
2167
+ * @param {number} q - 第二个元素
2168
+ */
2169
+ union(p, q) {
2170
+ const rootP = this.find(p);
2171
+ const rootQ = this.find(q);
2172
+ if (rootP === rootQ) {
2173
+ return; // 已经在同一个集合中
2174
+ }
2175
+ // 将较小的根节点作为父节点(按秩合并的简化版本)
2176
+ if (rootP < rootQ) {
2177
+ this.parent[rootQ] = rootP;
2178
+ }
2179
+ else {
2180
+ this.parent[rootP] = rootQ;
2181
+ }
2182
+ // 两个集合合并成一个集合,连通分量减1
2183
+ this.count--;
2184
+ }
2185
+ /**
2186
+ * 获取当前的连通分量个数
2187
+ * @returns {number} 连通分量数量
2188
+ */
2189
+ getCount() {
2190
+ return this.count;
2191
+ }
2192
+ /**
2193
+ * 获取联通的组
2194
+ * @param {Array} list - 原始元素列表
2195
+ * @returns {Array<Set>} 联通组列表
2196
+ */
2197
+ getConnectedGroup(list) {
2198
+ if (!list || list.length === 0 || !this.parent || this.parent.length === 0) {
2199
+ return null;
2200
+ }
2201
+ if (list.length !== this.parent.length) {
2202
+ return null;
2203
+ }
2204
+ const map = new Map();
2205
+ // 遍历所有元素,按根节点分组
2206
+ for (let i = 0; i < this.parent.length; i++) {
2207
+ const root = this.parent[i];
2208
+ if (!map.has(root)) {
2209
+ map.set(root, new Set());
2210
+ }
2211
+ map.get(root).add(list[i]);
2212
+ }
2213
+ return Array.from(map.values());
2214
+ }
2215
+ /**
2216
+ * 重置并查集
2217
+ * @param {number} n - 新的节点总数
2218
+ */
2219
+ reset(n) {
2220
+ this.count = n;
2221
+ this.parent = new Array(n);
2222
+ for (let i = 0; i < n; i++) {
2223
+ this.parent[i] = i;
2224
+ }
2225
+ }
2226
+ }
2227
+
2228
+ function isTunnelConnected(a, b, connectIds) {
2229
+ if (!a || !b)
2230
+ return false;
2231
+ if (!connectIds || connectIds?.length === 0)
2232
+ return false;
2233
+ const temp = [a?.id, b?.id];
2234
+ temp.sort();
2235
+ return connectIds?.includes(temp?.join('-'));
2236
+ }
2237
+ function isOverlayConnected(a, b) {
2238
+ if (!a || !b) {
2239
+ return false;
2240
+ }
2241
+ if (!a?.points?.length || !b?.points?.length) {
2242
+ return false;
2243
+ }
2244
+ const aPoints = a?.points?.map(item => ({ x: item[0], y: item[1] }));
2245
+ const bPoints = b?.points?.map(item => ({ x: item[0], y: item[1] }));
2246
+ try {
2247
+ if (isOutsideToEachOther(aPoints, bPoints)) {
2248
+ return false;
2249
+ }
2250
+ }
2251
+ catch (error) {
2252
+ console.log('error->', error);
2253
+ }
2254
+ return true;
2255
+ }
1775
2256
  /**
1776
2257
  * 通过 mapData 和 pathData 生成所有 boundary 的数据
1777
2258
  * @param mapData 地图数据
@@ -1780,41 +2261,21 @@ function calculateMapGpsCenter(mapData) {
1780
2261
  */
1781
2262
  function generateBoundaryData(mapData, pathData) {
1782
2263
  const boundaryData = [];
2264
+ let chargingPileBoundary = undefined;
1783
2265
  if (!mapData || !mapData.sub_maps) {
1784
2266
  return boundaryData;
1785
2267
  }
1786
2268
  // 第一步:收集所有TUNNEL数据的connection信息
1787
- const connectedBoundaryIds = new Set();
1788
- // 1.1 遍历子地图中的TUNNEL元素
1789
- for (const subMap of mapData.sub_maps) {
1790
- if (!subMap.elements)
1791
- continue;
1792
- // 找到该子地图中所有 type 为 TUNNEL 的元素
1793
- const tunnelElements = subMap.elements.filter(element => element.type === 'TUNNEL');
1794
- for (const tunnelElement of tunnelElements) {
1795
- const connection = tunnelElement.connection;
1796
- if (connection) {
1797
- // connection可能是单个数字或数组
1798
- if (Array.isArray(connection)) {
1799
- connection.forEach(id => connectedBoundaryIds.add(id));
1800
- }
1801
- else if (typeof connection === 'number') {
1802
- connectedBoundaryIds.add(connection);
1803
- }
1804
- }
1805
- }
1806
- }
1807
- // 1.2 遍历mapData中的tunnels字段
2269
+ const connectIds = [];
2270
+ // 遍历mapData中的tunnels字段
1808
2271
  if (mapData.tunnels && Array.isArray(mapData.tunnels)) {
1809
2272
  for (const tunnel of mapData.tunnels) {
1810
2273
  const connection = tunnel.connection;
1811
2274
  if (connection) {
1812
2275
  // connection可能是单个数字或数组
1813
2276
  if (Array.isArray(connection)) {
1814
- connection.forEach(id => connectedBoundaryIds.add(id));
1815
- }
1816
- else if (typeof connection === 'number') {
1817
- connectedBoundaryIds.add(connection);
2277
+ connection.sort();
2278
+ connectIds.push(connection.join('-'));
1818
2279
  }
1819
2280
  }
1820
2281
  }
@@ -1825,7 +2286,9 @@ function generateBoundaryData(mapData, pathData) {
1825
2286
  if (!subMap.elements)
1826
2287
  continue;
1827
2288
  // 每个sub_map的elements是边界坐标,没有sub_map只有一个boundary数据
1828
- const boundaryElement = subMap.elements.find(element => element.type === 'BOUNDARY');
2289
+ const boundaryElement = subMap.elements.find((element) => element.type === 'BOUNDARY');
2290
+ // 如果当前subMap存在充电桩且充电桩存在tunnel,说明当前subMap中的boundary是初始boundary,这个boundary不为孤立区域
2291
+ const hasTunnelToChargingPile = subMap.elements.some((element) => element.type === 'CHARGING_PILE' && element.tunnel);
1829
2292
  // 创建基础的 boundary 数据(来自 mapData)
1830
2293
  const boundary = {
1831
2294
  // 从 BOUNDARY 元素复制属性
@@ -1834,8 +2297,6 @@ function generateBoundaryData(mapData, pathData) {
1834
2297
  area: subMap?.area,
1835
2298
  points: convertPointsFormat(boundaryElement?.points) || [],
1836
2299
  type: boundaryElement.type,
1837
- // 判断是否为孤立子区域
1838
- isIsolated: !connectedBoundaryIds.has(boundaryElement.id)
1839
2300
  };
1840
2301
  // 如果有 pathData,尝试匹配对应的分区数据
1841
2302
  if (pathData) {
@@ -1851,185 +2312,43 @@ function generateBoundaryData(mapData, pathData) {
1851
2312
  boundary.endTime = partitionData.endTime;
1852
2313
  }
1853
2314
  }
2315
+ if (hasTunnelToChargingPile) {
2316
+ chargingPileBoundary = boundary;
2317
+ }
1854
2318
  boundaryData.push(boundary);
1855
2319
  }
1856
- return boundaryData;
1857
- }
1858
-
1859
- /**
1860
- * 数据格式化转换类 - TypeScript版本
1861
- * 对应Java FormatUtils.java中的bytes2Int函数
1862
- */
1863
- class FormatUtils {
1864
- /**
1865
- * 从data数组的第0位开始取4个字节组成一个int型数
1866
- * 对应Java: public static int bytes2Int(byte[] data)
1867
- *
1868
- * @param data 源字节数组
1869
- * @returns 返回int型数
1870
- */
1871
- // public static bytes2Int(data: Uint8Array): number {
1872
- // return FormatUtils.bytes2IntWithOffset(data, 0, false);
1873
- // }
1874
- /**
1875
- * 从data数组的第position位置开始取4个字节组成int型数
1876
- * 对应Java: public static int bytes2Int(byte[] data, int offset, boolean bigEndian)
1877
- *
1878
- * @param data 源字节数组
1879
- * @param offset 要组成int型数据的起始位置
1880
- * @param bigEndian 是否为大端
1881
- * @returns 返回int型数
1882
- */
1883
- static bytes2Int(data, offset, bigEndian) {
1884
- return FormatUtils.bytes2IntWidthLength(data, offset, bigEndian, 4);
1885
- }
1886
- /**
1887
- * 从data数组的第position位置开始取几个字节组成int型数
1888
- * 对应Java: public static int bytes2Int(byte[] data, int offset, boolean bigEndian, int length)
1889
- *
1890
- * @param data 源字节数组
1891
- * @param offset 要组成int型数据的起始位置
1892
- * @param bigEndian 是否为大端
1893
- * @param length 取的字节长度
1894
- * @returns 返回int型数
1895
- */
1896
- static bytes2IntWidthLength(data, offset, bigEndian, length) {
1897
- if (!data || offset < 0 || offset > data.length) {
1898
- return 0;
1899
- }
1900
- let result = 0;
1901
- try {
1902
- // 创建DataView来处理字节序
1903
- const buffer = new ArrayBuffer(length);
1904
- const view = new DataView(buffer);
1905
- // 将指定长度的数据复制到buffer中
1906
- const tempArray = new Uint8Array(buffer);
1907
- for (let i = 0; i < length && offset + i < data.length; i++) {
1908
- tempArray[i] = data[offset + i];
1909
- }
1910
- // 根据字节序读取数据(使用有符号整数,与Java保持一致)
1911
- if (length === 1) {
1912
- result = view.getInt8(0);
1913
- }
1914
- else if (length === 2) {
1915
- result = view.getInt16(0, !bigEndian); // DataView的littleEndian参数与bigEndian相反
1916
- }
1917
- else if (length === 4) {
1918
- result = view.getInt32(0, !bigEndian); // 改为 getInt32,返回有符号整数
1919
- }
1920
- else {
1921
- // 对于其他长度,手动处理
1922
- result = FormatUtils.manualBytes2Int(data, offset, bigEndian, length);
2320
+ const unionFind = new UnionFind(boundaryData?.length);
2321
+ for (let i = 0; i < boundaryData?.length - 1; i++) {
2322
+ for (let j = i + 1; j < boundaryData?.length; j++) {
2323
+ const boundary1 = boundaryData[i];
2324
+ const boundary2 = boundaryData[j];
2325
+ const isChannelConnect = isTunnelConnected(boundary1, boundary2, connectIds);
2326
+ const isOverlayConnect = isOverlayConnected(boundary1, boundary2);
2327
+ if (isChannelConnect || isOverlayConnect) {
2328
+ unionFind.union(i, j);
1923
2329
  }
1924
2330
  }
1925
- catch (e) {
1926
- // console.error(
1927
- // `${FormatUtils.TAG}: bytes2Int: Exception : data = ${data}, offset = ${offset}`,
1928
- // e
1929
- // );
1930
- // console.log(
1931
- // `${FormatUtils.TAG}: bytes2Int: Exception = ${e instanceof Error ? e.message : e}`
1932
- // );
1933
- }
1934
- return result;
1935
2331
  }
1936
- /**
1937
- * 手动处理字节到整数的转换(用于非标准长度)
1938
- *
1939
- * @param data 源字节数组
1940
- * @param offset 起始位置
1941
- * @param bigEndian 是否为大端
1942
- * @param length 字节长度
1943
- * @returns 转换后的整数
1944
- */
1945
- static manualBytes2Int(data, offset, bigEndian, length) {
1946
- let result = 0;
1947
- if (bigEndian) {
1948
- // 大端序:最高位字节在最低地址
1949
- for (let i = 0; i < length && offset + i < data.length; i++) {
1950
- result = (result << 8) | (data[offset + i] & 0xff);
1951
- }
2332
+ const tunnelAndOverlayList = unionFind.getConnectedGroup(boundaryData);
2333
+ const chargingPileConnectBoundarys = tunnelAndOverlayList?.find(item => item?.has(chargingPileBoundary));
2334
+ for (let boundary of boundaryData) {
2335
+ if (chargingPileConnectBoundarys?.has(boundary)) {
2336
+ boundary.isIsolated = false;
1952
2337
  }
1953
2338
  else {
1954
- // 小端序:最低位字节在最低地址
1955
- for (let i = length - 1; i >= 0; i--) {
1956
- if (offset + i < data.length) {
1957
- result = (result << 8) | (data[offset + i] & 0xff);
1958
- }
1959
- }
1960
- }
1961
- return result;
1962
- }
1963
- /**
1964
- * 将字节数组转换为十六进制字符串
1965
- * 对应Java: public static String bytesToHex(byte[] bytes)
1966
- *
1967
- * @param bytes 字节数组
1968
- * @returns 十六进制字符串
1969
- */
1970
- static bytesToHex(bytes) {
1971
- if (!bytes) {
1972
- return 'null';
1973
- }
1974
- const hexArray = '0123456789ABCDEF'.split('');
1975
- const hexChars = [];
1976
- for (let j = 0; j < bytes.length; j++) {
1977
- const v = bytes[j] & 0xff;
1978
- hexChars.push(hexArray[v >>> 4]);
1979
- hexChars.push(hexArray[v & 0x0f]);
1980
- }
1981
- return hexChars.join('');
1982
- }
1983
- /**
1984
- * 将十六进制字符串转换为字节数组
1985
- * 对应Java: public static byte[] hexStringToByteArray(String s)
1986
- *
1987
- * @param s 十六进制字符串
1988
- * @returns 字节数组
1989
- */
1990
- static hexStringToByteArray(s) {
1991
- if (!s || s.length <= 0) {
1992
- return null;
1993
- }
1994
- s = s.trim();
1995
- const len = s.length;
1996
- const data = new Uint8Array(len / 2);
1997
- for (let i = 0; i < len - 1; i += 2) {
1998
- try {
1999
- const high = parseInt(s.charAt(i), 16);
2000
- const low = parseInt(s.charAt(i + 1), 16);
2001
- data[i / 2] = (high << 4) + low;
2002
- }
2003
- catch (e) {
2004
- console.log('hexStringToByteArray: ' + e);
2005
- }
2339
+ boundary.isIsolated = true;
2006
2340
  }
2007
- return data;
2008
2341
  }
2342
+ return boundaryData;
2009
2343
  }
2010
- FormatUtils.TAG = 'FormatUtils';
2011
- // 使用示例
2012
- // export function example() {
2013
- // // 示例1:从字节数组开头读取4字节整数(小端序)
2014
- // const data1 = new Uint8Array([0x12, 0x34, 0x56, 0x78]);
2015
- // const result1 = FormatUtils.bytes2Int(data1); // 应该返回 0x78563412
2016
- // // 示例2:从指定位置读取4字节整数(大端序)
2017
- // const data2 = new Uint8Array([0x00, 0x00, 0x12, 0x34, 0x56, 0x78]);
2018
- // const result2 = FormatUtils.bytes2Int(data2, 2, true); // 应该返回 0x12345678
2019
- // // 示例3:读取2字节整数
2020
- // const data3 = new Uint8Array([0x12, 0x34]);
2021
- // const result3 = FormatUtils.bytes2Int(data3, 0, false, 2); // 应该返回 0x3412
2022
- // console.log(`Result1: 0x${result1.toString(16)}`);
2023
- // console.log(`Result2: 0x${result2.toString(16)}`);
2024
- // console.log(`Result3: 0x${result3.toString(16)}`);
2025
- // // 示例4:十六进制字符串转换
2026
- // const hexString = "12345678";
2027
- // const byteArray = FormatUtils.hexStringToByteArray(hexString);
2028
- // const backToHex = FormatUtils.bytesToHex(byteArray);
2029
- // console.log(`Original: ${hexString}`);
2030
- // console.log(`To bytes: ${byteArray}`);
2031
- // console.log(`Back to hex: ${backToHex}`);
2032
- // }
2344
+
2345
+ var RealTimeDataType;
2346
+ (function (RealTimeDataType) {
2347
+ RealTimeDataType[RealTimeDataType["LOCATION"] = 1] = "LOCATION";
2348
+ RealTimeDataType[RealTimeDataType["PROCESS"] = 2] = "PROCESS";
2349
+ RealTimeDataType[RealTimeDataType["PARTITION"] = 3] = "PARTITION";
2350
+ RealTimeDataType[RealTimeDataType["STATUS"] = 4] = "STATUS";
2351
+ })(RealTimeDataType || (RealTimeDataType = {}));
2033
2352
 
2034
2353
  /**
2035
2354
  * 射线法判断点是否在多边形内部
@@ -2125,50 +2444,6 @@ const getPartitionId = (partitionBoundary, postureX, postureY) => {
2125
2444
  })?.id;
2126
2445
  return partitionId;
2127
2446
  };
2128
- /**
2129
- *
2130
- * 支持分区割草后的解析,加入了分区数量、分区列表
2131
- * 8 割草路径类型
2132
- * 4 割草启动类型 0:自动 1:手动
2133
- * 4 当前割草边界id
2134
- * 4 当前边界割草百分比 比如9800----98%
2135
- * @returns
2136
- */
2137
- const parseMapWorkPosition = (mapWorkPosition) => {
2138
- let isMowing = false;
2139
- let mowStartType = null;
2140
- let currentMowBoundaryId = null;
2141
- let currentMowProgress = null;
2142
- const bytes = new Uint8Array(mapWorkPosition.match(/.{1,2}/g)?.map((byte) => parseInt(byte, 16)) || []);
2143
- if (mapWorkPosition.length >= 8) {
2144
- // 以下两种状态认为是割草中
2145
- // 1. action为8,且subAction为6
2146
- // 2. action为5
2147
- const action = FormatUtils.bytes2Int(bytes, 0, true);
2148
- if (action === ACTION_BOUNDARY_TASK) {
2149
- const subAction = FormatUtils.bytes2Int(bytes, 4, true);
2150
- if (subAction === ACTION_BLOCK_TRANSFER) {
2151
- // action=8且subAction=6:边界任务中的块转移
2152
- isMowing = true;
2153
- }
2154
- }
2155
- else if (action === ACTION_BLOCK_COVER) {
2156
- // action=5:块覆盖割草
2157
- isMowing = true;
2158
- }
2159
- if (mapWorkPosition.length >= 16) {
2160
- mowStartType = FormatUtils.bytes2Int(bytes, 8, true);
2161
- currentMowBoundaryId = FormatUtils.bytes2Int(bytes, 12, true);
2162
- currentMowProgress = FormatUtils.bytes2Int(bytes, 16, true);
2163
- }
2164
- }
2165
- return {
2166
- isMowing,
2167
- mowStartType,
2168
- currentMowBoundaryId,
2169
- currentMowProgress,
2170
- };
2171
- };
2172
2447
  /**
2173
2448
  * 处理实时数据的消息,这里的实时数据消息有两种,一种是实时轨迹,一种是割草进度,其中这两种下发的时间频次不一样
2174
2449
  * 实时轨迹的路径需要依靠割草进度时候的割草状态判断,目前只能根据上一次获取到的割草进度的状态来处理,如果一开始没有割草的状态,则默认为不割草,后续会根据割草进度来更新
@@ -2176,7 +2451,7 @@ const parseMapWorkPosition = (mapWorkPosition) => {
2176
2451
  * @param param0
2177
2452
  * @returns
2178
2453
  */
2179
- const handleRealTimeData = ({ realTimeData, isMowing, pathData, partitionBoundary, }) => {
2454
+ const handleMultipleRealTimeData = ({ realTimeData, isMowing, pathData, partitionBoundary, }) => {
2180
2455
  // 先将数据进行倒排,这样好插入数据
2181
2456
  if (realTimeData.length > 0) {
2182
2457
  realTimeData.reverse();
@@ -2192,7 +2467,6 @@ const handleRealTimeData = ({ realTimeData, isMowing, pathData, partitionBoundar
2192
2467
  // 割草轨迹
2193
2468
  const { postureX, postureY, vehicleState } = item;
2194
2469
  const currentPartitionId = getPartitionId(partitionBoundary, Number(postureX), Number(postureY));
2195
- console.log('currentPartitionId===', currentPartitionId);
2196
2470
  if (currentPartitionId && newPathData?.[currentPartitionId]) {
2197
2471
  const currentPathData = newPathData[currentPartitionId];
2198
2472
  newPathData[currentPartitionId] = {
@@ -2204,7 +2478,7 @@ const handleRealTimeData = ({ realTimeData, isMowing, pathData, partitionBoundar
2204
2478
  postureY: Number(postureY),
2205
2479
  knifeRotation: mowingStatus && vehicleState === RobotStatus.MOWING ? '01' : '00', // "knifeRotation": "01",//刀盘是否转动 00-否 01-是
2206
2480
  // knifeRotation: '01', // "knifeRotation": "01",//刀盘是否转动 00-否 01-是
2207
- pathType: '01', //"pathType": "01",//路径类型 : 00-巡边 01-弓字型割草 02-地图测试 03-转移路径 04-避障路径 05-恢复/脱困路径
2481
+ pathType: '', //"pathType": "01",//路径类型 : 00-巡边 01-弓字型割草 02-地图测试 03-转移路径 04-避障路径 05-恢复/脱困路径
2208
2482
  partitionId: currentPartitionId.toString(), // TODO:不知道为什么这里的id需要是字符串类型?
2209
2483
  },
2210
2484
  ],
@@ -2213,21 +2487,20 @@ const handleRealTimeData = ({ realTimeData, isMowing, pathData, partitionBoundar
2213
2487
  }
2214
2488
  else if (item.type === REAL_TIME_DATA_TYPE.PROCESS) {
2215
2489
  // 割草进度
2216
- const { mapWorkPosition } = item;
2217
- if (mapWorkPosition) {
2218
- // 解析mapWorkPosition数据,获取到当前的割草状态,割草进度,割草边界id
2219
- const { isMowing, currentMowBoundaryId, currentMowProgress } = parseMapWorkPosition(mapWorkPosition);
2220
- // const { isMowing, currentMowBoundaryId, currentMowProgress } = parseMapWorkPosition(
2221
- // '000000060000000000000001000000030000001BA5A5A5A5A5A5A5A5240000A024010D50FFFFFFED000000000000000000000000000000010000000000000000'
2222
- // );
2223
- console.log('isMowing===', isMowing, currentMowBoundaryId, currentMowProgress);
2224
- mowingStatus = isMowing;
2225
- const currentPartitionId = currentMowBoundaryId ? currentMowBoundaryId.toString() : null;
2226
- if (currentMowProgress && currentPartitionId && newPathData?.[currentPartitionId]) {
2227
- newPathData[currentPartitionId].partitionPercentage = currentMowProgress / 100;
2228
- newPathData[currentPartitionId].finishedArea =
2229
- (newPathData[currentPartitionId].area * currentMowProgress) / 10000;
2230
- }
2490
+ const { action, subAction, currentMowBoundary, currentMowProgress } = item;
2491
+ // 设置状态
2492
+ if ((action === ACTION_BOUNDARY_TASK && subAction && subAction === ACTION_BLOCK_TRANSFER) ||
2493
+ action === ACTION_BLOCK_COVER) {
2494
+ mowingStatus = true;
2495
+ }
2496
+ else {
2497
+ mowingStatus = false;
2498
+ }
2499
+ const currentPartitionId = currentMowBoundary ? currentMowBoundary.toString() : null;
2500
+ if (currentMowProgress && currentPartitionId && newPathData?.[currentPartitionId]) {
2501
+ newPathData[currentPartitionId].partitionPercentage = currentMowProgress / 100;
2502
+ newPathData[currentPartitionId].finishedArea =
2503
+ (newPathData[currentPartitionId].area * currentMowProgress) / 10000;
2231
2504
  }
2232
2505
  }
2233
2506
  });
@@ -2236,6 +2509,39 @@ const handleRealTimeData = ({ realTimeData, isMowing, pathData, partitionBoundar
2236
2509
  isMowing: mowingStatus,
2237
2510
  };
2238
2511
  };
2512
+ /**
2513
+ * 根据实时数据,获取到割草状态
2514
+ * @param realTimeData 实时数据
2515
+ * @param isMowing 上一次的割草状态
2516
+ * @returns 新的割草状态
2517
+ */
2518
+ const getProcessMowingDataFromRealTimeData = ({ realTimeData, isMowing, pathData, }) => {
2519
+ let newMowingStatus = isMowing;
2520
+ let newPathData = pathData || {};
2521
+ // 找到返回的第一个实时进度的点
2522
+ const firstProcessData = realTimeData.find((item) => item.type === RealTimeDataType.PROCESS);
2523
+ if (firstProcessData) {
2524
+ // console.log('firstProcessData==', firstProcessData);
2525
+ const { action, subAction, currentMowBoundary, currentMowProgress } = firstProcessData;
2526
+ // 设置状态
2527
+ if ((action === ACTION_BOUNDARY_TASK && subAction && subAction === ACTION_BLOCK_TRANSFER) ||
2528
+ action === ACTION_BLOCK_COVER) {
2529
+ newMowingStatus = true;
2530
+ }
2531
+ else {
2532
+ newMowingStatus = false;
2533
+ }
2534
+ if (currentMowBoundary && newPathData?.[currentMowBoundary]) {
2535
+ newPathData[currentMowBoundary].partitionPercentage = currentMowProgress / 100;
2536
+ newPathData[currentMowBoundary].finishedArea =
2537
+ (newPathData[currentMowBoundary].area * currentMowProgress) / 10000;
2538
+ }
2539
+ }
2540
+ return {
2541
+ isMowing: newMowingStatus,
2542
+ pathData: newPathData,
2543
+ };
2544
+ };
2239
2545
 
2240
2546
  var iMower = "";
2241
2547
 
@@ -2249,13 +2555,15 @@ var hNoPosition = "
2249
2555
 
2250
2556
  var hDisabled = "";
2251
2557
 
2558
+ var x3Edger = "";
2559
+
2252
2560
  var x3Mower = "";
2253
2561
 
2254
2562
  var x3NoPosition = "";
2255
2563
 
2256
2564
  var x3Disabled = "";
2257
2565
 
2258
- function getMowerImageByModal(mowerModal) {
2566
+ function getMowerImageByModal(mowerModal, hasEdger) {
2259
2567
  if (mowerModal.includes('i')) {
2260
2568
  return iMower;
2261
2569
  }
@@ -2263,7 +2571,7 @@ function getMowerImageByModal(mowerModal) {
2263
2571
  return hMower;
2264
2572
  }
2265
2573
  else if (mowerModal.includes('x3')) {
2266
- return x3Mower;
2574
+ return hasEdger ? x3Edger : x3Mower;
2267
2575
  }
2268
2576
  return iMower;
2269
2577
  }
@@ -2291,10 +2599,12 @@ function getNoPositionMowerImageByModal(mowerModal) {
2291
2599
  }
2292
2600
  return iNoPosition;
2293
2601
  }
2294
- function getMowerImage(positonConfig) {
2295
- const model = positonConfig.vehicleModel || '';
2602
+ function getMowerImage(positonConfig, modelType, hasEdger) {
2603
+ if (!positonConfig)
2604
+ return '';
2605
+ const model = modelType?.toLowerCase() || 'i';
2296
2606
  const state = positonConfig.vehicleState;
2297
- const mowerImage = getMowerImageByModal(model);
2607
+ const mowerImage = getMowerImageByModal(model, hasEdger);
2298
2608
  const disabledImage = getDisabledMowerImageByModal(model);
2299
2609
  const noPositionImage = getNoPositionMowerImageByModal(model);
2300
2610
  const positonOutOfRange = isOutOfRange(positonConfig);
@@ -2329,9 +2639,9 @@ function isInvalidPosition(positonConfig) {
2329
2639
  }
2330
2640
  function isOutOfRange(positonConfig) {
2331
2641
  return (positonConfig.postureX != null &&
2332
- Math.abs(positonConfig.postureX) > 1000 * SCALE_FACTOR &&
2642
+ Math.abs(positonConfig.postureX) > 1000 &&
2333
2643
  positonConfig.postureY != null &&
2334
- Math.abs(positonConfig.postureY) > 1000 * SCALE_FACTOR);
2644
+ Math.abs(positonConfig.postureY) > 1000);
2335
2645
  }
2336
2646
 
2337
2647
  /** Detect free variable `global` from Node.js. */
@@ -4397,7 +4707,7 @@ function cloneBuffer(buffer, isDeep) {
4397
4707
  }
4398
4708
 
4399
4709
  /** Built-in value references. */
4400
- var Uint8Array$1 = root.Uint8Array;
4710
+ var Uint8Array = root.Uint8Array;
4401
4711
 
4402
4712
  /**
4403
4713
  * Creates a clone of `arrayBuffer`.
@@ -4408,7 +4718,7 @@ var Uint8Array$1 = root.Uint8Array;
4408
4718
  */
4409
4719
  function cloneArrayBuffer(arrayBuffer) {
4410
4720
  var result = new arrayBuffer.constructor(arrayBuffer.byteLength);
4411
- new Uint8Array$1(result).set(new Uint8Array$1(arrayBuffer));
4721
+ new Uint8Array(result).set(new Uint8Array(arrayBuffer));
4412
4722
  return result;
4413
4723
  }
4414
4724
 
@@ -4759,6 +5069,36 @@ var merge = createAssigner(function(object, source, srcIndex) {
4759
5069
  */
4760
5070
  var round = createRound('round');
4761
5071
 
5072
+ /**
5073
+ * 工具模块类型定义
5074
+ */
5075
+ /**
5076
+ * 路径段类型枚举
5077
+ */
5078
+ var PathSegmentType;
5079
+ (function (PathSegmentType) {
5080
+ PathSegmentType["EDGE"] = "edge";
5081
+ PathSegmentType["MOWING"] = "mowing";
5082
+ PathSegmentType["TRANS"] = "trans";
5083
+ })(PathSegmentType || (PathSegmentType = {}));
5084
+ /**
5085
+ * 单位类型枚举
5086
+ */
5087
+ var UnitsType;
5088
+ (function (UnitsType) {
5089
+ UnitsType["Metric"] = "Metric";
5090
+ UnitsType["Imperial"] = "Imperial";
5091
+ })(UnitsType || (UnitsType = {}));
5092
+ /**
5093
+ * 面积单位类型枚举
5094
+ */
5095
+ var UnitsAreaType;
5096
+ (function (UnitsAreaType) {
5097
+ UnitsAreaType["SQUARE_METER"] = "m\u00B2";
5098
+ UnitsAreaType["SQUARE_FOOT"] = "ft\u00B2";
5099
+ UnitsAreaType["ACRE"] = "ac";
5100
+ })(UnitsAreaType || (UnitsAreaType = {}));
5101
+
4762
5102
  /**
4763
5103
  * 默认航向相对于canvas的偏移角度: 航向默认是东
4764
5104
  */
@@ -4786,10 +5126,6 @@ function radToDegree(radian) {
4786
5126
  function distance(x1, y1, x2, y2) {
4787
5127
  return Math.sqrt(Math.pow(x1 - x2, 2) + Math.pow(y1 - y2, 2));
4788
5128
  }
4789
- // 计算前进方向和车头的夹角
4790
- function calAngle(x1, y1, x2, y2) {
4791
- return Math.atan2((y2 - y1), (x2 - x1));
4792
- }
4793
5129
  const mathRound = (value, decimals = 2) => {
4794
5130
  return Number.isInteger(value) ? value : round(value, decimals);
4795
5131
  };
@@ -4836,17 +5172,6 @@ function formatNumberWithMetricPrefix(value, round = true, decimals = 2) {
4836
5172
  return `${mathFn(value / 1000000000, decimals)}B`;
4837
5173
  }
4838
5174
  }
4839
- var UnitsType;
4840
- (function (UnitsType) {
4841
- UnitsType["Metric"] = "metric";
4842
- UnitsType["Imperial"] = "imperial";
4843
- })(UnitsType || (UnitsType = {}));
4844
- var UnitsAreaType;
4845
- (function (UnitsAreaType) {
4846
- UnitsAreaType["SQUARE_METER"] = "m\u00B2";
4847
- UnitsAreaType["SQUARE_FOOT"] = "ft\u00B2";
4848
- UnitsAreaType["ACRE"] = "ac";
4849
- })(UnitsAreaType || (UnitsAreaType = {}));
4850
5175
  /**
4851
5176
  * 转换割草面积的方法
4852
5177
  * @param area 面积数值(单位:m²)
@@ -5035,6 +5360,25 @@ class BoundaryBorderLayer extends BaseLayer {
5035
5360
  this.scale = 1;
5036
5361
  this.level = LAYER_LEVELS.BOUNDARY_BORDER; // 中等层级
5037
5362
  this.type = LAYER_DEFAULT_TYPE.BOUNDARY_BORDER;
5363
+ this.boudaryBorderPaths = {};
5364
+ this.mowingBoundarys = [];
5365
+ }
5366
+ /**
5367
+ * 设置当前割草任务的边界
5368
+ */
5369
+ setMowingBoundarys(mowingBoundarys) {
5370
+ if (!mowingBoundarys) {
5371
+ this.mowingBoundarys = this.elements?.map(item => item?.originalData?.id);
5372
+ }
5373
+ else {
5374
+ this.mowingBoundarys = mowingBoundarys;
5375
+ }
5376
+ }
5377
+ /**
5378
+ * 获取当前割草任务的边界
5379
+ */
5380
+ getMowingBoundarys() {
5381
+ return this.mowingBoundarys;
5038
5382
  }
5039
5383
  /**
5040
5384
  * SVG渲染方法
@@ -5044,18 +5388,38 @@ class BoundaryBorderLayer extends BaseLayer {
5044
5388
  return;
5045
5389
  }
5046
5390
  this.scale = scale;
5047
- // 只渲染边界边框类型的元素
5391
+ // 将元素分为两组:非割草边界和割草边界
5392
+ const nonMowingElements = [];
5393
+ const mowingElements = [];
5394
+ // 只处理边界边框类型的元素
5048
5395
  for (const element of this.elements) {
5049
5396
  if (element.type === 'boundary_border') {
5050
- this.renderBoundaryBorder(svgGroup, element);
5397
+ const { originalData } = element;
5398
+ const { id } = originalData || {};
5399
+ // 检查是否为割草边界
5400
+ if (this.mowingBoundarys.includes(Number(id))) {
5401
+ mowingElements.push(element);
5402
+ }
5403
+ else {
5404
+ nonMowingElements.push(element);
5405
+ }
5051
5406
  }
5052
5407
  }
5408
+ // 先渲染非割草边界
5409
+ for (const element of nonMowingElements) {
5410
+ this.renderBoundaryBorder(svgGroup, element);
5411
+ }
5412
+ // 再渲染割草边界(放在最后)
5413
+ for (const element of mowingElements) {
5414
+ this.renderBoundaryBorder(svgGroup, element);
5415
+ }
5053
5416
  }
5054
5417
  /**
5055
5418
  * 渲染边界边框
5056
5419
  */
5057
5420
  renderBoundaryBorder(svgGroup, element) {
5058
- const { coordinates, style } = element;
5421
+ const { coordinates, style, originalData } = element;
5422
+ const { id } = originalData || {};
5059
5423
  if (coordinates.length < 2)
5060
5424
  return;
5061
5425
  // 1. 先遍历所有的coordinates,把所有点分为若干段的path
@@ -5066,20 +5430,21 @@ class BoundaryBorderLayer extends BaseLayer {
5066
5430
  return;
5067
5431
  if (segment.type === 2) {
5068
5432
  // type=2: 直接添加到svgGroup中
5069
- this.createDirectPath(svgGroup, segment.points, style);
5433
+ this.createDirectPath(svgGroup, segment.points, style, id);
5070
5434
  }
5071
5435
  else if (segment.type === 1) {
5072
5436
  // type=1: 使用PathMeasure逻辑生成平行路径
5073
5437
  // this.createDirectPath(svgGroup, segment.points, style);
5074
- this.createParallelPathsWithMeasure(svgGroup, segment.points, style);
5438
+ this.createParallelPathsWithMeasure(svgGroup, segment.points, style, id);
5075
5439
  }
5076
5440
  });
5077
5441
  }
5078
5442
  /**
5079
5443
  * 创建直接路径(type=2)
5080
5444
  */
5081
- createDirectPath(svgGroup, points, style) {
5082
- const strokeColor = style.lineColor;
5445
+ createDirectPath(svgGroup, points, style, id) {
5446
+ const isMowing = this.mowingBoundarys.includes(Number(id));
5447
+ const strokeColor = isMowing ? style.mowingLineColor : style.lineColor;
5083
5448
  const lineWidth = dp2px(style.lineWidth || 3);
5084
5449
  const path = document.createElementNS('http://www.w3.org/2000/svg', 'path');
5085
5450
  // 构建路径数据
@@ -5102,13 +5467,18 @@ class BoundaryBorderLayer extends BaseLayer {
5102
5467
  path.setAttribute('opacity', (style.opacity || 1).toString());
5103
5468
  path.setAttribute('vector-effect', 'non-scaling-stroke');
5104
5469
  path.classList.add('vector-boundary-solid');
5470
+ if (!this.boudaryBorderPaths[id]) {
5471
+ this.boudaryBorderPaths[id] = [];
5472
+ }
5473
+ this.boudaryBorderPaths[id].push(path);
5105
5474
  svgGroup.appendChild(path);
5106
5475
  }
5107
5476
  /**
5108
5477
  * 使用PathMeasure逻辑创建平行路径(type=1)
5109
5478
  */
5110
- createParallelPathsWithMeasure(svgGroup, points, style) {
5111
- const strokeColor = style.lineColor;
5479
+ createParallelPathsWithMeasure(svgGroup, points, style, id) {
5480
+ const isMowing = this.mowingBoundarys.includes(Number(id));
5481
+ const strokeColor = isMowing ? style.mowingLineColor : style.lineColor;
5112
5482
  const lineWidth = dp2px(style.lineWidth || 3);
5113
5483
  // 获取当前SVG的缩放级别,计算固定屏幕像素间距
5114
5484
  const fixedScreenDistance = lineWidth; // 固定的屏幕像素距离
@@ -5131,6 +5501,10 @@ class BoundaryBorderLayer extends BaseLayer {
5131
5501
  // 或者可以根据当前缩放级别动态计算dash array
5132
5502
  path.style.strokeDasharray = `${lineWidth}px ${lineWidth * 2}px`;
5133
5503
  path.classList.add(`vector-boundary-parallel-${index + 1}`);
5504
+ if (!this.boudaryBorderPaths[id]) {
5505
+ this.boudaryBorderPaths[id] = [];
5506
+ }
5507
+ this.boudaryBorderPaths[id].push(path);
5134
5508
  svgGroup.appendChild(path);
5135
5509
  });
5136
5510
  }
@@ -5220,6 +5594,9 @@ class BoundaryBorderLayer extends BaseLayer {
5220
5594
  }
5221
5595
  return segments;
5222
5596
  }
5597
+ resetPaths() {
5598
+ this.boudaryBorderPaths = {};
5599
+ }
5223
5600
  }
5224
5601
 
5225
5602
  var antennaOneOnline = "";
@@ -5432,7 +5809,7 @@ class DrawLayer extends BaseLayer {
5432
5809
  this.pointLayer,
5433
5810
  this.svgElementLayer,
5434
5811
  this.visionOffLayer,
5435
- ];
5812
+ ]?.filter((layer) => layer.getElements().length > 0);
5436
5813
  }
5437
5814
  getPathLayers() {
5438
5815
  return this.pathLayer;
@@ -5447,6 +5824,14 @@ class BoundaryDataBuilder {
5447
5824
  * 创建边界元素数据
5448
5825
  */
5449
5826
  static create(type, coordinates, style) {
5827
+ const len = coordinates?.length || 0;
5828
+ const firstPoint = coordinates?.[0];
5829
+ const lastPoint = coordinates?.[len - 1];
5830
+ const isClosed = firstPoint?.[0] === lastPoint?.[0] && firstPoint?.[1] === lastPoint?.[1];
5831
+ // 如果地图没有闭合,则手动新增闭合点,避免border最后一部分没有闭合的情况
5832
+ if (!isClosed) {
5833
+ coordinates.push([firstPoint?.[0], firstPoint?.[1], lastPoint?.[2]]);
5834
+ }
5450
5835
  return {
5451
5836
  type,
5452
5837
  coordinates,
@@ -5761,7 +6146,6 @@ class MapDataProcessor {
5761
6146
  if (mapData.vision_off_areas && mapData.vision_off_areas.length > 0) {
5762
6147
  allElements.push(...mapData.vision_off_areas);
5763
6148
  }
5764
- console.log('allElements', allElements);
5765
6149
  // 按照元素类型分组并设置不同的层级
5766
6150
  return this.createLayeredMapData(allElements);
5767
6151
  }
@@ -5850,8 +6234,13 @@ class MapDataProcessor {
5850
6234
  // 为ObstacleData创建兼容的MapElement接口
5851
6235
  const mapElement = element;
5852
6236
  const obstacleElement = ObstacleDataBuilder.fromMapElement(mapElement, this.mapConfig.obstacle);
5853
- if (obstacleElement)
6237
+ if (obstacleElement) {
5854
6238
  result.push(obstacleElement);
6239
+ const { addObstacles } = useSubBoundaryBorderStore.getState();
6240
+ addObstacles(`obstacle-${obstacleElement.originalData.id}`, {
6241
+ ...obstacleElement,
6242
+ });
6243
+ }
5855
6244
  }
5856
6245
  catch (error) {
5857
6246
  console.warn(`Error processing OBSTACLE element:`, element, error);
@@ -5912,9 +6301,13 @@ class MapDataProcessor {
5912
6301
  element.direction !== undefined) {
5913
6302
  const mapElement = element;
5914
6303
  const svgElement = SvgElementDataBuilder.fromMapElement(mapElement, this.mapConfig.doodle);
5915
- console.log('svgElement==', svgElement);
5916
- if (svgElement)
6304
+ if (svgElement) {
5917
6305
  result.push(svgElement);
6306
+ const { addSvgElements } = useSubBoundaryBorderStore.getState();
6307
+ addSvgElements(`time-limit-obstacle-${svgElement.originalData.id}`, {
6308
+ ...svgElement,
6309
+ });
6310
+ }
5918
6311
  }
5919
6312
  // 如果有points数据,按传统方式绘制
5920
6313
  else if ('points' in element &&
@@ -5923,8 +6316,13 @@ class MapDataProcessor {
5923
6316
  element.points.length >= 3) {
5924
6317
  const mapElement = element;
5925
6318
  const polygonElement = ObstacleDataBuilder.createTimeLimitObstacle(mapElement, this.mapConfig.obstacle);
5926
- if (polygonElement)
6319
+ if (polygonElement) {
5927
6320
  result.push(polygonElement);
6321
+ const { addObstacles } = useSubBoundaryBorderStore.getState();
6322
+ addObstacles(`time-limit-obstacle-${polygonElement.originalData.id}`, {
6323
+ ...polygonElement,
6324
+ });
6325
+ }
5928
6326
  }
5929
6327
  }
5930
6328
  catch (error) {
@@ -5969,35 +6367,15 @@ class PathDataBuilder {
5969
6367
  style,
5970
6368
  };
5971
6369
  }
5972
- // /**
5973
- // * 创建隧道元素数据
5974
- // */
5975
- // static createTunnel(element: MapElement): DrawElement | null {
5976
- // const convertedPoints = convertPointsFormat(element.tunnel?.points || element.points);
5977
- // if (!convertedPoints || convertedPoints.length < 2) return null;
5978
- // return this.create(convertedPoints, DEFAULT_STYLES.PATH_EDGE);
5979
- // }
5980
- // /**
5981
- // * 创建充电桩内部隧道路径(用于CHARGING_PILE元素中的tunnel.points)
5982
- // */
5983
- // static createChargingPileTunnelPath(element: MapElement): DrawElement | null {
5984
- // if (!element.tunnel || !element.tunnel.points) return null;
5985
- // const convertedTunnelPoints = convertPointsFormat(element.tunnel.points);
5986
- // if (!convertedTunnelPoints || convertedTunnelPoints.length < 2) return null;
5987
- // return this.create(convertedTunnelPoints, {
5988
- // strokeColor: DEFAULT_STYLES.CHANNEL.strokeColor,
5989
- // lineWidth: 2,
5990
- // opacity: 1.0
5991
- // });
5992
- // }
5993
6370
  /**
5994
6371
  * 创建边缘路径
5995
6372
  */
5996
6373
  static createEdgePath(points, config) {
5997
- const drawElement = this.create(points, {
5998
- strokeColor: config?.color || DEFAULT_STYLES.PATH_EDGE.strokeColor,
5999
- lineWidth: config?.lineWidth || 2,
6000
- opacity: config?.opacity || 1.0,
6374
+ const convertedPoints = convertPointsFormat(points);
6375
+ const drawElement = this.create(convertedPoints, {
6376
+ lineColor: config?.color,
6377
+ lineWidth: config?.lineWidth,
6378
+ opacity: config?.opacity,
6001
6379
  });
6002
6380
  drawElement.originalData = points;
6003
6381
  return drawElement;
@@ -6006,10 +6384,11 @@ class PathDataBuilder {
6006
6384
  * 创建割草路径
6007
6385
  */
6008
6386
  static createMowingPath(points, config) {
6009
- const drawElement = this.create(points, {
6010
- strokeColor: config?.color || DEFAULT_STYLES.PATH_EDGE.strokeColor,
6011
- lineWidth: config?.lineWidth || DEFAULT_STYLES.PATH_EDGE.lineWidth,
6012
- opacity: config?.opacity || 1.0,
6387
+ const convertedPoints = convertPointsFormat(points);
6388
+ const drawElement = this.create(convertedPoints, {
6389
+ lineColor: config?.color,
6390
+ lineWidth: config?.lineWidth,
6391
+ opacity: config?.opacity,
6013
6392
  });
6014
6393
  drawElement.originalData = points;
6015
6394
  return drawElement;
@@ -6018,11 +6397,11 @@ class PathDataBuilder {
6018
6397
  * 创建传输路径
6019
6398
  */
6020
6399
  static createTransPath(points, config) {
6021
- const drawElement = this.create(points, {
6022
- strokeColor: config?.color || DEFAULT_STYLES.PATH_EDGE.strokeColor,
6023
- lineWidth: config?.lineWidth || DEFAULT_STYLES.PATH_EDGE.lineWidth,
6024
- lineDash: [5, 5], // 虚线样式
6025
- opacity: config?.opacity || 1.0,
6400
+ const convertedPoints = convertPointsFormat(points);
6401
+ const drawElement = this.create(convertedPoints, {
6402
+ lineColor: config?.color,
6403
+ lineWidth: config?.lineWidth,
6404
+ opacity: config?.opacity,
6026
6405
  });
6027
6406
  drawElement.originalData = points;
6028
6407
  return drawElement;
@@ -6041,55 +6420,59 @@ class PathDataProcessor {
6041
6420
  if (!pathData || typeof pathData !== 'object') {
6042
6421
  return [];
6043
6422
  }
6044
- // 获取所有分区的路径数据
6045
- const allPathItems = Object.values(pathData).reduce((acc, partitionData) => {
6046
- if (partitionData && partitionData.points && partitionData.points.length > 0) {
6047
- acc.push(...partitionData.points);
6048
- }
6049
- return acc;
6050
- }, []);
6051
6423
  // 合并配置
6052
6424
  const config = mapConfig.path;
6053
- // 使用Python相同的逻辑:按线段分组而不是按点分组
6054
- const pathSegments = createPathSegmentsByType(allPathItems);
6055
- const elements = [];
6056
- // 处理边缘路径段
6057
- for (const segment of pathSegments.edge) {
6058
- // 转换 Point[] 为 number[][]
6059
- const points = segment.points.map((point) => [point.x, point.y]);
6060
- const element = PathDataBuilder.createEdgePath(points, {
6061
- lineWidth: config.lineWidth,
6062
- color: config.edgeLineColor,
6063
- opacity: config.opacity,
6064
- });
6065
- element.originalData = points;
6066
- elements.push(element);
6067
- }
6068
- // 处理割草路径段
6069
- for (const segment of pathSegments.mowing) {
6070
- // 转换 Point[] 为 number[][]
6071
- const points = segment.points.map((point) => [point.x, point.y]);
6072
- const element = PathDataBuilder.createMowingPath(points, {
6073
- lineWidth: config.lineWidth,
6074
- color: config.mowingLineColor,
6075
- opacity: config.opacity,
6076
- });
6077
- element.originalData = points;
6078
- elements.push(element);
6079
- }
6080
- // 处理传输路径段(只有在showTransPaths为true时才添加)
6081
- for (const segment of pathSegments.trans) {
6082
- // 转换 Point[] 为 number[][]
6083
- const points = segment.points.map((point) => [point.x, point.y]);
6084
- const element = PathDataBuilder.createTransPath(points, {
6085
- lineWidth: config.lineWidth,
6086
- color: config.transLineColor,
6087
- opacity: config.opacity,
6088
- });
6089
- element.originalData = points;
6090
- elements.push(element);
6091
- }
6092
- return elements;
6425
+ const result = Object.keys(pathData).map((key) => {
6426
+ const id = Number(key);
6427
+ const points = pathData[key].points || [];
6428
+ // 使用Python相同的逻辑:按线段分组而不是按点分组
6429
+ const pathSegments = createPathSegmentsByType(points);
6430
+ const elements = [];
6431
+ // 处理边缘路径段
6432
+ for (const segment of pathSegments.edge) {
6433
+ // 转换 Point[] 为 number[][]
6434
+ const points = segment.points.map((point) => [point.x, point.y]);
6435
+ const element = PathDataBuilder.createEdgePath(points, {
6436
+ lineWidth: config.lineWidth,
6437
+ color: config.edgeLineColor,
6438
+ opacity: config.opacity,
6439
+ });
6440
+ element.originalData = points;
6441
+ element.pathType = PathSegmentType.EDGE;
6442
+ elements.push(element);
6443
+ }
6444
+ // 处理割草路径段
6445
+ for (const segment of pathSegments.mowing) {
6446
+ // 转换 Point[] 为 number[][]
6447
+ const points = segment.points.map((point) => [point.x, point.y]);
6448
+ const element = PathDataBuilder.createMowingPath(points, {
6449
+ lineWidth: config.lineWidth,
6450
+ color: config.mowedLineColor,
6451
+ opacity: config.opacity,
6452
+ });
6453
+ element.originalData = points;
6454
+ element.pathType = PathSegmentType.MOWING;
6455
+ elements.push(element);
6456
+ }
6457
+ // 处理传输路径段(只有在showTransPaths为true时才添加)
6458
+ for (const segment of pathSegments.trans) {
6459
+ // 转换 Point[] 为 number[][]
6460
+ const points = segment.points.map((point) => [point.x, point.y]);
6461
+ const element = PathDataBuilder.createTransPath(points, {
6462
+ lineWidth: config.lineWidth,
6463
+ color: config.transLineColor,
6464
+ opacity: config.opacity,
6465
+ });
6466
+ element.originalData = points;
6467
+ element.pathType = PathSegmentType.TRANS;
6468
+ elements.push(element);
6469
+ }
6470
+ return {
6471
+ id,
6472
+ elements,
6473
+ };
6474
+ });
6475
+ return result;
6093
6476
  }
6094
6477
  }
6095
6478
 
@@ -6098,15 +6481,19 @@ class PathDataProcessor {
6098
6481
  * 专门处理边界标签的创建、定位和管理
6099
6482
  */
6100
6483
  class BoundaryLabelsManager {
6101
- constructor(svgView, boundaryData) {
6484
+ constructor(svgView, boundaryData, { unitType, language }) {
6102
6485
  this.container = null;
6103
6486
  this.overlayDiv = null;
6104
6487
  this.globalClickHandler = null;
6105
6488
  // 当前展开的边界id
6106
6489
  this.currentExpandedBoundaryId = null;
6490
+ // 旋转角度
6491
+ this.rotation = 0;
6107
6492
  this.svgView = svgView;
6108
6493
  this.boundaryData = boundaryData;
6109
6494
  this.initializeContainer();
6495
+ this.unitType = unitType;
6496
+ this.language = language;
6110
6497
  }
6111
6498
  /**
6112
6499
  * 初始化容器
@@ -6172,7 +6559,7 @@ class BoundaryLabelsManager {
6172
6559
  labelDiv.setAttribute('data-boundary-id', boundary.id.toString());
6173
6560
  // 样式设置
6174
6561
  labelDiv.style.position = 'absolute';
6175
- labelDiv.style.backgroundColor = 'rgba(30, 30, 31, 0.3)';
6562
+ labelDiv.style.backgroundColor = 'rgba(30, 30, 31, 0.6)';
6176
6563
  labelDiv.style.color = 'rgba(255, 255, 255, 1)';
6177
6564
  labelDiv.style.padding = '6px';
6178
6565
  labelDiv.style.borderRadius = '12px';
@@ -6180,17 +6567,17 @@ class BoundaryLabelsManager {
6180
6567
  labelDiv.style.fontWeight = 'bold';
6181
6568
  labelDiv.style.whiteSpace = 'nowrap';
6182
6569
  labelDiv.style.maxWidth = '220px';
6183
- labelDiv.style.transform = 'translate(-50%, -50%)';
6570
+ labelDiv.style.transform = `translate(-50%, -50%) rotate(${-this.rotation}deg)`;
6184
6571
  labelDiv.style.pointerEvents = 'auto';
6185
6572
  labelDiv.style.boxShadow = '0 2px 8px rgba(0,0,0,0.4)';
6186
6573
  labelDiv.style.cursor = 'pointer';
6187
- labelDiv.style.transition = 'background-color 0.2s ease, transform 0.2s ease';
6574
+ labelDiv.style.transition = 'background-color 0.2s ease';
6188
6575
  labelDiv.style.userSelect = 'none';
6189
6576
  labelDiv.style.zIndex = BoundaryLabelsManager.Z_INDEX.DEFAULT.toString();
6190
6577
  // 计算进度
6191
- const progress = boundary.finishedArea && boundary.area ?
6192
- `${Math.round((boundary.finishedArea / boundary.area) * 100)}%` :
6193
- '0%';
6578
+ const progress = boundary.finishedArea && boundary.area
6579
+ ? `${Math.floor((boundary.finishedArea / boundary.area) * 100)}%`
6580
+ : '0%';
6194
6581
  // 基础内容(始终显示)
6195
6582
  const baseContent = document.createElement('div');
6196
6583
  baseContent.className = 'boundary-label-base';
@@ -6202,15 +6589,19 @@ class BoundaryLabelsManager {
6202
6589
  extendedContent.style.marginTop = '6px';
6203
6590
  extendedContent.style.fontSize = '11px';
6204
6591
  extendedContent.style.opacity = '0.9';
6205
- extendedContent.style.display = this.currentExpandedBoundaryId === boundary.id ? 'block' : 'none';
6592
+ extendedContent.style.display =
6593
+ this.currentExpandedBoundaryId === boundary.id ? 'block' : 'none';
6206
6594
  extendedContent.style.borderTop = '1px solid rgba(255,255,255,0.2)';
6207
6595
  extendedContent.style.paddingTop = '6px';
6596
+ const boundaryLayer = this.svgView.getLayer(LAYER_DEFAULT_TYPE.BOUNDARY_BORDER);
6597
+ const mowingBoundarys = boundaryLayer.getMowingBoundarys();
6208
6598
  // 面积信息
6209
- const totalArea = convertAreaByUnits(boundary.area || 0, 'metric');
6210
- const finishedArea = convertAreaByUnits(boundary.finishedArea || 0, 'metric');
6599
+ const totalArea = convertAreaByUnits(boundary.area || 0, this.unitType);
6600
+ const finishedArea = convertAreaByUnits(boundary.finishedArea || 0, this.unitType);
6211
6601
  const coverageText = `Coverage: ${finishedArea.value}/${totalArea.value}`;
6602
+ const isMowing = mowingBoundarys.includes(boundary.id);
6212
6603
  // 日期信息
6213
- const dateText = formatBoundaryDateText(boundary.endTime || 0);
6604
+ const dateText = formatBoundaryDateText(isMowing ? Date.now() / 1000 : boundary.endTime || 0);
6214
6605
  const covertHtml = `<div style="margin-bottom: 3px; font-weight: bold;">${coverageText}</div>`;
6215
6606
  const dateHtml = `<div>${dateText}</div>`;
6216
6607
  extendedContent.innerHTML = boundary.finishedArea > 0 ? `${covertHtml}${dateHtml}` : covertHtml;
@@ -6273,7 +6664,6 @@ class BoundaryLabelsManager {
6273
6664
  this.collapseOtherLabels(boundaryId);
6274
6665
  // 展开当前标签
6275
6666
  extendedContent.style.display = 'block';
6276
- labelDiv.style.whiteSpace = 'normal';
6277
6667
  this.currentExpandedBoundaryId = boundaryId;
6278
6668
  }
6279
6669
  /**
@@ -6283,7 +6673,7 @@ class BoundaryLabelsManager {
6283
6673
  if (!this.container)
6284
6674
  return;
6285
6675
  const allLabels = this.container.querySelectorAll('.boundary-label');
6286
- allLabels.forEach(label => {
6676
+ allLabels.forEach((label) => {
6287
6677
  const labelBoundaryId = label.getAttribute('data-boundary-id');
6288
6678
  if (labelBoundaryId && labelBoundaryId !== boundaryId.toString()) {
6289
6679
  const extendedContent = label.querySelector('.boundary-label-extended');
@@ -6329,7 +6719,6 @@ class BoundaryLabelsManager {
6329
6719
  // 计算边界中心点的地图坐标
6330
6720
  const mapCenter = this.calculatePolygonCentroid(boundary.points);
6331
6721
  if (!mapCenter) {
6332
- console.warn(`BoundaryLabelsManager: 无法计算边界 ${boundary.name} (ID: ${boundary.id}) 的中心点`);
6333
6722
  return;
6334
6723
  }
6335
6724
  // 直接使用预计算的数据进行坐标转换
@@ -6386,7 +6775,7 @@ class BoundaryLabelsManager {
6386
6775
  if (points.length < 3)
6387
6776
  return null;
6388
6777
  // 过滤有效点
6389
- const validPoints = points.filter(point => point.length >= 2);
6778
+ const validPoints = points.filter((point) => point.length >= 2);
6390
6779
  if (validPoints.length < 3)
6391
6780
  return null;
6392
6781
  // 确保多边形是封闭的(如果不是,自动闭合)
@@ -6414,7 +6803,6 @@ class BoundaryLabelsManager {
6414
6803
  area = area / 2;
6415
6804
  // 如果面积为0,回退到简单的平均值计算
6416
6805
  if (Math.abs(area) < 1e-10) {
6417
- console.warn('BoundaryLabelsManager: 多边形面积为0,使用平均值计算重心');
6418
6806
  return this.calculateAverageCenter(validPoints);
6419
6807
  }
6420
6808
  centroidX = centroidX / (6 * area);
@@ -6435,7 +6823,7 @@ class BoundaryLabelsManager {
6435
6823
  });
6436
6824
  return {
6437
6825
  x: sumX / points.length,
6438
- y: sumY / points.length
6826
+ y: sumY / points.length,
6439
6827
  };
6440
6828
  }
6441
6829
  /**
@@ -6509,12 +6897,6 @@ class BoundaryLabelsManager {
6509
6897
  this.container.style.display = visible ? 'block' : 'none';
6510
6898
  }
6511
6899
  }
6512
- /**
6513
- * 获取边界数据
6514
- */
6515
- getBoundaryData() {
6516
- return this.boundaryData;
6517
- }
6518
6900
  /**
6519
6901
  * 根据ID获取特定边界的标签元素
6520
6902
  */
@@ -6531,7 +6913,7 @@ class BoundaryLabelsManager {
6531
6913
  return;
6532
6914
  const allLabels = this.container.querySelectorAll('.boundary-label');
6533
6915
  this.currentExpandedBoundaryId = null;
6534
- allLabels.forEach(label => {
6916
+ allLabels.forEach((label) => {
6535
6917
  const extendedContent = label.querySelector('.boundary-label-extended');
6536
6918
  if (extendedContent) {
6537
6919
  extendedContent.style.display = 'none';
@@ -6546,7 +6928,7 @@ class BoundaryLabelsManager {
6546
6928
  if (!this.container)
6547
6929
  return;
6548
6930
  const allLabels = this.container.querySelectorAll('.boundary-label');
6549
- allLabels.forEach(label => {
6931
+ allLabels.forEach((label) => {
6550
6932
  const labelElement = label;
6551
6933
  if (interactive) {
6552
6934
  labelElement.style.pointerEvents = 'auto';
@@ -6565,34 +6947,26 @@ class BoundaryLabelsManager {
6565
6947
  });
6566
6948
  }
6567
6949
  /**
6568
- * 重置标签层级(公共方法)
6569
- */
6570
- resetZIndex() {
6571
- this.resetLabelZIndex();
6572
- }
6573
- /**
6574
- * 获取层级常量(静态方法)
6575
- */
6576
- static getZIndexConstants() {
6577
- return BoundaryLabelsManager.Z_INDEX;
6578
- }
6579
- /**
6580
- * 调试方法:检查当前所有边界标签的层级
6950
+ * 设置标签旋转角度,使其与地图旋转相反,保持水平状态
6951
+ * @param rotation 地图的旋转角度(度)
6581
6952
  */
6582
- debugCheckZIndex() {
6953
+ setRotation(rotation) {
6583
6954
  if (!this.container)
6584
6955
  return;
6585
- const allLabels = this.container.querySelectorAll('.boundary-label');
6586
- allLabels.forEach((label, index) => {
6956
+ this.rotation = rotation;
6957
+ const labels = this.container.querySelectorAll('.boundary-label');
6958
+ labels.forEach((label) => {
6587
6959
  const labelElement = label;
6588
- labelElement.style.zIndex;
6960
+ // 应用与地图旋转相反的旋转,保持标签水平
6961
+ const counterRotation = -rotation;
6962
+ labelElement.style.transform = `translate(-50%, -50%) rotate(${counterRotation}deg)`;
6589
6963
  });
6590
6964
  }
6591
6965
  }
6592
6966
  // 简化的层级定义
6593
6967
  BoundaryLabelsManager.Z_INDEX = {
6594
6968
  DEFAULT: 900, // 默认层级
6595
- ACTIVE: 9999 // 点击激活时的高层级
6969
+ ACTIVE: 9999, // 点击激活时的高层级
6596
6970
  };
6597
6971
 
6598
6972
  /**
@@ -6605,6 +6979,10 @@ class ChargingPileManager {
6605
6979
  this.container = null;
6606
6980
  this.overlayDiv = null;
6607
6981
  this.pileElements = new Map();
6982
+ // 原始旋转角度
6983
+ this.originalRotation = 0;
6984
+ // 旋转角度
6985
+ this.rotation = 0;
6608
6986
  this.svgView = svgView;
6609
6987
  this.initializeContainer();
6610
6988
  }
@@ -6659,7 +7037,9 @@ class ChargingPileManager {
6659
7037
  // 将弧度转换为角度
6660
7038
  const angle = (direction * 180) / Math.PI;
6661
7039
  const rotationDegree = 270 - angle; // 坐标系转换
6662
- pileDiv.style.transform = `translate(-50%, -50%) rotate(${rotationDegree}deg)`;
7040
+ this.originalRotation = rotationDegree;
7041
+ const actualRotation = rotationDegree - this.rotation;
7042
+ pileDiv.style.transform = `translate(-50%, -50%) rotate(${actualRotation}deg)`;
6663
7043
  // 添加动画
6664
7044
  // this.addChargingPileAnimation(pileDiv, imgElement, rotationDegree);
6665
7045
  // 生成唯一ID
@@ -6708,30 +7088,15 @@ class ChargingPileManager {
6708
7088
  updatePositions() {
6709
7089
  if (!this.overlayDiv || !this.container)
6710
7090
  return;
6711
- const divRect = this.overlayDiv.getBoundingClientRect();
6712
- const divWidth = divRect.width;
6713
- const divHeight = divRect.height;
6714
- // 获取SVG的viewBox
6715
- const svg = this.svgView.getSVG();
6716
- if (!svg)
6717
- return;
6718
- const viewBox = svg.viewBox.baseVal;
6719
- const viewBoxData = {
6720
- x: viewBox.x,
6721
- y: viewBox.y,
6722
- width: viewBox.width,
6723
- height: viewBox.height,
6724
- };
6725
- this.updatePositionsWithPrecomputedData(divWidth, divHeight, viewBoxData);
7091
+ this.updatePositionsWithPrecomputedData();
6726
7092
  }
6727
7093
  /**
6728
7094
  * 使用预计算数据更新位置
6729
7095
  */
6730
- updatePositionsWithPrecomputedData(divWidth, divHeight, viewBox) {
7096
+ updatePositionsWithPrecomputedData() {
6731
7097
  this.chargingPileElements.forEach((element, _index) => {
6732
7098
  const center = element.coordinates[0];
6733
- const pixelPosition = this.convertMapCoordinateToPixelWithPrecomputedData(center[0], center[1], divWidth, divHeight, viewBox);
6734
- console.log('updatePositionsWithPrecomputedData----->', element, pixelPosition, divWidth, divHeight, viewBox);
7099
+ const pixelPosition = this.convertMapCoordinateToPixelWithPrecomputedData(center[0], center[1]);
6735
7100
  if (pixelPosition) {
6736
7101
  const pileId = `pile_${center[0]}_${center[1]}`;
6737
7102
  const pileElement = this.pileElements.get(pileId);
@@ -6745,7 +7110,15 @@ class ChargingPileManager {
6745
7110
  /**
6746
7111
  * 使用预计算数据进行坐标转换
6747
7112
  */
6748
- convertMapCoordinateToPixelWithPrecomputedData(mapX, mapY, divWidth, divHeight, viewBox) {
7113
+ convertMapCoordinateToPixelWithPrecomputedData(mapX, mapY) {
7114
+ // 获取叠加层div的CSS尺寸
7115
+ const divWidth = parseFloat(this.overlayDiv.style.width) || this.overlayDiv.offsetWidth;
7116
+ const divHeight = parseFloat(this.overlayDiv.style.height) || this.overlayDiv.offsetHeight;
7117
+ // 获取SVG的viewBox
7118
+ const svg = this.svgView.getSVG();
7119
+ if (!svg)
7120
+ return { x: 0, y: 0 };
7121
+ const viewBox = svg.viewBox.baseVal;
6749
7122
  // 计算地图坐标在viewBox中的相对位置
6750
7123
  const relativeX = (mapX - viewBox.x) / viewBox.width;
6751
7124
  const relativeY = (mapY - viewBox.y) / viewBox.height;
@@ -6764,26 +7137,6 @@ class ChargingPileManager {
6764
7137
  this.container.innerHTML = '';
6765
7138
  }
6766
7139
  }
6767
- /**
6768
- * 设置可见性
6769
- */
6770
- setVisible(visible) {
6771
- if (this.container) {
6772
- this.container.style.display = visible ? 'block' : 'none';
6773
- }
6774
- }
6775
- /**
6776
- * 获取充电桩数量
6777
- */
6778
- getElementCount() {
6779
- return this.chargingPileElements.length;
6780
- }
6781
- /**
6782
- * 充电桩不需要动态层级调整(为了接口统一而保留)
6783
- */
6784
- resetZIndex() {
6785
- // 充电桩层级始终保持固定,无需重置
6786
- }
6787
7140
  /**
6788
7141
  * 销毁管理器
6789
7142
  */
@@ -6795,10 +7148,21 @@ class ChargingPileManager {
6795
7148
  this.container = null;
6796
7149
  this.overlayDiv = null;
6797
7150
  }
7151
+ /**
7152
+ * 设置充电桩旋转角度
7153
+ */
7154
+ setRotation(rotation) {
7155
+ this.rotation = rotation;
7156
+ const allContainers = this.container.querySelectorAll('.charging-pile');
7157
+ allContainers.forEach((container) => {
7158
+ const pileElement = container;
7159
+ pileElement.style.transform = `translate(-50%, -50%) rotate(${this.originalRotation - this.rotation}deg)`;
7160
+ });
7161
+ }
6798
7162
  }
6799
7163
  // 简化的层级定义 - 充电桩只需要一个固定层级
6800
7164
  ChargingPileManager.Z_INDEX = {
6801
- CHARGING_PILE: 800, // 充电桩图标固定层级
7165
+ CHARGING_PILE: 750, // 充电桩图标固定层级
6802
7166
  };
6803
7167
 
6804
7168
  /**
@@ -6816,6 +7180,8 @@ class AntennaManager {
6816
7180
  this.globalClickHandler = null;
6817
7181
  this.antennaTooltipFlag = false;
6818
7182
  this.singleAntennaTooltipFlag = false;
7183
+ // 旋转角度
7184
+ this.rotation = 0;
6819
7185
  this.svgView = svgView;
6820
7186
  this.initializeContainer();
6821
7187
  this.setupGlobalClickHandler();
@@ -6888,7 +7254,7 @@ class AntennaManager {
6888
7254
  const antennaContainer = document.createElement('div');
6889
7255
  antennaContainer.className = 'antenna-container-item';
6890
7256
  antennaContainer.style.position = 'absolute';
6891
- antennaContainer.style.transform = 'translate(-50%, -50%)';
7257
+ antennaContainer.style.transform = `translate(-50%, -50%) rotate(${-this.rotation}deg)`;
6892
7258
  antennaContainer.style.pointerEvents = 'auto';
6893
7259
  antennaContainer.style.zIndex = AntennaManager.Z_INDEX.DEFAULT.toString();
6894
7260
  antennaContainer.setAttribute('data-antenna-id', antennaData.type.toString());
@@ -6929,8 +7295,6 @@ class AntennaManager {
6929
7295
  this.singleAntennaTooltipFlag = true;
6930
7296
  }
6931
7297
  this.expandTooltip(antennaContainer);
6932
- // 调试:检查层级变化
6933
- this.debugCheckZIndex();
6934
7298
  });
6935
7299
  // 添加悬停效果
6936
7300
  antennaDiv.addEventListener('mouseenter', () => {
@@ -7180,39 +7544,6 @@ class AntennaManager {
7180
7544
  this.container.innerHTML = '';
7181
7545
  }
7182
7546
  }
7183
- /**
7184
- * 设置可见性
7185
- */
7186
- setVisible(visible) {
7187
- if (this.container) {
7188
- this.container.style.display = visible ? 'block' : 'none';
7189
- }
7190
- }
7191
- /**
7192
- * 获取天线数量
7193
- */
7194
- getElementCount() {
7195
- return this.antennaElements.length;
7196
- }
7197
- /**
7198
- * 重置天线层级(公共方法)
7199
- */
7200
- resetZIndex() {
7201
- this.resetAntennaZIndex();
7202
- }
7203
- /**
7204
- * 获取层级常量(静态方法)
7205
- */
7206
- static getZIndexConstants() {
7207
- return AntennaManager.Z_INDEX;
7208
- }
7209
- /**
7210
- * 调试方法:检查当前所有天线的层级
7211
- */
7212
- debugCheckZIndex() {
7213
- if (!this.container)
7214
- return;
7215
- }
7216
7547
  /**
7217
7548
  * 销毁管理器
7218
7549
  */
@@ -7229,10 +7560,21 @@ class AntennaManager {
7229
7560
  this.container = null;
7230
7561
  this.overlayDiv = null;
7231
7562
  }
7563
+ /**
7564
+ * 设置天线旋转角度
7565
+ */
7566
+ setRotation(rotation) {
7567
+ this.rotation = rotation;
7568
+ const allContainers = this.container.querySelectorAll('.antenna-container-item');
7569
+ allContainers.forEach((container) => {
7570
+ const antennaContainer = container;
7571
+ antennaContainer.style.transform = `translate(-50%, -50%) rotate(${-this.rotation}deg)`;
7572
+ });
7573
+ }
7232
7574
  }
7233
7575
  // 简化的层级定义
7234
7576
  AntennaManager.Z_INDEX = {
7235
- DEFAULT: 750, // 默认层级
7577
+ DEFAULT: 800, // 默认层级
7236
7578
  ACTIVE: 9999, // 点击激活时的高层级
7237
7579
  };
7238
7580
 
@@ -7283,8 +7625,8 @@ const EDIT_BEHAVIOR = {
7283
7625
  SUCCESS_MESSAGE_DURATION_MS: 3000,
7284
7626
  };
7285
7627
 
7286
- class MowerPostionManager {
7287
- constructor(svgView, mowerPositonConfig, overlayDiv, onAnimationComplete) {
7628
+ class MowerPositionManager {
7629
+ constructor(svgView, mowerPositionConfig, modelType, overlayDiv, onAnimationComplete, onMowingPositionChange) {
7288
7630
  this.container = null;
7289
7631
  this.overlayDiv = null;
7290
7632
  this.mowerElement = null;
@@ -7301,9 +7643,11 @@ class MowerPostionManager {
7301
7643
  this.deltaPosition = null;
7302
7644
  this.onlyUpdateTheta = false;
7303
7645
  this.svgView = svgView;
7304
- this.mowerPositonConfig = mowerPositonConfig;
7646
+ this.mowerPositionConfig = mowerPositionConfig;
7647
+ this.modelType = modelType;
7305
7648
  this.overlayDiv = overlayDiv;
7306
7649
  this.onAnimationComplete = onAnimationComplete;
7650
+ this.onMowingPositionChange = onMowingPositionChange;
7307
7651
  this.initializeContainer();
7308
7652
  }
7309
7653
  get animationFlag() {
@@ -7349,12 +7693,13 @@ class MowerPostionManager {
7349
7693
  imgElement.style.height = '100%';
7350
7694
  imgElement.style.objectFit = 'contain';
7351
7695
  // 获取图片源
7352
- const imageSrc = getMowerImage(this.mowerPositonConfig);
7696
+ const imageSrc = getMowerImage(this.mowerPositionConfig);
7353
7697
  if (imageSrc) {
7354
7698
  imgElement.src = imageSrc;
7355
7699
  }
7356
7700
  this.mowerElement.appendChild(imgElement);
7357
7701
  this.container.appendChild(this.mowerElement);
7702
+ this.updatePosition(this.mowerPositionConfig);
7358
7703
  }
7359
7704
  /**
7360
7705
  * 设置叠加层div引用(用于坐标转换)
@@ -7368,54 +7713,69 @@ class MowerPostionManager {
7368
7713
  getElement() {
7369
7714
  return this.container;
7370
7715
  }
7716
+ //
7717
+ setEdger(edger) {
7718
+ this.hasEdger = edger;
7719
+ }
7371
7720
  /**
7372
7721
  * 根据最后一次有效的位置更新数据
7373
7722
  */
7374
- updatePostionByLastPosition(chargingPilesPositonConfig, animationTime = 0) {
7375
- this.mowerPositonConfig = chargingPilesPositonConfig;
7376
- console.log('updatePostionByLastPosition----->', chargingPilesPositonConfig, this.lastPosition);
7723
+ updatePositionByLastPosition(chargingPilesPositionConfig) {
7724
+ if (!chargingPilesPositionConfig)
7725
+ return;
7726
+ this.mowerPositionConfig = chargingPilesPositionConfig;
7727
+ const positonOutOfRange = isOutOfRange(chargingPilesPositionConfig);
7728
+ const positionInValid = isInvalidPosition(chargingPilesPositionConfig);
7729
+ let postureX = 0;
7730
+ let postureY = 0;
7731
+ let postureTheta = 0;
7377
7732
  const lastPosition = this.lastPosition;
7378
- if (!lastPosition) {
7379
- this.updatePosition(chargingPilesPositonConfig, animationTime);
7733
+ if (positonOutOfRange || positionInValid) {
7734
+ postureX = lastPosition?.x || 0;
7735
+ postureY = lastPosition?.y || 0;
7736
+ postureTheta = lastPosition?.rotation || 0;
7380
7737
  }
7381
7738
  else {
7382
- this.updatePosition({
7383
- ...chargingPilesPositonConfig,
7384
- postureX: lastPosition.x,
7385
- postureY: lastPosition.y,
7386
- postureTheta: lastPosition.rotation,
7387
- }, 0);
7739
+ postureX = chargingPilesPositionConfig.postureX || 0;
7740
+ postureY = chargingPilesPositionConfig.postureY || 0;
7741
+ postureTheta = chargingPilesPositionConfig.postureTheta || 0;
7388
7742
  }
7743
+ // 检查是否需要更新图片
7744
+ this.updateMowerImage(chargingPilesPositionConfig);
7745
+ // 立即更新位置
7746
+ this.setElementPosition(postureX, postureY, postureTheta);
7389
7747
  }
7390
7748
  /**
7391
7749
  * 更新割草机位置
7392
7750
  */
7393
- updatePosition(positonConfig, animationTime = 0) {
7394
- console.log('updatePosition----->', positonConfig);
7751
+ updatePosition(positionConfig, animationTime = 0) {
7395
7752
  // 检查是否需要更新图片
7396
- this.updateMowerImage(positonConfig);
7753
+ this.updateMowerImage(positionConfig);
7754
+ // 更新配置
7755
+ this.mowerPositionConfig = positionConfig;
7756
+ const postureX = positionConfig?.postureX || this.lastPosition?.x || 0;
7757
+ const postureY = positionConfig?.postureY || this.lastPosition?.y || 0;
7758
+ const postureTheta = positionConfig?.postureTheta || this.lastPosition?.rotation || 0;
7397
7759
  // 停止当前动画(如果有)
7398
7760
  this.stopAnimation();
7399
7761
  // 第一个点
7400
7762
  if (!this.currentPosition) {
7401
7763
  this.currentPosition = {
7402
- x: positonConfig.postureX,
7403
- y: positonConfig.postureY,
7404
- rotation: positonConfig.postureTheta,
7764
+ x: postureX,
7765
+ y: postureY,
7766
+ rotation: postureTheta,
7405
7767
  };
7406
7768
  this.setElementPosition(this.currentPosition.x, this.currentPosition.y, this.currentPosition.rotation);
7407
7769
  return;
7408
7770
  }
7409
7771
  // 根据动画时长决定更新方式
7410
7772
  if (animationTime > 0) {
7411
- this.startAnimationToPosition(positonConfig, animationTime);
7773
+ this.startAnimationToPosition(positionConfig, animationTime);
7412
7774
  }
7413
7775
  else {
7414
7776
  // 立即更新位置
7415
- this.setElementPosition(positonConfig.postureX, positonConfig.postureY, positonConfig.postureTheta);
7777
+ this.setElementPosition(positionConfig.postureX, positionConfig.postureY, positionConfig.postureTheta);
7416
7778
  }
7417
- // 更新配置
7418
- this.mowerPositonConfig = positonConfig;
7419
7779
  }
7420
7780
  /**
7421
7781
  * 更新割草机图片
@@ -7426,9 +7786,14 @@ class MowerPostionManager {
7426
7786
  const imgElement = this.mowerElement.querySelector('img');
7427
7787
  if (!imgElement)
7428
7788
  return;
7429
- const imageSrc = getMowerImage(positonConfig);
7789
+ const imageSrc = getMowerImage(positonConfig, this.modelType, this.hasEdger);
7430
7790
  if (imageSrc) {
7431
7791
  imgElement.src = imageSrc;
7792
+ imgElement.style.display = 'block';
7793
+ }
7794
+ else {
7795
+ imgElement.style.display = 'none';
7796
+ return;
7432
7797
  }
7433
7798
  }
7434
7799
  /**
@@ -7438,7 +7803,9 @@ class MowerPostionManager {
7438
7803
  const { x: pointX, y: pointY } = convertCoordinate(x, y);
7439
7804
  const targetPixelPosition = this.convertMapCoordinateToOverlayPixel(pointX, pointY);
7440
7805
  const targetRotation = radToDegree(theta);
7806
+ this.currentPosition = { x: x, y: y, rotation: theta };
7441
7807
  this.lastPosition = { x, y, rotation: theta };
7808
+ // console.log('setElementPosition', x, y, theta, targetRotation, positonOutOfRange, positionValid, targetPixelPosition);
7442
7809
  if (!this.mowerElement)
7443
7810
  return;
7444
7811
  this.mowerElement.style.left = `${targetPixelPosition?.x}px`;
@@ -7464,29 +7831,48 @@ class MowerPostionManager {
7464
7831
  startAnimationToPosition(positionConfig, duration) {
7465
7832
  if (!this.mowerElement || !this.currentPosition)
7466
7833
  return;
7467
- console.log('before startAnimationToPosition----->', this.startPosition, this.currentPosition, this.targetPosition, positionConfig);
7468
- this.startPosition = { ...this.currentPosition, rotation: radNormalize(this.currentPosition.rotation) };
7469
7834
  this.targetPosition = {
7470
7835
  x: positionConfig.postureX,
7471
7836
  y: positionConfig.postureY,
7472
7837
  rotation: positionConfig.postureTheta,
7473
7838
  };
7839
+ const isTargetPositionInvalid = isInvalidPosition({
7840
+ postureX: this.targetPosition.x,
7841
+ postureY: this.targetPosition.y,
7842
+ postureTheta: this.targetPosition.rotation,
7843
+ });
7844
+ const isTargetPositionOutOfRange = isOutOfRange({
7845
+ postureX: this.targetPosition.x,
7846
+ postureY: this.targetPosition.y,
7847
+ postureTheta: this.targetPosition.rotation,
7848
+ });
7849
+ // 如果目标坐标点不合理,则舍弃不使用
7850
+ if (isTargetPositionInvalid || isTargetPositionOutOfRange) {
7851
+ return;
7852
+ }
7853
+ this.startPosition = {
7854
+ ...this.currentPosition,
7855
+ rotation: radNormalize(this.currentPosition.rotation),
7856
+ };
7474
7857
  this.animationDuration = duration;
7475
7858
  this.startTime = window.performance.now();
7476
7859
  this.isAnimating = true;
7477
7860
  this.onlyUpdateTheta =
7478
7861
  distance(this.currentPosition.x, this.currentPosition.y, positionConfig.postureX, positionConfig.postureY) < 0.2;
7479
7862
  // 为了实现倒车时,割草机图标旋转方向正确,需要计算当前位置和目标位置的夹角
7480
- const calTheta = calAngle(this.startPosition.x, this.startPosition.y, this.targetPosition.x, this.targetPosition.y);
7863
+ // const calTheta = calAngle(
7864
+ // this.startPosition.x,
7865
+ // this.startPosition.y,
7866
+ // this.targetPosition.x,
7867
+ // this.targetPosition.y
7868
+ // );
7481
7869
  const startTheta = this.startPosition.rotation;
7482
- const targetTheta = this.onlyUpdateTheta ? this.targetPosition.rotation : calTheta;
7483
- console.log('begain startAnimationToPosition----->', this.startPosition, this.targetPosition, this.onlyUpdateTheta);
7870
+ const targetTheta = this.targetPosition.rotation;
7484
7871
  this.deltaPosition = {
7485
7872
  x: this.onlyUpdateTheta ? 0 : this.targetPosition.x - this.startPosition.x,
7486
7873
  y: this.onlyUpdateTheta ? 0 : this.targetPosition.y - this.startPosition.y,
7487
- rotation: radNormalize(targetTheta - startTheta)
7874
+ rotation: radNormalize(targetTheta - startTheta),
7488
7875
  };
7489
- console.log('deltaPosition----->', targetTheta, startTheta, calTheta, this.deltaPosition);
7490
7876
  // 开始动画循环
7491
7877
  this.animateStep();
7492
7878
  }
@@ -7497,7 +7883,7 @@ class MowerPostionManager {
7497
7883
  * 动画步骤
7498
7884
  */
7499
7885
  animateStep() {
7500
- if (!this.isAnimating || !this.currentPosition || !this.targetPosition || !this.startPosition)
7886
+ if (!this.isAnimating || !this.targetPosition || !this.startPosition)
7501
7887
  return;
7502
7888
  const currentTime = window.performance.now();
7503
7889
  const elapsed = currentTime - this.startTime;
@@ -7508,20 +7894,24 @@ class MowerPostionManager {
7508
7894
  const currentX = this.startPosition.x + this.deltaPosition.x * easedProgress;
7509
7895
  const currentY = this.startPosition.y + this.deltaPosition.y * easedProgress;
7510
7896
  const currentRotation = this.startPosition.rotation + this.deltaPosition.rotation * easedProgress;
7511
- this.currentPosition = { x: currentX, y: currentY, rotation: currentRotation };
7512
- // console.log('animateStep----->', this.onlyUpdateTheta, currentX, currentY, currentRotation);
7513
- // console.log('animateStep1----->', onlyUpdateTheta, progress, calTheta, currentRotation);
7897
+ // 假设在这里进行更新路径数据
7898
+ if (this.onMowingPositionChange) {
7899
+ this.onMowingPositionChange({
7900
+ x: currentX,
7901
+ y: currentY,
7902
+ vehicleState: this.mowerPositionConfig?.vehicleState,
7903
+ });
7904
+ }
7905
+ // console.log('animateStep-->', this.startPosition, this.deltaPosition, this.targetPosition, easedProgress)
7906
+ this.setElementPosition(currentX, currentY, currentRotation);
7514
7907
  // 继续动画或结束
7515
7908
  if (progress < 1) {
7516
7909
  // 设置当前位置
7517
- this.setElementPosition(currentX, currentY, currentRotation);
7518
7910
  this.animationId = window.requestAnimationFrame(() => this.animateStep());
7519
7911
  }
7520
7912
  else {
7521
- this.setElementPosition(currentX, currentY, currentRotation);
7522
7913
  // 动画完成
7523
7914
  this.stopAnimation();
7524
- console.log('end animateStep----->', this.lastPosition, this.currentPosition, this.targetPosition);
7525
7915
  // 通知动画完成
7526
7916
  if (this.onAnimationComplete) {
7527
7917
  this.onAnimationComplete();
@@ -7566,12 +7956,6 @@ class MowerPostionManager {
7566
7956
  this.container.style.display = visible ? 'block' : 'none';
7567
7957
  }
7568
7958
  }
7569
- /**
7570
- * 检查是否正在动画中
7571
- */
7572
- getIsAnimating() {
7573
- return this.isAnimating;
7574
- }
7575
7959
  /**
7576
7960
  * 销毁管理器
7577
7961
  */
@@ -7587,13 +7971,86 @@ class MowerPostionManager {
7587
7971
  }
7588
7972
  }
7589
7973
 
7974
+ // 记录割草状态,状态变更的时候,变量不触发重新渲染
7975
+ const useProcessMowingState = create((set) => ({
7976
+ processStateIsMowing: false,
7977
+ updateProcessStateIsMowing: (isMowing) => set({ processStateIsMowing: isMowing }),
7978
+ resetProcessStateIsMowing: () => set({ processStateIsMowing: false }),
7979
+ }));
7980
+
7981
+ /**
7982
+ * 高级节流函数
7983
+ * @param func 要节流的函数
7984
+ * @param delay 延迟时间(毫秒)
7985
+ * @param options 配置选项
7986
+ * @returns 节流后的函数
7987
+ */
7988
+ function throttleAdvanced(func, delay, options = { leading: true, trailing: true }) {
7989
+ let lastExecTime = 0;
7990
+ let timeoutId = null;
7991
+ let lastArgs = null;
7992
+ return function throttled(...args) {
7993
+ const currentTime = Date.now();
7994
+ lastArgs = args;
7995
+ // 如果距离上次执行的时间小于延迟时间
7996
+ if (currentTime - lastExecTime < delay) {
7997
+ // 清除之前的定时器
7998
+ if (timeoutId) {
7999
+ clearTimeout(timeoutId);
8000
+ }
8001
+ // 设置新的定时器
8002
+ timeoutId = setTimeout(() => {
8003
+ if (options.trailing && lastArgs) {
8004
+ lastExecTime = Date.now();
8005
+ func.apply(this, lastArgs);
8006
+ lastArgs = null;
8007
+ }
8008
+ timeoutId = null;
8009
+ }, delay - (currentTime - lastExecTime));
8010
+ }
8011
+ else {
8012
+ // 如果距离上次执行的时间已经超过延迟时间
8013
+ if (options.leading) {
8014
+ lastExecTime = currentTime;
8015
+ func.apply(this, args);
8016
+ }
8017
+ }
8018
+ };
8019
+ }
8020
+ /**
8021
+ * 检测当前设备是否为移动设备
8022
+ * @returns {boolean} 如果是移动设备返回true,否则返回false
8023
+ */
8024
+ function isMobileDevice() {
8025
+ // 确保在浏览器环境中运行
8026
+ if (typeof window === 'undefined' || typeof navigator === 'undefined') {
8027
+ return false;
8028
+ }
8029
+ // 检查用户代理字符串
8030
+ const userAgent = navigator.userAgent.toLowerCase();
8031
+ const mobileKeywords = [
8032
+ 'android', 'webos', 'iphone', 'ipad', 'ipod',
8033
+ 'blackberry', 'windows phone', 'mobile'
8034
+ ];
8035
+ const isMobileUserAgent = mobileKeywords.some(keyword => userAgent.includes(keyword));
8036
+ // 检查触摸屏支持
8037
+ const hasTouchScreen = 'ontouchstart' in window ||
8038
+ (navigator.maxTouchPoints && navigator.maxTouchPoints > 0);
8039
+ // 检查屏幕尺寸(移动设备通常屏幕较小)
8040
+ const isSmallScreen = window.innerWidth <= 768;
8041
+ // 综合判断:用户代理包含移动设备关键词,或者有触摸屏且屏幕较小
8042
+ return isMobileUserAgent || (hasTouchScreen && isSmallScreen);
8043
+ }
8044
+
7590
8045
  // Google Maps 叠加层类 - 带编辑功能
7591
8046
  class MowerMapOverlay {
7592
- constructor(bounds, mapData, mowerPositonConfig, pathData, isEditMode = false, mapConfig = {}, antennaConfig = {}, onMapLoad, onPathLoad, dragCallbacks) {
8047
+ constructor(bounds, mapData, partitionBoundary, mowerPositionConfig, modelType, pathData, isEditMode = false, unitType = UnitsType.Imperial, language = 'en', mapConfig = {}, antennaConfig = {}, mowPartitionData = null, defaultTransform, onMapLoad, onPathLoad, dragCallbacks) {
7593
8048
  this.div = null;
7594
8049
  this.svgMapView = null;
7595
8050
  this.offscreenContainer = null;
7596
8051
  this.overlayView = null;
8052
+ this.defaultTransform = { x: 0, y: 0, rotation: 0 };
8053
+ this.hasEdger = false;
7597
8054
  // boundary数据
7598
8055
  this.boundaryData = [];
7599
8056
  // 边界标签管理器
@@ -7603,7 +8060,7 @@ class MowerMapOverlay {
7603
8060
  // 天线管理器
7604
8061
  this.antennaManager = null;
7605
8062
  // 割草机位置管理器
7606
- this.mowerPostionManager = null;
8063
+ this.mowerPositionManager = null;
7607
8064
  // 当前动画时长
7608
8065
  this.currentAnimationTime = 0;
7609
8066
  // 是否正在用户动画中(区分用户主动触发的动画和地图重绘)
@@ -7626,16 +8083,38 @@ class MowerMapOverlay {
7626
8083
  // 初始状态记录(用于计算相对于初始状态的偏移)
7627
8084
  this.initialOffset = { x: 0, y: 0 };
7628
8085
  this.initialRotation = 0;
8086
+ this.mowPartitionData = null;
8087
+ this.updatePathDataByMowingPositionThrottled = throttleAdvanced(this.updatePathDataByMowingPosition, 300, {
8088
+ leading: true,
8089
+ trailing: false,
8090
+ });
7629
8091
  this.bounds = bounds;
7630
8092
  this.mapData = mapData;
8093
+ this.partitionBoundary = partitionBoundary;
7631
8094
  this.pathData = pathData;
7632
8095
  this.isEditMode = isEditMode;
8096
+ this.unitType = unitType;
8097
+ this.language = language;
7633
8098
  this.mapConfig = mapConfig;
7634
8099
  this.antennaConfig = antennaConfig;
7635
8100
  this.onMapLoad = onMapLoad;
7636
8101
  this.onPathLoad = onPathLoad;
7637
8102
  this.dragCallbacks = dragCallbacks;
7638
- this.mowerPositonConfig = mowerPositonConfig;
8103
+ this.mowerPositionConfig = mowerPositionConfig;
8104
+ this.mowPartitionData = mowPartitionData;
8105
+ this.modelType = modelType;
8106
+ // 设置默认的transform
8107
+ if (defaultTransform) {
8108
+ this.defaultTransform = {
8109
+ x: defaultTransform.x ?? 0,
8110
+ y: defaultTransform.y ?? 0,
8111
+ rotation: defaultTransform.rotation ?? 0,
8112
+ };
8113
+ // defaultTransform的x对应经度偏移量,y对应纬度偏移量
8114
+ this.latLngOffset.lng = this.defaultTransform.x;
8115
+ this.latLngOffset.lat = this.defaultTransform.y;
8116
+ this.currentRotation = this.defaultTransform.rotation;
8117
+ }
7639
8118
  // 创建 OverlayView 实例
7640
8119
  if (window.google && window.google.maps) {
7641
8120
  this.overlayView = new window.google.maps.OverlayView();
@@ -7650,26 +8129,28 @@ class MowerMapOverlay {
7650
8129
  this.overlayView.handleSave = this.handleSave.bind(this);
7651
8130
  this.overlayView.setCustomIcons = this.setCustomIcons.bind(this);
7652
8131
  this.overlayView.getCurrentDragState = this.getCurrentDragState.bind(this);
8132
+ this.overlayView.setTransform = this.setTransform.bind(this);
8133
+ this.overlayView.resetToDefaultTransform = this.resetToDefaultTransform.bind(this);
7653
8134
  }
7654
8135
  this.boundaryData = generateBoundaryData(mapData, pathData);
7655
8136
  }
7656
- updatePosition(positonConfig, animationTime = 2200) {
7657
- console.log('updatePosition==', positonConfig, animationTime);
8137
+ updatePosition(positionConfig, animationTime = 2200) {
7658
8138
  // 保存当前动画时长
7659
8139
  this.currentAnimationTime = animationTime;
7660
8140
  // 标记是否为用户动画(大于0表示用户主动触发的动画)
7661
8141
  this.isUserAnimation = animationTime > 0;
7662
8142
  // 更新割草机位置配置
7663
- this.mowerPositonConfig = positonConfig;
8143
+ this.mowerPositionConfig = positionConfig;
7664
8144
  // 更新割草机位置管理器
7665
- if (this.mowerPostionManager) {
7666
- this.mowerPostionManager.updatePosition(positonConfig, animationTime);
8145
+ if (this.mowerPositionManager) {
8146
+ this.mowerPositionManager.updatePosition(positionConfig, animationTime);
7667
8147
  }
7668
8148
  }
7669
- updatePostionByLastPosition(chargingPilesPositonConfig, animationTime = 2200) {
8149
+ updatePositionByLastPosition(chargingPilesPositionConfig) {
8150
+ this.mowerPositionConfig = chargingPilesPositionConfig;
7670
8151
  // 更新配置
7671
- if (this.mowerPostionManager) {
7672
- this.mowerPostionManager.updatePostionByLastPosition(chargingPilesPositonConfig, animationTime);
8152
+ if (this.mowerPositionManager) {
8153
+ this.mowerPositionManager.updatePositionByLastPosition(chargingPilesPositionConfig);
7673
8154
  }
7674
8155
  }
7675
8156
  setMap(map) {
@@ -7677,6 +8158,12 @@ class MowerMapOverlay {
7677
8158
  this.overlayView.setMap(map);
7678
8159
  }
7679
8160
  }
8161
+ setEdger(edger) {
8162
+ this.hasEdger = edger;
8163
+ if (this.mowerPositionManager) {
8164
+ this.mowerPositionManager.setEdger(edger);
8165
+ }
8166
+ }
7680
8167
  getMap() {
7681
8168
  return this.overlayView ? this.overlayView.getMap() : null;
7682
8169
  }
@@ -7686,6 +8173,23 @@ class MowerMapOverlay {
7686
8173
  getPanes() {
7687
8174
  return this.overlayView ? this.overlayView.getPanes() : null;
7688
8175
  }
8176
+ resetBorderLayerHighlight() {
8177
+ this.mowPartitionData = null;
8178
+ const boundaryBorderLayer = this.svgMapView?.getLayer(LAYER_DEFAULT_TYPE.BOUNDARY_BORDER);
8179
+ if (!boundaryBorderLayer)
8180
+ return;
8181
+ boundaryBorderLayer.setMowingBoundarys([]);
8182
+ this.svgMapView?.renderLayer(LAYER_DEFAULT_TYPE.BOUNDARY_BORDER);
8183
+ }
8184
+ setBorderLayerHighlight(mowPartitionData) {
8185
+ this.mowPartitionData = mowPartitionData;
8186
+ const boundaryBorderLayer = this.svgMapView?.getLayer(LAYER_DEFAULT_TYPE.BOUNDARY_BORDER);
8187
+ const partitionIds = mowPartitionData?.partitionIds || [];
8188
+ if (!boundaryBorderLayer)
8189
+ return;
8190
+ boundaryBorderLayer.setMowingBoundarys(partitionIds);
8191
+ this.svgMapView?.renderLayer(LAYER_DEFAULT_TYPE.BOUNDARY_BORDER);
8192
+ }
7689
8193
  onAdd() {
7690
8194
  // 创建包含SVG的div
7691
8195
  this.div = document.createElement('div');
@@ -7708,6 +8212,7 @@ class MowerMapOverlay {
7708
8212
  this.createBoundaryLabelsManager();
7709
8213
  // 创建割草机位置管理器
7710
8214
  this.createMowerPositionManager();
8215
+ this.setManagerRotation(this.defaultTransform.rotation);
7711
8216
  // 如果处于编辑模式,创建编辑界面
7712
8217
  if (this.isEditMode) {
7713
8218
  this.createEditInterface();
@@ -7746,23 +8251,17 @@ class MowerMapOverlay {
7746
8251
  const map = this.getMap();
7747
8252
  if (!map || !this.svgMapView)
7748
8253
  return;
7749
- map.getZoom();
7750
- // const center = map.getCenter();
7751
- // 基础公式:像素/米 = 156543.03392 * cos(latitude) / (2 ^ zoom)
7752
- // const metersPerPixel =
7753
- // (156543.03392 * Math.cos((center.lat() * Math.PI) / 180)) / Math.pow(2, currentZoom);
7754
- // 缩放比例 = 1 / 米/像素
7755
- // const scale = (1 / metersPerPixel) * 50;
7756
- // 应用缩放到SVG
7757
- // this.svgMapView.setZoom(scale);
8254
+ this.draw();
7758
8255
  }
7759
8256
  // 创建边界标签管理器
7760
8257
  createBoundaryLabelsManager() {
7761
8258
  if (!this.div || !this.svgMapView)
7762
8259
  return;
7763
- console.log('this.boundaryData----->', this.boundaryData);
7764
8260
  // 创建边界标签管理器
7765
- this.boundaryLabelsManager = new BoundaryLabelsManager(this.svgMapView, this.boundaryData);
8261
+ this.boundaryLabelsManager = new BoundaryLabelsManager(this.svgMapView, this.boundaryData, {
8262
+ unitType: this.unitType,
8263
+ language: this.language,
8264
+ });
7766
8265
  // 设置叠加层div引用
7767
8266
  this.boundaryLabelsManager.setOverlayDiv(this.div);
7768
8267
  // 添加所有边界标签
@@ -7806,11 +8305,12 @@ class MowerMapOverlay {
7806
8305
  if (!this.div || !this.svgMapView)
7807
8306
  return;
7808
8307
  // 创建割草机位置管理器,传入动画完成回调
7809
- this.mowerPostionManager = new MowerPostionManager(this.svgMapView, this.mowerPositonConfig, this.div);
8308
+ this.mowerPositionManager = new MowerPositionManager(this.svgMapView, this.mowerPositionConfig, this.modelType, this.div, () => { }, this.updatePathDataByMowingPositionThrottled.bind(this));
7810
8309
  // 设置叠加层div引用
7811
- this.mowerPostionManager.setOverlayDiv(this.div);
8310
+ this.mowerPositionManager.setOverlayDiv(this.div);
8311
+ this.mowerPositionManager.setEdger(this.hasEdger);
7812
8312
  // 获取容器并添加到主div
7813
- const container = this.mowerPostionManager.getElement();
8313
+ const container = this.mowerPositionManager.getElement();
7814
8314
  if (container) {
7815
8315
  this.div.appendChild(container);
7816
8316
  }
@@ -7837,23 +8337,9 @@ class MowerMapOverlay {
7837
8337
  this.boundaryLabelsManager.updatePositionsWithPrecomputedData(width, height, viewBoxInfo);
7838
8338
  }
7839
8339
  // 更新管理器位置
7840
- updateManagerPositions(width, height) {
8340
+ updateManagerPositions(_width, _height) {
7841
8341
  if (!this.div)
7842
8342
  return;
7843
- // 获取SVG元素和其viewBox
7844
- const svgElement = this.div.querySelector('svg');
7845
- if (!svgElement)
7846
- return;
7847
- const viewBox = svgElement.viewBox.baseVal;
7848
- if (!viewBox)
7849
- return;
7850
- // 构造viewBox信息对象
7851
- ({
7852
- x: viewBox.x,
7853
- y: viewBox.y,
7854
- width: viewBox.width,
7855
- height: viewBox.height,
7856
- });
7857
8343
  // 更新充电桩位置
7858
8344
  if (this.chargingPileManager) {
7859
8345
  this.chargingPileManager.updatePositions();
@@ -7902,7 +8388,7 @@ class MowerMapOverlay {
7902
8388
  this.rotateHandle.style.pointerEvents = 'auto';
7903
8389
  this.rotateHandle.innerHTML = DEFAULT_ROTATE_ICON;
7904
8390
  this.editContainer.appendChild(this.rotateHandle);
7905
- // 创建拖拽手柄(左上角)
8391
+ // 创建拖拽手柄(左下角)- 仅在移动设备上显示
7906
8392
  this.dragHandle = document.createElement('div');
7907
8393
  this.dragHandle.style.position = 'absolute';
7908
8394
  this.dragHandle.style.bottom = '-20px';
@@ -7913,6 +8399,10 @@ class MowerMapOverlay {
7913
8399
  this.dragHandle.style.zIndex = EDIT_STYLES.Z_INDEX.HANDLE;
7914
8400
  this.dragHandle.style.pointerEvents = 'auto';
7915
8401
  this.dragHandle.innerHTML = DEFAULT_DRAG_ICON;
8402
+ // 在PC设备上隐藏拖拽手柄
8403
+ if (!isMobileDevice()) {
8404
+ this.dragHandle.style.display = 'none';
8405
+ }
7916
8406
  this.editContainer.appendChild(this.dragHandle);
7917
8407
  // 将编辑容器添加到主div
7918
8408
  this.div.appendChild(this.editContainer);
@@ -7928,9 +8418,10 @@ class MowerMapOverlay {
7928
8418
  }
7929
8419
  // 获取当前拖拽状态
7930
8420
  getCurrentDragState() {
8421
+ // 返回基于地图中心点的经纬度偏移量
7931
8422
  return {
7932
- offsetX: this.latLngOffset.lng * 111320 + this.tempPixelOffset.x - this.initialOffset.x, // 简化的经度到米的转换
7933
- offsetY: this.latLngOffset.lat * 110540 + this.tempPixelOffset.y - this.initialOffset.y, // 简化的纬度到米的转换
8423
+ x: this.latLngOffset.lng + this.tempPixelOffset.x - this.initialOffset.x,
8424
+ y: this.latLngOffset.lat + this.tempPixelOffset.y - this.initialOffset.y,
7934
8425
  rotation: this.currentRotation - this.initialRotation,
7935
8426
  isDragging: this.isDragging,
7936
8427
  isRotating: this.isRotating,
@@ -7953,7 +8444,6 @@ class MowerMapOverlay {
7953
8444
  this.boundaryLabelsManager.collapseAllLabels();
7954
8445
  }
7955
8446
  this.dragCallbacks?.onDragStart?.(this.getCurrentDragState());
7956
- console.log('开始旋转操作');
7957
8447
  });
7958
8448
  // 旋转手柄的触摸事件
7959
8449
  this.rotateHandle.addEventListener('touchstart', (e) => {
@@ -7969,39 +8459,41 @@ class MowerMapOverlay {
7969
8459
  this.boundaryLabelsManager.collapseAllLabels();
7970
8460
  }
7971
8461
  this.dragCallbacks?.onDragStart?.(this.getCurrentDragState());
7972
- console.log('开始旋转操作(触摸)');
7973
- }, { passive: false });
7974
- // 拖拽手柄的鼠标事件
7975
- this.dragHandle.addEventListener('mousedown', (e) => {
7976
- e.preventDefault();
7977
- e.stopPropagation();
7978
- e.stopImmediatePropagation();
7979
- this.isDragging = true;
7980
- this.startPos = { x: e.clientX, y: e.clientY };
7981
- this.dragHandle.style.cursor = 'grabbing';
7982
- // 开始编辑时关闭所有展开的边界标签
7983
- if (this.boundaryLabelsManager) {
7984
- this.boundaryLabelsManager.collapseAllLabels();
7985
- }
7986
- this.dragCallbacks?.onDragStart?.(this.getCurrentDragState());
7987
- console.log('开始拖动操作(通过手柄)');
7988
- });
7989
- // 拖拽手柄的触摸事件
7990
- this.dragHandle.addEventListener('touchstart', (e) => {
7991
- e.preventDefault();
7992
- e.stopPropagation();
7993
- e.stopImmediatePropagation();
7994
- this.isDragging = true;
7995
- const touch = e.touches[0];
7996
- this.startPos = { x: touch.clientX, y: touch.clientY };
7997
- this.dragHandle.style.cursor = 'grabbing';
7998
- this.dragCallbacks?.onDragStart?.(this.getCurrentDragState());
7999
- console.log('开始拖动操作(通过手柄,触摸)');
8000
8462
  }, { passive: false });
8463
+ // 拖拽手柄的鼠标事件 - 仅在移动设备上启用
8464
+ if (isMobileDevice()) {
8465
+ this.dragHandle.addEventListener('mousedown', (e) => {
8466
+ e.preventDefault();
8467
+ e.stopPropagation();
8468
+ e.stopImmediatePropagation();
8469
+ this.isDragging = true;
8470
+ this.startPos = { x: e.clientX, y: e.clientY };
8471
+ this.dragHandle.style.cursor = 'grabbing';
8472
+ // 开始编辑时关闭所有展开的边界标签
8473
+ if (this.boundaryLabelsManager) {
8474
+ this.boundaryLabelsManager.collapseAllLabels();
8475
+ }
8476
+ this.dragCallbacks?.onDragStart?.(this.getCurrentDragState());
8477
+ });
8478
+ // 拖拽手柄的触摸事件
8479
+ this.dragHandle.addEventListener('touchstart', (e) => {
8480
+ e.preventDefault();
8481
+ e.stopPropagation();
8482
+ e.stopImmediatePropagation();
8483
+ this.isDragging = true;
8484
+ const touch = e.touches[0];
8485
+ this.startPos = { x: touch.clientX, y: touch.clientY };
8486
+ this.dragHandle.style.cursor = 'grabbing';
8487
+ this.dragCallbacks?.onDragStart?.(this.getCurrentDragState());
8488
+ }, { passive: false });
8489
+ }
8001
8490
  // 编辑容器的鼠标事件(整个区域拖拽)
8002
8491
  this.editContainer.addEventListener('mousedown', (e) => {
8003
- console.log('开始拖动操作(整个叠加层)');
8004
- if (e.target === this.dragHandle || e.target === this.rotateHandle) {
8492
+ // 在移动设备上,检查是否点击了拖拽手柄或旋转手柄
8493
+ // 在PC设备上,只检查旋转手柄(拖拽手柄已隐藏)
8494
+ const isDragHandleClick = isMobileDevice() && e.target === this.dragHandle;
8495
+ const isRotateHandleClick = e.target === this.rotateHandle;
8496
+ if (isDragHandleClick || isRotateHandleClick) {
8005
8497
  return;
8006
8498
  }
8007
8499
  e.preventDefault();
@@ -8018,8 +8510,11 @@ class MowerMapOverlay {
8018
8510
  });
8019
8511
  // 编辑容器的触摸事件(整个区域拖拽)
8020
8512
  this.editContainer.addEventListener('touchstart', (e) => {
8021
- console.log('开始拖动操作(整个叠加层,触摸)');
8022
- if (e.target === this.dragHandle || e.target === this.rotateHandle) {
8513
+ // 在移动设备上,检查是否点击了拖拽手柄或旋转手柄
8514
+ // 在PC设备上,只检查旋转手柄(拖拽手柄已隐藏)
8515
+ const isDragHandleClick = isMobileDevice() && e.target === this.dragHandle;
8516
+ const isRotateHandleClick = e.target === this.rotateHandle;
8517
+ if (isDragHandleClick || isRotateHandleClick) {
8023
8518
  return;
8024
8519
  }
8025
8520
  e.preventDefault();
@@ -8079,7 +8574,6 @@ class MowerMapOverlay {
8079
8574
  e.preventDefault();
8080
8575
  e.stopPropagation();
8081
8576
  e.stopImmediatePropagation();
8082
- console.log('结束编辑操作');
8083
8577
  }
8084
8578
  // 如果是拖拽结束,将像素偏移量转换为地理坐标偏移量
8085
8579
  if (this.isDragging) {
@@ -8101,7 +8595,6 @@ class MowerMapOverlay {
8101
8595
  e.preventDefault();
8102
8596
  e.stopPropagation();
8103
8597
  e.stopImmediatePropagation();
8104
- console.log('结束编辑操作(触摸)');
8105
8598
  }
8106
8599
  // 如果是拖拽结束,将像素偏移量转换为地理坐标偏移量
8107
8600
  if (this.isDragging) {
@@ -8207,11 +8700,12 @@ class MowerMapOverlay {
8207
8700
  // 确保旋转角度在0-360度范围内
8208
8701
  this.currentRotation = ((this.currentRotation % 360) + 360) % 360;
8209
8702
  // 应用旋转变换到DOM元素,同时保持位移
8703
+ // 更新边界标签的旋转角度,使其保持水平状态
8704
+ this.setManagerRotation(this.currentRotation);
8210
8705
  const transform = `translate(${this.tempPixelOffset.x}px, ${this.tempPixelOffset.y}px) rotate(${this.currentRotation}deg)`;
8211
8706
  this.div.style.transform = transform;
8212
8707
  // 更新鼠标起始位置为当前位置,为下次计算做准备
8213
8708
  this.startPos = { x: mouseCurrentX, y: mouseCurrentY };
8214
- console.log('旋转角度:', this.currentRotation, '角度增量:', angleDifferenceDegrees);
8215
8709
  }
8216
8710
  // 将像素偏移量转换为地理坐标偏移量
8217
8711
  convertPixelOffsetToLatLng() {
@@ -8241,17 +8735,8 @@ class MowerMapOverlay {
8241
8735
  // 累积更新地理坐标偏移量(不是直接赋值!)
8242
8736
  this.latLngOffset.lat += latOffset;
8243
8737
  this.latLngOffset.lng += lngOffset;
8244
- console.log('精确转换偏移量:', {
8245
- pixelOffset: this.tempPixelOffset,
8246
- centerLatLng: { lat: centerLatLng.lat(), lng: centerLatLng.lng() },
8247
- offsetLatLng: { lat: offsetLatLng.lat(), lng: offsetLatLng.lng() },
8248
- latOffset,
8249
- lngOffset,
8250
- newLatLngOffset: this.latLngOffset,
8251
- });
8252
8738
  // 重置临时像素偏移量
8253
8739
  this.tempPixelOffset = { x: 0, y: 0 };
8254
- // 重新绘制以应用新的地理坐标偏移量
8255
8740
  this.draw();
8256
8741
  }
8257
8742
  // 获取编辑数据
@@ -8287,8 +8772,6 @@ class MowerMapOverlay {
8287
8772
  editData: editData,
8288
8773
  timestamp: new Date().toISOString(),
8289
8774
  };
8290
- // 在这里可以添加保存逻辑,比如发送到服务器
8291
- console.log('保存编辑数据:', saveData);
8292
8775
  // 显示保存成功提示
8293
8776
  this.showSaveSuccess();
8294
8777
  return saveData;
@@ -8354,6 +8837,63 @@ class MowerMapOverlay {
8354
8837
  this.div.style.pointerEvents = 'none';
8355
8838
  }
8356
8839
  }
8840
+ /**
8841
+ * 设置旋转角度
8842
+ */
8843
+ setManagerRotation(rotation) {
8844
+ if (this.boundaryLabelsManager) {
8845
+ this.boundaryLabelsManager.setRotation(rotation);
8846
+ }
8847
+ if (this.antennaManager) {
8848
+ this.antennaManager.setRotation(rotation);
8849
+ }
8850
+ if (this.chargingPileManager) {
8851
+ this.chargingPileManager.setRotation(rotation);
8852
+ }
8853
+ }
8854
+ // 设置transform
8855
+ setTransform(transform) {
8856
+ if (typeof transform.x === 'number')
8857
+ this.tempPixelOffset.x = transform.x;
8858
+ if (typeof transform.y === 'number')
8859
+ this.tempPixelOffset.y = transform.y;
8860
+ if (typeof transform.rotation === 'number')
8861
+ this.currentRotation = transform.rotation;
8862
+ this.defaultTransform = {
8863
+ x: transform.x,
8864
+ y: transform.y,
8865
+ rotation: transform.rotation,
8866
+ };
8867
+ // defaultTransform的x对应经度偏移量,y对应纬度偏移量
8868
+ this.latLngOffset.lng = this.defaultTransform.x;
8869
+ this.latLngOffset.lat = this.defaultTransform.y;
8870
+ this.setManagerRotation(this.currentRotation);
8871
+ this.draw();
8872
+ }
8873
+ // 重置到默认的transform
8874
+ resetToDefaultTransform() {
8875
+ // 重置所有偏移和旋转相关的状态,x对应经度偏移,y对应纬度偏移
8876
+ this.latLngOffset.lng = this.defaultTransform.x;
8877
+ this.latLngOffset.lat = this.defaultTransform.y;
8878
+ this.tempPixelOffset = { x: 0, y: 0 };
8879
+ this.currentRotation = this.defaultTransform.rotation;
8880
+ this.initialOffset = { x: 0, y: 0 };
8881
+ this.initialRotation = 0;
8882
+ // 重新绘制
8883
+ // 重置边界标签的旋转角度
8884
+ if (this.boundaryLabelsManager) {
8885
+ this.boundaryLabelsManager.setRotation(this.defaultTransform.rotation);
8886
+ }
8887
+ if (this.antennaManager) {
8888
+ this.antennaManager.setRotation(this.defaultTransform.rotation);
8889
+ }
8890
+ if (this.chargingPileManager) {
8891
+ this.chargingPileManager.setRotation(this.defaultTransform.rotation);
8892
+ }
8893
+ this.draw();
8894
+ // 触发拖拽回调,通知外部状态更新
8895
+ this.dragCallbacks?.onDragEnd?.(this.getCurrentDragState());
8896
+ }
8357
8897
  initializeSvgMapView() {
8358
8898
  if (!this.offscreenContainer)
8359
8899
  return;
@@ -8364,8 +8904,10 @@ class MowerMapOverlay {
8364
8904
  this.loadMapData();
8365
8905
  // 加载路径数据
8366
8906
  if (this.pathData && this.svgMapView) {
8367
- this.loadPathData(this.pathData);
8907
+ this.loadPathData(this.pathData, this.mowPartitionData);
8368
8908
  }
8909
+ // 刷新绘制图层
8910
+ this.svgMapView.refresh();
8369
8911
  // 获取生成的SVG并添加到叠加层div中
8370
8912
  const svgElement = this.svgMapView.getSVG();
8371
8913
  if (svgElement) {
@@ -8388,9 +8930,6 @@ class MowerMapOverlay {
8388
8930
  // 使用现有的MapDataProcessor处理地图数据
8389
8931
  const elements = MapDataProcessor.processMapData(this.mapData, this.mapConfig);
8390
8932
  // 分离充电桩和天线元素,其他元素添加到SVG图层
8391
- // const svgElements = elements.filter(element =>
8392
- // element.type !== 'charging_pile' && element.type !== 'antenna'
8393
- // );
8394
8933
  const svgElements = elements.filter((element) => element.type !== 'charging_pile' && element.type !== 'antenna');
8395
8934
  const chargingPileElements = elements.filter((element) => element.type === 'charging_pile');
8396
8935
  // 处理SVG图层元素
@@ -8399,14 +8938,16 @@ class MowerMapOverlay {
8399
8938
  // 处理天线数据
8400
8939
  let antennaElements = [];
8401
8940
  if (this.antennaConfig.length > 0) {
8402
- antennaElements = AntennaDataBuilder.fromAntennaData(this.antennaConfig);
8941
+ antennaElements = AntennaDataBuilder.fromAntennaData(this.antennaConfig, this.mapConfig);
8403
8942
  }
8404
8943
  // 添加图层到SvgMapView
8405
8944
  const layers = drawLayer.getLayers();
8406
8945
  this.svgMapView.clear();
8407
- layers.forEach((layer) => {
8408
- this.svgMapView.addLayer(layer);
8409
- });
8946
+ this.svgMapView.addLayers(layers);
8947
+ const boundaryBorderLayer = this.svgMapView.getLayer(LAYER_DEFAULT_TYPE.BOUNDARY_BORDER);
8948
+ if (boundaryBorderLayer) {
8949
+ boundaryBorderLayer.setMowingBoundarys(this.mowPartitionData?.partitionIds || []);
8950
+ }
8410
8951
  this.createChargingPileManager();
8411
8952
  // 使用管理器处理充电桩和天线
8412
8953
  if (this.chargingPileManager && chargingPileElements.length > 0) {
@@ -8432,40 +8973,94 @@ class MowerMapOverlay {
8432
8973
  console.error('加载地图数据时出错:', error);
8433
8974
  }
8434
8975
  }
8435
- loadPathData(pathData) {
8976
+ loadPathData(pathData, mowPartitionData) {
8436
8977
  try {
8437
8978
  // 使用现有的PathDataProcessor处理路径数据
8438
8979
  const pathElements = PathDataProcessor.processPathData(pathData, this.mapConfig);
8980
+ const newPathElements = pathElements.map((pathElement) => {
8981
+ const { id, elements } = pathElement;
8982
+ const isMowBoundary = mowPartitionData && mowPartitionData?.partitionIds?.includes(id);
8983
+ if (isMowBoundary) {
8984
+ return {
8985
+ id: id,
8986
+ elements: elements.map((element) => {
8987
+ const isTransPath = element.pathType === PathSegmentType.TRANS;
8988
+ if (isTransPath) {
8989
+ return element;
8990
+ }
8991
+ return {
8992
+ ...element,
8993
+ style: {
8994
+ ...element.style,
8995
+ lineColor: DEFAULT_STYLES.path.mowingLineColor,
8996
+ },
8997
+ };
8998
+ }),
8999
+ };
9000
+ }
9001
+ else {
9002
+ return pathElement;
9003
+ }
9004
+ });
8439
9005
  const pathLayer = new PathLayer();
8440
- pathLayer.addElements(pathElements);
9006
+ pathLayer.addElements(newPathElements);
8441
9007
  // 添加图层到SvgMapView
8442
9008
  this.svgMapView.removeLayerByType(LAYER_DEFAULT_TYPE.PATH);
8443
9009
  this.svgMapView.addLayer(pathLayer);
8444
- console.log('pathElements----->', this.svgMapView.getLayers());
9010
+ this.svgMapView.renderLayer(LAYER_DEFAULT_TYPE.PATH);
8445
9011
  // 调用回调
8446
- const elementCount = pathElements.length;
9012
+ const elementCount = newPathElements.length;
8447
9013
  this.onPathLoad?.(elementCount);
8448
9014
  }
8449
9015
  catch (error) {
8450
9016
  console.error('加载路径数据时出错:', error);
8451
9017
  }
8452
9018
  }
9019
+ /**
9020
+ * 根据割草机位置,更新路径数据
9021
+ * @param position 割草机位置
9022
+ */
9023
+ updatePathDataByMowingPosition(position) {
9024
+ // 找到当前position所在的分区id,将该点更新到pathData中
9025
+ const currentPartitionId = getPartitionId(this.partitionBoundary, position.x, position.y);
9026
+ const processStateIsMowing = useProcessMowingState.getState().processStateIsMowing;
9027
+ if (currentPartitionId && this.pathData?.[currentPartitionId]) {
9028
+ const currentPathData = this.pathData[currentPartitionId];
9029
+ this.pathData[currentPartitionId] = {
9030
+ ...currentPathData,
9031
+ points: [
9032
+ ...(currentPathData?.points || []),
9033
+ {
9034
+ postureX: Number(position.x),
9035
+ postureY: Number(position.y),
9036
+ knifeRotation: processStateIsMowing && position.vehicleState === RobotStatus.MOWING ? '01' : '00', // "knifeRotation": "01",//刀盘是否转动 00-否 01-是
9037
+ pathType: '', //这里由于从实时路径是获取不到pathType的,所以这里暂时不传。让路径是否绘制取决于knifeRotation
9038
+ partitionId: currentPartitionId.toString(),
9039
+ },
9040
+ ],
9041
+ };
9042
+ this.updatePathData(this.pathData, this.mowPartitionData);
9043
+ }
9044
+ }
9045
+ updateMowPartitionData(mowPartitionData) {
9046
+ this.mowPartitionData = mowPartitionData;
9047
+ }
8453
9048
  /** 更新历史路径数据 */
8454
- updatePathData(pathData) {
9049
+ updatePathData(pathData, mowPartitionData) {
8455
9050
  if (!this.svgMapView || !pathData)
8456
9051
  return;
8457
9052
  // 找到pathLayer,将其删除,然后重新添加
8458
- this.loadPathData(pathData);
9053
+ this.loadPathData(pathData, mowPartitionData);
8459
9054
  }
9055
+ /** 更新边界标签信息 */
8460
9056
  updateBoundaryLabelInfo(pathJson) {
8461
9057
  if (!pathJson)
8462
9058
  return;
8463
9059
  const boundaryData = generateBoundaryData(this.mapData, pathJson);
8464
- console.log('boundaryData==', boundaryData);
9060
+ // console.log('boundaryData==', boundaryData);
8465
9061
  this.boundaryLabelsManager?.updateBoundaryData(boundaryData);
8466
9062
  }
8467
9063
  draw() {
8468
- console.log('draw方法被调用');
8469
9064
  // 防御性检查:如果this.div为null,说明onAdd还没被调用,直接返回
8470
9065
  if (!this.div) {
8471
9066
  return;
@@ -8523,13 +9118,17 @@ class MowerMapOverlay {
8523
9118
  this.div.style.transform = transform;
8524
9119
  }
8525
9120
  else {
8526
- // 非拖拽时:只应用旋转(如果有的话)
8527
- if (this.currentRotation !== 0) {
8528
- this.div.style.transform = `rotate(${this.currentRotation}deg)`;
9121
+ // 非拖拽时:应用当前偏移和旋转(包括默认值)
9122
+ const transforms = [];
9123
+ // 应用像素偏移
9124
+ if (this.tempPixelOffset.x !== 0 || this.tempPixelOffset.y !== 0) {
9125
+ transforms.push(`translate(${this.tempPixelOffset.x}px, ${this.tempPixelOffset.y}px)`);
8529
9126
  }
8530
- else {
8531
- this.div.style.transform = '';
9127
+ // 应用旋转
9128
+ if (this.currentRotation !== 0) {
9129
+ transforms.push(`rotate(${this.currentRotation}deg)`);
8532
9130
  }
9131
+ this.div.style.transform = transforms.join(' ');
8533
9132
  }
8534
9133
  // 更新SVG视图框以适应新的尺寸
8535
9134
  if (this.svgMapView) {
@@ -8549,11 +9148,11 @@ class MowerMapOverlay {
8549
9148
  this.updateManagerPositions(width, height);
8550
9149
  // 重绘的时候可能在动画中,这时候需要强制更新一次数据,不然会出现抖动的效果
8551
9150
  // 非动画的情况下,根据最新的数据实时同步就可以了
8552
- if (!this.mowerPostionManager?.animationFlag) {
8553
- this.mowerPostionManager?.updatePostionByLastPosition(this.mowerPositonConfig, 0);
9151
+ if (!this.mowerPositionManager?.animationFlag) {
9152
+ this.mowerPositionManager?.updatePositionByLastPosition(this.mowerPositionConfig);
8554
9153
  }
8555
9154
  else {
8556
- this.mowerPostionManager?.forceUpdatePosition();
9155
+ this.mowerPositionManager?.forceUpdatePosition();
8557
9156
  }
8558
9157
  if (this.antennaManager) {
8559
9158
  this.antennaManager.updateAntennaPosition();
@@ -8589,9 +9188,9 @@ class MowerMapOverlay {
8589
9188
  this.antennaManager = null;
8590
9189
  }
8591
9190
  // 清理割草机位置管理器
8592
- if (this.mowerPostionManager) {
8593
- this.mowerPostionManager.destroy();
8594
- this.mowerPostionManager = null;
9191
+ if (this.mowerPositionManager) {
9192
+ this.mowerPositionManager.destroy();
9193
+ this.mowerPositionManager = null;
8595
9194
  }
8596
9195
  // 清理编辑界面
8597
9196
  this.removeEditInterface();
@@ -8608,12 +9207,45 @@ class MowerMapOverlay {
8608
9207
  }
8609
9208
  // 显示/隐藏割草机位置
8610
9209
  setMowerPositionVisible(visible) {
8611
- if (this.mowerPostionManager) {
8612
- this.mowerPostionManager.setVisible(visible);
8613
- }
9210
+ if (this.mowerPositionManager) {
9211
+ this.mowerPositionManager.setVisible(visible);
9212
+ }
9213
+ }
9214
+ // 获取SvgMapView实例(用于debug)
9215
+ getSvgMapView() {
9216
+ return this.svgMapView;
9217
+ }
9218
+ }
9219
+
9220
+ // 获取车辆状态的中文文案
9221
+ const getVehicleStateText = (vehicleState) => {
9222
+ switch (vehicleState) {
9223
+ case RobotStatus.PARKED:
9224
+ return 'PARKED';
9225
+ case RobotStatus.CHARGING:
9226
+ return 'CHARGING';
9227
+ case RobotStatus.STANDBY:
9228
+ return 'STANDBY';
9229
+ case RobotStatus.MOWING:
9230
+ return 'MOWING';
9231
+ case RobotStatus.WORKING:
9232
+ return 'WORKING';
9233
+ case RobotStatus.MAPPING:
9234
+ return 'MAPPING';
9235
+ case RobotStatus.ERROR:
9236
+ return 'ERROR';
9237
+ case RobotStatus.UPGRADING:
9238
+ return 'UPGRADING';
9239
+ case RobotStatus.DISCONNECTED:
9240
+ return 'DISCONNECTED';
9241
+ case RobotStatus.TASK_DELAY:
9242
+ return 'TASK_DELAY';
9243
+ case RobotStatus.UNKNOWN:
9244
+ return '未知';
9245
+ default:
9246
+ return `未知状态(${vehicleState})`;
8614
9247
  }
8615
- }
8616
-
9248
+ };
8617
9249
  // 验证GPS坐标是否有效
8618
9250
  const isValidGpsCoordinate = (coordinate) => {
8619
9251
  if (!coordinate || coordinate.length < 2)
@@ -8627,47 +9259,89 @@ const isValidGpsCoordinate = (coordinate) => {
8627
9259
  !(Math.abs(lng) < 0.001 && Math.abs(lat) < 0.001) // 排除接近(0,0)的坐标
8628
9260
  );
8629
9261
  };
9262
+ // 旋转坐标点
9263
+ const rotateCoordinate = (point, center, angleRadians) => {
9264
+ const [x, y] = point;
9265
+ const [cx, cy] = center;
9266
+ // 将点移动到原点
9267
+ const dx = x - cx;
9268
+ const dy = y - cy;
9269
+ // 应用旋转矩阵
9270
+ const cos = Math.cos(angleRadians);
9271
+ const sin = Math.sin(angleRadians);
9272
+ const rotatedX = dx * cos - dy * sin;
9273
+ const rotatedY = dx * sin + dy * cos;
9274
+ // 移回原位置
9275
+ return [rotatedX + cx, rotatedY + cy];
9276
+ };
8630
9277
  // 获取有效的GPS边界
8631
- const getValidGpsBounds = (mapData) => {
9278
+ const getValidGpsBounds = (mapData, rotation = 0) => {
9279
+ let bounds;
8632
9280
  // 首先尝试使用地图数据中的GPS坐标
8633
9281
  if (isValidGpsCoordinate(mapData.sw_gps) && isValidGpsCoordinate(mapData.ne_gps)) {
8634
- return {
9282
+ bounds = {
8635
9283
  sw: mapData.sw_gps,
8636
9284
  ne: mapData.ne_gps,
8637
9285
  };
8638
9286
  }
8639
- // 如果GPS坐标无效,尝试从地图几何数据估算
8640
- const { sw, ne } = estimateGpsFromMapBounds(mapData);
8641
- console.log('sw, ne==', sw, ne);
8642
- if (sw && ne) {
8643
- console.warn('GPS坐标无效,使用地图几何数据估算边界:', sw, ne);
8644
- return {
8645
- sw: [sw[0], sw[1]],
8646
- ne: [ne[0], ne[1]],
9287
+ else {
9288
+ // 如果GPS坐标无效,尝试从地图几何数据估算
9289
+ const { sw, ne } = estimateGpsFromMapBounds(mapData);
9290
+ if (sw && ne) {
9291
+ console.warn('GPS坐标无效,使用地图几何数据估算边界:', sw, ne);
9292
+ bounds = {
9293
+ sw: [sw[0], sw[1]],
9294
+ ne: [ne[0], ne[1]],
9295
+ };
9296
+ }
9297
+ else {
9298
+ // 最后的fallback:使用默认坐标
9299
+ console.warn('无法获取有效的GPS边界,使用默认坐标');
9300
+ bounds = {
9301
+ sw: [-9.1562, -37.7503],
9302
+ ne: [31.247, 5.797],
9303
+ };
9304
+ }
9305
+ }
9306
+ // 如果有旋转角度,计算旋转后的边界
9307
+ if (rotation !== 0) {
9308
+ const angleRadians = (rotation * Math.PI) / 180; // 转换为弧度
9309
+ // 计算边界中心点
9310
+ const centerLng = (bounds.sw[0] + bounds.ne[0]) / 2;
9311
+ const centerLat = (bounds.sw[1] + bounds.ne[1]) / 2;
9312
+ const center = [centerLng, centerLat];
9313
+ // 旋转四个角点
9314
+ const sw = rotateCoordinate(bounds.sw, center, angleRadians);
9315
+ const ne = rotateCoordinate(bounds.ne, center, angleRadians);
9316
+ const se = rotateCoordinate([bounds.ne[0], bounds.sw[1]], center, angleRadians);
9317
+ const nw = rotateCoordinate([bounds.sw[0], bounds.ne[1]], center, angleRadians);
9318
+ // 计算旋转后的边界框(包含所有旋转后的点)
9319
+ const lngs = [sw[0], ne[0], se[0], nw[0]];
9320
+ const lats = [sw[1], ne[1], se[1], nw[1]];
9321
+ bounds = {
9322
+ sw: [Math.min(...lngs), Math.min(...lats)],
9323
+ ne: [Math.max(...lngs), Math.max(...lats)],
8647
9324
  };
8648
9325
  }
8649
- // 最后的fallback:使用默认坐标
8650
- console.warn('无法获取有效的GPS边界,使用默认坐标');
8651
- return {
8652
- sw: [-9.1562, -37.7503],
8653
- ne: [31.247, 5.797],
8654
- };
9326
+ return bounds;
8655
9327
  };
8656
9328
  // 默认配置
8657
9329
  const defaultMapConfig = DEFAULT_STYLES;
8658
9330
  // 地图渲染器组件
8659
- const MowerMapRenderer = forwardRef(({ mapConfig, modelType, mapRef, mapJson, pathJson, realTimeData, currentPositionData, antennaConfig, onMapLoad, onPathLoad, onZoomChange, onError, className, style, googleMapInstance, isEditMode = false, dragCallbacks, }, ref) => {
8660
- const svgMapViewRef = useRef(null);
9331
+ const MowerMapRenderer = forwardRef(({ edger = false, unitType = UnitsType.Imperial, language = 'en', mapConfig, modelType, mapRef, mapJson, pathJson, realTimeData, antennaConfig, onMapLoad, onPathLoad, onError, className, style, googleMapInstance, isEditMode = false, dragCallbacks, defaultTransform, debug = false, }, ref) => {
8661
9332
  const [elementCount, setElementCount] = useState(0);
8662
9333
  const [pathCount, setPathCount] = useState(0);
8663
- const [zoom, setZoom] = useState(1);
8664
9334
  const [currentError, setCurrentError] = useState(null);
8665
9335
  const overlayRef = useRef(null);
8666
9336
  // const mapRef = useMap();
8667
9337
  const [isGoogleMapsReady, setIsGoogleMapsReady] = useState(false);
8668
9338
  const [hasInitializedBounds, setHasInitializedBounds] = useState(false);
8669
- const { clearSubBoundaryBorder } = useSubBoundaryBorderStore();
9339
+ const { clearSubBoundaryBorder, clearObstacles, clearSvgElements } = useSubBoundaryBorderStore();
8670
9340
  const currentProcessMowingStatusRef = useRef(false);
9341
+ const { updateProcessStateIsMowing, processStateIsMowing } = useProcessMowingState();
9342
+ const [mowPartitionData, setMowPartitionData] = useState(null);
9343
+ // Debug相关状态
9344
+ const [debugInfo, setDebugInfo] = useState({});
8671
9345
  // 处理地图分区边界
8672
9346
  const partitionBoundary = useMemo(() => {
8673
9347
  const allBoundaryElements = [];
@@ -8690,22 +9364,67 @@ const MowerMapRenderer = forwardRef(({ mapConfig, modelType, mapRef, mapJson, pa
8690
9364
  }, [mapConfig]);
8691
9365
  const mergedAntennaConfig = useMemo(() => antennaConfig, [antennaConfig]);
8692
9366
  const mowerPositionData = useMemo(() => {
9367
+ // realTimeData 中包含三个种类的数据,之需要实时坐标的数据即可。
9368
+ if (!realTimeData || realTimeData.length === 0)
9369
+ return {
9370
+ postureX: 0,
9371
+ postureY: 0,
9372
+ postureTheta: 0,
9373
+ vehicleState: RobotStatus.DISCONNECTED,
9374
+ };
9375
+ let currentPositionData;
9376
+ if (realTimeData.length === 1 && realTimeData[0].type === RealTimeDataType.LOCATION) {
9377
+ currentPositionData = realTimeData[0];
9378
+ }
9379
+ else {
9380
+ currentPositionData = realTimeData?.find((item) => item.type === RealTimeDataType.LOCATION);
9381
+ }
9382
+ if (!currentPositionData)
9383
+ return undefined;
8693
9384
  return {
8694
9385
  postureTheta: currentPositionData?.postureTheta
8695
9386
  ? Number(currentPositionData.postureTheta)
8696
9387
  : 0,
8697
9388
  postureX: currentPositionData?.postureX ? Number(currentPositionData.postureX) : 0,
8698
9389
  postureY: currentPositionData?.postureY ? Number(currentPositionData.postureY) : 0,
8699
- vehicleState: currentPositionData?.vehicleState || RobotStatus.STANDBY,
8700
- vehicleModel: modelType?.toLowerCase() || '',
9390
+ lastPostureTheta: currentPositionData?.lastPostureTheta
9391
+ ? Number(currentPositionData.lastPostureTheta)
9392
+ : 0,
9393
+ lastPostureX: currentPositionData?.lastPostureX
9394
+ ? Number(currentPositionData.lastPostureX)
9395
+ : 0,
9396
+ lastPostureY: currentPositionData?.lastPostureY
9397
+ ? Number(currentPositionData.lastPostureY)
9398
+ : 0,
9399
+ vehicleState: currentPositionData?.vehicleState || RobotStatus.DISCONNECTED,
8701
9400
  };
8702
- }, [currentPositionData, modelType]);
8703
- console.log('mowerPositionData==', currentPositionData, mowerPositionData);
9401
+ }, [realTimeData, modelType]);
8704
9402
  // 处理错误
8705
9403
  const handleError = (error) => {
8706
9404
  setCurrentError(error);
8707
9405
  onError?.(error);
8708
9406
  };
9407
+ const fitBounds = useCallback(() => {
9408
+ if (!mapJson || !mapRef)
9409
+ return null;
9410
+ // 计算边界
9411
+ const bounds = calculateMapBounds(mapJson);
9412
+ if (!bounds) {
9413
+ handleError('无法计算地图边界');
9414
+ return;
9415
+ }
9416
+ // 将自定义边界转换为Google Maps LatLngBounds(使用有效的GPS坐标)
9417
+ const validBounds = getValidGpsBounds(mapJson, defaultTransform?.rotation);
9418
+ // 地图数据中的坐标格式是 [longitude, latitude]
9419
+ const swLat = validBounds.sw[1] + defaultTransform.y;
9420
+ const swLng = validBounds.sw[0] + defaultTransform.x;
9421
+ const neLat = validBounds.ne[1] + defaultTransform.y;
9422
+ const neLng = validBounds.ne[0] + defaultTransform.x;
9423
+ const googleBounds = new window.google.maps.LatLngBounds(new window.google.maps.LatLng(swLat, swLng), // 西南角
9424
+ new window.google.maps.LatLng(neLat, neLng) // 东北角
9425
+ );
9426
+ mapRef.fitBounds(googleBounds);
9427
+ }, [mapJson, mapRef, defaultTransform]);
8709
9428
  // 初始化Google Maps叠加层
8710
9429
  const initializeGoogleMapsOverlay = async () => {
8711
9430
  if (!mapJson)
@@ -8729,7 +9448,8 @@ const MowerMapRenderer = forwardRef(({ mapConfig, modelType, mapRef, mapJson, pa
8729
9448
  return;
8730
9449
  }
8731
9450
  // 将自定义边界转换为Google Maps LatLngBounds(使用有效的GPS坐标)
8732
- const validBounds = getValidGpsBounds(mapJson);
9451
+ // 这里需要使用0度,需要原始的坐标点去计算元素的实际位置,只有在fitbounds的时候才需要使用旋转后的坐标点
9452
+ const validBounds = getValidGpsBounds(mapJson, 0);
8733
9453
  // 地图数据中的坐标格式是 [longitude, latitude]
8734
9454
  const swLat = validBounds.sw[1];
8735
9455
  const swLng = validBounds.sw[0];
@@ -8738,17 +9458,13 @@ const MowerMapRenderer = forwardRef(({ mapConfig, modelType, mapRef, mapJson, pa
8738
9458
  const googleBounds = new window.google.maps.LatLngBounds(new window.google.maps.LatLng(swLat, swLng), // 西南角
8739
9459
  new window.google.maps.LatLng(neLat, neLng) // 东北角
8740
9460
  );
8741
- console.log('使用有效GPS坐标创建边界:', {
8742
- sw: { lat: swLat, lng: swLng },
8743
- ne: { lat: neLat, lng: neLng },
8744
- });
8745
9461
  // 如果已经存在叠加层,先移除它
8746
9462
  if (overlayRef.current) {
8747
9463
  overlayRef.current.setMap(null);
8748
9464
  overlayRef.current = null;
8749
9465
  }
8750
9466
  // 创建叠加层
8751
- const overlay = new MowerMapOverlay(googleBounds, mapJson, mowerPositionData, pathJson || {}, isEditMode, mergedMapConfig, mergedAntennaConfig, (count) => {
9467
+ const overlay = new MowerMapOverlay(googleBounds, mapJson, partitionBoundary, mowerPositionData, modelType, pathJson || {}, isEditMode, unitType, language, mergedMapConfig, mergedAntennaConfig, null, defaultTransform, (count) => {
8752
9468
  setElementCount(count);
8753
9469
  onMapLoad?.(count);
8754
9470
  }, (count) => {
@@ -8758,6 +9474,7 @@ const MowerMapRenderer = forwardRef(({ mapConfig, modelType, mapRef, mapJson, pa
8758
9474
  // 设置地图
8759
9475
  overlay.setMap(mapInstance);
8760
9476
  overlayRef.current = overlay;
9477
+ overlay.setEdger(edger);
8761
9478
  // 只在首次初始化时自适应视图
8762
9479
  if (!hasInitializedBounds) {
8763
9480
  mapInstance.fitBounds(googleBounds);
@@ -8769,39 +9486,33 @@ const MowerMapRenderer = forwardRef(({ mapConfig, modelType, mapRef, mapJson, pa
8769
9486
  handleError(`初始化Google Maps叠加层失败: ${error instanceof Error ? error.message : String(error)}`);
8770
9487
  }
8771
9488
  };
8772
- const formatRealTimeData = async (realTimeData) => {
8773
- // await sleep(1000);
8774
- const newRealTimeData = handleRealTimeData({
8775
- realTimeData,
8776
- isMowing: currentProcessMowingStatusRef.current,
8777
- pathData: pathJson,
8778
- partitionBoundary,
8779
- });
8780
- console.log('newRealTimeData==', newRealTimeData);
8781
- // currentProcessMowingStatusRef.current = newRealTimeData.isMowing;
8782
- currentProcessMowingStatusRef.current = true;
8783
- // 调用overlay的updatePathData方法,更新历史路径数据
8784
- if (overlayRef.current) {
8785
- overlayRef.current.updatePathData(newRealTimeData.pathData);
8786
- }
8787
- overlayRef.current.updateBoundaryLabelInfo(newRealTimeData.pathData);
8788
- };
9489
+ const resetInCharginPie = useCallback(() => {
9490
+ const elements = MapDataProcessor.processMapData(mapJson, mergedMapConfig);
9491
+ const chargingPiles = elements.find((element) => element.type === 'charging_pile');
9492
+ if (!overlayRef.current)
9493
+ return;
9494
+ // 如果在充电桩上,则直接更新位置到充电桩的位置
9495
+ overlayRef.current.updatePosition({
9496
+ ...mowerPositionData,
9497
+ postureX: chargingPiles?.originalData.position[0],
9498
+ postureY: chargingPiles?.originalData.position[1],
9499
+ postureTheta: chargingPiles?.originalData.direction - Math.PI || 0,
9500
+ }, 0);
9501
+ }, [mapJson, mowerPositionData]);
8789
9502
  // 初始化效果
8790
9503
  useEffect(() => {
8791
9504
  initializeGoogleMapsOverlay();
8792
- console.log('init mow map');
8793
9505
  // 清理函数
8794
9506
  return () => {
8795
9507
  clearSubBoundaryBorder();
9508
+ clearObstacles();
9509
+ clearSvgElements();
9510
+ updateProcessStateIsMowing(false);
8796
9511
  currentProcessMowingStatusRef.current = false;
8797
9512
  if (overlayRef.current) {
8798
9513
  overlayRef.current.setMap(null);
8799
9514
  overlayRef.current = null;
8800
9515
  }
8801
- if (svgMapViewRef.current) {
8802
- svgMapViewRef.current.destroy();
8803
- svgMapViewRef.current = null;
8804
- }
8805
9516
  };
8806
9517
  }, [mapJson, pathJson, mergedMapConfig, mergedAntennaConfig]);
8807
9518
  // 监听编辑模式变化
@@ -8821,13 +9532,12 @@ const MowerMapRenderer = forwardRef(({ mapConfig, modelType, mapRef, mapJson, pa
8821
9532
  return;
8822
9533
  const elements = MapDataProcessor.processMapData(mapJson, mergedMapConfig);
8823
9534
  const chargingPiles = elements.find((element) => element.type === 'charging_pile');
8824
- console.log('chargingPiles==', chargingPiles, currentPositionData);
8825
9535
  if (!mowerPositionData || !overlayRef.current)
8826
9536
  return;
8827
9537
  const inChargingPiles = [RobotStatus.CHARGING, RobotStatus.PARKED];
8828
9538
  const isOffLine = mowerPositionData.vehicleState === RobotStatus.DISCONNECTED;
8829
9539
  const isInChargingPile = inChargingPiles.includes(mowerPositionData.vehicleState);
8830
- // 如果在充电桩上,则更新位置到充电桩的位置
9540
+ // 如果在充电桩上,则直接更新位置到充电桩的位置
8831
9541
  if (isInChargingPile) {
8832
9542
  overlayRef.current.updatePosition({
8833
9543
  ...mowerPositionData,
@@ -8837,68 +9547,247 @@ const MowerMapRenderer = forwardRef(({ mapConfig, modelType, mapRef, mapJson, pa
8837
9547
  }, 0);
8838
9548
  }
8839
9549
  else {
9550
+ // 如果车辆是disabled状态或者超出边界(默认超过1000m),则更新位置到上一次的位置
8840
9551
  const positonOutOfRange = isOutOfRange(mowerPositionData);
8841
9552
  const positionValid = isInvalidPosition(mowerPositionData);
8842
- const isStandyBy = mowerPositionData.vehicleState === RobotStatus.STANDBY;
9553
+ const isStandby = mowerPositionData.vehicleState === RobotStatus.STANDBY;
8843
9554
  if (positonOutOfRange || positionValid || isOffLine) {
8844
- overlayRef.current.updatePostionByLastPosition({
8845
- ...mowerPositionData,
8846
- postureX: chargingPiles?.originalData.position[0],
8847
- postureY: chargingPiles?.originalData.position[1],
8848
- postureTheta: chargingPiles?.originalData.direction - Math.PI || 0,
8849
- }, isStandyBy ? 0 : 2200);
9555
+ // 初始信息是通过后端接口获取的,此时当前位置数据不可用的时候,可以取上一次的位置数据
9556
+ // mowerPositionData 中可能会包含上一次的位置数据,
9557
+ const lastPostureX = mowerPositionData.lastPostureX;
9558
+ const lastPostureY = mowerPositionData.lastPostureY;
9559
+ const lastPostureTheta = mowerPositionData.lastPostureTheta;
9560
+ if (lastPostureX && lastPostureY && lastPostureTheta) {
9561
+ overlayRef.current.updatePositionByLastPosition(mowerPositionData);
9562
+ }
9563
+ else {
9564
+ overlayRef.current.updatePositionByLastPosition({
9565
+ ...mowerPositionData,
9566
+ postureX: chargingPiles?.originalData.position[0],
9567
+ postureY: chargingPiles?.originalData.position[1],
9568
+ postureTheta: chargingPiles?.originalData.direction - Math.PI || 0,
9569
+ });
9570
+ }
8850
9571
  }
8851
9572
  else {
8852
- overlayRef.current.updatePosition(mowerPositionData, isStandyBy ? 0 : 2200);
9573
+ overlayRef.current.updatePosition(mowerPositionData, isStandby ? 0 : 2000);
8853
9574
  }
8854
9575
  }
8855
- }, [currentPositionData]);
9576
+ }, [mowerPositionData]);
9577
+ // 更新debug信息
8856
9578
  useEffect(() => {
8857
- if (!mapJson ||
8858
- !pathJson ||
8859
- !realTimeData ||
8860
- realTimeData.length === 0 ||
8861
- !Array.isArray(realTimeData))
9579
+ if (!debug)
8862
9580
  return;
8863
- // 根据后端推送的实时数据,进行不同处理
8864
- // TODO:需要根据返回的数据,处理车辆的移动位置
8865
- // 这里是根据返回的数据,处理实时轨迹
8866
- formatRealTimeData(realTimeData);
8867
- }, [realTimeData, mapJson, pathJson]);
8868
- // 提供ref方法
8869
- useImperativeHandle(ref, () => ({
8870
- setZoom: (newZoom) => {
8871
- setZoom(newZoom);
8872
- if (svgMapViewRef.current) {
8873
- svgMapViewRef.current.setZoom(newZoom);
9581
+ const updateDebugInfo = () => {
9582
+ const newDebugInfo = {};
9583
+ // 获取地图GPS边界
9584
+ if (mapJson) {
9585
+ newDebugInfo.mapBounds = getValidGpsBounds(mapJson, defaultTransform?.rotation);
8874
9586
  }
8875
- onZoomChange?.(newZoom);
8876
- },
8877
- getZoom: () => zoom,
8878
- resetView: () => {
8879
- if (svgMapViewRef.current) {
8880
- svgMapViewRef.current.resetTransform();
9587
+ // 获取SVG viewBox信息
9588
+ if (overlayRef.current) {
9589
+ const overlay = overlayRef.current;
9590
+ const svgMapView = overlay.getSvgMapView?.();
9591
+ if (svgMapView) {
9592
+ const viewBoxInfo = svgMapView.getViewBoxInfo();
9593
+ // 计算实际的米单位数据(除以缩放比例)
9594
+ const SCALE_FACTOR = 50; // 根据项目中的缩放因子
9595
+ newDebugInfo.viewBox = {
9596
+ x: viewBoxInfo.x / SCALE_FACTOR,
9597
+ y: viewBoxInfo.y / SCALE_FACTOR,
9598
+ width: viewBoxInfo.width / SCALE_FACTOR,
9599
+ height: viewBoxInfo.height / SCALE_FACTOR,
9600
+ scale: SCALE_FACTOR,
9601
+ // 计算左下角和右上角坐标
9602
+ sw: {
9603
+ x: viewBoxInfo.x / SCALE_FACTOR,
9604
+ y: viewBoxInfo.y / SCALE_FACTOR,
9605
+ },
9606
+ ne: {
9607
+ x: (viewBoxInfo.x + viewBoxInfo.width) / SCALE_FACTOR,
9608
+ y: (viewBoxInfo.y + viewBoxInfo.height) / SCALE_FACTOR,
9609
+ },
9610
+ };
9611
+ }
8881
9612
  }
8882
- setZoom(1);
8883
- onZoomChange?.(1);
8884
- },
8885
- clear: () => {
8886
- if (svgMapViewRef.current) {
8887
- svgMapViewRef.current.clear();
9613
+ // 获取当前割草机位置
9614
+ if (mowerPositionData) {
9615
+ newDebugInfo.mowerPosition = {
9616
+ x: mowerPositionData.postureX || 0,
9617
+ y: mowerPositionData.postureY || 0,
9618
+ theta: mowerPositionData.postureTheta || 0,
9619
+ lastX: mowerPositionData.lastPostureX || 0,
9620
+ lastY: mowerPositionData.lastPostureY || 0,
9621
+ lastTheta: mowerPositionData.lastPostureTheta || 0,
9622
+ vehicleState: mowerPositionData.vehicleState || RobotStatus.UNKNOWN,
9623
+ vehicleStateText: getVehicleStateText(mowerPositionData.vehicleState || RobotStatus.UNKNOWN),
9624
+ };
8888
9625
  }
8889
- setElementCount(0);
8890
- setPathCount(0);
8891
- },
8892
- getSvgElement: () => {
8893
- return svgMapViewRef.current ? svgMapViewRef.current.getSVG() : null;
8894
- },
8895
- fitToView: () => {
8896
- if (svgMapViewRef.current && mapJson) {
8897
- const bounds = calculateMapBounds(mapJson);
8898
- if (bounds) {
8899
- svgMapViewRef.current.fitToView(bounds);
9626
+ // 获取当前割草地块数据
9627
+ newDebugInfo.partitionData = mowPartitionData;
9628
+ setDebugInfo(newDebugInfo);
9629
+ };
9630
+ updateDebugInfo();
9631
+ }, [debug, mapJson, mowerPositionData, mowPartitionData, defaultTransform]);
9632
+ // 当关键数据变化时立即更新debug信息
9633
+ useEffect(() => {
9634
+ if (!debug)
9635
+ return;
9636
+ const updateDebugInfo = () => {
9637
+ const newDebugInfo = {};
9638
+ // 获取地图GPS边界
9639
+ if (mapJson) {
9640
+ newDebugInfo.mapBounds = getValidGpsBounds(mapJson, defaultTransform?.rotation);
9641
+ }
9642
+ // 获取SVG viewBox信息
9643
+ if (overlayRef.current) {
9644
+ const overlay = overlayRef.current;
9645
+ const svgMapView = overlay.getSvgMapView?.();
9646
+ if (svgMapView) {
9647
+ const viewBoxInfo = svgMapView.getViewBoxInfo();
9648
+ // 计算实际的米单位数据(除以缩放比例)
9649
+ const SCALE_FACTOR = 50; // 根据项目中的缩放因子
9650
+ newDebugInfo.viewBox = {
9651
+ x: viewBoxInfo.x / SCALE_FACTOR,
9652
+ y: viewBoxInfo.y / SCALE_FACTOR,
9653
+ width: viewBoxInfo.width / SCALE_FACTOR,
9654
+ height: viewBoxInfo.height / SCALE_FACTOR,
9655
+ scale: SCALE_FACTOR,
9656
+ // 计算左下角和右上角坐标
9657
+ sw: {
9658
+ x: viewBoxInfo.x / SCALE_FACTOR,
9659
+ y: viewBoxInfo.y / SCALE_FACTOR,
9660
+ },
9661
+ ne: {
9662
+ x: (viewBoxInfo.x + viewBoxInfo.width) / SCALE_FACTOR,
9663
+ y: (viewBoxInfo.y + viewBoxInfo.height) / SCALE_FACTOR,
9664
+ },
9665
+ };
8900
9666
  }
8901
9667
  }
9668
+ // 获取当前割草机位置
9669
+ if (mowerPositionData) {
9670
+ newDebugInfo.mowerPosition = {
9671
+ x: mowerPositionData.postureX || 0,
9672
+ y: mowerPositionData.postureY || 0,
9673
+ theta: mowerPositionData.postureTheta || 0,
9674
+ lastX: mowerPositionData.lastPostureX || 0,
9675
+ lastY: mowerPositionData.lastPostureY || 0,
9676
+ lastTheta: mowerPositionData.lastPostureTheta || 0,
9677
+ vehicleState: mowerPositionData.vehicleState || RobotStatus.UNKNOWN,
9678
+ vehicleStateText: getVehicleStateText(mowerPositionData.vehicleState || RobotStatus.UNKNOWN),
9679
+ };
9680
+ }
9681
+ // 获取当前割草地块数据
9682
+ newDebugInfo.partitionData = mowPartitionData;
9683
+ setDebugInfo(newDebugInfo);
9684
+ };
9685
+ updateDebugInfo();
9686
+ }, [debug, mowerPositionData, mowPartitionData]);
9687
+ useEffect(() => {
9688
+ if (!realTimeData || realTimeData.length === 0 || !Array.isArray(realTimeData)) {
9689
+ return;
9690
+ }
9691
+ let curMowPartitionData = mowPartitionData;
9692
+ // realtime中包含当前割草任务的数据,根据数据进行path路径和边界的高亮操作,
9693
+ const mowingPartition = realTimeData.find((item) => item.type === RealTimeDataType.PARTITION);
9694
+ if (mowingPartition) {
9695
+ setMowPartitionData(mowingPartition);
9696
+ curMowPartitionData = mowingPartition;
9697
+ }
9698
+ const positionData = realTimeData?.find((item) => item?.type === RealTimeDataType.LOCATION);
9699
+ const statusData = realTimeData?.find((item) => item?.type === RealTimeDataType.STATUS);
9700
+ if (statusData || positionData) {
9701
+ const currentStatus = statusData?.vehicleState || positionData?.vehicleState;
9702
+ // 车辆回桩不会回传最后的park的位置,所以根据实时数据的状态数据判断车辆回到桩上
9703
+ if ([RobotStatus.CHARGING, RobotStatus.PARKED].includes(currentStatus || RobotStatus.UNKNOWN)) {
9704
+ resetInCharginPie();
9705
+ }
9706
+ else if (currentStatus === RobotStatus.WORKING) {
9707
+ // 兜底收不到割草地块的实时数据,使用状态来兜底
9708
+ overlayRef.current.resetBorderLayerHighlight();
9709
+ setMowPartitionData({});
9710
+ curMowPartitionData = {};
9711
+ }
9712
+ else if (currentStatus === RobotStatus.MOWING &&
9713
+ curMowPartitionData &&
9714
+ !curMowPartitionData?.partitionIds) {
9715
+ // 如果当前是割草状态,但是地块数据初始化过且不存在则认为是全局割草,则把所有地块都高亮
9716
+ const allPartitionIds = mapJson?.sub_maps?.map((item) => item?.id);
9717
+ setMowPartitionData({
9718
+ partitionIds: allPartitionIds,
9719
+ });
9720
+ curMowPartitionData = {
9721
+ partitionIds: allPartitionIds,
9722
+ };
9723
+ }
9724
+ }
9725
+ if (!mapJson || !pathJson || !overlayRef.current)
9726
+ return;
9727
+ // 根据后端推送的实时数据,进行不同处理
9728
+ if (curMowPartitionData) {
9729
+ const isMowing = curMowPartitionData?.partitionIds && curMowPartitionData.partitionIds.length > 0;
9730
+ overlayRef.current.updateMowPartitionData(curMowPartitionData);
9731
+ if (!isMowing) {
9732
+ overlayRef.current.resetBorderLayerHighlight();
9733
+ }
9734
+ else {
9735
+ overlayRef.current.setBorderLayerHighlight(curMowPartitionData);
9736
+ }
9737
+ }
9738
+ // 如果一次性推送的是多条数据,则把多条数据处理后存入pathData,然后更新路径数据和边界标签信息
9739
+ // 如果一次只推送一条数据,则只解析里面的进度数据,然后更新边界标签信息,剩下的实时轨迹数据由车辆运动产生时存入pathData
9740
+ if (realTimeData.length > 1) {
9741
+ const { pathData, isMowing } = handleMultipleRealTimeData({
9742
+ realTimeData,
9743
+ isMowing: processStateIsMowing,
9744
+ pathData: pathJson,
9745
+ partitionBoundary,
9746
+ });
9747
+ updateProcessStateIsMowing(isMowing);
9748
+ if (pathData) {
9749
+ overlayRef.current.updatePathData(pathData, curMowPartitionData);
9750
+ overlayRef.current.updateBoundaryLabelInfo(pathData);
9751
+ }
9752
+ }
9753
+ else {
9754
+ const { isMowing, pathData } = getProcessMowingDataFromRealTimeData({
9755
+ realTimeData,
9756
+ isMowing: processStateIsMowing,
9757
+ pathData: pathJson,
9758
+ });
9759
+ updateProcessStateIsMowing(isMowing);
9760
+ overlayRef.current.updatePathData(pathData, curMowPartitionData);
9761
+ // 更新进度数据
9762
+ if (pathData) {
9763
+ overlayRef.current.updateBoundaryLabelInfo(pathData);
9764
+ }
9765
+ }
9766
+ }, [realTimeData, mapJson, pathJson]);
9767
+ useEffect(() => {
9768
+ if (!overlayRef.current || !defaultTransform)
9769
+ return;
9770
+ overlayRef.current?.setTransform(defaultTransform);
9771
+ const validBounds = getValidGpsBounds(mapJson, defaultTransform?.rotation);
9772
+ // 地图数据中的坐标格式是 [longitude, latitude]
9773
+ const swLat = validBounds.sw[1] + defaultTransform.y;
9774
+ const swLng = validBounds.sw[0] + defaultTransform.x;
9775
+ const neLat = validBounds.ne[1] + defaultTransform.y;
9776
+ const neLng = validBounds.ne[0] + defaultTransform.x;
9777
+ const googleBounds = new window.google.maps.LatLngBounds(new window.google.maps.LatLng(swLat, swLng), // 西南角
9778
+ new window.google.maps.LatLng(neLat, neLng) // 东北角
9779
+ );
9780
+ mapRef.fitBounds(googleBounds);
9781
+ }, [defaultTransform]);
9782
+ useEffect(() => {
9783
+ if (!overlayRef || !overlayRef.current)
9784
+ return;
9785
+ overlayRef.current.setEdger(edger);
9786
+ }, [edger]);
9787
+ // 提供ref方法
9788
+ useImperativeHandle(ref, () => ({
9789
+ fitToView: () => {
9790
+ fitBounds();
8902
9791
  },
8903
9792
  getOverlay: () => {
8904
9793
  return overlayRef.current;
@@ -8927,13 +9816,35 @@ const MowerMapRenderer = forwardRef(({ mapConfig, modelType, mapRef, mapJson, pa
8927
9816
  getElementCount: () => elementCount,
8928
9817
  getPathCount: () => pathCount,
8929
9818
  isGoogleMapsReady: () => isGoogleMapsReady,
9819
+ setTransform: (t) => overlayRef.current?.setTransform(t),
9820
+ resetToDefaultTransform: () => overlayRef.current?.resetToDefaultTransform(),
8930
9821
  }));
9822
+ // Debug信息组件
9823
+ const DebugInfo = () => {
9824
+ if (!debug)
9825
+ return null;
9826
+ return (jsxs("div", { style: {
9827
+ position: 'fixed',
9828
+ bottom: '10px',
9829
+ left: '10px',
9830
+ backgroundColor: 'rgba(0, 0, 0, 0.8)',
9831
+ color: 'white',
9832
+ padding: '10px',
9833
+ borderRadius: '5px',
9834
+ fontSize: '12px',
9835
+ fontFamily: 'monospace',
9836
+ zIndex: 10000,
9837
+ maxWidth: '300px',
9838
+ lineHeight: '1.4',
9839
+ }, children: [jsx("div", { style: { fontWeight: 'bold', marginBottom: '8px' }, children: "\uD83D\uDC1B Debug Info" }), debugInfo.mapBounds && (jsxs("div", { style: { marginBottom: '6px' }, children: [jsx("div", { style: { fontWeight: 'bold' }, children: "\uD83D\uDCCD Map GPS Bounds:" }), jsxs("div", { children: ["SW: [", debugInfo.mapBounds.sw[0].toFixed(6), ", ", debugInfo.mapBounds.sw[1].toFixed(6), "]"] }), jsxs("div", { children: ["NE: [", debugInfo.mapBounds.ne[0].toFixed(6), ", ", debugInfo.mapBounds.ne[1].toFixed(6), "]"] })] })), debugInfo.viewBox && (jsxs("div", { style: { marginBottom: '6px' }, children: [jsx("div", { style: { fontWeight: 'bold' }, children: "\uD83D\uDCD0 SVG ViewBox (meters):" }), jsxs("div", { children: ["SW: [", debugInfo.viewBox.sw.x.toFixed(2), ", ", debugInfo.viewBox.sw.y.toFixed(2), "]"] }), jsxs("div", { children: ["NE: [", debugInfo.viewBox.ne.x.toFixed(2), ", ", debugInfo.viewBox.ne.y.toFixed(2), "]"] }), jsxs("div", { children: ["Size: ", debugInfo.viewBox.width.toFixed(2), "m \u00D7 ", debugInfo.viewBox.height.toFixed(2), "m"] }), jsxs("div", { children: ["Scale: 1:", debugInfo.viewBox.scale] })] })), debugInfo.mowerPosition && (jsxs("div", { style: { marginBottom: '6px' }, children: [jsx("div", { style: { fontWeight: 'bold' }, children: "\uD83D\uDE9C Mower Position:" }), jsxs("div", { children: ["Current: X=", debugInfo.mowerPosition.x.toFixed(2), ", Y=", debugInfo.mowerPosition.y.toFixed(2)] }), jsxs("div", { children: ["Theta: ", ((debugInfo.mowerPosition.theta * 180) / Math.PI).toFixed(1), "\u00B0"] }), jsxs("div", { children: ["Last: X=", debugInfo.mowerPosition.lastX.toFixed(2), ", Y=", debugInfo.mowerPosition.lastY.toFixed(2)] }), jsxs("div", { children: ["Last Theta: ", ((debugInfo.mowerPosition.lastTheta * 180) / Math.PI).toFixed(1), "\u00B0"] }), jsxs("div", { children: ["Status: ", debugInfo.mowerPosition.vehicleStateText, " (", debugInfo.mowerPosition.vehicleState, ")"] })] })), debugInfo.partitionData && (jsxs("div", { style: { marginBottom: '6px' }, children: [jsx("div", { style: { fontWeight: 'bold' }, children: "\uD83D\uDD32 Mow Partition Data:" }), jsxs("div", { children: ["Type: ", debugInfo.partitionData.type || 'N/A'] }), debugInfo.partitionData.partitionIds &&
9840
+ debugInfo.partitionData.partitionIds.length > 0 ? (jsxs("div", { children: ["Active IDs: [", debugInfo.partitionData.partitionIds.join(', '), "]"] })) : (jsx("div", { children: "No active partitions" })), debugInfo.partitionData.time && (jsxs("div", { children: ["Updated: ", new Date(debugInfo.partitionData.time).toLocaleTimeString()] }))] })), !debugInfo.partitionData && (jsxs("div", { style: { marginBottom: '6px' }, children: [jsx("div", { style: { fontWeight: 'bold' }, children: "\uD83D\uDD32 Mow Partition Data:" }), jsx("div", { style: { color: '#888' }, children: "No partition data available" })] }))] }));
9841
+ };
8931
9842
  // 错误显示
8932
9843
  if (currentError) {
8933
- return (jsx("div", { className: className, style: style, children: jsxs("div", { style: { color: 'red', padding: '10px' }, children: ["\u9519\u8BEF: ", currentError] }) }));
9844
+ return (jsxs("div", { className: className, style: style, children: [jsxs("div", { style: { color: 'red', padding: '10px' }, children: ["\u9519\u8BEF: ", currentError] }), jsx(DebugInfo, {})] }));
8934
9845
  }
8935
- // 使用goole maps自定义叠加层,不需要实际返回组件内容
8936
- return null;
9846
+ // 使用goole maps自定义叠加层,返回debug信息(如果启用)
9847
+ return debug ? jsx(DebugInfo, {}) : null;
8937
9848
  });
8938
9849
  MowerMapRenderer.displayName = 'MowerMapRenderer';
8939
9850