/*
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.yfileswrap.Gui.GraphWindows;
import com.google.common.base.Preconditions;
import com.google.security.zynamics.binnavi.Gui.GraphWindows.CGraphModel;
import com.google.security.zynamics.binnavi.Gui.GraphWindows.Extensions.ICodeNodeExtension;
import com.google.security.zynamics.binnavi.Gui.GraphWindows.Implementations.CNodeHoverer;
import com.google.security.zynamics.binnavi.Gui.GraphWindows.Menu.CGraphWindowMenuBar;
import com.google.security.zynamics.binnavi.ZyGraph.Implementations.CProximityFunctions;
import com.google.security.zynamics.binnavi.ZyGraph.Menus.CEdgeMenu;
import com.google.security.zynamics.binnavi.ZyGraph.Menus.CProximityNodeMenu;
import com.google.security.zynamics.binnavi.ZyGraph.Updaters.CNodeUpdaterInitializer;
import com.google.security.zynamics.binnavi.disassembly.INaviViewNode;
import com.google.security.zynamics.binnavi.disassembly.views.CViewListenerAdapter;
import com.google.security.zynamics.binnavi.disassembly.views.INaviView;
import com.google.security.zynamics.binnavi.yfileswrap.zygraph.NaviEdge;
import com.google.security.zynamics.binnavi.yfileswrap.zygraph.NaviGraphListenerAdapter;
import com.google.security.zynamics.binnavi.yfileswrap.zygraph.NaviNode;
import com.google.security.zynamics.binnavi.yfileswrap.zygraph.ZyGraph;
import com.google.security.zynamics.zylib.yfileswrap.gui.zygraph.proximity.ZyProximityNode;
import java.awt.event.MouseEvent;
import java.util.ArrayList;
import java.util.List;
import javax.swing.JPopupMenu;
/**
* Synchronizes events in graph and view objects with the displayed graph.
*/
public final class CGraphPanelSynchronizer {
/**
* Model of the displayed graph.
*/
private final CGraphModel m_model;
/**
* Graph-specific menu bar that is shown when the graph is active.
*/
private final CGraphWindowMenuBar m_menuBar;
/**
* Handles clicks on nodes.
*/
private final CNodeClickHandler m_clickHandler;
/**
* Updates the displayed graph on relevant changes in the underlying view object.
*/
private final InternalViewListener m_viewListener = new InternalViewListener();
/**
* Updates the displayed graph on relevant changes in the graph.
*/
private final InternalGraphListener m_graphListener = new InternalGraphListener();
/**
* Flag that indicates whether the graph panel has requested the view to be closed or not.
*/
private boolean closing = false;
/**
* Extension objects that add menu items to the context menu of code nodes.
*/
private final List<ICodeNodeExtension> m_codeNodeExtensions = new ArrayList<ICodeNodeExtension>();
/**
* Creates a new synchronizer object.
*
* @param model Model of the displayed graph.
* @param menuBar Graph-specific menu bar that is shown when the graph is active.
*/
public CGraphPanelSynchronizer(final CGraphModel model, final CGraphWindowMenuBar menuBar) {
m_model = Preconditions.checkNotNull(model, "IE01618: Model argument can not be null");
m_menuBar = Preconditions.checkNotNull(menuBar, "IE01620: Menu bar argument can not be null");
m_clickHandler = new CNodeClickHandler(model);
m_model.getGraph().addListener(m_graphListener);
m_model.getGraph().getRawView().addListener(m_viewListener);
}
/**
* Frees allocated resources.
*/
public void dispose() {
closing = true;
m_model.getGraph().getRawView().removeListener(m_viewListener);
m_model.getGraph().removeListener(m_graphListener);
}
/**
* Adds a new extension object that adds entries to the context menu of code nodes.
*
* @param extension The extension object to add.
*/
public void registerCodeNodeContextMenuExtension(final ICodeNodeExtension extension) {
Preconditions.checkNotNull(extension, "IE01621: Extension argument can not be null");
m_codeNodeExtensions.add(extension);
}
/**
* Updates the displayed graph on relevant changes in the graph.
*/
private class InternalGraphListener extends NaviGraphListenerAdapter {
/**
* Highlights lines on mouse-over.
*/
private final CNodeHoverer m_nodeHoverer = new CNodeHoverer();
/**
* Handles proximity-browsing related updating events.
*/
private final CProximityFunctions m_proximityFunctions = new CProximityFunctions();
@Override
public void addedNode(final ZyGraph graph, final NaviNode node) {
CNodeUpdaterInitializer.addUpdaters(m_model, node);
}
@Override
public void changedModel(final ZyGraph graph, final NaviNode node) {
CNodeUpdaterInitializer.addUpdaters(m_model, node);
}
@Override
public void changedView(final INaviView oldView, final INaviView newView) {
oldView.removeListener(m_viewListener);
newView.addListener(m_viewListener);
m_menuBar.updateGui();
}
@Override
public void edgeClicked(final NaviEdge edge, final MouseEvent event, final double x,
final double y) {
if (event.getButton() == MouseEvent.BUTTON3) {
final JPopupMenu menu = new CEdgeMenu(m_model.getParent(), m_model.getGraph(), edge);
menu.show(m_model.getGraph().getView(), event.getX(), event.getY());
}
}
@Override
public void nodeClicked(final NaviNode node, final MouseEvent event, final double x,
final double y) {
m_clickHandler.nodeClicked(node, event, x, y, m_codeNodeExtensions);
}
@Override
public void nodeHovered(final NaviNode node, final double x, final double y) {
m_nodeHoverer.nodeHovered(node, y);
m_model.getGraph().updateViews();
}
@Override
public void nodeLeft(final NaviNode node) {
if (m_nodeHoverer.clear(node)) {
m_model.getGraph().updateViews();
}
}
@SuppressWarnings("unchecked")
@Override
public void proximityBrowserNodeClicked(final ZyProximityNode<?> proximityNode,
final MouseEvent event, final double x, final double y) {
if (event.getButton() == MouseEvent.BUTTON1) {
if (!event.isShiftDown() && !event.isControlDown()) {
m_proximityFunctions.showHiddenNodes(m_model.getParent(), m_model.getGraph(),
(ZyProximityNode<INaviViewNode>) proximityNode);
} else if (event.isShiftDown()) {
m_proximityFunctions.unhideAndSelect(m_model.getGraph(),
(ZyProximityNode<INaviViewNode>) proximityNode);
} else if (event.isControlDown()) {
m_proximityFunctions.unhideAndSelectOnly(m_model.getGraph(),
(ZyProximityNode<INaviViewNode>) proximityNode);
}
} else if (event.getButton() == MouseEvent.BUTTON3) {
final JPopupMenu menu =
new CProximityNodeMenu(m_model.getParent(), m_model.getGraph(),
(ZyProximityNode<INaviViewNode>) proximityNode);
menu.show(m_model.getGraph().getView(), event.getX(), event.getY());
}
}
}
/**
* Updates the displayed graph on relevant changes in the underlying view object.
*/
private class InternalViewListener extends CViewListenerAdapter {
@Override
public boolean closingView(final INaviView view) {
return view != m_model.getGraph().getRawView() || closing;
}
}
}