feat(copy-paste): copy error, escalation, message and signal references

* copy references when copying element
* add referenced root element if it doesn't exist
* do NOT add referenced root element if root element with same ID exists

Related to camunda/camunda-modeler#1049.
Related to camunda/camunda-modeler#1463.
This commit is contained in:
Philipp Fromme 2019-11-20 15:11:11 +01:00 committed by Nico Rehwaldt
parent 3ad47af299
commit 477217c891
4 changed files with 478 additions and 0 deletions

View File

@ -0,0 +1,165 @@
import inherits from 'inherits';
import {
find,
isArray,
matchPattern,
some
} from 'min-dash';
import CommandInterceptor from 'diagram-js/lib/command/CommandInterceptor';
import {
add as collectionAdd,
remove as collectionRemove
} from 'diagram-js/lib/util/Collections';
import {
getBusinessObject,
is
} from '../../../util/ModelUtil';
import { hasEventDefinition } from '../../../util/DiUtil';
var LOW_PRIORITY = 500;
/**
* Add referenced root elements (error, escalation, message, signal) if they don't exist.
* Copy referenced root elements on copy & paste.
*/
export default function RootElementReferenceBehavior(bpmnjs, eventBus, injector) {
injector.invoke(CommandInterceptor, this);
function hasRootElement(rootElement) {
var definitions = bpmnjs.getDefinitions(),
rootElements = definitions.get('rootElements');
return !!find(rootElements, matchPattern({ id: rootElement.id }));
}
function getRootElementReferencePropertyName(eventDefinition) {
if (is(eventDefinition, 'bpmn:ErrorEventDefinition')) {
return 'errorRef';
} else if (is(eventDefinition, 'bpmn:EscalationEventDefinition')) {
return 'escalationRef';
} else if (is(eventDefinition, 'bpmn:MessageEventDefinition')) {
return 'messageRef';
} else if (is(eventDefinition, 'bpmn:SignalEventDefinition')) {
return 'signalRef';
}
}
function getRootElementReferenced(eventDefinition) {
return eventDefinition.get(getRootElementReferencePropertyName(eventDefinition));
}
// create shape
this.executed('shape.create', function(context) {
var shape = context.shape;
if (!hasAnyEventDefinition(shape, [
'bpmn:ErrorEventDefinition',
'bpmn:EscalationEventDefinition',
'bpmn:MessageEventDefinition',
'bpmn:SignalEventDefinition'
])) {
return;
}
var businessObject = getBusinessObject(shape),
eventDefinitions = businessObject.get('eventDefinitions'),
eventDefinition = eventDefinitions[ 0 ],
rootElement = getRootElementReferenced(eventDefinition),
rootElements;
if (rootElement && !hasRootElement(rootElement)) {
rootElements = bpmnjs.getDefinitions().get('rootElements');
// add root element
collectionAdd(rootElements, rootElement);
context.addedRootElement = rootElement;
}
}, true);
this.reverted('shape.create', function(context) {
var addedRootElement = context.addedRootElement;
if (!addedRootElement) {
return;
}
var rootElements = bpmnjs.getDefinitions().get('rootElements');
// remove root element
collectionRemove(rootElements, addedRootElement);
}, true);
eventBus.on('copyPaste.copyElement', function(context) {
var descriptor = context.descriptor,
element = context.element;
if (!hasAnyEventDefinition(element, [
'bpmn:ErrorEventDefinition',
'bpmn:EscalationEventDefinition',
'bpmn:MessageEventDefinition',
'bpmn:SignalEventDefinition'
])) {
return;
}
var businessObject = getBusinessObject(element),
eventDefinitions = businessObject.get('eventDefinitions'),
eventDefinition = eventDefinitions[ 0 ],
rootElement = getRootElementReferenced(eventDefinition);
if (rootElement) {
descriptor.referencedRootElement = rootElement;
}
});
eventBus.on('copyPaste.pasteElement', LOW_PRIORITY, function(context) {
var descriptor = context.descriptor,
businessObject = descriptor.businessObject;
if (!hasAnyEventDefinition(businessObject, [
'bpmn:ErrorEventDefinition',
'bpmn:EscalationEventDefinition',
'bpmn:MessageEventDefinition',
'bpmn:SignalEventDefinition'
])) {
return;
}
var eventDefinitions = businessObject.get('eventDefinitions'),
eventDefinition = eventDefinitions[ 0 ],
referencedRootElement = descriptor.referencedRootElement;
if (!referencedRootElement) {
return;
}
eventDefinition.set(getRootElementReferencePropertyName(eventDefinition), referencedRootElement);
});
}
RootElementReferenceBehavior.$inject = [
'bpmnjs',
'eventBus',
'injector'
];
inherits(RootElementReferenceBehavior, CommandInterceptor);
// helpers //////////
function hasAnyEventDefinition(element, types) {
if (!isArray(types)) {
types = [ types ];
}
return some(types, function(type) {
return hasEventDefinition(element, type);
});
}

