首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >包含来自一个API源的多个部分的RxDataSources tableView

包含来自一个API源的多个部分的RxDataSources tableView
EN

Stack Overflow用户
提问于 2019-03-27 21:38:16
回答 2查看 3.9K关注 0票数 1

目前,对于我们的API请求,我们使用Rx。我们如何使用它的一个例子是:

代码语言:javascript
复制
let orderRxService = OrderRxService.listAsShop(shopId, status: .active)
    .repeatRequest(delay: 4)
    .observeOn(MainScheduler.instance)
    .subscribe( onNext: { [weak self] orders in
        self?.orders = orders
        self?.tableView.reloadData()
    })
    .disposed(by: disposeBag)

这将获取状态为.active的给定shopId的所有订单。每次更新时,都会替换本地orders对象并重新加载tableView。

这会重新加载整个tableView,这是我们想要避免的。我现在正在研究RxDataSources,但不能真正弄清楚让它工作的方法。

Order对象有另一个属性currentStatus,它可以是3个不同的值。我们有一个包含3个不同部分的tableView,每个部分显示一个currentStatus的所有订单。

这应该如何在RxDataSources中实现?理想的做法是将它绑定到我之前展示的服务(OrderRxService.....subscribe()..)。

我现在要设置的RxDataSources-types是:

代码语言:javascript
复制
extension Order: IdentifiableType, Equatable {
    public typealias Identity = String

    public var identity: String {
        return String(id)
    }

    public static func == (lhs: Order, rhs: Order) -> Bool {
        return (lhs.timeCreated ?? 0) > (rhs.timeCreated ?? 0)
    }
}

struct OrdersSection {
    var header: String
    var orders: [Order]
}

extension OrdersSection: AnimatableSectionModelType {
    typealias Item = Order
    typealias Identity = String

    var identity: String {
        return header
    }

    var items: [Item] {
        set {
            orders = items
        }
        get {
            return orders
        }
    }

    init(original: OrdersSection, items: [Order]) {
        self = original
        self.items = items
    }
}

我试图让它工作的是:

代码语言:javascript
复制
// I tried to make our local orders a Variable (I don't like this between-step and would like this to be just [Order]).
var orders: Variable<[Order]> = Variable([])


fun viewDidLoad() {
    super.viewDidLoad()

    // Then I set the local orders-variable's value to the new value coming from our Rx service.
    let orderRxDisposable: Disposable = OrderRxService.listAsShop(shopId, status: .active)
        .repeatRequest(delay: 4)
        .observeOn(MainScheduler.instance)
        .map { $0.items }.subscribe( onNext: { [weak self] orders in
            self?.orders.value = orders
        })

    // Here I setup the dataSource
    let dataSource = RxTableViewSectionedAnimatedDataSource<OrdersSection>(
        configureCell: { ds, tv, ip, item in
            let cell = tv.dequeueReusableCell(withIdentifier: "OrderCell", for: ip) as! OrderCell
            cell.addContent(item, tableView: tv, viewController: self, spotDelegate: self)
            return cell
        },

        titleForHeaderInSection: { ds, ip in
            return ds.sectionModels[ip].header
        }
    )

    // Here I set up the three different sections.
    self.orders.asObservable().observeOn(MainScheduler.instance)
        .map { o in
            o.filter { $0.currentStatus == .status_one }
        }
        .map { [OrdersSection(header: "Status one", orders: $0)] }
        .bind(to: self.tableView.rx.items(dataSource: dataSource))

    self.orders.asObservable().observeOn(MainScheduler.instance)
        .map { o in
            o.filter { $0.currentStatus == .status_two }
        }
        .map { [OrdersSection(header: "Status two", orders: $0)] }
        .bind(to: self.tableView.rx.items(dataSource: dataSource))

    self.orders.asObservable().observeOn(MainScheduler.instance)
        .map { o in
            o.filter { $0.currentStatus == .status_three }
        }
        .map { [OrdersSection(header: "Status three", orders: $0)] }
        .bind(to: self.tableView.rx.items(dataSource: dataSource))

}

