Skip to content

Commit

Permalink
feat: support outline providers
Browse files Browse the repository at this point in the history
  • Loading branch information
smbea committed Oct 26, 2023
1 parent 89cf76d commit d6cd1c1
Show file tree
Hide file tree
Showing 4 changed files with 223 additions and 47 deletions.
110 changes: 78 additions & 32 deletions lib/features/outline/Outline.js
Original file line number Diff line number Diff line change
@@ -1,5 +1,3 @@
import { getBBox } from '../../util/Elements';

var LOW_PRIORITY = 500;

import {
Expand All @@ -13,9 +11,13 @@ import {
} from 'min-dom';

import {
assign
assign,
forEach,
isFunction
} from 'min-dash';

var DEFAULT_PRIORITY = 1000;

/**
* @typedef {import('../../model/Types').Element} Element
*
Expand All @@ -34,25 +36,25 @@ import {
*/
export default function Outline(eventBus, styles) {

this.offset = 6;
this._eventBus = eventBus;

this.offset = 5;

var OUTLINE_STYLE = styles.cls('djs-outline', [ 'no-fill' ]);

var self = this;

function createOutline(gfx, bounds) {
function createOutline(gfx, element) {
var outline = svgCreate('rect');

svgAttr(outline, assign({
x: 10,
y: 10,
x: - self.offset,
y: - self.offset,
rx: 4,
width: 100,
height: 100
width: element.width + self.offset * 2,
height: element.height + self.offset * 2
}, OUTLINE_STYLE));

svgAppend(gfx, outline);

return outline;
}

Expand All @@ -65,10 +67,12 @@ export default function Outline(eventBus, styles) {
var outline = domQuery('.djs-outline', gfx);

if (!outline) {
outline = createOutline(gfx, element);
}
outline = self.getOutline(element) || createOutline(gfx, element);
svgAppend(gfx, outline);

self.updateShapeOutline(outline, element);
} else {
self.updateShapeOutline(outline, element);
}
});

}
Expand All @@ -83,35 +87,77 @@ export default function Outline(eventBus, styles) {
*/
Outline.prototype.updateShapeOutline = function(outline, element) {

svgAttr(outline, {
x: -this.offset,
y: -this.offset,
width: element.width + this.offset * 2,
height: element.height + this.offset * 2
});
var updated = false;
var providers = this._getProviders();

};
if (providers.length) {
forEach(providers, function(provider) {
updated = provider.updateOutline(element, outline) || updated;

Check warning on line 95 in lib/features/outline/Outline.js

View check run for this annotation

Codecov / codecov/patch

lib/features/outline/Outline.js#L94-L95

Added lines #L94 - L95 were not covered by tests
});
}

if (!updated) {
svgAttr(outline, {
x: -this.offset,
y: -this.offset,
width: element.width + this.offset * 2,
height: element.height + this.offset * 2
});
}
};

