@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.js CHANGED
@@ -69,7 +69,8 @@ const DEFAULT_LINE_WIDTHS = {
69
69
  const DEFAULT_OPACITIES = {
70
70
  FULL: 1.0,
71
71
  HIGH: 0.7,
72
- MEDIUM: 0.6};
72
+ MEDIUM: 0.6,
73
+ DOODLE: 0.8};
73
74
  /**
74
75
  * 默认半径设置
75
76
  */
@@ -228,7 +229,6 @@ class SvgMapView {
228
229
  */
229
230
  removeLayer(layer) {
230
231
  const index = this.layers.indexOf(layer);
231
- console.log('removeLayer----->', index);
232
232
  if (index !== -1) {
233
233
  this.layers.splice(index, 1);
234
234
  this.refresh();
@@ -276,7 +276,6 @@ class SvgMapView {
276
276
  width: boundWidth + padding * 2,
277
277
  height: boundHeight + padding * 2,
278
278
  };
279
- console.log('viewbox->', this.viewBox);
280
279
  // 根据宽高比选择合适的preserveAspectRatio设置
281
280
  if (Math.abs(contentAspectRatio - containerAspectRatio) < 0.01) {
282
281
  // 宽高比接近,使用slice填满容器
@@ -286,7 +285,6 @@ class SvgMapView {
286
285
  // 宽高比差异较大,使用meet确保内容完全可见
287
286
  this.svg.setAttribute('preserveAspectRatio', 'xMidYMid meet');
288
287
  }
289
- console.log('fitToView');
290
288
  this.updateViewBox();
291
289
  }
292
290
  /**
@@ -357,7 +355,6 @@ class SvgMapView {
357
355
  * 绘制图层,不传参数则默认绘制所有图层
358
356
  */
359
357
  onDrawLayers(type) {
360
- console.log('onDrawLayers----->', type);
361
358
  if (type) {
362
359
  const layer = this.layers.find((layer) => layer.getType() === type);
363
360
  if (layer) {
@@ -434,7 +431,6 @@ class SvgMapView {
434
431
  refresh() {
435
432
  if (this.destroyed)
436
433
  return;
437
- console.log('refresh----->');
438
434
  this.render();
439
435
  }
440
436
  // ==================== 拖拽功能 ====================
@@ -1051,15 +1047,37 @@ class PathLayer extends BaseLayer {
1051
1047
  d += ' Z ';
1052
1048
  }
1053
1049
  });
1054
- // 3. svgElements(直接拼接path字符串,建议逆时针)
1055
- if (Array.isArray(svgElements)) {
1056
- svgElements.forEach((svgPath) => {
1057
- const svgPathString = svgPath?.metadata?.svg;
1058
- if (svgPathString && typeof svgPathString === 'string' && svgPathString.trim()) {
1059
- d += svgPathString + ' ';
1050
+ // 3. svgElements(解析 SVG 字符串并提取 path 数据)
1051
+ Object.values(svgElements).forEach((svgPath) => {
1052
+ const svgPathString = svgPath?.metadata?.svg;
1053
+ if (svgPathString && typeof svgPathString === 'string' && svgPathString.trim()) {
1054
+ // 处理转义字符
1055
+ const processedSvgString = svgPathString.replace(/\\n/g, '\n').replace(/\\"/g, '"');
1056
+ // 解析 SVG 字符串
1057
+ const parser = new DOMParser();
1058
+ const svgDoc = parser.parseFromString(processedSvgString, 'image/svg+xml');
1059
+ const svgElement = svgDoc.documentElement;
1060
+ if (svgElement.tagName === 'svg') {
1061
+ // 查找 path 元素
1062
+ const pathElement = svgElement.querySelector('path');
1063
+ if (pathElement) {
1064
+ const pathData = pathElement.getAttribute('d');
1065
+ if (pathData) {
1066
+ // 获取 SVG 元素的变换参数
1067
+ const centerCoords = svgPath.coordinates?.[0] || [0, 0];
1068
+ const center = [centerCoords[0], centerCoords[1]];
1069
+ const userScale = svgPath.metadata.scale || 1;
1070
+ const direction = svgPath.metadata?.direction || 0;
1071
+ const originalWidth = parseFloat(svgElement.getAttribute('width') || '76');
1072
+ const originalHeight = parseFloat(svgElement.getAttribute('height') || '68');
1073
+ // 应用变换到路径数据
1074
+ const transformedPathData = this.transformSvgPath(pathData, center, userScale, direction, originalWidth, originalHeight);
1075
+ d += transformedPathData + ' ';
1076
+ }
1077
+ }
1060
1078
  }
1061
- });
1062
- }
1079
+ }
1080
+ });
1063
1081
  const path = document.createElementNS('http://www.w3.org/2000/svg', 'path');
1064
1082
  path.setAttribute('d', d);
1065
1083
  const clipPath = document.createElementNS('http://www.w3.org/2000/svg', 'clipPath');
@@ -1084,47 +1102,132 @@ class PathLayer extends BaseLayer {
1084
1102
  // 2. 创建一个组,应用 clipPath
1085
1103
  const group = document.createElementNS('http://www.w3.org/2000/svg', 'g');
1086
1104
  group.setAttribute('clip-path', `url(#${clipPathId})`);
1087
- group.setAttribute('opacity', '0.6'); // 统一透明度,防止叠加脏乱
1088
- // 3. 渲染所有路径
1105
+ group.setAttribute('opacity', '0.5'); // 统一透明度,防止叠加脏乱
1106
+ // 3. 优化渲染:按样式分组并合并路径
1107
+ this.renderOptimizedPaths(group);
1108
+ svgGroup.appendChild(group);
1109
+ }
1110
+ /**
1111
+ * 优化渲染:按样式分组并合并路径,减少 DOM 节点数量
1112
+ */
1113
+ renderOptimizedPaths(group) {
1114
+ // 按样式分组存储路径数据
1115
+ const styleGroups = new Map();
1116
+ // 收集所有路径数据并按样式分组
1089
1117
  for (const element of this.elements) {
1090
- const { id, elements } = element;
1118
+ // 类型断言:PathLayer 中的 elements 实际上是 PathElements 结构
1119
+ const pathElement = element;
1120
+ const { id, elements } = pathElement;
1091
1121
  this.boundaryPaths[id] = [];
1122
+ elements.forEach((pathElement) => {
1123
+ const { coordinates, style } = pathElement;
1124
+ if (coordinates.length < 2)
1125
+ return;
1126
+ // 生成样式键(用于分组)
1127
+ const styleKey = this.generateStyleKey(style);
1128
+ // 构建路径数据
1129
+ let pathData = `M ${coordinates[0][0]} ${coordinates[0][1]}`;
1130
+ for (let i = 1; i < coordinates.length; i++) {
1131
+ pathData += ` L ${coordinates[i][0]} ${coordinates[i][1]}`;
1132
+ }
1133
+ // 按样式分组存储
1134
+ if (!styleGroups.has(styleKey)) {
1135
+ styleGroups.set(styleKey, { pathData: [], elements: [] });
1136
+ }
1137
+ styleGroups.get(styleKey).pathData.push(pathData);
1138
+ styleGroups.get(styleKey).elements.push(pathElement);
1139
+ });
1140
+ }
1141
+ // 为每种样式创建一个合并的 path 元素
1142
+ styleGroups.forEach((groupData) => {
1143
+ const { pathData, elements } = groupData;
1144
+ if (pathData.length === 0)
1145
+ return;
1146
+ // 使用第一个元素的样式作为该组的样式
1147
+ const firstElement = elements[0];
1148
+ const style = firstElement.style;
1149
+ // 创建合并的 path 元素
1150
+ const path = document.createElementNS('http://www.w3.org/2000/svg', 'path');
1151
+ // 合并所有路径数据
1152
+ const mergedPathData = pathData.join(' ');
1153
+ path.setAttribute('d', mergedPathData);
1154
+ // 设置样式属性
1155
+ path.setAttribute('fill', 'none');
1156
+ path.setAttribute('stroke', style.lineColor || '#000000');
1157
+ path.setAttribute('mix-blend-mode', 'normal');
1158
+ const lineWidth = Math.max(style.lineWidth || 1, 0.5);
1159
+ path.setAttribute('stroke-width', lineWidth.toString());
1160
+ path.setAttribute('stroke-linecap', 'round');
1161
+ path.setAttribute('stroke-linejoin', 'round');
1162
+ path.classList.add('vector-path');
1163
+ // 将合并的 path 添加到组中
1164
+ group.appendChild(path);
1165
+ // 保存引用到 boundaryPaths 中(保持兼容性)
1092
1166
  elements.forEach((element) => {
1093
- this.renderPathToGroup(group, id, element);
1167
+ const { id } = element;
1168
+ if (!this.boundaryPaths[id]) {
1169
+ this.boundaryPaths[id] = [];
1170
+ }
1171
+ this.boundaryPaths[id].push(path);
1094
1172
  });
1173
+ });
1174
+ }
1175
+ /**
1176
+ * 变换 SVG 路径数据
1177
+ */
1178
+ transformSvgPath(pathData, center, scale, direction, originalWidth, originalHeight) {
1179
+ // 解析路径数据并应用变换
1180
+ const commands = pathData.match(/[MmLlHhVvCcSsQqTtAaZz][^MmLlHhVvCcSsQqTtAaZz]*/g) || [];
1181
+ let transformedCommands = [];
1182
+ for (const command of commands) {
1183
+ const type = command[0];
1184
+ const params = command
1185
+ .slice(1)
1186
+ .trim()
1187
+ .split(/[\s,]+/)
1188
+ .filter(Boolean)
1189
+ .map(Number);
1190
+ if (type === 'Z' || type === 'z') {
1191
+ // 闭合路径,不需要变换
1192
+ transformedCommands.push(command);
1193
+ continue;
1194
+ }
1195
+ // 处理坐标参数
1196
+ let transformedParams = [];
1197
+ for (let i = 0; i < params.length; i += 2) {
1198
+ if (i + 1 < params.length) {
1199
+ let x = params[i];
1200
+ let y = params[i + 1];
1201
+ // 应用变换:先平移到中心,然后缩放、旋转,最后平移到目标位置
1202
+ // 1. 平移到原点(相对于原始尺寸的中心)
1203
+ x -= originalWidth / 2;
1204
+ y -= originalHeight / 2;
1205
+ // 2. 应用缩放
1206
+ x *= scale;
1207
+ y *= scale;
1208
+ // 3. 应用旋转
1209
+ const cos = Math.cos(-direction);
1210
+ const sin = Math.sin(-direction);
1211
+ const newX = x * cos - y * sin;
1212
+ const newY = x * sin + y * cos;
1213
+ // 4. 平移到目标位置
1214
+ x = newX + center[0];
1215
+ y = newY + center[1];
1216
+ transformedParams.push(x, y);
1217
+ }
1218
+ }
1219
+ // 重建命令
1220
+ if (transformedParams.length > 0) {
1221
+ transformedCommands.push(type + transformedParams.join(' '));
1222
+ }
1095
1223
  }
1096
- svgGroup.appendChild(group);
1224
+ return transformedCommands.join(' ');
1097
1225
  }
1098
1226
  /**
1099
- * 渲染单个路径到指定的组中
1227
+ * 生成样式键,用于路径分组
1100
1228
  */
1101
- renderPathToGroup(group, id, element) {
1102
- const { coordinates, style } = element;
1103
- if (coordinates.length < 2)
1104
- return;
1105
- const path = document.createElementNS('http://www.w3.org/2000/svg', 'path');
1106
- // 构建路径数据
1107
- let pathData = `M ${coordinates[0][0]} ${coordinates[0][1]}`;
1108
- for (let i = 1; i < coordinates.length; i++) {
1109
- pathData += ` L ${coordinates[i][0]} ${coordinates[i][1]}`;
1110
- }
1111
- path.style.mixBlendMode = 'normal';
1112
- // 设置路径属性
1113
- path.setAttribute('d', pathData);
1114
- // 直接给fill的颜色设置透明度会导致path重叠的部分颜色叠加,所以使用fill填充实色,通过fill-opacity设置透明度
1115
- path.setAttribute('fill', 'none');
1116
- // path.setAttribute('fill-opacity', '0.4');
1117
- path.setAttribute('stroke', style.lineColor || '#000000');
1118
- path.setAttribute('mix-blend-mode', 'normal');
1119
- const lineWidth = Math.max(style.lineWidth || 1, 0.5);
1120
- path.setAttribute('stroke-width', lineWidth.toString());
1121
- path.setAttribute('stroke-linecap', 'round');
1122
- path.setAttribute('stroke-linejoin', 'round');
1123
- // 注意:这里不设置 opacity,因为透明度由父组控制
1124
- // path.setAttribute('vector-effect', 'non-scaling-stroke');
1125
- path.classList.add('vector-path');
1126
- this.boundaryPaths[id].push(path);
1127
- group.appendChild(path);
1229
+ generateStyleKey(style) {
1230
+ return `${style.lineColor || '#000000'}-${style.lineWidth || 1}-${style.opacity || 1}`;
1128
1231
  }
1129
1232
  }
1130
1233
 
@@ -1548,7 +1651,7 @@ const DOODLE_STYLES = {
1548
1651
  lineColor: '#ff5722',
1549
1652
  fillColor: '#ff9800', // 粉色半透明填充
1550
1653
  lineWidth: DEFAULT_LINE_WIDTHS.TIME_LIMIT_OBSTACLE,
1551
- opacity: DEFAULT_OPACITIES.HIGH,
1654
+ opacity: DEFAULT_OPACITIES.DOODLE,
1552
1655
  };
1553
1656
  const PATH_EDGE_STYLES = {
1554
1657
  lineWidth: DEFAULT_LINE_WIDTHS.PATH,
@@ -1590,13 +1693,13 @@ const DEFAULT_STYLES = {
1590
1693
  function convertPointsFormat(points) {
1591
1694
  if (!points || points.length === 0)
1592
1695
  return null;
1593
- return points.map(point => {
1696
+ return points.map((point) => {
1594
1697
  if (point.length >= 2) {
1595
1698
  // 对前两个元素应用缩放因子,保留其他元素
1596
1699
  return [
1597
1700
  point[0] * SCALE_FACTOR,
1598
1701
  -point[1] * SCALE_FACTOR, // Y轴翻转,与Python代码一致
1599
- ...point.slice(2) // 保留第三个及以后的元素
1702
+ ...point.slice(2), // 保留第三个及以后的元素
1600
1703
  ];
1601
1704
  }
1602
1705
  return point;
@@ -1611,7 +1714,7 @@ function convertPositionFormat(position) {
1611
1714
  return null;
1612
1715
  return {
1613
1716
  x: position[0] * SCALE_FACTOR,
1614
- y: -position[1] * SCALE_FACTOR // Y轴翻转
1717
+ y: -position[1] * SCALE_FACTOR, // Y轴翻转
1615
1718
  };
1616
1719
  }
1617
1720
  /**
@@ -1620,9 +1723,146 @@ function convertPositionFormat(position) {
1620
1723
  function convertCoordinate(x, y) {
1621
1724
  return {
1622
1725
  x: x * SCALE_FACTOR,
1623
- y: -y * SCALE_FACTOR // Y轴翻转
1726
+ y: -y * SCALE_FACTOR, // Y轴翻转
1624
1727
  };
1625
1728
  }
1729
+ /**
1730
+ * @param x x坐标
1731
+ * @param y y坐标
1732
+ * @param isAllowInBoundary 是否允许点在边界上的判断
1733
+ * @return ture-点在边界上即可视为在边界内,false-严格判断点在边界内
1734
+ */
1735
+ function isPointIn$1(x, y, pointList, isAllowInBoundary) {
1736
+ let count = 0;
1737
+ let size = pointList.length;
1738
+ let p1, p2, p3;
1739
+ for (let i = 0; i < size; i++) {
1740
+ p1 = pointList[i];
1741
+ p2 = pointList[(i + 1) % size];
1742
+ if (p1.y == null || p2.y == null || p1.x == null || p2.x == null) {
1743
+ continue;
1744
+ }
1745
+ if (p1.y === p2.y) {
1746
+ continue;
1747
+ }
1748
+ if (y > Math.min(p1.y, p2.y) && y < Math.max(p1.y, p2.y)) {
1749
+ const interX = ((y - p1.y) * (p2.x - p1.x)) / (p2.y - p1.y) + p1.x;
1750
+ if (interX >= x) {
1751
+ count++;
1752
+ }
1753
+ else if (interX == x) {
1754
+ return isAllowInBoundary;
1755
+ }
1756
+ }
1757
+ else {
1758
+ if (y == p2.y && x <= p2.x) {
1759
+ p3 = pointList[(i + 2) % size];
1760
+ if (y >= Math.min(p1.y, p3.y) && y <= Math.max(p1.y, p3.y)) {
1761
+ // 若当前点的y坐标位于 p1和p3组成的线段关于y轴的投影中,则记为该点的射线只穿过端点一次。
1762
+ ++count;
1763
+ }
1764
+ else {
1765
+ // 若当前点的y坐标不能包含在p1和p3组成的线段关于y轴的投影中,则点射线通过的两条线段组成了一个弯折的部分,
1766
+ // 此时我们记射线穿过该端点两次
1767
+ count += 2;
1768
+ }
1769
+ }
1770
+ }
1771
+ }
1772
+ return count % 2 == 1;
1773
+ }
1774
+ /**
1775
+ * 用于判断三个点的方向的辅助方法
1776
+ */
1777
+ function orientation(p, q, r) {
1778
+ const val = (q.y - p.y) * (r.x - q.x) - (q.x - p.x) * (r.y - q.y);
1779
+ if (val == 0)
1780
+ return 0; // colinear
1781
+ return val > 0 ? 1 : 2; // clock or counterclock wise
1782
+ }
1783
+ /**
1784
+ * 检查点q是否在线段pr上的辅助方法
1785
+ */
1786
+ function onSegment(p, q, r) {
1787
+ if (q.x <= Math.max(p.x, r.x) &&
1788
+ q.x >= Math.min(p.x, r.x) &&
1789
+ q.y <= Math.max(p.y, r.y) &&
1790
+ q.y >= Math.min(p.y, r.y)) {
1791
+ return true;
1792
+ }
1793
+ return false;
1794
+ }
1795
+ /**
1796
+ * 判断两条线段是否相交的方法
1797
+ */
1798
+ function doTwoLinesIntersect(p1, q1, p2, q2) {
1799
+ //处理p1和q1两个点相同的情况
1800
+ if (p1.x - q1.x == 0 && p1.y - q1.y == 0) {
1801
+ return false;
1802
+ }
1803
+ if (p2.x - q2.x == 0 && p2.y - q2.y == 0) {
1804
+ return false;
1805
+ }
1806
+ // 计算四个点的方向
1807
+ const o1 = orientation(p1, q1, p2);
1808
+ const o2 = orientation(p1, q1, q2);
1809
+ const o3 = orientation(p2, q2, p1);
1810
+ const o4 = orientation(p2, q2, q1);
1811
+ // 一般情况,如果四个方向两两不同,则线段相交
1812
+ if (o1 != o2 && o3 != o4) {
1813
+ return true;
1814
+ }
1815
+ // 特殊情况,当线段的端点在另一条线段上时
1816
+ if (o1 == 0 && onSegment(p1, q1, p2))
1817
+ return true;
1818
+ if (o2 == 0 && onSegment(p1, q1, q2))
1819
+ return true;
1820
+ if (o3 == 0 && onSegment(p2, q2, p1))
1821
+ return true;
1822
+ if (o4 == 0 && onSegment(p2, q2, q1))
1823
+ return true;
1824
+ // 如果以上情况都不满足,则线段不相交
1825
+ return false;
1826
+ }
1827
+ /**
1828
+ * 判断多点折线是否相交
1829
+ */
1830
+ function doIntersect(points1, points2) {
1831
+ if (points1 == null || points2 == null || points1.length < 3 || points2.length < 3) {
1832
+ return false;
1833
+ }
1834
+ for (let i = 0; i < points1.length - 1; i++) {
1835
+ for (let j = 0; j < points2.length - 1; j++) {
1836
+ if (doTwoLinesIntersect(points1[i], points1[i + 1], points2[j], points2[j + 1])) {
1837
+ return true;
1838
+ }
1839
+ }
1840
+ }
1841
+ return false;
1842
+ }
1843
+ /**
1844
+ * 两个图形是否完全分离,互相不包含
1845
+ */
1846
+ function isOutsideToEachOther(points1, points2) {
1847
+ // 相交关系
1848
+ if (doIntersect(points1, points2)) {
1849
+ return false;
1850
+ }
1851
+ // 点关系,判断每个图形的点都在另一个图形外部
1852
+ for (let point of points1) {
1853
+ if (isPointIn$1(point.x, point.y, points2, true)) {
1854
+ // Log.i("ycf", "isOutsideToEachOther: mapPoint1=" + mapPoint);
1855
+ return false;
1856
+ }
1857
+ }
1858
+ for (let point of points2) {
1859
+ if (isPointIn$1(point.x, point.y, points1, true)) {
1860
+ // Log.i("ycf", "isOutsideToEachOther: mapPoint2=" + mapPoint);
1861
+ return false;
1862
+ }
1863
+ }
1864
+ return true;
1865
+ }
1626
1866
 
1627
1867
  /**
1628
1868
  * 按Python逻辑创建路径段:根据连续的两点之间的关系确定线段类型
@@ -1885,6 +2125,136 @@ function calculateMapGpsCenter(mapData) {
1885
2125
  };
1886
2126
  }
1887
2127
 
2128
+ /**
2129
+ * 并查集(Union-Find)是一种非常高效的数据结构,用于处理动态连通性问题。
2130
+ * 它可以快速判断网络中任意两点是否连通,并能将不连通的集合合并。
2131
+ */
2132
+ class UnionFind {
2133
+ /**
2134
+ * 构造函数,n为图的节点总数
2135
+ * @param {number} n - 节点总数
2136
+ */
2137
+ constructor(n) {
2138
+ this.count = n; // 连通分量的数量
2139
+ this.parent = new Array(n); // parent[i]表示第i个元素所指向的父节点
2140
+ // 初始时,每个节点的父节点是自己
2141
+ for (let i = 0; i < n; i++) {
2142
+ this.parent[i] = i;
2143
+ }
2144
+ }
2145
+ /**
2146
+ * 查找元素p所对应的集合编号(根节点)
2147
+ * @param {number} p - 要查找的元素
2148
+ * @returns {number} 根节点的编号
2149
+ */
2150
+ find(p) {
2151
+ while (p !== this.parent[p]) {
2152
+ this.parent[p] = this.parent[this.parent[p]]; // 路径压缩
2153
+ p = this.parent[p];
2154
+ }
2155
+ return p;
2156
+ }
2157
+ /**
2158
+ * 判断元素p和元素q是否属于同一集合
2159
+ * @param {number} p - 第一个元素
2160
+ * @param {number} q - 第二个元素
2161
+ * @returns {boolean} 是否连通
2162
+ */
2163
+ isConnected(p, q) {
2164
+ return this.find(p) === this.find(q);
2165
+ }
2166
+ /**
2167
+ * 合并元素p和元素q所属的集合
2168
+ * @param {number} p - 第一个元素
2169
+ * @param {number} q - 第二个元素
2170
+ */
2171
+ union(p, q) {
2172
+ const rootP = this.find(p);
2173
+ const rootQ = this.find(q);
2174
+ if (rootP === rootQ) {
2175
+ return; // 已经在同一个集合中
2176
+ }
2177
+ // 将较小的根节点作为父节点(按秩合并的简化版本)
2178
+ if (rootP < rootQ) {
2179
+ this.parent[rootQ] = rootP;
2180
+ }
2181
+ else {
2182
+ this.parent[rootP] = rootQ;
2183
+ }
2184
+ // 两个集合合并成一个集合,连通分量减1
2185
+ this.count--;
2186
+ }
2187
+ /**
2188
+ * 获取当前的连通分量个数
2189
+ * @returns {number} 连通分量数量
2190
+ */
2191
+ getCount() {
2192
+ return this.count;
2193
+ }
2194
+ /**
2195
+ * 获取联通的组
2196
+ * @param {Array} list - 原始元素列表
2197
+ * @returns {Array<Set>} 联通组列表
2198
+ */
2199
+ getConnectedGroup(list) {
2200
+ if (!list || list.length === 0 || !this.parent || this.parent.length === 0) {
2201
+ return null;
2202
+ }
2203
+ if (list.length !== this.parent.length) {
2204
+ return null;
2205
+ }
2206
+ const map = new Map();
2207
+ // 遍历所有元素,按根节点分组
2208
+ for (let i = 0; i < this.parent.length; i++) {
2209
+ const root = this.parent[i];
2210
+ if (!map.has(root)) {
2211
+ map.set(root, new Set());
2212
+ }
2213
+ map.get(root).add(list[i]);
2214
+ }
2215
+ return Array.from(map.values());
2216
+ }
2217
+ /**
2218
+ * 重置并查集
2219
+ * @param {number} n - 新的节点总数
2220
+ */
2221
+ reset(n) {
2222
+ this.count = n;
2223
+ this.parent = new Array(n);
2224
+ for (let i = 0; i < n; i++) {
2225
+ this.parent[i] = i;
2226
+ }
2227
+ }
2228
+ }
2229
+
2230
+ function isTunnelConnected(a, b, connectIds) {
2231
+ if (!a || !b)
2232
+ return false;
2233
+ if (!connectIds || connectIds?.length === 0)
2234
+ return false;
2235
+ const temp = [a?.id, b?.id];
2236
+ temp.sort();
2237
+ return connectIds?.includes(temp?.join('-'));
2238
+ }
2239
+ function isOverlayConnected(a, b) {
2240
+ if (!a || !b) {
2241
+ return false;
2242
+ }
2243
+ if (!a?.points?.length || !b?.points?.length) {
2244
+ return false;
2245
+ }
2246
+ const aPoints = a?.points?.map(item => ({ x: item[0], y: item[1] }));
2247
+ const bPoints = b?.points?.map(item => ({ x: item[0], y: item[1] }));
2248
+ try {
2249
+ if (isOutsideToEachOther(aPoints, bPoints)) {
2250
+ return false;
2251
+ }
2252
+ }
2253
+ catch (error) {
2254
+ console.log('error->', error);
2255
+ }
2256
+ return true;
2257
+ }
1888
2258
  /**
1889
2259
  * 通过 mapData 和 pathData 生成所有 boundary 的数据
1890
2260
  * @param mapData 地图数据
@@ -1893,11 +2263,12 @@ function calculateMapGpsCenter(mapData) {
1893
2263
  */
1894
2264
  function generateBoundaryData(mapData, pathData) {
1895
2265
  const boundaryData = [];
2266
+ let chargingPileBoundary = undefined;
1896
2267
  if (!mapData || !mapData.sub_maps) {
1897
2268
  return boundaryData;
1898
2269
  }
1899
2270
  // 第一步:收集所有TUNNEL数据的connection信息
1900
- const connectedBoundaryIds = new Set();
2271
+ const connectIds = [];
1901
2272
  // 遍历mapData中的tunnels字段
1902
2273
  if (mapData.tunnels && Array.isArray(mapData.tunnels)) {
1903
2274
  for (const tunnel of mapData.tunnels) {
@@ -1905,10 +2276,8 @@ function generateBoundaryData(mapData, pathData) {
1905
2276
  if (connection) {
1906
2277
  // connection可能是单个数字或数组
1907
2278
  if (Array.isArray(connection)) {
1908
- connection.forEach(id => connectedBoundaryIds.add(id));
1909
- }
1910
- else if (typeof connection === 'number') {
1911
- connectedBoundaryIds.add(connection);
2279
+ connection.sort();
2280
+ connectIds.push(connection.join('-'));
1912
2281
  }
1913
2282
  }
1914
2283
  }
@@ -1919,9 +2288,9 @@ function generateBoundaryData(mapData, pathData) {
1919
2288
  if (!subMap.elements)
1920
2289
  continue;
1921
2290
  // 每个sub_map的elements是边界坐标,没有sub_map只有一个boundary数据
1922
- const boundaryElement = subMap.elements.find(element => element.type === 'BOUNDARY');
2291
+ const boundaryElement = subMap.elements.find((element) => element.type === 'BOUNDARY');
1923
2292
  // 如果当前subMap存在充电桩且充电桩存在tunnel,说明当前subMap中的boundary是初始boundary,这个boundary不为孤立区域
1924
- const hasTunnelToChargingPile = subMap.elements.some(element => element.type === 'CHARGING_PILE' && element.tunnel);
2293
+ const hasTunnelToChargingPile = subMap.elements.some((element) => element.type === 'CHARGING_PILE' && element.tunnel);
1925
2294
  // 创建基础的 boundary 数据(来自 mapData)
1926
2295
  const boundary = {
1927
2296
  // 从 BOUNDARY 元素复制属性
@@ -1930,8 +2299,6 @@ function generateBoundaryData(mapData, pathData) {
1930
2299
  area: subMap?.area,
1931
2300
  points: convertPointsFormat(boundaryElement?.points) || [],
1932
2301
  type: boundaryElement.type,
1933
- // 判断是否为孤立子区域
1934
- isIsolated: hasTunnelToChargingPile ? false : !connectedBoundaryIds.has(boundaryElement.id)
1935
2302
  };
1936
2303
  // 如果有 pathData,尝试匹配对应的分区数据
1937
2304
  if (pathData) {
@@ -1947,8 +2314,33 @@ function generateBoundaryData(mapData, pathData) {
1947
2314
  boundary.endTime = partitionData.endTime;
1948
2315
  }
1949
2316
  }
2317
+ if (hasTunnelToChargingPile) {
2318
+ chargingPileBoundary = boundary;
2319
+ }
1950
2320
  boundaryData.push(boundary);
1951
2321
  }
2322
+ const unionFind = new UnionFind(boundaryData?.length);
2323
+ for (let i = 0; i < boundaryData?.length - 1; i++) {
2324
+ for (let j = i + 1; j < boundaryData?.length; j++) {
2325
+ const boundary1 = boundaryData[i];
2326
+ const boundary2 = boundaryData[j];
2327
+ const isChannelConnect = isTunnelConnected(boundary1, boundary2, connectIds);
2328
+ const isOverlayConnect = isOverlayConnected(boundary1, boundary2);
2329
+ if (isChannelConnect || isOverlayConnect) {
2330
+ unionFind.union(i, j);
2331
+ }
2332
+ }
2333
+ }
2334
+ const tunnelAndOverlayList = unionFind.getConnectedGroup(boundaryData);
2335
+ const chargingPileConnectBoundarys = tunnelAndOverlayList?.find(item => item?.has(chargingPileBoundary));
2336
+ for (let boundary of boundaryData) {
2337
+ if (chargingPileConnectBoundarys?.has(boundary)) {
2338
+ boundary.isIsolated = false;
2339
+ }
2340
+ else {
2341
+ boundary.isIsolated = true;
2342
+ }
2343
+ }
1952
2344
  return boundaryData;
1953
2345
  }
1954
2346
 
@@ -2165,13 +2557,15 @@ var hNoPosition = "
2165
2557
 
2166
2558
  var hDisabled = "";
2167
2559
 
2560
+ var x3Edger = "";
2561
+
2168
2562
  var x3Mower = "";
2169
2563
 
2170
2564
  var x3NoPosition = "";
2171
2565
 
2172
2566
  var x3Disabled = "";
2173
2567
 
2174
- function getMowerImageByModal(mowerModal) {
2568
+ function getMowerImageByModal(mowerModal, hasEdger) {
2175
2569
  if (mowerModal.includes('i')) {
2176
2570
  return iMower;
2177
2571
  }
@@ -2179,7 +2573,7 @@ function getMowerImageByModal(mowerModal) {
2179
2573
  return hMower;
2180
2574
  }
2181
2575
  else if (mowerModal.includes('x3')) {
2182
- return x3Mower;
2576
+ return hasEdger ? x3Edger : x3Mower;
2183
2577
  }
2184
2578
  return iMower;
2185
2579
  }
@@ -2207,12 +2601,12 @@ function getNoPositionMowerImageByModal(mowerModal) {
2207
2601
  }
2208
2602
  return iNoPosition;
2209
2603
  }
2210
- function getMowerImage(positonConfig, modelType) {
2604
+ function getMowerImage(positonConfig, modelType, hasEdger) {
2211
2605
  if (!positonConfig)
2212
2606
  return '';
2213
2607
  const model = modelType?.toLowerCase() || 'i';
2214
2608
  const state = positonConfig.vehicleState;
2215
- const mowerImage = getMowerImageByModal(model);
2609
+ const mowerImage = getMowerImageByModal(model, hasEdger);
2216
2610
  const disabledImage = getDisabledMowerImageByModal(model);
2217
2611
  const noPositionImage = getNoPositionMowerImageByModal(model);
2218
2612
  const positonOutOfRange = isOutOfRange(positonConfig);
@@ -4694,8 +5088,8 @@ var PathSegmentType;
4694
5088
  */
4695
5089
  var UnitsType;
4696
5090
  (function (UnitsType) {
4697
- UnitsType["Metric"] = "metric";
4698
- UnitsType["Imperial"] = "imperial";
5091
+ UnitsType["Metric"] = "Metric";
5092
+ UnitsType["Imperial"] = "Imperial";
4699
5093
  })(UnitsType || (UnitsType = {}));
4700
5094
  /**
4701
5095
  * 面积单位类型枚举
@@ -4982,6 +5376,12 @@ class BoundaryBorderLayer extends BaseLayer {
4982
5376
  this.mowingBoundarys = mowingBoundarys;
4983
5377
  }
4984
5378
  }
5379
+ /**
5380
+ * 获取当前割草任务的边界
5381
+ */
5382
+ getMowingBoundarys() {
5383
+ return this.mowingBoundarys;
5384
+ }
4985
5385
  /**
4986
5386
  * SVG渲染方法
4987
5387
  */
@@ -4990,13 +5390,31 @@ class BoundaryBorderLayer extends BaseLayer {
4990
5390
  return;
4991
5391
  }
4992
5392
  this.scale = scale;
4993
- console.log('draw boundary border->', this.elements, this.mowingBoundarys);
4994
- // 只渲染边界边框类型的元素
5393
+ // 将元素分为两组:非割草边界和割草边界
5394
+ const nonMowingElements = [];
5395
+ const mowingElements = [];
5396
+ // 只处理边界边框类型的元素
4995
5397
  for (const element of this.elements) {
4996
5398
  if (element.type === 'boundary_border') {
4997
- this.renderBoundaryBorder(svgGroup, element);
5399
+ const { originalData } = element;
5400
+ const { id } = originalData || {};
5401
+ // 检查是否为割草边界
5402
+ if (this.mowingBoundarys.includes(Number(id))) {
5403
+ mowingElements.push(element);
5404
+ }
5405
+ else {
5406
+ nonMowingElements.push(element);
5407
+ }
4998
5408
  }
4999
5409
  }
5410
+ // 先渲染非割草边界
5411
+ for (const element of nonMowingElements) {
5412
+ this.renderBoundaryBorder(svgGroup, element);
5413
+ }
5414
+ // 再渲染割草边界(放在最后)
5415
+ for (const element of mowingElements) {
5416
+ this.renderBoundaryBorder(svgGroup, element);
5417
+ }
5000
5418
  }
5001
5419
  /**
5002
5420
  * 渲染边界边框
@@ -6065,7 +6483,7 @@ class PathDataProcessor {
6065
6483
  * 专门处理边界标签的创建、定位和管理
6066
6484
  */
6067
6485
  class BoundaryLabelsManager {
6068
- constructor(svgView, boundaryData) {
6486
+ constructor(svgView, boundaryData, { unitType, language }) {
6069
6487
  this.container = null;
6070
6488
  this.overlayDiv = null;
6071
6489
  this.globalClickHandler = null;
@@ -6076,6 +6494,8 @@ class BoundaryLabelsManager {
6076
6494
  this.svgView = svgView;
6077
6495
  this.boundaryData = boundaryData;
6078
6496
  this.initializeContainer();
6497
+ this.unitType = unitType;
6498
+ this.language = language;
6079
6499
  }
6080
6500
  /**
6081
6501
  * 初始化容器
@@ -6141,7 +6561,7 @@ class BoundaryLabelsManager {
6141
6561
  labelDiv.setAttribute('data-boundary-id', boundary.id.toString());
6142
6562
  // 样式设置
6143
6563
  labelDiv.style.position = 'absolute';
6144
- labelDiv.style.backgroundColor = 'rgba(30, 30, 31, 0.3)';
6564
+ labelDiv.style.backgroundColor = 'rgba(30, 30, 31, 0.6)';
6145
6565
  labelDiv.style.color = 'rgba(255, 255, 255, 1)';
6146
6566
  labelDiv.style.padding = '6px';
6147
6567
  labelDiv.style.borderRadius = '12px';
@@ -6158,7 +6578,7 @@ class BoundaryLabelsManager {
6158
6578
  labelDiv.style.zIndex = BoundaryLabelsManager.Z_INDEX.DEFAULT.toString();
6159
6579
  // 计算进度
6160
6580
  const progress = boundary.finishedArea && boundary.area
6161
- ? `${Math.round((boundary.finishedArea / boundary.area) * 100)}%`
6581
+ ? `${Math.floor((boundary.finishedArea / boundary.area) * 100)}%`
6162
6582
  : '0%';
6163
6583
  // 基础内容(始终显示)
6164
6584
  const baseContent = document.createElement('div');
@@ -6175,12 +6595,15 @@ class BoundaryLabelsManager {
6175
6595
  this.currentExpandedBoundaryId === boundary.id ? 'block' : 'none';
6176
6596
  extendedContent.style.borderTop = '1px solid rgba(255,255,255,0.2)';
6177
6597
  extendedContent.style.paddingTop = '6px';
6598
+ const boundaryLayer = this.svgView.getLayer(LAYER_DEFAULT_TYPE.BOUNDARY_BORDER);
6599
+ const mowingBoundarys = boundaryLayer.getMowingBoundarys();
6178
6600
  // 面积信息
6179
- const totalArea = convertAreaByUnits(boundary.area || 0, 'metric');
6180
- const finishedArea = convertAreaByUnits(boundary.finishedArea || 0, 'metric');
6601
+ const totalArea = convertAreaByUnits(boundary.area || 0, this.unitType);
6602
+ const finishedArea = convertAreaByUnits(boundary.finishedArea || 0, this.unitType);
6181
6603
  const coverageText = `Coverage: ${finishedArea.value}/${totalArea.value}`;
6604
+ const isMowing = mowingBoundarys.includes(boundary.id);
6182
6605
  // 日期信息
6183
- const dateText = formatBoundaryDateText(boundary.endTime || 0);
6606
+ const dateText = formatBoundaryDateText(isMowing ? Date.now() / 1000 : boundary.endTime || 0);
6184
6607
  const covertHtml = `<div style="margin-bottom: 3px; font-weight: bold;">${coverageText}</div>`;
6185
6608
  const dateHtml = `<div>${dateText}</div>`;
6186
6609
  extendedContent.innerHTML = boundary.finishedArea > 0 ? `${covertHtml}${dateHtml}` : covertHtml;
@@ -6298,7 +6721,6 @@ class BoundaryLabelsManager {
6298
6721
  // 计算边界中心点的地图坐标
6299
6722
  const mapCenter = this.calculatePolygonCentroid(boundary.points);
6300
6723
  if (!mapCenter) {
6301
- console.warn(`BoundaryLabelsManager: 无法计算边界 ${boundary.name} (ID: ${boundary.id}) 的中心点`);
6302
6724
  return;
6303
6725
  }
6304
6726
  // 直接使用预计算的数据进行坐标转换
@@ -6383,7 +6805,6 @@ class BoundaryLabelsManager {
6383
6805
  area = area / 2;
6384
6806
  // 如果面积为0,回退到简单的平均值计算
6385
6807
  if (Math.abs(area) < 1e-10) {
6386
- console.warn('BoundaryLabelsManager: 多边形面积为0,使用平均值计算重心');
6387
6808
  return this.calculateAverageCenter(validPoints);
6388
6809
  }
6389
6810
  centroidX = centroidX / (6 * area);
@@ -7294,6 +7715,10 @@ class MowerPositionManager {
7294
7715
  getElement() {
7295
7716
  return this.container;
7296
7717
  }
7718
+ //
7719
+ setEdger(edger) {
7720
+ this.hasEdger = edger;
7721
+ }
7297
7722
  /**
7298
7723
  * 根据最后一次有效的位置更新数据
7299
7724
  */
@@ -7317,7 +7742,6 @@ class MowerPositionManager {
7317
7742
  postureY = chargingPilesPositionConfig.postureY || 0;
7318
7743
  postureTheta = chargingPilesPositionConfig.postureTheta || 0;
7319
7744
  }
7320
- console.log('updatePositionByLastPosition->', postureX, postureY, postureTheta, chargingPilesPositionConfig);
7321
7745
  // 检查是否需要更新图片
7322
7746
  this.updateMowerImage(chargingPilesPositionConfig);
7323
7747
  // 立即更新位置
@@ -7334,7 +7758,6 @@ class MowerPositionManager {
7334
7758
  const postureX = positionConfig?.postureX || this.lastPosition?.x || 0;
7335
7759
  const postureY = positionConfig?.postureY || this.lastPosition?.y || 0;
7336
7760
  const postureTheta = positionConfig?.postureTheta || this.lastPosition?.rotation || 0;
7337
- console.log('updatePosition manager', JSON.stringify(this.currentPosition), this.currentPosition, !this.currentPosition, positionConfig, this.lastPosition, animationTime);
7338
7761
  // 停止当前动画(如果有)
7339
7762
  this.stopAnimation();
7340
7763
  // 第一个点
@@ -7344,7 +7767,6 @@ class MowerPositionManager {
7344
7767
  y: postureY,
7345
7768
  rotation: postureTheta,
7346
7769
  };
7347
- console.log('updatePosition first->', this.currentPosition);
7348
7770
  this.setElementPosition(this.currentPosition.x, this.currentPosition.y, this.currentPosition.rotation);
7349
7771
  return;
7350
7772
  }
@@ -7366,7 +7788,7 @@ class MowerPositionManager {
7366
7788
  const imgElement = this.mowerElement.querySelector('img');
7367
7789
  if (!imgElement)
7368
7790
  return;
7369
- const imageSrc = getMowerImage(positonConfig, this.modelType);
7791
+ const imageSrc = getMowerImage(positonConfig, this.modelType, this.hasEdger);
7370
7792
  if (imageSrc) {
7371
7793
  imgElement.src = imageSrc;
7372
7794
  imgElement.style.display = 'block';
@@ -7453,12 +7875,10 @@ class MowerPositionManager {
7453
7875
  y: this.onlyUpdateTheta ? 0 : this.targetPosition.y - this.startPosition.y,
7454
7876
  rotation: radNormalize(targetTheta - startTheta),
7455
7877
  };
7456
- console.log('startAnimationToPosition-->', this.deltaPosition, this.onlyUpdateTheta, this.targetPosition, this.startPosition);
7457
7878
  // 开始动画循环
7458
7879
  this.animateStep();
7459
7880
  }
7460
7881
  forceUpdatePosition() {
7461
- console.log('forceUpdatePosition-->', this.currentPosition, this.targetPosition, this.startPosition);
7462
7882
  this.animateStep();
7463
7883
  }
7464
7884
  /**
@@ -7599,15 +8019,40 @@ function throttleAdvanced(func, delay, options = { leading: true, trailing: true
7599
8019
  }
7600
8020
  };
7601
8021
  }
8022
+ /**
8023
+ * 检测当前设备是否为移动设备
8024
+ * @returns {boolean} 如果是移动设备返回true,否则返回false
8025
+ */
8026
+ function isMobileDevice() {
8027
+ // 确保在浏览器环境中运行
8028
+ if (typeof window === 'undefined' || typeof navigator === 'undefined') {
8029
+ return false;
8030
+ }
8031
+ // 检查用户代理字符串
8032
+ const userAgent = navigator.userAgent.toLowerCase();
8033
+ const mobileKeywords = [
8034
+ 'android', 'webos', 'iphone', 'ipad', 'ipod',
8035
+ 'blackberry', 'windows phone', 'mobile'
8036
+ ];
8037
+ const isMobileUserAgent = mobileKeywords.some(keyword => userAgent.includes(keyword));
8038
+ // 检查触摸屏支持
8039
+ const hasTouchScreen = 'ontouchstart' in window ||
8040
+ (navigator.maxTouchPoints && navigator.maxTouchPoints > 0);
8041
+ // 检查屏幕尺寸(移动设备通常屏幕较小)
8042
+ const isSmallScreen = window.innerWidth <= 768;
8043
+ // 综合判断:用户代理包含移动设备关键词,或者有触摸屏且屏幕较小
8044
+ return isMobileUserAgent || (hasTouchScreen && isSmallScreen);
8045
+ }
7602
8046
 
7603
8047
  // Google Maps 叠加层类 - 带编辑功能
7604
8048
  class MowerMapOverlay {
7605
- constructor(bounds, mapData, partitionBoundary, mowerPositionConfig, modelType, pathData, isEditMode = false, mapConfig = {}, antennaConfig = {}, mowPartitionData = null, defaultTransform, onMapLoad, onPathLoad, dragCallbacks) {
8049
+ constructor(bounds, mapData, partitionBoundary, mowerPositionConfig, modelType, pathData, isEditMode = false, unitType = UnitsType.Imperial, language = 'en', mapConfig = {}, antennaConfig = {}, mowPartitionData = null, defaultTransform, onMapLoad, onPathLoad, dragCallbacks) {
7606
8050
  this.div = null;
7607
8051
  this.svgMapView = null;
7608
8052
  this.offscreenContainer = null;
7609
8053
  this.overlayView = null;
7610
8054
  this.defaultTransform = { x: 0, y: 0, rotation: 0 };
8055
+ this.hasEdger = false;
7611
8056
  // boundary数据
7612
8057
  this.boundaryData = [];
7613
8058
  // 边界标签管理器
@@ -7650,6 +8095,8 @@ class MowerMapOverlay {
7650
8095
  this.partitionBoundary = partitionBoundary;
7651
8096
  this.pathData = pathData;
7652
8097
  this.isEditMode = isEditMode;
8098
+ this.unitType = unitType;
8099
+ this.language = language;
7653
8100
  this.mapConfig = mapConfig;
7654
8101
  this.antennaConfig = antennaConfig;
7655
8102
  this.onMapLoad = onMapLoad;
@@ -7696,7 +8143,6 @@ class MowerMapOverlay {
7696
8143
  this.isUserAnimation = animationTime > 0;
7697
8144
  // 更新割草机位置配置
7698
8145
  this.mowerPositionConfig = positionConfig;
7699
- console.log('updatePosition overlay', positionConfig);
7700
8146
  // 更新割草机位置管理器
7701
8147
  if (this.mowerPositionManager) {
7702
8148
  this.mowerPositionManager.updatePosition(positionConfig, animationTime);
@@ -7714,6 +8160,12 @@ class MowerMapOverlay {
7714
8160
  this.overlayView.setMap(map);
7715
8161
  }
7716
8162
  }
8163
+ setEdger(edger) {
8164
+ this.hasEdger = edger;
8165
+ if (this.mowerPositionManager) {
8166
+ this.mowerPositionManager.setEdger(edger);
8167
+ }
8168
+ }
7717
8169
  getMap() {
7718
8170
  return this.overlayView ? this.overlayView.getMap() : null;
7719
8171
  }
@@ -7741,7 +8193,6 @@ class MowerMapOverlay {
7741
8193
  this.svgMapView?.renderLayer(LAYER_DEFAULT_TYPE.BOUNDARY_BORDER);
7742
8194
  }
7743
8195
  onAdd() {
7744
- console.log('onAdd');
7745
8196
  // 创建包含SVG的div
7746
8197
  this.div = document.createElement('div');
7747
8198
  this.div.style.borderStyle = 'none';
@@ -7809,7 +8260,10 @@ class MowerMapOverlay {
7809
8260
  if (!this.div || !this.svgMapView)
7810
8261
  return;
7811
8262
  // 创建边界标签管理器
7812
- this.boundaryLabelsManager = new BoundaryLabelsManager(this.svgMapView, this.boundaryData);
8263
+ this.boundaryLabelsManager = new BoundaryLabelsManager(this.svgMapView, this.boundaryData, {
8264
+ unitType: this.unitType,
8265
+ language: this.language,
8266
+ });
7813
8267
  // 设置叠加层div引用
7814
8268
  this.boundaryLabelsManager.setOverlayDiv(this.div);
7815
8269
  // 添加所有边界标签
@@ -7853,11 +8307,10 @@ class MowerMapOverlay {
7853
8307
  if (!this.div || !this.svgMapView)
7854
8308
  return;
7855
8309
  // 创建割草机位置管理器,传入动画完成回调
7856
- this.mowerPositionManager = new MowerPositionManager(this.svgMapView, this.mowerPositionConfig, this.modelType, this.div, () => {
7857
- console.log('动画完成');
7858
- }, this.updatePathDataByMowingPositionThrottled.bind(this));
8310
+ this.mowerPositionManager = new MowerPositionManager(this.svgMapView, this.mowerPositionConfig, this.modelType, this.div, () => { }, this.updatePathDataByMowingPositionThrottled.bind(this));
7859
8311
  // 设置叠加层div引用
7860
8312
  this.mowerPositionManager.setOverlayDiv(this.div);
8313
+ this.mowerPositionManager.setEdger(this.hasEdger);
7861
8314
  // 获取容器并添加到主div
7862
8315
  const container = this.mowerPositionManager.getElement();
7863
8316
  if (container) {
@@ -7937,7 +8390,7 @@ class MowerMapOverlay {
7937
8390
  this.rotateHandle.style.pointerEvents = 'auto';
7938
8391
  this.rotateHandle.innerHTML = DEFAULT_ROTATE_ICON;
7939
8392
  this.editContainer.appendChild(this.rotateHandle);
7940
- // 创建拖拽手柄(左上角)
8393
+ // 创建拖拽手柄(左下角)- 仅在移动设备上显示
7941
8394
  this.dragHandle = document.createElement('div');
7942
8395
  this.dragHandle.style.position = 'absolute';
7943
8396
  this.dragHandle.style.bottom = '-20px';
@@ -7948,6 +8401,10 @@ class MowerMapOverlay {
7948
8401
  this.dragHandle.style.zIndex = EDIT_STYLES.Z_INDEX.HANDLE;
7949
8402
  this.dragHandle.style.pointerEvents = 'auto';
7950
8403
  this.dragHandle.innerHTML = DEFAULT_DRAG_ICON;
8404
+ // 在PC设备上隐藏拖拽手柄
8405
+ if (!isMobileDevice()) {
8406
+ this.dragHandle.style.display = 'none';
8407
+ }
7951
8408
  this.editContainer.appendChild(this.dragHandle);
7952
8409
  // 将编辑容器添加到主div
7953
8410
  this.div.appendChild(this.editContainer);
@@ -7989,7 +8446,6 @@ class MowerMapOverlay {
7989
8446
  this.boundaryLabelsManager.collapseAllLabels();
7990
8447
  }
7991
8448
  this.dragCallbacks?.onDragStart?.(this.getCurrentDragState());
7992
- console.log('开始旋转操作');
7993
8449
  });
7994
8450
  // 旋转手柄的触摸事件
7995
8451
  this.rotateHandle.addEventListener('touchstart', (e) => {
@@ -8005,39 +8461,41 @@ class MowerMapOverlay {
8005
8461
  this.boundaryLabelsManager.collapseAllLabels();
8006
8462
  }
8007
8463
  this.dragCallbacks?.onDragStart?.(this.getCurrentDragState());
8008
- console.log('开始旋转操作(触摸)');
8009
- }, { passive: false });
8010
- // 拖拽手柄的鼠标事件
8011
- this.dragHandle.addEventListener('mousedown', (e) => {
8012
- e.preventDefault();
8013
- e.stopPropagation();
8014
- e.stopImmediatePropagation();
8015
- this.isDragging = true;
8016
- this.startPos = { x: e.clientX, y: e.clientY };
8017
- this.dragHandle.style.cursor = 'grabbing';
8018
- // 开始编辑时关闭所有展开的边界标签
8019
- if (this.boundaryLabelsManager) {
8020
- this.boundaryLabelsManager.collapseAllLabels();
8021
- }
8022
- this.dragCallbacks?.onDragStart?.(this.getCurrentDragState());
8023
- console.log('开始拖动操作(通过手柄)');
8024
- });
8025
- // 拖拽手柄的触摸事件
8026
- this.dragHandle.addEventListener('touchstart', (e) => {
8027
- e.preventDefault();
8028
- e.stopPropagation();
8029
- e.stopImmediatePropagation();
8030
- this.isDragging = true;
8031
- const touch = e.touches[0];
8032
- this.startPos = { x: touch.clientX, y: touch.clientY };
8033
- this.dragHandle.style.cursor = 'grabbing';
8034
- this.dragCallbacks?.onDragStart?.(this.getCurrentDragState());
8035
- console.log('开始拖动操作(通过手柄,触摸)');
8036
8464
  }, { passive: false });
8465
+ // 拖拽手柄的鼠标事件 - 仅在移动设备上启用
8466
+ if (isMobileDevice()) {
8467
+ this.dragHandle.addEventListener('mousedown', (e) => {
8468
+ e.preventDefault();
8469
+ e.stopPropagation();
8470
+ e.stopImmediatePropagation();
8471
+ this.isDragging = true;
8472
+ this.startPos = { x: e.clientX, y: e.clientY };
8473
+ this.dragHandle.style.cursor = 'grabbing';
8474
+ // 开始编辑时关闭所有展开的边界标签
8475
+ if (this.boundaryLabelsManager) {
8476
+ this.boundaryLabelsManager.collapseAllLabels();
8477
+ }
8478
+ this.dragCallbacks?.onDragStart?.(this.getCurrentDragState());
8479
+ });
8480
+ // 拖拽手柄的触摸事件
8481
+ this.dragHandle.addEventListener('touchstart', (e) => {
8482
+ e.preventDefault();
8483
+ e.stopPropagation();
8484
+ e.stopImmediatePropagation();
8485
+ this.isDragging = true;
8486
+ const touch = e.touches[0];
8487
+ this.startPos = { x: touch.clientX, y: touch.clientY };
8488
+ this.dragHandle.style.cursor = 'grabbing';
8489
+ this.dragCallbacks?.onDragStart?.(this.getCurrentDragState());
8490
+ }, { passive: false });
8491
+ }
8037
8492
  // 编辑容器的鼠标事件(整个区域拖拽)
8038
8493
  this.editContainer.addEventListener('mousedown', (e) => {
8039
- console.log('开始拖动操作(整个叠加层)');
8040
- if (e.target === this.dragHandle || e.target === this.rotateHandle) {
8494
+ // 在移动设备上,检查是否点击了拖拽手柄或旋转手柄
8495
+ // 在PC设备上,只检查旋转手柄(拖拽手柄已隐藏)
8496
+ const isDragHandleClick = isMobileDevice() && e.target === this.dragHandle;
8497
+ const isRotateHandleClick = e.target === this.rotateHandle;
8498
+ if (isDragHandleClick || isRotateHandleClick) {
8041
8499
  return;
8042
8500
  }
8043
8501
  e.preventDefault();
@@ -8054,8 +8512,11 @@ class MowerMapOverlay {
8054
8512
  });
8055
8513
  // 编辑容器的触摸事件(整个区域拖拽)
8056
8514
  this.editContainer.addEventListener('touchstart', (e) => {
8057
- console.log('开始拖动操作(整个叠加层,触摸)');
8058
- if (e.target === this.dragHandle || e.target === this.rotateHandle) {
8515
+ // 在移动设备上,检查是否点击了拖拽手柄或旋转手柄
8516
+ // 在PC设备上,只检查旋转手柄(拖拽手柄已隐藏)
8517
+ const isDragHandleClick = isMobileDevice() && e.target === this.dragHandle;
8518
+ const isRotateHandleClick = e.target === this.rotateHandle;
8519
+ if (isDragHandleClick || isRotateHandleClick) {
8059
8520
  return;
8060
8521
  }
8061
8522
  e.preventDefault();
@@ -8115,7 +8576,6 @@ class MowerMapOverlay {
8115
8576
  e.preventDefault();
8116
8577
  e.stopPropagation();
8117
8578
  e.stopImmediatePropagation();
8118
- console.log('结束编辑操作');
8119
8579
  }
8120
8580
  // 如果是拖拽结束,将像素偏移量转换为地理坐标偏移量
8121
8581
  if (this.isDragging) {
@@ -8137,7 +8597,6 @@ class MowerMapOverlay {
8137
8597
  e.preventDefault();
8138
8598
  e.stopPropagation();
8139
8599
  e.stopImmediatePropagation();
8140
- console.log('结束编辑操作(触摸)');
8141
8600
  }
8142
8601
  // 如果是拖拽结束,将像素偏移量转换为地理坐标偏移量
8143
8602
  if (this.isDragging) {
@@ -8249,7 +8708,6 @@ class MowerMapOverlay {
8249
8708
  this.div.style.transform = transform;
8250
8709
  // 更新鼠标起始位置为当前位置,为下次计算做准备
8251
8710
  this.startPos = { x: mouseCurrentX, y: mouseCurrentY };
8252
- console.log('旋转角度:', this.currentRotation, '角度增量:', angleDifferenceDegrees);
8253
8711
  }
8254
8712
  // 将像素偏移量转换为地理坐标偏移量
8255
8713
  convertPixelOffsetToLatLng() {
@@ -8279,14 +8737,6 @@ class MowerMapOverlay {
8279
8737
  // 累积更新地理坐标偏移量(不是直接赋值!)
8280
8738
  this.latLngOffset.lat += latOffset;
8281
8739
  this.latLngOffset.lng += lngOffset;
8282
- console.log('精确转换偏移量:', {
8283
- pixelOffset: this.tempPixelOffset,
8284
- centerLatLng: { lat: centerLatLng.lat(), lng: centerLatLng.lng() },
8285
- offsetLatLng: { lat: offsetLatLng.lat(), lng: offsetLatLng.lng() },
8286
- latOffset,
8287
- lngOffset,
8288
- newLatLngOffset: this.latLngOffset,
8289
- });
8290
8740
  // 重置临时像素偏移量
8291
8741
  this.tempPixelOffset = { x: 0, y: 0 };
8292
8742
  this.draw();
@@ -8324,8 +8774,6 @@ class MowerMapOverlay {
8324
8774
  editData: editData,
8325
8775
  timestamp: new Date().toISOString(),
8326
8776
  };
8327
- // 在这里可以添加保存逻辑,比如发送到服务器
8328
- console.log('保存编辑数据:', saveData);
8329
8777
  // 显示保存成功提示
8330
8778
  this.showSaveSuccess();
8331
8779
  return saveData;
@@ -8418,6 +8866,9 @@ class MowerMapOverlay {
8418
8866
  y: transform.y,
8419
8867
  rotation: transform.rotation,
8420
8868
  };
8869
+ // defaultTransform的x对应经度偏移量,y对应纬度偏移量
8870
+ this.latLngOffset.lng = this.defaultTransform.x;
8871
+ this.latLngOffset.lat = this.defaultTransform.y;
8421
8872
  this.setManagerRotation(this.currentRotation);
8422
8873
  this.draw();
8423
8874
  }
@@ -8457,7 +8908,6 @@ class MowerMapOverlay {
8457
8908
  if (this.pathData && this.svgMapView) {
8458
8909
  this.loadPathData(this.pathData, this.mowPartitionData);
8459
8910
  }
8460
- console.log('initializeSvgMapView');
8461
8911
  // 刷新绘制图层
8462
8912
  this.svgMapView.refresh();
8463
8913
  // 获取生成的SVG并添加到叠加层div中
@@ -8613,7 +9063,6 @@ class MowerMapOverlay {
8613
9063
  this.boundaryLabelsManager?.updateBoundaryData(boundaryData);
8614
9064
  }
8615
9065
  draw() {
8616
- console.log('ondraw');
8617
9066
  // 防御性检查:如果this.div为null,说明onAdd还没被调用,直接返回
8618
9067
  if (!this.div) {
8619
9068
  return;
@@ -8881,7 +9330,7 @@ const getValidGpsBounds = (mapData, rotation = 0) => {
8881
9330
  // 默认配置
8882
9331
  const defaultMapConfig = DEFAULT_STYLES;
8883
9332
  // 地图渲染器组件
8884
- const MowerMapRenderer = React.forwardRef(({ mapConfig, modelType, mapRef, mapJson, pathJson, realTimeData, antennaConfig, onMapLoad, onPathLoad, onError, className, style, googleMapInstance, isEditMode = false, dragCallbacks, defaultTransform, debug = false, }, ref) => {
9333
+ const MowerMapRenderer = React.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) => {
8885
9334
  const [elementCount, setElementCount] = React.useState(0);
8886
9335
  const [pathCount, setPathCount] = React.useState(0);
8887
9336
  const [currentError, setCurrentError] = React.useState(null);
@@ -8889,7 +9338,7 @@ const MowerMapRenderer = React.forwardRef(({ mapConfig, modelType, mapRef, mapJs
8889
9338
  // const mapRef = useMap();
8890
9339
  const [isGoogleMapsReady, setIsGoogleMapsReady] = React.useState(false);
8891
9340
  const [hasInitializedBounds, setHasInitializedBounds] = React.useState(false);
8892
- const { clearSubBoundaryBorder, clearObstacles } = useSubBoundaryBorderStore();
9341
+ const { clearSubBoundaryBorder, clearObstacles, clearSvgElements } = useSubBoundaryBorderStore();
8893
9342
  const currentProcessMowingStatusRef = React.useRef(false);
8894
9343
  const { updateProcessStateIsMowing, processStateIsMowing } = useProcessMowingState();
8895
9344
  const [mowPartitionData, setMowPartitionData] = React.useState(null);
@@ -8923,7 +9372,7 @@ const MowerMapRenderer = React.forwardRef(({ mapConfig, modelType, mapRef, mapJs
8923
9372
  postureX: 0,
8924
9373
  postureY: 0,
8925
9374
  postureTheta: 0,
8926
- vehicleState: RobotStatus.PARKED,
9375
+ vehicleState: RobotStatus.DISCONNECTED,
8927
9376
  };
8928
9377
  let currentPositionData;
8929
9378
  if (realTimeData.length === 1 && realTimeData[0].type === RealTimeDataType.LOCATION) {
@@ -8949,10 +9398,9 @@ const MowerMapRenderer = React.forwardRef(({ mapConfig, modelType, mapRef, mapJs
8949
9398
  lastPostureY: currentPositionData?.lastPostureY
8950
9399
  ? Number(currentPositionData.lastPostureY)
8951
9400
  : 0,
8952
- vehicleState: currentPositionData?.vehicleState || RobotStatus.CHARGING,
9401
+ vehicleState: currentPositionData?.vehicleState || RobotStatus.DISCONNECTED,
8953
9402
  };
8954
9403
  }, [realTimeData, modelType]);
8955
- console.log('mowerPositionData', mowerPositionData);
8956
9404
  // 处理错误
8957
9405
  const handleError = (error) => {
8958
9406
  setCurrentError(error);
@@ -8977,7 +9425,6 @@ const MowerMapRenderer = React.forwardRef(({ mapConfig, modelType, mapRef, mapJs
8977
9425
  const googleBounds = new window.google.maps.LatLngBounds(new window.google.maps.LatLng(swLat, swLng), // 西南角
8978
9426
  new window.google.maps.LatLng(neLat, neLng) // 东北角
8979
9427
  );
8980
- console.log('fitBounds----->', googleBounds);
8981
9428
  mapRef.fitBounds(googleBounds);
8982
9429
  }, [mapJson, mapRef, defaultTransform]);
8983
9430
  // 初始化Google Maps叠加层
@@ -9018,9 +9465,8 @@ const MowerMapRenderer = React.forwardRef(({ mapConfig, modelType, mapRef, mapJs
9018
9465
  overlayRef.current.setMap(null);
9019
9466
  overlayRef.current = null;
9020
9467
  }
9021
- console.log('initializeGoogleMapsOverlay', mowPartitionData);
9022
9468
  // 创建叠加层
9023
- const overlay = new MowerMapOverlay(googleBounds, mapJson, partitionBoundary, mowerPositionData, modelType, pathJson || {}, isEditMode, mergedMapConfig, mergedAntennaConfig, null, defaultTransform, (count) => {
9469
+ const overlay = new MowerMapOverlay(googleBounds, mapJson, partitionBoundary, mowerPositionData, modelType, pathJson || {}, isEditMode, unitType, language, mergedMapConfig, mergedAntennaConfig, null, defaultTransform, (count) => {
9024
9470
  setElementCount(count);
9025
9471
  onMapLoad?.(count);
9026
9472
  }, (count) => {
@@ -9030,6 +9476,7 @@ const MowerMapRenderer = React.forwardRef(({ mapConfig, modelType, mapRef, mapJs
9030
9476
  // 设置地图
9031
9477
  overlay.setMap(mapInstance);
9032
9478
  overlayRef.current = overlay;
9479
+ overlay.setEdger(edger);
9033
9480
  // 只在首次初始化时自适应视图
9034
9481
  if (!hasInitializedBounds) {
9035
9482
  mapInstance.fitBounds(googleBounds);
@@ -9053,7 +9500,7 @@ const MowerMapRenderer = React.forwardRef(({ mapConfig, modelType, mapRef, mapJs
9053
9500
  postureY: chargingPiles?.originalData.position[1],
9054
9501
  postureTheta: chargingPiles?.originalData.direction - Math.PI || 0,
9055
9502
  }, 0);
9056
- }, [mapJson]);
9503
+ }, [mapJson, mowerPositionData]);
9057
9504
  // 初始化效果
9058
9505
  React.useEffect(() => {
9059
9506
  initializeGoogleMapsOverlay();
@@ -9061,6 +9508,7 @@ const MowerMapRenderer = React.forwardRef(({ mapConfig, modelType, mapRef, mapJs
9061
9508
  return () => {
9062
9509
  clearSubBoundaryBorder();
9063
9510
  clearObstacles();
9511
+ clearSvgElements();
9064
9512
  updateProcessStateIsMowing(false);
9065
9513
  currentProcessMowingStatusRef.current = false;
9066
9514
  if (overlayRef.current) {
@@ -9092,7 +9540,6 @@ const MowerMapRenderer = React.forwardRef(({ mapConfig, modelType, mapRef, mapJs
9092
9540
  const isOffLine = mowerPositionData.vehicleState === RobotStatus.DISCONNECTED;
9093
9541
  const isInChargingPile = inChargingPiles.includes(mowerPositionData.vehicleState);
9094
9542
  // 如果在充电桩上,则直接更新位置到充电桩的位置
9095
- console.log('usefeect mowerPositionData----->', mowerPositionData);
9096
9543
  if (isInChargingPile) {
9097
9544
  overlayRef.current.updatePosition({
9098
9545
  ...mowerPositionData,
@@ -9125,7 +9572,6 @@ const MowerMapRenderer = React.forwardRef(({ mapConfig, modelType, mapRef, mapJs
9125
9572
  }
9126
9573
  }
9127
9574
  else {
9128
- console.log('hook updatePosition----->', mowerPositionData);
9129
9575
  overlayRef.current.updatePosition(mowerPositionData, isStandby ? 0 : 2000);
9130
9576
  }
9131
9577
  }
@@ -9244,7 +9690,6 @@ const MowerMapRenderer = React.forwardRef(({ mapConfig, modelType, mapRef, mapJs
9244
9690
  if (!realTimeData || realTimeData.length === 0 || !Array.isArray(realTimeData)) {
9245
9691
  return;
9246
9692
  }
9247
- console.log('usefeect realTimeData----->', realTimeData, mapJson, pathJson, overlayRef.current);
9248
9693
  let curMowPartitionData = mowPartitionData;
9249
9694
  // realtime中包含当前割草任务的数据,根据数据进行path路径和边界的高亮操作,
9250
9695
  const mowingPartition = realTimeData.find((item) => item.type === RealTimeDataType.PARTITION);
@@ -9252,9 +9697,8 @@ const MowerMapRenderer = React.forwardRef(({ mapConfig, modelType, mapRef, mapJs
9252
9697
  setMowPartitionData(mowingPartition);
9253
9698
  curMowPartitionData = mowingPartition;
9254
9699
  }
9255
- const positionData = realTimeData?.find(item => item?.type === RealTimeDataType.LOCATION);
9256
- const statusData = realTimeData?.find(item => item?.type === RealTimeDataType.STATUS);
9257
- console.log('current->1', positionData, statusData);
9700
+ const positionData = realTimeData?.find((item) => item?.type === RealTimeDataType.LOCATION);
9701
+ const statusData = realTimeData?.find((item) => item?.type === RealTimeDataType.STATUS);
9258
9702
  if (statusData || positionData) {
9259
9703
  const currentStatus = statusData?.vehicleState || positionData?.vehicleState;
9260
9704
  // 车辆回桩不会回传最后的park的位置,所以根据实时数据的状态数据判断车辆回到桩上
@@ -9264,32 +9708,28 @@ const MowerMapRenderer = React.forwardRef(({ mapConfig, modelType, mapRef, mapJs
9264
9708
  else if (currentStatus === RobotStatus.WORKING) {
9265
9709
  // 兜底收不到割草地块的实时数据,使用状态来兜底
9266
9710
  overlayRef.current.resetBorderLayerHighlight();
9267
- setMowPartitionData(null);
9268
- curMowPartitionData = null;
9711
+ setMowPartitionData({});
9712
+ curMowPartitionData = {};
9269
9713
  }
9270
- else if (currentStatus === RobotStatus.MOWING && (curMowPartitionData && !curMowPartitionData?.partitionIds)) {
9714
+ else if (currentStatus === RobotStatus.MOWING &&
9715
+ curMowPartitionData &&
9716
+ !curMowPartitionData?.partitionIds) {
9271
9717
  // 如果当前是割草状态,但是地块数据初始化过且不存在则认为是全局割草,则把所有地块都高亮
9272
- const allPartitionIds = mapJson?.sub_maps?.map(item => item?.id);
9273
- console.log('allPartitionIds->', allPartitionIds, mapJson);
9718
+ const allPartitionIds = mapJson?.sub_maps?.map((item) => item?.id);
9274
9719
  setMowPartitionData({
9275
- partitionIds: allPartitionIds
9720
+ partitionIds: allPartitionIds,
9276
9721
  });
9277
9722
  curMowPartitionData = {
9278
- partitionIds: allPartitionIds
9723
+ partitionIds: allPartitionIds,
9279
9724
  };
9280
9725
  }
9281
9726
  }
9282
- if (!mapJson ||
9283
- !pathJson ||
9284
- !overlayRef.current)
9727
+ if (!mapJson || !pathJson || !overlayRef.current)
9285
9728
  return;
9286
9729
  // 根据后端推送的实时数据,进行不同处理
9287
- // TODO:需要根据返回的数据,处理车辆的移动位置
9288
- console.log('realTimeData----->', realTimeData, curMowPartitionData);
9289
9730
  if (curMowPartitionData) {
9290
9731
  const isMowing = curMowPartitionData?.partitionIds && curMowPartitionData.partitionIds.length > 0;
9291
9732
  overlayRef.current.updateMowPartitionData(curMowPartitionData);
9292
- console.log('isMowing', isMowing, curMowPartitionData);
9293
9733
  if (!isMowing) {
9294
9734
  overlayRef.current.resetBorderLayerHighlight();
9295
9735
  }
@@ -9327,7 +9767,6 @@ const MowerMapRenderer = React.forwardRef(({ mapConfig, modelType, mapRef, mapJs
9327
9767
  }
9328
9768
  }, [realTimeData, mapJson, pathJson]);
9329
9769
  React.useEffect(() => {
9330
- console.log('defaultTransform----->', defaultTransform, overlayRef.current, mapJson);
9331
9770
  if (!overlayRef.current || !defaultTransform)
9332
9771
  return;
9333
9772
  overlayRef.current?.setTransform(defaultTransform);
@@ -9342,6 +9781,11 @@ const MowerMapRenderer = React.forwardRef(({ mapConfig, modelType, mapRef, mapJs
9342
9781
  );
9343
9782
  mapRef.fitBounds(googleBounds);
9344
9783
  }, [defaultTransform]);
9784
+ React.useEffect(() => {
9785
+ if (!overlayRef || !overlayRef.current)
9786
+ return;
9787
+ overlayRef.current.setEdger(edger);
9788
+ }, [edger]);
9345
9789
  // 提供ref方法
9346
9790
  React.useImperativeHandle(ref, () => ({
9347
9791
  fitToView: () => {