UICollectionViewCompositionalLayout 不能使用与 UICollectionViewDiffableDataSource 相同的部分吗?

UICollectionViewCompositionalLayout can't use the same sections as UICollectionViewDiffableDataSource?

我正在构建一个带有 collection 视图的视图控制器,使用(有点)新的可区分数据源和可组合的 collection 视图布局。我制作了自己的 Section 和 Row 枚举,可以对我的模型(一组照片)的变化做出反应,这会触发 applySnapshot 并且一切正常。

但是..在我的 createLayout 函数中我只有一个部分索引可以使用??这对我来说似乎很奇怪,为什么这两种技术不能更好地协同工作?一个使用 SectionIdentifierTypeItemIdentifierType(太棒了!),另一个仍然基于索引?呃

我想要做的是这样的:Section.header 部分中的所有单元格都需要全屏宽度,并具有自动高度。 Section.photos 部分中的每个单元格都将获得固定大小。并且也可以有其他具有不同单元格大小的部分,并且某些部分可以是可选的。所以 index-based 系统真的很难用,不能硬编码第 0 部分是 header 而第 1 部分是照片:顺序可以改变,事情可能是可选的,等等

有什么更好的方法来处理这个问题?我当然可以在视图控制器上存储数组 [Section],使用节索引以这种方式获取 Section 情况,但是没有 built-in 方法获取节标识符指数?这两项技术是同时发布的,这似乎是合乎逻辑的事情。

class PhotosViewController: UIViewController {
  enum Section: Equatable, Hashable {
    case header
    case photos
  }

  enum Row: Equatable, Hashable {
    case header
    case photo(Photo)
  }

  private lazy var dataSource = makeDataSource()
  var photos: AnyPublisher<[Photo], Never>!

  override func viewDidLoad() {
    super.viewDidLoad()

    collectionView.collectionViewLayout = createLayout()

    photos
      .sink(receiveValue: applySnapshot)
      .store(in: &subscriptions)
  }

  private func makeDataSource() -> UICollectionViewDiffableDataSource<Section, Row> {
    return UICollectionViewDiffableDataSource(collectionView: collectionView) { (collectionView, indexPath, item) -> UICollectionViewCell? in
      switch item {
        case .header:
          let cell = collectionView.dequeueReusableCell(for: indexPath, cellType: ProfileHeaderCollectionViewCell.self)
          return cell

        case .photo(let photo):
          let cell = collectionView.dequeueReusableCell(for: indexPath, cellType: PhotoCollectionViewCell.self)
          cell.configure(photo: photo)
          return cell
      }
    }
  }

  private func applySnapshot(photos: [Photo]) {
    var snapshot = NSDiffableDataSourceSnapshot<Section, Row>()

    let headerSection = Section.header
    snapshot.appendSections([headerSection])
    snapshot.appendItems([.header], toSection: headerSection)

    let items = photos.map { Row.photo([=10=]) }
    let photosSection = Section.photos
    snapshot.appendSections([photosSection])
    snapshot.appendItems(items, toSection: photosSection)

    dataSource.apply(snapshot, animatingDifferences: false)
  }

  private func createLayout() -> UICollectionViewLayout {
    let layout = UICollectionViewCompositionalLayout { [unowned self] sectionIndex, _ in
      // I have to work with a sectionIndex instead of a Section case?
    }

    return layout
  }
}

不用在单独的数组中跟踪部分标识符,幸运的是您可以只使用 dataSource.snapshot().sectionIdentifiers[sectionIndex].