0

这是我的课:

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?我怎样才能做到这一点?

4

1 回答 1

1
  var snapshot = DataSourceSnapshot()
  SectionLayoutKind.allCases.forEach {
      snapshot.appendSections([$0])
      let records_copy = records
      snapshot.appendItems(records_copy, toSection: $0)
  }

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

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

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

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

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

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

于 2020-10-02T02:59:37.167 回答