@momo-kits/native-kits 0.160.2-beta.1 → 0.160.2-beta.2-debug

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 (136) hide show
  1. package/build.gradle.kts +11 -0
  2. package/compose/build.gradle.kts +186 -0
  3. package/compose/build.gradle.kts.backup +186 -0
  4. package/compose/compose.podspec +47 -0
  5. package/compose/src/androidMain/kotlin/vn/momo/kits/navigation/ScrollToTop.android.kt +6 -0
  6. package/compose/src/androidMain/kotlin/vn/momo/kits/platform/Platform.android.kt +123 -0
  7. package/compose/src/commonMain/composeResources/font/momosignature.otf +0 -0
  8. package/compose/src/commonMain/composeResources/font/momotrustdisplay.otf +0 -0
  9. package/compose/src/commonMain/composeResources/font/sfprotext_black.otf +0 -0
  10. package/compose/src/commonMain/composeResources/font/sfprotext_black.ttf +0 -0
  11. package/compose/src/commonMain/composeResources/font/sfprotext_bold.ttf +0 -0
  12. package/compose/src/commonMain/composeResources/font/sfprotext_heavy.ttf +0 -0
  13. package/compose/src/commonMain/composeResources/font/sfprotext_light.ttf +0 -0
  14. package/compose/src/commonMain/composeResources/font/sfprotext_medium.ttf +0 -0
  15. package/compose/src/commonMain/composeResources/font/sfprotext_regular.ttf +0 -0
  16. package/compose/src/commonMain/composeResources/font/sfprotext_semibold.ttf +0 -0
  17. package/compose/src/commonMain/composeResources/font/sfprotext_thin.otf +0 -0
  18. package/compose/src/commonMain/composeResources/font/sfprotext_thin.ttf +0 -0
  19. package/compose/src/commonMain/composeResources/font/sfprotext_ultralight.otf +0 -0
  20. package/compose/src/commonMain/composeResources/font/sfprotext_ultralight.ttf +0 -0
  21. package/compose/src/commonMain/kotlin/vn/momo/kits/application/AnimationSearchInput.kt +57 -0
  22. package/compose/src/commonMain/kotlin/vn/momo/kits/application/Context.kt +109 -0
  23. package/compose/src/commonMain/kotlin/vn/momo/kits/application/FloatingButton.kt +201 -0
  24. package/compose/src/commonMain/kotlin/vn/momo/kits/application/Header.kt +222 -0
  25. package/compose/src/commonMain/kotlin/vn/momo/kits/application/HeaderAnimated.kt +48 -0
  26. package/compose/src/commonMain/kotlin/vn/momo/kits/application/HeaderBackground.kt +86 -0
  27. package/compose/src/commonMain/kotlin/vn/momo/kits/application/HeaderDefault.kt +76 -0
  28. package/compose/src/commonMain/kotlin/vn/momo/kits/application/HeaderExtended.kt +76 -0
  29. package/compose/src/commonMain/kotlin/vn/momo/kits/application/HeaderRight.kt +305 -0
  30. package/compose/src/commonMain/kotlin/vn/momo/kits/application/HeaderTitle.kt +33 -0
  31. package/compose/src/commonMain/kotlin/vn/momo/kits/application/LiteScreen.kt +927 -0
  32. package/compose/src/commonMain/kotlin/vn/momo/kits/application/NavigationContainer.kt +121 -0
  33. package/compose/src/commonMain/kotlin/vn/momo/kits/application/Screen.kt +403 -0
  34. package/compose/src/commonMain/kotlin/vn/momo/kits/application/useHeaderSearchAnimation.kt +69 -0
  35. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Avatar.kt +157 -0
  36. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Badge.kt +85 -0
  37. package/compose/src/commonMain/kotlin/vn/momo/kits/components/BadgeDot.kt +32 -0
  38. package/compose/src/commonMain/kotlin/vn/momo/kits/components/BadgeRibbon.kt +340 -0
  39. package/compose/src/commonMain/kotlin/vn/momo/kits/components/BaselineView.kt +198 -0
  40. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Button.kt +357 -0
  41. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Carousel.kt +123 -0
  42. package/compose/src/commonMain/kotlin/vn/momo/kits/components/CheckBox.kt +94 -0
  43. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Chip.kt +136 -0
  44. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Collapse.kt +224 -0
  45. package/compose/src/commonMain/kotlin/vn/momo/kits/components/CupertinoOverscroll.kt +543 -0
  46. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Divider.kt +23 -0
  47. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Icon.kt +76 -0
  48. package/compose/src/commonMain/kotlin/vn/momo/kits/components/IconButton.kt +148 -0
  49. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Image.kt +188 -0
  50. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Information.kt +116 -0
  51. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Input.kt +452 -0
  52. package/compose/src/commonMain/kotlin/vn/momo/kits/components/InputDropDown.kt +172 -0
  53. package/compose/src/commonMain/kotlin/vn/momo/kits/components/InputMoney.kt +255 -0
  54. package/compose/src/commonMain/kotlin/vn/momo/kits/components/InputOTP.kt +235 -0
  55. package/compose/src/commonMain/kotlin/vn/momo/kits/components/InputPhoneNumber.kt +233 -0
  56. package/compose/src/commonMain/kotlin/vn/momo/kits/components/InputSearch.kt +259 -0
  57. package/compose/src/commonMain/kotlin/vn/momo/kits/components/InputTextArea.kt +241 -0
  58. package/compose/src/commonMain/kotlin/vn/momo/kits/components/LazyColumnWithBouncing.kt +364 -0
  59. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Loader.kt +108 -0
  60. package/compose/src/commonMain/kotlin/vn/momo/kits/components/PaginationDot.kt +56 -0
  61. package/compose/src/commonMain/kotlin/vn/momo/kits/components/PaginationNumber.kt +41 -0
  62. package/compose/src/commonMain/kotlin/vn/momo/kits/components/PaginationScroll.kt +92 -0
  63. package/compose/src/commonMain/kotlin/vn/momo/kits/components/PaginationWhiteDot.kt +40 -0
  64. package/compose/src/commonMain/kotlin/vn/momo/kits/components/PopupNotify.kt +352 -0
  65. package/compose/src/commonMain/kotlin/vn/momo/kits/components/PopupPromotion.kt +103 -0
  66. package/compose/src/commonMain/kotlin/vn/momo/kits/components/ProgressInfo.kt +338 -0
  67. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Radio.kt +70 -0
  68. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Rating.kt +87 -0
  69. package/compose/src/commonMain/kotlin/vn/momo/kits/components/ScaleSizeScope.kt +17 -0
  70. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Skeleton.kt +96 -0
  71. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Slider.kt +348 -0
  72. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Stepper.kt +256 -0
  73. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Steps.kt +494 -0
  74. package/compose/src/commonMain/kotlin/vn/momo/kits/components/SuggestAction.kt +131 -0
  75. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Swipe.kt +215 -0
  76. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Switch.kt +96 -0
  77. package/compose/src/commonMain/kotlin/vn/momo/kits/components/TabView.kt +531 -0
  78. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Tag.kt +92 -0
  79. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Text.kt +130 -0
  80. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Title.kt +214 -0
  81. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Tooltip.kt +590 -0
  82. package/compose/src/commonMain/kotlin/vn/momo/kits/components/TrustBanner.kt +177 -0
  83. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Uploader.kt +192 -0
  84. package/compose/src/commonMain/kotlin/vn/momo/kits/components/datetimepicker/DateTimePicker.kt +205 -0
  85. package/compose/src/commonMain/kotlin/vn/momo/kits/components/datetimepicker/DateTimePickerTypes.kt +29 -0
  86. package/compose/src/commonMain/kotlin/vn/momo/kits/components/datetimepicker/DateTimePickerUtils.kt +239 -0
  87. package/compose/src/commonMain/kotlin/vn/momo/kits/components/datetimepicker/WheelPicker.kt +191 -0
  88. package/compose/src/commonMain/kotlin/vn/momo/kits/const/Colors.kt +306 -0
  89. package/compose/src/commonMain/kotlin/vn/momo/kits/const/Radius.kt +12 -0
  90. package/compose/src/commonMain/kotlin/vn/momo/kits/const/Spacing.kt +16 -0
  91. package/compose/src/commonMain/kotlin/vn/momo/kits/const/Theme.kt +188 -0
  92. package/compose/src/commonMain/kotlin/vn/momo/kits/const/Typography.kt +285 -0
  93. package/compose/src/commonMain/kotlin/vn/momo/kits/layout/Card.kt +2 -0
  94. package/compose/src/commonMain/kotlin/vn/momo/kits/layout/Item.kt +35 -0
  95. package/compose/src/commonMain/kotlin/vn/momo/kits/layout/Section.kt +2 -0
  96. package/compose/src/commonMain/kotlin/vn/momo/kits/modifier/AutomationId.kt +57 -0
  97. package/compose/src/commonMain/kotlin/vn/momo/kits/modifier/Clickable.kt +68 -0
  98. package/compose/src/commonMain/kotlin/vn/momo/kits/modifier/Conditional.kt +11 -0
  99. package/compose/src/commonMain/kotlin/vn/momo/kits/modifier/DeprecatedModifier.kt +14 -0
  100. package/compose/src/commonMain/kotlin/vn/momo/kits/modifier/Shadow.kt +50 -0
  101. package/compose/src/commonMain/kotlin/vn/momo/kits/modifier/Size.kt +51 -0
  102. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/BottomSheet.kt +253 -0
  103. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/ModalScreen.kt +133 -0
  104. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/Navigation.kt +100 -0
  105. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/NavigationContainer.kt +147 -0
  106. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/Navigator.kt +345 -0
  107. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/ScrollToTop.kt +8 -0
  108. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/StackScreen.kt +558 -0
  109. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/bottomtab/BottomTab.kt +161 -0
  110. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/bottomtab/BottomTabBar.kt +243 -0
  111. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/bottomtab/CurvedContainer.kt +86 -0
  112. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/FloatingButton.kt +187 -0
  113. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/Header.kt +279 -0
  114. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/HeaderBackground.kt +80 -0
  115. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/HeaderRight.kt +306 -0
  116. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/HeaderTitle.kt +32 -0
  117. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/HeaderUser.kt +370 -0
  118. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/SnackBar.kt +131 -0
  119. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/tracking/ScreenTracker.kt +167 -0
  120. package/compose/src/commonMain/kotlin/vn/momo/kits/platform/ComposeLottieAnimation.kt +62 -0
  121. package/compose/src/commonMain/kotlin/vn/momo/kits/platform/Platform.kt +66 -0
  122. package/compose/src/commonMain/kotlin/vn/momo/kits/utils/Icons.kt +1329 -0
  123. package/compose/src/commonMain/kotlin/vn/momo/kits/utils/Resources.kt +70 -0
  124. package/compose/src/commonMain/kotlin/vn/momo/kits/utils/Tracking.kt +15 -0
  125. package/compose/src/commonMain/kotlin/vn/momo/kits/utils/Utils.kt +105 -0
  126. package/compose/src/iosMain/kotlin/vn/momo/kits/navigation/ScrollToTop.ios.kt +33 -0
  127. package/compose/src/iosMain/kotlin/vn/momo/kits/platform/Platform.ios.kt +186 -0
  128. package/gradle/libs.versions.toml +60 -0
  129. package/gradle/wrapper/gradle-wrapper.jar +0 -0
  130. package/gradle/wrapper/gradle-wrapper.properties +8 -0
  131. package/gradle.properties +26 -0
  132. package/gradlew +252 -0
  133. package/gradlew.bat +94 -0
  134. package/local.properties +8 -0
  135. package/package.json +1 -1
  136. package/settings.gradle.kts +64 -0
