From e7928f47a7edc0eb48bcd441254ff6193e556a5a Mon Sep 17 00:00:00 2001 From: Oleksandr Danylchenko Date: Wed, 14 Aug 2024 17:00:42 +0300 Subject: [PATCH 1/2] Added timeout before processing the pointer up event --- .../text-annotator/src/SelectionHandler.ts | 35 +++++++++++++------ 1 file changed, 24 insertions(+), 11 deletions(-) diff --git a/packages/text-annotator/src/SelectionHandler.ts b/packages/text-annotator/src/SelectionHandler.ts index eedef19e..4fa9488c 100644 --- a/packages/text-annotator/src/SelectionHandler.ts +++ b/packages/text-annotator/src/SelectionHandler.ts @@ -76,13 +76,13 @@ export const SelectionHandler = ( const selectionRange = sel.getRangeAt(0); if (isWhitespaceOrEmpty(selectionRange)) return; - + const annotatableRanges = splitAnnotatableRanges(selectionRange.cloneRange()); const hasChanged = annotatableRanges.length !== currentTarget.selector.length || annotatableRanges.some((r, i) => r.toString() !== currentTarget.selector[i]?.quote); - + if (!hasChanged) return; currentTarget = { @@ -96,7 +96,7 @@ export const SelectionHandler = ( } else { // Proper lifecycle management: clear selection first... selection.clear(); - + // ...then add annotation to store... store.addAnnotation({ id: currentTarget.annotation, @@ -148,13 +148,26 @@ export const SelectionHandler = ( const timeDifference = evt.timeStamp - lastPointerDown.timeStamp; - // Just a click, not a selection - if (document.getSelection().isCollapsed && timeDifference < 300) { - currentTarget = undefined; - clickSelect(); - } else if (currentTarget) { - selection.userSelect(currentTarget.annotation, evt); - } + /** + * We must check the `isCollapsed` within the 0-timeout + * to handle the annotation dismissal after a click properly. + * + * Otherwise, the `isCollapsed` will return an obsolete `false` value, + * click won't be processed, and the annotation will get falsely re-selected. + * + * @see https://github.com/recogito/text-annotator-js/issues/136 + */ + setTimeout(() => { + const { isCollapsed } = document.getSelection() + + // Just a click, not a selection + if (isCollapsed && timeDifference < 300) { + currentTarget = undefined; + clickSelect(); + } else if (currentTarget) { + selection.userSelect(currentTarget.annotation, evt); + } + }); } document.addEventListener('pointerup', onPointerUp); @@ -162,7 +175,7 @@ export const SelectionHandler = ( const destroy = () => { container.removeEventListener('selectstart', onSelectStart); document.removeEventListener('selectionchange', onSelectionChange); - + container.removeEventListener('pointerdown', onPointerDown); document.removeEventListener('pointerup', onPointerUp); } From 0d894e5bd42d6bd1b86bc3dc40da019537b4d4f7 Mon Sep 17 00:00:00 2001 From: Oleksandr Danylchenko Date: Fri, 23 Aug 2024 20:51:30 +0300 Subject: [PATCH 2/2] Accounted for the missing selection --- packages/text-annotator/src/SelectionHandler.ts | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/packages/text-annotator/src/SelectionHandler.ts b/packages/text-annotator/src/SelectionHandler.ts index 4fa9488c..4244e23b 100644 --- a/packages/text-annotator/src/SelectionHandler.ts +++ b/packages/text-annotator/src/SelectionHandler.ts @@ -158,10 +158,10 @@ export const SelectionHandler = ( * @see https://github.com/recogito/text-annotator-js/issues/136 */ setTimeout(() => { - const { isCollapsed } = document.getSelection() + const sel = document.getSelection() // Just a click, not a selection - if (isCollapsed && timeDifference < 300) { + if (sel?.isCollapsed && timeDifference < 300) { currentTarget = undefined; clickSelect(); } else if (currentTarget) {