@react-aria/interactions 3.0.0-nightly-641446f65-240905

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (109) hide show
  1. package/README.md +3 -0
  2. package/dist/PressResponder.main.js +62 -0
  3. package/dist/PressResponder.main.js.map +1 -0
  4. package/dist/PressResponder.mjs +52 -0
  5. package/dist/PressResponder.module.js +52 -0
  6. package/dist/PressResponder.module.js.map +1 -0
  7. package/dist/Pressable.main.js +43 -0
  8. package/dist/Pressable.main.js.map +1 -0
  9. package/dist/Pressable.mjs +34 -0
  10. package/dist/Pressable.module.js +34 -0
  11. package/dist/Pressable.module.js.map +1 -0
  12. package/dist/context.main.js +30 -0
  13. package/dist/context.main.js.map +1 -0
  14. package/dist/context.mjs +21 -0
  15. package/dist/context.module.js +21 -0
  16. package/dist/context.module.js.map +1 -0
  17. package/dist/createEventHandler.main.js +42 -0
  18. package/dist/createEventHandler.main.js.map +1 -0
  19. package/dist/createEventHandler.mjs +37 -0
  20. package/dist/createEventHandler.module.js +37 -0
  21. package/dist/createEventHandler.module.js.map +1 -0
  22. package/dist/import.mjs +39 -0
  23. package/dist/main.js +62 -0
  24. package/dist/main.js.map +1 -0
  25. package/dist/module.js +39 -0
  26. package/dist/module.js.map +1 -0
  27. package/dist/textSelection.main.js +77 -0
  28. package/dist/textSelection.main.js.map +1 -0
  29. package/dist/textSelection.mjs +71 -0
  30. package/dist/textSelection.module.js +71 -0
  31. package/dist/textSelection.module.js.map +1 -0
  32. package/dist/types.d.ts +218 -0
  33. package/dist/types.d.ts.map +1 -0
  34. package/dist/useFocus.main.js +64 -0
  35. package/dist/useFocus.main.js.map +1 -0
  36. package/dist/useFocus.mjs +59 -0
  37. package/dist/useFocus.module.js +59 -0
  38. package/dist/useFocus.module.js.map +1 -0
  39. package/dist/useFocusVisible.main.js +248 -0
  40. package/dist/useFocusVisible.main.js.map +1 -0
  41. package/dist/useFocusVisible.mjs +237 -0
  42. package/dist/useFocusVisible.module.js +237 -0
  43. package/dist/useFocusVisible.module.js.map +1 -0
  44. package/dist/useFocusWithin.main.js +76 -0
  45. package/dist/useFocusWithin.main.js.map +1 -0
  46. package/dist/useFocusWithin.mjs +71 -0
  47. package/dist/useFocusWithin.module.js +71 -0
  48. package/dist/useFocusWithin.module.js.map +1 -0
  49. package/dist/useHover.main.js +142 -0
  50. package/dist/useHover.main.js.map +1 -0
  51. package/dist/useHover.mjs +137 -0
  52. package/dist/useHover.module.js +137 -0
  53. package/dist/useHover.module.js.map +1 -0
  54. package/dist/useInteractOutside.main.js +101 -0
  55. package/dist/useInteractOutside.main.js.map +1 -0
  56. package/dist/useInteractOutside.mjs +96 -0
  57. package/dist/useInteractOutside.module.js +96 -0
  58. package/dist/useInteractOutside.module.js.map +1 -0
  59. package/dist/useKeyboard.main.js +30 -0
  60. package/dist/useKeyboard.main.js.map +1 -0
  61. package/dist/useKeyboard.mjs +25 -0
  62. package/dist/useKeyboard.module.js +25 -0
  63. package/dist/useKeyboard.module.js.map +1 -0
  64. package/dist/useLongPress.main.js +84 -0
  65. package/dist/useLongPress.main.js.map +1 -0
  66. package/dist/useLongPress.mjs +79 -0
  67. package/dist/useLongPress.module.js +79 -0
  68. package/dist/useLongPress.module.js.map +1 -0
  69. package/dist/useMove.main.js +236 -0
  70. package/dist/useMove.main.js.map +1 -0
  71. package/dist/useMove.mjs +231 -0
  72. package/dist/useMove.module.js +231 -0
  73. package/dist/useMove.module.js.map +1 -0
  74. package/dist/usePress.main.js +612 -0
  75. package/dist/usePress.main.js.map +1 -0
  76. package/dist/usePress.mjs +607 -0
  77. package/dist/usePress.module.js +607 -0
  78. package/dist/usePress.module.js.map +1 -0
  79. package/dist/useScrollWheel.main.js +41 -0
  80. package/dist/useScrollWheel.main.js.map +1 -0
  81. package/dist/useScrollWheel.mjs +36 -0
  82. package/dist/useScrollWheel.module.js +36 -0
  83. package/dist/useScrollWheel.module.js.map +1 -0
  84. package/dist/utils.main.js +120 -0
  85. package/dist/utils.main.js.map +1 -0
  86. package/dist/utils.mjs +115 -0
  87. package/dist/utils.module.js +115 -0
  88. package/dist/utils.module.js.map +1 -0
  89. package/package.json +37 -0
  90. package/src/DOMPropsContext.ts +39 -0
  91. package/src/DOMPropsResponder.tsx +47 -0
  92. package/src/PressResponder.tsx +64 -0
  93. package/src/Pressable.tsx +31 -0
  94. package/src/context.ts +23 -0
  95. package/src/createEventHandler.ts +48 -0
  96. package/src/index.ts +44 -0
  97. package/src/textSelection.ts +99 -0
  98. package/src/useDOMPropsResponder.ts +27 -0
  99. package/src/useFocus.ts +87 -0
  100. package/src/useFocusVisible.ts +331 -0
  101. package/src/useFocusWithin.ts +103 -0
  102. package/src/useHover.ts +206 -0
  103. package/src/useInteractOutside.ts +134 -0
  104. package/src/useKeyboard.ts +36 -0
  105. package/src/useLongPress.ts +129 -0
  106. package/src/useMove.ts +231 -0
  107. package/src/usePress.ts +955 -0
  108. package/src/useScrollWheel.ts +41 -0
  109. package/src/utils.ts +130 -0
