diff --git a/Snapshots/iPhone/InputFieldTests/testInputFields.1.png b/Snapshots/iPhone/InputFieldTests/testInputFields.1.png index f0e3707bf25..ca5d691406c 100644 Binary files a/Snapshots/iPhone/InputFieldTests/testInputFields.1.png and b/Snapshots/iPhone/InputFieldTests/testInputFields.1.png differ diff --git a/Snapshots/iPhone/InputFieldTests/testInputFields.2.png b/Snapshots/iPhone/InputFieldTests/testInputFields.2.png index 035580911b2..23c775dd92c 100644 Binary files a/Snapshots/iPhone/InputFieldTests/testInputFields.2.png and b/Snapshots/iPhone/InputFieldTests/testInputFields.2.png differ diff --git a/Snapshots/iPhone/InputFieldTests/testInputFieldsPassword.1.png b/Snapshots/iPhone/InputFieldTests/testInputFieldsPassword.1.png index 9fe309906b6..337f6239999 100644 Binary files a/Snapshots/iPhone/InputFieldTests/testInputFieldsPassword.1.png and b/Snapshots/iPhone/InputFieldTests/testInputFieldsPassword.1.png differ diff --git a/Sources/Orbit/Components/InputField.swift b/Sources/Orbit/Components/InputField.swift index 1f55fd23d99..08d8b5b01e5 100644 --- a/Sources/Orbit/Components/InputField.swift +++ b/Sources/Orbit/Components/InputField.swift @@ -71,7 +71,6 @@ public struct InputField: private let isSecure: Bool private let passwordStrength: PasswordStrengthIndicator.PasswordStrength? private let message: Message? - @Binding private var messageHeight: CGFloat // Builder properties (keyboard related) var autocapitalizationType: UITextAutocapitalizationType = .none @@ -82,7 +81,7 @@ public struct InputField: var shouldDeleteBackwardAction: (String) -> Bool = { _ in true } public var body: some View { - FieldWrapper(message: message, messageHeight: $messageHeight) { + FieldWrapper(message: message) { InputContent(state: state, message: message, isFocused: isFocused) { textField } label: { @@ -119,7 +118,8 @@ public struct InputField: isSecureTextEntry: isSecure && isSecureTextRedacted, state: state, leadingPadding: .small, - trailingPadding: .small + trailingPadding: .small, + keyboardSpacing: keyboardSpacing ) .returnKeyType(returnKeyType) .autocorrectionDisabled(isAutocorrectionDisabled) @@ -135,6 +135,12 @@ public struct InputField: isFocused = false inputFieldEndEditingAction() } + // Reverts the additional keyboard spacing used for native keyboard avoidance + .padding(.bottom, -keyboardSpacing) + .overlay( + resolvedPrompt, + alignment: .leadingFirstTextBaseline + ) .accessibility(children: nil) { label } value: { @@ -155,6 +161,10 @@ public struct InputField: } } } + + private var keyboardSpacing: CGFloat { + .medium + } @ViewBuilder private var defaultLabel: some View { switch labelStyle { @@ -195,7 +205,6 @@ public struct InputField: /// /// - Parameters: /// - message: Optional message below the text field. - /// - messageHeight: Binding to the current height of the optional message. public init( value: Binding, state: InputState = .default, @@ -203,7 +212,6 @@ public struct InputField: isSecure: Bool = false, passwordStrength: PasswordStrengthIndicator.PasswordStrength? = nil, message: Message? = nil, - messageHeight: Binding = .constant(0), @ViewBuilder label: () -> Label, @ViewBuilder prompt: () -> Prompt = { EmptyView() }, @ViewBuilder prefix: () -> Prefix = { EmptyView() }, @@ -215,7 +223,6 @@ public struct InputField: self.isSecure = isSecure self.passwordStrength = passwordStrength self.message = message - self._messageHeight = messageHeight self.label = label() self.prompt = prompt() self.prefix = prefix() @@ -230,7 +237,6 @@ public extension InputField where Label == Text, Prompt == Text, Prefix == Icon, /// /// - Parameters: /// - message: Optional message below the text field. - /// - messageHeight: Binding to the current height of the optional message. @_disfavoredOverload init( _ label: some StringProtocol = String(""), @@ -242,8 +248,7 @@ public extension InputField where Label == Text, Prompt == Text, Prefix == Icon, labelStyle: InputLabelStyle = .default, isSecure: Bool = false, passwordStrength: PasswordStrengthIndicator.PasswordStrength? = nil, - message: Message? = nil, - messageHeight: Binding = .constant(0) + message: Message? = nil ) { self.init( value: value, @@ -251,8 +256,7 @@ public extension InputField where Label == Text, Prompt == Text, Prefix == Icon, labelStyle: labelStyle, isSecure: isSecure, passwordStrength: passwordStrength, - message: message, - messageHeight: messageHeight + message: message ) { Text(label) } prompt: { @@ -268,7 +272,6 @@ public extension InputField where Label == Text, Prompt == Text, Prefix == Icon, /// /// - Parameters: /// - message: Optional message below the text field. - /// - messageHeight: Binding to the current height of the optional message. @_semantics("swiftui.init_with_localization") init( _ label: LocalizedStringKey = "", @@ -281,7 +284,6 @@ public extension InputField where Label == Text, Prompt == Text, Prefix == Icon, isSecure: Bool = false, passwordStrength: PasswordStrengthIndicator.PasswordStrength? = nil, message: Message? = nil, - messageHeight: Binding = .constant(0), tableName: String? = nil, bundle: Bundle? = nil, labelComment: StaticString? = nil @@ -292,8 +294,7 @@ public extension InputField where Label == Text, Prompt == Text, Prefix == Icon, labelStyle: labelStyle, isSecure: isSecure, passwordStrength: passwordStrength, - message: message, - messageHeight: messageHeight + message: message ) { Text(label, tableName: tableName, bundle: bundle) } prompt: { diff --git a/Sources/Orbit/Components/Select.swift b/Sources/Orbit/Components/Select.swift index 510d17c4e34..a0b4f215847 100644 --- a/Sources/Orbit/Components/Select.swift +++ b/Sources/Orbit/Components/Select.swift @@ -42,8 +42,6 @@ public struct Select = .constant(0), action: @escaping () -> Void, @ViewBuilder label: () -> Label = { EmptyView() }, @ViewBuilder value: () -> Value = { EmptyView() }, @@ -154,7 +151,6 @@ public struct Select = .constant(0), action: @escaping () -> Void ) { self.init( state: state, labelStyle: labelStyle, - message: message, - messageHeight: messageHeight + message: message ) { action() } label: { @@ -213,7 +207,6 @@ public extension Select where Prefix == Icon, Suffix == Icon, Label == Text, Val state: InputState = .default, labelStyle: InputLabelStyle = .default, message: Message? = nil, - messageHeight: Binding = .constant(0), tableName: String? = nil, bundle: Bundle? = nil, labelComment: StaticString? = nil, @@ -222,8 +215,7 @@ public extension Select where Prefix == Icon, Suffix == Icon, Label == Text, Val self.init( state: state, labelStyle: labelStyle, - message: message, - messageHeight: messageHeight + message: message ) { action() } label: { diff --git a/Sources/Orbit/Components/Textarea.swift b/Sources/Orbit/Components/Textarea.swift index d808794b341..3546686fee9 100644 --- a/Sources/Orbit/Components/Textarea.swift +++ b/Sources/Orbit/Components/Textarea.swift @@ -35,7 +35,6 @@ public struct Textarea: View, TextFieldBuildable { private let state: InputState private let message: Message? - @Binding private var messageHeight: CGFloat @ViewBuilder private let label: Label @ViewBuilder private let prompt: Prompt @@ -48,7 +47,7 @@ public struct Textarea: View, TextFieldBuildable { var shouldDeleteBackwardAction: (String) -> Bool = { _ in true } public var body: some View { - FieldWrapper(message: message, messageHeight: $messageHeight) { + FieldWrapper(message: message) { InputContent(state: state, message: message, isFocused: isFocused) { textView .alignmentGuide(.firstTextBaseline) { dimension in @@ -109,19 +108,16 @@ public struct Textarea: View, TextFieldBuildable { /// /// - Parameters: /// - message: Optional message below the text field. - /// - messageHeight: Binding to the current height of the optional message. public init( value: Binding, state: InputState = .default, message: Message? = nil, - messageHeight: Binding = .constant(0), @ViewBuilder label: () -> Label, @ViewBuilder prompt: () -> Prompt = { EmptyView() } ) { self._value = value self.state = state self.message = message - self._messageHeight = messageHeight self.label = label() self.prompt = prompt() } @@ -134,21 +130,18 @@ public extension Textarea where Label == Text, Prompt == Text { /// /// - Parameters: /// - message: Optional message below the text field. - /// - messageHeight: Binding to the current height of the optional message. @_disfavoredOverload init( _ label: some StringProtocol = String(""), value: Binding, prompt: some StringProtocol = String(""), state: InputState = .default, - message: Message? = nil, - messageHeight: Binding = .constant(0) + message: Message? = nil ) { self.init( value: value, state: state, - message: message, - messageHeight: messageHeight + message: message ) { Text(label) } prompt: { @@ -160,7 +153,6 @@ public extension Textarea where Label == Text, Prompt == Text { /// /// - Parameters: /// - message: Optional message below the text field. - /// - messageHeight: Binding to the current height of the optional message. @_semantics("swiftui.init_with_localization") init( _ label: LocalizedStringKey = "", @@ -168,7 +160,6 @@ public extension Textarea where Label == Text, Prompt == Text { prompt: LocalizedStringKey = "", state: InputState = .default, message: Message? = nil, - messageHeight: Binding = .constant(0), tableName: String? = nil, bundle: Bundle? = nil, labelComment: StaticString? = nil @@ -176,8 +167,7 @@ public extension Textarea where Label == Text, Prompt == Text { self.init( value: value, state: state, - message: message, - messageHeight: messageHeight + message: message ) { Text(label, tableName: tableName, bundle: bundle) } prompt: { diff --git a/Sources/Orbit/Support/Forms/FieldWrapper.swift b/Sources/Orbit/Support/Forms/FieldWrapper.swift index 16802fa2613..f231f7c006f 100644 --- a/Sources/Orbit/Support/Forms/FieldWrapper.swift +++ b/Sources/Orbit/Support/Forms/FieldWrapper.swift @@ -1,14 +1,12 @@ import SwiftUI -/// Orbit support component that orovides label and message around input field. +/// Orbit support component that provides label and message around the form field. public struct FieldWrapper: View { - @Binding private var messageHeight: CGFloat - private let message: Message? - @ViewBuilder private let content: Content @ViewBuilder private let label: Label @ViewBuilder private let footer: Footer + @ViewBuilder private let content: Content public var body: some View { VStack(alignment: .leading, spacing: 0) { @@ -18,13 +16,11 @@ public struct FieldWrapper: View { content - ContentHeightReader(height: $messageHeight) { - VStack(alignment: .leading, spacing: 0) { - footer + VStack(alignment: .leading, spacing: 0) { + footer - FieldMessage(message) - .padding(.top, .xxSmall) - } + FieldMessage(message) + .padding(.top, .xxSmall) } } } @@ -32,13 +28,11 @@ public struct FieldWrapper: View { /// Creates Orbit ``FieldWrapper`` around form field content with a custom label and an additional message content. public init( message: Message? = nil, - messageHeight: Binding = .constant(0), @ViewBuilder content: () -> Content, @ViewBuilder label: () -> Label, @ViewBuilder footer: () -> Footer = { EmptyView() } ) { self.message = message - self._messageHeight = messageHeight self.content = content() self.label = label() self.footer = footer() @@ -53,13 +47,11 @@ public extension FieldWrapper where Label == Text { init( _ label: some StringProtocol = String(""), message: Message? = nil, - messageHeight: Binding = .constant(0), @ViewBuilder content: () -> Content, @ViewBuilder footer: () -> Footer = { EmptyView() } ) { self.init( message: message, - messageHeight: messageHeight, content: content, label: { Text(label) @@ -73,13 +65,11 @@ public extension FieldWrapper where Label == Text { init( _ label: LocalizedStringKey = "", message: Message? = nil, - messageHeight: Binding = .constant(0), @ViewBuilder content: () -> Content, @ViewBuilder footer: () -> Footer = { EmptyView() } ) { self.init( message: message, - messageHeight: messageHeight, content: content, label: { Text(label) @@ -114,32 +104,29 @@ struct FieldWrapperPreviews: PreviewProvider { contentPlaceholder } - StateWrapper((true, true, CGFloat(0), false)) { state in + StateWrapper((true, true, false)) { state in VStack(alignment: .leading, spacing: .large) { FieldWrapper( state.0.wrappedValue ? "Form Field Label" : "", - message: state.1.wrappedValue ? .error("Error message") : .none, - messageHeight: state.2 + message: state.1.wrappedValue ? .error("Error message") : .none ) { contentPlaceholder } - Text("Message height: \(state.2.wrappedValue)") - HStack(spacing: .medium) { Button("Toggle label") { state.0.wrappedValue.toggle() - state.3.wrappedValue.toggle() + state.2.wrappedValue.toggle() } Button("Toggle message") { state.1.wrappedValue.toggle() - state.3.wrappedValue.toggle() + state.2.wrappedValue.toggle() } } Spacer() } - .animation(.easeOut(duration: 1), value: state.3.wrappedValue) + .animation(.easeOut(duration: 1), value: state.2.wrappedValue) } .previewDisplayName("Live preview") } diff --git a/Sources/Orbit/Support/TextFields/InsetableTextField.swift b/Sources/Orbit/Support/TextFields/InsetableTextField.swift index b878b747da3..9dd98f1ab8b 100644 --- a/Sources/Orbit/Support/TextFields/InsetableTextField.swift +++ b/Sources/Orbit/Support/TextFields/InsetableTextField.swift @@ -3,18 +3,40 @@ import UIKit /// Orbit `UITextField` wrapper with a larger touch area. public class InsetableTextField: UITextField { - // Using .small vertical padding would cause resize issue in secure mode - public var insets = UIEdgeInsets(top: 11, left: 0, bottom: 11, right: 0) + /// Insets for setting overall control touch area. + public var insets = UIEdgeInsets(top: .small, left: 0, bottom: .small, right: 0) { + didSet { + invalidateIntrinsicContentSize() + } + } + + /// Additonal spacing between insets and keyboard. + public var keyboardSpacing: CGFloat = .medium { + didSet { + invalidateIntrinsicContentSize() + } + } + public var shouldDeleteBackwardAction: (String) -> Bool = { _ in true } - + + /// Insets for overall control touch area and native keyboard avoidance. + public var resolvedInsets: UIEdgeInsets { + .init( + top: insets.top, + left: insets.left, + bottom: insets.bottom + keyboardSpacing, + right: insets.right + ) + } + public override func textRect(forBounds bounds: CGRect) -> CGRect { guard Thread.isMainThread else { return .zero } - return super.textRect(forBounds: bounds).inset(by: insets) + return super.textRect(forBounds: bounds).inset(by: resolvedInsets) } public override func editingRect(forBounds bounds: CGRect) -> CGRect { guard Thread.isMainThread else { return .zero } - return super.textRect(forBounds: bounds).inset(by: insets) + return super.textRect(forBounds: bounds).inset(by: resolvedInsets) } public override func deleteBackward() { diff --git a/Sources/Orbit/Support/TextFields/TextField.swift b/Sources/Orbit/Support/TextFields/TextField.swift index 62049beee81..71554f28cba 100644 --- a/Sources/Orbit/Support/TextFields/TextField.swift +++ b/Sources/Orbit/Support/TextFields/TextField.swift @@ -37,6 +37,7 @@ public struct TextField: UIViewRepresentable, TextFieldBuildable { private var state: InputState private var leadingPadding: CGFloat private var trailingPadding: CGFloat + private var keyboardSpacing: CGFloat // Builder properties (keyboard related) var returnKeyType: UIReturnKeyType = .default @@ -69,6 +70,7 @@ public struct TextField: UIViewRepresentable, TextFieldBuildable { uiView.updateIfNeeded(\.insets.left, to: leadingPadding) uiView.updateIfNeeded(\.insets.right, to: trailingPadding) + uiView.updateIfNeeded(\.keyboardSpacing, to: keyboardSpacing) uiView.updateIfNeeded(\.isSecureTextEntry, to: isSecureTextEntry) // Keyboard related @@ -175,13 +177,15 @@ public extension TextField { isSecureTextEntry: Bool = false, state: InputState = .default, leadingPadding: CGFloat = 0, - trailingPadding: CGFloat = 0 + trailingPadding: CGFloat = 0, + keyboardSpacing: CGFloat = 0 ) { self._value = value self.isSecureTextEntry = isSecureTextEntry self.state = state self.leadingPadding = leadingPadding self.trailingPadding = trailingPadding + self.keyboardSpacing = keyboardSpacing } }