@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/config/constants.d.ts +1 -0
- package/dist/config/constants.d.ts.map +1 -1
- package/dist/config/styles.d.ts.map +1 -1
- package/dist/index.esm.js +613 -169
- package/dist/index.js +613 -169
- package/dist/render/BoundaryLabelsManager.d.ts +7 -2
- package/dist/render/BoundaryLabelsManager.d.ts.map +1 -1
- package/dist/render/MowerMapOverlay.d.ts +6 -1
- package/dist/render/MowerMapOverlay.d.ts.map +1 -1
- package/dist/render/MowerMapRenderer.d.ts.map +1 -1
- package/dist/render/MowerPositionManager.d.ts +2 -0
- package/dist/render/MowerPositionManager.d.ts.map +1 -1
- package/dist/render/SvgMapView.d.ts.map +1 -1
- package/dist/render/layers/BoundaryBorderLayer.d.ts +4 -0
- package/dist/render/layers/BoundaryBorderLayer.d.ts.map +1 -1
- package/dist/render/layers/PathLayer.d.ts +10 -2
- package/dist/render/layers/PathLayer.d.ts.map +1 -1
- package/dist/types/renderer.d.ts +4 -0
- package/dist/types/renderer.d.ts.map +1 -1
- package/dist/types/utils.d.ts +2 -2
- package/dist/utils/boundaryUtils.d.ts.map +1 -1
- package/dist/utils/common.d.ts +10 -0
- package/dist/utils/common.d.ts.map +1 -1
- package/dist/utils/coordinates.d.ts +27 -0
- package/dist/utils/coordinates.d.ts.map +1 -1
- package/dist/utils/math.d.ts +1 -1
- package/dist/utils/math.d.ts.map +1 -1
- package/dist/utils/mower.d.ts +2 -2
- package/dist/utils/mower.d.ts.map +1 -1
- package/dist/utils/unionFind.d.ts +49 -0
- package/dist/utils/unionFind.d.ts.map +1 -0
- package/package.json +1 -1
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
|
|
1055
|
-
|
|
1056
|
-
|
|
1057
|
-
|
|
1058
|
-
|
|
1059
|
-
|
|
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.
|
|
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
|
-
|
|
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
|
-
|
|
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
|
-
|
|
1224
|
+
return transformedCommands.join(' ');
|
|
1097
1225
|
}
|
|
1098
1226
|
/**
|
|
1099
|
-
*
|
|
1227
|
+
* 生成样式键,用于路径分组
|
|
1100
1228
|
*/
|
|
1101
|
-
|
|
1102
|
-
|
|
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.
|
|
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
|
|
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.
|
|
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"] = "
|
|
4698
|
-
UnitsType["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
|
-
|
|
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
|
-
|
|
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.
|
|
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.
|
|
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,
|
|
6180
|
-
const finishedArea = convertAreaByUnits(boundary.finishedArea || 0,
|
|
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
|
-
|
|
8040
|
-
|
|
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
|
-
|
|
8058
|
-
|
|
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.
|
|
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.
|
|
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(
|
|
9268
|
-
curMowPartitionData =
|
|
9711
|
+
setMowPartitionData({});
|
|
9712
|
+
curMowPartitionData = {};
|
|
9269
9713
|
}
|
|
9270
|
-
else if (currentStatus === RobotStatus.MOWING &&
|
|
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: () => {
|