From bd4b613802d6c34e79db4f1840a264bda0766d5c Mon Sep 17 00:00:00 2001 From: Will Eastcott Date: Mon, 29 Apr 2024 01:10:00 +0100 Subject: [PATCH 1/8] Migrate SliderInput to Pointer Events --- src/components/SliderInput/index.ts | 111 ++++++---------------------- 1 file changed, 22 insertions(+), 89 deletions(-) diff --git a/src/components/SliderInput/index.ts b/src/components/SliderInput/index.ts index cbcc82ef..5b3bbe9b 100644 --- a/src/components/SliderInput/index.ts +++ b/src/components/SliderInput/index.ts @@ -72,7 +72,7 @@ class SliderInput extends Element implements IBindable, IFocusable, IPlaceholder protected _cursorHandleOffset = 0; - protected _touchId: number = null; + protected _pointerId: number = null; /** * Creates a new SliderInput. @@ -132,8 +132,7 @@ class SliderInput extends Element implements IBindable, IFocusable, IPlaceholder this._domHandle.classList.add(CLASS_SLIDER_HANDLE); this._domBar.appendChild(this._domHandle); - this._domSlider.addEventListener('mousedown', this._onMouseDown); - this._domSlider.addEventListener('touchstart', this._onTouchStart, { passive: true }); + this._domSlider.addEventListener('pointerdown', this._onPointerDown); this._domHandle.addEventListener('keydown', this._onKeyDown); if (args.value !== undefined) { @@ -153,81 +152,34 @@ class SliderInput extends Element implements IBindable, IFocusable, IPlaceholder destroy() { if (this._destroyed) return; - this._domSlider.removeEventListener('mousedown', this._onMouseDown); - this._domSlider.removeEventListener('touchstart', this._onTouchStart); - + this._domSlider.removeEventListener('pointerdown', this._onPointerDown); this._domHandle.removeEventListener('keydown', this._onKeyDown); - this.dom.removeEventListener('mouseup', this._onMouseUp); - this.dom.removeEventListener('mousemove', this._onMouseMove); - this.dom.removeEventListener('touchmove', this._onTouchMove); - this.dom.removeEventListener('touchend', this._onTouchEnd); - super.destroy(); } - protected _onMouseDown = (evt: MouseEvent) => { - if (evt.button !== 0 || !this.enabled || this.readOnly) return; + protected _onPointerDown = (evt: PointerEvent) => { + if (evt.button !== 0 || !this.enabled || this.readOnly || this._pointerId !== null) return; + this._domSlider.setPointerCapture(evt.pointerId); + this._pointerId = evt.pointerId; + window.addEventListener('pointermove', this._onPointerMove); + window.addEventListener('pointerup', this._onPointerUp); this._onSlideStart(evt.pageX); }; - protected _onMouseMove = (evt: MouseEvent) => { - evt.stopPropagation(); + protected _onPointerMove = (evt: PointerEvent) => { + if (evt.pointerId !== this._pointerId) return; evt.preventDefault(); this._onSlideMove(evt.pageX); }; - protected _onMouseUp = (evt: MouseEvent) => { - evt.stopPropagation(); - evt.preventDefault(); + protected _onPointerUp = (evt: PointerEvent) => { + if (evt.pointerId !== this._pointerId || this._pointerId === null) return; + this._domSlider.releasePointerCapture(evt.pointerId); this._onSlideEnd(evt.pageX); - }; - - protected _onTouchStart = (evt: TouchEvent) => { - if (!this.enabled || this.readOnly) return; - - for (let i = 0; i < evt.changedTouches.length; i++) { - const touch = evt.changedTouches[i]; - const node = touch.target as Node; - - if (!node.ui || node.ui !== this) - continue; - - this._touchId = touch.identifier; - this._onSlideStart(touch.pageX); - break; - } - }; - - protected _onTouchMove = (evt: TouchEvent) => { - for (let i = 0; i < evt.changedTouches.length; i++) { - const touch = evt.changedTouches[i]; - - if (touch.identifier !== this._touchId) - continue; - - evt.stopPropagation(); - evt.preventDefault(); - - this._onSlideMove(touch.pageX); - break; - } - }; - - protected _onTouchEnd = (evt: TouchEvent) => { - for (let i = 0; i < evt.changedTouches.length; i++) { - const touch = evt.changedTouches[i]; - - if (touch.identifier !== this._touchId) - continue; - - evt.stopPropagation(); - evt.preventDefault(); - - this._onSlideEnd(touch.pageX); - this._touchId = null; - break; - } + window.removeEventListener('pointermove', this._onPointerMove); + window.removeEventListener('pointerup', this._onPointerUp); + this._pointerId = null; // Reset the pointer ID }; protected _onKeyDown = (evt: KeyboardEvent) => { @@ -289,22 +241,12 @@ class SliderInput extends Element implements IBindable, IFocusable, IPlaceholder protected _onSlideStart(pageX: number) { this._domHandle.focus(); - if (this._touchId === null) { - window.addEventListener('mousemove', this._onMouseMove); - window.addEventListener('mouseup', this._onMouseUp); - } else { - window.addEventListener('touchmove', this._onTouchMove); - window.addEventListener('touchend', this._onTouchEnd); - } - this.class.add(CLASS_SLIDER_ACTIVE); - // calculate the cursor - handle offset. If there is - // an offset that means the cursor is on the handle so + // Calculate the cursor - handle offset. If there is an offset, // do not move the handle until the cursor moves. - if (!this._calculateCursorHandleOffset(pageX)) { - this._onSlideMove(pageX); - } + this._calculateCursorHandleOffset(pageX); + this._onSlideMove(pageX); if (this.binding) { this._historyCombine = this.binding.historyCombine; @@ -317,7 +259,7 @@ class SliderInput extends Element implements IBindable, IFocusable, IPlaceholder protected _onSlideMove(pageX: number) { const rect = this._domBar.getBoundingClientRect(); - // reduce pageX by the initial cursor - handle offset + // Reduce pageX by the initial cursor - handle offset pageX -= this._cursorHandleOffset; const x = Math.max(0, Math.min(1, (pageX - rect.left) / rect.width)); @@ -329,7 +271,7 @@ class SliderInput extends Element implements IBindable, IFocusable, IPlaceholder } protected _onSlideEnd(pageX: number) { - // when slide ends only move the handle if the cursor is no longer + // When slide ends only move the handle if the cursor is no longer // on the handle if (!this._calculateCursorHandleOffset(pageX)) { this._onSlideMove(pageX); @@ -337,14 +279,6 @@ class SliderInput extends Element implements IBindable, IFocusable, IPlaceholder this.class.remove(CLASS_SLIDER_ACTIVE); - if (this._touchId === null) { - window.removeEventListener('mousemove', this._onMouseMove); - window.removeEventListener('mouseup', this._onMouseUp); - } else { - window.removeEventListener('touchmove', this._onTouchMove); - window.removeEventListener('touchend', this._onTouchEnd); - } - if (this.binding) { this.binding.historyCombine = this._historyCombine; this.binding.historyPostfix = this._historyPostfix; @@ -352,7 +286,6 @@ class SliderInput extends Element implements IBindable, IFocusable, IPlaceholder this._historyCombine = false; this._historyPostfix = null; } - } focus() { From 103b1d2f2d7cb9fc7dc8d2a5658fe3c7fde1d589 Mon Sep 17 00:00:00 2001 From: Will Eastcott Date: Mon, 29 Apr 2024 01:12:32 +0100 Subject: [PATCH 2/8] Revert case change --- src/components/SliderInput/index.ts | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/components/SliderInput/index.ts b/src/components/SliderInput/index.ts index 5b3bbe9b..6988c7ab 100644 --- a/src/components/SliderInput/index.ts +++ b/src/components/SliderInput/index.ts @@ -259,7 +259,7 @@ class SliderInput extends Element implements IBindable, IFocusable, IPlaceholder protected _onSlideMove(pageX: number) { const rect = this._domBar.getBoundingClientRect(); - // Reduce pageX by the initial cursor - handle offset + // reduce pageX by the initial cursor - handle offset pageX -= this._cursorHandleOffset; const x = Math.max(0, Math.min(1, (pageX - rect.left) / rect.width)); @@ -271,7 +271,7 @@ class SliderInput extends Element implements IBindable, IFocusable, IPlaceholder } protected _onSlideEnd(pageX: number) { - // When slide ends only move the handle if the cursor is no longer + // when slide ends only move the handle if the cursor is no longer // on the handle if (!this._calculateCursorHandleOffset(pageX)) { this._onSlideMove(pageX); From b30ef806079e745c0b915fe5758d78f9c9218a22 Mon Sep 17 00:00:00 2001 From: Will Eastcott Date: Mon, 29 Apr 2024 01:15:39 +0100 Subject: [PATCH 3/8] Move handlers --- src/components/SliderInput/index.ts | 11 +++++++---- 1 file changed, 7 insertions(+), 4 deletions(-) diff --git a/src/components/SliderInput/index.ts b/src/components/SliderInput/index.ts index 6988c7ab..0e8f8b15 100644 --- a/src/components/SliderInput/index.ts +++ b/src/components/SliderInput/index.ts @@ -162,8 +162,6 @@ class SliderInput extends Element implements IBindable, IFocusable, IPlaceholder if (evt.button !== 0 || !this.enabled || this.readOnly || this._pointerId !== null) return; this._domSlider.setPointerCapture(evt.pointerId); this._pointerId = evt.pointerId; - window.addEventListener('pointermove', this._onPointerMove); - window.addEventListener('pointerup', this._onPointerUp); this._onSlideStart(evt.pageX); }; @@ -177,8 +175,6 @@ class SliderInput extends Element implements IBindable, IFocusable, IPlaceholder if (evt.pointerId !== this._pointerId || this._pointerId === null) return; this._domSlider.releasePointerCapture(evt.pointerId); this._onSlideEnd(evt.pageX); - window.removeEventListener('pointermove', this._onPointerMove); - window.removeEventListener('pointerup', this._onPointerUp); this._pointerId = null; // Reset the pointer ID }; @@ -241,6 +237,10 @@ class SliderInput extends Element implements IBindable, IFocusable, IPlaceholder protected _onSlideStart(pageX: number) { this._domHandle.focus(); + + window.addEventListener('pointermove', this._onPointerMove); + window.addEventListener('pointerup', this._onPointerUp); + this.class.add(CLASS_SLIDER_ACTIVE); // Calculate the cursor - handle offset. If there is an offset, @@ -279,6 +279,9 @@ class SliderInput extends Element implements IBindable, IFocusable, IPlaceholder this.class.remove(CLASS_SLIDER_ACTIVE); + window.removeEventListener('pointermove', this._onPointerMove); + window.removeEventListener('pointerup', this._onPointerUp); + if (this.binding) { this.binding.historyCombine = this._historyCombine; this.binding.historyPostfix = this._historyPostfix; From 0f482d2993997b27dc092dfcba46152c093c0ab8 Mon Sep 17 00:00:00 2001 From: Will Eastcott Date: Mon, 29 Apr 2024 01:16:50 +0100 Subject: [PATCH 4/8] Remove newline --- src/components/SliderInput/index.ts | 1 - 1 file changed, 1 deletion(-) diff --git a/src/components/SliderInput/index.ts b/src/components/SliderInput/index.ts index 0e8f8b15..5f1668d6 100644 --- a/src/components/SliderInput/index.ts +++ b/src/components/SliderInput/index.ts @@ -237,7 +237,6 @@ class SliderInput extends Element implements IBindable, IFocusable, IPlaceholder protected _onSlideStart(pageX: number) { this._domHandle.focus(); - window.addEventListener('pointermove', this._onPointerMove); window.addEventListener('pointerup', this._onPointerUp); From 493d639382913176a756124e147dc25ca8af5959 Mon Sep 17 00:00:00 2001 From: Will Eastcott Date: Mon, 29 Apr 2024 01:22:22 +0100 Subject: [PATCH 5/8] Revert some more changes --- src/components/SliderInput/index.ts | 8 +++++--- 1 file changed, 5 insertions(+), 3 deletions(-) diff --git a/src/components/SliderInput/index.ts b/src/components/SliderInput/index.ts index 5f1668d6..12ea3069 100644 --- a/src/components/SliderInput/index.ts +++ b/src/components/SliderInput/index.ts @@ -242,10 +242,12 @@ class SliderInput extends Element implements IBindable, IFocusable, IPlaceholder this.class.add(CLASS_SLIDER_ACTIVE); - // Calculate the cursor - handle offset. If there is an offset, + // calculate the cursor - handle offset. If there is + // an offset that means the cursor is on the handle so // do not move the handle until the cursor moves. - this._calculateCursorHandleOffset(pageX); - this._onSlideMove(pageX); + if (!this._calculateCursorHandleOffset(pageX)) { + this._onSlideMove(pageX); + } if (this.binding) { this._historyCombine = this.binding.historyCombine; From 1a3ff966e988f209746e7325b430fcc551fab80d Mon Sep 17 00:00:00 2001 From: Will Eastcott Date: Mon, 29 Apr 2024 09:16:30 +0100 Subject: [PATCH 6/8] Check pointer is mouse --- src/components/SliderInput/index.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/components/SliderInput/index.ts b/src/components/SliderInput/index.ts index 12ea3069..76a1c6cc 100644 --- a/src/components/SliderInput/index.ts +++ b/src/components/SliderInput/index.ts @@ -159,7 +159,7 @@ class SliderInput extends Element implements IBindable, IFocusable, IPlaceholder } protected _onPointerDown = (evt: PointerEvent) => { - if (evt.button !== 0 || !this.enabled || this.readOnly || this._pointerId !== null) return; + if ((evt.pointerType === 'mouse' && evt.button !== 0) || !this.enabled || this.readOnly || this._pointerId !== null) return; this._domSlider.setPointerCapture(evt.pointerId); this._pointerId = evt.pointerId; this._onSlideStart(evt.pageX); From e88f083903e1213699d517edbf8eedb446fbb311 Mon Sep 17 00:00:00 2001 From: Will Eastcott Date: Mon, 29 Apr 2024 19:18:02 +0100 Subject: [PATCH 7/8] Handle pointercancel --- src/components/SliderInput/index.ts | 20 +++++++++++++++++--- src/components/SliderInput/style.scss | 1 + 2 files changed, 18 insertions(+), 3 deletions(-) diff --git a/src/components/SliderInput/index.ts b/src/components/SliderInput/index.ts index 76a1c6cc..749061f4 100644 --- a/src/components/SliderInput/index.ts +++ b/src/components/SliderInput/index.ts @@ -160,6 +160,7 @@ class SliderInput extends Element implements IBindable, IFocusable, IPlaceholder protected _onPointerDown = (evt: PointerEvent) => { if ((evt.pointerType === 'mouse' && evt.button !== 0) || !this.enabled || this.readOnly || this._pointerId !== null) return; + evt.stopPropagation(); this._domSlider.setPointerCapture(evt.pointerId); this._pointerId = evt.pointerId; this._onSlideStart(evt.pageX); @@ -167,15 +168,26 @@ class SliderInput extends Element implements IBindable, IFocusable, IPlaceholder protected _onPointerMove = (evt: PointerEvent) => { if (evt.pointerId !== this._pointerId) return; + evt.stopPropagation(); evt.preventDefault(); this._onSlideMove(evt.pageX); }; + protected _endPointerInteraction = (evt: PointerEvent, pageX: number) => { + evt.stopPropagation(); + this._domSlider.releasePointerCapture(evt.pointerId); + this._onSlideEnd(pageX); + this._pointerId = null; + }; + protected _onPointerUp = (evt: PointerEvent) => { if (evt.pointerId !== this._pointerId || this._pointerId === null) return; - this._domSlider.releasePointerCapture(evt.pointerId); - this._onSlideEnd(evt.pageX); - this._pointerId = null; // Reset the pointer ID + this._endPointerInteraction(evt, evt.pageX); + }; + + protected _onPointerCancel = (evt: PointerEvent) => { + if (evt.pointerId !== this._pointerId || this._pointerId === null) return; + this._endPointerInteraction(evt, evt.pageX); }; protected _onKeyDown = (evt: KeyboardEvent) => { @@ -239,6 +251,7 @@ class SliderInput extends Element implements IBindable, IFocusable, IPlaceholder this._domHandle.focus(); window.addEventListener('pointermove', this._onPointerMove); window.addEventListener('pointerup', this._onPointerUp); + window.addEventListener('pointercancel', this._onPointerCancel); this.class.add(CLASS_SLIDER_ACTIVE); @@ -282,6 +295,7 @@ class SliderInput extends Element implements IBindable, IFocusable, IPlaceholder window.removeEventListener('pointermove', this._onPointerMove); window.removeEventListener('pointerup', this._onPointerUp); + window.removeEventListener('pointercancel', this._onPointerCancel); if (this.binding) { this.binding.historyCombine = this._historyCombine; diff --git a/src/components/SliderInput/style.scss b/src/components/SliderInput/style.scss index d2d1d977..b71e5f38 100644 --- a/src/components/SliderInput/style.scss +++ b/src/components/SliderInput/style.scss @@ -5,6 +5,7 @@ height: 24px; margin: $element-margin; align-items: center; + touch-action: none; // Prevents gestures and touch event interference. > .pcui-numeric-input { flex: 1; From 8dd574a9ca7c3e8a09e8745bfc17331c5a40a853 Mon Sep 17 00:00:00 2001 From: Will Eastcott Date: Mon, 29 Apr 2024 19:26:05 +0100 Subject: [PATCH 8/8] Remove pointercancel handling --- src/components/SliderInput/index.ts | 17 +++-------------- 1 file changed, 3 insertions(+), 14 deletions(-) diff --git a/src/components/SliderInput/index.ts b/src/components/SliderInput/index.ts index 749061f4..97e70155 100644 --- a/src/components/SliderInput/index.ts +++ b/src/components/SliderInput/index.ts @@ -173,23 +173,14 @@ class SliderInput extends Element implements IBindable, IFocusable, IPlaceholder this._onSlideMove(evt.pageX); }; - protected _endPointerInteraction = (evt: PointerEvent, pageX: number) => { + protected _onPointerUp = (evt: PointerEvent) => { + if (evt.pointerId !== this._pointerId || this._pointerId === null) return; evt.stopPropagation(); this._domSlider.releasePointerCapture(evt.pointerId); - this._onSlideEnd(pageX); + this._onSlideEnd(evt.pageX); this._pointerId = null; }; - protected _onPointerUp = (evt: PointerEvent) => { - if (evt.pointerId !== this._pointerId || this._pointerId === null) return; - this._endPointerInteraction(evt, evt.pageX); - }; - - protected _onPointerCancel = (evt: PointerEvent) => { - if (evt.pointerId !== this._pointerId || this._pointerId === null) return; - this._endPointerInteraction(evt, evt.pageX); - }; - protected _onKeyDown = (evt: KeyboardEvent) => { if (evt.key === 'Escape') { this.blur(); @@ -251,7 +242,6 @@ class SliderInput extends Element implements IBindable, IFocusable, IPlaceholder this._domHandle.focus(); window.addEventListener('pointermove', this._onPointerMove); window.addEventListener('pointerup', this._onPointerUp); - window.addEventListener('pointercancel', this._onPointerCancel); this.class.add(CLASS_SLIDER_ACTIVE); @@ -295,7 +285,6 @@ class SliderInput extends Element implements IBindable, IFocusable, IPlaceholder window.removeEventListener('pointermove', this._onPointerMove); window.removeEventListener('pointerup', this._onPointerUp); - window.removeEventListener('pointercancel', this._onPointerCancel); if (this.binding) { this.binding.historyCombine = this._historyCombine;