Rxswift在发送api请求之前验证输入

问题描述 投票:1回答:1

我有一个登录屏幕,我在文本字段上方显示以前的错误输入,并通过组合电子邮件/密码Bool Observable禁用按钮,但是现在设计已更改,我想在发送请求或显示警报之前检查电子邮件/密码是否为空对话

我的问题是如何在命令之前验证输入是否有效:

LoginViewModel:

  var emailValid: Observable<Bool> {
        return emailSubject.asObservable().map { $0.count > 0 && $0.isEmail}
    }

  signInDidTapSubject
       .withLatestFrom(credentialsObservable)
        .flatMapLatest { credentials -> Observable<Event<Result<UserResponse>>> in
            self.loadInProgress.accept(true)
            return network.login(with: credentials).materialize()
        }
        .subscribe(onNext: { [weak self] event in
            self?.loadInProgress.accept(false)
            switch event {
            case .next(let result):
                switch result{
                case .Success(let user):
                    self?.loginResultSubject.onNext(user)
                case .Failure(let error):
                    self?.errorsSubject.onNext(error)
                }
            case .error( _):
                print("error")
            default:
                break
            }
        })
        .disposed(by: disposeBag)

VC:

   loginButton.rx.tap.asObservable()
        .debounce(0.5, scheduler: MainScheduler.instance)
        .subscribe(viewModel.input.loginButtonTapped).disposed(by: disposeBag)
swift mvvm observable rx-swift
1个回答
2
投票

你想出这个问题的方法是问自己,“我需要什么输入来产生这个输出?”

这里有两个输出“User”和“Error”。让我们先做快乐的道路吧。您需要哪些输入才能生成用户?只是一个成功的网络响应。但是,您需要哪些输入才能产生网络响应?那么你需要电子邮件和密码,但是煽动事件正在点击登录按钮,如果数据无效,则不应该发出请求:

let networkResponse = loginTrigger
    .withLatestFrom(credentials)
    .filter { $0.email.isEmail && !$0.password.isEmpty }
    .flatMapLatest {
        network.login(with: $0)
            .materialize()
    }
    .share(replay: 1)

let user = networkResponse
    .filter { $0.element != nil }
    .map { $0.element! }

现在,您需要哪些输入才能生成错误输出?您需要网络请求中的错误(如果有),并且您需要知道电子邮件或密码是否无效。但是,如果点击登录按钮,您只想知道无效输入...

这是很多,但这是它如何崩溃:

let networkError = networkResponse
    .filter { $0.error != nil }
    .map { $0.error! }

let invalidEmailError = loginTrigger
    .withLatestFrom(email)
    .filter { !$0.isEmail }
    .map { _ in UIError.badEmail as Error }

let invalidPasswordError = loginTrigger
    .withLatestFrom(password)
    .filter { $0.isEmpty }
    .map { _ in UIError.badPassword as Error }

let error = Observable.merge(networkError, invalidEmailError, invalidPasswordError)

看看它是如何工作的?依次获取每个输出并找出使输出工作所需的输入。

© www.soinside.com 2019 - 2024. All rights reserved.