/*
Copyright 2011-2016 Google Inc. All Rights Reserved.
Licensed 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 com.google.security.zynamics.binnavi.Gui.GraphWindows.NodeTaggingTree;
import com.google.common.base.Preconditions;
import com.google.security.zynamics.binnavi.Gui.HotKeys;
import com.google.security.zynamics.binnavi.Gui.Actions.CActionProxy;
import com.google.security.zynamics.binnavi.Gui.DragAndDrop.CDefaultTransferHandler;
import com.google.security.zynamics.binnavi.Gui.DragAndDrop.IDropHandler;
import com.google.security.zynamics.binnavi.Gui.GraphWindows.NodeTaggingTree.Actions.CAddRootTagNodeAction;
import com.google.security.zynamics.binnavi.Gui.GraphWindows.NodeTaggingTree.DragDrop.CTagSortingHandler;
import com.google.security.zynamics.binnavi.Gui.GraphWindows.NodeTaggingTree.Implementations.CTagFunctions;
import com.google.security.zynamics.binnavi.Gui.GraphWindows.NodeTaggingTree.Nodes.CTagTreeNode;
import com.google.security.zynamics.binnavi.Gui.GraphWindows.NodeTaggingTree.Nodes.CTaggedGraphNodeNode;
import com.google.security.zynamics.binnavi.Gui.GraphWindows.NodeTaggingTree.Nodes.CTaggedGraphNodesContainerNode;
import com.google.security.zynamics.binnavi.Gui.GraphWindows.NodeTaggingTree.Nodes.ITagTreeNode;
import com.google.security.zynamics.binnavi.Help.CHelpFunctions;
import com.google.security.zynamics.binnavi.Help.IHelpInformation;
import com.google.security.zynamics.binnavi.Help.IHelpProvider;
import com.google.security.zynamics.binnavi.Tagging.ITagManager;
import com.google.security.zynamics.binnavi.yfileswrap.Gui.GraphWindows.NodeTaggingTree.Nodes.CRootTagTreeNode;
import com.google.security.zynamics.binnavi.yfileswrap.zygraph.NaviNode;
import com.google.security.zynamics.binnavi.yfileswrap.zygraph.ZyGraph;
import com.google.security.zynamics.zylib.gui.SwingInvoker;
import com.google.security.zynamics.zylib.gui.dndtree.DNDTree;
import com.google.security.zynamics.zylib.gui.jtree.TreeHelpers;
import com.google.security.zynamics.zylib.gui.zygraph.IZyGraphSelectionListener;
import com.google.security.zynamics.zylib.gui.zygraph.IZyGraphVisibilityListener;
import com.google.security.zynamics.zylib.yfileswrap.gui.zygraph.functions.MoveFunctions;
import com.google.security.zynamics.zylib.yfileswrap.gui.zygraph.functions.ZoomFunctions;
import java.awt.dnd.DnDConstants;
import java.awt.event.ActionEvent;
import java.awt.event.MouseAdapter;
import java.awt.event.MouseEvent;
import java.net.URL;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Enumeration;
import java.util.List;
import java.util.Objects;
import javax.swing.AbstractAction;
import javax.swing.InputMap;
import javax.swing.JComponent;
import javax.swing.JFrame;
import javax.swing.JPopupMenu;
import javax.swing.event.TreeModelEvent;
import javax.swing.event.TreeModelListener;
import javax.swing.tree.DefaultMutableTreeNode;
import javax.swing.tree.DefaultTreeSelectionModel;
import javax.swing.tree.TreePath;
import javax.swing.tree.TreeSelectionModel;
/**
* Tree where the available node tags are shown in graph windows.
*/
public final class CTagsTree extends DNDTree implements IHelpProvider {
/**
* Used for serialization.
*/
private static final long serialVersionUID = -1074808285623025354L;
/**
* Parent window used for dialogs.
*/
private final JFrame m_parent;
/**
* Graph shown in the window.
*/
private final ZyGraph m_graph;
/**
* Provides tag information.
*/
private final ITagManager m_tagManager;
/**
* Model of the tags tree.
*/
private final CTagsTreeModel m_model;
/**
* Root node of the tags tree. This is an invisible dummy node.
*/
private final CRootTagTreeNode m_rootNode;
/**
* Handles clicks on the tree.
*/
private final InternalMouseListener m_mouseListener = new InternalMouseListener();
/**
* Updates the tree GUI on selection changes in the graph.
*/
private final InternalGraphSelectionListener m_graphSelectionListener =
new InternalGraphSelectionListener();
/**
* Updates the tree GUI on visibility changes in the graph.
*/
private final InternalGraphVisibilityListener m_graphVisibilityListener =
new InternalGraphVisibilityListener();
/**
* The last selection path that can actually be selected.
*/
private TreePath m_lastValidSelectionPath = null;
/**
* The last selected node that can actually be selected.
*/
private CTagTreeNode m_lastValidSelectedNode = null;
/**
* Creates a new tree object.
*
* @param parent Parent window used for dialogs.
* @param graph Graph shown in the window.
* @param manager Provides tag information.
*/
public CTagsTree(final JFrame parent, final ZyGraph graph, final ITagManager manager) {
m_parent = Preconditions.checkNotNull(parent, "IE02308: Perent argument can not be null");
m_graph = Preconditions.checkNotNull(graph, "IE01776: Graph can not be null");
m_tagManager = Preconditions.checkNotNull(manager, "IE01777: Manager argument can not be null");
m_model = new CTagsTreeModel(this);
setModel(m_model);
getModel().addTreeModelListener(new InternalModelListener());
addMouseListener(m_mouseListener);
m_graph.addListener(m_graphSelectionListener);
m_graph.addListener(m_graphVisibilityListener);
setRootVisible(false);
m_rootNode = new CRootTagTreeNode(parent, this, graph, m_tagManager);
m_model.setRoot(m_rootNode);
setCellRenderer(new CTagTreeCellRenderer()); // ATTENTION: UNDER NO CIRCUMSTANCES MOVE THIS LINE
// ABOVE THE SETROOT LINE
m_model.nodeStructureChanged(m_rootNode);
final List<IDropHandler> handlers = new ArrayList<IDropHandler>();
handlers.add(new CTagSortingHandler());
new CDefaultTransferHandler(this, DnDConstants.ACTION_COPY_OR_MOVE, handlers);
final DefaultTreeSelectionModel selectionModel = new DefaultTreeSelectionModel();
selectionModel.setSelectionMode(TreeSelectionModel.SINGLE_TREE_SELECTION);
setSelectionModel(selectionModel);
final InputMap windowImap = getInputMap(JComponent.WHEN_FOCUSED);
windowImap.put(HotKeys.DELETE_HK.getKeyStroke(), "DELETE");
getActionMap().put("DELETE", CActionProxy.proxy(new DeleteAction()));
}
/**
* Shows the context menu for a given mouse event.
*
* @param event The mouse event that triggered the popup menu.
*/
private void showPopupMenu(final MouseEvent event) {
final ITagTreeNode selectedNode =
(ITagTreeNode) TreeHelpers.getNodeAt(this, event.getX(), event.getY());
if (selectedNode == null) {
// Show the default menu
final JPopupMenu popupMenu = new JPopupMenu();
popupMenu.add(CActionProxy.proxy(new CAddRootTagNodeAction(m_parent, m_tagManager, m_rootNode
.getTag())));
popupMenu.show(this, event.getX(), event.getY());
} else {
final JPopupMenu menu = selectedNode.getPopupMenu();
if (menu != null) {
menu.show(this, event.getX(), event.getY());
}
}
}
/**
* Frees allocated resources.
*/
public void dispose() {
m_rootNode.dispose();
removeMouseListener(m_mouseListener);
m_graph.removeListener(m_graphSelectionListener);
m_graph.removeListener(m_graphVisibilityListener);
}
@Override
public IHelpInformation getHelpInformation() {
return new IHelpInformation() {
@Override
public String getText() {
return "This tree is used to configure and assign node tags. "
+ "Once you have assigned tags to nodes you can use this "
+ "tree to quickly select all nodes tagged with given tags.";
}
@Override
public URL getUrl() {
return CHelpFunctions.urlify(CHelpFunctions.MAIN_WINDOW_FILE);
}
};
}
@Override
public CTagsTreeModel getModel() {
return m_model;
}
/**
* Action class used to delete the selected tag.
*/
private class DeleteAction extends AbstractAction {
@Override
public void actionPerformed(final ActionEvent event) {
final Object component = getLastSelectedPathComponent();
if (component instanceof CTagTreeNode) {
CTagFunctions.deleteTag(m_parent, m_tagManager, ((CTagTreeNode) component).getTag());
}
}
}
/**
* Updates the tree GUI on selection changes in the graph.
*/
private class InternalGraphSelectionListener implements IZyGraphSelectionListener {
@Override
public void selectionChanged() {
}
}
/**
* Updates the tree GUI on visibility changes in the graph.
*/
private class InternalGraphVisibilityListener implements IZyGraphVisibilityListener {
@Override
public void nodeDeleted() {
}
@Override
public void visibilityChanged() {
}
}
/**
* Makes sure to update the tree if the model changed.
*/
private class InternalModelListener implements TreeModelListener {
/**
* Updates the tree.
*/
private void update() {
if (m_lastValidSelectedNode == null) {
validate();
} else {
final Integer tagId = (Integer) m_lastValidSelectedNode.getUserObject();
final Enumeration<?> nodes = m_rootNode.breadthFirstEnumeration();
while (nodes.hasMoreElements()) {
final DefaultMutableTreeNode node = (DefaultMutableTreeNode) nodes.nextElement();
if (Objects.equals(node.getUserObject(), tagId)) {
new SwingInvoker() {
@Override
protected void operation() {
m_lastValidSelectionPath = new TreePath(getModel().getPathToRoot(node));
getSelectionModel().setSelectionPath(m_lastValidSelectionPath);
}
};
return;
}
}
}
}
@Override
public void treeNodesChanged(final TreeModelEvent event) {
update();
}
@Override
public void treeNodesInserted(final TreeModelEvent event) {
update();
}
@Override
public void treeNodesRemoved(final TreeModelEvent event) {
update();
}
@Override
public void treeStructureChanged(final TreeModelEvent event) {
update();
}
}
/**
* Handles clicks on the tree.
*/
private class InternalMouseListener extends MouseAdapter {
@Override
public void mousePressed(final MouseEvent event) {
if (event.isPopupTrigger()) {
showPopupMenu(event);
} else {
final int y = event.getY();
final int x = event.getX();
final TreePath path = getPathForLocation(x, y);
if (path == null) {
return;
}
final Object treenode = path.getLastPathComponent();
if (event.getButton() == 1) {
if (treenode instanceof CTagTreeNode) {
m_lastValidSelectedNode = (CTagTreeNode) treenode;
m_lastValidSelectionPath = path;
// avoids flickering
((CTagTreeCellRenderer) getCellRenderer()).setSelectedNode(m_lastValidSelectedNode);
} else if (treenode instanceof CTaggedGraphNodesContainerNode) {
final CTaggedGraphNodesContainerNode containerNode =
(CTaggedGraphNodesContainerNode) treenode;
final Collection<NaviNode> nodes = containerNode.getGraphNodes();
boolean select = false;
int countunselected = 0;
int countinvisible = 0;
for (final NaviNode nn : nodes) {
if (!nn.getRawNode().isSelected()) {
select = true;
countunselected++;
}
if (!nn.getRawNode().isVisible()) {
countinvisible++;
}
}
if (((countinvisible == countunselected) || !select)
&& !m_graph.getSettings().getProximitySettings().getProximityBrowsingFrozen()) {
m_graph.selectNodes(nodes, select);
} else {
final Collection<NaviNode> visiblenodes = new ArrayList<NaviNode>();
for (final NaviNode nn : nodes) {
if (nn.isVisible()) {
visiblenodes.add(nn);
}
}
m_graph.selectNodes(visiblenodes, select);
}
} else if (treenode instanceof CTaggedGraphNodeNode) {
final CTaggedGraphNodeNode graphNode = (CTaggedGraphNodeNode) treenode;
final boolean graphNodeSelected = graphNode.getGraphNode().getRawNode().isSelected();
if (!(m_graph.getSettings().getProximitySettings().getProximityBrowsingFrozen() && !graphNode
.getGraphNode().isVisible())) {
m_graph.selectNode(graphNode.getGraphNode(), !graphNodeSelected);
}
}
new SwingInvoker() {
@Override
protected void operation() {
getSelectionModel().setSelectionPath(m_lastValidSelectionPath);
}
}.invokeLater();
}
}
}
@Override
public void mouseReleased(final MouseEvent event) {
if (event.isPopupTrigger()) {
showPopupMenu(event);
}
final int y = event.getY();
final int x = event.getX();
final TreePath path = getPathForLocation(x, y);
if (path == null) {
return;
}
final Object treenode = path.getLastPathComponent();
if ((event.getButton() == 3) && (treenode instanceof CTaggedGraphNodeNode)) {
final CTaggedGraphNodeNode treeNode = (CTaggedGraphNodeNode) treenode;
final NaviNode graphNode = treeNode.getGraphNode();
if (graphNode.isVisible()) {
if (event.getClickCount() == 1) {
MoveFunctions.centerNode(m_graph, graphNode);
} else if (event.getClickCount() == 2) {
ZoomFunctions.zoomToNode(m_graph, graphNode);
}
}
}
}
}
}