@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.
@@ -40,7 +40,7 @@ kotlin {
40
40
  }
41
41
 
42
42
  cocoapods {
43
- version = "0.156.4-debug"
43
+ version = "0.156.5-debug"
44
44
  summary = "IOS Shared module"
45
45
  homepage = "https://momo.vn"
46
46
  ios.deploymentTarget = "15.0"
@@ -50,7 +50,7 @@ fun PopupPromotion(
50
50
  indication = null,
51
51
  onClick = onPress
52
52
  ),
53
- options = Options(alignment = Alignment.Center, contentScale = ContentScale.Crop)
53
+ options = Options(alignment = Alignment.Center, contentScale = ContentScale.Fit)
54
54
  ), null
55
55
  )
56
56
  Box(
package/gradle.properties CHANGED
@@ -18,7 +18,7 @@ kotlin.apple.xcodeCompatibility.nowarn=true
18
18
  name="ComposeKits"
19
19
  group=vn.momo.kits
20
20
  artifact.id=kits
21
- version=0.156.4
21
+ version=0.156.5
22
22
 
23
23
  repo=GitLab
24
24
  url=https://gitlab.mservice.com.vn/api/v4/projects/5400/packages/maven
@@ -106,11 +106,12 @@ extension ButtonSize {
106
106
  }
107
107
  }
108
108
 
109
- var font: Font {
109
+ var typographyStyle: TypographyStyle {
110
110
  switch self {
111
- case .large: return .system(size: 16, weight: .bold)
112
- case .medium: return .system(size: 14, weight: .bold)
113
- case .small: return .system(size: 12, weight: .bold)
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
- Text(title)
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
 
@@ -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
- @FocusState private var isTextFieldFocused: Bool
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: $text, onCommit: {})
131
- .font(fontWeight == .bold ? .action_s_bold : .body_default_regular)
132
- .foregroundColor(getTextColor())
133
- .disabled(disabled || readOnly)
134
- .applyPrimaryCursorColor()
135
- .focused($isTextFieldFocused)
136
- .onChange(of: text) { newValue in
137
- onChangeText?(newValue)
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: $text, onEditingChanged: { focused in
159
+ TextField("", text: textBinding, onEditingChanged: { focused in
141
160
  handleFocusChange(focused)
142
161
  })
143
- .keyboardType(secureTextEntry ? .default : 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: { text = "" }) {
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
- SwiftUI.Button(action: togglePasswordVisibility) {
173
- Icon(
174
- source: isPasswordHidden ? "24_security_eye_off" : "24_security_eye_open",
175
- size: 24,
176
- color: rightIconColor
177
- )
178
- .padding(.leading, Spacing.S)
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
- if autofocus {
208
- DispatchQueue.main.asyncAfter(deadline: .now() + 0.1) {
209
- isTextFieldFocused = true
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
- .onChange(of: isTextFieldFocused) { newValue in
214
- isFocused = newValue
215
- if newValue {
216
- onFocus?()
217
- } else {
218
- onBlur?()
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: $text, onEditingChanged: { focused in
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: { text = "" }) {
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 {
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@momo-kits/native-kits",
3
- "version": "0.156.4-debug",
3
+ "version": "0.156.5-debug",
4
4
  "private": false,
5
5
  "dependencies": {},
6
6
  "devDependencies": {},