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