@fleet-frontend/mower-maps 0.0.9-beta.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.
package/dist/index.esm.js CHANGED
@@ -67,7 +67,8 @@ const DEFAULT_LINE_WIDTHS = {
67
67
  const DEFAULT_OPACITIES = {
68
68
  FULL: 1.0,
69
69
  HIGH: 0.7,
70
- MEDIUM: 0.6};
70
+ MEDIUM: 0.6,
71
+ DOODLE: 0.8};
71
72
  /**
72
73
  * 默认半径设置
73
74
  */
@@ -226,7 +227,6 @@ class SvgMapView {
226
227
  */
227
228
  removeLayer(layer) {
228
229
  const index = this.layers.indexOf(layer);
229
- console.log('removeLayer----->', index);
230
230
  if (index !== -1) {
231
231
  this.layers.splice(index, 1);
232
232
  this.refresh();
@@ -274,7 +274,6 @@ class SvgMapView {
274
274
  width: boundWidth + padding * 2,
275
275
  height: boundHeight + padding * 2,
276
276
  };
277
- console.log('viewbox->', this.viewBox);
278
277
  // 根据宽高比选择合适的preserveAspectRatio设置
279
278
  if (Math.abs(contentAspectRatio - containerAspectRatio) < 0.01) {
280
279
  // 宽高比接近,使用slice填满容器
@@ -284,7 +283,6 @@ class SvgMapView {
284
283
  // 宽高比差异较大,使用meet确保内容完全可见
285
284
  this.svg.setAttribute('preserveAspectRatio', 'xMidYMid meet');
286
285
  }
287
- console.log('fitToView');
288
286
  this.updateViewBox();
289
287
  }
290
288
  /**
@@ -355,7 +353,6 @@ class SvgMapView {
355
353
  * 绘制图层,不传参数则默认绘制所有图层
356
354
  */
357
355
  onDrawLayers(type) {
358
- console.log('onDrawLayers----->', type);
359
356
  if (type) {
360
357
  const layer = this.layers.find((layer) => layer.getType() === type);
361
358
  if (layer) {
@@ -432,7 +429,6 @@ class SvgMapView {
432
429
  refresh() {
433
430
  if (this.destroyed)
434
431
  return;
435
- console.log('refresh----->');
436
432
  this.render();
437
433
  }
438
434
  // ==================== 拖拽功能 ====================
@@ -1049,15 +1045,37 @@ class PathLayer extends BaseLayer {
1049
1045
  d += ' Z ';
1050
1046
  }
1051
1047
  });
1052
- // 3. svgElements(直接拼接path字符串,建议逆时针)
1053
- if (Array.isArray(svgElements)) {
1054
- svgElements.forEach((svgPath) => {
1055
- const svgPathString = svgPath?.metadata?.svg;
1056
- if (svgPathString && typeof svgPathString === 'string' && svgPathString.trim()) {
1057
- d += svgPathString + ' ';
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
+ }
1058
1076
  }
1059
- });
1060
- }
1077
+ }
1078
+ });
1061
1079
  const path = document.createElementNS('http://www.w3.org/2000/svg', 'path');
1062
1080
  path.setAttribute('d', d);
1063
1081
  const clipPath = document.createElementNS('http://www.w3.org/2000/svg', 'clipPath');
@@ -1082,47 +1100,132 @@ class PathLayer extends BaseLayer {
1082
1100
  // 2. 创建一个组,应用 clipPath
1083
1101
  const group = document.createElementNS('http://www.w3.org/2000/svg', 'g');
1084
1102
  group.setAttribute('clip-path', `url(#${clipPathId})`);
1085
- group.setAttribute('opacity', '0.6'); // 统一透明度,防止叠加脏乱
1086
- // 3. 渲染所有路径
1103
+ group.setAttribute('opacity', '0.5'); // 统一透明度,防止叠加脏乱
1104
+ // 3. 优化渲染:按样式分组并合并路径
1105
+ this.renderOptimizedPaths(group);
1106
+ svgGroup.appendChild(group);
1107
+ }
1108
+ /**
1109
+ * 优化渲染:按样式分组并合并路径,减少 DOM 节点数量
1110
+ */
1111
+ renderOptimizedPaths(group) {
1112
+ // 按样式分组存储路径数据
1113
+ const styleGroups = new Map();
1114
+ // 收集所有路径数据并按样式分组
1087
1115
  for (const element of this.elements) {
1088
- const { id, elements } = element;
1116
+ // 类型断言:PathLayer 中的 elements 实际上是 PathElements 结构
1117
+ const pathElement = element;
1118
+ const { id, elements } = pathElement;
1089
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
+ });
1138
+ }
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 中(保持兼容性)
1090
1164
  elements.forEach((element) => {
1091
- this.renderPathToGroup(group, id, element);
1165
+ const { id } = element;
1166
+ if (!this.boundaryPaths[id]) {
1167
+ this.boundaryPaths[id] = [];
1168
+ }
1169
+ this.boundaryPaths[id].push(path);
1092
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
+ }
1093
1221
  }
1094
- svgGroup.appendChild(group);
1222
+ return transformedCommands.join(' ');
1095
1223
  }
1096
1224
  /**
1097
- * 渲染单个路径到指定的组中
1225
+ * 生成样式键,用于路径分组
1098
1226
  */
1099
- renderPathToGroup(group, id, element) {
1100
- const { coordinates, style } = element;
1101
- if (coordinates.length < 2)
1102
- return;
1103
- const path = document.createElementNS('http://www.w3.org/2000/svg', 'path');
1104
- // 构建路径数据
1105
- let pathData = `M ${coordinates[0][0]} ${coordinates[0][1]}`;
1106
- for (let i = 1; i < coordinates.length; i++) {
1107
- pathData += ` L ${coordinates[i][0]} ${coordinates[i][1]}`;
1108
- }
1109
- path.style.mixBlendMode = 'normal';
1110
- // 设置路径属性
1111
- path.setAttribute('d', pathData);
1112
- // 直接给fill的颜色设置透明度会导致path重叠的部分颜色叠加,所以使用fill填充实色,通过fill-opacity设置透明度
1113
- path.setAttribute('fill', 'none');
1114
- // path.setAttribute('fill-opacity', '0.4');
1115
- path.setAttribute('stroke', style.lineColor || '#000000');
1116
- path.setAttribute('mix-blend-mode', 'normal');
1117
- const lineWidth = Math.max(style.lineWidth || 1, 0.5);
1118
- path.setAttribute('stroke-width', lineWidth.toString());
1119
- path.setAttribute('stroke-linecap', 'round');
1120
- path.setAttribute('stroke-linejoin', 'round');
1121
- // 注意:这里不设置 opacity,因为透明度由父组控制
1122
- // path.setAttribute('vector-effect', 'non-scaling-stroke');
1123
- path.classList.add('vector-path');
1124
- this.boundaryPaths[id].push(path);
1125
- group.appendChild(path);
1227
+ generateStyleKey(style) {
1228
+ return `${style.lineColor || '#000000'}-${style.lineWidth || 1}-${style.opacity || 1}`;
1126
1229
  }
1127
1230
  }