@@ -0,0 +1,955 @@
1
+ /*
2
+ * Copyright 2020 Adobe. All rights reserved.
3
+ * This file is licensed to you under the Apache License, Version 2.0 (the "License");
4
+ * you may not use this file except in compliance with the License. You may obtain a copy
5
+ * of the License at http://www.apache.org/licenses/LICENSE-2.0
6
+ *
7
+ * Unless required by applicable law or agreed to in writing, software distributed under
8
+ * the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS
9
+ * OF ANY KIND, either express or implied. See the License for the specific language
10
+ * governing permissions and limitations under the License.
11
+ */
12
+
13
+ // Portions of the code in this file are based on code from react.
14
+ // Original licensing for the following can be found in the
15
+ // NOTICE file in the root directory of this source tree.
16
+ // See https://github.com/facebook/react/tree/cc7c1aece46a6b69b41958d731e0fd27c94bfc6c/packages/react-interactions
17
+
18
+ import {chain, focusWithoutScrolling, getOwnerDocument, getOwnerWindow, isMac, isVirtualClick, isVirtualPointerEvent, mergeProps, openLink, useEffectEvent, useGlobalListeners, useSyncRef} from '@react-aria/utils';
19
+ import {disableTextSelection, restoreTextSelection} from './textSelection';
20
+ import {DOMAttributes, FocusableElement, PressEvent as IPressEvent, PointerType, PressEvents, RefObject} from '@react-types/shared';
21
+ import {PressResponderContext} from './context';
22
+ import {TouchEvent as RTouchEvent, useContext, useEffect, useMemo, useRef, useState} from 'react';
23
+
24
+ export interface PressProps extends PressEvents {
25
+ /** Whether the target is in a controlled press state (e.g. an overlay it triggers is open). */
26
+ isPressed?: boolean,
27
+ /** Whether the press events should be disabled. */
28
+ isDisabled?: boolean,
29
+ /** Whether the target should not receive focus on press. */
30
+ preventFocusOnPress?: boolean,
31
+ /**
32
+ * Whether press events should be canceled when the pointer leaves the target while pressed.
33
+ * By default, this is `false`, which means if the pointer returns back over the target while
34
+ * still pressed, onPressStart will be fired again. If set to `true`, the press is canceled
35
+ * when the pointer leaves the target and onPressStart will not be fired if the pointer returns.
36
+ */
37
+ shouldCancelOnPointerExit?: boolean,
38
+ /** Whether text selection should be enabled on the pressable element. */
39
+ allowTextSelectionOnPress?: boolean
40
+ }
41
+
42
+ export interface PressHookProps extends PressProps {
43
+ /** A ref to the target element. */
44
+ ref?: RefObject<Element | null>
45
+ }
46
+
47
+ interface PressState {
48
+ isPressed: boolean,
49
+ ignoreEmulatedMouseEvents: boolean,
50
+ ignoreClickAfterPress: boolean,
51
+ didFirePressStart: boolean,
52
+ isTriggeringEvent: boolean,
53
+ activePointerId: any,
54
+ target: FocusableElement | null,
55
+ isOverTarget: boolean,
56
+ pointerType: PointerType | null,
57
+ userSelect?: string,
58
+ metaKeyEvents?: Map<string, KeyboardEvent>
59
+ }
60
+
61
+ interface EventBase {
62
+ currentTarget: EventTarget | null,
63
+ shiftKey: boolean,
64
+ ctrlKey: boolean,
65
+ metaKey: boolean,
66
+ altKey: boolean,
67
+ clientX?: number,
68
+ clientY?: number,
69
+ targetTouches?: Array<{clientX?: number, clientY?: number}>
70
+ }
71
+
72
+ export interface PressResult {
73
+ /** Whether the target is currently pressed. */
74
+ isPressed: boolean,
75
+ /** Props to spread on the target element. */
76
+ pressProps: DOMAttributes
77
+ }
78
+
79
+ function usePressResponderContext(props: PressHookProps): PressHookProps {
80
+ // Consume context from <PressResponder> and merge with props.
81
+ let context = useContext(PressResponderContext);
82
+ if (context) {
83
+ let {register, ...contextProps} = context;
84
+ props = mergeProps(contextProps, props) as PressHookProps;
85
+ register();
86
+ }
87
+ useSyncRef(context, props.ref);
88
+
89
+ return props;
90
+ }
91
+
92
+ class PressEvent implements IPressEvent {
93
+ type: IPressEvent['type'];
94
+ pointerType: PointerType;
95
+ target: Element;
96
+ shiftKey: boolean;
97
+ ctrlKey: boolean;
98
+ metaKey: boolean;
99
+ altKey: boolean;
100
+ x: number;
101
+ y: number;
102
+ #shouldStopPropagation = true;
103
+
104
+ constructor(type: IPressEvent['type'], pointerType: PointerType, originalEvent: EventBase, state?: PressState) {
105
+ let currentTarget = state?.target ?? originalEvent.currentTarget;
106
+ const rect: DOMRect | undefined = (currentTarget as Element)?.getBoundingClientRect();
107
+ let x, y = 0;
108
+ let clientX, clientY: number | null = null;
109
+ if (originalEvent.clientX != null && originalEvent.clientY != null) {
110
+ clientX = originalEvent.clientX;
111
+ clientY = originalEvent.clientY;
112
+ }
113
+ if (rect) {
114
+ if (clientX != null && clientY != null) {
115
+ x = clientX - rect.left;
116
+ y = clientY - rect.top;
117
+ } else {
118
+ x = rect.width / 2;
119
+ y = rect.height / 2;
120
+ }
121
+ }
122
+ this.type = type;
123
+ this.pointerType = pointerType;
124
+ this.target = originalEvent.currentTarget as Element;
125
+ this.shiftKey = originalEvent.shiftKey;
126
+ this.metaKey = originalEvent.metaKey;
127
+ this.ctrlKey = originalEvent.ctrlKey;
128
+ this.altKey = originalEvent.altKey;
129
+ this.x = x;
130
+ this.y = y;
131
+ }
132
+
133
+ continuePropagation() {
134
+ this.#shouldStopPropagation = false;
135
+ }
136
+
137
+ get shouldStopPropagation() {
138
+ return this.#shouldStopPropagation;
139
+ }
140
+ }
141
+
142
+ const LINK_CLICKED = Symbol('linkClicked');
143
+
144
+ /**
145
+ * Handles press interactions across mouse, touch, keyboard, and screen readers.
146
+ * It normalizes behavior across browsers and platforms, and handles many nuances
147
+ * of dealing with pointer and keyboard events.
148
+ */
149
+ export function usePress(props: PressHookProps): PressResult {
150
+ let {
151
+ onPress,
152
+ onPressChange,
153
+ onPressStart,
154
+ onPressEnd,
155
+ onPressUp,
156
+ isDisabled,
157
+ isPressed: isPressedProp,
158
+ preventFocusOnPress,
159
+ shouldCancelOnPointerExit,
160
+ allowTextSelectionOnPress,
161
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
162
+ ref: _, // Removing `ref` from `domProps` because TypeScript is dumb
163
+ ...domProps
164
+ } = usePressResponderContext(props);
165
+
166
+ let [isPressed, setPressed] = useState(false);
167
+ let ref = useRef<PressState>({
168
+ isPressed: false,
169
+ ignoreEmulatedMouseEvents: false,
170
+ ignoreClickAfterPress: false,
171
+ didFirePressStart: false,
172
+ isTriggeringEvent: false,
173
+ activePointerId: null,
174
+ target: null,
175
+ isOverTarget: false,
176
+ pointerType: null
177
+ });
178
+
179
+ let {addGlobalListener, removeAllGlobalListeners} = useGlobalListeners();
180
+
181
+ let triggerPressStart = useEffectEvent((originalEvent: EventBase, pointerType: PointerType) => {
182
+ let state = ref.current;
183
+ if (isDisabled || state.didFirePressStart) {
184
+ return false;
185
+ }
186
+
187
+ let shouldStopPropagation = true;
188
+ state.isTriggeringEvent = true;
189
+ if (onPressStart) {
190
+ let event = new PressEvent('pressstart', pointerType, originalEvent);
191
+ onPressStart(event);
192
+ shouldStopPropagation = event.shouldStopPropagation;
193
+ }
194
+
195
+ if (onPressChange) {
196
+ onPressChange(true);
197
+ }
198
+
199
+ state.isTriggeringEvent = false;
200
+ state.didFirePressStart = true;
201
+ setPressed(true);
202
+ return shouldStopPropagation;
203
+ });
204
+
205
+ let triggerPressEnd = useEffectEvent((originalEvent: EventBase, pointerType: PointerType, wasPressed = true) => {
206
+ let state = ref.current;
207
+ if (!state.didFirePressStart) {
208
+ return false;
209
+ }
210
+
211
+ state.ignoreClickAfterPress = true;
212
+ state.didFirePressStart = false;
213
+ state.isTriggeringEvent = true;
214
+
215
+ let shouldStopPropagation = true;
216
+ if (onPressEnd) {
217
+ let event = new PressEvent('pressend', pointerType, originalEvent);
218
+ onPressEnd(event);
219
+ shouldStopPropagation = event.shouldStopPropagation;
220
+ }
221
+
222
+ if (onPressChange) {
223
+ onPressChange(false);
224
+ }
225
+
226
+ setPressed(false);
227
+
228
+ if (onPress && wasPressed && !isDisabled) {
229
+ let event = new PressEvent('press', pointerType, originalEvent);
230
+ onPress(event);
231
+ shouldStopPropagation &&= event.shouldStopPropagation;
232
+ }
233
+
234
+ state.isTriggeringEvent = false;
235
+ return shouldStopPropagation;
236
+ });
237
+
238
+ let triggerPressUp = useEffectEvent((originalEvent: EventBase, pointerType: PointerType) => {
239
+ let state = ref.current;
240
+ if (isDisabled) {
241
+ return false;
242
+ }
243
+
244
+ if (onPressUp) {
245
+ state.isTriggeringEvent = true;
246
+ let event = new PressEvent('pressup', pointerType, originalEvent);
247
+ onPressUp(event);
248
+ state.isTriggeringEvent = false;
249
+ return event.shouldStopPropagation;
250
+ }
251
+
252
+ return true;
253
+ });
254
+
255
+ let cancel = useEffectEvent((e: EventBase) => {
256
+ let state = ref.current;
257
+ if (state.isPressed && state.target) {
258
+ if (state.isOverTarget && state.pointerType != null) {
259
+ triggerPressEnd(createEvent(state.target, e), state.pointerType, false);
260
+ }
261
+ state.isPressed = false;
262
+ state.isOverTarget = false;
263
+ state.activePointerId = null;
264
+ state.pointerType = null;
265
+ removeAllGlobalListeners();
266
+ if (!allowTextSelectionOnPress) {
267
+ restoreTextSelection(state.target);
268
+ }
269
+ }
270
+ });
271
+
272
+ let cancelOnPointerExit = useEffectEvent((e: EventBase) => {
273
+ if (shouldCancelOnPointerExit) {
274
+ cancel(e);
275
+ }
276
+ });
277
+
278
+ let pressProps = useMemo(() => {
279
+ let state = ref.current;
280
+ let pressProps: DOMAttributes = {
281
+ onKeyDown(e) {
282
+ if (isValidKeyboardEvent(e.nativeEvent, e.currentTarget) && e.currentTarget.contains(e.target as Element)) {
283
+ if (shouldPreventDefaultKeyboard(e.target as Element, e.key)) {
284
+ e.preventDefault();
285
+ }
286
+
287
+ // If the event is repeating, it may have started on a different element
288
+ // after which focus moved to the current element. Ignore these events and
289
+ // only handle the first key down event.
290
+ let shouldStopPropagation = true;
291
+ if (!state.isPressed && !e.repeat) {
292
+ state.target = e.currentTarget;
293
+ state.isPressed = true;
294
+ shouldStopPropagation = triggerPressStart(e, 'keyboard');
295
+
296
+ // Focus may move before the key up event, so register the event on the document
297
+ // instead of the same element where the key down event occurred. Make it capturing so that it will trigger
298
+ // before stopPropagation from useKeyboard on a child element may happen and thus we can still call triggerPress for the parent element.
299
+ let originalTarget = e.currentTarget;
300
+ let pressUp = (e) => {
301
+ if (isValidKeyboardEvent(e, originalTarget) && !e.repeat && originalTarget.contains(e.target as Element) && state.target) {
302
+ triggerPressUp(createEvent(state.target, e), 'keyboard');
303
+ }
304
+ };
305
+
306
+ addGlobalListener(getOwnerDocument(e.currentTarget), 'keyup', chain(pressUp, onKeyUp), true);
307
+ }
308
+
309
+ if (shouldStopPropagation) {
310
+ e.stopPropagation();
311
+ }
312
+
313
+ // Keep track of the keydown events that occur while the Meta (e.g. Command) key is held.
314
+ // macOS has a bug where keyup events are not fired while the Meta key is down.
315
+ // When the Meta key itself is released we will get an event for that, and we'll act as if
316
+ // all of these other keys were released as well.
317
+ // https://bugs.chromium.org/p/chromium/issues/detail?id=1393524
318
+ // https://bugs.webkit.org/show_bug.cgi?id=55291
319
+ // https://bugzilla.mozilla.org/show_bug.cgi?id=1299553
320
+ if (e.metaKey && isMac()) {
321
+ state.metaKeyEvents?.set(e.key, e.nativeEvent);
322
+ }
323
+ } else if (e.key === 'Meta') {
324
+ state.metaKeyEvents = new Map();
325
+ }
326
+ },
327
+ onClick(e) {
328
+ if (e && !e.currentTarget.contains(e.target as Element)) {
329
+ return;
330
+ }
331
+
332
+ if (e && e.button === 0 && !state.isTriggeringEvent && !(openLink as any).isOpening) {
333
+ let shouldStopPropagation = true;
334
+ if (isDisabled) {
335
+ e.preventDefault();
336
+ }
337
+
338
+ // If triggered from a screen reader or by using element.click(),
339
+ // trigger as if it were a keyboard click.
340
+ if (!state.ignoreClickAfterPress && !state.ignoreEmulatedMouseEvents && !state.isPressed && (state.pointerType === 'virtual' || isVirtualClick(e.nativeEvent))) {
341
+ // Ensure the element receives focus (VoiceOver on iOS does not do this)
342
+ if (!isDisabled && !preventFocusOnPress) {
343
+ focusWithoutScrolling(e.currentTarget);
344
+ }
345
+
346
+ let stopPressStart = triggerPressStart(e, 'virtual');
347
+ let stopPressUp = triggerPressUp(e, 'virtual');
348
+ let stopPressEnd = triggerPressEnd(e, 'virtual');
349
+ shouldStopPropagation = stopPressStart && stopPressUp && stopPressEnd;
350
+ }
351
+
352
+ state.ignoreEmulatedMouseEvents = false;
353
+ state.ignoreClickAfterPress = false;
354
+ if (shouldStopPropagation) {
355
+ e.stopPropagation();
356
+ }
357
+ }
358
+ }
359
+ };
360
+
361
+ let onKeyUp = (e: KeyboardEvent) => {
362
+ if (state.isPressed && state.target && isValidKeyboardEvent(e, state.target)) {
363
+ if (shouldPreventDefaultKeyboard(e.target as Element, e.key)) {
364
+ e.preventDefault();
365
+ }
366
+
367
+ let target = e.target as Element;
368
+ triggerPressEnd(createEvent(state.target, e), 'keyboard', state.target.contains(target));
369
+ removeAllGlobalListeners();
370
+
371
+ // If a link was triggered with a key other than Enter, open the URL ourselves.
372
+ // This means the link has a role override, and the default browser behavior
373
+ // only applies when using the Enter key.
374
+ if (e.key !== 'Enter' && isHTMLAnchorLink(state.target) && state.target.contains(target) && !e[LINK_CLICKED]) {
375
+ // Store a hidden property on the event so we only trigger link click once,
376
+ // even if there are multiple usePress instances attached to the element.
377
+ e[LINK_CLICKED] = true;
378
+ openLink(state.target, e, false);
379
+ }
380
+
381
+ state.isPressed = false;
382
+ state.metaKeyEvents?.delete(e.key);
383
+ } else if (e.key === 'Meta' && state.metaKeyEvents?.size) {
384
+ // If we recorded keydown events that occurred while the Meta key was pressed,
385
+ // and those haven't received keyup events already, fire keyup events ourselves.
386
+ // See comment above for more info about the macOS bug causing this.
387
+ let events = state.metaKeyEvents;
388
+ state.metaKeyEvents = undefined;
389
+ for (let event of events.values()) {
390
+ state.target?.dispatchEvent(new KeyboardEvent('keyup', event));
391
+ }
392
+ }
393
+ };
394
+
395
+ if (typeof PointerEvent !== 'undefined') {
396
+ pressProps.onPointerDown = (e) => {
397
+ // Only handle left clicks, and ignore events that bubbled through portals.
398
+ if (e.button !== 0 || !e.currentTarget.contains(e.target as Element)) {
399
+ return;
400
+ }
401
+
402
+ // iOS safari fires pointer events from VoiceOver with incorrect coordinates/target.
403
+ // Ignore and let the onClick handler take care of it instead.
404
+ // https://bugs.webkit.org/show_bug.cgi?id=222627
405
+ // https://bugs.webkit.org/show_bug.cgi?id=223202
406
+ if (isVirtualPointerEvent(e.nativeEvent)) {
407
+ state.pointerType = 'virtual';
408
+ return;
409
+ }
410
+
411
+ // Due to browser inconsistencies, especially on mobile browsers, we prevent
412
+ // default on pointer down and handle focusing the pressable element ourselves.
413
+ if (shouldPreventDefault(e.currentTarget as Element)) {
414
+ e.preventDefault();
415
+ }
416
+
417
+ state.pointerType = e.pointerType;
418
+
419
+ let shouldStopPropagation = true;
420
+ if (!state.isPressed) {
421
+ state.isPressed = true;
422
+ state.isOverTarget = true;
423
+ state.activePointerId = e.pointerId;
424
+ state.target = e.currentTarget;
425
+
426
+ if (!isDisabled && !preventFocusOnPress) {
427
+ focusWithoutScrolling(e.currentTarget);
428
+ }
429
+
430
+ if (!allowTextSelectionOnPress) {
431
+ disableTextSelection(state.target);
432
+ }
433
+
434
+ shouldStopPropagation = triggerPressStart(e, state.pointerType);
435
+
436
+ addGlobalListener(getOwnerDocument(e.currentTarget), 'pointermove', onPointerMove, false);
437
+ addGlobalListener(getOwnerDocument(e.currentTarget), 'pointerup', onPointerUp, false);
438
+ addGlobalListener(getOwnerDocument(e.currentTarget), 'pointercancel', onPointerCancel, false);
439
+ }
440
+
441
+ if (shouldStopPropagation) {
442
+ e.stopPropagation();
443
+ }
444
+ };
445
+
446
+ pressProps.onMouseDown = (e) => {
447
+ if (!e.currentTarget.contains(e.target as Element)) {
448
+ return;
449
+ }
450
+
451
+ if (e.button === 0) {
452
+ // Chrome and Firefox on touch Windows devices require mouse down events
453
+ // to be canceled in addition to pointer events, or an extra asynchronous
454
+ // focus event will be fired.
455
+ if (shouldPreventDefault(e.currentTarget as Element)) {
456
+ e.preventDefault();
457
+ }
458
+
459
+ e.stopPropagation();
460
+ }
461
+ };
462
+
463
+ pressProps.onPointerUp = (e) => {
464
+ // iOS fires pointerup with zero width and height, so check the pointerType recorded during pointerdown.
465
+ if (!e.currentTarget.contains(e.target as Element) || state.pointerType === 'virtual') {
466
+ return;
467
+ }
468
+
469
+ // Only handle left clicks
470
+ // Safari on iOS sometimes fires pointerup events, even
471
+ // when the touch isn't over the target, so double check.
472
+ if (e.button === 0 && isOverTarget(e, e.currentTarget)) {
473
+ triggerPressUp(e, state.pointerType || e.pointerType);
474
+ }
475
+ };
476
+
477
+ // Safari on iOS < 13.2 does not implement pointerenter/pointerleave events correctly.
478
+ // Use pointer move events instead to implement our own hit testing.
479
+ // See https://bugs.webkit.org/show_bug.cgi?id=199803
480
+ let onPointerMove = (e: PointerEvent) => {
481
+ if (e.pointerId !== state.activePointerId) {
482
+ return;
483
+ }
484
+
485
+ if (state.target && isOverTarget(e, state.target)) {
486
+ if (!state.isOverTarget && state.pointerType != null) {
487
+ state.isOverTarget = true;
488
+ triggerPressStart(createEvent(state.target, e), state.pointerType);
489
+ }
490
+ } else if (state.target && state.isOverTarget && state.pointerType != null) {
491
+ state.isOverTarget = false;
492
+ triggerPressEnd(createEvent(state.target, e), state.pointerType, false);
493
+ cancelOnPointerExit(e);
494
+ }
495
+ };
496
+
497
+ let onPointerUp = (e: PointerEvent) => {
498
+ if (e.pointerId === state.activePointerId && state.isPressed && e.button === 0 && state.target) {
499
+ if (isOverTarget(e, state.target) && state.pointerType != null) {
500
+ triggerPressEnd(createEvent(state.target, e), state.pointerType);
501
+ } else if (state.isOverTarget && state.pointerType != null) {
502
+ triggerPressEnd(createEvent(state.target, e), state.pointerType, false);
503
+ }
504
+
505
+ state.isPressed = false;
506
+ state.isOverTarget = false;
507
+ state.activePointerId = null;
508
+ state.pointerType = null;
509
+ removeAllGlobalListeners();
510
+ if (!allowTextSelectionOnPress) {
511
+ restoreTextSelection(state.target);
512
+ }
513
+ }
514
+ };
515
+
516
+ let onPointerCancel = (e: PointerEvent) => {
517
+ cancel(e);
518
+ };
519
+
520
+ pressProps.onDragStart = (e) => {
521
+ if (!e.currentTarget.contains(e.target as Element)) {
522
+ return;
523
+ }
524
+
525
+ // Safari does not call onPointerCancel when a drag starts, whereas Chrome and Firefox do.
526
+ cancel(e);
527
+ };
528
+ } else {
529
+ pressProps.onMouseDown = (e) => {
530
+ // Only handle left clicks
531
+ if (e.button !== 0 || !e.currentTarget.contains(e.target as Element)) {
532
+ return;
533
+ }
534
+
535
+ // Due to browser inconsistencies, especially on mobile browsers, we prevent
536
+ // default on mouse down and handle focusing the pressable element ourselves.
537
+ if (shouldPreventDefault(e.currentTarget)) {
538
+ e.preventDefault();
539
+ }
540
+
541
+ if (state.ignoreEmulatedMouseEvents) {
542
+ e.stopPropagation();
543
+ return;
544
+ }
545
+
546
+ state.isPressed = true;
547
+ state.isOverTarget = true;
548
+ state.target = e.currentTarget;
549
+ state.pointerType = isVirtualClick(e.nativeEvent) ? 'virtual' : 'mouse';
550
+
551
+ if (!isDisabled && !preventFocusOnPress) {
552
+ focusWithoutScrolling(e.currentTarget);
553
+ }
554
+
555
+ let shouldStopPropagation = triggerPressStart(e, state.pointerType);
556
+ if (shouldStopPropagation) {
557
+ e.stopPropagation();
558
+ }
559
+
560
+ addGlobalListener(getOwnerDocument(e.currentTarget), 'mouseup', onMouseUp, false);
561
+ };
562
+
563
+ pressProps.onMouseEnter = (e) => {
564
+ if (!e.currentTarget.contains(e.target as Element)) {
565
+ return;
566
+ }
567
+
568
+ let shouldStopPropagation = true;
569
+ if (state.isPressed && !state.ignoreEmulatedMouseEvents && state.pointerType != null) {
570
+ state.isOverTarget = true;
571
+ shouldStopPropagation = triggerPressStart(e, state.pointerType);
572
+ }
573
+
574
+ if (shouldStopPropagation) {
575
+ e.stopPropagation();
576
+ }
577
+ };
578
+
579
+ pressProps.onMouseLeave = (e) => {
580
+ if (!e.currentTarget.contains(e.target as Element)) {
581
+ return;
582
+ }
583
+
584
+ let shouldStopPropagation = true;
585
+ if (state.isPressed && !state.ignoreEmulatedMouseEvents && state.pointerType != null) {
586
+ state.isOverTarget = false;
587
+ shouldStopPropagation = triggerPressEnd(e, state.pointerType, false);
588
+ cancelOnPointerExit(e);
589
+ }
590
+
591
+ if (shouldStopPropagation) {
592
+ e.stopPropagation();
593
+ }
594
+ };
595
+
596
+ pressProps.onMouseUp = (e) => {
597
+ if (!e.currentTarget.contains(e.target as Element)) {
598
+ return;
599
+ }
600
+
601
+ if (!state.ignoreEmulatedMouseEvents && e.button === 0) {
602
+ triggerPressUp(e, state.pointerType || 'mouse');
603
+ }
604
+ };
605
+
606
+ let onMouseUp = (e: MouseEvent) => {
607
+ // Only handle left clicks
608
+ if (e.button !== 0) {
609
+ return;
610
+ }
611
+
612
+ state.isPressed = false;
613
+ removeAllGlobalListeners();
614
+
615
+ if (state.ignoreEmulatedMouseEvents) {
616
+ state.ignoreEmulatedMouseEvents = false;
617
+ return;
618
+ }
619
+
620
+ if (state.target && isOverTarget(e, state.target) && state.pointerType != null) {
621
+ triggerPressEnd(createEvent(state.target, e), state.pointerType);
622
+ } else if (state.target && state.isOverTarget && state.pointerType != null) {
623
+ triggerPressEnd(createEvent(state.target, e), state.pointerType, false);
624
+ }
625
+
626
+ state.isOverTarget = false;
627
+ };
628
+
629
+ pressProps.onTouchStart = (e) => {
630
+ if (!e.currentTarget.contains(e.target as Element)) {
631
+ return;
632
+ }
633
+
634
+ let touch = getTouchFromEvent(e.nativeEvent);
635
+ if (!touch) {
636
+ return;
637
+ }
638
+ state.activePointerId = touch.identifier;
639
+ state.ignoreEmulatedMouseEvents = true;
640
+ state.isOverTarget = true;
641
+ state.isPressed = true;
642
+ state.target = e.currentTarget;
643
+ state.pointerType = 'touch';
644
+
645
+ // Due to browser inconsistencies, especially on mobile browsers, we prevent default
646
+ // on the emulated mouse event and handle focusing the pressable element ourselves.
647
+ if (!isDisabled && !preventFocusOnPress) {
648
+ focusWithoutScrolling(e.currentTarget);
649
+ }
650
+
651
+ if (!allowTextSelectionOnPress) {
652
+ disableTextSelection(state.target);
653
+ }
654
+
655
+ let shouldStopPropagation = triggerPressStart(createTouchEvent(state.target, e), state.pointerType);
656
+ if (shouldStopPropagation) {
657
+ e.stopPropagation();
658
+ }
659
+
660
+ addGlobalListener(getOwnerWindow(e.currentTarget), 'scroll', onScroll, true);
661
+ };
662
+
663
+ pressProps.onTouchMove = (e) => {
664
+ if (!e.currentTarget.contains(e.target as Element)) {
665
+ return;
666
+ }
667
+
668
+ if (!state.isPressed) {
669
+ e.stopPropagation();
670
+ return;
671
+ }
672
+
673
+ let touch = getTouchById(e.nativeEvent, state.activePointerId);
674
+ let shouldStopPropagation = true;
675
+ if (touch && isOverTarget(touch, e.currentTarget)) {
676
+ if (!state.isOverTarget && state.pointerType != null) {
677
+ state.isOverTarget = true;
678
+ shouldStopPropagation = triggerPressStart(createTouchEvent(state.target!, e), state.pointerType);
679
+ }
680
+ } else if (state.isOverTarget && state.pointerType != null) {
681
+ state.isOverTarget = false;
682
+ shouldStopPropagation = triggerPressEnd(createTouchEvent(state.target!, e), state.pointerType, false);
683
+ cancelOnPointerExit(createTouchEvent(state.target!, e));
684
+ }
685
+
686
+ if (shouldStopPropagation) {
687
+ e.stopPropagation();
688
+ }
689
+ };
690
+
691
+ pressProps.onTouchEnd = (e) => {
692
+ if (!e.currentTarget.contains(e.target as Element)) {
693
+ return;
694
+ }
695
+
696
+ if (!state.isPressed) {
697
+ e.stopPropagation();
698
+ return;
699
+ }
700
+
701
+ let touch = getTouchById(e.nativeEvent, state.activePointerId);
702
+ let shouldStopPropagation = true;
703
+ if (touch && isOverTarget(touch, e.currentTarget) && state.pointerType != null) {
704
+ triggerPressUp(createTouchEvent(state.target!, e), state.pointerType);
705
+ shouldStopPropagation = triggerPressEnd(createTouchEvent(state.target!, e), state.pointerType);
706
+ } else if (state.isOverTarget && state.pointerType != null) {
707
+ shouldStopPropagation = triggerPressEnd(createTouchEvent(state.target!, e), state.pointerType, false);
708
+ }
709
+
710
+ if (shouldStopPropagation) {
711
+ e.stopPropagation();
712
+ }
713
+
714
+ state.isPressed = false;
715
+ state.activePointerId = null;
716
+ state.isOverTarget = false;
717
+ state.ignoreEmulatedMouseEvents = true;
718
+ if (state.target && !allowTextSelectionOnPress) {
719
+ restoreTextSelection(state.target);
720
+ }
721
+ removeAllGlobalListeners();
722
+ };
723
+
724
+ pressProps.onTouchCancel = (e) => {
725
+ if (!e.currentTarget.contains(e.target as Element)) {
726
+ return;
727
+ }
728
+
729
+ e.stopPropagation();
730
+ if (state.isPressed) {
731
+ cancel(createTouchEvent(state.target!, e));
732
+ }
733
+ };
734
+
735
+ let onScroll = (e: Event) => {
736
+ if (state.isPressed && (e.target as Element).contains(state.target)) {
737
+ cancel({
738
+ currentTarget: state.target,
739
+ shiftKey: false,
740
+ ctrlKey: false,
741
+ metaKey: false,
742
+ altKey: false
743
+ });
744
+ }
745
+ };
746
+
747
+ pressProps.onDragStart = (e) => {
748
+ if (!e.currentTarget.contains(e.target as Element)) {
749
+ return;
750
+ }
751
+
752
+ cancel(e);
753
+ };
754
+ }
755
+
756
+ return pressProps;
757
+ }, [
758
+ addGlobalListener,
759
+ isDisabled,
760
+ preventFocusOnPress,
761
+ removeAllGlobalListeners,
762
+ allowTextSelectionOnPress,
763
+ cancel,
764
+ cancelOnPointerExit,
765
+ triggerPressEnd,
766
+ triggerPressStart,
767
+ triggerPressUp
768
+ ]);
769
+
770
+ // Remove user-select: none in case component unmounts immediately after pressStart
771
+ // eslint-disable-next-line arrow-body-style
772
+ useEffect(() => {
773
+ return () => {
774
+ if (!allowTextSelectionOnPress) {
775
+ // eslint-disable-next-line react-hooks/exhaustive-deps
776
+ restoreTextSelection(ref.current.target ?? undefined);
777
+ }
778
+ };
779
+ }, [allowTextSelectionOnPress]);
780
+
781
+ return {
782
+ isPressed: isPressedProp || isPressed,
783
+ pressProps: mergeProps(domProps, pressProps)
784
+ };
785
+ }
786
+
787
+ function isHTMLAnchorLink(target: Element): target is HTMLAnchorElement {
788
+ return target.tagName === 'A' && target.hasAttribute('href');
789
+ }
790
+
791
+ function isValidKeyboardEvent(event: KeyboardEvent, currentTarget: Element): boolean {
792
+ const {key, code} = event;
793
+ const element = currentTarget as HTMLElement;
794
+ const role = element.getAttribute('role');
795
+ // Accessibility for keyboards. Space and Enter only.
796
+ // "Spacebar" is for IE 11
797
+ return (
798
+ (key === 'Enter' || key === ' ' || key === 'Spacebar' || code === 'Space') &&
799
+ !((element instanceof getOwnerWindow(element).HTMLInputElement && !isValidInputKey(element, key)) ||
800
+ element instanceof getOwnerWindow(element).HTMLTextAreaElement ||
801
+ element.isContentEditable) &&
802
+ // Links should only trigger with Enter key
803
+ !((role === 'link' || (!role && isHTMLAnchorLink(element))) && key !== 'Enter')
804
+ );
805
+ }
806
+
807
+ function getTouchFromEvent(event: TouchEvent): Touch | null {
808
+ const {targetTouches} = event;
809
+ if (targetTouches.length > 0) {
810
+ return targetTouches[0];
811
+ }
812
+ return null;
813
+ }
814
+
815
+ function getTouchById(
816
+ event: TouchEvent,
817
+ pointerId: null | number
818
+ ): null | Touch {
819
+ const changedTouches = event.changedTouches;
820
+ for (let i = 0; i < changedTouches.length; i++) {
821
+ const touch = changedTouches[i];
822
+ if (touch.identifier === pointerId) {
823
+ return touch;
824
+ }
825
+ }
826
+ return null;
827
+ }
828
+
829
+ function createTouchEvent(target: FocusableElement, e: RTouchEvent<FocusableElement>): EventBase {
830
+ let clientX = 0;
831
+ let clientY = 0;
832
+ if (e.targetTouches && e.targetTouches.length === 1) {
833
+ clientX = e.targetTouches[0].clientX;
834
+ clientY = e.targetTouches[0].clientY;
835
+ }
836
+ return {
837
+ currentTarget: target,
838
+ shiftKey: e.shiftKey,
839
+ ctrlKey: e.ctrlKey,
840
+ metaKey: e.metaKey,
841
+ altKey: e.altKey,
842
+ clientX,
843
+ clientY
844
+ };
845
+ }
846
+
847
+ function createEvent(target: FocusableElement, e: EventBase): EventBase {
848
+ let clientX = e.clientX;
849
+ let clientY = e.clientY;
850
+ return {
851
+ currentTarget: target,
852
+ shiftKey: e.shiftKey,
853
+ ctrlKey: e.ctrlKey,
854
+ metaKey: e.metaKey,
855
+ altKey: e.altKey,
856
+ clientX,
857
+ clientY
858
+ };
859
+ }
860
+
861
+ interface Rect {
862
+ top: number,
863
+ right: number,
864
+ bottom: number,
865
+ left: number
866
+ }
867
+
868
+ interface EventPoint {
869
+ clientX: number,
870
+ clientY: number,
871
+ width?: number,
872
+ height?: number,
873
+ radiusX?: number,
874
+ radiusY?: number
875
+ }
876
+
877
+ function getPointClientRect(point: EventPoint): Rect {
878
+ let offsetX = 0;
879
+ let offsetY = 0;
880
+ if (point.width !== undefined) {
881
+ offsetX = (point.width / 2);
882
+ } else if (point.radiusX !== undefined) {
883
+ offsetX = point.radiusX;
884
+ }
885
+ if (point.height !== undefined) {
886
+ offsetY = (point.height / 2);
887
+ } else if (point.radiusY !== undefined) {
888
+ offsetY = point.radiusY;
889
+ }
890
+
891
+ return {
892
+ top: point.clientY - offsetY,
893
+ right: point.clientX + offsetX,
894
+ bottom: point.clientY + offsetY,
895
+ left: point.clientX - offsetX
896
+ };
897
+ }
898
+
899
+ function areRectanglesOverlapping(a: Rect, b: Rect) {
900
+ // check if they cannot overlap on x axis
901
+ if (a.left > b.right || b.left > a.right) {
902
+ return false;
903
+ }
904
+ // check if they cannot overlap on y axis
905
+ if (a.top > b.bottom || b.top > a.bottom) {
906
+ return false;
907
+ }
908
+ return true;
909
+ }
910
+
911
+ function isOverTarget(point: EventPoint, target: Element) {
912
+ let rect = target.getBoundingClientRect();
913
+ let pointRect = getPointClientRect(point);
914
+ return areRectanglesOverlapping(rect, pointRect);
915
+ }
916
+
917
+ function shouldPreventDefault(target: Element) {
918
+ // We cannot prevent default if the target is a draggable element.
919
+ return !(target instanceof HTMLElement) || !target.hasAttribute('draggable');
920
+ }
921
+
922
+ function shouldPreventDefaultKeyboard(target: Element, key: string) {
923
+ if (target instanceof HTMLInputElement) {
924
+ return !isValidInputKey(target, key);
925
+ }
926
+
927
+ if (target instanceof HTMLButtonElement) {
928
+ return target.type !== 'submit' && target.type !== 'reset';
929
+ }
930
+
931
+ if (isHTMLAnchorLink(target)) {
932
+ return false;
933
+ }
934
+
935
+ return true;
936
+ }
937
+
938
+ const nonTextInputTypes = new Set([
939
+ 'checkbox',
940
+ 'radio',
941
+ 'range',
942
+ 'color',
943
+ 'file',
944
+ 'image',
945
+ 'button',
946
+ 'submit',
947
+ 'reset'
948
+ ]);
949
+
950
+ function isValidInputKey(target: HTMLInputElement, key: string) {
951
+ // Only space should toggle checkboxes and radios, not enter.
952
+ return target.type === 'checkbox' || target.type === 'radio'
953
+ ? key === ' '
954
+ : nonTextInputTypes.has(target.type);
955
+ }