UIViewControllerRepresentable не получает обновления фокуса, когда в Scrollview - PullRequest
0 голосов
/ 11 ноября 2019

У меня есть SwiftUI UIViewControllerRepresentable, который я встроил в ScrollView (см. Ниже)

import SwiftUI

struct ContactView: View {
    @State private var text = ""
    var body: some View {
        ScrollView{
            TextFieldView(text: $text, dismissKeyboardCallback: nil)
        }
    }
}

Когда я нажимаю на TextField, ничего не происходит. Если я заменю родительский ScrollView на VStack, TextFieldRepresentable сможет получать фокус в обычном режиме (появляется программная клавиатура, и я могу вводить данные). Это ошибка или я что-то делаю неправильно? UIViewControllerRepresentable включен ниже, если он вообще помогает:

import SwiftUI

struct TextFieldView: View {
    var text: Binding<String>
    var onDismissKeyboard: (() -> Void)?

    var body: some View {
        TextFieldRepresentable(text: text, dismissKeyboardCallback: self.onDismissKeyboard)
            .frame(height: 32, alignment: .leading)
    }
}

struct TextFieldRepresentable: UIViewControllerRepresentable {
    let dismissKeyboardCallback: (() -> Void)?
    let viewController: TextFieldViewController

    init (text: Binding<String>, dismissKeyboardCallback: (() -> Void)?) {
        self.dismissKeyboardCallback = dismissKeyboardCallback
        self.viewController = TextFieldViewController(text: text, onDismiss: dismissKeyboardCallback)
    }

    func makeUIViewController(context: Context) -> UIViewController {
        return viewController
    }

    func updateUIViewController(_ viewController: UIViewController, context: Context) {}
}

И сам TextFieldViewController:

class TextFieldViewController: UIViewController, UITextFieldDelegate {
    let text: Binding<String>?
    let onDismiss: (() -> Void)?

    init (text: Binding<String>, onDismiss: (() -> Void)?) {
        self.text = text
        self.onDismiss = onDismiss
        super.init( nibName: "TextField", bundle: Bundle.main)
    }

    required init?(coder: NSCoder) {
        self.text = nil
        self.onDismiss = nil
        super.init(coder: coder)
    }

    fileprivate func getTextField() -> UITextField? {
        return view.subviews.first as? UITextField
    }

    override func viewDidLoad() {
        let textField = self.getTextField()
        textField?.delegate = self

        let toolbar = UIToolbar(frame: CGRect(x: 0, y: 0, width: UIScreen.main.bounds.width, height: 50))
        toolbar.barStyle = UIBarStyle.default
        toolbar.items = [UIBarButtonItem(title: "Done", style: UIBarButtonItem.Style.done, target: self, action: #selector(self.onSet))]
        textField?.inputAccessoryView = toolbar
    }

    @objc private func onSet() {
        let textField = self.getTextField()
        textField?.resignFirstResponder()
        self.text?.wrappedValue = textField?.text ?? ""
        self.onDismiss?()
    }
}
...