view java/src/uk/ac/qmul/eecs/ccmi/gui/Diagram.java @ 8:ea7885bd9bff tip

fixed bug : render solid line as dotted/dashed when moving the stylus from dotted/dashed to solid
author ccmi-guest
date Thu, 03 Jul 2014 16:12:20 +0100
parents 9e67171477bc
children
line wrap: on
line source
/*  
 CCmI Editor - A Collaborative Cross-Modal Diagram Editing Tool
  
 Copyright (C) 2011  Queen Mary University of London (http://ccmi.eecs.qmul.ac.uk/)

 This program is free software: you can redistribute it and/or modify
 it under the terms of the GNU General Public License as published by
 the Free Software Foundation, either version 3 of the License, or
 (at your option) any later version.

 This program 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 General Public License for more details.

 You should have received a copy of the GNU General Public License
 along with this program.  If not, see <http://www.gnu.org/licenses/>.
*/
package uk.ac.qmul.eecs.ccmi.gui;

import java.awt.geom.Point2D;
import java.util.Set;

import uk.ac.qmul.eecs.ccmi.diagrammodel.CollectionModel;
import uk.ac.qmul.eecs.ccmi.diagrammodel.DiagramElement;
import uk.ac.qmul.eecs.ccmi.diagrammodel.DiagramModel;
import uk.ac.qmul.eecs.ccmi.diagrammodel.DiagramTreeNode;
import uk.ac.qmul.eecs.ccmi.diagrammodel.NodeProperties;
import uk.ac.qmul.eecs.ccmi.diagrammodel.TreeModel;
import uk.ac.qmul.eecs.ccmi.gui.persistence.PrototypePersistenceDelegate;
import uk.ac.qmul.eecs.ccmi.network.AwarenessMessage;
import uk.ac.qmul.eecs.ccmi.network.DiagramEventActionSource;

/**
 * The {@code Diagram} class holds all the data needed for a representation of the diagram. It is used by component classes 
 * such as {@link GraphPanel} and {@link DiagramTree} to draw the diagram by accessing the diagram model or by 
 * {@link EditorTabbedPane} to assign a title to the tabs out of the diagram name. 
 *
 */
public abstract class Diagram implements Cloneable {
	
	/**
	 * Crates a new instance of a Diagram. The diagram created through this method is not shared with any peer via 
	 * a server. 
	 * @param name the name of the diagram.
	 * @param nodes an array of node prototypes. Nodes inserted by users in the diagram will be created by cloning these nodes. 
	 * @param edges an array of edge prototypes. Edges inserted by users in the diagram will be created by cloning these edges.
	 * @param prototypePersistenceDelegate a delegate class to handle nodes and edges persistence.
	 * @return a new instance of {@code Diagram}
	 */
	public static Diagram newInstance(String name, Node[] nodes, Edge[] edges, PrototypePersistenceDelegate prototypePersistenceDelegate){
		return new LocalDiagram(name,nodes,edges,prototypePersistenceDelegate);
	}
	
	/**
	 * Returns the name of the diagram. The name identifies the diagram uniquely in the editor. There cannot
	 * be two diagrams with the same name open at the same time. This makes things easier when sharing diagrams
	 * with other users via the network.
	 * 
	 * @return the name of the diagram
	 */
	public abstract String getName();

	/**
	 * Assign this diagram a new name. 
	 * @param name the new name of the diagram
	 */
	public abstract void setName(String name);

	/**
	 * Returns an array with the node prototypes. Node prototypes are used when creating new node 
	 * instances via the {@code clone()} method. 
	 * 
	 * @return an array of nodes 
	 */
	public abstract Node[] getNodePrototypes();

	/**
	 * Returns an array with the edge prototypes. Edge prototypes are used when creating new edge 
	 * instances via the {@code clone()} method.
	 * 
	 * @return an array of edges
	 */
	public abstract Edge[] getEdgePrototypes();

