@momo-kits/native-kits 0.156.3 → 0.156.4-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 (114) 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 +113 -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 +95 -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 +715 -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/Badge.kt +83 -0
  35. package/compose/src/commonMain/kotlin/vn/momo/kits/components/BadgeDot.kt +32 -0
  36. package/compose/src/commonMain/kotlin/vn/momo/kits/components/BadgeRibbon.kt +340 -0
  37. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Button.kt +348 -0
  38. package/compose/src/commonMain/kotlin/vn/momo/kits/components/CheckBox.kt +94 -0
  39. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Chip.kt +136 -0
  40. package/compose/src/commonMain/kotlin/vn/momo/kits/components/CupertinoOverscroll.kt +543 -0
  41. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Divider.kt +23 -0
  42. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Icon.kt +76 -0
  43. package/compose/src/commonMain/kotlin/vn/momo/kits/components/IconButton.kt +148 -0
  44. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Image.kt +188 -0
  45. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Information.kt +116 -0
  46. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Input.kt +447 -0
  47. package/compose/src/commonMain/kotlin/vn/momo/kits/components/InputDropDown.kt +172 -0
  48. package/compose/src/commonMain/kotlin/vn/momo/kits/components/InputMoney.kt +244 -0
  49. package/compose/src/commonMain/kotlin/vn/momo/kits/components/InputOTP.kt +231 -0
  50. package/compose/src/commonMain/kotlin/vn/momo/kits/components/InputPhoneNumber.kt +234 -0
  51. package/compose/src/commonMain/kotlin/vn/momo/kits/components/InputSearch.kt +254 -0
  52. package/compose/src/commonMain/kotlin/vn/momo/kits/components/InputTextArea.kt +241 -0
  53. package/compose/src/commonMain/kotlin/vn/momo/kits/components/LazyColumnWithBouncing.kt +364 -0
  54. package/compose/src/commonMain/kotlin/vn/momo/kits/components/PaginationDot.kt +56 -0
  55. package/compose/src/commonMain/kotlin/vn/momo/kits/components/PaginationNumber.kt +41 -0
  56. package/compose/src/commonMain/kotlin/vn/momo/kits/components/PaginationScroll.kt +92 -0
  57. package/compose/src/commonMain/kotlin/vn/momo/kits/components/PaginationWhiteDot.kt +40 -0
  58. package/compose/src/commonMain/kotlin/vn/momo/kits/components/PopupNotify.kt +352 -0
  59. package/compose/src/commonMain/kotlin/vn/momo/kits/components/PopupPromotion.kt +103 -0
  60. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Radio.kt +70 -0
  61. package/compose/src/commonMain/kotlin/vn/momo/kits/components/ScaleSizeScope.kt +17 -0
  62. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Skeleton.kt +96 -0
  63. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Switch.kt +96 -0
  64. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Tag.kt +92 -0
  65. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Text.kt +130 -0
  66. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Title.kt +214 -0
  67. package/compose/src/commonMain/kotlin/vn/momo/kits/components/TrustBanner.kt +177 -0
  68. package/compose/src/commonMain/kotlin/vn/momo/kits/components/datetimepicker/DateTimePicker.kt +205 -0
  69. package/compose/src/commonMain/kotlin/vn/momo/kits/components/datetimepicker/DateTimePickerTypes.kt +29 -0
  70. package/compose/src/commonMain/kotlin/vn/momo/kits/components/datetimepicker/DateTimePickerUtils.kt +239 -0
  71. package/compose/src/commonMain/kotlin/vn/momo/kits/components/datetimepicker/WheelPicker.kt +191 -0
  72. package/compose/src/commonMain/kotlin/vn/momo/kits/const/Colors.kt +306 -0
  73. package/compose/src/commonMain/kotlin/vn/momo/kits/const/Radius.kt +12 -0
  74. package/compose/src/commonMain/kotlin/vn/momo/kits/const/Spacing.kt +13 -0
  75. package/compose/src/commonMain/kotlin/vn/momo/kits/const/Theme.kt +189 -0
  76. package/compose/src/commonMain/kotlin/vn/momo/kits/const/Typography.kt +285 -0
  77. package/compose/src/commonMain/kotlin/vn/momo/kits/layout/Card.kt +2 -0
  78. package/compose/src/commonMain/kotlin/vn/momo/kits/layout/Item.kt +35 -0
  79. package/compose/src/commonMain/kotlin/vn/momo/kits/layout/Section.kt +2 -0
  80. package/compose/src/commonMain/kotlin/vn/momo/kits/modifier/AutomationId.kt +59 -0
  81. package/compose/src/commonMain/kotlin/vn/momo/kits/modifier/Clickable.kt +68 -0
  82. package/compose/src/commonMain/kotlin/vn/momo/kits/modifier/Conditional.kt +11 -0
  83. package/compose/src/commonMain/kotlin/vn/momo/kits/modifier/DeprecatedModifier.kt +14 -0
  84. package/compose/src/commonMain/kotlin/vn/momo/kits/modifier/Shadow.kt +50 -0
  85. package/compose/src/commonMain/kotlin/vn/momo/kits/modifier/Size.kt +51 -0
  86. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/BottomSheet.kt +239 -0
  87. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/ModalScreen.kt +119 -0
  88. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/Navigation.kt +98 -0
  89. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/NavigationContainer.kt +163 -0
  90. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/Navigator.kt +331 -0
  91. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/StackScreen.kt +497 -0
  92. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/bottomtab/BottomTab.kt +162 -0
  93. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/bottomtab/BottomTabBar.kt +226 -0
  94. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/bottomtab/CurvedContainer.kt +86 -0
  95. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/FloatingButton.kt +187 -0
  96. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/Header.kt +274 -0
  97. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/HeaderBackground.kt +80 -0
  98. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/HeaderRight.kt +306 -0
  99. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/HeaderTitle.kt +31 -0
  100. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/HeaderUser.kt +370 -0
  101. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/SnackBar.kt +132 -0
  102. package/compose/src/commonMain/kotlin/vn/momo/kits/platform/Platform.kt +42 -0
  103. package/compose/src/commonMain/kotlin/vn/momo/kits/utils/Icons.kt +1329 -0
  104. package/compose/src/commonMain/kotlin/vn/momo/kits/utils/Resources.kt +62 -0
  105. package/compose/src/commonMain/kotlin/vn/momo/kits/utils/Utils.kt +88 -0
  106. package/compose/src/iosMain/kotlin/vn/momo/kits/platform/Platform.ios.kt +152 -0
  107. package/gradle/libs.versions.toml +57 -0
  108. package/gradle/wrapper/gradle-wrapper.jar +0 -0
  109. package/gradle/wrapper/gradle-wrapper.properties +8 -0
  110. package/gradle.properties +26 -0
  111. package/gradlew +252 -0
  112. package/gradlew.bat +94 -0
  113. package/package.json +1 -1
  114. package/settings.gradle.kts +52 -0
