Changeset 18965 in josm
Legend:
- Unmodified
- Added
- Removed
-
trunk/src/org/openstreetmap/josm/actions/mapmode/ImproveWayAccuracyAction.java
r18274 r18965 32 32 import org.openstreetmap.josm.data.UndoRedoHandler; 33 33 import org.openstreetmap.josm.data.coor.EastNorth; 34 import org.openstreetmap.josm.data.coor.ILatLon; 34 35 import org.openstreetmap.josm.data.osm.DataSelectionListener; 35 36 import org.openstreetmap.josm.data.osm.DataSet; … … 269 270 // Non-native highlighting is used here as well. 270 271 271 // Finding endpoints272 Node p1 = null;273 Node p2 = null;274 272 if (ctrl && candidateSegment != null) { 275 g.setStroke(ADD_NODE_STROKE.get()); 276 try { 277 p1 = candidateSegment.getFirstNode(); 278 p2 = candidateSegment.getSecondNode(); 279 } catch (ArrayIndexOutOfBoundsException e) { 280 Logging.error(e); 281 } 273 paintAddNodeStroke(g); 282 274 } else if (!alt && !ctrl && candidateNode != null) { 283 g.setStroke(MOVE_NODE_STROKE.get()); 284 List<Pair<Node, Node>> wpps = targetWay.getNodePairs(false); 285 for (Pair<Node, Node> wpp : wpps) { 286 if (wpp.a == candidateNode) { 287 p1 = wpp.b; 288 } 289 if (wpp.b == candidateNode) { 290 p2 = wpp.a; 291 } 292 if (p1 != null && p2 != null) { 293 break; 294 } 295 } 275 paintMoveNodeStroke(g); 296 276 } else if (alt && !ctrl && candidateNode != null) { 297 g.setStroke(DELETE_NODE_STROKE.get()); 298 List<Node> nodes = targetWay.getNodes(); 299 int index = nodes.indexOf(candidateNode); 300 301 // Only draw line if node is not first and/or last 302 if (index > 0 && index < (nodes.size() - 1)) { 303 p1 = nodes.get(index - 1); 304 p2 = nodes.get(index + 1); 305 } else if (targetWay.isClosed()) { 306 p1 = targetWay.getNode(1); 307 p2 = targetWay.getNode(nodes.size() - 2); 308 } 309 // TODO: indicate what part that will be deleted? (for end nodes) 310 } 311 312 313 // Drawing preview lines 314 MapViewPath b = new MapViewPath(mv); 315 if (alt && !ctrl) { 316 // In delete mode 317 if (p1 != null && p2 != null) { 318 b.moveTo(p1); 319 b.lineTo(p2); 320 } 321 } else { 322 // In add or move mode 323 if (p1 != null) { 324 b.moveTo(mousePos.x, mousePos.y); 325 b.lineTo(p1); 326 } 327 if (p2 != null) { 328 b.moveTo(mousePos.x, mousePos.y); 329 b.lineTo(p2); 330 } 331 } 277 paintDeleteStroke(g); 278 } 279 } 280 } 281 282 /** 283 * Paint the add stroke for {@link State#IMPROVING} 284 * @param g The graphics 285 */ 286 private void paintAddNodeStroke(Graphics2D g) { 287 g.setStroke(ADD_NODE_STROKE.get()); 288 ILatLon p1 = null; 289 ILatLon p2 = null; 290 try { 291 p1 = candidateSegment.getFirstNode(); 292 p2 = candidateSegment.getSecondNode(); 293 } catch (ArrayIndexOutOfBoundsException e) { 294 Logging.error(e); 295 } 296 paintImprovingPreviewLines(g, p1, p2); 297 } 298 299 /** 300 * Paint the move stroke for {@link State#IMPROVING} 301 * @param g The graphics 302 */ 303 private void paintMoveNodeStroke(Graphics2D g) { 304 g.setStroke(MOVE_NODE_STROKE.get()); 305 List<Pair<Node, Node>> wpps = targetWay.getNodePairs(false); 306 ILatLon p1 = null; 307 ILatLon p2 = null; 308 for (Pair<Node, Node> wpp : wpps) { 309 if (wpp.a == candidateNode) { 310 p1 = wpp.b; 311 } 312 if (wpp.b == candidateNode) { 313 p2 = wpp.a; 314 } 315 if (p1 != null && p2 != null) { 316 break; 317 } 318 } 319 paintImprovingPreviewLines(g, p1, p2); 320 } 321 322 /** 323 * Paint the delete stroke for {@link State#IMPROVING} 324 * @param g The graphics 325 */ 326 private void paintDeleteStroke(Graphics2D g) { 327 g.setStroke(DELETE_NODE_STROKE.get()); 328 List<Node> nodes = targetWay.getNodes(); 329 int index = nodes.indexOf(candidateNode); 330 ILatLon p1 = null; 331 ILatLon p2 = null; 332 // Only draw line if node is not first and/or last 333 if (index > 0 && index < (nodes.size() - 1)) { 334 p1 = nodes.get(index - 1); 335 p2 = nodes.get(index + 1); 336 } else if (targetWay.isClosed()) { 337 p1 = targetWay.getNode(1); 338 p2 = targetWay.getNode(nodes.size() - 2); 339 } 340 paintImprovingPreviewLines(g, p1, p2); 341 // TODO: indicate what part that will be deleted? (for end nodes) 342 } 343 344 /** 345 * Paint the preview lines for {@link State#IMPROVING} 346 * @param g The graphics 347 * @param p1 The first endpoint 348 * @param p2 The second endpoint 349 */ 350 private void paintImprovingPreviewLines(Graphics2D g, ILatLon p1, ILatLon p2) { 351 // Drawing preview lines 352 MapViewPath b = new MapViewPath(mv); 353 if (alt && !ctrl) { 354 // In delete mode 355 if (p1 != null && p2 != null) { 356 b.moveTo(p1); 357 b.lineTo(p2); 358 } 359 } else { 360 // In add or move mode 361 if (p1 != null) { 362 b.moveTo(mousePos.x, mousePos.y); 363 b.lineTo(p1); 364 } 365 if (p2 != null) { 366 b.moveTo(mousePos.x, mousePos.y); 367 b.lineTo(p2); 368 } 369 } 370 g.draw(b.computeClippedLine(g.getStroke())); 371 372 // Highlighting candidateNode 373 if (candidateNode != null) { 374 p1 = candidateNode; 375 g.fill(new MapViewPath(mv).shapeAround(p1, SymbolShape.SQUARE, DOT_SIZE.get())); 376 } 377 378 if (!alt && !ctrl && candidateNode != null) { 379 b.reset(); 380 drawIntersectingWayHelperLines(mv, b); 381 g.setStroke(MOVE_NODE_INTERSECTING_STROKE.get()); 332 382 g.draw(b.computeClippedLine(g.getStroke())); 333 334 // Highlighting candidateNode335 if (candidateNode != null) {336 p1 = candidateNode;337 g.fill(new MapViewPath(mv).shapeAround(p1, SymbolShape.SQUARE, DOT_SIZE.get()));338 }339 340 if (!alt && !ctrl && candidateNode != null) {341 b.reset();342 drawIntersectingWayHelperLines(mv, b);343 g.setStroke(MOVE_NODE_INTERSECTING_STROKE.get());344 g.draw(b.computeClippedLine(g.getStroke()));345 }346 347 383 } 348 384 } … … 440 476 441 477 if (ctrl && !alt && candidateSegment != null) { 442 // Add a new node to the highlighted segment. 443 Collection<WaySegment> virtualSegments = new LinkedList<>(); 444 445 // Check if other ways have the same segment. 446 // We have to make sure that we add the new node to all of them. 447 Set<Way> commonParentWays = new HashSet<>(candidateSegment.getFirstNode().getParentWays()); 448 commonParentWays.retainAll(candidateSegment.getSecondNode().getParentWays()); 449 for (Way w : commonParentWays) { 450 for (int i = 0; i < w.getNodesCount() - 1; i++) { 451 WaySegment testWS = new WaySegment(w, i); 452 if (testWS.isSimilar(candidateSegment)) { 453 virtualSegments.add(testWS); 454 } 455 } 456 } 457 458 Collection<Command> virtualCmds = new LinkedList<>(); 459 // Create the new node 460 Node virtualNode = new Node(mv.getEastNorth(mousePos.x, mousePos.y)); 461 virtualCmds.add(new AddCommand(ds, virtualNode)); 462 463 // Adding the node to all segments found 464 for (WaySegment virtualSegment : virtualSegments) { 465 Way w = virtualSegment.getWay(); 466 List<Node> modNodes = w.getNodes(); 467 modNodes.add(virtualSegment.getUpperIndex(), virtualNode); 468 virtualCmds.add(new ChangeNodesCommand(w, modNodes)); 469 } 470 471 // Finishing the sequence command 472 String text = trn("Add a new node to way", 473 "Add a new node to {0} ways", 474 virtualSegments.size(), virtualSegments.size()); 475 476 UndoRedoHandler.getInstance().add(new SequenceCommand(text, virtualCmds)); 477 478 addNode(ds); 478 479 } else if (alt && !ctrl && candidateNode != null) { 479 // Deleting the highlighted node 480 481 //check to see if node is in use by more than one object 482 long referrersCount = candidateNode.referrers(OsmPrimitive.class).count(); 483 long referrerWayCount = candidateNode.referrers(Way.class).count(); 484 if (referrersCount != 1 || referrerWayCount != 1) { 485 // detach node from way 486 final List<Node> nodes = targetWay.getNodes(); 487 nodes.remove(candidateNode); 488 if (nodes.size() < 2) { 489 final Command deleteCmd = DeleteCommand.delete(Collections.singleton(targetWay), true); 490 if (deleteCmd != null) { 491 UndoRedoHandler.getInstance().add(deleteCmd); 492 } 493 } else { 494 UndoRedoHandler.getInstance().add(new ChangeNodesCommand(targetWay, nodes)); 495 } 496 } else if (candidateNode.isTagged()) { 497 JOptionPane.showMessageDialog(MainApplication.getMainFrame(), 498 tr("Cannot delete node that has tags"), 499 tr("Error"), JOptionPane.ERROR_MESSAGE); 500 } else { 501 final Command deleteCmd = DeleteCommand.delete(Collections.singleton(candidateNode), true); 502 if (deleteCmd != null) { 503 UndoRedoHandler.getInstance().add(deleteCmd); 504 } 505 } 506 480 deleteHighlightedNode(); 507 481 } else if (candidateNode != null) { 508 // Moving the highlighted node 509 EastNorth nodeEN = candidateNode.getEastNorth(); 510 EastNorth cursorEN = mv.getEastNorth(mousePos.x, mousePos.y); 511 512 UndoRedoHandler.getInstance().add( 513 new MoveCommand(candidateNode, cursorEN.east() - nodeEN.east(), cursorEN.north() - nodeEN.north())); 514 515 SelectAction.checkCommandForLargeDistance(UndoRedoHandler.getInstance().getLastCommand()); 482 moveHighlightedNode(); 516 483 } 517 484 } … … 523 490 } 524 491 492 /** 493 * Add a new node to the currently highlighted segment 494 * @param ds The dataset to add the node to 495 */ 496 private void addNode(DataSet ds) { 497 // Add a new node to the highlighted segment. 498 Collection<WaySegment> virtualSegments = new LinkedList<>(); 499 500 // Check if other ways have the same segment. 501 // We have to make sure that we add the new node to all of them. 502 Set<Way> commonParentWays = new HashSet<>(candidateSegment.getFirstNode().getParentWays()); 503 commonParentWays.retainAll(candidateSegment.getSecondNode().getParentWays()); 504 for (Way w : commonParentWays) { 505 for (int i = 0; i < w.getNodesCount() - 1; i++) { 506 WaySegment testWS = new WaySegment(w, i); 507 if (testWS.isSimilar(candidateSegment)) { 508 virtualSegments.add(testWS); 509 } 510 } 511 } 512 513 Collection<Command> virtualCmds = new LinkedList<>(); 514 // Create the new node 515 Node virtualNode = new Node(mv.getEastNorth(mousePos.x, mousePos.y)); 516 virtualCmds.add(new AddCommand(ds, virtualNode)); 517 518 // Adding the node to all segments found 519 for (WaySegment virtualSegment : virtualSegments) { 520 Way w = virtualSegment.getWay(); 521 List<Node> modNodes = w.getNodes(); 522 modNodes.add(virtualSegment.getUpperIndex(), virtualNode); 523 virtualCmds.add(new ChangeNodesCommand(w, modNodes)); 524 } 525 526 // Finishing the sequence command 527 String text = trn("Add a new node to way", 528 "Add a new node to {0} ways", 529 virtualSegments.size(), virtualSegments.size()); 530 531 UndoRedoHandler.getInstance().add(new SequenceCommand(text, virtualCmds)); 532 } 533 534 /** 535 * Delete the highlighted node 536 */ 537 private void deleteHighlightedNode() { 538 // Deleting the highlighted node 539 540 //check to see if node is in use by more than one object 541 long referrersCount = candidateNode.referrers(OsmPrimitive.class).count(); 542 long referrerWayCount = candidateNode.referrers(Way.class).count(); 543 if (referrersCount != 1 || referrerWayCount != 1) { 544 // detach node from way 545 final List<Node> nodes = targetWay.getNodes(); 546 nodes.remove(candidateNode); 547 if (nodes.size() < 2) { 548 final Command deleteCmd = DeleteCommand.delete(Collections.singleton(targetWay), true); 549 if (deleteCmd != null) { 550 UndoRedoHandler.getInstance().add(deleteCmd); 551 } 552 } else { 553 UndoRedoHandler.getInstance().add(new ChangeNodesCommand(targetWay, nodes)); 554 } 555 } else if (candidateNode.isTagged()) { 556 JOptionPane.showMessageDialog(MainApplication.getMainFrame(), 557 tr("Cannot delete node that has tags"), 558 tr("Error"), JOptionPane.ERROR_MESSAGE); 559 } else { 560 final Command deleteCmd = DeleteCommand.delete(Collections.singleton(candidateNode), true); 561 if (deleteCmd != null) { 562 UndoRedoHandler.getInstance().add(deleteCmd); 563 } 564 } 565 } 566 567 /** 568 * Move the highlighted node 569 */ 570 private void moveHighlightedNode() { 571 // Moving the highlighted node 572 EastNorth nodeEN = candidateNode.getEastNorth(); 573 EastNorth cursorEN = mv.getEastNorth(mousePos.x, mousePos.y); 574 575 UndoRedoHandler.getInstance().add( 576 new MoveCommand(candidateNode, cursorEN.east() - nodeEN.east(), cursorEN.north() - nodeEN.north())); 577 578 SelectAction.checkCommandForLargeDistance(UndoRedoHandler.getInstance().getLastCommand()); 579 } 580 525 581 @Override 526 582 public void mouseExited(MouseEvent e) { … … 551 607 : cursorSelectHover, this); 552 608 } else if (state == State.IMPROVING) { 553 if (alt && !ctrl) { 554 mv.setNewCursor(cursorImproveDelete, this); 555 } else if (shift || dragging) { 556 if (ctrl) { 557 mv.setNewCursor(cursorImproveAddLock, this); 558 } else { 559 mv.setNewCursor(cursorImproveLock, this); 560 } 561 } else if (ctrl && !alt) { 562 mv.setNewCursor(cursorImproveAdd, this); 609 updateCursorImproving(); 610 } 611 } 612 613 /** 614 * Update the mouse cursor for the {@link State#IMPROVING} mode 615 */ 616 private void updateCursorImproving() { 617 if (alt && !ctrl) { 618 mv.setNewCursor(cursorImproveDelete, this); 619 } else if (shift || dragging) { 620 if (ctrl) { 621 mv.setNewCursor(cursorImproveAddLock, this); 563 622 } else { 564 mv.setNewCursor(cursorImprove, this); 565 } 623 mv.setNewCursor(cursorImproveLock, this); 624 } 625 } else if (ctrl && !alt) { 626 mv.setNewCursor(cursorImproveAdd, this); 627 } else { 628 mv.setNewCursor(cursorImprove, this); 566 629 } 567 630 } … … 642 705 */ 643 706 private void updateStateByCurrentSelection() { 707 final DataSet ds = getLayerManager().getEditDataSet(); 708 if (ds != null && selectWay(ds)) { 709 return; 710 } 711 712 // Starting selecting by default 713 startSelecting(); 714 } 715 716 /** 717 * Select the initial way 718 * @param ds The dataset to get the selection from 719 * @return {@code true} if a way was selected 720 */ 721 private boolean selectWay(DataSet ds) { 644 722 final List<Node> nodeList = new ArrayList<>(); 645 723 final List<Way> wayList = new ArrayList<>(); 646 final DataSet ds = getLayerManager().getEditDataSet(); 647 if (ds != null) { 648 final Collection<OsmPrimitive> sel = ds.getSelected(); 649 650 // Collecting nodes and ways from the selection 651 for (OsmPrimitive p : sel) { 652 if (p instanceof Way) { 653 wayList.add((Way) p); 654 } 655 if (p instanceof Node) { 656 nodeList.add((Node) p); 657 } 658 } 659 660 if (wayList.size() == 1) { 661 // Starting improving the single selected way 662 startImproving(wayList.get(0)); 663 return; 664 } else if (nodeList.size() == 1) { 665 // Starting improving the only way of the single selected node 666 List<OsmPrimitive> r = nodeList.get(0).getReferrers(); 667 if (r.size() == 1 && (r.get(0) instanceof Way)) { 668 startImproving((Way) r.get(0)); 669 return; 670 } 671 } 672 } 673 674 // Starting selecting by default 675 startSelecting(); 724 final Collection<OsmPrimitive> sel = ds.getSelected(); 725 726 // Collecting nodes and ways from the selection 727 for (OsmPrimitive p : sel) { 728 if (p instanceof Way) { 729 wayList.add((Way) p); 730 } 731 if (p instanceof Node) { 732 nodeList.add((Node) p); 733 } 734 } 735 736 if (wayList.size() == 1) { 737 // Starting improving the single selected way 738 startImproving(wayList.get(0)); 739 return true; 740 } else if (nodeList.size() == 1) { 741 // Starting improving the only way of the single selected node 742 List<OsmPrimitive> r = nodeList.get(0).getReferrers(); 743 r.removeIf(osm -> !osm.isUsable()); 744 if (r.size() == 1 && (r.get(0) instanceof Way)) { 745 startImproving((Way) r.get(0)); 746 return true; 747 } 748 } 749 return false; 676 750 } 677 751 -
trunk/test/unit/org/openstreetmap/josm/actions/mapmode/ImproveWayAccuracyActionTest.java
r18963 r18965 2 2 package org.openstreetmap.josm.actions.mapmode; 3 3 4 import static org.junit.jupiter.api.Assertions.assertAll; 4 5 import static org.junit.jupiter.api.Assertions.assertDoesNotThrow; 5 6 import static org.junit.jupiter.api.Assertions.assertEquals; 6 7 import static org.junit.jupiter.api.Assertions.assertFalse; 8 import static org.junit.jupiter.api.Assertions.assertNotEquals; 7 9 import static org.junit.jupiter.api.Assertions.assertTrue; 8 10 9 11 import java.awt.Graphics2D; 10 12 import java.awt.Point; 13 import java.awt.Rectangle; 14 import java.awt.event.InputEvent; 11 15 import java.awt.event.MouseEvent; 12 16 import java.awt.image.BufferedImage; 13 17 import java.util.Collection; 18 import java.util.concurrent.atomic.AtomicReference; 19 20 import org.apache.commons.lang3.function.TriConsumer; 14 21 import org.junit.jupiter.api.Test; 15 import org.junit.jupiter.api.extension.RegisterExtension;16 22 import org.openstreetmap.josm.TestUtils; 17 23 import org.openstreetmap.josm.actions.mapmode.ImproveWayAccuracyAction.State; 24 import org.openstreetmap.josm.command.DeleteCommand; 18 25 import org.openstreetmap.josm.data.Bounds; 19 26 import org.openstreetmap.josm.data.coor.ILatLon; … … 21 28 import org.openstreetmap.josm.data.osm.DataSet; 22 29 import org.openstreetmap.josm.data.osm.Node; 30 import org.openstreetmap.josm.data.osm.OsmPrimitive; 31 import org.openstreetmap.josm.data.osm.Relation; 32 import org.openstreetmap.josm.data.osm.RelationToChildReference; 23 33 import org.openstreetmap.josm.data.osm.Way; 34 import org.openstreetmap.josm.data.osm.visitor.BoundingXYVisitor; 24 35 import org.openstreetmap.josm.gui.MainApplication; 25 36 import org.openstreetmap.josm.gui.MapFrame; 26 37 import org.openstreetmap.josm.gui.MapView; 38 import org.openstreetmap.josm.gui.layer.Layer; 27 39 import org.openstreetmap.josm.gui.layer.OsmDataLayer; 28 40 import org.openstreetmap.josm.gui.util.GuiHelper; 29 41 import org.openstreetmap.josm.testutils.annotations.Main; 30 42 import org.openstreetmap.josm.testutils.annotations.Projection; 31 import org.openstreetmap.josm.testutils.mockers.WindowlessMapViewStateMocker;32 import org.openstreetmap.josm.testutils.mockers.WindowlessNavigatableComponentMocker;33 34 import mockit.Mock;35 43 36 44 /** 37 45 * Unit tests for class {@link ImproveWayAccuracyAction}. 38 46 */ 47 @Main 39 48 @Projection 40 49 class ImproveWayAccuracyActionTest { 41 @RegisterExtension42 Main.MainExtension mainExtension = new Main.MainExtension().setMapViewMocker(SizedWindowlessMapViewStateMocker::new)43 .setNavigableComponentMocker(SizedWindowlessNavigatableComponentMocker::new);44 45 50 private static final int WIDTH = 800; 46 51 private static final int HEIGHT = 600; 47 private static class SizedWindowlessMapViewStateMocker extends WindowlessMapViewStateMocker { 48 @Mock 49 public int getWidth() { 50 return WIDTH; 51 } 52 53 @Mock 54 public int getHeight() { 55 return HEIGHT; 56 } 57 } 58 59 private static class SizedWindowlessNavigatableComponentMocker extends WindowlessNavigatableComponentMocker { 60 @Mock 61 public int getWidth() { 62 return WIDTH; 63 } 64 65 @Mock 66 public int getHeight() { 67 return HEIGHT; 68 } 69 } 70 71 private static MouseEvent generateEvent(MapView mapView, ILatLon location) { 52 53 private static class AlwaysDeleteCallback implements DeleteCommand.DeletionCallback { 54 @Override 55 public boolean checkAndConfirmOutlyingDelete(Collection<? extends OsmPrimitive> primitives, Collection<? extends OsmPrimitive> ignore) { 56 return true; 57 } 58 59 @Override 60 public boolean confirmRelationDeletion(Collection<Relation> relations) { 61 return true; 62 } 63 64 @Override 65 public boolean confirmDeletionFromRelation(Collection<RelationToChildReference> references) { 66 return true; 67 } 68 } 69 70 private static void setupMapView(DataSet ds) { 71 // setup a reasonable size for the edit window 72 MainApplication.getMap().mapView.setBounds(new Rectangle(WIDTH, HEIGHT)); 73 if (ds.getDataSourceBoundingBox() != null) { 74 MainApplication.getMap().mapView.zoomTo(ds.getDataSourceBoundingBox()); 75 } else { 76 BoundingXYVisitor v = new BoundingXYVisitor(); 77 for (Layer l : MainApplication.getLayerManager().getLayers()) { 78 l.visitBoundingBox(v); 79 } 80 MainApplication.getMap().mapView.zoomTo(v); 81 } 82 } 83 84 /** 85 * Generate a mouse event 86 * @param mapView The current map view 87 * @param location The location to generate the event for 88 * @param modifiers The modifiers for {@link MouseEvent} (see {@link InputEvent#getModifiersEx()}) 89 * @return The generated event 90 */ 91 private static MouseEvent generateEvent(MapView mapView, ILatLon location, int modifiers) { 72 92 final Point p = mapView.getPoint(location); 73 return new MouseEvent(mapView, 0, 0, 0, p.x, p.y, p.x, p.y, 1, false, MouseEvent.BUTTON1);93 return new MouseEvent(mapView, 0, 0, modifiers, p.x, p.y, p.x, p.y, 1, false, MouseEvent.BUTTON1); 74 94 } 75 95 … … 102 122 103 123 @Test 124 void testNonRegression23444Selection() { 125 final DataSet dataSet = new DataSet(); 126 final OsmDataLayer layer = new OsmDataLayer(dataSet, "ImproveWayAccuracyActionTest#testNonRegression23444Selection", null); 127 MainApplication.getLayerManager().addLayer(layer); 128 final ImproveWayAccuracyAction mapMode = new ImproveWayAccuracyAction(); 129 final MapFrame map = MainApplication.getMap(); 130 final Way testWay = TestUtils.newWay("", new Node(1, 1), new Node(2, 1), 131 new Node(3), new Node(4, 1), new Node(5, 1)); 132 testWay.firstNode().setCoor(new LatLon(0, 0)); 133 testWay.lastNode().setCoor(new LatLon(0.001, 0.001)); 134 testWay.getNode(1).setCoor(new LatLon(0.0001, 0.0001)); 135 testWay.getNode(3).setCoor(new LatLon(0.0009, 0.0009)); 136 dataSet.addPrimitiveRecursive(testWay); 137 assertFalse(testWay.getNode(2).isLatLonKnown(), "The second node should not have valid coordinates"); 138 dataSet.setSelected(testWay.firstNode()); 139 assertTrue(map.selectMapMode(mapMode)); 140 assertEquals(mapMode, map.mapMode); 141 // This is where the exception occurs; we shouldn't be setting the incomplete way as the target when we enter the mode. 142 setupMapView(dataSet); 143 assertDoesNotThrow(() -> GuiHelper.runInEDTAndWaitWithException(() -> { 144 mapMode.mouseMoved(generateEvent(map.mapView, new LatLon(0.0001, 0.0001), 0)); 145 })); 146 } 147 148 @Test 104 149 void testNonRegression23444() { 150 testSimplifyWayAction((mapMode, map, testWay) -> { 151 testWay.getNode(2).setCoor(null); 152 assertFalse(testWay.getNode(2).isLatLonKnown(), "The second node should not have valid coordinates"); 153 mapMode.startSelecting(); 154 mapMode.mouseMoved(generateEvent(map.mapView, testWay.getNode(1), 0)); 155 }); 156 } 157 158 @Test 159 void testAdd() { 160 AtomicReference<Way> referenceWay = new AtomicReference<>(); 161 testSimplifyWayAction((mapMode, map, testWay) -> { 162 // Add a node at 0.0001, 0.0005 (not on the direct line) 163 mapMode.mouseMoved(generateEvent(map.mapView, new LatLon(0.0001, 0.0005), InputEvent.CTRL_DOWN_MASK)); 164 mapMode.mouseReleased(generateEvent(map.mapView, new LatLon(0.0001, 0.0005), InputEvent.CTRL_DOWN_MASK)); 165 referenceWay.set(testWay); 166 }); 167 final Way testWay = referenceWay.get(); 168 // There should be a new node between nodes 1 and 2 (old node 2 is now node 3) 169 assertAll(() -> assertEquals(6, testWay.getNodesCount()), 170 () -> assertFalse(testWay.getNode(1).isNew()), 171 () -> assertTrue(testWay.getNode(2).isNew()), 172 () -> assertFalse(testWay.getNode(3).isNew())); 173 // These aren't expected to be 0.0001 and 0.0005 exactly, due zoom and conversions between point and latlon. 174 assertAll(() -> assertEquals(0.0001, testWay.getNode(2).lat(), 1e-5), 175 () -> assertEquals(0.0005, testWay.getNode(2).lon(), 1e-5)); 176 } 177 178 @Test 179 void testAddLock() { 180 final AtomicReference<Way> referenceWay = new AtomicReference<>(); 181 testSimplifyWayAction((mapMode, map, testWay) -> { 182 // Add a node at 0.0009, 0.0005 (not on the direct line) that is between nodes 1 and 2 but not 2 and 3. 183 // First get the waysegment selected 184 mapMode.mouseMoved(generateEvent(map.mapView, new LatLon(0.0001, 0.0005), InputEvent.CTRL_DOWN_MASK)); 185 // Then move to another location with ctrl+shift 186 mapMode.mouseMoved(generateEvent(map.mapView, new LatLon(0.0009, 0.0005), InputEvent.CTRL_DOWN_MASK | InputEvent.SHIFT_DOWN_MASK)); 187 // Finally, release the mouse with ctrl+shift 188 mapMode.mouseReleased(generateEvent(map.mapView, new LatLon(0.0009, 0.0005), 189 InputEvent.CTRL_DOWN_MASK | InputEvent.SHIFT_DOWN_MASK)); 190 referenceWay.set(testWay); 191 }); 192 final Way testWay = referenceWay.get(); 193 // There should be a new node between nodes 1 and 2 (old node 2 is now node 3) 194 assertAll(() -> assertEquals(6, testWay.getNodesCount()), 195 () -> assertFalse(testWay.getNode(1).isNew()), 196 () -> assertTrue(testWay.getNode(2).isNew()), 197 () -> assertFalse(testWay.getNode(3).isNew())); 198 // These aren't expected to be 0.0009 and 0.0005 exactly, due zoom and conversions between point and latlon. 199 assertAll(() -> assertEquals(0.0009, testWay.getNode(2).lat(), 1e-5), 200 () -> assertEquals(0.0005, testWay.getNode(2).lon(), 1e-5)); 201 } 202 203 @Test 204 void testMove() { 205 final AtomicReference<Way> referenceWay = new AtomicReference<>(); 206 testSimplifyWayAction((mapMode, map, testWay) -> { 207 // Move node to 0.0001, 0.0005 (not on the direct line) 208 // First get the waysegment selected 209 mapMode.mouseMoved(generateEvent(map.mapView, new LatLon(0.0001, 0.0005), 0)); 210 // Finally, release the mouse 211 mapMode.mouseReleased(generateEvent(map.mapView, new LatLon(0.0001, 0.0005), 0)); 212 referenceWay.set(testWay); 213 }); 214 final Way testWay = referenceWay.get(); 215 assertEquals(5, testWay.getNodesCount()); 216 // These aren't expected to be 0.0001 and 0.0005 exactly, due zoom and conversions between point and latlon. 217 assertAll(() -> assertEquals(0.0001, testWay.getNode(2).lat(), 1e-5), 218 () -> assertEquals(0.0005, testWay.getNode(2).lon(), 1e-5)); 219 } 220 221 @Test 222 void testMoveLock() { 223 final AtomicReference<Way> referenceWay = new AtomicReference<>(); 224 testSimplifyWayAction((mapMode, map, testWay) -> { 225 // Move node to 0.0001, 0.0005 (not on the direct line) 226 // First get the waysegment selected 227 mapMode.mouseMoved(generateEvent(map.mapView, new LatLon(0.0001, 0.0005), 0)); 228 // Then move to another location 229 mapMode.mouseMoved(generateEvent(map.mapView, new LatLon(0.0009, 0.0005), InputEvent.SHIFT_DOWN_MASK)); 230 // Finally, release the mouse 231 mapMode.mouseReleased(generateEvent(map.mapView, new LatLon(0.0009, 0.0005), InputEvent.SHIFT_DOWN_MASK)); 232 referenceWay.set(testWay); 233 }); 234 final Way testWay = referenceWay.get(); 235 assertEquals(5, testWay.getNodesCount()); 236 // These aren't expected to be 0.0009 and 0.0005 exactly, due zoom and conversions between point and latlon. 237 assertAll(() -> assertEquals(0.0009, testWay.getNode(2).lat(), 1e-5), 238 () -> assertEquals(0.0005, testWay.getNode(2).lon(), 1e-5)); 239 } 240 241 @Test 242 void testDelete() { 243 DeleteCommand.setDeletionCallback(new AlwaysDeleteCallback()); 244 final AtomicReference<Way> referenceWay = new AtomicReference<>(); 245 testSimplifyWayAction((mapMode, map, testWay) -> { 246 // Move node to 0.0001, 0.0005 (not on the direct line) 247 // First get the waysegment selected 248 mapMode.mouseMoved(generateEvent(map.mapView, new LatLon(0.0001, 0.0005), InputEvent.ALT_DOWN_MASK)); 249 // Finally, release the mouse 250 mapMode.mouseReleased(generateEvent(map.mapView, new LatLon(0.0001, 0.0005), InputEvent.ALT_DOWN_MASK)); 251 referenceWay.set(testWay); 252 }); 253 final Way testWay = referenceWay.get(); 254 assertEquals(4, testWay.getNodesCount()); 255 assertAll(testWay.getNodes().stream().map(n -> () -> assertNotEquals(3, n.getUniqueId()))); 256 } 257 258 @Test 259 void testDeleteLock() { 260 DeleteCommand.setDeletionCallback(new AlwaysDeleteCallback()); 261 final AtomicReference<Way> referenceWay = new AtomicReference<>(); 262 testSimplifyWayAction((mapMode, map, testWay) -> { 263 // Move node to 0.0001, 0.0005 (not on the direct line) 264 // First get the waysegment selected 265 mapMode.mouseMoved(generateEvent(map.mapView, new LatLon(0.0001, 0.0005), 0)); 266 // Then move to another location 267 mapMode.mouseMoved(generateEvent(map.mapView, new LatLon(0.0009, 0.0005), InputEvent.SHIFT_DOWN_MASK | InputEvent.ALT_DOWN_MASK)); 268 // Finally, release the mouse 269 mapMode.mouseReleased(generateEvent(map.mapView, new LatLon(0.0009, 0.0005), InputEvent.SHIFT_DOWN_MASK | InputEvent.ALT_DOWN_MASK)); 270 referenceWay.set(testWay); 271 }); 272 final Way testWay = referenceWay.get(); 273 assertEquals(4, testWay.getNodesCount()); 274 assertAll(testWay.getNodes().stream().map(n -> () -> assertNotEquals(3, n.getUniqueId()))); 275 } 276 277 private void testSimplifyWayAction(TriConsumer<ImproveWayAccuracyAction, MapFrame, Way> runnable) { 105 278 final DataSet dataSet = new DataSet(); 106 279 final OsmDataLayer layer = new OsmDataLayer(dataSet, "ImproveWayAccuracyActionT", null); … … 115 288 testWay.lastNode().setCoor(new LatLon(0.001, 0.001)); 116 289 testWay.getNode(1).setCoor(new LatLon(0.0001, 0.0001)); 290 testWay.getNode(2).setCoor(new LatLon(0.0005, 0.0005)); 117 291 testWay.getNode(3).setCoor(new LatLon(0.0009, 0.0009)); 118 292 dataSet.addPrimitiveRecursive(testWay); 119 assertFalse(testWay.getNode(2).isLatLonKnown(), "The second node should not have valid coordinates");293 setupMapView(dataSet); 120 294 final Graphics2D g2d = new BufferedImage(WIDTH, HEIGHT, BufferedImage.TYPE_INT_ARGB).createGraphics(); 121 295 g2d.setClip(0, 0, WIDTH, HEIGHT); … … 124 298 assertDoesNotThrow(() -> map.mapView.paint(g2d), "The mapview should be able to handle a null coordinate node"); 125 299 // Ensure that the test way is selected (and use the methods from the action to do so) 126 GuiHelper.runInEDTAndWaitWithException(() -> { 127 map.mapView.zoomTo(new Bounds(0, 0, 0.001, 0.001));128 // Get the target way selected (note: not selected in dataset -- usemapMode.mapReleased for that)129 mapMode.mouse Moved(generateEvent(map.mapView, testWay.getNode(1)));130 });131 // mouseMoved shouldn't cause the way to get selected132 assertFalse(dataSet.getAllSelected().contains(testWay));300 assertDoesNotThrow(() -> GuiHelper.runInEDTAndWaitWithException(() -> { 301 // Set the way as selected 302 mapMode.mouseMoved(generateEvent(map.mapView, testWay.getNode(1), 0)); 303 mapMode.mouseReleased(generateEvent(map.mapView, testWay.getNode(1), 0)); 304 // And then run the test case 305 runnable.accept(mapMode, map, testWay); 306 })); 133 307 134 308 // Now check painting (where the problem should occur; the mapMode.paint call should be called as part of the map.mapView.paint call) … … 136 310 assertDoesNotThrow(() -> mapMode.paint(g2d, map.mapView, new Bounds(0, 0, 0.001, 0.001))); 137 311 138 // Finally, check painting during selection312 // Then perform the action(s) 139 313 GuiHelper.runInEDTAndWaitWithException(() -> { 140 // Set the way as selected141 mapMode.mouseReleased(generateEvent(map.mapView, new LatLon(0.0001, 0.0001)));142 314 // Set the mouse location (unset in mouseReleased call) 143 mapMode.mouseMoved(generateEvent(map.mapView, new LatLon(0.0001, 0.0001))); 315 // This is required by testNonRegression23444, and it doesn't hurt during the other tests 316 mapMode.mouseMoved(generateEvent(map.mapView, new LatLon(0.0001, 0.0001), 0)); 144 317 }); 145 // The way shouldn't be selected, since it isn't usable for the improve way tool146 assertFalse(dataSet.getAllSelected().contains(testWay));147 318 // Now check painting again (just in case) 148 319 assertDoesNotThrow(() -> map.paint(g2d));
Note:
See TracChangeset
for help on using the changeset viewer.