UITextField EditingDidBegin 在 IOS 14.4 上触发了两次

UITextField EditingDidBegin triggered twice on IOS 14.4

我有两个文本框。我想在点击第一个文本字段时打开 bottomsheet 。另一个文本字段定期接受用户输入。

produklistrikDropdown 文本字段带有 edittingDidBegin 时,这是我的触发器。

produkListrikDropDown.rx.controlEvent(.editingDidBegin)
            .subscribe(onNext:{
                self.dismissKeyboard()
                print("produk Listrik Field did begin")
                self.viewModel.showProdukListrik()
            })
            .disposed(by: disposeBag)

在我从底部选择一个产品后 sheet 它只是将字段的文本更改为我选择的那个。

viewModel.currentProdukListBehaviorSubject
            .map{[=12=]?.rawValue}
            .subscribe(onNext: { [unowned self] in
                if produkListrikDropDown.text != [=12=] {
                    produkListrikDropDown.text = [=12=]
                }
            })
            .disposed(by: disposeBag)

这是我的第二个文本域配置。只是一个接收用户输入的常规文本字段。

idPelangganField.rx.text
            .bind(to: viewModel.currentIdPelangganBehaviorSubject)
            .disposed(by: disposeBag)

发生了什么

当我点击第一个文本字段并选择一个产品时,它的表现还不错。 但是,当我点击第二个文本字段并返回第一个文本字段时,第一个文本字段上的 edittingDidBegin 触发了两次。这不会发生在 IOS 12.4.

您可以克隆此存储库以重现该问题 https://github.com/Bobbyphtr/TextfieldDidEdittingProblem/tree/master

问题是您在 textFieldShouldBeginEditing(_:) 中执行副作用。操作系统可以根据需要随意调用此函数,因为该函数的唯一用途是确定 SDK 是否应允许文本字段开始编辑。

继续阅读 Command-Query Separation(不仅仅是维基百科文章。进行互联网搜索并阅读其他相关文章。)

It states that every method should either be a command that performs an action, or a query that returns data to the caller, but not both. In other words, asking a question should not change the answer.1 More formally, methods should return a value only if they are referentially transparent and hence possess no side effects.

您的问题的解决方案是使用带有点击手势识别器的 UILabel 而不是文本字段。确保将标签的 isInteractionEnabled 设置为 true。


我在您的代码中也发现了第二个问题。 showBottomSheetPicker(selectionList:) 返回的 Observable 永远不会完成。确保在发送下一个事件后调用 observer.onCompleted()

我找到了一种更好的方法,同时又不牺牲文本字段的使用。正如@Daniel T. 所说,我们可以在其上使用点击手势识别器。我为 UITextField 放置了一个标记手势识别器,并将其设置为接收 touch down 事件。它消除了 textFieldShouldBeginEditting(_:) 的任何副作用,也不使用任何 RxSwift 进行调用。