RxSwift MVVM如何使用项目管理器设置视图模型?

fin*_*elp 5 swift rx-swift

假设我有以下SwiftRx(2.0.0-beta.4)MVVM情况:

我有4件事:

  • ItemListViewController
  • ItemsViewModel
  • ItemsManager
  • 项目

ItemsManager有一个名为items()的函数,它将以可观察的RxSwift方式返回Items.

ItemsViewModel目前只需要传递项目.稍后可能会在视图控制器的Item属性上应用显示逻辑(比如正确显示日期).

ItemListViewController将把项放在一个表中,每行一个Item.

一个Item有4个属性(如标识符,日期......),它们将显示在表格行单元格中.

如何在ItemsViewModel和ItemsManager中设置它,以便在管理器中添加,删除,更改项目时,它们会通过ItemsViewModel?

从阅读文档和查看Rx.playground看起来现在使用的东西似乎是一个RxSwift PublishSubject <Item>或者可能是一个RxSwift 地图,它以某种方式订阅了经理的项目()

怎么做得好?

ItemsManager现在有类似的东西:

func items() -> Observable<Item> {
    // placeholder for now
    return [Item(identification: "123", content: ""), Item(identification: "456", content:""), Item(identification: "789", content:"")].toObservable()
}
Run Code Online (Sandbox Code Playgroud)

View Model有这个:

let items = Variable(/* how to subsribe to the items in the manager? */)
Run Code Online (Sandbox Code Playgroud)

fin*_*elp 8

一个人为的解决方案,也就是展示一般方法并且更易于理解的内容如下:

import UIKit
import RxSwift
import RxCocoa

struct Item {

    let identification: String
    let content: String
}

struct ItemsManager {

    let items: Variable<[Item]> = Variable<[Item]>(
        [Item(identification: "some id1", content: "some content"),
        Item(identification: "some id2", content: "some more content")]
    )
}

struct ItemsViewModel {

    let itemsManager = ItemsManager()

    let myItems:Observable<[String]>

    init() {

        myItems = itemsManager.items
        .map({ someArrayOfItems in
            return someArrayOfItems.map {$0.content }
        })
    }
}

class ItemListViewController: UIViewController {

    @IBOutlet weak var tableView: UITableView!

    let itemsViewModel = ItemsViewModel()

    let disposeBag = DisposeBag()

    override func viewDidLoad() {
        super.viewDidLoad()

        itemsViewModel.myItems
            .bindTo(tableView.rx_itemsWithCellIdentifier("itemListCell")) { (row, element, cell) in

                guard let myCell: UITableViewCell = cell else {
                    return
                }

                myCell.textLabel?.text = element
            }
            .addDisposableTo(disposeBag)

    }
}
Run Code Online (Sandbox Code Playgroud)