Обновление RxSwift tableView на основе свойства
У меня есть UITableView, заполненный моделями продуктов. У меня есть 2 проблемы прямо сейчас. Во-первых, у меня есть количество Label в ячейках, которое я хочу обновить, когда пользователь нажимает кнопку увеличения или уменьшения в ячейке. В настоящее время для этого я обновляю модель и принимаю совершенно новый массив, который вызывает перезагрузку tableView, что не идеально. Вторая проблема заключается в том, что, когда количество Продукта становится равным 0, эта ячейка должна быть удалена из tableView с анимацией. В настоящее время я просто нахожу индекс продукта в источнике данных, удаляю его и перезагружаю таблицу. Я хотел бы знать правильный способ Rx сделать это. Вот пример моего кода. Спасибо!
struct Product: Equatable {
var i: String
var quantity: Int
init(i: Int, quantity: Int) {
self.i = "item: \(i)"
self.quantity = quantity
}
}
extension Product {
static func ==(lhs: Product, rhs: Product) -> Bool {
return lhs.i == rhs.i
}
}
class ProductSource {
var items: BehaviorRelay<[Product]> = BehaviorRelay<[Product]>(value: [])
var itemsObservable: Observable<[Product]>
init() {
itemsObservable = items.asObservable().share(replay: 1)
items.accept((0..<20).map { Product(i: $0, quantity: 2) })
}
func add(product: Product) {}
func remove(product: Product) {
guard let index = self.items.value.index(of: product) else {return}
// decrement quantity, if quantity is 0, remove from the datasource and update the tableView
}
}
class ViewController: UIViewController {
@IBOutlet weak var tableView: UITableView!
var disposeBag = DisposeBag()
var data = ProductSource()
override func viewDidLoad() {
super.viewDidLoad()
tableView.rx.setDelegate(self).disposed(by: disposeBag)
data.itemsObservable
.bind(to: tableView.rx.items(cellIdentifier: "productCell", cellType: ProductTableViewCell.self)) { row, element, cell in
cell.nameLabel.text = element.i
cell.quantityLabel.text = String(element.quantity)
}.disposed(by: disposeBag)
}
}
extension ViewController: UITableViewDelegate {
func tableView(_ tableView: UITableView, heightForRowAt indexPath: IndexPath) -> CGFloat {
return 100
}
}
1 ответ
Обе ваши проблемы могут быть решены с помощью RxDataSources
Cocoapod.
Кроме того, вы можете свернуть свой собственный, как я сделал здесь: https://github.com/dtartaglia/RxMultiCounter/blob/master/RxMultiCounter/RxExtensions/RxSimpleAnimatableDataSource.swift
Идея здесь состоит в том, чтобы использовать что-то (я использую DifferenceKit), чтобы выяснить, какие элементы изменились, и перезагрузить только эти конкретные элементы. Для этого вам нужен специализированный объект источника данных, а не универсальный, который поставляется с RxCocoa.