View File

@ -3,6 +3,7 @@ import AppendBehavior from './AppendBehavior';
import AssociationBehavior from './AssociationBehavior'; import AssociationBehavior from './AssociationBehavior';
import AttachEventBehavior from './AttachEventBehavior'; import AttachEventBehavior from './AttachEventBehavior';
import BoundaryEventBehavior from './BoundaryEventBehavior'; import BoundaryEventBehavior from './BoundaryEventBehavior';
import RootElementReferenceBehavior from './RootElementReferenceBehavior';
import CreateBehavior from './CreateBehavior'; import CreateBehavior from './CreateBehavior';
import FixHoverBehavior from './FixHoverBehavior'; import FixHoverBehavior from './FixHoverBehavior';
import CreateDataObjectBehavior from './CreateDataObjectBehavior'; import CreateDataObjectBehavior from './CreateDataObjectBehavior';
@ -37,6 +38,7 @@ export default {
'associationBehavior', 'associationBehavior',
'attachEventBehavior', 'attachEventBehavior',
'boundaryEventBehavior', 'boundaryEventBehavior',
'rootElementReferenceBehavior',
'createBehavior', 'createBehavior',
'fixHoverBehavior', 'fixHoverBehavior',
'createDataObjectBehavior', 'createDataObjectBehavior',
@ -69,6 +71,7 @@ export default {
associationBehavior: [ 'type', AssociationBehavior ], associationBehavior: [ 'type', AssociationBehavior ],
attachEventBehavior: [ 'type', AttachEventBehavior ], attachEventBehavior: [ 'type', AttachEventBehavior ],
boundaryEventBehavior: [ 'type', BoundaryEventBehavior ], boundaryEventBehavior: [ 'type', BoundaryEventBehavior ],
rootElementReferenceBehavior: [ 'type', RootElementReferenceBehavior ],
createBehavior: [ 'type', CreateBehavior ], createBehavior: [ 'type', CreateBehavior ],
fixHoverBehavior: [ 'type', FixHoverBehavior ], fixHoverBehavior: [ 'type', FixHoverBehavior ],
createDataObjectBehavior: [ 'type', CreateDataObjectBehavior ], createDataObjectBehavior: [ 'type', CreateDataObjectBehavior ],

View File

@ -0,0 +1,45 @@
<?xml version="1.0" encoding="UTF-8"?>
<bpmn:definitions xmlns:bpmn="http://www.omg.org/spec/BPMN/20100524/MODEL" xmlns:bpmndi="http://www.omg.org/spec/BPMN/20100524/DI" xmlns:dc="http://www.omg.org/spec/DD/20100524/DC" xmlns:camunda="http://camunda.org/schema/1.0/bpmn" id="Definitions_1" targetNamespace="http://bpmn.io/schema/bpmn" exporter="Camunda Modeler" exporterVersion="3.4.1">
<bpmn:process id="Process_1" isExecutable="true">
<bpmn:task id="Task_1" />
<bpmn:boundaryEvent id="MessageBoundaryEvent_1" attachedToRef="Task_1">
<bpmn:messageEventDefinition messageRef="Message_1" />
</bpmn:boundaryEvent>
<bpmn:boundaryEvent id="EscalationBoundaryEvent_1" attachedToRef="Task_1">
<bpmn:escalationEventDefinition escalationRef="Escalation_1" camunda:escalationCodeVariable="foo" />
</bpmn:boundaryEvent>
<bpmn:boundaryEvent id="ErrorBoundaryEvent_1" attachedToRef="Task_1">
<bpmn:errorEventDefinition errorRef="Error_1" camunda:errorCodeVariable="foo" camunda:errorMessageVariable="bar" />
</bpmn:boundaryEvent>
<bpmn:boundaryEvent id="SignalBoundaryEvent_1" attachedToRef="Task_1">
<bpmn:signalEventDefinition signalRef="Signal_1" />
</bpmn:boundaryEvent>
<bpmn:task id="Task_2" />
</bpmn:process>
<bpmn:message id="Message_1" name="Message_1" />
<bpmn:escalation id="Escalation_1" name="Escalation_1" escalationCode="42" />
<bpmn:error id="Error_1" name="Error_1" errorCode="1" />
<bpmn:signal id="Signal_1" name="Signal_1" />
<bpmndi:BPMNDiagram id="BPMNDiagram_1">
<bpmndi:BPMNPlane id="BPMNPlane_1" bpmnElement="Process_1">
<bpmndi:BPMNShape id="Task_1_di" bpmnElement="Task_1">
<dc:Bounds x="50" y="60" width="100" height="80" />
</bpmndi:BPMNShape>
<bpmndi:BPMNShape id="MessageBoundaryEvent_1_di" bpmnElement="MessageBoundaryEvent_1">
<dc:Bounds x="32" y="122" width="36" height="36" />
</bpmndi:BPMNShape>
<bpmndi:BPMNShape id="EscalationBoundaryEvent_1_di" bpmnElement="EscalationBoundaryEvent_1">
<dc:Bounds x="132" y="42" width="36" height="36" />
</bpmndi:BPMNShape>
<bpmndi:BPMNShape id="ErrorBoundaryEvent_1_di" bpmnElement="ErrorBoundaryEvent_1">
<dc:Bounds x="32" y="42" width="36" height="36" />
</bpmndi:BPMNShape>
<bpmndi:BPMNShape id="SignalBoundaryEvent_1_di" bpmnElement="SignalBoundaryEvent_1">
<dc:Bounds x="132" y="122" width="36" height="36" />
</bpmndi:BPMNShape>
<bpmndi:BPMNShape id="Task_2_di" bpmnElement="Task_2">
<dc:Bounds x="250" y="60" width="100" height="80" />
</bpmndi:BPMNShape>
</bpmndi:BPMNPlane>
</bpmndi:BPMNDiagram>
</bpmn:definitions>

View File

@ -0,0 +1,265 @@
import {
bootstrapModeler,
getBpmnJS,
inject
} from 'test/TestHelper';
import coreModule from 'lib/core';
import modelingModule from 'lib/features/modeling';
import {
getBusinessObject,
is
} from 'lib/util/ModelUtil';
import {
remove as collectionRemove
} from 'diagram-js/lib/util/Collections';
import {
filter,
find,
forEach,
matchPattern
} from 'min-dash';
var testModules = [
coreModule,
modelingModule
];
describe('features/modeling - root element reference behavior', function() {
var diagramXML = require('./RootElementReferenceBehavior.bpmn');
beforeEach(bootstrapModeler(diagramXML, { modules: testModules }));
describe('add root element', function() {
forEach([
'error',
'escalation',
'message',
'signal'
], function(type) {
describe(type, function() {
var id = capitalizeFirstChar(type) + 'BoundaryEvent_1';
var boundaryEvent,
host,
rootElement;
describe('should add', function() {
beforeEach(inject(function(bpmnjs, copyPaste, elementRegistry, modeling) {
// given
boundaryEvent = elementRegistry.get(id);
host = elementRegistry.get('Task_2');
var businessObject = getBusinessObject(boundaryEvent),
eventDefinitions = businessObject.get('eventDefinitions'),
eventDefinition = eventDefinitions[ 0 ];
rootElement = getRootElementReferenced(eventDefinition);
// when
copyPaste.copy(boundaryEvent);
modeling.removeShape(boundaryEvent);
collectionRemove(bpmnjs.getDefinitions().get('rootElements'), rootElement);
expect(hasRootElement(rootElement)).to.be.false;
boundaryEvent = copyPaste.paste({
element: host,
point: {
x: host.x,
y: host.y
},
hints: {
attach: 'attach'
}
})[0];
}));
it('<do>', function() {
// then
expect(hasRootElement(rootElement)).to.be.true;
});
it('<undo>', inject(function(commandStack) {
// when
commandStack.undo();
// then
expect(hasRootElement(rootElement)).to.be.false;
}));
it('<redo>', inject(function(commandStack) {
// given
commandStack.undo();
// when
commandStack.redo();
// then
expect(hasRootElement(rootElement)).to.be.true;
}));
});
it('should NOT add', inject(function(bpmnFactory, bpmnjs, copyPaste, elementRegistry, moddleCopy, modeling) {
// given
boundaryEvent = elementRegistry.get(id);
host = elementRegistry.get('Task_2');
var businessObject = getBusinessObject(boundaryEvent),
eventDefinitions = businessObject.get('eventDefinitions'),
eventDefinition = eventDefinitions[ 0 ],
rootElements = bpmnjs.getDefinitions().get('rootElements');
rootElement = getRootElementReferenced(eventDefinition);
copyPaste.copy(boundaryEvent);
modeling.removeShape(boundaryEvent);
collectionRemove(rootElements, rootElement);
expect(hasRootElement(rootElement)).to.be.false;
var rootElementWithSameId = bpmnFactory.create(rootElement.$type);
moddleCopy.copyElement(rootElement, rootElementWithSameId);
collectionRemove(rootElements, rootElementWithSameId);
// when
boundaryEvent = copyPaste.paste({
element: host,
point: {
x: host.x,
y: host.y
},
hints: {
attach: 'attach'
}
})[0];
// then
var rootElementsOfType = filter(rootElements, matchPattern({ $type: rootElement.$type }));
expect(rootElementsOfType).to.have.lengthOf(1);
}));
});
});
});
describe('copy root element reference', function() {
forEach([
'error',
'escalation',
'message',
'signal'
], function(type) {
describe(type, function() {
var id = capitalizeFirstChar(type) + 'BoundaryEvent_1';
var boundaryEvent,
host,
rootElement;
beforeEach(inject(function(copyPaste, elementRegistry) {
// given
boundaryEvent = elementRegistry.get(id);
host = elementRegistry.get('Task_2');
var businessObject = getBusinessObject(boundaryEvent),
eventDefinitions = businessObject.get('eventDefinitions'),
eventDefinition = eventDefinitions[ 0 ];
rootElement = getRootElementReferenced(eventDefinition);
copyPaste.copy(boundaryEvent);
// when
boundaryEvent = copyPaste.paste({
element: host,
point: {
x: host.x,
y: host.y
},
hints: {
attach: 'attach'
}
})[0];
}));
it('should copy root element reference', function() {
// then
var businessObject = getBusinessObject(boundaryEvent),
eventDefinitions = businessObject.get('eventDefinitions'),
eventDefinition = eventDefinitions[ 0 ];
expect(getRootElementReferenced(eventDefinition)).to.equal(rootElement);
});
});
});
});
});
// helpers //////////
function getRootElementReferenced(eventDefinition) {
if (is(eventDefinition, 'bpmn:ErrorEventDefinition')) {
return eventDefinition.get('errorRef');
} else if (is(eventDefinition, 'bpmn:EscalationEventDefinition')) {
return eventDefinition.get('escalationRef');
} else if (is(eventDefinition, 'bpmn:MessageEventDefinition')) {
return eventDefinition.get('messageRef');
} else if (is(eventDefinition, 'bpmn:SignalEventDefinition')) {
return eventDefinition.get('signalRef');
}
}
function hasRootElement(rootElement) {
var definitions = getBpmnJS().getDefinitions(),
rootElements = definitions.get('rootElements');
return !!find(rootElements, matchPattern({ id: rootElement.id }));
}
function capitalizeFirstChar(string) {
return string.charAt(0).toUpperCase() + string.slice(1);
}