@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,108 @@
1
+ package vn.momo.kits.components
2
+
3
+ import androidx.compose.animation.core.FastOutSlowInEasing
4
+ import androidx.compose.animation.core.animateFloatAsState
5
+ import androidx.compose.animation.core.tween
6
+ import androidx.compose.foundation.background
7
+ import androidx.compose.foundation.border
8
+ import androidx.compose.foundation.layout.Box
9
+ import androidx.compose.foundation.layout.fillMaxWidth
10
+ import androidx.compose.foundation.layout.height
11
+ import androidx.compose.foundation.layout.size
12
+ import androidx.compose.foundation.shape.RoundedCornerShape
13
+ import androidx.compose.runtime.Composable
14
+ import androidx.compose.runtime.getValue
15
+ import androidx.compose.ui.Modifier
16
+ import androidx.compose.ui.draw.clip
17
+ import androidx.compose.ui.graphics.Color
18
+ import androidx.compose.ui.unit.dp
19
+ import vn.momo.kits.application.IsShowBaseLineDebug
20
+ import vn.momo.kits.const.AppTheme
21
+ import vn.momo.kits.const.Colors
22
+ import vn.momo.kits.const.Radius
23
+ import vn.momo.kits.modifier.conditional
24
+ import vn.momo.kits.platform.LottieAnimation
25
+
26
+ enum class LoaderType {
27
+ DOT, SPINNER
28
+ }
29
+
30
+ @Composable
31
+ fun Loader(
32
+ type: LoaderType = LoaderType.DOT,
33
+ tintColor: Color? = null,
34
+ modifier: Modifier = Modifier
35
+ ) {
36
+ when (type) {
37
+ LoaderType.DOT -> DotLoader(tintColor = tintColor, modifier = modifier)
38
+ LoaderType.SPINNER -> SpinnerLoader(tintColor = tintColor, modifier = modifier)
39
+ }
40
+ }
41
+
42
+ @Composable
43
+ fun DotLoader(
44
+ tintColor: Color? = null,
45
+ modifier: Modifier = Modifier
46
+ ) {
47
+ LottieAnimation(
48
+ path = "files/dot_loading",
49
+ tintColor = tintColor,
50
+ modifier = modifier
51
+ .size(width = 52.dp, height = 18.dp)
52
+ .conditional(IsShowBaseLineDebug) {
53
+ border(1.dp, Colors.blue_03)
54
+ }
55
+ )
56
+ }
57
+
58
+ @Composable
59
+ fun SpinnerLoader(
60
+ tintColor: Color? = null,
61
+ modifier: Modifier = Modifier
62
+ ) {
63
+ LottieAnimation(
64
+ path = "files/lottie_circle_loader",
65
+ tintColor = tintColor,
66
+ modifier = modifier
67
+ .size(24.dp)
68
+ .conditional(IsShowBaseLineDebug) {
69
+ border(1.dp, Colors.blue_03)
70
+ }
71
+ )
72
+ }
73
+
74
+ @Composable
75
+ fun ProgressBar(
76
+ percent: Float = 0f,
77
+ color: Color? = null,
78
+ modifier: Modifier = Modifier
79
+ ) {
80
+ val clampedPercent = percent.coerceIn(0f, 100f)
81
+ val animatedProgress by animateFloatAsState(
82
+ targetValue = clampedPercent / 100f,
83
+ animationSpec = tween(durationMillis = 200, easing = FastOutSlowInEasing),
84
+ label = "progressAnimation"
85
+ )
86
+
87
+ val trackColor = AppTheme.current.colors.background.default
88
+ val fillColor = color ?: AppTheme.current.colors.primary
89
+
90
+ Box(
91
+ modifier = modifier
92
+ .fillMaxWidth()
93
+ .height(4.dp)
94
+ .clip(RoundedCornerShape(Radius.XXS))
95
+ .background(trackColor)
96
+ .conditional(IsShowBaseLineDebug) {
97
+ border(1.dp, Colors.blue_03)
98
+ }
99
+ ) {
100
+ Box(
101
+ modifier = Modifier
102
+ .fillMaxWidth(animatedProgress)
103
+ .height(4.dp)
104
+ .clip(RoundedCornerShape(Radius.XXS))
105
+ .background(fillColor)
106
+ )
107
+ }
108
+ }
@@ -0,0 +1,56 @@
1
+ package vn.momo.kits.components
2
+
3
+ import androidx.compose.foundation.background
4
+ import androidx.compose.foundation.border
5
+ import androidx.compose.foundation.layout.Box
6
+ import androidx.compose.foundation.layout.Row
7
+ import androidx.compose.foundation.layout.padding
8
+ import androidx.compose.foundation.layout.size
9
+ import androidx.compose.foundation.shape.RoundedCornerShape
10
+ import androidx.compose.runtime.Composable
11
+ import androidx.compose.ui.Modifier
12
+ import androidx.compose.ui.graphics.Color
13
+ import androidx.compose.ui.unit.dp
14
+ import vn.momo.kits.application.IsShowBaseLineDebug
15
+ import vn.momo.kits.const.AppTheme
16
+ import vn.momo.kits.const.Colors
17
+ import vn.momo.kits.const.Spacing
18
+ import vn.momo.kits.modifier.conditional
19
+
20
+ @Composable
21
+ fun Dot(active: Boolean = false, activeColor: Color, modifier: Modifier = Modifier) {
22
+ return if (active) {
23
+ Box(
24
+ modifier = modifier.size(width = 12.dp, height = 4.dp).background(
25
+ color = activeColor,
26
+ shape = RoundedCornerShape(4.dp)
27
+ )
28
+ )
29
+ } else {
30
+ Box(
31
+ modifier = modifier.size(4.dp)
32
+ .background(
33
+ color = AppTheme.current.colors.background.pressed,
34
+ shape = RoundedCornerShape(4.dp)
35
+ )
36
+ )
37
+ }
38
+ }
39
+
40
+ @Composable
41
+ fun PaginationDot(
42
+ activeIndex: Int = 0,
43
+ dataLength: Int = 3
44
+ ) {
45
+ Row(modifier = Modifier.conditional(IsShowBaseLineDebug) {
46
+ border(1.dp, Colors.blue_03)
47
+ }) {
48
+ for (i in 0 until dataLength) {
49
+ Dot(
50
+ i == activeIndex,
51
+ AppTheme.current.colors.primary,
52
+ modifier = Modifier.padding(end = if (i != dataLength - 1) Spacing.XS else 0.dp)
53
+ )
54
+ }
55
+ }
56
+ }
@@ -0,0 +1,41 @@
1
+ package vn.momo.kits.components
2
+
3
+ import androidx.compose.foundation.background
4
+ import androidx.compose.foundation.border
5
+ import androidx.compose.foundation.layout.BoxWithConstraints
6
+ import androidx.compose.foundation.layout.padding
7
+ import androidx.compose.foundation.shape.RoundedCornerShape
8
+ import androidx.compose.runtime.Composable
9
+ import androidx.compose.ui.Alignment
10
+ import androidx.compose.ui.Modifier
11
+ import androidx.compose.ui.graphics.Color
12
+ import androidx.compose.ui.unit.dp
13
+ import vn.momo.kits.application.IsShowBaseLineDebug
14
+ import vn.momo.kits.const.Colors
15
+ import vn.momo.kits.const.Spacing
16
+ import vn.momo.kits.const.Typography
17
+ import vn.momo.kits.modifier.conditional
18
+
19
+ @Composable
20
+ fun PaginationNumber(
21
+ activeIndex: Int = 0,
22
+ dataLength: Int = 2
23
+ ) {
24
+ BoxWithConstraints(
25
+ modifier = Modifier.background(
26
+ color = Color(0x33000000),
27
+ shape = RoundedCornerShape(Spacing.L)
28
+ )
29
+ .conditional(IsShowBaseLineDebug) {
30
+ border(1.dp, Colors.blue_03)
31
+ },
32
+ contentAlignment = Alignment.Center
33
+ ) {
34
+ Text(
35
+ modifier = Modifier.padding(horizontal = Spacing.S),
36
+ text = "${1 + activeIndex}/$dataLength",
37
+ color = Colors.black_01,
38
+ style = Typography.labelDefaultMedium
39
+ )
40
+ }
41
+ }
@@ -0,0 +1,92 @@
1
+ package vn.momo.kits.components
2
+
3
+ import androidx.compose.foundation.background
4
+ import androidx.compose.foundation.border
5
+ import androidx.compose.foundation.horizontalScroll
6
+ import androidx.compose.foundation.layout.Box
7
+ import androidx.compose.foundation.layout.Column
8
+ import androidx.compose.foundation.layout.Row
9
+ import androidx.compose.foundation.layout.fillMaxWidth
10
+ import androidx.compose.foundation.layout.height
11
+ import androidx.compose.foundation.layout.offset
12
+ import androidx.compose.foundation.layout.padding
13
+ import androidx.compose.foundation.layout.width
14
+ import androidx.compose.foundation.rememberScrollState
15
+ import androidx.compose.foundation.shape.RoundedCornerShape
16
+ import androidx.compose.runtime.Composable
17
+ import androidx.compose.runtime.derivedStateOf
18
+ import androidx.compose.runtime.getValue
19
+ import androidx.compose.runtime.remember
20
+ import androidx.compose.ui.Alignment
21
+ import androidx.compose.ui.Modifier
22
+ import androidx.compose.ui.draw.clip
23
+ import androidx.compose.ui.unit.Dp
24
+ import androidx.compose.ui.unit.dp
25
+ import vn.momo.kits.application.IsShowBaseLineDebug
26
+ import vn.momo.kits.const.AppTheme
27
+ import vn.momo.kits.const.Colors
28
+ import vn.momo.kits.const.Spacing
29
+ import vn.momo.kits.modifier.conditional
30
+
31
+ val INDICATOR_WIDTH = 24.dp
32
+ val PROGRESS_WIDTH = 72.dp
33
+
34
+ // Memoized offset calculation to avoid repeated computations during scroll
35
+ private fun calculateOffset(currentPosition: Int, maxBound: Int): Dp {
36
+ return if (maxBound == 0) 0.dp else {
37
+ (PROGRESS_WIDTH - INDICATOR_WIDTH) * (currentPosition / maxBound.toFloat())
38
+ }
39
+ }
40
+
41
+ @Composable
42
+ fun PaginationScroll(
43
+ modifier: Modifier = Modifier,
44
+ content: @Composable () -> Unit
45
+ ) {
46
+ val scrollState = rememberScrollState()
47
+ val theme = AppTheme.current
48
+
49
+ // Use derivedStateOf to optimize scroll position calculations
50
+ val indicatorOffset by remember {
51
+ derivedStateOf {
52
+ calculateOffset(scrollState.value, scrollState.maxValue)
53
+ }
54
+ }
55
+
56
+ // Cache theme colors to avoid repeated access
57
+ val (progressBackground, indicatorColor) = remember(theme) {
58
+ theme.colors.background.pressed to theme.colors.primary
59
+ }
60
+
61
+ Column(horizontalAlignment = Alignment.CenterHorizontally) {
62
+ Row(
63
+ modifier = modifier
64
+ .fillMaxWidth()
65
+ .conditional(IsShowBaseLineDebug) {
66
+ border(1.dp, Colors.blue_03)
67
+ }
68
+ .padding(bottom = Spacing.L)
69
+ .horizontalScroll(scrollState)
70
+ ) {
71
+ content()
72
+ }
73
+
74
+ // Progress indicator
75
+ Box(
76
+ modifier = Modifier
77
+ .clip(RoundedCornerShape(Spacing.XS))
78
+ .width(PROGRESS_WIDTH)
79
+ .height(4.dp)
80
+ .background(color = progressBackground)
81
+ ) {
82
+ Box(
83
+ modifier = Modifier
84
+ .offset(x = indicatorOffset)
85
+ .clip(RoundedCornerShape(Spacing.XS))
86
+ .width(INDICATOR_WIDTH)
87
+ .height(4.dp)
88
+ .background(color = indicatorColor)
89
+ )
90
+ }
91
+ }
92
+ }
@@ -0,0 +1,40 @@
1
+ package vn.momo.kits.components
2
+
3
+ import androidx.compose.foundation.background
4
+ import androidx.compose.foundation.border
5
+ import androidx.compose.foundation.layout.Row
6
+ import androidx.compose.foundation.layout.padding
7
+ import androidx.compose.foundation.shape.RoundedCornerShape
8
+ import androidx.compose.runtime.Composable
9
+ import androidx.compose.ui.Modifier
10
+ import androidx.compose.ui.graphics.Color
11
+ import androidx.compose.ui.unit.dp
12
+ import vn.momo.kits.application.IsShowBaseLineDebug
13
+ import vn.momo.kits.const.Colors
14
+ import vn.momo.kits.const.Spacing
15
+ import vn.momo.kits.modifier.conditional
16
+
17
+ @Composable
18
+ fun PaginationWhiteDot(
19
+ activeIndex: Int = 0,
20
+ dataLength: Int = 3
21
+ ) {
22
+ Row(
23
+ modifier = Modifier.background(
24
+ color = Color(0x33000000),
25
+ shape = RoundedCornerShape(Spacing.S)
26
+ )
27
+ .conditional(IsShowBaseLineDebug) {
28
+ border(1.dp, Colors.blue_03)
29
+ }
30
+ .padding(Spacing.XS)
31
+ ) {
32
+ for (i in 0 until dataLength) {
33
+ Dot(
34
+ i == activeIndex,
35
+ Colors.black_01,
36
+ modifier = Modifier.padding(end = if (i != dataLength - 1) Spacing.XS else 0.dp)
37
+ )
38
+ }
39
+ }
40
+ }
@@ -0,0 +1,352 @@
1
+ package vn.momo.kits.components
2
+
3
+ import androidx.compose.foundation.background
4
+ import androidx.compose.foundation.border
5
+ import androidx.compose.foundation.clickable
6
+ import androidx.compose.foundation.interaction.MutableInteractionSource
7
+ import androidx.compose.foundation.layout.Arrangement
8
+ import androidx.compose.foundation.layout.Box
9
+ import androidx.compose.foundation.layout.Column
10
+ import androidx.compose.foundation.layout.Row
11
+ import androidx.compose.foundation.layout.Spacer
12
+ import androidx.compose.foundation.layout.aspectRatio
13
+ import androidx.compose.foundation.layout.fillMaxWidth
14
+ import androidx.compose.foundation.layout.height
15
+ import androidx.compose.foundation.layout.offset
16
+ import androidx.compose.foundation.layout.padding
17
+ import androidx.compose.foundation.layout.width
18
+ import androidx.compose.foundation.lazy.LazyColumn
19
+ import androidx.compose.foundation.shape.RoundedCornerShape
20
+ import androidx.compose.runtime.Composable
21
+ import androidx.compose.runtime.LaunchedEffect
22
+ import androidx.compose.runtime.MutableState
23
+ import androidx.compose.runtime.getValue
24
+ import androidx.compose.runtime.mutableStateOf
25
+ import androidx.compose.runtime.remember
26
+ import androidx.compose.runtime.setValue
27
+ import androidx.compose.ui.Alignment
28
+ import androidx.compose.ui.Modifier
29
+ import androidx.compose.ui.draw.clip
30
+ import androidx.compose.ui.text.TextLayoutResult
31
+ import androidx.compose.ui.unit.dp
32
+ import vn.momo.kits.const.AppTheme
33
+ import vn.momo.kits.const.Radius
34
+ import vn.momo.kits.const.Spacing
35
+ import vn.momo.kits.application.IsShowBaseLineDebug
36
+ import vn.momo.kits.const.Colors
37
+ import vn.momo.kits.const.Typography
38
+ import vn.momo.kits.modifier.conditional
39
+ import vn.momo.kits.modifier.setAutomationId
40
+ import androidx.compose.ui.platform.LocalDensity
41
+ import androidx.compose.ui.unit.Dp
42
+ import vn.momo.kits.application.AppLanguage
43
+ import kotlin.math.min
44
+
45
+ data class PopupAction(
46
+ val title: String,
47
+ val onPress: (() -> Unit)?,
48
+ )
49
+
50
+ enum class PopupActionDirection {
51
+ ROW,
52
+ COLUMN,
53
+ AUTO
54
+ }
55
+
56
+ private fun heightForLines(
57
+ layout: TextLayoutResult,
58
+ targetLines: Float
59
+ ): Float {
60
+ val clampedLines = targetLines.coerceAtLeast(1f)
61
+ val wholeLines = kotlin.math.floor(clampedLines).toInt()
62
+ val fractionalPart = clampedLines - wholeLines
63
+
64
+ val baseLineIndex = (wholeLines - 1)
65
+ .coerceAtLeast(0)
66
+ .coerceAtMost(layout.lineCount - 1)
67
+
68
+ var heightPx = layout.getLineBottom(baseLineIndex)
69
+
70
+ if (fractionalPart > 0f) {
71
+ val nextLineIndex = (baseLineIndex + 1)
72
+ .coerceAtMost(layout.lineCount - 1)
73
+ val nextBottomPx = layout.getLineBottom(nextLineIndex)
74
+ heightPx += (nextBottomPx - heightPx) * fractionalPart
75
+ }
76
+ return heightPx
77
+ }
78
+
79
+ data class PopupNotifyProps(
80
+ val image: String = "",
81
+ val title: String = "Title",
82
+ val description: String = "Description",
83
+ val error: String = "",
84
+ val primary: PopupAction? = null,
85
+ val secondary: PopupAction? = null,
86
+ val buttonDirection: PopupActionDirection = PopupActionDirection.ROW,
87
+ val onIconClose: () -> Unit,
88
+ val isShowCloseIcon: Boolean = true,
89
+ )
90
+
91
+ @Composable
92
+ fun PopupNotify(
93
+ props: PopupNotifyProps,
94
+ ) {
95
+ var isScroll by remember { mutableStateOf(false) }
96
+ val layoutResult = remember { mutableStateOf<TextLayoutResult?>(null) }
97
+
98
+ val density = LocalDensity.current
99
+ var scrollHeight: Dp by remember { mutableStateOf(0.dp) }
100
+
101
+ LaunchedEffect(layoutResult.value) {
102
+ if ((layoutResult.value?.lineCount ?: 0) > 3) {
103
+ isScroll = true
104
+ }
105
+ }
106
+
107
+ val language = AppLanguage.current ?: "vi"
108
+ val errorCode: Map<String, String> = mapOf("vi" to "Mã lỗi: ", "en" to "Error code: ")
109
+
110
+ LaunchedEffect(layoutResult.value) {
111
+ layoutResult.value?.let { v ->
112
+ // cap visible height at ~8.5 lines
113
+ val textPx = v.size.height.toFloat()
114
+ val capPx = heightForLines(v, 8.5f)
115
+
116
+ isScroll = textPx >= capPx
117
+ scrollHeight = with(density) { min(textPx, capPx).toDp() }
118
+ }
119
+ }
120
+
121
+ val content: @Composable (Modifier) -> Unit = if (isScroll) {
122
+ { modifier ->
123
+ LazyColumn(
124
+ modifier = modifier.height(scrollHeight)
125
+ ) {
126
+ item {
127
+ Text(
128
+ text = props.description,
129
+ onTextLayout = { layoutResult.value = it },
130
+ style = Typography.bodyDefaultRegular
131
+ )
132
+ }
133
+ }
134
+ }
135
+ } else {
136
+ { modifier ->
137
+ Box(modifier = modifier) {
138
+ Text(
139
+ text = props.description,
140
+ onTextLayout = { layoutResult.value = it },
141
+ style = Typography.bodyDefaultRegular
142
+ )
143
+ }
144
+ }
145
+ }
146
+
147
+ fun onClose(callback: (() -> Unit)?) {
148
+ callback?.invoke()
149
+ }
150
+
151
+ Box(
152
+ Modifier.setAutomationId("popup_notify"), contentAlignment = Alignment.TopEnd
153
+ ) {
154
+ Column(
155
+ modifier = Modifier
156
+ .fillMaxWidth()
157
+ .padding(horizontal = 12.dp)
158
+ .clip(RoundedCornerShape(Radius.L))
159
+ .background(
160
+ AppTheme.current.colors.background.surface,
161
+ RoundedCornerShape(Radius.L)
162
+ )
163
+ .conditional(IsShowBaseLineDebug) {
164
+ border(1.dp, Colors.blue_03)
165
+ }
166
+ ) {
167
+ if(props.image.isNotEmpty()) {
168
+ Image(
169
+ source = props.image,
170
+ modifier = Modifier.fillMaxWidth().aspectRatio(1.777f),
171
+ options = Options(alignment = Alignment.Center)
172
+ )
173
+ }
174
+
175
+ Column(modifier = Modifier.padding(Spacing.XL)) {
176
+ Text(
177
+ style = Typography.headerDefaultBold,
178
+ maxLines = 2,
179
+ text = props.title,
180
+ modifier = Modifier.setAutomationId("title_popup_permission")
181
+ )
182
+ content(Modifier.padding(top = Spacing.S))
183
+ if (props.error.isNotEmpty()) {
184
+ Text(
185
+ text = errorCode[language] + props.error,
186
+ style = Typography.descriptionXsRegular,
187
+ color = AppTheme.current.colors.text.hint,
188
+ maxLines = 1,
189
+ modifier = Modifier.padding(top = Spacing.S)
190
+ )
191
+ }
192
+ }
193
+
194
+ Box(
195
+ modifier = Modifier.padding(horizontal = Spacing.XL)
196
+ .padding(bottom = Spacing.XL)
197
+ ) {
198
+ BuildAction(props.primary, props.secondary, props.buttonDirection, onAction = { callback ->
199
+ onClose(callback)
200
+ })
201
+ }
202
+ }
203
+
204
+ if(props.isShowCloseIcon) {
205
+ Box(
206
+ Modifier
207
+ .width(22.dp)
208
+ .height(22.dp)
209
+ .offset(x = -(1).dp, y = (-11).dp)
210
+ .background(
211
+ color = AppTheme.current.colors.text.default,
212
+ shape = RoundedCornerShape(Radius.M)
213
+ )
214
+ .border(
215
+ width = 2.dp,
216
+ color = AppTheme.current.colors.background.surface,
217
+ shape = RoundedCornerShape(Radius.M)
218
+ )
219
+ .clip(RoundedCornerShape(100))
220
+ .clickable(
221
+ interactionSource = remember { MutableInteractionSource() },
222
+ indication = null,
223
+ onClick = { onClose { props.onIconClose() } }
224
+ ),
225
+ contentAlignment = Alignment.Center
226
+ ) {
227
+ Icon(
228
+ source = "navigation_close",
229
+ color = AppTheme.current.colors.background.surface,
230
+ size = 16.dp,
231
+ modifier = Modifier.setAutomationId("ic_popup_close")
232
+ )
233
+ }
234
+ }
235
+ }
236
+ }
237
+
238
+ @Composable
239
+ fun BuildAction(
240
+ primary: PopupAction?,
241
+ secondary: PopupAction?,
242
+ buttonDirection: PopupActionDirection = PopupActionDirection.ROW,
243
+ onAction: (onPress: (() -> Unit)?) -> Unit,
244
+ ) {
245
+ val closeAction = remember { mutableStateOf("button_action") }
246
+ when (buttonDirection) {
247
+ PopupActionDirection.AUTO -> {
248
+ if (secondary != null && (secondary.title.length > 12 || primary?.title?.length!! > 12)) {
249
+ renderColumn(
250
+ secondary = secondary,
251
+ onAction = onAction,
252
+ closeAction = closeAction,
253
+ primary = primary
254
+ )
255
+ } else {
256
+ renderRow(
257
+ secondary = secondary,
258
+ onAction = onAction,
259
+ closeAction = closeAction,
260
+ primary = primary
261
+ )
262
+ }
263
+ }
264
+
265
+ PopupActionDirection.ROW -> renderRow(
266
+ secondary = secondary,
267
+ onAction = onAction,
268
+ closeAction = closeAction,
269
+ primary = primary
270
+ )
271
+
272
+ else -> renderColumn(
273
+ secondary = secondary,
274
+ onAction = onAction,
275
+ closeAction = closeAction,
276
+ primary = primary
277
+ )
278
+ }
279
+ }
280
+
281
+ @Composable
282
+ fun renderRow(
283
+ secondary: PopupAction?,
284
+ closeAction: MutableState<String>,
285
+ primary: PopupAction?,
286
+ onAction: (onPress: (() -> Unit)?) -> Unit,
287
+ ) {
288
+ Row(horizontalArrangement = Arrangement.Center) {
289
+ secondary?.let {
290
+ Box(Modifier.weight(1f)) {
291
+ Button(
292
+ size = Size.MEDIUM,
293
+ onClick = {
294
+ closeAction.value = "button_action"
295
+ onAction(it.onPress)
296
+ },
297
+ title = it.title,
298
+ type = ButtonType.TEXT,
299
+ modifier = Modifier.setAutomationId("btn_popup_cancel")
300
+ )
301
+ }
302
+ Spacer(modifier = Modifier.width(Spacing.S))
303
+ }
304
+ Box(Modifier.weight(1f)) {
305
+ Button(
306
+ size = Size.MEDIUM,
307
+ onClick = {
308
+ closeAction.value = "button_action"
309
+ onAction(primary?.onPress)
310
+ },
311
+ title = primary?.title ?: "",
312
+ modifier = Modifier.setAutomationId("btn_popup_allow")
313
+ )
314
+ }
315
+ }
316
+ }
317
+
318
+ @Composable
319
+ fun renderColumn(
320
+ secondary: PopupAction?,
321
+ closeAction: MutableState<String>,
322
+ primary: PopupAction?,
323
+ onAction: (onPress: (() -> Unit)?) -> Unit,
324
+ ) {
325
+ Column(horizontalAlignment = Alignment.CenterHorizontally) {
326
+ Button(
327
+ size = Size.MEDIUM,
328
+ onClick = {
329
+ closeAction.value = "button_action"
330
+ onAction(primary?.onPress)
331
+ },
332
+ title = primary?.title ?: "",
333
+ modifier = Modifier.setAutomationId("btn_popup_allow")
334
+ )
335
+ secondary?.let {
336
+ Spacer(modifier = Modifier.height(Spacing.S))
337
+ secondary.let {
338
+ Button(
339
+ size = Size.MEDIUM,
340
+ onClick = {
341
+ closeAction.value = "button_action"
342
+ onAction(it.onPress)
343
+ },
344
+ title = it.title,
345
+ type = ButtonType.TEXT,
346
+ modifier = Modifier.setAutomationId("btn_popup_cancel")
347
+ )
348
+ }
349
+ }
350
+ }
351
+ }
352
+