UICollectionViewCompositionalLayout 不能使用与 UICollectionViewDiffableDataSource 相同的部分吗?
UICollectionViewCompositionalLayout can't use the same sections as UICollectionViewDiffableDataSource?
我正在构建一个带有 collection 视图的视图控制器,使用(有点)新的可区分数据源和可组合的 collection 视图布局。我制作了自己的 Section 和 Row 枚举,可以对我的模型(一组照片)的变化做出反应,这会触发 applySnapshot
并且一切正常。
但是..在我的 createLayout
函数中我只有一个部分索引可以使用??这对我来说似乎很奇怪,为什么这两种技术不能更好地协同工作?一个使用 SectionIdentifierType
和 ItemIdentifierType
(太棒了!),另一个仍然基于索引?呃
我想要做的是这样的:Section.header
部分中的所有单元格都需要全屏宽度,并具有自动高度。 Section.photos
部分中的每个单元格都将获得固定大小。并且也可以有其他具有不同单元格大小的部分,并且某些部分可以是可选的。所以 index-based 系统真的很难用,不能硬编码第 0 部分是 header 而第 1 部分是照片:顺序可以改变,事情可能是可选的,等等
有什么更好的方法来处理这个问题?我当然可以在视图控制器上存储数组 [Section]
,使用节索引以这种方式获取 Section
情况,但是没有 built-in 方法获取节标识符指数?这两项技术是同时发布的,这似乎是合乎逻辑的事情。
class PhotosViewController: UIViewController {
enum Section: Equatable, Hashable {
case header
case photos
}
enum Row: Equatable, Hashable {
case header
case photo(Photo)
}
private lazy var dataSource = makeDataSource()
var photos: AnyPublisher<[Photo], Never>!
override func viewDidLoad() {
super.viewDidLoad()
collectionView.collectionViewLayout = createLayout()
photos
.sink(receiveValue: applySnapshot)
.store(in: &subscriptions)
}
private func makeDataSource() -> UICollectionViewDiffableDataSource<Section, Row> {
return UICollectionViewDiffableDataSource(collectionView: collectionView) { (collectionView, indexPath, item) -> UICollectionViewCell? in
switch item {
case .header:
let cell = collectionView.dequeueReusableCell(for: indexPath, cellType: ProfileHeaderCollectionViewCell.self)
return cell
case .photo(let photo):
let cell = collectionView.dequeueReusableCell(for: indexPath, cellType: PhotoCollectionViewCell.self)
cell.configure(photo: photo)
return cell
}
}
}
private func applySnapshot(photos: [Photo]) {
var snapshot = NSDiffableDataSourceSnapshot<Section, Row>()
let headerSection = Section.header
snapshot.appendSections([headerSection])
snapshot.appendItems([.header], toSection: headerSection)
let items = photos.map { Row.photo([=10=]) }
let photosSection = Section.photos
snapshot.appendSections([photosSection])
snapshot.appendItems(items, toSection: photosSection)
dataSource.apply(snapshot, animatingDifferences: false)
}
private func createLayout() -> UICollectionViewLayout {
let layout = UICollectionViewCompositionalLayout { [unowned self] sectionIndex, _ in
// I have to work with a sectionIndex instead of a Section case?
}
return layout
}
}
不用在单独的数组中跟踪部分标识符,幸运的是您可以只使用 dataSource.snapshot().sectionIdentifiers[sectionIndex]
.
我正在构建一个带有 collection 视图的视图控制器,使用(有点)新的可区分数据源和可组合的 collection 视图布局。我制作了自己的 Section 和 Row 枚举,可以对我的模型(一组照片)的变化做出反应,这会触发 applySnapshot
并且一切正常。
但是..在我的 createLayout
函数中我只有一个部分索引可以使用??这对我来说似乎很奇怪,为什么这两种技术不能更好地协同工作?一个使用 SectionIdentifierType
和 ItemIdentifierType
(太棒了!),另一个仍然基于索引?呃
我想要做的是这样的:Section.header
部分中的所有单元格都需要全屏宽度,并具有自动高度。 Section.photos
部分中的每个单元格都将获得固定大小。并且也可以有其他具有不同单元格大小的部分,并且某些部分可以是可选的。所以 index-based 系统真的很难用,不能硬编码第 0 部分是 header 而第 1 部分是照片:顺序可以改变,事情可能是可选的,等等
有什么更好的方法来处理这个问题?我当然可以在视图控制器上存储数组 [Section]
,使用节索引以这种方式获取 Section
情况,但是没有 built-in 方法获取节标识符指数?这两项技术是同时发布的,这似乎是合乎逻辑的事情。
class PhotosViewController: UIViewController {
enum Section: Equatable, Hashable {
case header
case photos
}
enum Row: Equatable, Hashable {
case header
case photo(Photo)
}
private lazy var dataSource = makeDataSource()
var photos: AnyPublisher<[Photo], Never>!
override func viewDidLoad() {
super.viewDidLoad()
collectionView.collectionViewLayout = createLayout()
photos
.sink(receiveValue: applySnapshot)
.store(in: &subscriptions)
}
private func makeDataSource() -> UICollectionViewDiffableDataSource<Section, Row> {
return UICollectionViewDiffableDataSource(collectionView: collectionView) { (collectionView, indexPath, item) -> UICollectionViewCell? in
switch item {
case .header:
let cell = collectionView.dequeueReusableCell(for: indexPath, cellType: ProfileHeaderCollectionViewCell.self)
return cell
case .photo(let photo):
let cell = collectionView.dequeueReusableCell(for: indexPath, cellType: PhotoCollectionViewCell.self)
cell.configure(photo: photo)
return cell
}
}
}
private func applySnapshot(photos: [Photo]) {
var snapshot = NSDiffableDataSourceSnapshot<Section, Row>()
let headerSection = Section.header
snapshot.appendSections([headerSection])
snapshot.appendItems([.header], toSection: headerSection)
let items = photos.map { Row.photo([=10=]) }
let photosSection = Section.photos
snapshot.appendSections([photosSection])
snapshot.appendItems(items, toSection: photosSection)
dataSource.apply(snapshot, animatingDifferences: false)
}
private func createLayout() -> UICollectionViewLayout {
let layout = UICollectionViewCompositionalLayout { [unowned self] sectionIndex, _ in
// I have to work with a sectionIndex instead of a Section case?
}
return layout
}
}
不用在单独的数组中跟踪部分标识符,幸运的是您可以只使用 dataSource.snapshot().sectionIdentifiers[sectionIndex]
.