@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,720 +0,0 @@
1
- package vn.momo.kits.application
2
-
3
- import androidx.compose.animation.animateContentSize
4
- import androidx.compose.foundation.ScrollState
5
- import androidx.compose.foundation.background
6
- import androidx.compose.foundation.border
7
- import androidx.compose.foundation.gestures.detectTapGestures
8
- import androidx.compose.foundation.interaction.FocusInteraction
9
- import androidx.compose.foundation.interaction.MutableInteractionSource
10
- import androidx.compose.foundation.layout.Arrangement
11
- import androidx.compose.foundation.layout.Box
12
- import androidx.compose.foundation.layout.Column
13
- import androidx.compose.foundation.layout.Row
14
- import androidx.compose.foundation.layout.WindowInsets
15
- import androidx.compose.foundation.layout.asPaddingValues
16
- import androidx.compose.foundation.layout.fillMaxSize
17
- import androidx.compose.foundation.layout.fillMaxWidth
18
- import androidx.compose.foundation.layout.height
19
- import androidx.compose.foundation.layout.ime
20
- import androidx.compose.foundation.layout.navigationBars
21
- import androidx.compose.foundation.layout.offset
22
- import androidx.compose.foundation.layout.padding
23
- import androidx.compose.foundation.layout.size
24
- import androidx.compose.foundation.layout.sizeIn
25
- import androidx.compose.foundation.rememberScrollState
26
- import androidx.compose.foundation.shape.CircleShape
27
- import androidx.compose.foundation.shape.RoundedCornerShape
28
- import androidx.compose.foundation.text.BasicTextField
29
- import androidx.compose.foundation.text.KeyboardActions
30
- import androidx.compose.foundation.text.KeyboardOptions
31
- import androidx.compose.foundation.verticalScroll
32
- import androidx.compose.runtime.Composable
33
- import androidx.compose.runtime.LaunchedEffect
34
- import androidx.compose.runtime.MutableState
35
- import androidx.compose.runtime.Stable
36
- import androidx.compose.runtime.State
37
- import androidx.compose.runtime.derivedStateOf
38
- import androidx.compose.runtime.getValue
39
- import androidx.compose.runtime.mutableStateOf
40
- import androidx.compose.runtime.produceState
41
- import androidx.compose.runtime.remember
42
- import androidx.compose.runtime.rememberUpdatedState
43
- import androidx.compose.runtime.snapshotFlow
44
- import androidx.compose.ui.Alignment
45
- import androidx.compose.ui.Modifier
46
- import androidx.compose.ui.composed
47
- import androidx.compose.ui.draw.clip
48
- import androidx.compose.ui.draw.drawBehind
49
- import androidx.compose.ui.draw.drawWithContent
50
- import androidx.compose.ui.geometry.Offset
51
- import androidx.compose.ui.graphics.Brush
52
- import androidx.compose.ui.graphics.Color
53
- import androidx.compose.ui.graphics.graphicsLayer
54
- import androidx.compose.ui.input.pointer.pointerInput
55
- import androidx.compose.ui.layout.Layout
56
- import androidx.compose.ui.layout.Measurable
57
- import androidx.compose.ui.layout.MeasurePolicy
58
- import androidx.compose.ui.layout.MeasureResult
59
- import androidx.compose.ui.layout.MeasureScope
60
- import androidx.compose.ui.layout.Placeable
61
- import androidx.compose.ui.layout.layoutId
62
- import androidx.compose.ui.platform.LocalDensity
63
- import androidx.compose.ui.platform.LocalFocusManager
64
- import androidx.compose.ui.platform.LocalSoftwareKeyboardController
65
- import androidx.compose.ui.text.style.TextOverflow
66
- import androidx.compose.ui.unit.Constraints
67
- import androidx.compose.ui.unit.Dp
68
- import androidx.compose.ui.unit.IntOffset
69
- import androidx.compose.ui.unit.LayoutDirection
70
- import androidx.compose.ui.unit.dp
71
- import androidx.compose.ui.unit.sp
72
- import kotlinx.coroutines.flow.collectLatest
73
- import kotlinx.coroutines.flow.mapNotNull
74
- import vn.momo.kits.components.Icon
75
- import vn.momo.kits.components.Text
76
- import vn.momo.kits.const.AppTheme
77
- import vn.momo.kits.const.Colors
78
- import vn.momo.kits.const.Radius
79
- import vn.momo.kits.const.Spacing
80
- import vn.momo.kits.const.Typography
81
- import vn.momo.kits.modifier.kitsAutomationId
82
- import vn.momo.kits.modifier.noFeedbackClickable
83
- import vn.momo.kits.modifier.setAutomationId
84
- import vn.momo.kits.modifier.shadow
85
- import vn.momo.kits.utils.getAppStatusBarHeight
86
- import kotlin.math.max
87
-
88
- @Composable
89
- fun LiteScreen(
90
- scrollable: Boolean = true,
91
- scrollState: ScrollState? = null,
92
- headerType: HeaderType = HeaderType.DEFAULT,
93
- verticalArrangement: Arrangement.Vertical = Arrangement.Top,
94
- horizontalAlignment: Alignment.Horizontal = Alignment.Start,
95
- backgroundColor: Color = AppTheme.current.colors.background.default,
96
- /* Begin of header props */
97
- title: String? = null,
98
- inputSearchProps: LiteInputSearchProps? = null,
99
- goBack: (() -> Unit)? = null,
100
- headerRight: @Composable() (() -> Unit)? = null,
101
- useAnimationSearch: Boolean = true,
102
- titlePosition: TitlePosition = TitlePosition.LEFT,
103
- headerRightData: HeaderRightData? = null,
104
- /* End of header props */
105
-
106
- screenContent: @Composable () -> Unit,
107
- ) {
108
- val content by rememberUpdatedState(screenContent)
109
-
110
- val finalScrollState = scrollState ?: rememberScrollState()
111
-
112
- Column(
113
- modifier = Modifier
114
- .fillMaxSize()
115
- .background(color = backgroundColor)
116
- .hideKeyboardOnTap(),
117
- verticalArrangement = verticalArrangement,
118
- horizontalAlignment = horizontalAlignment,
119
- ) {
120
- val contentModifier = remember(scrollable, finalScrollState) {
121
- var res = Modifier.weight(1f)
122
- if (scrollable) {
123
- res = res.verticalScroll(finalScrollState)
124
- }
125
- res
126
- }
127
-
128
- LiteScreenHeader(
129
- scrollState = finalScrollState,
130
- title = title,
131
- headerRight = headerRight,
132
- headerType = headerType,
133
- onGoBack = goBack,
134
- inputSearchProps = inputSearchProps,
135
- titlePosition = titlePosition,
136
- useAnimationSearch = useAnimationSearch,
137
- headerRightData = headerRightData,
138
- )
139
-
140
- Box(
141
- modifier = contentModifier,
142
- contentAlignment = Alignment.TopCenter,
143
- ) {
144
- content()
145
- }
146
- }
147
- }
148
-
149
- private object HeaderId {
150
- private const val PACKAGE_NAME = "vn.momo.compose.kits"
151
- const val BACK_ID = "${PACKAGE_NAME}.back"
152
- const val HEADER_RIGHT_ID = "${PACKAGE_NAME}.headerRight"
153
- const val INPUT_SEARCH_ID = "${PACKAGE_NAME}.inputSearch"
154
- const val TITLE_ID = "${PACKAGE_NAME}.title"
155
-
156
- val EXTENDED_HEADER_HEIGHT = 154.dp
157
- }
158
-
159
- @Composable
160
- private fun LiteScreenHeader(
161
- scrollState: ScrollState?,
162
- title: String? = null,
163
- tintColor: Color? = null,
164
- headerRightData: HeaderRightData? = null,
165
- headerType: HeaderType = HeaderType.DEFAULT,
166
- titlePosition: TitlePosition = TitlePosition.LEFT,
167
- useAnimationSearch: Boolean = true,
168
- onGoBack: (() -> Unit)? = null,
169
- inputSearchProps: LiteInputSearchProps? = null,
170
- headerRight: @Composable (() -> Unit)? = null,
171
- ) {
172
- val statusBarHeight = getAppStatusBarHeight()
173
- if (headerType == HeaderType.NONE) {
174
- Box(modifier = Modifier.height(statusBarHeight))
175
- return
176
- }
177
- val theme = AppTheme.current
178
- val density = LocalDensity.current
179
-
180
- val isHeaderExtend = remember(headerType) {
181
- headerType == HeaderType.EXTENDED
182
- }
183
- val backgroundHeight = remember(isHeaderExtend, statusBarHeight) {
184
- if (!isHeaderExtend) statusBarHeight + HEADER_HEIGHT.dp
185
- else HeaderId.EXTENDED_HEADER_HEIGHT
186
- }
187
- val listGradientColors = remember {
188
- listOf(
189
- Color(0xFFFDCADE),
190
- Color(0x00FDCADE),
191
- )
192
- }
193
-
194
- val headerColor = remember(tintColor, theme) {
195
- if (tintColor == Colors.black_01) HeaderColor(
196
- tintIconColor = tintColor,
197
- backgroundButton = Colors.black_20.copy(alpha = 0.6f),
198
- borderColor = Colors.black_01.copy(alpha = 0.2f)
199
- )
200
- else HeaderColor(
201
- tintIconColor = tintColor ?: theme.colors.text.default,
202
- backgroundButton = Colors.black_01.copy(alpha = 0.6f),
203
- borderColor = Colors.black_20.copy(alpha = 0.2f)
204
- )
205
- }
206
-
207
- val scrollPercentage = produceState(
208
- initialValue = 0f,
209
- key1 = useAnimationSearch,
210
- key2 = scrollState,
211
- key3 = backgroundHeight,
212
- ) {
213
- if (!useAnimationSearch) return@produceState
214
- scrollState ?: return@produceState
215
- val rangePx = with(density) { backgroundHeight.toPx() }
216
- snapshotFlow { scrollState.value }.collectLatest {
217
- value = (it / rangePx).coerceIn(0f, 1f)
218
- }
219
- }
220
-
221
- val titleStyle = remember {
222
- Typography.actionSBold.copy(
223
- fontSize = 15.sp,
224
- lineHeight = 22.sp,
225
- )
226
- }
227
-
228
- val titleModifier = remember {
229
- Modifier
230
- .kitsAutomationId("title_navigation_header")
231
- .layoutId(HeaderId.TITLE_ID)
232
- .graphicsLayer {
233
- alpha = if (isHeaderExtend && inputSearchProps != null && useAnimationSearch)
234
- (1 - scrollPercentage.value * 2).coerceIn(0f, 1f)
235
- else 1f
236
- }
237
- }
238
-
239
- val policy = remember(
240
- useAnimationSearch,
241
- isHeaderExtend,
242
- statusBarHeight,
243
- titlePosition,
244
- scrollPercentage,
245
- ) {
246
- LiteScreenHeaderPolicy(
247
- useAnimationSearch = useAnimationSearch,
248
- isHeaderExtend = isHeaderExtend,
249
- statusBarHeight = statusBarHeight,
250
- titlePosition = titlePosition,
251
- scrollPercentage = scrollPercentage,
252
- )
253
- }
254
-
255
- Layout(
256
- modifier = Modifier
257
- .animateContentSize()
258
- .drawBehind {
259
- val headerHeight = max(
260
- HeaderId.EXTENDED_HEADER_HEIGHT.toPx(),
261
- size.height,
262
- )
263
- drawRect(color = Colors.black_01)
264
- drawRect(
265
- brush = Brush.linearGradient(
266
- colors = listGradientColors,
267
- start = Offset.Zero,
268
- end = Offset(
269
- x = 0f,
270
- y = headerHeight * (1 - scrollPercentage.value),
271
- ),
272
- )
273
- )
274
- },
275
- content = {
276
- if (onGoBack != null) {
277
- Box(
278
- modifier = Modifier
279
- .size(28.dp)
280
- .layoutId(HeaderId.BACK_ID)
281
- .clip(CircleShape)
282
- .border(
283
- width = 0.2.dp,
284
- color = headerColor.borderColor,
285
- shape = CircleShape,
286
- )
287
- .background(color = headerColor.backgroundButton)
288
- .noFeedbackClickable(onClick = onGoBack)
289
- .setAutomationId("btn_navigation_back")
290
- .padding(Spacing.XS),
291
- ) {
292
- Icon(
293
- source = "arrow-back",
294
- color = headerColor.tintIconColor,
295
- size = 20.dp,
296
- )
297
- }
298
- }
299
-
300
- Box(
301
- modifier = Modifier
302
- .layoutId(HeaderId.HEADER_RIGHT_ID)
303
- ) {
304
- if (headerRight != null) {
305
- headerRight()
306
- } else {
307
- HeaderRight(
308
- headerRight = headerRightData,
309
- tintColor = tintColor,
310
- )
311
- }
312
- }
313
-
314
- if (inputSearchProps != null) {
315
- LiteInputSearch(
316
- modifier = Modifier.layoutId(HeaderId.INPUT_SEARCH_ID),
317
- inputSearchProps = inputSearchProps,
318
- )
319
- }
320
- if (title != null && (inputSearchProps == null || isHeaderExtend)) {
321
- Text(
322
- text = title,
323
- color = headerColor.tintIconColor,
324
- style = titleStyle,
325
- modifier = titleModifier,
326
- maxLines = 1,
327
- overflow = TextOverflow.Ellipsis
328
- )
329
- }
330
- },
331
- measurePolicy = policy,
332
- )
333
- }
334
-
335
- private class LiteScreenHeaderPolicy(
336
- private val useAnimationSearch: Boolean,
337
- private val isHeaderExtend: Boolean,
338
- private val statusBarHeight: Dp,
339
- private val scrollPercentage: State<Float>,
340
- private val titlePosition: TitlePosition,
341
- ) : MeasurePolicy {
342
-
343
- override fun MeasureScope.measure(
344
- measurables: List<Measurable>,
345
- constraints: Constraints
346
- ): MeasureResult {
347
- val spacing12 = Spacing.M.roundToPx()
348
- val statusBarPx = statusBarHeight.roundToPx()
349
- val scrollPercent = scrollPercentage.value
350
-
351
- val realConstraints = constraints.copy(
352
- minWidth = 0,
353
- minHeight = 0,
354
- maxWidth = (constraints.maxWidth - spacing12 * 2)
355
- .coerceAtLeast(0),
356
- )
357
- val backIconPlaceable =
358
- measurables.find { it.layoutId == HeaderId.BACK_ID }?.measure(realConstraints)
359
- val headerRightPlaceable =
360
- measurables.find { it.layoutId == HeaderId.HEADER_RIGHT_ID }?.measure(
361
- realConstraints.copy(
362
- maxWidth = realConstraints.maxWidth / 2,
363
- )
364
- )
365
- val inputSearchConstraints = if (isHeaderExtend) {
366
- val minWidth =
367
- if (useAnimationSearch) realConstraints.maxWidth - backIconPlaceable.safeWidth - headerRightPlaceable.safeWidth - spacing12 * 2
368
- else realConstraints.maxWidth
369
- realConstraints.copy(
370
- maxWidth = (realConstraints.maxWidth * (1 - scrollPercent)).toInt()
371
- .coerceAtLeast(minWidth)
372
- )
373
- } else {
374
- var spaceConsumed = 0
375
- if (backIconPlaceable.safeWidth != 0) spaceConsumed += backIconPlaceable.safeWidth + spacing12
376
- if (headerRightPlaceable.safeWidth != 0) spaceConsumed += headerRightPlaceable.safeWidth + spacing12
377
- realConstraints.copy(
378
- maxWidth = realConstraints.maxWidth - spaceConsumed
379
- )
380
- }
381
- val inputSearchPlaceable = measurables.find { it.layoutId == HeaderId.INPUT_SEARCH_ID }
382
- ?.measure(inputSearchConstraints)
383
- val titlePlaceable = measurables.find { it.layoutId == HeaderId.TITLE_ID }?.measure(
384
- constraints = realConstraints.copy(
385
- maxWidth = realConstraints.maxWidth - backIconPlaceable.safeWidth - headerRightPlaceable.safeWidth - spacing12 * 2
386
- )
387
- )
388
-
389
- val firstRowMaxHeight = buildList {
390
- add(backIconPlaceable.safeHeight)
391
- add(headerRightPlaceable.safeHeight)
392
- if (!isHeaderExtend) {
393
- add(inputSearchPlaceable.safeHeight)
394
- }
395
- if (isHeaderExtend) {
396
- add(titlePlaceable.safeHeight)
397
- }
398
- }.max()
399
-
400
- var defaultHeight = statusBarPx + spacing12 + firstRowMaxHeight + spacing12
401
- if (isHeaderExtend) {
402
- defaultHeight += inputSearchPlaceable.safeHeight + spacing12
403
- }
404
- val height = when {
405
- !useAnimationSearch && !isHeaderExtend -> defaultHeight
406
- else -> (defaultHeight - scrollPercent * (defaultHeight - statusBarPx - HEADER_HEIGHT.dp.roundToPx())).toInt()
407
- }
408
-
409
- return layout(
410
- width = constraints.maxWidth,
411
- height = height,
412
- ) {
413
- val startX = spacing12
414
- val startY = statusBarPx + spacing12
415
- var curX = startX
416
- var curY = startY
417
-
418
- if (backIconPlaceable != null) {
419
- backIconPlaceable.place(
420
- x = startX,
421
- y = startY + backIconPlaceable.verticalCenterOffset(firstRowMaxHeight),
422
- )
423
- curX += backIconPlaceable.safeWidth + spacing12
424
- }
425
-
426
- headerRightPlaceable?.place(
427
- x = constraints.maxWidth - spacing12 - headerRightPlaceable.safeWidth,
428
- y = startY + headerRightPlaceable.verticalCenterOffset(firstRowMaxHeight),
429
- )
430
-
431
- val titleOffset = IntOffset(
432
- x = if (titlePosition == TitlePosition.LEFT) curX
433
- else titlePlaceable.horizontalCenterOffset(
434
- space = constraints.maxWidth,
435
- layoutDirection = layoutDirection,
436
- ),
437
- y = startY + titlePlaceable.verticalCenterOffset(firstRowMaxHeight),
438
- )
439
-
440
- titlePlaceable?.place(titleOffset)
441
-
442
- if (backIconPlaceable != null || headerRightPlaceable != null || titlePlaceable != null) {
443
- curY += firstRowMaxHeight + spacing12
444
- }
445
-
446
- val inputSearchOffset = if (isHeaderExtend) {
447
- IntOffset(
448
- x = startX + ((backIconPlaceable.safeWidth + spacing12) * (scrollPercent * 2f).coerceIn(
449
- 0f, 1f
450
- )).toInt(),
451
- y = (curY * (1 - scrollPercent)).toInt().coerceAtLeast(
452
- startY + inputSearchPlaceable.verticalCenterOffset(firstRowMaxHeight)
453
- ),
454
- )
455
- } else {
456
- IntOffset(
457
- x = curX,
458
- y = startY + inputSearchPlaceable.verticalCenterOffset(firstRowMaxHeight),
459
- )
460
- }
461
-
462
- inputSearchPlaceable?.place(inputSearchOffset)
463
- }
464
- }
465
-
466
- private val Placeable?.safeHeight
467
- get() = this?.height ?: 0
468
- private val Placeable?.safeWidth
469
- get() = this?.width ?: 0
470
-
471
- private fun Placeable?.verticalCenterOffset(space: Int): Int {
472
- if (this == null) return 0
473
- return Alignment.CenterVertically.align(safeHeight, space)
474
- }
475
-
476
- private fun Placeable?.horizontalCenterOffset(
477
- space: Int,
478
- layoutDirection: LayoutDirection
479
- ): Int {
480
- if (this == null) return 0
481
- return Alignment.CenterHorizontally.align(safeWidth, space, layoutDirection)
482
- }
483
- }
484
-
485
- private val EMPTY_FUNC = {}
486
-
487
- @Stable
488
- data class LiteInputSearchProps(
489
- val textFieldState: MutableState<String>,
490
- val onValueChange: (String) -> Unit,
491
- val onClear: () -> Unit = EMPTY_FUNC,
492
- val clearCondition: (() -> Boolean)? = null,
493
- val modifier: Modifier = Modifier,
494
- val onFocused: (() -> Unit) = EMPTY_FUNC,
495
- val keyboardOptions: KeyboardOptions = KeyboardOptions.Default,
496
- val keyboardActions: KeyboardActions = KeyboardActions.Default,
497
- val onKeyboardAction: (() -> Unit) = EMPTY_FUNC,
498
-
499
- val btnText: String? = null,
500
- val isShowBtnText: Boolean = !btnText.isNullOrEmpty(),
501
- val onPressBtnText: (() -> Unit) = EMPTY_FUNC,
502
-
503
- val disabled: Boolean = false,
504
-
505
- val placeHolder: String? = null,
506
-
507
- val iconRightTextField: @Composable ((Modifier) -> Unit)? = null,
508
- )
509
-
510
- @Composable
511
- private fun LiteInputSearch(
512
- modifier: Modifier = Modifier,
513
- inputSearchProps: LiteInputSearchProps? = null,
514
- ) {
515
- inputSearchProps ?: return
516
- val theme = AppTheme.current
517
-
518
- val isFocused = remember { mutableStateOf(false) }
519
- val interactionSource = remember { MutableInteractionSource() }
520
- val textState by remember { inputSearchProps.textFieldState }
521
-
522
- val isShowBtnText by remember(inputSearchProps.isShowBtnText, inputSearchProps.btnText) {
523
- derivedStateOf {
524
- inputSearchProps.isShowBtnText && !inputSearchProps.btnText.isNullOrEmpty()
525
- }
526
- }
527
- val inputFieldStyle = remember(theme) {
528
- Typography.bodyDefaultRegular.copy(
529
- color = theme.colors.text.default,
530
- )
531
- }
532
-
533
- LaunchedEffect(inputSearchProps.onFocused) {
534
- interactionSource.interactions.mapNotNull { it as? FocusInteraction }.collectLatest {
535
- val isFocus = it is FocusInteraction.Focus
536
- isFocused.value = isFocus
537
- if (isFocus) inputSearchProps.onFocused()
538
- }
539
- }
540
-
541
- Row(
542
- modifier = modifier,
543
- verticalAlignment = Alignment.CenterVertically,
544
- ) {
545
- val textFieldModifier = remember(inputSearchProps.modifier) {
546
- inputSearchProps.modifier
547
- .weight(1f)
548
- .sizeIn(minHeight = 36.dp)
549
- }
550
- BasicTextField(
551
- value = textState,
552
- onValueChange = inputSearchProps.onValueChange,
553
- enabled = !inputSearchProps.disabled,
554
- keyboardOptions = inputSearchProps.keyboardOptions,
555
- keyboardActions = inputSearchProps.keyboardActions,
556
- modifier = textFieldModifier,
557
- textStyle = inputFieldStyle,
558
- singleLine = true,
559
- interactionSource = interactionSource,
560
- decorationBox = { innerTextField ->
561
- val isShowClear by remember(inputSearchProps.clearCondition, isFocused.value) {
562
- derivedStateOf {
563
- inputSearchProps.clearCondition?.invoke() == true || (isFocused.value && textState.isNotEmpty())
564
- }
565
- }
566
- val placeHolder by produceState<String?>(null, inputSearchProps.placeHolder) {
567
- if (inputSearchProps.placeHolder.isNullOrEmpty()) return@produceState
568
- snapshotFlow { textState }.collectLatest {
569
- value = if (it.isEmpty()) inputSearchProps.placeHolder else null
570
- }
571
- }
572
- val iconRightModifier = remember {
573
- Modifier
574
- .padding(start = Spacing.L)
575
- .drawWithContent {
576
- val offsetX = -Spacing.S.toPx()
577
- drawLine(
578
- start = Offset(
579
- x = offsetX,
580
- y = 0f,
581
- ),
582
- end = Offset(
583
- x = offsetX,
584
- y = size.height,
585
- ),
586
- color = theme.colors.primary,
587
- strokeWidth = 1.dp.toPx(),
588
- )
589
- drawContent()
590
- }
591
- }
592
-
593
- Row(
594
- modifier = Modifier
595
- .background(
596
- color = theme.colors.background.surface,
597
- shape = RoundedCornerShape(Radius.XL),
598
- )
599
- .padding(
600
- horizontal = Spacing.M,
601
- vertical = Spacing.S,
602
- ),
603
- horizontalArrangement = Arrangement.Start,
604
- verticalAlignment = Alignment.CenterVertically,
605
- ) {
606
- Icon(
607
- source = "navigation_search",
608
- modifier = Modifier.padding(end = Spacing.XS),
609
- size = 24.dp,
610
- color = theme.colors.text.hint
611
- )
612
- Box(
613
- modifier = Modifier.weight(1f),
614
- contentAlignment = Alignment.CenterStart,
615
- ) {
616
- if (!placeHolder.isNullOrEmpty()) {
617
- Text(
618
- text = placeHolder ?: "",
619
- style = Typography.bodyDefaultRegular,
620
- maxLines = 1,
621
- color = theme.colors.text.hint,
622
- overflow = TextOverflow.Ellipsis
623
- )
624
- }
625
- innerTextField()
626
- }
627
-
628
- if (isShowClear) {
629
- Icon(
630
- source = "24_navigation_close_circle_full",
631
- size = 16.dp,
632
- color = theme.colors.text.hint,
633
- modifier = Modifier.padding(start = Spacing.XS)
634
- .noFeedbackClickable(onClick = inputSearchProps.onClear),
635
- )
636
- }
637
-
638
- inputSearchProps.iconRightTextField?.invoke(iconRightModifier)
639
- }
640
- }
641
- )
642
-
643
- if (isShowBtnText) {
644
- Text(
645
- text = inputSearchProps.btnText ?: "",
646
- style = Typography.actionDefaultBold,
647
- color = theme.colors.text.default,
648
- modifier = Modifier.padding(start = Spacing.L)
649
- .noFeedbackClickable(onClick = inputSearchProps.onPressBtnText)
650
- )
651
- }
652
- }
653
- }
654
-
655
- @Composable
656
- private fun footerOffset(
657
- useAvoidKeyboard: Boolean = true,
658
- ): State<IntOffset> {
659
- if (!useAvoidKeyboard) return remember { mutableStateOf(IntOffset.Zero) }
660
- val density = LocalDensity.current
661
- val navPaddingValue = WindowInsets.navigationBars.asPaddingValues()
662
- val keyboardSize = keyboardSizeState()
663
-
664
- return produceState(IntOffset.Zero, density, navPaddingValue) {
665
- val navSystemOffset = with(density) { navPaddingValue.calculateBottomPadding().roundToPx() }
666
- snapshotFlow { keyboardSize.value }.collectLatest {
667
- value = if (it == 0.dp) IntOffset.Zero
668
- else IntOffset(
669
- x = 0, y = with(density) {
670
- navSystemOffset - it.roundToPx()
671
- })
672
- }
673
- }
674
- }
675
-
676
- @Composable
677
- fun keyboardSizeState(): State<Dp> {
678
- val bottom = WindowInsets.ime.asPaddingValues()
679
- return rememberUpdatedState(bottom.calculateBottomPadding())
680
- }
681
-
682
- @Composable
683
- private fun Footer(
684
- useAvoidKeyboard: Boolean,
685
- footerContent: (@Composable () -> Unit)?,
686
- ) {
687
- footerContent ?: return
688
- val theme = AppTheme.current
689
- val navPaddingValue = WindowInsets.navigationBars.asPaddingValues()
690
- val offsetMove = footerOffset(useAvoidKeyboard)
691
-
692
- Box(
693
- modifier = Modifier.padding(navPaddingValue).fillMaxWidth().offset {
694
- offsetMove.value
695
- }.shadow(
696
- color = Colors.black_20.copy(alpha = 0.05f),
697
- blurRadius = 24f,
698
- offsetX = 0.dp,
699
- offsetY = (-4).dp
700
- ).background(theme.colors.background.surface).padding(
701
- start = Spacing.M,
702
- top = Spacing.M,
703
- end = Spacing.M,
704
- )
705
- ) {
706
- footerContent()
707
- }
708
- }
709
-
710
- fun Modifier.hideKeyboardOnTap() = composed {
711
- val focusManager = LocalFocusManager.current
712
- val keyboardManager = LocalSoftwareKeyboardController.current
713
-
714
- pointerInput(Unit) {
715
- detectTapGestures {
716
- keyboardManager?.hide()
717
- focusManager.clearFocus()
718
- }
719
- }
720
- }