![](/img/trans.png)
[英]Update UIButton title and select the action accordingly using RxSwift
[英]RxSwift, binding Action to UIButton stops producing events after view transitions
我正在嘗試使用RxSwift實現MVVM登錄屏幕,並且遇到了一些困難。
我的登錄屏幕(用戶,密碼,登錄按鈕)過渡到相機查看屏幕,在該屏幕上,應用程序檢查相機權限,如果未批准,則將用戶注銷並返回登錄屏幕。
我的LoginViewModel
中有一個loginAction
返回一個Action<Void, LoginResult>
,其中LoginResult
是一個Result<Bool, Error>
而我的loginProvider
服務返回一個Observable<LoginResult>
:
struct LoginViewModel {
let sceneCoordinator: SceneCoordinatorType
let loginProvider: LoginProviderType
var usernameText = Variable<String>("")
var passwordText = Variable<String>("")
var isValid: Observable<Bool> {
return Observable.combineLatest(usernameText.asObservable(), passwordText.asObservable()) { username, password in
username.count > 0 && password.count > 0
}
}
init(loginProvider: LoginProvider, coordinator: SceneCoordinatorType) {
self.loginProvider = loginProvider
self.sceneCoordinator = coordinator
}
lazy var loginAction: Action<Void, LoginResult> = { (coordinator: SceneCoordinatorType, service: LoginProviderType, username: String, password: String) in
return Action<Void, LoginResult>(enabledIf: self.isValid) { _ in
return service.login(username: username, password: password)
.observeOn(MainScheduler.instance)
.do(onNext: { result in
guard let loggedIn = result.value else { return }
if loggedIn {
let cameraViewModel = CameraViewModel(coordinator: coordinator)
coordinator.transition(to: Scene.camera(cameraViewModel), type: .modal)
}
}(self.sceneCoordinator, self.loginProvider, self.companyText.value, self.usernameText.value, self.passwordText.value)
}
一切正常,有效的輸入成功登錄( loginProvider
將請求發送到我的服務器,獲取響應並相應地處理所有其他步驟)。
如果用戶未授予相機權限,則我的CameraViewModel
有一個Observable
,我綁定到CameraViewController
,進行訂閱,並在需要時注銷用戶,並使用CocoaAction
將視圖彈出回到登錄屏幕彈出當前視圖(使用場景協調器類)。
問題是,當我轉換回“登錄”屏幕后嘗試再次登錄時, loginAction
發出的元素的訂閱未收到任何元素。
這是LoginViewController
的代碼:
class LoginViewController: UIViewController, BindableType {
var viewModel: LoginViewModel!
private let disposeBag = DisposeBag()
private var loginAction: Action<Void, LoginResult>!
@IBOutlet weak var usernameTextField: UITextField!
@IBOutlet weak var passwordTextField: UITextField!
@IBOutlet weak var loginButton: UIButton!
@IBOutlet weak var loadingIndicator: UIActivityIndicatorView!
private var usernameObservable: Observable<String> {
return usernameTextField.rx.text
.throttle(0.5, scheduler: MainScheduler.instance)
.map() { text in
return text ?? ""
}
}
private var passwordObservable: Observable<String> {
return passwordTextField.rx.text
.throttle(0.5, scheduler: MainScheduler.instance)
.map() { text in
return text ?? ""
}
}
override func viewDidLoad() {
super.viewDidLoad()
}
func bindViewModel() {
usernameObservable
.bind(to: viewModel.usernameText)
.disposed(by: disposeBag)
passwordObservable
.debug()
.bind(to: viewModel.passwordText)
.disposed(by: disposeBag)
loginAction = viewModel.loginAction
loginAction.elements
.subscribe(onNext: { [weak self] result in
self?.loadingIndicator.stopAnimating()
var message = ""
switch result {
case .failure(.unknownError):
message = "unknown error"
case .failure(.wrongCredentials):
message = "wrong credentials"
case .failure(.serviceUnavailable):
message = "service unavailable"
case let .success(loggedIn):
return
}
self?.errorMessage(message: message)
}).disposed(by: disposeBag)
loginButton.rx.tap
.subscribe(onNext: { [unowned self] in
self.loadingIndicator.startAnimating()
self.loginAction.execute(Void())
})
.disposed(by: disposeBag)
viewModel.isValid
.bind(to: loginButton.rx.isEnabled)
.disposed(by: disposeBag)
}
我看到點擊登錄按鈕會產生點擊事件,但是Action本身不再被調用。 知道我缺少什么嗎?
我不確定ScreenCoordinator的工作方式,但我將從將usernameObservable / passwordObservable的創建移動到viewDidLoad()開始,因為該問題似乎與生命周期相關。 另外,您可以添加更多.debug()調用(使用參數可以在日志中區分它們)。
發現了問題:顯然,該操作未達到完整狀態,因為基礎服務上的一個返回了一個在登錄視圖控制器中訂閱的觀察者,而從攝像機視圖控制器返回該觀察者時未將其丟棄。
我在該服務中添加了一種清除方法,該方法可以在返回登錄視圖時創建一個新的DisposeBag
,該操作可以完成操作並將其釋放以供重用。
聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.