@momo-kits/native-kits 0.152.4-beta.6 → 0.152.4-beta.7

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 (156) hide show
  1. package/CODE_OF_CONDUCT.md +133 -0
  2. package/CONTRIBUTING.md +114 -0
  3. package/LICENSE +20 -0
  4. package/README.md +7 -0
  5. package/build.gradle.kts +32 -0
  6. package/compose/MoMoComposeKits.podspec +54 -0
  7. package/compose/build.gradle.kts +149 -0
  8. package/compose/src/androidMain/AndroidManifest.xml +2 -0
  9. package/compose/src/androidMain/kotlin/vn/momo/kits/platform/Platform.android.kt +105 -0
  10. package/compose/src/commonMain/composeResources/files/lottie_circle_loader.json +1 -0
  11. package/compose/src/commonMain/composeResources/font/momosignature.otf +0 -0
  12. package/compose/src/commonMain/composeResources/font/momotrustdisplay.otf +0 -0
  13. package/compose/src/commonMain/composeResources/font/sfprotext_black.otf +0 -0
  14. package/compose/src/commonMain/composeResources/font/sfprotext_black.ttf +0 -0
  15. package/compose/src/commonMain/composeResources/font/sfprotext_bold.ttf +0 -0
  16. package/compose/src/commonMain/composeResources/font/sfprotext_heavy.ttf +0 -0
  17. package/compose/src/commonMain/composeResources/font/sfprotext_light.ttf +0 -0
  18. package/compose/src/commonMain/composeResources/font/sfprotext_medium.ttf +0 -0
  19. package/compose/src/commonMain/composeResources/font/sfprotext_regular.ttf +0 -0
  20. package/compose/src/commonMain/composeResources/font/sfprotext_semibold.ttf +0 -0
  21. package/compose/src/commonMain/composeResources/font/sfprotext_thin.otf +0 -0
  22. package/compose/src/commonMain/composeResources/font/sfprotext_thin.ttf +0 -0
  23. package/compose/src/commonMain/composeResources/font/sfprotext_ultralight.otf +0 -0
  24. package/compose/src/commonMain/composeResources/font/sfprotext_ultralight.ttf +0 -0
  25. package/compose/src/commonMain/kotlin/vn/momo/kits/application/AnimationSearchInput.kt +57 -0
  26. package/compose/src/commonMain/kotlin/vn/momo/kits/application/FloatingButton.kt +201 -0
  27. package/compose/src/commonMain/kotlin/vn/momo/kits/application/Header.kt +222 -0
  28. package/compose/src/commonMain/kotlin/vn/momo/kits/application/HeaderAnimated.kt +48 -0
  29. package/compose/src/commonMain/kotlin/vn/momo/kits/application/HeaderBackground.kt +86 -0
  30. package/compose/src/commonMain/kotlin/vn/momo/kits/application/HeaderDefault.kt +76 -0
  31. package/compose/src/commonMain/kotlin/vn/momo/kits/application/HeaderExtended.kt +76 -0
  32. package/compose/src/commonMain/kotlin/vn/momo/kits/application/HeaderRight.kt +306 -0
  33. package/compose/src/commonMain/kotlin/vn/momo/kits/application/HeaderTitle.kt +33 -0
  34. package/compose/src/commonMain/kotlin/vn/momo/kits/application/LiteScreen.kt +715 -0
  35. package/compose/src/commonMain/kotlin/vn/momo/kits/application/NavigationContainer.kt +214 -0
  36. package/compose/src/commonMain/kotlin/vn/momo/kits/application/Screen.kt +392 -0
  37. package/compose/src/commonMain/kotlin/vn/momo/kits/application/useHeaderSearchAnimation.kt +69 -0
  38. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Badge.kt +77 -0
  39. package/compose/src/commonMain/kotlin/vn/momo/kits/components/BadgeDot.kt +27 -0
  40. package/compose/src/commonMain/kotlin/vn/momo/kits/components/BadgeRibbon.kt +334 -0
  41. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Button.kt +345 -0
  42. package/compose/src/commonMain/kotlin/vn/momo/kits/components/CheckBox.kt +90 -0
  43. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Chip.kt +131 -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 +58 -0
  47. package/compose/src/commonMain/kotlin/vn/momo/kits/components/IconButton.kt +143 -0
  48. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Image.kt +179 -0
  49. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Information.kt +111 -0
  50. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Input.kt +384 -0
  51. package/compose/src/commonMain/kotlin/vn/momo/kits/components/InputDropDown.kt +160 -0
  52. package/compose/src/commonMain/kotlin/vn/momo/kits/components/InputMoney.kt +234 -0
  53. package/compose/src/commonMain/kotlin/vn/momo/kits/components/InputOTP.kt +223 -0
  54. package/compose/src/commonMain/kotlin/vn/momo/kits/components/InputPhoneNumber.kt +232 -0
  55. package/compose/src/commonMain/kotlin/vn/momo/kits/components/InputSearch.kt +236 -0
  56. package/compose/src/commonMain/kotlin/vn/momo/kits/components/InputTextArea.kt +228 -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/PaginationDot.kt +50 -0
  59. package/compose/src/commonMain/kotlin/vn/momo/kits/components/PaginationNumber.kt +34 -0
  60. package/compose/src/commonMain/kotlin/vn/momo/kits/components/PaginationScroll.kt +85 -0
  61. package/compose/src/commonMain/kotlin/vn/momo/kits/components/PaginationWhiteDot.kt +33 -0
  62. package/compose/src/commonMain/kotlin/vn/momo/kits/components/PopupNotify.kt +338 -0
  63. package/compose/src/commonMain/kotlin/vn/momo/kits/components/PopupPromotion.kt +95 -0
  64. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Radio.kt +64 -0
  65. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Skeleton.kt +89 -0
  66. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Switch.kt +91 -0
  67. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Tag.kt +86 -0
  68. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Text.kt +84 -0
  69. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Title.kt +208 -0
  70. package/compose/src/commonMain/kotlin/vn/momo/kits/components/TrustBanner.kt +172 -0
  71. package/compose/src/commonMain/kotlin/vn/momo/kits/components/datetimepicker/DateTimePicker.kt +199 -0
  72. package/compose/src/commonMain/kotlin/vn/momo/kits/components/datetimepicker/DateTimePickerTypes.kt +29 -0
  73. package/compose/src/commonMain/kotlin/vn/momo/kits/components/datetimepicker/DateTimePickerUtils.kt +237 -0
  74. package/compose/src/commonMain/kotlin/vn/momo/kits/components/datetimepicker/WheelPicker.kt +191 -0
  75. package/compose/src/commonMain/kotlin/vn/momo/kits/const/Colors.kt +306 -0
  76. package/compose/src/commonMain/kotlin/vn/momo/kits/const/Radius.kt +12 -0
  77. package/compose/src/commonMain/kotlin/vn/momo/kits/const/Spacing.kt +13 -0
  78. package/compose/src/commonMain/kotlin/vn/momo/kits/const/Theme.kt +191 -0
  79. package/compose/src/commonMain/kotlin/vn/momo/kits/const/Typography.kt +258 -0
  80. package/compose/src/commonMain/kotlin/vn/momo/kits/layout/Card.kt +2 -0
  81. package/compose/src/commonMain/kotlin/vn/momo/kits/layout/Item.kt +35 -0
  82. package/compose/src/commonMain/kotlin/vn/momo/kits/layout/Section.kt +2 -0
  83. package/compose/src/commonMain/kotlin/vn/momo/kits/modifier/AutomationId.kt +59 -0
  84. package/compose/src/commonMain/kotlin/vn/momo/kits/modifier/Clickable.kt +68 -0
  85. package/compose/src/commonMain/kotlin/vn/momo/kits/modifier/Conditional.kt +11 -0
  86. package/compose/src/commonMain/kotlin/vn/momo/kits/modifier/Shadow.kt +49 -0
  87. package/compose/src/commonMain/kotlin/vn/momo/kits/modifier/Size.kt +51 -0
  88. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/BottomSheet.kt +232 -0
  89. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/ModalScreen.kt +111 -0
  90. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/Navigation.kt +94 -0
  91. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/NavigationContainer.kt +159 -0
  92. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/Navigator.kt +302 -0
  93. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/ScaleSizeScope.kt +17 -0
  94. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/StackScreen.kt +484 -0
  95. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/bottomtab/BottomTab.kt +169 -0
  96. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/bottomtab/BottomTabBar.kt +216 -0
  97. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/bottomtab/CurvedContainer.kt +86 -0
  98. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/FloatingButton.kt +180 -0
  99. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/Header.kt +251 -0
  100. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/HeaderBackground.kt +80 -0
  101. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/HeaderRight.kt +306 -0
  102. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/HeaderTitle.kt +31 -0
  103. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/HeaderUser.kt +385 -0
  104. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/SnackBar.kt +123 -0
  105. package/compose/src/commonMain/kotlin/vn/momo/kits/platform/Platform.kt +38 -0
  106. package/compose/src/commonMain/kotlin/vn/momo/kits/utils/Icons.kt +1329 -0
  107. package/compose/src/commonMain/kotlin/vn/momo/kits/utils/Resources.kt +62 -0
  108. package/compose/src/commonMain/kotlin/vn/momo/kits/utils/Utils.kt +88 -0
  109. package/compose/src/iosMain/kotlin/vn/momo/kits/platform/Platform.ios.kt +144 -0
  110. package/gradle.properties +19 -0
  111. package/gradlew +240 -0
  112. package/gradlew.bat +91 -0
  113. package/ios/Application/ApplicationEnvironment.swift +50 -0
  114. package/ios/Application/Components.swift +263 -0
  115. package/ios/Application/ComposeApi.swift +22 -0
  116. package/ios/Application/FloatingButton.swift +172 -0
  117. package/ios/Application/HeaderRight.swift +271 -0
  118. package/ios/Application/Screen.swift +249 -0
  119. package/ios/Badge/BadgeDot.swift +31 -0
  120. package/ios/Button/Button.swift +211 -0
  121. package/ios/CalculatorKeyboard/CalculatorKeyboard.swift +126 -0
  122. package/ios/Checkbox/Checkbox.swift +81 -0
  123. package/ios/Chip/Chip.swift +96 -0
  124. package/ios/Colors+Radius+Spacing/Colors.swift +172 -0
  125. package/ios/Colors+Radius+Spacing/Radius.swift +22 -0
  126. package/ios/Colors+Radius+Spacing/Spacing.swift +12 -0
  127. package/ios/Extensions/Color++.swift +25 -0
  128. package/ios/Icon/Icon.swift +51 -0
  129. package/ios/Image/Image.swift +70 -0
  130. package/ios/Input/Input.swift +207 -0
  131. package/ios/Input/InputPhoneNumber.swift +176 -0
  132. package/ios/Input/InputSearch.swift +238 -0
  133. package/ios/Input/InputTextArea.swift +242 -0
  134. package/ios/Lottie/LottieView.swift +86 -0
  135. package/ios/OTPKeyboard/KeyboardButton.swift +41 -0
  136. package/ios/OTPKeyboard/OTPKeyboard.swift +145 -0
  137. package/ios/Popup/PopupDisplay.swift +284 -0
  138. package/ios/Popup/PopupInput.swift +96 -0
  139. package/ios/Popup/PopupPromotion.swift +73 -0
  140. package/ios/PopupView/FullscreenPopup.swift +251 -0
  141. package/ios/PopupView/Modifiers.swift +158 -0
  142. package/ios/PopupView/PopupView.swift +289 -0
  143. package/ios/PopupView/Utils++.swift +281 -0
  144. package/ios/ScrollIndicator/ScrollIndicator.swift +110 -0
  145. package/ios/Swipeable/SwipeCell.swift +278 -0
  146. package/ios/Swipeable/SwipeCellModel.swift +86 -0
  147. package/ios/Switch/Switch.swift +44 -0
  148. package/ios/Template/Logo/Logo.swift +75 -0
  149. package/ios/Template/TrustBanner/TrustBanner.swift +120 -0
  150. package/ios/Theme.md +18 -0
  151. package/ios/Typography/Text.swift +140 -0
  152. package/ios/Typography/Typography.swift +95 -0
  153. package/ios/native-kits.podspec +18 -0
  154. package/package.json +6 -7
  155. package/settings.gradle.kts +25 -0
  156. package/shared/build.gradle.kts +0 -74