	/**
	 * Returns the tree model of this diagram. Note that each diagram holds a {@code DiagramModel} 
	 * which has  two sub-models ({@code TreeModel} and {@code CollectionModel}). Changes on one 
	 * sub-model will affect the other model as well. 
	 * 
	 * @return the tree model of this diagram 
	 * 
	 * @see uk.ac.qmul.eecs.ccmi.diagrammodel.DiagramModel uk.ac.qmul.eecs.ccmi.diagrammodel.DiagramModel
	 */
	public abstract TreeModel<Node,Edge> getTreeModel();

	/**
	 * Returns the collection model of this diagram. Note that each diagram holds a {@code DiagramModel} 
	 * which has  two sub-models ({@code TreeModel} and {@code CollectionModel}). Changes on one 
	 * sub-model will affect the other model as well. 
	 * 
	 * @return the tree model of this diagram 
	 * 
	 * @see uk.ac.qmul.eecs.ccmi.diagrammodel.DiagramModel uk.ac.qmul.eecs.ccmi.diagrammodel.DiagramModel
	 */
	public abstract CollectionModel<Node,Edge> getCollectionModel();

	/**
	 * Returns the model updater of this diagram. The model updater is the delegate for all the 
	 * update operations affecting the diagram model.
	 * 
	 * @return the model updater for this diagram
	 */
	public abstract DiagramModelUpdater getModelUpdater();
	
	/**
	 * Returns the label of the diagram. The label is slightly different from the name as it's the string 
	 * appearing in the tabbed pane of the editor. It includes asterisk character at the end when the {@code DiagramModel}
	 * of this class has been changed and not  yet saved on hard disk.   
	 *  
	 * @return a label for this diagram
	 */
	public abstract String getLabel();
	
	/**
	 * Returns the delegates for this diagram for nodes and edges prototypes persistence. 
	 * When saving a diagram to an xml file each node and edge of the prototypes is encoded  
	 * in the xml file. Indeed the template of a diagram is made of of its prototypes. 
	 * In the template is held the general attributes common to all the nodes and edges, like 
	 * for instance the type of a node but not its current position. 
	 *  
	 * @return the PrototypePersistenceDelegate for this diagram 
	 */
	public abstract PrototypePersistenceDelegate getPrototypePersistenceDelegate();

	@Override
	public Object clone(){
		try {
			return super.clone();
		} catch (CloneNotSupportedException e) {
			throw new RuntimeException(e);
		}
	}
	
	private static class LocalDiagram extends Diagram {

		private LocalDiagram(String name, Node[] nodes, Edge[] edges,PrototypePersistenceDelegate prototypePersistenceDelegate){
			this.name = name;
			this.nodes = nodes;
			this.edges = edges;
			this.prototypePersistenceDelegate = prototypePersistenceDelegate;
			diagramModel = new DiagramModel<Node,Edge>(nodes,edges);
			innerModelUpdater = new InnerModelUpdater();
		}

		@Override
		public String getName(){
			return name;
		}

		@Override
		public void setName(String name){
			this.name = name;
		}

		@Override
		public Node[] getNodePrototypes(){
			return nodes;
		}

		@Override
		public Edge[] getEdgePrototypes(){
			return edges;
		}

		@Override
		public TreeModel<Node,Edge> getTreeModel(){
			return diagramModel.getTreeModel();
		}

		@Override
		public CollectionModel<Node,Edge> getCollectionModel(){
			return diagramModel.getDiagramCollection();
		}

		@Override
		public String getLabel(){
			return name;
		}
		
		@Override
		public DiagramModelUpdater getModelUpdater(){
			return innerModelUpdater;
		}
		
		@Override
		public String toString(){
			return name;
		}
		
		@Override
		public PrototypePersistenceDelegate getPrototypePersistenceDelegate(){
			return prototypePersistenceDelegate;
		}
		
		/**
		 * Creates a new {@code Diagram} by clonation.  
		 */
		@Override
		public Object clone(){
			LocalDiagram clone = (LocalDiagram)super.clone();
			clone.name = getName();
			clone.nodes = getNodePrototypes();
			clone.edges = getEdgePrototypes();
			/* constructor with no args makes just a dummy wrapper */
			clone.diagramModel = new DiagramModel<Node,Edge>(nodes,edges);
			clone.innerModelUpdater = clone.new InnerModelUpdater();
			return clone;
		}

