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