@momo-kits/native-kits 0.159.1-beta.8 → 0.160.1-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 (142) hide show
  1. package/.claude/settings.local.json +8 -0
  2. package/build.gradle.kts +11 -0
  3. package/compose/build.gradle.kts +180 -0
  4. package/compose/build.gradle.kts.backup +180 -0
  5. package/compose/compose.podspec +47 -0
  6. package/compose/src/androidMain/kotlin/vn/momo/kits/platform/Platform.android.kt +116 -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 +112 -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 +720 -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 +402 -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 +194 -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 +448 -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 +231 -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 +254 -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/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 +273 -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 +57 -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 +140 -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 +552 -0
  107. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/bottomtab/BottomTab.kt +161 -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 +131 -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 +45 -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 +105 -0
  123. package/compose/src/iosMain/kotlin/vn/momo/kits/platform/Platform.ios.kt +176 -0
  124. package/example/ios/Example.xcodeproj/xcuserdata/huynhdung.xcuserdatad/xcschemes/xcschememanagement.plist +14 -0
  125. package/example/ios/Example.xcworkspace/xcuserdata/huynhdung.xcuserdatad/UserInterfaceState.xcuserstate +0 -0
  126. package/example/ios/Example.xcworkspace/xcuserdata/huynhdung.xcuserdatad/xcschemes/xcschememanagement.plist +5 -0
  127. package/example/ios/Pods/Pods.xcodeproj/xcuserdata/huynhdung.xcuserdatad/xcschemes/MoMoUIKits.xcscheme +58 -0
  128. package/example/ios/Pods/Pods.xcodeproj/xcuserdata/huynhdung.xcuserdatad/xcschemes/Pods-Example.xcscheme +58 -0
  129. package/example/ios/Pods/Pods.xcodeproj/xcuserdata/huynhdung.xcuserdatad/xcschemes/SDWebImage.xcscheme +58 -0
  130. package/example/ios/Pods/Pods.xcodeproj/xcuserdata/huynhdung.xcuserdatad/xcschemes/SDWebImageSwiftUI.xcscheme +58 -0
  131. package/example/ios/Pods/Pods.xcodeproj/xcuserdata/huynhdung.xcuserdatad/xcschemes/SkeletonUI.xcscheme +58 -0
  132. package/example/ios/Pods/Pods.xcodeproj/xcuserdata/huynhdung.xcuserdatad/xcschemes/xcschememanagement.plist +46 -0
  133. package/gradle/libs.versions.toml +58 -0
  134. package/gradle/wrapper/gradle-wrapper.jar +0 -0
  135. package/gradle/wrapper/gradle-wrapper.properties +8 -0
  136. package/gradle.properties +26 -0
  137. package/gradlew +252 -0
  138. package/gradlew.bat +94 -0
  139. package/ios/Typography/Text.swift +1 -1
  140. package/local.properties +8 -0
  141. package/package.json +1 -1
  142. package/settings.gradle.kts +52 -0
