@momo-kits/native-kits 0.157.2-debug → 0.157.2

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 (117) hide show
  1. package/package.json +1 -1
  2. package/build.gradle.kts +0 -11
  3. package/compose/build.gradle.kts +0 -180
  4. package/compose/build.gradle.kts.backup +0 -180
  5. package/compose/compose.podspec +0 -54
  6. package/compose/src/androidMain/kotlin/vn/momo/kits/platform/Platform.android.kt +0 -110
  7. package/compose/src/commonMain/composeResources/font/momosignature.otf +0 -0
  8. package/compose/src/commonMain/composeResources/font/momotrustdisplay.otf +0 -0
  9. package/compose/src/commonMain/composeResources/font/sfprotext_black.otf +0 -0
  10. package/compose/src/commonMain/composeResources/font/sfprotext_black.ttf +0 -0
  11. package/compose/src/commonMain/composeResources/font/sfprotext_bold.ttf +0 -0
  12. package/compose/src/commonMain/composeResources/font/sfprotext_heavy.ttf +0 -0
  13. package/compose/src/commonMain/composeResources/font/sfprotext_light.ttf +0 -0
  14. package/compose/src/commonMain/composeResources/font/sfprotext_medium.ttf +0 -0
  15. package/compose/src/commonMain/composeResources/font/sfprotext_regular.ttf +0 -0
  16. package/compose/src/commonMain/composeResources/font/sfprotext_semibold.ttf +0 -0
  17. package/compose/src/commonMain/composeResources/font/sfprotext_thin.otf +0 -0
  18. package/compose/src/commonMain/composeResources/font/sfprotext_thin.ttf +0 -0
  19. package/compose/src/commonMain/composeResources/font/sfprotext_ultralight.otf +0 -0
  20. package/compose/src/commonMain/composeResources/font/sfprotext_ultralight.ttf +0 -0
  21. package/compose/src/commonMain/kotlin/vn/momo/kits/application/AnimationSearchInput.kt +0 -57
  22. package/compose/src/commonMain/kotlin/vn/momo/kits/application/Context.kt +0 -107
  23. package/compose/src/commonMain/kotlin/vn/momo/kits/application/FloatingButton.kt +0 -201
  24. package/compose/src/commonMain/kotlin/vn/momo/kits/application/Header.kt +0 -222
  25. package/compose/src/commonMain/kotlin/vn/momo/kits/application/HeaderAnimated.kt +0 -48
  26. package/compose/src/commonMain/kotlin/vn/momo/kits/application/HeaderBackground.kt +0 -86
  27. package/compose/src/commonMain/kotlin/vn/momo/kits/application/HeaderDefault.kt +0 -76
  28. package/compose/src/commonMain/kotlin/vn/momo/kits/application/HeaderExtended.kt +0 -76
  29. package/compose/src/commonMain/kotlin/vn/momo/kits/application/HeaderRight.kt +0 -305
  30. package/compose/src/commonMain/kotlin/vn/momo/kits/application/HeaderTitle.kt +0 -33
  31. package/compose/src/commonMain/kotlin/vn/momo/kits/application/LiteScreen.kt +0 -720
  32. package/compose/src/commonMain/kotlin/vn/momo/kits/application/NavigationContainer.kt +0 -121
  33. package/compose/src/commonMain/kotlin/vn/momo/kits/application/Screen.kt +0 -405
  34. package/compose/src/commonMain/kotlin/vn/momo/kits/application/useHeaderSearchAnimation.kt +0 -69
  35. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Badge.kt +0 -85
  36. package/compose/src/commonMain/kotlin/vn/momo/kits/components/BadgeDot.kt +0 -32
  37. package/compose/src/commonMain/kotlin/vn/momo/kits/components/BadgeRibbon.kt +0 -340
  38. package/compose/src/commonMain/kotlin/vn/momo/kits/components/BaselineView.kt +0 -198
  39. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Button.kt +0 -357
  40. package/compose/src/commonMain/kotlin/vn/momo/kits/components/CheckBox.kt +0 -94
  41. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Chip.kt +0 -136
  42. package/compose/src/commonMain/kotlin/vn/momo/kits/components/CupertinoOverscroll.kt +0 -543
  43. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Divider.kt +0 -23
  44. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Icon.kt +0 -76
  45. package/compose/src/commonMain/kotlin/vn/momo/kits/components/IconButton.kt +0 -148
  46. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Image.kt +0 -188
  47. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Information.kt +0 -116
  48. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Input.kt +0 -448
  49. package/compose/src/commonMain/kotlin/vn/momo/kits/components/InputDropDown.kt +0 -172
  50. package/compose/src/commonMain/kotlin/vn/momo/kits/components/InputMoney.kt +0 -255
  51. package/compose/src/commonMain/kotlin/vn/momo/kits/components/InputOTP.kt +0 -231
  52. package/compose/src/commonMain/kotlin/vn/momo/kits/components/InputPhoneNumber.kt +0 -233
  53. package/compose/src/commonMain/kotlin/vn/momo/kits/components/InputSearch.kt +0 -254
  54. package/compose/src/commonMain/kotlin/vn/momo/kits/components/InputTextArea.kt +0 -241
  55. package/compose/src/commonMain/kotlin/vn/momo/kits/components/LazyColumnWithBouncing.kt +0 -364
  56. package/compose/src/commonMain/kotlin/vn/momo/kits/components/PaginationDot.kt +0 -56
  57. package/compose/src/commonMain/kotlin/vn/momo/kits/components/PaginationNumber.kt +0 -41
  58. package/compose/src/commonMain/kotlin/vn/momo/kits/components/PaginationScroll.kt +0 -92
  59. package/compose/src/commonMain/kotlin/vn/momo/kits/components/PaginationWhiteDot.kt +0 -40
  60. package/compose/src/commonMain/kotlin/vn/momo/kits/components/PopupNotify.kt +0 -352
  61. package/compose/src/commonMain/kotlin/vn/momo/kits/components/PopupPromotion.kt +0 -103
  62. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Radio.kt +0 -70
  63. package/compose/src/commonMain/kotlin/vn/momo/kits/components/ScaleSizeScope.kt +0 -17
  64. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Skeleton.kt +0 -96
  65. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Switch.kt +0 -96
  66. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Tag.kt +0 -92
  67. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Text.kt +0 -130
  68. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Title.kt +0 -214
  69. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Tooltip.kt +0 -576
  70. package/compose/src/commonMain/kotlin/vn/momo/kits/components/TrustBanner.kt +0 -177
  71. package/compose/src/commonMain/kotlin/vn/momo/kits/components/datetimepicker/DateTimePicker.kt +0 -205
  72. package/compose/src/commonMain/kotlin/vn/momo/kits/components/datetimepicker/DateTimePickerTypes.kt +0 -29
  73. package/compose/src/commonMain/kotlin/vn/momo/kits/components/datetimepicker/DateTimePickerUtils.kt +0 -239
  74. package/compose/src/commonMain/kotlin/vn/momo/kits/components/datetimepicker/WheelPicker.kt +0 -191
  75. package/compose/src/commonMain/kotlin/vn/momo/kits/const/Colors.kt +0 -306
  76. package/compose/src/commonMain/kotlin/vn/momo/kits/const/Radius.kt +0 -12
  77. package/compose/src/commonMain/kotlin/vn/momo/kits/const/Spacing.kt +0 -13
  78. package/compose/src/commonMain/kotlin/vn/momo/kits/const/Theme.kt +0 -185
  79. package/compose/src/commonMain/kotlin/vn/momo/kits/const/Typography.kt +0 -285
  80. package/compose/src/commonMain/kotlin/vn/momo/kits/layout/Card.kt +0 -2
  81. package/compose/src/commonMain/kotlin/vn/momo/kits/layout/Item.kt +0 -35
  82. package/compose/src/commonMain/kotlin/vn/momo/kits/layout/Section.kt +0 -2
  83. package/compose/src/commonMain/kotlin/vn/momo/kits/modifier/AutomationId.kt +0 -59
  84. package/compose/src/commonMain/kotlin/vn/momo/kits/modifier/Clickable.kt +0 -68
  85. package/compose/src/commonMain/kotlin/vn/momo/kits/modifier/Conditional.kt +0 -11
  86. package/compose/src/commonMain/kotlin/vn/momo/kits/modifier/DeprecatedModifier.kt +0 -14
  87. package/compose/src/commonMain/kotlin/vn/momo/kits/modifier/Shadow.kt +0 -50
  88. package/compose/src/commonMain/kotlin/vn/momo/kits/modifier/Size.kt +0 -51
  89. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/BottomSheet.kt +0 -239
  90. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/ModalScreen.kt +0 -119
  91. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/Navigation.kt +0 -98
  92. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/NavigationContainer.kt +0 -161
  93. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/Navigator.kt +0 -331
  94. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/StackScreen.kt +0 -497
  95. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/bottomtab/BottomTab.kt +0 -162
  96. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/bottomtab/BottomTabBar.kt +0 -243
  97. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/bottomtab/CurvedContainer.kt +0 -86
  98. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/FloatingButton.kt +0 -187
  99. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/Header.kt +0 -279
  100. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/HeaderBackground.kt +0 -80
  101. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/HeaderRight.kt +0 -306
  102. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/HeaderTitle.kt +0 -32
  103. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/HeaderUser.kt +0 -370
  104. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/SnackBar.kt +0 -132
  105. package/compose/src/commonMain/kotlin/vn/momo/kits/platform/Platform.kt +0 -42
  106. package/compose/src/commonMain/kotlin/vn/momo/kits/utils/Icons.kt +0 -1329
  107. package/compose/src/commonMain/kotlin/vn/momo/kits/utils/Resources.kt +0 -62
  108. package/compose/src/commonMain/kotlin/vn/momo/kits/utils/Tracking.kt +0 -15
  109. package/compose/src/commonMain/kotlin/vn/momo/kits/utils/Utils.kt +0 -88
  110. package/compose/src/iosMain/kotlin/vn/momo/kits/platform/Platform.ios.kt +0 -149
  111. package/gradle/libs.versions.toml +0 -57
  112. package/gradle/wrapper/gradle-wrapper.jar +0 -0
  113. package/gradle/wrapper/gradle-wrapper.properties +0 -8
  114. package/gradle.properties +0 -26
  115. package/gradlew +0 -252
  116. package/gradlew.bat +0 -94
  117. package/settings.gradle.kts +0 -52
