如何用RxSwift替换简单的委托协议?

时间:2019-03-25 19:05:30

标签: rx-swift

我正在寻找避免定义的最佳方法

protocol SomeTableviewCellDelegate: class {
    func didSelectTopic(topic: RTopic)
}

并改用RxSwift。

我已经定义了

var didSelectTopic: Observable<RTopic> {
    return _didSelectTopic.asObservable()
}
private let _didSelectTopic = PublishSubject<RTopic>()

var didDeselectTopic: Observable<RTopic> {
     return _didDeselectTopic.asObservable()
}
private let _didDeselectTopic = PublishSubject<RTopic>()

我找不到发出主题的正确方法(我是RxSwift的新手)。

1 个答案:

答案 0 :(得分:1)

您尝试传递有关选定项目的信息(如果我输入错了,请纠正我)。

当前选择的项目

我们不在乎选择/取消选择,我们只关心在每个时刻选择的项目。

这可能是最常见的用例,因为您的用户(组件的用户)不需要具有本地状态。

protocol TopicSelectionProvider {
    var selectedTopicsStream: Observable<Set<RTopic>> { get }
}

class MyTableDelegate: NSObject, UITableViewDelegate {
    private var selectedTopics = BehaviourRelay<Set<RTopic>>([])
    ...
    func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) {
        var selected = selectedTopics.value
        selected.insert(topics[indexPath.row])
        selectedTopics.accept(selected)
    }

    func tableView(_ tableView: UITableView, didDeselectRowAt indexPath: IndexPath) {
        var selected = selectedTopics.value
        selected.remove(topics[indexPath.row])
        selectedTopics.accept(selected)
    }
}

extension MyTableDelegate: TopicSelectionProvider {
    var selectedTopicsStream: Observable<Set<RTopic>> {
        return selectedTopics.asObservable()
    }
}

选择/取消选择事件

我们不在乎所选项目,我们只对它们自身发生的事件感兴趣。

当您不需要项而是事件(例如用于api调用)时,可以使用此用例。

enum TopicSelectionEvent {
    case select(RTopic)
    case deselect(RTopic)
}

protocol TopicSelectionProvider {
    var topicSelectionStream: Observable<TopicSelectionEvent> { get }
}

class MyTableDelegate: NSObject, UITableViewDelegate {
    private var topicSelection = PublishSubject<TopicSelectionEvent>()
    ...
    func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) {
        topicSelection.onNext(.select(topics[indexPath.row]))
    }

    func tableView(_ tableView: UITableView, didDeselectRowAt indexPath: IndexPath) {
        topicSelection.onNext(.deselect(topics[indexPath.row]))
    }
}

extension MyTableDelegate: TopicSelectionProvider {
    var topicSelectionStream: Observable<TopicSelectionEvent> {
        return topicSelection.asObservable()
    }
}

这两种方法均有效,并且具有proc和cons。选择最适合您的一种。