如何测试 RxSwift 变量和 RxCocoa Observable 之间的 UI 绑定?

How to test the UI Binding between RxSwift Variable and RxCocoa Observable?

我有一个简单的 ViewModel 和一个 属性:

class ViewModel {
    var name = Variable<String>("")
}

我将它绑定到我的 ViewController 中的 UITextField:

class ViewController : UIViewController {

    var viewModel : ViewModel!

    @IBOutlet weak var nameField : UITextField!

    private let disposeBag = DisposeBag()

    override func viewDidLoad() {
        super.viewDidLoad()
        // Binding
        nameField.rx.text
            .orEmpty
            .bind(to: viewModel.name)
            .disposed(by: disposeBag)
    }
}

现在,我想对其进行单元测试。

我可以使用 .onNext 事件通过 Rx 填充 nameField.text 属性 - nameField.rx.text.onNext("My Name Here").

但是 viewModel.name 没有被填充。为什么?我怎样才能让这个 Rx 行为在我的 XCTestCase 中起作用?

请看下面我上次测试的结果运行:

我认为您遇到的问题是没有在主线程上明确安排绑定。我建议在这种情况下使用 Driver

class ViewModel {
  var name = Variable<String>("")
}

class ViewController: UIViewController {

  let textField = UITextField()
  let disposeBag = DisposeBag()
  let vm = ViewModel()

  override func viewDidLoad() {
    super.viewDidLoad()
    textField.rx.text
        .orEmpty
        .asDriver()
        .drive(vm.name)
        .disposed(by: disposeBag)
}

// ..

使用 Driver,绑定将始终发生在主线程上,不会出错。

此外,在您的测试中,我会在绑定上调用 skip

sut.nameTextField.rx.text.skip(1).onNext(name)

因为Driver订阅时会重播第一个元素。

最后,我建议使用 RxTest 和 TestScheduler 而不是 GCD。

如果有帮助请告诉我。

rx.text 依赖于以下 UIControlEvents:.allEditingEvents.valueChanged。因此,显式发送一个 onNext 事件不会为这些事件发送动作。您应该手动发送操作。

sut.nameField.rx.text.onNext(name)
sut.nameField.sendActions(for: .valueChanged)