All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.eclipse.draw2d.graph.BreakCycles Maven / Gradle / Ivy

/*******************************************************************************
 * Copyright (c) 2003, 2010, 2012 IBM Corporation, Gerhardt Informatics Kft. and others.
 * All rights reserved. This program and the accompanying materials
 * are made available under the terms of the Eclipse Public License v1.0
 * which accompanies this distribution, and is available at
 * http://www.eclipse.org/legal/epl-v10.html
 *
 * Contributors:
 *     IBM Corporation - initial API and implementation
 *     Gerhardt Informatics Kft. - GEFGWT port
 *******************************************************************************/
package org.eclipse.draw2d.graph;

import java.util.ArrayList;
import java.util.List;

/**
 * This visitor eliminates cycles in the graph using a "greedy" heuristic. Nodes
 * which are sources and sinks are marked and placed in a source and sink list,
 * leaving only nodes involved in cycles. A remaining node with the highest
 * (outgoing-incoming) edges score is then chosen greedily as if it were a
 * source. The process is repeated until all nodes have been marked and placed
 * in a list. The lists are then concatenated, and any edges which go backwards
 * in this list will be inverted during the layout procedure.
 * 
 * @author Daniel Lee
 * @since 2.1.2
 */
class BreakCycles extends GraphVisitor {

	// Used in identifying cycles and in cycle removal.
	// Flag field indicates "presence". If true, the node has been removed from
	// the list.
	NodeList graphNodes = new NodeList();

	private boolean allNodesFlagged() {
		for (int i = 0; i < graphNodes.size(); i++) {
			if (graphNodes.getNode(i).flag == false)
				return false;
		}
		return true;
	}

	private void breakCycles(DirectedGraph g) {
		initializeDegrees(g);
		greedyCycleRemove(g);
		invertEdges(g);
	}

	/*
	 * Returns true if g contains cycles, false otherwise
	 */
	private boolean containsCycles(DirectedGraph g) {
		List noLefts = new ArrayList();
		// Identify all initial nodes for removal
		for (int i = 0; i < graphNodes.size(); i++) {
			Node node = graphNodes.getNode(i);
			if (getIncomingCount(node) == 0)
				sortedInsert(noLefts, node);
		}

		while (noLefts.size() > 0) {
			Node node = (Node) noLefts.remove(noLefts.size() - 1);
			node.flag = true;
			for (int i = 0; i < node.outgoing.size(); i++) {
				Node right = node.outgoing.getEdge(i).target;
				setIncomingCount(right, getIncomingCount(right) - 1);
				if (getIncomingCount(right) == 0)
					sortedInsert(noLefts, right);
			}
		}

		if (allNodesFlagged())
			return false;
		return true;
	}

	/*
	 * Returns the node in graphNodes with the largest (outgoing edge count -
	 * incoming edge count) value
	 */
	private Node findNodeWithMaxDegree() {
		int max = Integer.MIN_VALUE;
		Node maxNode = null;

		for (int i = 0; i < graphNodes.size(); i++) {
			Node node = graphNodes.getNode(i);
			if (getDegree(node) >= max && node.flag == false) {
				max = getDegree(node);
				maxNode = node;
			}
		}
		return maxNode;
	}

	private int getDegree(Node n) {
		return n.workingInts[3];
	}

	private int getIncomingCount(Node n) {
		return n.workingInts[0];
	}

	private int getInDegree(Node n) {
		return n.workingInts[1];
	}

	private int getOrderIndex(Node n) {
		return n.workingInts[0];
	}

	private int getOutDegree(Node n) {
		return n.workingInts[2];
	}

