@fleet-frontend/mower-maps 0.0.9-beta.5 → 0.0.9-beta.6
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 -25
- package/dist/config/constants.d.ts.map +1 -1
- package/dist/config/styles.d.ts +1 -1
- package/dist/config/styles.d.ts.map +1 -1
- package/dist/index.d.ts +2 -1
- package/dist/index.d.ts.map +1 -1
- package/dist/index.esm.js +278 -130
- package/dist/index.js +278 -130
- package/dist/processor/MapDataProcessor.d.ts +1 -1
- package/dist/processor/MapDataProcessor.d.ts.map +1 -1
- package/dist/processor/PathDataProcessor.d.ts +2 -21
- package/dist/processor/PathDataProcessor.d.ts.map +1 -1
- package/dist/processor/builder/AntennaDataBuilder.d.ts +2 -1
- package/dist/processor/builder/AntennaDataBuilder.d.ts.map +1 -1
- package/dist/processor/builder/BoundaryDataBuilder.d.ts +1 -1
- package/dist/processor/builder/BoundaryDataBuilder.d.ts.map +1 -1
- package/dist/processor/builder/ChannelDataBuilder.d.ts +1 -1
- package/dist/processor/builder/ChannelDataBuilder.d.ts.map +1 -1
- package/dist/processor/builder/ChargingPileDataBuilder.d.ts +1 -1
- package/dist/processor/builder/ChargingPileDataBuilder.d.ts.map +1 -1
- package/dist/processor/builder/ObstacleDataBuilder.d.ts +2 -1
- package/dist/processor/builder/ObstacleDataBuilder.d.ts.map +1 -1
- package/dist/processor/builder/PathDataBuilder.d.ts +1 -1
- package/dist/processor/builder/PathDataBuilder.d.ts.map +1 -1
- package/dist/processor/builder/PointDataBuilder.d.ts +2 -1
- package/dist/processor/builder/PointDataBuilder.d.ts.map +1 -1
- package/dist/processor/builder/SvgElementDataBuilder.d.ts +1 -1
- package/dist/processor/builder/SvgElementDataBuilder.d.ts.map +1 -1
- package/dist/processor/builder/VisionOffDataBuilder.d.ts +2 -1
- package/dist/processor/builder/VisionOffDataBuilder.d.ts.map +1 -1
- package/dist/processor/index.d.ts +2 -1
- package/dist/processor/index.d.ts.map +1 -1
- package/dist/render/BoundaryLabelsManager.d.ts +1 -1
- package/dist/render/BoundaryLabelsManager.d.ts.map +1 -1
- package/dist/render/MowerMapOverlay.d.ts +2 -2
- package/dist/render/MowerMapOverlay.d.ts.map +1 -1
- package/dist/render/MowerMapRenderer.d.ts.map +1 -1
- 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 +2 -2
- package/dist/render/layers/BoundaryBorderLayer.d.ts.map +1 -1
- package/dist/render/layers/ChannelLayer.d.ts.map +1 -1
- package/dist/render/layers/DrawLayer.d.ts +1 -1
- package/dist/render/layers/DrawLayer.d.ts.map +1 -1
- package/dist/render/layers/PathLayer.d.ts.map +1 -1
- package/dist/render/layers/SvgElementLayer.d.ts.map +1 -1
- package/dist/render/layers/types.d.ts +1 -38
- package/dist/render/layers/types.d.ts.map +1 -1
- package/dist/store/processMowingState.d.ts +1 -5
- package/dist/store/processMowingState.d.ts.map +1 -1
- package/dist/store/useSubBoundaryBorderStore.d.ts +9 -5
- package/dist/store/useSubBoundaryBorderStore.d.ts.map +1 -1
- package/dist/types/constants.d.ts +38 -0
- package/dist/types/constants.d.ts.map +1 -0
- package/dist/types/index.d.ts +6 -0
- package/dist/types/index.d.ts.map +1 -1
- package/dist/types/layers.d.ts +50 -0
- package/dist/types/layers.d.ts.map +1 -0
- package/dist/types/processor.d.ts +26 -0
- package/dist/types/processor.d.ts.map +1 -0
- package/dist/types/realTime.d.ts +8 -1
- package/dist/types/realTime.d.ts.map +1 -1
- package/dist/types/renderer.d.ts +3 -3
- package/dist/types/renderer.d.ts.map +1 -1
- package/dist/types/store.d.ts +22 -0
- package/dist/types/store.d.ts.map +1 -0
- package/dist/types/utils.d.ts +102 -0
- package/dist/types/utils.d.ts.map +1 -0
- package/dist/utils/boundaryUtils.d.ts +1 -29
- package/dist/utils/boundaryUtils.d.ts.map +1 -1
- package/dist/utils/common.d.ts +1 -5
- package/dist/utils/common.d.ts.map +1 -1
- package/dist/utils/coordinates.d.ts +1 -7
- package/dist/utils/coordinates.d.ts.map +1 -1
- package/dist/utils/handleRealTime.d.ts +2 -7
- package/dist/utils/handleRealTime.d.ts.map +1 -1
- package/dist/utils/mapBounds.d.ts +1 -10
- package/dist/utils/mapBounds.d.ts.map +1 -1
- package/dist/utils/math.d.ts +2 -9
- package/dist/utils/math.d.ts.map +1 -1
- package/dist/utils/mower.d.ts +4 -4
- package/dist/utils/mower.d.ts.map +1 -1
- package/dist/utils/pathSegments.d.ts +1 -23
- package/dist/utils/pathSegments.d.ts.map +1 -1
- package/package.json +1 -1
package/dist/index.esm.js
CHANGED
|
@@ -118,7 +118,7 @@ class SvgMapView {
|
|
|
118
118
|
* 设置自适应视图变换 - 让SVG刚好包裹住图形
|
|
119
119
|
*/
|
|
120
120
|
fitToView(bounds) {
|
|
121
|
-
const padding =
|
|
121
|
+
const padding = 20; // 添加一些边距以避免内容贴边
|
|
122
122
|
const boundWidth = bounds.maxX - bounds.minX;
|
|
123
123
|
const boundHeight = bounds.maxY - bounds.minY;
|
|
124
124
|
// 防止宽高为0的情况
|
|
@@ -141,9 +141,10 @@ class SvgMapView {
|
|
|
141
141
|
this.viewBox = {
|
|
142
142
|
x: bounds.minX - padding,
|
|
143
143
|
y: bounds.minY - padding,
|
|
144
|
-
width: boundWidth + padding,
|
|
145
|
-
height: boundHeight + padding,
|
|
144
|
+
width: boundWidth + padding * 2,
|
|
145
|
+
height: boundHeight + padding * 2,
|
|
146
146
|
};
|
|
147
|
+
console.log('viewbox->', this.viewBox);
|
|
147
148
|
// 根据宽高比选择合适的preserveAspectRatio设置
|
|
148
149
|
if (Math.abs(contentAspectRatio - containerAspectRatio) < 0.01) {
|
|
149
150
|
// 宽高比接近,使用slice填满容器
|
|
@@ -649,8 +650,6 @@ const create = (createState) => createState ? createImpl(createState) : createIm
|
|
|
649
650
|
|
|
650
651
|
const useSubBoundaryBorderStore = create((set, get) => ({
|
|
651
652
|
subBoundaryBorder: {},
|
|
652
|
-
// 覆盖所有数据
|
|
653
|
-
setSubBoundaryBorder: (subBoundaryBorder) => set({ subBoundaryBorder }),
|
|
654
653
|
// 追加单个数据
|
|
655
654
|
addSubBoundaryBorder: (key, element) => set((state) => ({
|
|
656
655
|
subBoundaryBorder: {
|
|
@@ -658,17 +657,69 @@ const useSubBoundaryBorderStore = create((set, get) => ({
|
|
|
658
657
|
[key]: element,
|
|
659
658
|
},
|
|
660
659
|
})),
|
|
661
|
-
// 追加多个数据
|
|
662
|
-
addMultipleSubBoundaryBorders: (borders) => set((state) => ({
|
|
663
|
-
subBoundaryBorder: {
|
|
664
|
-
...state.subBoundaryBorder,
|
|
665
|
-
...borders,
|
|
666
|
-
},
|
|
667
|
-
})),
|
|
668
660
|
// 清空所有数据
|
|
669
661
|
clearSubBoundaryBorder: () => set({ subBoundaryBorder: {} }),
|
|
662
|
+
// 障碍物
|
|
663
|
+
obstacles: {},
|
|
664
|
+
addObstacles: (key, element) => set((state) => ({
|
|
665
|
+
obstacles: {
|
|
666
|
+
...state.obstacles,
|
|
667
|
+
[key]: element,
|
|
668
|
+
},
|
|
669
|
+
})),
|
|
670
|
+
clearObstacles: () => set({ obstacles: {} }),
|
|
671
|
+
// svg数据
|
|
672
|
+
svgElements: {},
|
|
673
|
+
addSvgElements: (key, element) => set((state) => ({
|
|
674
|
+
svgElements: {
|
|
675
|
+
...state.svgElements,
|
|
676
|
+
[key]: element,
|
|
677
|
+
},
|
|
678
|
+
})),
|
|
679
|
+
clearSvgElements: () => set({ svgElements: {} }),
|
|
670
680
|
}));
|
|
671
681
|
|
|
682
|
+
/**
|
|
683
|
+
* 常量和枚举类型定义
|
|
684
|
+
*/
|
|
685
|
+
/**
|
|
686
|
+
* 机器人状态枚举
|
|
687
|
+
*/
|
|
688
|
+
var RobotStatus;
|
|
689
|
+
(function (RobotStatus) {
|
|
690
|
+
RobotStatus[RobotStatus["PARKED"] = 1] = "PARKED";
|
|
691
|
+
RobotStatus[RobotStatus["CHARGING"] = 2] = "CHARGING";
|
|
692
|
+
RobotStatus[RobotStatus["STANDBY"] = 3] = "STANDBY";
|
|
693
|
+
RobotStatus[RobotStatus["MOWING"] = 4] = "MOWING";
|
|
694
|
+
RobotStatus[RobotStatus["WORKING"] = 5] = "WORKING";
|
|
695
|
+
RobotStatus[RobotStatus["MAPPING"] = 6] = "MAPPING";
|
|
696
|
+
RobotStatus[RobotStatus["ERROR"] = 7] = "ERROR";
|
|
697
|
+
RobotStatus[RobotStatus["UPGRADING"] = 8] = "UPGRADING";
|
|
698
|
+
RobotStatus[RobotStatus["DISCONNECTED"] = 9] = "DISCONNECTED";
|
|
699
|
+
RobotStatus[RobotStatus["UNKNOWN"] = -1] = "UNKNOWN";
|
|
700
|
+
RobotStatus[RobotStatus["TASK_DELAY"] = 10] = "TASK_DELAY";
|
|
701
|
+
})(RobotStatus || (RobotStatus = {}));
|
|
702
|
+
/**
|
|
703
|
+
* RTK状态枚举
|
|
704
|
+
*/
|
|
705
|
+
var RTK_STATE;
|
|
706
|
+
(function (RTK_STATE) {
|
|
707
|
+
RTK_STATE[RTK_STATE["LOW_RTK"] = 1] = "LOW_RTK";
|
|
708
|
+
RTK_STATE[RTK_STATE["MIDDLE_RTK"] = 2] = "MIDDLE_RTK";
|
|
709
|
+
RTK_STATE[RTK_STATE["HIGH_RTK"] = 3] = "HIGH_RTK";
|
|
710
|
+
RTK_STATE[RTK_STATE["NO_POSTURE"] = 10] = "NO_POSTURE";
|
|
711
|
+
RTK_STATE[RTK_STATE["OUT_OF_RANGE"] = 11] = "OUT_OF_RANGE";
|
|
712
|
+
RTK_STATE[RTK_STATE["OFF_LINE"] = 19] = "OFF_LINE";
|
|
713
|
+
})(RTK_STATE || (RTK_STATE = {}));
|
|
714
|
+
/**
|
|
715
|
+
* 实时数据类型枚举
|
|
716
|
+
*/
|
|
717
|
+
var REAL_TIME_DATA_TYPE;
|
|
718
|
+
(function (REAL_TIME_DATA_TYPE) {
|
|
719
|
+
REAL_TIME_DATA_TYPE[REAL_TIME_DATA_TYPE["LOCATION"] = 1] = "LOCATION";
|
|
720
|
+
REAL_TIME_DATA_TYPE[REAL_TIME_DATA_TYPE["PROCESS"] = 2] = "PROCESS";
|
|
721
|
+
})(REAL_TIME_DATA_TYPE || (REAL_TIME_DATA_TYPE = {}));
|
|
722
|
+
|
|
672
723
|
/**
|
|
673
724
|
* 地图渲染相关常量配置
|
|
674
725
|
*/
|
|
@@ -721,36 +772,6 @@ const LAYER_DEFAULT_TYPE = {
|
|
|
721
772
|
VISION_OFF_AREA: 'vision_off_area',
|
|
722
773
|
ANTENNA: 'antenna',
|
|
723
774
|
};
|
|
724
|
-
var RobotStatus;
|
|
725
|
-
(function (RobotStatus) {
|
|
726
|
-
RobotStatus[RobotStatus["PARKED"] = 1] = "PARKED";
|
|
727
|
-
RobotStatus[RobotStatus["CHARGING"] = 2] = "CHARGING";
|
|
728
|
-
RobotStatus[RobotStatus["STANDBY"] = 3] = "STANDBY";
|
|
729
|
-
RobotStatus[RobotStatus["MOWING"] = 4] = "MOWING";
|
|
730
|
-
RobotStatus[RobotStatus["WORKING"] = 5] = "WORKING";
|
|
731
|
-
RobotStatus[RobotStatus["MAPPING"] = 6] = "MAPPING";
|
|
732
|
-
RobotStatus[RobotStatus["ERROR"] = 7] = "ERROR";
|
|
733
|
-
RobotStatus[RobotStatus["UPGRADING"] = 8] = "UPGRADING";
|
|
734
|
-
RobotStatus[RobotStatus["DISCONNECTED"] = 9] = "DISCONNECTED";
|
|
735
|
-
RobotStatus[RobotStatus["UNKNOWN"] = -1] = "UNKNOWN";
|
|
736
|
-
RobotStatus[RobotStatus["TASK_DELAY"] = 10] = "TASK_DELAY";
|
|
737
|
-
// WAITING = 'Waiting',
|
|
738
|
-
})(RobotStatus || (RobotStatus = {}));
|
|
739
|
-
// RTK状态
|
|
740
|
-
var RTK_STATE;
|
|
741
|
-
(function (RTK_STATE) {
|
|
742
|
-
RTK_STATE[RTK_STATE["LOW_RTK"] = 1] = "LOW_RTK";
|
|
743
|
-
RTK_STATE[RTK_STATE["MIDDLE_RTK"] = 2] = "MIDDLE_RTK";
|
|
744
|
-
RTK_STATE[RTK_STATE["HIGH_RTK"] = 3] = "HIGH_RTK";
|
|
745
|
-
RTK_STATE[RTK_STATE["NO_POSTURE"] = 10] = "NO_POSTURE";
|
|
746
|
-
RTK_STATE[RTK_STATE["OUT_OF_RANGE"] = 11] = "OUT_OF_RANGE";
|
|
747
|
-
RTK_STATE[RTK_STATE["OFF_LINE"] = 19] = "OFF_LINE";
|
|
748
|
-
})(RTK_STATE || (RTK_STATE = {}));
|
|
749
|
-
var REAL_TIME_DATA_TYPE;
|
|
750
|
-
(function (REAL_TIME_DATA_TYPE) {
|
|
751
|
-
REAL_TIME_DATA_TYPE[REAL_TIME_DATA_TYPE["LOCATION"] = 1] = "LOCATION";
|
|
752
|
-
REAL_TIME_DATA_TYPE[REAL_TIME_DATA_TYPE["PROCESS"] = 2] = "PROCESS";
|
|
753
|
-
})(REAL_TIME_DATA_TYPE || (REAL_TIME_DATA_TYPE = {}));
|
|
754
775
|
const ISOLATED_BOUNDARY_SVG = `<svg width="24" height="24" viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg">
|
|
755
776
|
<g opacity="0.6">
|
|
756
777
|
<rect width="24" height="24" rx="12" fill="#1E1E1F" fill-opacity="0.5"/>
|
|
@@ -850,37 +871,66 @@ class ChannelLayer extends BaseLayer {
|
|
|
850
871
|
}
|
|
851
872
|
// 2. 再计算所有通道的边界
|
|
852
873
|
for (const element of this.elements) {
|
|
853
|
-
const tunnelConnection = element.originalData?.connection;
|
|
854
|
-
if (tunnelConnection && Array.isArray(tunnelConnection)) {
|
|
855
|
-
|
|
856
|
-
|
|
857
|
-
|
|
858
|
-
|
|
859
|
-
|
|
860
|
-
|
|
861
|
-
|
|
862
|
-
|
|
863
|
-
|
|
864
|
-
|
|
865
|
-
|
|
866
|
-
|
|
867
|
-
|
|
868
|
-
|
|
869
|
-
|
|
870
|
-
|
|
871
|
-
|
|
872
|
-
|
|
873
|
-
|
|
874
|
+
// const tunnelConnection = element.originalData?.connection;
|
|
875
|
+
// if (tunnelConnection && Array.isArray(tunnelConnection)) {
|
|
876
|
+
// const clipPathId = `channel-exclude-${
|
|
877
|
+
// element.originalData?.id || Math.random().toString(36).substr(2, 9)
|
|
878
|
+
// }`;
|
|
879
|
+
// // 检查是否已存在该 clipPath
|
|
880
|
+
// const existingClipPath = defs.querySelector(`#${clipPathId}`);
|
|
881
|
+
// if (existingClipPath) continue;
|
|
882
|
+
// // 创建 clipPath
|
|
883
|
+
// const clipPath = document.createElementNS('http://www.w3.org/2000/svg', 'clipPath');
|
|
884
|
+
// clipPath.setAttribute('id', clipPathId);
|
|
885
|
+
// clipPath.setAttribute('clipPathUnits', 'userSpaceOnUse');
|
|
886
|
+
// // === 合成一个 path ===
|
|
887
|
+
// let d = `M ${minX} ${minY} L ${maxX} ${minY} L ${maxX} ${maxY} L ${minX} ${maxY} Z`;
|
|
888
|
+
// for (const partitionId of tunnelConnection) {
|
|
889
|
+
// const boundaryData = subBoundaryBorder[partitionId];
|
|
890
|
+
// if (boundaryData && boundaryData.coordinates.length >= 3) {
|
|
891
|
+
// d += ` M ${boundaryData.coordinates[0][0]} ${boundaryData.coordinates[0][1]}`;
|
|
892
|
+
// for (let i = 1; i < boundaryData.coordinates.length; i++) {
|
|
893
|
+
// d += ` L ${boundaryData.coordinates[i][0]} ${boundaryData.coordinates[i][1]}`;
|
|
894
|
+
// }
|
|
895
|
+
// d += ' Z';
|
|
896
|
+
// }
|
|
897
|
+
// }
|
|
898
|
+
// const path = document.createElementNS('http://www.w3.org/2000/svg', 'path');
|
|
899
|
+
// path.setAttribute('d', d);
|
|
900
|
+
// path.setAttribute('clip-rule', 'evenodd'); // 关键
|
|
901
|
+
// clipPath.appendChild(path);
|
|
902
|
+
// defs.appendChild(clipPath);
|
|
903
|
+
// clipPathIdsMap[element.originalData?.id.toString()] = clipPathId;
|
|
904
|
+
// } else {
|
|
905
|
+
const clipPathId = `channel-exclude-all-${element.originalData?.id || Math.random().toString(36).substr(2, 9)}`;
|
|
906
|
+
// 检查是否已存在该 clipPath
|
|
907
|
+
const existingClipPath = defs.querySelector(`#${clipPathId}`);
|
|
908
|
+
if (existingClipPath)
|
|
909
|
+
continue;
|
|
910
|
+
// 创建 clipPath
|
|
911
|
+
const clipPath = document.createElementNS('http://www.w3.org/2000/svg', 'clipPath');
|
|
912
|
+
clipPath.setAttribute('id', clipPathId);
|
|
913
|
+
clipPath.setAttribute('clipPathUnits', 'userSpaceOnUse');
|
|
914
|
+
// === 合成一个 path ===
|
|
915
|
+
let d = `M ${minX} ${minY} L ${maxX} ${minY} L ${maxX} ${maxY} L ${minX} ${maxY} Z`;
|
|
916
|
+
for (const partitionId in subBoundaryBorder) {
|
|
917
|
+
const boundaryData = subBoundaryBorder[partitionId];
|
|
918
|
+
if (boundaryData && boundaryData.coordinates.length >= 3) {
|
|
919
|
+
d += ` M ${boundaryData.coordinates[0][0]} ${boundaryData.coordinates[0][1]}`;
|
|
920
|
+
for (let i = 1; i < boundaryData.coordinates.length; i++) {
|
|
921
|
+
d += ` L ${boundaryData.coordinates[i][0]} ${boundaryData.coordinates[i][1]}`;
|
|
874
922
|
}
|
|
923
|
+
d += ' Z';
|
|
875
924
|
}
|
|
876
|
-
const path = document.createElementNS('http://www.w3.org/2000/svg', 'path');
|
|
877
|
-
path.setAttribute('d', d);
|
|
878
|
-
path.setAttribute('clip-rule', 'evenodd'); // 关键
|
|
879
|
-
clipPath.appendChild(path);
|
|
880
|
-
defs.appendChild(clipPath);
|
|
881
|
-
clipPathIdsMap[element.originalData?.id.toString()] = clipPathId;
|
|
882
925
|
}
|
|
926
|
+
const path = document.createElementNS('http://www.w3.org/2000/svg', 'path');
|
|
927
|
+
path.setAttribute('d', d);
|
|
928
|
+
path.setAttribute('clip-rule', 'evenodd'); // 关键
|
|
929
|
+
clipPath.appendChild(path);
|
|
930
|
+
defs.appendChild(clipPath);
|
|
931
|
+
clipPathIdsMap[element.originalData?.id.toString()] = clipPathId;
|
|
883
932
|
}
|
|
933
|
+
// }
|
|
884
934
|
return clipPathIdsMap;
|
|
885
935
|
}
|
|
886
936
|
/**
|
|
@@ -951,7 +1001,7 @@ class PathLayer extends BaseLayer {
|
|
|
951
1001
|
* 创建所有分区并集的 clipPath
|
|
952
1002
|
*/
|
|
953
1003
|
createUnionClipPath(svgGroup) {
|
|
954
|
-
const { subBoundaryBorder } = useSubBoundaryBorderStore.getState();
|
|
1004
|
+
const { subBoundaryBorder, obstacles, svgElements } = useSubBoundaryBorderStore.getState();
|
|
955
1005
|
// 确保 defs 元素存在
|
|
956
1006
|
let defs = svgGroup.querySelector('defs');
|
|
957
1007
|
if (!defs) {
|
|
@@ -965,7 +1015,7 @@ class PathLayer extends BaseLayer {
|
|
|
965
1015
|
defs.removeChild(existing);
|
|
966
1016
|
// 合成所有分区的 path
|
|
967
1017
|
let d = '';
|
|
968
|
-
//
|
|
1018
|
+
// 1. 外圈(主边界,顺时针)
|
|
969
1019
|
Object.values(subBoundaryBorder).forEach((item) => {
|
|
970
1020
|
const bCoords = item.coordinates;
|
|
971
1021
|
if (bCoords.length >= 3) {
|
|
@@ -976,6 +1026,27 @@ class PathLayer extends BaseLayer {
|
|
|
976
1026
|
d += ' Z ';
|
|
977
1027
|
}
|
|
978
1028
|
});
|
|
1029
|
+
// 2. 内圈(禁区,逆时针)
|
|
1030
|
+
Object.values(obstacles).forEach((item) => {
|
|
1031
|
+
const bCoords = item.coordinates;
|
|
1032
|
+
if (bCoords.length >= 3) {
|
|
1033
|
+
d += `M ${bCoords[bCoords.length - 1][0]} ${bCoords[bCoords.length - 1][1]}`;
|
|
1034
|
+
for (let i = bCoords.length - 2; i >= 0; i--) {
|
|
1035
|
+
d += ` L ${bCoords[i][0]} ${bCoords[i][1]}`;
|
|
1036
|
+
}
|
|
1037
|
+
d += ' Z ';
|
|
1038
|
+
}
|
|
1039
|
+
});
|
|
1040
|
+
console.error('svgElements----', svgElements);
|
|
1041
|
+
// 3. svgElements(直接拼接path字符串,建议逆时针)
|
|
1042
|
+
if (Array.isArray(svgElements)) {
|
|
1043
|
+
svgElements.forEach((svgPath) => {
|
|
1044
|
+
const svgPathString = svgPath?.metadata?.svg;
|
|
1045
|
+
if (svgPathString && typeof svgPathString === 'string' && svgPathString.trim()) {
|
|
1046
|
+
d += svgPathString + ' ';
|
|
1047
|
+
}
|
|
1048
|
+
});
|
|
1049
|
+
}
|
|
979
1050
|
const path = document.createElementNS('http://www.w3.org/2000/svg', 'path');
|
|
980
1051
|
path.setAttribute('d', d);
|
|
981
1052
|
const clipPath = document.createElementNS('http://www.w3.org/2000/svg', 'clipPath');
|
|
@@ -1148,7 +1219,7 @@ class ObstacleLayer extends BaseLayer {
|
|
|
1148
1219
|
}
|
|
1149
1220
|
}
|
|
1150
1221
|
|
|
1151
|
-
var chargingPileImage = "";
|
|
1222
|
+
var chargingPileImage = "";
|
|
1152
1223
|
|
|
1153
1224
|
/**
|
|
1154
1225
|
* 充电桩图层
|
|
@@ -1338,7 +1409,7 @@ class SvgElementLayer extends BaseLayer {
|
|
|
1338
1409
|
// 在transformGroup上应用变换:平移到中心,旋转,缩放,然后居中SVG
|
|
1339
1410
|
const transform = [
|
|
1340
1411
|
`translate(${center[0]}, ${center[1]})`,
|
|
1341
|
-
`rotate(${(direction * 180) / Math.PI})`,
|
|
1412
|
+
`rotate(${-(direction * 180) / Math.PI})`,
|
|
1342
1413
|
`scale(${userScale})`,
|
|
1343
1414
|
`translate(${-originalWidth / 2}, ${-originalHeight / 2})`,
|
|
1344
1415
|
].join(' ');
|
|
@@ -1540,15 +1611,6 @@ function convertCoordinate(x, y) {
|
|
|
1540
1611
|
};
|
|
1541
1612
|
}
|
|
1542
1613
|
|
|
1543
|
-
/**
|
|
1544
|
-
* 路径段类型
|
|
1545
|
-
*/
|
|
1546
|
-
var PathSegmentType;
|
|
1547
|
-
(function (PathSegmentType) {
|
|
1548
|
-
PathSegmentType["EDGE"] = "edge";
|
|
1549
|
-
PathSegmentType["MOWING"] = "mowing";
|
|
1550
|
-
PathSegmentType["TRANS"] = "trans";
|
|
1551
|
-
})(PathSegmentType || (PathSegmentType = {}));
|
|
1552
1614
|
/**
|
|
1553
1615
|
* 按Python逻辑创建路径段:根据连续的两点之间的关系确定线段类型
|
|
1554
1616
|
*/
|
|
@@ -1877,6 +1939,14 @@ function generateBoundaryData(mapData, pathData) {
|
|
|
1877
1939
|
return boundaryData;
|
|
1878
1940
|
}
|
|
1879
1941
|
|
|
1942
|
+
var RealTimeDataType;
|
|
1943
|
+
(function (RealTimeDataType) {
|
|
1944
|
+
RealTimeDataType[RealTimeDataType["LOCATION"] = 1] = "LOCATION";
|
|
1945
|
+
RealTimeDataType[RealTimeDataType["PROCESS"] = 2] = "PROCESS";
|
|
1946
|
+
RealTimeDataType[RealTimeDataType["PARTITION"] = 3] = "PARTITION";
|
|
1947
|
+
RealTimeDataType[RealTimeDataType["STATUS"] = 4] = "STATUS";
|
|
1948
|
+
})(RealTimeDataType || (RealTimeDataType = {}));
|
|
1949
|
+
|
|
1880
1950
|
/**
|
|
1881
1951
|
* 射线法判断点是否在多边形内部
|
|
1882
1952
|
* @param x 点的x坐标
|
|
@@ -2046,7 +2116,7 @@ const getProcessMowingDataFromRealTimeData = ({ realTimeData, isMowing, pathData
|
|
|
2046
2116
|
let newMowingStatus = isMowing;
|
|
2047
2117
|
let newPathData = pathData || {};
|
|
2048
2118
|
// 找到返回的第一个实时进度的点
|
|
2049
|
-
const firstProcessData = realTimeData.find((item) => item.type ===
|
|
2119
|
+
const firstProcessData = realTimeData.find((item) => item.type === RealTimeDataType.PROCESS);
|
|
2050
2120
|
if (firstProcessData) {
|
|
2051
2121
|
// console.log('firstProcessData==', firstProcessData);
|
|
2052
2122
|
const { action, subAction, currentMowBoundary, currentMowProgress } = firstProcessData;
|
|
@@ -4594,6 +4664,36 @@ var merge = createAssigner(function(object, source, srcIndex) {
|
|
|
4594
4664
|
*/
|
|
4595
4665
|
var round = createRound('round');
|
|
4596
4666
|
|
|
4667
|
+
/**
|
|
4668
|
+
* 工具模块类型定义
|
|
4669
|
+
*/
|
|
4670
|
+
/**
|
|
4671
|
+
* 路径段类型枚举
|
|
4672
|
+
*/
|
|
4673
|
+
var PathSegmentType;
|
|
4674
|
+
(function (PathSegmentType) {
|
|
4675
|
+
PathSegmentType["EDGE"] = "edge";
|
|
4676
|
+
PathSegmentType["MOWING"] = "mowing";
|
|
4677
|
+
PathSegmentType["TRANS"] = "trans";
|
|
4678
|
+
})(PathSegmentType || (PathSegmentType = {}));
|
|
4679
|
+
/**
|
|
4680
|
+
* 单位类型枚举
|
|
4681
|
+
*/
|
|
4682
|
+
var UnitsType;
|
|
4683
|
+
(function (UnitsType) {
|
|
4684
|
+
UnitsType["Metric"] = "metric";
|
|
4685
|
+
UnitsType["Imperial"] = "imperial";
|
|
4686
|
+
})(UnitsType || (UnitsType = {}));
|
|
4687
|
+
/**
|
|
4688
|
+
* 面积单位类型枚举
|
|
4689
|
+
*/
|
|
4690
|
+
var UnitsAreaType;
|
|
4691
|
+
(function (UnitsAreaType) {
|
|
4692
|
+
UnitsAreaType["SQUARE_METER"] = "m\u00B2";
|
|
4693
|
+
UnitsAreaType["SQUARE_FOOT"] = "ft\u00B2";
|
|
4694
|
+
UnitsAreaType["ACRE"] = "ac";
|
|
4695
|
+
})(UnitsAreaType || (UnitsAreaType = {}));
|
|
4696
|
+
|
|
4597
4697
|
/**
|
|
4598
4698
|
* 默认航向相对于canvas的偏移角度: 航向默认是东
|
|
4599
4699
|
*/
|
|
@@ -4667,17 +4767,6 @@ function formatNumberWithMetricPrefix(value, round = true, decimals = 2) {
|
|
|
4667
4767
|
return `${mathFn(value / 1000000000, decimals)}B`;
|
|
4668
4768
|
}
|
|
4669
4769
|
}
|
|
4670
|
-
var UnitsType;
|
|
4671
|
-
(function (UnitsType) {
|
|
4672
|
-
UnitsType["Metric"] = "metric";
|
|
4673
|
-
UnitsType["Imperial"] = "imperial";
|
|
4674
|
-
})(UnitsType || (UnitsType = {}));
|
|
4675
|
-
var UnitsAreaType;
|
|
4676
|
-
(function (UnitsAreaType) {
|
|
4677
|
-
UnitsAreaType["SQUARE_METER"] = "m\u00B2";
|
|
4678
|
-
UnitsAreaType["SQUARE_FOOT"] = "ft\u00B2";
|
|
4679
|
-
UnitsAreaType["ACRE"] = "ac";
|
|
4680
|
-
})(UnitsAreaType || (UnitsAreaType = {}));
|
|
4681
4770
|
/**
|
|
4682
4771
|
* 转换割草面积的方法
|
|
4683
4772
|
* @param area 面积数值(单位:m²)
|
|
@@ -4873,7 +4962,12 @@ class BoundaryBorderLayer extends BaseLayer {
|
|
|
4873
4962
|
* 设置当前割草任务的边界
|
|
4874
4963
|
*/
|
|
4875
4964
|
setMowingBoundarys(mowingBoundarys) {
|
|
4876
|
-
|
|
4965
|
+
if (!mowingBoundarys) {
|
|
4966
|
+
this.mowingBoundarys = this.elements?.map(item => item?.originalData?.id);
|
|
4967
|
+
}
|
|
4968
|
+
else {
|
|
4969
|
+
this.mowingBoundarys = mowingBoundarys;
|
|
4970
|
+
}
|
|
4877
4971
|
}
|
|
4878
4972
|
/**
|
|
4879
4973
|
* SVG渲染方法
|
|
@@ -5301,6 +5395,14 @@ class BoundaryDataBuilder {
|
|
|
5301
5395
|
* 创建边界元素数据
|
|
5302
5396
|
*/
|
|
5303
5397
|
static create(type, coordinates, style) {
|
|
5398
|
+
const len = coordinates?.length || 0;
|
|
5399
|
+
const firstPoint = coordinates?.[0];
|
|
5400
|
+
const lastPoint = coordinates?.[len - 1];
|
|
5401
|
+
const isClosed = firstPoint?.[0] === lastPoint?.[0] && firstPoint?.[1] === lastPoint?.[1];
|
|
5402
|
+
// 如果地图没有闭合,则手动新增闭合点,避免border最后一部分没有闭合的情况
|
|
5403
|
+
if (!isClosed) {
|
|
5404
|
+
coordinates.push([firstPoint?.[0], firstPoint?.[1], lastPoint?.[2]]);
|
|
5405
|
+
}
|
|
5304
5406
|
return {
|
|
5305
5407
|
type,
|
|
5306
5408
|
coordinates,
|
|
@@ -5703,8 +5805,13 @@ class MapDataProcessor {
|
|
|
5703
5805
|
// 为ObstacleData创建兼容的MapElement接口
|
|
5704
5806
|
const mapElement = element;
|
|
5705
5807
|
const obstacleElement = ObstacleDataBuilder.fromMapElement(mapElement, this.mapConfig.obstacle);
|
|
5706
|
-
if (obstacleElement)
|
|
5808
|
+
if (obstacleElement) {
|
|
5707
5809
|
result.push(obstacleElement);
|
|
5810
|
+
const { addObstacles } = useSubBoundaryBorderStore.getState();
|
|
5811
|
+
addObstacles(`obstacle-${obstacleElement.originalData.id}`, {
|
|
5812
|
+
...obstacleElement,
|
|
5813
|
+
});
|
|
5814
|
+
}
|
|
5708
5815
|
}
|
|
5709
5816
|
catch (error) {
|
|
5710
5817
|
console.warn(`Error processing OBSTACLE element:`, element, error);
|
|
@@ -5765,8 +5872,13 @@ class MapDataProcessor {
|
|
|
5765
5872
|
element.direction !== undefined) {
|
|
5766
5873
|
const mapElement = element;
|
|
5767
5874
|
const svgElement = SvgElementDataBuilder.fromMapElement(mapElement, this.mapConfig.doodle);
|
|
5768
|
-
if (svgElement)
|
|
5875
|
+
if (svgElement) {
|
|
5769
5876
|
result.push(svgElement);
|
|
5877
|
+
const { addSvgElements } = useSubBoundaryBorderStore.getState();
|
|
5878
|
+
addSvgElements(`time-limit-obstacle-${svgElement.originalData.id}`, {
|
|
5879
|
+
...svgElement,
|
|
5880
|
+
});
|
|
5881
|
+
}
|
|
5770
5882
|
}
|
|
5771
5883
|
// 如果有points数据,按传统方式绘制
|
|
5772
5884
|
else if ('points' in element &&
|
|
@@ -5775,8 +5887,13 @@ class MapDataProcessor {
|
|
|
5775
5887
|
element.points.length >= 3) {
|
|
5776
5888
|
const mapElement = element;
|
|
5777
5889
|
const polygonElement = ObstacleDataBuilder.createTimeLimitObstacle(mapElement, this.mapConfig.obstacle);
|
|
5778
|
-
if (polygonElement)
|
|
5890
|
+
if (polygonElement) {
|
|
5779
5891
|
result.push(polygonElement);
|
|
5892
|
+
const { addObstacles } = useSubBoundaryBorderStore.getState();
|
|
5893
|
+
addObstacles(`time-limit-obstacle-${polygonElement.originalData.id}`, {
|
|
5894
|
+
...polygonElement,
|
|
5895
|
+
});
|
|
5896
|
+
}
|
|
5780
5897
|
}
|
|
5781
5898
|
}
|
|
5782
5899
|
catch (error) {
|
|
@@ -6113,7 +6230,6 @@ class BoundaryLabelsManager {
|
|
|
6113
6230
|
this.collapseOtherLabels(boundaryId);
|
|
6114
6231
|
// 展开当前标签
|
|
6115
6232
|
extendedContent.style.display = 'block';
|
|
6116
|
-
labelDiv.style.whiteSpace = 'normal';
|
|
6117
6233
|
this.currentExpandedBoundaryId = boundaryId;
|
|
6118
6234
|
}
|
|
6119
6235
|
/**
|
|
@@ -7173,9 +7289,10 @@ class MowerPositionManager {
|
|
|
7173
7289
|
return;
|
|
7174
7290
|
this.mowerPositionConfig = chargingPilesPositionConfig;
|
|
7175
7291
|
const lastPosition = this.lastPosition;
|
|
7176
|
-
const postureX = chargingPilesPositionConfig.postureX
|
|
7177
|
-
const postureY = chargingPilesPositionConfig.postureY
|
|
7178
|
-
const postureTheta = chargingPilesPositionConfig.postureTheta
|
|
7292
|
+
const postureX = chargingPilesPositionConfig.postureX || chargingPilesPositionConfig.lastPostureX || lastPosition?.x || 0;
|
|
7293
|
+
const postureY = chargingPilesPositionConfig.postureY || chargingPilesPositionConfig.lastPostureY || lastPosition?.y || 0;
|
|
7294
|
+
const postureTheta = chargingPilesPositionConfig.postureTheta || chargingPilesPositionConfig.lastPostureTheta || lastPosition?.rotation || 0;
|
|
7295
|
+
console.log('updatePositionByLastPosition->', postureX, postureY, postureTheta, chargingPilesPositionConfig);
|
|
7179
7296
|
// 检查是否需要更新图片
|
|
7180
7297
|
this.updateMowerImage(chargingPilesPositionConfig);
|
|
7181
7298
|
// 立即更新位置
|
|
@@ -7189,10 +7306,10 @@ class MowerPositionManager {
|
|
|
7189
7306
|
this.updateMowerImage(positionConfig);
|
|
7190
7307
|
// 更新配置
|
|
7191
7308
|
this.mowerPositionConfig = positionConfig;
|
|
7192
|
-
const postureX = positionConfig?.postureX
|
|
7193
|
-
const postureY = positionConfig?.postureY
|
|
7194
|
-
const postureTheta = positionConfig?.postureTheta
|
|
7195
|
-
console.log('updatePosition manager', positionConfig, this.lastPosition);
|
|
7309
|
+
const postureX = positionConfig?.postureX || this.lastPosition?.x || 0;
|
|
7310
|
+
const postureY = positionConfig?.postureY || this.lastPosition?.y || 0;
|
|
7311
|
+
const postureTheta = positionConfig?.postureTheta || this.lastPosition?.rotation || 0;
|
|
7312
|
+
console.log('updatePosition manager', JSON.stringify(this.currentPosition), this.currentPosition, !this.currentPosition, positionConfig, this.lastPosition, animationTime);
|
|
7196
7313
|
// 停止当前动画(如果有)
|
|
7197
7314
|
this.stopAnimation();
|
|
7198
7315
|
// 第一个点
|
|
@@ -7202,6 +7319,7 @@ class MowerPositionManager {
|
|
|
7202
7319
|
y: postureY,
|
|
7203
7320
|
rotation: postureTheta,
|
|
7204
7321
|
};
|
|
7322
|
+
console.log('updatePosition first->', this.currentPosition);
|
|
7205
7323
|
this.setElementPosition(this.currentPosition.x, this.currentPosition.y, this.currentPosition.rotation);
|
|
7206
7324
|
return;
|
|
7207
7325
|
}
|
|
@@ -7251,7 +7369,7 @@ class MowerPositionManager {
|
|
|
7251
7369
|
if (!positonOutOfRange && !positionValid) {
|
|
7252
7370
|
this.lastPosition = { x, y, rotation: theta };
|
|
7253
7371
|
}
|
|
7254
|
-
// console.log('setElementPosition', x, y, targetRotation, positonOutOfRange, positionValid, targetPixelPosition);
|
|
7372
|
+
// console.log('setElementPosition', x, y, theta, targetRotation, positonOutOfRange, positionValid, targetPixelPosition);
|
|
7255
7373
|
if (!this.mowerElement)
|
|
7256
7374
|
return;
|
|
7257
7375
|
this.mowerElement.style.left = `${targetPixelPosition?.x}px`;
|
|
@@ -7305,10 +7423,12 @@ class MowerPositionManager {
|
|
|
7305
7423
|
y: this.onlyUpdateTheta ? 0 : this.targetPosition.y - this.startPosition.y,
|
|
7306
7424
|
rotation: radNormalize(targetTheta - startTheta),
|
|
7307
7425
|
};
|
|
7426
|
+
console.log('startAnimationToPosition-->', this.deltaPosition, this.onlyUpdateTheta, this.targetPosition, this.startPosition);
|
|
7308
7427
|
// 开始动画循环
|
|
7309
7428
|
this.animateStep();
|
|
7310
7429
|
}
|
|
7311
7430
|
forceUpdatePosition() {
|
|
7431
|
+
console.log('forceUpdatePosition-->', this.currentPosition, this.targetPosition, this.startPosition);
|
|
7312
7432
|
this.animateStep();
|
|
7313
7433
|
}
|
|
7314
7434
|
/**
|
|
@@ -7335,6 +7455,7 @@ class MowerPositionManager {
|
|
|
7335
7455
|
vehicleState: this.mowerPositionConfig?.vehicleState,
|
|
7336
7456
|
});
|
|
7337
7457
|
}
|
|
7458
|
+
// console.log('animateStep-->', this.startPosition, this.deltaPosition, this.targetPosition, easedProgress)
|
|
7338
7459
|
// 继续动画或结束
|
|
7339
7460
|
if (progress < 1) {
|
|
7340
7461
|
// 设置当前位置
|
|
@@ -8621,13 +8742,6 @@ class MowerMapOverlay {
|
|
|
8621
8742
|
}
|
|
8622
8743
|
}
|
|
8623
8744
|
|
|
8624
|
-
var RealTimeDataType;
|
|
8625
|
-
(function (RealTimeDataType) {
|
|
8626
|
-
RealTimeDataType[RealTimeDataType["LOCATION"] = 1] = "LOCATION";
|
|
8627
|
-
RealTimeDataType[RealTimeDataType["PROCESS"] = 2] = "PROCESS";
|
|
8628
|
-
RealTimeDataType[RealTimeDataType["PARTITION"] = 3] = "PARTITION";
|
|
8629
|
-
})(RealTimeDataType || (RealTimeDataType = {}));
|
|
8630
|
-
|
|
8631
8745
|
// 获取车辆状态的中文文案
|
|
8632
8746
|
const getVehicleStateText = (vehicleState) => {
|
|
8633
8747
|
switch (vehicleState) {
|
|
@@ -8747,7 +8861,7 @@ const MowerMapRenderer = forwardRef(({ mapConfig, modelType, mapRef, mapJson, pa
|
|
|
8747
8861
|
// const mapRef = useMap();
|
|
8748
8862
|
const [isGoogleMapsReady, setIsGoogleMapsReady] = useState(false);
|
|
8749
8863
|
const [hasInitializedBounds, setHasInitializedBounds] = useState(false);
|
|
8750
|
-
const { clearSubBoundaryBorder } = useSubBoundaryBorderStore();
|
|
8864
|
+
const { clearSubBoundaryBorder, clearObstacles } = useSubBoundaryBorderStore();
|
|
8751
8865
|
const currentProcessMowingStatusRef = useRef(false);
|
|
8752
8866
|
const { updateProcessStateIsMowing, processStateIsMowing } = useProcessMowingState();
|
|
8753
8867
|
const [mowPartitionData, setMowPartitionData] = useState(null);
|
|
@@ -8801,8 +8915,12 @@ const MowerMapRenderer = forwardRef(({ mapConfig, modelType, mapRef, mapJson, pa
|
|
|
8801
8915
|
lastPostureTheta: currentPositionData?.lastPostureTheta
|
|
8802
8916
|
? Number(currentPositionData.lastPostureTheta)
|
|
8803
8917
|
: 0,
|
|
8804
|
-
lastPostureX: currentPositionData?.lastPostureX
|
|
8805
|
-
|
|
8918
|
+
lastPostureX: currentPositionData?.lastPostureX
|
|
8919
|
+
? Number(currentPositionData.lastPostureX)
|
|
8920
|
+
: 0,
|
|
8921
|
+
lastPostureY: currentPositionData?.lastPostureY
|
|
8922
|
+
? Number(currentPositionData.lastPostureY)
|
|
8923
|
+
: 0,
|
|
8806
8924
|
vehicleState: currentPositionData?.vehicleState || RobotStatus.CHARGING,
|
|
8807
8925
|
};
|
|
8808
8926
|
}, [realTimeData, modelType]);
|
|
@@ -8895,12 +9013,26 @@ const MowerMapRenderer = forwardRef(({ mapConfig, modelType, mapRef, mapJson, pa
|
|
|
8895
9013
|
handleError(`初始化Google Maps叠加层失败: ${error instanceof Error ? error.message : String(error)}`);
|
|
8896
9014
|
}
|
|
8897
9015
|
};
|
|
9016
|
+
const resetInCharginPie = useCallback(() => {
|
|
9017
|
+
const elements = MapDataProcessor.processMapData(mapJson, mergedMapConfig);
|
|
9018
|
+
const chargingPiles = elements.find((element) => element.type === 'charging_pile');
|
|
9019
|
+
if (!overlayRef.current)
|
|
9020
|
+
return;
|
|
9021
|
+
// 如果在充电桩上,则直接更新位置到充电桩的位置
|
|
9022
|
+
overlayRef.current.updatePosition({
|
|
9023
|
+
...mowerPositionData,
|
|
9024
|
+
postureX: chargingPiles?.originalData.position[0],
|
|
9025
|
+
postureY: chargingPiles?.originalData.position[1],
|
|
9026
|
+
postureTheta: chargingPiles?.originalData.direction - Math.PI || 0,
|
|
9027
|
+
}, 0);
|
|
9028
|
+
}, [mapJson]);
|
|
8898
9029
|
// 初始化效果
|
|
8899
9030
|
useEffect(() => {
|
|
8900
9031
|
initializeGoogleMapsOverlay();
|
|
8901
9032
|
// 清理函数
|
|
8902
9033
|
return () => {
|
|
8903
9034
|
clearSubBoundaryBorder();
|
|
9035
|
+
clearObstacles();
|
|
8904
9036
|
updateProcessStateIsMowing(false);
|
|
8905
9037
|
currentProcessMowingStatusRef.current = false;
|
|
8906
9038
|
if (overlayRef.current) {
|
|
@@ -8997,12 +9129,12 @@ const MowerMapRenderer = forwardRef(({ mapConfig, modelType, mapRef, mapJson, pa
|
|
|
8997
9129
|
// 计算左下角和右上角坐标
|
|
8998
9130
|
sw: {
|
|
8999
9131
|
x: viewBoxInfo.x / SCALE_FACTOR,
|
|
9000
|
-
y: viewBoxInfo.y / SCALE_FACTOR
|
|
9132
|
+
y: viewBoxInfo.y / SCALE_FACTOR,
|
|
9001
9133
|
},
|
|
9002
9134
|
ne: {
|
|
9003
9135
|
x: (viewBoxInfo.x + viewBoxInfo.width) / SCALE_FACTOR,
|
|
9004
|
-
y: (viewBoxInfo.y + viewBoxInfo.height) / SCALE_FACTOR
|
|
9005
|
-
}
|
|
9136
|
+
y: (viewBoxInfo.y + viewBoxInfo.height) / SCALE_FACTOR,
|
|
9137
|
+
},
|
|
9006
9138
|
};
|
|
9007
9139
|
}
|
|
9008
9140
|
}
|
|
@@ -9016,7 +9148,7 @@ const MowerMapRenderer = forwardRef(({ mapConfig, modelType, mapRef, mapJson, pa
|
|
|
9016
9148
|
lastY: mowerPositionData.lastPostureY || 0,
|
|
9017
9149
|
lastTheta: mowerPositionData.lastPostureTheta || 0,
|
|
9018
9150
|
vehicleState: mowerPositionData.vehicleState || RobotStatus.UNKNOWN,
|
|
9019
|
-
vehicleStateText: getVehicleStateText(mowerPositionData.vehicleState || RobotStatus.UNKNOWN)
|
|
9151
|
+
vehicleStateText: getVehicleStateText(mowerPositionData.vehicleState || RobotStatus.UNKNOWN),
|
|
9020
9152
|
};
|
|
9021
9153
|
}
|
|
9022
9154
|
// 获取当前割草地块数据
|
|
@@ -9024,9 +9156,6 @@ const MowerMapRenderer = forwardRef(({ mapConfig, modelType, mapRef, mapJson, pa
|
|
|
9024
9156
|
setDebugInfo(newDebugInfo);
|
|
9025
9157
|
};
|
|
9026
9158
|
updateDebugInfo();
|
|
9027
|
-
// 设置定时更新(仅在没有数据变化时使用)
|
|
9028
|
-
const interval = setInterval(updateDebugInfo, 1000);
|
|
9029
|
-
return () => clearInterval(interval);
|
|
9030
9159
|
}, [debug, mapJson, mowerPositionData, mowPartitionData, defaultTransform]);
|
|
9031
9160
|
// 当关键数据变化时立即更新debug信息
|
|
9032
9161
|
useEffect(() => {
|
|
@@ -9055,12 +9184,12 @@ const MowerMapRenderer = forwardRef(({ mapConfig, modelType, mapRef, mapJson, pa
|
|
|
9055
9184
|
// 计算左下角和右上角坐标
|
|
9056
9185
|
sw: {
|
|
9057
9186
|
x: viewBoxInfo.x / SCALE_FACTOR,
|
|
9058
|
-
y: viewBoxInfo.y / SCALE_FACTOR
|
|
9187
|
+
y: viewBoxInfo.y / SCALE_FACTOR,
|
|
9059
9188
|
},
|
|
9060
9189
|
ne: {
|
|
9061
9190
|
x: (viewBoxInfo.x + viewBoxInfo.width) / SCALE_FACTOR,
|
|
9062
|
-
y: (viewBoxInfo.y + viewBoxInfo.height) / SCALE_FACTOR
|
|
9063
|
-
}
|
|
9191
|
+
y: (viewBoxInfo.y + viewBoxInfo.height) / SCALE_FACTOR,
|
|
9192
|
+
},
|
|
9064
9193
|
};
|
|
9065
9194
|
}
|
|
9066
9195
|
}
|
|
@@ -9074,7 +9203,7 @@ const MowerMapRenderer = forwardRef(({ mapConfig, modelType, mapRef, mapJson, pa
|
|
|
9074
9203
|
lastY: mowerPositionData.lastPostureY || 0,
|
|
9075
9204
|
lastTheta: mowerPositionData.lastPostureTheta || 0,
|
|
9076
9205
|
vehicleState: mowerPositionData.vehicleState || RobotStatus.UNKNOWN,
|
|
9077
|
-
vehicleStateText: getVehicleStateText(mowerPositionData.vehicleState || RobotStatus.UNKNOWN)
|
|
9206
|
+
vehicleStateText: getVehicleStateText(mowerPositionData.vehicleState || RobotStatus.UNKNOWN),
|
|
9078
9207
|
};
|
|
9079
9208
|
}
|
|
9080
9209
|
// 获取当前割草地块数据
|
|
@@ -9094,6 +9223,24 @@ const MowerMapRenderer = forwardRef(({ mapConfig, modelType, mapRef, mapJson, pa
|
|
|
9094
9223
|
setMowPartitionData(mowingPartition);
|
|
9095
9224
|
}
|
|
9096
9225
|
const curMowPartitionData = mowingPartition || mowPartitionData;
|
|
9226
|
+
const statusData = realTimeData?.find(item => item?.type === RealTimeDataType.STATUS);
|
|
9227
|
+
if (statusData) {
|
|
9228
|
+
// 车辆回桩不会回传最后的park的位置,所以根据实时数据的状态数据判断车辆回到桩上
|
|
9229
|
+
if ([RobotStatus.CHARGING, RobotStatus.PARKED].includes(statusData?.vehicleState || RobotStatus.UNKNOWN)) {
|
|
9230
|
+
resetInCharginPie();
|
|
9231
|
+
}
|
|
9232
|
+
else if (statusData?.vehicleState === RobotStatus.MOWING) {
|
|
9233
|
+
// 兜底收不到割草地块的实时数据,使用状态来兜底
|
|
9234
|
+
const partitionids = curMowPartitionData?.partitionIds;
|
|
9235
|
+
if (!partitionids) {
|
|
9236
|
+
overlayRef.current.setBorderLayerHighlight();
|
|
9237
|
+
}
|
|
9238
|
+
}
|
|
9239
|
+
else if (statusData?.vehicleState === RobotStatus.WORKING) {
|
|
9240
|
+
// 同上
|
|
9241
|
+
overlayRef.current.resetBorderLayerHighlight();
|
|
9242
|
+
}
|
|
9243
|
+
}
|
|
9097
9244
|
if (!mapJson ||
|
|
9098
9245
|
!pathJson ||
|
|
9099
9246
|
!overlayRef.current)
|
|
@@ -9209,8 +9356,9 @@ const MowerMapRenderer = forwardRef(({ mapConfig, modelType, mapRef, mapJson, pa
|
|
|
9209
9356
|
fontFamily: 'monospace',
|
|
9210
9357
|
zIndex: 10000,
|
|
9211
9358
|
maxWidth: '300px',
|
|
9212
|
-
lineHeight: '1.4'
|
|
9213
|
-
}, children: [jsx("div", { style: { fontWeight: 'bold', marginBottom: '8px' }, children: "\uD83D\uDC1B Debug Info" }), debugInfo.mapBounds && (jsxs("div", { style: { marginBottom: '6px' }, children: [jsx("div", { style: { fontWeight: 'bold' }, children: "\uD83D\uDCCD Map GPS Bounds:" }), jsxs("div", { children: ["SW: [", debugInfo.mapBounds.sw[0].toFixed(6), ", ", debugInfo.mapBounds.sw[1].toFixed(6), "]"] }), jsxs("div", { children: ["NE: [", debugInfo.mapBounds.ne[0].toFixed(6), ", ", debugInfo.mapBounds.ne[1].toFixed(6), "]"] })] })), debugInfo.viewBox && (jsxs("div", { style: { marginBottom: '6px' }, children: [jsx("div", { style: { fontWeight: 'bold' }, children: "\uD83D\uDCD0 SVG ViewBox (meters):" }), jsxs("div", { children: ["SW: [", debugInfo.viewBox.sw.x.toFixed(2), ", ", debugInfo.viewBox.sw.y.toFixed(2), "]"] }), jsxs("div", { children: ["NE: [", debugInfo.viewBox.ne.x.toFixed(2), ", ", debugInfo.viewBox.ne.y.toFixed(2), "]"] }), jsxs("div", { children: ["Size: ", debugInfo.viewBox.width.toFixed(2), "m \u00D7 ", debugInfo.viewBox.height.toFixed(2), "m"] }), jsxs("div", { children: ["Scale: 1:", debugInfo.viewBox.scale] })] })), debugInfo.mowerPosition && (jsxs("div", { style: { marginBottom: '6px' }, children: [jsx("div", { style: { fontWeight: 'bold' }, children: "\uD83D\uDE9C Mower Position:" }), jsxs("div", { children: ["Current: X=", debugInfo.mowerPosition.x.toFixed(2), ", Y=", debugInfo.mowerPosition.y.toFixed(2)] }), jsxs("div", { children: ["Theta: ", (debugInfo.mowerPosition.theta * 180 / Math.PI).toFixed(1), "\u00B0"] }), jsxs("div", { children: ["Last: X=", debugInfo.mowerPosition.lastX.toFixed(2), ", Y=", debugInfo.mowerPosition.lastY.toFixed(2)] }), jsxs("div", { children: ["Last Theta: ", (debugInfo.mowerPosition.lastTheta * 180 / Math.PI).toFixed(1), "\u00B0"] }), jsxs("div", { children: ["Status: ", debugInfo.mowerPosition.vehicleStateText, " (", debugInfo.mowerPosition.vehicleState, ")"] })] })), debugInfo.partitionData && (jsxs("div", { style: { marginBottom: '6px' }, children: [jsx("div", { style: { fontWeight: 'bold' }, children: "\uD83D\uDD32 Mow Partition Data:" }), jsxs("div", { children: ["Type: ", debugInfo.partitionData.type || 'N/A'] }), debugInfo.partitionData.partitionIds &&
|
|
9359
|
+
lineHeight: '1.4',
|
|
9360
|
+
}, children: [jsx("div", { style: { fontWeight: 'bold', marginBottom: '8px' }, children: "\uD83D\uDC1B Debug Info" }), debugInfo.mapBounds && (jsxs("div", { style: { marginBottom: '6px' }, children: [jsx("div", { style: { fontWeight: 'bold' }, children: "\uD83D\uDCCD Map GPS Bounds:" }), jsxs("div", { children: ["SW: [", debugInfo.mapBounds.sw[0].toFixed(6), ", ", debugInfo.mapBounds.sw[1].toFixed(6), "]"] }), jsxs("div", { children: ["NE: [", debugInfo.mapBounds.ne[0].toFixed(6), ", ", debugInfo.mapBounds.ne[1].toFixed(6), "]"] })] })), debugInfo.viewBox && (jsxs("div", { style: { marginBottom: '6px' }, children: [jsx("div", { style: { fontWeight: 'bold' }, children: "\uD83D\uDCD0 SVG ViewBox (meters):" }), jsxs("div", { children: ["SW: [", debugInfo.viewBox.sw.x.toFixed(2), ", ", debugInfo.viewBox.sw.y.toFixed(2), "]"] }), jsxs("div", { children: ["NE: [", debugInfo.viewBox.ne.x.toFixed(2), ", ", debugInfo.viewBox.ne.y.toFixed(2), "]"] }), jsxs("div", { children: ["Size: ", debugInfo.viewBox.width.toFixed(2), "m \u00D7 ", debugInfo.viewBox.height.toFixed(2), "m"] }), jsxs("div", { children: ["Scale: 1:", debugInfo.viewBox.scale] })] })), debugInfo.mowerPosition && (jsxs("div", { style: { marginBottom: '6px' }, children: [jsx("div", { style: { fontWeight: 'bold' }, children: "\uD83D\uDE9C Mower Position:" }), jsxs("div", { children: ["Current: X=", debugInfo.mowerPosition.x.toFixed(2), ", Y=", debugInfo.mowerPosition.y.toFixed(2)] }), jsxs("div", { children: ["Theta: ", ((debugInfo.mowerPosition.theta * 180) / Math.PI).toFixed(1), "\u00B0"] }), jsxs("div", { children: ["Last: X=", debugInfo.mowerPosition.lastX.toFixed(2), ", Y=", debugInfo.mowerPosition.lastY.toFixed(2)] }), jsxs("div", { children: ["Last Theta: ", ((debugInfo.mowerPosition.lastTheta * 180) / Math.PI).toFixed(1), "\u00B0"] }), jsxs("div", { children: ["Status: ", debugInfo.mowerPosition.vehicleStateText, " (", debugInfo.mowerPosition.vehicleState, ")"] })] })), debugInfo.partitionData && (jsxs("div", { style: { marginBottom: '6px' }, children: [jsx("div", { style: { fontWeight: 'bold' }, children: "\uD83D\uDD32 Mow Partition Data:" }), jsxs("div", { children: ["Type: ", debugInfo.partitionData.type || 'N/A'] }), debugInfo.partitionData.partitionIds &&
|
|
9361
|
+
debugInfo.partitionData.partitionIds.length > 0 ? (jsxs("div", { children: ["Active IDs: [", debugInfo.partitionData.partitionIds.join(', '), "]"] })) : (jsx("div", { children: "No active partitions" })), debugInfo.partitionData.time && (jsxs("div", { children: ["Updated: ", new Date(debugInfo.partitionData.time).toLocaleTimeString()] }))] })), !debugInfo.partitionData && (jsxs("div", { style: { marginBottom: '6px' }, children: [jsx("div", { style: { fontWeight: 'bold' }, children: "\uD83D\uDD32 Mow Partition Data:" }), jsx("div", { style: { color: '#888' }, children: "No partition data available" })] }))] }));
|
|
9214
9362
|
};
|
|
9215
9363
|
// 错误显示
|
|
9216
9364
|
if (currentError) {
|