1128
1231
 
@@ -1546,7 +1649,7 @@ const DOODLE_STYLES = {
1546
1649
  lineColor: '#ff5722',
1547
1650
  fillColor: '#ff9800', // 粉色半透明填充
1548
1651
  lineWidth: DEFAULT_LINE_WIDTHS.TIME_LIMIT_OBSTACLE,
1549
- opacity: DEFAULT_OPACITIES.HIGH,
1652
+ opacity: DEFAULT_OPACITIES.DOODLE,
1550
1653
  };
1551
1654
  const PATH_EDGE_STYLES = {
1552
1655
  lineWidth: DEFAULT_LINE_WIDTHS.PATH,
@@ -1588,13 +1691,13 @@ const DEFAULT_STYLES = {
1588
1691
  function convertPointsFormat(points) {
1589
1692
  if (!points || points.length === 0)
1590
1693
  return null;
1591
- return points.map(point => {
1694
+ return points.map((point) => {
1592
1695
  if (point.length >= 2) {
1593
1696
  // 对前两个元素应用缩放因子,保留其他元素
1594
1697
  return [
1595
1698
  point[0] * SCALE_FACTOR,
1596
1699
  -point[1] * SCALE_FACTOR, // Y轴翻转,与Python代码一致
1597
- ...point.slice(2) // 保留第三个及以后的元素
1700
+ ...point.slice(2), // 保留第三个及以后的元素
1598
1701
  ];
1599
1702
  }
1600
1703
  return point;
@@ -1609,7 +1712,7 @@ function convertPositionFormat(position) {
1609
1712
  return null;
1610
1713
  return {
1611
1714
  x: position[0] * SCALE_FACTOR,
1612
- y: -position[1] * SCALE_FACTOR // Y轴翻转
1715
+ y: -position[1] * SCALE_FACTOR, // Y轴翻转
1613
1716
  };
1614
1717
  }
1615
1718
  /**
@@ -1618,9 +1721,146 @@ function convertPositionFormat(position) {
1618
1721
  function convertCoordinate(x, y) {
1619
1722
  return {
1620
1723
  x: x * SCALE_FACTOR,
1621
- y: -y * SCALE_FACTOR // Y轴翻转
1724
+ y: -y * SCALE_FACTOR, // Y轴翻转
1622
1725
  };
1623
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
+ }
1624
1864
 
1625
1865
  /**
1626
1866
  * 按Python逻辑创建路径段:根据连续的两点之间的关系确定线段类型
@@ -1883,6 +2123,136 @@ function calculateMapGpsCenter(mapData) {
1883
2123
  };
1884
2124
  }
1885
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
+ }
1886
2256
  /**
1887
2257
  * 通过 mapData 和 pathData 生成所有 boundary 的数据
1888
2258
  * @param mapData 地图数据
@@ -1891,11 +2261,12 @@ function calculateMapGpsCenter(mapData) {
1891
2261
  */
1892
2262
  function generateBoundaryData(mapData, pathData) {
1893
2263
  const boundaryData = [];
2264
+ let chargingPileBoundary = undefined;
1894
2265
  if (!mapData || !mapData.sub_maps) {
1895
2266
  return boundaryData;
1896
2267
  }
1897
2268
  // 第一步:收集所有TUNNEL数据的connection信息
1898
- const connectedBoundaryIds = new Set();
2269
+ const connectIds = [];
1899
2270
  // 遍历mapData中的tunnels字段
1900
2271
  if (mapData.tunnels && Array.isArray(mapData.tunnels)) {
1901
2272
  for (const tunnel of mapData.tunnels) {
@@ -1903,10 +2274,8 @@ function generateBoundaryData(mapData, pathData) {
1903
2274
  if (connection) {
1904
2275
  // connection可能是单个数字或数组
1905
2276
  if (Array.isArray(connection)) {
1906
- connection.forEach(id => connectedBoundaryIds.add(id));
1907
- }
1908
- else if (typeof connection === 'number') {
1909
- connectedBoundaryIds.add(connection);
2277
+ connection.sort();
2278
+ connectIds.push(connection.join('-'));
1910
2279
  }
1911
2280
  }
1912
2281
  }
@@ -1917,9 +2286,9 @@ function generateBoundaryData(mapData, pathData) {
1917
2286
  if (!subMap.elements)
1918
2287
  continue;
1919
2288
  // 每个sub_map的elements是边界坐标,没有sub_map只有一个boundary数据
1920
- const boundaryElement = subMap.elements.find(element => element.type === 'BOUNDARY');
2289
+ const boundaryElement = subMap.elements.find((element) => element.type === 'BOUNDARY');
1921
2290
  // 如果当前subMap存在充电桩且充电桩存在tunnel,说明当前subMap中的boundary是初始boundary,这个boundary不为孤立区域
1922
- const hasTunnelToChargingPile = subMap.elements.some(element => element.type === 'CHARGING_PILE' && element.tunnel);
2291
+ const hasTunnelToChargingPile = subMap.elements.some((element) => element.type === 'CHARGING_PILE' && element.tunnel);
1923
2292
  // 创建基础的 boundary 数据(来自 mapData)
1924
2293
  const boundary = {
1925
2294
  // 从 BOUNDARY 元素复制属性
@@ -1928,8 +2297,6 @@ function generateBoundaryData(mapData, pathData) {
1928
2297
  area: subMap?.area,
1929
2298
  points: convertPointsFormat(boundaryElement?.points) || [],
1930
2299
  type: boundaryElement.type,
1931
- // 判断是否为孤立子区域
1932
- isIsolated: hasTunnelToChargingPile ? false : !connectedBoundaryIds.has(boundaryElement.id)
1933
2300
  };
1934
2301
  // 如果有 pathData,尝试匹配对应的分区数据
1935
2302
  if (pathData) {
@@ -1945,8 +2312,33 @@ function generateBoundaryData(mapData, pathData) {
1945
2312
  boundary.endTime = partitionData.endTime;
1946
2313
  }
1947
2314
  }
2315
+ if (hasTunnelToChargingPile) {
2316
+ chargingPileBoundary = boundary;
2317
+ }
1948
2318
  boundaryData.push(boundary);
1949
2319
  }
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);
2329
+ }
2330
+ }
2331
+ }
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;
2337
+ }
2338
+ else {
2339
+ boundary.isIsolated = true;
2340
+ }
2341
+ }
1950
2342
  return boundaryData;
1951
2343
  }
1952
2344
 
@@ -2163,13 +2555,15 @@ var hNoPosition = "
2163
2555
 
2164
2556
  var hDisabled = "";
2165
2557
 
2558
+ var x3Edger = "";
2559
+
2166
2560
  var x3Mower = "";
2167
2561
 
2168
2562
  var x3NoPosition = "";
2169
2563
 
2170
2564
  var x3Disabled = "";
2171
2565
 
2172
- function getMowerImageByModal(mowerModal) {
2566
+ function getMowerImageByModal(mowerModal, hasEdger) {
2173
2567
  if (mowerModal.includes('i')) {
2174
2568
  return iMower;
2175
2569
  }
@@ -2177,7 +2571,7 @@ function getMowerImageByModal(mowerModal) {
2177
2571
  return hMower;
2178
2572
  }
2179
2573
  else if (mowerModal.includes('x3')) {
2180
- return x3Mower;
2574
+ return hasEdger ? x3Edger : x3Mower;
2181
2575
  }
2182
2576
  return iMower;
2183
2577
  }
@@ -2205,12 +2599,12 @@ function getNoPositionMowerImageByModal(mowerModal) {
2205
2599
  }
2206
2600
  return iNoPosition;
2207
2601
  }
2208
- function getMowerImage(positonConfig, modelType) {
2602
+ function getMowerImage(positonConfig, modelType, hasEdger) {
2209
2603
  if (!positonConfig)
2210
2604
  return '';
2211
2605
  const model = modelType?.toLowerCase() || 'i';
2212
2606
  const state = positonConfig.vehicleState;
2213
- const mowerImage = getMowerImageByModal(model);
2607
+ const mowerImage = getMowerImageByModal(model, hasEdger);
2214
2608
  const disabledImage = getDisabledMowerImageByModal(model);
2215
2609
  const noPositionImage = getNoPositionMowerImageByModal(model);
2216
2610
  const positonOutOfRange = isOutOfRange(positonConfig);
@@ -4692,8 +5086,8 @@ var PathSegmentType;
4692
5086
  */
4693
5087
  var UnitsType;
4694
5088
  (function (UnitsType) {
4695
- UnitsType["Metric"] = "metric";
4696
- UnitsType["Imperial"] = "imperial";
5089
+ UnitsType["Metric"] = "Metric";
5090
+ UnitsType["Imperial"] = "Imperial";
4697
5091
  })(UnitsType || (UnitsType = {}));
4698
5092
  /**
4699
5093
  * 面积单位类型枚举
@@ -4980,6 +5374,12 @@ class BoundaryBorderLayer extends BaseLayer {
4980
5374
  this.mowingBoundarys = mowingBoundarys;
4981
5375
  }
4982
5376
  }
5377
+ /**
5378
+ * 获取当前割草任务的边界
5379
+ */
5380
+ getMowingBoundarys() {
5381
+ return this.mowingBoundarys;
5382
+ }
4983
5383
  /**
4984
5384
  * SVG渲染方法
4985
5385
  */
@@ -4988,13 +5388,31 @@ class BoundaryBorderLayer extends BaseLayer {
4988
5388
  return;
4989
5389
  }
4990
5390
  this.scale = scale;
4991
- console.log('draw boundary border->', this.elements, this.mowingBoundarys);
4992
- // 只渲染边界边框类型的元素
5391
+ // 将元素分为两组:非割草边界和割草边界
5392
+ const nonMowingElements = [];
5393
+ const mowingElements = [];
5394
+ // 只处理边界边框类型的元素
4993
5395
  for (const element of this.elements) {
4994
5396
  if (element.type === 'boundary_border') {
4995
- 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
+ }
4996
5406
  }
4997
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
+ }
4998
5416
  }
4999
5417
  /**
5000
5418
  * 渲染边界边框
@@ -6063,7 +6481,7 @@ class PathDataProcessor {
6063
6481
  * 专门处理边界标签的创建、定位和管理
6064
6482
  */
6065
6483
  class BoundaryLabelsManager {
6066
- constructor(svgView, boundaryData) {
6484
+ constructor(svgView, boundaryData, { unitType, language }) {
6067
6485
  this.container = null;
6068
6486
  this.overlayDiv = null;
6069
6487
  this.globalClickHandler = null;
@@ -6074,6 +6492,8 @@ class BoundaryLabelsManager {
6074
6492
  this.svgView = svgView;
6075
6493
  this.boundaryData = boundaryData;
6076
6494
  this.initializeContainer();
6495
+ this.unitType = unitType;
6496
+ this.language = language;
6077
6497
  }
6078
6498
  /**
6079
6499
  * 初始化容器
@@ -6139,7 +6559,7 @@ class BoundaryLabelsManager {
6139
6559
  labelDiv.setAttribute('data-boundary-id', boundary.id.toString());
6140
6560
  // 样式设置
6141
6561
  labelDiv.style.position = 'absolute';
6142
- labelDiv.style.backgroundColor = 'rgba(30, 30, 31, 0.3)';
6562
+ labelDiv.style.backgroundColor = 'rgba(30, 30, 31, 0.6)';
6143
6563
  labelDiv.style.color = 'rgba(255, 255, 255, 1)';
6144
6564
  labelDiv.style.padding = '6px';
6145
6565
  labelDiv.style.borderRadius = '12px';
@@ -6156,7 +6576,7 @@ class BoundaryLabelsManager {
6156
6576
  labelDiv.style.zIndex = BoundaryLabelsManager.Z_INDEX.DEFAULT.toString();
6157
6577
  // 计算进度
6158
6578
  const progress = boundary.finishedArea && boundary.area
6159
- ? `${Math.round((boundary.finishedArea / boundary.area) * 100)}%`
6579
+ ? `${Math.floor((boundary.finishedArea / boundary.area) * 100)}%`
6160
6580
  : '0%';
6161
6581
  // 基础内容(始终显示)
6162
6582
  const baseContent = document.createElement('div');
@@ -6173,12 +6593,15 @@ class BoundaryLabelsManager {
6173
6593
  this.currentExpandedBoundaryId === boundary.id ? 'block' : 'none';
6174
6594
  extendedContent.style.borderTop = '1px solid rgba(255,255,255,0.2)';
6175
6595
  extendedContent.style.paddingTop = '6px';
6596
+ const boundaryLayer = this.svgView.getLayer(LAYER_DEFAULT_TYPE.BOUNDARY_BORDER);
6597
+ const mowingBoundarys = boundaryLayer.getMowingBoundarys();
6176
6598
  // 面积信息
6177
- const totalArea = convertAreaByUnits(boundary.area || 0, 'metric');
6178
- 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);
6179
6601
  const coverageText = `Coverage: ${finishedArea.value}/${totalArea.value}`;
6602
+ const isMowing = mowingBoundarys.includes(boundary.id);
6180
6603
  // 日期信息
6181
- const dateText = formatBoundaryDateText(boundary.endTime || 0);
6604
+ const dateText = formatBoundaryDateText(isMowing ? Date.now() / 1000 : boundary.endTime || 0);
6182
6605
  const covertHtml = `<div style="margin-bottom: 3px; font-weight: bold;">${coverageText}</div>`;
6183
6606
  const dateHtml = `<div>${dateText}</div>`;
6184
6607
  extendedContent.innerHTML = boundary.finishedArea > 0 ? `${covertHtml}${dateHtml}` : covertHtml;
@@ -6296,7 +6719,6 @@ class BoundaryLabelsManager {
6296
6719
  // 计算边界中心点的地图坐标
6297
6720
  const mapCenter = this.calculatePolygonCentroid(boundary.points);
6298
6721
  if (!mapCenter) {
6299
- console.warn(`BoundaryLabelsManager: 无法计算边界 ${boundary.name} (ID: ${boundary.id}) 的中心点`);
6300
6722
  return;
6301
6723
  }
6302
6724
  // 直接使用预计算的数据进行坐标转换
@@ -6381,7 +6803,6 @@ class BoundaryLabelsManager {
6381
6803
  area = area / 2;
6382
6804
  // 如果面积为0,回退到简单的平均值计算
6383
6805
  if (Math.abs(area) < 1e-10) {
6384
- console.warn('BoundaryLabelsManager: 多边形面积为0,使用平均值计算重心');
6385
6806
  return this.calculateAverageCenter(validPoints);
6386
6807
  }
6387
6808
  centroidX = centroidX / (6 * area);
@@ -7292,6 +7713,10 @@ class MowerPositionManager {
7292
7713
  getElement() {
7293
7714
  return this.container;
7294
7715
  }
7716
+ //
7717
+ setEdger(edger) {
7718
+ this.hasEdger = edger;
7719
+ }
7295
7720
  /**
7296
7721
  * 根据最后一次有效的位置更新数据
7297
7722
  */
@@ -7315,7 +7740,6 @@ class MowerPositionManager {
7315
7740
  postureY = chargingPilesPositionConfig.postureY || 0;
7316
7741
  postureTheta = chargingPilesPositionConfig.postureTheta || 0;
7317
7742
  }
7318
- console.log('updatePositionByLastPosition->', postureX, postureY, postureTheta, chargingPilesPositionConfig);
7319
7743
  // 检查是否需要更新图片
7320
7744
  this.updateMowerImage(chargingPilesPositionConfig);
7321
7745
  // 立即更新位置
@@ -7332,7 +7756,6 @@ class MowerPositionManager {
7332
7756
  const postureX = positionConfig?.postureX || this.lastPosition?.x || 0;
7333
7757
  const postureY = positionConfig?.postureY || this.lastPosition?.y || 0;
7334
7758
  const postureTheta = positionConfig?.postureTheta || this.lastPosition?.rotation || 0;
7335
- console.log('updatePosition manager', JSON.stringify(this.currentPosition), this.currentPosition, !this.currentPosition, positionConfig, this.lastPosition, animationTime);
7336
7759
  // 停止当前动画(如果有)
7337
7760
  this.stopAnimation();
7338
7761
  // 第一个点
@@ -7342,7 +7765,6 @@ class MowerPositionManager {
7342
7765
  y: postureY,
7343
7766
  rotation: postureTheta,
7344
7767
  };
7345
- console.log('updatePosition first->', this.currentPosition);
7346
7768
  this.setElementPosition(this.currentPosition.x, this.currentPosition.y, this.currentPosition.rotation);
7347
7769
  return;
7348
7770
  }
@@ -7364,7 +7786,7 @@ class MowerPositionManager {
7364
7786
  const imgElement = this.mowerElement.querySelector('img');
7365
7787
  if (!imgElement)
7366
7788
  return;
7367
- const imageSrc = getMowerImage(positonConfig, this.modelType);
7789
+ const imageSrc = getMowerImage(positonConfig, this.modelType, this.hasEdger);
7368
7790
  if (imageSrc) {
7369
7791
  imgElement.src = imageSrc;
7370
7792
  imgElement.style.display = 'block';
@@ -7451,12 +7873,10 @@ class MowerPositionManager {
7451
7873
  y: this.onlyUpdateTheta ? 0 : this.targetPosition.y - this.startPosition.y,
7452
7874
  rotation: radNormalize(targetTheta - startTheta),
7453
7875
  };
7454
- console.log('startAnimationToPosition-->', this.deltaPosition, this.onlyUpdateTheta, this.targetPosition, this.startPosition);
7455
7876
  // 开始动画循环
7456
7877
  this.animateStep();
7457
7878
  }
7458
7879
  forceUpdatePosition() {
7459
- console.log('forceUpdatePosition-->', this.currentPosition, this.targetPosition, this.startPosition);
7460
7880
  this.animateStep();
7461
7881
  }
7462
7882
  /**
@@ -7597,15 +8017,40 @@ function throttleAdvanced(func, delay, options = { leading: true, trailing: true
7597
8017
  }
7598
8018
  };
7599
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
+ }
7600
8044
 
7601
8045
  // Google Maps 叠加层类 - 带编辑功能
7602
8046
  class MowerMapOverlay {
7603
- constructor(bounds, mapData, partitionBoundary, mowerPositionConfig, modelType, pathData, isEditMode = false, mapConfig = {}, antennaConfig = {}, mowPartitionData = null, defaultTransform, 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) {
7604
8048
  this.div = null;
7605
8049
  this.svgMapView = null;
7606
8050
  this.offscreenContainer = null;
7607
8051
  this.overlayView = null;
7608
8052
  this.defaultTransform = { x: 0, y: 0, rotation: 0 };
8053
+ this.hasEdger = false;
7609
8054
  // boundary数据
7610
8055
  this.boundaryData = [];
7611
8056
  // 边界标签管理器
@@ -7648,6 +8093,8 @@ class MowerMapOverlay {
7648
8093
  this.partitionBoundary = partitionBoundary;
7649
8094
  this.pathData = pathData;
7650
8095
  this.isEditMode = isEditMode;
8096
+ this.unitType = unitType;
8097
+ this.language = language;
7651
8098
  this.mapConfig = mapConfig;
7652
8099
  this.antennaConfig = antennaConfig;
7653
8100
  this.onMapLoad = onMapLoad;
@@ -7694,7 +8141,6 @@ class MowerMapOverlay {
7694
8141
  this.isUserAnimation = animationTime > 0;
7695
8142
  // 更新割草机位置配置
7696
8143
  this.mowerPositionConfig = positionConfig;
7697
- console.log('updatePosition overlay', positionConfig);
7698
8144
  // 更新割草机位置管理器
7699
8145
  if (this.mowerPositionManager) {
7700
8146
  this.mowerPositionManager.updatePosition(positionConfig, animationTime);
@@ -7712,6 +8158,12 @@ class MowerMapOverlay {
7712
8158
  this.overlayView.setMap(map);
7713
8159
  }
7714
8160
  }
8161
+ setEdger(edger) {
8162
+ this.hasEdger = edger;
8163
+ if (this.mowerPositionManager) {
8164
+ this.mowerPositionManager.setEdger(edger);
8165
+ }
8166
+ }
7715
8167
  getMap() {
7716
8168
  return this.overlayView ? this.overlayView.getMap() : null;
7717
8169
  }
@@ -7739,7 +8191,6 @@ class MowerMapOverlay {
7739
8191
  this.svgMapView?.renderLayer(LAYER_DEFAULT_TYPE.BOUNDARY_BORDER);
7740
8192
  }
7741
8193
  onAdd() {
7742
- console.log('onAdd');
7743
8194
  // 创建包含SVG的div
7744
8195
  this.div = document.createElement('div');
7745
8196
  this.div.style.borderStyle = 'none';
@@ -7807,7 +8258,10 @@ class MowerMapOverlay {
7807
8258
  if (!this.div || !this.svgMapView)
7808
8259
  return;
7809
8260
  // 创建边界标签管理器
7810
- 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
+ });
7811
8265
  // 设置叠加层div引用
7812
8266
  this.boundaryLabelsManager.setOverlayDiv(this.div);
7813
8267
  // 添加所有边界标签
@@ -7851,11 +8305,10 @@ class MowerMapOverlay {
7851
8305
  if (!this.div || !this.svgMapView)
7852
8306
  return;
7853
8307
  // 创建割草机位置管理器,传入动画完成回调
7854
- this.mowerPositionManager = new MowerPositionManager(this.svgMapView, this.mowerPositionConfig, this.modelType, this.div, () => {
7855
- console.log('动画完成');
7856
- }, this.updatePathDataByMowingPositionThrottled.bind(this));
8308
+ this.mowerPositionManager = new MowerPositionManager(this.svgMapView, this.mowerPositionConfig, this.modelType, this.div, () => { }, this.updatePathDataByMowingPositionThrottled.bind(this));
7857
8309
  // 设置叠加层div引用
7858
8310
  this.mowerPositionManager.setOverlayDiv(this.div);
8311
+ this.mowerPositionManager.setEdger(this.hasEdger);
7859
8312
  // 获取容器并添加到主div
7860
8313
  const container = this.mowerPositionManager.getElement();
7861
8314
  if (container) {
@@ -7935,7 +8388,7 @@ class MowerMapOverlay {
7935
8388
  this.rotateHandle.style.pointerEvents = 'auto';
7936
8389
  this.rotateHandle.innerHTML = DEFAULT_ROTATE_ICON;
7937
8390
  this.editContainer.appendChild(this.rotateHandle);
7938
- // 创建拖拽手柄(左上角)
8391
+ // 创建拖拽手柄(左下角)- 仅在移动设备上显示
7939
8392
  this.dragHandle = document.createElement('div');
7940
8393
  this.dragHandle.style.position = 'absolute';
7941
8394
  this.dragHandle.style.bottom = '-20px';
@@ -7946,6 +8399,10 @@ class MowerMapOverlay {
7946
8399
  this.dragHandle.style.zIndex = EDIT_STYLES.Z_INDEX.HANDLE;
7947
8400
  this.dragHandle.style.pointerEvents = 'auto';
7948
8401
  this.dragHandle.innerHTML = DEFAULT_DRAG_ICON;
8402
+ // 在PC设备上隐藏拖拽手柄
8403
+ if (!isMobileDevice()) {
8404
+ this.dragHandle.style.display = 'none';
8405
+ }
7949
8406
  this.editContainer.appendChild(this.dragHandle);
7950
8407
  // 将编辑容器添加到主div
7951
8408
  this.div.appendChild(this.editContainer);
@@ -7987,7 +8444,6 @@ class MowerMapOverlay {
7987
8444
  this.boundaryLabelsManager.collapseAllLabels();
7988
8445
  }
7989
8446
  this.dragCallbacks?.onDragStart?.(this.getCurrentDragState());
7990
- console.log('开始旋转操作');
7991
8447
  });
7992
8448
  // 旋转手柄的触摸事件
7993
8449
  this.rotateHandle.addEventListener('touchstart', (e) => {
@@ -8003,39 +8459,41 @@ class MowerMapOverlay {
8003
8459
  this.boundaryLabelsManager.collapseAllLabels();
8004
8460
  }
8005
8461
  this.dragCallbacks?.onDragStart?.(this.getCurrentDragState());
8006
- console.log('开始旋转操作(触摸)');
8007
- }, { passive: false });
8008
- // 拖拽手柄的鼠标事件
8009
- this.dragHandle.addEventListener('mousedown', (e) => {
8010
- e.preventDefault();
8011
- e.stopPropagation();
8012
- e.stopImmediatePropagation();
8013
- this.isDragging = true;
8014
- this.startPos = { x: e.clientX, y: e.clientY };
8015
- this.dragHandle.style.cursor = 'grabbing';
8016
- // 开始编辑时关闭所有展开的边界标签
8017
- if (this.boundaryLabelsManager) {
8018
- this.boundaryLabelsManager.collapseAllLabels();
8019
- }
8020
- this.dragCallbacks?.onDragStart?.(this.getCurrentDragState());
8021
- console.log('开始拖动操作(通过手柄)');
8022
- });
8023
- // 拖拽手柄的触摸事件
8024
- this.dragHandle.addEventListener('touchstart', (e) => {
8025
- e.preventDefault();
8026
- e.stopPropagation();
8027
- e.stopImmediatePropagation();
8028
- this.isDragging = true;
8029
- const touch = e.touches[0];
8030
- this.startPos = { x: touch.clientX, y: touch.clientY };
8031
- this.dragHandle.style.cursor = 'grabbing';
8032
- this.dragCallbacks?.onDragStart?.(this.getCurrentDragState());
8033
- console.log('开始拖动操作(通过手柄,触摸)');
8034
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
+ }
8035
8490
  // 编辑容器的鼠标事件(整个区域拖拽)
8036
8491
  this.editContainer.addEventListener('mousedown', (e) => {
8037
- console.log('开始拖动操作(整个叠加层)');
8038
- 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) {
8039
8497
  return;
8040
8498
  }
8041
8499
  e.preventDefault();
@@ -8052,8 +8510,11 @@ class MowerMapOverlay {
8052
8510
  });
8053
8511
  // 编辑容器的触摸事件(整个区域拖拽)
8054
8512
  this.editContainer.addEventListener('touchstart', (e) => {
8055
- console.log('开始拖动操作(整个叠加层,触摸)');
8056
- 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) {
8057
8518
  return;
8058
8519
  }
8059
8520
  e.preventDefault();
@@ -8113,7 +8574,6 @@ class MowerMapOverlay {
8113
8574
  e.preventDefault();
8114
8575
  e.stopPropagation();
8115
8576
  e.stopImmediatePropagation();
8116
- console.log('结束编辑操作');
8117
8577
  }
8118
8578
  // 如果是拖拽结束,将像素偏移量转换为地理坐标偏移量
8119
8579
  if (this.isDragging) {
@@ -8135,7 +8595,6 @@ class MowerMapOverlay {
8135
8595
  e.preventDefault();
8136
8596
  e.stopPropagation();
8137
8597
  e.stopImmediatePropagation();
8138
- console.log('结束编辑操作(触摸)');
8139
8598
  }
8140
8599
  // 如果是拖拽结束,将像素偏移量转换为地理坐标偏移量
8141
8600
  if (this.isDragging) {
@@ -8247,7 +8706,6 @@ class MowerMapOverlay {
8247
8706
  this.div.style.transform = transform;
8248
8707
  // 更新鼠标起始位置为当前位置,为下次计算做准备
8249
8708
  this.startPos = { x: mouseCurrentX, y: mouseCurrentY };
8250
- console.log('旋转角度:', this.currentRotation, '角度增量:', angleDifferenceDegrees);
8251
8709
  }
8252
8710
  // 将像素偏移量转换为地理坐标偏移量
8253
8711
  convertPixelOffsetToLatLng() {
@@ -8277,14 +8735,6 @@ class MowerMapOverlay {
8277
8735
  // 累积更新地理坐标偏移量(不是直接赋值!)
8278
8736
  this.latLngOffset.lat += latOffset;
8279
8737
  this.latLngOffset.lng += lngOffset;
8280
- console.log('精确转换偏移量:', {
8281
- pixelOffset: this.tempPixelOffset,
8282
- centerLatLng: { lat: centerLatLng.lat(), lng: centerLatLng.lng() },
8283
- offsetLatLng: { lat: offsetLatLng.lat(), lng: offsetLatLng.lng() },
8284
- latOffset,
8285
- lngOffset,
8286
- newLatLngOffset: this.latLngOffset,
8287
- });
8288
8738
  // 重置临时像素偏移量
8289
8739
  this.tempPixelOffset = { x: 0, y: 0 };
8290
8740
  this.draw();
@@ -8322,8 +8772,6 @@ class MowerMapOverlay {
8322
8772
  editData: editData,
8323
8773
  timestamp: new Date().toISOString(),
8324
8774
  };
8325
- // 在这里可以添加保存逻辑,比如发送到服务器
8326
- console.log('保存编辑数据:', saveData);
8327
8775
  // 显示保存成功提示
8328
8776
  this.showSaveSuccess();
8329
8777
  return saveData;
@@ -8416,6 +8864,9 @@ class MowerMapOverlay {
8416
8864
  y: transform.y,
8417
8865
  rotation: transform.rotation,
8418
8866
  };
8867
+ // defaultTransform的x对应经度偏移量,y对应纬度偏移量
8868
+ this.latLngOffset.lng = this.defaultTransform.x;
8869
+ this.latLngOffset.lat = this.defaultTransform.y;
8419
8870
  this.setManagerRotation(this.currentRotation);
8420
8871
  this.draw();
8421
8872
  }
@@ -8455,7 +8906,6 @@ class MowerMapOverlay {
8455
8906
  if (this.pathData && this.svgMapView) {
8456
8907
  this.loadPathData(this.pathData, this.mowPartitionData);
8457
8908
  }
8458
- console.log('initializeSvgMapView');
8459
8909
  // 刷新绘制图层
8460
8910
  this.svgMapView.refresh();
8461
8911
  // 获取生成的SVG并添加到叠加层div中
@@ -8611,7 +9061,6 @@ class MowerMapOverlay {
8611
9061
  this.boundaryLabelsManager?.updateBoundaryData(boundaryData);
8612
9062
  }
8613
9063
  draw() {
8614
- console.log('ondraw');
8615
9064
  // 防御性检查:如果this.div为null,说明onAdd还没被调用,直接返回
8616
9065
  if (!this.div) {
8617
9066
  return;
@@ -8879,7 +9328,7 @@ const getValidGpsBounds = (mapData, rotation = 0) => {
8879
9328
  // 默认配置
8880
9329
  const defaultMapConfig = DEFAULT_STYLES;
8881
9330
  // 地图渲染器组件
8882
- const MowerMapRenderer = forwardRef(({ mapConfig, modelType, mapRef, mapJson, pathJson, realTimeData, antennaConfig, onMapLoad, onPathLoad, onError, className, style, googleMapInstance, isEditMode = false, dragCallbacks, defaultTransform, debug = false, }, ref) => {
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) => {
8883
9332
  const [elementCount, setElementCount] = useState(0);
8884
9333
  const [pathCount, setPathCount] = useState(0);
8885
9334
  const [currentError, setCurrentError] = useState(null);
@@ -8887,7 +9336,7 @@ const MowerMapRenderer = forwardRef(({ mapConfig, modelType, mapRef, mapJson, pa
8887
9336
  // const mapRef = useMap();
8888
9337
  const [isGoogleMapsReady, setIsGoogleMapsReady] = useState(false);
8889
9338
  const [hasInitializedBounds, setHasInitializedBounds] = useState(false);
8890
- const { clearSubBoundaryBorder, clearObstacles } = useSubBoundaryBorderStore();
9339
+ const { clearSubBoundaryBorder, clearObstacles, clearSvgElements } = useSubBoundaryBorderStore();
8891
9340
  const currentProcessMowingStatusRef = useRef(false);
8892
9341
  const { updateProcessStateIsMowing, processStateIsMowing } = useProcessMowingState();
8893
9342
  const [mowPartitionData, setMowPartitionData] = useState(null);
@@ -8921,7 +9370,7 @@ const MowerMapRenderer = forwardRef(({ mapConfig, modelType, mapRef, mapJson, pa
8921
9370
  postureX: 0,
8922
9371
  postureY: 0,
8923
9372
  postureTheta: 0,
8924
- vehicleState: RobotStatus.PARKED,
9373
+ vehicleState: RobotStatus.DISCONNECTED,
8925
9374
  };
8926
9375
  let currentPositionData;
8927
9376
  if (realTimeData.length === 1 && realTimeData[0].type === RealTimeDataType.LOCATION) {
@@ -8947,10 +9396,9 @@ const MowerMapRenderer = forwardRef(({ mapConfig, modelType, mapRef, mapJson, pa
8947
9396
  lastPostureY: currentPositionData?.lastPostureY
8948
9397
  ? Number(currentPositionData.lastPostureY)
8949
9398
  : 0,
8950
- vehicleState: currentPositionData?.vehicleState || RobotStatus.CHARGING,
9399
+ vehicleState: currentPositionData?.vehicleState || RobotStatus.DISCONNECTED,
8951
9400
  };
8952
9401
  }, [realTimeData, modelType]);
8953
- console.log('mowerPositionData', mowerPositionData);
8954
9402
  // 处理错误
8955
9403
  const handleError = (error) => {
8956
9404
  setCurrentError(error);
@@ -8975,7 +9423,6 @@ const MowerMapRenderer = forwardRef(({ mapConfig, modelType, mapRef, mapJson, pa
8975
9423
  const googleBounds = new window.google.maps.LatLngBounds(new window.google.maps.LatLng(swLat, swLng), // 西南角
8976
9424
  new window.google.maps.LatLng(neLat, neLng) // 东北角
8977
9425
  );
8978
- console.log('fitBounds----->', googleBounds);
8979
9426
  mapRef.fitBounds(googleBounds);
8980
9427
  }, [mapJson, mapRef, defaultTransform]);
8981
9428
  // 初始化Google Maps叠加层
@@ -9016,9 +9463,8 @@ const MowerMapRenderer = forwardRef(({ mapConfig, modelType, mapRef, mapJson, pa
9016
9463
  overlayRef.current.setMap(null);
9017
9464
  overlayRef.current = null;
9018
9465
  }
9019
- console.log('initializeGoogleMapsOverlay', mowPartitionData);
9020
9466
  // 创建叠加层
9021
- const overlay = new MowerMapOverlay(googleBounds, mapJson, partitionBoundary, mowerPositionData, modelType, pathJson || {}, isEditMode, mergedMapConfig, mergedAntennaConfig, null, defaultTransform, (count) => {
9467
+ const overlay = new MowerMapOverlay(googleBounds, mapJson, partitionBoundary, mowerPositionData, modelType, pathJson || {}, isEditMode, unitType, language, mergedMapConfig, mergedAntennaConfig, null, defaultTransform, (count) => {
9022
9468
  setElementCount(count);
9023
9469
  onMapLoad?.(count);
9024
9470
  }, (count) => {
@@ -9028,6 +9474,7 @@ const MowerMapRenderer = forwardRef(({ mapConfig, modelType, mapRef, mapJson, pa
9028
9474
  // 设置地图
9029
9475
  overlay.setMap(mapInstance);
9030
9476
  overlayRef.current = overlay;
9477
+ overlay.setEdger(edger);
9031
9478
  // 只在首次初始化时自适应视图
9032
9479
  if (!hasInitializedBounds) {
9033
9480
  mapInstance.fitBounds(googleBounds);
@@ -9051,7 +9498,7 @@ const MowerMapRenderer = forwardRef(({ mapConfig, modelType, mapRef, mapJson, pa
9051
9498
  postureY: chargingPiles?.originalData.position[1],
9052
9499
  postureTheta: chargingPiles?.originalData.direction - Math.PI || 0,
9053
9500
  }, 0);
9054
- }, [mapJson]);
9501
+ }, [mapJson, mowerPositionData]);
9055
9502
  // 初始化效果
9056
9503
  useEffect(() => {
9057
9504
  initializeGoogleMapsOverlay();
@@ -9059,6 +9506,7 @@ const MowerMapRenderer = forwardRef(({ mapConfig, modelType, mapRef, mapJson, pa
9059
9506
  return () => {
9060
9507
  clearSubBoundaryBorder();
9061
9508
  clearObstacles();
9509
+ clearSvgElements();
9062
9510
  updateProcessStateIsMowing(false);
9063
9511
  currentProcessMowingStatusRef.current = false;
9064
9512
  if (overlayRef.current) {
@@ -9090,7 +9538,6 @@ const MowerMapRenderer = forwardRef(({ mapConfig, modelType, mapRef, mapJson, pa
9090
9538
  const isOffLine = mowerPositionData.vehicleState === RobotStatus.DISCONNECTED;
9091
9539
  const isInChargingPile = inChargingPiles.includes(mowerPositionData.vehicleState);
9092
9540
  // 如果在充电桩上,则直接更新位置到充电桩的位置
9093
- console.log('usefeect mowerPositionData----->', mowerPositionData);
9094
9541
  if (isInChargingPile) {
9095
9542
  overlayRef.current.updatePosition({
9096
9543
  ...mowerPositionData,
@@ -9123,7 +9570,6 @@ const MowerMapRenderer = forwardRef(({ mapConfig, modelType, mapRef, mapJson, pa
9123
9570
  }
9124
9571
  }
9125
9572
  else {
9126
- console.log('hook updatePosition----->', mowerPositionData);
9127
9573
  overlayRef.current.updatePosition(mowerPositionData, isStandby ? 0 : 2000);
9128
9574
  }
9129
9575
  }
@@ -9242,7 +9688,6 @@ const MowerMapRenderer = forwardRef(({ mapConfig, modelType, mapRef, mapJson, pa
9242
9688
  if (!realTimeData || realTimeData.length === 0 || !Array.isArray(realTimeData)) {
9243
9689
  return;
9244
9690
  }
9245
- console.log('usefeect realTimeData----->', realTimeData, mapJson, pathJson, overlayRef.current);
9246
9691
  let curMowPartitionData = mowPartitionData;
9247
9692
  // realtime中包含当前割草任务的数据,根据数据进行path路径和边界的高亮操作,
9248
9693
  const mowingPartition = realTimeData.find((item) => item.type === RealTimeDataType.PARTITION);
@@ -9250,9 +9695,8 @@ const MowerMapRenderer = forwardRef(({ mapConfig, modelType, mapRef, mapJson, pa
9250
9695
  setMowPartitionData(mowingPartition);
9251
9696
  curMowPartitionData = mowingPartition;
9252
9697
  }
9253
- const positionData = realTimeData?.find(item => item?.type === RealTimeDataType.LOCATION);
9254
- const statusData = realTimeData?.find(item => item?.type === RealTimeDataType.STATUS);
9255
- console.log('current->1', positionData, statusData);
9698
+ const positionData = realTimeData?.find((item) => item?.type === RealTimeDataType.LOCATION);
9699
+ const statusData = realTimeData?.find((item) => item?.type === RealTimeDataType.STATUS);
9256
9700
  if (statusData || positionData) {
9257
9701
  const currentStatus = statusData?.vehicleState || positionData?.vehicleState;
9258
9702
  // 车辆回桩不会回传最后的park的位置,所以根据实时数据的状态数据判断车辆回到桩上
@@ -9262,32 +9706,28 @@ const MowerMapRenderer = forwardRef(({ mapConfig, modelType, mapRef, mapJson, pa
9262
9706
  else if (currentStatus === RobotStatus.WORKING) {
9263
9707
  // 兜底收不到割草地块的实时数据,使用状态来兜底
9264
9708
  overlayRef.current.resetBorderLayerHighlight();
9265
- setMowPartitionData(null);
9266
- curMowPartitionData = null;
9709
+ setMowPartitionData({});
9710
+ curMowPartitionData = {};
9267
9711
  }
9268
- else if (currentStatus === RobotStatus.MOWING && (curMowPartitionData && !curMowPartitionData?.partitionIds)) {
9712
+ else if (currentStatus === RobotStatus.MOWING &&
9713
+ curMowPartitionData &&
9714
+ !curMowPartitionData?.partitionIds) {
9269
9715
  // 如果当前是割草状态,但是地块数据初始化过且不存在则认为是全局割草,则把所有地块都高亮
9270
- const allPartitionIds = mapJson?.sub_maps?.map(item => item?.id);
9271
- console.log('allPartitionIds->', allPartitionIds, mapJson);
9716
+ const allPartitionIds = mapJson?.sub_maps?.map((item) => item?.id);
9272
9717
  setMowPartitionData({
9273
- partitionIds: allPartitionIds
9718
+ partitionIds: allPartitionIds,
9274
9719
  });
9275
9720
  curMowPartitionData = {
9276
- partitionIds: allPartitionIds
9721
+ partitionIds: allPartitionIds,
9277
9722
  };
9278
9723
  }
9279
9724
  }
9280
- if (!mapJson ||
9281
- !pathJson ||
9282
- !overlayRef.current)
9725
+ if (!mapJson || !pathJson || !overlayRef.current)
9283
9726
  return;
9284
9727
  // 根据后端推送的实时数据,进行不同处理
9285
- // TODO:需要根据返回的数据,处理车辆的移动位置
9286
- console.log('realTimeData----->', realTimeData, curMowPartitionData);
9287
9728
  if (curMowPartitionData) {
9288
9729
  const isMowing = curMowPartitionData?.partitionIds && curMowPartitionData.partitionIds.length > 0;
9289
9730
  overlayRef.current.updateMowPartitionData(curMowPartitionData);
9290
- console.log('isMowing', isMowing, curMowPartitionData);
9291
9731
  if (!isMowing) {
9292
9732
  overlayRef.current.resetBorderLayerHighlight();
9293
9733
  }
@@ -9325,7 +9765,6 @@ const MowerMapRenderer = forwardRef(({ mapConfig, modelType, mapRef, mapJson, pa
9325
9765
  }
9326
9766
  }, [realTimeData, mapJson, pathJson]);
9327
9767
  useEffect(() => {
9328
- console.log('defaultTransform----->', defaultTransform, overlayRef.current, mapJson);
9329
9768
  if (!overlayRef.current || !defaultTransform)
9330
9769
  return;
9331
9770
  overlayRef.current?.setTransform(defaultTransform);
@@ -9340,6 +9779,11 @@ const MowerMapRenderer = forwardRef(({ mapConfig, modelType, mapRef, mapJson, pa
9340
9779
  );
9341
9780
  mapRef.fitBounds(googleBounds);
9342
9781
  }, [defaultTransform]);
9782
+ useEffect(() => {
9783
+ if (!overlayRef || !overlayRef.current)
9784
+ return;
9785
+ overlayRef.current.setEdger(edger);
9786
+ }, [edger]);
9343
9787
  // 提供ref方法
9344
9788
  useImperativeHandle(ref, () => ({
9345
9789
  fitToView: () => {