feat(modeling): allow label and group movement everywhere, round two

This partially reverts 0c0932d4c6f54181d7f06a626ef109ae7c00dccb.

Closes #1076
This commit is contained in:
Nico Rehwaldt 2019-06-14 13:24:09 +02:00 committed by merge-me[bot]
parent 5a69f9c0bc
commit 055fdf75e1
6 changed files with 243 additions and 100 deletions

View File

@ -7,13 +7,14 @@ import { isAny } from '../util/ModelingUtil';
var HIGH_PRIORITY = 1500; var HIGH_PRIORITY = 1500;
/** /**
* Ensure we correct hover targets to improve diagram interaction * Correct hover targets in certain situations to improve diagram interaction.
* during create and move.
* *
* @param {ElementRegistry} elementRegistry * @param {ElementRegistry} elementRegistry
* @param {EventBus} eventBus * @param {EventBus} eventBus
* @param {Canvas} canvas
*/ */
export default function FixHoverBehavior(elementRegistry, eventBus) { export default function FixHoverBehavior(elementRegistry, eventBus, canvas) {
eventBus.on([ eventBus.on([
'create.hover', 'create.hover',
'create.move', 'create.move',
@ -23,17 +24,30 @@ export default function FixHoverBehavior(elementRegistry, eventBus) {
'shape.move.end' 'shape.move.end'
], HIGH_PRIORITY, function(event) { ], HIGH_PRIORITY, function(event) {
var context = event.context, var context = event.context,
shape = context.shape, shape = context.shape || event.shape,
hover = event.hover; hover = event.hover;
// ensure elements are not dropped onto a bpmn:Lane but onto
// the underlying bpmn:Participant
if (is(hover, 'bpmn:Lane') && !isAny(shape, [ 'bpmn:Lane', 'bpmn:Participant' ])) { if (is(hover, 'bpmn:Lane') && !isAny(shape, [ 'bpmn:Lane', 'bpmn:Participant' ])) {
event.hover = getLanesRoot(hover); event.hover = getLanesRoot(hover);
event.hoverGfx = elementRegistry.getGraphics(event.hover); event.hoverGfx = elementRegistry.getGraphics(event.hover);
} }
var rootElement = canvas.getRootElement();
// ensure bpmn:Group and label elements are dropped
// always onto the root
if (hover !== rootElement && (shape.labelTarget || is(shape, 'bpmn:Group'))) {
event.hover = rootElement;
event.hoverGfx = elementRegistry.getGraphics(event.hover);
}
}); });
} }
FixHoverBehavior.$inject = [ FixHoverBehavior.$inject = [
'elementRegistry', 'elementRegistry',
'eventBus' 'eventBus',
'canvas'
]; ];

View File

@ -123,25 +123,10 @@ BpmnRules.prototype.init = function() {
shapes = context.shapes, shapes = context.shapes,
position = context.position; position = context.position;
var attach = canAttach(shapes, target, null, position); return canAttach(shapes, target, null, position) ||
canReplace(shapes, target, position) ||
if (attach || attach === null) { canMove(shapes, target, position) ||
return attach; canInsert(shapes, target, position);
}
var replace = canReplace(shapes, target, position);
if (replace || replace === null) {
return replace;
}
var move = canMove(shapes, target, position);
if (move || move === null) {
return move;
}
return canInsert(shapes, target, position);
}); });
this.addRule('shape.create', function(context) { this.addRule('shape.create', function(context) {
@ -464,11 +449,12 @@ function canConnect(source, target, connection) {
*/ */
function canDrop(element, target, position) { function canDrop(element, target, position) {
// can move labels everywhere // can move labels and groups everywhere
if (isLabel(element)) { if (isLabel(element) || isGroup(element)) {
return null; return true;
} }
// disallow to create elements on collapsed pools // disallow to create elements on collapsed pools
if (is(target, 'bpmn:Participant') && !isExpanded(target)) { if (is(target, 'bpmn:Participant') && !isExpanded(target)) {
return false; return false;
@ -766,21 +752,9 @@ function canMove(elements, target) {
return true; return true;
} }
var move, return elements.every(function(element) {
currentMove; return canDrop(element, target);
});
for (var i = 0; i < elements.length; i++) {
currentMove = canDrop(elements[i], target);
if (currentMove === false) {
return false;
}
move = move || currentMove;
}
return move;
} }
function canCreate(shape, target, source, position) { function canCreate(shape, target, source, position) {
@ -789,8 +763,8 @@ function canCreate(shape, target, source, position) {
return false; return false;
} }
if (isLabel(target)) { if (isLabel(target) || isGroup(target)) {
return null; return true;
} }
if (isSame(source, target)) { if (isSame(source, target)) {

View File

@ -0,0 +1,20 @@
<?xml version="1.0" encoding="UTF-8"?>
<definitions xmlns="http://www.omg.org/spec/BPMN/20100524/MODEL" xmlns:bpmndi="http://www.omg.org/spec/BPMN/20100524/DI" xmlns:omgdc="http://www.omg.org/spec/DD/20100524/DC" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" id="sid-38422fae-e03e-43a3-bef4-bd33b32041b2" targetNamespace="http://bpmn.io/bpmn" exporter="bpmn-js (https://demo.bpmn.io)" exporterVersion="4.0.0-beta.1">
<process id="Process">
<task id="Task" />
<group id="Group" categoryValueRef="CategoryValue_1heb3m9" />
</process>
<category id="Category_1g60m2v">
<categoryValue id="CategoryValue_1heb3m9" />
</category>
<bpmndi:BPMNDiagram id="BpmnDiagram_1">
<bpmndi:BPMNPlane id="BpmnPlane_1" bpmnElement="Process">
<bpmndi:BPMNShape id="Task_di" bpmnElement="Task">
<omgdc:Bounds x="264" y="81" width="100" height="80" />
</bpmndi:BPMNShape>
<bpmndi:BPMNShape id="Group_di" bpmnElement="Group">
<omgdc:Bounds x="185" y="200" width="150" height="120" />
</bpmndi:BPMNShape>
</bpmndi:BPMNPlane>
</bpmndi:BPMNDiagram>
</definitions>

View File

@ -0,0 +1,20 @@
<?xml version="1.0" encoding="UTF-8"?>
<definitions xmlns="http://www.omg.org/spec/BPMN/20100524/MODEL" xmlns:bpmndi="http://www.omg.org/spec/BPMN/20100524/DI" xmlns:omgdc="http://www.omg.org/spec/DD/20100524/DC" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" id="sid-38422fae-e03e-43a3-bef4-bd33b32041b2" targetNamespace="http://bpmn.io/bpmn" exporter="bpmn-js (https://demo.bpmn.io)" exporterVersion="4.0.0-beta.1">
<process id="Process">
<startEvent id="StartEvent" name="A" />
<task id="Task" />
</process>
<bpmndi:BPMNDiagram id="BpmnDiagram_1">
<bpmndi:BPMNPlane id="BpmnPlane_1" bpmnElement="Process">
<bpmndi:BPMNShape id="StartEvent_di" bpmnElement="StartEvent">
<omgdc:Bounds x="156" y="103" width="36" height="36" />
<bpmndi:BPMNLabel>
<omgdc:Bounds x="171" y="146" width="7" height="14" />
</bpmndi:BPMNLabel>
</bpmndi:BPMNShape>
<bpmndi:BPMNShape id="Task_di" bpmnElement="Task">
<omgdc:Bounds x="264" y="81" width="100" height="80" />
</bpmndi:BPMNShape>
</bpmndi:BPMNPlane>
</bpmndi:BPMNDiagram>
</definitions>

View File

@ -11,14 +11,16 @@ import moveModule from 'diagram-js/lib/features/move';
import { createCanvasEvent as canvasEvent } from '../../../../util/MockEvents'; import { createCanvasEvent as canvasEvent } from '../../../../util/MockEvents';
describe('features/modeling/behavior - fix hover', function() { var testModules = [
var testModules = [
coreModule, coreModule,
createModule, createModule,
moveModule, moveModule,
modelingModule modelingModule
]; ];
describe('features/modeling/behavior - fix hover', function() {
describe('drop on lane', function() {
var diagramXML = require('./FixHoverBehavior.participant.bpmn'); var diagramXML = require('./FixHoverBehavior.participant.bpmn');
@ -41,9 +43,10 @@ describe('features/modeling/behavior - fix hover', function() {
laneGfx = elementRegistry.getGraphics(lane); laneGfx = elementRegistry.getGraphics(lane);
})); }));
describe('create', function() { describe('create', function() {
it('should ensure hovering participant', inject( it('should set participant as hover element', inject(
function(create, dragging, elementFactory) { function(create, dragging, elementFactory) {
// given // given
@ -68,7 +71,7 @@ describe('features/modeling/behavior - fix hover', function() {
describe('move', function() { describe('move', function() {
it('should ensure hovering participant', inject( it('should set participant as hover element', inject(
function(dragging, elementRegistry, move) { function(dragging, elementRegistry, move) {
// given // given
@ -90,4 +93,116 @@ describe('features/modeling/behavior - fix hover', function() {
}); });
});
describe('label', function() {
var diagramXML = require('./FixHoverBehavior.label.bpmn');
beforeEach(bootstrapModeler(diagramXML, {
modules: testModules
}));
beforeEach(inject(function(dragging) {
dragging.setOptions({ manual: true });
}));
describe('move', function() {
it('should set root as hover element', inject(
function(dragging, elementRegistry, move, canvas) {
// given
var startEvent = elementRegistry.get('StartEvent');
var label = startEvent.label;
move.start(canvasEvent({ x: 175, y: 150 }), label, true);
// when
dragging.hover({ element: startEvent, gfx: elementRegistry.getGraphics(startEvent) });
dragging.move(canvasEvent({ x: 240, y: 220 }));
dragging.end();
// then
expect(label.parent).to.equal(canvas.getRootElement());
}
));
});
});
describe('group', function() {
var diagramXML = require('./FixHoverBehavior.group.bpmn');
beforeEach(bootstrapModeler(diagramXML, {
modules: testModules
}));
beforeEach(inject(function(dragging) {
dragging.setOptions({ manual: true });
}));
describe('create', function() {
it('should set root as hover element', inject(
function(dragging, elementFactory, elementRegistry, create, canvas) {
// given
var task = elementRegistry.get('Task');
var group = elementFactory.createShape({ type: 'bpmn:Group' });
create.start(canvasEvent({ x: 0, y: 0 }), group, true);
// when
dragging.hover({ element: task, gfx: elementRegistry.getGraphics(task) });
dragging.move(canvasEvent({ x: 240, y: 220 }));
dragging.end();
// then
expect(group.parent).to.equal(canvas.getRootElement());
}
));
});
describe('move', function() {
it('should set root as hover element', inject(
function(dragging, elementRegistry, move, canvas) {
// given
var task = elementRegistry.get('Task');
var group = elementRegistry.get('Group');
move.start(canvasEvent({ x: 175, y: 150 }), group, true);
// when
dragging.hover({ element: task, gfx: elementRegistry.getGraphics(task) });
dragging.move(canvasEvent({ x: 240, y: 220 }));
dragging.end();
// then
expect(group.parent).to.equal(canvas.getRootElement());
}
));
});
});
}); });

View File

@ -1015,38 +1015,38 @@ describe('features/modeling/rules - BpmnRules', function() {
it('-> MessageFlow', function() { it('-> MessageFlow', function() {
expectCanDrop(label, 'MessageFlow_labeled', null); expectCanDrop(label, 'MessageFlow_labeled', true);
}); });
it('-> CollapsedParticipant', function() { it('-> CollapsedParticipant', function() {
expectCanDrop(label, 'CollapsedParticipant', null); expectCanDrop(label, 'CollapsedParticipant', true);
}); });
it('-> Collaboration', function() { it('-> Collaboration', function() {
// then // then
expectCanDrop(label, 'Collaboration', null); expectCanDrop(label, 'Collaboration', true);
}); });
it('-> Task_in_SubProcess', function() { it('-> Task_in_SubProcess', function() {
expectCanDrop(label, 'Task_in_SubProcess', null); expectCanDrop(label, 'Task_in_SubProcess', true);
}); });
it('-> SequenceFlow', function() { it('-> SequenceFlow', function() {
expectCanDrop(label, 'SequenceFlow', null); expectCanDrop(label, 'SequenceFlow', true);
}); });
it('-> DataOutputAssociation', function() { it('-> DataOutputAssociation', function() {
expectCanDrop(label, 'DataOutputAssociation', null); expectCanDrop(label, 'DataOutputAssociation', true);
}); });
it('-> Group', function() { it('-> Group', function() {
expectCanDrop(label, 'Group', null); expectCanDrop(label, 'Group', true);
}); });
}); });