2018-04-02 21:01:53 +02:00
|
|
|
import {
|
|
|
|
assign
|
|
|
|
} from 'min-dash';
|
2017-06-27 09:37:58 +02:00
|
|
|
|
2018-04-02 21:01:53 +02:00
|
|
|
import {
|
|
|
|
getLabel
|
|
|
|
} from './LabelUtil';
|
2015-04-27 16:50:09 +02:00
|
|
|
|
2018-04-02 21:01:53 +02:00
|
|
|
import { is } from '../../util/ModelUtil';
|
|
|
|
import { isAny } from '../modeling/util/ModelingUtil';
|
|
|
|
import { isExpanded } from '../../util/DiUtil';
|
2014-06-17 11:53:07 +02:00
|
|
|
|
2018-04-30 11:06:26 +02:00
|
|
|
import {
|
|
|
|
getExternalLabelMid,
|
|
|
|
isLabelExternal,
|
|
|
|
hasExternalLabel,
|
|
|
|
isLabel
|
|
|
|
} from '../../util/LabelUtil';
|
|
|
|
|
2018-01-24 20:25:28 +01:00
|
|
|
|
2018-04-02 21:01:53 +02:00
|
|
|
export default function LabelEditingProvider(
|
2018-01-24 20:25:28 +01:00
|
|
|
eventBus, canvas, directEditing,
|
2018-05-25 13:56:12 +02:00
|
|
|
modeling, resizeHandles, textRenderer) {
|
2014-06-11 15:08:45 +02:00
|
|
|
|
2016-09-05 14:27:34 +02:00
|
|
|
this._canvas = canvas;
|
2018-01-24 20:25:28 +01:00
|
|
|
this._modeling = modeling;
|
2018-05-25 13:56:12 +02:00
|
|
|
this._textRenderer = textRenderer;
|
2016-09-05 14:27:34 +02:00
|
|
|
|
2014-06-11 15:08:45 +02:00
|
|
|
directEditing.registerProvider(this);
|
2016-09-05 14:27:34 +02:00
|
|
|
|
2014-12-17 21:54:25 +01:00
|
|
|
// listen to dblclick on non-root elements
|
|
|
|
eventBus.on('element.dblclick', function(event) {
|
2017-12-09 21:30:10 +01:00
|
|
|
activateDirectEdit(event.element, true);
|
2014-06-24 15:34:57 +02:00
|
|
|
});
|
2014-06-17 11:53:07 +02:00
|
|
|
|
2014-12-23 16:49:28 +01:00
|
|
|
// complete on followup canvas operation
|
2018-01-24 20:25:28 +01:00
|
|
|
eventBus.on([
|
|
|
|
'element.mousedown',
|
|
|
|
'drag.init',
|
2018-02-15 23:04:02 +01:00
|
|
|
'canvas.viewbox.changing',
|
2018-04-12 16:17:06 +02:00
|
|
|
'autoPlace',
|
|
|
|
'popupMenu.open'
|
2018-01-24 20:25:28 +01:00
|
|
|
], function(event) {
|
2018-04-12 16:17:06 +02:00
|
|
|
|
|
|
|
if (directEditing.isActive()) {
|
|
|
|
directEditing.complete();
|
|
|
|
}
|
2014-06-11 15:08:45 +02:00
|
|
|
});
|
|
|
|
|
2014-12-23 16:49:28 +01:00
|
|
|
// cancel on command stack changes
|
2017-06-27 09:37:58 +02:00
|
|
|
eventBus.on([ 'commandStack.changed' ], function(e) {
|
2018-04-12 16:17:06 +02:00
|
|
|
if (directEditing.isActive()) {
|
|
|
|
directEditing.cancel();
|
|
|
|
}
|
2014-06-17 11:53:07 +02:00
|
|
|
});
|
|
|
|
|
2017-12-08 19:32:13 +01:00
|
|
|
|
|
|
|
eventBus.on('directEditing.activate', function(event) {
|
|
|
|
resizeHandles.removeResizers();
|
|
|
|
});
|
|
|
|
|
|
|
|
eventBus.on('create.end', 500, function(event) {
|
|
|
|
|
|
|
|
var element = event.shape,
|
|
|
|
canExecute = event.context.canExecute,
|
2017-12-09 23:56:31 +01:00
|
|
|
isTouch = event.isTouch;
|
2017-12-08 19:32:13 +01:00
|
|
|
|
2018-01-24 20:25:28 +01:00
|
|
|
// TODO(nikku): we need to find a way to support the
|
|
|
|
// direct editing on mobile devices; right now this will
|
|
|
|
// break for desworkflowediting on mobile devices
|
2014-12-23 16:49:28 +01:00
|
|
|
// as it breaks the user interaction workflow
|
2014-09-15 14:42:43 +02:00
|
|
|
|
2018-01-24 20:25:28 +01:00
|
|
|
// TODO(nre): we should temporarily focus the edited element
|
|
|
|
// here and release the focused viewport after the direct edit
|
|
|
|
// operation is finished
|
2017-12-08 19:32:13 +01:00
|
|
|
if (isTouch) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!canExecute) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-12-09 21:30:10 +01:00
|
|
|
activateDirectEdit(element);
|
|
|
|
});
|
|
|
|
|
2017-12-08 21:06:08 +01:00
|
|
|
eventBus.on('autoPlace.end', 500, function(event) {
|
|
|
|
activateDirectEdit(event.shape);
|
|
|
|
});
|
|
|
|
|
2017-12-09 21:30:10 +01:00
|
|
|
|
|
|
|
function activateDirectEdit(element, force) {
|
|
|
|
if (force ||
|
|
|
|
isAny(element, [ 'bpmn:Task', 'bpmn:TextAnnotation' ]) ||
|
|
|
|
isCollapsedSubProcess(element)) {
|
|
|
|
|
2017-12-08 19:32:13 +01:00
|
|
|
directEditing.activate(element);
|
|
|
|
}
|
2017-12-09 21:30:10 +01:00
|
|
|
}
|
2017-12-08 19:32:13 +01:00
|
|
|
|
2014-06-11 15:08:45 +02:00
|
|
|
}
|
|
|
|
|
2018-01-24 20:25:28 +01:00
|
|
|
LabelEditingProvider.$inject = [
|
|
|
|
'eventBus',
|
|
|
|
'canvas',
|
|
|
|
'directEditing',
|
|
|
|
'modeling',
|
2018-05-25 13:56:12 +02:00
|
|
|
'resizeHandles',
|
|
|
|
'textRenderer'
|
2018-01-24 20:25:28 +01:00
|
|
|
];
|
2014-09-15 14:42:43 +02:00
|
|
|
|
2014-06-11 15:08:45 +02:00
|
|
|
|
2016-08-19 16:16:37 +02:00
|
|
|
/**
|
|
|
|
* Activate direct editing for activities and text annotations.
|
|
|
|
*
|
|
|
|
* @param {djs.model.Base} element
|
|
|
|
*
|
2017-06-27 09:37:58 +02:00
|
|
|
* @return {Object} an object with properties bounds (position and size), text and options
|
2016-08-19 16:16:37 +02:00
|
|
|
*/
|
2014-06-11 15:08:45 +02:00
|
|
|
LabelEditingProvider.prototype.activate = function(element) {
|
|
|
|
|
2017-06-27 09:37:58 +02:00
|
|
|
// text
|
2018-04-02 21:01:53 +02:00
|
|
|
var text = getLabel(element);
|
2014-06-11 15:08:45 +02:00
|
|
|
|
2014-06-17 11:53:07 +02:00
|
|
|
if (text === undefined) {
|
|
|
|
return;
|
2014-06-11 15:08:45 +02:00
|
|
|
}
|
|
|
|
|
2017-06-27 09:37:58 +02:00
|
|
|
var context = {
|
|
|
|
text: text
|
|
|
|
};
|
|
|
|
|
|
|
|
// bounds
|
|
|
|
var bounds = this.getEditingBBox(element);
|
|
|
|
|
|
|
|
assign(context, bounds);
|
|
|
|
|
|
|
|
var options = {};
|
2016-05-18 07:54:58 +02:00
|
|
|
|
2017-06-27 09:37:58 +02:00
|
|
|
// tasks
|
2018-01-24 20:25:28 +01:00
|
|
|
if (
|
|
|
|
isAny(element, [
|
|
|
|
'bpmn:Task',
|
|
|
|
'bpmn:Participant',
|
|
|
|
'bpmn:Lane',
|
|
|
|
'bpmn:CallActivity'
|
|
|
|
]) ||
|
|
|
|
isCollapsedSubProcess(element)
|
|
|
|
) {
|
2017-06-27 09:37:58 +02:00
|
|
|
assign(options, {
|
|
|
|
centerVertically: true
|
|
|
|
});
|
|
|
|
}
|
2014-06-17 11:53:07 +02:00
|
|
|
|
2017-06-27 09:37:58 +02:00
|
|
|
// external labels
|
2018-04-30 11:06:26 +02:00
|
|
|
if (isLabelExternal(element)) {
|
2017-06-27 09:37:58 +02:00
|
|
|
assign(options, {
|
|
|
|
autoResize: true
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
// text annotations
|
|
|
|
if (is(element, 'bpmn:TextAnnotation')) {
|
|
|
|
assign(options, {
|
|
|
|
resizable: true,
|
|
|
|
autoResize: true
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
assign(context, {
|
|
|
|
options: options
|
|
|
|
});
|
|
|
|
|
|
|
|
return context;
|
2016-08-19 16:16:37 +02:00
|
|
|
};
|
2014-06-11 15:08:45 +02:00
|
|
|
|
|
|
|
|
2016-08-19 16:16:37 +02:00
|
|
|
/**
|
|
|
|
* Get the editing bounding box based on the element's size and position
|
|
|
|
*
|
|
|
|
* @param {djs.model.Base} element
|
|
|
|
*
|
2018-01-24 20:25:28 +01:00
|
|
|
* @return {Object} an object containing information about position
|
|
|
|
* and size (fixed or minimum and/or maximum)
|
2016-08-19 16:16:37 +02:00
|
|
|
*/
|
|
|
|
LabelEditingProvider.prototype.getEditingBBox = function(element) {
|
2016-09-05 14:27:34 +02:00
|
|
|
var canvas = this._canvas;
|
2016-03-07 17:06:24 +01:00
|
|
|
|
2016-08-19 16:16:37 +02:00
|
|
|
var target = element.label || element;
|
2016-03-07 17:06:24 +01:00
|
|
|
|
2016-09-05 14:27:34 +02:00
|
|
|
var bbox = canvas.getAbsoluteBBox(target);
|
2016-08-19 16:16:37 +02:00
|
|
|
|
|
|
|
var mid = {
|
|
|
|
x: bbox.x + bbox.width / 2,
|
|
|
|
y: bbox.y + bbox.height / 2
|
|
|
|
};
|
2016-03-07 17:06:24 +01:00
|
|
|
|
2016-08-19 16:16:37 +02:00
|
|
|
// default position
|
|
|
|
var bounds = { x: bbox.x, y: bbox.y };
|
2014-06-17 11:53:07 +02:00
|
|
|
|
2017-06-27 09:37:58 +02:00
|
|
|
var zoom = canvas.zoom();
|
|
|
|
|
2018-05-25 13:56:12 +02:00
|
|
|
var defaultStyle = this._textRenderer.getDefaultStyle(),
|
|
|
|
externalStyle = this._textRenderer.getExternalStyle();
|
2017-06-27 09:37:58 +02:00
|
|
|
|
2018-05-25 13:56:12 +02:00
|
|
|
// take zoom into account
|
|
|
|
var externalFontSize = externalStyle.fontSize * zoom,
|
|
|
|
externalLineHeight = externalStyle.lineHeight,
|
|
|
|
defaultFontSize = defaultStyle.fontSize * zoom,
|
|
|
|
defaultLineHeight = defaultStyle.lineHeight;
|
|
|
|
|
|
|
|
var style = {
|
|
|
|
fontFamily: this._textRenderer.getDefaultStyle().fontFamily,
|
|
|
|
fontWeight: this._textRenderer.getDefaultStyle().fontWeight
|
|
|
|
};
|
2014-06-11 15:08:45 +02:00
|
|
|
|
2016-08-19 16:16:37 +02:00
|
|
|
// adjust for expanded pools AND lanes
|
2017-12-09 21:30:10 +01:00
|
|
|
if (is(element, 'bpmn:Lane') || isExpandedPool(element)) {
|
2017-06-27 09:37:58 +02:00
|
|
|
|
|
|
|
assign(bounds, {
|
|
|
|
width: bbox.height,
|
|
|
|
height: 30 * zoom,
|
|
|
|
x: bbox.x - bbox.height / 2 + (15 * zoom),
|
|
|
|
y: mid.y - (30 * zoom) / 2
|
|
|
|
});
|
2016-08-19 16:16:37 +02:00
|
|
|
|
2017-06-27 09:37:58 +02:00
|
|
|
assign(style, {
|
2018-05-25 13:56:12 +02:00
|
|
|
fontSize: defaultFontSize + 'px',
|
|
|
|
lineHeight: defaultLineHeight,
|
2017-06-27 09:37:58 +02:00
|
|
|
paddingTop: (7 * zoom) + 'px',
|
|
|
|
paddingBottom: (7 * zoom) + 'px',
|
|
|
|
paddingLeft: (5 * zoom) + 'px',
|
|
|
|
paddingRight: (5 * zoom) + 'px',
|
|
|
|
transform: 'rotate(-90deg)'
|
|
|
|
});
|
2014-06-11 15:08:45 +02:00
|
|
|
}
|
2014-06-17 11:53:07 +02:00
|
|
|
|
2014-06-11 15:08:45 +02:00
|
|
|
|
2018-01-24 20:25:28 +01:00
|
|
|
// internal labels for tasks and collapsed call activities,
|
|
|
|
// sub processes and participants
|
2017-12-09 21:30:10 +01:00
|
|
|
if (isAny(element, [ 'bpmn:Task', 'bpmn:CallActivity']) ||
|
|
|
|
isCollapsedPool(element) ||
|
|
|
|
isCollapsedSubProcess(element)) {
|
|
|
|
|
2017-06-27 09:37:58 +02:00
|
|
|
assign(bounds, {
|
|
|
|
width: bbox.width,
|
|
|
|
height: bbox.height
|
|
|
|
});
|
2014-06-11 15:08:45 +02:00
|
|
|
|
2017-06-27 09:37:58 +02:00
|
|
|
assign(style, {
|
2018-05-25 13:56:12 +02:00
|
|
|
fontSize: defaultFontSize + 'px',
|
|
|
|
lineHeight: defaultLineHeight,
|
2017-06-27 09:37:58 +02:00
|
|
|
paddingTop: (7 * zoom) + 'px',
|
|
|
|
paddingBottom: (7 * zoom) + 'px',
|
|
|
|
paddingLeft: (5 * zoom) + 'px',
|
|
|
|
paddingRight: (5 * zoom) + 'px'
|
|
|
|
});
|
2016-08-19 16:16:37 +02:00
|
|
|
}
|
2014-06-11 15:08:45 +02:00
|
|
|
|
2016-05-18 07:54:58 +02:00
|
|
|
|
2016-08-19 16:16:37 +02:00
|
|
|
// internal labels for expanded sub processes
|
2017-12-09 21:30:10 +01:00
|
|
|
if (isExpandedSubProcess(element)) {
|
2017-06-27 09:37:58 +02:00
|
|
|
assign(bounds, {
|
|
|
|
width: bbox.width,
|
|
|
|
x: bbox.x
|
|
|
|
});
|
2014-06-11 15:08:45 +02:00
|
|
|
|
2017-06-27 09:37:58 +02:00
|
|
|
assign(style, {
|
2018-05-25 13:56:12 +02:00
|
|
|
fontSize: defaultFontSize + 'px',
|
|
|
|
lineHeight: defaultLineHeight,
|
2017-06-27 09:37:58 +02:00
|
|
|
paddingTop: (7 * zoom) + 'px',
|
|
|
|
paddingBottom: (7 * zoom) + 'px',
|
|
|
|
paddingLeft: (5 * zoom) + 'px',
|
|
|
|
paddingRight: (5 * zoom) + 'px'
|
|
|
|
});
|
2016-08-19 16:16:37 +02:00
|
|
|
}
|
2014-06-11 15:08:45 +02:00
|
|
|
|
2018-04-30 11:06:26 +02:00
|
|
|
var width = 90 * zoom,
|
|
|
|
paddingTop = 7 * zoom,
|
|
|
|
paddingBottom = 4 * zoom;
|
2016-08-19 16:16:37 +02:00
|
|
|
|
|
|
|
// external labels for events, data elements, gateways and connections
|
2014-09-09 15:21:21 +02:00
|
|
|
if (target.labelTarget) {
|
2017-06-27 09:37:58 +02:00
|
|
|
assign(bounds, {
|
|
|
|
width: width,
|
|
|
|
height: bbox.height + paddingTop + paddingBottom,
|
|
|
|
x: mid.x - width / 2,
|
|
|
|
y: bbox.y - paddingTop
|
|
|
|
});
|
2014-06-11 15:08:45 +02:00
|
|
|
|
2017-06-27 09:37:58 +02:00
|
|
|
assign(style, {
|
2018-05-25 13:56:12 +02:00
|
|
|
fontSize: externalFontSize + 'px',
|
|
|
|
lineHeight: externalLineHeight,
|
2017-06-27 09:37:58 +02:00
|
|
|
paddingTop: paddingTop + 'px',
|
|
|
|
paddingBottom: paddingBottom + 'px'
|
|
|
|
});
|
2014-09-09 15:21:21 +02:00
|
|
|
}
|
2014-06-11 15:08:45 +02:00
|
|
|
|
2018-04-30 11:06:26 +02:00
|
|
|
// external label not yet created
|
|
|
|
if (isLabelExternal(target)
|
|
|
|
&& !hasExternalLabel(target)
|
|
|
|
&& !isLabel(target)) {
|
|
|
|
|
|
|
|
var externalLabelMid = getExternalLabelMid(element);
|
|
|
|
|
|
|
|
var absoluteBBox = canvas.getAbsoluteBBox({
|
|
|
|
x: externalLabelMid.x,
|
|
|
|
y: externalLabelMid.y,
|
|
|
|
width: 0,
|
|
|
|
height: 0
|
|
|
|
});
|
|
|
|
|
2018-05-25 13:56:12 +02:00
|
|
|
var height = externalFontSize + paddingTop + paddingBottom;
|
2018-04-30 11:06:26 +02:00
|
|
|
|
|
|
|
assign(bounds, {
|
|
|
|
width: width,
|
|
|
|
height: height,
|
|
|
|
x: absoluteBBox.x - width / 2,
|
|
|
|
y: absoluteBBox.y - height / 2
|
|
|
|
});
|
|
|
|
|
|
|
|
assign(style, {
|
2018-05-25 13:56:12 +02:00
|
|
|
fontSize: externalFontSize + 'px',
|
|
|
|
lineHeight: externalLineHeight,
|
2018-04-30 11:06:26 +02:00
|
|
|
paddingTop: paddingTop + 'px',
|
|
|
|
paddingBottom: paddingBottom + 'px'
|
|
|
|
});
|
|
|
|
}
|
2014-06-17 11:53:07 +02:00
|
|
|
|
2016-08-19 16:16:37 +02:00
|
|
|
// text annotations
|
|
|
|
if (is(element, 'bpmn:TextAnnotation')) {
|
2017-06-27 09:37:58 +02:00
|
|
|
assign(bounds, {
|
|
|
|
width: bbox.width,
|
|
|
|
height: bbox.height,
|
|
|
|
minWidth: 30 * zoom,
|
|
|
|
minHeight: 10 * zoom
|
|
|
|
});
|
2016-08-19 16:16:37 +02:00
|
|
|
|
2017-06-27 09:37:58 +02:00
|
|
|
assign(style, {
|
|
|
|
textAlign: 'left',
|
2018-06-13 09:01:16 +02:00
|
|
|
paddingTop: (5 * zoom) + 'px',
|
2017-06-27 09:37:58 +02:00
|
|
|
paddingBottom: (7 * zoom) + 'px',
|
2018-06-13 09:01:16 +02:00
|
|
|
paddingLeft: (7 * zoom) + 'px',
|
2017-06-27 09:37:58 +02:00
|
|
|
paddingRight: (5 * zoom) + 'px',
|
2018-05-25 13:56:12 +02:00
|
|
|
fontSize: defaultFontSize + 'px',
|
|
|
|
lineHeight: defaultLineHeight
|
2017-06-27 09:37:58 +02:00
|
|
|
});
|
2016-08-19 16:16:37 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return { bounds: bounds, style: style };
|
2014-06-11 15:08:45 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2018-01-24 20:25:28 +01:00
|
|
|
LabelEditingProvider.prototype.update = function(
|
|
|
|
element, newLabel,
|
|
|
|
activeContextText, bounds) {
|
|
|
|
|
|
|
|
var newBounds,
|
|
|
|
bbox;
|
|
|
|
|
|
|
|
if (is(element, 'bpmn:TextAnnotation')) {
|
|
|
|
|
|
|
|
bbox = this._canvas.getAbsoluteBBox(element);
|
2017-06-27 09:37:58 +02:00
|
|
|
|
2018-01-24 20:25:28 +01:00
|
|
|
newBounds = {
|
2017-06-27 09:37:58 +02:00
|
|
|
x: element.x,
|
|
|
|
y: element.y,
|
2018-01-24 20:25:28 +01:00
|
|
|
width: element.width / bbox.width * bounds.width,
|
|
|
|
height: element.height / bbox.height * bounds.height
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2018-06-08 16:13:18 +02:00
|
|
|
if (isEmptyText(newLabel)) {
|
|
|
|
newLabel = null;
|
|
|
|
}
|
|
|
|
|
2018-01-24 20:25:28 +01:00
|
|
|
this._modeling.updateLabel(element, newLabel, newBounds);
|
2016-03-07 17:06:24 +01:00
|
|
|
};
|
2017-12-09 21:30:10 +01:00
|
|
|
|
|
|
|
|
|
|
|
|
2018-02-27 09:57:22 +01:00
|
|
|
// helpers //////////////////////
|
2017-12-09 21:30:10 +01:00
|
|
|
|
|
|
|
function isCollapsedSubProcess(element) {
|
|
|
|
return is(element, 'bpmn:SubProcess') && !isExpanded(element);
|
|
|
|
}
|
|
|
|
|
|
|
|
function isExpandedSubProcess(element) {
|
|
|
|
return is(element, 'bpmn:SubProcess') && isExpanded(element);
|
|
|
|
}
|
|
|
|
|
|
|
|
function isCollapsedPool(element) {
|
|
|
|
return is(element, 'bpmn:Participant') && !isExpanded(element);
|
|
|
|
}
|
|
|
|
|
|
|
|
function isExpandedPool(element) {
|
|
|
|
return is(element, 'bpmn:Participant') && isExpanded(element);
|
2018-06-08 16:13:18 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
function isEmptyText(label) {
|
|
|
|
return !label || !label.trim();
|
2017-12-09 21:30:10 +01:00
|
|
|
}
|