From 5d6d890cfd10466d5d14ff5177adcc888baaa0e4 Mon Sep 17 00:00:00 2001
From: 张世豪 <979909237@qq.com>
Date: 星期三, 17 十二月 2025 17:46:13 +0800
Subject: [PATCH] 新增了边界距离显示优化了设置页面布局
---
src/zhuye/MapRenderer.java | 1387 +++++++++++++++++++++++++++++++++++++++------------------
1 files changed, 937 insertions(+), 450 deletions(-)
diff --git a/src/zhuye/MapRenderer.java b/src/zhuye/MapRenderer.java
index b5665ee..78bbf86 100644
--- a/src/zhuye/MapRenderer.java
+++ b/src/zhuye/MapRenderer.java
@@ -10,32 +10,42 @@
import java.awt.geom.Point2D;
import java.awt.geom.Rectangle2D;
import java.math.BigDecimal;
-import java.text.SimpleDateFormat;
+import java.util.ArrayDeque;
import java.util.ArrayList;
import java.util.Collections;
-import java.util.Date;
+import java.util.Deque;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Locale;
import java.util.Set;
+import java.io.File;
+import set.Setsys;
import gecaoji.Device;
import gecaoji.Gecaoji;
+import gecaoji.GecaojiMeg;
+import gecaoji.gecaolunjing;
+import gecaoji.lujingdraw;
import dikuai.Dikuaiguanli;
import dikuai.Dikuai;
import zhangaiwu.Obstacledraw;
import zhangaiwu.Obstacledge;
import zhangaiwu.yulanzhangaiwu;
+import yaokong.Control03;
/**
* 鍦板浘娓叉煋鍣� - 璐熻矗鍧愭爣绯荤粯鍒躲�佽鍥惧彉鎹㈢瓑鍔熻兘
*/
public class MapRenderer {
// 瑙嗗浘鍙樻崲鍙傛暟
- private double scale = 1.0;
+ private static final double DEFAULT_SCALE = 20.0; // 榛樿缂╂斁姣斾緥
+ private double scale = DEFAULT_SCALE;
private double translateX = 0.0;
private double translateY = 0.0;
private Point lastDragPoint;
- private Point mousePoint;
+ private static final double MIN_SCALE = 0.05d;
+ private static final double MAX_SCALE = 50.0d;
+ private static final double SCALE_EPSILON = 1e-6d;
+ private static final String MAP_SCALE_PROPERTY = "mapScale"; // 灞炴�ф枃浠朵腑鐨勯敭鍚�
// 涓婚棰滆壊
private final Color THEME_COLOR = new Color(46, 139, 87);
@@ -43,16 +53,12 @@
private static final Color GRASS_FILL_COLOR = new Color(144, 238, 144, 120);
private static final Color GRASS_BORDER_COLOR = new Color(60, 179, 113);
private static final Color BOUNDARY_POINT_COLOR = new Color(128, 0, 128);
- private static final Color BOUNDARY_LABEL_COLOR = Color.BLACK;
+ private static final Color OBSTACLE_POINT_COLOR = new Color(255, 140, 0); // 姗欒壊锛岀敤浜庡尯鍒嗛殰纰嶇墿鐐�
private static final Color CIRCLE_SAMPLE_COLOR = new Color(220, 20, 60, 230);
private static final double CIRCLE_SAMPLE_SIZE = 0.54d;
private static final double BOUNDARY_POINT_MERGE_THRESHOLD = 0.05;
- private static final SimpleDateFormat TIMESTAMP_FORMAT = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
- private static final Color HANDHELD_BOUNDARY_FILL = new Color(51, 153, 255, 60);
- private static final Color HANDHELD_BOUNDARY_BORDER = new Color(51, 102, 204, 220);
- private static final Color HANDHELD_BOUNDARY_POINT = new Color(51, 102, 204);
- private static final Color HANDHELD_BOUNDARY_LABEL = new Color(22, 62, 138);
private static final double BOUNDARY_CONTAINS_TOLERANCE = 0.05;
+ private static final double PREVIEW_BOUNDARY_MARKER_SCALE = 0.25d;
// 缁勪欢寮曠敤
private JPanel visualizationPanel;
@@ -67,24 +73,21 @@
private String currentObstacleLandNumber;
private String boundaryName;
private boolean boundaryPointsVisible;
+ private boolean obstaclePointsVisible;
+ private boolean boundaryLengthVisible = false; // 鏄惁鏄剧ず杈圭晫璺濈锛岄粯璁ゅ叧闂�
+ private double boundaryPointSizeScale = 1.0d;
+ private boolean previewSizingEnabled;
private String currentBoundaryLandNumber;
private boolean dragInProgress;
private final Gecaoji mower;
private final Timer mowerUpdateTimer;
- private JDialog mowerInfoDialog;
- private Timer mowerInfoRefreshTimer;
- private JLabel mowerNumberValueLabel;
- private JLabel realtimeXValueLabel;
- private JLabel realtimeYValueLabel;
- private JLabel positioningStatusValueLabel;
- private JLabel satelliteCountValueLabel;
- private JLabel realtimeSpeedValueLabel;
- private JLabel headingValueLabel;
- private JLabel updateTimeValueLabel;
+ private final GecaojiMeg mowerInfoManager;
private CircleCaptureOverlay circleCaptureOverlay;
private final List<double[]> circleSampleMarkers = new ArrayList<>();
private final List<Point2D.Double> realtimeMowingTrack = new ArrayList<>();
+ private final Deque<tuowei.TrailSample> idleMowerTrail = new ArrayDeque<>();
private final List<Point2D.Double> handheldBoundaryPreview = new ArrayList<>();
+ private double boundaryPreviewMarkerScale = 1.0d;
private boolean realtimeTrackRecording;
private String realtimeTrackLandNumber;
private double mowerEffectiveWidthMeters;
@@ -97,15 +100,82 @@
private boolean trackDirty;
private boolean handheldBoundaryPreviewActive;
private boolean pendingTrackBreak = true;
+ private boolean idleTrailSuppressed;
+ private Path2D.Double realtimeBoundaryPathCache;
+ private String realtimeBoundaryPathLand;
- private static final double TRACK_SAMPLE_MIN_DISTANCE_METERS = 0.1d;
+ private static final double TRACK_SAMPLE_MIN_DISTANCE_METERS = 0.2d;
+ private static final double TRACK_DUPLICATE_TOLERANCE_METERS = 1e-3d;
private static final long TRACK_PERSIST_INTERVAL_MS = 5_000L;
+ public static final int DEFAULT_IDLE_TRAIL_DURATION_SECONDS = 60;
+ private static final double IDLE_TRAIL_SAMPLE_DISTANCE_METERS = 0.05d;
+ private long idleTrailDurationMs = DEFAULT_IDLE_TRAIL_DURATION_SECONDS * 1_000L;
+ private static final double ZOOM_STEP_FACTOR = 1.2d;
public MapRenderer(JPanel visualizationPanel) {
this.visualizationPanel = visualizationPanel;
this.mower = new Gecaoji();
this.mowerUpdateTimer = createMowerTimer();
+ this.mowerInfoManager = new GecaojiMeg(visualizationPanel, mower);
setupMouseListeners();
+ // 浠庨厤缃枃浠惰鍙栦笂娆′繚瀛樼殑缂╂斁姣斾緥鍜岃鍥句腑蹇冨潗鏍�
+ loadViewSettingsFromProperties();
+ }
+
+ /**
+ * 浠庨厤缃枃浠惰鍙栫缉鏀炬瘮渚嬪拰瑙嗗浘涓績鍧愭爣
+ */
+ private void loadViewSettingsFromProperties() {
+ // 鍔犺浇缂╂斁姣斾緥
+ String scaleValue = Setsys.getPropertyValue(MAP_SCALE_PROPERTY);
+ if (scaleValue != null && !scaleValue.trim().isEmpty()) {
+ try {
+ double savedScale = Double.parseDouble(scaleValue.trim());
+ // 楠岃瘉缂╂斁姣斾緥鏄惁鍦ㄦ湁鏁堣寖鍥村唴
+ if (savedScale >= MIN_SCALE && savedScale <= MAX_SCALE) {
+ scale = savedScale;
+ } else {
+ scale = DEFAULT_SCALE;
+ }
+ } catch (NumberFormatException e) {
+ // 濡傛灉瑙f瀽澶辫触锛屼娇鐢ㄩ粯璁ゅ��
+ scale = DEFAULT_SCALE;
+ }
+ } else {
+ // 濡傛灉娌℃湁淇濆瓨鐨勫�硷紝浣跨敤榛樿鍊�
+ scale = DEFAULT_SCALE;
+ }
+
+ // 鍔犺浇瑙嗗浘涓績鍧愭爣
+ String viewCenterXValue = Setsys.getPropertyValue("viewCenterX");
+ String viewCenterYValue = Setsys.getPropertyValue("viewCenterY");
+ if (viewCenterXValue != null && !viewCenterXValue.trim().isEmpty()) {
+ try {
+ translateX = Double.parseDouble(viewCenterXValue.trim());
+ } catch (NumberFormatException e) {
+ translateX = 0.0;
+ }
+ } else {
+ translateX = 0.0;
+ }
+ if (viewCenterYValue != null && !viewCenterYValue.trim().isEmpty()) {
+ try {
+ translateY = Double.parseDouble(viewCenterYValue.trim());
+ } catch (NumberFormatException e) {
+ translateY = 0.0;
+ }
+ } else {
+ translateY = 0.0;
+ }
+ }
+
+ /**
+ * 淇濆瓨缂╂斁姣斾緥鍒伴厤缃枃浠�
+ */
+ private void saveScaleToProperties() {
+ Setsys setsys = new Setsys();
+ // 淇濈暀2浣嶅皬鏁�
+ setsys.updateProperty(MAP_SCALE_PROPERTY, String.format("%.2f", scale));
}
/**
@@ -114,10 +184,10 @@
private void setupMouseListeners() {
// 榧犳爣婊氳疆缂╂斁
visualizationPanel.addMouseWheelListener(e -> {
- mousePoint = e.getPoint();
+ Point referencePoint = e.getPoint();
int notches = e.getWheelRotation();
- double zoomFactor = notches < 0 ? 1.2 : 1/1.2;
- zoomAtPoint(zoomFactor);
+ double zoomFactor = notches < 0 ? ZOOM_STEP_FACTOR : 1 / ZOOM_STEP_FACTOR;
+ zoomAtPoint(referencePoint, zoomFactor);
});
// 榧犳爣鎷栨嫿绉诲姩
@@ -147,6 +217,11 @@
if (handleMowerClick(e.getPoint())) {
return;
}
+ // 浼樺厛澶勭悊闅滅鐗╄竟鐣岀偣鐐瑰嚮锛堝鏋滃彲瑙侊級
+ if (obstaclePointsVisible && handleObstaclePointClick(e.getPoint())) {
+ return;
+ }
+ // 鐒跺悗澶勭悊鍦板潡杈圭晫鐐圭偣鍑�
if (boundaryPointsVisible) {
handleBoundaryPointClick(e.getPoint());
}
@@ -167,18 +242,13 @@
visualizationPanel.repaint();
}
}
-
- public void mouseMoved(MouseEvent e) {
- // 鏇存柊榧犳爣浣嶇疆鐢ㄤ簬鏄剧ず鍧愭爣
- mousePoint = e.getPoint();
- visualizationPanel.repaint();
- }
});
}
private Timer createMowerTimer() {
Timer timer = new Timer(300, e -> {
mower.refreshFromDevice();
+ updateIdleMowerTrail();
if (realtimeTrackRecording) {
captureRealtimeTrackPoint();
}
@@ -195,34 +265,71 @@
/**
* 鍩轰簬榧犳爣浣嶇疆鐨勭缉鏀�
*/
- private void zoomAtPoint(double zoomFactor) {
- if (mousePoint == null) return;
-
- // 璁$畻榧犳爣浣嶇疆鍦ㄤ笘鐣屽潗鏍囩郴涓殑鍧愭爣
- double worldX = (mousePoint.x - visualizationPanel.getWidth()/2) / scale - translateX;
- double worldY = (mousePoint.y - visualizationPanel.getHeight()/2) / scale - translateY;
+ private void zoomAtPoint(Point referencePoint, double zoomFactor) {
+ if (visualizationPanel == null) {
+ return;
+ }
+ if (referencePoint == null) {
+ referencePoint = new Point(visualizationPanel.getWidth() / 2, visualizationPanel.getHeight() / 2);
+ }
- scale *= zoomFactor;
- scale = Math.max(0.05, Math.min(scale, 50.0)); // 闄愬埗缂╂斁鑼冨洿锛屽厑璁告渶澶氱缉灏忚嚦鍘熷鐨�1/20
-
- // 璁$畻缂╂斁鍚庡悓涓�榧犳爣浣嶇疆鍦ㄤ笘鐣屽潗鏍囩郴涓殑鏂板潗鏍�
- double newWorldX = (mousePoint.x - visualizationPanel.getWidth()/2) / scale - translateX;
- double newWorldY = (mousePoint.y - visualizationPanel.getHeight()/2) / scale - translateY;
-
- // 璋冩暣骞崇Щ閲忥紝浣块紶鏍囨寚鍚戠殑涓栫晫鍧愭爣淇濇寔涓嶅彉
+ double panelCenterX = visualizationPanel.getWidth() / 2.0;
+ double panelCenterY = visualizationPanel.getHeight() / 2.0;
+
+ double worldX = (referencePoint.x - panelCenterX) / scale - translateX;
+ double worldY = (referencePoint.y - panelCenterY) / scale - translateY;
+
+ scale *= zoomFactor;
+ scale = Math.max(MIN_SCALE, Math.min(scale, MAX_SCALE)); // 闄愬埗缂╂斁鑼冨洿锛屽厑璁告渶澶氱缉灏忚嚦鍘熷鐨�1/20
+
+ double newWorldX = (referencePoint.x - panelCenterX) / scale - translateX;
+ double newWorldY = (referencePoint.y - panelCenterY) / scale - translateY;
+
translateX += (newWorldX - worldX);
translateY += (newWorldY - worldY);
-
+
+ // 淇濆瓨缂╂斁姣斾緥鍒伴厤缃枃浠�
+ saveScaleToProperties();
visualizationPanel.repaint();
}
+
+ public void zoomInFromCenter() {
+ zoomAtPoint(null, ZOOM_STEP_FACTOR);
+ }
+
+ public void zoomOutFromCenter() {
+ zoomAtPoint(null, 1 / ZOOM_STEP_FACTOR);
+ }
+
+ public boolean canZoomIn() {
+ return scale < MAX_SCALE - SCALE_EPSILON;
+ }
+
+ public boolean canZoomOut() {
+ return scale > MIN_SCALE + SCALE_EPSILON;
+ }
+
+ public double getScale() {
+ return scale;
+ }
+
+ public double getMaxScale() {
+ return MAX_SCALE;
+ }
+
+ public double getMinScale() {
+ return MIN_SCALE;
+ }
/**
* 閲嶇疆瑙嗗浘
*/
public void resetView() {
- scale = 1.0;
+ scale = DEFAULT_SCALE;
translateX = 0.0;
translateY = 0.0;
+ // 淇濆瓨缂╂斁姣斾緥鍒伴厤缃枃浠�
+ saveScaleToProperties();
visualizationPanel.repaint();
}
@@ -248,14 +355,11 @@
boolean hasPlannedPath = currentPlannedPath != null && currentPlannedPath.size() >= 2;
boolean hasObstacles = currentObstacles != null && !currentObstacles.isEmpty();
+ // 缁樺埗鍦板潡杈圭晫锛堝簳灞傦級
if (hasBoundary) {
drawCurrentBoundary(g2d);
}
- if (hasObstacles) {
- Obstacledraw.drawObstacles(g2d, currentObstacles, scale, selectedObstacleName);
- }
-
yulanzhangaiwu.renderPreview(g2d, scale);
if (!circleSampleMarkers.isEmpty()) {
@@ -266,23 +370,43 @@
drawCircleCaptureOverlay(g2d, circleCaptureOverlay, scale);
}
- if (handheldBoundaryPreviewActive) {
- drawHandheldBoundaryPreview(g2d);
- }
+ adddikuaiyulan.drawPreview(g2d, handheldBoundaryPreview, scale, handheldBoundaryPreviewActive, boundaryPreviewMarkerScale);
+ // 缁樺埗瀵艰埅璺緞锛堜腑灞傦級
if (hasPlannedPath) {
drawCurrentPlannedPath(g2d);
}
- if (boundaryPointsVisible && hasBoundary) {
- pointandnumber.drawBoundaryPoints(
- g2d,
- currentBoundary,
- scale,
- BOUNDARY_POINT_MERGE_THRESHOLD,
- BOUNDARY_POINT_COLOR,
- BOUNDARY_LABEL_COLOR
- );
+ // 缁樺埗闅滅鐗╋紙椤跺眰锛屾樉绀哄湪鍦板潡鍜屽鑸矾寰勪笂鏂癸級
+ if (hasObstacles) {
+ Obstacledraw.drawObstacles(g2d, currentObstacles, scale, selectedObstacleName);
+ }
+
+ // 鏄剧ず杈圭晫鐐癸紙濡傛灉杈圭晫鐐瑰彲瑙侊紝鎴栬�呰竟鐣岃窛绂诲彲瑙侊級
+ if ((boundaryPointsVisible || boundaryLengthVisible) && hasBoundary) {
+ // 棰勮妯″紡涓嬫樉绀哄簭鍙�
+ if (previewSizingEnabled) {
+ drawBoundaryPointsWithNumbers(g2d, currentBoundary, scale);
+ } else {
+ double markerScale = boundaryPointSizeScale;
+ pointandnumber.drawBoundaryPoints(
+ g2d,
+ currentBoundary,
+ scale,
+ BOUNDARY_POINT_MERGE_THRESHOLD,
+ BOUNDARY_POINT_COLOR,
+ markerScale
+ );
+ }
+ }
+
+ // 缁樺埗闅滅鐗╁潗鏍囩偣锛堝甫搴忓彿锛�
+ if (obstaclePointsVisible && hasObstacles) {
+ drawObstaclePointsWithNumbers(g2d, currentObstacles, scale);
+ }
+
+ if (shouldRenderIdleTrail()) {
+ tuowei.draw(g2d, idleMowerTrail, scale);
}
if (!realtimeMowingTrack.isEmpty()) {
@@ -291,9 +415,18 @@
drawMower(g2d);
+ // 淇濆瓨褰撳墠鍙樻崲锛堝寘鍚鍥惧彉鎹級鐢ㄤ簬鍧愭爣杞崲
+ AffineTransform currentTransformForLength = g2d.getTransform();
+
// 鎭㈠鍘熷鍙樻崲
g2d.setTransform(originalTransform);
+ // 缁樺埗杈圭晫闀垮害锛堝鏋滃惎鐢級- 鍦ㄦ仮澶嶅師濮嬪彉鎹㈠悗缁樺埗
+ if (boundaryLengthVisible && hasBoundary) {
+ bianjie.BoundaryLengthDrawer.drawBoundaryLengths(g2d, currentBoundary, scale,
+ visualizationPanel.getWidth(), visualizationPanel.getHeight(), translateX, translateY);
+ }
+
// 缁樺埗瑙嗗浘淇℃伅
drawViewInfo(g2d);
}
@@ -304,7 +437,7 @@
private void drawCoordinateSystem(Graphics2D g2d) {
// 缁樺埗鍘熺偣 - 绾㈣壊瀹炲績灏忓渾鍦�
g2d.setColor(Color.RED);
- g2d.fillOval(-1, -1, 2, 2);
+ g2d.fill(new Ellipse2D.Double(-0.5d, -0.5d, 1d, 1d));
}
@@ -317,121 +450,62 @@
}
private void drawRealtimeMowingCoverage(Graphics2D g2d) {
- if (realtimeMowingTrack.size() < 2) {
+ if (realtimeMowingTrack == null || realtimeMowingTrack.size() < 2) {
return;
}
- Path2D.Double path = new Path2D.Double();
- boolean started = false;
- for (Point2D.Double point : realtimeMowingTrack) {
- if (point == null || !Double.isFinite(point.x) || !Double.isFinite(point.y)) {
- continue;
- }
- if (!started) {
- path.moveTo(point.x, point.y);
- started = true;
- } else {
- path.lineTo(point.x, point.y);
- }
- }
-
- if (!started) {
- return;
- }
-
- Stroke originalStroke = g2d.getStroke();
- Color originalColor = g2d.getColor();
-
+ Path2D.Double boundaryPath = getRealtimeBoundaryPath();
double effectiveWidth = getEffectiveMowerWidthMeters();
- if (effectiveWidth > 0) {
- float coverageWidth = (float) Math.max(effectiveWidth, 0.1d);
- g2d.setStroke(new BasicStroke(coverageWidth, BasicStroke.CAP_ROUND, BasicStroke.JOIN_ROUND));
- g2d.setColor(new Color(46, 139, 87, 80));
- g2d.draw(path);
- }
-
- float spineWidth = (float) (effectiveWidth > 0 ? Math.max(effectiveWidth / 4.0, 0.18d) : 0.3d);
- g2d.setStroke(new BasicStroke(spineWidth, BasicStroke.CAP_ROUND, BasicStroke.JOIN_ROUND));
- g2d.setColor(new Color(34, 139, 34, 200));
- g2d.draw(path);
-
- g2d.setStroke(originalStroke);
- g2d.setColor(originalColor);
+ gecaolunjing.draw(g2d, realtimeMowingTrack, effectiveWidth, boundaryPath);
}
- private void drawHandheldBoundaryPreview(Graphics2D g2d) {
- if (!handheldBoundaryPreviewActive || handheldBoundaryPreview.isEmpty()) {
- return;
+ private Path2D.Double getRealtimeBoundaryPath() {
+ if (realtimeTrackLandNumber == null) {
+ return null;
}
- Path2D.Double path = new Path2D.Double();
- boolean started = false;
- for (Point2D.Double point : handheldBoundaryPreview) {
- if (point == null || !Double.isFinite(point.x) || !Double.isFinite(point.y)) {
- continue;
+ if (currentBoundaryLandNumber != null && realtimeTrackLandNumber.equals(currentBoundaryLandNumber)) {
+ if (currentBoundaryPath == null) {
+ currentBoundaryPath = buildBoundaryPath(currentBoundary);
}
- if (!started) {
- path.moveTo(point.x, point.y);
- started = true;
- } else {
- path.lineTo(point.x, point.y);
- }
+ return currentBoundaryPath;
}
- if (!started) {
- return;
+ if (realtimeBoundaryPathCache != null && realtimeTrackLandNumber.equals(realtimeBoundaryPathLand)) {
+ return realtimeBoundaryPathCache;
}
- Stroke originalStroke = g2d.getStroke();
- Color originalColor = g2d.getColor();
-
- if (handheldBoundaryPreview.size() >= 3) {
- path.closePath();
- g2d.setColor(HANDHELD_BOUNDARY_FILL);
- g2d.fill(path);
+ Dikuai dikuai = Dikuai.getDikuai(realtimeTrackLandNumber);
+ if (dikuai == null) {
+ realtimeBoundaryPathCache = null;
+ realtimeBoundaryPathLand = null;
+ return null;
}
- float outlineWidth = (float) Math.max(1.5f / scale, 0.2f);
- g2d.setStroke(new BasicStroke(outlineWidth, BasicStroke.CAP_ROUND, BasicStroke.JOIN_ROUND));
- g2d.setColor(HANDHELD_BOUNDARY_BORDER);
- g2d.draw(path);
-
- double markerSize = Math.max(0.5d, 3.0d / scale);
- double markerRadius = markerSize / 2.0d;
- Font originalFont = g2d.getFont();
- float labelFontSize = (float) Math.max(6.0f, 16.0f / Math.max(scale, 0.2));
- Font labelFont = originalFont.deriveFont(Font.BOLD, labelFontSize);
- FontMetrics metrics = g2d.getFontMetrics(labelFont);
-
- for (Point2D.Double point : handheldBoundaryPreview) {
- if (point == null || !Double.isFinite(point.x) || !Double.isFinite(point.y)) {
- continue;
- }
- Shape marker = new Ellipse2D.Double(point.x - markerRadius, point.y - markerRadius, markerSize, markerSize);
- g2d.setColor(HANDHELD_BOUNDARY_POINT);
- g2d.fill(marker);
+ String normalized = normalizeValue(dikuai.getBoundaryCoordinates());
+ if (normalized == null) {
+ realtimeBoundaryPathCache = null;
+ realtimeBoundaryPathLand = null;
+ return null;
}
- g2d.setFont(labelFont);
- g2d.setColor(HANDHELD_BOUNDARY_LABEL);
- int index = 1;
- for (Point2D.Double point : handheldBoundaryPreview) {
- if (point == null || !Double.isFinite(point.x) || !Double.isFinite(point.y)) {
- index++;
- continue;
- }
- String label = String.valueOf(index++);
- int textWidth = metrics.stringWidth(label);
- int ascent = metrics.getAscent();
- int descent = metrics.getDescent();
- float textX = (float) (point.x - textWidth / 2.0);
- float textY = (float) (point.y + (ascent - descent) / 2.0);
- g2d.drawString(label, textX, textY);
+ List<Point2D.Double> parsed = parseBoundary(normalized);
+ if (parsed.size() < 3) {
+ realtimeBoundaryPathCache = null;
+ realtimeBoundaryPathLand = null;
+ return null;
}
- g2d.setStroke(originalStroke);
- g2d.setFont(originalFont);
- g2d.setColor(originalColor);
+ realtimeBoundaryPathCache = buildBoundaryPath(parsed);
+ realtimeBoundaryPathLand = realtimeTrackLandNumber;
+ return realtimeBoundaryPathCache;
+ }
+
+ private boolean shouldRenderIdleTrail() {
+ return !idleTrailSuppressed
+ && !realtimeTrackRecording
+ && !handheldBoundaryPreviewActive
+ && idleMowerTrail.size() >= 2;
}
private void captureRealtimeTrackPoint() {
@@ -464,29 +538,144 @@
return;
}
+ Point2D.Double candidate = new Point2D.Double(position.x, position.y);
Point2D.Double lastPoint = realtimeMowingTrack.isEmpty() ? null : realtimeMowingTrack.get(realtimeMowingTrack.size() - 1);
- if (pendingTrackBreak) {
- lastPoint = null;
- }
- double distance = 0.0;
+ double distance = Double.NaN;
if (lastPoint != null) {
- double dx = position.x - lastPoint.x;
- double dy = position.y - lastPoint.y;
+ double dx = candidate.x - lastPoint.x;
+ double dy = candidate.y - lastPoint.y;
distance = Math.hypot(dx, dy);
+ if (distance <= TRACK_DUPLICATE_TOLERANCE_METERS) {
+ return;
+ }
if (distance < TRACK_SAMPLE_MIN_DISTANCE_METERS) {
return;
}
}
- realtimeMowingTrack.add(new Point2D.Double(position.x, position.y));
- if (distance > 0.0) {
+ realtimeMowingTrack.add(candidate);
+ if (!pendingTrackBreak && lastPoint != null && Double.isFinite(distance)) {
trackLengthMeters += distance;
}
updateCompletionMetrics();
trackDirty = true;
maybePersistRealtimeTrack(false);
- pendingTrackBreak = false;
+ pendingTrackBreak = false;
+ }
+
+ private void updateIdleMowerTrail() {
+ long now = System.currentTimeMillis();
+ pruneIdleMowerTrail(now);
+
+ if (idleTrailSuppressed || realtimeTrackRecording) {
+ if (!idleMowerTrail.isEmpty()) {
+ clearIdleMowerTrail();
+ }
+ return;
+ }
+
+ Device device = Device.getGecaoji();
+ if (device == null) {
+ return;
+ }
+ // 浣跨敤鏇村鏉剧殑瀹氫綅鐘舵�佸垽鏂紝鍏佽鐘舵��1鍜�4鏄剧ず鎷栧熬
+ if (!isValidFixForTrail(device.getPositioningStatus())) {
+ return;
+ }
+
+ Point2D.Double position = mower.getPosition();
+ if (position == null || !Double.isFinite(position.x) || !Double.isFinite(position.y)) {
+ return;
+ }
+
+ tuowei.TrailSample lastSample = idleMowerTrail.peekLast();
+ if (lastSample != null) {
+ Point2D.Double lastPoint = lastSample.getPoint();
+ double dx = position.x - lastPoint.x;
+ double dy = position.y - lastPoint.y;
+ if (Math.hypot(dx, dy) < IDLE_TRAIL_SAMPLE_DISTANCE_METERS) {
+ return;
+ }
+ }
+
+ idleMowerTrail.addLast(new tuowei.TrailSample(now, new Point2D.Double(position.x, position.y)));
+ pruneIdleMowerTrail(now);
+ }
+
+ /**
+ * 寮哄埗鏇存柊鎷栧熬锛堢敤浜庢敹鍒�$GNGGA鏁版嵁鏃剁珛鍗虫洿鏂帮級
+ * 杩欎釜鏂规硶浼氬埛鏂癿ower浣嶇疆骞剁珛鍗虫坊鍔犲埌鎷栧熬
+ */
+ public void forceUpdateIdleMowerTrail() {
+ long now = System.currentTimeMillis();
+ pruneIdleMowerTrail(now);
+
+ if (idleTrailSuppressed || realtimeTrackRecording) {
+ if (!idleMowerTrail.isEmpty()) {
+ clearIdleMowerTrail();
+ }
+ return;
+ }
+
+ Device device = Device.getGecaoji();
+ if (device == null) {
+ return;
+ }
+ // 浣跨敤鏇村鏉剧殑瀹氫綅鐘舵�佸垽鏂紝鍏佽鐘舵��1鍜�4鏄剧ず鎷栧熬
+ if (!isValidFixForTrail(device.getPositioningStatus())) {
+ return;
+ }
+
+ // 鍒锋柊mower浣嶇疆锛屼娇鐢ㄦ渶鏂扮殑Device鏁版嵁
+ mower.refreshFromDevice();
+ Point2D.Double position = mower.getPosition();
+ if (position == null || !Double.isFinite(position.x) || !Double.isFinite(position.y)) {
+ return;
+ }
+
+ tuowei.TrailSample lastSample = idleMowerTrail.peekLast();
+ if (lastSample != null) {
+ Point2D.Double lastPoint = lastSample.getPoint();
+ double dx = position.x - lastPoint.x;
+ double dy = position.y - lastPoint.y;
+ if (Math.hypot(dx, dy) < IDLE_TRAIL_SAMPLE_DISTANCE_METERS) {
+ return;
+ }
+ }
+
+ idleMowerTrail.addLast(new tuowei.TrailSample(now, new Point2D.Double(position.x, position.y)));
+ pruneIdleMowerTrail(now);
+
+ // 绔嬪嵆閲嶇粯锛岀‘淇濇嫋灏惧強鏃舵樉绀�
+ if (visualizationPanel != null) {
+ visualizationPanel.repaint();
+ }
+ }
+
+ private void pruneIdleMowerTrail(long now) {
+ if (idleMowerTrail.isEmpty()) {
+ return;
+ }
+ long cutoff = now - idleTrailDurationMs;
+ boolean modified = false;
+ while (!idleMowerTrail.isEmpty() && idleMowerTrail.peekFirst().getTimestamp() < cutoff) {
+ idleMowerTrail.removeFirst();
+ modified = true;
+ }
+ if (modified && visualizationPanel != null) {
+ visualizationPanel.repaint();
+ }
+ }
+
+ private void clearIdleMowerTrail() {
+ if (idleMowerTrail.isEmpty()) {
+ return;
+ }
+ idleMowerTrail.clear();
+ if (visualizationPanel != null) {
+ visualizationPanel.repaint();
+ }
}
private void updateCompletionMetrics() {
@@ -601,6 +790,9 @@
mowerEffectiveWidthMeters = defaultMowerWidthMeters;
}
+ idleTrailSuppressed = true;
+ clearIdleMowerTrail();
+
realtimeTrackLandNumber = normalizedLand;
realtimeTrackRecording = true;
pendingTrackBreak = true;
@@ -610,12 +802,14 @@
public void pauseRealtimeTrackRecording() {
realtimeTrackRecording = false;
pendingTrackBreak = true;
+ idleTrailSuppressed = false;
maybePersistRealtimeTrack(true);
}
public void stopRealtimeTrackRecording() {
realtimeTrackRecording = false;
pendingTrackBreak = true;
+ idleTrailSuppressed = false;
maybePersistRealtimeTrack(true);
}
@@ -634,10 +828,35 @@
mowingCompletionRatio = 0.0;
trackDirty = true;
pendingTrackBreak = true;
+ idleTrailSuppressed = false;
maybePersistRealtimeTrack(true);
visualizationPanel.repaint();
}
+ public void clearIdleTrail() {
+ clearIdleMowerTrail();
+ }
+
+ public void setIdleTrailDurationSeconds(int seconds) {
+ int sanitized = seconds;
+ if (sanitized < 5 || sanitized > 600) {
+ sanitized = DEFAULT_IDLE_TRAIL_DURATION_SECONDS;
+ }
+ idleTrailDurationMs = sanitized * 1_000L;
+ pruneIdleMowerTrail(System.currentTimeMillis());
+ }
+
+ public int getIdleTrailDurationSeconds() {
+ long seconds = idleTrailDurationMs / 1_000L;
+ if (seconds <= 0L) {
+ return DEFAULT_IDLE_TRAIL_DURATION_SECONDS;
+ }
+ if (seconds > Integer.MAX_VALUE) {
+ return DEFAULT_IDLE_TRAIL_DURATION_SECONDS;
+ }
+ return (int) seconds;
+ }
+
public double getMowingCompletionRatio() {
if (!isMowerInsideSelectedBoundary()) {
return 0.0;
@@ -682,6 +901,8 @@
trackDirty = false;
lastTrackPersistTimeMillis = 0L;
pendingTrackBreak = true;
+ realtimeBoundaryPathCache = null;
+ realtimeBoundaryPathLand = null;
String trimmed = normalizeValue(trackData);
if (trimmed == null || trimmed.isEmpty()) {
@@ -690,6 +911,7 @@
}
String[] segments = trimmed.split(";");
+ Path2D.Double boundaryPath = getRealtimeBoundaryPath();
Point2D.Double lastPoint = null;
for (String segment : segments) {
if (segment == null || segment.trim().isEmpty()) {
@@ -702,11 +924,26 @@
try {
double x = Double.parseDouble(parts[0].trim());
double y = Double.parseDouble(parts[1].trim());
- Point2D.Double current = new Point2D.Double(x, y);
- realtimeMowingTrack.add(current);
- if (lastPoint != null) {
- trackLengthMeters += Math.hypot(current.x - lastPoint.x, current.y - lastPoint.y);
+ if (!Double.isFinite(x) || !Double.isFinite(y)) {
+ continue;
}
+ Point2D.Double current = new Point2D.Double(x, y);
+ if (boundaryPath != null && !isPointInsideBoundary(current, boundaryPath)) {
+ continue;
+ }
+ if (lastPoint != null) {
+ double dx = current.x - lastPoint.x;
+ double dy = current.y - lastPoint.y;
+ double distance = Math.hypot(dx, dy);
+ if (distance <= TRACK_DUPLICATE_TOLERANCE_METERS) {
+ continue;
+ }
+ if (distance < TRACK_SAMPLE_MIN_DISTANCE_METERS) {
+ continue;
+ }
+ trackLengthMeters += distance;
+ }
+ realtimeMowingTrack.add(current);
lastPoint = current;
} catch (NumberFormatException ignored) {
// 璺宠繃寮傚父鏉$洰
@@ -817,7 +1054,8 @@
}
private void drawCurrentPlannedPath(Graphics2D g2d) {
- lujingdraw.drawPlannedPath(g2d, currentPlannedPath, scale);
+ double arrowScale = previewSizingEnabled ? 0.5d : 1.0d;
+ lujingdraw.drawPlannedPath(g2d, currentPlannedPath, scale, arrowScale);
}
private void drawCircleSampleMarkers(Graphics2D g2d, List<double[]> markers, double scale) {
@@ -951,248 +1189,142 @@
}
}
- private void showMowerInfo() {
- Device device = Device.getGecaoji();
- if (device == null) {
+ public void showMowerInfo() {
+ if (mowerInfoManager != null) {
+ mowerInfoManager.showMowerInfo();
+ }
+ }
+
+ /**
+ * 澶勭悊闅滅鐗╄竟鐣岀偣鐐瑰嚮
+ * @param screenPoint 灞忓箷鍧愭爣鐐�
+ * @return 濡傛灉澶勭悊浜嗙偣鍑昏繑鍥瀟rue锛屽惁鍒欒繑鍥瀎alse
+ */
+ private boolean handleObstaclePointClick(Point screenPoint) {
+ if (currentObstacles == null || currentObstacles.isEmpty() || currentObstacleLandNumber == null) {
+ return false;
+ }
+
+ double threshold = computeSelectionThresholdPixels();
+
+ // 閬嶅巻鎵�鏈夐殰纰嶇墿锛屾壘鍒拌鐐瑰嚮鐨勭偣
+ for (Obstacledge.Obstacle obstacle : currentObstacles) {
+ if (obstacle == null) {
+ continue;
+ }
+
+ List<Obstacledge.XYCoordinate> xyCoords = obstacle.getXyCoordinates();
+ if (xyCoords == null || xyCoords.isEmpty()) {
+ continue;
+ }
+
+ // 妫�鏌ユ瘡涓偣
+ for (int i = 0; i < xyCoords.size(); i++) {
+ Obstacledge.XYCoordinate coord = xyCoords.get(i);
+ Point2D.Double worldPoint = new Point2D.Double(coord.getX(), coord.getY());
+ Point2D.Double screenPosition = worldToScreen(worldPoint);
+
+ double dx = screenPosition.x - screenPoint.x;
+ double dy = screenPosition.y - screenPoint.y;
+ if (Math.hypot(dx, dy) <= threshold) {
+ // 鎵惧埌琚偣鍑荤殑鐐�
+ String obstacleName = obstacle.getObstacleName();
+ String pointLabel = (i + 1) + "";
+ String message = "纭畾瑕佸垹闄ら殰纰嶇墿 \"" + obstacleName + "\" 鐨勭" + pointLabel + "鍙疯竟鐣岀偣鍚楋紵";
+
+ int choice = JOptionPane.showConfirmDialog(
+ visualizationPanel,
+ message,
+ "鍒犻櫎闅滅鐗╄竟鐣岀偣",
+ JOptionPane.OK_CANCEL_OPTION,
+ JOptionPane.WARNING_MESSAGE
+ );
+
+ if (choice == JOptionPane.OK_OPTION) {
+ removeObstaclePoint(obstacle, i);
+ }
+ return true;
+ }
+ }
+ }
+
+ return false;
+ }
+
+ /**
+ * 鍒犻櫎闅滅鐗╃殑鎸囧畾杈圭晫鐐�
+ */
+ private void removeObstaclePoint(Obstacledge.Obstacle obstacle, int pointIndex) {
+ if (obstacle == null || currentObstacleLandNumber == null) {
+ return;
+ }
+
+ List<Obstacledge.XYCoordinate> xyCoords = obstacle.getXyCoordinates();
+ if (xyCoords == null || pointIndex < 0 || pointIndex >= xyCoords.size()) {
+ return;
+ }
+
+ // 妫�鏌ュ垹闄ゅ悗鏄惁杩樻湁瓒冲鐨勭偣
+ Obstacledge.ObstacleShape shape = obstacle.getShape();
+ int minPoints = (shape == Obstacledge.ObstacleShape.CIRCLE) ? 2 : 3;
+
+ if (xyCoords.size() <= minPoints) {
JOptionPane.showMessageDialog(
visualizationPanel,
- "鏃犺澶囨暟鎹�",
- "鍓茶崏鏈轰俊鎭�",
+ "闅滅鐗╄嚦灏戦渶瑕�" + minPoints + "涓偣锛屾棤娉曞垹闄�",
+ "鎻愮ず",
JOptionPane.INFORMATION_MESSAGE
);
return;
}
-
- ensureMowerInfoDialog();
- updateMowerInfoLabels();
- positionMowerDialog();
- if (!mowerInfoDialog.isVisible()) {
- mowerInfoDialog.setVisible(true);
- } else {
- mowerInfoDialog.toFront();
- }
- startMowerInfoTimer();
- }
-
- private void ensureMowerInfoDialog() {
- if (mowerInfoDialog != null) {
- return;
- }
-
- Window owner = SwingUtilities.getWindowAncestor(visualizationPanel);
- JDialog dialog = new JDialog(owner, "鍓茶崏鏈轰俊鎭�", Dialog.ModalityType.MODELESS);
- dialog.setDefaultCloseOperation(WindowConstants.DISPOSE_ON_CLOSE);
-
- JPanel content = new JPanel(new BorderLayout(0, 12));
- content.setBorder(BorderFactory.createEmptyBorder(16, 20, 16, 20));
-
- JPanel grid = new JPanel(new GridLayout(0, 2, 12, 8));
-
- grid.add(new JLabel("璁惧缂栧彿锛�"));
- mowerNumberValueLabel = createValueLabel();
- grid.add(mowerNumberValueLabel);
-
- grid.add(new JLabel("瀹炴椂X鍧愭爣锛�"));
- realtimeXValueLabel = createValueLabel();
- grid.add(realtimeXValueLabel);
-
- grid.add(new JLabel("瀹炴椂Y鍧愭爣锛�"));
- realtimeYValueLabel = createValueLabel();
- grid.add(realtimeYValueLabel);
-
- grid.add(new JLabel("瀹氫綅璐ㄩ噺锛�"));
- positioningStatusValueLabel = createValueLabel();
- grid.add(positioningStatusValueLabel);
-
- grid.add(new JLabel("鍗槦棰楁暟锛�"));
- satelliteCountValueLabel = createValueLabel();
- grid.add(satelliteCountValueLabel);
-
- grid.add(new JLabel("琛岄┒閫熷害锛�"));
- realtimeSpeedValueLabel = createValueLabel();
- grid.add(realtimeSpeedValueLabel);
-
- grid.add(new JLabel("杩愬姩鑸悜锛�"));
- headingValueLabel = createValueLabel();
- grid.add(headingValueLabel);
-
- grid.add(new JLabel("鏇存柊鏃堕棿锛�"));
- updateTimeValueLabel = createValueLabel();
- grid.add(updateTimeValueLabel);
-
- content.add(grid, BorderLayout.CENTER);
-
- JButton closeButton = new JButton("鍏抽棴");
- closeButton.addActionListener(e -> dialog.dispose());
- JPanel buttonPanel = new JPanel(new FlowLayout(FlowLayout.RIGHT));
- buttonPanel.add(closeButton);
- content.add(buttonPanel, BorderLayout.SOUTH);
-
- dialog.setContentPane(content);
- dialog.pack();
- dialog.setResizable(false);
- dialog.addWindowListener(new WindowAdapter() {
- @Override
- public void windowClosed(WindowEvent e) {
- stopMowerInfoTimer();
- resetMowerInfoLabels();
- mowerInfoDialog = null;
+
+ // 鍒涘缓鏂扮殑鍧愭爣鍒楄〃锛堢Щ闄ゆ寚瀹氱偣锛�
+ List<Obstacledge.XYCoordinate> updatedCoords = new ArrayList<>(xyCoords);
+ updatedCoords.remove(pointIndex);
+
+ // 鏇存柊闅滅鐗╁潗鏍�
+ obstacle.setXyCoordinates(updatedCoords);
+
+ // 淇濆瓨鍒伴厤缃枃浠�
+ try {
+ File configFile = new File("Obstacledge.properties");
+ Obstacledge.ConfigManager manager = new Obstacledge.ConfigManager();
+ if (configFile.exists()) {
+ manager.loadFromFile(configFile.getAbsolutePath());
}
- });
-
- mowerInfoDialog = dialog;
- }
-
- private void positionMowerDialog() {
- if (mowerInfoDialog == null || visualizationPanel == null) {
- return;
- }
- int panelWidth = visualizationPanel.getWidth();
- int panelHeight = visualizationPanel.getHeight();
- int dialogHeight = mowerInfoDialog.getHeight();
- if (dialogHeight <= 0) {
- dialogHeight = mowerInfoDialog.getPreferredSize().height;
- }
- if (panelWidth > 0) {
- mowerInfoDialog.setSize(panelWidth, dialogHeight);
- }
-
- try {
- Point panelOnScreen = visualizationPanel.getLocationOnScreen();
- int dialogWidth = mowerInfoDialog.getWidth();
- int targetX = panelOnScreen.x + (panelWidth - dialogWidth) / 2;
- int targetY = panelOnScreen.y + panelHeight / 3 - dialogHeight / 2;
- mowerInfoDialog.setLocation(targetX, targetY);
- } catch (IllegalComponentStateException ex) {
- // component not yet showing; ignore positioning
- }
- }
-
- private JLabel createValueLabel() {
- JLabel label = new JLabel("--");
- label.setHorizontalAlignment(SwingConstants.LEFT);
- return label;
- }
-
- private void startMowerInfoTimer() {
- if (mowerInfoRefreshTimer == null) {
- mowerInfoRefreshTimer = new Timer(1000, e -> updateMowerInfoLabels());
- mowerInfoRefreshTimer.setRepeats(true);
- }
- if (!mowerInfoRefreshTimer.isRunning()) {
- mowerInfoRefreshTimer.start();
- }
- }
-
- private void stopMowerInfoTimer() {
- if (mowerInfoRefreshTimer != null) {
- mowerInfoRefreshTimer.stop();
- mowerInfoRefreshTimer = null;
- }
- }
-
- private void resetMowerInfoLabels() {
- mowerNumberValueLabel = null;
- realtimeXValueLabel = null;
- realtimeYValueLabel = null;
- positioningStatusValueLabel = null;
- satelliteCountValueLabel = null;
- realtimeSpeedValueLabel = null;
- headingValueLabel = null;
- updateTimeValueLabel = null;
- }
-
- private void updateMowerInfoLabels() {
- if (mowerNumberValueLabel == null) {
- return;
- }
-
- Device device = Device.getGecaoji();
- if (device == null) {
- setMowerInfoLabels("--");
- return;
- }
-
- mower.refreshFromDevice();
-
- mowerNumberValueLabel.setText(formatDeviceValue(device.getMowerNumber()));
- realtimeXValueLabel.setText(formatDeviceValue(device.getRealtimeX()));
- realtimeYValueLabel.setText(formatDeviceValue(device.getRealtimeY()));
- positioningStatusValueLabel.setText(formatFixQualityValue(device.getPositioningStatus()));
- satelliteCountValueLabel.setText(formatDeviceValue(device.getSatelliteCount()));
- realtimeSpeedValueLabel.setText(formatDeviceValue(device.getRealtimeSpeed()));
- headingValueLabel.setText(formatDeviceValue(device.getHeading()));
- updateTimeValueLabel.setText(formatTimestamp(device.getGupdateTime()));
- }
-
- private void setMowerInfoLabels(String value) {
- if (mowerNumberValueLabel != null) mowerNumberValueLabel.setText(value);
- if (realtimeXValueLabel != null) realtimeXValueLabel.setText(value);
- if (realtimeYValueLabel != null) realtimeYValueLabel.setText(value);
- if (positioningStatusValueLabel != null) positioningStatusValueLabel.setText(value);
- if (satelliteCountValueLabel != null) satelliteCountValueLabel.setText(value);
- if (realtimeSpeedValueLabel != null) realtimeSpeedValueLabel.setText(value);
- if (headingValueLabel != null) headingValueLabel.setText(value);
- if (updateTimeValueLabel != null) updateTimeValueLabel.setText(value);
- }
-
- private String sanitizeDeviceValue(String value) {
- if (value == null) {
- return null;
- }
- String trimmed = value.trim();
- if (trimmed.isEmpty() || "-1".equals(trimmed)) {
- return null;
- }
- return trimmed;
- }
-
- private String formatDeviceValue(String value) {
- String sanitized = sanitizeDeviceValue(value);
- return sanitized == null ? "--" : sanitized;
- }
-
- private String formatFixQualityValue(String value) {
- String sanitized = sanitizeDeviceValue(value);
- if (sanitized == null) {
- return "--";
- }
- switch (sanitized) {
- case "0":
- return "鏈畾浣�";
- case "1":
- return "鍗曠偣瀹氫綅";
- case "2":
- return "鐮佸樊鍒�";
- case "3":
- return "鏃犳晥PPS";
- case "4":
- return "鍥哄畾瑙�";
- case "5":
- return "娴偣瑙�";
- case "6":
- return "姝e湪浼扮畻";
- case "7":
- return "浜哄伐杈撳叆鍥哄畾鍊�";
- case "8":
- return "妯℃嫙妯″紡";
- case "9":
- return "WAAS宸垎";
- default:
- return sanitized;
- }
- }
-
- private String formatTimestamp(String value) {
- String sanitized = sanitizeDeviceValue(value);
- if (sanitized == null) {
- return "--";
- }
- try {
- long millis = Long.parseLong(sanitized);
- return TIMESTAMP_FORMAT.format(new Date(millis));
- } catch (NumberFormatException ex) {
- return sanitized;
+
+ Obstacledge.Plot plot = manager.getPlotById(currentObstacleLandNumber.trim());
+ if (plot != null) {
+ // 绉婚櫎鏃ч殰纰嶇墿骞舵坊鍔犳洿鏂板悗鐨勯殰纰嶇墿
+ plot.removeObstacleByName(obstacle.getObstacleName());
+ plot.addObstacle(obstacle);
+ manager.saveToFile(configFile.getAbsolutePath());
+
+ // 鏇存柊鍦板潡鏇存柊鏃堕棿
+ Dikuai.updateField(currentObstacleLandNumber.trim(), "updateTime",
+ new java.text.SimpleDateFormat("yyyy-MM-dd HH:mm:ss").format(new java.util.Date()));
+ Dikuai.saveToProperties();
+
+ // 閲嶆柊鍔犺浇闅滅鐗╂暟鎹互鍒锋柊鏄剧ず
+ List<Obstacledge.Obstacle> updatedObstacles = new ArrayList<>();
+ for (Obstacledge.Obstacle obs : currentObstacles) {
+ if (obs.getObstacleName().equals(obstacle.getObstacleName())) {
+ updatedObstacles.add(obstacle); // 浣跨敤鏇存柊鍚庣殑闅滅鐗�
+ } else {
+ updatedObstacles.add(obs); // 淇濇寔鍏朵粬闅滅鐗╀笉鍙�
+ }
+ }
+ applyObstaclesToRenderer(updatedObstacles, currentObstacleLandNumber);
+ visualizationPanel.repaint();
+ }
+ } catch (Exception ex) {
+ ex.printStackTrace();
+ JOptionPane.showMessageDialog(
+ visualizationPanel,
+ "淇濆瓨澶辫触: " + ex.getMessage(),
+ "閿欒",
+ JOptionPane.ERROR_MESSAGE
+ );
}
}
@@ -1245,7 +1377,11 @@
private double computeSelectionThresholdPixels() {
double scaleFactor = Math.max(0.5, scale);
- double markerDiameterWorld = Math.max(1.0, (10.0 / scaleFactor) * 0.2);
+ double diameterScale = boundaryPointSizeScale * (previewSizingEnabled ? PREVIEW_BOUNDARY_MARKER_SCALE : 1.0d);
+ if (!Double.isFinite(diameterScale) || diameterScale <= 0.0d) {
+ diameterScale = 1.0d;
+ }
+ double markerDiameterWorld = Math.max(1.0, (10.0 / scaleFactor) * 0.2 * diameterScale);
double markerDiameterPixels = markerDiameterWorld * scale;
return Math.max(8.0, markerDiameterPixels * 1.5);
}
@@ -1378,6 +1514,32 @@
return false;
}
}
+
+ /**
+ * 鍒ゆ柇瀹氫綅鐘舵�佹槸鍚︽湁鏁堬紝鍙敤浜庢樉绀烘嫋灏�
+ * 鎺ュ彈鐘舵��1锛堝崟鐐瑰畾浣嶏級銆�2锛堢爜宸垎锛夈��3锛堟棤鏁圥PS锛夈��4锛堝浐瀹氳В锛夈��5锛堟诞鐐硅В锛�
+ */
+ private boolean isValidFixForTrail(String fixQuality) {
+ if (fixQuality == null) {
+ return false;
+ }
+ String trimmed = fixQuality.trim();
+ if (trimmed.isEmpty()) {
+ return false;
+ }
+ // 鎺ュ彈鐘舵��1,2,3,4,5锛堝彧瑕佷笉鏄�0鎴栨棤鏁堢姸鎬侊級
+ if ("1".equals(trimmed) || "2".equals(trimmed) || "3".equals(trimmed) ||
+ "4".equals(trimmed) || "5".equals(trimmed)) {
+ return true;
+ }
+ try {
+ double value = Double.parseDouble(trimmed);
+ // 鎺ュ彈1.0, 2.0, 3.0, 4.0, 5.0锛堝彧瑕佷笉鏄�0锛�
+ return value >= 1.0 && value <= 5.0;
+ } catch (NumberFormatException ex) {
+ return false;
+ }
+ }
private boolean isPointInsideActiveBoundary(Point2D.Double point) {
if (point == null || !Double.isFinite(point.x) || !Double.isFinite(point.y)) {
@@ -1386,16 +1548,12 @@
if (realtimeTrackLandNumber == null) {
return false;
}
- if (currentBoundaryLandNumber != null && !currentBoundaryLandNumber.equals(realtimeTrackLandNumber)) {
- return false;
- }
+ Path2D.Double path = getRealtimeBoundaryPath();
+ return isPointInsideBoundary(point, path);
+ }
- Path2D.Double path = currentBoundaryPath;
- if (path == null) {
- path = buildBoundaryPath(currentBoundary);
- currentBoundaryPath = path;
- }
- if (path == null) {
+ private boolean isPointInsideBoundary(Point2D.Double point, Path2D.Double path) {
+ if (point == null || path == null || !Double.isFinite(point.x) || !Double.isFinite(point.y)) {
return false;
}
if (path.contains(point.x, point.y)) {
@@ -1437,33 +1595,214 @@
/**
* 缁樺埗瑙嗗浘淇℃伅
*/
- private void drawViewInfo(Graphics2D g2d) {
- g2d.setColor(new Color(80, 80, 80));
- g2d.setFont(new Font("寰蒋闆呴粦", Font.PLAIN, 11));
-
- // 鍙樉绀虹缉鏀炬瘮渚嬶紝鍘绘帀骞崇Щ淇℃伅
- String info = String.format("缂╂斁: %.2fx", scale);
- g2d.drawString(info, 15, visualizationPanel.getHeight() - 15);
-
- if (mousePoint != null) {
- // 璁$畻榧犳爣浣嶇疆鍦ㄤ笘鐣屽潗鏍囩郴涓殑鍧愭爣
- double worldX = (mousePoint.x - visualizationPanel.getWidth()/2) / scale - translateX;
- double worldY = (mousePoint.y - visualizationPanel.getHeight()/2) / scale - translateY;
-
- String mouseCoord = String.format("鍧愭爣: (%.1f, %.1f)m", worldX, worldY);
- g2d.drawString(mouseCoord, visualizationPanel.getWidth() - 150, visualizationPanel.getHeight() - 15);
+ /**
+ * 缁樺埗闅滅鐗╁潗鏍囩偣锛堝甫搴忓彿锛�
+ * 搴忓彿鏄剧ず鍦ㄧ偣涓績锛屽瓧浣撳ぇ灏忎笌闅滅鐗╁悕绉颁竴鑷达紙11鍙凤級锛屼笉闅忕缉鏀惧彉鍖�
+ */
+ private void drawObstaclePointsWithNumbers(Graphics2D g2d, List<Obstacledge.Obstacle> obstacles, double scale) {
+ if (obstacles == null || obstacles.isEmpty()) {
+ return;
}
- // 鍘绘帀鎿嶄綔鎻愮ず锛�"婊氳疆缂╂斁 | 鎷栨嫿绉诲姩 | 鍙抽敭閲嶇疆"
- // String tips = "婊氳疆缂╂斁 | 鎷栨嫿绉诲姩 | 鍙抽敭閲嶇疆";
- // g2d.drawString(tips, visualizationPanel.getWidth() - 200, 30);
+ // 淇濆瓨鍘熷鍙樻崲
+ AffineTransform originalTransform = g2d.getTransform();
+
+ // 璁剧疆鐐圭殑澶у皬锛堥殢缂╂斁鍙樺寲锛�
+ double scaleFactor = Math.max(0.5, scale);
+ double clampedScale = boundaryPointSizeScale * (previewSizingEnabled ? PREVIEW_BOUNDARY_MARKER_SCALE : 1.0d);
+ if (!Double.isFinite(clampedScale) || clampedScale <= 0.0d) {
+ clampedScale = 1.0d;
+ }
+ double minimumDiameter = clampedScale < 1.0 ? 0.5 : 1.0;
+ double markerDiameter = Math.max(minimumDiameter, (10.0 / scaleFactor) * 0.2 * clampedScale);
+ double markerRadius = markerDiameter / 2.0;
+
+ // 璁剧疆瀛椾綋锛堜笌闅滅鐗╁悕绉颁竴鑷达紝涓嶉殢缂╂斁鍙樺寲锛�
+ Font labelFont = new Font("寰蒋闆呴粦", Font.PLAIN, 11);
+ g2d.setFont(labelFont);
+ FontMetrics fontMetrics = g2d.getFontMetrics(labelFont);
+
+ // 閬嶅巻鎵�鏈夐殰纰嶇墿
+ for (Obstacledge.Obstacle obstacle : obstacles) {
+ if (obstacle == null || !obstacle.isValid()) {
+ continue;
+ }
+
+ List<Obstacledge.XYCoordinate> xyCoords = obstacle.getXyCoordinates();
+ if (xyCoords == null || xyCoords.isEmpty()) {
+ continue;
+ }
+
+ // 缁樺埗姣忎釜鐐瑰強鍏跺簭鍙�
+ for (int i = 0; i < xyCoords.size(); i++) {
+ Obstacledge.XYCoordinate coord = xyCoords.get(i);
+ double x = coord.getX();
+ double y = coord.getY();
+
+ // 缁樺埗鐐癸紙鍦ㄤ笘鐣屽潗鏍囩郴涓紝闅忕缉鏀惧彉鍖栵級
+ g2d.setColor(OBSTACLE_POINT_COLOR);
+ Ellipse2D.Double marker = new Ellipse2D.Double(
+ x - markerRadius,
+ y - markerRadius,
+ markerDiameter,
+ markerDiameter
+ );
+ g2d.fill(marker);
+
+ // 灏嗕笘鐣屽潗鏍囪浆鎹负灞忓箷鍧愭爣浠ョ粯鍒跺簭鍙凤紙涓嶉殢缂╂斁鍙樺寲锛�
+ Point2D.Double worldPoint = new Point2D.Double(x, y);
+ Point2D.Double screenPoint = new Point2D.Double();
+ originalTransform.transform(worldPoint, screenPoint);
+
+ // 淇濆瓨褰撳墠鍙樻崲
+ AffineTransform savedTransform = g2d.getTransform();
+
+ // 閲嶇疆鍙樻崲涓哄睆骞曞潗鏍囩郴缁�
+ g2d.setTransform(new AffineTransform());
+
+ // 缁樺埗搴忓彿锛堝湪灞忓箷鍧愭爣绯讳腑锛屼笉闅忕缉鏀惧彉鍖栵級
+ String numberText = String.valueOf(i + 1);
+ int textWidth = fontMetrics.stringWidth(numberText);
+ int textHeight = fontMetrics.getHeight();
+
+ // 鍦ㄧ偣涓績缁樺埗搴忓彿
+ int textX = (int)(screenPoint.x - textWidth / 2.0);
+ int textY = (int)(screenPoint.y + textHeight / 4.0);
+
+ // 缁樺埗搴忓彿鏂囧瓧锛堟棤鑳屾櫙锛�
+ g2d.setColor(Color.BLACK);
+ g2d.drawString(numberText, textX, textY);
+
+ // 鎭㈠鍙樻崲
+ g2d.setTransform(savedTransform);
+ }
+ }
+
+ // 鎭㈠鍘熷鍙樻崲
+ g2d.setTransform(originalTransform);
}
/**
- * 鑾峰彇褰撳墠缂╂斁姣斾緥
+ * 缁樺埗杈圭晫鐐癸紙甯﹀簭鍙凤級
+ * 搴忓彿鏄剧ず鍦ㄧ偣涓績锛屽瓧浣撳ぇ灏忎笌闅滅鐗╁簭鍙蜂竴鑷达紙11鍙凤級锛屼笉闅忕缉鏀惧彉鍖�
*/
- public double getScale() {
- return scale;
+ private void drawBoundaryPointsWithNumbers(Graphics2D g2d, List<Point2D.Double> boundary, double scale) {
+ if (boundary == null || boundary.size() < 2) {
+ return;
+ }
+
+ // 淇濆瓨鍘熷鍙樻崲
+ AffineTransform originalTransform = g2d.getTransform();
+
+ int totalPoints = boundary.size();
+ boolean closed = totalPoints > 2 && areBoundaryPointsClose(boundary.get(0), boundary.get(totalPoints - 1));
+ int effectiveCount = closed ? totalPoints - 1 : totalPoints;
+ if (effectiveCount <= 0) {
+ return;
+ }
+
+ // 璁剧疆鐐圭殑澶у皬锛堜笌杈圭晫绾垮搴︿竴鑷达級
+ // 杈圭晫绾垮搴︼細3 / Math.max(0.5, scale)
+ double scaleFactor = Math.max(0.5, scale);
+ double markerDiameter = 3.0 / scaleFactor; // 涓庤竟鐣岀嚎瀹藉害涓�鑷�
+ double markerRadius = markerDiameter / 2.0;
+
+ // 璁剧疆瀛椾綋锛堜笌闅滅鐗╁簭鍙蜂竴鑷达紝涓嶉殢缂╂斁鍙樺寲锛�
+ Font labelFont = new Font("寰蒋闆呴粦", Font.PLAIN, 11);
+ g2d.setFont(labelFont);
+ FontMetrics fontMetrics = g2d.getFontMetrics(labelFont);
+
+ // 缁樺埗姣忎釜鐐瑰強鍏跺簭鍙�
+ for (int i = 0; i < effectiveCount; i++) {
+ Point2D.Double point = boundary.get(i);
+ double x = point.x;
+ double y = point.y;
+
+ // 缁樺埗鐐癸紙鍦ㄤ笘鐣屽潗鏍囩郴涓紝闅忕缉鏀惧彉鍖栵級
+ g2d.setColor(BOUNDARY_POINT_COLOR);
+ Ellipse2D.Double marker = new Ellipse2D.Double(
+ x - markerRadius,
+ y - markerRadius,
+ markerDiameter,
+ markerDiameter
+ );
+ g2d.fill(marker);
+
+ // 灏嗕笘鐣屽潗鏍囪浆鎹负灞忓箷鍧愭爣浠ョ粯鍒跺簭鍙凤紙涓嶉殢缂╂斁鍙樺寲锛�
+ Point2D.Double worldPoint = new Point2D.Double(x, y);
+ Point2D.Double screenPoint = new Point2D.Double();
+ originalTransform.transform(worldPoint, screenPoint);
+
+ // 淇濆瓨褰撳墠鍙樻崲
+ AffineTransform savedTransform = g2d.getTransform();
+
+ // 閲嶇疆鍙樻崲涓哄睆骞曞潗鏍囩郴缁�
+ g2d.setTransform(new AffineTransform());
+
+ // 缁樺埗搴忓彿锛堝湪灞忓箷鍧愭爣绯讳腑锛屼笉闅忕缉鏀惧彉鍖栵級
+ String numberText = String.valueOf(i + 1);
+ int textWidth = fontMetrics.stringWidth(numberText);
+ int textHeight = fontMetrics.getHeight();
+
+ // 鍦ㄧ偣涓績缁樺埗搴忓彿
+ int textX = (int)(screenPoint.x - textWidth / 2.0);
+ int textY = (int)(screenPoint.y + textHeight / 4.0);
+
+ // 缁樺埗搴忓彿鏂囧瓧锛堟棤鑳屾櫙锛�
+ g2d.setColor(Color.BLACK);
+ g2d.drawString(numberText, textX, textY);
+
+ // 鎭㈠鍙樻崲
+ g2d.setTransform(savedTransform);
+ }
+
+ // 鎭㈠鍘熷鍙樻崲
+ g2d.setTransform(originalTransform);
+ }
+
+ /**
+ * 妫�鏌ヤ袱涓竟鐣岀偣鏄惁鎺ヨ繎锛堢敤浜庡垽鏂竟鐣屾槸鍚﹂棴鍚堬級
+ */
+ private boolean areBoundaryPointsClose(Point2D.Double a, Point2D.Double b) {
+ if (a == null || b == null) {
+ return false;
+ }
+ double dx = a.x - b.x;
+ double dy = a.y - b.y;
+ return Math.hypot(dx, dy) <= BOUNDARY_POINT_MERGE_THRESHOLD;
+ }
+
+ private void drawViewInfo(Graphics2D g2d) {
+ g2d.setColor(new Color(80, 80, 80));
+ g2d.setFont(new Font("寰蒋闆呴粦", Font.PLAIN, 11));
+
+ // 鍦ㄥ湴鍥鹃《閮ㄥ乏渚ф樉绀洪仴鎺ф憞鏉嗗搴旈�熷害锛堣嫢闈為浂锛�
+ try {
+ int forward = Control03.getCurrentForwardSpeed();
+ int steer = Control03.getCurrentSteeringSpeed();
+ if (forward != 0 || steer != 0) {
+ String speedInfo = String.format("琛岃繘:%d 杞悜:%d", forward, steer);
+ // 鑳屾櫙鍗婇�忔槑鐭╁舰澧炲己鍙鎬�
+ FontMetrics fm = g2d.getFontMetrics();
+ int padding = 6;
+ int w = fm.stringWidth(speedInfo) + padding * 2;
+ int h = fm.getHeight() + padding;
+ int x = 12;
+ int y = 12;
+ Color bg = new Color(255, 255, 255, 180);
+ g2d.setColor(bg);
+ g2d.fillRoundRect(x, y, w, h, 8, 8);
+ g2d.setColor(new Color(120, 120, 120));
+ g2d.drawString(speedInfo, x + padding, y + fm.getAscent() + (padding/2));
+ }
+ } catch (Throwable t) {
+ // 涓嶅簲闃诲娓叉煋锛岄潤榛樺鐞嗕换浣曞紓甯�
+ }
+
+ // 淇濈暀搴曢儴鐨勭缉鏀炬瘮渚嬩俊鎭�
+ String info = String.format("缂╂斁: %.2fx", scale);
+ g2d.setColor(new Color(80, 80, 80));
+ g2d.drawString(info, 15, visualizationPanel.getHeight() - 15);
}
/**
@@ -1484,7 +1823,15 @@
* 璁剧疆瑙嗗浘鍙樻崲鍙傛暟锛堢敤浜庣▼搴忓寲鎺у埗锛�
*/
public void setViewTransform(double scale, double translateX, double translateY) {
- this.scale = scale;
+ // 闄愬埗缂╂斁鑼冨洿
+ scale = Math.max(MIN_SCALE, Math.min(scale, MAX_SCALE));
+ // 濡傛灉缂╂斁姣斾緥鏀瑰彉浜嗭紝淇濆瓨鍒伴厤缃枃浠�
+ if (Math.abs(this.scale - scale) > SCALE_EPSILON) {
+ this.scale = scale;
+ saveScaleToProperties();
+ } else {
+ this.scale = scale;
+ }
this.translateX = translateX;
this.translateY = translateY;
visualizationPanel.repaint();
@@ -1493,6 +1840,8 @@
public void setCurrentBoundary(String boundaryCoordinates, String landNumber, String landName) {
this.boundaryName = landName;
this.currentBoundaryLandNumber = landNumber;
+ this.realtimeBoundaryPathCache = null;
+ this.realtimeBoundaryPathLand = null;
if (boundaryCoordinates == null) {
clearBoundaryData();
@@ -1534,6 +1883,8 @@
boundaryPointsVisible = false;
currentBoundaryLandNumber = null;
pendingTrackBreak = true;
+ realtimeBoundaryPathCache = null;
+ realtimeBoundaryPathLand = null;
}
public void setCurrentObstacles(String obstaclesData, String landNumber) {
@@ -1734,6 +2085,7 @@
obstacleBounds = null;
selectedObstacleName = null;
currentObstacleLandNumber = null;
+ obstaclePointsVisible = false;
}
private List<Obstacledge.Obstacle> parseObstacles(String obstaclesData, String landNumber) {
@@ -1844,6 +2196,9 @@
return coords;
}
+ // Remove wrapper characters like parentheses that are used when persisting payloads
+ sanitized = sanitized.replace("(", "").replace(")", "");
+
String[] pairs = sanitized.split(";");
for (String pair : pairs) {
if (pair == null) {
@@ -1853,6 +2208,10 @@
if (trimmed.isEmpty()) {
continue;
}
+ trimmed = trimmed.replace("(", "").replace(")", "");
+ if (trimmed.isEmpty()) {
+ continue;
+ }
String[] parts = trimmed.split(",");
if (parts.length < 2) {
continue;
@@ -2014,6 +2373,87 @@
visualizationPanel.repaint();
}
+ public void setObstaclePointsVisible(boolean visible) {
+ this.obstaclePointsVisible = visible;
+ visualizationPanel.repaint();
+ }
+
+ /**
+ * 璁剧疆鏄惁鏄剧ず杈圭晫璺濈
+ */
+ public void setBoundaryLengthVisible(boolean visible) {
+ boundaryLengthVisible = visible;
+ if (visualizationPanel != null) {
+ visualizationPanel.repaint();
+ }
+ }
+
+ /**
+ * 鑾峰彇鏄惁鏄剧ず杈圭晫璺濈
+ */
+ public boolean isBoundaryLengthVisible() {
+ return boundaryLengthVisible;
+ }
+
+ public void setBoundaryPointSizeScale(double sizeScale) {
+ double normalized = (Double.isFinite(sizeScale) && sizeScale > 0.0d) ? sizeScale : 1.0d;
+ if (Math.abs(boundaryPointSizeScale - normalized) < 1e-6) {
+ return;
+ }
+ boundaryPointSizeScale = normalized;
+ if (visualizationPanel == null) {
+ return;
+ }
+ if (SwingUtilities.isEventDispatchThread()) {
+ visualizationPanel.repaint();
+ } else {
+ SwingUtilities.invokeLater(visualizationPanel::repaint);
+ }
+ }
+
+ public void setPathPreviewSizingEnabled(boolean enabled) {
+ previewSizingEnabled = enabled;
+ if (visualizationPanel == null) {
+ return;
+ }
+ if (SwingUtilities.isEventDispatchThread()) {
+ visualizationPanel.repaint();
+ } else {
+ SwingUtilities.invokeLater(visualizationPanel::repaint);
+ }
+ }
+
+ public void setBoundaryPreviewMarkerScale(double markerScale) {
+ double normalized = Double.isFinite(markerScale) && markerScale > 0.0d ? markerScale : 1.0d;
+ if (Math.abs(boundaryPreviewMarkerScale - normalized) < 1e-6) {
+ return;
+ }
+ boundaryPreviewMarkerScale = normalized;
+ if (visualizationPanel == null) {
+ return;
+ }
+ if (SwingUtilities.isEventDispatchThread()) {
+ visualizationPanel.repaint();
+ } else {
+ SwingUtilities.invokeLater(visualizationPanel::repaint);
+ }
+ }
+
+ public boolean setHandheldMowerIconActive(boolean handheldActive) {
+ if (mower == null) {
+ return false;
+ }
+ boolean changed = mower.useHandheldIcon(handheldActive);
+ if (changed && visualizationPanel != null) {
+ if (SwingUtilities.isEventDispatchThread()) {
+ visualizationPanel.repaint();
+ } else {
+ SwingUtilities.invokeLater(visualizationPanel::repaint);
+ }
+ }
+ return changed;
+ }
+
public void beginHandheldBoundaryPreview() {
handheldBoundaryPreviewActive = true;
handheldBoundaryPreview.clear();
@@ -2043,6 +2483,7 @@
public void clearHandheldBoundaryPreview() {
handheldBoundaryPreviewActive = false;
handheldBoundaryPreview.clear();
+ boundaryPreviewMarkerScale = 1.0d;
visualizationPanel.repaint();
}
@@ -2098,8 +2539,10 @@
return;
}
- double width = Math.max(bounds.width, 1);
- double height = Math.max(bounds.height, 1);
+ Rectangle2D.Double targetBounds = includeMowerInBounds(bounds);
+
+ double width = Math.max(targetBounds.width, 1);
+ double height = Math.max(targetBounds.height, 1);
double targetWidth = width * 1.2;
double targetHeight = height * 1.2;
@@ -2111,18 +2554,62 @@
newScale = Math.max(0.05, Math.min(newScale, 50.0));
this.scale = newScale;
- this.translateX = -bounds.getCenterX();
- this.translateY = -bounds.getCenterY();
+ this.translateX = -targetBounds.getCenterX();
+ this.translateY = -targetBounds.getCenterY();
+ }
+
+ // Keep the mower marker inside the viewport whenever the camera refits to scene bounds.
+ private Rectangle2D.Double includeMowerInBounds(Rectangle2D.Double bounds) {
+ Rectangle2D.Double expanded = new Rectangle2D.Double(
+ bounds.x,
+ bounds.y,
+ Math.max(0.0, bounds.width),
+ Math.max(0.0, bounds.height)
+ );
+
+ if (mower == null || !mower.hasValidPosition()) {
+ return expanded;
+ }
+
+ Point2D.Double mowerPosition = mower.getPosition();
+ if (mowerPosition == null
+ || !Double.isFinite(mowerPosition.x)
+ || !Double.isFinite(mowerPosition.y)) {
+ return expanded;
+ }
+
+ double minX = Math.min(expanded.x, mowerPosition.x);
+ double minY = Math.min(expanded.y, mowerPosition.y);
+ double maxX = Math.max(expanded.x + expanded.width, mowerPosition.x);
+ double maxY = Math.max(expanded.y + expanded.height, mowerPosition.y);
+
+ expanded.x = minX;
+ expanded.y = minY;
+ expanded.width = Math.max(0.0, maxX - minX);
+ expanded.height = Math.max(0.0, maxY - minY);
+
+ return expanded;
}
public void dispose() {
mowerUpdateTimer.stop();
- stopMowerInfoTimer();
- if (mowerInfoDialog != null) {
- mowerInfoDialog.dispose();
- mowerInfoDialog = null;
- }
- resetMowerInfoLabels();
+ mowerInfoManager.dispose();
+ }
+
+ /**
+ * 鑾峰彇褰撳墠杈圭晫鐐瑰垪琛�
+ * @return 褰撳墠杈圭晫鐐瑰垪琛紝濡傛灉娌℃湁杈圭晫鍒欒繑鍥瀗ull
+ */
+ public List<Point2D.Double> getCurrentBoundary() {
+ return currentBoundary;
+ }
+
+ /**
+ * 鑾峰彇鍓茶崏鏈哄疄渚�
+ * @return 鍓茶崏鏈哄疄渚�
+ */
+ public Gecaoji getMower() {
+ return mower;
}
}
\ No newline at end of file
--
Gitblit v1.10.0