- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在尝试理解 UICollectionViewDiffableDataSource
和 NSDiffableDataSourceSnapshot
。
我在下面创建了一个非常粗糙的版本。基本上在加载时它应该获取照片。
点击导航栏中的按钮,它会获取下一页。然而,这只是替换现有数据,我期望它将值附加到数组。
我应该如何更新我的数据而不是替换它?
class ViewController: UIViewController {
lazy var collectionView = UICollectionView(frame: view.frame, collectionViewLayout: UICollectionViewFlowLayout())
enum Section {
case main
}
typealias DataSource = UICollectionViewDiffableDataSource<Section, AnyHashable>
typealias DataSourceSnapshot = NSDiffableDataSourceSnapshot<Section, AnyHashable>
private var dataSource: DataSource!
private var snapshot = DataSourceSnapshot()
override func viewDidLoad() {
super.viewDidLoad()
let updateButton = UIBarButtonItem(title: "Add", style: .plain, target: self, action: #selector(onTapLoad))
navigationItem.rightBarButtonItem = updateButton
view.addSubview(collectionView)
collectionView.autoresizingMask = [.flexibleHeight, .flexibleWidth]
collectionView.delegate = self
collectionView.backgroundColor = .systemBackground
collectionView.register(PhotoCell.self, forCellWithReuseIdentifier: "PhotoCell")
dataSource = DataSource(collectionView: collectionView, cellProvider: { (cv, indexPath, object) -> PhotoCell? in
if let object = object as? Photo {
let cell = cv.dequeueReusableCell(withReuseIdentifier: "PhotoCell", for: indexPath) as! PhotoCell
cell.backgroundColor = indexPath.item % 2 == 0 ? .systemTeal : .systemPink
cell.label.text = object.title
return cell
}
return nil
})
load()
}
@objc func onTapLoad() {
load(page: 1)
}
func load(page: Int = 0) {
PhotoLoader.shared.load(page: page) { result in
if let photos = try? result.get() {
self.apply(photos)
}
}
}
func apply(_ photos: [Photo]) {
snapshot = DataSourceSnapshot()
snapshot.appendSections([Section.main])
snapshot.appendItems(photos)
dataSource.apply(snapshot, animatingDifferences: false)
}
}
extension ViewController: UICollectionViewDelegateFlowLayout {
func collectionView(_ collectionView: UICollectionView, layout collectionViewLayout: UICollectionViewLayout, sizeForItemAt indexPath: IndexPath) -> CGSize {
return .init(width: collectionView.frame.width - 32, height: 100)
}
}
struct Photo: Decodable, Hashable {
let id = UUID()
let title: String
}
final class PhotoLoader {
static let shared = PhotoLoader()
func load(page: Int, completion: @escaping (Result<[Photo], Error>) -> Void) {
URLSession.shared.dataTask(with: URL(string: "https://jsonplaceholder.typicode.com/photos?_start=\(page)&_limit=5")!, completionHandler: { data, response, error in
if let data = data, let model = try? JSONDecoder().decode([Photo].self, from: data) {
completion(.success(model))
}
}).resume()
}
}
final class PhotoCell: UICollectionViewCell {
lazy var label = UILabel(frame: .zero)
override init(frame: CGRect) {
super.init(frame: frame)
label.translatesAutoresizingMaskIntoConstraints = false
label.numberOfLines = 3
addSubview(label)
NSLayoutConstraint.activate([
label.topAnchor.constraint(equalTo: topAnchor),
label.leadingAnchor.constraint(equalTo: leadingAnchor),
label.bottomAnchor.constraint(equalTo: bottomAnchor),
label.trailingAnchor.constraint(equalTo: trailingAnchor)
])
}
required init?(coder: NSCoder) {
return nil
}
}
最佳答案
在不使用额外数组的情况下初始化 viewDidLoad
中的快照
override func viewDidLoad() {
super.viewDidLoad()
...
let snapshot = DataSourceSnapshot()
snapshot.appendSections([Section.main])
dataSource.apply(snapshot, animatingDifferences: false)
}
并在apply(_ photos)
将照片附加到当前快照而不是创建新快照
func apply(_ photos: [Photo]) {
var snapshot = dataSource.snapshot()
snapshot.appendItems(photos, toSection: .main)
dataSource.apply(snapshot, animatingDifferences: true)
}
不需要该属性
<罢工> 私有(private) var 快照 = DataSourceSnapshot()注意:
尽可能具体地声明数据源
typealias DataSource = UICollectionViewDiffableDataSource<Section, Photo>
typealias DataSourceSnapshot = NSDiffableDataSourceSnapshot<Section, Photo>
然后你就摆脱了不必要的类型检查
dataSource = DataSource(collectionView: collectionView, cellProvider: { (cv, indexPath, photo) -> PhotoCell? in
let cell = cv.dequeueReusableCell(withReuseIdentifier: "PhotoCell", for: indexPath) as! PhotoCell
cell.backgroundColor = indexPath.item % 2 == 0 ? .systemTeal : .systemPink
cell.label.text = photo.title
return cell
})
关于ios - UICollectionViewDiffableDataSource 正在替换数据而不是更新,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/63162333/
我正在尝试理解 UICollectionViewDiffableDataSource 和 NSDiffableDataSourceSnapshot。 我在下面创建了一个非常粗糙的版本。基本上在加载时它
在使用传统的UICollectionView时,我们经常使用如下代码来处理空状态: func collectionView(_ collectionView: UICollectionView, nu
我创建了以下演示 View Controller 以在最小示例中重现该问题。 在这里,我使用 UICollectionViewDiffableDataSource 将相同数据的快照重复应用到相同的集合
我正在使用 UICollectionViewDiffableDataSource 来填充我的 UICollectionView。通过 REST API 收到项目列表后,我创建了一个新快照并像这样应用它
我采用了新的 UICollectionViewDiffableDataSource。每次删除项目时,我都会应用数据源快照: var snapshot = NSDiffableDataSourceSna
我正在构建一个带有 Collection View 的 View Controller ,使用(有点)新的可区分数据源和可组合的 Collection View 布局。我制作了自己的 Section
我有一个 UICollectionView,其中我使用 UICollectionViewCompositionalLayout 布置了两个部分(一个水平滚动,一个垂直滚动)。我将其与 UICollec
我正在尝试将 iOS 13 中内置的一些新差异类与 Core Data 一起使用。我遇到的问题是 controllerdidChangeContentWith没有按预期工作。它传递给我一个快照引用,它
我正在使用 UICollectionViewDiffableDataSource对于我的 collectionView 的数据源。我的 CollectionView 中有 3 个部分: enum Se
我正在使用 UICollectionViewDiffableDataSource和一个 NSFetchedResultsController填充我的 UICollectionView在我的 UIVie
我正在尝试实现 UICollectionViewDiffableDataSource我的 collectionView .我的代码编译得很好,但是我第一次对它应用快照时一直遇到这个错误,并出现以下错误
我正在使用 UICollectionView 和 UICollectionViewDiffableDataSource 并异步和同步获取数据以填充 View 。应用程序崩溃并显示以下消息:“由于未捕获
我需要使用 moveItem(at:to:) 委托(delegate)方法来移动我的 Collection View 单元格。当我实现数据源委托(delegate)方法时它工作正常,但我需要将我的数据
我是一名优秀的程序员,十分优秀!