	private void greedyCycleRemove(DirectedGraph g) {
		NodeList sL = new NodeList();
		NodeList sR = new NodeList();

		do {
			// Add all sinks and isolated nodes to sR
			boolean hasSink;
			do {
				hasSink = false;
				for (int i = 0; i < graphNodes.size(); i++) {
					Node node = graphNodes.getNode(i);
					if (getOutDegree(node) == 0 && node.flag == false) {
						hasSink = true;
						node.flag = true;
						updateIncoming(node);
						sR.add(node);
						break;
					}
				}
			} while (hasSink);

			// Add all sources to sL
			boolean hasSource;
			do {
				hasSource = false;
				for (int i = 0; i < graphNodes.size(); i++) {
					Node node = graphNodes.getNode(i);
					if (getInDegree(node) == 0 && node.flag == false) {
						hasSource = true;
						node.flag = true;
						updateOutgoing(node);
						sL.add(node);
						break;
					}
				}
			} while (hasSource);

			// When all sinks and sources are removed, choose a node with the
			// maximum degree (outDegree - inDegree) and add it to sL
			Node max = findNodeWithMaxDegree();
			if (max != null) {
				sL.add(max);
				max.flag = true;
				updateIncoming(max);
				updateOutgoing(max);
			}
		} while (!allNodesFlagged());

		// Assign order indexes
		int orderIndex = 0;
		for (int i = 0; i < sL.size(); i++) {
			setOrderIndex(sL.getNode(i), orderIndex++);
		}
		for (int i = sR.size() - 1; i >= 0; i--) {
			setOrderIndex(sR.getNode(i), orderIndex++);
		}
	}

	private void initializeDegrees(DirectedGraph g) {
		graphNodes.resetFlags();
		for (int i = 0; i < g.nodes.size(); i++) {
			Node n = graphNodes.getNode(i);
			setInDegree(n, n.incoming.size());
			setOutDegree(n, n.outgoing.size());
			setDegree(n, n.outgoing.size() - n.incoming.size());
		}
	}

	private void invertEdges(DirectedGraph g) {
		for (int i = 0; i < g.edges.size(); i++) {
			Edge e = g.edges.getEdge(i);
			if (getOrderIndex(e.source) > getOrderIndex(e.target)) {
				e.invert();
				e.isFeedback = true;
			}
		}
	}

	private void setDegree(Node n, int deg) {
		n.workingInts[3] = deg;
	}

	private void setIncomingCount(Node n, int count) {
		n.workingInts[0] = count;
	}

	private void setInDegree(Node n, int deg) {
		n.workingInts[1] = deg;
	}

	private void setOutDegree(Node n, int deg) {
		n.workingInts[2] = deg;
	}

	private void setOrderIndex(Node n, int index) {
		n.workingInts[0] = index;
	}

	private void sortedInsert(List list, Node node) {
		int insert = 0;
		while (insert < list.size()
				&& ((Node) list.get(insert)).sortValue > node.sortValue)
			insert++;
		list.add(insert, node);
	}

	/*
	 * Called after removal of n. Updates the degree values of n's incoming
	 * nodes.
	 */
	private void updateIncoming(Node n) {
		for (int i = 0; i < n.incoming.size(); i++) {
			Node in = n.incoming.getEdge(i).source;
			if (in.flag == false) {
				setOutDegree(in, getOutDegree(in) - 1);
				setDegree(in, getOutDegree(in) - getInDegree(in));
			}
		}
	}

	/*
	 * Called after removal of n. Updates the degree values of n's outgoing
	 * nodes.
	 */
	private void updateOutgoing(Node n) {
		for (int i = 0; i < n.outgoing.size(); i++) {
			Node out = n.outgoing.getEdge(i).target;
			if (out.flag == false) {
				setInDegree(out, getInDegree(out) - 1);
				setDegree(out, getOutDegree(out) - getInDegree(out));
			}
		}
	}

	public void revisit(DirectedGraph g) {
		for (int i = 0; i < g.edges.size(); i++) {
			Edge e = g.edges.getEdge(i);
			if (e.isFeedback())
				e.invert();
		}
	}

	/**
	 * @see GraphVisitor#visit(org.eclipse.draw2d.graph.DirectedGraph)
	 */
	public void visit(DirectedGraph g) {
		// put all nodes in list, initialize index
		graphNodes.resetFlags();
		for (int i = 0; i < g.nodes.size(); i++) {
			Node n = g.nodes.getNode(i);
			setIncomingCount(n, n.incoming.size());
			graphNodes.add(n);
		}
		if (containsCycles(g)) {
			breakCycles(g);
		}
	}

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy