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