@office-iss/react-native-win32 0.76.2 → 0.77.0-preview.1

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 (248) hide show
  1. package/.eslintignore +1 -1
  2. package/.flowconfig +5 -1
  3. package/CHANGELOG.json +164 -53
  4. package/CHANGELOG.md +74 -28
  5. package/Libraries/ActionSheetIOS/ActionSheetIOS.d.ts +1 -0
  6. package/Libraries/ActionSheetIOS/ActionSheetIOS.js +13 -0
  7. package/Libraries/Animated/AnimatedEvent.js +1 -1
  8. package/Libraries/Animated/AnimatedImplementation.js +2 -2
  9. package/Libraries/Animated/NativeAnimatedAllowlist.js +20 -9
  10. package/Libraries/Animated/animations/Animation.js +60 -25
  11. package/Libraries/Animated/animations/DecayAnimation.js +26 -38
  12. package/Libraries/Animated/animations/SpringAnimation.js +33 -39
  13. package/Libraries/Animated/animations/TimingAnimation.js +34 -42
  14. package/Libraries/Animated/components/AnimatedFlatList.js +1 -1
  15. package/Libraries/Animated/components/AnimatedSectionList.js +3 -1
  16. package/Libraries/Animated/createAnimatedComponent.js +60 -33
  17. package/Libraries/Animated/nodes/AnimatedColor.js +1 -1
  18. package/Libraries/Animated/nodes/AnimatedInterpolation.js +1 -1
  19. package/Libraries/Animated/nodes/AnimatedNode.js +39 -45
  20. package/Libraries/Animated/nodes/AnimatedObject.js +13 -3
  21. package/Libraries/Animated/nodes/AnimatedProps.js +96 -46
  22. package/Libraries/Animated/nodes/AnimatedStyle.js +108 -39
  23. package/Libraries/Animated/nodes/AnimatedTransform.js +56 -23
  24. package/Libraries/Animated/nodes/AnimatedValue.js +1 -1
  25. package/Libraries/Animated/nodes/AnimatedWithChildren.js +1 -3
  26. package/Libraries/Animated/useAnimatedProps.js +41 -35
  27. package/Libraries/Components/AccessibilityInfo/AccessibilityInfo.d.ts +19 -3
  28. package/Libraries/Components/AccessibilityInfo/AccessibilityInfo.js +77 -5
  29. package/Libraries/Components/AccessibilityInfo/AccessibilityInfo.win32.js +82 -5
  30. package/Libraries/Components/ActivityIndicator/ActivityIndicator.js +4 -4
  31. package/Libraries/Components/Button.js +9 -4
  32. package/Libraries/Components/Button.win32.js +12 -4
  33. package/Libraries/Components/DrawerAndroid/DrawerLayoutAndroid.js +3 -1
  34. package/Libraries/Components/Keyboard/KeyboardAvoidingView.js +7 -0
  35. package/Libraries/Components/Pressable/Pressable.js +4 -4
  36. package/Libraries/Components/Pressable/Pressable.win32.js +4 -4
  37. package/Libraries/Components/ProgressBarAndroid/ProgressBarAndroid.android.js +13 -7
  38. package/Libraries/Components/RefreshControl/__mocks__/RefreshControlMock.js +3 -2
  39. package/Libraries/Components/SafeAreaView/SafeAreaView.js +4 -4
  40. package/Libraries/Components/SafeAreaView/SafeAreaView.win32.js +4 -4
  41. package/Libraries/Components/ScrollView/AndroidHorizontalScrollViewNativeComponent.js +0 -1
  42. package/Libraries/Components/ScrollView/ScrollView.js +49 -88
  43. package/Libraries/Components/ScrollView/ScrollViewCommands.js +1 -1
  44. package/Libraries/Components/ScrollView/ScrollViewContext.js +2 -0
  45. package/Libraries/Components/ScrollView/ScrollViewNativeComponent.js +0 -2
  46. package/Libraries/Components/ScrollView/ScrollViewStickyHeader.js +8 -9
  47. package/Libraries/Components/Switch/Switch.js +8 -6
  48. package/Libraries/Components/TextInput/InputAccessoryView.js +1 -1
  49. package/Libraries/Components/TextInput/RCTMultilineTextInputNativeComponent.js +4 -4
  50. package/Libraries/Components/TextInput/RCTSingelineTextInputNativeComponent.js +6 -4
  51. package/Libraries/Components/TextInput/RCTTextInputViewConfig.js +2 -1
  52. package/Libraries/Components/TextInput/TextInput.d.ts +27 -4
  53. package/Libraries/Components/TextInput/TextInput.flow.js +36 -19
  54. package/Libraries/Components/TextInput/TextInput.js +37 -13
  55. package/Libraries/Components/TextInput/TextInput.win32.js +40 -15
  56. package/Libraries/Components/TextInput/TextInputState.js +11 -13
  57. package/Libraries/Components/TextInput/TextInputState.win32.js +13 -16
  58. package/Libraries/Components/TextInput/Win32TextInputNativeComponent.js +3 -0
  59. package/Libraries/Components/Touchable/BoundingDimensions.js +11 -3
  60. package/Libraries/Components/Touchable/Position.js +7 -2
  61. package/Libraries/Components/Touchable/Touchable.js +4 -0
  62. package/Libraries/Components/Touchable/Touchable.win32.js +4 -0
  63. package/Libraries/Components/Touchable/TouchableBounce.js +6 -2
  64. package/Libraries/Components/Touchable/TouchableHighlight.js +5 -5
  65. package/Libraries/Components/Touchable/TouchableOpacity.js +6 -5
  66. package/Libraries/Components/Touchable/TouchableWithoutFeedback.js +1 -2
  67. package/Libraries/Components/View/ReactNativeStyleAttributes.js +6 -1
  68. package/Libraries/Components/View/View.js +4 -4
  69. package/Libraries/Components/View/View.win32.js +4 -4
  70. package/Libraries/Components/View/ViewNativeComponent.js +6 -98
  71. package/Libraries/Components/View/ViewPropTypes.d.ts +7 -0
  72. package/Libraries/Components/View/ViewPropTypes.js +0 -3
  73. package/Libraries/Components/View/ViewPropTypes.win32.js +0 -3
  74. package/Libraries/Components/View/ViewWin32.js +1 -0
  75. package/Libraries/Core/ExceptionsManager.js +50 -29
  76. package/Libraries/Core/ReactNativeVersion.js +3 -3
  77. package/Libraries/Core/__mocks__/NativeExceptionsManager.js +0 -1
  78. package/Libraries/Core/setUpBatchedBridge.js +1 -10
  79. package/Libraries/Core/setUpDeveloperTools.js +1 -5
  80. package/Libraries/Core/setUpErrorHandling.js +20 -18
  81. package/Libraries/Core/setUpReactDevTools.js +107 -8
  82. package/Libraries/Core/setUpSegmentFetcher.js +1 -0
  83. package/Libraries/Core/setUpTimers.js +21 -18
  84. package/Libraries/Debugging/DebuggingOverlay.js +4 -5
  85. package/Libraries/Image/AssetSourceResolver.js +12 -1
  86. package/Libraries/Image/Image.android.js +1 -5
  87. package/Libraries/Image/Image.d.ts +20 -29
  88. package/Libraries/Image/Image.ios.js +0 -2
  89. package/Libraries/Image/Image.win32.js +0 -2
  90. package/Libraries/Image/ImageBackground.js +2 -5
  91. package/Libraries/Image/ImageProps.js +7 -6
  92. package/Libraries/Image/ImageResizeMode.d.ts +8 -1
  93. package/Libraries/Image/ImageResizeMode.js +4 -1
  94. package/Libraries/Image/ImageSource.d.ts +0 -2
  95. package/Libraries/Image/ImageSource.js +0 -2
  96. package/Libraries/Image/ImageTypes.flow.js +11 -9
  97. package/Libraries/Image/ImageUtils.js +6 -3
  98. package/Libraries/Image/ImageViewNativeComponent.js +5 -3
  99. package/Libraries/Inspector/Inspector.js +1 -0
  100. package/Libraries/Inspector/Inspector.win32.js +2 -1
  101. package/Libraries/Inspector/NetworkOverlay.js +4 -0
  102. package/Libraries/Inspector/ReactDevToolsOverlay.js +8 -14
  103. package/Libraries/Inspector/getInspectorDataForViewAtPoint.js +3 -5
  104. package/Libraries/Interaction/InteractionManager.js +6 -1
  105. package/Libraries/Interaction/InteractionManagerStub.js +176 -0
  106. package/Libraries/Interaction/TouchHistoryMath.js +22 -19
  107. package/Libraries/JSInspector/NetworkAgent.js +1 -1
  108. package/Libraries/Lists/FlatList.d.ts +1 -2
  109. package/Libraries/Lists/FlatList.js +2 -2
  110. package/Libraries/Lists/SectionListModern.js +7 -7
  111. package/Libraries/Lists/__flowtests__/FlatList-flowtest.js +2 -2
  112. package/Libraries/Lists/__flowtests__/SectionList-flowtest.js +1 -1
  113. package/Libraries/LogBox/Data/LogBoxData.js +3 -3
  114. package/Libraries/LogBox/LogBox.js +18 -5
  115. package/Libraries/LogBox/LogBoxInspectorContainer.js +1 -1
  116. package/Libraries/LogBox/LogBoxNotificationContainer.js +2 -2
  117. package/Libraries/LogBox/UI/AnsiHighlight.js +26 -17
  118. package/Libraries/LogBox/UI/LogBoxInspectorCodeFrame.js +6 -1
  119. package/Libraries/LogBox/UI/LogBoxInspectorCodeFrame.win32.js +6 -1
  120. package/Libraries/LogBox/UI/LogBoxInspectorHeader.js +1 -1
  121. package/Libraries/LogBox/UI/LogBoxInspectorHeader.win32.js +1 -1
  122. package/Libraries/LogBox/UI/LogBoxInspectorStackFrames.js +1 -1
  123. package/Libraries/LogBox/UI/LogBoxMessage.js +2 -2
  124. package/Libraries/Modal/Modal.d.ts +12 -0
  125. package/Libraries/Modal/Modal.js +31 -4
  126. package/Libraries/NativeComponent/BaseViewConfig.android.js +72 -1
  127. package/Libraries/NativeComponent/BaseViewConfig.ios.js +2 -1
  128. package/Libraries/NativeComponent/BaseViewConfig.win32.js +3 -11
  129. package/Libraries/NativeComponent/NativeComponentRegistry.js +3 -3
  130. package/Libraries/NativeComponent/StaticViewConfigValidator.js +0 -1
  131. package/Libraries/Network/XHRInterceptor.js +63 -14
  132. package/Libraries/Network/XMLHttpRequest.js +26 -1
  133. package/Libraries/NewAppScreen/components/HermesBadge.js +1 -1
  134. package/Libraries/PermissionsAndroid/PermissionsAndroid.d.ts +49 -2
  135. package/Libraries/PermissionsAndroid/PermissionsAndroid.js +4 -4
  136. package/Libraries/Pressability/HoverState.js +2 -0
  137. package/Libraries/Pressability/HoverState.win32.js +2 -0
  138. package/Libraries/Pressability/Pressability.js +2 -3
  139. package/Libraries/Pressability/Pressability.win32.js +2 -3
  140. package/Libraries/Pressability/usePressability.js +4 -1
  141. package/Libraries/ReactNative/AppContainer.js +1 -1
  142. package/Libraries/ReactNative/AppRegistry.js +1 -11
  143. package/Libraries/ReactNative/DisplayMode.js +1 -1
  144. package/Libraries/ReactNative/ReactFabricPublicInstance/ReactFabricHostComponent.js +2 -3
  145. package/Libraries/ReactNative/RendererImplementation.js +18 -17
  146. package/Libraries/ReactNative/getCachedComponentWithDebugName.js +1 -3
  147. package/Libraries/ReactNative/renderApplication.js +9 -8
  148. package/Libraries/ReactNative/requireNativeComponent.js +5 -2
  149. package/Libraries/Renderer/shims/ReactFabric.js +3 -3
  150. package/Libraries/Renderer/shims/ReactFeatureFlags.js +2 -2
  151. package/Libraries/Renderer/shims/ReactNative.js +3 -3
  152. package/Libraries/Renderer/shims/ReactNativeTypes.js +22 -35
  153. package/Libraries/Renderer/shims/ReactNativeViewConfigRegistry.js +5 -6
  154. package/Libraries/Renderer/shims/createReactNativeComponentClass.js +2 -2
  155. package/Libraries/StyleSheet/StyleSheet.js +7 -1
  156. package/Libraries/StyleSheet/StyleSheet.win32.js +7 -1
  157. package/Libraries/StyleSheet/StyleSheetTypes.d.ts +13 -2
  158. package/Libraries/StyleSheet/StyleSheetTypes.js +24 -6
  159. package/Libraries/StyleSheet/processBackgroundImage.js +87 -110
  160. package/Libraries/StyleSheet/processTransform.js +3 -34
  161. package/Libraries/Text/Text.js +248 -249
  162. package/Libraries/Text/Text.win32.js +282 -295
  163. package/Libraries/Text/TextNativeComponent.js +0 -1
  164. package/Libraries/TurboModule/TurboModuleRegistry.js +5 -5
  165. package/Libraries/Types/CoreEventTypes.d.ts +3 -10
  166. package/Libraries/Types/CoreEventTypes.js +4 -6
  167. package/Libraries/Types/CoreEventTypes.win32.js +4 -6
  168. package/Libraries/Utilities/Appearance.js +3 -1
  169. package/Libraries/Utilities/BackHandler.android.js +6 -18
  170. package/Libraries/Utilities/BackHandler.d.ts +0 -4
  171. package/Libraries/Utilities/BackHandler.ios.js +0 -7
  172. package/Libraries/Utilities/BackHandler.win32.js +6 -18
  173. package/Libraries/Utilities/FocusManager.win32.js +1 -1
  174. package/Libraries/Utilities/HMRClient.js +3 -4
  175. package/Libraries/Utilities/Platform.flow.js +2 -2
  176. package/Libraries/Utilities/Platform.flow.win32.js +3 -2
  177. package/Libraries/Utilities/__mocks__/BackHandler.js +3 -8
  178. package/Libraries/Utilities/codegenNativeComponent.js +1 -1
  179. package/Libraries/Utilities/useMergeRefs.js +26 -7
  180. package/Libraries/WebSocket/WebSocketEvent.js +4 -1
  181. package/Libraries/WebSocket/WebSocketInterceptor.js +31 -13
  182. package/Libraries/__flowtests__/ReactNativeTypes-flowtest.js +6 -5
  183. package/Libraries/promiseRejectionTrackingOptions.js +1 -1
  184. package/index.js +10 -3
  185. package/index.win32.js +10 -3
  186. package/jest/setup.js +36 -1
  187. package/overrides.json +37 -37
  188. package/package.json +20 -20
  189. package/src/private/animated/NativeAnimatedHelper.js +18 -16
  190. package/src/private/animated/NativeAnimatedHelper.win32.js +18 -15
  191. package/src/private/animated/useAnimatedPropsMemo.js +348 -0
  192. package/src/private/components/HScrollViewNativeComponents.js +1 -27
  193. package/src/private/components/SafeAreaView_INTERNAL_DO_NOT_USE.js +11 -8
  194. package/src/private/components/VScrollViewNativeComponents.js +2 -25
  195. package/src/private/debugging/ReactDevToolsSettingsManager.android.js +20 -0
  196. package/src/private/debugging/ReactDevToolsSettingsManager.ios.js +30 -0
  197. package/src/private/debugging/ReactDevToolsSettingsManager.win32.js +20 -0
  198. package/src/private/{fusebox → debugging}/setUpFuseboxReactDevToolsDispatcher.js +6 -0
  199. package/src/private/devmenu/DevMenu.d.ts +20 -0
  200. package/src/private/devmenu/DevMenu.js +31 -0
  201. package/src/private/featureflags/ReactNativeFeatureFlags.js +95 -86
  202. package/src/private/featureflags/ReactNativeFeatureFlagsBase.js +8 -2
  203. package/src/private/featureflags/specs/NativeReactNativeFeatureFlags.js +17 -19
  204. package/src/private/fusebox/specs/NativeReactDevToolsRuntimeSettingsModule.js +34 -0
  205. package/src/private/setup/setUpDOM.js +14 -6
  206. package/src/private/setup/setUpMutationObserver.js +5 -0
  207. package/src/private/specs/components/AndroidHorizontalScrollContentViewNativeComponent.js +1 -0
  208. package/src/private/specs/components/RCTModalHostViewNativeComponent.js +8 -0
  209. package/src/private/specs/modules/NativeAccessibilityInfo.js +9 -0
  210. package/src/private/specs/modules/NativeAccessibilityInfoWin32.js +9 -0
  211. package/src/private/specs/modules/NativeAccessibilityManager.js +4 -0
  212. package/src/private/specs/modules/NativeActionSheetManager.js +2 -0
  213. package/src/private/specs/modules/NativeAppearance.js +4 -10
  214. package/src/private/specs/modules/NativeExceptionsManager.js +0 -12
  215. package/src/private/specs/modules/{NativeDevToolsSettingsManager.js → NativeReactDevToolsSettingsManager.js} +3 -5
  216. package/src/private/webapis/dom/geometry/DOMRect.js +2 -2
  217. package/src/private/webapis/dom/geometry/DOMRectReadOnly.js +2 -2
  218. package/src/private/webapis/dom/nodes/ReactNativeElement.js +2 -3
  219. package/src/private/webapis/intersectionobserver/IntersectionObserver.js +102 -11
  220. package/src/private/webapis/intersectionobserver/IntersectionObserverEntry.js +26 -0
  221. package/src/private/webapis/intersectionobserver/IntersectionObserverManager.js +1 -0
  222. package/src/private/webapis/intersectionobserver/specs/NativeIntersectionObserver.js +1 -0
  223. package/src/private/webapis/intersectionobserver/specs/__mocks__/NativeIntersectionObserver.js +9 -0
  224. package/src/private/webapis/performance/EventTiming.js +13 -8
  225. package/src/private/webapis/performance/Performance.js +66 -73
  226. package/src/private/webapis/performance/PerformanceEntry.js +2 -5
  227. package/src/private/webapis/performance/PerformanceObserver.js +65 -164
  228. package/src/private/webapis/performance/RawPerformanceEntry.js +1 -1
  229. package/src/private/webapis/performance/UserTiming.js +11 -7
  230. package/src/private/webapis/performance/Utilities.js +18 -0
  231. package/src/private/webapis/performance/specs/NativePerformance.js +71 -2
  232. package/src/private/webapis/performance/specs/__mocks__/NativePerformanceMock.js +267 -0
  233. package/src-win/Libraries/Components/AccessibilityInfo/AccessibilityInfo.d.ts +19 -3
  234. package/src-win/Libraries/Components/View/ViewPropTypes.d.ts +7 -0
  235. package/types/index.d.ts +1 -1
  236. package/types/public/ReactNativeTypes.d.ts +4 -8
  237. package/Libraries/DevToolsSettings/DevToolsSettingsManager.android.js +0 -35
  238. package/Libraries/DevToolsSettings/DevToolsSettingsManager.d.ts +0 -20
  239. package/Libraries/DevToolsSettings/DevToolsSettingsManager.ios.js +0 -49
  240. package/Libraries/DevToolsSettings/DevToolsSettingsManager.win32.js +0 -35
  241. package/Libraries/DevToolsSettings/NativeDevToolsSettingsManager.js +0 -13
  242. package/Libraries/ReactNative/ReactFabricInternals.js +0 -17
  243. package/src/private/components/useSyncOnScroll.js +0 -48
  244. package/src/private/webapis/performance/specs/NativePerformanceObserver.js +0 -61
  245. package/src/private/webapis/performance/specs/__mocks__/NativePerformance.js +0 -67
  246. package/src/private/webapis/performance/specs/__mocks__/NativePerformanceObserver.js +0 -127
  247. package/types/experimental.d.ts +0 -59
  248. /package/src/private/{fusebox → debugging}/FuseboxSessionObserver.js +0 -0
