import { bootstrapModeler, getBpmnJS, inject } from 'test/TestHelper'; import { createEvent as globalEvent } from '../../../util/MockEvents'; import autoResizeModule from 'lib/features/auto-resize'; import coreModule from 'lib/core'; import customRulesModule from '../../../util/custom-rules'; import modelingModule from 'lib/features/modeling'; import replaceMenuProviderModule from 'lib/features/popup-menu'; import { query as domQuery, queryAll as domQueryAll, classes as domClasses } from 'min-dom'; import { is } from 'lib/util/ModelUtil'; import { isExpanded } from 'lib/util/DiUtil'; describe('features/popup-menu - replace menu provider', function() { var diagramXMLMarkers = require('../../../fixtures/bpmn/draw/activity-markers-simple.bpmn'), diagramXMLReplace = require('../../../fixtures/bpmn/features/replace/01_replace.bpmn'); var testModules = [ coreModule, modelingModule, replaceMenuProviderModule, customRulesModule ]; var openPopup = function(element, offset) { offset = offset || 100; getBpmnJS().invoke(function(popupMenu) { popupMenu.open(element, 'bpmn-replace', { x: element.x + offset, y: element.y + offset }); }); }; describe('toggle', function() { beforeEach(bootstrapModeler(diagramXMLMarkers, { modules: testModules })); var toggleActive; beforeEach(inject(function(popupMenu) { toggleActive = function(entryCls) { return popupMenu._getEntry(entryCls).active; }; })); describe('active attribute', function() { it('should be true for parallel marker', inject(function(bpmnReplace, elementRegistry) { // given var task = elementRegistry.get('ParallelTask'), loopCharacteristics = task.businessObject.loopCharacteristics; // assume expect(loopCharacteristics.isSequential).to.be.false; expect(loopCharacteristics.isSequential).to.exist; // when openPopup(task); // then expect(is(loopCharacteristics, 'bpmn:MultiInstanceLoopCharacteristics')).to.be.true; expect(toggleActive('toggle-parallel-mi')).to.be.true; })); it('should be true for sequential marker', inject(function(bpmnReplace, elementRegistry) { // given var task = elementRegistry.get('SequentialTask'), loopCharacteristics = task.businessObject.loopCharacteristics; // assume expect(loopCharacteristics.isSequential).to.be.true; expect(is(loopCharacteristics, 'bpmn:MultiInstanceLoopCharacteristics')).to.be.true; // when openPopup(task); // then expect(toggleActive('toggle-sequential-mi')).to.be.true; })); it('should be true for loop marker', inject(function(bpmnReplace, elementRegistry) { // given var task = elementRegistry.get('LoopTask'), loopCharacteristics = task.businessObject.loopCharacteristics; // assume expect(loopCharacteristics.isSequential).not.to.exist; expect(is(loopCharacteristics, 'bpmn:MultiInstanceLoopCharacteristics')).to.be.false; // when openPopup(task); // then expect(toggleActive('toggle-loop')).to.be.true; })); it('should be true for ad hoc marker', inject(function(bpmnReplace, elementRegistry) { // given var AdHocSubProcess = elementRegistry.get('AdHocSubProcess'); // when openPopup(AdHocSubProcess); // then expect(toggleActive('toggle-adhoc')).to.be.true; })); }); describe('exclusive toggle buttons', function() { it('should not toggle non exclusive buttons off', inject(function(bpmnReplace, elementRegistry) { var subProcess = elementRegistry.get('AdHocSubProcess'); openPopup(subProcess); // when triggerAction('toggle-parallel-mi'); openPopup(subProcess); // then expect(domClasses(queryEntry('toggle-adhoc')).has('active')).to.be.true; })); }); describe('non exclusive toggle buttons', function() { it('should not toggle exclusive buttons off', inject(function(bpmnReplace, elementRegistry) { // given var subProcess = elementRegistry.get('SubProcess'); // when // toggle parallel on openPopup(subProcess); triggerAction('toggle-parallel-mi'); // toggle ad hoc on openPopup(subProcess); var adHocSubProcess = triggerAction('toggle-adhoc'); openPopup(adHocSubProcess); // then var parallelEntry = queryEntry('toggle-parallel-mi'); var adHocEntry = queryEntry('toggle-adhoc'); expect(domClasses(parallelEntry).has('active')).to.be.true; expect(domClasses(adHocEntry).has('active')).to.be.true; }) ); }); describe('parallel toggle button', function() { it('should toggle parallel marker off', inject(function(bpmnReplace, elementRegistry) { // given var task = elementRegistry.get('ParallelTask'); openPopup(task); // when triggerAction('toggle-parallel-mi'); openPopup(task); var parallelEntry = queryEntry('toggle-parallel-mi'); // then expect(task.businessObject.loopCharacteristics).not.to.exist; expect(domClasses(parallelEntry).has('active')).to.be.false; }) ); it('should toggle parallel marker on', inject(function(bpmnReplace, elementRegistry) { // given var task = elementRegistry.get('Task'); openPopup(task); // when triggerAction('toggle-parallel-mi'); openPopup(task); var parallelEntry = queryEntry('toggle-parallel-mi'); // then expect(domClasses(parallelEntry).has('active')).to.be.true; expect(task.businessObject.loopCharacteristics.isSequential).to.be.false; expect(is(task.businessObject.loopCharacteristics, 'bpmn:MultiInstanceLoopCharacteristics')).to.be.true; })); it('should set sequential button inactive', inject(function(bpmnReplace, elementRegistry) { // given var task = elementRegistry.get('SequentialTask'); openPopup(task); // when triggerAction('toggle-parallel-mi'); openPopup(task); var sequentialEntry = queryEntry('toggle-sequential-mi'); // then expect(domClasses(sequentialEntry).has('active')).to.be.false; })); it('should set loop button inactive', inject(function(bpmnReplace, elementRegistry) { // given var task = elementRegistry.get('LoopTask'); openPopup(task); // when triggerAction('toggle-parallel-mi'); openPopup(task); var loopEntry = queryEntry('toggle-loop'); // then expect(domClasses(loopEntry).has('active')).to.be.false; })); }); describe('sequential toggle button', function() { it('should toggle sequential marker off', inject(function(bpmnReplace, elementRegistry) { // given var task = elementRegistry.get('SequentialTask'); openPopup(task); // when triggerAction('toggle-sequential-mi'); openPopup(task); var sequentialEntry = queryEntry('toggle-sequential-mi'); // then expect(task.businessObject.loopCharacteristics).not.to.exist; expect(domClasses(sequentialEntry).has('active')).to.be.false; })); it('should toggle sequential marker on', inject(function(bpmnReplace, elementRegistry) { // given var task = elementRegistry.get('Task'); openPopup(task); // when triggerAction('toggle-sequential-mi'); openPopup(task); var sequentialEntry = queryEntry('toggle-sequential-mi'); // then expect(domClasses(sequentialEntry).has('active')).to.be.true; expect(task.businessObject.loopCharacteristics.isSequential).to.be.true; expect(is(task.businessObject.loopCharacteristics, 'bpmn:MultiInstanceLoopCharacteristics')).to.be.true; })); it('should set loop button inactive', inject(function(bpmnReplace, elementRegistry) { // given var task = elementRegistry.get('LoopTask'); openPopup(task); // when triggerAction('toggle-sequential-mi'); openPopup(task); var loopEntry = queryEntry('toggle-loop'); // then expect(domClasses(loopEntry).has('active')).to.be.false; })); it('should set parallel button inactive', inject(function(bpmnReplace, elementRegistry) { // given var task = elementRegistry.get('ParallelTask'); openPopup(task); // when triggerAction('toggle-sequential-mi'); openPopup(task); var parallelEntry = queryEntry('toggle-parallel-mi'); // then expect(domClasses(parallelEntry).has('active')).to.be.false; })); }); describe('loop toggle button', function() { it('should toggle loop marker off', inject(function(bpmnReplace, elementRegistry) { // given var task = elementRegistry.get('LoopTask'); openPopup(task); // when triggerAction('toggle-loop'); openPopup(task); var loopEntry = queryEntry('toggle-loop'); // then expect(domClasses(loopEntry).has('active')).to.be.false; expect(is(task.businessObject.loopCharacteristics, 'bpmn:MultiInstanceLoopCharacteristics')).not.to.exist; })); it('should toggle loop marker on', inject(function(bpmnReplace, elementRegistry) { // given var task = elementRegistry.get('Task'); openPopup(task); // when triggerAction('toggle-loop'); openPopup(task); var loopEntry = queryEntry('toggle-loop'); // then expect(domClasses(loopEntry).has('active')).to.be.true; expect(is(task.businessObject.loopCharacteristics, 'bpmn:StandardLoopCharacteristics')).to.be.true; })); it('should set sequential button inactive', inject(function(bpmnReplace, elementRegistry) { // given var task = elementRegistry.get('SequentialTask'); openPopup(task); triggerAction('toggle-loop'); // when openPopup(task); var sequentialEntry = queryEntry('toggle-sequential-mi'); // then expect(domClasses(sequentialEntry).has('active')).to.be.false; })); it('should set parallel button inactive', inject(function(bpmnReplace, elementRegistry) { // given var task = elementRegistry.get('ParallelTask'); openPopup(task); // when triggerAction('toggle-loop'); openPopup(task); var parallelEntry = queryEntry('toggle-parallel-mi'); // then expect(domClasses(parallelEntry).has('active')).to.be.false; })); }); }); describe('replacing', function() { beforeEach(bootstrapModeler(diagramXMLMarkers, { modules: testModules })); it('should retain the loop characteristics', inject(function(bpmnReplace, elementRegistry) { // given var task = elementRegistry.get('SequentialTask'); openPopup(task); // when // replacing the task with a send task var sendTask = triggerAction('replace-with-send-task'); // then expect(sendTask.businessObject.loopCharacteristics).to.exist; expect(sendTask.businessObject.loopCharacteristics.isSequential).to.be.true; expect(is(sendTask.businessObject.loopCharacteristics, 'bpmn:MultiInstanceLoopCharacteristics')).to.be.true; })); it('should retain the loop characteristics for call activites', inject(function(bpmnReplace, elementRegistry) { // given var task = elementRegistry.get('SequentialTask'); openPopup(task); // when // replacing the task with a call activity var callActivity = triggerAction('replace-with-call-activity'); // then expect(callActivity.businessObject.loopCharacteristics).to.exist; expect(callActivity.businessObject.loopCharacteristics.isSequential).to.be.true; expect(is(callActivity.businessObject.loopCharacteristics, 'bpmn:MultiInstanceLoopCharacteristics')).to.be.true; }) ); it('should retain expanded status for sub processes', inject(function(bpmnReplace, elementRegistry) { // given var subProcess = elementRegistry.get('SubProcess'); openPopup(subProcess); // when // replacing the expanded sub process with a transaction var transaction = triggerAction('replace-with-transaction'); // then expect(isExpanded(transaction)).to.equal(isExpanded(subProcess)); }) ); it('should replace sub processes -> event sub process', inject(function(bpmnReplace, elementRegistry) { // given var subProcess = elementRegistry.get('SubProcess'); openPopup(subProcess); // when // replacing the expanded sub process with a eventSubProcess var eventSubProcess = triggerAction('replace-with-event-subprocess'); // then expect(eventSubProcess.businessObject.triggeredByEvent).to.be.true; }) ); it('should replace event sub processes -> sub process', inject(function(bpmnReplace, elementRegistry) { // given var eventSubProcess = elementRegistry.get('EventSubProcess'); openPopup(eventSubProcess); // when // replacing the expanded sub process with a eventSubProcess var subProcess = triggerAction('replace-with-subprocess'); // then expect(subProcess.businessObject.triggeredByEvent).to.be.false; }) ); it('should retain the loop characteristics and the expanded status for transactions', inject(function(bpmnReplace, elementRegistry) { // given var transaction = elementRegistry.get('Transaction'); openPopup(transaction); // when // replacing the transaction with an expanded sub process var subProcess = triggerAction('replace-with-subprocess'); // then expect(isExpanded(subProcess)).to.equal(isExpanded(transaction)); }) ); it('should not retain the loop characteristics morphing to an event sub process', inject(function(bpmnReplace, elementRegistry, modeling) { // given var transaction = elementRegistry.get('Transaction'); modeling.updateProperties(transaction, { loopCharacteristics: { isparallel: true } }); openPopup(transaction); // when // replacing the transaction with an event sub process var subProcess = triggerAction('replace-with-event-subprocess'); // then expect(isExpanded(subProcess)).to.equal(isExpanded(transaction)); }) ); it('should retain the expanded property morphing to an event sub processes', inject(function(bpmnReplace, elementRegistry) { // given var transaction = elementRegistry.get('Transaction'); openPopup(transaction); // when // replacing the transaction with an expanded sub process var eventSubProcess = triggerAction('replace-with-event-subprocess'); // then expect(isExpanded(eventSubProcess)).to.equal(isExpanded(transaction)); }) ); }); describe('replace menu', function() { describe('events', function() { beforeEach(bootstrapModeler(diagramXMLReplace, { modules: testModules })); it('should contain all except the current one', inject(function(bpmnReplace, elementRegistry) { // given var startEvent = elementRegistry.get('StartEvent_1'); // when openPopup(startEvent); // then expect(queryEntry('replace-with-none-start')).to.be.null; expect(queryEntries()).to.have.length(6); }) ); it('should contain all start events inside event sub process except the current one', inject(function(bpmnReplace, elementRegistry) { // given var startEvent = elementRegistry.get('StartEvent_3'); // when openPopup(startEvent); // then expect(queryEntry('replace-with-non-interrupting-message-start')).to.be.null; expect(queryEntry('replace-with-message-start')).to.exist; expect(queryEntries()).to.have.length(11); }) ); it('should contain all non interrupting start events inside event sub process except the current one', inject(function(bpmnReplace, elementRegistry) { // given var startEvent = elementRegistry.get('StartEvent_3'); var newElement = bpmnReplace.replaceElement(startEvent, { type: 'bpmn:StartEvent', eventDefinitionType: 'bpmn:ConditionalEventDefinition', isInterrupting: false }); // when openPopup(newElement); // then expect(queryEntry('replace-with-conditional-start')).to.exist; expect(queryEntry('replace-with-non-interrupting-conditional-start')).to.be.null; expect(queryEntries()).to.have.length(11); }) ); it('should contain all intermediate events except the current one', inject(function(bpmnReplace, elementRegistry) { // given var intermediateEvent = elementRegistry.get('IntermediateThrowEvent_1'); // when openPopup(intermediateEvent); // then expect(queryEntry('replace-with-none-intermediate-throw')).to.be.null; expect(queryEntries()).to.have.length(12); }) ); it('should contain all end events except the current one', inject(function(bpmnReplace, elementRegistry) { // given var endEvent = elementRegistry.get('EndEvent_1'); // when openPopup(endEvent); // then expect(queryEntry('replace-with-none-end')).to.be.null; expect(queryEntries()).to.have.length(8); }) ); }); describe('cancel events', function() { var diagramXML = require('../../../fixtures/bpmn/features/replace/cancel-events.bpmn'); beforeEach(bootstrapModeler(diagramXML, { modules: testModules })); it('should contain cancel event replace option', inject(function(elementRegistry) { // given var endEvent = elementRegistry.get('EndEvent_1'); // when openPopup(endEvent); // then expect(queryEntries()).to.have.length(9); }) ); it('should NOT contain cancel event replace option', inject(function(elementRegistry) { // given var endEvent = elementRegistry.get('EndEvent_2'); // when openPopup(endEvent); // then expect(queryEntries()).to.have.length(8); }) ); it('should contain cancel event replace option (boundary events)', inject(function(elementRegistry) { // given var boundaryEvent = elementRegistry.get('BoundaryEvent_1'); // when openPopup(boundaryEvent); // then expect(queryEntries()).to.have.length(13); }) ); it('should NOT contain cancel event replace option (boundary events)', inject(function(elementRegistry) { // given var boundaryEvent = elementRegistry.get('BoundaryEvent_2'); // when openPopup(boundaryEvent, 40); // then expect(queryEntries()).to.have.length(13); }) ); }); describe('boundary events', function() { beforeEach(bootstrapModeler(diagramXMLReplace, { modules: testModules })); it('should contain all boundary events for an interrupting boundary event', inject(function(bpmnReplace, elementRegistry) { // given var boundaryEvent = elementRegistry.get('BoundaryEvent_1'); // when openPopup(boundaryEvent, 40); // then expect(queryEntry('replace-with-conditional-intermediate-catch')).to.be.null; expect(queryEntries()).to.have.length(12); }) ); it('should contain all boundary events for a non interrupting boundary event', inject(function(bpmnReplace, elementRegistry) { // given var boundaryEvent = elementRegistry.get('BoundaryEvent_2'); // when openPopup(boundaryEvent, 40); // then expect(queryEntry('replace-with-non-interrupting-message-intermediate-catch')).to.be.null; expect(queryEntries()).to.have.length(12); }) ); it('should contain compensation boundary event', inject(function(bpmnReplace, elementRegistry) { // given var boundaryEvent = elementRegistry.get('BoundaryEvent_1'); // when openPopup(boundaryEvent, 40); // then expect(queryEntry('replace-with-compensation-boundary')).to.exist; }) ); }); describe('default flows', function() { var diagramXML = require('./ReplaceMenuProvider.defaultFlows.bpmn'); beforeEach(bootstrapModeler(diagramXML, { modules: testModules })); it('should show default replace option [gateway]', inject(function(elementRegistry) { // given var sequenceFlow = elementRegistry.get('SequenceFlow_3'); // when openPopup(sequenceFlow); // then expect(queryEntries()).to.have.length(1); })); it('should show Default replace option [task]', inject(function(elementRegistry) { // given var sequenceFlow = elementRegistry.get('SequenceFlow_15f5knn'); // when openPopup(sequenceFlow); // then expect(queryEntries()).to.have.length(2); })); it('should NOT show default replace option', inject(function(elementRegistry) { // given var sequenceFlow = elementRegistry.get('SequenceFlow_4'); // when openPopup(sequenceFlow); // then expect(queryEntries()).to.have.length(0); })); }); describe('default flows from inclusive gateways', function() { var diagramXML = require('./ReplaceMenuProvider.defaultFlowsFromInclusiveGateways.bpmn'); beforeEach(bootstrapModeler(diagramXML, { modules: testModules })); it('should show default replace option', inject(function(elementRegistry) { // given var sequenceFlow = elementRegistry.get('SequenceFlow_2'); // when openPopup(sequenceFlow); var sequenceFlowEntry = queryEntry('replace-with-sequence-flow'), defaultFlowEntry = queryEntry('replace-with-default-flow'); // then expect(sequenceFlowEntry).not.to.exist; expect(defaultFlowEntry).to.exist; })); it('should NOT show default replace option', inject(function(elementRegistry) { // given var sequenceFlow = elementRegistry.get('SequenceFlow_1'); // when openPopup(sequenceFlow); var sequenceFlowEntry = queryEntry('replace-with-sequence-flow'), defaultFlowEntry = queryEntry('replace-with-default-flow'); // then expect(sequenceFlowEntry).to.exist; expect(defaultFlowEntry).not.to.exist; })); }); describe('default flows from complex gateways', function() { var diagramXML = require('./ReplaceMenuProvider.defaultFlowsFromComplexGateways.bpmn'); beforeEach(bootstrapModeler(diagramXML, { modules: testModules })); it('should show default replace option', inject(function(elementRegistry) { // given var sequenceFlow = elementRegistry.get('SequenceFlow_2'); // when openPopup(sequenceFlow); var sequenceFlowEntry = queryEntry('replace-with-sequence-flow'), defaultFlowEntry = queryEntry('replace-with-default-flow'); // then expect(sequenceFlowEntry).not.to.exist; expect(defaultFlowEntry).to.exist; })); it('should NOT show default replace option', inject(function(elementRegistry) { // given var sequenceFlow = elementRegistry.get('SequenceFlow_1'); // when openPopup(sequenceFlow); var sequenceFlowEntry = queryEntry('replace-with-sequence-flow'), defaultFlowEntry = queryEntry('replace-with-default-flow'); // then expect(sequenceFlowEntry).to.exist; expect(defaultFlowEntry).not.to.exist; })); }); describe('conditional flows', function() { var diagramXML = require('./ReplaceMenuProvider.conditionalFlows.bpmn'); beforeEach(bootstrapModeler(diagramXML, { modules: testModules })); it('should show ConditionalFlow replace option', inject(function(elementRegistry) { // given var sequenceFlow = elementRegistry.get('SequenceFlow_3'); // when openPopup(sequenceFlow); var conditionalFlowEntry = queryEntry('replace-with-conditional-flow'); // then expect(conditionalFlowEntry).to.exist; expect(queryEntries()).to.have.length(2); })); it('should NOT show ConditionalFlow replace option', inject(function(elementRegistry) { // given var sequenceFlow = elementRegistry.get('SequenceFlow_1'); // when openPopup(sequenceFlow); // then expect(queryEntries()).to.have.length(0); })); }); describe('compensate activities', function() { var diagramXML = require('./ReplaceMenuProvider.compensation-activity.bpmn'); beforeEach(bootstrapModeler(diagramXML, { modules: testModules })); it('options should include subProcesses and callActivity', inject(function(elementRegistry) { // given var taskElement = elementRegistry.get('Task_1'); // when openPopup(taskElement); var callActivityEntry = queryEntry('replace-with-call-activity'), subProcessEntry = queryEntry('replace-with-collapsed-subprocess'); // then expect(callActivityEntry).to.exist; expect(subProcessEntry).to.exist; })); }); describe('collapsed subprocesses', function() { var diagramXML = require('./ReplaceMenuProvider.collapsedSubProcess.bpmn'); beforeEach(bootstrapModeler(diagramXML, { modules: testModules })); it('options do not include collapsed subprocesses itself', inject(function(elementRegistry) { // given var collapsedSubProcess = elementRegistry.get('Task_1'); // when openPopup(collapsedSubProcess); var collapsedSubProcessEntry = queryEntry('replace-with-collapsed-subprocess'); // then expect(collapsedSubProcessEntry).not.to.exist; })); }); }); describe('integration', function() { describe('default flows', function() { var diagramXML = require('./ReplaceMenuProvider.defaultFlows.bpmn'); beforeEach(bootstrapModeler(diagramXML, { modules: testModules })); it('should replace SequenceFlow with DefaultFlow [gateway]', inject(function(elementRegistry) { // given var sequenceFlow = elementRegistry.get('SequenceFlow_3'); // when openPopup(sequenceFlow); triggerAction('replace-with-default-flow'); var gateway = elementRegistry.get('ExclusiveGateway_1'); // then expect(gateway.businessObject.default).to.equal(sequenceFlow.businessObject); })); it('should replace SequenceFlow with DefaultFlow [task]', inject(function(elementRegistry) { // given var sequenceFlow = elementRegistry.get('SequenceFlow_15f5knn'); // when openPopup(sequenceFlow); triggerAction('replace-with-default-flow'); var task = elementRegistry.get('Task_1ei94kl'); // then expect(task.businessObject.default).to.equal(sequenceFlow.businessObject); })); it('should morph DefaultFlow into a SequenceFlow [task]', inject(function(elementRegistry) { // given var sequenceFlow = elementRegistry.get('SequenceFlow_15f5knn'); openPopup(sequenceFlow); triggerAction('replace-with-default-flow'); // when openPopup(sequenceFlow); triggerAction('replace-with-sequence-flow'); var task = elementRegistry.get('Task_1ei94kl'); // then expect(task.businessObject.default).not.to.exist; })); it('should morph DefaultFlow into a SequenceFlow [task] -> undo', inject(function(elementRegistry, commandStack) { // given var sequenceFlow = elementRegistry.get('SequenceFlow_15f5knn'); openPopup(sequenceFlow); triggerAction('replace-with-default-flow'); // when openPopup(sequenceFlow); triggerAction('replace-with-sequence-flow'); commandStack.undo(); var task = elementRegistry.get('Task_1ei94kl'); // then expect(task.businessObject.default).to.equal(sequenceFlow.businessObject); }) ); it('should morph DefaultFlow into a ConditionalFlow [task]', inject(function(elementRegistry) { // given var sequenceFlow = elementRegistry.get('SequenceFlow_15f5knn'), task = elementRegistry.get('Task_1ei94kl'); openPopup(sequenceFlow); triggerAction('replace-with-default-flow'); // when openPopup(sequenceFlow); triggerAction('replace-with-conditional-flow'); // then expect(task.businessObject.default).not.to.exist; })); it('should morph DefaultFlow into a ConditionalFlow [task] -> undo', inject(function(elementRegistry, commandStack) { // given var sequenceFlow = elementRegistry.get('SequenceFlow_15f5knn'), task = elementRegistry.get('Task_1ei94kl'); openPopup(sequenceFlow); triggerAction('replace-with-default-flow'); // when openPopup(sequenceFlow); triggerAction('replace-with-conditional-flow'); commandStack.undo(); // then expect(task.businessObject.default).to.equal(sequenceFlow.businessObject); }) ); it('should replace SequenceFlow with DefaultFlow [gateway] -> undo', inject(function(elementRegistry, commandStack) { // given var sequenceFlow = elementRegistry.get('SequenceFlow_3'), gateway = elementRegistry.get('ExclusiveGateway_1'); // when openPopup(sequenceFlow); triggerAction('replace-with-default-flow'); commandStack.undo(); // then expect(gateway.businessObject.default).not.to.exist; }) ); it('should replace SequenceFlow with DefaultFlow [task] -> undo', inject(function(elementRegistry, commandStack) { // given var sequenceFlow = elementRegistry.get('SequenceFlow_15f5knn'); // when openPopup(sequenceFlow); triggerAction('replace-with-default-flow'); commandStack.undo(); var task = elementRegistry.get('Task_1ei94kl'); // then expect(task.businessObject.default).not.to.exist; }) ); it('should only have one DefaultFlow', inject(function(elementRegistry) { // given var sequenceFlow = elementRegistry.get('SequenceFlow_1'), sequenceFlow3 = elementRegistry.get('SequenceFlow_3'); // when // trigger morphing sequenceFlow3 to default flow openPopup(sequenceFlow3); triggerAction('replace-with-default-flow'); // trigger morphing sequenceFlow to default flow openPopup(sequenceFlow); triggerAction('replace-with-default-flow'); var gateway = elementRegistry.get('ExclusiveGateway_1'); // then expect(gateway.businessObject.default).to.equal(sequenceFlow.businessObject); })); it('should replace DefaultFlow with SequenceFlow when changing source', inject(function(elementRegistry, modeling) { // given var sequenceFlow = elementRegistry.get('SequenceFlow_1'), task = elementRegistry.get('Task_2'); openPopup(sequenceFlow); triggerAction('replace-with-default-flow'); // when modeling.reconnectStart(sequenceFlow, task, [ { x: 686, y: 267, original: { x: 686, y: 307 } }, { x: 686, y: 207, original: { x: 686, y: 187 } } ]); var gateway = elementRegistry.get('ExclusiveGateway_1'); // then expect(gateway.businessObject.default).not.to.exist; }) ); it('should replace DefaultFlow with SequenceFlow when changing source -> undo', inject(function(elementRegistry, modeling, commandStack) { // given var sequenceFlow = elementRegistry.get('SequenceFlow_1'), task = elementRegistry.get('Task_2'); openPopup(sequenceFlow); triggerAction('replace-with-default-flow'); // when modeling.reconnectStart(sequenceFlow, task, [ { x: 686, y: 267, original: { x: 686, y: 307 } }, { x: 686, y: 207, original: { x: 686, y: 187 } } ]); commandStack.undo(); var gateway = elementRegistry.get('ExclusiveGateway_1'); // then expect(gateway.businessObject.default).equal(sequenceFlow.businessObject); }) ); [ 'bpmn:Activity', 'bpmn:EndEvent', 'bpmn:IntermediateThrowEvent', 'bpmn:IntermediateCatchEvent' ].forEach(function(type) { it('should keep DefaultFlow when changing target to ' + type, inject(function(elementRegistry, elementFactory, canvas, modeling) { // given var sequenceFlow = elementRegistry.get('SequenceFlow_1'), root = canvas.getRootElement(); var intermediateEvent = elementFactory.createShape({ type: type }); modeling.createShape(intermediateEvent, { x: 686, y: 50 }, root); openPopup(sequenceFlow); triggerAction('replace-with-default-flow'); // when modeling.reconnectEnd(sequenceFlow, intermediateEvent, [ { x: 686, y: 267, original: { x: 686, y: 307 } }, { x: 686, y: 50, original: { x: 686, y: 75 } } ]); var gateway = elementRegistry.get('ExclusiveGateway_1'); // then expect(gateway.businessObject.default).to.exist; }) ); }); it('should replace DefaultFlow with SequenceFlow when changing target -> undo', inject(function(elementRegistry, elementFactory, canvas, modeling, commandStack) { // given var sequenceFlow = elementRegistry.get('SequenceFlow_1'), rootElement = canvas.getRootElement(); var intermediateEvent = elementFactory.createShape({ type: 'bpmn:StartEvent' }); modeling.createShape(intermediateEvent, { x: 686, y: 50 }, rootElement); openPopup(sequenceFlow); triggerAction('replace-with-default-flow'); // when modeling.reconnectEnd(sequenceFlow, intermediateEvent, [ { x: 686, y: 267, original: { x: 686, y: 307 } }, { x: 686, y: 50, original: { x: 686, y: 75 } } ]); commandStack.undo(); var gateway = elementRegistry.get('ExclusiveGateway_1'); // then expect(gateway.businessObject.default).equal(sequenceFlow.businessObject); }) ); it('should keep DefaultFlow when morphing Gateway', inject(function(elementRegistry, bpmnReplace) { // given var sequenceFlow = elementRegistry.get('SequenceFlow_3'), exclusiveGateway = elementRegistry.get('ExclusiveGateway_1'); // when openPopup(sequenceFlow); triggerAction('replace-with-default-flow'); var inclusiveGateway = bpmnReplace.replaceElement(exclusiveGateway, { type: 'bpmn:InclusiveGateway' }); // then expect(inclusiveGateway.businessObject.default).to.equal(sequenceFlow.businessObject); })); it('should keep DefaultFlow when morphing Task', inject( function(elementRegistry, bpmnReplace) { // given var sequenceFlow = elementRegistry.get('SequenceFlow_15f5knn'), task = elementRegistry.get('Task_1ei94kl'); // when openPopup(sequenceFlow); // trigger DefaultFlow replacement triggerAction('replace-with-default-flow'); var sendTask = bpmnReplace.replaceElement(task, { type: 'bpmn:SendTask' }); // then expect(sendTask.businessObject.default).to.equal(sequenceFlow.businessObject); }) ); it('should keep DefaultFlow when morphing Gateway -> undo', inject(function(elementRegistry, bpmnReplace, commandStack) { // given var sequenceFlow = elementRegistry.get('SequenceFlow_3'), exclusiveGateway = elementRegistry.get('ExclusiveGateway_1'); // when openPopup(sequenceFlow); triggerAction('replace-with-default-flow'); bpmnReplace.replaceElement(exclusiveGateway, { type: 'bpmn:InclusiveGateway' }); commandStack.undo(); // then expect(exclusiveGateway.businessObject.default).to.equal(sequenceFlow.businessObject); }) ); it('should remove any conditionExpression when morphing to DefaultFlow', inject(function(elementRegistry, modeling, moddle) { // given var sequenceFlow = elementRegistry.get('SequenceFlow_3'), exclusiveGateway = elementRegistry.get('ExclusiveGateway_1'); var conditionExpression = moddle.create('bpmn:FormalExpression', { body: '' }); modeling.updateProperties(sequenceFlow, { conditionExpression: conditionExpression }); // when openPopup(sequenceFlow); // trigger DefaultFlow replacement triggerAction('replace-with-default-flow'); // then expect(exclusiveGateway.businessObject.default).to.equal(sequenceFlow.businessObject); expect(sequenceFlow.businessObject.conditionExpression).not.to.exist; }) ); it('should remove any conditionExpression when morphing to DefaultFlow -> undo', inject(function(elementRegistry, modeling, moddle, commandStack) { // given var sequenceFlow = elementRegistry.get('SequenceFlow_3'), exclusiveGateway = elementRegistry.get('ExclusiveGateway_1'); var conditionExpression = moddle.create('bpmn:FormalExpression', { body: '' }); modeling.updateProperties(sequenceFlow, { conditionExpression: conditionExpression }); // when openPopup(sequenceFlow); // trigger DefaultFlow replacement triggerAction('replace-with-default-flow'); commandStack.undo(); // then expect(exclusiveGateway.businessObject.default).not.to.exist; expect(sequenceFlow.businessObject.conditionExpression).to.equal(conditionExpression); }) ); }); describe('conditional flows', function() { var diagramXML = require('./ReplaceMenuProvider.conditionalFlows.bpmn'); beforeEach(bootstrapModeler(diagramXML, { modules: testModules })); it('should morph into a ConditionalFlow', inject(function(elementRegistry) { // given var sequenceFlow = elementRegistry.get('SequenceFlow_2'); // when openPopup(sequenceFlow); triggerAction('replace-with-conditional-flow'); // then expect(sequenceFlow.businessObject.conditionExpression.$type).to.equal('bpmn:FormalExpression'); })); it('should morph into a ConditionalFlow -> undo', inject( function(elementRegistry, commandStack) { // given var sequenceFlow = elementRegistry.get('SequenceFlow_2'); // when openPopup(sequenceFlow); triggerAction('replace-with-conditional-flow'); commandStack.undo(); // then expect(sequenceFlow.businessObject.conditionExpression).not.to.exist; } )); it('should morph back into a SequenceFlow', inject(function(elementRegistry) { // given var sequenceFlow = elementRegistry.get('SequenceFlow_2'); // when openPopup(sequenceFlow); // trigger ConditionalFlow replacement triggerAction('replace-with-conditional-flow'); openPopup(sequenceFlow); // replace with SequenceFlow triggerAction('replace-with-sequence-flow'); // then expect(sequenceFlow.businessObject.conditionExpression).not.to.exist; })); it('should replace ConditionalFlow with SequenceFlow when changing source', inject(function(elementRegistry, modeling) { // given var sequenceFlow = elementRegistry.get('SequenceFlow_3'), startEvent = elementRegistry.get('StartEvent_1'); // when openPopup(sequenceFlow); triggerAction('replace-with-conditional-flow'); // when modeling.reconnectStart(sequenceFlow, startEvent, [ { x: 196, y: 197, original: { x: 178, y: 197 } }, { x: 497, y: 278, original: { x: 547, y: 278 } } ]); // then expect(sequenceFlow.businessObject.conditionExpression).not.to.exist; }) ); it('should replace ConditionalFlow with SequenceFlow when changing source -> undo', inject(function(elementRegistry, modeling, commandStack) { // given var sequenceFlow = elementRegistry.get('SequenceFlow_3'), startEvent = elementRegistry.get('StartEvent_1'); // when openPopup(sequenceFlow); triggerAction('replace-with-conditional-flow'); // when modeling.reconnectStart(sequenceFlow, startEvent, [ { x: 196, y: 197, original: { x: 178, y: 197 } }, { x: 497, y: 278, original: { x: 547, y: 278 } } ]); commandStack.undo(); // then expect(sequenceFlow.businessObject.conditionExpression.$type).to.equal('bpmn:FormalExpression'); }) ); [ 'bpmn:Activity', 'bpmn:EndEvent', 'bpmn:IntermediateThrowEvent', 'bpmn:IntermediateCatchEvent' ].forEach(function(type) { it('should keep ConditionalFlow when changing target to ' + type, inject(function(elementRegistry, elementFactory, canvas, modeling) { // given var sequenceFlow = elementRegistry.get('SequenceFlow_3'), root = canvas.getRootElement(), intermediateEvent = elementFactory.createShape({ type: type }); modeling.createShape(intermediateEvent, { x: 497, y: 197 }, root); openPopup(sequenceFlow); triggerAction('replace-with-conditional-flow'); // when modeling.reconnectEnd(sequenceFlow, intermediateEvent, [ { x: 389, y: 197, original: { x: 389, y: 197 } }, { x: 497, y: 197, original: { x: 497, y: 197 } } ]); // then expect(sequenceFlow.businessObject.conditionExpression).to.exist; }) ); }); it('should replace ConditionalFlow with SequenceFlow when changing target -> undo', inject(function(elementRegistry, elementFactory, canvas, modeling, commandStack) { // given var sequenceFlow = elementRegistry.get('SequenceFlow_3'), root = canvas.getRootElement(), intermediateEvent = elementFactory.createShape({ type: 'bpmn:StartEvent' }); modeling.createShape(intermediateEvent, { x: 497, y: 197 }, root); openPopup(sequenceFlow); // trigger ConditionalFlow replacement triggerAction('replace-with-conditional-flow'); // when modeling.reconnectEnd(sequenceFlow, intermediateEvent, [ { x: 389, y: 197, original: { x: 389, y: 197 } }, { x: 497, y: 197, original: { x: 497, y: 197 } } ]); commandStack.undo(); // then expect(sequenceFlow.businessObject.conditionExpression.$type).to.equal('bpmn:FormalExpression'); }) ); }); describe('adhoc sub process', function() { var diagramXML = require('./ReplaceMenuProvider.collapsedSubProcess.bpmn'); beforeEach(bootstrapModeler(diagramXML, { modules: testModules.concat(autoResizeModule) })); describe('sub process -> adhoc', function() { it('should not resize', inject(function(elementRegistry, modeling) { // given var subProcess = elementRegistry.get('SubProcess_1'); var resizeShapeSpy = sinon.spy(modeling, 'resizeShape'); // when openPopup(subProcess); triggerAction('toggle-adhoc'); // then expect(resizeShapeSpy).not.to.have.been.called; })); it('should not lay out connection', inject(function(elementRegistry, modeling) { // given var subProcess = elementRegistry.get('SubProcess_1'); var layoutConnectionSpy = sinon.spy(modeling, 'layoutConnection'); // when openPopup(subProcess); triggerAction('toggle-adhoc'); // then expect(layoutConnectionSpy).not.to.have.been.called; })); }); describe('adhoc -> sub process', function() { it('should not resize', inject(function(elementRegistry, modeling) { // given var adhocSubProcess = elementRegistry.get('AdhocSubProcess_1'); var resizeShapeSpy = sinon.spy(modeling, 'resizeShape'); // when openPopup(adhocSubProcess); triggerAction('toggle-adhoc'); // then expect(resizeShapeSpy).not.to.have.been.called; })); it('should not lay out connection', inject(function(elementRegistry, modeling) { // given var adhocSubProcess = elementRegistry.get('AdhocSubProcess_1'); var layoutConnectionSpy = sinon.spy(modeling, 'layoutConnection'); // when openPopup(adhocSubProcess); triggerAction('toggle-adhoc'); // then expect(layoutConnectionSpy).not.to.have.been.called; })); }); }); }); describe('rules', function() { var diagramXML = require('../../../fixtures/bpmn/basic.bpmn'); beforeEach(bootstrapModeler(diagramXML, { modules: testModules.concat([ customRulesModule ]) })); it('should get entries by default', inject(function(elementRegistry) { // given var startEvent = elementRegistry.get('StartEvent_1'); // when openPopup(startEvent); // then expect(queryEntries()).to.have.length.above(0); })); it('should get entries when custom rule returns true', inject(function(elementRegistry, customRules) { // given var startEvent = elementRegistry.get('StartEvent_1'); customRules.addRule('shape.replace', function() { return true; }); // when openPopup(startEvent); // then expect(queryEntries()).to.have.length.above(0); }) ); it('should get no entries when custom rule returns false', inject(function(elementRegistry, customRules) { // given var startEvent = elementRegistry.get('StartEvent_1'); customRules.addRule('shape.replace', function() { return false; }); // when openPopup(startEvent); // then expect(queryEntries()).to.have.length(0); }) ); it('should provide element to custom rules', inject(function(elementRegistry, customRules) { // given var startEvent = elementRegistry.get('StartEvent_1'); var actual; customRules.addRule('shape.replace', function(context) { actual = context.element; }); // when openPopup(startEvent); // then expect(actual).to.equal(startEvent); })); it('should evaluate rule once', inject(function(elementRegistry, customRules) { // given var callCount = 0; var startEvent = elementRegistry.get('StartEvent_1'); customRules.addRule('shape.replace', function() { callCount++; }); // when openPopup(startEvent); // then expect(callCount).to.equal(1); })); }); }); // helpers //////////// function queryEntry(id) { var container = getBpmnJS().get('canvas').getContainer(); return domQuery('.djs-popup [data-id="' + id + '"]', container); } function queryEntries() { var container = getBpmnJS().get('canvas').getContainer(); return domQueryAll('.djs-popup .entry', container); } function triggerAction(id) { var entry = queryEntry(id); if (!entry) { throw new Error('entry "'+ id +'" not found in replace menu'); } var popupMenu = getBpmnJS().get('popupMenu'); return popupMenu.trigger(globalEvent(entry, { x: 0, y: 0 })); }