/* Copyright 2008-2010 Gephi Authors : Mathieu Bastian <mathieu.bastian@gephi.org> Website : http://www.gephi.org This file is part of Gephi. Gephi is free software: you can redistribute it and/or modify it under the terms of the GNU Affero General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. Gephi is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public License for more details. You should have received a copy of the GNU Affero General Public License along with Gephi. If not, see <http://www.gnu.org/licenses/>. */ package org.gephi.tools.plugin; import java.awt.Color; import javax.swing.Icon; import javax.swing.ImageIcon; import javax.swing.JPanel; import org.gephi.algorithms.shortestpath.AbstractShortestPathAlgorithm; import org.gephi.algorithms.shortestpath.BellmanFordShortestPathAlgorithm; import org.gephi.algorithms.shortestpath.DijkstraShortestPathAlgorithm; import org.gephi.graph.api.DirectedGraph; import org.gephi.graph.api.Edge; import org.gephi.graph.api.GraphController; import org.gephi.graph.api.Node; import org.gephi.tools.spi.MouseClickEventListener; import org.gephi.tools.spi.NodeClickEventListener; import org.gephi.tools.spi.Tool; import org.gephi.tools.spi.ToolEventListener; import org.gephi.tools.spi.ToolSelectionType; import org.gephi.ui.tools.plugin.ShortestPathPanel; import org.gephi.tools.spi.ToolUI; import org.gephi.visualization.VizController; import org.openide.util.Lookup; import org.openide.util.NbBundle; import org.openide.util.lookup.ServiceProvider; /** * * @author Mathieu Bastian */ @ServiceProvider(service = Tool.class) public class ShortestPath implements Tool { //Architecture private ToolEventListener[] listeners; private ShortestPathPanel shortestPathPanel; //Settings private Color color; private boolean settingEdgeSourceColor; //State private Node sourceNode; public ShortestPath() { //Default settings color = Color.RED; } public boolean select() { settingEdgeSourceColor = !VizController.getInstance().getVizModel().isEdgeHasUniColor(); VizController.getInstance().getVizModel().setEdgeHasUniColor(true); VizController.getInstance().getVizConfig().setEnableAutoSelect(false); return false; } public void unselect() { listeners = null; sourceNode = null; shortestPathPanel = null; VizController.getInstance().getVizModel().setEdgeHasUniColor(settingEdgeSourceColor); VizController.getInstance().getVizConfig().setEnableAutoSelect(true); } public ToolEventListener[] getListeners() { listeners = new ToolEventListener[2]; listeners[0] = new NodeClickEventListener() { public void clickNodes(Node[] nodes) { Node n = nodes[0]; if (sourceNode == null) { sourceNode = n; shortestPathPanel.setResult(""); shortestPathPanel.setStatus(NbBundle.getMessage(ShortestPath.class, "ShortestPath.status2")); } else if (n != sourceNode) { color = shortestPathPanel.getColor(); float[] colorArray = new float[]{color.getRed() / 255f, color.getGreen() / 255f, color.getBlue() / 255f}; Node targetNode = n; GraphController gc = Lookup.getDefault().lookup(GraphController.class); AbstractShortestPathAlgorithm algorithm; if (gc.getModel().getGraphVisible() instanceof DirectedGraph) { algorithm = new BellmanFordShortestPathAlgorithm((DirectedGraph) gc.getModel().getGraphVisible(), sourceNode); } else { algorithm = new DijkstraShortestPathAlgorithm(gc.getModel().getGraphVisible(), sourceNode); } algorithm.compute(); double distance; if ((distance = algorithm.getDistances().get(targetNode)) != Double.POSITIVE_INFINITY) { targetNode.getNodeData().setColor(colorArray[0], colorArray[1], colorArray[2]); VizController.getInstance().selectNode(targetNode); Edge predecessorEdge = algorithm.getPredecessorIncoming(targetNode); Node predecessor = algorithm.getPredecessor(targetNode); while (predecessorEdge != null && predecessor != sourceNode) { predecessorEdge.getEdgeData().setColor(colorArray[0], colorArray[1], colorArray[2]); VizController.getInstance().selectEdge(predecessorEdge); predecessor.getNodeData().setColor(colorArray[0], colorArray[1], colorArray[2]); VizController.getInstance().selectNode(predecessor); predecessorEdge = algorithm.getPredecessorIncoming(predecessor); predecessor = algorithm.getPredecessor(predecessor); } predecessorEdge.getEdgeData().setColor(colorArray[0], colorArray[1], colorArray[2]); VizController.getInstance().selectEdge(predecessorEdge); sourceNode.getNodeData().setColor(colorArray[0], colorArray[1], colorArray[2]); VizController.getInstance().selectNode(sourceNode); shortestPathPanel.setResult(NbBundle.getMessage(ShortestPath.class, "ShortestPath.result", distance)); } else { //No path shortestPathPanel.setResult(NbBundle.getMessage(ShortestPath.class, "ShortestPath.noresult")); } sourceNode = null; shortestPathPanel.setStatus(NbBundle.getMessage(ShortestPath.class, "ShortestPath.status1")); } } }; listeners[1] = new MouseClickEventListener() { public void mouseClick(int[] positionViewport, float[] position3d) { if (sourceNode != null) { //Cancel shortestPathPanel.setStatus(NbBundle.getMessage(ShortestPath.class, "ShortestPath.status1")); sourceNode = null; } else { VizController.getInstance().resetSelection(); } } }; return listeners; } public ToolUI getUI() { return new ToolUI() { public JPanel getPropertiesBar(Tool tool) { shortestPathPanel = new ShortestPathPanel(); shortestPathPanel.setColor(color); shortestPathPanel.setStatus(NbBundle.getMessage(ShortestPath.class, "ShortestPath.status1")); return shortestPathPanel; } public String getName() { return NbBundle.getMessage(ShortestPath.class, "ShortestPath.name"); } public Icon getIcon() { return new ImageIcon(getClass().getResource("/org/gephi/tools/plugin/resources/shortestpath.png")); } public String getDescription() { return NbBundle.getMessage(ShortestPath.class, "ShortestPath.description"); } public int getPosition() { return 140; } }; } public ToolSelectionType getSelectionType() { return ToolSelectionType.SELECTION; } }