'use strict'; var assign = require('lodash/object/assign'), forEach = require('lodash/collection/forEach'), is = require('../../util/ModelUtil').is, isEventSubProcess = require('../../util/DiUtil').isEventSubProcess; /** * A provider for BPMN 2.0 elements context pad */ function ContextPadProvider(contextPad, modeling, elementFactory, connect, create, bpmnReplace, canvas) { contextPad.registerProvider(this); this._contextPad = contextPad; this._modeling = modeling; this._elementFactory = elementFactory; this._connect = connect; this._create = create; this._bpmnReplace = bpmnReplace; this._canvas = canvas; } ContextPadProvider.$inject = [ 'contextPad', 'modeling', 'elementFactory', 'connect', 'create', 'bpmnReplace', 'canvas' ]; ContextPadProvider.prototype.getContextPadEntries = function(element) { var contextPad = this._contextPad, modeling = this._modeling, elementFactory = this._elementFactory, connect = this._connect, create = this._create, bpmnReplace = this._bpmnReplace, canvas = this._canvas; var actions = {}; if (element.type === 'label') { return actions; } var bpmnElement = element.businessObject; function startConnect(event, element, autoActivate) { connect.start(event, element, autoActivate); } function removeElement(e) { if (element.waypoints) { modeling.removeConnection(element); } else { modeling.removeShape(element); } } function getReplaceMenuPosition(element) { var Y_OFFSET = 5; var diagramContainer = canvas.getContainer(), pad = contextPad.getPad(element).html; var diagramRect = diagramContainer.getBoundingClientRect(), padRect = pad.getBoundingClientRect(); var top = padRect.top - diagramRect.top; var left = padRect.left - diagramRect.left; var pos = { x: left, y: top + padRect.height + Y_OFFSET }; return pos; } function appendAction(type, className, options) { function appendListener(event, element) { var shape = elementFactory.createShape(assign({ type: type }, options)); create.start(event, shape, element); } var shortType = type.replace(/^bpmn\:/, ''); return { group: 'model', className: className, title: 'Append ' + shortType, action: { dragstart: appendListener, click: appendListener } }; } if (is(bpmnElement, 'bpmn:FlowNode')) { if (!is(bpmnElement, 'bpmn:EndEvent') && !is(bpmnElement, 'bpmn:EventBasedGateway') && !isEventType(bpmnElement, 'bpmn:IntermediateThrowEvent', 'bpmn:LinkEventDefinition') && !isEventSubProcess(bpmnElement)) { assign(actions, { 'append.end-event': appendAction('bpmn:EndEvent', 'icon-end-event-none'), 'append.gateway': appendAction('bpmn:ExclusiveGateway', 'icon-gateway-xor'), 'append.append-task': appendAction('bpmn:Task', 'icon-task'), 'append.intermediate-event': appendAction('bpmn:IntermediateThrowEvent', 'icon-intermediate-event-none') }); } if (is(bpmnElement, 'bpmn:EventBasedGateway')) { assign(actions, { 'append.receive-task': appendAction('bpmn:ReceiveTask', 'icon-receive-task'), 'append.message-intermediate-event': appendAction('bpmn:IntermediateCatchEvent', 'icon-intermediate-event-catch-message', { _eventDefinitionType: 'bpmn:MessageEventDefinition'}), 'append.timer-intermediate-event': appendAction('bpmn:IntermediateCatchEvent', 'icon-intermediate-event-catch-timer', { _eventDefinitionType: 'bpmn:TimerEventDefinition'}), 'append.condtion-intermediate-event': appendAction('bpmn:IntermediateCatchEvent', 'icon-intermediate-event-catch-condition', { _eventDefinitionType: 'bpmn:ConditionalEventDefinition'}), 'append.signal-intermediate-event': appendAction('bpmn:IntermediateCatchEvent', 'icon-intermediate-event-catch-signal', { _eventDefinitionType: 'bpmn:SignalEventDefinition'}) }); } } if (is(bpmnElement, 'bpmn:FlowNode') || (is(bpmnElement, 'bpmn:SequenceFlow') && (is(bpmnElement.sourceRef, 'bpmn:Activity') || is(bpmnElement.sourceRef, 'bpmn:ExclusiveGateway') || is(bpmnElement.sourceRef, 'bpmn:InclusiveGateway')))) { // Replace menu entry assign(actions, { 'replace': { group: 'edit', className: 'icon-screw-wrench', title: 'Change type', action: { click: function(event, element) { bpmnReplace.openChooser(getReplaceMenuPosition(element), element); } } } }); } if (is(bpmnElement, 'bpmn:FlowNode') || is(bpmnElement, 'bpmn:InteractionNode')) { assign(actions, { 'append.text-annotation': appendAction('bpmn:TextAnnotation', 'icon-text-annotation'), 'connect': { group: 'connect', className: 'icon-connection-multi', title: 'Connect using Sequence/MessageFlow', action: { click: startConnect, dragstart: startConnect } } }); } if (is(bpmnElement, 'bpmn:DataObjectReference')) { assign(actions, { 'connect': { group: 'connect', className: 'icon-connection-multi', title: 'Connect using DataInputAssociation', action: { click: startConnect, dragstart: startConnect } } }); } // Delete Element Entry assign(actions, { 'delete': { group: 'edit', className: 'icon-trash', title: 'Remove', action: { click: removeElement, dragstart: removeElement } } }); return actions; }; function isEventType(eventBo, type, definition) { var isType = eventBo.$instanceOf(type); var isDefinition = false; var definitions = eventBo.eventDefinitions || []; forEach(definitions, function(def) { if (def.$type === definition) { isDefinition = true; } }); return isType && isDefinition; } module.exports = ContextPadProvider;