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 包装在结构中时