@@ -8,6 +8,8 @@
8
8
  * @format
9
9
  */
10
10
 
11
+ import type {AnimatedPropsAllowlist} from './nodes/AnimatedProps';
12
+
11
13
  import composeStyles from '../../src/private/styles/composeStyles';
12
14
  import View from '../Components/View/View';
13
15
  import useMergeRefs from '../Utilities/useMergeRefs';
@@ -25,46 +27,71 @@ export type AnimatedProps<Props: {...}> = {
25
27
  }>)]: any,
26
28
  };
27
29
 
28
- export type AnimatedComponentType<
30
+ // We could use a mapped type here to introduce acceptable Animated variants
31
+ // of properties, instead of doing so in the core StyleSheetTypes
32
+ // Inexact Props are not supported, they'll be made exact here.
33
+ export type StrictAnimatedProps<Props: {...}> = $ReadOnly<{
34
+ ...$Exact<Props>,
35
+ passthroughAnimatedPropExplicitValues?: ?Props,
36
+ }>;
37
+
38
+ export type AnimatedComponentType<Props: {...}, +Instance = mixed> = component(
39
+ ref: React.RefSetter<Instance>,
40
+ ...AnimatedProps<Props>
41
+ );
42
+
43
+ export type StrictAnimatedComponentType<
29
44
  Props: {...},
30
45
  +Instance = mixed,
31
- > = React.AbstractComponent<AnimatedProps<Props>, Instance>;
46
+ > = component(ref: React.RefSetter<Instance>, ...StrictAnimatedProps<Props>);
32
47
 
33
48
  export default function createAnimatedComponent<TProps: {...}, TInstance>(
34
- Component: React.AbstractComponent<TProps, TInstance>,
49
+ Component: component(ref: React.RefSetter<TInstance>, ...TProps),
35
50
  ): AnimatedComponentType<TProps, TInstance> {
36
- const AnimatedComponent = React.forwardRef<AnimatedProps<TProps>, TInstance>(
37
- (props, forwardedRef) => {
38
- const [reducedProps, callbackRef] = useAnimatedProps<TProps, TInstance>(
39
- // $FlowFixMe[incompatible-call]
40
- props,
41
- );
42
- const ref = useMergeRefs<TInstance>(callbackRef, forwardedRef);
51
+ return unstable_createAnimatedComponentWithAllowlist(Component, null);
52
+ }
53
+
54
+ export function unstable_createAnimatedComponentWithAllowlist<
55
+ TProps: {...},
56
+ TInstance,
57
+ >(
58
+ Component: component(ref: React.RefSetter<TInstance>, ...TProps),
59
+ allowlist: ?AnimatedPropsAllowlist,
60
+ ): StrictAnimatedComponentType<TProps, TInstance> {
61
+ const AnimatedComponent = React.forwardRef<
62
+ StrictAnimatedProps<TProps>,
63
+ TInstance,
64
+ >((props, forwardedRef) => {
65
+ const [reducedProps, callbackRef] = useAnimatedProps<TProps, TInstance>(
66
+ // $FlowFixMe[incompatible-call]
67
+ props,
68
+ allowlist,
69
+ );
70
+ const ref = useMergeRefs<TInstance>(callbackRef, forwardedRef);
43
71
 
44
- // Some components require explicit passthrough values for animation
45
- // to work properly. For example, if an animated component is
46
- // transformed and Pressable, onPress will not work after transform
47
- // without these passthrough values.
48
- // $FlowFixMe[prop-missing]
49
- const {passthroughAnimatedPropExplicitValues, style} = reducedProps;
50
- const passthroughStyle = passthroughAnimatedPropExplicitValues?.style;
51
- const mergedStyle = useMemo(
52
- () => composeStyles(style, passthroughStyle),
53
- [passthroughStyle, style],
54
- );
72
+ // Some components require explicit passthrough values for animation
73
+ // to work properly. For example, if an animated component is
74
+ // transformed and Pressable, onPress will not work after transform
75
+ // without these passthrough values.
76
+ // $FlowFixMe[prop-missing]
77
+ const {passthroughAnimatedPropExplicitValues, style} = reducedProps;
78
+ const passthroughStyle = passthroughAnimatedPropExplicitValues?.style;
79
+ const mergedStyle = useMemo(
80
+ () => composeStyles(style, passthroughStyle),
81
+ [passthroughStyle, style],
82
+ );
55
83
 
56
- // NOTE: It is important that `passthroughAnimatedPropExplicitValues` is
57
- // spread after `reducedProps` but before `style`.
58
- return (
59
- <Component
60
- {...reducedProps}
61
- {...passthroughAnimatedPropExplicitValues}
62
- style={mergedStyle}
63
- ref={ref}
64
- />
65
- );
66
- },
67
- );
84
+ // NOTE: It is important that `passthroughAnimatedPropExplicitValues` is
85
+ // spread after `reducedProps` but before `style`.
86
+ return (
87
+ <Component
88
+ {...reducedProps}
89
+ {...passthroughAnimatedPropExplicitValues}
90
+ style={mergedStyle}
91
+ ref={ref}
92
+ />
93
+ );
94
+ });
68
95
 
69
96
  AnimatedComponent.displayName = `Animated(${
70
97
  Component.displayName || 'Anonymous'
@@ -15,9 +15,9 @@ import type {ColorValue} from '../../StyleSheet/StyleSheet';
15
15
  import type {NativeColorValue} from '../../StyleSheet/StyleSheetTypes';
16
16
  import type {PlatformConfig} from '../AnimatedPlatformConfig';
17
17
 
18
+ import NativeAnimatedHelper from '../../../src/private/animated/NativeAnimatedHelper';
18
19
  import normalizeColor from '../../StyleSheet/normalizeColor';
19
20
  import {processColorObject} from '../../StyleSheet/PlatformColorValueTypes';
20
- import NativeAnimatedHelper from '../../../src/private/animated/NativeAnimatedHelper';
21
21
  import AnimatedValue, {flushValue} from './AnimatedValue';
22
22
  import AnimatedWithChildren from './AnimatedWithChildren';
23
23
 
@@ -15,11 +15,11 @@
15
15
  import type {PlatformConfig} from '../AnimatedPlatformConfig';
16
16
  import type AnimatedNode from './AnimatedNode';
17
17
 
18
+ import NativeAnimatedHelper from '../../../src/private/animated/NativeAnimatedHelper';
18
19
  import {validateInterpolation} from '../../../src/private/animated/NativeAnimatedValidation';
19
20
  import normalizeColor from '../../StyleSheet/normalizeColor';
20
21
  import processColor from '../../StyleSheet/processColor';
21
22
  import Easing from '../Easing';
22
- import NativeAnimatedHelper from '../../../src/private/animated/NativeAnimatedHelper';
23
23
  import AnimatedWithChildren from './AnimatedWithChildren';
24
24
  import invariant from 'invariant';
25
25
 
@@ -8,8 +8,6 @@
8
8
  * @format
9
9
  */
10
10
 
11
- 'use strict';
12
-
13
11
  import type {EventSubscription} from '../../vendor/emitter/EventEmitter';
14
12
  import type {PlatformConfig} from '../AnimatedPlatformConfig';
15
13
 
@@ -29,12 +27,11 @@ let _assertNativeAnimatedModule: ?() => void = () => {
29
27
  _assertNativeAnimatedModule = null;
30
28
  };
31
29
 
32
- // Note(vjeux): this would be better as an interface but flow doesn't
33
- // support them yet
34
30
  export default class AnimatedNode {
35
31
  #listeners: Map<string, ValueListenerCallback> = new Map();
32
+ #updateSubscription: ?EventSubscription = null;
33
+
36
34
  _platformConfig: ?PlatformConfig = undefined;
37
- __nativeAnimatedValueListener: ?EventSubscription = null;
38
35
  __attach(): void {}
39
36
  __detach(): void {
40
37
  this.removeAllListeners();
@@ -56,16 +53,17 @@ export default class AnimatedNode {
56
53
  /* Methods and props used by native Animated impl */
57
54
  __isNative: boolean = false;
58
55
  __nativeTag: ?number = undefined;
59
- __shouldUpdateListenersForNewNativeTag: boolean = false;
60
56
 
61
57
  __makeNative(platformConfig: ?PlatformConfig): void {
62
- if (!this.__isNative) {
63
- throw new Error('This node cannot be made a "native" animated node');
64
- }
58
+ // Subclasses are expected to set `__isNative` to true before this.
59
+ invariant(
60
+ this.__isNative,
61
+ 'This node cannot be made a "native" animated node',
62
+ );
65
63
 
66
64
  this._platformConfig = platformConfig;
67
65
  if (this.#listeners.size > 0) {
68
- this._startListeningToNativeValueUpdates();
66
+ this.#ensureUpdateSubscriptionExists();
69
67
  }
70
68
  }
71
69
 
@@ -80,7 +78,7 @@ export default class AnimatedNode {
80
78
  const id = String(_uniqueId++);
81
79
  this.#listeners.set(id, callback);
82
80
  if (this.__isNative) {
83
- this._startListeningToNativeValueUpdates();
81
+ this.#ensureUpdateSubscriptionExists();
84
82
  }
85
83
  return id;
86
84
  }
@@ -94,7 +92,7 @@ export default class AnimatedNode {
94
92
  removeListener(id: string): void {
95
93
  this.#listeners.delete(id);
96
94
  if (this.__isNative && this.#listeners.size === 0) {
97
- this._stopListeningForNativeValueUpdates();
95
+ this.#updateSubscription?.remove();
98
96
  }
99
97
  }
100
98
 
@@ -106,7 +104,7 @@ export default class AnimatedNode {
106
104
  removeAllListeners(): void {
107
105
  this.#listeners.clear();
108
106
  if (this.__isNative) {
109
- this._stopListeningForNativeValueUpdates();
107
+ this.#updateSubscription?.remove();
110
108
  }
111
109
  }
112
110
 
@@ -114,33 +112,36 @@ export default class AnimatedNode {
114
112
  return this.#listeners.size > 0;
115
113
  }
116
114
 
117
- _startListeningToNativeValueUpdates() {
118
- if (
119
- this.__nativeAnimatedValueListener &&
120
- !this.__shouldUpdateListenersForNewNativeTag
121
- ) {
115
+ #ensureUpdateSubscriptionExists(): void {
116
+ if (this.#updateSubscription != null) {
122
117
  return;
123
118
  }
124
-
125
- if (this.__shouldUpdateListenersForNewNativeTag) {
126
- this.__shouldUpdateListenersForNewNativeTag = false;
127
- this._stopListeningForNativeValueUpdates();
128
- }
129
-
130
- startListeningToAnimatedNodeValue(this.__getNativeTag());
131
- this.__nativeAnimatedValueListener =
119
+ const nativeTag = this.__getNativeTag();
120
+ startListeningToAnimatedNodeValue(nativeTag);
121
+ const subscription: EventSubscription =
132
122
  NativeAnimatedHelper.nativeEventEmitter.addListener(
133
123
  'onAnimatedValueUpdate',
134
124
  data => {
135
- if (data.tag !== this.__getNativeTag()) {
136
- return;
125
+ if (data.tag === nativeTag) {
126
+ this.__onAnimatedValueUpdateReceived(data.value);
137
127
  }
138
- this.__onAnimatedValueUpdateReceived(data.value);
139
128
  },
140
129
  );
130
+
131
+ this.#updateSubscription = {
132
+ remove: () => {
133
+ // Only this function assigns to `this.#updateSubscription`.
134
+ if (this.#updateSubscription == null) {
135
+ return;
136
+ }
137
+ this.#updateSubscription = null;
138
+ subscription.remove();
139
+ stopListeningToAnimatedNodeValue(nativeTag);
140
+ },
141
+ };
141
142
  }
142
143
 
143
- __onAnimatedValueUpdateReceived(value: number) {
144
+ __onAnimatedValueUpdateReceived(value: number): void {
144
145
  this.__callListeners(value);
145
146
  }
146
147
 
@@ -151,16 +152,6 @@ export default class AnimatedNode {
151
152
  });
152
153
  }
153
154
 
154
- _stopListeningForNativeValueUpdates() {
155
- if (!this.__nativeAnimatedValueListener) {
156
- return;
157
- }
158
-
159
- this.__nativeAnimatedValueListener.remove();
160
- this.__nativeAnimatedValueListener = null;
161
- stopListeningToAnimatedNodeValue(this.__getNativeTag());
162
- }
163
-
164
155
  __getNativeTag(): number {
165
156
  let nativeTag = this.__nativeTag;
166
157
  if (nativeTag == null) {
@@ -181,7 +172,6 @@ export default class AnimatedNode {
181
172
  config.platformConfig = this._platformConfig;
182
173
  }
183
174
  NativeAnimatedHelper.API.createAnimatedNode(nativeTag, config);
184
- this.__shouldUpdateListenersForNewNativeTag = true;
185
175
  }
186
176
  return nativeTag;
187
177
  }
@@ -192,10 +182,6 @@ export default class AnimatedNode {
192
182
  );
