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

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (131) hide show
  1. package/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 -117
  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/Avatar.kt +0 -157
  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 -357
  41. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Carousel.kt +0 -123
  42. package/compose/src/commonMain/kotlin/vn/momo/kits/components/CheckBox.kt +0 -94
  43. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Chip.kt +0 -136
  44. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Collapse.kt +0 -224
  45. package/compose/src/commonMain/kotlin/vn/momo/kits/components/CupertinoOverscroll.kt +0 -543
  46. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Divider.kt +0 -23
  47. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Icon.kt +0 -76
  48. package/compose/src/commonMain/kotlin/vn/momo/kits/components/IconButton.kt +0 -148
  49. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Image.kt +0 -188
  50. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Information.kt +0 -116
  51. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Input.kt +0 -448
  52. package/compose/src/commonMain/kotlin/vn/momo/kits/components/InputDropDown.kt +0 -172
  53. package/compose/src/commonMain/kotlin/vn/momo/kits/components/InputMoney.kt +0 -255
  54. package/compose/src/commonMain/kotlin/vn/momo/kits/components/InputOTP.kt +0 -231
  55. package/compose/src/commonMain/kotlin/vn/momo/kits/components/InputPhoneNumber.kt +0 -233
  56. package/compose/src/commonMain/kotlin/vn/momo/kits/components/InputSearch.kt +0 -254
  57. package/compose/src/commonMain/kotlin/vn/momo/kits/components/InputTextArea.kt +0 -241
  58. package/compose/src/commonMain/kotlin/vn/momo/kits/components/LazyColumnWithBouncing.kt +0 -364
  59. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Loader.kt +0 -108
  60. package/compose/src/commonMain/kotlin/vn/momo/kits/components/PaginationDot.kt +0 -56
  61. package/compose/src/commonMain/kotlin/vn/momo/kits/components/PaginationNumber.kt +0 -41
  62. package/compose/src/commonMain/kotlin/vn/momo/kits/components/PaginationScroll.kt +0 -92
  63. package/compose/src/commonMain/kotlin/vn/momo/kits/components/PaginationWhiteDot.kt +0 -40
  64. package/compose/src/commonMain/kotlin/vn/momo/kits/components/PopupNotify.kt +0 -352
  65. package/compose/src/commonMain/kotlin/vn/momo/kits/components/PopupPromotion.kt +0 -103
  66. package/compose/src/commonMain/kotlin/vn/momo/kits/components/ProgressInfo.kt +0 -338
  67. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Radio.kt +0 -70
  68. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Rating.kt +0 -87
  69. package/compose/src/commonMain/kotlin/vn/momo/kits/components/ScaleSizeScope.kt +0 -17
  70. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Skeleton.kt +0 -96
  71. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Slider.kt +0 -348
  72. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Stepper.kt +0 -256
  73. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Steps.kt +0 -494
  74. package/compose/src/commonMain/kotlin/vn/momo/kits/components/SuggestAction.kt +0 -131
  75. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Swipe.kt +0 -215
  76. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Switch.kt +0 -96
  77. package/compose/src/commonMain/kotlin/vn/momo/kits/components/TabView.kt +0 -531
  78. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Tag.kt +0 -92
  79. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Text.kt +0 -130
  80. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Title.kt +0 -214
  81. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Tooltip.kt +0 -590
  82. package/compose/src/commonMain/kotlin/vn/momo/kits/components/TrustBanner.kt +0 -177
  83. package/compose/src/commonMain/kotlin/vn/momo/kits/components/Uploader.kt +0 -192
  84. package/compose/src/commonMain/kotlin/vn/momo/kits/components/datetimepicker/DateTimePicker.kt +0 -205
  85. package/compose/src/commonMain/kotlin/vn/momo/kits/components/datetimepicker/DateTimePickerTypes.kt +0 -29
  86. package/compose/src/commonMain/kotlin/vn/momo/kits/components/datetimepicker/DateTimePickerUtils.kt +0 -239
  87. package/compose/src/commonMain/kotlin/vn/momo/kits/components/datetimepicker/WheelPicker.kt +0 -191
  88. package/compose/src/commonMain/kotlin/vn/momo/kits/const/Colors.kt +0 -306
  89. package/compose/src/commonMain/kotlin/vn/momo/kits/const/Radius.kt +0 -12
  90. package/compose/src/commonMain/kotlin/vn/momo/kits/const/Spacing.kt +0 -16
  91. package/compose/src/commonMain/kotlin/vn/momo/kits/const/Theme.kt +0 -188
  92. package/compose/src/commonMain/kotlin/vn/momo/kits/const/Typography.kt +0 -285
  93. package/compose/src/commonMain/kotlin/vn/momo/kits/layout/Card.kt +0 -2
  94. package/compose/src/commonMain/kotlin/vn/momo/kits/layout/Item.kt +0 -35
  95. package/compose/src/commonMain/kotlin/vn/momo/kits/layout/Section.kt +0 -2
  96. package/compose/src/commonMain/kotlin/vn/momo/kits/modifier/AutomationId.kt +0 -50
  97. package/compose/src/commonMain/kotlin/vn/momo/kits/modifier/Clickable.kt +0 -68
  98. package/compose/src/commonMain/kotlin/vn/momo/kits/modifier/Conditional.kt +0 -11
  99. package/compose/src/commonMain/kotlin/vn/momo/kits/modifier/DeprecatedModifier.kt +0 -14
  100. package/compose/src/commonMain/kotlin/vn/momo/kits/modifier/Shadow.kt +0 -50
  101. package/compose/src/commonMain/kotlin/vn/momo/kits/modifier/Size.kt +0 -51
  102. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/BottomSheet.kt +0 -253
  103. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/ModalScreen.kt +0 -133
  104. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/Navigation.kt +0 -99
  105. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/NavigationContainer.kt +0 -164
  106. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/Navigator.kt +0 -333
  107. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/StackScreen.kt +0 -552
  108. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/bottomtab/BottomTab.kt +0 -162
  109. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/bottomtab/BottomTabBar.kt +0 -243
  110. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/bottomtab/CurvedContainer.kt +0 -86
  111. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/FloatingButton.kt +0 -187
  112. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/Header.kt +0 -279
  113. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/HeaderBackground.kt +0 -80
  114. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/HeaderRight.kt +0 -306
  115. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/HeaderTitle.kt +0 -32
  116. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/HeaderUser.kt +0 -370
  117. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/component/SnackBar.kt +0 -132
  118. package/compose/src/commonMain/kotlin/vn/momo/kits/navigation/tracking/ScreenTracker.kt +0 -167
  119. package/compose/src/commonMain/kotlin/vn/momo/kits/platform/Platform.kt +0 -46
  120. package/compose/src/commonMain/kotlin/vn/momo/kits/utils/Icons.kt +0 -1329
  121. package/compose/src/commonMain/kotlin/vn/momo/kits/utils/Resources.kt +0 -62
  122. package/compose/src/commonMain/kotlin/vn/momo/kits/utils/Tracking.kt +0 -15
  123. package/compose/src/commonMain/kotlin/vn/momo/kits/utils/Utils.kt +0 -88
  124. package/compose/src/iosMain/kotlin/vn/momo/kits/platform/Platform.ios.kt +0 -161
  125. package/gradle/libs.versions.toml +0 -57
  126. package/gradle/wrapper/gradle-wrapper.jar +0 -0
  127. package/gradle/wrapper/gradle-wrapper.properties +0 -8
  128. package/gradle.properties +0 -26
  129. package/gradlew +0 -252
  130. package/gradlew.bat +0 -94
  131. 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
- }