'use strict'; var assign = require('lodash/object/assign'), inherits = require('inherits'); var LabelUtil = require('../../../util/LabelUtil'), LabelLayoutUtil = require('./util/LabelLayoutUtil'), ModelUtil = require('../../../util/ModelUtil'), is = ModelUtil.is, getBusinessObject = ModelUtil.getBusinessObject; var hasExternalLabel = LabelUtil.hasExternalLabel, getExternalLabelMid = LabelUtil.getExternalLabelMid, getLabelAdjustment = LabelLayoutUtil.getLabelAdjustment; var CommandInterceptor = require('diagram-js/lib/command/CommandInterceptor'); /** * A component that makes sure that external labels are added * together with respective elements and properly updated (DI wise) * during move. * * @param {EventBus} eventBus * @param {Modeling} modeling * @param {BpmnFactory} bpmnFactory */ function LabelSupport(eventBus, modeling, bpmnFactory) { CommandInterceptor.call(this, eventBus); // create external labels on shape creation this.postExecute([ 'shape.create', 'connection.create' ], function(e) { var context = e.context; var element = context.shape || context.connection, businessObject = element.businessObject; var position; if (hasExternalLabel(businessObject)) { position = getExternalLabelMid(element); modeling.createLabel(element, position, { id: businessObject.id + '_label', hidden: !businessObject.name, businessObject: businessObject }); } }); // update label position on waypoints change if still hidden this.postExecute([ 'connection.updateWaypoints' ], function(e) { var context = e.context, connection = context.connection, label = connection.label; if (!label) { return; } if (label.hidden) { var position = getExternalLabelMid(connection); var delta = { x: position.x - label.x - label.width / 2, y: position.y - label.y - label.height / 2 }; modeling.moveShape(label, delta); } }); this.postExecute([ 'connection.layout', 'connection.reconnectEnd', 'connection.reconnectStart' ], function(e) { var context = e.context, oldWaypoints = context.oldWaypoints, newWaypoints = context.newWaypoints || context.connection.waypoints, label = context.connection.label, hints = context.hints || {}, delta; if (e.command != 'connection.layout') { hints.startChanged = e.command == 'connection.reconnectStart' ? true : false; hints.endChanged = e.command == 'connection.reconnectEnd' ? true : false; } if (!label) { return; } delta = getLabelAdjustment(label, newWaypoints, oldWaypoints, hints); modeling.moveShape(label, delta); }); this.postExecute([ 'connection.updateWaypoints' ], function(e) { var context = e.context, oldWaypoints = context.oldWaypoints, newWaypoints = context.newWaypoints, label = context.connection.label, hints = context.hints || {}, delta; if (!label) { return; } delta = getLabelAdjustment(label, newWaypoints, oldWaypoints, hints); modeling.moveShape(label, delta); }); this.postExecute([ 'shape.replace' ], function(e) { var context = e.context, newShape = context.newShape, oldShape = context.oldShape; var businessObject = getBusinessObject(newShape); if (businessObject && hasExternalLabel(businessObject)) { newShape.label.x = oldShape.label.x; newShape.label.y = oldShape.label.y; } }); // update di information on label creation this.executed([ 'label.create' ], function(e) { var element = e.context.shape, businessObject, di; // we want to trigger on real labels only if (!element.labelTarget) { return; } // we want to trigger on BPMN elements only if (!is(element.labelTarget || element, 'bpmn:BaseElement')) { return; } businessObject = element.businessObject, di = businessObject.di; if (!di.label) { di.label = bpmnFactory.create('bpmndi:BPMNLabel', { bounds: bpmnFactory.create('dc:Bounds') }); } assign(di.label.bounds, { x: element.x, y: element.y, width: element.width, height: element.height }); }); } inherits(LabelSupport, CommandInterceptor); LabelSupport.$inject = [ 'eventBus', 'modeling', 'bpmnFactory' ]; module.exports = LabelSupport;