@@ -0,0 +1,253 @@
1
+ package vn.momo.kits.navigation
2
+
3
+ import androidx.compose.animation.core.Animatable
4
+ import androidx.compose.animation.core.CubicBezierEasing
5
+ import androidx.compose.animation.core.tween
6
+ import androidx.compose.foundation.background
7
+ import androidx.compose.foundation.border
8
+ import androidx.compose.foundation.gestures.detectDragGestures
9
+ import androidx.compose.foundation.layout.Box
10
+ import androidx.compose.foundation.layout.BoxWithConstraints
11
+ import androidx.compose.foundation.layout.Column
12
+ import androidx.compose.foundation.layout.Row
13
+ import androidx.compose.foundation.layout.Spacer
14
+ import androidx.compose.foundation.layout.fillMaxHeight
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.heightIn
19
+ import androidx.compose.foundation.layout.offset
20
+ import androidx.compose.foundation.layout.padding
21
+ import androidx.compose.foundation.layout.size
22
+ import androidx.compose.foundation.layout.width
23
+ import androidx.compose.foundation.shape.RoundedCornerShape
24
+ import androidx.compose.runtime.Composable
25
+ import androidx.compose.runtime.DisposableEffect
26
+ import androidx.compose.runtime.LaunchedEffect
27
+ import androidx.compose.runtime.remember
28
+ import androidx.compose.runtime.rememberCoroutineScope
29
+ import androidx.compose.ui.Alignment
30
+ import androidx.compose.ui.Modifier
31
+ import androidx.compose.ui.graphics.Color
32
+ import androidx.compose.ui.input.pointer.pointerInput
33
+ import androidx.compose.ui.platform.LocalDensity
34
+ import androidx.compose.ui.unit.IntOffset
35
+ import androidx.compose.ui.unit.dp
36
+ import kotlinx.coroutines.launch
37
+ import vn.momo.kits.components.Divider
38
+ import vn.momo.kits.components.Icon
39
+ import vn.momo.kits.components.Text
40
+ import vn.momo.kits.const.AppTheme
41
+ import vn.momo.kits.const.Colors
42
+ import vn.momo.kits.const.Radius
43
+ import vn.momo.kits.const.Spacing
44
+ import vn.momo.kits.application.ApplicationContext
45
+ import vn.momo.kits.application.IsShowBaseLineDebug
46
+ import vn.momo.kits.const.Typography
47
+ import vn.momo.kits.navigation.tracking.ScreenTracker
48
+ import vn.momo.kits.modifier.conditional
49
+ import vn.momo.kits.modifier.noFeedbackClickable
50
+ import vn.momo.kits.platform.BackHandler
51
+ import vn.momo.kits.platform.getScreenHeight
52
+
53
+ @Composable
54
+ internal fun BottomSheet(
55
+ content: @Composable () -> Unit,
56
+ header: BottomHeader,
57
+ isSurface: Boolean = false,
58
+ barrierDismissible: Boolean = true,
59
+ onDismiss: (() -> Unit)?
60
+ ) {
61
+ val navigator = LocalNavigator.current
62
+
63
+ val screenHeightDp = getScreenHeight()
64
+ val screenHeightPx = with(LocalDensity.current) { screenHeightDp.toPx() }
65
+ val sheetOffset = remember { Animatable(screenHeightPx) }
66
+
67
+ val backgroundAlpha = remember { Animatable(0f) }
68
+ val dynamicAlpha = ((1f - (sheetOffset.value / screenHeightPx)).coerceIn(0f, 1f)) * 0.3f
69
+ val alpha = backgroundAlpha.value + dynamicAlpha
70
+
71
+ val sheetCloseOffset = with(LocalDensity.current) { 100.dp.toPx() }
72
+
73
+ val coroutineScope = rememberCoroutineScope()
74
+
75
+ val maxApi = LocalMaxApi.current
76
+ val context = ApplicationContext.current
77
+
78
+ DisposableEffect(Unit) {
79
+ ScreenTracker.trackPopupDisplayed(
80
+ maxApi = maxApi,
81
+ context = context,
82
+ parentScreenName = ScreenTracker.getLastScreenName()
83
+ )
84
+ onDispose { }
85
+ }
86
+
87
+ suspend fun openEvent(){
88
+ backgroundAlpha.animateTo(
89
+ targetValue = 0.3f,
90
+ animationSpec = tween(
91
+ durationMillis = 100,
92
+ easing = CubicBezierEasing(0.05f, 0.7f, 0.1f, 1f)
93
+ )
94
+ )
95
+ sheetOffset.animateTo(
96
+ targetValue = 0f,
97
+ animationSpec = tween(
98
+ durationMillis = 350,
99
+ easing = CubicBezierEasing(0.05f, 0.7f, 0.1f, 1f)
100
+ )
101
+ )
102
+ }
103
+
104
+ fun closeEvent(){
105
+ coroutineScope.launch {
106
+ sheetOffset.animateTo(
107
+ targetValue = screenHeightPx,
108
+ animationSpec = tween(
109
+ durationMillis = 200,
110
+ easing = CubicBezierEasing(0.3f, 0.0f, 0.8f, 0.15f)
111
+ )
112
+ )
113
+ backgroundAlpha.animateTo(
114
+ targetValue = 0f,
115
+ animationSpec = tween(
116
+ durationMillis = 100,
117
+ easing = CubicBezierEasing(0.3f, 0.0f, 0.8f, 0.15f)
118
+ )
119
+ )
120
+ navigator.pop()
121
+ }
122
+ }
123
+
124
+ LaunchedEffect(Unit) {
125
+ openEvent()
126
+ }
127
+
128
+ DisposableEffect(Unit) {
129
+ OverplayComponentRegistry.bindClose { closeEvent() }
130
+ onDispose {
131
+ onDismiss?.invoke()
132
+ }
133
+ }
134
+
135
+ BackHandler(true){
136
+ closeEvent()
137
+ }
138
+
139
+ BoxWithConstraints(
140
+ modifier = Modifier
141
+ .fillMaxSize()
142
+ .background(Color.Black.copy(alpha = alpha))
143
+ .noFeedbackClickable {
144
+ if(barrierDismissible) closeEvent()
145
+ }
146
+ ) {
147
+ val parentHeight = with(LocalDensity.current) { constraints.maxHeight.toDp() }
148
+
149
+ Box(
150
+ modifier = Modifier
151
+ .align(Alignment.BottomCenter)
152
+ .offset { IntOffset(x = 0, y = sheetOffset.value.toInt()) }
153
+ .fillMaxWidth()
154
+ .heightIn(max = parentHeight - 90.dp)
155
+ .background(
156
+ color = if (isSurface) AppTheme.current.colors.background.surface else AppTheme.current.colors.background.default,
157
+ shape = RoundedCornerShape(topStart = Radius.M, topEnd = Radius.M)
158
+ )
159
+ .conditional(IsShowBaseLineDebug) {
160
+ border(1.dp, Colors.blue_03)
161
+ }
162
+ .noFeedbackClickable { },
163
+ contentAlignment = Alignment.BottomCenter
164
+ ) {
165
+ Column {
166
+ Box(
167
+ modifier = Modifier
168
+ .height(72.dp)
169
+ .fillMaxWidth()
170
+ .pointerInput(Unit) {
171
+ detectDragGestures(
172
+ onDrag = { change, dragAmount ->
173
+ change.consume()
174
+ coroutineScope.launch {
175
+ val newOffset = (sheetOffset.value + dragAmount.y).coerceAtLeast(0f)
176
+ sheetOffset.snapTo(newOffset)
177
+ }
178
+ },
179
+ onDragEnd = {
180
+ coroutineScope.launch {
181
+ if (sheetOffset.value > sheetCloseOffset) {
182
+ closeEvent()
183
+ } else {
184
+ sheetOffset.animateTo(0f)
185
+ }
186
+ }
187
+ }
188
+ )
189
+ }
190
+ ) {
191
+ Column(
192
+ modifier = Modifier
193
+ .fillMaxSize()
194
+ .padding(horizontal = Spacing.M),
195
+ horizontalAlignment = Alignment.CenterHorizontally
196
+ ) {
197
+ Spacer(Modifier.height(8.dp))
198
+ Box(
199
+ modifier = Modifier
200
+ .height(4.dp)
201
+ .width(40.dp)
202
+ .background(
203
+ color = Colors.black_06,
204
+ shape = RoundedCornerShape(Radius.S)
205
+ )
206
+ )
207
+ Spacer(Modifier.height(4.dp))
208
+ Box(Modifier.fillMaxWidth().height(56.dp), contentAlignment = Alignment.Center){
209
+ when (header) {
210
+ is BottomHeader.Title -> {
211
+ Row(modifier = Modifier.fillMaxSize(), verticalAlignment = Alignment.CenterVertically) {
212
+ Box(Modifier.size(24.dp))
213
+ Box(Modifier.fillMaxHeight().weight(1f), contentAlignment = Alignment.Center){
214
+ Text(
215
+ text = header.data,
216
+ color = Colors.black_17,
217
+ style = Typography.headerDefaultBold
218
+ )
219
+ }
220
+ Box(Modifier
221
+ .size(24.dp)
222
+ .noFeedbackClickable {
223
+ closeEvent()
224
+ },
225
+ contentAlignment = Alignment.Center
226
+ ){
227
+ Icon(source = "navigation_close")
228
+ }
229
+ }
230
+ }
231
+ is BottomHeader.Custom -> {
232
+ header.content()
233
+ }
234
+ }
235
+ }
236
+ }
237
+ }
238
+ Divider()
239
+ content()
240
+ }
241
+ }
242
+ }
243
+ }
244
+
245
+ sealed class BottomHeader {
246
+ data class Title(
247
+ val data: String = "Bottom Sheet Title",
248
+ ) : BottomHeader()
249
+
250
+ data class Custom(
251
+ val content: @Composable () -> Unit
252
+ ) : BottomHeader()
253
+ }
@@ -0,0 +1,133 @@
1
+ package vn.momo.kits.navigation
2
+
3
+ import androidx.compose.animation.core.Animatable
4
+ import androidx.compose.animation.core.CubicBezierEasing
5
+ import androidx.compose.animation.core.tween
6
+ import androidx.compose.foundation.background
7
+ import androidx.compose.foundation.border
8
+ import androidx.compose.foundation.clickable
9
+ import androidx.compose.foundation.interaction.MutableInteractionSource
10
+ import androidx.compose.foundation.layout.Box
11
+ import androidx.compose.foundation.layout.fillMaxSize
12
+ import androidx.compose.foundation.layout.wrapContentSize
13
+ import androidx.compose.runtime.Composable
14
+ import androidx.compose.runtime.DisposableEffect
15
+ import androidx.compose.runtime.LaunchedEffect
16
+ import androidx.compose.runtime.remember
17
+ import androidx.compose.runtime.rememberCoroutineScope
18
+ import androidx.compose.ui.Alignment
19
+ import androidx.compose.ui.Modifier
20
+ import androidx.compose.ui.draw.alpha
21
+ import androidx.compose.ui.draw.scale
22
+ import androidx.compose.ui.graphics.Color
23
+ import androidx.compose.ui.unit.dp
24
+ import kotlinx.coroutines.launch
25
+ import vn.momo.kits.application.ApplicationContext
26
+ import vn.momo.kits.application.IsShowBaseLineDebug
27
+ import vn.momo.kits.const.Colors
28
+ import vn.momo.kits.modifier.conditional
29
+ import vn.momo.kits.modifier.noFeedbackClickable
30
+ import vn.momo.kits.navigation.tracking.ScreenTracker
31
+ import vn.momo.kits.platform.BackHandler
32
+
33
+ @Composable
34
+ internal fun ModalScreen(
35
+ content: @Composable () -> Unit,
36
+ barrierDismissible: Boolean = true,
37
+ onDismiss: (() -> Unit)? = null
38
+ ) {
39
+ val navigator = LocalNavigator.current
40
+
41
+ val alpha = remember { Animatable(0f) }
42
+ val scale = remember { Animatable(0.8f) }
43
+
44
+ val coroutineScope = rememberCoroutineScope()
45
+
46
+ val maxApi = LocalMaxApi.current
47
+ val context = ApplicationContext.current
48
+
49
+ DisposableEffect(Unit) {
50
+ ScreenTracker.trackPopupDisplayed(
51
+ maxApi = maxApi,
52
+ context = context,
53
+ parentScreenName = ScreenTracker.getLastScreenName()
54
+ )
55
+ onDispose { }
56
+ }
57
+
58
+ fun openEvent() {
59
+ coroutineScope.launch {
60
+ launch {
61
+ alpha.animateTo(
62
+ targetValue = 1f,
63
+ animationSpec = tween(durationMillis = 250)
64
+ )
65
+ }
66
+ launch {
67
+ scale.animateTo(
68
+ targetValue = 1f,
69
+ animationSpec = tween(
70
+ durationMillis = 250,
71
+ easing = CubicBezierEasing(0.2f, 0.0f, 0f, 1f)
72
+ )
73
+ )
74
+ }
75
+ }
76
+ }
77
+
78
+ fun closeEvent() {
79
+ coroutineScope.launch {
80
+ alpha.animateTo(
81
+ targetValue = 0f,
82
+ animationSpec = tween(durationMillis = 200)
83
+ )
84
+ scale.animateTo(
85
+ targetValue = 0.8f,
86
+ animationSpec = tween(durationMillis = 200)
87
+ )
88
+ navigator.pop()
89
+ }
90
+ }
91
+
92
+ LaunchedEffect(Unit) {
93
+ openEvent()
94
+ }
95
+
96
+ DisposableEffect(Unit) {
97
+ OverplayComponentRegistry.bindClose { closeEvent() }
98
+ onDispose {
99
+ onDismiss?.invoke()
100
+ }
101
+ }
102
+
103
+ BackHandler(enabled = true) {
104
+ closeEvent()
105
+ }
106
+
107
+ Box(
108
+ modifier = Modifier
109
+ .fillMaxSize()
110
+ .background(Color.Black.copy(alpha = 0.6f * alpha.value))
111
+ .noFeedbackClickable {
112
+ if(barrierDismissible) closeEvent()
113
+ },
114
+ contentAlignment = Alignment.Center
115
+ ) {
116
+ Box(
117
+ modifier = Modifier
118
+ .scale(scale.value)
119
+ .alpha(alpha.value)
120
+ .wrapContentSize()
121
+ .conditional(IsShowBaseLineDebug) {
122
+ border(1.dp, Colors.blue_03)
123
+ }
124
+ .clickable(
125
+ indication = null,
126
+ interactionSource = remember { MutableInteractionSource() }
127
+ ) {},
128
+ contentAlignment = Alignment.Center
129
+ ) {
130
+ content()
131
+ }
132
+ }
133
+ }
@@ -0,0 +1,99 @@
1
+ package vn.momo.kits.navigation
2
+
3
+ import androidx.compose.foundation.gestures.ScrollableState
4
+ import androidx.compose.runtime.Composable
5
+ import androidx.compose.runtime.Stable
6
+ import androidx.compose.runtime.State
7
+ import androidx.compose.runtime.mutableStateOf
8
+ import androidx.compose.runtime.staticCompositionLocalOf
9
+ import androidx.compose.ui.graphics.Color
10
+ import vn.momo.kits.navigation.bottomtab.setBottomTabOption
11
+ import vn.momo.kits.navigation.component.FloatingButtonProps
12
+ import vn.momo.kits.navigation.component.HeaderBackProps
13
+ import vn.momo.kits.navigation.component.HeaderRight
14
+ import vn.momo.kits.navigation.component.HeaderTitle
15
+ import vn.momo.kits.navigation.component.HeaderType
16
+
17
+ class Navigation(
18
+ val id: Int = -1,
19
+ val bottomTabIndex: Int = -1,
20
+ val initOptions: NavigationOptions? = null
21
+ ) {
22
+ private val _options = mutableStateOf(initOptions ?: NavigationOptions())
23
+ val currentOptions: State<NavigationOptions> get() = _options
24
+ val options: NavigationOptions get() = _options.value
25
+
26
+ fun setOptions(
27
+ onBackHandler: (() -> Unit)? = null,
28
+ hiddenBack: Boolean? = null,
29
+ headerBackProps: HeaderBackProps? = null,
30
+ headerTitle: HeaderTitle? = null,
31
+ headerRight: HeaderRight? = null,
32
+ headerType: HeaderType? = null,
33
+ scrollData: ScrollData? = null,
34
+ backgroundColor: Color? = null,
35
+ tintColor: Color? = null,
36
+ footerComponent: (@Composable () -> Unit)? = null,
37
+ floatingButtonProps: FloatingButtonProps? = null,
38
+ keyboardOptions: KeyboardOptions? = null
39
+ ) {
40
+ updateOptions(
41
+ options.copy(
42
+ onBackHandler = onBackHandler ?: options.onBackHandler,
43
+ hiddenBack = hiddenBack ?: options.hiddenBack,
44
+ headerBackProps = headerBackProps ?: options.headerBackProps,
45
+ headerTitle = headerTitle ?: options.headerTitle,
46
+ headerRight = headerRight ?: options.headerRight,
47
+ headerType = headerType ?: options.headerType,
48
+ scrollData = scrollData ?: options.scrollData,
49
+ backgroundColor = backgroundColor ?: options.backgroundColor,
50
+ tintColor = tintColor ?: options.tintColor,
51
+ footerComponent = footerComponent ?: options.footerComponent,
52
+ floatingButtonProps = floatingButtonProps ?: options.floatingButtonProps,
53
+ keyboardOptions = keyboardOptions ?: options.keyboardOptions,
54
+ )
55
+ )
56
+ }
57
+
58
+ fun setOptions(newOptions: NavigationOptions) {
59
+ updateOptions(newOptions)
60
+ }
61
+
62
+ private fun updateOptions(updated: NavigationOptions) {
63
+ _options.value = updated
64
+ if (bottomTabIndex != -1) setBottomTabOption(bottomTabIndex, updated)
65
+ if (id != -1) DynamicScreenRegistry.setOptions(id, updated)
66
+ }
67
+ }
68
+
69
+ val LocalNavigation = staticCompositionLocalOf<Navigation> {
70
+ error("No NavigationStack provided")
71
+ }
72
+
73
+ @Stable
74
+ data class NavigationOptions(
75
+ val screenName: String? = null,
76
+ val onBackHandler: (() -> Unit)? = null,
77
+ val hiddenBack: Boolean = false,
78
+ val headerBackProps: HeaderBackProps = HeaderBackProps(),
79
+ val headerTitle: HeaderTitle = HeaderTitle.Default("Stack"),
80
+ val headerRight: HeaderRight = HeaderRight.Toolkit(),
81
+ val headerType: HeaderType = HeaderType.Default(),
82
+ val scrollData: ScrollData = ScrollData(),
83
+ val backgroundColor: Color? = null,
84
+ val tintColor: Color? = null,
85
+ val footerComponent: @Composable (() -> Unit)? = null,
86
+ val floatingButtonProps: FloatingButtonProps? = null,
87
+ val keyboardOptions: KeyboardOptions = KeyboardOptions()
88
+ )
89
+
90
+ data class KeyboardOptions(
91
+ val keyboardShouldPersistTaps: Boolean = false,
92
+ val useAvoidKeyboard: Boolean = true
93
+ )
94
+
95
+
96
+ data class ScrollData(
97
+ val scrollable: Boolean = true,
98
+ val scrollState: ScrollableState? = null,
99
+ )
@@ -0,0 +1,140 @@
1
+ package vn.momo.kits.navigation
2
+
3
+ import androidx.compose.animation.*
4
+ import androidx.compose.animation.core.tween
5
+ import androidx.compose.runtime.*
6
+ import androidx.compose.ui.unit.Dp
7
+ import androidx.navigation.compose.NavHost
8
+ import androidx.navigation.compose.composable
9
+ import androidx.navigation.compose.rememberNavController
10
+ import androidx.navigation.toRoute
11
+ import vn.momo.kits.application.*
12
+ import vn.momo.kits.const.*
13
+ import vn.momo.kits.platform.ProvideNavigationEventDispatcherOwner
14
+ import vn.momo.kits.utils.getAppStatusBarHeight
15
+ import vn.momo.kits.utils.getNavigationBarHeight
16
+ import vn.momo.maxapi.IMaxApi
17
+
18
+ @Composable
19
+ fun NavigationContainer(
20
+ initialScreenName: String,
21
+ initialScreen: @Composable () -> Unit,
22
+ options: NavigationOptions? = null,
23
+ initialTheme: Theme = defaultTheme,
24
+ applicationContext: MiniAppContext? = null,
25
+ maxApi: IMaxApi? = null,
26
+ setNavigator: ((Navigator) -> Unit)? = null,
27
+ statusBarHeight: Dp? = null,
28
+ config: KitConfig? = null,
29
+ language: String? = null
30
+ ){
31
+ val navController = rememberNavController()
32
+ val navigator = remember { Navigator(navController = navController, maxApi = maxApi) }
33
+ val statusBarHeight = statusBarHeight ?: getAppStatusBarHeight()
34
+ val navigationBarHeight = getNavigationBarHeight()
35
+
36
+ val parentContext = ApplicationContext.current
37
+ val mergedContext = MiniAppContext.merge(parentContext, applicationContext)
38
+
39
+ val theme = remember { mutableStateOf(initialTheme) }
40
+
41
+ LaunchedEffect(Unit) {
42
+ val headerBar = config?.headerBar
43
+ if (headerBar != null && theme.value.assets.headerBackground == null) {
44
+ theme.value = theme.value.copy(
45
+ assets = ThemeAssets(
46
+ headerBackground = headerBar
47
+ )
48
+ )
49
+ }
50
+ }
51
+
52
+ val startDestination = DynamicScreenRegistry.register(initialScreenName, initialScreen, options)
53
+
54
+ ProvideNavigationEventDispatcherOwner {
55
+ CompositionLocalProvider(
56
+ LocalNavigator provides navigator,
57
+ LocalMaxApi provides maxApi,
58
+ AppTheme provides theme.value,
59
+ AppStatusBar provides statusBarHeight,
60
+ AppNavigationBar provides navigationBarHeight,
61
+ ApplicationContext provides mergedContext,
62
+ AppConfig provides config,
63
+ AppLanguage provides language,
64
+ ) {
65
+ LaunchedEffect(Unit) {
66
+ setNavigator?.invoke(navigator)
67
+ }
68
+
69
+ NavHost(navController, startDestination = startDestination) {
70
+ composable<DynamicScreenRoute>(
71
+ enterTransition = {
72
+ slideInHorizontally(
73
+ animationSpec = tween(300),
74
+ initialOffsetX = { it }
75
+ )
76
+ },
77
+ exitTransition = null,
78
+ popEnterTransition = { fadeIn(animationSpec = tween(0)) },
79
+ popExitTransition = {
80
+ slideOutHorizontally(
81
+ animationSpec = tween(300),
82
+ targetOffsetX = { it }
83
+ )
84
+ }
85
+ ) { backStackEntry ->
86
+ val route = backStackEntry.toRoute<DynamicScreenRoute>()
87
+ val screen = DynamicScreenRegistry.getScreen(route.id)
88
+
89
+ if (screen != null) {
90
+ StackScreen(
91
+ id = route.id,
92
+ name = screen.name,
93
+ content = screen.content,
94
+ navigationOptions = screen.options
95
+ )
96
+ }
97
+ }
98
+
99
+ composable<DynamicDialogRoute>(
100
+ enterTransition = {
101
+ slideInVertically(
102
+ animationSpec = tween(300),
103
+ initialOffsetY = { it }
104
+ )
105
+ },
106
+ exitTransition = null,
107
+ popEnterTransition = { fadeIn(animationSpec = tween(0)) },
108
+ popExitTransition = {
109
+ slideOutVertically(
110
+ animationSpec = tween(300),
111
+ targetOffsetY = { it }
112
+ )
113
+ }
114
+ ) { backStackEntry ->
115
+ val route = backStackEntry.toRoute<DynamicDialogRoute>()
116
+ val screen = DynamicScreenRegistry.getScreen(route.id)
117
+
118
+ if (screen != null) {
119
+ StackScreen(
120
+ id = route.id,
121
+ name = screen.name,
122
+ content = screen.content,
123
+ navigationOptions = screen.options
124
+ )
125
+ }
126
+ }
127
+ }
128
+ }
129
+ }
130
+
131
+ DisposableEffect(Unit) {
132
+ onDispose {
133
+ navigator.dispose()
134
+ }
135
+ }
136
+ }
137
+
138
+ val LocalMaxApi = staticCompositionLocalOf<IMaxApi?> {
139
+ error("No MaxApi provided")
140
+ }