From af76a2e04ddc6d4aa9717dac069e6311ed0d72a5 Mon Sep 17 00:00:00 2001 From: Philipp Date: Mon, 6 Nov 2023 16:18:48 +0100 Subject: [PATCH] fix(context-pad): clear hover timeout on close Related to bpmn-io/bpmn-js#2004 --- lib/features/context-pad/ContextPad.js | 12 +++++- .../context-pad/ContextPadProvider.js | 13 +++++- .../features/context-pad/ContextPadSpec.js | 41 +++++++++++++++++++ 3 files changed, 63 insertions(+), 3 deletions(-) diff --git a/lib/features/context-pad/ContextPad.js b/lib/features/context-pad/ContextPad.js index 18ee112eb..b71a1be11 100644 --- a/lib/features/context-pad/ContextPad.js +++ b/lib/features/context-pad/ContextPad.js @@ -235,7 +235,11 @@ ContextPad.prototype.trigger = function(action, event, autoActivate) { this._mouseout = this.triggerEntry(entry, 'hover', originalEvent, autoActivate); }, HOVER_DELAY); } else if (action === 'mouseout') { - clearTimeout(this._timeout); + if (this._timeout) { + clearTimeout(this._timeout); + + this._timeout = null; + } if (this._mouseout) { this._mouseout(); @@ -442,6 +446,12 @@ ContextPad.prototype.close = function() { return; } + if (this._timeout) { + clearTimeout(this._timeout); + + this._timeout = null; + } + this._overlays.remove(this._overlayId); this._overlayId = null; diff --git a/test/spec/features/context-pad/ContextPadProvider.js b/test/spec/features/context-pad/ContextPadProvider.js index 1a407f9f7..88a0c2f91 100755 --- a/test/spec/features/context-pad/ContextPadProvider.js +++ b/test/spec/features/context-pad/ContextPadProvider.js @@ -5,13 +5,17 @@ import imageC from './resources/c.png'; import { every } from 'min-dash'; -export default function ContextPadProvider(contextPad) { +export default function ContextPadProvider(contextPad, elementRegistry) { + this._contextPad = contextPad; + this._elementRegistry = elementRegistry; + contextPad.registerProvider(this); } -ContextPadProvider.$inject = [ 'contextPad' ]; +ContextPadProvider.$inject = [ 'contextPad', 'elementRegistry' ]; ContextPadProvider.prototype.getContextPadEntries = function(element) { + var self = this; if (element.type === 'A') { return { @@ -47,6 +51,11 @@ ContextPadProvider.prototype.getContextPadEntries = function(element) { 'action.hover': { className: 'hover', action: { + click: function(e) { + self._contextPad.open(self._elementRegistry.get('s2')); + + return 'action.click'; + }, hover: function(e) { e.__handled = true; diff --git a/test/spec/features/context-pad/ContextPadSpec.js b/test/spec/features/context-pad/ContextPadSpec.js index 5ce23f9c6..7c27c9edb 100755 --- a/test/spec/features/context-pad/ContextPadSpec.js +++ b/test/spec/features/context-pad/ContextPadSpec.js @@ -838,6 +838,47 @@ describe('features/context-pad', function() { })); + it('should not handle hover event', inject(function(canvas, contextPad) { + + // given + var shape = canvas.addShape({ + id: 's1', + width: 100, height: 100, + x: 10, y: 10, + type: 'hover' + }); + + canvas.addShape({ + id: 's2', + width: 100, height: 100, + x: 10, y: 10, + type: 'hover' + }); + + contextPad.open(shape); + + var pad = contextPad.getPad(shape), + html = pad.html, + target = domQuery('[data-action="action.hover"]', html); + + var event = globalEvent(target, { x: 0, y: 0 }); + + // when + contextPad.trigger('mouseover', event); + + expect(event.__handled).not.to.exist; + + clock.tick(250); + + contextPad.trigger('click', globalEvent(target, { x: 0, y: 0 })); + + clock.tick(500); + + // then + expect(event.__handled).not.to.exist; + })); + + it('should prevent unhandled events', inject(function(canvas, contextPad) { // given