可能有不同的方面可以改进。例如,我想只做[Order]Variable<[Order]>。与其让这一点可见,是否可以完全跳过,通过观察我们的OrderRxService来创建三个不同的部分?

有没有可能有这样的东西:

代码语言:javascript
复制
OrderRxService.listAsshop(shopId, status: .active).observeOn(MainScheduler.instance)
    // First section
    .map { o in
        o.filter { $0.status == .status_one }
    }
    .map { [OrdersSection(header: "Status one", orders: $0)] }
    .bind(to: self.tableView.rx.items(dataSource: dataSource))
    // Second section
    .map { o in
        o.filter { $0.status == .status_two }
    }
    .map { [OrdersSection(header: "Status two", orders: $0)] }
    .bind(to: self.tableView.rx.items(dataSource: dataSource))
    // Etc...

谢谢你的帮助!

EN

回答 2

Stack Overflow用户

回答已采纳

发布于 2019-03-27 22:31:39

您可以像这样创建一个模型:

代码语言:javascript
复制
enum SectionModel {
  case SectionOne(items: [SectionItem])
  case SectionTwo(items: [SectionItem])
  case SectionThree(items: [SectionItem])
}

enum SectionItem {
  case StatusOne()
  case StatusTwo()
  case StatusThree()
}

extension SectionModel: SectionModelType {
  typealias Item = SectionItem

  var items: [SectionItem] {
      switch self {
      case .SectionOne(items: let items):
          return items.map { $0 }
      case .SectionTwo(items: let items):
          return items.map { $0 }
      case.SectionThree(items: let items):
          return items.map { $0 }
      }
  }

  init(original: SectionModel, items: [Item]) {
      switch  original {
      case .SectionOne(items: _):
          self = .SectionOne(items: items)
      case .SectionTwo(items: _):
          self = .SectionTwo(items: items)
      case .SectionThree(items: _):
          self = .SectionThree(items: items)
      }
  }
}

并处理数据源中的不同项。

代码语言:javascript
复制
dataSource = RxCollectionViewSectionedReloadDataSource<SectionModel>(configureCell: { (datasource, collectionView, indexPath, _) in
        switch datasource[indexPath] {
        case .StatusOne:
            let cell = collectionView.dequeueReusableCell(withReuseIdentifier: R.reuseIdentifier.statusCellOne, for: indexPath)!
            // do stuff
            return cell
        case .StatusTwo:
            let cell = collectionView.dequeueReusableCell(withReuseIdentifier: R.reuseIdentifier.statusCellTwo, for: indexPath)!
            // do stuff
            return cell
        case .StatusThree:
            let cell = collectionView.dequeueReusableCell(withReuseIdentifier: R.reuseIdentifier.statusCellThree, for: indexPath)!
            // do stuff
            return cell
        }
    })

然后将您的oders映射到SectionModelSectionItem,并将其绑定到dataSource

票数 5
EN

Stack Overflow用户

发布于 2021-10-20 20:58:21

您不需要映射->绑定->映射->绑定...您可以简单地在一个“map”中处理所有这些操作,如下所示:

代码语言:javascript
复制
OrderRxService.listAsshop(shopId, status: .active).observeOn(MainScheduler.instance)
    .map { orders in
        let sections: [OrdersSection] = []

        sections.append(OrdersSection(header: "Status one", orders: orders.filter { $0.status == .status_one })
        
        sections.append(OrdersSection(header: "Status two", orders: orders.filter { $0.status == .status_two })
 
        sections.append(OrdersSection(header: "Status three", orders: orders.filter { $0.status == .status_three })
        
        return sections
    }
    .bind(to: self.tableView.rx.items(dataSource: dataSource))
    .disposed(by: disposeBag)

此外,如果您经常使用RxSwift /RxDataSources/等,并且需要指导,那么加入RxSwift Slack频道是一个很好的资源:Invite to RxSwift Slack

票数 0
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/55378613

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档