@momo-kits/native-kits 0.157.1-debug → 0.157.1-sp.1

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 (118) hide show
  1. package/ios/Application/HeaderRight.swift +1 -1
  2. package/package.json +1 -1
  3. package/build.gradle.kts +0 -11
  4. package/compose/build.gradle.kts +0 -180
  5. package/compose/build.gradle.kts.backup +0 -180
  6. package/compose/compose.podspec +0 -54
  7. package/compose/src/androidMain/kotlin/vn/momo/kits/platform/Platform.android.kt +0 -110
  8. package/compose/src/commonMain/composeResources/font/momosignature.otf +0 -0
  9. package/compose/src/commonMain/composeResources/font/momotrustdisplay.otf +0 -0
  10. package/compose/src/commonMain/composeResources/font/sfprotext_black.otf +0 -0
  11. package/compose/src/commonMain/composeResources/font/sfprotext_black.ttf +0 -0
  12. package/compose/src/commonMain/composeResources/font/sfprotext_bold.ttf +0 -0
  13. package/compose/src/commonMain/composeResources/font/sfprotext_heavy.ttf +0 -0
  14. package/compose/src/commonMain/composeResources/font/sfprotext_light.ttf +0 -0
  15. package/compose/src/commonMain/composeResources/font/sfprotext_medium.ttf +0 -0
  16. package/compose/src/commonMain/composeResources/font/sfprotext_regular.ttf +0 -0
  17. package/compose/src/commonMain/composeResources/font/sfprotext_semibold.ttf +0 -0
  18. package/compose/src/commonMain/composeResources/font/sfprotext_thin.otf +0 -0
  19. package/compose/src/commonMain/composeResources/font/sfprotext_thin.ttf +0 -0
  20. package/compose/src/commonMain/composeResources/font/sfprotext_ultralight.otf +0 -0
  21. package/compose/src/commonMain/composeResources/font/sfprotext_ultralight.ttf +0 -0
  22. package/compose/src/commonMain/kotlin/vn/momo/kits/application/AnimationSearchInput.kt +0 -57
  23. package/compose/src/commonMain/kotlin/vn/momo/kits/application/Context.kt +0 -107
  24. package/compose/src/commonMain/kotlin/vn/momo/kits/application/FloatingButton.kt +0 -201
  25. package/compose/src/commonMain/kotlin/vn/momo/kits/application/Header.kt +0 -222
  26. package/compose/src/commonMain/kotlin/vn/momo/kits/application/HeaderAnimated.kt +0 -48
  27. package/compose/src/commonMain/kotlin/vn/momo/kits/application/HeaderBackground.kt +0 -86
  28. package/compose/src/commonMain/kotlin/vn/momo/kits/application/HeaderDefault.kt +0 -76
  29. package/compose/src/commonMain/kotlin/vn/momo/kits/application/HeaderExtended.kt +0 -76
  30. package/compose/src/commonMain/kotlin/vn/momo/kits/application/HeaderRight.kt +0 -305
  31. package/compose/src/commonMain/kotlin/vn/momo/kits/application/HeaderTitle.kt +0 -33
  32. package/compose/src/commonMain/kotlin/vn/momo/kits/application/LiteScreen.kt +0 -720
  33. package/compose/src/commonMain/kotlin/vn/momo/kits/application/NavigationContainer.kt +0 -121
  34. package/compose/src/commonMain/kotlin/vn/momo/kits/application/Screen.kt +0 -405
  35. package/compose/src/commonMain/kotlin/vn/momo/kits/application/useHeaderSearchAnimation.kt +0 -69
  36. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Badge.kt +0 -85
  37. package/compose/src/commonMain/kotlin/vn/momo/kits/components/BadgeDot.kt +0 -32
  38. package/compose/src/commonMain/kotlin/vn/momo/kits/components/BadgeRibbon.kt +0 -340
  39. package/compose/src/commonMain/kotlin/vn/momo/kits/components/BaselineView.kt +0 -198
  40. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Button.kt +0 -339
  41. package/compose/src/commonMain/kotlin/vn/momo/kits/components/CheckBox.kt +0 -94
  42. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Chip.kt +0 -136
  43. package/compose/src/commonMain/kotlin/vn/momo/kits/components/CupertinoOverscroll.kt +0 -543
  44. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Divider.kt +0 -23
  45. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Icon.kt +0 -76
  46. package/compose/src/commonMain/kotlin/vn/momo/kits/components/IconButton.kt +0 -148
  47. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Image.kt +0 -188
  48. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Information.kt +0 -116
  49. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Input.kt +0 -448
  50. package/compose/src/commonMain/kotlin/vn/momo/kits/components/InputDropDown.kt +0 -172
  51. package/compose/src/commonMain/kotlin/vn/momo/kits/components/InputMoney.kt +0 -255
  52. package/compose/src/commonMain/kotlin/vn/momo/kits/components/InputOTP.kt +0 -231
  53. package/compose/src/commonMain/kotlin/vn/momo/kits/components/InputPhoneNumber.kt +0 -233
  54. package/compose/src/commonMain/kotlin/vn/momo/kits/components/InputSearch.kt +0 -254
  55. package/compose/src/commonMain/kotlin/vn/momo/kits/components/InputTextArea.kt +0 -241
  56. package/compose/src/commonMain/kotlin/vn/momo/kits/components/LazyColumnWithBouncing.kt +0 -364
  57. package/compose/src/commonMain/kotlin/vn/momo/kits/components/PaginationDot.kt +0 -56
  58. package/compose/src/commonMain/kotlin/vn/momo/kits/components/PaginationNumber.kt +0 -41
  59. package/compose/src/commonMain/kotlin/vn/momo/kits/components/PaginationScroll.kt +0 -92
  60. package/compose/src/commonMain/kotlin/vn/momo/kits/components/PaginationWhiteDot.kt +0 -40
  61. package/compose/src/commonMain/kotlin/vn/momo/kits/components/PopupNotify.kt +0 -352
  62. package/compose/src/commonMain/kotlin/vn/momo/kits/components/PopupPromotion.kt +0 -103
  63. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Radio.kt +0 -70
  64. package/compose/src/commonMain/kotlin/vn/momo/kits/components/ScaleSizeScope.kt +0 -17
  65. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Skeleton.kt +0 -96
  66. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Switch.kt +0 -96
  67. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Tag.kt +0 -92
  68. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Text.kt +0 -130
  69. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Title.kt +0 -214
  70. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Tooltip.kt +0 -576
  71. package/compose/src/commonMain/kotlin/vn/momo/kits/components/TrustBanner.kt +0 -177
  72. package/compose/src/commonMain/kotlin/vn/momo/kits/components/datetimepicker/DateTimePicker.kt +0 -205
  73. package/compose/src/commonMain/kotlin/vn/momo/kits/components/datetimepicker/DateTimePickerTypes.kt +0 -29
  74. package/compose/src/commonMain/kotlin/vn/momo/kits/components/datetimepicker/DateTimePickerUtils.kt +0 -239
  75. package/compose/src/commonMain/kotlin/vn/momo/kits/components/datetimepicker/WheelPicker.kt +0 -191
  76. package/compose/src/commonMain/kotlin/vn/momo/kits/const/Colors.kt +0 -306
  77. package/compose/src/commonMain/kotlin/vn/momo/kits/const/Radius.kt +0 -12
  78. package/compose/src/commonMain/kotlin/vn/momo/kits/const/Spacing.kt +0 -13
  79. package/compose/src/commonMain/kotlin/vn/momo/kits/const/Theme.kt +0 -185
  80. package/compose/src/commonMain/kotlin/vn/momo/kits/const/Typography.kt +0 -285
  81. package/compose/src/commonMain/kotlin/vn/momo/kits/layout/Card.kt +0 -2
  82. package/compose/src/commonMain/kotlin/vn/momo/kits/layout/Item.kt +0 -35
  83. package/compose/src/commonMain/kotlin/vn/momo/kits/layout/Section.kt +0 -2
  84. package/compose/src/commonMain/kotlin/vn/momo/kits/modifier/AutomationId.kt +0 -59
  85. package/compose/src/commonMain/kotlin/vn/momo/kits/modifier/Clickable.kt +0 -68
  86. package/compose/src/commonMain/kotlin/vn/momo/kits/modifier/Conditional.kt +0 -11
  87. package/compose/src/commonMain/kotlin/vn/momo/kits/modifier/DeprecatedModifier.kt +0 -14
  88. package/compose/src/commonMain/kotlin/vn/momo/kits/modifier/Shadow.kt +0 -50
  89. package/compose/src/commonMain/kotlin/vn/momo/kits/modifier/Size.kt +0 -51
  90. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/BottomSheet.kt +0 -239
  91. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/ModalScreen.kt +0 -119
  92. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/Navigation.kt +0 -98
  93. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/NavigationContainer.kt +0 -161
  94. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/Navigator.kt +0 -331
  95. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/StackScreen.kt +0 -497
  96. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/bottomtab/BottomTab.kt +0 -162
  97. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/bottomtab/BottomTabBar.kt +0 -243
  98. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/bottomtab/CurvedContainer.kt +0 -86
  99. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/FloatingButton.kt +0 -187
  100. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/Header.kt +0 -279
  101. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/HeaderBackground.kt +0 -80
  102. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/HeaderRight.kt +0 -306
  103. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/HeaderTitle.kt +0 -32
  104. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/HeaderUser.kt +0 -370
  105. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/SnackBar.kt +0 -132
  106. package/compose/src/commonMain/kotlin/vn/momo/kits/platform/Platform.kt +0 -42
  107. package/compose/src/commonMain/kotlin/vn/momo/kits/utils/Icons.kt +0 -1329
  108. package/compose/src/commonMain/kotlin/vn/momo/kits/utils/Resources.kt +0 -62
  109. package/compose/src/commonMain/kotlin/vn/momo/kits/utils/Tracking.kt +0 -15
  110. package/compose/src/commonMain/kotlin/vn/momo/kits/utils/Utils.kt +0 -88
  111. package/compose/src/iosMain/kotlin/vn/momo/kits/platform/Platform.ios.kt +0 -149
  112. package/gradle/libs.versions.toml +0 -57
  113. package/gradle/wrapper/gradle-wrapper.jar +0 -0
  114. package/gradle/wrapper/gradle-wrapper.properties +0 -8
  115. package/gradle.properties +0 -26
  116. package/gradlew +0 -252
  117. package/gradlew.bat +0 -94
  118. 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
- }