@@ -0,0 +1,558 @@
1
+ package vn.momo.kits.navigation
2
+
3
+ import androidx.compose.animation.animateColorAsState
4
+ import androidx.compose.animation.core.animateDpAsState
5
+ import androidx.compose.foundation.ScrollState
6
+ import androidx.compose.foundation.background
7
+ import androidx.compose.foundation.border
8
+ import androidx.compose.foundation.gestures.ScrollableState
9
+ import androidx.compose.foundation.layout.Box
10
+ import androidx.compose.foundation.layout.Column
11
+ import androidx.compose.foundation.layout.ColumnScope
12
+ import androidx.compose.foundation.layout.Spacer
13
+ import androidx.compose.foundation.layout.WindowInsets
14
+ import androidx.compose.foundation.layout.aspectRatio
15
+ import androidx.compose.foundation.layout.fillMaxSize
16
+ import androidx.compose.foundation.layout.fillMaxWidth
17
+ import androidx.compose.foundation.layout.height
18
+ import androidx.compose.foundation.layout.ime
19
+ import androidx.compose.foundation.layout.imePadding
20
+ import androidx.compose.foundation.layout.offset
21
+ import androidx.compose.foundation.layout.padding
22
+ import androidx.compose.foundation.lazy.LazyListState
23
+ import androidx.compose.foundation.lazy.rememberLazyListState
24
+ import androidx.compose.foundation.rememberScrollState
25
+ import androidx.compose.foundation.verticalScroll
26
+ import androidx.compose.material.ExperimentalMaterialApi
27
+ import androidx.compose.runtime.Composable
28
+ import androidx.compose.runtime.CompositionLocalProvider
29
+ import androidx.compose.runtime.LaunchedEffect
30
+ import androidx.compose.runtime.getValue
31
+ import androidx.compose.runtime.mutableIntStateOf
32
+ import androidx.compose.runtime.mutableStateOf
33
+ import androidx.compose.runtime.remember
34
+ import androidx.compose.runtime.snapshotFlow
35
+ import androidx.compose.runtime.staticCompositionLocalOf
36
+ import androidx.compose.ui.Alignment
37
+ import androidx.compose.ui.Modifier
38
+ import androidx.compose.ui.graphics.Brush
39
+ import androidx.compose.ui.graphics.Color
40
+ import androidx.compose.ui.graphics.lerp
41
+ import androidx.compose.ui.layout.onGloballyPositioned
42
+ import androidx.compose.ui.platform.LocalDensity
43
+ import androidx.compose.ui.unit.Dp
44
+ import androidx.compose.ui.unit.dp
45
+ import androidx.compose.ui.unit.min
46
+ import androidx.compose.ui.zIndex
47
+ import vn.momo.kits.components.InputSearch
48
+ import vn.momo.kits.const.AppNavigationBar
49
+ import vn.momo.kits.const.AppStatusBar
50
+ import vn.momo.kits.const.AppTheme
51
+ import vn.momo.kits.application.IsShowBaseLineDebug
52
+ import vn.momo.kits.const.Colors
53
+ import vn.momo.kits.const.Spacing
54
+ import vn.momo.kits.modifier.conditional
55
+ import vn.momo.kits.modifier.hideKeyboardOnTap
56
+ import vn.momo.kits.navigation.component.FABPosition
57
+ import vn.momo.kits.navigation.component.FloatingButton
58
+ import vn.momo.kits.navigation.component.HEADER_HEIGHT
59
+ import vn.momo.kits.navigation.component.Header
60
+ import vn.momo.kits.navigation.component.HeaderBackground
61
+ import vn.momo.kits.navigation.component.HeaderRight
62
+ import vn.momo.kits.navigation.component.HeaderType
63
+ import vn.momo.kits.navigation.component.InputSearchType
64
+ import vn.momo.kits.platform.BackHandler
65
+ import vn.momo.kits.platform.supportsImePadding
66
+ import vn.momo.kits.navigation.tracking.ScreenTracker
67
+ import vn.momo.kits.navigation.tracking.ScreenTrackingState
68
+ import kotlinx.coroutines.delay
69
+ import vn.momo.kits.application.ApplicationContext
70
+ import kotlin.time.Clock
71
+ import kotlin.time.ExperimentalTime
72
+
73
+ internal val LocalFooterHeightPx = staticCompositionLocalOf { mutableIntStateOf(0) }
74
+ internal val LocalHeaderRightWidthPx = staticCompositionLocalOf { mutableIntStateOf(0) }
75
+
76
+ @OptIn(ExperimentalMaterialApi::class, ExperimentalTime::class)
77
+ @Composable
78
+ internal fun StackScreen(
79
+ content: @Composable () -> Unit,
80
+ navigationOptions: NavigationOptions? = null,
81
+ id: Int = -1,
82
+ name: String = "",
83
+ bottomTabIndex: Int = -1,
84
+ onBackHandler: (() -> Unit)? = null,
85
+ ) {
86
+ val navigator = LocalNavigator.current
87
+ val maxApi = LocalMaxApi.current
88
+ val context = ApplicationContext.current
89
+ val statusBar = AppStatusBar.current
90
+ val density = LocalDensity.current
91
+ val navigation = remember { Navigation(id = id, bottomTabIndex = bottomTabIndex, initOptions = navigationOptions) }
92
+
93
+ val options by navigation.currentOptions
94
+
95
+ // Auto tracking state
96
+ val trackingState = remember {
97
+ ScreenTrackingState().apply {
98
+ startTime = Clock.System.now().toEpochMilliseconds()
99
+ }
100
+ }
101
+
102
+ // Determine action: push or back
103
+ val action = remember {
104
+ if (ScreenTracker.getLastScreenName() != null) "push" else "back"
105
+ }
106
+
107
+ // Auto tracking effects
108
+ LaunchedEffect(name) {
109
+ // Track screen navigated immediately
110
+ ScreenTracker.trackScreenNavigated(
111
+ maxApi = maxApi,
112
+ context = context,
113
+ screenName = name,
114
+ action = action,
115
+ state = trackingState
116
+ )
117
+
118
+ // Track screen displayed after 2 seconds (debounce)
119
+ delay(2000)
120
+ val loadTime = Clock.System.now().toEpochMilliseconds() - trackingState.startTime
121
+ ScreenTracker.trackScreenDisplayed(
122
+ maxApi = maxApi,
123
+ context = context,
124
+ screenName = name,
125
+ duration = loadTime,
126
+ state = trackingState
127
+ )
128
+
129
+ // Track screen interacted after displayed
130
+ val interactionTime = Clock.System.now().toEpochMilliseconds() - trackingState.startTime
131
+ ScreenTracker.trackScreenInteracted(
132
+ maxApi = maxApi,
133
+ context = context,
134
+ screenName = name,
135
+ duration = interactionTime - loadTime,
136
+ totalDuration = interactionTime,
137
+ state = trackingState
138
+ )
139
+ }
140
+
141
+ val limit = with(density) {
142
+ (statusBar).toPx() + HEADER_HEIGHT
143
+ }.toInt()
144
+
145
+ val (scrollState, scrollInProcess) = if (options.scrollData.scrollState is LazyListState)
146
+ (options.scrollData.scrollState as? LazyListState ?: rememberLazyListState()).proxyScrollState(limit, 15)
147
+ else
148
+ (options.scrollData.scrollState as? ScrollState ?: rememberScrollState()).proxyLimitedScrollState(limit, 15)
149
+
150
+ val footerHeightPx = remember { mutableIntStateOf(0) }
151
+ val headerRightWidthPx = remember { mutableIntStateOf(0) }
152
+
153
+ if (navigation.options.scrollData.scrollToTopCallback != null) {
154
+ RegisterScrollToTop(navigation.options.scrollData.scrollToTopCallback)
155
+ }
156
+
157
+ BackHandler(true) { navigator.onBackSafe() }
158
+
159
+ CompositionLocalProvider(
160
+ StackScreenScrollableState provides options.scrollData.scrollState,
161
+ LocalNavigation provides navigation,
162
+ LocalScrollState provides scrollState,
163
+ LocalOptions provides options,
164
+ LocalFooterHeightPx provides footerHeightPx,
165
+ LocalHeaderRightWidthPx provides headerRightWidthPx
166
+ ) {
167
+ Box(
168
+ Modifier
169
+ .fillMaxSize()
170
+ .background(options.backgroundColor ?: AppTheme.current.colors.background.default)
171
+ .conditional(options.keyboardOptions.keyboardShouldPersistTaps) {
172
+ hideKeyboardOnTap()
173
+ }
174
+ .conditional(options.keyboardOptions.useAvoidKeyboard && supportsImePadding()) {
175
+ imePadding()
176
+ }
177
+ ) {
178
+ Box(Modifier.zIndex(1f)) {
179
+ HeaderBackground()
180
+ }
181
+
182
+ Box(Modifier.zIndex(5f)) {
183
+ Header(onBackHandler)
184
+ }
185
+
186
+ Column(Modifier.zIndex(6f)) {
187
+ SearchAnimated(isScrollInProgress = scrollInProcess)
188
+ }
189
+
190
+ Column(Modifier.zIndex(2f).fillMaxSize()) {
191
+ MainContent(content = content)
192
+ FooterContent()
193
+ }
194
+
195
+ Box(Modifier.zIndex(7f)) {
196
+ FloatingContent()
197
+ }
198
+
199
+ OverplayView(bottomTabIndex = bottomTabIndex, id = id)
200
+ }
201
+ }
202
+ }
203
+
204
+ @Composable
205
+ fun FloatingContent() {
206
+ val options = LocalOptions.current
207
+ val density = LocalDensity.current
208
+ val footerHeightPx = LocalFooterHeightPx.current
209
+ val scrollState = LocalScrollState.current
210
+
211
+ val fabProps = options.floatingButtonProps ?: return
212
+ val bottomPadding = fabProps.bottom
213
+ ?: (Spacing.M + if (options.footerComponent != null) with(density) { footerHeightPx.intValue.toDp() } else 0.dp)
214
+
215
+ FloatingButton(
216
+ scrollPosition = fabProps.scrollState?.value ?: scrollState.value,
217
+ onClick = fabProps.onClick,
218
+ containerColor = AppTheme.current.colors.primary,
219
+ bottom = bottomPadding,
220
+ icon = fabProps.icon,
221
+ iconColor = fabProps.iconColor,
222
+ text = fabProps.label,
223
+ size = fabProps.size,
224
+ position = fabProps.position ?: FABPosition.END,
225
+ )
226
+ }
227
+
228
+ @Composable
229
+ fun ColumnScope.MainContent(content: @Composable () -> Unit) {
230
+ val options = LocalOptions.current
231
+ val inputSearchType = getInputSearchType(options)
232
+ val density = LocalDensity.current
233
+ val scrollState = LocalScrollState.current
234
+
235
+ Spacer(
236
+ Modifier.height(
237
+ if (options.headerType is HeaderType.DefaultOrExtended || (options.headerType is HeaderType.Transparent && !options.headerType.isFullScreenContent))
238
+ AppStatusBar.current + HEADER_HEIGHT.dp else 0.dp
239
+ )
240
+ )
241
+ if (inputSearchType == InputSearchType.Animated) {
242
+ val scrollDp = with(density) { scrollState.value.toDp() }
243
+
244
+ val animatedTopPadding by animateDpAsState(
245
+ targetValue = (HEADER_HEIGHT.dp - scrollDp).coerceIn(0.dp, HEADER_HEIGHT.dp),
246
+ label = "AnimatedTopPadding"
247
+ )
248
+ Spacer(Modifier.height(animatedTopPadding))
249
+ }
250
+ Column(
251
+ Modifier
252
+ .fillMaxWidth()
253
+ .weight(1f)
254
+ .conditional(options.scrollData.scrollable && options.scrollData.scrollState is ScrollState) {
255
+ verticalScroll(scrollState)
256
+ }
257
+ ) {
258
+ ScreenContent(content = content)
259
+ }
260
+
261
+ }
262
+
263
+ @Composable
264
+ fun ScreenContent(content: @Composable () -> Unit) {
265
+ val scrollState = LocalScrollState.current
266
+ val options = LocalOptions.current
267
+
268
+ if (options.headerType is HeaderType.Animated) {
269
+ val animatedHeader = options.headerType
270
+ Box {
271
+ Box(Modifier.fillMaxWidth().aspectRatio(animatedHeader.aspectRatio.value)) {
272
+ animatedHeader.composable.invoke(scrollState.value)
273
+ }
274
+ Box(Modifier.offset(x = 0.dp, y = AppStatusBar.current + HEADER_HEIGHT.dp + animatedHeader.layoutOffSet)) {
275
+ content()
276
+ }
277
+ }
278
+ } else {
279
+ content()
280
+ }
281
+ }
282
+
283
+ @Composable
284
+ fun FooterContent() {
285
+ val options = LocalOptions.current
286
+ if (options.footerComponent != null) {
287
+ val ime = WindowInsets.ime
288
+ val density = LocalDensity.current
289
+ val imeBottom = ime.getBottom(density)
290
+ val thresholdPx = with(density) { 50.dp.toPx() }
291
+ val isKeyboardVisible = imeBottom > thresholdPx
292
+ val bottomPadding = if (isKeyboardVisible) 0.dp else AppNavigationBar.current
293
+ Footer(footerComponent = options.footerComponent, bottomPadding = bottomPadding)
294
+ }
295
+ }
296
+
297
+ @Composable
298
+ fun OverplayView(bottomTabIndex: Int, id: Int) {
299
+ val overplayType = OverplayComponentRegistry.getOverplayType()
300
+
301
+ if (overplayType != null) {
302
+ Box(Modifier.zIndex(if (overplayType == OverplayComponentType.SNACK_BAR) 3f else 8f).fillMaxSize()) {
303
+ if (bottomTabIndex != -1) return@Box
304
+ if (OverplayComponentRegistry.currentRootId() != id) return@Box
305
+ OverplayComponentRegistry.OverlayComponent()
306
+ }
307
+ }
308
+ }
309
+
310
+ @Composable
311
+ fun Footer(footerComponent: @Composable (() -> Unit)?, bottomPadding: Dp) {
312
+ if (footerComponent == null) return
313
+
314
+ val footerHeightPx = LocalFooterHeightPx.current
315
+
316
+ val shadowBrush = remember {
317
+ Brush.verticalGradient(
318
+ colors = listOf(Color.Transparent, Color.Black.copy(alpha = 0.05f))
319
+ )
320
+ }
321
+
322
+ Box(Modifier.onGloballyPositioned {
323
+ if (footerHeightPx.intValue != it.size.height) footerHeightPx.intValue = it.size.height
324
+ }) {
325
+ Box(
326
+ Modifier
327
+ .fillMaxWidth()
328
+ .background(AppTheme.current.colors.background.surface)
329
+ .conditional(IsShowBaseLineDebug) {
330
+ border(1.dp, Colors.blue_03)
331
+ }
332
+ .padding(top = Spacing.S, start = Spacing.M, end = Spacing.M, bottom = bottomPadding + Spacing.S)
333
+ ) {
334
+ footerComponent.invoke()
335
+ }
336
+
337
+ Box(
338
+ modifier = Modifier
339
+ .fillMaxWidth()
340
+ .height(6.dp)
341
+ .offset(x = 0.dp, y = (-6).dp)
342
+ .background(shadowBrush)
343
+ )
344
+ }
345
+ }
346
+
347
+ data class InputSearchLayoutParams(
348
+ val topPadding: Dp,
349
+ val startPadding: Dp,
350
+ val endPadding: Dp
351
+ )
352
+
353
+ @Composable
354
+ fun SearchAnimated(
355
+ isScrollInProgress: Boolean
356
+ ) {
357
+ val scrollState = LocalScrollState.current
358
+ val options = LocalOptions.current
359
+ val navigator = LocalNavigator.current
360
+ val density = LocalDensity.current
361
+
362
+ val scrollDp = with(density) { scrollState.value.toDp() }
363
+
364
+ val inputSearchType = getInputSearchType(options)
365
+ val headerRightWidthPx = LocalHeaderRightWidthPx.current
366
+ val headerRightWidthDp = with(density) { headerRightWidthPx.intValue.toDp() }
367
+
368
+ if (inputSearchType == InputSearchType.None) return
369
+ val inputSearchProps = (options.headerType as? HeaderType.DefaultOrExtended)?.inputSearchProps ?: return
370
+
371
+ val minTopPadding = AppStatusBar.current
372
+ val maxTopPadding = AppStatusBar.current + HEADER_HEIGHT.dp
373
+ val minStartPadding = Spacing.M
374
+ val maxStartPadding = if (options.hiddenBack) Spacing.M else 52.dp
375
+ val minEndPadding = Spacing.M
376
+ val maxEndPadding = headerRightWidthDp + if (options.headerRight is HeaderRight.None) Spacing.M else Spacing.M * 2
377
+
378
+ val targetColor = lerp(
379
+ AppTheme.current.colors.background.surface,
380
+ AppTheme.current.colors.background.default,
381
+ (scrollDp / minTopPadding).coerceIn(0f, 1f)
382
+ )
383
+ val animatedColor by animateColorAsState(targetValue = targetColor, label = "BackgroundColor")
384
+
385
+ val layoutParams = when (inputSearchType) {
386
+ InputSearchType.Header -> {
387
+ InputSearchLayoutParams(
388
+ topPadding = minTopPadding,
389
+ startPadding = maxStartPadding,
390
+ endPadding = maxEndPadding
391
+ )
392
+ }
393
+
394
+ InputSearchType.Animated -> {
395
+ val animatedTopPadding by animateDpAsState(
396
+ targetValue = (maxTopPadding - scrollDp).coerceIn(minTopPadding, maxTopPadding),
397
+ label = "AnimatedTopPadding"
398
+ )
399
+
400
+ val animatedStartPadding by animateDpAsState(
401
+ targetValue = (minStartPadding + scrollDp).coerceIn(minStartPadding, maxStartPadding),
402
+ label = "AnimatedStartPadding"
403
+ )
404
+
405
+ val animatedEndPadding by animateDpAsState(
406
+ targetValue = (minEndPadding + scrollDp).coerceIn(minEndPadding, maxEndPadding),
407
+ label = "AnimatedEndPadding"
408
+ )
409
+
410
+ val maxPadding = remember(maxTopPadding, maxStartPadding, maxEndPadding) {
411
+ maxOf(maxEndPadding, maxStartPadding, maxTopPadding)
412
+ }
413
+ val snapScrollValue = with(density) { maxPadding.toPx().toInt() }
414
+
415
+ LaunchedEffect(isScrollInProgress && scrollState.value != 0 && scrollState.value != snapScrollValue) {
416
+ if (scrollDp < maxPadding) {
417
+ if (!isScrollInProgress) {
418
+ val midpoint = (maxTopPadding - minTopPadding) / 2
419
+
420
+ if (scrollDp < midpoint) {
421
+ scrollState.animateScrollTo(0)
422
+ } else {
423
+ scrollState.animateScrollTo(snapScrollValue)
424
+ }
425
+ }
426
+ }
427
+ }
428
+
429
+ InputSearchLayoutParams(
430
+ topPadding = animatedTopPadding,
431
+ startPadding = animatedStartPadding,
432
+ endPadding = animatedEndPadding
433
+ )
434
+ }
435
+
436
+ InputSearchType.None -> return
437
+ }
438
+
439
+ Spacer(Modifier.height(layoutParams.topPadding))
440
+ Box(
441
+ modifier = Modifier
442
+ .height(HEADER_HEIGHT.dp)
443
+ .fillMaxWidth()
444
+ .padding(
445
+ start = layoutParams.startPadding,
446
+ end = layoutParams.endPadding
447
+ ),
448
+ contentAlignment = Alignment.Center
449
+ ) {
450
+ InputSearch(
451
+ inputSearchProps = inputSearchProps.copy(
452
+ backgroundColor = animatedColor,
453
+ onPressButtonText = {
454
+ navigator.pop()
455
+ }
456
+ )
457
+ )
458
+ }
459
+ }
460
+
461
+ @Composable
462
+ internal fun isKeyboardVisible(): Boolean {
463
+ val ime = WindowInsets.ime
464
+ val density = LocalDensity.current
465
+ return ime.getBottom(density) > 0
466
+ }
467
+
468
+ private fun quantize(value: Int, stepPx: Int): Int {
469
+ if (stepPx <= 1) return value
470
+ return (value / stepPx) * stepPx
471
+ }
472
+
473
+ @Composable
474
+ fun LazyListState.proxyScrollState(
475
+ thresholdPx: Int = 200,
476
+ stepPx: Int = 4
477
+ ): Pair<ScrollState, Boolean> {
478
+ val scrollState = rememberScrollState()
479
+ val locked = remember { mutableStateOf(false) }
480
+
481
+ LaunchedEffect(this, scrollState, thresholdPx, stepPx) {
482
+ snapshotFlow { firstVisibleItemIndex to firstVisibleItemScrollOffset }
483
+ .collect { (index, offset) ->
484
+ val rawTarget = if (index == 0) offset else SCROLLED_PAST_FIRST_ITEM_THRESHOLD
485
+ val shouldLock = rawTarget > thresholdPx
486
+ if (locked.value != shouldLock) locked.value = shouldLock
487
+
488
+ val desired = if (shouldLock) thresholdPx else quantize(rawTarget, stepPx)
489
+
490
+ if (scrollState.value != desired) {
491
+ scrollState.scrollTo(desired.coerceAtLeast(0))
492
+ }
493
+ }
494
+ }
495
+
496
+ val inProgress = remember { mutableStateOf(false) }
497
+ LaunchedEffect(this, thresholdPx) {
498
+ snapshotFlow { isScrollInProgress }
499
+ .collect { progressing ->
500
+ inProgress.value = if (locked.value) false else progressing
501
+ }
502
+ }
503
+
504
+ return scrollState to inProgress.value
505
+ }
506
+
507
+ @Composable
508
+ fun ScrollState.proxyLimitedScrollState(
509
+ thresholdPx: Int = 200,
510
+ stepPx: Int = 4
511
+ ): Pair<ScrollState, Boolean> {
512
+ val proxy = rememberScrollState()
513
+ val locked = remember { mutableStateOf(false) }
514
+
515
+ LaunchedEffect(this, proxy, thresholdPx, stepPx) {
516
+ snapshotFlow { value }
517
+ .collect { rawTarget ->
518
+ val shouldLock = rawTarget > thresholdPx
519
+ if (locked.value != shouldLock) locked.value = shouldLock
520
+
521
+ val desired = if (shouldLock) thresholdPx else quantize(rawTarget, stepPx)
522
+
523
+ if (proxy.value != desired) {
524
+ proxy.scrollTo(desired.coerceAtLeast(0))
525
+ }
526
+ }
527
+ }
528
+
529
+ val inProgress = remember { mutableStateOf(false) }
530
+ LaunchedEffect(this, thresholdPx) {
531
+ snapshotFlow { isScrollInProgress }
532
+ .collect { progressing ->
533
+ inProgress.value = if (locked.value) false else progressing
534
+ }
535
+ }
536
+
537
+ return proxy to inProgress.value
538
+ }
539
+
540
+ private const val SCROLLED_PAST_FIRST_ITEM_THRESHOLD = 10_000
541
+
542
+ internal val LocalScrollState = staticCompositionLocalOf<ScrollState> { error("No Scroll State provided") }
543
+ internal val LocalOptions = staticCompositionLocalOf<NavigationOptions> { error("No NavigationOptions provided") }
544
+
545
+ val StackScreenScrollableState = staticCompositionLocalOf<ScrollableState?> { null }
546
+
547
+ internal fun getInputSearchType(options: NavigationOptions): InputSearchType {
548
+ return when (val headerType = options.headerType) {
549
+ is HeaderType.DefaultOrExtended -> when {
550
+ headerType.inputSearchProps == null -> InputSearchType.None
551
+ headerType.useAnimated -> InputSearchType.Animated
552
+ else -> InputSearchType.Header
553
+ }
554
+
555
+ else -> InputSearchType.None
556
+ }
557
+ }
558
+