@@ -1,241 +0,0 @@
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.Box
8
- import androidx.compose.foundation.layout.Column
9
- import androidx.compose.foundation.layout.Row
10
- import androidx.compose.foundation.layout.Spacer
11
- import androidx.compose.foundation.layout.fillMaxWidth
12
- import androidx.compose.foundation.layout.height
13
- import androidx.compose.foundation.layout.offset
14
- import androidx.compose.foundation.layout.padding
15
- import androidx.compose.foundation.layout.width
16
- import androidx.compose.foundation.layout.wrapContentSize
17
- import androidx.compose.foundation.shape.RoundedCornerShape
18
- import androidx.compose.foundation.text.BasicTextField
19
- import androidx.compose.foundation.text.KeyboardOptions
20
- import androidx.compose.runtime.Composable
21
- import androidx.compose.runtime.MutableState
22
- import androidx.compose.runtime.getValue
23
- import androidx.compose.runtime.mutableStateOf
24
- import androidx.compose.runtime.remember
25
- import androidx.compose.runtime.setValue
26
- import androidx.compose.ui.Alignment
27
- import androidx.compose.ui.Modifier
28
- import androidx.compose.ui.focus.onFocusChanged
29
- import androidx.compose.ui.graphics.Color
30
- import androidx.compose.ui.text.TextStyle
31
- import androidx.compose.ui.text.input.KeyboardType
32
- import androidx.compose.ui.unit.Dp
33
- import androidx.compose.ui.unit.dp
34
- import androidx.compose.ui.unit.sp
35
- import androidx.compose.ui.zIndex
36
- import vn.momo.kits.const.AppTheme
37
- import vn.momo.kits.const.Spacing
38
- import vn.momo.kits.application.IsShowBaseLineDebug
39
- import vn.momo.kits.const.Colors
40
- import vn.momo.kits.const.Typography
41
- import vn.momo.kits.const.scaleSize
42
- import vn.momo.kits.modifier.conditional
43
-
44
- const val MAX_LENGTH = 300
45
- val DEFAULT_HEIGHT = 104.dp
46
-
47
- @Composable
48
- fun InputTextArea(
49
- text: MutableState<String> = remember { mutableStateOf("") },
50
- maxLength: Int = MAX_LENGTH,
51
- height: Dp = DEFAULT_HEIGHT,
52
- floatingValue: String = "",
53
- floatingValueColor: Color = AppTheme.current.colors.text.hint,
54
- floatingIcon: String = "",
55
- floatingIconColor: Color = AppTheme.current.colors.text.default,
56
- placeholder: String = "",
57
- size: InputSize = InputSize.SMALL,
58
- onChangeText: (String) -> Unit = {},
59
- error: String = "",
60
- errorSpacing: Boolean = false,
61
- disabled: Boolean = false,
62
- icon: String = "",
63
- iconColor: Color = AppTheme.current.colors.text.default,
64
- onRightIconPressed: () -> Unit = {},
65
- onFocus: () -> Unit = {},
66
- onBlur: () -> Unit = {},
67
- loading: Boolean = false,
68
- required: Boolean = false,
69
- fontWeight: InputFontWeight = InputFontWeight.REGULAR,
70
- keyboardType: KeyboardType = KeyboardType.Text,
71
- ) {
72
- var isFocused by remember { mutableStateOf(false) }
73
- var isBlurred = false
74
- val disabledColor = AppTheme.current.colors.text.disable
75
- var textColor = AppTheme.current.colors.text.default
76
- var placeholderColor = AppTheme.current.colors.text.hint
77
- var iconTintColor = iconColor
78
- val floatingTitleColor = when {
79
- disabled -> AppTheme.current.colors.text.disable
80
- else -> floatingValueColor
81
- }
82
- val floatingIconTintColor = when {
83
- disabled -> AppTheme.current.colors.text.disable
84
- else -> floatingIconColor
85
- }
86
-
87
- if (disabled) {
88
- textColor = disabledColor
89
- placeholderColor = disabledColor
90
- iconTintColor = disabledColor
91
- }
92
-
93
- val fontSize = 16.sp
94
- val lineHeight = 24.sp
95
- val scaleFontSize = scaleSize(16.sp)
96
- val scaleLineHeight = scaleSize(24.sp)
97
- val scaleHeight = scaleSize(height)
98
-
99
- Column(modifier = Modifier.conditional(IsShowBaseLineDebug) {
100
- border(1.dp, Colors.blue_03)
101
- }) {
102
- BasicTextField(
103
- enabled = !disabled,
104
- singleLine = false,
105
- value = text.value,
106
- textStyle = TextStyle(
107
- color = textColor,
108
- fontSize = scaleFontSize,
109
- lineHeight = scaleLineHeight,
110
- fontWeight = fontWeight.value
111
- ),
112
- keyboardOptions = KeyboardOptions.Default.copy(keyboardType = keyboardType),
113
- modifier = Modifier.height(scaleHeight).onFocusChanged {
114
- isFocused = it.isFocused
115
- if (it.isFocused) {
116
- onFocus()
117
- }
118
- if (!it.isFocused && isBlurred) onBlur()
119
- if (it.isFocused && !isBlurred) isBlurred = true
120
- },
121
- onValueChange = {
122
- if (it.length <= maxLength) {
123
- onChangeText(it)
124
- }
125
- },
126
- decorationBox = { innerTextField ->
127
- // Floating Icon
128
- if (floatingValue.isNotEmpty() || floatingIcon.isNotEmpty()) {
129
- Box(
130
- modifier = Modifier.wrapContentSize()
131
- .offset(y = (-scaleHeight / 2), x = (Spacing.S))
132
- .background(AppTheme.current.colors.background.surface)
133
- .zIndex(10f),
134
- ) {
135
- Row(
136
- modifier = Modifier
137
- .padding(horizontal = Spacing.S),
138
- verticalAlignment = Alignment.CenterVertically
139
- ) {
140
- Text(
141
- floatingValue,
142
- style = Typography.labelSMedium,
143
- color = floatingTitleColor
144
- )
145
- if (required) {
146
- Text(
147
- "*",
148
- style = Typography.labelSMedium,
149
- color = AppTheme.current.colors.error.primary,
150
- )
151
- }
152
- if (floatingIcon.isNotEmpty()) {
153
- Icon(
154
- source = floatingIcon,
155
- modifier = Modifier.padding(start = Spacing.XS),
156
- size = 16.dp,
157
- color = floatingIconTintColor
158
- )
159
- }
160
- }
161
- }
162
- }
163
- //input box wrapper
164
- Box(
165
- modifier = Modifier.fillMaxWidth()
166
- .background(
167
- AppTheme.current.colors.background.surface,
168
- RoundedCornerShape(size.values.borderRadius)
169
- )
170
- .border(
171
- 1.dp,
172
- getBorderColor(isFocused, error, disabled),
173
- RoundedCornerShape(size.values.borderRadius)
174
- ),
175
- contentAlignment = Alignment.TopStart
176
- ) {
177
- Column {
178
- Row(
179
- modifier = Modifier.weight(1f).padding(
180
- start = Spacing.M,
181
- end = Spacing.M,
182
- top = Spacing.M
183
- ),
184
- verticalAlignment = Alignment.Top
185
- ) {
186
- Box(Modifier.weight(1f)) {
187
- if (text.value.isEmpty()) {
188
- Text(
189
- text = placeholder,
190
- style = TextStyle(
191
- fontSize = fontSize,
192
- lineHeight = lineHeight,
193
- fontWeight = fontWeight.value
194
- ),
195
- color = placeholderColor
196
- )
197
- }
198
- innerTextField()
199
- }
200
- if (isFocused && text.value.isNotEmpty()) {
201
- Row {
202
- Spacer(Modifier.width(Spacing.XS))
203
- Icon(
204
- source = "24_navigation_close_circle_full",
205
- size = 16.dp,
206
- color = AppTheme.current.colors.text.hint,
207
- modifier = Modifier.clickable(
208
- onClick = {
209
- text.value = ""
210
- onChangeText.invoke("")
211
- },
212
- interactionSource = remember { MutableInteractionSource() },
213
- indication = null
214
- )
215
- )
216
- }
217
- }
218
- RenderRightIcon(loading, icon, iconTintColor, onRightIconPressed)
219
- }
220
-
221
- Box(
222
- modifier = Modifier
223
- .fillMaxWidth()
224
- .padding(end = Spacing.M),
225
- contentAlignment = Alignment.CenterEnd
226
- ) {
227
- Text(
228
- "${text.value.length}/$maxLength",
229
- style = Typography.descriptionXsRegular,
230
- color = AppTheme.current.colors.text.hint
231
- )
232
- }
233
- Spacer(Modifier.height(Spacing.M))
234
- }
235
- }
236
- },
237
- )
238
- // show error
239
- ErrorView(error, errorSpacing, "")
240
- }
241
- }
@@ -1,364 +0,0 @@
1
- package vn.momo.kits.components
2
-
3
- import androidx.compose.animation.core.LinearEasing
4
- import androidx.compose.animation.core.RepeatMode
5
- import androidx.compose.animation.core.animateFloat
6
- import androidx.compose.animation.core.infiniteRepeatable
7
- import androidx.compose.animation.core.rememberInfiniteTransition
8
- import androidx.compose.animation.core.tween
9
- import androidx.compose.foundation.Canvas
10
- import androidx.compose.foundation.OverscrollEffect
11
- import androidx.compose.foundation.background
12
- import androidx.compose.foundation.gestures.FlingBehavior
13
- import androidx.compose.foundation.gestures.ScrollableDefaults
14
- import androidx.compose.foundation.layout.Arrangement
15
- import androidx.compose.foundation.layout.Box
16
- import androidx.compose.foundation.layout.PaddingValues
17
- import androidx.compose.foundation.layout.fillMaxWidth
18
- import androidx.compose.foundation.layout.padding
19
- import androidx.compose.foundation.layout.size
20
- import androidx.compose.foundation.lazy.LazyColumn
21
- import androidx.compose.foundation.lazy.LazyListLayoutInfo
22
- import androidx.compose.foundation.lazy.LazyListScope
23
- import androidx.compose.foundation.lazy.LazyListState
24
- import androidx.compose.foundation.lazy.rememberLazyListState
25
- import androidx.compose.material.ExperimentalMaterialApi
26
- import androidx.compose.material.pullrefresh.PullRefreshIndicator
27
- import androidx.compose.material.pullrefresh.pullRefresh
28
- import androidx.compose.material.pullrefresh.rememberPullRefreshState
29
- import androidx.compose.runtime.Composable
30
- import androidx.compose.runtime.LaunchedEffect
31
- import androidx.compose.runtime.MutableState
32
- import androidx.compose.runtime.State
33
- import androidx.compose.runtime.mutableIntStateOf
34
- import androidx.compose.runtime.mutableStateOf
35
- import androidx.compose.runtime.remember
36
- import androidx.compose.runtime.snapshotFlow
37
- import androidx.compose.ui.Alignment
38
- import androidx.compose.ui.Modifier
39
- import androidx.compose.ui.composed
40
- import androidx.compose.ui.geometry.CornerRadius
41
- import androidx.compose.ui.geometry.Offset
42
- import androidx.compose.ui.geometry.Size
43
- import androidx.compose.ui.graphics.Color
44
- import androidx.compose.ui.graphics.drawscope.rotate
45
- import androidx.compose.ui.hapticfeedback.HapticFeedbackType
46
- import androidx.compose.ui.layout.Measurable
47
- import androidx.compose.ui.layout.MeasureResult
48
- import androidx.compose.ui.layout.MeasureScope
49
- import androidx.compose.ui.layout.layout
50
- import androidx.compose.ui.platform.LocalDensity
51
- import androidx.compose.ui.platform.LocalHapticFeedback
52
- import androidx.compose.ui.unit.Constraints
53
- import androidx.compose.ui.unit.IntOffset
54
- import androidx.compose.ui.unit.dp
55
- import kotlinx.coroutines.flow.collect
56
- import kotlinx.coroutines.flow.combine
57
- import kotlinx.coroutines.flow.distinctUntilChanged
58
- import vn.momo.kits.const.Spacing
59
- import vn.momo.kits.modifier.conditional
60
- import vn.momo.kits.platform.getPlatformName
61
-
62
- private const val BOX_KEY = "LazyColumnWithBouncing_BOX_KEY"
63
-
64
-
65
- @OptIn(ExperimentalMaterialApi::class)
66
- @Composable
67
- fun overScrollWithPullToRefresh(
68
- pullRefreshState: PullToRefreshCustomState?,
69
- ): OverscrollEffect {
70
- val density = LocalDensity.current
71
- val hapticFeedback = LocalHapticFeedback.current
72
-
73
- return remember(density) {
74
- CupertinoOverscrollEffect(
75
- pullRefreshState = if (pullRefreshState != null && pullRefreshState.sendHaptic != null) pullRefreshState
76
- else pullRefreshState?.copy(
77
- sendHaptic = {
78
- hapticFeedback.performHapticFeedback(HapticFeedbackType.LongPress)
79
- }),
80
- density = density.density,
81
- applyClip = false,
82
- )
83
- }
84
- }
85
-
86
- @OptIn(ExperimentalMaterialApi::class)
87
- @Composable
88
- fun LazyColumnWithBouncing(
89
- modifier: Modifier = Modifier,
90
- modifierIndicator: Modifier = Modifier,
91
- state: LazyListState = rememberLazyListState(),
92
- contentPadding: PaddingValues = PaddingValues(0.dp),
93
- reverseLayout: Boolean = false,
94
- verticalArrangement: Arrangement.Vertical = if (!reverseLayout) Arrangement.Top else Arrangement.Bottom,
95
- horizontalAlignment: Alignment.Horizontal = Alignment.Start,
96
- flingBehavior: FlingBehavior = ScrollableDefaults.flingBehavior(),
97
- userScrollEnabled: Boolean = true,
98
- pullRefreshState: PullToRefreshCustomState? = null,
99
- content: LazyListScope.() -> Unit,
100
- ) {
101
- val height = remember { mutableIntStateOf(0) }
102
- val maxConstraint = remember { mutableIntStateOf(0) }
103
-
104
- LaunchedEffect(Unit) {
105
- val addOn = 1
106
- combine(
107
- snapshotFlow { maxConstraint.value }.distinctUntilChanged(),
108
- snapshotFlow { state.layoutInfo }
109
- ) { cHeight, layoutInfo ->
110
- height.value = when {
111
- height.value == 0 && (state.canScrollForward || state.canScrollBackward) -> 0
112
- layoutInfo.viewportSize.height >= cHeight && height.value == 0 -> 0
113
- layoutInfo.visibleItemsInfo.lastOrNull()?.key != BOX_KEY -> 0
114
- else -> {
115
- val vH = layoutInfo.totalHeight
116
- (cHeight - (vH - height.value) + addOn).coerceIn(0, cHeight)
117
- }
118
- }
119
- }.collect()
120
- }
121
-
122
- val measureLayout =
123
- remember<MeasureScope.(Measurable, Constraints) -> MeasureResult>(maxConstraint) {
124
- { m, c ->
125
- val p = m.measure(c.copy(minHeight = 0))
126
- maxConstraint.value = c.maxHeight
127
- layout(p.width, p.height) {
128
- p.place(IntOffset.Zero)
129
- }
130
- }
131
- }
132
-
133
- if (getPlatformName() == "iOS") {
134
-
135
- val overscroll = overScrollWithPullToRefresh(
136
- pullRefreshState = remember {
137
- pullRefreshState
138
- })
139
-
140
- Box {
141
- IosIndicator(
142
- effect = overscroll,
143
- modifier = modifierIndicator.align(Alignment.TopCenter).padding(top = Spacing.L).size(24.dp),
144
- )
145
- LazyColumn(
146
- modifier = modifier
147
- .layout(measure = measureLayout),
148
- state = state,
149
- contentPadding = contentPadding,
150
- flingBehavior = flingBehavior,
151
- horizontalAlignment = horizontalAlignment,
152
- verticalArrangement = verticalArrangement,
153
- reverseLayout = reverseLayout,
154
- userScrollEnabled = userScrollEnabled,
155
- overscrollEffect = overscroll,
156
- ) {
157
- content()
158
- item(key = BOX_KEY) {
159
- FakeBox(height)
160
- }
161
- }
162
- }
163
- } else {
164
- val pullState = rememberPullRefreshState(
165
- refreshing = pullRefreshState?.refreshingState?.value ?: false,
166
- onRefresh = {
167
- pullRefreshState?.onRefresh()
168
- },
169
- )
170
-
171
- Box(
172
- contentAlignment = Alignment.TopCenter,
173
- modifier = Modifier.conditional(pullRefreshState != null) {
174
- pullRefresh(pullState)
175
- }) {
176
- LazyColumn(
177
- modifier = modifier.calculateHeight(height),
178
- state = state,
179
- contentPadding = contentPadding,
180
- flingBehavior = flingBehavior,
181
- horizontalAlignment = horizontalAlignment,
182
- verticalArrangement = verticalArrangement,
183
- reverseLayout = reverseLayout,
184
- userScrollEnabled = userScrollEnabled,
185
- ) {
186
- content()
187
- }
188
- if (pullRefreshState != null) {
189
- PullRefreshIndicator(
190
- pullRefreshState.refreshingState.value,
191
- pullState,
192
- modifierIndicator.align(Alignment.TopCenter)
193
- )
194
- }
195
-
196
- }
197
- }
198
- }
199
-
200
- @Composable
201
- private fun IosIndicator(
202
- effect: OverscrollEffect?,
203
- modifier: Modifier = Modifier,
204
- ) {
205
- if (effect !is CupertinoOverscrollEffect) return
206
- if (effect.pullRefreshState == null) return
207
- val state = effect.pullRefreshState
208
-
209
-
210
- SpinningProgressBarLoading(
211
- progress = state.position, modifier = modifier
212
- )
213
- }
214
-
215
- @Composable
216
- private fun SpinningProgressBarLoading(
217
- progress: State<Float>, modifier: Modifier = Modifier
218
- ) {
219
- if (progress.value <= 0) {
220
- return
221
- }
222
-
223
- val hapticFeedback = LocalHapticFeedback.current
224
- val isSendHaptic = remember { mutableStateOf(false) }
225
-
226
- val totalStep = 8
227
- val stepAngle = 360f / totalStep
228
- val baseOpacity = 1.0f / (totalStep + 2)
229
- val minAlpha = 0.2f
230
-
231
- val infiniteTransition = rememberInfiniteTransition()
232
- val angle = infiniteTransition.animateFloat(
233
- initialValue = 0f, targetValue = totalStep.toFloat(), animationSpec = infiniteRepeatable(
234
- animation = tween(totalStep * 100, easing = LinearEasing),
235
- repeatMode = RepeatMode.Restart
236
- )
237
- )
238
-
239
- LaunchedEffect(Unit) {
240
- snapshotFlow { progress.value }.collect {
241
- if (it >= 1.0f && !isSendHaptic.value) {
242
- hapticFeedback.performHapticFeedback(HapticFeedbackType.LongPress)
243
- isSendHaptic.value = true
244
- } else if (progress.value < 1.0f && isSendHaptic.value) {
245
- isSendHaptic.value = false
246
- }
247
- }
248
-
249
- }
250
-
251
- Canvas(modifier = modifier) {
252
- val percentage = progress.value
253
- val clampedProgress = percentage.coerceIn(0f, 1f)
254
- val value = (totalStep * clampedProgress).coerceIn(0f, totalStep.toFloat())
255
- val integerPart = value.toInt()
256
- val decimalPart = value - integerPart
257
-
258
- if (integerPart <= 0) return@Canvas
259
-
260
- val canvasWidth = size.width
261
- val canvasHeight = size.height
262
- val width = size.width * 0.3f
263
- val height = size.height / 8
264
- val cornerRadius = (width.coerceAtMost(height) / 2)
265
-
266
- if (percentage >= 1f) {
267
- for (i in 0..360 step 360 / totalStep) {
268
- rotate(i.toFloat()) {
269
- drawRoundRect(
270
- color = Color.LightGray.copy(alpha = .7f),
271
- topLeft = Offset(canvasWidth - width, (canvasHeight - height) / 2),
272
- size = Size(width, height),
273
- cornerRadius = CornerRadius(cornerRadius, cornerRadius)
274
- )
275
- }
276
- }
277
-
278
- val coefficient = 360f / totalStep
279
-
280
- for (i in 1..4) {
281
- rotate((angle.value.toInt() + i) * coefficient) {
282
- drawRoundRect(
283
- color = Color.Gray.copy(alpha = (0.2f + 0.2f * i).coerceIn(0f, 1f)),
284
- topLeft = Offset(canvasWidth - width, (canvasHeight - height) / 2),
285
- size = Size(width, height),
286
- cornerRadius = CornerRadius(cornerRadius, cornerRadius)
287
- )
288
- }
289
- }
290
- return@Canvas
291
- }
292
-
293
- val topLeft = Offset(canvasWidth - width, (canvasHeight - height) / 2)
294
- val rectSize = Size(width, height)
295
- val cornerRadiusValue = CornerRadius(cornerRadius, cornerRadius)
296
- repeat(integerPart) { i ->
297
- val angle = i * stepAngle - 90f
298
- val stepOpacity = 1f - (baseOpacity * i)
299
-
300
- val alpha = if (i == integerPart - 1) {
301
- (decimalPart * stepOpacity).coerceIn(minAlpha, 1f)
302
- } else {
303
- stepOpacity.coerceIn(minAlpha, 1f)
304
- }
305
-
306
- rotate(angle) {
307
- drawRoundRect(
308
- color = Color.Gray.copy(alpha = alpha),
309
- topLeft = topLeft,
310
- size = rectSize,
311
- cornerRadius = cornerRadiusValue
312
- )
313
- }
314
- }
315
- }
316
- }
317
-
318
- @Composable
319
- private fun FakeBox(height: MutableState<Int>) {
320
- Box(
321
- modifier = Modifier
322
- .dynamicHeight(height).fillMaxWidth().background(Color.Red),
323
- )
324
- }
325
-
326
- private fun Modifier.calculateHeight(
327
- height: MutableState<Int>,
328
- ) = this.layout { m, c ->
329
- val p = m.measure(
330
- constraints = c.copy(
331
- minHeight = 0
332
- )
333
- )
334
- val vH = p.height
335
- val cMaxHeight = c.maxHeight
336
- val addOn = 0.5.dp.roundToPx()
337
- var extraSpace = addOn + height.value
338
- if (height.value == 0) {
339
- extraSpace = 0
340
- }
341
- if (vH - extraSpace >= cMaxHeight && height.value != 0) {
342
- height.value = 0
343
- } else if (vH - extraSpace < cMaxHeight) {
344
- height.value = cMaxHeight - (vH - extraSpace) + addOn
345
- }
346
-
347
- layout(c.maxWidth, cMaxHeight) {
348
- p.place(IntOffset.Zero)
349
- }
350
- }
351
-
352
-
353
- private fun Modifier.dynamicHeight(height: MutableState<Int>) = composed {
354
- val measure = remember<MeasureScope.(Measurable, Constraints) -> MeasureResult>(this, height) {
355
- { m, c ->
356
- val p = m.measure(c)
357
- layout(c.maxWidth, height.value) { p.place(IntOffset.Zero) }
358
- }
359
- }
360
- this.layout(measure = measure)
361
- }
362
-
363
- private val LazyListLayoutInfo.totalHeight: Int
364
- get() = visibleItemsInfo.sumOf { it.size }
@@ -1,56 +0,0 @@
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
- }
@@ -1,41 +0,0 @@
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
- }