@@ -0,0 +1,340 @@
1
+ package vn.momo.kits.components
2
+
3
+ import androidx.compose.foundation.Canvas
4
+ import androidx.compose.foundation.background
5
+ import androidx.compose.foundation.border
6
+ import androidx.compose.foundation.layout.Box
7
+ import androidx.compose.foundation.layout.Row
8
+ import androidx.compose.foundation.layout.height
9
+ import androidx.compose.foundation.layout.padding
10
+ import androidx.compose.foundation.layout.size
11
+ import androidx.compose.foundation.layout.width
12
+ import androidx.compose.foundation.shape.RoundedCornerShape
13
+ import androidx.compose.runtime.Composable
14
+ import androidx.compose.runtime.remember
15
+ import androidx.compose.ui.Alignment
16
+ import androidx.compose.ui.Modifier
17
+ import androidx.compose.ui.geometry.CornerRadius
18
+ import androidx.compose.ui.geometry.Offset
19
+ import androidx.compose.ui.geometry.Rect
20
+ import androidx.compose.ui.geometry.RoundRect
21
+ import androidx.compose.ui.geometry.Size
22
+ import androidx.compose.ui.graphics.Color
23
+ import androidx.compose.ui.graphics.Path
24
+ import androidx.compose.ui.graphics.graphicsLayer
25
+ import androidx.compose.ui.layout.ContentScale
26
+ import androidx.compose.ui.platform.LocalDensity
27
+ import androidx.compose.ui.text.rememberTextMeasurer
28
+ import androidx.compose.ui.text.style.TextAlign
29
+ import androidx.compose.ui.text.style.TextOverflow
30
+ import androidx.compose.ui.unit.Dp
31
+ import androidx.compose.ui.unit.dp
32
+ import androidx.compose.ui.unit.sp
33
+ import vn.momo.kits.application.IsShowBaseLineDebug
34
+ import vn.momo.kits.const.AppTheme
35
+ import vn.momo.kits.const.Colors
36
+ import vn.momo.kits.const.Typography
37
+ import vn.momo.kits.const.getFontFamily
38
+ import vn.momo.kits.const.scaleSize
39
+ import vn.momo.kits.modifier.conditional
40
+
41
+ @Composable
42
+ fun BadgeRibbon(
43
+ position: RibbonPosition = RibbonPosition.TopRight,
44
+ label: String = "Label",
45
+ isRound: Boolean = false,
46
+ modifier: Modifier = Modifier,
47
+ ) {
48
+ val theme = AppTheme.current
49
+
50
+ val rotate = if (position == RibbonPosition.TopRight || position == RibbonPosition.BottomRight) 180f else 0f
51
+ val useUpTail = position == RibbonPosition.BottomLeft || position == RibbonPosition.TopRight
52
+ val verticalAlignment = when(position){
53
+ RibbonPosition.TopLeft, RibbonPosition.BottomRight -> Alignment.Top
54
+ RibbonPosition.BottomLeft, RibbonPosition.TopRight -> Alignment.Bottom
55
+ }
56
+
57
+ Row(
58
+ modifier = modifier
59
+ .height(ribbonHeight)
60
+ .conditional(IsShowBaseLineDebug) {
61
+ border(1.dp, Colors.blue_03)
62
+ }
63
+ .graphicsLayer { rotationZ = rotate },
64
+ verticalAlignment = verticalAlignment
65
+ ) {
66
+ if (useUpTail) {
67
+ UpTail()
68
+ } else {
69
+ DownTail()
70
+ }
71
+
72
+ if (isRound) {
73
+ renderRoundContent(label, rotate, theme.colors.warning.primary)
74
+ } else {
75
+ renderSkewContent(label, rotate, theme.colors.warning.primary)
76
+ }
77
+ }
78
+ }
79
+
80
+ @Composable
81
+ fun renderRoundContent(label: String, rotate: Float, backgroundColor: Color){
82
+ Box(
83
+ modifier = Modifier
84
+ .height(roundHeight)
85
+ .background(
86
+ color = backgroundColor,
87
+ shape = RoundedCornerShape(
88
+ topEnd = roundRightRadius,
89
+ bottomEnd = roundRightRadius,
90
+ )
91
+ )
92
+ .padding(end = roundPaddingEnd),
93
+ contentAlignment = Alignment.Center,
94
+ ) {
95
+ Label(label, rotate)
96
+ }
97
+ }
98
+
99
+ @Composable
100
+ fun renderSkewContent(label: String, rotate: Float, backgroundColor: Color){
101
+ Box(
102
+ modifier = Modifier
103
+ .height(skewBodyHeight)
104
+ .background(backgroundColor),
105
+ contentAlignment = Alignment.Center
106
+ ) {
107
+ Label(label, rotate)
108
+ }
109
+ RightTail()
110
+ }
111
+
112
+ @Composable
113
+ fun Label(label: String, rotate: Float){
114
+ Text(
115
+ text = label,
116
+ color = Colors.black_01,
117
+ maxLines = 1,
118
+ overflow = TextOverflow.Ellipsis,
119
+ style = Typography.labelXsMedium,
120
+ modifier = Modifier.graphicsLayer { rotationZ = rotate }
121
+ )
122
+ }
123
+
124
+ @Composable
125
+ fun UpTail() {
126
+ Image(
127
+ source = "https://static.momocdn.net/app/img/kits/utils/Head_down_4x.png",
128
+ modifier = Modifier
129
+ .graphicsLayer { rotationZ = 180f }
130
+ .width(headTailWidth)
131
+ .height(headTailHeight),
132
+ options = Options(
133
+ contentScale = ContentScale.FillBounds,
134
+ )
135
+ )
136
+ }
137
+
138
+ @Composable
139
+ fun DownTail() {
140
+ Image(
141
+ source = "https://static.momocdn.net/app/img/kits/utils/Head_4x.png",
142
+ modifier = Modifier
143
+ .width(headTailWidth)
144
+ .height(headTailHeight),
145
+ options = Options(
146
+ contentScale = ContentScale.FillBounds,
147
+ )
148
+ )
149
+ }
150
+
151
+ @Composable
152
+ fun RightTail() {
153
+ Image(
154
+ source = "https://static.momocdn.net/app/img/kits/utils/Tail_4x.png",
155
+ modifier = Modifier
156
+ .width(skewTailWidth)
157
+ .height(skewTailHeight),
158
+ options = Options(
159
+ contentScale = ContentScale.FillBounds,
160
+ )
161
+ )
162
+ }
163
+
164
+ @Composable
165
+ fun RoundedBadgeRibbon(
166
+ text: String = "Label",
167
+ position: RibbonPosition = RibbonPosition.TopLeft,
168
+ modifier: Modifier = Modifier
169
+ ) {
170
+ val theme = AppTheme.current
171
+ val density = LocalDensity.current
172
+ val style = Typography.labelXsMedium
173
+ val fontFamily = style.fontFamily
174
+
175
+ val scaledFontSize = scaleSize(style.fontSize.value).sp
176
+ val fontFamilyResult = getFontFamily(fontFamily?.toString() ?: theme.font, style.fontWeight)
177
+
178
+ val fontSize = remember(scaledFontSize) { scaledFontSize }
179
+ val font = remember(fontFamilyResult) { fontFamilyResult }
180
+ val textMeasurer = rememberTextMeasurer()
181
+
182
+ val textLayoutResult = remember(text, fontSize, font) {
183
+ textMeasurer.measure(
184
+ text = text,
185
+ style = style.copy(fontSize = fontSize, fontFamily = font)
186
+ )
187
+ }
188
+
189
+ val textWidth = with(density) { textLayoutResult.size.width.toDp() }
190
+ val textHeight = with(density) { textLayoutResult.size.height.toDp() }
191
+
192
+ val minRibbonHeight = (textHeight * 1.2f).coerceAtLeast(16.dp)
193
+ val paddingBottom = minRibbonHeight / 4f
194
+ val badgeHeight = minRibbonHeight + paddingBottom
195
+ val horizontalPadding = paddingBottom
196
+ val badgeWidth = (textWidth + horizontalPadding * 2).coerceAtLeast(28.dp)
197
+
198
+ val (rotateZ, scaleY, scaleX) = when (position) {
199
+ RibbonPosition.TopLeft -> Triple(0f, 1f, -1f)
200
+ RibbonPosition.TopRight -> Triple(0f, 1f, 1f)
201
+ RibbonPosition.BottomRight -> Triple(0f, -1f, 1f)
202
+ RibbonPosition.BottomLeft -> Triple(0f, -1f, -1f)
203
+ }
204
+
205
+ Box(
206
+ modifier = modifier
207
+ .size(width = badgeWidth, height = badgeHeight)
208
+ .graphicsLayer {
209
+ rotationZ = rotateZ
210
+ this.scaleY = scaleY
211
+ this.scaleX = scaleX
212
+ }
213
+ ) {
214
+ val roundedRect = (badgeHeight - paddingBottom) / 2f
215
+
216
+ Canvas(modifier = Modifier.matchParentSize()) {
217
+ val width = size.width
218
+ val height = size.height
219
+ val ribbonHeight = height - paddingBottom.toPx()
220
+ val cornerRadius = roundedRect.toPx()
221
+
222
+ val mainColor = Color(0xFFFA541C)
223
+ val tailColor = Color(0xFFC41B24)
224
+
225
+ val headWidth = 4.dp.toPx()
226
+ val leftSectionWidth = width * (9f / 28f)
227
+ val middleSectionWidth = width - leftSectionWidth - headWidth
228
+ val rightX = leftSectionWidth + middleSectionWidth
229
+
230
+ // 1. Tail + 2. BG
231
+ val leftMiddlePath = Path().apply {
232
+ addRoundRect(
233
+ RoundRect(
234
+ rect = Rect(
235
+ left = 0f,
236
+ top = 0f,
237
+ right = rightX + 1f,
238
+ bottom = ribbonHeight
239
+ ),
240
+ topLeft = CornerRadius(cornerRadius, cornerRadius),
241
+ topRight = CornerRadius.Zero,
242
+ bottomRight = CornerRadius.Zero,
243
+ bottomLeft = CornerRadius(cornerRadius, cornerRadius)
244
+ )
245
+ )
246
+ }
247
+ drawPath(
248
+ path = leftMiddlePath,
249
+ color = mainColor
250
+ )
251
+
252
+ // 3. Head
253
+ val headPath = Path().apply {
254
+ addRoundRect(
255
+ RoundRect(
256
+ rect = Rect(
257
+ left = rightX,
258
+ top = 0f,
259
+ right = rightX + headWidth,
260
+ bottom = ribbonHeight
261
+ ),
262
+ topLeft = CornerRadius.Zero,
263
+ topRight = CornerRadius(cornerRadius, cornerRadius),
264
+ bottomRight = CornerRadius.Zero,
265
+ bottomLeft = CornerRadius.Zero
266
+ )
267
+ )
268
+ }
269
+ drawPath(
270
+ path = headPath,
271
+ color = mainColor
272
+ )
273
+
274
+ // Draw bottom tail section
275
+ val tailY = ribbonHeight
276
+ val tailHeight = paddingBottom.toPx()
277
+ val halfHeadWidth = headWidth / 2f
278
+ val tailStartX = rightX + headWidth - halfHeadWidth
279
+
280
+ // Background square (main color)
281
+ drawRect(
282
+ color = mainColor,
283
+ topLeft = Offset(tailStartX, tailY),
284
+ size = Size(halfHeadWidth, halfHeadWidth)
285
+ )
286
+
287
+ val tailRoundRadius = tailHeight
288
+ val tailRightPath = Path().apply {
289
+ addRoundRect(
290
+ RoundRect(
291
+ rect = Rect(
292
+ left = tailStartX,
293
+ top = tailY,
294
+ right = tailStartX + halfHeadWidth,
295
+ bottom = tailY + tailHeight
296
+ ),
297
+ topLeft = CornerRadius.Zero,
298
+ topRight = CornerRadius(tailRoundRadius, tailRoundRadius),
299
+ bottomRight = CornerRadius(tailRoundRadius, tailRoundRadius),
300
+ bottomLeft = CornerRadius.Zero
301
+ )
302
+ )
303
+ }
304
+ drawPath(
305
+ path = tailRightPath,
306
+ color = tailColor
307
+ )
308
+ }
309
+
310
+ Text(
311
+ text = text,
312
+ modifier = Modifier
313
+ .padding(bottom = paddingBottom)
314
+ .padding(horizontal = paddingBottom / 2)
315
+ .align(Alignment.Center)
316
+ .graphicsLayer {
317
+ rotationZ = rotateZ
318
+ this.scaleY = scaleY
319
+ this.scaleX = scaleX
320
+ },
321
+ textAlign = TextAlign.Center,
322
+ style = Typography.labelXsMedium,
323
+ maxLines = 1,
324
+ overflow = TextOverflow.Ellipsis,
325
+ color = Colors.black_01
326
+ )
327
+ }
328
+ }
329
+
330
+ val ribbonHeight: Dp = 20.dp
331
+ val roundHeight: Dp = 16.dp
332
+ val skewBodyHeight: Dp = 16.dp
333
+ val roundRightRadius: Dp = 12.dp
334
+ val roundPaddingEnd: Dp = 6.dp
335
+ val skewTailWidth: Dp = 8.dp
336
+ val skewTailHeight: Dp = 16.dp
337
+ val headTailWidth: Dp = 5.dp
338
+ val headTailHeight: Dp = 20.dp
339
+
340
+ enum class RibbonPosition {TopLeft, TopRight, BottomLeft, BottomRight}
@@ -0,0 +1,348 @@
1
+ package vn.momo.kits.components
2
+
3
+ import androidx.compose.animation.core.animateDpAsState
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.clickable
9
+ import androidx.compose.foundation.interaction.MutableInteractionSource
10
+ import androidx.compose.foundation.interaction.collectIsPressedAsState
11
+ import androidx.compose.foundation.layout.Arrangement
12
+ import androidx.compose.foundation.layout.Box
13
+ import androidx.compose.foundation.layout.Row
14
+ import androidx.compose.foundation.layout.defaultMinSize
15
+ import androidx.compose.foundation.layout.fillMaxWidth
16
+ import androidx.compose.foundation.layout.height
17
+ import androidx.compose.foundation.layout.padding
18
+ import androidx.compose.foundation.layout.size
19
+ import androidx.compose.foundation.shape.RoundedCornerShape
20
+ import androidx.compose.runtime.Composable
21
+ import androidx.compose.runtime.CompositionLocalProvider
22
+ import androidx.compose.runtime.getValue
23
+ import androidx.compose.runtime.remember
24
+ import androidx.compose.runtime.staticCompositionLocalOf
25
+ import androidx.compose.ui.Alignment
26
+ import androidx.compose.ui.Modifier
27
+ import androidx.compose.ui.draw.alpha
28
+ import androidx.compose.ui.draw.clip
29
+ import androidx.compose.ui.graphics.Color
30
+ import androidx.compose.ui.text.TextStyle
31
+ import androidx.compose.ui.text.style.TextOverflow
32
+ import androidx.compose.ui.unit.Dp
33
+ import androidx.compose.ui.unit.dp
34
+ import vn.momo.kits.application.IsShowBaseLineDebug
35
+ import vn.momo.kits.const.AppTheme
36
+ import vn.momo.kits.const.Colors
37
+ import vn.momo.kits.const.Radius
38
+ import vn.momo.kits.const.Spacing
39
+ import vn.momo.kits.const.Typography
40
+ import vn.momo.kits.modifier.conditional
41
+ import vn.momo.kits.platform.LottieAnimation
42
+
43
+ enum class ButtonType {
44
+ PRIMARY,
45
+ SECONDARY,
46
+ TONAL,
47
+ OUTLINE,
48
+ DANGER,
49
+ TEXT,
50
+ DISABLED
51
+ }
52
+
53
+ data class ButtonSpecs(
54
+ val height: Dp,
55
+ val radius: Dp,
56
+ val padding: Dp,
57
+ val width: Dp,
58
+ )
59
+
60
+ enum class Size(val value: ButtonSpecs) {
61
+ LARGE(
62
+ ButtonSpecs(
63
+ height = 48.dp,
64
+ radius = Radius.S,
65
+ padding = Spacing.L,
66
+ width = 128.dp
67
+ )
68
+ ),
69
+ MEDIUM(
70
+ ButtonSpecs(
71
+ height = 36.dp,
72
+ radius = Radius.S,
73
+ padding = Spacing.M,
74
+ width = 80.dp
75
+ )
76
+ ),
77
+ SMALL(
78
+ ButtonSpecs(
79
+ height = 28.dp,
80
+ radius = Radius.S,
81
+ padding = Spacing.S,
82
+ width = 60.dp
83
+ )
84
+ )
85
+ }
86
+
87
+ private val styleCache = mapOf(
88
+ Size.LARGE to Typography.actionDefaultBold,
89
+ Size.MEDIUM to Typography.actionSBold,
90
+ Size.SMALL to Typography.actionXsBold
91
+ )
92
+
93
+ private val iconSizeCache = mapOf(
94
+ Size.LARGE to 24.dp,
95
+ Size.MEDIUM to 16.dp,
96
+ Size.SMALL to 16.dp
97
+ )
98
+
99
+ private val iconSpaceCache = mapOf(
100
+ Size.SMALL to Spacing.XS,
101
+ Size.MEDIUM to Spacing.S,
102
+ Size.LARGE to Spacing.S
103
+ )
104
+
105
+ fun getStyle(size: Size): TextStyle {
106
+ return styleCache[size] ?: Typography.actionDefaultBold
107
+ }
108
+
109
+ fun getIconSize(size: Size): Dp = iconSizeCache[size] ?: 24.dp
110
+ fun getIconSpace(size: Size): Dp = iconSpaceCache[size] ?: Spacing.S
111
+
112
+ @Composable
113
+ fun getTextColor(loading: Boolean, type: ButtonType): Color {
114
+ val theme = AppTheme.current
115
+
116
+ return remember(type, theme, loading) {
117
+ when (type) {
118
+ ButtonType.DISABLED -> theme.colors.text.disable
119
+ ButtonType.PRIMARY -> Colors.black_01
120
+ ButtonType.SECONDARY -> theme.colors.text.default
121
+ ButtonType.OUTLINE -> theme.colors.primary
122
+ ButtonType.TONAL -> theme.colors.primary
123
+ ButtonType.DANGER -> Colors.black_01
124
+ ButtonType.TEXT -> theme.colors.primary
125
+ }.withLoading(loading)
126
+ }
127
+ }
128
+
129
+ @Composable
130
+ fun RenderTitle(size: Size, title: String = "", type: ButtonType) {
131
+ val style = remember(size) { getStyle(size) }
132
+ val color = TextColor.current
133
+ Text(
134
+ style = style,
135
+ text = title,
136
+ color = color,
137
+ overflow = TextOverflow.Ellipsis,
138
+ maxLines = 1
139
+ )
140
+ }
141
+
142
+ @Composable
143
+ fun RenderIcon(
144
+ size: Size,
145
+ isIconLeft: Boolean,
146
+ useTintColor: Boolean = true,
147
+ icon: String = "",
148
+ forceLoading: Boolean = false
149
+ ) {
150
+ val bgColor = BackgroundColor.current
151
+ val iconSize = remember(size) { getIconSize(size) }
152
+ val margin = remember(size) { getIconSpace(size) }
153
+ val color = if (useTintColor) TextColor.current else Color.Unspecified
154
+
155
+ val showLoading = forceLoading
156
+
157
+ val modifier = Modifier.padding(
158
+ start = if (isIconLeft) 0.dp else margin,
159
+ end = if (isIconLeft) margin else 0.dp
160
+ )
161
+
162
+ if (showLoading) {
163
+ Box(modifier) {
164
+ LottieAnimation(
165
+ modifier = Modifier.size(iconSize),
166
+ bgColor = bgColor,
167
+ tintColor = color,
168
+ path = "files/lottie_circle_loader"
169
+ )
170
+ }
171
+ } else if (icon.isNotEmpty()) {
172
+ Icon(
173
+ source = icon,
174
+ color = color,
175
+ size = iconSize,
176
+ modifier = modifier
177
+ )
178
+ }
179
+ }
180
+
181
+ private fun shouldLoadingOnLeft(iconLeft: String, iconRight: String): Boolean {
182
+ val hasLeft = iconLeft.isNotEmpty()
183
+ val hasRight = iconRight.isNotEmpty()
184
+ return when {
185
+ !hasLeft && !hasRight -> true
186
+ hasLeft && !hasRight -> true
187
+ !hasLeft && hasRight -> false
188
+ hasLeft && hasRight -> false
189
+ else -> true
190
+ }
191
+ }
192
+
193
+ @Composable
194
+ fun getTypeStyle(
195
+ type: ButtonType,
196
+ color: Color? = AppTheme.current.colors.primary,
197
+ size: Size,
198
+ ): Modifier {
199
+ val theme = AppTheme.current
200
+ val bgColor = BackgroundColor.current
201
+ val radius = remember(size) { size.value.radius }
202
+ val modifier = Modifier.background(bgColor)
203
+
204
+ return remember(type, color, theme, radius, bgColor) {
205
+ when (type) {
206
+ ButtonType.DISABLED -> modifier
207
+ .border(0.dp, Color.Unspecified, RoundedCornerShape(radius))
208
+
209
+ ButtonType.PRIMARY -> modifier
210
+ .border(0.dp, Color.Unspecified, RoundedCornerShape(radius))
211
+
212
+ ButtonType.SECONDARY -> modifier
213
+ .border(1.dp, theme.colors.border.default, RoundedCornerShape(radius))
214
+
215
+ ButtonType.OUTLINE -> modifier
216
+ .border(1.dp, color ?: theme.colors.primary, RoundedCornerShape(radius))
217
+
218
+ ButtonType.TONAL -> modifier
219
+ .border(0.dp, Color.Unspecified, RoundedCornerShape(radius))
220
+
221
+ ButtonType.DANGER -> modifier
222
+ .border(0.dp, Color.Unspecified, RoundedCornerShape(radius))
223
+
224
+ ButtonType.TEXT -> modifier
225
+ }
226
+ }
227
+ }
228
+
229
+ @Composable
230
+ fun getButtonBackgroundColor(
231
+ loading: Boolean,
232
+ type: ButtonType
233
+ ): Color {
234
+ val theme = AppTheme.current
235
+
236
+ return remember(loading, type, theme) {
237
+ when (type) {
238
+ ButtonType.DISABLED -> theme.colors.background.disable.withLoading(loading)
239
+ ButtonType.PRIMARY -> theme.colors.primary.withLoading(loading)
240
+ ButtonType.SECONDARY -> theme.colors.background.surface.withLoading(loading)
241
+ ButtonType.OUTLINE -> theme.colors.background.surface.withLoading(loading)
242
+ ButtonType.TONAL -> theme.colors.background.tonal.withLoading(loading)
243
+ ButtonType.DANGER -> theme.colors.error.primary.withLoading(loading)
244
+ ButtonType.TEXT -> Color.Unspecified
245
+ }
246
+ }
247
+ }
248
+
249
+ fun Color.withLoading(loading: Boolean): Color =
250
+ this.copy(alpha = if (loading) 0.75f else 1f)
251
+
252
+ @Composable
253
+ fun Button(
254
+ onClick: () -> Unit,
255
+ type: ButtonType = ButtonType.PRIMARY,
256
+ size: Size = Size.LARGE,
257
+ iconRight: String = "",
258
+ iconLeft: String = "",
259
+ title: String = "Button",
260
+ loading: Boolean = false,
261
+ useTintColor: Boolean = true,
262
+ isFull: Boolean = true,
263
+ modifier: Modifier = Modifier,
264
+ ) {
265
+ val radius = remember(size) { size.value.radius }
266
+ val isEnabled = remember(type) { type != ButtonType.DISABLED }
267
+ val loadingOnLeft = remember(iconLeft, iconRight) {
268
+ shouldLoadingOnLeft(iconLeft, iconRight)
269
+ }
270
+
271
+ val sizeSpecs = remember(size) { size.value }
272
+
273
+ val interactionSource = remember { MutableInteractionSource() }
274
+ val isPressed by interactionSource.collectIsPressedAsState()
275
+
276
+ val animatedPadding by animateDpAsState(
277
+ targetValue = if (isPressed && isEnabled && !loading) 2.dp else 0.dp,
278
+ animationSpec = tween(100),
279
+ label = "pressPadding"
280
+ )
281
+
282
+ val targetAlpha = if (isPressed && isEnabled && !loading) 0.5f else 1f
283
+ val alpha by animateFloatAsState(
284
+ targetValue = targetAlpha,
285
+ animationSpec = tween(100),
286
+ label = "buttonPressAlpha"
287
+ )
288
+
289
+ val clickableModifier =
290
+ if (isEnabled && !loading) {
291
+ modifier
292
+ .then(if (isFull) Modifier.fillMaxWidth() else Modifier)
293
+ .clip(RoundedCornerShape(radius))
294
+ .clickable(
295
+ enabled = isEnabled && !loading,
296
+ interactionSource = interactionSource,
297
+ indication = null,
298
+ onClick = onClick
299
+ )
300
+ .alpha(alpha)
301
+ } else {
302
+ modifier
303
+ .then(if (isFull) Modifier.fillMaxWidth() else Modifier)
304
+ .clip(RoundedCornerShape(radius))
305
+ }
306
+
307
+ CompositionLocalProvider(
308
+ IsLoading provides loading,
309
+ BackgroundColor provides getButtonBackgroundColor(loading, type),
310
+ TextColor provides getTextColor(loading, type)
311
+ ) {
312
+ Row(
313
+ modifier = clickableModifier
314
+ .padding(horizontal = animatedPadding)
315
+ .clip(RoundedCornerShape(radius))
316
+ .then(getTypeStyle(type, size = size))
317
+ .conditional(IsShowBaseLineDebug) {
318
+ border(1.dp, Colors.blue_03)
319
+ }
320
+ .padding(horizontal = sizeSpecs.padding)
321
+ .height(sizeSpecs.height),
322
+ horizontalArrangement = Arrangement.Center,
323
+ verticalAlignment = Alignment.CenterVertically,
324
+ ) {
325
+ RenderIcon(
326
+ size = size,
327
+ isIconLeft = true,
328
+ useTintColor = useTintColor,
329
+ icon = iconLeft,
330
+ forceLoading = loading && loadingOnLeft
331
+ )
332
+ RenderTitle(size, title, type = type)
333
+ RenderIcon(
334
+ size = size,
335
+ isIconLeft = false,
336
+ useTintColor = useTintColor,
337
+ icon = iconRight,
338
+ forceLoading = loading && !loadingOnLeft
339
+ )
340
+ }
341
+ }
342
+ }
343
+
344
+
345
+ private val IsLoading = staticCompositionLocalOf<Boolean> { false }
346
+ private val BackgroundColor = staticCompositionLocalOf<Color> { Color.Transparent }
347
+ private val TextColor = staticCompositionLocalOf<Color> { Color.Transparent }
348
+