Глюк
Я использую CoreData
с NSFetchResultController
, чтобы данные отображались в UITableView
. У меня есть одна проблема: UITableView
изменяет contentOffSet.y
при вставке / перемещении / удалении новой строки. Когда пользователь выполнил прокрутку до, например, в середине UITableView
отскакивает, когда вставляется новая строка.
Проект репродукции
Эта ссылка github на проект, который содержит минимальный код для воспроизведения этого поведения: https://github.com/Jasperav/FetchResultControllerGlitch (код также ниже)
Это показывает глюк. Я стою в середине моего UITableView
и постоянно вижу, как вставляются новые строки, независимо от текущего contentOffSet.y
.:
Похожие вопросы
Беспокойство
Я также попытался переключиться на performBatchUpdates
вместо begin/endUpdates
, что тоже не сработало.
UITableView
просто не должен перемещаться при вставке / удалении / перемещении строк, когда эти строки не видны пользователю . Я ожидаю, что что-то вроде этого просто должно работать из коробки.
Конечная цель
Это то, что я в конечном итоге хочу (просто репликация экрана чата в WhatsApp):
- Когда пользователь полностью прокручивается до вершины (для WhatsApp это низ), куда вставляются новые строки,
UITableView
должен анимировать новую вставленную строку и изменять текущую contentOffSet.y
.
- Когда пользователь не полностью прокручивает вверх (или вниз, в зависимости от того, где вставляются новые строки), ячейки, которые видит пользователь, не должны подпрыгивать при вставке новой строки. Это действительно плохо для пользовательского опыта приложения.
- Должно работать для динамических ячеек высоты.
- Я также вижу это поведение при перемещении / удалении ячеек. Есть ли какое-нибудь простое исправление для всех глюков здесь?
Если бы UICollectionView
было бы лучше, это было бы хорошо.
Вариант использования
Я пытаюсь скопировать экран чата WhatsApp. Я не уверен, что они используют NSFetchResultController, но, кроме того, конечной целью является предоставление им точного пользовательского опыта. Поэтому вставка, перемещение, удаление и обновление ячеек должны выполняться так, как это делает WhatsApp. Итак, для рабочего примера: перейдите в WhatsApp, для неработающего примера: загрузите проект.
Скопировать код вставки
Код (ViewController.swift):
import CoreData
import UIKit
class ViewController: UIViewController, NSFetchedResultsControllerDelegate, UITableViewDataSource, UITableViewDelegate {
let tableView = MyTableView()
let resultController = ViewController.createResultController()
override func viewDidLoad() {
super.viewDidLoad()
// Initial cells
for i in 0...40 {
let x = SomeEntity(context: CoreDataContext.persistentContainer.viewContext)
x.something = randomString(length: i + 1)
x.date = Date()
x.height = Float.random(in: 50...100)
}
Timer.scheduledTimer(withTimeInterval: 1, repeats: true) { (_) in
let x = SomeEntity(context: CoreDataContext.persistentContainer.viewContext)
x.something = self.randomString(length: Int.random(in: 10...50))
x.date = Date()
x.height = Float.random(in: 50...100)
}
resultController.delegate = self
view.addSubview(tableView)
tableView.translatesAutoresizingMaskIntoConstraints = false
tableView.topAnchor.constraint(equalTo: view.safeAreaLayoutGuide.topAnchor).isActive = true
tableView.bottomAnchor.constraint(equalTo: view.safeAreaLayoutGuide.bottomAnchor).isActive = true
tableView.leadingAnchor.constraint(equalTo: view.safeAreaLayoutGuide.leadingAnchor).isActive = true
tableView.trailingAnchor.constraint(equalTo: view.safeAreaLayoutGuide.trailingAnchor).isActive = true
tableView.delegate = self
tableView.dataSource = self
tableView.estimatedRowHeight = 75
try! resultController.performFetch()
}
public func controller(_ controller: NSFetchedResultsController<NSFetchRequestResult>, didChange anObject: Any, at indexPath: IndexPath?, for type: NSFetchedResultsChangeType, newIndexPath: IndexPath?) {
switch type {
case .insert:
tableView.insertRows(at: [newIndexPath!], with: .automatic)
case .delete:
tableView.deleteRows(at: [indexPath!], with: .automatic)
case .move:
tableView.deleteRows(at: [indexPath!], with: .automatic)
tableView.insertRows(at: [newIndexPath!], with: .automatic)
case .update:
tableView.moveRow(at: indexPath!, to: newIndexPath!)
}
}
func controllerWillChangeContent(_ controller: NSFetchedResultsController<NSFetchRequestResult>) {
tableView.beginUpdates()
}
public func controllerDidChangeContent(_ controller: NSFetchedResultsController<NSFetchRequestResult>) {
tableView.endUpdates()
}
func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
return resultController.fetchedObjects?.count ?? 0
}
func tableView(_ tableView: UITableView, heightForRowAt indexPath: IndexPath) -> CGFloat {
return CGFloat(resultController.object(at: indexPath).height)
}
func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
let cell = tableView.dequeueReusableCell(withIdentifier: "cell", for: indexPath) as! MyTableViewCell
cell.textLabel?.text = resultController.object(at: indexPath).something
return cell
}
private static func createResultController() -> NSFetchedResultsController<SomeEntity> {
let fetchRequest: NSFetchRequest<SomeEntity> = SomeEntity.fetchRequest()
fetchRequest.sortDescriptors = [NSSortDescriptor(key: "date", ascending: false)]
return NSFetchedResultsController(fetchRequest: fetchRequest, managedObjectContext: CoreDataContext.persistentContainer.viewContext, sectionNameKeyPath: nil, cacheName: nil)
}
func randomString(length: Int) -> String {
let letters = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789"
return String((0...length-1).map{ _ in letters.randomElement()! })
}
}
class MyTableView: UITableView {
init() {
super.init(frame: .zero, style: .plain)
register(MyTableViewCell.self, forCellReuseIdentifier: "cell")
}
required init?(coder aDecoder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}
}
class MyTableViewCell: UITableViewCell {
}
class CoreDataContext {
static let persistentContainer: NSPersistentContainer = {
let container = NSPersistentContainer(name: "FetchViewControllerGlitch")
container.loadPersistentStores(completionHandler: { (nsPersistentStoreDescription, error) in
guard let error = error else {
return
}
fatalError(error.localizedDescription)
})
return container
}()
}