react-native-screens 3.32.0 → 3.34.0

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 (142) hide show
  1. package/README.md +17 -13
  2. package/RNScreens.podspec +10 -52
  3. package/android/CMakeLists.txt +48 -4
  4. package/android/build.gradle +9 -81
  5. package/android/src/fabric/java/com/swmansion/rnscreens/FabricEnabledViewGroup.kt +25 -16
  6. package/android/src/fabric/java/com/swmansion/rnscreens/NativeProxy.kt +53 -0
  7. package/android/src/main/cpp/NativeProxy.cpp +51 -0
  8. package/android/src/main/cpp/NativeProxy.h +35 -0
  9. package/android/src/main/cpp/OnLoad.cpp +8 -0
  10. package/android/src/main/java/com/swmansion/rnscreens/CustomSearchView.kt +5 -2
  11. package/android/src/main/java/com/swmansion/rnscreens/CustomToolbar.kt +4 -1
  12. package/android/src/main/java/com/swmansion/rnscreens/FragmentBackPressOverrider.kt +2 -2
  13. package/android/src/main/java/com/swmansion/rnscreens/RNScreensPackage.kt +36 -17
  14. package/android/src/main/java/com/swmansion/rnscreens/Screen.kt +134 -38
  15. package/android/src/main/java/com/swmansion/rnscreens/ScreenContainer.kt +52 -30
  16. package/android/src/main/java/com/swmansion/rnscreens/ScreenContainerViewManager.kt +17 -7
  17. package/android/src/main/java/com/swmansion/rnscreens/ScreenEventDispatcher.kt +10 -2
  18. package/android/src/main/java/com/swmansion/rnscreens/ScreenFragment.kt +56 -27
  19. package/android/src/main/java/com/swmansion/rnscreens/ScreenFragmentWrapper.kt +8 -1
  20. package/android/src/main/java/com/swmansion/rnscreens/ScreenStack.kt +50 -19
  21. package/android/src/main/java/com/swmansion/rnscreens/ScreenStackFragment.kt +60 -37
  22. package/android/src/main/java/com/swmansion/rnscreens/ScreenStackFragmentWrapper.kt +4 -0
  23. package/android/src/main/java/com/swmansion/rnscreens/ScreenStackHeaderConfig.kt +85 -58
  24. package/android/src/main/java/com/swmansion/rnscreens/ScreenStackHeaderConfigViewManager.kt +128 -37
  25. package/android/src/main/java/com/swmansion/rnscreens/ScreenStackHeaderSubview.kt +19 -4
  26. package/android/src/main/java/com/swmansion/rnscreens/ScreenStackHeaderSubviewManager.kt +16 -10
  27. package/android/src/main/java/com/swmansion/rnscreens/ScreenStackViewManager.kt +28 -25
  28. package/android/src/main/java/com/swmansion/rnscreens/ScreenViewManager.kt +173 -78
  29. package/android/src/main/java/com/swmansion/rnscreens/ScreenWindowTraits.kt +59 -24
  30. package/android/src/main/java/com/swmansion/rnscreens/ScreensModule.kt +30 -8
  31. package/android/src/main/java/com/swmansion/rnscreens/ScreensShadowNode.kt +3 -1
  32. package/android/src/main/java/com/swmansion/rnscreens/SearchBarManager.kt +101 -50
  33. package/android/src/main/java/com/swmansion/rnscreens/SearchBarView.kt +29 -22
  34. package/android/src/main/java/com/swmansion/rnscreens/SearchViewFormatter.kt +7 -2
  35. package/android/src/main/java/com/swmansion/rnscreens/events/HeaderAttachedEvent.kt +4 -1
  36. package/android/src/main/java/com/swmansion/rnscreens/events/HeaderBackButtonClickedEvent.kt +4 -1
  37. package/android/src/main/java/com/swmansion/rnscreens/events/HeaderDetachedEvent.kt +4 -1
  38. package/android/src/main/java/com/swmansion/rnscreens/events/HeaderHeightChangeEvent.kt +5 -5
  39. package/android/src/main/java/com/swmansion/rnscreens/events/ScreenAppearEvent.kt +4 -1
  40. package/android/src/main/java/com/swmansion/rnscreens/events/ScreenDisappearEvent.kt +4 -1
  41. package/android/src/main/java/com/swmansion/rnscreens/events/ScreenDismissedEvent.kt +8 -4
  42. package/android/src/main/java/com/swmansion/rnscreens/events/ScreenTransitionProgressEvent.kt +7 -6
  43. package/android/src/main/java/com/swmansion/rnscreens/events/ScreenWillAppearEvent.kt +4 -1
  44. package/android/src/main/java/com/swmansion/rnscreens/events/ScreenWillDisappearEvent.kt +4 -1
  45. package/android/src/main/java/com/swmansion/rnscreens/events/SearchBarBlurEvent.kt +4 -1
  46. package/android/src/main/java/com/swmansion/rnscreens/events/SearchBarChangeTextEvent.kt +4 -3
  47. package/android/src/main/java/com/swmansion/rnscreens/events/SearchBarCloseEvent.kt +4 -1
  48. package/android/src/main/java/com/swmansion/rnscreens/events/SearchBarFocusEvent.kt +4 -1
  49. package/android/src/main/java/com/swmansion/rnscreens/events/SearchBarOpenEvent.kt +4 -1
  50. package/android/src/main/java/com/swmansion/rnscreens/events/SearchBarSearchButtonPressEvent.kt +9 -4
  51. package/android/src/main/java/com/swmansion/rnscreens/events/StackFinishTransitioningEvent.kt +4 -1
  52. package/android/src/main/java/com/swmansion/rnscreens/utils/DeviceUtils.kt +1 -5
  53. package/android/src/main/java/com/swmansion/rnscreens/utils/ScreenDummyLayoutHelper.kt +245 -0
  54. package/android/src/main/jni/CMakeLists.txt +5 -4
  55. package/android/src/paper/java/com/facebook/react/viewmanagers/RNSScreenManagerDelegate.java +3 -0
  56. package/android/src/paper/java/com/facebook/react/viewmanagers/RNSScreenManagerInterface.java +1 -0
  57. package/android/src/paper/java/com/swmansion/rnscreens/FabricEnabledViewGroup.kt +10 -5
  58. package/android/src/paper/java/com/swmansion/rnscreens/NativeProxy.kt +19 -0
  59. package/android/src/paper/java/com/swmansion/rnscreens/NativeScreensModuleSpec.java +4 -0
  60. package/common/cpp/react/renderer/components/rnscreens/FrameCorrectionModes.h +51 -0
  61. package/common/cpp/react/renderer/components/rnscreens/RNSModalScreenShadowNode.cpp +2 -1
  62. package/common/cpp/react/renderer/components/rnscreens/RNSModalScreenShadowNode.h +1 -1
  63. package/common/cpp/react/renderer/components/rnscreens/RNSScreenComponentDescriptor.h +140 -1
  64. package/common/cpp/react/renderer/components/rnscreens/RNSScreenShadowNode.cpp +51 -1
  65. package/common/cpp/react/renderer/components/rnscreens/RNSScreenShadowNode.h +23 -1
  66. package/common/cpp/react/renderer/components/rnscreens/RNSScreenState.cpp +20 -0
  67. package/common/cpp/react/renderer/components/rnscreens/RNSScreenState.h +23 -1
  68. package/common/cpp/react/renderer/components/rnscreens/utils/RectUtil.h +36 -0
  69. package/cpp/RNSScreenRemovalListener.cpp +25 -0
  70. package/cpp/RNSScreenRemovalListener.h +20 -0
  71. package/ios/RNSConvert.h +1 -0
  72. package/ios/RNSModalScreen.mm +22 -0
  73. package/ios/RNSModule.mm +1 -1
  74. package/ios/RNSScreen.h +2 -1
  75. package/ios/RNSScreen.mm +27 -19
  76. package/ios/RNSScreenStack.mm +24 -77
  77. package/ios/RNSScreenStackAnimator.mm +43 -6
  78. package/ios/RNSScreenStackHeaderConfig.mm +49 -11
  79. package/ios/RNSScreenStackHeaderSubview.mm +8 -0
  80. package/ios/utils/UIView+RNSUtility.h +23 -0
  81. package/ios/utils/UIView+RNSUtility.mm +55 -0
  82. package/lib/commonjs/components/ScreenStack.js +8 -1
  83. package/lib/commonjs/components/ScreenStack.js.map +1 -1
  84. package/lib/commonjs/fabric/ModalScreenNativeComponent.js.map +1 -1
  85. package/lib/commonjs/fabric/ScreenNativeComponent.js.map +1 -1
  86. package/lib/commonjs/native-stack/views/NativeStackView.js +2 -0
  87. package/lib/commonjs/native-stack/views/NativeStackView.js.map +1 -1
  88. package/lib/module/components/ScreenStack.js +8 -1
  89. package/lib/module/components/ScreenStack.js.map +1 -1
  90. package/lib/module/fabric/ModalScreenNativeComponent.js.map +1 -1
  91. package/lib/module/fabric/ScreenNativeComponent.js.map +1 -1
  92. package/lib/module/native-stack/views/NativeStackView.js +2 -0
  93. package/lib/module/native-stack/views/NativeStackView.js.map +1 -1
  94. package/lib/typescript/components/ScreenStack.d.ts.map +1 -1
  95. package/lib/typescript/fabric/ModalScreenNativeComponent.d.ts +1 -0
  96. package/lib/typescript/fabric/ModalScreenNativeComponent.d.ts.map +1 -1
  97. package/lib/typescript/fabric/ScreenNativeComponent.d.ts +1 -0
  98. package/lib/typescript/fabric/ScreenNativeComponent.d.ts.map +1 -1
  99. package/lib/typescript/native-stack/types.d.ts +10 -0
  100. package/lib/typescript/native-stack/types.d.ts.map +1 -1
  101. package/lib/typescript/native-stack/views/NativeStackView.d.ts.map +1 -1
  102. package/lib/typescript/types.d.ts +10 -0
  103. package/lib/typescript/types.d.ts.map +1 -1
  104. package/native-stack/README.md +110 -99
  105. package/package.json +6 -3
  106. package/react-native.config.js +17 -15
  107. package/src/TransitionProgressContext.tsx +1 -1
  108. package/src/components/Screen.tsx +4 -4
  109. package/src/components/ScreenStack.tsx +11 -1
  110. package/src/components/ScreenStackHeaderConfig.tsx +5 -5
  111. package/src/components/ScreenStackHeaderConfig.web.tsx +6 -6
  112. package/src/components/SearchBar.tsx +4 -4
  113. package/src/core.ts +1 -1
  114. package/src/fabric/ModalScreenNativeComponent.ts +1 -0
  115. package/src/fabric/ScreenNativeComponent.ts +1 -0
  116. package/src/fabric/ScreenNavigationContainerNativeComponent.ts +1 -1
  117. package/src/fabric/ScreenStackHeaderConfigNativeComponent.ts +1 -1
  118. package/src/fabric/ScreenStackHeaderSubviewNativeComponent.ts +1 -1
  119. package/src/fabric/SearchBarNativeComponent.ts +1 -1
  120. package/src/gesture-handler/ScreenGestureDetector.tsx +5 -5
  121. package/src/gesture-handler/constraints.ts +5 -5
  122. package/src/gesture-handler/fabricUtils.ts +1 -1
  123. package/src/native-stack/contexts/GHContext.tsx +1 -1
  124. package/src/native-stack/navigators/createNativeStackNavigator.tsx +3 -3
  125. package/src/native-stack/types.tsx +14 -4
  126. package/src/native-stack/utils/getDefaultHeaderHeight.tsx +1 -1
  127. package/src/native-stack/utils/getStatusBarHeight.tsx +1 -1
  128. package/src/native-stack/utils/useAnimatedHeaderHeight.tsx +1 -1
  129. package/src/native-stack/utils/useBackPressSubscription.tsx +1 -1
  130. package/src/native-stack/utils/useHeaderHeight.tsx +1 -1
  131. package/src/native-stack/views/FontProcessor.tsx +1 -1
  132. package/src/native-stack/views/HeaderConfig.tsx +1 -1
  133. package/src/native-stack/views/NativeStackView.tsx +11 -9
  134. package/src/reanimated/ReanimatedHeaderHeightContext.tsx +1 -1
  135. package/src/reanimated/ReanimatedNativeStackScreen.tsx +5 -5
  136. package/src/reanimated/ReanimatedScreen.tsx +2 -2
  137. package/src/reanimated/ReanimatedScreenProvider.tsx +1 -1
  138. package/src/reanimated/useReanimatedHeaderHeight.tsx +1 -1
  139. package/src/reanimated/useReanimatedTransitionProgress.tsx +1 -1
  140. package/src/types.tsx +15 -5
  141. package/src/useTransitionProgress.tsx +1 -1
  142. package/windows/README.md +4 -1
@@ -4,7 +4,10 @@ import com.facebook.react.bridge.Arguments
4
4
  import com.facebook.react.bridge.WritableMap
5
5
  import com.facebook.react.uimanager.events.Event
6
6
 
7
- class SearchBarCloseEvent(surfaceId: Int, viewId: Int) : Event<SearchBarCloseEvent>(surfaceId, viewId) {
7
+ class SearchBarCloseEvent(
8
+ surfaceId: Int,
9
+ viewId: Int,
10
+ ) : Event<SearchBarCloseEvent>(surfaceId, viewId) {
8
11
  override fun getEventName(): String = EVENT_NAME
9
12
 
10
13
  // All events for a given view can be coalesced.
@@ -4,7 +4,10 @@ import com.facebook.react.bridge.Arguments
4
4
  import com.facebook.react.bridge.WritableMap
5
5
  import com.facebook.react.uimanager.events.Event
6
6
 
7
- class SearchBarFocusEvent(surfaceId: Int, viewId: Int) : Event<SearchBarFocusEvent>(surfaceId, viewId) {
7
+ class SearchBarFocusEvent(
8
+ surfaceId: Int,
9
+ viewId: Int,
10
+ ) : Event<SearchBarFocusEvent>(surfaceId, viewId) {
8
11
  override fun getEventName(): String = EVENT_NAME
9
12
 
10
13
  // All events for a given view can be coalesced.
@@ -4,7 +4,10 @@ import com.facebook.react.bridge.Arguments
4
4
  import com.facebook.react.bridge.WritableMap
5
5
  import com.facebook.react.uimanager.events.Event
6
6
 
7
- class SearchBarOpenEvent(surfaceId: Int, viewId: Int) : Event<SearchBarOpenEvent>(surfaceId, viewId) {
7
+ class SearchBarOpenEvent(
8
+ surfaceId: Int,
9
+ viewId: Int,
10
+ ) : Event<SearchBarOpenEvent>(surfaceId, viewId) {
8
11
  override fun getEventName(): String = EVENT_NAME
9
12
 
10
13
  // All events for a given view can be coalesced.
@@ -4,15 +4,20 @@ import com.facebook.react.bridge.Arguments
4
4
  import com.facebook.react.bridge.WritableMap
5
5
  import com.facebook.react.uimanager.events.Event
6
6
 
7
- class SearchBarSearchButtonPressEvent(surfaceId: Int, viewId: Int, private val text: String?) : Event<SearchBarSearchButtonPressEvent>(surfaceId, viewId) {
7
+ class SearchBarSearchButtonPressEvent(
8
+ surfaceId: Int,
9
+ viewId: Int,
10
+ private val text: String?,
11
+ ) : Event<SearchBarSearchButtonPressEvent>(surfaceId, viewId) {
8
12
  override fun getEventName(): String = EVENT_NAME
9
13
 
10
14
  // All events for a given view can be coalesced.
11
15
  override fun getCoalescingKey(): Short = 0
12
16
 
13
- override fun getEventData(): WritableMap? = Arguments.createMap().apply {
14
- putString("text", text)
15
- }
17
+ override fun getEventData(): WritableMap? =
18
+ Arguments.createMap().apply {
19
+ putString("text", text)
20
+ }
16
21
 
17
22
  companion object {
18
23
  const val EVENT_NAME = "topSearchButtonPress"
@@ -4,7 +4,10 @@ import com.facebook.react.bridge.Arguments
4
4
  import com.facebook.react.bridge.WritableMap
5
5
  import com.facebook.react.uimanager.events.Event
6
6
 
7
- class StackFinishTransitioningEvent(surfaceId: Int, viewId: Int) : Event<StackFinishTransitioningEvent>(surfaceId, viewId) {
7
+ class StackFinishTransitioningEvent(
8
+ surfaceId: Int,
9
+ viewId: Int,
10
+ ) : Event<StackFinishTransitioningEvent>(surfaceId, viewId) {
8
11
  override fun getEventName() = EVENT_NAME
9
12
 
10
13
  // All events for a given view can be coalesced.
@@ -4,9 +4,5 @@ import android.content.Context
4
4
  import android.content.pm.PackageManager
5
5
 
6
6
  object DeviceUtils {
7
-
8
- fun isPlatformAndroidTV(context: Context?): Boolean {
9
- return context?.packageManager?.hasSystemFeature(PackageManager.FEATURE_LEANBACK) == true
10
- }
11
-
7
+ fun isPlatformAndroidTV(context: Context?): Boolean = context?.packageManager?.hasSystemFeature(PackageManager.FEATURE_LEANBACK) == true
12
8
  }
@@ -0,0 +1,245 @@
1
+ package com.swmansion.rnscreens.utils
2
+
3
+ import android.app.Activity
4
+ import android.util.Log
5
+ import android.view.View
6
+ import androidx.appcompat.widget.Toolbar
7
+ import androidx.coordinatorlayout.widget.CoordinatorLayout
8
+ import com.facebook.react.bridge.LifecycleEventListener
9
+ import com.facebook.react.bridge.ReactApplicationContext
10
+ import com.facebook.react.uimanager.PixelUtil
11
+ import com.google.android.material.appbar.AppBarLayout
12
+ import com.swmansion.rnscreens.ScreenStackHeaderConfig
13
+ import java.lang.ref.WeakReference
14
+
15
+ /**
16
+ * This class provides methods to create dummy layout (that mimics Screen setup), and to compute
17
+ * expected header height. It is meant to be accessed from C++ layer via JNI.
18
+ * See https://github.com/software-mansion/react-native-screens/pull/2169
19
+ * for more detailed description of the issue this code solves.
20
+ */
21
+ internal class ScreenDummyLayoutHelper(
22
+ reactContext: ReactApplicationContext,
23
+ ) : LifecycleEventListener {
24
+ // The state required to compute header dimensions. We want this on instance rather than on class
25
+ // for context access & being tied to instance lifetime.
26
+ private lateinit var coordinatorLayout: CoordinatorLayout
27
+ private lateinit var appBarLayout: AppBarLayout
28
+ private lateinit var dummyContentView: View
29
+ private lateinit var toolbar: Toolbar
30
+ private var defaultFontSize: Float = 0f
31
+ private var defaultContentInsetStartWithNavigation: Int = 0
32
+
33
+ // LRU with size 1
34
+ private var cache: CacheEntry = CacheEntry.EMPTY
35
+
36
+ // We do not want to be responsible for the context lifecycle. If it's null, we're fine.
37
+ // This same context is being passed down to our view components so it is destroyed
38
+ // only if our views also are.
39
+ private var reactContextRef: WeakReference<ReactApplicationContext> =
40
+ WeakReference(reactContext)
41
+
42
+ init {
43
+ // We load the library so that we are able to communicate with our C++ code (descriptor & shadow nodes).
44
+ // Basically we leak this object to C++, as its lifecycle should span throughout whole application
45
+ // lifecycle anyway.
46
+ try {
47
+ System.loadLibrary(LIBRARY_NAME)
48
+ } catch (e: UnsatisfiedLinkError) {
49
+ Log.w(TAG, "Failed to load $LIBRARY_NAME")
50
+ }
51
+
52
+ weakInstance = WeakReference(this)
53
+
54
+ if (!(reactContext.hasCurrentActivity() && maybeInitDummyLayoutWithHeader(reactContext))) {
55
+ reactContext.addLifecycleEventListener(this)
56
+ }
57
+ }
58
+
59
+ /**
60
+ * Initializes dummy view hierarchy with CoordinatorLayout, AppBarLayout and dummy View.
61
+ * We utilize this to compute header height (app bar layout height) from C++ layer when its needed.
62
+ *
63
+ * @return boolean whether the layout was initialised or not
64
+ */
65
+ private fun maybeInitDummyLayoutWithHeader(reactContext: ReactApplicationContext): Boolean {
66
+ if (isLayoutInitialized) {
67
+ return true
68
+ }
69
+
70
+ if (!reactContext.hasCurrentActivity()) {
71
+ return false
72
+ }
73
+
74
+ // We need to use activity here, as react context does not have theme attributes required by
75
+ // AppBarLayout attached leading to crash.
76
+ val contextWithTheme =
77
+ requireNotNull(reactContext.currentActivity) { "[RNScreens] Attempt to use context detached from activity" }
78
+
79
+ coordinatorLayout = CoordinatorLayout(contextWithTheme)
80
+
81
+ appBarLayout =
82
+ AppBarLayout(contextWithTheme).apply {
83
+ layoutParams =
84
+ CoordinatorLayout.LayoutParams(
85
+ CoordinatorLayout.LayoutParams.MATCH_PARENT,
86
+ CoordinatorLayout.LayoutParams.WRAP_CONTENT,
87
+ )
88
+ }
89
+
90
+ toolbar =
91
+ Toolbar(contextWithTheme).apply {
92
+ title = DEFAULT_HEADER_TITLE
93
+ layoutParams =
94
+ AppBarLayout
95
+ .LayoutParams(
96
+ AppBarLayout.LayoutParams.MATCH_PARENT,
97
+ AppBarLayout.LayoutParams.WRAP_CONTENT,
98
+ ).apply { scrollFlags = 0 }
99
+ }
100
+
101
+ // We know the title text view will be there, cause we've just set title.
102
+ defaultFontSize = ScreenStackHeaderConfig.findTitleTextViewInToolbar(toolbar)!!.textSize
103
+ defaultContentInsetStartWithNavigation = toolbar.contentInsetStartWithNavigation
104
+
105
+ appBarLayout.addView(toolbar)
106
+
107
+ dummyContentView =
108
+ View(contextWithTheme).apply {
109
+ layoutParams =
110
+ CoordinatorLayout.LayoutParams(
111
+ CoordinatorLayout.LayoutParams.MATCH_PARENT,
112
+ CoordinatorLayout.LayoutParams.MATCH_PARENT,
113
+ )
114
+ }
115
+
116
+ coordinatorLayout.apply {
117
+ addView(appBarLayout)
118
+ addView(dummyContentView)
119
+ }
120
+
121
+ isLayoutInitialized = true
122
+ return true
123
+ }
124
+
125
+ /**
126
+ * Triggers layout pass on dummy view hierarchy, taking into consideration selected
127
+ * ScreenStackHeaderConfig props that might have impact on final header height.
128
+ *
129
+ * @param fontSize font size value as passed from JS
130
+ * @return header height in dp as consumed by Yoga
131
+ */
132
+ private fun computeDummyLayout(
133
+ fontSize: Int,
134
+ isTitleEmpty: Boolean,
135
+ ): Float {
136
+ if (!isLayoutInitialized) {
137
+ val reactContext =
138
+ requireReactContext { "[RNScreens] Context was null-ed before dummy layout was initialized" }
139
+ if (!maybeInitDummyLayoutWithHeader(reactContext)) {
140
+ // This theoretically might happen at Fabric + "bridgefull" combination, due to race condition where `reactContext.currentActivity`
141
+ // is still null at this execution point. We don't wanna crash in such case, thus returning zeroed height.
142
+ Log.e(
143
+ TAG,
144
+ "[RNScreens] Failed to late-init layout while computing header height. This is a race-condition-bug in react-native-screens, please file an issue at https://github.com/software-mansion/react-native-screens/issues"
145
+ )
146
+ return 0.0f
147
+ }
148
+ }
149
+
150
+ if (cache.hasKey(CacheKey(fontSize, isTitleEmpty))) {
151
+ return cache.headerHeight
152
+ }
153
+
154
+ val topLevelDecorView = requireActivity().window.decorView
155
+
156
+ // These dimensions are not accurate, as they do include status bar & navigation bar, however
157
+ // it is ok for our purposes.
158
+ val decorViewWidth = topLevelDecorView.width
159
+ val decorViewHeight = topLevelDecorView.height
160
+
161
+ val widthMeasureSpec =
162
+ View.MeasureSpec.makeMeasureSpec(decorViewWidth, View.MeasureSpec.EXACTLY)
163
+ val heightMeasureSpec =
164
+ View.MeasureSpec.makeMeasureSpec(decorViewHeight, View.MeasureSpec.EXACTLY)
165
+
166
+ if (isTitleEmpty) {
167
+ toolbar.title = ""
168
+ toolbar.contentInsetStartWithNavigation = 0
169
+ } else {
170
+ toolbar.title = DEFAULT_HEADER_TITLE
171
+ toolbar.contentInsetStartWithNavigation = defaultContentInsetStartWithNavigation
172
+ }
173
+
174
+ val textView = ScreenStackHeaderConfig.findTitleTextViewInToolbar(toolbar)
175
+ textView?.textSize =
176
+ if (fontSize != FONT_SIZE_UNSET) fontSize.toFloat() else defaultFontSize
177
+
178
+ coordinatorLayout.measure(widthMeasureSpec, heightMeasureSpec)
179
+
180
+ // It seems that measure pass would be enough, however I'm not certain whether there are no
181
+ // scenarios when layout violates measured dimensions.
182
+ coordinatorLayout.layout(0, 0, decorViewWidth, decorViewHeight)
183
+
184
+ val headerHeight = PixelUtil.toDIPFromPixel(appBarLayout.height.toFloat())
185
+ cache = CacheEntry(CacheKey(fontSize, isTitleEmpty), headerHeight)
186
+ return headerHeight
187
+ }
188
+
189
+ private fun requireReactContext(lazyMessage: (() -> Any)? = null): ReactApplicationContext =
190
+ requireNotNull(
191
+ reactContextRef.get(),
192
+ lazyMessage ?: { "[RNScreens] Attempt to require missing react context" })
193
+
194
+ private fun requireActivity(): Activity =
195
+ requireNotNull(requireReactContext().currentActivity) {
196
+ "[RNScreens] Attempt to use context detached from activity"
197
+ }
198
+
199
+ companion object {
200
+ const val TAG = "ScreenDummyLayoutHelper"
201
+
202
+ const val LIBRARY_NAME = "react_codegen_rnscreens"
203
+
204
+ const val FONT_SIZE_UNSET = -1
205
+
206
+ private const val DEFAULT_HEADER_TITLE: String = "FontSize123!#$"
207
+
208
+ // We access this field from C++ layer, through `getInstance` method.
209
+ // We don't care what instance we get access to as long as it has initialized
210
+ // dummy view hierarchy.
211
+ private var weakInstance = WeakReference<ScreenDummyLayoutHelper>(null)
212
+
213
+ @JvmStatic
214
+ fun getInstance(): ScreenDummyLayoutHelper? = weakInstance.get()
215
+ }
216
+
217
+ private var isLayoutInitialized = false
218
+
219
+ override fun onHostResume() {
220
+ // This is the earliest we have guarantee that the context has a reference to an activity.
221
+ val reactContext = requireReactContext { "[RNScreens] ReactContext missing in onHostResume! This should not happen." }
222
+ check(maybeInitDummyLayoutWithHeader(reactContext)) { "[RNScreens] Failed to initialise dummy layout in onHostResume. This is not expected."}
223
+ reactContext.removeLifecycleEventListener(this)
224
+ }
225
+
226
+ override fun onHostPause() = Unit
227
+
228
+ override fun onHostDestroy() = Unit
229
+ }
230
+
231
+ private data class CacheKey(
232
+ val fontSize: Int,
233
+ val isTitleEmpty: Boolean,
234
+ )
235
+
236
+ private class CacheEntry(
237
+ val cacheKey: CacheKey,
238
+ val headerHeight: Float,
239
+ ) {
240
+ fun hasKey(key: CacheKey) = cacheKey.fontSize != Int.MIN_VALUE && cacheKey == key
241
+
242
+ companion object {
243
+ val EMPTY = CacheEntry(CacheKey(Int.MIN_VALUE, false), 0f)
244
+ }
245
+ }
@@ -6,6 +6,7 @@ set(LIB_TARGET_NAME react_codegen_${LIB_LITERAL})
6
6
 
7
7
  set(LIB_ANDROID_DIR ${CMAKE_CURRENT_SOURCE_DIR}/../../..)
8
8
  set(LIB_COMMON_DIR ${LIB_ANDROID_DIR}/../common/cpp)
9
+ set(LIB_COMMON_COMPONENTS_DIR ${LIB_COMMON_DIR}/react/renderer/components/${LIB_LITERAL})
9
10
  set(LIB_ANDROID_GENERATED_JNI_DIR ${LIB_ANDROID_DIR}/build/generated/source/codegen/jni)
10
11
  set(LIB_ANDROID_GENERATED_COMPONENTS_DIR ${LIB_ANDROID_GENERATED_JNI_DIR}/react/renderer/components/${LIB_LITERAL})
11
12
 
@@ -18,20 +19,20 @@ add_compile_options(
18
19
  -Wno-gnu-zero-variadic-macro-arguments
19
20
  )
20
21
 
21
- file(GLOB LIB_CUSTOM_SRCS CONFIGURE_DEPENDS *.cpp ${LIB_COMMON_DIR}/react/renderer/components/${LIB_LITERAL}/*.cpp)
22
+ file(GLOB LIB_CUSTOM_SRCS CONFIGURE_DEPENDS *.cpp ${LIB_COMMON_COMPONENTS_DIR}/*.cpp ${LIB_COMMON_COMPONENTS_DIR}/utils/*.cpp)
22
23
  file(GLOB LIB_CODEGEN_SRCS CONFIGURE_DEPENDS ${LIB_ANDROID_GENERATED_COMPONENTS_DIR}/*.cpp)
23
24
 
24
25
  add_library(
25
26
  ${LIB_TARGET_NAME}
26
- SHARED
27
+ SHARED
27
28
  ${LIB_CUSTOM_SRCS}
28
29
  ${LIB_CODEGEN_SRCS}
29
30
  )
30
31
 
31
32
  target_include_directories(
32
33
  ${LIB_TARGET_NAME}
33
- PUBLIC
34
- .
34
+ PUBLIC
35
+ .
35
36
  ${LIB_COMMON_DIR}
36
37
  ${LIB_ANDROID_GENERATED_JNI_DIR}
37
38
  ${LIB_ANDROID_GENERATED_COMPONENTS_DIR}
@@ -44,6 +44,9 @@ public class RNSScreenManagerDelegate<T extends View, U extends BaseViewManagerI
44
44
  case "fullScreenSwipeEnabled":
45
45
  mViewManager.setFullScreenSwipeEnabled(view, value == null ? false : (boolean) value);
46
46
  break;
47
+ case "fullScreenSwipeShadowEnabled":
48
+ mViewManager.setFullScreenSwipeShadowEnabled(view, value == null ? false : (boolean) value);
49
+ break;
47
50
  case "homeIndicatorHidden":
48
51
  mViewManager.setHomeIndicatorHidden(view, value == null ? false : (boolean) value);
49
52
  break;
@@ -21,6 +21,7 @@ public interface RNSScreenManagerInterface<T extends View> {
21
21
  void setSheetExpandsWhenScrolledToEdge(T view, boolean value);
22
22
  void setCustomAnimationOnSwipe(T view, boolean value);
23
23
  void setFullScreenSwipeEnabled(T view, boolean value);
24
+ void setFullScreenSwipeShadowEnabled(T view, boolean value);
24
25
  void setHomeIndicatorHidden(T view, boolean value);
25
26
  void setPreventNativeDismiss(T view, boolean value);
26
27
  void setGestureEnabled(T view, boolean value);
@@ -2,13 +2,18 @@ package com.swmansion.rnscreens
2
2
 
3
3
  import android.view.ViewGroup
4
4
  import com.facebook.react.bridge.ReactContext
5
- import com.facebook.react.uimanager.FabricViewStateManager
5
+ import com.facebook.react.uimanager.StateWrapper
6
6
 
7
- abstract class FabricEnabledViewGroup constructor(context: ReactContext?) : ViewGroup(context) {
7
+ abstract class FabricEnabledViewGroup(
8
+ context: ReactContext?,
9
+ ) : ViewGroup(context) {
10
+ fun setStateWrapper(wrapper: StateWrapper?) = Unit
8
11
 
9
- val fabricViewStateManager get() = null as FabricViewStateManager?
10
-
11
- protected fun updateScreenSizeFabric(width: Int, height: Int, headerHeight: Double) {
12
+ protected fun updateScreenSizeFabric(
13
+ width: Int,
14
+ height: Int,
15
+ headerHeight: Double,
16
+ ) {
12
17
  // do nothing
13
18
  }
14
19
  }
@@ -0,0 +1,19 @@
1
+ package com.swmansion.rnscreens
2
+
3
+ import com.facebook.react.fabric.FabricUIManager
4
+
5
+ class NativeProxy {
6
+ // do nothing on Paper
7
+ fun nativeAddMutationsListener(fabricUIManager: FabricUIManager) = Unit
8
+
9
+ companion object {
10
+ fun addScreenToMap(
11
+ tag: Int,
12
+ view: Screen,
13
+ ) = Unit
14
+
15
+ fun removeScreenFromMap(tag: Int) = Unit
16
+
17
+ fun clearMapOnInvalidate() = Unit
18
+ }
19
+ }
@@ -12,8 +12,10 @@
12
12
 
13
13
  package com.swmansion.rnscreens;
14
14
 
15
+ import com.facebook.proguard.annotations.DoNotStrip;
15
16
  import com.facebook.react.bridge.ReactApplicationContext;
16
17
  import com.facebook.react.bridge.ReactContextBaseJavaModule;
18
+ import com.facebook.react.bridge.ReactMethod;
17
19
  import com.facebook.react.bridge.ReactModuleWithSpec;
18
20
  import com.facebook.react.turbomodule.core.interfaces.TurboModule;
19
21
  import javax.annotation.Nonnull;
@@ -29,4 +31,6 @@ public abstract class NativeScreensModuleSpec extends ReactContextBaseJavaModule
29
31
  public @Nonnull String getName() {
30
32
  return NAME;
31
33
  }
34
+
35
+
32
36
  }
@@ -0,0 +1,51 @@
1
+ #pragma once
2
+
3
+ #include <cstdint>
4
+
5
+ namespace rnscreens {
6
+
7
+ /**
8
+ * Flags describing types of corrections to apply to Screen frame
9
+ * after layout process.
10
+ */
11
+ class FrameCorrectionModes final {
12
+ public:
13
+ enum Mode : std::uint8_t {
14
+ /**
15
+ * No correction should be applied to layout metrics of Screen
16
+ */
17
+ None = 0,
18
+
19
+ /**
20
+ * Screen's frame height should be corrected
21
+ */
22
+ FrameHeightCorrection = 1 << 0,
23
+
24
+ /**
25
+ * Screen's frame origin should be corrected
26
+ */
27
+ FrameOriginCorrection = 1 << 1,
28
+ };
29
+
30
+ inline void set(Mode mode) {
31
+ modes_ = Mode(modes_ | mode);
32
+ }
33
+
34
+ inline void unset(Mode mode) {
35
+ modes_ = Mode(modes_ & ~mode);
36
+ }
37
+
38
+ // Check whether current set of flags contains all flags set in argument.
39
+ inline bool check(Mode mode) const {
40
+ return Mode(modes_ & mode) == mode;
41
+ }
42
+
43
+ inline Mode getAll() const {
44
+ return modes_;
45
+ }
46
+
47
+ private:
48
+ Mode modes_{Mode::None};
49
+ };
50
+
51
+ } // namespace rnscreens
@@ -5,7 +5,8 @@ namespace react {
5
5
 
6
6
  extern const char RNSModalScreenComponentName[] = "RNSModalScreen";
7
7
 
8
- Point RNSModalScreenShadowNode::getContentOriginOffset() const {
8
+ Point RNSModalScreenShadowNode::getContentOriginOffset(
9
+ bool /*includeTransform*/) const {
9
10
  auto stateData = getStateData();
10
11
  auto contentOffset = stateData.contentOffset;
11
12
  return {contentOffset.x, contentOffset.y};
@@ -20,7 +20,7 @@ class JSI_EXPORT RNSModalScreenShadowNode final
20
20
  public:
21
21
  using ConcreteViewShadowNode::ConcreteViewShadowNode;
22
22
 
23
- Point getContentOriginOffset() const override;
23
+ Point getContentOriginOffset(bool includeTransform) const override;
24
24
  static ShadowNodeTraits BaseTraits() {
25
25
  auto traits = ConcreteViewShadowNode::BaseTraits();
26
26
  traits.set(ShadowNodeTraits::Trait::RootNodeKind);