mirror of
https://github.com/sartography/bpmn-js.git
synced 2025-01-18 04:51:10 +00:00
b8ed73b7f8
closes #1539
379 lines
8.6 KiB
JavaScript
379 lines
8.6 KiB
JavaScript
import inherits from 'inherits';
|
|
|
|
import CommandInterceptor from 'diagram-js/lib/command/CommandInterceptor';
|
|
|
|
import { find } from 'min-dash';
|
|
import { isExpanded } from '../../../util/DiUtil';
|
|
import { getBusinessObject, is } from '../../../util/ModelUtil';
|
|
import { getMid } from 'diagram-js/lib/layout/LayoutUtil';
|
|
import { getBBox } from 'diagram-js/lib/util/Elements';
|
|
import { asPlaneId, planeId } from '../../../util/DrilldownUtil';
|
|
|
|
|
|
var LOW_PRIORITY = 400;
|
|
var HIGH_PRIORITY = 600;
|
|
|
|
var DEFAULT_POSITION = {
|
|
x: 180,
|
|
y: 160
|
|
};
|
|
|
|
|
|
/**
|
|
* Creates diPlanes and canvas planes when collapsed subprocesses are created.
|
|
*
|
|
*
|
|
* @param {Canvas} canvas
|
|
* @param {EventBus} eventBus
|
|
* @param {Modeling} modeling
|
|
* @param {ElementFactory} elementFactory
|
|
* @param {BpmnFactory} bpmnFactory
|
|
* @param {Bpmnjs} bpmnjs
|
|
* @param {ElementRegistry} elementRegistry
|
|
*/
|
|
export default function SubProcessPlaneBehavior(
|
|
canvas, eventBus, modeling,
|
|
elementFactory, bpmnFactory, bpmnjs, elementRegistry) {
|
|
|
|
CommandInterceptor.call(this, eventBus);
|
|
|
|
this._canvas = canvas;
|
|
this._eventBus = eventBus;
|
|
this._modeling = modeling;
|
|
this._elementFactory = elementFactory;
|
|
this._bpmnFactory = bpmnFactory;
|
|
this._bpmnjs = bpmnjs;
|
|
this._elementRegistry = elementRegistry;
|
|
|
|
var self = this;
|
|
|
|
function isCollapsedSubProcess(element) {
|
|
return is(element, 'bpmn:SubProcess') && !isExpanded(element);
|
|
}
|
|
|
|
function createRoot(context) {
|
|
var shape = context.shape,
|
|
rootElement = context.newRootElement;
|
|
|
|
var businessObject = getBusinessObject(shape);
|
|
|
|
rootElement = self._addDiagram(rootElement || businessObject);
|
|
|
|
context.newRootElement = canvas.addRootElement(rootElement);
|
|
}
|
|
|
|
function removeRoot(context) {
|
|
var shape = context.shape;
|
|
|
|
var businessObject = getBusinessObject(shape);
|
|
self._removeDiagram(businessObject);
|
|
|
|
var rootElement = context.newRootElement = elementRegistry.get(planeId(businessObject));
|
|
|
|
canvas.removeRootElement(rootElement);
|
|
}
|
|
|
|
// add plane elements for newly created sub-processes
|
|
// this ensures we can actually drill down into the element
|
|
this.executed('shape.create', function(context) {
|
|
var shape = context.shape;
|
|
if (!isCollapsedSubProcess(shape)) {
|
|
return;
|
|
}
|
|
|
|
createRoot(context);
|
|
}, true);
|
|
|
|
|
|
this.reverted('shape.create', function(context) {
|
|
var shape = context.shape;
|
|
if (!isCollapsedSubProcess(shape)) {
|
|
return;
|
|
}
|
|
|
|
removeRoot(context);
|
|
}, true);
|
|
|
|
|
|
// rename secondary elements (roots) when the primary element changes
|
|
// this ensures rootElement.id = element.id + '_plane'
|
|
this.executed('element.updateProperties', function(context) {
|
|
var shape = context.element;
|
|
|
|
if (!isCollapsedSubProcess(shape)) {
|
|
return;
|
|
}
|
|
|
|
var properties = context.properties;
|
|
var oldProperties = context.oldProperties;
|
|
|
|
var oldId = oldProperties.id,
|
|
newId = properties.id;
|
|
|
|
if (oldId === newId) {
|
|
return;
|
|
}
|
|
|
|
var planeElement = elementRegistry.get(asPlaneId(oldId));
|
|
|
|
if (!planeElement) {
|
|
return;
|
|
}
|
|
|
|
elementRegistry.updateId(planeElement, asPlaneId(newId));
|
|
}, true);
|
|
|
|
|
|
this.reverted('element.updateProperties', function(context) {
|
|
var shape = context.element;
|
|
|
|
if (!isCollapsedSubProcess(shape)) {
|
|
return;
|
|
}
|
|
|
|
var properties = context.properties;
|
|
var oldProperties = context.oldProperties;
|
|
|
|
var oldId = oldProperties.id,
|
|
newId = properties.id;
|
|
|
|
if (oldId === newId) {
|
|
return;
|
|
}
|
|
|
|
var planeElement = elementRegistry.get(asPlaneId(newId));
|
|
|
|
if (!planeElement) {
|
|
return;
|
|
}
|
|
|
|
elementRegistry.updateId(planeElement, asPlaneId(oldId));
|
|
}, true);
|
|
|
|
|
|
// create/remove plane for the subprocess
|
|
this.executed('shape.toggleCollapse', LOW_PRIORITY, function(context) {
|
|
var shape = context.shape;
|
|
|
|
if (!is(shape, 'bpmn:SubProcess')) {
|
|
return;
|
|
}
|
|
|
|
if (!isExpanded(shape)) {
|
|
createRoot(context);
|
|
self._showRecursively(shape.children);
|
|
} else {
|
|
removeRoot(context);
|
|
}
|
|
|
|
}, true);
|
|
|
|
|
|
// create/remove plane for the subprocess
|
|
this.reverted('shape.toggleCollapse', LOW_PRIORITY, function(context) {
|
|
var shape = context.shape;
|
|
|
|
if (!is(shape, 'bpmn:SubProcess')) {
|
|
return;
|
|
}
|
|
|
|
if (!isExpanded(shape)) {
|
|
createRoot(context);
|
|
self._showRecursively(shape.children);
|
|
} else {
|
|
removeRoot(context);
|
|
}
|
|
|
|
}, true);
|
|
|
|
// move elements between planes
|
|
this.postExecuted('shape.toggleCollapse', HIGH_PRIORITY, function(context) {
|
|
var shape = context.shape;
|
|
|
|
if (!is(shape, 'bpmn:SubProcess')) {
|
|
return;
|
|
}
|
|
|
|
var rootElement = context.newRootElement;
|
|
|
|
if (!rootElement) {
|
|
return;
|
|
}
|
|
|
|
if (!isExpanded(shape)) {
|
|
|
|
// collapsed
|
|
self._moveChildrenToShape(shape, rootElement);
|
|
|
|
} else {
|
|
self._moveChildrenToShape(rootElement, shape);
|
|
}
|
|
}, true);
|
|
|
|
}
|
|
|
|
inherits(SubProcessPlaneBehavior, CommandInterceptor);
|
|
|
|
/**
|
|
* Moves the child elements from source to target.
|
|
*
|
|
* If the target is a plane, the children are moved to the top left corner.
|
|
* Otherwise, the center of the target is used.
|
|
*
|
|
* @param {Object|djs.model.Base} source
|
|
* @param {Object|djs.model.Base} target
|
|
*/
|
|
SubProcessPlaneBehavior.prototype._moveChildrenToShape = function(source, target) {
|
|
var modeling = this._modeling;
|
|
|
|
var children = source.children;
|
|
var offset;
|
|
|
|
if (!children) {
|
|
return;
|
|
}
|
|
|
|
// Only change plane if there are no visible children, but don't move them
|
|
var visibleChildren = children.filter(function(child) {
|
|
return !child.hidden;
|
|
});
|
|
|
|
if (!visibleChildren.length) {
|
|
modeling.moveElements(children, { x: 0, y: 0 }, target, { autoResize: false });
|
|
return;
|
|
}
|
|
|
|
// target is a plane
|
|
if (!target.x) {
|
|
offset = {
|
|
x: DEFAULT_POSITION.x - source.x ,
|
|
y: DEFAULT_POSITION.y - source.y
|
|
};
|
|
}
|
|
|
|
// source is a plane
|
|
else {
|
|
|
|
// move relative to the center of the shape
|
|
var targetMid = getMid(target);
|
|
var childrenBounds = getBBox(visibleChildren);
|
|
var childrenMid = getMid(childrenBounds);
|
|
|
|
offset = {
|
|
x: targetMid.x - childrenMid.x,
|
|
y: targetMid.y - childrenMid.y
|
|
};
|
|
}
|
|
|
|
modeling.moveElements(children, offset, target, { autoResize: false });
|
|
};
|
|
|
|
/**
|
|
* Sets `hidden` property on all children of the given shape.
|
|
*
|
|
* @param {Array} elements
|
|
* @param {Boolean} [hidden]
|
|
* @returns {Array} all child elements
|
|
*/
|
|
SubProcessPlaneBehavior.prototype._showRecursively = function(elements, hidden) {
|
|
var self = this;
|
|
|
|
var result = [];
|
|
elements.forEach(function(element) {
|
|
element.hidden = !!hidden;
|
|
|
|
result = result.concat(element);
|
|
|
|
if (element.children) {
|
|
result = result.concat(
|
|
self._showRecursively(element.children, element.collapsed || hidden)
|
|
);
|
|
}
|
|
});
|
|
|
|
return result;
|
|
};
|
|
|
|
/**
|
|
* Adds a given rootElement to the bpmnDi diagrams.
|
|
*
|
|
* @param {Object} rootElement
|
|
* @returns {Object} planeElement
|
|
*/
|
|
SubProcessPlaneBehavior.prototype._addDiagram = function(planeElement) {
|
|
var bpmnjs = this._bpmnjs;
|
|
var diagrams = bpmnjs.getDefinitions().diagrams;
|
|
|
|
if (!planeElement.businessObject) {
|
|
planeElement = this._createNewDiagram(planeElement);
|
|
}
|
|
|
|
diagrams.push(planeElement.di.$parent);
|
|
|
|
return planeElement;
|
|
};
|
|
|
|
|
|
/**
|
|
* Creates a new plane element for the given sub process.
|
|
*
|
|
* @param {Object} bpmnElement
|
|
*
|
|
* @return {Object} new diagram element
|
|
*/
|
|
SubProcessPlaneBehavior.prototype._createNewDiagram = function(bpmnElement) {
|
|
var bpmnFactory = this._bpmnFactory;
|
|
var elementFactory = this._elementFactory;
|
|
|
|
var diPlane = bpmnFactory.create('bpmndi:BPMNPlane', {
|
|
bpmnElement: bpmnElement
|
|
});
|
|
var diDiagram = bpmnFactory.create('bpmndi:BPMNDiagram', {
|
|
plane: diPlane
|
|
});
|
|
diPlane.$parent = diDiagram;
|
|
|
|
// add a virtual element (not being drawn),
|
|
// a copy cat of our BpmnImporter code
|
|
var planeElement = elementFactory.createRoot({
|
|
id: planeId(bpmnElement),
|
|
type: bpmnElement.$type,
|
|
di: diPlane,
|
|
businessObject: bpmnElement,
|
|
collapsed: true
|
|
});
|
|
|
|
return planeElement;
|
|
};
|
|
|
|
/**
|
|
* Removes the diagram for a given root element
|
|
*
|
|
* @param {Object} rootElement
|
|
* @returns {Object} removed bpmndi:BPMNDiagram
|
|
*/
|
|
SubProcessPlaneBehavior.prototype._removeDiagram = function(rootElement) {
|
|
var bpmnjs = this._bpmnjs;
|
|
|
|
var diagrams = bpmnjs.getDefinitions().diagrams;
|
|
|
|
var removedDiagram = find(diagrams, function(diagram) {
|
|
return diagram.plane.bpmnElement.id === rootElement.id;
|
|
});
|
|
|
|
diagrams.splice(diagrams.indexOf(removedDiagram), 1);
|
|
|
|
return removedDiagram;
|
|
};
|
|
|
|
|
|
SubProcessPlaneBehavior.$inject = [
|
|
'canvas',
|
|
'eventBus',
|
|
'modeling',
|
|
'elementFactory',
|
|
'bpmnFactory',
|
|
'bpmnjs',
|
|
'elementRegistry'
|
|
];
|