Skip to content

Commit c055dfc

Browse files
committedOct 24, 2022
8294565: IGV: ClassCastException when clicking on an edge in the graph
Reviewed-by: rcastanedalo, thartmann
1 parent 3898385 commit c055dfc

File tree

2 files changed

+73
-70
lines changed

2 files changed

+73
-70
lines changed
 

‎src/utils/IdealGraphVisualizer/View/src/main/java/com/sun/hotspot/igv/view/DiagramScene.java

+45-45
Original file line numberDiff line numberDiff line change
@@ -723,50 +723,52 @@ private void doCFGLayout(HashSet<Figure> figures, HashSet<Connection> edges) {
723723

724724
private Set<Pair<Point, Point>> lineCache = new HashSet<>();
725725

726-
private void relayoutWithoutLayout(Set<Widget> oldVisibleWidgets) {
727-
728-
Diagram diagram = getModel().getDiagram();
729-
730-
SceneAnimator animator = getSceneAnimator();
731-
connectionLayer.removeChildren();
726+
private boolean shouldAnimate() {
732727
int visibleFigureCount = 0;
733-
for (Figure f : diagram.getFigures()) {
734-
if (getWidget(f, FigureWidget.class).isVisible()) {
728+
for (Figure figure : getModel().getDiagram().getFigures()) {
729+
if (getWidget(figure, FigureWidget.class).isVisible()) {
735730
visibleFigureCount++;
736731
}
737732
}
733+
return visibleFigureCount <= ANIMATION_LIMIT;
734+
}
738735

736+
private void relayoutWithoutLayout(Set<Widget> oldVisibleWidgets) {
737+
assert oldVisibleWidgets != null;
738+
739+
Diagram diagram = getModel().getDiagram();
740+
connectionLayer.removeChildren();
741+
742+
SceneAnimator connectionAnimator = getSceneAnimator();
743+
boolean doAnimation = shouldAnimate();
744+
if (!doAnimation) {
745+
connectionAnimator = null;
746+
}
739747

740748
Set<Pair<Point, Point>> lastLineCache = lineCache;
741749
lineCache = new HashSet<>();
742-
for (Figure f : diagram.getFigures()) {
743-
for (OutputSlot s : f.getOutputSlots()) {
744-
SceneAnimator anim = animator;
745-
if (visibleFigureCount > ANIMATION_LIMIT || oldVisibleWidgets == null) {
746-
anim = null;
747-
}
748-
List<Connection> cl = new ArrayList<>(s.getConnections().size());
749-
for (FigureConnection c : s.getConnections()) {
750-
cl.add((Connection) c);
751-
}
752-
processOutputSlot(lastLineCache, s, cl, 0, null, null, 0, 0, anim);
750+
for (Figure figure : diagram.getFigures()) {
751+
for (OutputSlot outputSlot : figure.getOutputSlots()) {
752+
List<Connection> connectionList = new ArrayList<>(outputSlot.getConnections());
753+
processOutputSlot(lastLineCache, outputSlot, connectionList, 0, null, null, connectionAnimator);
753754
}
754755
}
755756

756757
if (getModel().getShowCFG()) {
757758
for (BlockConnection c : diagram.getBlockConnections()) {
758759
if (isVisible(c)) {
759-
processOutputSlot(lastLineCache, null, Collections.singletonList(c), 0, null, null, 0, 0, animator);
760+
processOutputSlot(lastLineCache, null, Collections.singletonList(c), 0, null, null, connectionAnimator);
760761
}
761762
}
762763
}
763764

765+
SceneAnimator animator = getSceneAnimator();
764766
for (Figure f : diagram.getFigures()) {
765767
FigureWidget w = getWidget(f);
766768
if (w.isVisible()) {
767769
Point p = f.getPosition();
768770
Point p2 = new Point(p.x, p.y);
769-
if ((visibleFigureCount <= ANIMATION_LIMIT && oldVisibleWidgets != null && oldVisibleWidgets.contains(w))) {
771+
if (doAnimation && oldVisibleWidgets.contains(w)) {
770772
animator.animatePreferredLocation(w, p2);
771773
} else {
772774
w.setPreferredLocation(p2);
@@ -782,7 +784,7 @@ private void relayoutWithoutLayout(Set<Widget> oldVisibleWidgets) {
782784
Point location = new Point(b.getBounds().x, b.getBounds().y);
783785
Rectangle r = new Rectangle(location.x, location.y, b.getBounds().width, b.getBounds().height);
784786

785-
if ((visibleFigureCount <= ANIMATION_LIMIT && oldVisibleWidgets != null && oldVisibleWidgets.contains(w))) {
787+
if (doAnimation && oldVisibleWidgets.contains(w)) {
786788
animator.animatePreferredBounds(w, r);
787789
} else {
788790
w.setPreferredBounds(r);
@@ -796,44 +798,42 @@ private void relayoutWithoutLayout(Set<Widget> oldVisibleWidgets) {
796798
}
797799
private final Point specialNullPoint = new Point(Integer.MAX_VALUE, Integer.MAX_VALUE);
798800

799-
private void processOutputSlot(Set<Pair<Point, Point>> lastLineCache, OutputSlot outputSlot, List<Connection> connections, int controlPointIndex, Point lastPoint, LineWidget predecessor, int offx, int offy, SceneAnimator animator) {
801+
private void processOutputSlot(Set<Pair<Point, Point>> lastLineCache, OutputSlot outputSlot, List<Connection> connections, int controlPointIndex, Point lastPoint, LineWidget predecessor, SceneAnimator animator) {
800802
Map<Point, List<Connection>> pointMap = new HashMap<>(connections.size());
801803

802-
for (Connection c : connections) {
803-
804-
if (!isVisible(c)) {
804+
for (Connection connection : connections) {
805+
if (!isVisible(connection)) {
805806
continue;
806807
}
807808

808-
List<Point> controlPoints = c.getControlPoints();
809+
List<Point> controlPoints = connection.getControlPoints();
809810
if (controlPointIndex >= controlPoints.size()) {
810811
continue;
811812
}
812813

813-
Point cur = controlPoints.get(controlPointIndex);
814-
if (cur == null) { // Long connection, has been cut vertically.
815-
cur = specialNullPoint;
816-
} else if (c.hasSlots()) {
814+
Point currentPoint = controlPoints.get(controlPointIndex);
815+
if (currentPoint == null) { // Long connection, has been cut vertically.
816+
currentPoint = specialNullPoint;
817+
} else if (connection.hasSlots()) {
817818
if (controlPointIndex == 0 && !outputSlot.shouldShowName()) {
818-
cur = new Point(cur.x, cur.y - SLOT_OFFSET);
819+
currentPoint = new Point(currentPoint.x, currentPoint.y - SLOT_OFFSET);
819820
} else if (controlPointIndex == controlPoints.size() - 1 &&
820-
!((Slot)c.getTo()).shouldShowName()) {
821-
cur = new Point(cur.x, cur.y + SLOT_OFFSET);
821+
!((Slot)connection.getTo()).shouldShowName()) {
822+
currentPoint = new Point(currentPoint.x, currentPoint.y + SLOT_OFFSET);
822823
}
823824
}
824825

825-
if (pointMap.containsKey(cur)) {
826-
pointMap.get(cur).add(c);
826+
if (pointMap.containsKey(currentPoint)) {
827+
pointMap.get(currentPoint).add(connection);
827828
} else {
828829
List<Connection> newList = new ArrayList<>(2);
829-
newList.add(c);
830-
pointMap.put(cur, newList);
830+
newList.add(connection);
831+
pointMap.put(currentPoint, newList);
831832
}
832-
833833
}
834834

835-
for (Point p : pointMap.keySet()) {
836-
List<Connection> connectionList = pointMap.get(p);
835+
for (Point currentPoint : pointMap.keySet()) {
836+
List<Connection> connectionList = pointMap.get(currentPoint);
837837

838838
boolean isBold = false;
839839
boolean isDashed = true;
@@ -855,9 +855,9 @@ private void processOutputSlot(Set<Pair<Point, Point>> lastLineCache, OutputSlot
855855
}
856856

857857
LineWidget newPredecessor = predecessor;
858-
if (p != specialNullPoint && lastPoint != specialNullPoint && lastPoint != null) {
859-
Point p1 = new Point(lastPoint.x + offx, lastPoint.y + offy);
860-
Point p2 = new Point(p.x + offx, p.y + offy);
858+
if (currentPoint != specialNullPoint && lastPoint != specialNullPoint && lastPoint != null) {
859+
Point p1 = new Point(lastPoint.x, lastPoint.y);
860+
Point p2 = new Point(currentPoint.x, currentPoint.y);
861861

862862
Pair<Point, Point> curPair = new Pair<>(p1, p2);
863863
SceneAnimator curAnimator = animator;
@@ -874,7 +874,7 @@ private void processOutputSlot(Set<Pair<Point, Point>> lastLineCache, OutputSlot
874874
lineWidget.getActions().addAction(hoverAction);
875875
}
876876

877-
processOutputSlot(lastLineCache, outputSlot, connectionList, controlPointIndex + 1, p, newPredecessor, offx, offy, animator);
877+
processOutputSlot(lastLineCache, outputSlot, connectionList, controlPointIndex + 1, currentPoint, newPredecessor, animator);
878878
}
879879
}
880880

‎src/utils/IdealGraphVisualizer/View/src/main/java/com/sun/hotspot/igv/view/widgets/LineWidget.java

+28-25
Original file line numberDiff line numberDiff line change
@@ -27,8 +27,10 @@
2727
import com.sun.hotspot.igv.graph.Connection;
2828
import com.sun.hotspot.igv.graph.Figure;
2929
import com.sun.hotspot.igv.graph.OutputSlot;
30+
import com.sun.hotspot.igv.layout.Vertex;
3031
import com.sun.hotspot.igv.util.StringUtils;
3132
import com.sun.hotspot.igv.view.DiagramScene;
33+
import com.sun.hotspot.igv.view.actions.CustomSelectAction;
3234
import java.awt.*;
3335
import java.awt.geom.Line2D;
3436
import java.util.ArrayList;
@@ -57,20 +59,19 @@ public class LineWidget extends Widget implements PopupMenuProvider {
5759
public final int HOVER_ARROW_SIZE = 8;
5860
public final int BOLD_STROKE_WIDTH = 2;
5961
public final int HOVER_STROKE_WIDTH = 3;
60-
private static double ZOOM_FACTOR = 0.1;
61-
private OutputSlot outputSlot;
62-
private DiagramScene scene;
63-
private List<Connection> connections;
64-
private Point from;
65-
private Point to;
66-
private Rectangle clientArea;
67-
private Color color = Color.BLACK;
68-
private LineWidget predecessor;
69-
private List<LineWidget> successors;
62+
private final static double ZOOM_FACTOR = 0.1;
63+
private final OutputSlot outputSlot;
64+
private final DiagramScene scene;
65+
private final List<Connection> connections;
66+
private final Point from;
67+
private final Point to;
68+
private final Rectangle clientArea;
69+
private final LineWidget predecessor;
70+
private final List<LineWidget> successors;
7071
private boolean highlighted;
7172
private boolean popupVisible;
72-
private boolean isBold;
73-
private boolean isDashed;
73+
private final boolean isBold;
74+
private final boolean isDashed;
7475

7576
public LineWidget(DiagramScene scene, OutputSlot s, List<Connection> connections, Point from, Point to, LineWidget predecessor, SceneAnimator animator, boolean isBold, boolean isDashed) {
7677
super(scene);
@@ -107,43 +108,44 @@ public LineWidget(DiagramScene scene, OutputSlot s, List<Connection> connections
107108
clientArea = new Rectangle(minX, minY, maxX - minX + 1, maxY - minY + 1);
108109
clientArea.grow(BORDER, BORDER);
109110

111+
Color color = Color.BLACK;
110112
if (connections.size() > 0) {
111113
color = connections.get(0).getColor();
112114
}
113-
this.setToolTipText("<HTML>" + generateToolTipText(this.connections) + "</HTML>");
115+
setToolTipText("<HTML>" + generateToolTipText(this.connections) + "</HTML>");
114116

115-
this.setCheckClipping(true);
117+
setCheckClipping(true);
116118

117-
this.getActions().addAction(ActionFactory.createPopupMenuAction(this));
119+
getActions().addAction(ActionFactory.createPopupMenuAction(this));
118120
if (animator == null) {
119121
this.setBackground(color);
120122
} else {
121123
this.setBackground(Color.WHITE);
122124
animator.animateBackgroundColor(this, color);
123125
}
124126

125-
this.getActions().addAction(ActionFactory.createSelectAction(new SelectProvider() {
127+
getActions().addAction(new CustomSelectAction(new SelectProvider() {
126128

127129
@Override
128-
public boolean isAimingAllowed(Widget arg0, Point arg1, boolean arg2) {
130+
public boolean isAimingAllowed(Widget widget, Point localLocation, boolean invertSelection) {
129131
return true;
130132
}
131133

132134
@Override
133-
public boolean isSelectionAllowed(Widget arg0, Point arg1, boolean arg2) {
135+
public boolean isSelectionAllowed(Widget widget, Point localLocation, boolean invertSelection) {
134136
return true;
135137
}
136138

137139
@Override
138-
public void select(Widget arg0, Point arg1, boolean arg2) {
139-
Set<Figure> set = new HashSet<>();
140-
for (Connection c : LineWidget.this.connections) {
141-
if (c.hasSlots()) {
142-
set.add(scene.getWidget(c.getTo()));
143-
set.add(scene.getWidget(c.getFrom()));
140+
public void select(Widget widget, Point localLocation, boolean invertSelection) {
141+
Set<Vertex> vertexSet = new HashSet<>();
142+
for (Connection connection : connections) {
143+
if (connection.hasSlots()) {
144+
vertexSet.add(connection.getTo().getVertex());
145+
vertexSet.add(connection.getFrom().getVertex());
144146
}
145147
}
146-
LineWidget.this.scene.setSelectedObjects(set);
148+
scene.userSelectionSuggested(vertexSet, invertSelection);
147149
}
148150
}));
149151
}
@@ -209,6 +211,7 @@ protected void paintWidget() {
209211
for (LineWidget w : successors) {
210212
if (w.getFrom().equals(getTo())) {
211213
sameTo = true;
214+
break;
212215
}
213216
}
214217

0 commit comments

Comments
 (0)
Please sign in to comment.