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 进行调用。
我有两个文本框。我想在点击第一个文本字段时打开 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 进行调用。