2018-04-02 21:01:53 +02:00
|
|
|
import inherits from 'inherits';
|
2015-04-22 09:08:38 +02:00
|
|
|
|
2018-04-02 21:01:53 +02:00
|
|
|
import {
|
|
|
|
assign
|
|
|
|
} from 'min-dash';
|
2015-04-22 09:08:38 +02:00
|
|
|
|
2018-04-02 21:01:53 +02:00
|
|
|
import BaseLayouter from 'diagram-js/lib/layout/BaseLayouter';
|
2015-04-22 09:08:38 +02:00
|
|
|
|
2018-04-02 21:01:53 +02:00
|
|
|
import {
|
2018-07-04 09:13:38 +02:00
|
|
|
repairConnection,
|
|
|
|
withoutRedundantPoints
|
2018-04-02 21:01:53 +02:00
|
|
|
} from 'diagram-js/lib/layout/ManhattanLayout';
|
2015-07-29 15:11:54 +02:00
|
|
|
|
2018-04-02 21:01:53 +02:00
|
|
|
import {
|
|
|
|
getMid,
|
|
|
|
getOrientation
|
|
|
|
} from 'diagram-js/lib/layout/LayoutUtil';
|
2017-12-10 12:29:54 +01:00
|
|
|
|
2018-04-02 21:01:53 +02:00
|
|
|
import {
|
|
|
|
isExpanded
|
|
|
|
} from '../../util/DiUtil';
|
2015-07-29 15:11:54 +02:00
|
|
|
|
2018-04-02 21:01:53 +02:00
|
|
|
import { is } from '../../util/ModelUtil';
|
2015-04-22 09:08:38 +02:00
|
|
|
|
|
|
|
|
2018-04-02 21:01:53 +02:00
|
|
|
export default function BpmnLayouter() {}
|
2015-04-22 09:08:38 +02:00
|
|
|
|
|
|
|
inherits(BpmnLayouter, BaseLayouter);
|
|
|
|
|
|
|
|
|
2016-06-22 14:38:44 +02:00
|
|
|
BpmnLayouter.prototype.layoutConnection = function(connection, hints) {
|
|
|
|
hints = hints || {};
|
|
|
|
|
2015-04-22 09:08:38 +02:00
|
|
|
var source = connection.source,
|
|
|
|
target = connection.target,
|
|
|
|
waypoints = connection.waypoints,
|
2016-06-22 14:38:44 +02:00
|
|
|
start = hints.connectionStart,
|
|
|
|
end = hints.connectionEnd;
|
2015-04-22 09:08:38 +02:00
|
|
|
|
2015-07-29 15:11:54 +02:00
|
|
|
var manhattanOptions,
|
2015-04-22 09:08:38 +02:00
|
|
|
updatedWaypoints;
|
|
|
|
|
2016-06-27 00:29:54 +02:00
|
|
|
if (!start) {
|
|
|
|
start = getConnectionDocking(waypoints && waypoints[0], source);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!end) {
|
|
|
|
end = getConnectionDocking(waypoints && waypoints[waypoints.length - 1], target);
|
|
|
|
}
|
2015-07-29 15:11:54 +02:00
|
|
|
|
2016-06-22 14:36:44 +02:00
|
|
|
// TODO(nikku): support vertical modeling
|
2015-07-29 15:11:54 +02:00
|
|
|
// and invert preferredLayouts accordingly
|
|
|
|
|
2016-01-27 09:45:05 +01:00
|
|
|
if (is(connection, 'bpmn:Association') ||
|
|
|
|
is(connection, 'bpmn:DataAssociation')) {
|
|
|
|
|
|
|
|
if (waypoints && !isCompensationAssociation(connection)) {
|
2016-06-27 00:29:54 +02:00
|
|
|
return [].concat([ start ], waypoints.slice(1, -1), [ end ]);
|
2016-01-27 09:45:05 +01:00
|
|
|
}
|
2015-10-13 11:07:35 +02:00
|
|
|
}
|
2015-04-22 09:08:38 +02:00
|
|
|
|
|
|
|
// manhattan layout sequence / message flows
|
|
|
|
if (is(connection, 'bpmn:MessageFlow')) {
|
2018-08-22 13:54:24 +02:00
|
|
|
manhattanOptions = getMessageFlowManhattanOptions(source, target);
|
2015-08-24 11:28:54 +02:00
|
|
|
|
2015-07-29 15:11:54 +02:00
|
|
|
} else
|
|
|
|
|
2015-04-22 09:08:38 +02:00
|
|
|
|
2015-07-29 15:11:54 +02:00
|
|
|
// layout all connection between flow elements h:h,
|
|
|
|
//
|
|
|
|
// except for
|
|
|
|
//
|
2018-07-04 09:13:38 +02:00
|
|
|
// (1) outgoing of BoundaryEvents -> layout based on attach orientation and target orientation
|
2015-07-29 15:11:54 +02:00
|
|
|
// (2) incoming / outgoing of Gateway -> v:h (outgoing), h:v (incoming)
|
2018-07-11 16:33:42 +02:00
|
|
|
// (3) loops from / to the same element
|
2015-07-29 15:11:54 +02:00
|
|
|
//
|
2016-01-27 09:45:05 +01:00
|
|
|
if (is(connection, 'bpmn:SequenceFlow') ||
|
|
|
|
isCompensationAssociation(connection)) {
|
2015-07-29 15:11:54 +02:00
|
|
|
|
2018-07-11 16:33:42 +02:00
|
|
|
if (source === target) {
|
|
|
|
manhattanOptions = {
|
|
|
|
preferredLayouts: [ 'b:l' ]
|
|
|
|
};
|
|
|
|
} else
|
|
|
|
|
2015-07-29 15:11:54 +02:00
|
|
|
if (is(source, 'bpmn:BoundaryEvent')) {
|
|
|
|
|
2018-07-04 09:13:38 +02:00
|
|
|
manhattanOptions = {
|
|
|
|
preferredLayouts: getBoundaryEventPreferredLayouts(source, target)
|
|
|
|
};
|
2015-07-29 15:11:54 +02:00
|
|
|
|
|
|
|
} else
|
|
|
|
|
|
|
|
if (is(source, 'bpmn:Gateway')) {
|
|
|
|
|
|
|
|
manhattanOptions = {
|
|
|
|
preferredLayouts: [ 'v:h' ]
|
|
|
|
};
|
|
|
|
} else
|
|
|
|
|
|
|
|
if (is(target, 'bpmn:Gateway')) {
|
|
|
|
|
|
|
|
manhattanOptions = {
|
|
|
|
preferredLayouts: [ 'h:v' ]
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
else {
|
|
|
|
manhattanOptions = {
|
|
|
|
preferredLayouts: [ 'h:h' ]
|
|
|
|
};
|
|
|
|
}
|
2018-07-11 16:33:42 +02:00
|
|
|
|
2015-04-22 09:08:38 +02:00
|
|
|
}
|
|
|
|
|
2015-07-29 15:11:54 +02:00
|
|
|
if (manhattanOptions) {
|
2015-04-22 09:08:38 +02:00
|
|
|
|
2016-06-22 14:38:44 +02:00
|
|
|
manhattanOptions = assign(manhattanOptions, hints);
|
2015-04-22 09:08:38 +02:00
|
|
|
|
|
|
|
updatedWaypoints =
|
2018-07-04 09:13:38 +02:00
|
|
|
withoutRedundantPoints(
|
|
|
|
repairConnection(
|
|
|
|
source, target,
|
|
|
|
start, end,
|
|
|
|
waypoints,
|
|
|
|
manhattanOptions
|
|
|
|
)
|
|
|
|
);
|
2015-04-22 09:08:38 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return updatedWaypoints || [ start, end ];
|
2015-07-13 10:37:43 +02:00
|
|
|
};
|
2015-07-29 15:11:54 +02:00
|
|
|
|
|
|
|
|
|
|
|
function getAttachOrientation(attachedElement) {
|
|
|
|
|
|
|
|
var hostElement = attachedElement.host,
|
|
|
|
padding = -10;
|
|
|
|
|
|
|
|
return getOrientation(getMid(attachedElement), hostElement, padding);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-08-22 13:54:24 +02:00
|
|
|
function getMessageFlowManhattanOptions(source, target) {
|
|
|
|
return {
|
|
|
|
preferredLayouts: [ 'straight', 'v:v' ],
|
|
|
|
preserveDocking: getMessageFlowPreserveDocking(source, target)
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
function getMessageFlowPreserveDocking(source, target) {
|
|
|
|
|
|
|
|
// (1) docking element connected to participant has precedence
|
|
|
|
|
|
|
|
if (is(target, 'bpmn:Participant')) {
|
|
|
|
return 'source';
|
|
|
|
}
|
|
|
|
|
|
|
|
if (is(source, 'bpmn:Participant')) {
|
|
|
|
return 'target';
|
|
|
|
}
|
|
|
|
|
|
|
|
// (2) docking element connected to expanded sub-process has precedence
|
|
|
|
|
|
|
|
if (isExpandedSubProcess(target)) {
|
|
|
|
return 'source';
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isExpandedSubProcess(source)) {
|
|
|
|
return 'target';
|
|
|
|
}
|
|
|
|
|
|
|
|
// (3) docking event has precedence
|
|
|
|
|
|
|
|
if (is(target, 'bpmn:Event')) {
|
|
|
|
return 'target';
|
|
|
|
}
|
|
|
|
|
|
|
|
if (is(source, 'bpmn:Event')) {
|
|
|
|
return 'source';
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-06-27 00:29:54 +02:00
|
|
|
function getConnectionDocking(point, shape) {
|
|
|
|
return point ? (point.original || point) : getMid(shape);
|
2015-07-29 15:11:54 +02:00
|
|
|
}
|
2016-01-27 09:45:05 +01:00
|
|
|
|
|
|
|
function isCompensationAssociation(connection) {
|
|
|
|
|
|
|
|
var source = connection.source,
|
|
|
|
target = connection.target;
|
|
|
|
|
|
|
|
return is(target, 'bpmn:Activity') &&
|
|
|
|
is(source, 'bpmn:BoundaryEvent') &&
|
|
|
|
target.businessObject.isForCompensation;
|
2016-06-21 11:59:33 +02:00
|
|
|
}
|
2016-06-22 14:36:44 +02:00
|
|
|
|
|
|
|
|
|
|
|
function isExpandedSubProcess(element) {
|
|
|
|
return is(element, 'bpmn:SubProcess') && isExpanded(element);
|
2018-07-04 09:13:38 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
function isSame(a, b) {
|
|
|
|
return a === b;
|
|
|
|
}
|
|
|
|
|
|
|
|
function isAnyOrientation(orientation, orientations) {
|
|
|
|
return orientations.indexOf(orientation) !== -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
var oppositeOrientationMapping = {
|
|
|
|
'top': 'bottom',
|
|
|
|
'top-right': 'bottom-left',
|
|
|
|
'top-left': 'bottom-right',
|
|
|
|
'right': 'left',
|
|
|
|
'bottom': 'top',
|
|
|
|
'bottom-right': 'top-left',
|
|
|
|
'bottom-left': 'top-right',
|
|
|
|
'left': 'right'
|
|
|
|
};
|
|
|
|
|
|
|
|
var orientationDirectionMapping = {
|
|
|
|
top: 't',
|
|
|
|
right: 'r',
|
|
|
|
bottom: 'b',
|
|
|
|
left: 'l'
|
|
|
|
};
|
|
|
|
|
|
|
|
function getHorizontalOrientation(orientation) {
|
|
|
|
var matches = /right|left/.exec(orientation);
|
|
|
|
|
|
|
|
return matches && matches[0];
|
|
|
|
}
|
|
|
|
|
|
|
|
function getVerticalOrientation(orientation) {
|
|
|
|
var matches = /top|bottom/.exec(orientation);
|
|
|
|
|
|
|
|
return matches && matches[0];
|
|
|
|
}
|
|
|
|
|
|
|
|
function isOppositeOrientation(a, b) {
|
|
|
|
return oppositeOrientationMapping[a] === b;
|
|
|
|
}
|
|
|
|
|
|
|
|
function isOppositeHorizontalOrientation(a, b) {
|
|
|
|
var horizontalOrientation = getHorizontalOrientation(a);
|
|
|
|
|
|
|
|
var oppositeHorizontalOrientation = oppositeOrientationMapping[horizontalOrientation];
|
|
|
|
|
|
|
|
return b.indexOf(oppositeHorizontalOrientation) !== -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
function isOppositeVerticalOrientation(a, b) {
|
|
|
|
var verticalOrientation = getVerticalOrientation(a);
|
|
|
|
|
|
|
|
var oppositeVerticalOrientation = oppositeOrientationMapping[verticalOrientation];
|
|
|
|
|
|
|
|
return b.indexOf(oppositeVerticalOrientation) !== -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
function isHorizontalOrientation(orientation) {
|
|
|
|
return orientation === 'right' || orientation === 'left';
|
|
|
|
}
|
|
|
|
|
|
|
|
function getBoundaryEventPreferredLayouts(source, target) {
|
|
|
|
var sourceMid = getMid(source),
|
|
|
|
targetMid = getMid(target),
|
|
|
|
attachOrientation = getAttachOrientation(source),
|
|
|
|
sourceLayout,
|
|
|
|
targetlayout;
|
|
|
|
|
|
|
|
var isLoop = isSame(source.host, target);
|
|
|
|
|
|
|
|
var attachedToSide = isAnyOrientation(attachOrientation, [ 'top', 'right', 'bottom', 'left' ]);
|
|
|
|
|
|
|
|
var isHorizontalAttachOrientation = isHorizontalOrientation(attachOrientation);
|
|
|
|
|
|
|
|
var targetOrientation = getOrientation(targetMid, sourceMid, {
|
|
|
|
x: source.width / 2 + target.width / 2,
|
|
|
|
y: source.height / 2 + target.height / 2
|
|
|
|
});
|
|
|
|
|
|
|
|
// source layout
|
|
|
|
|
|
|
|
// attached to either top, right, bottom or left side
|
|
|
|
if (attachedToSide) {
|
|
|
|
|
|
|
|
sourceLayout = orientationDirectionMapping[
|
|
|
|
isHorizontalAttachOrientation ?
|
|
|
|
getHorizontalOrientation(attachOrientation) :
|
|
|
|
getVerticalOrientation(attachOrientation)
|
|
|
|
];
|
|
|
|
|
|
|
|
} else
|
|
|
|
|
|
|
|
// attached to either top-right, top-left, bottom-right or bottom-left corner
|
|
|
|
{
|
|
|
|
|
|
|
|
// loop, same vertical or opposite horizontal orientation
|
|
|
|
if (isLoop ||
|
|
|
|
isSame(getVerticalOrientation(attachOrientation), getVerticalOrientation(targetOrientation)) ||
|
|
|
|
isOppositeOrientation(getHorizontalOrientation(attachOrientation), getHorizontalOrientation(targetOrientation))
|
|
|
|
) {
|
|
|
|
sourceLayout = orientationDirectionMapping[getVerticalOrientation(attachOrientation)];
|
|
|
|
} else {
|
|
|
|
sourceLayout = orientationDirectionMapping[getHorizontalOrientation(attachOrientation)];
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// target layout
|
|
|
|
|
|
|
|
// attached to either top, right, bottom or left side
|
|
|
|
if (attachedToSide) {
|
|
|
|
|
|
|
|
// loop or opposite horizontal/vertical orientation
|
|
|
|
if (
|
|
|
|
isLoop ||
|
|
|
|
(isHorizontalAttachOrientation ?
|
|
|
|
isOppositeHorizontalOrientation(attachOrientation, targetOrientation) :
|
|
|
|
isOppositeVerticalOrientation(attachOrientation, targetOrientation))
|
|
|
|
) {
|
|
|
|
targetlayout = isHorizontalAttachOrientation ? 'h' : 'v';
|
|
|
|
} else {
|
|
|
|
targetlayout = isHorizontalAttachOrientation ? 'v' : 'h';
|
|
|
|
}
|
|
|
|
|
|
|
|
} else
|
|
|
|
|
|
|
|
// attached to either top-right, top-left, bottom-right or bottom-left corner
|
|
|
|
{
|
|
|
|
|
|
|
|
// orientation is 'right', 'left'
|
|
|
|
// or same vertical orientation but also 'right' or 'left'
|
|
|
|
if (
|
|
|
|
isHorizontalOrientation(targetOrientation) ||
|
|
|
|
(isSame(getVerticalOrientation(attachOrientation), getVerticalOrientation(targetOrientation)) &&
|
|
|
|
getHorizontalOrientation(targetOrientation))
|
|
|
|
) {
|
|
|
|
targetlayout = 'h';
|
|
|
|
} else {
|
|
|
|
targetlayout = 'v';
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return [ sourceLayout + ':' + targetlayout ];
|
2016-06-22 14:36:44 +02:00
|
|
|
}
|