mirror of
https://github.com/radareorg/radare2.git
synced 2025-01-07 13:51:16 +00:00
1714 lines
47 KiB
JavaScript
1714 lines
47 KiB
JavaScript
/*********************
|
|
* browser detection *
|
|
*********************/
|
|
|
|
var ie=document.all;
|
|
var nn6=document.getElementById&&!document.all;
|
|
|
|
/****************************************************
|
|
* This class is a scanner for the visitor pattern. *
|
|
****************************************************/
|
|
|
|
/**
|
|
* Constructor, parameters are:
|
|
* visitor: the visitor implementation, it must be a class with a visit(element) method.
|
|
* scanElementsOnly: a flag telling whether to scan html elements only or all html nodes.
|
|
*/
|
|
function DocumentScanner(visitor, scanElementsOnly) {
|
|
this.visitor = visitor;
|
|
this.scanElementsOnly = scanElementsOnly;
|
|
|
|
/**
|
|
* Scans the element
|
|
*/
|
|
this.scan = function(element) {
|
|
if (this.visitor.visit(element)) {
|
|
// visit child elements
|
|
var children = element.childNodes;
|
|
for(var i = 0; i < children.length; i++) {
|
|
if(!this.scanElementsOnly || children[i].nodeType == 1)
|
|
this.scan(children[i]);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/*****************
|
|
* drag and drop *
|
|
*****************/
|
|
|
|
var isdrag=false; // this flag indicates that the mouse movement is actually a drag.
|
|
var mouseStartX, mouseStartY; // mouse position when drag starts
|
|
var elementStartX, elementStartY; // element position when drag starts
|
|
|
|
/**
|
|
* the html element being dragged.
|
|
*/
|
|
var elementToMove;
|
|
|
|
/**
|
|
* an array containing the blocks being dragged. This is used to notify them of move.
|
|
*/
|
|
var blocksToMove;
|
|
|
|
/**
|
|
* this variable stores the orginal z-index of the object being dragged in order
|
|
* to restore it upon drop.
|
|
*/
|
|
var originalZIndex;
|
|
|
|
/**
|
|
* an array containing bounds to be respected while dragging elements,
|
|
* these bounds are left, top, left + width, top + height of the parent element.
|
|
*/
|
|
var bounds = new Array(4);
|
|
|
|
/**
|
|
* this visitor is used to find blocks nested in the element being moved.
|
|
*/
|
|
function BlocksToMoveVisitor() {
|
|
this.visit = function(element) {
|
|
if (isBlock(element)) {
|
|
blocksToMove.push(findBlock(element.id));
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
}
|
|
|
|
var blocksToMoveScanner = new DocumentScanner(new BlocksToMoveVisitor(), true);
|
|
|
|
function movemouse(e) {
|
|
if (isdrag) {
|
|
var currentMouseX = nn6 ? e.clientX : event.clientX;
|
|
var currentMouseY = nn6 ? e.clientY : event.clientY;
|
|
var newElementX = elementStartX + currentMouseX - mouseStartX;
|
|
var newElementY = elementStartY + currentMouseY - mouseStartY;
|
|
|
|
// check bounds
|
|
// note: the "-1" and "+1" is to avoid borders overlap
|
|
if(newElementX < bounds[0])
|
|
newElementX = bounds[0] + 1;
|
|
if(newElementX + elementToMove.offsetWidth > bounds[2])
|
|
newElementX = bounds[2] - elementToMove.offsetWidth - 1;
|
|
if(newElementY < bounds[1])
|
|
newElementY = bounds[1] + 1;
|
|
if(newElementY + elementToMove.offsetHeight > bounds[3])
|
|
newElementY = bounds[3] - elementToMove.offsetHeight - 1;
|
|
|
|
// move element
|
|
elementToMove.style.left = newElementX + 'px';
|
|
elementToMove.style.top = newElementY + 'px';
|
|
|
|
// elementToMove.style.left = newElementX / elementToMove.parentNode.offsetWidth * 100 + '%';
|
|
// elementToMove.style.top = newElementY / elementToMove.parentNode.offsetHeight * 100 + '%';
|
|
|
|
elementToMove.style.right = null;
|
|
elementToMove.style.bottom = null;
|
|
|
|
for (var i = 0; i < blocksToMove.length; i++) {
|
|
if (blocksToMove[i])
|
|
blocksToMove[i].onMove();
|
|
}
|
|
return false;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* finds the innermost draggable element starting from the one that generated the event "e"
|
|
* (i.e.: the html element under mouse pointer), then setup the document's onmousemove function to
|
|
* move the element around.
|
|
*/
|
|
function startDrag(e) {
|
|
var eventSource = nn6 ? e.target : event.srcElement;
|
|
if (eventSource.tagName == 'HTML')
|
|
return;
|
|
|
|
while (eventSource != document.body && !hasClass(eventSource, "draggable"))
|
|
{
|
|
eventSource = nn6 ? eventSource.parentNode : eventSource.parentElement;
|
|
}
|
|
|
|
// if a draggable element was found, calculate its actual position
|
|
if (hasClass(eventSource, "draggable")) {
|
|
isdrag = true;
|
|
elementToMove = eventSource;
|
|
|
|
// set absolute positioning on the element
|
|
elementToMove.style.position = "absolute";
|
|
|
|
// calculate start point
|
|
elementStartX = elementToMove.offsetLeft;
|
|
elementStartY = elementToMove.offsetTop;
|
|
|
|
// calculate mouse start point
|
|
mouseStartX = nn6 ? e.clientX : event.clientX;
|
|
mouseStartY = nn6 ? e.clientY : event.clientY;
|
|
|
|
// calculate bounds as left, top, width, height of the parent element
|
|
if(getStyle(elementToMove.parentNode, "position") == 'absolute') {
|
|
bounds[0] = bounds[1] = 0;
|
|
} else {
|
|
bounds[0] = calculateOffsetLeft(elementToMove.parentNode);
|
|
bounds[1] = calculateOffsetTop(elementToMove.parentNode);
|
|
}
|
|
bounds[2] = bounds[0] + elementToMove.parentNode.offsetWidth;
|
|
bounds[3] = bounds[1] + elementToMove.parentNode.offsetHeight;
|
|
|
|
// either find the block related to the dragging element to call its onMove method
|
|
blocksToMove = new Array();
|
|
|
|
blocksToMoveScanner.scan(eventSource);
|
|
document.onmousemove = movemouse;
|
|
|
|
originalZIndex = getStyle(elementToMove, "z-index");
|
|
elementToMove.style.zIndex = "3";
|
|
|
|
return false;
|
|
}
|
|
}
|
|
|
|
function stopDrag(e) {
|
|
isdrag = false;
|
|
if (elementToMove)
|
|
elementToMove.style.zIndex=originalZIndex;
|
|
elementToMove = null;
|
|
document.onmousemove = null;
|
|
}
|
|
|
|
document.onmousedown = startDrag;
|
|
document.onmouseup = stopDrag;
|
|
|
|
function touch2move (x) {
|
|
var ev = {}
|
|
ev.target = x.target;
|
|
ev.clientX = x.targetTouches[0].pageX;
|
|
ev.clientY = x.targetTouches[0].pageY;
|
|
return ev;
|
|
}
|
|
document.ontouchstart = function(x) {
|
|
if (x.touches.length >1) return;
|
|
startDrag(touch2move (x));
|
|
isdrag = true;
|
|
}
|
|
document.ontouchmove = function(x) {
|
|
if (x.touches.length >1) return;
|
|
isdrag = true;
|
|
movemouse (touch2move (x));
|
|
x.preventDefault();
|
|
}
|
|
document.ontouchend = stopDrag;
|
|
|
|
|
|
|
|
/*************
|
|
* Constants *
|
|
*************/
|
|
var LEFT = 1;
|
|
var RIGHT = 2;
|
|
var UP = 4;
|
|
var DOWN = 8;
|
|
var HORIZONTAL = LEFT + RIGHT;
|
|
var VERTICAL = UP + DOWN;
|
|
var AUTO = HORIZONTAL + VERTICAL;
|
|
|
|
var START = 0;
|
|
var END = 1;
|
|
var SCROLLBARS_WIDTH = 18;
|
|
|
|
/**************
|
|
* Inspectors *
|
|
**************/
|
|
|
|
var inspectors = new Array();
|
|
|
|
/**
|
|
* The canvas class.
|
|
* This class is built on a div html element.
|
|
*/
|
|
function Canvas(htmlElement) {
|
|
/*
|
|
* initialization
|
|
*/
|
|
this.id = htmlElement.id;
|
|
this.htmlElement = htmlElement;
|
|
this.blocks = new Array();
|
|
this.connectors = new Array();
|
|
this.offsetLeft = calculateOffsetLeft(this.htmlElement);
|
|
this.offsetTop = calculateOffsetTop(this.htmlElement);
|
|
|
|
this.width;
|
|
this.height;
|
|
|
|
// create the inner div element
|
|
this.innerDiv = document.createElement ("div");
|
|
|
|
this.initCanvas = function() {
|
|
// setup the inner div
|
|
var children = this.htmlElement.childNodes;
|
|
var el;
|
|
var n = children.length;
|
|
for(var i = 0; i < n; i++) {
|
|
el = children[0];
|
|
this.htmlElement.removeChild(el);
|
|
this.innerDiv.appendChild(el);
|
|
if(el.style)
|
|
el.style.zIndex = "2";
|
|
}
|
|
this.htmlElement.appendChild(this.innerDiv);
|
|
|
|
this.htmlElement.style.overflow = "auto";
|
|
this.htmlElement.style.position = "relative";
|
|
this.innerDiv.id = this.id + "_innerDiv";
|
|
this.innerDiv.style.border = "none";
|
|
this.innerDiv.style.padding = "0px";
|
|
this.innerDiv.style.margin = "0px";
|
|
this.innerDiv.style.position = "absolute";
|
|
this.innerDiv.style.top = "0px";
|
|
this.innerDiv.style.left = "0px";
|
|
this.width = 0;
|
|
this.height = 0;
|
|
this.offsetLeft = calculateOffsetLeft(this.innerDiv);
|
|
this.offsetTop = calculateOffsetTop(this.innerDiv);
|
|
|
|
// inspect canvas children to identify first level blocks
|
|
new DocumentScanner(this, true).scan(this.htmlElement);
|
|
|
|
// now this.width and this.height are populated with minimum values needed for the inner
|
|
// blocks to fit, add 2 to avoid border overlap;
|
|
this.height += 2;
|
|
this.width += 2;
|
|
|
|
var visibleWidth = this.htmlElement.offsetWidth - 2; // - 2 is to avoid border overlap
|
|
var visibleHeight = this.htmlElement.offsetHeight - 2; // - 2 is to avoid border overlap
|
|
|
|
// consider the scrollbars width calculating the inner div size
|
|
if(this.height > visibleHeight)
|
|
visibleWidth -= SCROLLBARS_WIDTH;
|
|
if(this.width > visibleWidth)
|
|
visibleHeight -= SCROLLBARS_WIDTH;
|
|
|
|
this.height = Math.max(this.height, visibleHeight);
|
|
this.width = Math.max(this.width, visibleWidth);
|
|
|
|
this.innerDiv.style.width = this.width + "px";
|
|
this.innerDiv.style.height = this.height + "px";
|
|
|
|
// init connectors
|
|
for(i = 0; i < this.connectors.length; i++) {
|
|
this.connectors[i].initConnector();
|
|
}
|
|
}
|
|
|
|
this.visit = function(element) {
|
|
if (element == this.htmlElement)
|
|
return true;
|
|
|
|
// check the element dimensions against the acutal size of the canvas
|
|
this.width = Math.max(this.width, calculateOffsetLeft(element) - this.offsetLeft + element.offsetWidth);
|
|
this.height = Math.max(this.height, calculateOffsetTop(element) - this.offsetTop + element.offsetHeight);
|
|
|
|
if(isBlock(element)) {
|
|
// block found initialize it
|
|
var newBlock = new Block(element, this);
|
|
newBlock.initBlock();
|
|
this.blocks.push(newBlock);
|
|
return false;
|
|
} else if(isConnector(element)) {
|
|
// connector found, just create it, source or destination blocks may not
|
|
// have been initialized yet
|
|
var newConnector = new Connector(element, this);
|
|
this.connectors.push(newConnector);
|
|
return false;
|
|
} else {
|
|
// continue searching nested elements
|
|
return true;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* methods
|
|
*/
|
|
this.print = function()
|
|
{
|
|
var output = '<ul><legend>canvas: ' + this.id + '</legend>';
|
|
var i;
|
|
for(i = 0; i < this.blocks.length; i++)
|
|
{
|
|
output += '<li>';
|
|
output += this.blocks[i].print();
|
|
output += '</li>';
|
|
}
|
|
output += '</ul>';
|
|
return output;
|
|
}
|
|
|
|
this.alignBlocks = function() {
|
|
var i;
|
|
var roof = 0;
|
|
// TODO: implement proper layout
|
|
for (i = 0; i < this.blocks.length ; i++) {
|
|
var b = this.blocks[i]; //.findBlock(blockId);
|
|
b.onMove();
|
|
b.htmlElement.style.top =roof;
|
|
roof += b.htmlElement.style.height+20;
|
|
// TODO: alert ("align "+b);
|
|
}
|
|
for (i = 0; i < this.connectors.length; i++) {
|
|
this.connectors[i].repaint();
|
|
console.log( this.connectors[i]);
|
|
}
|
|
}
|
|
|
|
this.fitBlocks = function() {
|
|
for (var i = 0; i < this.blocks.length ; i++) {
|
|
var b = this.blocks[i]; //.findBlock(blockId);
|
|
this.blocks[i].fit ();
|
|
}
|
|
}
|
|
/*
|
|
* This function searches for a nested block with a given id
|
|
*/
|
|
this.findBlock = function(blockId) {
|
|
var result;
|
|
for(var i = 0; i < this.blocks.length && !result; i++)
|
|
result = this.blocks[i].findBlock(blockId);
|
|
return result;
|
|
}
|
|
|
|
this.toString = function() {
|
|
return 'canvas: ' + this.id;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Block class
|
|
*/
|
|
function Block(htmlElement, canvas)
|
|
{
|
|
/*
|
|
* initialization
|
|
*/
|
|
|
|
this.canvas = canvas;
|
|
this.htmlElement = htmlElement;
|
|
this.id = htmlElement.id;
|
|
this.blocks = new Array();
|
|
this.moveListeners = new Array();
|
|
|
|
if(this.id == 'description2_out1')
|
|
var merda = 0;
|
|
this.currentTop = calculateOffsetTop(this.htmlElement) - this.canvas.offsetTop;
|
|
this.currentLeft = calculateOffsetLeft(this.htmlElement) - this.canvas.offsetLeft;
|
|
|
|
this.visit = function(element) {
|
|
if (element == this.htmlElement) {
|
|
// exclude itself
|
|
return true;
|
|
}
|
|
if (isBlock(element)) {
|
|
var innerBlock = new Block(element, this.canvas);
|
|
innerBlock.initBlock();
|
|
this.blocks.push(innerBlock);
|
|
this.moveListeners.push(innerBlock);
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
this.initBlock = function() {
|
|
// inspect block children to identify nested blocks
|
|
new DocumentScanner(this, true).scan(this.htmlElement);
|
|
}
|
|
|
|
this.top = function() {
|
|
return this.currentTop;
|
|
}
|
|
|
|
this.left = function() {
|
|
return this.currentLeft;
|
|
}
|
|
|
|
this.width = function() {
|
|
return this.htmlElement.offsetWidth;
|
|
}
|
|
|
|
this.height = function() {
|
|
return this.htmlElement.offsetHeight;
|
|
}
|
|
|
|
/*
|
|
* methods
|
|
*/
|
|
this.print = function() {
|
|
var output = 'block: ' + this.id;
|
|
if (this.blocks.length > 0) {
|
|
output += '<ul>';
|
|
for(var i = 0; i < this.blocks.length; i++) {
|
|
output += '<li>';
|
|
output += this.blocks[i].print();
|
|
output += '</li>';
|
|
}
|
|
output += '</ul>';
|
|
}
|
|
return output;
|
|
}
|
|
|
|
/*
|
|
* This function searches for a nested block (or the block itself) with a given id
|
|
*/
|
|
this.findBlock = function(blockId) {
|
|
if(this.id == blockId)
|
|
return this;
|
|
var result;
|
|
for(var i = 0; i < this.blocks.length && !result; i++)
|
|
result = this.blocks[i].findBlock(blockId);
|
|
return result;
|
|
}
|
|
|
|
this.fit = function() {
|
|
function getlines(txt) {
|
|
return (12*txt.split ("\n").length);
|
|
}
|
|
function getcolumns(txt) {
|
|
var cols = 0;
|
|
var txts = txt.split ("\n");
|
|
for (var x in txts) {
|
|
const len = txts[x].length;
|
|
if (len>cols)
|
|
cols = len;
|
|
}
|
|
return 10+ (7*cols);
|
|
}
|
|
var text = this.htmlElement.innerHTML;
|
|
this.htmlElement.style.width = getcolumns (text);
|
|
this.htmlElement.style.height = getlines (text);
|
|
}
|
|
|
|
this.move = function(left, top) {
|
|
this.htmlElement.style.left = left;
|
|
this.htmlElement.style.top = top;
|
|
this.onMove();
|
|
}
|
|
|
|
this.onMove = function() {
|
|
this.currentLeft = calculateOffsetLeft(this.htmlElement) - this.canvas.offsetLeft;
|
|
this.currentTop = calculateOffsetTop(this.htmlElement) - this.canvas.offsetTop;
|
|
// notify listeners
|
|
for(var i = 0; i < this.moveListeners.length; i++)
|
|
this.moveListeners[i].onMove();
|
|
}
|
|
|
|
this.toString = function() {
|
|
return 'block: ' + this.id;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* This class represents a connector segment, it is drawn via a div element.
|
|
* A segment has a starting point defined by the properties startX and startY, a length,
|
|
* a thickness and an orientation.
|
|
* Allowed values for the orientation property are defined by the constants UP, LEFT, DOWN and RIGHT.
|
|
*/
|
|
function Segment(id, parentElement)
|
|
{
|
|
this.id = id;
|
|
this.htmlElement = document.createElement('div');
|
|
this.htmlElement.id = id;
|
|
this.htmlElement.style.position = 'absolute';
|
|
this.htmlElement.style.overflow = 'hidden';
|
|
parentElement.appendChild(this.htmlElement);
|
|
|
|
this.startX;
|
|
this.startY;
|
|
this.length;
|
|
this.thickness;
|
|
this.orientation;
|
|
this.nextSegment;
|
|
this.visible = true;
|
|
|
|
/**
|
|
* draw the segment. This operation is cascaded to next segment if any.
|
|
*/
|
|
this.draw = function()
|
|
{
|
|
// set properties to next segment
|
|
if(this.nextSegment)
|
|
{
|
|
this.nextSegment.startX = this.getEndX();
|
|
this.nextSegment.startY = this.getEndY();
|
|
}
|
|
|
|
this.htmlElement.style.display = this.visible?'block':'none';
|
|
|
|
switch (this.orientation) {
|
|
case LEFT:
|
|
this.htmlElement.style.left = (this.startX - this.length) + "px";
|
|
this.htmlElement.style.top = this.startY + "px";
|
|
this.htmlElement.style.width = this.length + "px";
|
|
this.htmlElement.style.height = this.thickness + "px";
|
|
break;
|
|
case RIGHT:
|
|
this.htmlElement.style.left = this.startX + "px";
|
|
this.htmlElement.style.top = this.startY + "px";
|
|
if(this.nextSegment)
|
|
this.htmlElement.style.width = this.length + this.thickness + "px";
|
|
else
|
|
this.htmlElement.style.width = this.length + "px";
|
|
this.htmlElement.style.height = this.thickness + "px";
|
|
break;
|
|
case UP:
|
|
this.htmlElement.style.left = this.startX + "px";
|
|
this.htmlElement.style.top = (this.startY - this.length) + "px";
|
|
this.htmlElement.style.width = this.thickness + "px";
|
|
this.htmlElement.style.height = this.length + "px";
|
|
break;
|
|
case DOWN:
|
|
this.htmlElement.style.left = this.startX + "px";
|
|
this.htmlElement.style.top = this.startY + "px";
|
|
this.htmlElement.style.width = this.thickness + "px";
|
|
if(this.nextSegment)
|
|
this.htmlElement.style.height = this.length + this.thickness + "px";
|
|
else
|
|
this.htmlElement.style.height = this.length + "px";
|
|
break;
|
|
}
|
|
|
|
if(this.nextSegment)
|
|
this.nextSegment.draw();
|
|
}
|
|
|
|
/**
|
|
* Returns the "left" coordinate of the end point of this segment
|
|
*/
|
|
this.getEndX = function()
|
|
{
|
|
switch(this.orientation)
|
|
{
|
|
case LEFT: return this.startX - this.length;
|
|
case RIGHT: return this.startX + this.length;
|
|
case DOWN: return this.startX;
|
|
case UP: return this.startX;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Returns the "top" coordinate of the end point of this segment
|
|
*/
|
|
this.getEndY = function() {
|
|
switch (this.orientation) {
|
|
case LEFT: return this.startY;
|
|
case RIGHT: return this.startY;
|
|
case DOWN: return this.startY + this.length;
|
|
case UP: return this.startY - this.length;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Append another segment to the end point of this.
|
|
* If another segment is already appended to this, cascades the operation so
|
|
* the given next segment will be appended to the tail of the segments chain.
|
|
*/
|
|
this.append = function(nextSegment) {
|
|
if(!nextSegment)
|
|
return;
|
|
if(!this.nextSegment) {
|
|
this.nextSegment = nextSegment;
|
|
this.nextSegment.startX = this.getEndX();
|
|
this.nextSegment.startY = this.getEndY();
|
|
} else this.nextSegment.append(nextSegment);
|
|
}
|
|
|
|
this.detach = function() {
|
|
var s = this.nextSegment;
|
|
this.nextSegment = null;
|
|
return s;
|
|
}
|
|
|
|
/**
|
|
* hides this segment and all the following
|
|
*/
|
|
this.cascadeHide = function() {
|
|
this.visible = false;
|
|
if(this.nextSegment)
|
|
this.nextSegment.cascadeHide();
|
|
}
|
|
}
|
|
/**
|
|
* Connector class.
|
|
* The init function takes two Block objects as arguments representing
|
|
* the source and destination of the connector
|
|
*/
|
|
function Connector(htmlElement, canvas)
|
|
{
|
|
/**
|
|
* declaring html element
|
|
*/
|
|
this.htmlElement = htmlElement;
|
|
|
|
/**
|
|
* the canvas this connector is in
|
|
*/
|
|
this.canvas = canvas;
|
|
|
|
/**
|
|
* the source block
|
|
*/
|
|
this.source = null;
|
|
|
|
/**
|
|
* the destination block
|
|
*/
|
|
this.destination = null;
|
|
|
|
/**
|
|
* preferred orientation
|
|
*/
|
|
this.preferredSourceOrientation = AUTO;
|
|
this.preferredDestinationOrientation = AUTO;
|
|
|
|
/**
|
|
* css class to be applied to the connector's segments
|
|
*/
|
|
this.connectorClass;
|
|
|
|
/**
|
|
* minimum length for a connector segment.
|
|
*/
|
|
this.minSegmentLength = 10;
|
|
|
|
/**
|
|
* size of the connector, i.e.: thickness of the segments.
|
|
*/
|
|
this.size = 1;
|
|
|
|
/**
|
|
* connector's color
|
|
*/
|
|
this.color = 'black';
|
|
|
|
/**
|
|
* move listeners, they are notify when connector moves
|
|
*/
|
|
this.moveListeners = new Array();
|
|
|
|
this.firstSegment;
|
|
|
|
this.segmentsPool;
|
|
|
|
this.segmentsNumber = 0;
|
|
|
|
this.strategy;
|
|
|
|
this.initConnector = function()
|
|
{
|
|
// detect the connector id
|
|
if(this.htmlElement.id)
|
|
this.id = this.htmlElement.id;
|
|
else
|
|
this.id = this.htmlElement.className;
|
|
|
|
// split the class name to get the ids of the source and destination blocks
|
|
var splitted = htmlElement.className.split(' ');
|
|
if(splitted.length < 3)
|
|
{
|
|
alert('Unable to create connector \'' + id + '\', class is not in the correct format: connector <sourceBlockId>, <destBlockId>');
|
|
return;
|
|
}
|
|
|
|
this.connectorClass = splitted[0] + ' ' + splitted[1] + ' ' + splitted[2];
|
|
|
|
this.source = this.canvas.findBlock(splitted[1]);
|
|
if(!this.source)
|
|
{
|
|
alert('cannot find source block with id \'' + splitted[1] + '\'');
|
|
return;
|
|
}
|
|
|
|
this.destination = this.canvas.findBlock(splitted[2]);
|
|
if(!this.destination)
|
|
{
|
|
// alert('cannot find destination block with id \'' + splitted[2] + '\'');
|
|
return;
|
|
}
|
|
|
|
// check preferred orientation
|
|
if(hasClass(this.htmlElement, 'vertical'))
|
|
{
|
|
this.preferredSourceOrientation = VERTICAL;
|
|
this.preferredDestinationOrientation = VERTICAL;
|
|
}
|
|
else if(hasClass(this.htmlElement, 'horizontal'))
|
|
{
|
|
this.preferredSourceOrientation = HORIZONTAL;
|
|
this.preferredDestinationOrientation = HORIZONTAL;
|
|
}
|
|
else
|
|
{
|
|
// check preferred orientation on source side
|
|
if(hasClass(this.htmlElement, 'vertical_start'))
|
|
this.preferredSourceOrientation = VERTICAL;
|
|
else if(hasClass(this.htmlElement, 'horizontal_start'))
|
|
this.preferredSourceOrientation = HORIZONTAL;
|
|
else if(hasClass(this.htmlElement, 'left_start'))
|
|
this.preferredSourceOrientation = LEFT;
|
|
else if(hasClass(this.htmlElement, 'right_start'))
|
|
this.preferredSourceOrientation = RIGHT;
|
|
else if(hasClass(this.htmlElement, 'up_start'))
|
|
this.preferredSourceOrientation = UP;
|
|
else if(hasClass(this.htmlElement, 'down_start'))
|
|
this.preferredSourceOrientation = DOWN;
|
|
|
|
// check preferred orientation on destination side
|
|
if(hasClass(this.htmlElement, 'vertical_end'))
|
|
this.preferredDestinationOrientation = VERTICAL;
|
|
else if(hasClass(this.htmlElement, 'horizontal_end'))
|
|
this.preferredDestinationOrientation = HORIZONTAL;
|
|
else if(hasClass(this.htmlElement, 'left_end'))
|
|
this.preferredDestinationOrientation = LEFT;
|
|
else if(hasClass(this.htmlElement, 'right_end'))
|
|
this.preferredDestinationOrientation = RIGHT;
|
|
else if(hasClass(this.htmlElement, 'up_end'))
|
|
this.preferredDestinationOrientation = UP;
|
|
else if(hasClass(this.htmlElement, 'down_end'))
|
|
this.preferredDestinationOrientation = DOWN;
|
|
}
|
|
|
|
// get the first strategy as default
|
|
this.strategy = strategies[0](this);
|
|
this.repaint();
|
|
|
|
this.source.moveListeners.push(this);
|
|
this.destination.moveListeners.push(this);
|
|
|
|
// call inspectors for this connector
|
|
var i;
|
|
for(i = 0; i < inspectors.length; i++)
|
|
{
|
|
inspectors[i].inspect(this);
|
|
}
|
|
|
|
// remove old html element
|
|
this.htmlElement.parentNode.removeChild(this.htmlElement);
|
|
}
|
|
|
|
this.getStartSegment = function() {
|
|
return this.firstSegment;
|
|
}
|
|
|
|
this.getEndSegment = function() {
|
|
var s = this.firstSegment;
|
|
while (s.nextSegment)
|
|
s = s.nextSegment;
|
|
return s;
|
|
}
|
|
|
|
this.getMiddleSegment = function() {
|
|
return this.strategy? this.strategy.getMiddleSegment(): null;
|
|
}
|
|
|
|
this.createSegment = function() {
|
|
var segment;
|
|
|
|
// if the pool contains more objects, borrow the segment, create it otherwise
|
|
if(this.segmentsPool) {
|
|
segment = this.segmentsPool;
|
|
this.segmentsPool = this.segmentsPool.detach();
|
|
} else {
|
|
segment = new Segment(this.id + "_" + (this.segmentsNumber + 1), this.canvas.htmlElement);
|
|
segment.htmlElement.className = this.connectorClass;
|
|
if(!getStyle(segment.htmlElement, 'background-color'))
|
|
segment.htmlElement.style.backgroundColor = this.color;
|
|
segment.thickness = this.size;
|
|
}
|
|
this.segmentsNumber++;
|
|
|
|
if(this.firstSegment)
|
|
this.firstSegment.append(segment);
|
|
else
|
|
this.firstSegment = segment;
|
|
segment.visible = true;
|
|
return segment;
|
|
}
|
|
|
|
/**
|
|
* Repaints the connector
|
|
*/
|
|
this.repaint = function() {
|
|
// check strategies fitness and choose the best fitting one
|
|
var maxFitness = 0;
|
|
var fitness;
|
|
var s;
|
|
|
|
// check if any strategy is possible with preferredOrientation
|
|
for(var i = 0; i < strategies.length; i++) {
|
|
this.clearSegments();
|
|
|
|
fitness = 0;
|
|
s = strategies[i](this);
|
|
if(s.isApplicable()) {
|
|
fitness++;
|
|
s.paint();
|
|
// check resulting orientation against the preferred orientations
|
|
if((this.firstSegment.orientation & this.preferredSourceOrientation) != 0)
|
|
fitness++;
|
|
if((this.getEndSegment().orientation & this.preferredDestinationOrientation) != 0)
|
|
fitness++;
|
|
}
|
|
|
|
if(fitness > maxFitness) {
|
|
this.strategy = s;
|
|
maxFitness = fitness;
|
|
}
|
|
}
|
|
|
|
this.clearSegments();
|
|
|
|
this.strategy.paint();
|
|
this.firstSegment.draw();
|
|
|
|
// this is needed to actually hide unused html elements
|
|
if(this.segmentsPool)
|
|
this.segmentsPool.draw();
|
|
}
|
|
|
|
/**
|
|
* Hide all the segments and return them to pool
|
|
*/
|
|
this.clearSegments = function() {
|
|
if (this.firstSegment) {
|
|
this.firstSegment.cascadeHide();
|
|
this.firstSegment.append(this.segmentsPool);
|
|
this.segmentsPool = this.firstSegment;
|
|
this.firstSegment = null;
|
|
}
|
|
}
|
|
|
|
this.onMove = function() {
|
|
this.repaint();
|
|
// notify listeners
|
|
for (var i = 0; i < this.moveListeners.length; i++)
|
|
this.moveListeners[i].onMove();
|
|
}
|
|
}
|
|
|
|
var strategies = new Array();
|
|
|
|
function ConnectorEnd(htmlElement, connector, side) {
|
|
this.side = side;
|
|
this.htmlElement = htmlElement;
|
|
this.connector = connector;
|
|
connector.canvas.htmlElement.appendChild(htmlElement);
|
|
// strip extension
|
|
if(this.htmlElement.tagName.toLowerCase() == "img")
|
|
{
|
|
this.src = this.htmlElement.src.substring(0, this.htmlElement.src.lastIndexOf('.'));
|
|
this.srcExtension = this.htmlElement.src.substring(this.htmlElement.src.lastIndexOf('.'));
|
|
this.htmlElement.style.zIndex = getStyle(this.connector.htmlElement, "z-index");
|
|
}
|
|
|
|
this.orientation;
|
|
|
|
this.repaint = function() {
|
|
this.htmlElement.style.position = 'absolute';
|
|
|
|
var left;
|
|
var top;
|
|
var segment;
|
|
var orientation;
|
|
|
|
if(this.side == START) {
|
|
segment = connector.getStartSegment();
|
|
left = segment.startX;
|
|
top = segment.startY;
|
|
orientation = segment.orientation;
|
|
// swap orientation
|
|
if((orientation & VERTICAL) != 0)
|
|
orientation = (~orientation) & VERTICAL;
|
|
else
|
|
orientation = (~orientation) & HORIZONTAL;
|
|
} else {
|
|
segment = connector.getEndSegment();
|
|
left = segment.getEndX();
|
|
top = segment.getEndY();
|
|
orientation = segment.orientation;
|
|
}
|
|
|
|
switch(orientation) {
|
|
case LEFT:
|
|
top -= (this.htmlElement.offsetHeight - segment.thickness) / 2;
|
|
break;
|
|
case RIGHT:
|
|
left -= this.htmlElement.offsetWidth;
|
|
top -= (this.htmlElement.offsetHeight - segment.thickness) / 2;
|
|
break;
|
|
case DOWN:
|
|
top -= this.htmlElement.offsetHeight;
|
|
left -= (this.htmlElement.offsetWidth - segment.thickness) / 2;
|
|
break;
|
|
case UP:
|
|
left -= (this.htmlElement.offsetWidth - segment.thickness) / 2;
|
|
break;
|
|
}
|
|
|
|
this.htmlElement.style.left = Math.ceil(left) + "px";
|
|
this.htmlElement.style.top = Math.ceil(top) + "px";
|
|
|
|
if(this.htmlElement.tagName.toLowerCase() == "img" && this.orientation != orientation)
|
|
{
|
|
var orientationSuffix;
|
|
switch(orientation)
|
|
{
|
|
case UP: orientationSuffix = "u"; break;
|
|
case DOWN: orientationSuffix = "d"; break;
|
|
case LEFT: orientationSuffix = "l"; break;
|
|
case RIGHT: orientationSuffix = "r"; break;
|
|
}
|
|
this.htmlElement.src = this.src + "_" + orientationSuffix + this.srcExtension;
|
|
}
|
|
this.orientation = orientation;
|
|
}
|
|
|
|
this.onMove = function()
|
|
{
|
|
this.repaint();
|
|
}
|
|
}
|
|
|
|
function SideConnectorLabel(connector, htmlElement, side)
|
|
{
|
|
this.connector = connector;
|
|
this.htmlElement = htmlElement;
|
|
this.side = side;
|
|
this.connector.htmlElement.parentNode.appendChild(htmlElement);
|
|
|
|
this.repaint = function()
|
|
{
|
|
this.htmlElement.style.position = 'absolute';
|
|
var left;
|
|
var top;
|
|
var segment;
|
|
|
|
if(this.side == START)
|
|
{
|
|
segment = this.connector.getStartSegment();
|
|
left = segment.startX;
|
|
top = segment.startY;
|
|
if(segment.orientation == LEFT)
|
|
left -= this.htmlElement.offsetWidth;
|
|
if(segment.orientation == UP)
|
|
top -= this.htmlElement.offsetHeight;
|
|
|
|
if((segment.orientation & HORIZONTAL) != 0 && top < this.connector.getEndSegment().getEndY())
|
|
top -= this.htmlElement.offsetHeight;
|
|
if((segment.orientation & VERTICAL) != 0 && left < this.connector.getEndSegment().getEndX())
|
|
left -= this.htmlElement.offsetWidth;
|
|
}
|
|
else
|
|
{
|
|
segment = this.connector.getEndSegment();
|
|
left = segment.getEndX();
|
|
top = segment.getEndY();
|
|
if(segment.orientation == RIGHT)
|
|
left -= this.htmlElement.offsetWidth;
|
|
if(segment.orientation == DOWN)
|
|
top -= this.htmlElement.offsetHeight;
|
|
if((segment.orientation & HORIZONTAL) != 0 && top < this.connector.getStartSegment().startY)
|
|
top -= this.htmlElement.offsetHeight;
|
|
if((segment.orientation & VERTICAL) != 0 && left < this.connector.getStartSegment().startX)
|
|
left -= this.htmlElement.offsetWidth;
|
|
}
|
|
|
|
this.htmlElement.style.left = Math.ceil(left) + "px";
|
|
this.htmlElement.style.top = Math.ceil(top) + "px";
|
|
}
|
|
|
|
this.onMove = function()
|
|
{
|
|
this.repaint();
|
|
}
|
|
}
|
|
|
|
function MiddleConnectorLabel(connector, htmlElement)
|
|
{
|
|
this.connector = connector;
|
|
this.htmlElement = htmlElement;
|
|
this.connector.canvas.htmlElement.appendChild(htmlElement);
|
|
|
|
this.repaint = function()
|
|
{
|
|
this.htmlElement.style.position = 'absolute';
|
|
|
|
var left;
|
|
var top;
|
|
var segment = connector.getMiddleSegment();
|
|
|
|
if((segment.orientation & VERTICAL) != 0)
|
|
{
|
|
// put label at middle height on right side of the connector
|
|
top = segment.htmlElement.offsetTop + (segment.htmlElement.offsetHeight - this.htmlElement.offsetHeight) / 2;
|
|
left = segment.htmlElement.offsetLeft;
|
|
}
|
|
else
|
|
{
|
|
// put connector below the connector at middle widths
|
|
top = segment.htmlElement.offsetTop;
|
|
left = segment.htmlElement.offsetLeft + (segment.htmlElement.offsetWidth - this.htmlElement.offsetWidth) / 2;;
|
|
}
|
|
|
|
this.htmlElement.style.left = Math.ceil(left) + "px";
|
|
this.htmlElement.style.top = Math.ceil(top) + "px";
|
|
}
|
|
|
|
this.onMove = function()
|
|
{
|
|
this.repaint();
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Inspector classes
|
|
*/
|
|
|
|
function ConnectorEndsInspector()
|
|
{
|
|
this.inspect = function(connector)
|
|
{
|
|
var children = connector.htmlElement.childNodes;
|
|
var i;
|
|
for(i = 0; i < children.length; i++)
|
|
{
|
|
if(hasClass(children[i], "connector-end"))
|
|
{
|
|
var newElement = new ConnectorEnd(children[i], connector, END);
|
|
newElement.repaint();
|
|
connector.moveListeners.push(newElement);
|
|
}
|
|
else if(hasClass(children[i], "connector-start"))
|
|
{
|
|
var newElement = new ConnectorEnd(children[i], connector, START);
|
|
newElement.repaint();
|
|
connector.moveListeners.push(newElement);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
function ConnectorLabelsInspector()
|
|
{
|
|
this.inspect = function(connector)
|
|
{
|
|
var children = connector.htmlElement.childNodes;
|
|
var i;
|
|
for(i = 0; i < children.length; i++)
|
|
{
|
|
if(hasClass(children[i], "source-label"))
|
|
{
|
|
var newElement = new SideConnectorLabel(connector, children[i], START);
|
|
newElement.repaint();
|
|
connector.moveListeners.push(newElement);
|
|
}
|
|
else if(hasClass(children[i], "middle-label"))
|
|
{
|
|
var newElement = new MiddleConnectorLabel(connector, children[i]);
|
|
newElement.repaint();
|
|
connector.moveListeners.push(newElement);
|
|
}
|
|
else if(hasClass(children[i], "destination-label"))
|
|
{
|
|
var newElement = new SideConnectorLabel(connector, children[i], END);
|
|
newElement.repaint();
|
|
connector.moveListeners.push(newElement);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Inspector registration
|
|
*/
|
|
|
|
inspectors.push(new ConnectorEndsInspector());
|
|
inspectors.push(new ConnectorLabelsInspector());
|
|
|
|
/*
|
|
* an array containing all the canvases in document
|
|
*/
|
|
var canvases = new Array();
|
|
|
|
/*
|
|
* This function initializes the js_graph objects inspecting the html document
|
|
*/
|
|
function initPageObjects()
|
|
{
|
|
if(isCanvas(document.body))
|
|
{
|
|
var newCanvas = new Canvas(document.body);
|
|
newCanvas.initCanvas();
|
|
canvases.push(newCanvas);
|
|
}
|
|
else
|
|
{
|
|
var divs = document.getElementsByTagName('div');
|
|
var i;
|
|
for(i = 0; i < divs.length; i++)
|
|
{
|
|
if(isCanvas(divs[i]) && !findCanvas(divs[i].id))
|
|
{
|
|
var newCanvas = new Canvas(divs[i]);
|
|
newCanvas.initCanvas();
|
|
canvases.push(newCanvas);
|
|
newCanvas.fitBlocks();
|
|
newCanvas.alignBlocks();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
* Utility functions
|
|
*/
|
|
|
|
function findCanvas(canvasId) {
|
|
for (var i = 0; i < canvases.length; i++)
|
|
if(canvases[i].id == canvasId)
|
|
return canvases[i];
|
|
return null;
|
|
}
|
|
|
|
function findBlock(blockId) {
|
|
for (var i = 0; i < canvases.length; i++) {
|
|
var block = canvases[i].findBlock(blockId);
|
|
if (block) return block;
|
|
}
|
|
return null;
|
|
}
|
|
|
|
/*
|
|
* This function determines whether a html element is to be considered a canvas
|
|
*/
|
|
function isBlock(htmlElement)
|
|
{
|
|
return hasClass(htmlElement, 'block');
|
|
}
|
|
|
|
/*
|
|
* This function determines whether a html element is to be considered a block
|
|
*/
|
|
function isCanvas(htmlElement) {
|
|
return hasClass(htmlElement, 'canvas');
|
|
}
|
|
|
|
/*
|
|
* This function determines whether a html element is to be considered a connector
|
|
*/
|
|
function isConnector(htmlElement) {
|
|
return htmlElement.className && htmlElement.className.match(new RegExp('connector .*'));
|
|
}
|
|
|
|
/*
|
|
* This function calculates the absolute 'top' value for a html node
|
|
*/
|
|
function calculateOffsetTop(obj) {
|
|
var curtop = 0;
|
|
if (obj.offsetParent) {
|
|
curtop = obj.offsetTop
|
|
while (obj = obj.offsetParent)
|
|
curtop += obj.offsetTop
|
|
} else if (obj.y)
|
|
curtop += obj.y;
|
|
return curtop;
|
|
}
|
|
|
|
/*
|
|
* This function calculates the absolute 'left' value for a html node
|
|
*/
|
|
function calculateOffsetLeft(obj)
|
|
{
|
|
var curleft = 0;
|
|
if (obj.offsetParent) {
|
|
curleft = obj.offsetLeft
|
|
while (obj = obj.offsetParent)
|
|
curleft += obj.offsetLeft;
|
|
} else if (obj.x)
|
|
curleft += obj.x;
|
|
return curleft;
|
|
}
|
|
|
|
function parseBorder(obj, side) {
|
|
var sizeString = getStyle(obj, "border-" + side + "-width");
|
|
if(sizeString && sizeString != "") {
|
|
if(sizeString.substring(sizeString.length - 2) == "px")
|
|
return parseInt(sizeString.substring(0, sizeString.length - 2));
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
function hasClass(element, className) {
|
|
if (!element || !element.className)
|
|
return false;
|
|
|
|
var classes = element.className.split(' ');
|
|
for (var i = 0; i < classes.length; i++)
|
|
if (classes[i] == className)
|
|
return true;
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* This function retrieves the actual value of a style property even if it is set via css.
|
|
*/
|
|
function getStyle(node, styleProp) {
|
|
// if not an element
|
|
if( node.nodeType != 1)
|
|
return;
|
|
|
|
var value;
|
|
if (node.currentStyle) {
|
|
// ie case
|
|
styleProp = replaceDashWithCamelNotation(styleProp);
|
|
value = node.currentStyle[styleProp];
|
|
} else if (window.getComputedStyle) {
|
|
// mozilla case
|
|
value = document.defaultView.getComputedStyle(node, null).getPropertyValue(styleProp);
|
|
}
|
|
|
|
return value;
|
|
}
|
|
|
|
function replaceDashWithCamelNotation(value) {
|
|
var pos = value.indexOf('-');
|
|
while(pos > 0 && value.length > pos + 1) {
|
|
value = value.substring(0, pos) + value.substring(pos + 1, pos + 2).toUpperCase() + value.substring(pos + 2);
|
|
pos = value.indexOf('-');
|
|
}
|
|
return value;
|
|
}
|
|
/*******************************
|
|
* Connector paint strategies. *
|
|
*******************************/
|
|
|
|
/**
|
|
* Horizontal "S" routing strategy.
|
|
*/
|
|
function HorizontalSStrategy(connector) {
|
|
this.connector = connector;
|
|
this.startSegment;
|
|
this.middleSegment;
|
|
this.endSegment;
|
|
this.strategyName = "horizontal_s";
|
|
this.getMiddleSegment = function() {
|
|
return this.middleSegment;
|
|
}
|
|
this.isApplicable = function() {
|
|
var sourceLeft = this.connector.source.left();
|
|
var sourceWidth = this.connector.source.width();
|
|
var destinationLeft = this.connector.destination.left();
|
|
var destinationWidth = this.connector.destination.width();
|
|
|
|
return Math.abs(2 * destinationLeft + destinationWidth - (2 * sourceLeft + sourceWidth)) - (sourceWidth + destinationWidth) > 4 * this.connector.minSegmentLength;
|
|
}
|
|
|
|
this.paint = function() {
|
|
this.startSegment = connector.createSegment();
|
|
this.middleSegment = connector.createSegment();
|
|
this.endSegment = connector.createSegment();
|
|
|
|
var sourceLeft = this.connector.source.left();
|
|
var sourceTop = this.connector.source.top();
|
|
var sourceWidth = this.connector.source.width();
|
|
var sourceHeight = this.connector.source.height();
|
|
|
|
var destinationLeft = this.connector.destination.left();
|
|
var destinationTop = this.connector.destination.top();
|
|
var destinationWidth = this.connector.destination.width();
|
|
var destinationHeight = this.connector.destination.height();
|
|
|
|
var hLength;
|
|
|
|
this.startSegment.startY = Math.floor(sourceTop + sourceHeight / 2);
|
|
|
|
// deduce which face to use on source and destination blocks
|
|
if(sourceLeft + sourceWidth / 2 < destinationLeft + destinationWidth / 2)
|
|
{
|
|
// use left side of the source block and right side of the destination block
|
|
this.startSegment.startX = sourceLeft + sourceWidth;
|
|
hLength = destinationLeft - (sourceLeft + sourceWidth);
|
|
}
|
|
else
|
|
{
|
|
// use right side of the source block and left side of the destination block
|
|
this.startSegment.startX = sourceLeft;
|
|
hLength = destinationLeft + destinationWidth - sourceLeft;
|
|
}
|
|
|
|
// first horizontal segment positioning
|
|
this.startSegment.length = Math.floor(Math.abs(hLength) / 2);
|
|
this.startSegment.orientation = hLength > 0 ? RIGHT : LEFT;
|
|
|
|
// vertical segment positioning
|
|
var vLength = Math.floor(destinationTop + destinationHeight / 2 - (sourceTop + sourceHeight / 2));
|
|
this.middleSegment.length = Math.abs(vLength);
|
|
if(vLength == 0)
|
|
this.middleSegment.visible = false;
|
|
this.middleSegment.orientation = vLength > 0 ? DOWN : UP;
|
|
|
|
// second horizontal segment positioning
|
|
this.endSegment.length = Math.floor(Math.abs(hLength) / 2);
|
|
this.endSegment.orientation = hLength > 0 ? RIGHT : LEFT;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Vertical "S" routing strategy.
|
|
*/
|
|
function VerticalSStrategy(connector)
|
|
{
|
|
this.connector = connector;
|
|
|
|
this.startSegment;
|
|
this.middleSegment;
|
|
this.endSegment;
|
|
|
|
this.strategyName = "vertical_s";
|
|
|
|
this.getMiddleSegment = function()
|
|
{
|
|
return this.middleSegment;
|
|
}
|
|
|
|
this.isApplicable = function()
|
|
{
|
|
var sourceTop = this.connector.source.top();
|
|
var sourceHeight = this.connector.source.height();
|
|
var destinationTop = this.connector.destination.top();
|
|
var destinationHeight = this.connector.destination.height();
|
|
return Math.abs(2 * destinationTop + destinationHeight - (2 * sourceTop + sourceHeight)) - (sourceHeight + destinationHeight) > 4 * this.connector.minSegmentLength;
|
|
}
|
|
|
|
this.paint = function()
|
|
{
|
|
this.startSegment = connector.createSegment();
|
|
this.middleSegment = connector.createSegment();
|
|
this.endSegment = connector.createSegment();
|
|
|
|
var sourceLeft = this.connector.source.left();
|
|
var sourceTop = this.connector.source.top();
|
|
var sourceWidth = this.connector.source.width();
|
|
var sourceHeight = this.connector.source.height();
|
|
|
|
var destinationLeft = this.connector.destination.left();
|
|
var destinationTop = this.connector.destination.top();
|
|
var destinationWidth = this.connector.destination.width();
|
|
var destinationHeight = this.connector.destination.height();
|
|
|
|
var vLength;
|
|
|
|
this.startSegment.startX = Math.floor(sourceLeft + sourceWidth / 2);
|
|
|
|
// deduce which face to use on source and destination blocks
|
|
if(sourceTop + sourceHeight / 2 < destinationTop + destinationHeight / 2)
|
|
{
|
|
// use bottom side of the source block and top side of destination block
|
|
this.startSegment.startY = sourceTop + sourceHeight;
|
|
vLength = destinationTop - (sourceTop + sourceHeight);
|
|
}
|
|
else
|
|
{
|
|
// use top side of the source block and bottom side of the destination block
|
|
this.startSegment.startY = sourceTop;
|
|
vLength = destinationTop + destinationHeight - sourceTop;
|
|
}
|
|
|
|
// first vertical segment positioning
|
|
this.startSegment.length = Math.floor(Math.abs(vLength) / 2);
|
|
this.startSegment.orientation = vLength > 0 ? DOWN : UP;
|
|
|
|
// horizontal segment positioning
|
|
var hLength = Math.floor(destinationLeft + destinationWidth / 2 - (sourceLeft + sourceWidth / 2));
|
|
this.middleSegment.length = Math.abs(hLength);
|
|
this.middleSegment.orientation = hLength > 0 ? RIGHT : LEFT;
|
|
|
|
// second vertical segment positioning
|
|
this.endSegment.length = Math.floor(Math.abs(vLength) / 2);
|
|
this.endSegment.orientation = vLength > 0 ? DOWN : UP;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* A horizontal "L" connector routing strategy
|
|
*/
|
|
function HorizontalLStrategy(connector)
|
|
{
|
|
this.connector = connector;
|
|
|
|
this.destination;
|
|
|
|
this.startSegment;
|
|
this.endSegment;
|
|
|
|
this.strategyName = "horizontal_L";
|
|
|
|
this.isApplicable = function()
|
|
{
|
|
var destMiddle = Math.floor(this.connector.destination.left() + this.connector.destination.width() / 2);
|
|
var sl = this.connector.source.left();
|
|
var sw = this.connector.source.width();
|
|
var dt = this.connector.destination.top();
|
|
var dh = this.connector.destination.height();
|
|
var sourceMiddle = Math.floor(this.connector.source.top() + this.connector.source.height() / 2);
|
|
|
|
if(destMiddle > sl && destMiddle < sl + sw)
|
|
return false;
|
|
if(sourceMiddle > dt && sourceMiddle < dt + dh)
|
|
return false;
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Chooses the longest segment as the "middle" segment.
|
|
*/
|
|
this.getMiddleSegment = function()
|
|
{
|
|
if(this.startSegment.length > this.endSegment.length)
|
|
return this.startSegment;
|
|
else
|
|
return this.endSegment;
|
|
}
|
|
|
|
this.paint = function()
|
|
{
|
|
this.startSegment = this.connector.createSegment();
|
|
this.endSegment = this.connector.createSegment();
|
|
|
|
var destMiddleX = Math.floor(this.connector.destination.left() + this.connector.destination.width() / 2);
|
|
var sl = this.connector.source.left();
|
|
var sw = this.connector.source.width();
|
|
var dt = this.connector.destination.top();
|
|
var dh = this.connector.destination.height();
|
|
|
|
this.startSegment.startY = Math.floor(this.connector.source.top() + this.connector.source.height() / 2);
|
|
|
|
// decide which side of the source block to connect to
|
|
if(Math.abs(destMiddleX - sl) < Math.abs(destMiddleX - (sl + sw)))
|
|
{
|
|
// use the left face
|
|
this.startSegment.orientation = (destMiddleX < sl) ? LEFT : RIGHT;
|
|
this.startSegment.startX = sl;
|
|
}
|
|
else
|
|
{
|
|
// use the right face
|
|
this.startSegment.orientation = (destMiddleX > (sl + sw)) ? RIGHT : LEFT;
|
|
this.startSegment.startX = sl + sw;
|
|
}
|
|
|
|
this.startSegment.length = Math.abs(destMiddleX - this.startSegment.startX);
|
|
|
|
// decide which side of the destination block to connect to
|
|
if(Math.abs(this.startSegment.startY - dt) < Math.abs(this.startSegment.startY - (dt + dh)))
|
|
{
|
|
// use the upper face
|
|
this.endSegment.orientation = (this.startSegment.startY < dt) ? DOWN : UP;
|
|
this.endSegment.length = Math.abs(this.startSegment.startY - dt);
|
|
}
|
|
else
|
|
{
|
|
// use the lower face
|
|
this.endSegment.orientation = (this.startSegment.startY > (dt + dh)) ? UP : DOWN;
|
|
this.endSegment.length = Math.abs(this.startSegment.startY - (dt + dh));
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Vertical "L" connector routing strategy
|
|
*/
|
|
function VerticalLStrategy(connector)
|
|
{
|
|
this.connector = connector;
|
|
|
|
this.startSegment;
|
|
this.endSegment;
|
|
|
|
this.strategyName = "vertical_L";
|
|
|
|
this.isApplicable = function()
|
|
{
|
|
var sourceMiddle = Math.floor(this.connector.source.left() + this.connector.source.width() / 2);
|
|
var dl = this.connector.destination.left();
|
|
var dw = this.connector.destination.width();
|
|
var st = this.connector.source.top();
|
|
var sh = this.connector.source.height();
|
|
var destMiddle = Math.floor(this.connector.destination.top() + this.connector.destination.height() / 2);
|
|
|
|
if(sourceMiddle > dl && sourceMiddle < dl + dw)
|
|
return false;
|
|
if(destMiddle > st && destMiddle < st + sh)
|
|
return false;
|
|
return true;
|
|
}
|
|
/**
|
|
* Chooses the longest segment as the "middle" segment.
|
|
*/
|
|
this.getMiddleSegment = function() {
|
|
if(this.startSegment.length > this.endSegment.length)
|
|
return this.startSegment;
|
|
return this.endSegment;
|
|
}
|
|
this.paint = function() {
|
|
this.startSegment = this.connector.createSegment();
|
|
this.endSegment = this.connector.createSegment();
|
|
var destMiddleY = Math.floor(this.connector.destination.top() + this.connector.destination.height() / 2);
|
|
var dl = this.connector.destination.left();
|
|
var dw = this.connector.destination.width();
|
|
var st = this.connector.source.top();
|
|
var sh = this.connector.source.height();
|
|
this.startSegment.startX = Math.floor(this.connector.source.left() + this.connector.source.width() / 2);
|
|
// decide which side of the source block to connect to
|
|
if(Math.abs(destMiddleY - st) < Math.abs(destMiddleY - (st + sh))) {
|
|
// use the upper face
|
|
this.startSegment.orientation = (destMiddleY < st) ? UP : DOWN;
|
|
this.startSegment.startY = st;
|
|
} else {
|
|
// use the lower face
|
|
this.startSegment.orientation = (destMiddleY > (st + sh)) ? DOWN : UP;
|
|
this.startSegment.startY = st + sh;
|
|
}
|
|
|
|
this.startSegment.length = Math.abs(destMiddleY - this.startSegment.startY);
|
|
|
|
// decide which side of the destination block to connect to
|
|
if(Math.abs(this.startSegment.startX - dl) < Math.abs(this.startSegment.startX - (dl + dw)))
|
|
{
|
|
// use the left face
|
|
this.endSegment.orientation = (this.startSegment.startX < dl) ? RIGHT : LEFT;
|
|
this.endSegment.length = Math.abs(this.startSegment.startX - dl);
|
|
}
|
|
else
|
|
{
|
|
// use the right face
|
|
this.endSegment.orientation = (this.startSegment.startX > dl + dw) ? LEFT : RIGHT;
|
|
this.endSegment.length = Math.abs(this.startSegment.startX - (dl + dw));
|
|
}
|
|
}
|
|
}
|
|
|
|
function HorizontalCStrategy(connector, startOrientation)
|
|
{
|
|
this.connector = connector;
|
|
|
|
this.startSegment;
|
|
this.middleSegment;
|
|
this.endSegment;
|
|
|
|
this.strategyName = "horizontal_c";
|
|
|
|
this.getMiddleSegment = function() {
|
|
return this.middleSegment;
|
|
}
|
|
|
|
this.isApplicable = function() {
|
|
return true;
|
|
}
|
|
|
|
this.paint = function()
|
|
{
|
|
this.startSegment = connector.createSegment();
|
|
this.middleSegment = connector.createSegment();
|
|
this.endSegment = connector.createSegment();
|
|
|
|
var sign = 1;
|
|
if(startOrientation == RIGHT)
|
|
sign = -1;
|
|
|
|
var startX = this.connector.source.left();
|
|
if(startOrientation == RIGHT)
|
|
startX += this.connector.source.width();
|
|
var startY = Math.floor(this.connector.source.top() + this.connector.source.height() / 2);
|
|
|
|
var endX = this.connector.destination.left();
|
|
if(startOrientation == RIGHT)
|
|
endX += this.connector.destination.width();
|
|
var endY = Math.floor(this.connector.destination.top() + this.connector.destination.height() / 2);
|
|
|
|
this.startSegment.startX = startX;
|
|
this.startSegment.startY = startY;
|
|
this.startSegment.orientation = startOrientation;
|
|
this.startSegment.length = this.connector.minSegmentLength + Math.max(0, sign * (startX - endX));
|
|
|
|
var vLength = endY - startY;
|
|
this.middleSegment.orientation = vLength > 0 ? DOWN : UP;
|
|
this.middleSegment.length = Math.abs(vLength);
|
|
|
|
this.endSegment.orientation = startOrientation == LEFT ? RIGHT : LEFT;
|
|
this.endSegment.length = Math.max(0, sign * (endX - startX)) + this.connector.minSegmentLength;
|
|
}
|
|
}
|
|
|
|
function VerticalCStrategy(connector, startOrientation)
|
|
{
|
|
this.connector = connector;
|
|
|
|
this.startSegment;
|
|
this.middleSegment;
|
|
this.endSegment;
|
|
|
|
this.strategyName = "vertical_c";
|
|
|
|
this.getMiddleSegment = function()
|
|
{
|
|
return this.middleSegment;
|
|
}
|
|
|
|
this.isApplicable = function()
|
|
{
|
|
return true;
|
|
}
|
|
|
|
this.paint = function()
|
|
{
|
|
this.startSegment = connector.createSegment();
|
|
this.middleSegment = connector.createSegment();
|
|
this.endSegment = connector.createSegment();
|
|
|
|
var sign = 1;
|
|
if(startOrientation == DOWN)
|
|
sign = -1;
|
|
|
|
var startY = this.connector.source.top();
|
|
if(startOrientation == DOWN)
|
|
startY += this.connector.source.height();
|
|
var startX = Math.floor(this.connector.source.left() + this.connector.source.width() / 2);
|
|
|
|
var endY = this.connector.destination.top();
|
|
if(startOrientation == DOWN)
|
|
endY += this.connector.destination.height();
|
|
var endX = Math.floor(this.connector.destination.left() + this.connector.destination.width() / 2);
|
|
this.startSegment.startX = startX;
|
|
this.startSegment.startY = startY;
|
|
this.startSegment.orientation = startOrientation;
|
|
this.startSegment.length = this.connector.minSegmentLength + Math.max(0, sign * (startY - endY));
|
|
var hLength = endX - startX;
|
|
this.middleSegment.orientation = hLength > 0 ? RIGHT : LEFT;
|
|
this.middleSegment.length = Math.abs(hLength);
|
|
this.endSegment.orientation = startOrientation == UP ? DOWN : UP;
|
|
this.endSegment.length = Math.max(0, sign * (endY - startY)) + this.connector.minSegmentLength;
|
|
}
|
|
}
|
|
|
|
//strategies[0] = function(connector) {return new VerticalCStrategy(connector)};
|
|
strategies[0] = function(connector) {return new VerticalSStrategy(connector)};
|
|
strategies[1] = function(connector) {return new HorizontalSStrategy(connector)};
|
|
/*
|
|
strategies[2] = function(connector) {return new HorizontalCStrategy(connector, LEFT)};
|
|
strategies[1] = function(connector) {return new VerticalSStrategy(connector)};
|
|
strategies[2] = function(connector) {return new HorizontalLStrategy(connector)};
|
|
strategies[3] = function(connector) {return new VerticalLStrategy(connector)};
|
|
strategies[4] = function(connector) {return new HorizontalCStrategy(connector, LEFT)};
|
|
strategies[5] = function(connector) {return new HorizontalCStrategy(connector, RIGHT)};
|
|
strategies[6] = function(connector) {return new VerticalCStrategy(connector, UP)};
|
|
strategies[7] = function(connector) {return new VerticalCStrategy(connector, DOWN)};
|
|
*/
|