一旦我向CoreData添加新评论,我的UITableView
就会向上滚动。 NSFetchedResultsController知道它,将此注释放在表的底部,并将表滚动到顶部。这是正常的吗?
我的例子:
在我添加评论之前:
在我添加评论(不是预期的行为)之后:
它应该是这样的(手动轻扫后):
这可能与以下情况有关:
这是我的NSFetchedResultsController
:
NSSortDescriptor(key: "createdAt", ascending: false) //from the latest to the oldest
但我需要显示反向索引路径的注释(最新版本位于最底层)。换句话说,无论何时我需要使用indexPath
,我都会使用:
private func reversedIndexPathForIndexPath(indexPath: NSIndexPath) -> NSIndexPath {
return NSIndexPath(forRow: fetchedResultsController.fetchedObjects!.count - indexPath.row - 1, inSection: 0)
}
NSFetchedResultsControllerDelegate
:
//MARK: - NSFetchedResultsControllerDelegate
func controllerWillChangeContent(controller: NSFetchedResultsController) {
self.tableView.beginUpdates()
}
func controller(controller: NSFetchedResultsController, didChangeObject anObject: AnyObject, atIndexPath indexPath: NSIndexPath?, forChangeType type: NSFetchedResultsChangeType, newIndexPath: NSIndexPath?) {
switch type {
case .Insert:
if let newIndexPath = newIndexPath {
tableView.insertRowsAtIndexPaths([reversedIndexPathForIndexPath(newIndexPath)], withRowAnimation: .Fade)
}
case .Delete:
if let indexPath = indexPath {
tableView.deleteRowsAtIndexPaths([reversedIndexPathForIndexPath(indexPath)], withRowAnimation: .Fade)
}
case .Update:
if let indexPath = indexPath {
tableView.reloadRowsAtIndexPaths([reversedIndexPathForIndexPath(indexPath)], withRowAnimation: .Fade)
}
case .Move:
if let indexPath = indexPath, let newIndexPath = newIndexPath {
tableView.deleteRowsAtIndexPaths([reversedIndexPathForIndexPath(indexPath)], withRowAnimation: .Fade)
tableView.insertRowsAtIndexPaths([reversedIndexPathForIndexPath(newIndexPath)], withRowAnimation: .Fade)
}
}
}
func controllerDidChangeContent(controller: NSFetchedResultsController) {
tableView.endUpdates()
}
它与我的问题有关吗?
答案 0 :(得分:6)
编辑回答:
我能够通过以下步骤重现这个故障:
UITableView
,其rowHeight
设置为UITableViewAutomaticDimension
且非零estimatedRowHeight
。此行为的来源需要进一步调查。
目前唯一的解决方案是不使用UITableViewAutomaticDimension
并从tableView:heightForRowAtIndexPath:
返回行高。
顺便说一句,反向索引路径的实现有一个重大缺陷。当FRC调用它的委托方法controller:didChangeObject:atIndexPath:forChangeType:newIndexPath:
时,所有删除索引路径都属于更改前的数据集,而所有插入索引路径都属于更改后的路径。如果你在controllerWillChangeContent:
和controllerDidChangeContent:
之间,fetchedResultsController.fetchedObjects
将包含更改后的数据集,我想(但需要进行检查)。
答案 1 :(得分:3)
我找到了对我有用的东西。 我遇到了同样的问题,我的estimatedRowHeight是33,我的所有单元都大于33。
我刚刚将estimatedRowHeight更改为最大细胞高度(或大于最大细胞高度)
self.table.estimatedRowHeight = 310
self.table.rowHeight = UITableViewAutomaticDimension
像魅力一样工作,所有单元格现在都是自动调整的,当核心数据执行更新/删除/插入时,uitableview不会滚动到顶部或底部