/**
* Updates the outline of a connection respecting the bounding box of
* the connection and an outline offset.
* Register an outline provider with the given priority.
*
* @param {SVGElement} outline
* @param {Element} connection
* @param {number} priority
* @param {OutlineProvider} provider
*/
Outline.prototype.updateConnectionOutline = function(outline, connection) {
Outline.prototype.registerProvider = function(priority, provider) {
if (!provider) {
provider = priority;
priority = DEFAULT_PRIORITY;
}

var bbox = getBBox(connection);
this._eventBus.on('outline.getProviders', priority, function(event) {
event.providers.push(provider);
});
};

svgAttr(outline, {
x: bbox.x - this.offset,
y: bbox.y - this.offset,
width: bbox.width + this.offset * 2,
height: bbox.height + this.offset * 2
/**
* Returns the registered outline providers.
*
* @returns {OutlineProvider[]}
*/
Outline.prototype._getProviders = function() {
var event = this._eventBus.createEvent({
type: 'outline.getProviders',
providers: []
});

this._eventBus.fire(event);

return event.providers;
};

/**
* Returns the outline for an element.
*
* @param {Element} element
**/
Outline.prototype.getOutline = function(element) {
var outline;
var providers = this._getProviders();

forEach(providers, function(provider) {

if (!isFunction(provider.getOutline)) {
return;
}

outline = provider.getOutline(element);
});

return outline;
};

Outline.$inject = [ 'eventBus', 'styles', 'elementRegistry' ];
61 changes: 61 additions & 0 deletions lib/features/outline/OutlineProvider.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,61 @@
import { Element } from '../../model/Types';

export type Outline = SVGSVGElement | SVGCircleElement | SVGRectElement;

/**
* An interface to be implemented by an outline provider.
*/
export default interface OutlineProvider {

/**
* Returns an outline shape for the given element.
*
* @example
*
* ```javascript
* getOutline(element) {
* if(element.type === 'Foo') {
* const outline = svgCreate('circle');
*
* svgAttr(outline, {
* cx: element.width / 2,
* cy: element.height / 2,
* r: 23
* });
*
* return outline;
* };
* }
* ```
*
* @param element
*/
getOutline: (element: Element) => Outline;

/**
* Updates outline shape based on element's current size. Returns true if
* update was handled by provider.
*
* @example
*
* ```javascript
* updateOutline(element, outline) {
* if(element.type === 'Foo') {
* svgAttr(outline, {
* cx: element.width / 2,
* cy: element.height / 2,
* r: 23
* });
* } else if(element.type === 'Bar') {
* return true;
* }
*
* return false;
* }
* ```
*
* @param element
* @param outline
*/
updateOutline: (element: Element, outline: Outline) => boolean;
}
95 changes: 82 additions & 13 deletions test/spec/features/outline/OutlineSpec.js
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,8 @@ import {
} from 'min-dom';

import {
classes as svgClasses
classes as svgClasses,
create as svgCreate
} from 'tiny-svg';


Expand All @@ -22,7 +23,6 @@ describe('features/outline/Outline', function() {

expect(outline).to.exist;
expect(outline.updateShapeOutline).to.exist;
expect(outline.updateConnectionOutline).to.exist;
}));


Expand Down Expand Up @@ -51,7 +51,7 @@ describe('features/outline/Outline', function() {
}));


it('should add outline to connection', inject(function(selection, canvas, elementRegistry) {
it('should not add outline to connection', inject(function(selection, canvas, elementRegistry) {

// given
var connection = canvas.addConnection({ id: 'select1', waypoints: [ { x: 25, y: 25 }, { x: 115, y: 115 } ] });
Expand All @@ -63,7 +63,7 @@ describe('features/outline/Outline', function() {
var gfx = elementRegistry.getGraphics(connection);
var outline = domQuery('.djs-outline', gfx);

expect(outline).to.exist;
expect(outline).to.not.exist;
expect(svgClasses(gfx).has('selected')).to.be.true; // Outline class is set
}));

Expand Down Expand Up @@ -94,25 +94,94 @@ describe('features/outline/Outline', function() {
expect(outline).to.exist;
expect(svgClasses(gfx).has('selected')).to.be.false; // Outline class is not set
}));
});


describe('providers', function() {

it('should remove outline class from connection', inject(function(selection, canvas, elementRegistry) {
/**
* @constructor
*
* @param {*} [entriesOrUpdater]
*/
function Provider() {
this.getOutline = function(element) {
if (element === 'A') {
return svgCreate('circle');
} else if (element === 'B') {
return svgCreate('rect');
}
};
}

it('should register provider', inject(function(outline) {

// given
var connection = canvas.addConnection({ id: 'select3', waypoints: [ { x: 25, y: 25 }, { x: 115, y: 115 } ] });
var provider = new Provider();

// when
selection.select(connection);
selection.deselect(connection);
outline.registerProvider(provider);

// then
var gfx = elementRegistry.getGraphics(connection);
var outline = domQuery('.djs-outline', gfx);
expect(function() {
outline.registerProvider(provider);
}).not.to.throw;
}));

expect(outline).to.exist;
expect(svgClasses(gfx).has('selected')).to.be.false; // Outline class is not set

it('should get outline', inject(function(outline) {

// given
var provider = new Provider();

outline.registerProvider(provider);

// when
var outlineElement = outline.getOutline('A');

// then
expect(outlineElement).to.exist;
expect(outlineElement.tagName).to.equal('circle');
}));

});

it('missing provider API', inject(function(outline) {

// given
var provider = {};

// when
outline.registerProvider(provider);

// then
expect(outline.getOutline('FOO')).to.be.undefined;
}));


it('should set default ouline if no outline provided', inject(function(outline, canvas, selection, elementRegistry) {

// given
var provider = new Provider();
outline.registerProvider(provider);

var shape = canvas.addShape({
id: 'test',
x: 10,
y: 10,
width: 100,
height: 100
});

// when
selection.select(shape);

// then
expect(outline.getOutline(shape)).to.not.exist;

var gfx = elementRegistry.getGraphics(shape);
var outlineShape = domQuery('.djs-outline', gfx);
expect(outlineShape).to.exist;
expect(svgClasses(gfx).has('selected')).to.be.true;
}));
});
});
4 changes: 2 additions & 2 deletions test/spec/features/selection/SelectionVisualsSpec.js
Original file line number Diff line number Diff line change
Expand Up @@ -69,10 +69,10 @@ describe('features/selection/SelectionVisuals', function() {
it('should show box on select', inject(function(selection, canvas) {

// when
selection.select(connection);
selection.select(shape);

// then
var gfx = canvas.getGraphics(connection),
var gfx = canvas.getGraphics(shape),
outline = domQuery('.djs-outline', gfx);

expect(outline).to.exist;
Expand Down

0 comments on commit d6cd1c1

Please sign in to comment.