react-native-gesture-handler 2.16.2 → 2.17.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (212) hide show
  1. package/README.md +9 -3
  2. package/android/build.gradle +103 -0
  3. package/android/gradle.properties +7 -0
  4. package/android/paper/src/main/java/com/facebook/react/viewmanagers/RNGestureHandlerButtonManagerDelegate.java +7 -7
  5. package/android/src/main/java/com/swmansion/gesturehandler/core/NativeViewGestureHandler.kt +16 -8
  6. package/android/src/main/java/com/swmansion/gesturehandler/react/RNGestureHandlerButtonViewManager.kt +9 -5
  7. package/android/src/main/java/com/swmansion/gesturehandler/react/RNGestureHandlerInteractionManager.kt +4 -0
  8. package/android/src/main/java/com/swmansion/gesturehandler/react/RNGestureHandlerRootHelper.kt +1 -1
  9. package/android/src/main/java/com/swmansion/gesturehandler/react/RNGestureHandlerTouchEvent.kt +1 -0
  10. package/apple/RNGestureHandlerButtonComponentView.mm +10 -0
  11. package/apple/RNGestureHandlerModule.mm +2 -3
  12. package/lib/commonjs/components/GestureButtons.js +27 -12
  13. package/lib/commonjs/components/GestureButtons.js.map +1 -1
  14. package/lib/commonjs/handlers/createHandler.js +1 -3
  15. package/lib/commonjs/handlers/createHandler.js.map +1 -1
  16. package/lib/commonjs/handlers/gestureHandlerCommon.js +3 -3
  17. package/lib/commonjs/handlers/gestureHandlerCommon.js.map +1 -1
  18. package/lib/commonjs/handlers/gestures/GestureDetector/Wrap.js +42 -0
  19. package/lib/commonjs/handlers/gestures/GestureDetector/Wrap.js.map +1 -0
  20. package/lib/commonjs/handlers/gestures/GestureDetector/attachHandlers.js +83 -0
  21. package/lib/commonjs/handlers/gestures/GestureDetector/attachHandlers.js.map +1 -0
  22. package/lib/commonjs/handlers/gestures/GestureDetector/dropHandlers.js +25 -0
  23. package/lib/commonjs/handlers/gestures/GestureDetector/dropHandlers.js.map +1 -0
  24. package/lib/commonjs/handlers/gestures/GestureDetector/index.js +143 -0
  25. package/lib/commonjs/handlers/gestures/GestureDetector/index.js.map +1 -0
  26. package/lib/commonjs/handlers/gestures/GestureDetector/needsToReattach.js +25 -0
  27. package/lib/commonjs/handlers/gestures/GestureDetector/needsToReattach.js.map +1 -0
  28. package/lib/commonjs/handlers/gestures/GestureDetector/types.js +6 -0
  29. package/lib/commonjs/handlers/gestures/GestureDetector/types.js.map +1 -0
  30. package/lib/commonjs/handlers/gestures/GestureDetector/updateHandlers.js +80 -0
  31. package/lib/commonjs/handlers/gestures/GestureDetector/updateHandlers.js.map +1 -0
  32. package/lib/commonjs/handlers/gestures/GestureDetector/useAnimatedGesture.js +180 -0
  33. package/lib/commonjs/handlers/gestures/GestureDetector/useAnimatedGesture.js.map +1 -0
  34. package/lib/commonjs/handlers/gestures/GestureDetector/useDetectorUpdater.js +55 -0
  35. package/lib/commonjs/handlers/gestures/GestureDetector/useDetectorUpdater.js.map +1 -0
  36. package/lib/commonjs/handlers/gestures/GestureDetector/useViewRefHandler.js +47 -0
  37. package/lib/commonjs/handlers/gestures/GestureDetector/useViewRefHandler.js.map +1 -0
  38. package/lib/commonjs/handlers/gestures/GestureDetector/utils.js +176 -0
  39. package/lib/commonjs/handlers/gestures/GestureDetector/utils.js.map +1 -0
  40. package/lib/commonjs/handlers/gestures/gestureComposition.js +3 -1
  41. package/lib/commonjs/handlers/gestures/gestureComposition.js.map +1 -1
  42. package/lib/commonjs/handlers/gestures/gestureStateManager.js +8 -0
  43. package/lib/commonjs/handlers/gestures/gestureStateManager.js.map +1 -1
  44. package/lib/commonjs/handlers/gestures/reanimatedWrapper.js.map +1 -1
  45. package/lib/commonjs/utils.js +36 -0
  46. package/lib/commonjs/utils.js.map +1 -1
  47. package/lib/commonjs/web/detectors/RotationGestureDetector.js +6 -8
  48. package/lib/commonjs/web/detectors/RotationGestureDetector.js.map +1 -1
  49. package/lib/commonjs/web/detectors/ScaleGestureDetector.js +5 -6
  50. package/lib/commonjs/web/detectors/ScaleGestureDetector.js.map +1 -1
  51. package/lib/commonjs/web/handlers/FlingGestureHandler.js +1 -0
  52. package/lib/commonjs/web/handlers/FlingGestureHandler.js.map +1 -1
  53. package/lib/commonjs/web/handlers/GestureHandler.js +56 -64
  54. package/lib/commonjs/web/handlers/GestureHandler.js.map +1 -1
  55. package/lib/commonjs/web/handlers/LongPressGestureHandler.js +1 -0
  56. package/lib/commonjs/web/handlers/LongPressGestureHandler.js.map +1 -1
  57. package/lib/commonjs/web/handlers/ManualGestureHandler.js +1 -0
  58. package/lib/commonjs/web/handlers/ManualGestureHandler.js.map +1 -1
  59. package/lib/commonjs/web/handlers/NativeViewGestureHandler.js +7 -4
  60. package/lib/commonjs/web/handlers/NativeViewGestureHandler.js.map +1 -1
  61. package/lib/commonjs/web/handlers/PanGestureHandler.js +28 -18
  62. package/lib/commonjs/web/handlers/PanGestureHandler.js.map +1 -1
  63. package/lib/commonjs/web/handlers/PinchGestureHandler.js +1 -0
  64. package/lib/commonjs/web/handlers/PinchGestureHandler.js.map +1 -1
  65. package/lib/commonjs/web/handlers/RotationGestureHandler.js +1 -0
  66. package/lib/commonjs/web/handlers/RotationGestureHandler.js.map +1 -1
  67. package/lib/commonjs/web/handlers/TapGestureHandler.js +18 -12
  68. package/lib/commonjs/web/handlers/TapGestureHandler.js.map +1 -1
  69. package/lib/commonjs/web/interfaces.js.map +1 -1
  70. package/lib/commonjs/web/tools/GestureHandlerOrchestrator.js +1 -6
  71. package/lib/commonjs/web/tools/GestureHandlerOrchestrator.js.map +1 -1
  72. package/lib/commonjs/web/tools/PointerEventManager.js +7 -2
  73. package/lib/commonjs/web/tools/PointerEventManager.js.map +1 -1
  74. package/lib/commonjs/web/tools/PointerTracker.js +89 -57
  75. package/lib/commonjs/web/tools/PointerTracker.js.map +1 -1
  76. package/lib/commonjs/web/tools/TouchEventManager.js +6 -2
  77. package/lib/commonjs/web/tools/TouchEventManager.js.map +1 -1
  78. package/lib/commonjs/web/tools/Vector.js +2 -1
  79. package/lib/commonjs/web/tools/Vector.js.map +1 -1
  80. package/lib/commonjs/web/utils.js +31 -0
  81. package/lib/commonjs/web/utils.js.map +1 -1
  82. package/lib/module/components/GestureButtons.js +24 -6
  83. package/lib/module/components/GestureButtons.js.map +1 -1
  84. package/lib/module/handlers/createHandler.js +2 -4
  85. package/lib/module/handlers/createHandler.js.map +1 -1
  86. package/lib/module/handlers/gestureHandlerCommon.js +1 -1
  87. package/lib/module/handlers/gestureHandlerCommon.js.map +1 -1
  88. package/lib/module/handlers/gestures/GestureDetector/Wrap.js +26 -0
  89. package/lib/module/handlers/gestures/GestureDetector/Wrap.js.map +1 -0
  90. package/lib/module/handlers/gestures/GestureDetector/attachHandlers.js +65 -0
  91. package/lib/module/handlers/gestures/GestureDetector/attachHandlers.js.map +1 -0
  92. package/lib/module/handlers/gestures/GestureDetector/dropHandlers.js +12 -0
  93. package/lib/module/handlers/gestures/GestureDetector/dropHandlers.js.map +1 -0
  94. package/lib/module/handlers/gestures/GestureDetector/index.js +115 -0
  95. package/lib/module/handlers/gestures/GestureDetector/index.js.map +1 -0
  96. package/lib/module/handlers/gestures/GestureDetector/needsToReattach.js +18 -0
  97. package/lib/module/handlers/gestures/GestureDetector/needsToReattach.js.map +1 -0
  98. package/lib/module/handlers/gestures/GestureDetector/types.js +2 -0
  99. package/lib/module/handlers/gestures/GestureDetector/types.js.map +1 -0
  100. package/lib/module/handlers/gestures/GestureDetector/updateHandlers.js +64 -0
  101. package/lib/module/handlers/gestures/GestureDetector/updateHandlers.js.map +1 -0
  102. package/lib/module/handlers/gestures/GestureDetector/useAnimatedGesture.js +165 -0
  103. package/lib/module/handlers/gestures/GestureDetector/useAnimatedGesture.js.map +1 -0
  104. package/lib/module/handlers/gestures/GestureDetector/useDetectorUpdater.js +41 -0
  105. package/lib/module/handlers/gestures/GestureDetector/useDetectorUpdater.js.map +1 -0
  106. package/lib/module/handlers/gestures/GestureDetector/useViewRefHandler.js +36 -0
  107. package/lib/module/handlers/gestures/GestureDetector/useViewRefHandler.js.map +1 -0
  108. package/lib/module/handlers/gestures/GestureDetector/utils.js +142 -0
  109. package/lib/module/handlers/gestures/GestureDetector/utils.js.map +1 -0
  110. package/lib/module/handlers/gestures/gestureComposition.js +3 -1
  111. package/lib/module/handlers/gestures/gestureComposition.js.map +1 -1
  112. package/lib/module/handlers/gestures/gestureStateManager.js +8 -0
  113. package/lib/module/handlers/gestures/gestureStateManager.js.map +1 -1
  114. package/lib/module/handlers/gestures/reanimatedWrapper.js.map +1 -1
  115. package/lib/module/utils.js +34 -0
  116. package/lib/module/utils.js.map +1 -1
  117. package/lib/module/web/detectors/RotationGestureDetector.js +6 -8
  118. package/lib/module/web/detectors/RotationGestureDetector.js.map +1 -1
  119. package/lib/module/web/detectors/ScaleGestureDetector.js +5 -6
  120. package/lib/module/web/detectors/ScaleGestureDetector.js.map +1 -1
  121. package/lib/module/web/handlers/FlingGestureHandler.js +1 -0
  122. package/lib/module/web/handlers/FlingGestureHandler.js.map +1 -1
  123. package/lib/module/web/handlers/GestureHandler.js +56 -64
  124. package/lib/module/web/handlers/GestureHandler.js.map +1 -1
  125. package/lib/module/web/handlers/LongPressGestureHandler.js +1 -0
  126. package/lib/module/web/handlers/LongPressGestureHandler.js.map +1 -1
  127. package/lib/module/web/handlers/ManualGestureHandler.js +1 -0
  128. package/lib/module/web/handlers/ManualGestureHandler.js.map +1 -1
  129. package/lib/module/web/handlers/NativeViewGestureHandler.js +7 -4
  130. package/lib/module/web/handlers/NativeViewGestureHandler.js.map +1 -1
  131. package/lib/module/web/handlers/PanGestureHandler.js +28 -18
  132. package/lib/module/web/handlers/PanGestureHandler.js.map +1 -1
  133. package/lib/module/web/handlers/PinchGestureHandler.js +1 -0
  134. package/lib/module/web/handlers/PinchGestureHandler.js.map +1 -1
  135. package/lib/module/web/handlers/RotationGestureHandler.js +1 -0
  136. package/lib/module/web/handlers/RotationGestureHandler.js.map +1 -1
  137. package/lib/module/web/handlers/TapGestureHandler.js +18 -12
  138. package/lib/module/web/handlers/TapGestureHandler.js.map +1 -1
  139. package/lib/module/web/interfaces.js.map +1 -1
  140. package/lib/module/web/tools/GestureHandlerOrchestrator.js +1 -6
  141. package/lib/module/web/tools/GestureHandlerOrchestrator.js.map +1 -1
  142. package/lib/module/web/tools/PointerEventManager.js +8 -3
  143. package/lib/module/web/tools/PointerEventManager.js.map +1 -1
  144. package/lib/module/web/tools/PointerTracker.js +89 -57
  145. package/lib/module/web/tools/PointerTracker.js.map +1 -1
  146. package/lib/module/web/tools/TouchEventManager.js +7 -3
  147. package/lib/module/web/tools/TouchEventManager.js.map +1 -1
  148. package/lib/module/web/tools/Vector.js +2 -1
  149. package/lib/module/web/tools/Vector.js.map +1 -1
  150. package/lib/module/web/utils.js +29 -0
  151. package/lib/module/web/utils.js.map +1 -1
  152. package/lib/typescript/components/GestureButtons.d.ts +3 -34
  153. package/lib/typescript/handlers/gestureHandlerCommon.d.ts +3 -2
  154. package/lib/typescript/handlers/gestures/GestureDetector/Wrap.d.ts +13 -0
  155. package/lib/typescript/handlers/gestures/GestureDetector/attachHandlers.d.ts +13 -0
  156. package/lib/typescript/handlers/gestures/GestureDetector/dropHandlers.d.ts +2 -0
  157. package/lib/typescript/handlers/gestures/{GestureDetector.d.ts → GestureDetector/index.d.ts} +4 -12
  158. package/lib/typescript/handlers/gestures/GestureDetector/needsToReattach.d.ts +3 -0
  159. package/lib/typescript/handlers/gestures/GestureDetector/types.d.ts +20 -0
  160. package/lib/typescript/handlers/gestures/GestureDetector/updateHandlers.d.ts +4 -0
  161. package/lib/typescript/handlers/gestures/GestureDetector/useAnimatedGesture.d.ts +2 -0
  162. package/lib/typescript/handlers/gestures/GestureDetector/useDetectorUpdater.d.ts +5 -0
  163. package/lib/typescript/handlers/gestures/GestureDetector/useViewRefHandler.d.ts +3 -0
  164. package/lib/typescript/handlers/gestures/GestureDetector/utils.d.ts +12 -0
  165. package/lib/typescript/handlers/gestures/reanimatedWrapper.d.ts +1 -1
  166. package/lib/typescript/utils.d.ts +10 -0
  167. package/lib/typescript/web/handlers/GestureHandler.d.ts +1 -0
  168. package/lib/typescript/web/interfaces.d.ts +1 -0
  169. package/lib/typescript/web/tools/PointerTracker.d.ts +34 -31
  170. package/lib/typescript/web/utils.d.ts +4 -0
  171. package/package.json +5 -5
  172. package/src/components/GestureButtons.tsx +36 -4
  173. package/src/handlers/createHandler.tsx +1 -3
  174. package/src/handlers/gestureHandlerCommon.ts +4 -1
  175. package/src/handlers/gestures/GestureDetector/Wrap.tsx +35 -0
  176. package/src/handlers/gestures/GestureDetector/attachHandlers.ts +112 -0
  177. package/src/handlers/gestures/GestureDetector/dropHandlers.ts +14 -0
  178. package/src/handlers/gestures/GestureDetector/index.tsx +187 -0
  179. package/src/handlers/gestures/GestureDetector/needsToReattach.ts +27 -0
  180. package/src/handlers/gestures/GestureDetector/types.ts +32 -0
  181. package/src/handlers/gestures/GestureDetector/updateHandlers.ts +94 -0
  182. package/src/handlers/gestures/GestureDetector/useAnimatedGesture.ts +206 -0
  183. package/src/handlers/gestures/GestureDetector/useDetectorUpdater.ts +69 -0
  184. package/src/handlers/gestures/GestureDetector/useViewRefHandler.ts +54 -0
  185. package/src/handlers/gestures/GestureDetector/utils.ts +185 -0
  186. package/src/handlers/gestures/gestureComposition.ts +2 -0
  187. package/src/handlers/gestures/gestureStateManager.ts +12 -4
  188. package/src/handlers/gestures/reanimatedWrapper.ts +19 -17
  189. package/src/utils.ts +39 -0
  190. package/src/web/detectors/RotationGestureDetector.ts +6 -8
  191. package/src/web/detectors/ScaleGestureDetector.ts +5 -6
  192. package/src/web/handlers/FlingGestureHandler.ts +2 -0
  193. package/src/web/handlers/GestureHandler.ts +53 -62
  194. package/src/web/handlers/LongPressGestureHandler.ts +2 -0
  195. package/src/web/handlers/ManualGestureHandler.ts +2 -0
  196. package/src/web/handlers/NativeViewGestureHandler.ts +8 -4
  197. package/src/web/handlers/PanGestureHandler.ts +32 -19
  198. package/src/web/handlers/PinchGestureHandler.ts +2 -0
  199. package/src/web/handlers/RotationGestureHandler.ts +2 -0
  200. package/src/web/handlers/TapGestureHandler.ts +20 -12
  201. package/src/web/interfaces.ts +1 -0
  202. package/src/web/tools/GestureHandlerOrchestrator.ts +1 -7
  203. package/src/web/tools/PointerEventManager.ts +10 -3
  204. package/src/web/tools/PointerTracker.ts +81 -74
  205. package/src/web/tools/TouchEventManager.ts +5 -3
  206. package/src/web/tools/Vector.ts +2 -4
  207. package/src/web/utils.ts +34 -0
  208. package/lib/commonjs/handlers/gestures/GestureDetector.js +0 -704
  209. package/lib/commonjs/handlers/gestures/GestureDetector.js.map +0 -1
  210. package/lib/module/handlers/gestures/GestureDetector.js +0 -654
  211. package/lib/module/handlers/gestures/GestureDetector.js.map +0 -1
  212. package/src/handlers/gestures/GestureDetector.tsx +0 -889
