@momo-kits/native-kits 0.156.4-debug → 0.156.5-debug
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.
- package/compose/build.gradle.kts +1 -1
- package/compose/src/commonMain/kotlin/vn/momo/kits/components/PopupPromotion.kt +1 -1
- package/gradle.properties +1 -1
- package/ios/Button/Button.swift +6 -7
- package/ios/Input/Input.swift +73 -34
- package/ios/Input/InputPhoneNumber.swift +16 -15
- package/package.json +1 -1
package/compose/build.gradle.kts
CHANGED
|
@@ -50,7 +50,7 @@ fun PopupPromotion(
|
|
|
50
50
|
indication = null,
|
|
51
51
|
onClick = onPress
|
|
52
52
|
),
|
|
53
|
-
options = Options(alignment = Alignment.Center, contentScale = ContentScale.
|
|
53
|
+
options = Options(alignment = Alignment.Center, contentScale = ContentScale.Fit)
|
|
54
54
|
), null
|
|
55
55
|
)
|
|
56
56
|
Box(
|
package/gradle.properties
CHANGED
package/ios/Button/Button.swift
CHANGED
|
@@ -106,11 +106,12 @@ extension ButtonSize {
|
|
|
106
106
|
}
|
|
107
107
|
}
|
|
108
108
|
|
|
109
|
-
var
|
|
109
|
+
var typographyStyle: TypographyStyle {
|
|
110
110
|
switch self {
|
|
111
|
-
case .large: return .
|
|
112
|
-
case .medium: return .
|
|
113
|
-
case .small: return .
|
|
111
|
+
case .large: return .actionDefaultBold
|
|
112
|
+
case .medium: return .actionSBold
|
|
113
|
+
case .small: return .actionXsBold
|
|
114
|
+
default : return .actionDefaultBold
|
|
114
115
|
}
|
|
115
116
|
}
|
|
116
117
|
}
|
|
@@ -180,9 +181,7 @@ public struct Button: View {
|
|
|
180
181
|
iconLeft.frame(width: size.iconSize, height: size.iconSize)
|
|
181
182
|
}
|
|
182
183
|
|
|
183
|
-
|
|
184
|
-
.font(size.font)
|
|
185
|
-
.foregroundColor(fg)
|
|
184
|
+
MomoText(title, typography: size.typographyStyle, color: fg)
|
|
186
185
|
.lineLimit(1)
|
|
187
186
|
.truncationMode(.tail)
|
|
188
187
|
|
package/ios/Input/Input.swift
CHANGED
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
import Foundation
|
|
2
2
|
import SwiftUI
|
|
3
|
+
import Combine
|
|
3
4
|
|
|
4
5
|
public struct Input: View {
|
|
5
6
|
@Binding public var text: String
|
|
@@ -31,7 +32,7 @@ public struct Input: View {
|
|
|
31
32
|
|
|
32
33
|
@State private var isFocused: Bool = false
|
|
33
34
|
@State private var isPasswordHidden: Bool = true
|
|
34
|
-
@
|
|
35
|
+
@State private var lastTextValue: String = ""
|
|
35
36
|
|
|
36
37
|
public init(
|
|
37
38
|
text: Binding<String>,
|
|
@@ -89,6 +90,22 @@ public struct Input: View {
|
|
|
89
90
|
|
|
90
91
|
// MARK: - Body
|
|
91
92
|
public var body: some View {
|
|
93
|
+
let textBinding = Binding<String>(
|
|
94
|
+
get: { self.text },
|
|
95
|
+
set: { newValue in
|
|
96
|
+
// For SecureField, infer focus when text changes
|
|
97
|
+
if self.secureTextEntry && !self.isFocused && newValue != self.lastTextValue {
|
|
98
|
+
DispatchQueue.main.async {
|
|
99
|
+
self.isFocused = true
|
|
100
|
+
self.onFocus?()
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
self.lastTextValue = newValue
|
|
104
|
+
self.text = newValue
|
|
105
|
+
self.onChangeText?(newValue)
|
|
106
|
+
}
|
|
107
|
+
)
|
|
108
|
+
|
|
92
109
|
VStack(alignment: .leading, spacing: 4) {
|
|
93
110
|
ZStack(alignment: .topLeading) {
|
|
94
111
|
// Floating label
|
|
@@ -127,34 +144,35 @@ public struct Input: View {
|
|
|
127
144
|
}
|
|
128
145
|
|
|
129
146
|
if secureTextEntry && isPasswordHidden {
|
|
130
|
-
SecureField("", text:
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
147
|
+
SecureField("", text: textBinding, onCommit: {
|
|
148
|
+
handleSecureFieldBlur()
|
|
149
|
+
})
|
|
150
|
+
.keyboardType(keyboardType)
|
|
151
|
+
.font(fontWeight == .bold ? .action_s_bold : .body_default_regular)
|
|
152
|
+
.foregroundColor(getTextColor())
|
|
153
|
+
.disabled(disabled || readOnly)
|
|
154
|
+
.applyPrimaryCursorColor()
|
|
155
|
+
.simultaneousGesture(TapGesture().onEnded {
|
|
156
|
+
handleSecureFieldFocus()
|
|
157
|
+
})
|
|
139
158
|
} else {
|
|
140
|
-
TextField("", text:
|
|
159
|
+
TextField("", text: textBinding, onEditingChanged: { focused in
|
|
141
160
|
handleFocusChange(focused)
|
|
142
161
|
})
|
|
143
|
-
.keyboardType(
|
|
162
|
+
.keyboardType(keyboardType)
|
|
144
163
|
.font(fontWeight == .bold ? .action_s_bold : .body_default_regular)
|
|
145
164
|
.foregroundColor(getTextColor())
|
|
146
165
|
.disabled(disabled || readOnly)
|
|
147
166
|
.applyPrimaryCursorColor()
|
|
148
|
-
.focused($isTextFieldFocused)
|
|
149
|
-
.onChange(of: text) { newValue in
|
|
150
|
-
onChangeText?(newValue)
|
|
151
|
-
}
|
|
152
167
|
}
|
|
153
168
|
}
|
|
154
169
|
|
|
155
170
|
// Clear button (only show when focused and has text)
|
|
156
171
|
if isFocused && !text.isEmpty {
|
|
157
|
-
SwiftUI.Button(action: {
|
|
172
|
+
SwiftUI.Button(action: {
|
|
173
|
+
text = ""
|
|
174
|
+
onChangeText?("")
|
|
175
|
+
}) {
|
|
158
176
|
Icon(source: "24_navigation_close_circle_full", size: 16, color: Colors.black12)
|
|
159
177
|
.padding(.leading, Spacing.S)
|
|
160
178
|
}
|
|
@@ -169,13 +187,15 @@ public struct Input: View {
|
|
|
169
187
|
|
|
170
188
|
// Right icon (password toggle or custom icon)
|
|
171
189
|
if secureTextEntry {
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
|
|
190
|
+
if !text.isEmpty {
|
|
191
|
+
SwiftUI.Button(action: togglePasswordVisibility) {
|
|
192
|
+
Icon(
|
|
193
|
+
source: isPasswordHidden ? "24_security_eye_off" : "24_security_eye_open",
|
|
194
|
+
size: 24,
|
|
195
|
+
color: rightIconColor
|
|
196
|
+
)
|
|
197
|
+
.padding(.leading, Spacing.S)
|
|
198
|
+
}
|
|
179
199
|
}
|
|
180
200
|
} else if !rightIcon.isEmpty {
|
|
181
201
|
SwiftUI.Button(action: { onRightIconPressed?() }) {
|
|
@@ -204,18 +224,26 @@ public struct Input: View {
|
|
|
204
224
|
)
|
|
205
225
|
}
|
|
206
226
|
.onAppear {
|
|
207
|
-
|
|
208
|
-
|
|
209
|
-
|
|
227
|
+
lastTextValue = text
|
|
228
|
+
}
|
|
229
|
+
.onReceive(NotificationCenter.default.publisher(for: UIResponder.keyboardWillHideNotification)) { _ in
|
|
230
|
+
// Reset focus when keyboard is dismissed
|
|
231
|
+
if isFocused {
|
|
232
|
+
DispatchQueue.main.async {
|
|
233
|
+
if secureTextEntry {
|
|
234
|
+
handleSecureFieldBlur()
|
|
235
|
+
} else {
|
|
236
|
+
handleFocusChange(false)
|
|
237
|
+
}
|
|
210
238
|
}
|
|
211
239
|
}
|
|
212
240
|
}
|
|
213
|
-
.
|
|
214
|
-
|
|
215
|
-
if
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
|
|
241
|
+
.onReceive(NotificationCenter.default.publisher(for: UITextField.textDidEndEditingNotification)) { _ in
|
|
242
|
+
// Reset focus when any text field ends editing
|
|
243
|
+
if isFocused && secureTextEntry {
|
|
244
|
+
DispatchQueue.main.async {
|
|
245
|
+
self.handleSecureFieldBlur()
|
|
246
|
+
}
|
|
219
247
|
}
|
|
220
248
|
}
|
|
221
249
|
}
|
|
@@ -224,7 +252,6 @@ public struct Input: View {
|
|
|
224
252
|
|
|
225
253
|
private func handleFocusChange(_ focused: Bool) {
|
|
226
254
|
isFocused = focused
|
|
227
|
-
isTextFieldFocused = focused
|
|
228
255
|
if focused {
|
|
229
256
|
onFocus?()
|
|
230
257
|
} else {
|
|
@@ -232,6 +259,18 @@ public struct Input: View {
|
|
|
232
259
|
}
|
|
233
260
|
}
|
|
234
261
|
|
|
262
|
+
private func handleSecureFieldFocus() {
|
|
263
|
+
if !isFocused {
|
|
264
|
+
isFocused = true
|
|
265
|
+
onFocus?()
|
|
266
|
+
}
|
|
267
|
+
}
|
|
268
|
+
|
|
269
|
+
private func handleSecureFieldBlur() {
|
|
270
|
+
isFocused = false
|
|
271
|
+
onBlur?()
|
|
272
|
+
}
|
|
273
|
+
|
|
235
274
|
private func togglePasswordVisibility() {
|
|
236
275
|
isPasswordHidden.toggle()
|
|
237
276
|
onRightIconPressed?()
|
|
@@ -6,7 +6,6 @@
|
|
|
6
6
|
//
|
|
7
7
|
|
|
8
8
|
import SwiftUI
|
|
9
|
-
|
|
10
9
|
public struct InputPhoneNumber: View {
|
|
11
10
|
@Binding public var text: String
|
|
12
11
|
|
|
@@ -18,12 +17,12 @@ public struct InputPhoneNumber: View {
|
|
|
18
17
|
public var rightIcon: String
|
|
19
18
|
public var rightIconColor: Color
|
|
20
19
|
public var autofocus: Bool
|
|
20
|
+
public var onChangeText: ((String) -> Void)?
|
|
21
21
|
public var onFocus: (() -> Void)?
|
|
22
22
|
public var onBlur: (() -> Void)?
|
|
23
23
|
public var onRightIconPressed: (() -> Void)?
|
|
24
24
|
|
|
25
25
|
@State private var isFocused: Bool = false
|
|
26
|
-
@FocusState private var isTextFieldFocused: Bool
|
|
27
26
|
|
|
28
27
|
public init(
|
|
29
28
|
text: Binding<String>,
|
|
@@ -36,6 +35,7 @@ public struct InputPhoneNumber: View {
|
|
|
36
35
|
rightIcon: String = "",
|
|
37
36
|
rightIconColor: Color = Colors.black12,
|
|
38
37
|
autofocus: Bool = false,
|
|
38
|
+
onChangeText: ((String) -> Void)? = nil,
|
|
39
39
|
onFocus: (() -> Void)? = nil,
|
|
40
40
|
onBlur: (() -> Void)? = nil,
|
|
41
41
|
onRightIconPressed: (() -> Void)? = nil
|
|
@@ -49,6 +49,7 @@ public struct InputPhoneNumber: View {
|
|
|
49
49
|
self.rightIcon = rightIcon
|
|
50
50
|
self.rightIconColor = rightIconColor
|
|
51
51
|
self.autofocus = autofocus
|
|
52
|
+
self.onChangeText = onChangeText
|
|
52
53
|
self.onFocus = onFocus
|
|
53
54
|
self.onBlur = onBlur
|
|
54
55
|
self.onRightIconPressed = onRightIconPressed
|
|
@@ -56,6 +57,14 @@ public struct InputPhoneNumber: View {
|
|
|
56
57
|
|
|
57
58
|
// MARK: - Body
|
|
58
59
|
public var body: some View {
|
|
60
|
+
let textBinding = Binding<String>(
|
|
61
|
+
get: { self.text },
|
|
62
|
+
set: { newValue in
|
|
63
|
+
self.text = newValue
|
|
64
|
+
self.onChangeText?(newValue)
|
|
65
|
+
}
|
|
66
|
+
)
|
|
67
|
+
|
|
59
68
|
VStack(alignment: .leading, spacing: 4) {
|
|
60
69
|
HStack(spacing: 0) {
|
|
61
70
|
// 🇻🇳 Flag
|
|
@@ -77,7 +86,7 @@ public struct InputPhoneNumber: View {
|
|
|
77
86
|
MomoText(placeholder, typography: size == .small ? .headerSSemibold : .headerMBold, color: Colors.black12)
|
|
78
87
|
}
|
|
79
88
|
|
|
80
|
-
TextField("", text:
|
|
89
|
+
TextField("", text: textBinding, onEditingChanged: { focused in
|
|
81
90
|
handleFocusChange(focused)
|
|
82
91
|
})
|
|
83
92
|
.keyboardType(.numberPad)
|
|
@@ -85,13 +94,14 @@ public struct InputPhoneNumber: View {
|
|
|
85
94
|
.foregroundColor(Colors.black17)
|
|
86
95
|
.applyPrimaryCursorColor()
|
|
87
96
|
.lineLimit(1)
|
|
88
|
-
.focused($isTextFieldFocused)
|
|
89
|
-
|
|
90
97
|
}
|
|
91
98
|
|
|
92
99
|
// Clear button
|
|
93
100
|
if isFocused && !text.isEmpty {
|
|
94
|
-
SwiftUI.Button(action: {
|
|
101
|
+
SwiftUI.Button(action: {
|
|
102
|
+
text = ""
|
|
103
|
+
onChangeText?("")
|
|
104
|
+
}) {
|
|
95
105
|
Icon(source: "24_navigation_close_circle_full", size: 16, color: Colors.black12)
|
|
96
106
|
.padding(.leading, Spacing.S)
|
|
97
107
|
}
|
|
@@ -129,21 +139,12 @@ public struct InputPhoneNumber: View {
|
|
|
129
139
|
hintText: hintText
|
|
130
140
|
)
|
|
131
141
|
}
|
|
132
|
-
.onAppear {
|
|
133
|
-
if autofocus {
|
|
134
|
-
// Small delay to ensure the view is fully loaded
|
|
135
|
-
DispatchQueue.main.asyncAfter(deadline: .now() + 0.1) {
|
|
136
|
-
isTextFieldFocused = true
|
|
137
|
-
}
|
|
138
|
-
}
|
|
139
|
-
}
|
|
140
142
|
}
|
|
141
143
|
|
|
142
144
|
// MARK: - Helpers
|
|
143
145
|
|
|
144
146
|
private func handleFocusChange(_ focused: Bool) {
|
|
145
147
|
isFocused = focused
|
|
146
|
-
isTextFieldFocused = focused
|
|
147
148
|
if focused {
|
|
148
149
|
onFocus?()
|
|
149
150
|
} else {
|