241 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			241 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| /**
 | |
|  * Copyright (c) 2006-2015, JGraph Ltd
 | |
|  * Copyright (c) 2006-2015, Gaudenz Alder
 | |
|  */
 | |
| /**
 | |
|  * Class: mxPartitionLayout
 | |
|  * 
 | |
|  * Extends <mxGraphLayout> for partitioning the parent cell vertically or
 | |
|  * horizontally by filling the complete area with the child cells. A horizontal
 | |
|  * layout partitions the height of the given parent whereas a a non-horizontal
 | |
|  * layout partitions the width. If the parent is a layer (that is, a child of
 | |
|  * the root node), then the current graph size is partitioned. The children do
 | |
|  * not need to be connected for this layout to work.
 | |
|  * 
 | |
|  * Example:
 | |
|  * 
 | |
|  * (code)
 | |
|  * var layout = new mxPartitionLayout(graph, true, 10, 20);
 | |
|  * layout.execute(graph.getDefaultParent());
 | |
|  * (end)
 | |
|  * 
 | |
|  * Constructor: mxPartitionLayout
 | |
|  * 
 | |
|  * Constructs a new stack layout layout for the specified graph,
 | |
|  * spacing, orientation and offset.
 | |
|  */
 | |
| function mxPartitionLayout(graph, horizontal, spacing, border)
 | |
