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
проходит дважды.
В первый раз мы добавляем один раздел, а затем добавляем в него четыре записи.
Во второй раз мы добавляем еще один раздел, который затем добавляет к нему те же четыре записи. Это эффективно удаляет четыре записи из первого раздела и помещает их во второй раздел.
Я не уверен, что по какой-то причине плохо использовать один и тот же массив для обоих
Это не совсем "плохо", но определенно не приведет вас туда, куда вы хотите. Помните, что все элементы - не все элементы одного раздела, а все элементы - должны быть уникальными. Очевидно, что если вы используете одни и те же четыре записи дважды, это не уникально. Уникальность не означает, что это тот же объект или другой. Уникальность определяется реализацией Hashable / Equatable вашего типа идентификатора ячейки, которым в данном случае является
testRecord
. В этом смысле ваши копии идентичны исходному набору объектов, поэтому они считаются одинаковыми с точки зрения источника данных для различий.
(Вы не показали
testRecord
типа так что я не могу делать дальнейшие наблюдения. Но, пожалуйста, никогда больше не пишите код, где тип начинается с маленькой буквы.)