DiffableDataSource CollectionView 在部分中不返回任何项目
DiffableDataSource CollectionView returning no items in section
这是我的 class:
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([[=10=]])
let records_copy = records
snapshot.appendItems(records_copy, toSection: [=10=])
}
dataSource.apply(snapshot, animatingDifferences: animatingDifferences)
}
}
所以设置是有两个部分,记录和时间线。这些都是 运行 由相同的数据 - 记录数组。目前,当我每次应用快照时,我都从 class 复制这个数组 - 我不确定出于某种原因使用相同的数组是否不好..
然后在设置数据源时,对于 cellProvider,我有一个检查该部分的 switch 语句。如果它的第 0 节我将使用记录单元格,如果它的第 1 节我将使用时间轴单元格。
目前没有生成任何记录单元格。当我检查 collectionView.numberOfItems(inSection:0)
时,它是 0。
collectionView.numberOfItems(inSection:1)
为4(条数)
为什么两个部分都不是 4?我该怎么做?
var snapshot = DataSourceSnapshot()
SectionLayoutKind.allCases.forEach {
snapshot.appendSections([[=10=]])
let records_copy = records
snapshot.appendItems(records_copy, toSection: [=10=])
}
那么,让我们考虑一下该代码中发生了什么。 SectionLayoutKind.allCases
中有两种情况,所以forEach
运行了两次。
第一次,我们追加一个节,然后追加四条记录。
第二次,我们追加另一个部分,然后向其追加相同的四个记录。这有效地从第一部分中删除了四个记录并将它们放在第二部分中。
Im not sure its bad to use the same array for both for some reason
它并不完全“糟糕”,但它肯定不能让你到达你想去的地方。请记住,所有项目 — 不是同一部分的所有项目,而是 all 项目 — 必须是 unique。显然,如果您两次使用相同的四个记录,那不是唯一的。独特并不意味着它是相同的还是不同的对象。唯一性由您的单元格标识符类型的 Hashable / Equatable 实现决定,在本例中为 testRecord
。从这个意义上说,您的副本与原始对象集相同,因此就可比较数据源而言,它们算作相同。
(您没有显示 testRecord
类型,所以我无法进行进一步观察。但是请不要再编写类型以小写字母开头的代码。)
这是我的 class:
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([[=10=]])
let records_copy = records
snapshot.appendItems(records_copy, toSection: [=10=])
}
dataSource.apply(snapshot, animatingDifferences: animatingDifferences)
}
}
所以设置是有两个部分,记录和时间线。这些都是 运行 由相同的数据 - 记录数组。目前,当我每次应用快照时,我都从 class 复制这个数组 - 我不确定出于某种原因使用相同的数组是否不好..
然后在设置数据源时,对于 cellProvider,我有一个检查该部分的 switch 语句。如果它的第 0 节我将使用记录单元格,如果它的第 1 节我将使用时间轴单元格。
目前没有生成任何记录单元格。当我检查 collectionView.numberOfItems(inSection:0)
时,它是 0。
collectionView.numberOfItems(inSection:1)
为4(条数)
为什么两个部分都不是 4?我该怎么做?
var snapshot = DataSourceSnapshot()
SectionLayoutKind.allCases.forEach {
snapshot.appendSections([[=10=]])
let records_copy = records
snapshot.appendItems(records_copy, toSection: [=10=])
}
那么,让我们考虑一下该代码中发生了什么。 SectionLayoutKind.allCases
中有两种情况,所以forEach
运行了两次。
第一次,我们追加一个节,然后追加四条记录。
第二次,我们追加另一个部分,然后向其追加相同的四个记录。这有效地从第一部分中删除了四个记录并将它们放在第二部分中。
Im not sure its bad to use the same array for both for some reason
它并不完全“糟糕”,但它肯定不能让你到达你想去的地方。请记住,所有项目 — 不是同一部分的所有项目,而是 all 项目 — 必须是 unique。显然,如果您两次使用相同的四个记录,那不是唯一的。独特并不意味着它是相同的还是不同的对象。唯一性由您的单元格标识符类型的 Hashable / Equatable 实现决定,在本例中为 testRecord
。从这个意义上说,您的副本与原始对象集相同,因此就可比较数据源而言,它们算作相同。
(您没有显示 testRecord
类型,所以我无法进行进一步观察。但是请不要再编写类型以小写字母开头的代码。)