mirror of
https://github.com/sartography/bpmn-js.git
synced 2025-01-15 03:25:02 +00:00
2e27d74306
* allow copying boundary events without host * remove CreateBoundaryEventBehavior in favor of AttachEventBehavior Closes #1154 Closes #1202 Closes #1204 Closes #1205
686 lines
17 KiB
JavaScript
686 lines
17 KiB
JavaScript
import {
|
|
bootstrapModeler,
|
|
getBpmnJS,
|
|
inject
|
|
} from 'test/TestHelper';
|
|
|
|
import TestContainer from 'mocha-test-container-support';
|
|
|
|
import coreModule from 'lib/core';
|
|
import createModule from 'diagram-js/lib/features/create';
|
|
import modelingModule from 'lib/features/modeling';
|
|
import moveModule from 'diagram-js/lib/features/move';
|
|
import rulesModule from 'lib/features/rules';
|
|
import snappingModule from 'lib/features/snapping';
|
|
|
|
import {
|
|
isSnapped,
|
|
mid
|
|
} from 'diagram-js/lib/features/snapping/SnapUtil';
|
|
|
|
import { createCanvasEvent as canvasEvent } from '../../../util/MockEvents';
|
|
|
|
import {
|
|
DEFAULT_LABEL_SIZE,
|
|
getExternalLabelMid
|
|
} from 'lib/util/LabelUtil';
|
|
|
|
import { queryAll as domQueryAll } from 'min-dom';
|
|
|
|
import { attr as svgAttr } from 'tiny-svg';
|
|
|
|
|
|
describe('features/snapping - BpmnCreateMoveSnapping', function() {
|
|
|
|
var testModules = [
|
|
coreModule,
|
|
createModule,
|
|
modelingModule,
|
|
moveModule,
|
|
rulesModule,
|
|
snappingModule,
|
|
{
|
|
__init__: [ function(dragging) {
|
|
dragging.setOptions({ manual: true });
|
|
} ]
|
|
}
|
|
];
|
|
|
|
|
|
describe('create participant', function() {
|
|
|
|
describe('process', function() {
|
|
|
|
var diagramXML = require('./BpmnCreateMoveSnapping.process.bpmn');
|
|
|
|
|
|
it('should snap particpant if constrained', function(done) {
|
|
|
|
bootstrapModeler(diagramXML, {
|
|
container: TestContainer.get(this),
|
|
modules: testModules
|
|
})(function() {
|
|
|
|
// when
|
|
inject(function(canvas, create, dragging, elementFactory, eventBus) {
|
|
|
|
// given
|
|
dragging.setOptions({ manual: true });
|
|
|
|
var participantShape = elementFactory.createParticipantShape(false),
|
|
rootElement = canvas.getRootElement(),
|
|
rootGfx = canvas.getGraphics(rootElement);
|
|
|
|
create.start(canvasEvent({ x: 0, y: 0 }), participantShape);
|
|
|
|
dragging.hover({ element: rootElement, gfx: rootGfx });
|
|
|
|
eventBus.once('create.move', function(event) {
|
|
|
|
// then
|
|
// expect snapped to avoid snapping outside of constraints
|
|
expect(isSnapped(event)).to.be.true;
|
|
|
|
done();
|
|
});
|
|
|
|
// when
|
|
dragging.move(canvasEvent({ x: 1000, y: 1000 }));
|
|
})();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
describe('collaboration', function() {
|
|
|
|
var diagramXML = require('./BpmnCreateMoveSnapping.collaboration.bpmn');
|
|
|
|
|
|
it('should snap to participant border with higher priority', function(done) {
|
|
|
|
var container = TestContainer.get(this);
|
|
|
|
bootstrapModeler(diagramXML, {
|
|
container: container,
|
|
modules: testModules
|
|
})(function() {
|
|
|
|
// when
|
|
inject(function(create, dragging, elementFactory, elementRegistry, eventBus) {
|
|
|
|
// given
|
|
dragging.setOptions({ manual: true });
|
|
|
|
var participant = elementFactory.createParticipantShape(false),
|
|
collaboration = elementRegistry.get('Collaboration_1'),
|
|
collaborationGfx = elementRegistry.getGraphics(collaboration);
|
|
|
|
create.start(canvasEvent({ x: 0, y: 0 }), participant);
|
|
|
|
dragging.hover({ element: collaboration, gfx: collaborationGfx });
|
|
|
|
dragging.move(canvasEventTopLeft({ x: 0, y: 0 }, participant));
|
|
|
|
eventBus.once('create.move', function(event) {
|
|
|
|
// then
|
|
// expect snap line at left border of participant
|
|
expect(svgAttr(domQueryAll('.djs-snap-line', container)[1], 'd'))
|
|
.to.equal('M 100,-100000 L 100, +100000');
|
|
|
|
done();
|
|
});
|
|
|
|
// when
|
|
dragging.move(canvasEventTopLeft({ x: 95, y: 400 }, participant));
|
|
})();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
describe('boundary events', function() {
|
|
|
|
describe('creating boundary event', function() {
|
|
|
|
var diagramXML = require('./BpmnCreateMoveSnapping.process.bpmn');
|
|
|
|
beforeEach(bootstrapModeler(diagramXML, {
|
|
modules: testModules
|
|
}));
|
|
|
|
var task, taskGfx, intermediateThrowEvent;
|
|
|
|
|
|
describe('without label', function() {
|
|
|
|
beforeEach(inject(function(create, dragging, elementRegistry, elementFactory) {
|
|
task = elementRegistry.get('Task_1');
|
|
|
|
taskGfx = elementRegistry.getGraphics(task);
|
|
|
|
intermediateThrowEvent = elementFactory.createShape({
|
|
type: 'bpmn:IntermediateThrowEvent'
|
|
});
|
|
|
|
create.start(canvasEvent({ x: 0, y: 0 }), intermediateThrowEvent);
|
|
|
|
dragging.hover({ element: task, gfx: taskGfx });
|
|
}));
|
|
|
|
|
|
it('should snap to top', inject(function(dragging) {
|
|
|
|
// when
|
|
dragging.move(canvasEvent({ x: 150, y: 95 }));
|
|
|
|
dragging.end();
|
|
|
|
// then
|
|
var boundaryEvent = getBoundaryEvent(task);
|
|
|
|
expect(mid(boundaryEvent)).to.eql({
|
|
x: 150,
|
|
y: 100 // 95 snapped to 100
|
|
});
|
|
}));
|
|
|
|
|
|
it('should snap to right', inject(function(dragging) {
|
|
|
|
// when
|
|
dragging.move(canvasEvent({ x: 195, y: 140 }));
|
|
|
|
dragging.end();
|
|
|
|
// then
|
|
var boundaryEvent = getBoundaryEvent(task);
|
|
|
|
expect(mid(boundaryEvent)).to.eql({
|
|
x: 200, // 195 snapped to 200
|
|
y: 140
|
|
});
|
|
}));
|
|
|
|
|
|
it('should snap to bottom', inject(function(dragging) {
|
|
|
|
// when
|
|
dragging.move(canvasEvent({ x: 150, y: 175 }));
|
|
|
|
dragging.end();
|
|
|
|
// then
|
|
var boundaryEvent = getBoundaryEvent(task);
|
|
|
|
expect(mid(boundaryEvent)).to.eql({
|
|
x: 150,
|
|
y: 180 // 175 snapped to 180
|
|
});
|
|
}));
|
|
|
|
|
|
it('should snap to left', inject(function(dragging) {
|
|
|
|
// when
|
|
dragging.move(canvasEvent({ x: 95, y: 140 }));
|
|
|
|
dragging.end();
|
|
|
|
// then
|
|
var boundaryEvent = getBoundaryEvent(task);
|
|
|
|
expect(mid(boundaryEvent)).to.eql({
|
|
x: 100, // 95 snapped to 100
|
|
y: 140
|
|
});
|
|
}));
|
|
});
|
|
|
|
|
|
describe('with label', function() {
|
|
|
|
beforeEach(inject(function(
|
|
bpmnFactory,
|
|
create,
|
|
dragging,
|
|
elementFactory,
|
|
elementRegistry,
|
|
textRenderer
|
|
) {
|
|
task = elementRegistry.get('Task_1');
|
|
|
|
taskGfx = elementRegistry.getGraphics(task);
|
|
|
|
intermediateThrowEvent = elementFactory.createShape({
|
|
businessObject: bpmnFactory.create('bpmn:IntermediateThrowEvent', {
|
|
name: 'Foo'
|
|
}),
|
|
type: 'bpmn:IntermediateThrowEvent',
|
|
x: 0,
|
|
y: 0
|
|
});
|
|
|
|
var externalLabelMid = getExternalLabelMid(intermediateThrowEvent);
|
|
|
|
var externalLabelBounds = textRenderer.getExternalLabelBounds(DEFAULT_LABEL_SIZE, 'Foo');
|
|
|
|
var label = elementFactory.createLabel({
|
|
labelTarget: intermediateThrowEvent,
|
|
x: externalLabelMid.x - externalLabelBounds.width / 2,
|
|
y: externalLabelMid.y - externalLabelBounds.height / 2,
|
|
width: externalLabelBounds.width,
|
|
height: externalLabelBounds.height
|
|
});
|
|
|
|
create.start(canvasEvent({ x: 0, y: 0 }), [ intermediateThrowEvent, label ]);
|
|
|
|
dragging.hover({ element: task, gfx: taskGfx });
|
|
}));
|
|
|
|
|
|
it('should snap to top-left', inject(function(dragging) {
|
|
|
|
// when
|
|
dragging.move(canvasEvent({ x: 90, y: 95 }));
|
|
|
|
dragging.end();
|
|
|
|
// then
|
|
var boundaryEvent = getBoundaryEvent(task);
|
|
|
|
expect(mid(boundaryEvent)).to.eql({
|
|
x: 100, // 90 snapped to 100
|
|
y: 100 // 95 snapped to 100
|
|
});
|
|
}));
|
|
|
|
|
|
it('should snap to top-right', inject(function(dragging) {
|
|
|
|
// when
|
|
dragging.move(canvasEvent({ x: 210, y: 95 }));
|
|
|
|
dragging.end();
|
|
|
|
// then
|
|
var boundaryEvent = getBoundaryEvent(task);
|
|
|
|
expect(mid(boundaryEvent)).to.eql({
|
|
x: 200, // 210 snapped to 200
|
|
y: 100 // 95 snapped to 100
|
|
});
|
|
}));
|
|
|
|
|
|
it('should snap to bottom-left', inject(function(dragging) {
|
|
|
|
// when
|
|
dragging.move(canvasEvent({ x: 90, y: 190 }));
|
|
|
|
dragging.end();
|
|
|
|
// then
|
|
var boundaryEvent = getBoundaryEvent(task);
|
|
|
|
expect(mid(boundaryEvent)).to.eql({
|
|
x: 100, // 90 snapped to 100
|
|
y: 180 // 190 snapped to 180
|
|
});
|
|
}));
|
|
|
|
|
|
it('should snap to bottom-right', inject(function(dragging) {
|
|
|
|
// when
|
|
dragging.move(canvasEvent({ x: 210, y: 190 }));
|
|
|
|
dragging.end();
|
|
|
|
// then
|
|
var boundaryEvent = getBoundaryEvent(task);
|
|
|
|
expect(mid(boundaryEvent)).to.eql({
|
|
x: 200, // 210 snapped to 200
|
|
y: 180 // 190 snapped to 180
|
|
});
|
|
}));
|
|
});
|
|
|
|
});
|
|
|
|
|
|
describe('snapping to boundary events', function() {
|
|
|
|
var diagramXML = require('./BpmnCreateMoveSnapping.boundary-events.bpmn');
|
|
|
|
beforeEach(bootstrapModeler(diagramXML, {
|
|
modules: testModules
|
|
}));
|
|
|
|
var task;
|
|
|
|
beforeEach(inject(function(dragging, elementRegistry, move) {
|
|
task = elementRegistry.get('Task_1');
|
|
|
|
var process = elementRegistry.get('Process_1'),
|
|
processGfx = elementRegistry.getGraphics(process);
|
|
|
|
move.start(canvasEventTopLeft({ x: 100, y: 400 }, task), task, true);
|
|
|
|
dragging.hover({ element: process, gfx: processGfx });
|
|
|
|
dragging.move(canvasEventTopLeft({ x: 100, y: 400 }, task));
|
|
}));
|
|
|
|
|
|
it('should snap to boundary events', inject(function(dragging) {
|
|
|
|
// when
|
|
dragging.move(canvasEventTopLeft({ x: 245, y: 400 }, task));
|
|
|
|
dragging.end();
|
|
|
|
// then
|
|
expect(task).to.have.bounds({
|
|
x: 250, // 245 snapped to 250
|
|
y: 400,
|
|
width: 100,
|
|
height: 80
|
|
});
|
|
}));
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
describe('sequence flows', function() {
|
|
|
|
var diagramXML = require('./BpmnCreateMoveSnapping.sequence-flows.bpmn');
|
|
|
|
beforeEach(bootstrapModeler(diagramXML, {
|
|
modules: testModules
|
|
}));
|
|
|
|
var sequenceFlow, sequenceFlowGfx, task;
|
|
|
|
beforeEach(inject(function(create, dragging, elementRegistry, elementFactory) {
|
|
sequenceFlow = elementRegistry.get('SequenceFlow_1');
|
|
sequenceFlowGfx = elementRegistry.getGraphics(sequenceFlow);
|
|
|
|
task = elementFactory.createShape({
|
|
type: 'bpmn:Task'
|
|
});
|
|
|
|
create.start(canvasEvent({ x: 0, y: 0 }), task);
|
|
|
|
dragging.hover({ element: sequenceFlow, gfx: sequenceFlowGfx });
|
|
}));
|
|
|
|
|
|
it('should add snap targets of sequence flow parent', inject(function(dragging) {
|
|
|
|
// when
|
|
dragging.move(canvasEventTopLeft({ x: 195, y: 60 }, task));
|
|
|
|
dragging.end();
|
|
|
|
// then
|
|
expect(task).to.have.bounds({
|
|
x: 200, // 195 snapped to 200
|
|
y: 60,
|
|
width: 100,
|
|
height: 80
|
|
});
|
|
}));
|
|
|
|
});
|
|
|
|
|
|
describe('lanes', function() {
|
|
|
|
var diagramXML = require('./BpmnCreateMoveSnapping.collaboration.bpmn');
|
|
|
|
beforeEach(bootstrapModeler(diagramXML, {
|
|
modules: testModules
|
|
}));
|
|
|
|
var task;
|
|
|
|
beforeEach(inject(function(dragging, elementRegistry, move) {
|
|
task = elementRegistry.get('Task_1');
|
|
|
|
move.start(canvasEvent({ x: 200, y: 165 }), task);
|
|
}));
|
|
|
|
|
|
it('should should NOT snap to lanes', inject(function(dragging) {
|
|
|
|
// when
|
|
// lane mid is { x: 415, y: 162.5 }
|
|
dragging.move(canvasEvent({ x: 410, y: 160 }));
|
|
|
|
dragging.end();
|
|
|
|
// then
|
|
expect(task).to.have.bounds({
|
|
x: 360,
|
|
y: 120,
|
|
width: 100,
|
|
height: 80
|
|
});
|
|
}));
|
|
|
|
});
|
|
|
|
|
|
describe('docking points', function() {
|
|
|
|
var diagramXML = require('./BpmnCreateMoveSnapping.docking-points.bpmn');
|
|
|
|
beforeEach(bootstrapModeler(diagramXML, {
|
|
modules: testModules
|
|
}));
|
|
|
|
var participant,
|
|
participantGfx;
|
|
|
|
beforeEach(inject(function(dragging, elementRegistry, move) {
|
|
participant = elementRegistry.get('Participant_2');
|
|
participantGfx = elementRegistry.getGraphics(participant);
|
|
}));
|
|
|
|
|
|
it('should snap to docking point (incoming connections)', inject(
|
|
function(dragging, elementRegistry, move) {
|
|
|
|
// given
|
|
var task = elementRegistry.get('Task_2');
|
|
|
|
move.start(canvasEvent({ x: 400, y: 540 }), task);
|
|
|
|
dragging.hover({ element: participant, gfx: participantGfx });
|
|
|
|
dragging.move(canvasEvent({ x: 0, y: 0 }));
|
|
|
|
// when
|
|
dragging.move(canvasEvent({ x: 270, y: 540 }));
|
|
|
|
dragging.end();
|
|
|
|
// then
|
|
expect(mid(task)).to.eql({
|
|
x: 275,
|
|
y: 540
|
|
});
|
|
}
|
|
));
|
|
|
|
|
|
it('should snap to docking point (outgoing connections)', inject(
|
|
function(dragging, elementRegistry, move) {
|
|
|
|
// given
|
|
var task = elementRegistry.get('Task_4');
|
|
|
|
move.start(canvasEvent({ x: 600, y: 540 }), task);
|
|
|
|
dragging.hover({ element: participant, gfx: participantGfx });
|
|
|
|
dragging.move(canvasEvent({ x: 0, y: 0 }));
|
|
|
|
// when
|
|
dragging.move(canvasEvent({ x: 475, y: 540 }));
|
|
|
|
dragging.end();
|
|
|
|
// then
|
|
expect(mid(task)).to.eql({
|
|
x: 480,
|
|
y: 540
|
|
});
|
|
}
|
|
));
|
|
|
|
});
|
|
|
|
|
|
describe('TRBL snapping', function() {
|
|
|
|
var diagramXML = require('./BpmnCreateMoveSnapping.trbl-snapping.bpmn');
|
|
|
|
beforeEach(bootstrapModeler(diagramXML, {
|
|
modules: testModules
|
|
}));
|
|
|
|
|
|
function get(element) {
|
|
return getBpmnJS().invoke(function(elementRegistry) {
|
|
return elementRegistry.get(element);
|
|
});
|
|
}
|
|
|
|
function absoluteMove(element, toPosition) {
|
|
|
|
getBpmnJS().invoke(function(elementRegistry, move, dragging, canvas) {
|
|
|
|
var parent = element.parent;
|
|
|
|
move.start(canvasEvent({ x: 0, y: 0 }), element);
|
|
|
|
dragging.hover({
|
|
element: parent,
|
|
gfx: canvas.getGraphics(parent)
|
|
});
|
|
|
|
dragging.move(canvasEvent({ x: 100, y: 100 }), element);
|
|
|
|
dragging.move(canvasEvent({
|
|
x: toPosition.x - element.x,
|
|
y: toPosition.y - element.y
|
|
}));
|
|
|
|
dragging.end();
|
|
});
|
|
|
|
}
|
|
|
|
|
|
it('should snap text annotations', function() {
|
|
|
|
// given
|
|
var annotation = get('TEXT_1');
|
|
var otherAnnotation = get('TEXT_2');
|
|
|
|
// when
|
|
absoluteMove(annotation, {
|
|
x: otherAnnotation.x + 5,
|
|
y: otherAnnotation.y - 5
|
|
});
|
|
|
|
// then
|
|
expect(annotation).to.have.position(otherAnnotation);
|
|
});
|
|
|
|
|
|
it('should snap task to container', function() {
|
|
|
|
// given
|
|
var task = get('TASK');
|
|
var subProcess = get('SUB_PROCESS_1');
|
|
|
|
// when
|
|
absoluteMove(task, {
|
|
x: subProcess.x,
|
|
y: subProcess.y - 5
|
|
});
|
|
|
|
// then
|
|
expect(task).to.have.position(subProcess);
|
|
});
|
|
|
|
|
|
it('should snap container to container', function() {
|
|
|
|
// given
|
|
var participant = get('PARTICIPANT_1');
|
|
var otherParticipant = get('PARTICIPANT_2');
|
|
|
|
// when
|
|
absoluteMove(participant, {
|
|
x: otherParticipant.x + 5,
|
|
y: otherParticipant.y
|
|
});
|
|
|
|
// then
|
|
expect(participant).to.have.position(otherParticipant);
|
|
});
|
|
|
|
|
|
it('should snap container to container right', function() {
|
|
|
|
// given
|
|
var participant = get('PARTICIPANT_1');
|
|
var otherParticipant = get('PARTICIPANT_2');
|
|
|
|
// when
|
|
absoluteMove(participant, {
|
|
x: otherParticipant.x + otherParticipant.width - participant.width + 5,
|
|
y: 5
|
|
});
|
|
|
|
// then
|
|
expect(participant).to.have.position({
|
|
x: otherParticipant.x + otherParticipant.width - participant.width,
|
|
y: 5
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
// helpers //////////
|
|
|
|
function canvasEventTopLeft(position, shape) {
|
|
return canvasEvent({
|
|
x: position.x + shape.width / 2,
|
|
y: position.y + shape.height / 2
|
|
});
|
|
}
|
|
|
|
function getBoundaryEvent(element) {
|
|
return element.attachers[0];
|
|
}
|