RxSwift MVVM Как настроить модель просмотра с помощью диспетчера элементов?

Допустим, у меня есть следующий SwiftRx (2.0.0-beta.4) MVVM:

У меня 4 вещи:

  • ItemListViewController
  • ItemsViewModel
  • ItemsManager
  • Элемент

ItemsManager имеет функцию items (), которая возвращает элементы наблюдаемым способом RxSwift.

ItemsViewModel пока нужно только передать элементы. Позже можно применить логику отображения к атрибуту элемента для контроллера представления (например, для правильного отображения даты).

ItemListViewController поместит элементы в таблицу, по одному элементу в строке.

У элемента есть 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()
}

В модели представления есть следующее:

let items = Variable(/* how to subsribe to the items in the manager? */)

person finneycanhelp    schedule 08.12.2015    source источник


Ответы (1)


Выдуманное решение, также известное как нечто, демонстрирующее общий подход и более простое для понимания, приведено ниже:

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)

    }
}
person finneycanhelp    schedule 10.12.2015