		private DiagramModel<Node,Edge> diagramModel;
		private InnerModelUpdater innerModelUpdater;
		private PrototypePersistenceDelegate prototypePersistenceDelegate;
		private String name;
		private Node[] nodes;
		private Edge[] edges;

		private class InnerModelUpdater implements DiagramModelUpdater {

			@Override
			public boolean getLock(DiagramTreeNode treeNode, Lock lock, DiagramEventActionSource source) {
				/* using a non shared diagram requires no actual lock, therefore the answer is always yes */
				return true;
			}
			
			@Override
			public void yieldLock(DiagramTreeNode treeNode, Lock lock, DiagramEventActionSource actionSource) {}

			@Override 
			public void sendAwarenessMessage(AwarenessMessage.Name awMsgName, Object source){}
			
			@Override
			public void insertInCollection(DiagramElement element,DiagramEventSource source) {
				if(element instanceof Node)
					diagramModel.getDiagramCollection().insert((Node)element,source);
				else
					diagramModel.getDiagramCollection().insert((Edge)element,source);
			}

			@Override
			public void insertInTree(DiagramElement element) {
				if(element instanceof Node)
					diagramModel.getTreeModel().insertTreeNode((Node)element,DiagramEventSource.TREE);
				else
					diagramModel.getTreeModel().insertTreeNode((Edge)element,DiagramEventSource.TREE);
			}

			@Override
			public void takeOutFromCollection(DiagramElement element, DiagramEventSource source) {
				diagramModel.getDiagramCollection().takeOut(element,source);
			}

			@Override
			public void takeOutFromTree(DiagramElement element) {
				diagramModel.getTreeModel().takeTreeNodeOut(element,DiagramEventSource.TREE);
			}

			@Override
			public void setName(DiagramElement element, String name,DiagramEventSource source) {
				element.setName(name,source);
			}

			@Override
			public void setNotes(DiagramTreeNode treeNode, String notes,DiagramEventSource source) {
				diagramModel.getTreeModel().setNotes(treeNode, notes,source);
			}

			@Override
			public void setProperty(Node node, String type, int index,
					String value,DiagramEventSource source) {
				node.setProperty(type, index, value,source);
			}

			@Override
			public void setProperties(Node node, NodeProperties properties,DiagramEventSource source) {
				node.setProperties(properties,source);
			}

			@Override
			public void clearProperties(Node node,DiagramEventSource source) {
				node.clearProperties(source);
			}

			@Override
			public void addProperty(Node node, String type, String value,DiagramEventSource source) {
				node.addProperty(type, value,source);
			}

			@Override
			public void removeProperty(Node node, String type, int index,DiagramEventSource source) {
				node.removeProperty(type, index,source);
			}

			@Override
			public void setModifiers(Node node, String type, int index,
					Set<Integer> modifiers,DiagramEventSource source) {
				node.setModifierIndexes(type, index, modifiers,source);				
			}

			@Override
			public void setEndLabel(Edge edge, Node node, String label,DiagramEventSource source) {
				edge.setEndLabel(node, label,source);
			}

			@Override
			public void setEndDescription(Edge edge, Node node,
					int index,DiagramEventSource source) {
				edge.setEndDescription(node, index,source);
			}

			@Override
			public void translate(GraphElement ge, Point2D p, double x, double y,DiagramEventSource source) {
				ge.translate(p, x, y,source);
			}

			@Override
			public void startMove(GraphElement ge, Point2D p,DiagramEventSource source) {
				ge.startMove(p,source);
			}

			@Override
			public void bend(Edge edge, Point2D p,DiagramEventSource source) {
				edge.bend(p,source);
			}

			@Override
			public void stopMove(GraphElement ge,DiagramEventSource source) {
				ge.stopMove(source);
			}
		}
	}

}