@@ -0,0 +1,484 @@
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.gestures.ScrollableState
8
+ import androidx.compose.foundation.layout.Box
9
+ import androidx.compose.foundation.layout.BoxScope
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.const.Spacing
52
+ import vn.momo.kits.modifier.conditional
53
+ import vn.momo.kits.modifier.hideKeyboardOnTap
54
+ import vn.momo.kits.navigation.component.FABPosition
55
+ import vn.momo.kits.navigation.component.FloatingButton
56
+ import vn.momo.kits.navigation.component.HEADER_HEIGHT
57
+ import vn.momo.kits.navigation.component.Header
58
+ import vn.momo.kits.navigation.component.HeaderBackground
59
+ import vn.momo.kits.navigation.component.HeaderRight
60
+ import vn.momo.kits.navigation.component.HeaderType
61
+ import vn.momo.kits.navigation.component.InputSearchType
62
+ import vn.momo.kits.platform.BackHandler
63
+ import vn.momo.kits.platform.getAndroidBuildVersion
64
+
65
+ internal val LocalFooterHeightPx = staticCompositionLocalOf { mutableIntStateOf(0) }
66
+ internal val LocalHeaderRightWidthPx = staticCompositionLocalOf { mutableIntStateOf(0) }
67
+
68
+ @OptIn(ExperimentalMaterialApi::class)
69
+ @Composable
70
+ internal fun StackScreen(
71
+ content: @Composable () -> Unit,
72
+ navigationOptions: NavigationOptions? = null,
73
+ id: Int = -1,
74
+ bottomTabIndex: Int = -1,
75
+ ) {
76
+ val navigator = LocalNavigator.current
77
+ val statusBar = AppStatusBar.current
78
+ val density = LocalDensity.current
79
+ val navigation = remember { Navigation(id = id, bottomTabIndex = bottomTabIndex, initOptions = navigationOptions) }
80
+
81
+ val options by navigation.currentOptions
82
+
83
+ val limit = with(density) {
84
+ (statusBar).toPx() + HEADER_HEIGHT
85
+ }.toInt()
86
+
87
+ val (scrollState, scrollInProcess) = if (options.scrollData.scrollState is LazyListState)
88
+ (options.scrollData.scrollState as? LazyListState ?: rememberLazyListState()).proxyScrollState(limit, 15)
89
+ else
90
+ (options.scrollData.scrollState as? ScrollState ?: rememberScrollState()).proxyLimitedScrollState(limit, 15)
91
+
92
+ val footerHeightPx = remember { mutableIntStateOf(0) }
93
+ val headerRightWidthPx = remember { mutableIntStateOf(0) }
94
+
95
+ BackHandler(true) { navigator.pop() }
96
+
97
+ CompositionLocalProvider(
98
+ StackScreenScrollableState provides options.scrollData.scrollState,
99
+ LocalNavigation provides navigation,
100
+ LocalScrollState provides scrollState,
101
+ LocalOptions provides options,
102
+ LocalFooterHeightPx provides footerHeightPx,
103
+ LocalHeaderRightWidthPx provides headerRightWidthPx
104
+ ) {
105
+ Box(Modifier
106
+ .fillMaxSize()
107
+ .background(options.backgroundColor ?: AppTheme.current.colors.background.default)
108
+ .conditional(options.keyboardOptions.keyboardShouldPersistTaps) {
109
+ hideKeyboardOnTap()
110
+ }
111
+ .conditional(options.keyboardOptions.useAvoidKeyboard && getAndroidBuildVersion() > 29) {
112
+ imePadding()
113
+ }
114
+ ) {
115
+ Box(Modifier.zIndex(1f)) {
116
+ HeaderBackground()
117
+ }
118
+
119
+ Box(Modifier.zIndex(5f)) {
120
+ Header()
121
+ }
122
+
123
+ Column (Modifier.zIndex(6f)) {
124
+ SearchAnimated(isScrollInProgress = scrollInProcess)
125
+ }
126
+
127
+ Column(Modifier.zIndex(2f).fillMaxSize()) {
128
+ MainContent(content = content)
129
+ }
130
+
131
+ Box(Modifier.zIndex(4f).align(Alignment.BottomCenter)) {
132
+ FooterContent()
133
+ }
134
+
135
+ Box(Modifier.zIndex(7f)){
136
+ FloatingContent()
137
+ }
138
+
139
+ OverplayView(bottomTabIndex = bottomTabIndex, id = id)
140
+ }
141
+ }
142
+ }
143
+
144
+ @Composable
145
+ fun FloatingContent(){
146
+ val options = LocalOptions.current
147
+ val density = LocalDensity.current
148
+ val footerHeightPx = LocalFooterHeightPx.current
149
+ val scrollState = LocalScrollState.current
150
+
151
+ val fabProps = options.floatingButtonProps ?: return
152
+ val bottomPadding = fabProps.bottom ?:
153
+ (Spacing.M + if (options.footerComponent != null) with(density){ footerHeightPx.intValue.toDp() } else 0.dp)
154
+
155
+ FloatingButton(
156
+ scrollPosition = fabProps.scrollState?.value ?: scrollState.value,
157
+ onClick = fabProps.onClick,
158
+ containerColor = AppTheme.current.colors.primary,
159
+ bottom = bottomPadding,
160
+ icon = fabProps.icon,
161
+ iconColor = fabProps.iconColor,
162
+ text = fabProps.label,
163
+ size = fabProps.size,
164
+ position = fabProps.position ?: FABPosition.END,
165
+ )
166
+ }
167
+
168
+ @Composable
169
+ fun ColumnScope.MainContent(content: @Composable ()-> Unit){
170
+ val options = LocalOptions.current
171
+ val inputSearchType = getInputSearchType(options)
172
+ val density = LocalDensity.current
173
+ val scrollState = LocalScrollState.current
174
+
175
+ Spacer(Modifier.height(if (options.headerType is HeaderType.DefaultOrExtended) AppStatusBar.current + HEADER_HEIGHT.dp else 0.dp))
176
+ if (inputSearchType == InputSearchType.Animated){
177
+ val scrollDp = with(density) { scrollState.value.toDp() }
178
+
179
+ val animatedTopPadding by animateDpAsState(
180
+ targetValue = (HEADER_HEIGHT.dp - scrollDp).coerceIn(0.dp, HEADER_HEIGHT.dp),
181
+ label = "AnimatedTopPadding"
182
+ )
183
+ Spacer(Modifier.height(animatedTopPadding))
184
+ }
185
+ Column (Modifier
186
+ .fillMaxWidth()
187
+ .weight(1f)
188
+ .conditional(options.scrollData.scrollable && options.scrollData.scrollState is ScrollState) {
189
+ verticalScroll(scrollState)
190
+ }
191
+ ) {
192
+ ScreenContent(content = content)
193
+ }
194
+
195
+ if (options.footerComponent != null){
196
+ val footerHeight = LocalFooterHeightPx.current
197
+ val footerHeightDp = with(density) { footerHeight.value.toDp() }
198
+ Spacer(Modifier.height(footerHeightDp))
199
+ }
200
+ }
201
+
202
+ @Composable
203
+ fun ScreenContent(content: @Composable () -> Unit){
204
+ val scrollState = LocalScrollState.current
205
+ val options = LocalOptions.current
206
+
207
+ if (options.headerType is HeaderType.Animated){
208
+ val animatedHeader = options.headerType
209
+ Box {
210
+ Box(Modifier.fillMaxWidth().aspectRatio(animatedHeader.aspectRatio.value)){
211
+ animatedHeader.composable.invoke(scrollState.value)
212
+ }
213
+ Box(Modifier.offset(x = 0.dp, y = AppStatusBar.current + HEADER_HEIGHT.dp + animatedHeader.layoutOffSet)){
214
+ content()
215
+ }
216
+ }
217
+ } else {
218
+ content()
219
+ }
220
+ }
221
+
222
+ @Composable
223
+ fun FooterContent(){
224
+ val options = LocalOptions.current
225
+ if (options.footerComponent != null){
226
+ val isKeyboardVisible = isKeyboardVisible()
227
+ val bottomPadding = min(AppNavigationBar.current, if (isKeyboardVisible) 0.dp else 21.dp)
228
+ Footer(footerComponent = options.footerComponent, bottomPadding = bottomPadding)
229
+ }
230
+ }
231
+
232
+ @Composable
233
+ fun OverplayView(bottomTabIndex: Int, id: Int){
234
+ val overplayType = OverplayComponentRegistry.getOverplayType()
235
+
236
+ if (overplayType != null) {
237
+ Box(Modifier.zIndex(if (overplayType == OverplayComponentType.SNACK_BAR) 3f else 8f).fillMaxSize()){
238
+ if (bottomTabIndex != -1) return@Box
239
+ if (OverplayComponentRegistry.currentRootId() != id) return@Box
240
+ OverplayComponentRegistry.OverlayComponent()
241
+ }
242
+ }
243
+ }
244
+
245
+ @Composable
246
+ fun Footer(footerComponent: @Composable (() -> Unit)?, bottomPadding: Dp) {
247
+ if (footerComponent == null) return
248
+
249
+ val footerHeightPx = LocalFooterHeightPx.current
250
+
251
+ val shadowBrush = remember {
252
+ Brush.verticalGradient(
253
+ colors = listOf(Color.Transparent, Color.Black.copy(alpha = 0.05f))
254
+ )
255
+ }
256
+
257
+ Box(Modifier.onGloballyPositioned {
258
+ if (footerHeightPx.intValue != it.size.height) footerHeightPx.intValue = it.size.height
259
+ }) {
260
+ Box(Modifier
261
+ .fillMaxWidth()
262
+ .background(AppTheme.current.colors.background.surface)
263
+ .padding(top = Spacing.S, start = Spacing.M, end = Spacing.M, bottom = bottomPadding + Spacing.S)
264
+ ){
265
+ footerComponent.invoke()
266
+ }
267
+
268
+ Box(modifier = Modifier
269
+ .fillMaxWidth()
270
+ .height(6.dp)
271
+ .offset(x = 0.dp, y = (-6).dp)
272
+ .background(shadowBrush)
273
+ )
274
+ }
275
+ }
276
+
277
+ data class InputSearchLayoutParams(
278
+ val topPadding: Dp,
279
+ val startPadding: Dp,
280
+ val endPadding: Dp
281
+ )
282
+ @Composable
283
+ fun SearchAnimated(
284
+ isScrollInProgress: Boolean
285
+ ) {
286
+ val scrollState = LocalScrollState.current
287
+ val options = LocalOptions.current
288
+ val navigator = LocalNavigator.current
289
+ val density = LocalDensity.current
290
+
291
+ val scrollDp = with(density) { scrollState.value.toDp() }
292
+
293
+ val inputSearchType = getInputSearchType(options)
294
+ val headerRightWidthPx = LocalHeaderRightWidthPx.current
295
+ val headerRightWidthDp = with(density) { headerRightWidthPx.intValue.toDp() }
296
+
297
+ if (inputSearchType == InputSearchType.None) return
298
+ val inputSearchProps = (options.headerType as? HeaderType.DefaultOrExtended)?.inputSearchProps ?: return
299
+
300
+ val minTopPadding = AppStatusBar.current
301
+ val maxTopPadding = AppStatusBar.current + HEADER_HEIGHT.dp
302
+ val minStartPadding = Spacing.M
303
+ val maxStartPadding = if (options.hiddenBack) Spacing.M else 52.dp
304
+ val minEndPadding = Spacing.M
305
+ val maxEndPadding = headerRightWidthDp + if (options.headerRight is HeaderRight.None) Spacing.M else Spacing.M * 2
306
+
307
+ val targetColor = lerp(
308
+ AppTheme.current.colors.background.surface,
309
+ AppTheme.current.colors.background.default,
310
+ (scrollDp / minTopPadding).coerceIn(0f, 1f)
311
+ )
312
+ val animatedColor by animateColorAsState(targetValue = targetColor, label = "BackgroundColor")
313
+
314
+ val layoutParams = when (inputSearchType) {
315
+ InputSearchType.Header -> {
316
+ InputSearchLayoutParams(
317
+ topPadding = minTopPadding,
318
+ startPadding = maxStartPadding,
319
+ endPadding = maxEndPadding
320
+ )
321
+ }
322
+
323
+ InputSearchType.Animated -> {
324
+ val animatedTopPadding by animateDpAsState(
325
+ targetValue = (maxTopPadding - scrollDp).coerceIn(minTopPadding, maxTopPadding),
326
+ label = "AnimatedTopPadding"
327
+ )
328
+
329
+ val animatedStartPadding by animateDpAsState(
330
+ targetValue = (minStartPadding + scrollDp).coerceIn(minStartPadding, maxStartPadding),
331
+ label = "AnimatedStartPadding"
332
+ )
333
+
334
+ val animatedEndPadding by animateDpAsState(
335
+ targetValue = (minEndPadding + scrollDp).coerceIn(minEndPadding, maxEndPadding),
336
+ label = "AnimatedEndPadding"
337
+ )
338
+
339
+ val maxPadding = remember(maxTopPadding, maxStartPadding, maxEndPadding) {
340
+ maxOf(maxEndPadding, maxStartPadding, maxTopPadding)
341
+ }
342
+ val snapScrollValue = with(density) { maxPadding.toPx().toInt() }
343
+
344
+ LaunchedEffect(isScrollInProgress && scrollState.value != 0 && scrollState.value != snapScrollValue) {
345
+ if (scrollDp < maxPadding) {
346
+ if (!isScrollInProgress) {
347
+ val midpoint = (maxTopPadding - minTopPadding) / 2
348
+
349
+ if (scrollDp < midpoint) {
350
+ scrollState.animateScrollTo(0)
351
+ } else {
352
+ scrollState.animateScrollTo(snapScrollValue)
353
+ }
354
+ }
355
+ }
356
+ }
357
+
358
+ InputSearchLayoutParams(
359
+ topPadding = animatedTopPadding,
360
+ startPadding = animatedStartPadding,
361
+ endPadding = animatedEndPadding
362
+ )
363
+ }
364
+
365
+ InputSearchType.None -> return
366
+ }
367
+
368
+ Spacer(Modifier.height(layoutParams.topPadding))
369
+ Box(
370
+ modifier = Modifier
371
+ .height(HEADER_HEIGHT.dp)
372
+ .fillMaxWidth()
373
+ .padding(
374
+ start = layoutParams.startPadding,
375
+ end = layoutParams.endPadding
376
+ ),
377
+ contentAlignment = Alignment.Center
378
+ ) {
379
+ InputSearch(
380
+ inputSearchProps = inputSearchProps.copy(
381
+ backgroundColor = animatedColor,
382
+ onPressButtonText = {
383
+ navigator.pop()
384
+ }
385
+ )
386
+ )
387
+ }
388
+ }
389
+
390
+ @Composable
391
+ internal fun isKeyboardVisible(): Boolean {
392
+ val ime = WindowInsets.ime
393
+ val density = LocalDensity.current
394
+ return ime.getBottom(density) > 0
395
+ }
396
+ private fun quantize(value: Int, stepPx: Int): Int {
397
+ if (stepPx <= 1) return value
398
+ return (value / stepPx) * stepPx
399
+ }
400
+
401
+ @Composable
402
+ fun LazyListState.proxyScrollState(
403
+ thresholdPx: Int = 200,
404
+ stepPx: Int = 10
405
+ ): Pair<ScrollState, Boolean> {
406
+ val scrollState = rememberScrollState()
407
+ val locked = remember { mutableStateOf(false) }
408
+
409
+ LaunchedEffect(this, scrollState, thresholdPx, stepPx) {
410
+ snapshotFlow { firstVisibleItemIndex to firstVisibleItemScrollOffset }
411
+ .collect { (index, offset) ->
412
+ val rawTarget = if (index == 0) offset else SCROLLED_PAST_FIRST_ITEM_THRESHOLD
413
+ val shouldLock = rawTarget > thresholdPx
414
+ if (locked.value != shouldLock) locked.value = shouldLock
415
+
416
+ val desired = if (shouldLock) thresholdPx else quantize(rawTarget, stepPx)
417
+
418
+ if (scrollState.value != desired) {
419
+ scrollState.scrollTo(desired.coerceAtLeast(0))
420
+ }
421
+ }
422
+ }
423
+
424
+ val inProgress = remember { mutableStateOf(false) }
425
+ LaunchedEffect(this, thresholdPx) {
426
+ snapshotFlow { isScrollInProgress }
427
+ .collect { progressing ->
428
+ inProgress.value = if (locked.value) false else progressing
429
+ }
430
+ }
431
+
432
+ return scrollState to inProgress.value
433
+ }
434
+
435
+ @Composable
436
+ fun ScrollState.proxyLimitedScrollState(
437
+ thresholdPx: Int = 200,
438
+ stepPx: Int = 10
439
+ ): Pair<ScrollState, Boolean> {
440
+ val proxy = rememberScrollState()
441
+ val locked = remember { mutableStateOf(false) }
442
+
443
+ LaunchedEffect(this, proxy, thresholdPx, stepPx) {
444
+ snapshotFlow { value }
445
+ .collect { rawTarget ->
446
+ val shouldLock = rawTarget > thresholdPx
447
+ if (locked.value != shouldLock) locked.value = shouldLock
448
+
449
+ val desired = if (shouldLock) thresholdPx else quantize(rawTarget, stepPx)
450
+
451
+ if (proxy.value != desired) {
452
+ proxy.scrollTo(desired.coerceAtLeast(0))
453
+ }
454
+ }
455
+ }
456
+
457
+ val inProgress = remember { mutableStateOf(false) }
458
+ LaunchedEffect(this, thresholdPx) {
459
+ snapshotFlow { isScrollInProgress }
460
+ .collect { progressing ->
461
+ inProgress.value = if (locked.value) false else progressing
462
+ }
463
+ }
464
+
465
+ return proxy to inProgress.value
466
+ }
467
+
468
+ private const val SCROLLED_PAST_FIRST_ITEM_THRESHOLD = 10_000
469
+
470
+ internal val LocalScrollState = staticCompositionLocalOf<ScrollState> { error("No Scroll State provided") }
471
+ internal val LocalOptions = staticCompositionLocalOf<NavigationOptions> { error("No NavigationOptions provided") }
472
+
473
+ val StackScreenScrollableState = staticCompositionLocalOf<ScrollableState?> { null }
474
+
475
+ internal fun getInputSearchType(options: NavigationOptions): InputSearchType{
476
+ return when (val headerType = options.headerType) {
477
+ is HeaderType.DefaultOrExtended -> when {
478
+ headerType.inputSearchProps == null -> InputSearchType.None
479
+ headerType.useAnimated -> InputSearchType.Animated
480
+ else -> InputSearchType.Header
481
+ }
482
+ else -> InputSearchType.None
483
+ }
484
+ }
@@ -0,0 +1,169 @@
1
+ package vn.momo.kits.navigation.bottomtab
2
+
3
+ import androidx.compose.animation.core.tween
4
+ import androidx.compose.animation.fadeIn
5
+ import androidx.compose.animation.fadeOut
6
+ import androidx.compose.animation.scaleIn
7
+ import androidx.compose.animation.scaleOut
8
+ import androidx.compose.foundation.background
9
+ import androidx.compose.foundation.layout.Box
10
+ import androidx.compose.foundation.layout.Column
11
+ import androidx.compose.foundation.layout.Spacer
12
+ import androidx.compose.foundation.layout.fillMaxSize
13
+ import androidx.compose.foundation.layout.fillMaxWidth
14
+ import androidx.compose.foundation.layout.height
15
+ import androidx.compose.foundation.layout.padding
16
+ import androidx.compose.runtime.Composable
17
+ import androidx.compose.runtime.LaunchedEffect
18
+ import androidx.compose.ui.Alignment
19
+ import androidx.compose.ui.Modifier
20
+ import androidx.compose.ui.unit.dp
21
+ import androidx.compose.ui.unit.min
22
+ import androidx.navigation.compose.NavHost
23
+ import androidx.navigation.compose.composable
24
+ import androidx.navigation.compose.rememberNavController
25
+ import vn.momo.kits.const.AppNavigationBar
26
+ import vn.momo.kits.const.AppTheme
27
+ import vn.momo.kits.const.Spacing
28
+ import vn.momo.kits.navigation.LocalNavigation
29
+ import vn.momo.kits.navigation.LocalNavigator
30
+ import vn.momo.kits.navigation.NavigationOptions
31
+ import vn.momo.kits.navigation.StackScreen
32
+ import vn.momo.kits.navigation.component.HeaderType
33
+ import vn.momo.kits.platform.getScreenHeight
34
+
35
+ private var bottomTabOptionItems : MutableList<NavigationOptions?> = mutableListOf()
36
+ fun setBottomTabOption(index: Int, options: NavigationOptions){
37
+ if (index in bottomTabOptionItems.indices) {
38
+ bottomTabOptionItems[index] = options
39
+ }
40
+ }
41
+ fun getBottomTabOption(index: Int): NavigationOptions? {
42
+ return if (index in bottomTabOptionItems.indices) {
43
+ bottomTabOptionItems[index]
44
+ } else null
45
+ }
46
+
47
+ @Composable
48
+ fun BottomTab(
49
+ items: List<BottomTabItem>,
50
+ floatingButton: BottomTabFloatingButton? = null
51
+ ) {
52
+ val navigation = LocalNavigation.current
53
+ val navigator = LocalNavigator.current
54
+ val navController = rememberNavController()
55
+
56
+ bottomTabOptionItems = items.mapIndexed { index, item ->
57
+ val options = item.options ?: NavigationOptions()
58
+ options.copy(
59
+ onBackHandler = {
60
+ if (index != 0) {
61
+ navController.popBackStack()
62
+ } else {
63
+ navigator.pop()
64
+ }
65
+ }
66
+ )
67
+ }.toMutableList()
68
+
69
+
70
+ LaunchedEffect(Unit){
71
+ navigation.setOptions(
72
+ headerType = HeaderType.None
73
+ )
74
+ }
75
+
76
+ Box(modifier = Modifier.fillMaxWidth().height(getScreenHeight()), contentAlignment = Alignment.BottomCenter) {
77
+ Box(modifier = Modifier
78
+ .fillMaxSize()
79
+ .padding(bottom = BOTTOM_TAB_BAR_HEIGHT.dp + AppNavigationBar.current)
80
+ ) {
81
+ NavHost(
82
+ navController = navController,
83
+ startDestination = "option0"
84
+ ) {
85
+ items.forEachIndexed { index, item ->
86
+ composable(
87
+ route = "option$index",
88
+ enterTransition = {
89
+ fadeIn(animationSpec = tween(200)) +
90
+ scaleIn(
91
+ initialScale = 0.97f,
92
+ animationSpec = tween(200)
93
+ )
94
+ },
95
+ exitTransition = {
96
+ fadeOut(animationSpec = tween(200)) +
97
+ scaleOut(
98
+ targetScale = 0.97f,
99
+ animationSpec = tween(200)
100
+ )
101
+ },
102
+ popEnterTransition = {
103
+ fadeIn(animationSpec = tween(200)) +
104
+ scaleIn(
105
+ initialScale = 0.97f,
106
+ animationSpec = tween(200)
107
+ )
108
+ },
109
+ popExitTransition = {
110
+ fadeOut(animationSpec = tween(200)) +
111
+ scaleOut(
112
+ targetScale = 0.97f,
113
+ animationSpec = tween(200)
114
+ )
115
+ }
116
+ ) {
117
+ val option = getBottomTabOption(index)?.copy(
118
+ onBackHandler = {
119
+ if (index != 0) {
120
+ navController.popBackStack()
121
+ } else {
122
+ navigator.pop()
123
+ }
124
+ }
125
+ )
126
+
127
+ StackScreen(
128
+ content = item.screen,
129
+ navigationOptions = option,
130
+ bottomTabIndex = index
131
+ )
132
+ }
133
+ }
134
+ }
135
+ }
136
+ Column {
137
+ BottomTabBar(
138
+ items = items,
139
+ floatingButton = floatingButton,
140
+ navController = navController,
141
+ onTabSelected = {
142
+ val currentRoute = navController.currentBackStackEntry?.destination?.route
143
+ val targetRoute = "option$it"
144
+ if (currentRoute != targetRoute){
145
+ navController.navigate(targetRoute)
146
+ }
147
+ }
148
+ )
149
+ Spacer(modifier = Modifier.fillMaxWidth().height(min(AppNavigationBar.current, 21.dp) + Spacing.S).background(AppTheme.current.colors.background.surface))
150
+ }
151
+ }
152
+ }
153
+
154
+ data class BottomTabItem(
155
+ val name: String,
156
+ val label: String,
157
+ val icon: String,
158
+ val showDot: Boolean = false,
159
+ val badgeLabel: String? = null,
160
+ val screen: @Composable () -> Unit,
161
+ val options: NavigationOptions? = null,
162
+ val initialParams: Any? = null
163
+ )
164
+
165
+ data class BottomTabFloatingButton(
166
+ val icon: String,
167
+ val label: String,
168
+ val onPress: () -> Unit,
169
+ )