@momo-kits/native-kits 0.157.7 → 0.157.8-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 (131) 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 +552 -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/package.json +1 -1
  131. 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,164 @@
1
+ package vn.momo.kits.navigation
2
+
3
+ import androidx.compose.animation.core.tween
4
+ import androidx.compose.animation.fadeIn
5
+ import androidx.compose.animation.slideInHorizontally
6
+ import androidx.compose.animation.slideInVertically
7
+ import androidx.compose.animation.slideOutHorizontally
8
+ import androidx.compose.animation.slideOutVertically
9
+ import androidx.compose.foundation.layout.WindowInsets
10
+ import androidx.compose.foundation.layout.asPaddingValues
11
+ import androidx.compose.foundation.layout.systemBars
12
+ import androidx.compose.runtime.Composable
13
+ import androidx.compose.runtime.CompositionLocalProvider
14
+ import androidx.compose.runtime.DisposableEffect
15
+ import androidx.compose.runtime.LaunchedEffect
16
+ import androidx.compose.runtime.mutableStateOf
17
+ import androidx.compose.runtime.remember
18
+ import androidx.compose.runtime.staticCompositionLocalOf
19
+ import androidx.compose.ui.unit.Dp
20
+ import androidx.compose.ui.unit.dp
21
+ import androidx.navigation.compose.NavHost
22
+ import androidx.navigation.compose.composable
23
+ import androidx.navigation.compose.rememberNavController
24
+ import androidx.navigation.toRoute
25
+ import vn.momo.kits.application.AppConfig
26
+ import vn.momo.kits.application.AppLanguage
27
+ import vn.momo.kits.application.ApplicationContext
28
+ import vn.momo.kits.application.KitConfig
29
+ import vn.momo.kits.application.MiniAppContext
30
+ import vn.momo.kits.const.AppNavigationBar
31
+ import vn.momo.kits.const.AppStatusBar
32
+ import vn.momo.kits.const.AppTheme
33
+ import vn.momo.kits.const.Theme
34
+ import vn.momo.kits.const.ThemeAssets
35
+ import vn.momo.kits.const.defaultTheme
36
+ import vn.momo.kits.utils.getAppStatusBarHeight
37
+ import vn.momo.maxapi.IMaxApi
38
+
39
+ @Composable
40
+ fun NavigationContainer(
41
+ initialScreenName: String,
42
+ initialScreen: @Composable () -> Unit,
43
+ options: NavigationOptions? = null,
44
+ initialTheme: Theme = defaultTheme,
45
+ applicationContext: MiniAppContext? = null,
46
+ maxApi: IMaxApi? = null,
47
+ setNavigator: ((Navigator) -> Unit)? = null,
48
+ statusBarHeight: Dp? = null,
49
+ config: KitConfig? = null,
50
+ language: String? = null
51
+ ){
52
+ val navController = rememberNavController()
53
+ val navigator = remember { Navigator(navController = navController, maxApi = maxApi) }
54
+ val statusBarHeight = statusBarHeight ?: getAppStatusBarHeight()
55
+ val navigationBarHeight = if (AppNavigationBar.current == 0.dp) {
56
+ WindowInsets.systemBars.asPaddingValues().calculateBottomPadding()
57
+ } else {
58
+ AppNavigationBar.current
59
+ }
60
+
61
+ val parentContext = ApplicationContext.current
62
+ val mergedContext = MiniAppContext.merge(parentContext, applicationContext)
63
+
64
+ val theme = remember { mutableStateOf(initialTheme) }
65
+
66
+ LaunchedEffect(Unit) {
67
+ val headerBar = config?.headerBar
68
+ if (headerBar != null && theme.value.assets.headerBackground == null) {
69
+ theme.value = theme.value.copy(
70
+ assets = ThemeAssets(
71
+ headerBackground = headerBar
72
+ )
73
+ )
74
+ }
75
+ }
76
+
77
+ val startDestination = DynamicScreenRegistry.register(initialScreenName, initialScreen, options)
78
+
79
+ CompositionLocalProvider(
80
+ LocalNavigator provides navigator,
81
+ LocalMaxApi provides maxApi,
82
+ AppTheme provides theme.value,
83
+ AppStatusBar provides statusBarHeight,
84
+ AppNavigationBar provides navigationBarHeight,
85
+ ApplicationContext provides mergedContext,
86
+ AppConfig provides config,
87
+ AppLanguage provides language,
88
+ ) {
89
+ LaunchedEffect(Unit) {
90
+ setNavigator?.invoke(navigator)
91
+ }
92
+
93
+ NavHost(navController, startDestination = startDestination) {
94
+ composable<DynamicScreenRoute>(
95
+ enterTransition = {
96
+ slideInHorizontally(
97
+ animationSpec = tween(300),
98
+ initialOffsetX = { it }
99
+ )
100
+ },
101
+ exitTransition = null,
102
+ popEnterTransition = { fadeIn(animationSpec = tween(0)) },
103
+ popExitTransition = {
104
+ slideOutHorizontally(
105
+ animationSpec = tween(300),
106
+ targetOffsetX = { it }
107
+ )
108
+ }
109
+ ) { backStackEntry ->
110
+ val route = backStackEntry.toRoute<DynamicScreenRoute>()
111
+ val screen = DynamicScreenRegistry.getScreen(route.id)
112
+
113
+ if (screen != null){
114
+ StackScreen(
115
+ id = route.id,
116
+ name = screen.name,
117
+ content = screen.content,
118
+ navigationOptions = screen.options
119
+ )
120
+ }
121
+ }
122
+
123
+ composable<DynamicDialogRoute>(
124
+ enterTransition = {
125
+ slideInVertically (
126
+ animationSpec = tween(300),
127
+ initialOffsetY = { it }
128
+ )
129
+ },
130
+ exitTransition = null,
131
+ popEnterTransition = { fadeIn(animationSpec = tween(0)) },
132
+ popExitTransition = {
133
+ slideOutVertically (
134
+ animationSpec = tween(300),
135
+ targetOffsetY = { it }
136
+ )
137
+ }
138
+ ) { backStackEntry ->
139
+ val route = backStackEntry.toRoute<DynamicDialogRoute>()
140
+ val screen = DynamicScreenRegistry.getScreen(route.id)
141
+
142
+ if (screen != null){
143
+ StackScreen(
144
+ id = route.id,
145
+ name = screen.name,
146
+ content = screen.content,
147
+ navigationOptions = screen.options
148
+ )
149
+ }
150
+ }
151
+ }
152
+ }
153
+
154
+ DisposableEffect(Unit) {
155
+ onDispose {
156
+ navigator.dispose()
157
+ }
158
+ }
159
+ }
160
+
161
+ val LocalMaxApi = staticCompositionLocalOf<IMaxApi?> {
162
+ error("No MaxApi provided")
163
+ }
164
+