193
183
  }
194
184
 
195
- toJSON(): any {
196
- return this.__getValue();
197
- }
198
-
199
185
  __getPlatformConfig(): ?PlatformConfig {
200
186
  return this._platformConfig;
201
187
  }
@@ -203,4 +189,12 @@ export default class AnimatedNode {
203
189
  __setPlatformConfig(platformConfig: ?PlatformConfig) {
204
190
  this._platformConfig = platformConfig;
205
191
  }
192
+
193
+ /**
194
+ * NOTE: This is intended to prevent `JSON.stringify` from throwing "cyclic
195
+ * structure" errors in React DevTools. Avoid depending on this!
196
+ */
197
+ toJSON(): mixed {
198
+ return this.__getValue();
199
+ }
206
200
  }
@@ -19,9 +19,11 @@ import * as React from 'react';
19
19
 
20
20
  const MAX_DEPTH = 5;
21
21
 
22
- /* $FlowIssue[incompatible-type-guard] - Flow does not know that the prototype
23
- and ReactElement checks preserve the type refinement of `value`. */
24
- function isPlainObject(value: mixed): value is $ReadOnly<{[string]: mixed}> {
22
+ export function isPlainObject(
23
+ value: mixed,
24
+ /* $FlowIssue[incompatible-type-guard] - Flow does not know that the prototype
25
+ and ReactElement checks preserve the type refinement of `value`. */
26
+ ): value is $ReadOnly<{[string]: mixed}> {
25
27
  return (
26
28
  value !== null &&
27
29
  typeof value === 'object' &&
@@ -109,6 +111,14 @@ export default class AnimatedObject extends AnimatedWithChildren {
109
111
  });
110
112
  }
111
113
 
114
+ __getValueWithStaticObject(staticObject: mixed): any {
115
+ const nodes = this.#nodes;
116
+ let index = 0;
117
+ // NOTE: We can depend on `this._value` and `staticObject` sharing a
118
+ // structure because of `useAnimatedPropsMemo`.
119
+ return mapAnimatedNodes(staticObject, () => nodes[index++].__getValue());
120
+ }
121
+
112
122
  __getAnimatedValue(): any {
113
123
  return mapAnimatedNodes(this._value, node => {
114
124
  return node.__getAnimatedValue();
@@ -8,42 +8,44 @@
8
8
  * @format
9
9
  */
10
10
 
11
- 'use strict';
12
-
13
11
  import type {PlatformConfig} from '../AnimatedPlatformConfig';
12
+ import type {AnimatedStyleAllowlist} from './AnimatedStyle';
14
13
 
14
+ import NativeAnimatedHelper from '../../../src/private/animated/NativeAnimatedHelper';
15
15
  import {findNodeHandle} from '../../ReactNative/RendererProxy';
16
16
  import {AnimatedEvent} from '../AnimatedEvent';
17
- import NativeAnimatedHelper from '../../../src/private/animated/NativeAnimatedHelper';
18
17
  import AnimatedNode from './AnimatedNode';
19
18
  import AnimatedObject from './AnimatedObject';
20
19
  import AnimatedStyle from './AnimatedStyle';
21
20
  import invariant from 'invariant';
22
21
 
22
+ export type AnimatedPropsAllowlist = $ReadOnly<{
23
+ style?: ?AnimatedStyleAllowlist,
24
+ [string]: true,
25
+ }>;
26
+
23
27
  function createAnimatedProps(
24
- inputProps: Object,
25
- ): [$ReadOnlyArray<string>, $ReadOnlyArray<AnimatedNode>, Object] {
28
+ inputProps: {[string]: mixed},
29
+ allowlist: ?AnimatedPropsAllowlist,
30
+ ): [$ReadOnlyArray<string>, $ReadOnlyArray<AnimatedNode>, {[string]: mixed}] {
26
31
  const nodeKeys: Array<string> = [];
27
32
  const nodes: Array<AnimatedNode> = [];
28
- const props: Object = {};
33
+ const props: {[string]: mixed} = {};
29
34
 
30
35
  const keys = Object.keys(inputProps);
31
36
  for (let ii = 0, length = keys.length; ii < length; ii++) {
32
37
  const key = keys[ii];
33
38
  const value = inputProps[key];
34
39
 
35
- if (key === 'style') {
36
- const node = new AnimatedStyle(value);
37
- nodeKeys.push(key);
38
- nodes.push(node);
39
- props[key] = node;
40
- } else if (value instanceof AnimatedNode) {
41
- const node = value;
42
- nodeKeys.push(key);
43
- nodes.push(node);
44
- props[key] = node;
45
- } else {
46
- const node = AnimatedObject.from(value);
40
+ if (allowlist == null || Object.hasOwn(allowlist, key)) {
41
+ let node;
42
+ if (key === 'style') {
43
+ node = AnimatedStyle.from(value, allowlist?.style);
44
+ } else if (value instanceof AnimatedNode) {
45
+ node = value;
46
+ } else {
47
+ node = AnimatedObject.from(value);
48
+ }
47
49
  if (node == null) {
48
50
  props[key] = value;
49
51
  } else {
@@ -51,6 +53,20 @@ function createAnimatedProps(
51
53
  nodes.push(node);
52
54
  props[key] = node;
53
55
  }
56
+ } else {
57
+ if (__DEV__) {
58
+ // WARNING: This is a potentially expensive check that we should only
59
+ // do in development. Without this check in development, it might be
60
+ // difficult to identify which props need to be allowlisted.
61
+ if (AnimatedObject.from(inputProps[key]) != null) {
62
+ console.error(
63
+ `AnimatedProps: ${key} is not allowlisted for animation, but it ` +
64
+ 'contains AnimatedNode values; props allowing animation: ',
65
+ allowlist,
66
+ );
67
+ }
68
+ }
69
+ props[key] = value;
54
70
  }
55
71
  }
56
72
 
@@ -58,29 +74,32 @@ function createAnimatedProps(
58
74
  }
59
75
 
60
76
  export default class AnimatedProps extends AnimatedNode {
77
+ #animatedView: any = null;
78
+ #callback: () => void;
61
79
  #nodeKeys: $ReadOnlyArray<string>;
62
80
  #nodes: $ReadOnlyArray<AnimatedNode>;
81
+ #props: {[string]: mixed};
63
82
 
64
- _animatedView: any = null;
65
- _props: Object;
66
- _callback: () => void;
67
-
68
- constructor(inputProps: Object, callback: () => void) {
83
+ constructor(
84
+ inputProps: {[string]: mixed},
85
+ callback: () => void,
86
+ allowlist?: ?AnimatedPropsAllowlist,
87
+ ) {
69
88
  super();
70
- const [nodeKeys, nodes, props] = createAnimatedProps(inputProps);
89
+ const [nodeKeys, nodes, props] = createAnimatedProps(inputProps, allowlist);
71
90
  this.#nodeKeys = nodeKeys;
72
91
  this.#nodes = nodes;
73
- this._props = props;
74
- this._callback = callback;
92
+ this.#props = props;
93
+ this.#callback = callback;
75
94
  }
76
95
 
77
96
  __getValue(): Object {
78
- const props: {[string]: any | ((...args: any) => void)} = {};
97
+ const props: {[string]: mixed} = {};
79
98
 
80
- const keys = Object.keys(this._props);
99
+ const keys = Object.keys(this.#props);
81
100
  for (let ii = 0, length = keys.length; ii < length; ii++) {
82
101
  const key = keys[ii];
83
- const value = this._props[key];
102
+ const value = this.#props[key];
84
103
 
85
104
  if (value instanceof AnimatedNode) {
86
105
  props[key] = value.__getValue();
@@ -94,8 +113,33 @@ export default class AnimatedProps extends AnimatedNode {
94
113
  return props;
95
114
  }
96
115
 
116
+ /**
117
+ * Creates a new `props` object that contains the same props as the supplied
118
+ * `staticProps` object, except with animated nodes for any props that were
119
+ * created by this `AnimatedProps` instance.
120
+ */
121
+ __getValueWithStaticProps(staticProps: Object): Object {
122
+ const props: {[string]: mixed} = {...staticProps};
123
+
124
+ const keys = Object.keys(staticProps);
125
+ for (let ii = 0, length = keys.length; ii < length; ii++) {
126
+ const key = keys[ii];
127
+ const maybeNode = this.#props[key];
128
+
129
+ if (key === 'style' && maybeNode instanceof AnimatedStyle) {
130
+ props[key] = maybeNode.__getValueWithStaticStyle(staticProps.style);
131
+ } else if (maybeNode instanceof AnimatedNode) {
132
+ props[key] = maybeNode.__getValue();
133
+ } else if (maybeNode instanceof AnimatedEvent) {
134
+ props[key] = maybeNode.__getHandler();
135
+ }
136
+ }
137
+
138
+ return props;
139
+ }
140
+
97
141
  __getAnimatedValue(): Object {
98
- const props: {[string]: any} = {};
142
+ const props: {[string]: mixed} = {};
99
143
 
100
144
  const nodeKeys = this.#nodeKeys;
101
145
  const nodes = this.#nodes;
@@ -117,10 +161,10 @@ export default class AnimatedProps extends AnimatedNode {
117
161
  }
118
162
 
119
163
  __detach(): void {
120
- if (this.__isNative && this._animatedView) {
164
+ if (this.__isNative && this.#animatedView) {
121
165
  this.__disconnectAnimatedView();
122
166
  }
123
- this._animatedView = null;
167
+ this.#animatedView = null;
124
168
 
125
169
  const nodes = this.#nodes;
126
170
  for (let ii = 0, length = nodes.length; ii < length; ii++) {
@@ -132,7 +176,7 @@ export default class AnimatedProps extends AnimatedNode {
132
176
  }
133
177
 
134
178
  update(): void {
135
- this._callback();
179
+ this.#callback();
136
180
  }
137
181
 
138
182
  __makeNative(platformConfig: ?PlatformConfig): void {
@@ -150,17 +194,17 @@ export default class AnimatedProps extends AnimatedNode {
150
194
  // where it will be needed to traverse the graph of attached values.
151
195
  super.__setPlatformConfig(platformConfig);
152
196
 
153
- if (this._animatedView) {
197
+ if (this.#animatedView) {
154
198
  this.__connectAnimatedView();
155
199
  }
156
200
  }
157
201
  }
158
202
 
159
203
  setNativeView(animatedView: any): void {
160
- if (this._animatedView === animatedView) {
204
+ if (this.#animatedView === animatedView) {
161
205
  return;
162
206
  }
163
- this._animatedView = animatedView;
207
+ this.#animatedView = animatedView;
164
208
  if (this.__isNative) {
165
209
  this.__connectAnimatedView();
166
210
  }
@@ -168,11 +212,14 @@ export default class AnimatedProps extends AnimatedNode {
168
212
 
169
213
  __connectAnimatedView(): void {
170
214
  invariant(this.__isNative, 'Expected node to be marked as "native"');
171
- const nativeViewTag: ?number = findNodeHandle(this._animatedView);
172
- invariant(
173
- nativeViewTag != null,
174
- 'Unable to locate attached view in the native tree',
175
- );
215
+ let nativeViewTag: ?number = findNodeHandle(this.#animatedView);
216
+ if (nativeViewTag == null) {
217
+ if (process.env.NODE_ENV === 'test') {
218
+ nativeViewTag = -1;
219
+ } else {
220
+ throw new Error('Unable to locate attached view in the native tree');
221
+ }
222
+ }
176
223
  NativeAnimatedHelper.API.connectAnimatedNodeToView(
177
224
  this.__getNativeTag(),
178
225
  nativeViewTag,
@@ -181,11 +228,14 @@ export default class AnimatedProps extends AnimatedNode {
181
228
 
182
229
  __disconnectAnimatedView(): void {
183
230
  invariant(this.__isNative, 'Expected node to be marked as "native"');
184
- const nativeViewTag: ?number = findNodeHandle(this._animatedView);
185
- invariant(
186
- nativeViewTag != null,
187
- 'Unable to locate attached view in the native tree',
188
- );
231
+ let nativeViewTag: ?number = findNodeHandle(this.#animatedView);
232
+ if (nativeViewTag == null) {
233
+ if (process.env.NODE_ENV === 'test') {
234
+ nativeViewTag = -1;
235
+ } else {
236
+ throw new Error('Unable to locate attached view in the native tree');
237
+ }
238
+ }
189
239
  NativeAnimatedHelper.API.disconnectAnimatedNodeFromView(
190
240
  this.__getNativeTag(),
191
241
  nativeViewTag,