org.netbeans.modules.visual.router.OrthogonalSearchRouter Maven / Gradle / Ivy
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
package org.netbeans.modules.visual.router;
import java.util.EnumSet;
import org.netbeans.api.visual.anchor.Anchor;
import org.netbeans.api.visual.router.Router;
import org.netbeans.api.visual.router.CollisionsCollector;
import org.netbeans.api.visual.router.ConnectionWidgetCollisionsCollector;
import org.netbeans.api.visual.widget.ConnectionWidget;
import java.awt.*;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.List;
import org.netbeans.api.visual.anchor.Anchor.Direction;
import org.netbeans.api.visual.anchor.Anchor.Result;
/**
* @author David Kaspar
*/
public final class OrthogonalSearchRouter implements Router {
static final int SPACING_EDGE = 8;
static final int SPACING_NODE = 16;
private CollisionsCollector collector;
private ConnectionWidgetCollisionsCollector connectionWidgetCollector;
public OrthogonalSearchRouter (CollisionsCollector collector) {
this.collector = collector;
}
public OrthogonalSearchRouter (ConnectionWidgetCollisionsCollector collector) {
this.connectionWidgetCollector = collector;
}
public java.util.List routeConnection(ConnectionWidget widget) {
/* How this works:
* 1) get the source and target anchors
* 2) determine collisions - needed for the actual routing
* 3) get the centers of the source and target anchor
* 4) check to see if the anchor will allow the route to connect to
* any side. This really means that the router can test every
* orthogonal direction out of the center of the anchor. If not
* allowed then get the attach point from the anchor and use it to
* determine the route.
* 5) Iterate through the different allowed orthogonal directions from
* the target and source to find the actual "best" route. This route
* or Solution with contain a list of points for the route.
* 6) Send the list of best points to the source anchor and allow it to
* alter the points. The assumption is that the anchor will only move
* the end point to the appropriate intersection point.
* 7) Repeat (6) for the target.
* 8) Now that the "best" points have been moved they may not represent
* an orthogonal path any longer. Calculate the directions coming from/to
* the anchors in preparation for re-routing.
* 9) Using the new endpoints(6&7) and the directions(8), re-route
* and get the updated solution.
*
*/
Anchor sourceAnchor = widget.getSourceAnchor();
Anchor targetAnchor = widget.getTargetAnchor();
if (sourceAnchor == null || targetAnchor == null) {
return Collections.emptyList();
}
ArrayList verticalCollisions = new ArrayList();
ArrayList horizontalCollisions = new ArrayList();
if (collector != null) {
collector.collectCollisions(verticalCollisions, horizontalCollisions);
} else {
connectionWidgetCollector.collectCollisions(widget, verticalCollisions, horizontalCollisions);
}
if (sourceAnchor.getRelatedWidget() != null &&
(sourceAnchor.getRelatedWidget() == targetAnchor.getRelatedWidget()))
{
// try to avoid routing path inside node widget
Rectangle rec = sourceAnchor.getRelatedWidget().convertLocalToScene(sourceAnchor.getRelatedWidget().getBounds());
rec.grow(SPACING_NODE, SPACING_NODE);
verticalCollisions.add(rec);
horizontalCollisions.add(rec);
}
//the default is to point to the center of any anchor. If an extended anchor
//overwrites this class to return a different point than the center, then
//drawing problem will exist. Meaning that the connection may draw over the
//widget or run along its boundary.
final Point originalSourceCenterPoint = sourceAnchor.getRelatedSceneLocation();
final Point originalTargetCenterPoint = targetAnchor.getRelatedSceneLocation();
Point sourcePoint = originalSourceCenterPoint;
Point targetPoint = originalTargetCenterPoint;
//set the default test direction to be ANY in order to test all directions
//to find the best path.
EnumSet sourceDirections = Anchor.DIRECTION_ANY;
EnumSet targetDirections = Anchor.DIRECTION_ANY;
//if the anchor does not allow arbitrary connection points, then ask the
//anchor for the attach point.
if (!sourceAnchor.allowsArbitraryConnectionPlacement()) {
Result sourceResult = sourceAnchor.compute(widget.getSourceAnchorEntry());
sourceDirections = sourceResult.getDirections();
sourcePoint = sourceResult.getAnchorSceneLocation();
}
if (!targetAnchor.allowsArbitraryConnectionPlacement()) {
Result targetResult = targetAnchor.compute(widget.getTargetAnchorEntry());
targetDirections = targetResult.getDirections();
targetPoint = targetResult.getAnchorSceneLocation();
}
//set best solution to be replaced by first real solution
Solution bestSolution = new Solution(Integer.MAX_VALUE >> 2, null);
for (Anchor.Direction sourceDirection : sourceDirections) { //up to four choices (TOP, BOTTOM, RIGHT, LEFT)
for (Anchor.Direction targetDirection : targetDirections) {//up to four choices (TOP, BOTTOM, RIGHT, LEFT)
Solution solution =
new OrthogonalSearchRouterCore(widget.getScene(),
verticalCollisions, horizontalCollisions,
sourcePoint, sourceDirection,
targetPoint, targetDirection).route();
if (solution != null && solution.compareTo(bestSolution) > 0) {
bestSolution = solution;
}
}
}
List bestListOfPoints = bestSolution.getPoints();
//quick return if possible. If there is no adjusting that will be done,
//then simply return the best points.
if (!sourceAnchor.allowsArbitraryConnectionPlacement() && !targetAnchor.allowsArbitraryConnectionPlacement()) {
return bestListOfPoints;
}
List bestPoints = bestListOfPoints;
Direction sourceDirection = null;
Direction targetDirection = null;
Point firstPointOutsideAnchor = null;
if (sourceAnchor.allowsArbitraryConnectionPlacement()) {
//the initial route went to the center of the anchor. Now that we have the
//"best" route, ask the anchor to where to place its end point. These calls
//only adjust the end points within the anchor. Example would be with the
//rectangular anchor. The connection is meant to terminate on the boundary
//and not at the center. Since the "best" route ends at the center, it must
//be moved to the boundary.
bestPoints = sourceAnchor.compute(bestListOfPoints);
//best points now contain points that may no longer be orthogonal due to the
//adjusting of the endpoints. In order to re-align the points, first determine
//the direction that the points attach to the anchor and then re-route from the
//best attach points.
//Note: this methodology will fail miserably if the anchor moves the points
//around around they have been initially set. Meaning that this method assumes
//that the only sections of the connection not orthogonal are possibly the
//the first and last.
firstPointOutsideAnchor = bestPoints.get(1);
int deltaX = originalSourceCenterPoint.x - firstPointOutsideAnchor.x;
if (deltaX == 0) { // if deltaX is zero then we have a vertical connection
sourceDirection = originalSourceCenterPoint.y < firstPointOutsideAnchor.y ? Direction.BOTTOM : Direction.TOP;
} else {
sourceDirection = originalSourceCenterPoint.x < firstPointOutsideAnchor.x ? Direction.RIGHT : Direction.LEFT;
}
}
//repeat the above steps for the target anchor.
if (targetAnchor.allowsArbitraryConnectionPlacement()) {
bestPoints = targetAnchor.compute(bestPoints);
firstPointOutsideAnchor = bestPoints.get(bestPoints.size() - 2);
int deltaX = originalTargetCenterPoint.x - firstPointOutsideAnchor.x;
if (deltaX == 0) {
targetDirection = originalTargetCenterPoint.y < firstPointOutsideAnchor.y ? Direction.BOTTOM : Direction.TOP;
} else {
targetDirection = originalTargetCenterPoint.x < firstPointOutsideAnchor.x ? Direction.RIGHT : Direction.LEFT;
}
}
//re-route
if (sourceAnchor.allowsArbitraryConnectionPlacement() || targetAnchor.allowsArbitraryConnectionPlacement()) {
bestSolution = new OrthogonalSearchRouterCore(widget.getScene(),
verticalCollisions, horizontalCollisions,
bestPoints.get(0), sourceDirection,
bestPoints.get(bestPoints.size() - 1), targetDirection).route();
}
return bestSolution.getPoints();
}
static final class Solution implements Comparable {
private int price;
private List points;
public Solution (int price, List points) {
this.price = price;
this.points = points;
}
public int getPrice () {
return price;
}
public List getPoints () {
return points;
}
public int compareTo (Solution other) {
return other.price - price;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy