我是RxSwift
的新手,我发现的所有示例都处理的是简单案例。
我正在尝试为我的文本字段进行表单验证。
我的自定义TextField类有一个方法isValid()
和一个regexp
。 isValid
返回基于regexp
属性。
到目前为止,我写了以下内容:
let valids = [mLastName, mFirstName, mEmailField].map {
$0.rx.text.map {
text -> Bool in
// I want more complex logic here
// Like return field.isValid()
return text!.characters.count > 0
}
}
let _ = Observable.combineLatest(valids) { iterator -> Bool in
return iterator.reduce(true, { $0 && $1 })
}.subscribe(onNext: { allValid in
///update button according to AllValid
})
根据我的Observable<Bool>
方法而不是isValid()
text!.characters.count
基于 $canzone = $_POST['canzone'];
。
答案 0 :(得分:5)
可能有很多方法可以做到这一点。
您可以使用filter
转换自定义TextField类中的rx.text
Observable:
var isTextValid: Observable<Bool> {
return rx.text.filter { _ in
return self.isValid()
}
}
然后您可以将所有文本字段中的isTextValid
与combineLatest
合并。
您还可以从自定义文本字段中提取验证逻辑(可能您甚至根本不需要自定义文本字段)。
好处:
UITextView
,如果您曾使用过它)。验证员类的草稿:
class TextValidator {
var input: Observable<String>
var regex: NSRegularExpression
init(input: Observable<String>, regex: NSRegularExpression) {
self.input = input
self.regex = regex
}
func validate() -> Observable<Bool> {
return input.map { text in
//return true if regex matches the text
}
}
}
然后您可以按如下方式使用它:
let mailValidator = TextValidator(input: mLastName.rx.text, regex: /* actual regex*/)
let firstNameValidator = TextValidator(input: mFirstName.rx.text, regex: ...)
let _ = Observable.combineLatest(mailValidator.validate(), firstName.validate(), ...)
// and so on
现在,如果您想为验证器编写单元测试(您可能应该这样做),您只需将Observable.just("Some value")
作为input
传递给TextValidator
并验证validate()
返回的Observable。 {1}}确实。
答案 1 :(得分:3)
我自己找到了答案。问题出在第一张地图上,我不应该使用匿名参数。
见:
let valids = [mLastName, mFirstName, mEmailField].map { field in
field.rx.text.map({ _ in return field.isValid() })
}
_ = Observable.combineLatest(valids) { iterator -> Bool in
return iterator.reduce(true, { return $0 && $1 })
}.bindTo(self.mValidateButton.rx.isEnabled)