| | |
| | | import zhangaiwu.Obstacledge; |
| | | import zhangaiwu.yulanzhangaiwu; |
| | | import yaokong.Control03; |
| | | import bianjie.shudongdraw; |
| | | |
| | | /** |
| | | * 地图渲染器 - 负责坐标系绘制、视图变换等功能 |
| | |
| | | private CircleCaptureOverlay circleCaptureOverlay; |
| | | private final List<double[]> circleSampleMarkers = new ArrayList<>(); |
| | | private final List<Point2D.Double> realtimeMowingTrack = new ArrayList<>(); |
| | | private final List<Point2D.Double> navigationPreviewTrack = new ArrayList<>(); // 导航预览轨迹 |
| | | private final Deque<tuowei.TrailSample> idleMowerTrail = new ArrayDeque<>(); |
| | | private final List<Point2D.Double> handheldBoundaryPreview = new ArrayList<>(); |
| | | private double boundaryPreviewMarkerScale = 1.0d; |
| | |
| | | private double mowingCompletionRatio; |
| | | private long lastTrackPersistTimeMillis; |
| | | private boolean trackDirty; |
| | | private boolean measurementModeActive = false; // 测量模式是否激活 |
| | | private boolean handheldBoundaryPreviewActive; |
| | | private boolean pendingTrackBreak = true; |
| | | private bianjie.shudongdraw manualBoundaryDrawer = new bianjie.shudongdraw(); // 手动绘制边界绘制器 |
| | | private boolean idleTrailSuppressed; |
| | | private Path2D.Double realtimeBoundaryPathCache; |
| | | private String realtimeBoundaryPathLand; |
| | | private WangfanDraw returnPathDrawer; // 往返路径绘制管理器 |
| | | private List<Point2D.Double> currentReturnPath; // 当前地块的往返路径(用于显示) |
| | | private List<Point2D.Double> previewReturnPath; // 预览的往返路径 |
| | | private List<Point2D.Double> previewOriginalBoundary; // 预览的原始边界(紫色) |
| | | private List<Point2D.Double> previewOptimizedBoundary; // 预览的优化后边界 |
| | | private boolean boundaryPreviewActive; // 是否处于边界预览模式 |
| | | |
| | | private static final double TRACK_SAMPLE_MIN_DISTANCE_METERS = 0.2d; |
| | | private static final double TRACK_DUPLICATE_TOLERANCE_METERS = 1e-3d; |
| | |
| | | lastDragPoint = null; |
| | | dragInProgress = false; |
| | | } |
| | | |
| | | public void mouseExited(MouseEvent e) { |
| | | // 鼠标离开面板时,清除鼠标位置显示 |
| | | if (manualBoundaryDrawer.isManualBoundaryDrawingMode()) { |
| | | manualBoundaryDrawer.clearMousePosition(); |
| | | visualizationPanel.repaint(); |
| | | } |
| | | } |
| | | |
| | | public void mouseClicked(MouseEvent e) { |
| | | if (dragInProgress) { |
| | |
| | | if (!SwingUtilities.isLeftMouseButton(e) || e.getClickCount() != 1) { |
| | | return; |
| | | } |
| | | // 优先处理手动绘制边界模式点击 |
| | | if (manualBoundaryDrawer.isManualBoundaryDrawingMode()) { |
| | | Point2D.Double worldPoint = screenToWorld(e.getPoint()); |
| | | if (manualBoundaryDrawer.handleClick(worldPoint)) { |
| | | visualizationPanel.repaint(); |
| | | return; |
| | | } |
| | | } |
| | | // 优先处理测量模式点击 |
| | | if (measurementModeActive && handleMeasurementClick(e.getPoint())) { |
| | | return; |
| | | } |
| | | if (handleMowerClick(e.getPoint())) { |
| | | return; |
| | | } |
| | | // 优先处理优化后边界坐标点点击(边界预览模式下) |
| | | if (boundaryPreviewActive && handleOptimizedBoundaryPointClick(e.getPoint())) { |
| | | return; |
| | | } |
| | | // 优先处理障碍物边界点点击(如果可见) |
| | | if (obstaclePointsVisible && handleObstaclePointClick(e.getPoint())) { |
| | | return; |
| | |
| | | visualizationPanel.repaint(); |
| | | } |
| | | } |
| | | |
| | | public void mouseMoved(MouseEvent e) { |
| | | // 在手动绘制边界模式时,更新鼠标位置 |
| | | if (manualBoundaryDrawer.isManualBoundaryDrawingMode()) { |
| | | Point2D.Double worldPoint = screenToWorld(e.getPoint()); |
| | | manualBoundaryDrawer.updateMousePosition(worldPoint); |
| | | visualizationPanel.repaint(); |
| | | } else { |
| | | manualBoundaryDrawer.clearMousePosition(); |
| | | } |
| | | } |
| | | }); |
| | | } |
| | | |
| | |
| | | if (hasBoundary) { |
| | | drawCurrentBoundary(g2d); |
| | | } |
| | | |
| | | // 绘制边界预览(原始边界-紫色,优化后边界) |
| | | if (boundaryPreviewActive) { |
| | | drawBoundaryPreview(g2d); |
| | | } |
| | | |
| | | yulanzhangaiwu.renderPreview(g2d, scale); |
| | | |
| | |
| | | |
| | | adddikuaiyulan.drawPreview(g2d, handheldBoundaryPreview, scale, handheldBoundaryPreviewActive, boundaryPreviewMarkerScale); |
| | | |
| | | // 绘制导航路径(中层) |
| | | if (hasPlannedPath) { |
| | | // 绘制手动绘制的边界 |
| | | manualBoundaryDrawer.drawBoundary(g2d, scale); |
| | | |
| | | // 绘制鼠标实时位置(手动绘制边界模式时) |
| | | manualBoundaryDrawer.drawMousePosition(g2d, scale); |
| | | |
| | | // 绘制导航路径(中层)- 边界预览模式下不显示导航路径 |
| | | if (hasPlannedPath && !boundaryPreviewActive) { |
| | | drawCurrentPlannedPath(g2d); |
| | | } |
| | | |
| | |
| | | if (!realtimeMowingTrack.isEmpty()) { |
| | | drawRealtimeMowingCoverage(g2d); |
| | | } |
| | | |
| | | // 绘制导航预览已割区域 |
| | | if (!navigationPreviewTrack.isEmpty()) { |
| | | drawNavigationPreviewCoverage(g2d); |
| | | } |
| | | |
| | | // 先画往返路径(线+点),保证割草机图标在其上方 |
| | | if (returnPathDrawer != null && returnPathDrawer.isActive()) { |
| | | returnPathDrawer.draw(g2d, scale); |
| | | } else if (previewReturnPath != null && !previewReturnPath.isEmpty()) { |
| | | // 绘制预览的往返路径(铁线路图风格) |
| | | WangfanDraw.drawRailwayPath(g2d, previewReturnPath, scale); |
| | | } else if (currentReturnPath != null && !currentReturnPath.isEmpty()) { |
| | | // 绘制保存的往返路径(铁线路图风格) |
| | | WangfanDraw.drawRailwayPath(g2d, currentReturnPath, scale); |
| | | } |
| | | |
| | | drawMower(g2d); |
| | | |
| | | // 绘制导航预览速度(如果正在导航预览) |
| | | if (navigationPreviewSpeed > 0 && mower != null && mower.hasValidPosition()) { |
| | | drawNavigationPreviewSpeed(g2d, scale); |
| | | } |
| | | |
| | | // 绘制测量模式(如果激活) |
| | | if (measurementModeActive) { |
| | | drawMeasurementMode(g2d, scale); |
| | | } |
| | | |
| | | // 保存当前变换(包含视图变换)用于坐标转换 |
| | | AffineTransform currentTransformForLength = g2d.getTransform(); |
| | | |
| | |
| | | private void drawMower(Graphics2D g2d) { |
| | | mower.draw(g2d, scale); |
| | | } |
| | | |
| | | /** |
| | | * 绘制导航预览速度(在割草机图标上方) |
| | | */ |
| | | private void drawNavigationPreviewSpeed(Graphics2D g2d, double scale) { |
| | | if (mower == null || !mower.hasValidPosition()) { |
| | | return; |
| | | } |
| | | |
| | | Point2D.Double mowerPos = mower.getPosition(); |
| | | if (mowerPos == null) { |
| | | return; |
| | | } |
| | | |
| | | // 将速度从米/秒转换为KM/h |
| | | double speedKmh = navigationPreviewSpeed * 3.6; |
| | | String speedText = String.format("%.1f km/h", speedKmh); |
| | | |
| | | // 保存原始变换 |
| | | AffineTransform originalTransform = g2d.getTransform(); |
| | | |
| | | // 将世界坐标转换为屏幕坐标 |
| | | Point2D.Double screenPos = worldToScreen(mowerPos); |
| | | |
| | | // 恢复原始变换以绘制文字(固定大小,不随缩放变化) |
| | | g2d.setTransform(new AffineTransform()); |
| | | |
| | | // 设置字体(与缩放文字大小一致,11号字体) |
| | | Font labelFont = new Font("微软雅黑", Font.PLAIN, 11); |
| | | g2d.setFont(labelFont); |
| | | FontMetrics metrics = g2d.getFontMetrics(labelFont); |
| | | |
| | | // 计算文字位置(在割草机图标上方) |
| | | int textWidth = metrics.stringWidth(speedText); |
| | | int textHeight = metrics.getHeight(); |
| | | int textX = (int)Math.round(screenPos.x - textWidth / 2.0); |
| | | // 在割草机图标上方,留出一定间距 |
| | | // 图标在世界坐标系中的大小约为 48 * 0.8 / scale 米 |
| | | // 转换为屏幕像素:图标高度(像素)= (48 * 0.8 / scale) * scale = 48 * 0.8 = 38.4 像素 |
| | | double iconSizePixels = 48.0 * 0.8; // 图标在屏幕上的大小(像素) |
| | | int spacing = 8; // 间距(像素) |
| | | int textY = (int)Math.round(screenPos.y - iconSizePixels / 2.0 - spacing - textHeight); |
| | | |
| | | // 绘制文字背景(半透明白色,增强可读性) |
| | | g2d.setColor(new Color(255, 255, 255, 200)); |
| | | g2d.fillRoundRect(textX - 4, textY - metrics.getAscent() - 2, textWidth + 8, textHeight + 4, 4, 4); |
| | | |
| | | // 绘制文字 |
| | | g2d.setColor(new Color(46, 139, 87)); // 使用主题绿色 |
| | | g2d.drawString(speedText, textX, textY); |
| | | |
| | | // 恢复变换 |
| | | g2d.setTransform(originalTransform); |
| | | } |
| | | |
| | | private void drawRealtimeMowingCoverage(Graphics2D g2d) { |
| | | if (realtimeMowingTrack == null || realtimeMowingTrack.size() < 2) { |
| | |
| | | double effectiveWidth = getEffectiveMowerWidthMeters(); |
| | | gecaolunjing.draw(g2d, realtimeMowingTrack, effectiveWidth, boundaryPath); |
| | | } |
| | | |
| | | /** |
| | | * 绘制导航预览已割区域 |
| | | */ |
| | | private void drawNavigationPreviewCoverage(Graphics2D g2d) { |
| | | if (navigationPreviewTrack == null || navigationPreviewTrack.size() < 2) { |
| | | return; |
| | | } |
| | | |
| | | Path2D.Double boundaryPath = currentBoundaryPath; |
| | | // 获取导航预览的割草宽度(从daohangyulan获取) |
| | | double previewWidth = getNavigationPreviewWidth(); |
| | | if (previewWidth <= 0) { |
| | | previewWidth = 0.5; // 默认50厘米 |
| | | } |
| | | gecaolunjing.draw(g2d, navigationPreviewTrack, previewWidth, boundaryPath); |
| | | } |
| | | |
| | | /** |
| | | * 设置导航预览轨迹 |
| | | */ |
| | | public void setNavigationPreviewTrack(List<Point2D.Double> track) { |
| | | if (track == null) { |
| | | navigationPreviewTrack.clear(); |
| | | } else { |
| | | navigationPreviewTrack.clear(); |
| | | navigationPreviewTrack.addAll(track); |
| | | } |
| | | if (visualizationPanel != null) { |
| | | visualizationPanel.repaint(); |
| | | } |
| | | } |
| | | |
| | | /** |
| | | * 添加导航预览轨迹点 |
| | | */ |
| | | public void addNavigationPreviewTrackPoint(Point2D.Double point) { |
| | | if (point != null && Double.isFinite(point.x) && Double.isFinite(point.y)) { |
| | | navigationPreviewTrack.add(new Point2D.Double(point.x, point.y)); |
| | | if (visualizationPanel != null) { |
| | | visualizationPanel.repaint(); |
| | | } |
| | | } |
| | | } |
| | | |
| | | /** |
| | | * 清除导航预览轨迹 |
| | | */ |
| | | public void clearNavigationPreviewTrack() { |
| | | navigationPreviewTrack.clear(); |
| | | if (visualizationPanel != null) { |
| | | visualizationPanel.repaint(); |
| | | } |
| | | } |
| | | |
| | | private double navigationPreviewWidth = 0.5; // 导航预览的割草宽度(米) |
| | | private double navigationPreviewSpeed = 0.0; // 导航预览的割草机速度(米/秒) |
| | | |
| | | /** |
| | | * 设置导航预览的割草宽度 |
| | | */ |
| | | public void setNavigationPreviewWidth(double widthMeters) { |
| | | navigationPreviewWidth = widthMeters > 0 ? widthMeters : 0.5; |
| | | } |
| | | |
| | | /** |
| | | * 获取导航预览的割草宽度 |
| | | */ |
| | | private double getNavigationPreviewWidth() { |
| | | return navigationPreviewWidth; |
| | | } |
| | | |
| | | /** |
| | | * 设置导航预览的割草机速度(米/秒) |
| | | */ |
| | | public void setNavigationPreviewSpeed(double speedMetersPerSecond) { |
| | | navigationPreviewSpeed = speedMetersPerSecond >= 0 ? speedMetersPerSecond : 0.0; |
| | | } |
| | | |
| | | /** |
| | | * 获取导航预览的割草机速度(米/秒) |
| | | */ |
| | | private double getNavigationPreviewSpeed() { |
| | | return navigationPreviewSpeed; |
| | | } |
| | | |
| | | private Path2D.Double getRealtimeBoundaryPath() { |
| | | if (realtimeTrackLandNumber == null) { |
| | |
| | | mowerEffectiveWidthMeters = defaultMowerWidthMeters; |
| | | } |
| | | |
| | | // 加载往返路径 |
| | | String returnPathStr = dikuai != null ? dikuai.getReturnPathCoordinates() : null; |
| | | if (returnPathStr != null && !returnPathStr.isEmpty() && !"-1".equals(returnPathStr)) { |
| | | currentReturnPath = lujingdraw.parsePlannedPath(returnPathStr); |
| | | } else { |
| | | currentReturnPath = null; |
| | | } |
| | | |
| | | loadRealtimeTrack(landNumber, dikuai != null ? dikuai.getMowingTrack() : null); |
| | | visualizationPanel.repaint(); |
| | | } |
| | |
| | | double worldY = (screenPoint.y - visualizationPanel.getHeight() / 2.0) / scale - translateY; |
| | | return new Point2D.Double(worldX, worldY); |
| | | } |
| | | |
| | | /** |
| | | * 处理测量模式点击 |
| | | */ |
| | | private boolean handleMeasurementClick(Point screenPoint) { |
| | | if (!measurementModeActive) { |
| | | return false; |
| | | } |
| | | Point2D.Double worldPoint = screenToWorld(screenPoint); |
| | | celiangmoshi.addPoint(worldPoint); |
| | | visualizationPanel.repaint(); |
| | | return true; |
| | | } |
| | | |
| | | /** |
| | | * 设置手动绘制边界模式 |
| | | */ |
| | | public void setManualBoundaryDrawingMode(boolean active) { |
| | | manualBoundaryDrawer.setManualBoundaryDrawingMode(active); |
| | | if (visualizationPanel != null) { |
| | | visualizationPanel.repaint(); |
| | | } |
| | | } |
| | | |
| | | /** |
| | | * 获取手动绘制的边界点列表 |
| | | */ |
| | | public List<Point2D.Double> getManualBoundaryPoints() { |
| | | return manualBoundaryDrawer.getManualBoundaryPoints(); |
| | | } |
| | | |
| | | /** |
| | | * 清空手动绘制的边界点 |
| | | */ |
| | | public void clearManualBoundaryPoints() { |
| | | manualBoundaryDrawer.clearManualBoundaryPoints(); |
| | | if (visualizationPanel != null) { |
| | | visualizationPanel.repaint(); |
| | | } |
| | | } |
| | | |
| | | /** |
| | | * 设置测量模式 |
| | | */ |
| | | public void setMeasurementMode(boolean active) { |
| | | measurementModeActive = active; |
| | | if (!active) { |
| | | celiangmoshi.clear(); |
| | | } |
| | | if (visualizationPanel != null) { |
| | | visualizationPanel.repaint(); |
| | | } |
| | | } |
| | | |
| | | /** |
| | | * 触发地图重绘 |
| | | */ |
| | | public void repaint() { |
| | | if (visualizationPanel != null) { |
| | | visualizationPanel.repaint(); |
| | | } |
| | | } |
| | | |
| | | /** |
| | | * 绘制测量模式 |
| | | */ |
| | | private void drawMeasurementMode(Graphics2D g2d, double scale) { |
| | | List<Point2D.Double> points = celiangmoshi.getPoints(); |
| | | if (points.isEmpty()) { |
| | | return; |
| | | } |
| | | |
| | | // 保存原始变换 |
| | | AffineTransform originalTransform = g2d.getTransform(); |
| | | |
| | | // 设置测量模式颜色 |
| | | Color lineColor = new Color(255, 0, 0, 200); // 红色半透明 |
| | | Color pointColor = new Color(255, 0, 0, 255); // 红色 |
| | | Color textColor = new Color(33, 37, 41, 220); // 深灰色文字 |
| | | |
| | | // 设置线宽和点的大小(确保点和线连接) |
| | | float lineWidth = (float)(2.0 / scale); |
| | | // 点的大小(在世界坐标系中,米),确保点足够大以覆盖线的端点 |
| | | double pointSizeInWorld = 0.15d; // 点的大小(米) |
| | | double halfSize = pointSizeInWorld / 2.0; |
| | | |
| | | // 先绘制所有测量点(作为基础层) |
| | | g2d.setColor(pointColor); |
| | | for (Point2D.Double point : points) { |
| | | // 点的中心在 point.x, point.y |
| | | Ellipse2D.Double pointShape = new Ellipse2D.Double( |
| | | point.x - halfSize, |
| | | point.y - halfSize, |
| | | pointSizeInWorld, |
| | | pointSizeInWorld |
| | | ); |
| | | g2d.fill(pointShape); |
| | | } |
| | | |
| | | // 然后绘制连线,确保线从点的中心开始和结束,点和线连接在一起 |
| | | g2d.setColor(lineColor); |
| | | g2d.setStroke(new BasicStroke(lineWidth, BasicStroke.CAP_ROUND, BasicStroke.JOIN_ROUND)); |
| | | |
| | | // 绘制连线和距离文字(显示两个相邻点连线的长度) |
| | | for (int i = 0; i < points.size() - 1; i++) { |
| | | Point2D.Double p1 = points.get(i); |
| | | Point2D.Double p2 = points.get(i + 1); |
| | | |
| | | // 绘制连线(从第一个点的中心到第二个点的中心) |
| | | // 使用 Path2D 确保精确绘制,保持浮点精度 |
| | | Path2D.Double linePath = new Path2D.Double(); |
| | | linePath.moveTo(p1.x, p1.y); |
| | | linePath.lineTo(p2.x, p2.y); |
| | | g2d.draw(linePath); |
| | | |
| | | // 计算距离(两个相邻点连线的长度) |
| | | double distance = celiangmoshi.calculateDistance(p1, p2); |
| | | String distanceText = celiangmoshi.formatDistance(distance); |
| | | |
| | | // 计算中点位置(用于显示文字) |
| | | double midX = (p1.x + p2.x) / 2.0; |
| | | double midY = (p1.y + p2.y) / 2.0; |
| | | |
| | | // 将世界坐标转换为屏幕坐标(用于文字显示) |
| | | Point2D.Double worldMid = new Point2D.Double(midX, midY); |
| | | Point2D.Double screenMid = worldToScreen(worldMid); |
| | | |
| | | // 恢复原始变换以绘制文字(固定大小,不随缩放变化) |
| | | g2d.setTransform(new AffineTransform()); |
| | | |
| | | // 设置字体(与缩放文字大小一致,11号字体) |
| | | Font labelFont = new Font("微软雅黑", Font.PLAIN, 11); |
| | | g2d.setFont(labelFont); |
| | | FontMetrics metrics = g2d.getFontMetrics(labelFont); |
| | | |
| | | // 计算文字位置(居中显示) |
| | | int textWidth = metrics.stringWidth(distanceText); |
| | | int textHeight = metrics.getHeight(); |
| | | int textX = (int)Math.round(screenMid.x - textWidth / 2.0); |
| | | int textY = (int)Math.round(screenMid.y - textHeight / 2.0) + metrics.getAscent(); |
| | | |
| | | // 绘制文字背景(可选,用于提高可读性) |
| | | g2d.setColor(new Color(255, 255, 255, 200)); |
| | | g2d.fillRoundRect(textX - 2, textY - metrics.getAscent() - 2, textWidth + 4, textHeight + 4, 4, 4); |
| | | |
| | | // 绘制文字 |
| | | g2d.setColor(textColor); |
| | | g2d.drawString(distanceText, textX, textY); |
| | | |
| | | // 恢复变换 |
| | | g2d.setTransform(originalTransform); |
| | | } |
| | | |
| | | // 最后再次绘制测量点(在连线之上,确保点覆盖在线的端点上,点和线连接在一起) |
| | | g2d.setColor(pointColor); |
| | | for (Point2D.Double point : points) { |
| | | // 点的中心在 point.x, point.y,正好是线的端点位置 |
| | | Ellipse2D.Double pointShape = new Ellipse2D.Double( |
| | | point.x - halfSize, |
| | | point.y - halfSize, |
| | | pointSizeInWorld, |
| | | pointSizeInWorld |
| | | ); |
| | | g2d.fill(pointShape); |
| | | } |
| | | } |
| | | |
| | | private void drawCurrentBoundary(Graphics2D g2d) { |
| | | bianjiedrwa.drawBoundary(g2d, currentBoundary, scale, GRASS_FILL_COLOR, GRASS_BORDER_COLOR); |
| | |
| | | if (markers == null || markers.isEmpty()) { |
| | | return; |
| | | } |
| | | |
| | | // 保存原始变换 |
| | | AffineTransform originalTransform = g2d.getTransform(); |
| | | |
| | | Shape markerShape; |
| | | double half = CIRCLE_SAMPLE_SIZE / 2.0; |
| | | g2d.setColor(CIRCLE_SAMPLE_COLOR); |
| | | g2d.setStroke(new BasicStroke((float) (1.2f / scale), BasicStroke.CAP_ROUND, BasicStroke.JOIN_ROUND)); |
| | | Font originalFont = g2d.getFont(); |
| | | float baseSize = (float) Math.max(12f / scale, 9f); |
| | | float reducedSize = Math.max(baseSize / 3f, 3f); |
| | | Font labelFont = originalFont.deriveFont(reducedSize); |
| | | |
| | | // 设置字体(与缩放文字大小一致,11号字体,不随缩放变化) |
| | | Font labelFont = new Font("微软雅黑", Font.PLAIN, 11); |
| | | g2d.setFont(labelFont); |
| | | FontMetrics metrics = g2d.getFontMetrics(); |
| | | FontMetrics metrics = g2d.getFontMetrics(labelFont); |
| | | |
| | | for (double[] pt : markers) { |
| | | if (pt == null || pt.length < 2 || !Double.isFinite(pt[0]) || !Double.isFinite(pt[1])) { |
| | | continue; |
| | | } |
| | | double x = pt[0]; |
| | | double y = pt[1]; |
| | | |
| | | // 绘制点(在世界坐标系中,随缩放变化) |
| | | markerShape = new Ellipse2D.Double(x - half, y - half, CIRCLE_SAMPLE_SIZE, CIRCLE_SAMPLE_SIZE); |
| | | g2d.fill(markerShape); |
| | | |
| | | // 将世界坐标转换为屏幕坐标以绘制文字(不随缩放变化) |
| | | Point2D.Double worldPoint = new Point2D.Double(x, y); |
| | | Point2D.Double screenPoint = new Point2D.Double(); |
| | | originalTransform.transform(worldPoint, screenPoint); |
| | | |
| | | // 恢复原始变换以使用屏幕坐标绘制文字 |
| | | g2d.setTransform(new AffineTransform()); |
| | | |
| | | String label = String.format(Locale.US, "%.2f,%.2f", x, y); |
| | | int textWidth = metrics.stringWidth(label); |
| | | float textX = (float) (x - textWidth / 2.0); |
| | | float textY = (float) (y - half - 0.2d) - metrics.getDescent(); |
| | | int textHeight = metrics.getHeight(); |
| | | |
| | | // 在屏幕坐标系中绘制文字(不随缩放变化) |
| | | int textX = (int)(screenPoint.x - textWidth / 2.0); |
| | | int textY = (int)(screenPoint.y - half - 0.2d) - metrics.getDescent(); |
| | | g2d.setColor(new Color(33, 37, 41, 220)); |
| | | g2d.drawString(label, textX, textY); |
| | | |
| | | // 恢复原始变换 |
| | | g2d.setTransform(originalTransform); |
| | | g2d.setColor(CIRCLE_SAMPLE_COLOR); |
| | | } |
| | | g2d.setFont(originalFont); |
| | | } |
| | | |
| | | private void drawCircleCaptureOverlay(Graphics2D g2d, CircleCaptureOverlay overlay, double scale) { |
| | |
| | | return; |
| | | } |
| | | |
| | | // 设置点的大小(与边界线宽度一致) |
| | | // 设置点的大小(边界线宽度的2倍) |
| | | // 边界线宽度:3 / Math.max(0.5, scale) |
| | | double scaleFactor = Math.max(0.5, scale); |
| | | double markerDiameter = 3.0 / scaleFactor; // 与边界线宽度一致 |
| | | double boundaryLineWidth = 3.0 / scaleFactor; // 边界线宽度 |
| | | double markerDiameter = boundaryLineWidth * 2.0; // 边界点直径 = 边界线宽度的2倍 |
| | | double markerRadius = markerDiameter / 2.0; |
| | | |
| | | // 设置字体(与障碍物序号一致,不随缩放变化) |
| | |
| | | public Gecaoji getMower() { |
| | | return mower; |
| | | } |
| | | |
| | | /** |
| | | * 设置往返路径绘制管理器 |
| | | */ |
| | | public void setReturnPathDrawer(WangfanDraw drawer) { |
| | | this.returnPathDrawer = drawer; |
| | | } |
| | | |
| | | /** |
| | | * 设置预览的往返路径 |
| | | */ |
| | | public void setPreviewReturnPath(List<Point2D.Double> path) { |
| | | this.previewReturnPath = path; |
| | | if (visualizationPanel != null) { |
| | | visualizationPanel.repaint(); |
| | | } |
| | | } |
| | | |
| | | /** |
| | | * 开始往返路径绘制 |
| | | */ |
| | | public void startReturnPathDrawing() { |
| | | if (returnPathDrawer != null) { |
| | | // 禁用拖尾效果(在往返路径绘制模式下不显示实时轨迹拖尾) |
| | | idleTrailSuppressed = true; |
| | | clearIdleMowerTrail(); |
| | | // 清空之前的路径点(通过 WangfanDraw 管理) |
| | | repaint(); |
| | | } |
| | | } |
| | | |
| | | /** |
| | | * 停止往返路径绘制 |
| | | */ |
| | | public void stopReturnPathDrawing() { |
| | | // 恢复拖尾效果 |
| | | idleTrailSuppressed = false; |
| | | repaint(); |
| | | } |
| | | |
| | | /** |
| | | * 设置拖尾抑制状态 |
| | | * @param suppressed true表示抑制拖尾绘制,false表示允许拖尾绘制 |
| | | */ |
| | | public void setIdleTrailSuppressed(boolean suppressed) { |
| | | idleTrailSuppressed = suppressed; |
| | | if (suppressed && !idleMowerTrail.isEmpty()) { |
| | | clearIdleMowerTrail(); |
| | | } |
| | | if (visualizationPanel != null) { |
| | | visualizationPanel.repaint(); |
| | | } |
| | | } |
| | | |
| | | /** |
| | | * 添加往返路径点(已废弃,路径点由 WangfanDraw 直接管理) |
| | | */ |
| | | @Deprecated |
| | | public void addReturnPathPoint(double x, double y) { |
| | | // 路径点由 WangfanDraw 直接管理,这里只需要重绘 |
| | | repaint(); |
| | | } |
| | | |
| | | /** |
| | | * 获取往返路径点列表的快照 |
| | | */ |
| | | public List<Point2D.Double> getReturnPathPointsSnapshot() { |
| | | if (returnPathDrawer != null) { |
| | | return returnPathDrawer.getPointsSnapshot(); |
| | | } |
| | | return new ArrayList<>(); |
| | | } |
| | | |
| | | /** |
| | | * 设置边界预览数据(原始边界和优化后边界) |
| | | */ |
| | | public void setBoundaryPreview(String originalBoundaryXY, String optimizedBoundary) { |
| | | if (originalBoundaryXY != null && !originalBoundaryXY.trim().isEmpty() && !"-1".equals(originalBoundaryXY.trim())) { |
| | | previewOriginalBoundary = parseBoundary(originalBoundaryXY.trim()); |
| | | } else { |
| | | previewOriginalBoundary = null; |
| | | } |
| | | |
| | | if (optimizedBoundary != null && !optimizedBoundary.trim().isEmpty() && !"-1".equals(optimizedBoundary.trim())) { |
| | | previewOptimizedBoundary = parseBoundary(optimizedBoundary.trim()); |
| | | } else { |
| | | previewOptimizedBoundary = null; |
| | | } |
| | | |
| | | boundaryPreviewActive = (previewOriginalBoundary != null && previewOriginalBoundary.size() >= 2) || |
| | | (previewOptimizedBoundary != null && previewOptimizedBoundary.size() >= 2); |
| | | |
| | | if (boundaryPreviewActive) { |
| | | // 计算预览边界的边界框并调整视图 |
| | | List<Point2D.Double> allPoints = new ArrayList<>(); |
| | | if (previewOriginalBoundary != null) allPoints.addAll(previewOriginalBoundary); |
| | | if (previewOptimizedBoundary != null) allPoints.addAll(previewOptimizedBoundary); |
| | | if (!allPoints.isEmpty()) { |
| | | Rectangle2D.Double bounds = computeBounds(allPoints); |
| | | SwingUtilities.invokeLater(() -> { |
| | | fitBoundsToView(bounds); |
| | | visualizationPanel.repaint(); |
| | | }); |
| | | } |
| | | } else { |
| | | visualizationPanel.repaint(); |
| | | } |
| | | } |
| | | |
| | | /** |
| | | * 清除边界预览 |
| | | */ |
| | | public void clearBoundaryPreview() { |
| | | previewOriginalBoundary = null; |
| | | previewOptimizedBoundary = null; |
| | | boundaryPreviewActive = false; |
| | | visualizationPanel.repaint(); |
| | | } |
| | | |
| | | /** |
| | | * 绘制边界预览(原始边界-紫色,优化后边界-绿色) |
| | | */ |
| | | private void drawBoundaryPreview(Graphics2D g2d) { |
| | | // 绘制原始边界(紫色) |
| | | if (previewOriginalBoundary != null && previewOriginalBoundary.size() >= 2) { |
| | | Color purpleFill = new Color(128, 0, 128, 80); // 紫色半透明填充 |
| | | Color purpleBorder = new Color(128, 0, 128, 255); // 紫色边框 |
| | | bianjiedrwa.drawBoundary(g2d, previewOriginalBoundary, scale, purpleFill, purpleBorder); |
| | | } |
| | | |
| | | // 绘制优化后边界(绿色,与正常边界颜色一致) |
| | | if (previewOptimizedBoundary != null && previewOptimizedBoundary.size() >= 2) { |
| | | bianjiedrwa.drawBoundary(g2d, previewOptimizedBoundary, scale, GRASS_FILL_COLOR, GRASS_BORDER_COLOR); |
| | | |
| | | // 绘制优化后边界坐标点(紫色实心圆圈,显示序号) |
| | | drawOptimizedBoundaryPointsWithNumbers(g2d, previewOptimizedBoundary, scale); |
| | | } |
| | | } |
| | | |
| | | /** |
| | | * 绘制优化后边界坐标点(紫色实心圆圈,显示序号) |
| | | * 序号显示在点中心,字体大小11号,不随缩放变化 |
| | | */ |
| | | private void drawOptimizedBoundaryPointsWithNumbers(Graphics2D g2d, List<Point2D.Double> boundary, double scale) { |
| | | if (boundary == null || boundary.isEmpty()) { |
| | | return; |
| | | } |
| | | |
| | | // 保存原始变换 |
| | | AffineTransform originalTransform = g2d.getTransform(); |
| | | |
| | | // 设置点的大小(实心圆圈,直径约0.3米) |
| | | double scaleFactor = Math.max(0.5, scale); |
| | | double markerDiameter = 0.3; // 圆圈直径(米) |
| | | double markerRadius = markerDiameter / 2.0; |
| | | |
| | | // 设置字体(11号,不随缩放变化) |
| | | Font labelFont = new Font("微软雅黑", Font.PLAIN, 11); |
| | | g2d.setFont(labelFont); |
| | | FontMetrics fontMetrics = g2d.getFontMetrics(labelFont); |
| | | |
| | | // 紫色实心圆圈颜色 |
| | | Color purpleColor = new Color(128, 0, 128, 255); // 紫色 |
| | | |
| | | // 绘制每个点及其序号 |
| | | for (int i = 0; i < boundary.size(); i++) { |
| | | Point2D.Double point = boundary.get(i); |
| | | double x = point.x; |
| | | double y = point.y; |
| | | |
| | | // 绘制紫色实心圆圈(在世界坐标系中,随缩放变化) |
| | | g2d.setColor(purpleColor); |
| | | 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); |
| | | } |
| | | |
| | | /** |
| | | * 处理优化后边界坐标点点击 |
| | | * @param screenPoint 屏幕坐标点 |
| | | * @return 是否处理了点击 |
| | | */ |
| | | private boolean handleOptimizedBoundaryPointClick(Point screenPoint) { |
| | | if (previewOptimizedBoundary == null || previewOptimizedBoundary.isEmpty()) { |
| | | return false; |
| | | } |
| | | |
| | | // 计算选择阈值(像素) |
| | | double threshold = computeOptimizedBoundaryPointSelectionThreshold(); |
| | | |
| | | // 查找被点击的点 |
| | | int hitIndex = -1; |
| | | for (int i = 0; i < previewOptimizedBoundary.size(); i++) { |
| | | Point2D.Double worldPoint = previewOptimizedBoundary.get(i); |
| | | Point2D.Double screenPosition = worldToScreen(worldPoint); |
| | | double dx = screenPosition.x - screenPoint.x; |
| | | double dy = screenPosition.y - screenPoint.y; |
| | | if (Math.hypot(dx, dy) <= threshold) { |
| | | hitIndex = i; |
| | | break; |
| | | } |
| | | } |
| | | |
| | | if (hitIndex < 0) { |
| | | return false; |
| | | } |
| | | |
| | | // 弹出确认对话框 |
| | | String pointLabel = String.valueOf(hitIndex + 1); |
| | | int choice = JOptionPane.showConfirmDialog( |
| | | visualizationPanel, |
| | | "确定要删除第" + pointLabel + "号优化后边界坐标点吗?", |
| | | "删除边界坐标点", |
| | | JOptionPane.OK_CANCEL_OPTION, |
| | | JOptionPane.WARNING_MESSAGE |
| | | ); |
| | | |
| | | if (choice == JOptionPane.OK_OPTION) { |
| | | // 删除坐标点 |
| | | List<Point2D.Double> updated = new ArrayList<>(previewOptimizedBoundary); |
| | | updated.remove(hitIndex); |
| | | |
| | | // 更新预览边界 |
| | | previewOptimizedBoundary = updated; |
| | | |
| | | // 转换为字符串格式并保存 |
| | | String updatedBoundaryString = convertBoundaryToString(updated); |
| | | |
| | | // 通知 Shouye 保存更新后的边界坐标 |
| | | if (boundaryPreviewUpdateCallback != null) { |
| | | boundaryPreviewUpdateCallback.accept(updatedBoundaryString); |
| | | } |
| | | |
| | | // 刷新显示 |
| | | visualizationPanel.repaint(); |
| | | } |
| | | |
| | | return true; |
| | | } |
| | | |
| | | /** |
| | | * 计算优化后边界坐标点的选择阈值(像素) |
| | | */ |
| | | private double computeOptimizedBoundaryPointSelectionThreshold() { |
| | | double scaleFactor = Math.max(0.5, scale); |
| | | double markerDiameterWorld = 0.3; // 圆圈直径(米) |
| | | double markerDiameterPixels = markerDiameterWorld * scale; |
| | | return Math.max(8.0, markerDiameterPixels * 1.5); |
| | | } |
| | | |
| | | /** |
| | | * 将边界点列表转换为字符串格式 |
| | | */ |
| | | private String convertBoundaryToString(List<Point2D.Double> boundary) { |
| | | if (boundary == null || boundary.isEmpty()) { |
| | | return ""; |
| | | } |
| | | |
| | | StringBuilder sb = new StringBuilder(); |
| | | for (int i = 0; i < boundary.size(); i++) { |
| | | Point2D.Double point = boundary.get(i); |
| | | sb.append(String.format(Locale.US, "%.2f,%.2f", point.x, point.y)); |
| | | if (i < boundary.size() - 1) { |
| | | sb.append(";"); |
| | | } |
| | | } |
| | | return sb.toString(); |
| | | } |
| | | |
| | | /** |
| | | * 设置边界预览更新回调 |
| | | */ |
| | | private java.util.function.Consumer<String> boundaryPreviewUpdateCallback; |
| | | |
| | | public void setBoundaryPreviewUpdateCallback(java.util.function.Consumer<String> callback) { |
| | | this.boundaryPreviewUpdateCallback = callback; |
| | | } |
| | | |
| | | } |