Skip to content

Commit 5db4275

Browse files
bengfarrellWestbrook
authored andcommitted
fix(number-field): add support for scrubbing (#1535)
* feat(number-field): added features required for internal Adobe project - scrubbable (drag left/right for increment decrementing) - allow per pixel amount for scrubbing (defaults to using existing step) - shift modifier step multiplier for fast keyboard increment/decrement * feat(number-field): add shift modifier to multiply key input * feat(number-field): put stepper buttons in shift modifier path * fix(number-field): fix issue with PR where there are runtime errors when buttons are hidden * fix(number-field): add property to indicate that the user is scrubbing * fix(number-field): delay scrub input by 250ms in case user clicked/dragged a bit meaning to focus * fix(number-field): issue with scrubbing off component lights up focus ring without having focus * fix(number-field): issues with intermittent inner focus-visible and misfiring on negative distance
1 parent 39706da commit 5db4275

File tree

3 files changed

+222
-13
lines changed

3 files changed

+222
-13
lines changed

packages/number-field/src/NumberField.ts

+117-5
Original file line numberDiff line numberDiff line change
@@ -106,6 +106,9 @@ export class NumberField extends TextfieldBase {
106106

107107
_forcedUnit = '';
108108

109+
@property({ type: Boolean, reflect: true })
110+
public scrubbing = false;
111+
109112
/**
110113
* An `<sp-number-field>` element will process its numeric value with
111114
* `new Intl.NumberFormat(this.resolvedLanguage, this.formatOptions).format(this.valueAsNumber)`
@@ -149,6 +152,9 @@ export class NumberField extends TextfieldBase {
149152
@property({ type: Number, reflect: true, attribute: 'step-modifier' })
150153
public stepModifier = 10;
151154

155+
@property({ type: Number })
156+
public stepperpixel?: number;
157+
152158
@property({ type: Number })
153159
public override set value(rawValue: number) {
154160
const value = this.validateInput(rawValue);
@@ -242,13 +248,31 @@ export class NumberField extends TextfieldBase {
242248
private change!: (event: PointerEvent) => void;
243249
private safty!: number;
244250
private languageResolver = new LanguageResolutionController(this);
251+
private pointerDragXLocation?: number;
252+
private pointerDownTime?: number;
253+
private scrubDistance = 0;
245254

246255
private handlePointerdown(event: PointerEvent): void {
247256
if (event.button !== 0) {
248257
event.preventDefault();
249258
return;
250259
}
251260
this.managedInput = true;
261+
this.shiftPressed = event.shiftKey;
262+
263+
if (!this.focused) {
264+
this.setPointerCapture(event.pointerId);
265+
this.scrub(event);
266+
}
267+
}
268+
269+
private handleButtonPointerdown(event: PointerEvent): void {
270+
if (event.button !== 0) {
271+
event.preventDefault();
272+
return;
273+
}
274+
this.managedInput = true;
275+
this.shiftPressed = event.shiftKey;
252276
this.buttons.setPointerCapture(event.pointerId);
253277
const stepUpRect = this.buttons.children[0].getBoundingClientRect();
254278
const stepDownRect = this.buttons.children[1].getBoundingClientRect();
@@ -287,11 +311,11 @@ export class NumberField extends TextfieldBase {
287311
this.change(event);
288312
}
289313

290-
private handlePointermove(event: PointerEvent): void {
314+
private handleButtonPointermove(event: PointerEvent): void {
291315
this.findChange(event);
292316
}
293317

294-
private handlePointerup(event: PointerEvent): void {
318+
private handleButtonPointerup(event: PointerEvent): void {
295319
this.buttons.releasePointerCapture(event.pointerId);
296320
cancelAnimationFrame(this.nextChange);
297321
clearTimeout(this.safty);
@@ -340,8 +364,84 @@ export class NumberField extends TextfieldBase {
340364
this.stepBy(-1 * factor);
341365
}
342366

367+
private handlePointermove(event: PointerEvent): void {
368+
this.scrub(event);
369+
}
370+
371+
private handlePointerup(event: PointerEvent): void {
372+
this.releasePointerCapture(event.pointerId);
373+
this.scrub(event);
374+
cancelAnimationFrame(this.nextChange);
375+
clearTimeout(this.safty);
376+
this.managedInput = false;
377+
this.setValue();
378+
}
379+
380+
private scrub(event: PointerEvent): void {
381+
switch (event.type) {
382+
case 'pointerdown':
383+
this.scrubbing = true;
384+
this.pointerDragXLocation = event.clientX;
385+
this.pointerDownTime = Date.now();
386+
this.inputElement.disabled = true;
387+
this.addEventListener('pointermove', this.handlePointermove);
388+
this.addEventListener('pointerup', this.handlePointerup);
389+
this.addEventListener('pointercancel', this.handlePointerup);
390+
event.preventDefault();
391+
break;
392+
393+
case 'pointermove':
394+
if (
395+
this.pointerDragXLocation &&
396+
this.pointerDownTime &&
397+
Date.now() - this.pointerDownTime > 250
398+
) {
399+
const amtPerPixel = this.stepperpixel || this._step;
400+
const dist: number =
401+
event.clientX - this.pointerDragXLocation;
402+
const delta =
403+
Math.round(dist * amtPerPixel) *
404+
(event.shiftKey ? this.stepModifier : 1);
405+
this.scrubDistance += Math.abs(dist);
406+
this.pointerDragXLocation = event.clientX;
407+
this.stepBy(delta);
408+
event.preventDefault();
409+
}
410+
break;
411+
412+
default:
413+
this.pointerDragXLocation = undefined;
414+
this.scrubbing = false;
415+
this.inputElement.disabled = false;
416+
this.removeEventListener('pointermove', this.handlePointermove);
417+
this.removeEventListener('pointerup', this.handlePointerup);
418+
this.removeEventListener('pointercancel', this.handlePointerup);
419+
420+
// if user has scrubbed, disallow focus of field
421+
const bounds = this.getBoundingClientRect();
422+
if (
423+
this.scrubDistance > 0 &&
424+
this.pointerDownTime &&
425+
Date.now() - this.pointerDownTime > 250
426+
) {
427+
event.preventDefault();
428+
} else if (
429+
event.clientX >= bounds.x &&
430+
event.clientX <= bounds.x + bounds.width &&
431+
event.clientY >= bounds.y &&
432+
event.clientY <= bounds.y + bounds.height
433+
) {
434+
this.focus();
435+
}
436+
this.scrubDistance = 0;
437+
this.pointerDownTime = undefined;
438+
break;
439+
}
440+
}
441+
343442
private handleKeydown(event: KeyboardEvent): void {
344443
if (this.isComposing) return;
444+
this.shiftPressed = event.shiftKey;
345445
switch (event.code) {
346446
case 'ArrowUp':
347447
event.preventDefault();
@@ -356,6 +456,10 @@ export class NumberField extends TextfieldBase {
356456
}
357457
}
358458

459+
private handleKeyup(event: KeyboardEvent): void {
460+
this.shiftPressed = event.shiftKey;
461+
}
462+
359463
private queuedChangeEvent!: number;
360464

361465
protected onScroll(event: WheelEvent): void {
@@ -375,6 +479,9 @@ export class NumberField extends TextfieldBase {
375479
}
376480

377481
protected override onFocus(): void {
482+
if (this.pointerDragXLocation) {
483+
return;
484+
}
378485
super.onFocus();
379486
this._trackingValue = this.inputValue;
380487
this.keyboardFocused = !this.readonly && true;
@@ -639,7 +746,10 @@ export class NumberField extends TextfieldBase {
639746
@focusin=${this.handleFocusin}
640747
@focusout=${this.handleFocusout}
641748
${streamingListener({
642-
start: ['pointerdown', this.handlePointerdown],
749+
start: [
750+
'pointerdown',
751+
this.handleButtonPointerdown,
752+
],
643753
streamInside: [
644754
[
645755
'pointermove',
@@ -648,15 +758,15 @@ export class NumberField extends TextfieldBase {
648758
'pointerover',
649759
'pointerout',
650760
],
651-
this.handlePointermove,
761+
this.handleButtonPointermove,
652762
],
653763
end: [
654764
[
655765
'pointerup',
656766
'pointercancel',
657767
'pointerleave',
658768
],
659-
this.handlePointerup,
769+
this.handleButtonPointerup,
660770
],
661771
})}
662772
>
@@ -729,6 +839,8 @@ export class NumberField extends TextfieldBase {
729839
this.addEventListener('keydown', this.handleKeydown);
730840
this.addEventListener('compositionstart', this.handleCompositionStart);
731841
this.addEventListener('compositionend', this.handleCompositionEnd);
842+
this.addEventListener('keydown', this.handleKeyup);
843+
this.addEventListener('pointerdown', this.handlePointerdown);
732844
}
733845

734846
protected override updated(changes: PropertyValues<this>): void {

packages/number-field/src/number-field.css

+93
Original file line numberDiff line numberDiff line change
@@ -58,6 +58,99 @@ governing permissions and limitations under the License.
5858
visibility: hidden;
5959
}
6060

61+
:host(:not([focused], [disabled])) {
62+
cursor: ew-resize;
63+
}
64+
65+
:host(:not([focused], [disabled])) input {
66+
cursor: ew-resize;
67+
}
68+
69+
:host([hide-stepper]:not([quiet])) .input {
70+
border-radius: var(
71+
--spectrum-alias-border-radius-regular,
72+
var(--spectrum-global-dimension-size-50)
73+
);
74+
}
75+
76+
:host([dir='ltr'][invalid]:not([hide-stepper])) .icon {
77+
/* [dir=ltr] .spectrum-Textfield.is-invalid .spectrum-Textfield-validationIcon */
78+
right: calc(
79+
var(--spectrum-stepper-button-width) +
80+
var(--spectrum-textfield-error-icon-margin-left)
81+
);
82+
}
83+
84+
:host([dir='rtl'][invalid]:not([hide-stepper])) .icon {
85+
/* [dir=rtl] .spectrum-Textfield.is-invalid .spectrum-Textfield-validationIcon */
86+
left: calc(
87+
var(--spectrum-stepper-button-width) +
88+
var(--spectrum-textfield-error-icon-margin-left)
89+
);
90+
}
91+
92+
:host([dir='ltr'][valid]:not([hide-stepper])) .icon {
93+
/* [dir=ltr] .spectrum-Textfield.is-valid .spectrum-Textfield-validationIcon */
94+
right: calc(
95+
var(--spectrum-stepper-button-width) +
96+
var(--spectrum-textfield-error-icon-margin-left)
97+
);
98+
}
99+
100+
:host([dir='rtl'][valid]:not([hide-stepper])) .icon {
101+
/* [dir=rtl] .spectrum-Textfield.is-valid .spectrum-Textfield-validationIcon */
102+
left: calc(
103+
var(--spectrum-stepper-button-width) +
104+
var(--spectrum-textfield-error-icon-margin-left)
105+
);
106+
}
107+
108+
:host([dir='ltr'][quiet][invalid]:not([hide-stepper])) .icon {
109+
/* [dir=ltr] .spectrum-Textfield--quiet.spectrum-Textfield.is-invalid .spectrum-Textfield-validationIcon */
110+
right: var(--spectrum-stepper-button-width);
111+
}
112+
113+
:host([dir='rtl'][quiet][invalid]:not([hide-stepper])) .icon {
114+
/* [dir=rtl] .spectrum-Textfield--quiet.spectrum-Textfield.is-invalid .spectrum-Textfield-validationIcon */
115+
left: var(--spectrum-stepper-button-width);
116+
}
117+
118+
:host([dir='ltr'][quiet][valid]:not([hide-stepper])) .icon {
119+
/* [dir=ltr] .spectrum-Textfield--quiet.spectrum-Textfield.is-valid .spectrum-Textfield-validationIcon */
120+
right: var(--spectrum-stepper-button-width);
121+
}
122+
123+
:host([dir='rtl'][quiet][valid]:not([hide-stepper])) .icon {
124+
/* [dir=rtl] .spectrum-Textfield--quiet.spectrum-Textfield.is-valid .spectrum-Textfield-validationIcon */
125+
left: var(--spectrum-stepper-button-width);
126+
}
127+
128+
:host([dir='ltr']:not([hide-stepper])) .icon-workflow {
129+
/* [dir=ltr] .spectrum-Textfield-icon */
130+
left: calc(
131+
var(--spectrum-stepper-button-width) +
132+
var(--spectrum-textfield-error-icon-margin-left)
133+
);
134+
}
135+
136+
:host([dir='rtl']:not([hide-stepper])) .icon-workflow {
137+
/* [dir=rtl] .spectrum-Textfield-icon */
138+
right: calc(
139+
var(--spectrum-stepper-button-width) +
140+
var(--spectrum-textfield-error-icon-margin-left)
141+
);
142+
}
143+
144+
:host([dir='ltr'][quiet]:not([hide-stepper])) .icon-workflow {
145+
/* [dir=ltr] .spectrum-Textfield--quiet .spectrum-Textfield-icon */
146+
left: var(--spectrum-stepper-button-width);
147+
}
148+
149+
:host([dir='rtl'][quiet]:not([hide-stepper])) .icon-workflow {
150+
/* [dir=rtl] .spectrum-Textfield--quiet .spectrum-Textfield-icon */
151+
right: var(--spectrum-stepper-button-width);
152+
}
153+
61154
:host([readonly]:not([disabled], [invalid], [focused], [keyboard-focused]))
62155
#textfield:hover
63156
.input {

packages/number-field/test/number-field.test.ts

+12-8
Original file line numberDiff line numberDiff line change
@@ -557,6 +557,12 @@ describe('NumberField', () => {
557557
clientX: stepUpRect.x + 1,
558558
clientY: stepUpRect.y + 1,
559559
};
560+
el.setPointerCapture = () => {
561+
return;
562+
};
563+
el.releasePointerCapture = () => {
564+
return;
565+
};
560566
(
561567
el as unknown as {
562568
buttons: HTMLDivElement;
@@ -656,7 +662,7 @@ describe('NumberField', () => {
656662
expect(inputSpy.callCount).to.equal(5);
657663
expect(changeSpy.callCount).to.equal(1);
658664
});
659-
it('no change in committed value - using buttons', async () => {
665+
it('no change in committed value - using buttons', async function () {
660666
const buttonUp = el.shadowRoot.querySelector(
661667
'.step-up'
662668
) as HTMLElement;
@@ -692,19 +698,16 @@ describe('NumberField', () => {
692698
expect(el.value).to.equal(52);
693699
expect(inputSpy.callCount).to.equal(2);
694700
expect(changeSpy.callCount).to.equal(0);
695-
sendMouse({
701+
await sendMouse({
696702
steps: [
697703
{
698704
type: 'move',
699705
position: buttonDownPosition,
700706
},
701707
],
702708
});
703-
let framesToWait = FRAMES_PER_CHANGE * 2;
704-
while (framesToWait) {
705-
// input is only processed onces per FRAMES_PER_CHANGE number of frames
706-
framesToWait -= 1;
707-
await nextFrame();
709+
while (el.value > 50) {
710+
await oneEvent(el, 'input');
708711
}
709712
expect(inputSpy.callCount).to.equal(4);
710713
expect(changeSpy.callCount).to.equal(0);
@@ -715,14 +718,15 @@ describe('NumberField', () => {
715718
},
716719
],
717720
});
721+
expect(el.value).to.equal(50);
718722
expect(inputSpy.callCount).to.equal(4);
719723
expect(
720724
changeSpy.callCount,
721725
'value does not change from initial value so no "change" event is dispatched'
722726
).to.equal(0);
723727
});
724728
});
725-
it('accepts pointer interactions with the stepper UI', async () => {
729+
it('accepts pointer interactions with the stepper UI', async function () {
726730
const inputSpy = spy();
727731
const el = await getElFrom(Default({ value: 50 }));
728732
el.addEventListener('input', () => inputSpy());

0 commit comments

Comments
 (0)