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 для обоих разделов? Как я могу это сделать?

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типа так что я не могу делать дальнейшие наблюдения. Но, пожалуйста, никогда больше не пишите код, где тип начинается с маленькой буквы.)

Другие вопросы по тегам