如何在不滚动表格的情况下将行插入到UITableView的顶部?

如何在不滚动表格的情况下将行插入到UITableView的顶部? 就像当你拖动来在twitter上重新加载新消息或在whatsApp消息时一样

我试着做另一个开始/ endUpdate部分要求它滚动到第三个索引,但它没有工作。 它总是滚动到tableView的顶部。 我希望它保持在添加新行之前的相同位置。

self.tableView.beginUpdates() let indexPath = IndexPath(row: 3, section: 0) self.tableView.scrollToRow(at: indexPath, at: .top, animated: false) self.tableView.endUpdates() 

PS这是在我的真实项目中的示例项目,单元格的大小根据消息大小而变化。

这是该项目的链接: https : //github.com/akawther/TableView

 import UIKit class ViewController: UIViewController, UITableViewDelegate, UITableViewDataSource { @IBOutlet weak var tableView: UITableView! var loadMore: UIRefreshControl! var labels = ["A","B","C","D","E","F","G","H","I","J","K"] override func viewDidLoad() { super.viewDidLoad() self.loadMore = UIRefreshControl() self.loadMore.attributedTitle = NSAttributedString(string: "Pull to reload more") self.loadMore.addTarget(self, action: #selector(ViewController.loadMoreChats), for: .valueChanged) self.tableView.addSubview(self.loadMore) // Do any additional setup after loading the view, typically from a nib. } func loadMoreChats() { DispatchQueue.main.async(execute: {() -> Void in self.tableView.beginUpdates() self.labels.insert("3", at: 0) self.labels.insert("2", at: 0) self.labels.insert("1", at: 0) let indexPaths = [ IndexPath(row: 0, section: 0), IndexPath(row: 1, section: 0), IndexPath(row: 2, section: 0), ] self.tableView.insertRows(at: indexPaths, with: .top) self.tableView.endUpdates() self.loadMore.endRefreshing() self.tableView.beginUpdates() let indexPath = IndexPath(row: 3, section: 0) self.tableView.scrollToRow(at: indexPath, at: .top, animated: false) self.tableView.endUpdates() }) } override func didReceiveMemoryWarning() { super.didReceiveMemoryWarning() // Dispose of any resources that can be recreated. } func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int { return labels.count } func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell { let cell = self.tableView.dequeueReusableCell(withIdentifier: "cell") as! customCellTableViewCell cell.label.text = labels[indexPath.row] return cell } } 

更新#1:我刚刚尝试过,也围绕我的第二个/ CATransaction更新,但仍然是顶部显示是1而不是像我想要的:

  CATransaction.begin() self.tableView.beginUpdates() CATransaction.setCompletionBlock { () -> Void in let indexPath = IndexPath(row: 3, section: 0) self.tableView.scrollToRow(at: indexPath, at: .top, animated: false) } self.tableView.endUpdates() CATransaction.commit() 

更新#2: github项目更新了@beyowulf的答案

loadMoreChats是一个button的目标时,这段代码对我来说工作的很好:

  func loadMoreChats() { self.loadMore.endRefreshing() self.labels.insert("3", at: 0) self.labels.insert("2", at: 0) self.labels.insert("1", at: 0) self.tableView.reloadData() let indexPath = IndexPath(row: 3, section: 0) self.tableView.scrollToRow(at: indexPath, at: .top, animated: false) } 

使用刷新控件的情况下, scrollToRow被触发它的平移手势覆盖。

更新

请注意,您引用的应用程序不以这种方式使用刷新控制器,但是如果您坚持使用一个,我应该可以在编程方式滚动表视图时暂时禁用平移手势。 就像是:

 func loadMoreChats() { self.loadMore.endRefreshing() self.labels.insert("3", at: 0) self.labels.insert("2", at: 0) self.labels.insert("1", at: 0) self.tableView.reloadData() let indexPath = IndexPath(row: 3, section: 0) self.tableView.scrollToRow(at: indexPath, at: .top, animated: false) self.tableView.panGestureRecognizer.isEnabled = false let when = DispatchTime.now() + 0.05 DispatchQueue.main.asyncAfter(deadline: when) { self.tableView.panGestureRecognizer.isEnabled = true } } 

我觉得这是相当黑客,并会build议反对它。