DiffableDataSource CollectionView 在部分中不返回任何项目

Posted

技术标签:

【中文标题】DiffableDataSource CollectionView 在部分中不返回任何项目【英文标题】:DiffableDataSource CollectionView returning no items in section 【发布时间】:2020-10-02 02:43:10 【问题描述】:

这是我的课:

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?我该怎么做?

【问题讨论】:

【参考方案1】:
  var snapshot = DataSourceSnapshot()
  SectionLayoutKind.allCases.forEach 
      snapshot.appendSections([$0])
      let records_copy = records
      snapshot.appendItems(records_copy, toSection: $0)
  

那么,让我们考虑一下该代码中发生了什么。 SectionLayoutKind.allCases 有两种情况,所以forEach 运行两次。

    第一次,我们追加一个节,然后追加四条记录。

    第二次,我们附加另一个部分,然后将 相同的四个记录附加到它。这有效地从第一部分中删除了四条记录,并将它们放在第二部分中。

我不确定出于某种原因使用相同的数组是否不好

这并不完全是“坏”,但它肯定不会让你到达你想去的地方。请记住,所有项目——不是同一部分的所有项目,而是所有项目——必须是唯一的。显然,如果您两次使用相同的四个记录,那不是唯一的。唯一并不意味着它是相同的还是不同的对象。唯一性由单元标识符类型的 Hashable / Equatable 实现确定,在本例中为 testRecord。从这个意义上说,您的副本与原始对象集相同,因此就可区分数据源而言,它们算作相同。

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

【讨论】:

使用 records_copy 是我试图避免从第一部分中删除这四个记录的方式。我想将它们保留在那里并将它们也添加到部分部分 - 8 条记录。我该怎么做? 正如我在回答中所说,我不能告诉你,因为我不知道 testRecord 是什么。我已经回答了您提出的问题,即您在第一部分中获得零行的原因。希望对您有所帮助! 也仔细看看我的回答,它也解释了为什么records_copy 是无关紧要的。这是什么 object 并不重要,重要的是它的 Hashable/Equatable 是如何工作的。 “同一对象”的意思是“等于”。 您也许可以通过将 UUID 属性添加到 testRecord 来解决问题。

以上是关于DiffableDataSource CollectionView 在部分中不返回任何项目的主要内容,如果未能解决你的问题,请参考以下文章

UICollectionViewController + diffableDataSource - 当您的手指仍在其上时,单元格不会突出显示

DiffableDataSource CollectionView 在部分中不返回任何项目

DiffableDataSource 抛出“致命:提供的标识符不是唯一的。”将 MPMediaItem 包装在结构中时

DiffableDataSource引发“致命:提供的标识符不是唯一的。”将MPMediaItem包装在结构中时

UITableView diffable 数据源

mybatis里返回类型用resultmap映射的时候,如果resultmap写了一个collec