diff --git a/src/utils/IdealGraphVisualizer/Coordinator/src/main/java/com/sun/hotspot/igv/coordinator/FolderNode.java b/src/utils/IdealGraphVisualizer/Coordinator/src/main/java/com/sun/hotspot/igv/coordinator/FolderNode.java index 0521123890a..9188c8493f3 100644 --- a/src/utils/IdealGraphVisualizer/Coordinator/src/main/java/com/sun/hotspot/igv/coordinator/FolderNode.java +++ b/src/utils/IdealGraphVisualizer/Coordinator/src/main/java/com/sun/hotspot/igv/coordinator/FolderNode.java @@ -29,12 +29,11 @@ import com.sun.hotspot.igv.util.PropertiesSheet; import com.sun.hotspot.igv.util.StringUtils; import java.awt.Image; import java.util.HashMap; -import java.util.List; import java.util.Map; -import org.openide.nodes.AbstractNode; -import org.openide.nodes.Children; -import org.openide.nodes.Node; -import org.openide.nodes.Sheet; +import javax.swing.Action; +import org.openide.actions.PropertiesAction; +import org.openide.actions.RenameAction; +import org.openide.nodes.*; import org.openide.util.ImageUtilities; import org.openide.util.lookup.AbstractLookup; import org.openide.util.lookup.InstanceContent; @@ -49,7 +48,7 @@ public class FolderNode extends AbstractNode { private final FolderChildren children; // NetBeans node corresponding to each opened graph. Used to highlight the // focused graph in the Outline window. - private static final Map graphNode = new HashMap<>(); + private static final Map graphNodeMap = new HashMap<>(); private boolean selected = false; private static class FolderChildren extends Children.Keys implements ChangedListener { @@ -66,14 +65,14 @@ public class FolderNode extends AbstractNode { } @Override - protected Node[] createNodes(FolderElement e) { - if (e instanceof InputGraph) { - InputGraph g = (InputGraph) e; - GraphNode n = new GraphNode(g); - graphNode.put(g, n); - return new Node[]{n}; - } else if (e instanceof Folder) { - return new Node[]{new FolderNode((Folder) e)}; + protected Node[] createNodes(FolderElement folderElement) { + if (folderElement instanceof InputGraph) { + InputGraph inputGraph = (InputGraph) folderElement; + GraphNode graphNode = new GraphNode(inputGraph); + graphNodeMap.put(inputGraph, graphNode); + return new Node[]{graphNode}; + } else if (folderElement instanceof Group) { + return new Node[]{new FolderNode((Group) folderElement)}; } else { return null; } @@ -83,19 +82,23 @@ public class FolderNode extends AbstractNode { protected void destroyNodes(Node[] nodes) { for (Node n : nodes) { // Each node is only present once in the graphNode map. - graphNode.values().remove(n); + graphNodeMap.values().remove(n); + } + for (Node node : getNodes()) { + node.setDisplayName(node.getDisplayName()); } } @Override public void addNotify() { - this.setKeys(folder.getElements()); + super.addNotify(); + setKeys(folder.getElements()); } @Override public void changed(Object source) { addNotify(); - } + } } @Override @@ -128,7 +131,7 @@ public class FolderNode extends AbstractNode { if (folder instanceof FolderElement) { final FolderElement folderElement = (FolderElement) folder; this.setDisplayName(folderElement.getName()); - content.add((RemoveCookie) () -> { + this.content.add((RemoveCookie) () -> { children.destroyNodes(children.getNodes()); folderElement.getParent().removeElement(folderElement); }); @@ -141,6 +144,28 @@ public class FolderNode extends AbstractNode { fireIconChange(); } + @Override + public boolean canRename() { + return true; + } + + @Override + public void setName(String name) { + children.getFolder().setName(name); + fireDisplayNameChange(null, null); + } + + @Override + public String getName() { + return children.getFolder().getName(); + } + + @Override + public String getDisplayName() { + return children.getFolder().getDisplayName(); + } + + @Override public String getHtmlDisplayName() { String htmlDisplayName = StringUtils.escapeHTML(getDisplayName()); if (selected) { @@ -149,15 +174,6 @@ public class FolderNode extends AbstractNode { return htmlDisplayName; } - public void init(String name, List groups) { - this.setDisplayName(name); - children.addNotify(); - - for (Group g : groups) { - content.add(g); - } - } - public boolean isRootNode() { Folder folder = getFolder(); return (folder instanceof GraphDocument); @@ -168,12 +184,20 @@ public class FolderNode extends AbstractNode { return getIcon(i); } + @Override + public Action[] getActions(boolean b) { + return new Action[]{ + RenameAction.findObject(RenameAction.class, true), + PropertiesAction.findObject(PropertiesAction.class, true), + }; + } + public static void clearGraphNodeMap() { - graphNode.clear(); + graphNodeMap.clear(); } public static GraphNode getGraphNode(InputGraph graph) { - return graphNode.get(graph); + return graphNodeMap.get(graph); } public Folder getFolder() { diff --git a/src/utils/IdealGraphVisualizer/Coordinator/src/main/java/com/sun/hotspot/igv/coordinator/GraphNode.java b/src/utils/IdealGraphVisualizer/Coordinator/src/main/java/com/sun/hotspot/igv/coordinator/GraphNode.java index d3a0ed5232c..6c6974b7100 100644 --- a/src/utils/IdealGraphVisualizer/Coordinator/src/main/java/com/sun/hotspot/igv/coordinator/GraphNode.java +++ b/src/utils/IdealGraphVisualizer/Coordinator/src/main/java/com/sun/hotspot/igv/coordinator/GraphNode.java @@ -32,6 +32,7 @@ import com.sun.hotspot.igv.util.StringUtils; import java.awt.Image; import javax.swing.Action; import org.openide.actions.OpenAction; +import org.openide.actions.RenameAction; import org.openide.nodes.*; import org.openide.util.ImageUtilities; import org.openide.util.Lookup; @@ -44,7 +45,7 @@ import org.openide.util.lookup.InstanceContent; */ public class GraphNode extends AbstractNode { - private InputGraph graph; + private final InputGraph graph; private boolean selected = false; /** Creates a new instance of GraphNode */ @@ -52,12 +53,29 @@ public class GraphNode extends AbstractNode { this(graph, new InstanceContent()); } + @Override + public boolean canRename() { + return true; + } + + @Override + public void setName(String name) { + graph.setName(name); + fireDisplayNameChange(null, null); + } + + @Override + public String getName() { + return graph.getName(); + } + public void setSelected(boolean selected) { this.selected = selected; fireDisplayNameChange(null, null); fireIconChange(); } + @Override public String getHtmlDisplayName() { String htmlDisplayName = StringUtils.escapeHTML(getDisplayName()); if (selected) { @@ -65,6 +83,12 @@ public class GraphNode extends AbstractNode { } return htmlDisplayName; } + + @Override + public String getDisplayName() { + return graph.getDisplayName(); + } + private GraphNode(InputGraph graph, InstanceContent content) { super(Children.LEAF, new AbstractLookup(content)); this.graph = graph; @@ -86,13 +110,6 @@ public class GraphNode extends AbstractNode { // Action for cloning to the current graph content.add(new GraphCloneCookie(viewer, graph)); - - this.addNodeListener(new NodeAdapter() { - @Override - public void childrenRemoved(NodeMemberEvent ev) { - GraphNode.this.graph = null; - } - }); } @Override @@ -122,7 +139,12 @@ public class GraphNode extends AbstractNode { @Override public Action[] getActions(boolean b) { - return new Action[]{DiffGraphAction.findObject(DiffGraphAction.class, true), CloneGraphAction.findObject(CloneGraphAction.class, true), OpenAction.findObject(OpenAction.class, true)}; + return new Action[]{ + RenameAction.findObject(RenameAction.class, true), + DiffGraphAction.findObject(DiffGraphAction.class, true), + CloneGraphAction.findObject(CloneGraphAction.class, true), + OpenAction.findObject(OpenAction.class, true) + }; } @Override diff --git a/src/utils/IdealGraphVisualizer/Coordinator/src/main/resources/com/sun/hotspot/igv/coordinator/layer.xml b/src/utils/IdealGraphVisualizer/Coordinator/src/main/resources/com/sun/hotspot/igv/coordinator/layer.xml index 0f28ec80d95..e1e57a6dcc1 100644 --- a/src/utils/IdealGraphVisualizer/Coordinator/src/main/resources/com/sun/hotspot/igv/coordinator/layer.xml +++ b/src/utils/IdealGraphVisualizer/Coordinator/src/main/resources/com/sun/hotspot/igv/coordinator/layer.xml @@ -20,6 +20,9 @@ + + + @@ -58,27 +61,35 @@ - + + + + + + + + + - + - + - + - + - + diff --git a/src/utils/IdealGraphVisualizer/Data/src/main/java/com/sun/hotspot/igv/data/Folder.java b/src/utils/IdealGraphVisualizer/Data/src/main/java/com/sun/hotspot/igv/data/Folder.java index 3f93ba3a710..9ff7c32bd1f 100644 --- a/src/utils/IdealGraphVisualizer/Data/src/main/java/com/sun/hotspot/igv/data/Folder.java +++ b/src/utils/IdealGraphVisualizer/Data/src/main/java/com/sun/hotspot/igv/data/Folder.java @@ -26,6 +26,9 @@ package com.sun.hotspot.igv.data; import java.util.List; public interface Folder { + void setName(String name); + String getName(); + String getDisplayName(); List getElements(); void removeElement(FolderElement element); void addElement(FolderElement group); diff --git a/src/utils/IdealGraphVisualizer/Data/src/main/java/com/sun/hotspot/igv/data/FolderElement.java b/src/utils/IdealGraphVisualizer/Data/src/main/java/com/sun/hotspot/igv/data/FolderElement.java index 696479e3f91..eb0a874278f 100644 --- a/src/utils/IdealGraphVisualizer/Data/src/main/java/com/sun/hotspot/igv/data/FolderElement.java +++ b/src/utils/IdealGraphVisualizer/Data/src/main/java/com/sun/hotspot/igv/data/FolderElement.java @@ -24,8 +24,10 @@ package com.sun.hotspot.igv.data; public interface FolderElement { - - Folder getParent(); + ChangedEvent getDisplayNameChangedEvent(); + void setName(String name); String getName(); + String getDisplayName(); void setParent(Folder parent); + Folder getParent(); } diff --git a/src/utils/IdealGraphVisualizer/Data/src/main/java/com/sun/hotspot/igv/data/GraphDocument.java b/src/utils/IdealGraphVisualizer/Data/src/main/java/com/sun/hotspot/igv/data/GraphDocument.java index 744a3aa491e..1f13cb3b433 100644 --- a/src/utils/IdealGraphVisualizer/Data/src/main/java/com/sun/hotspot/igv/data/GraphDocument.java +++ b/src/utils/IdealGraphVisualizer/Data/src/main/java/com/sun/hotspot/igv/data/GraphDocument.java @@ -34,10 +34,12 @@ public class GraphDocument extends Properties.Entity implements ChangedEventProv private final List elements; private final ChangedEvent changedEvent; + private String name; public GraphDocument() { elements = new ArrayList<>(); changedEvent = new ChangedEvent<>(this); + setName("GraphDocument"); } public void clear() { @@ -61,10 +63,23 @@ public class GraphDocument extends Properties.Entity implements ChangedEventProv getChangedEvent().fire(); } + @Override + public void setName(String name) { + this.name = name; + } + + public String getName() { + return name; + } + + @Override + public String getDisplayName() { + return getName(); + } + @Override public String toString() { StringBuilder sb = new StringBuilder(); - sb.append("GraphDocument: ").append(getProperties().toString()).append(" \n\n"); for (FolderElement g : getElements()) { sb.append(g.toString()); @@ -84,6 +99,9 @@ public class GraphDocument extends Properties.Entity implements ChangedEventProv if (elements.remove(element)) { getChangedEvent().fire(); } + for (FolderElement folderElement : elements) { + folderElement.getDisplayNameChangedEvent().fire(); + } } @Override diff --git a/src/utils/IdealGraphVisualizer/Data/src/main/java/com/sun/hotspot/igv/data/Group.java b/src/utils/IdealGraphVisualizer/Data/src/main/java/com/sun/hotspot/igv/data/Group.java index 8926b67eb19..5eb4b416ad7 100644 --- a/src/utils/IdealGraphVisualizer/Data/src/main/java/com/sun/hotspot/igv/data/Group.java +++ b/src/utils/IdealGraphVisualizer/Data/src/main/java/com/sun/hotspot/igv/data/Group.java @@ -34,6 +34,8 @@ public class Group extends Properties.Entity implements ChangedEventProvider graphs; private InputMethod method; private final transient ChangedEvent changedEvent; + private final ChangedEvent displayNameChangedEvent = new ChangedEvent<>(this); + private Folder parent; public Group(Folder parent) { @@ -72,6 +74,10 @@ public class Group extends Properties.Entity implements ChangedEventProvider getDisplayNameChangedEvent() { + return displayNameChangedEvent; + } + + @Override + public void setName(String name) { + getProperties().setProperty("name", name); + displayNameChangedEvent.fire(); + } + @Override public String getName() { return getProperties().get("name"); } + @Override + public String getDisplayName() { + return getParent().getElements().indexOf(this)+1 + " - " + getName(); + } + public String getType() { return getProperties().get("type"); - } InputGraph getPrev(InputGraph graph) { diff --git a/src/utils/IdealGraphVisualizer/Data/src/main/java/com/sun/hotspot/igv/data/InputGraph.java b/src/utils/IdealGraphVisualizer/Data/src/main/java/com/sun/hotspot/igv/data/InputGraph.java index 6cfa0c74484..1a408a79caf 100644 --- a/src/utils/IdealGraphVisualizer/Data/src/main/java/com/sun/hotspot/igv/data/InputGraph.java +++ b/src/utils/IdealGraphVisualizer/Data/src/main/java/com/sun/hotspot/igv/data/InputGraph.java @@ -38,29 +38,35 @@ public class InputGraph extends Properties.Entity implements FolderElement { private final Map blocks; private final List blockEdges; private final Map nodeToBlock; - private boolean isDiffGraph; - private InputGraph firstGraph; - private InputGraph secondGraph; - + private final boolean isDiffGraph; + private final InputGraph firstGraph; + private final InputGraph secondGraph; + private final ChangedEvent displayNameChangedEvent = new ChangedEvent<>(this); public InputGraph(InputGraph firstGraph, InputGraph secondGraph) { - this(firstGraph.getName() + " Δ " + secondGraph.getName()); + this(firstGraph.getName() + " Δ " + secondGraph.getName(), firstGraph, secondGraph); assert !firstGraph.isDiffGraph() && !secondGraph.isDiffGraph(); - this.firstGraph = firstGraph; - this.secondGraph = secondGraph; - isDiffGraph = true; + } public InputGraph(String name) { + this(name, null, null); + } + + private InputGraph(String name, InputGraph firstGraph, InputGraph secondGraph) { setName(name); nodes = new LinkedHashMap<>(); edges = new ArrayList<>(); blocks = new LinkedHashMap<>(); blockEdges = new ArrayList<>(); nodeToBlock = new LinkedHashMap<>(); - firstGraph = null; - secondGraph = null; - isDiffGraph = false; + isDiffGraph = firstGraph != null && secondGraph != null; + this.firstGraph = firstGraph; + this.secondGraph = secondGraph; + if (isDiffGraph) { + this.firstGraph.getDisplayNameChangedEvent().addListener(l -> displayNameChangedEvent.fire()); + this.secondGraph.getDisplayNameChangedEvent().addListener(l -> displayNameChangedEvent.fire()); + } } public boolean isDiffGraph() { @@ -81,6 +87,9 @@ public class InputGraph extends Properties.Entity implements FolderElement { if (parent instanceof Group) { assert this.parentGroup == null; this.parentGroup = (Group) parent; + assert displayNameChangedEvent != null; + assert this.parentGroup.getDisplayNameChangedEvent() != null; + this.parentGroup.getDisplayNameChangedEvent().addListener(l -> displayNameChangedEvent.fire()); } } @@ -220,8 +229,15 @@ public class InputGraph extends Properties.Entity implements FolderElement { return parentGroup.getPrev(this); } - private void setName(String name) { - this.getProperties().setProperty("name", name); + @Override + public ChangedEvent getDisplayNameChangedEvent() { + return displayNameChangedEvent; + } + + @Override + public void setName(String name) { + getProperties().setProperty("name", name); + displayNameChangedEvent.fire(); } @Override @@ -229,6 +245,19 @@ public class InputGraph extends Properties.Entity implements FolderElement { return getProperties().get("name"); } + @Override + public String getDisplayName() { + if (isDiffGraph) { + return firstGraph.getDisplayName() + " Δ " + secondGraph.getDisplayName(); + } else { + return getIndex()+1 + ". " + getName(); + } + } + + public int getIndex() { + return getGroup().getGraphs().indexOf(this); + } + public Collection getNodes() { return Collections.unmodifiableCollection(nodes.values()); } diff --git a/src/utils/IdealGraphVisualizer/View/src/main/java/com/sun/hotspot/igv/view/DiagramViewModel.java b/src/utils/IdealGraphVisualizer/View/src/main/java/com/sun/hotspot/igv/view/DiagramViewModel.java index a4a4ba03d04..d43489580a7 100644 --- a/src/utils/IdealGraphVisualizer/View/src/main/java/com/sun/hotspot/igv/view/DiagramViewModel.java +++ b/src/utils/IdealGraphVisualizer/View/src/main/java/com/sun/hotspot/igv/view/DiagramViewModel.java @@ -37,6 +37,7 @@ import com.sun.hotspot.igv.settings.Settings; import com.sun.hotspot.igv.util.RangeSliderModel; import java.awt.Color; import java.util.*; +import java.util.function.Consumer; import org.openide.util.Lookup; /** @@ -45,7 +46,6 @@ import org.openide.util.Lookup; */ public class DiagramViewModel extends RangeSliderModel implements ChangedListener { - // Warning: Update setData method if fields are added private final Group group; private ArrayList graphs; private Set hiddenNodes; @@ -58,6 +58,7 @@ public class DiagramViewModel extends RangeSliderModel implements ChangedListene private final ChangedEvent graphChangedEvent; private final ChangedEvent selectedNodesChangedEvent; private final ChangedEvent hiddenNodesChangedEvent; + private ChangedListener titleChangedListener = g -> {}; private boolean showSea; private boolean showBlocks; private boolean showCFG; @@ -390,6 +391,9 @@ public class DiagramViewModel extends RangeSliderModel implements ChangedListene @Override public void changed(RangeSliderModel source) { + if (cachedInputGraph != null) { + cachedInputGraph.getDisplayNameChangedEvent().removeListener(titleChangedListener); + } if (getFirstGraph() != getSecondGraph()) { cachedInputGraph = Difference.createDiffGraph(getFirstGraph(), getSecondGraph()); } else { @@ -397,6 +401,12 @@ public class DiagramViewModel extends RangeSliderModel implements ChangedListene } rebuildDiagram(); graphChangedEvent.fire(); + assert titleChangedListener != null; + cachedInputGraph.getDisplayNameChangedEvent().addListener(titleChangedListener); + } + + void addTitleCallback(Consumer titleCallback) { + titleChangedListener = titleCallback::accept; } void close() { diff --git a/src/utils/IdealGraphVisualizer/View/src/main/java/com/sun/hotspot/igv/view/EditorTopComponent.java b/src/utils/IdealGraphVisualizer/View/src/main/java/com/sun/hotspot/igv/view/EditorTopComponent.java index 2e8044fb646..5a92db9ada5 100644 --- a/src/utils/IdealGraphVisualizer/View/src/main/java/com/sun/hotspot/igv/view/EditorTopComponent.java +++ b/src/utils/IdealGraphVisualizer/View/src/main/java/com/sun/hotspot/igv/view/EditorTopComponent.java @@ -135,12 +135,6 @@ public final class EditorTopComponent extends TopComponent { content.add(diagramViewModel); associateLookup(new ProxyLookup(scene.getLookup(), new AbstractLookup(graphContent), new AbstractLookup(content))); - diagramViewModel.getDiagramChangedEvent().addListener(model -> { - setDisplayName(model.getGraph().getName()); - setToolTipText(model.getGroup().getName()); - graphContent.set(Collections.singletonList(new EditorInputGraphProvider(this)), null); - }); - Group group = diagramViewModel.getGroup(); group.getChangedEvent().addListener(g -> closeOnRemovedOrEmptyGroup()); if (group.getParent() instanceof GraphDocument) { @@ -148,6 +142,17 @@ public final class EditorTopComponent extends TopComponent { doc.getChangedEvent().addListener(d -> closeOnRemovedOrEmptyGroup()); } + diagramViewModel.addTitleCallback(changedGraph -> { + setDisplayName(changedGraph.getDisplayName()); + setToolTipText(diagramViewModel.getGroup().getDisplayName()); + }); + + diagramViewModel.getDiagramChangedEvent().addListener(model -> { + setDisplayName(model.getGraph().getDisplayName()); + setToolTipText(model.getGroup().getDisplayName()); + graphContent.set(Collections.singletonList(new EditorInputGraphProvider(this)), null); + }); + cardLayout = new CardLayout(); centerPanel = new JPanel(); centerPanel.setLayout(cardLayout); diff --git a/src/utils/IdealGraphVisualizer/View/src/main/resources/com/sun/hotspot/igv/view/layer.xml b/src/utils/IdealGraphVisualizer/View/src/main/resources/com/sun/hotspot/igv/view/layer.xml index 867ded535f4..e705b2556a8 100644 --- a/src/utils/IdealGraphVisualizer/View/src/main/resources/com/sun/hotspot/igv/view/layer.xml +++ b/src/utils/IdealGraphVisualizer/View/src/main/resources/com/sun/hotspot/igv/view/layer.xml @@ -23,7 +23,7 @@ - +