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