DiffableDataSource CollectionView在节中不返回任何项目

问题描述

这是我的课程:

class MediaViewController: UIViewController{
    var collectionView: UICollectionView! = nil

    
    private lazy var dataSource = makeDataSource()

    fileprivate typealias DataSource = UICollectionViewDiffableDataSource<SectionLayoutKind,testRecord>

    fileprivate typealias DataSourceSnapshot = NSDiffableDataSourceSnapshot<SectionLayoutKind,testRecord>
   override func viewDidLoad() {
        super.viewDidLoad()
        setRecordItems()
        configureHierarchy()
        configureDataSource()
        applySnapshot()

    }
    
    func setRecordItems(){
        for i in 0...3{
            let record = testRecord(daysBack: i/2,progression: i/10)
            records.append(record)
        }
    }
extension MediaViewController {

    func configureHierarchy() {
        collectionView = UICollectionView(frame: view.bounds,collectionViewLayout: createLayout())
        collectionView.autoresizingMask = [.flexibleWidth,.flexibleHeight]
        collectionView.backgroundColor = .systemBackground
        view.addSubview(collectionView)
        collectionView.delegate = self
    }
}
extension MediaViewController {

    fileprivate enum SectionLayoutKind: Int,CaseIterable{
        case records
        case timeline
    }
    fileprivate func makeDataSource() -> DataSource {
        let dataSource = DataSource(
            collectionView: collectionView,cellProvider: { (collectionView,indexPath,testRecord) ->
              UICollectionViewCell? in
              // 2
                switch indexPath.section {
                case 0:
                    let cell = collectionView.dequeueReusableCell(withReuseIdentifier: RecordCollectionViewCell.identifier,for: indexPath) as? RecordCollectionViewCell
                    cell?.configure(with: testRecord)
                    return cell
                case 1:
                    let cell = collectionView.dequeueReusableCell(withReuseIdentifier: TimelineDayCell.identifier,for: indexPath) as? TimelineDayCell
                    cell?.configure(with: testRecord)
                    return cell
                default:
                    return UICollectionViewCell()
                }
          })
          return dataSource
    }
    func configureDataSource() {
        collectionView.register(RecordCollectionViewCell.nib,forCellWithReuseIdentifier: RecordCollectionViewCell.identifier)
        collectionView.register(TimelineDayCell.nib,forCellWithReuseIdentifier: TimelineDayCell.identifier)
    }
    func applySnapshot(animatingDifferences: Bool = true) {
      // 2
      var snapshot = DataSourceSnapshot()
        SectionLayoutKind.allCases.forEach {
            snapshot.appendSections([$0])
            let records_copy = records
            snapshot.appendItems(records_copy,toSection: $0)
        }
      dataSource.apply(snapshot,animatingDifferences: animatingDifferences)
    }
}

因此设置是分为两个部分:记录和时间轴。它们都由相同的数据-记录数组运行。目前,每次我应用快照时,我都从类中复制此数组-由于某些原因,我不确定同时使用同一数组是不好的。

然后在设置数据源时,对于cellProvider我有一个switch语句来检查该部分。如果它的第0部分不适用于一个记录单元,而如果其第1部分则使用一个时间轴单元。

目前,没有记录单元正在生成。当我检查collectionView.numberOfItems(inSection:0)时为0。 collectionView.numberOfItems(inSection:1)是4(记录数)

为什么两个部分都不是4?我该怎么办?

解决方法

const getPositionMain = applySpec ( {
   ratio : R.prop ('ratio'),resultY : R.zipWith ( resultY,R.prop ('ratio'),R.prop ('focusPoint') ),} )
const positionMain = getPositionMain ( Main )

console.log( 'positionMain : ',positionMain )

因此,让我们考虑一下该代码中发生的情况。 var snapshot = DataSourceSnapshot() SectionLayoutKind.allCases.forEach { snapshot.appendSections([$0]) let records_copy = records snapshot.appendItems(records_copy,toSection: $0) } 中有两种情况,因此SectionLayoutKind.allCases运行两次。

  1. 我们第一次附加一个部分,然后向其附加四个记录。

  2. 第二次,我们附加了另一个部分,然后向其添加了相同的四个记录。这样可以有效地从第一部分中删除这四个记录,并将它们放在第二部分中。

我不确定出于某种原因使用两个数组都不好

这并不完全是“坏”,但肯定不能使您到达想要去的地方。请记住,所有项目-不是同一节中的所有项目,而是 all 所有项目-必须是 unique 。显然,如果您两次使用相同的四个记录,那不是唯一的。唯一并不意味着它是相同还是不同的对象。唯一性由您的单元格标识符类型的Hashable / Equatable实现确定,在这种情况下为forEach。从某种意义上说,您的副本与原始对象集相同,因此就可扩散数据源而言,它们的数量相同。

(您没有显示testRecord类型,所以我无法做进一步的观察。但是,请,请不要再在类型以小写字母开头的地方编写代码。)