@@ -1,889 +0,0 @@
1
- import React, { useContext, useEffect, useRef, useState } from 'react';
2
- import {
3
- GestureType,
4
- HandlerCallbacks,
5
- BaseGesture,
6
- GestureRef,
7
- CALLBACK_TYPE,
8
- } from './gesture';
9
- import { Reanimated, SharedValue } from './reanimatedWrapper';
10
- import { registerHandler, unregisterHandler } from '../handlersRegistry';
11
- import RNGestureHandlerModule from '../../RNGestureHandlerModule';
12
- import {
13
- baseGestureHandlerWithMonitorProps,
14
- filterConfig,
15
- findNodeHandle,
16
- GestureTouchEvent,
17
- GestureUpdateEvent,
18
- GestureStateChangeEvent,
19
- HandlerStateChangeEvent,
20
- scheduleFlushOperations,
21
- UserSelect,
22
- TouchAction,
23
- } from '../gestureHandlerCommon';
24
- import {
25
- GestureStateManager,
26
- GestureStateManagerType,
27
- } from './gestureStateManager';
28
- import { flingGestureHandlerProps } from '../FlingGestureHandler';
29
- import { forceTouchGestureHandlerProps } from '../ForceTouchGestureHandler';
30
- import { longPressGestureHandlerProps } from '../LongPressGestureHandler';
31
- import {
32
- panGestureHandlerProps,
33
- panGestureHandlerCustomNativeProps,
34
- } from '../PanGestureHandler';
35
- import { tapGestureHandlerProps } from '../TapGestureHandler';
36
- import { hoverGestureHandlerProps } from './hoverGesture';
37
- import { State } from '../../State';
38
- import { TouchEventType } from '../../TouchEventType';
39
- import { ComposedGesture } from './gestureComposition';
40
- import { ActionType } from '../../ActionType';
41
- import { isFabric, isJestEnv, tagMessage } from '../../utils';
42
- import { getReactNativeVersion } from '../../getReactNativeVersion';
43
- import { getShadowNodeFromRef } from '../../getShadowNodeFromRef';
44
- import { Platform } from 'react-native';
45
- import type RNGestureHandlerModuleWeb from '../../RNGestureHandlerModule.web';
46
- import { onGestureHandlerEvent } from './eventReceiver';
47
- import { RNRenderer } from '../../RNRenderer';
48
- import { isNewWebImplementationEnabled } from '../../EnableNewWebImplementation';
49
- import { nativeViewGestureHandlerProps } from '../NativeViewGestureHandler';
50
- import GestureHandlerRootViewContext from '../../GestureHandlerRootViewContext';
51
- import { ghQueueMicrotask } from '../../ghQueueMicrotask';
52
-
53
- declare const global: {
54
- isFormsStackingContext: (node: unknown) => boolean | null; // JSI function
55
- };
56
-
57
- const ALLOWED_PROPS = [
58
- ...baseGestureHandlerWithMonitorProps,
59
- ...tapGestureHandlerProps,
60
- ...panGestureHandlerProps,
61
- ...panGestureHandlerCustomNativeProps,
62
- ...longPressGestureHandlerProps,
63
- ...forceTouchGestureHandlerProps,
64
- ...flingGestureHandlerProps,
65
- ...hoverGestureHandlerProps,
66
- ...nativeViewGestureHandlerProps,
67
- ];
68
-
69
- export type GestureConfigReference = {
70
- config: GestureType[];
71
- animatedEventHandler: unknown;
72
- animatedHandlers: SharedValue<
73
- HandlerCallbacks<Record<string, unknown>>[] | null
74
- > | null;
75
- firstExecution: boolean;
76
- useReanimatedHook: boolean;
77
- };
78
-
79
- function convertToHandlerTag(ref: GestureRef): number {
80
- if (typeof ref === 'number') {
81
- return ref;
82
- } else if (ref instanceof BaseGesture) {
83
- return ref.handlerTag;
84
- } else {
85
- // @ts-ignore in this case it should be a ref either to gesture object or
86
- // a gesture handler component, in both cases handlerTag property exists
87
- return ref.current?.handlerTag ?? -1;
88
- }
89
- }
90
-
91
- function extractValidHandlerTags(interactionGroup: GestureRef[] | undefined) {
92
- return (
93
- interactionGroup?.map(convertToHandlerTag)?.filter((tag) => tag > 0) ?? []
94
- );
95
- }
96
-
97
- function dropHandlers(preparedGesture: GestureConfigReference) {
98
- for (const handler of preparedGesture.config) {
99
- RNGestureHandlerModule.dropGestureHandler(handler.handlerTag);
100
-
101
- unregisterHandler(handler.handlerTag, handler.config.testId);
102
- }
103
-
104
- scheduleFlushOperations();
105
- }
106
-
107
- function checkGestureCallbacksForWorklets(gesture: GestureType) {
108
- // if a gesture is explicitly marked to run on the JS thread there is no need to check
109
- // if callbacks are worklets as the user is aware they will be ran on the JS thread
110
- if (gesture.config.runOnJS) {
111
- return;
112
- }
113
-
114
- const areSomeNotWorklets = gesture.handlers.isWorklet.includes(false);
115
- const areSomeWorklets = gesture.handlers.isWorklet.includes(true);
116
-
117
- // if some of the callbacks are worklets and some are not, and the gesture is not
118
- // explicitly marked with `.runOnJS(true)` show an error
119
- if (areSomeNotWorklets && areSomeWorklets) {
120
- console.error(
121
- tagMessage(
122
- `Some of the callbacks in the gesture are worklets and some are not. Either make sure that all calbacks are marked as 'worklet' if you wish to run them on the UI thread or use '.runOnJS(true)' modifier on the gesture explicitly to run all callbacks on the JS thread.`
123
- )
124
- );
125
- }
126
- }
127
-
128
- interface WebEventHandler {
129
- onGestureHandlerEvent: (event: HandlerStateChangeEvent<unknown>) => void;
130
- onGestureHandlerStateChange?: (
131
- event: HandlerStateChangeEvent<unknown>
132
- ) => void;
133
- }
134
-
135
- interface AttachHandlersConfig {
136
- preparedGesture: GestureConfigReference;
137
- gestureConfig: ComposedGesture | GestureType;
138
- gesture: GestureType[];
139
- viewTag: number;
140
- webEventHandlersRef: React.RefObject<WebEventHandler>;
141
- mountedRef: React.RefObject<boolean>;
142
- }
143
-
144
- function attachHandlers({
145
- preparedGesture,
146
- gestureConfig,
147
- gesture,
148
- viewTag,
149
- webEventHandlersRef,
150
- mountedRef,
151
- }: AttachHandlersConfig) {
152
- if (!preparedGesture.firstExecution) {
153
- gestureConfig.initialize();
154
- } else {
155
- preparedGesture.firstExecution = false;
156
- }
157
-
158
- // use queueMicrotask to extract handlerTags, because all refs should be initialized
159
- // when it's ran
160
- ghQueueMicrotask(() => {
161
- if (!mountedRef.current) {
162
- return;
163
- }
164
- gestureConfig.prepare();
165
- });
166
-
167
- for (const handler of gesture) {
168
- checkGestureCallbacksForWorklets(handler);
169
- RNGestureHandlerModule.createGestureHandler(
170
- handler.handlerName,
171
- handler.handlerTag,
172
- filterConfig(handler.config, ALLOWED_PROPS)
173
- );
174
-
175
- registerHandler(handler.handlerTag, handler, handler.config.testId);
176
- }
177
-
178
- // use queueMicrotask to extract handlerTags, because all refs should be initialized
179
- // when it's ran
180
- ghQueueMicrotask(() => {
181
- if (!mountedRef.current) {
182
- return;
183
- }
184
- for (const handler of gesture) {
185
- let requireToFail: number[] = [];
186
- if (handler.config.requireToFail) {
187
- requireToFail = extractValidHandlerTags(handler.config.requireToFail);
188
- }
189
-
190
- let simultaneousWith: number[] = [];
191
- if (handler.config.simultaneousWith) {
192
- simultaneousWith = extractValidHandlerTags(
193
- handler.config.simultaneousWith
194
- );
195
- }
196
-
197
- let blocksHandlers: number[] = [];
198
- if (handler.config.blocksHandlers) {
199
- blocksHandlers = extractValidHandlerTags(handler.config.blocksHandlers);
200
- }
201
-
202
- RNGestureHandlerModule.updateGestureHandler(
203
- handler.handlerTag,
204
- filterConfig(handler.config, ALLOWED_PROPS, {
205
- simultaneousHandlers: simultaneousWith,
206
- waitFor: requireToFail,
207
- blocksHandlers: blocksHandlers,
208
- })
209
- );
210
- }
211
-
212
- scheduleFlushOperations();
213
- });
214
-
215
- preparedGesture.config = gesture;
216
-
217
- for (const gesture of preparedGesture.config) {
218
- const actionType = gesture.shouldUseReanimated
219
- ? ActionType.REANIMATED_WORKLET
220
- : ActionType.JS_FUNCTION_NEW_API;
221
-
222
- if (Platform.OS === 'web') {
223
- (
224
- RNGestureHandlerModule.attachGestureHandler as typeof RNGestureHandlerModuleWeb.attachGestureHandler
225
- )(
226
- gesture.handlerTag,
227
- viewTag,
228
- ActionType.JS_FUNCTION_OLD_API, // ignored on web
229
- webEventHandlersRef
230
- );
231
- } else {
232
- RNGestureHandlerModule.attachGestureHandler(
233
- gesture.handlerTag,
234
- viewTag,
235
- actionType
236
- );
237
- }
238
- }
239
-
240
- if (preparedGesture.animatedHandlers) {
241
- const isAnimatedGesture = (g: GestureType) => g.shouldUseReanimated;
242
-
243
- preparedGesture.animatedHandlers.value = gesture
244
- .filter(isAnimatedGesture)
245
- .map((g) => g.handlers) as unknown as HandlerCallbacks<
246
- Record<string, unknown>
247
- >[];
248
- }
249
- }
250
-
251
- function updateHandlers(
252
- preparedGesture: GestureConfigReference,
253
- gestureConfig: ComposedGesture | GestureType,
254
- gesture: GestureType[],
255
- mountedRef: React.RefObject<boolean>
256
- ) {
257
- gestureConfig.prepare();
258
-
259
- for (let i = 0; i < gesture.length; i++) {
260
- const handler = preparedGesture.config[i];
261
- checkGestureCallbacksForWorklets(handler);
262
-
263
- // only update handlerTag when it's actually different, it may be the same
264
- // if gesture config object is wrapped with useMemo
265
- if (gesture[i].handlerTag !== handler.handlerTag) {
266
- gesture[i].handlerTag = handler.handlerTag;
267
- gesture[i].handlers.handlerTag = handler.handlerTag;
268
- }
269
- }
270
-
271
- // use queueMicrotask to extract handlerTags, because when it's ran, all refs should be updated
272
- // and handlerTags in BaseGesture references should be updated in the loop above (we need to wait
273
- // in case of external relations)
274
- ghQueueMicrotask(() => {
275
- if (!mountedRef.current) {
276
- return;
277
- }
278
- for (let i = 0; i < gesture.length; i++) {
279
- const handler = preparedGesture.config[i];
280
-
281
- handler.config = gesture[i].config;
282
- handler.handlers = gesture[i].handlers;
283
-
284
- const requireToFail = extractValidHandlerTags(
285
- handler.config.requireToFail
286
- );
287
-
288
- const simultaneousWith = extractValidHandlerTags(
289
- handler.config.simultaneousWith
290
- );
291
-
292
- RNGestureHandlerModule.updateGestureHandler(
293
- handler.handlerTag,
294
- filterConfig(handler.config, ALLOWED_PROPS, {
295
- simultaneousHandlers: simultaneousWith,
296
- waitFor: requireToFail,
297
- })
298
- );
299
-
300
- registerHandler(handler.handlerTag, handler, handler.config.testId);
301
- }
302
-
303
- if (preparedGesture.animatedHandlers) {
304
- const previousHandlersValue =
305
- preparedGesture.animatedHandlers.value ?? [];
306
- const newHandlersValue = preparedGesture.config
307
- .filter((g) => g.shouldUseReanimated) // ignore gestures that shouldn't run on UI
308
- .map((g) => g.handlers) as unknown as HandlerCallbacks<
309
- Record<string, unknown>
310
- >[];
311
-
312
- // if amount of gesture configs changes, we need to update the callbacks in shared value
313
- let shouldUpdateSharedValue =
314
- previousHandlersValue.length !== newHandlersValue.length;
315
-
316
- if (!shouldUpdateSharedValue) {
317
- // if the amount is the same, we need to check if any of the configs inside has changed
318
- for (let i = 0; i < newHandlersValue.length; i++) {
319
- if (
320
- // we can use the `gestureId` prop as it's unique for every config instance
321
- newHandlersValue[i].gestureId !== previousHandlersValue[i].gestureId
322
- ) {
323
- shouldUpdateSharedValue = true;
324
- break;
325
- }
326
- }
327
- }
328
-
329
- if (shouldUpdateSharedValue) {
330
- preparedGesture.animatedHandlers.value = newHandlersValue;
331
- }
332
- }
333
-
334
- scheduleFlushOperations();
335
- });
336
- }
337
-
338
- function needsToReattach(
339
- preparedGesture: GestureConfigReference,
340
- gesture: GestureType[]
341
- ) {
342
- if (gesture.length !== preparedGesture.config.length) {
343
- return true;
344
- }
345
- for (let i = 0; i < gesture.length; i++) {
346
- if (
347
- gesture[i].handlerName !== preparedGesture.config[i].handlerName ||
348
- gesture[i].shouldUseReanimated !==
349
- preparedGesture.config[i].shouldUseReanimated
350
- ) {
351
- return true;
352
- }
353
- }
354
-
355
- return false;
356
- }
357
-
358
- function isStateChangeEvent(
359
- event: GestureUpdateEvent | GestureStateChangeEvent | GestureTouchEvent
360
- ): event is GestureStateChangeEvent {
361
- 'worklet';
362
- // @ts-ignore Yes, the oldState prop is missing on GestureTouchEvent, that's the point
363
- return event.oldState != null;
364
- }
365
-
366
- function isTouchEvent(
367
- event: GestureUpdateEvent | GestureStateChangeEvent | GestureTouchEvent
368
- ): event is GestureTouchEvent {
369
- 'worklet';
370
- return event.eventType != null;
371
- }
372
-
373
- function getHandler(
374
- type: CALLBACK_TYPE,
375
- gesture: HandlerCallbacks<Record<string, unknown>>
376
- ) {
377
- 'worklet';
378
- switch (type) {
379
- case CALLBACK_TYPE.BEGAN:
380
- return gesture.onBegin;
381
- case CALLBACK_TYPE.START:
382
- return gesture.onStart;
383
- case CALLBACK_TYPE.UPDATE:
384
- return gesture.onUpdate;
385
- case CALLBACK_TYPE.CHANGE:
386
- return gesture.onChange;
387
- case CALLBACK_TYPE.END:
388
- return gesture.onEnd;
389
- case CALLBACK_TYPE.FINALIZE:
390
- return gesture.onFinalize;
391
- case CALLBACK_TYPE.TOUCHES_DOWN:
392
- return gesture.onTouchesDown;
393
- case CALLBACK_TYPE.TOUCHES_MOVE:
394
- return gesture.onTouchesMove;
395
- case CALLBACK_TYPE.TOUCHES_UP:
396
- return gesture.onTouchesUp;
397
- case CALLBACK_TYPE.TOUCHES_CANCELLED:
398
- return gesture.onTouchesCancelled;
399
- }
400
- }
401
-
402
- function touchEventTypeToCallbackType(
403
- eventType: TouchEventType
404
- ): CALLBACK_TYPE {
405
- 'worklet';
406
- switch (eventType) {
407
- case TouchEventType.TOUCHES_DOWN:
408
- return CALLBACK_TYPE.TOUCHES_DOWN;
409
- case TouchEventType.TOUCHES_MOVE:
410
- return CALLBACK_TYPE.TOUCHES_MOVE;
411
- case TouchEventType.TOUCHES_UP:
412
- return CALLBACK_TYPE.TOUCHES_UP;
413
- case TouchEventType.TOUCHES_CANCELLED:
414
- return CALLBACK_TYPE.TOUCHES_CANCELLED;
415
- }
416
- return CALLBACK_TYPE.UNDEFINED;
417
- }
418
-
419
- function runWorklet(
420
- type: CALLBACK_TYPE,
421
- gesture: HandlerCallbacks<Record<string, unknown>>,
422
- event: GestureStateChangeEvent | GestureUpdateEvent | GestureTouchEvent,
423
- ...args: any[]
424
- ) {
425
- 'worklet';
426
- const handler = getHandler(type, gesture);
427
- if (gesture.isWorklet[type]) {
428
- // @ts-ignore Logic below makes sure the correct event is send to the
429
- // correct handler.
430
- handler?.(event, ...args);
431
- } else if (handler) {
432
- console.warn(tagMessage('Animated gesture callback must be a worklet'));
433
- }
434
- }
435
-
436
- function useAnimatedGesture(
437
- preparedGesture: GestureConfigReference,
438
- needsRebuild: boolean
439
- ) {
440
- if (!Reanimated) {
441
- return;
442
- }
443
-
444
- // Hooks are called conditionally, but the condition is whether the
445
- // react-native-reanimated is installed, which shouldn't change while running
446
- // eslint-disable-next-line react-hooks/rules-of-hooks
447
- const sharedHandlersCallbacks = Reanimated.useSharedValue<
448
- HandlerCallbacks<Record<string, unknown>>[] | null
449
- >(null);
450
-
451
- // eslint-disable-next-line react-hooks/rules-of-hooks
452
- const lastUpdateEvent = Reanimated.useSharedValue<
453
- (GestureUpdateEvent | undefined)[]
454
- >([]);
455
-
456
- // not every gesture needs a state controller, init them lazily
457
- const stateControllers: GestureStateManagerType[] = [];
458
-
459
- const callback = (
460
- event: GestureStateChangeEvent | GestureUpdateEvent | GestureTouchEvent
461
- ) => {
462
- 'worklet';
463
-
464
- const currentCallback = sharedHandlersCallbacks.value;
465
- if (!currentCallback) {
466
- return;
467
- }
468
-
469
- for (let i = 0; i < currentCallback.length; i++) {
470
- const gesture = currentCallback[i];
471
-
472
- if (event.handlerTag === gesture.handlerTag) {
473
- if (isStateChangeEvent(event)) {
474
- if (
475
- event.oldState === State.UNDETERMINED &&
476
- event.state === State.BEGAN
477
- ) {
478
- runWorklet(CALLBACK_TYPE.BEGAN, gesture, event);
479
- } else if (
480
- (event.oldState === State.BEGAN ||
481
- event.oldState === State.UNDETERMINED) &&
482
- event.state === State.ACTIVE
483
- ) {
484
- runWorklet(CALLBACK_TYPE.START, gesture, event);
485
- lastUpdateEvent.value[gesture.handlerTag] = undefined;
486
- } else if (
487
- event.oldState !== event.state &&
488
- event.state === State.END
489
- ) {
490
- if (event.oldState === State.ACTIVE) {
491
- runWorklet(CALLBACK_TYPE.END, gesture, event, true);
492
- }
493
- runWorklet(CALLBACK_TYPE.FINALIZE, gesture, event, true);
494
- } else if (
495
- (event.state === State.FAILED || event.state === State.CANCELLED) &&
496
- event.state !== event.oldState
497
- ) {
498
- if (event.oldState === State.ACTIVE) {
499
- runWorklet(CALLBACK_TYPE.END, gesture, event, false);
500
- }
501
- runWorklet(CALLBACK_TYPE.FINALIZE, gesture, event, false);
502
- }
503
- } else if (isTouchEvent(event)) {
504
- if (!stateControllers[i]) {
505
- stateControllers[i] = GestureStateManager.create(event.handlerTag);
506
- }
507
-
508
- if (event.eventType !== TouchEventType.UNDETERMINED) {
509
- runWorklet(
510
- touchEventTypeToCallbackType(event.eventType),
511
- gesture,
512
- event,
513
- stateControllers[i]
514
- );
515
- }
516
- } else {
517
- runWorklet(CALLBACK_TYPE.UPDATE, gesture, event);
518
-
519
- if (gesture.onChange && gesture.changeEventCalculator) {
520
- runWorklet(
521
- CALLBACK_TYPE.CHANGE,
522
- gesture,
523
- gesture.changeEventCalculator?.(
524
- event,
525
- lastUpdateEvent.value[gesture.handlerTag]
526
- )
527
- );
528
-
529
- lastUpdateEvent.value[gesture.handlerTag] = event;
530
- }
531
- }
532
- }
533
- }
534
- };
535
-
536
- // eslint-disable-next-line react-hooks/rules-of-hooks
537
- const event = Reanimated.useEvent(
538
- callback,
539
- ['onGestureHandlerStateChange', 'onGestureHandlerEvent'],
540
- needsRebuild
541
- );
542
-
543
- preparedGesture.animatedEventHandler = event;
544
- preparedGesture.animatedHandlers = sharedHandlersCallbacks;
545
- }
546
-
547
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
548
- function validateDetectorChildren(ref: any) {
549
- // finds the first native view under the Wrap component and traverses the fiber tree upwards
550
- // to check whether there is more than one native view as a pseudo-direct child of GestureDetector
551
- // i.e. this is not ok:
552
- // Wrap
553
- // |
554
- // / \
555
- // / \
556
- // / \
557
- // / \
558
- // NativeView NativeView
559
- //
560
- // but this is fine:
561
- // Wrap
562
- // |
563
- // NativeView
564
- // |
565
- // / \
566
- // / \
567
- // / \
568
- // / \
569
- // NativeView NativeView
570
- if (__DEV__ && Platform.OS !== 'web') {
571
- const REACT_NATIVE_VERSION = getReactNativeVersion();
572
- // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment
573
- const wrapType =
574
- REACT_NATIVE_VERSION.minor > 63 || REACT_NATIVE_VERSION.major > 0
575
- ? // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access
576
- ref._reactInternals.elementType
577
- : // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access
578
- ref._reactInternalFiber.elementType;
579
- // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment
580
- let instance =
581
- RNRenderer.findHostInstance_DEPRECATED(
582
- ref
583
- )._internalFiberInstanceHandleDEV;
584
-
585
- // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access
586
- while (instance && instance.elementType !== wrapType) {
587
- // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access
588
- if (instance.sibling) {
589
- throw new Error(
590
- 'GestureDetector has more than one native view as its children. This can happen if you are using a custom component that renders multiple views, like React.Fragment. You should wrap content of GestureDetector with a <View> or <Animated.View>.'
591
- );
592
- }
593
-
594
- // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment, @typescript-eslint/no-unsafe-member-access
595
- instance = instance.return;
596
- }
597
- }
598
- }
599
-
600
- const applyUserSelectProp = (
601
- userSelect: UserSelect,
602
- gesture: ComposedGesture | GestureType
603
- ): void => {
604
- for (const g of gesture.toGestureArray()) {
605
- g.config.userSelect = userSelect;
606
- }
607
- };
608
-
609
- const applyEnableContextMenuProp = (
610
- enableContextMenu: boolean,
611
- gesture: ComposedGesture | GestureType
612
- ): void => {
613
- for (const g of gesture.toGestureArray()) {
614
- g.config.enableContextMenu = enableContextMenu;
615
- }
616
- };
617
-
618
- const applyTouchActionProp = (
619
- touchAction: TouchAction,
620
- gesture: ComposedGesture | GestureType
621
- ): void => {
622
- for (const g of gesture.toGestureArray()) {
623
- g.config.touchAction = touchAction;
624
- }
625
- };
626
-
627
- interface GestureDetectorProps {
628
- /**
629
- * A gesture object containing the configuration and callbacks.
630
- * Can be any of:
631
- * - base gestures (`Tap`, `Pan`, ...)
632
- * - `ComposedGesture` (`Race`, `Simultaneous`, `Exclusive`)
633
- */
634
- gesture: ComposedGesture | GestureType;
635
- children?: React.ReactNode;
636
-
637
- /**
638
- * #### Web only
639
- * This parameter allows to specify which `userSelect` property should be applied to underlying view.
640
- * Possible values are `"none" | "auto" | "text"`. Default value is set to `"none"`.
641
- */
642
- userSelect?: UserSelect;
643
- /**
644
- * #### Web only
645
- * Specifies whether context menu should be enabled after clicking on underlying view with right mouse button.
646
- * Default value is set to `false`.
647
- */
648
- enableContextMenu?: boolean;
649
- /**
650
- * #### Web only
651
- * This parameter allows to specify which `touchAction` property should be applied to underlying view.
652
- * Supports all CSS touch-action values (e.g. `"none"`, `"pan-y"`). Default value is set to `"none"`.
653
- */
654
- touchAction?: TouchAction;
655
- }
656
- interface GestureDetectorState {
657
- firstRender: boolean;
658
- viewRef: React.Component | null;
659
- previousViewTag: number;
660
- forceReattach: boolean;
661
- }
662
-
663
- /**
664
- * `GestureDetector` is responsible for creating and updating native gesture handlers based on the config of provided gesture.
665
- *
666
- * ### Props
667
- * - `gesture`
668
- * - `userSelect` (**Web only**)
669
- * - `enableContextMenu` (**Web only**)
670
- * - `touchAction` (**Web only**)
671
- *
672
- * ### Remarks
673
- * - Gesture Detector will use first native view in its subtree to recognize gestures, however if this view is used only to group its children it may get automatically collapsed.
674
- * - Using the same instance of a gesture across multiple Gesture Detectors is not possible.
675
- *
676
- * @see https://docs.swmansion.com/react-native-gesture-handler/docs/gestures/gesture-detector
677
- */
678
- export const GestureDetector = (props: GestureDetectorProps) => {
679
- const rootViewContext = useContext(GestureHandlerRootViewContext);
680
- if (__DEV__ && !rootViewContext && !isJestEnv() && Platform.OS !== 'web') {
681
- throw new Error(
682
- 'GestureDetector must be used as a descendant of GestureHandlerRootView. Otherwise the gestures will not be recognized. See https://docs.swmansion.com/react-native-gesture-handler/docs/installation for more details.'
683
- );
684
- }
685
-
686
- const gestureConfig = props.gesture;
687
-
688
- if (props.userSelect) {
689
- applyUserSelectProp(props.userSelect, gestureConfig);
690
- }
691
-
692
- if (props.enableContextMenu !== undefined) {
693
- applyEnableContextMenuProp(props.enableContextMenu, gestureConfig);
694
- }
695
-
696
- if (props.touchAction !== undefined) {
697
- applyTouchActionProp(props.touchAction, gestureConfig);
698
- }
699
-
700
- const gesture = gestureConfig.toGestureArray();
701
- const useReanimatedHook = gesture.some((g) => g.shouldUseReanimated);
702
-
703
- // store state in ref to prevent unnecessary renders
704
- const state = useRef<GestureDetectorState>({
705
- firstRender: true,
706
- viewRef: null,
707
- previousViewTag: -1,
708
- forceReattach: false,
709
- }).current;
710
- const mountedRef = useRef(false);
711
- const webEventHandlersRef = useRef<WebEventHandler>({
712
- onGestureHandlerEvent: (e: HandlerStateChangeEvent<unknown>) => {
713
- onGestureHandlerEvent(e.nativeEvent);
714
- },
715
- onGestureHandlerStateChange: isNewWebImplementationEnabled()
716
- ? (e: HandlerStateChangeEvent<unknown>) => {
717
- onGestureHandlerEvent(e.nativeEvent);
718
- }
719
- : undefined,
720
- });
721
-
722
- const [renderState, setRenderState] = useState(false);
723
- function forceRender() {
724
- setRenderState(!renderState);
725
- }
726
-
727
- const preparedGesture = React.useRef<GestureConfigReference>({
728
- config: gesture,
729
- animatedEventHandler: null,
730
- animatedHandlers: null,
731
- firstExecution: true,
732
- useReanimatedHook: useReanimatedHook,
733
- }).current;
734
-
735
- if (useReanimatedHook !== preparedGesture.useReanimatedHook) {
736
- throw new Error(
737
- tagMessage(
738
- 'You cannot change the thread the callbacks are ran on while the app is running'
739
- )
740
- );
741
- }
742
-
743
- function onHandlersUpdate(skipConfigUpdate?: boolean) {
744
- // if the underlying view has changed we need to reattach handlers to the new view
745
- const viewTag = findNodeHandle(state.viewRef) as number;
746
- const forceReattach = viewTag !== state.previousViewTag;
747
-
748
- if (forceReattach || needsToReattach(preparedGesture, gesture)) {
749
- validateDetectorChildren(state.viewRef);
750
- dropHandlers(preparedGesture);
751
- attachHandlers({
752
- preparedGesture,
753
- gestureConfig,
754
- gesture,
755
- webEventHandlersRef,
756
- viewTag,
757
- mountedRef,
758
- });
759
-
760
- state.previousViewTag = viewTag;
761
- state.forceReattach = forceReattach;
762
- if (forceReattach) {
763
- forceRender();
764
- }
765
- } else if (!skipConfigUpdate) {
766
- updateHandlers(preparedGesture, gestureConfig, gesture, mountedRef);
767
- }
768
- }
769
-
770
- // Reanimated event should be rebuilt only when gestures are reattached, otherwise
771
- // config update will be enough as all necessary items are stored in shared values anyway
772
- const needsToRebuildReanimatedEvent =
773
- preparedGesture.firstExecution ||
774
- needsToReattach(preparedGesture, gesture) ||
775
- state.forceReattach;
776
-
777
- state.forceReattach = false;
778
-
779
- if (preparedGesture.firstExecution) {
780
- gestureConfig.initialize();
781
- }
782
-
783
- if (useReanimatedHook) {
784
- // Whether animatedGesture or gesture is used shouldn't change while the app is running
785
- // eslint-disable-next-line react-hooks/rules-of-hooks
786
- useAnimatedGesture(preparedGesture, needsToRebuildReanimatedEvent);
787
- }
788
-
789
- useEffect(() => {
790
- const viewTag = findNodeHandle(state.viewRef) as number;
791
- state.firstRender = true;
792
- mountedRef.current = true;
793
-
794
- validateDetectorChildren(state.viewRef);
795
-
796
- attachHandlers({
797
- preparedGesture,
798
- gestureConfig,
799
- gesture,
800
- webEventHandlersRef,
801
- viewTag,
802
- mountedRef,
803
- });
804
-
805
- return () => {
806
- mountedRef.current = false;
807
- dropHandlers(preparedGesture);
808
- };
809
- }, []);
810
-
811
- useEffect(() => {
812
- if (!state.firstRender) {
813
- onHandlersUpdate();
814
- } else {
815
- state.firstRender = false;
816
- }
817
- }, [props]);
818
-
819
- const refFunction = (ref: unknown) => {
820
- if (ref !== null) {
821
- // @ts-ignore Just setting the view ref
822
- state.viewRef = ref;
823
-
824
- // if it's the first render, also set the previousViewTag to prevent reattaching gestures when not needed
825
- if (state.previousViewTag === -1) {
826
- state.previousViewTag = findNodeHandle(state.viewRef) as number;
827
- }
828
-
829
- // pass true as `skipConfigUpdate`, here we only want to trigger the eventual reattaching of handlers
830
- // in case the view has changed, while config update would be handled be the `useEffect` above
831
- onHandlersUpdate(true);
832
-
833
- if (isFabric() && global.isFormsStackingContext) {
834
- const node = getShadowNodeFromRef(ref);
835
- if (global.isFormsStackingContext(node) === false) {
836
- console.error(
837
- tagMessage(
838
- 'GestureDetector has received a child that may get view-flattened. ' +
839
- '\nTo prevent it from misbehaving you need to wrap the child with a `<View collapsable={false}>`.'
840
- )
841
- );
842
- }
843
- }
844
- }
845
- };
846
-
847
- if (useReanimatedHook) {
848
- return (
849
- <AnimatedWrap
850
- ref={refFunction}
851
- onGestureHandlerEvent={preparedGesture.animatedEventHandler}>
852
- {props.children}
853
- </AnimatedWrap>
854
- );
855
- } else {
856
- return <Wrap ref={refFunction}>{props.children}</Wrap>;
857
- }
858
- };
859
-
860
- class Wrap extends React.Component<{
861
- onGestureHandlerEvent?: unknown;
862
- // implicit `children` prop has been removed in @types/react^18.0.0
863
- children?: React.ReactNode;
864
- }> {
865
- render() {
866
- try {
867
- // I don't think that fighting with types over such a simple function is worth it
868
- // The only thing it does is add 'collapsable: false' to the child component
869
- // to make sure it is in the native view hierarchy so the detector can find
870
- // correct viewTag to attach to.
871
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
872
- const child: any = React.Children.only(this.props.children);
873
- return React.cloneElement(
874
- child,
875
- { collapsable: false },
876
- // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access
877
- child.props.children
878
- );
879
- } catch (e) {
880
- throw new Error(
881
- tagMessage(
882
- `GestureDetector got more than one view as a child. If you want the gesture to work on multiple views, wrap them with a common parent and attach the gesture to that view.`
883
- )
884
- );
885
- }
886
- }
887
- }
888
-
889
- const AnimatedWrap = Reanimated?.default?.createAnimatedComponent(Wrap) ?? Wrap;