3

我最近在玩 Combine 框架,想知道是否可以创建一些智能扩展来作为 Publisher 获取文本更改。

假设我有两个 UITextField:

firstTextField.textPub.sink {
    self.viewModel.first = $0
}

secondTextField.textPub.sink {
    self.viewModel.second = $0
}

其中第一个和第二个变量只是`@Published var first/second: String = ""

extension UITextField {
    var textPub: AnyPublisher<String, Never> {
        return NotificationCenter.default
            .publisher(for: UITextField.textDidChangeNotification)
            .map {
                guard let textField = $0.object as? UITextField else { return "" }
                return textField.text ?? ""
            }
            .eraseToAnyPublisher()
    }
}

这不起作用,因为我使用的是 NotificationCenter 的共享实例,因此当我对任何 textFields 进行任何更改时,它会将新值传播到两个sink闭包。你认为有什么方法可以实现类似于rx.textRxSwift 中可用的东西吗?我正在考虑使用addTargetwith 闭包,但它需要使用来自 Objective-C 的关联对象。

4

2 回答 2

16

我想通了。我们可以使用 NotificationCenter 传递对象,然后过滤所有与我们的实例不匹配的实例。它似乎按我的预期工作:

extension UITextField {
    var textPublisher: AnyPublisher<String, Never> {
        NotificationCenter.default
            .publisher(for: UITextField.textDidChangeNotification, object: self)
            .compactMap { $0.object as? UITextField }
            .map { $0.text ?? "" }
            .eraseToAnyPublisher()
    }
}
于 2019-06-27T17:55:12.687 回答
0

我建议您将订阅者添加到视图模式,并将它们连接到视图控制器上下文中的文本字段发布者。

NotificationCenter 对于在应用程序范围内调度事件很有用;连接完全由 View Controller 拥有的项目时,无需使用它。但是,一旦您更新了视图模式,将“视图模式发生变化”事件发布到 NotificationCenter 可能是有意义的。

于 2019-06-24T13:09:55.830 回答