| {
 | |
| 	mxGraphLayout.call(this, graph);
 | |
| 	this.horizontal = (horizontal != null) ? horizontal : true;
 | |
| 	this.spacing = spacing || 0;
 | |
| 	this.border = border || 0;
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * Extends mxGraphLayout.
 | |
|  */
 | |
| mxPartitionLayout.prototype = new mxGraphLayout();
 | |
| mxPartitionLayout.prototype.constructor = mxPartitionLayout;
 | |
| 
 | |
| /**
 | |
|  * Variable: horizontal
 | |
|  * 
 | |
|  * Boolean indicating the direction in which the space is partitioned.
 | |
|  * Default is true.
 | |
|  */
 | |
| mxPartitionLayout.prototype.horizontal = null;
 | |
| 
 | |
| /**
 | |
|  * Variable: spacing
 | |
|  * 
 | |
|  * Integer that specifies the absolute spacing in pixels between the
 | |
|  * children. Default is 0.
 | |
|  */
 | |
| mxPartitionLayout.prototype.spacing = null;
 | |
| 
 | |
| /**
 | |
|  * Variable: border
 | |
|  * 
 | |
|  * Integer that specifies the absolute inset in pixels for the parent that
 | |
|  * contains the children. Default is 0.
 | |
|  */
 | |
| mxPartitionLayout.prototype.border = null;
 | |
| 
 | |
| /**
 | |
|  * Variable: resizeVertices
 | |
|  * 
 | |
|  * Boolean that specifies if vertices should be resized. Default is true.
 | |
|  */
 | |
| mxPartitionLayout.prototype.resizeVertices = true;
 | |
| 
 | |
| /**
 | |
|  * Function: isHorizontal
 | |
|  * 
 | |
|  * Returns <horizontal>.
 | |
|  */
 | |
| mxPartitionLayout.prototype.isHorizontal = function()
 | |
| {
 | |
| 	return this.horizontal;
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * Function: moveCell
 | |
|  * 
 | |
|  * Implements <mxGraphLayout.moveCell>.
 | |
|  */
 | |
| mxPartitionLayout.prototype.moveCell = function(cell, x, y)
 | |
| {
 | |
| 	var model = this.graph.getModel();
 | |
| 	var parent = model.getParent(cell);
 | |
| 	
 | |
| 	if (cell != null &&
 | |
| 		parent != null)
 | |
| 	{
 | |
| 		var i = 0;
 | |
| 		var last = 0;
 | |
| 		var childCount = model.getChildCount(parent);
 | |
| 		
 | |
| 		// Finds index of the closest swimlane
 | |
| 		// TODO: Take into account the orientation
 | |
| 		for (i = 0; i < childCount; i++)
 | |
| 		{
 | |
| 			var child = model.getChildAt(parent, i);
 | |
| 			var bounds = this.getVertexBounds(child);
 | |
| 			
 | |
| 			if (bounds != null)
 | |
| 			{
 | |
| 				var tmp = bounds.x + bounds.width / 2;
 | |
| 				
 | |
| 				if (last < x && tmp > x)
 | |
| 				{
 | |
| 					break;
 | |
| 				}
 | |
| 				
 | |
| 				last = tmp;
 | |
| 			}
 | |
| 		}
 | |
| 		
 | |
| 		// Changes child order in parent
 | |
| 		var idx = parent.getIndex(cell);
 | |
| 		idx = Math.max(0, i - ((i > idx) ? 1 : 0));
 | |
| 		
 | |
| 		model.add(parent, cell, idx);
 | |
| 	}
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * Function: execute
 | |
|  * 
 | |
|  * Implements <mxGraphLayout.execute>. All children where <isVertexIgnored>
 | |
|  * returns false and <isVertexMovable> returns true are modified.
 | |
|  */
 | |
| mxPartitionLayout.prototype.execute = function(parent)
 | |
| {
 | |
| 	var horizontal = this.isHorizontal();
 | |
| 	var model = this.graph.getModel();
 | |
| 	var pgeo = model.getGeometry(parent);
 | |
| 	
 | |
| 	// Handles special case where the parent is either a layer with no
 | |
| 	// geometry or the current root of the view in which case the size
 | |
| 	// of the graph's container will be used.
 | |
| 	if (this.graph.container != null &&
 | |
| 		((pgeo == null &&
 | |
| 		model.isLayer(parent)) ||
 | |
| 		parent == this.graph.getView().currentRoot))
 | |
| 	{
 | |
| 		var width = this.graph.container.offsetWidth - 1;
 | |
| 		var height = this.graph.container.offsetHeight - 1;
 | |
| 		pgeo = new mxRectangle(0, 0, width, height);
 | |
| 	}
 | |
| 
 | |
| 	if (pgeo != null)
 | |
| 	{
 | |
| 		var children = [];
 | |
| 		var childCount = model.getChildCount(parent);
 | |
| 		
 | |
| 		for (var i = 0; i < childCount; i++)
 | |
| 		{
 | |
| 			var child = model.getChildAt(parent, i);
 | |
| 			
 | |
| 			if (!this.isVertexIgnored(child) &&
 | |
| 				this.isVertexMovable(child))
 | |
| 			{
 | |
| 				children.push(child);
 | |
| 			}
 | |
| 		}
 | |
| 		
 | |
| 		var n = children.length;
 | |
| 
 | |
| 		if (n > 0)
 | |
| 		{
 | |
| 			var x0 = this.border;
 | |
| 			var y0 = this.border;
 | |
| 			var other = (horizontal) ? pgeo.height : pgeo.width;
 | |
| 			other -= 2 * this.border;
 | |
| 
 | |
| 			var size = (this.graph.isSwimlane(parent)) ?
 | |
| 				this.graph.getStartSize(parent) :
 | |
| 				new mxRectangle();
 | |
| 
 | |
| 			other -= (horizontal) ? size.height : size.width;
 | |
| 			x0 = x0 + size.width;
 | |
| 			y0 = y0 + size.height;
 | |
| 
 | |
| 			var tmp = this.border + (n - 1) * this.spacing;
 | |
| 			var value = (horizontal) ?
 | |
| 				((pgeo.width - x0 - tmp) / n) :
 | |
| 				((pgeo.height - y0 - tmp) / n);
 | |
| 			
 | |
| 			// Avoids negative values, that is values where the sum of the
 | |
| 			// spacing plus the border is larger then the available space
 | |
| 			if (value > 0)
 | |
| 			{
 | |
| 				model.beginUpdate();
 | |
| 				try
 | |
| 				{
 | |
| 					for (var i = 0; i < n; i++)
 | |
| 					{
 | |
| 						var child = children[i];
 | |
| 						var geo = model.getGeometry(child);
 | |
| 					
 | |
| 						if (geo != null)
 | |
| 						{
 | |
| 							geo = geo.clone();
 | |
| 							geo.x = x0;
 | |
| 							geo.y = y0;
 | |
| 
 | |
| 							if (horizontal)
 | |
| 							{
 | |
| 								if (this.resizeVertices)
 | |
| 								{
 | |
| 									geo.width = value;
 | |
| 									geo.height = other;
 | |
| 								}
 | |
| 								
 | |
| 								x0 += value + this.spacing;
 | |
| 							}
 | |
| 							else
 | |
| 							{
 | |
| 								if (this.resizeVertices)
 | |
| 								{
 | |
| 									geo.height = value;
 | |
| 									geo.width = other;
 | |
| 								}
 | |
| 								
 | |
| 								y0 += value + this.spacing;
 | |
| 							}
 | |
| 
 | |
| 							model.setGeometry(child, geo);
 | |
| 						}
 | |
| 					}
 | |
| 				}
 | |
| 				finally
 | |
| 				{
 | |
| 					model.endUpdate();
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| };
 |