scrollToRowAtIndexPath始终从顶部滚动而不是从当前偏移滚动

时间:2015-09-09 09:54:09

标签: ios swift uitableviewautomaticdimension

在我的UITableView我使用方法滚动到最后一个单元格:

WLNetworkClient.sharedClient().createCommentWIthText(commentTextView.text, forItem: item) { error in

    defer {
        UIAlertController.showAlertFromError(error)
    }

    if error == nil {
        self.fetchedResultsController.fetchRequest.fetchLimit += 1
        try! self.fetchedResultsController.performFetch()
        self.tableView.reloadData()
        self.scrollTableViewToBottom()
    }
}

private func scrollTableViewToBottom() {

    tableView.scrollToRowAtIndexPath(NSIndexPath(forRow: fetchedResultsController.fetchedObjects!.count - 1, inSection: 0), atScrollPosition: UITableViewScrollPosition.Bottom, animated: true)
}

但是这没有按预期工作,因为ALWAYS表从顶部滚动,即使我在第二个单元格上也是如此。如何解决这个问题?

5 个答案:

答案 0 :(得分:7)

dosdos回答在 Swift 2 中为我工作 添加次要更新

宣布ivar

var heightAtIndexPath = NSMutableDictionary()

在func viewDidLoad()

func viewDidLoad() {
  .... your code
  self.tableView.rowHeight = UITableViewAutomaticDimension
}

然后添加以下两种方法:

override func tableView(tableView: UITableView, estimatedHeightForRowAtIndexPath indexPath: NSIndexPath) -> CGFloat {
   let height = self.heightAtIndexPath.objectForKey(indexPath)
   if ((height) != nil) {
     return CGFloat(height!.floatValue)
   } else {
    return UITableViewAutomaticDimension
   }
 }

override func tableView(tableView: UITableView, willDisplayCell cell: UITableViewCell, forRowAtIndexPath indexPath: NSIndexPath) {
  let height = cell.frame.size.height
  self.heightAtIndexPath.setObject(height, forKey: indexPath)
}

SWIFT 3:

var heightAtIndexPath = NSMutableDictionary()

func tableView(_ tableView: UITableView, estimatedHeightForRowAt indexPath: IndexPath) -> CGFloat {
    let height = self.heightAtIndexPath.object(forKey: indexPath)
    if ((height) != nil) {
        return CGFloat(height as! CGFloat)
    } else {
        return UITableViewAutomaticDimension
    }
}

func tableView(_ tableView: UITableView, willDisplay cell: UITableViewCell, forRowAt indexPath: IndexPath) {
    let height = cell.frame.size.height
    self.heightAtIndexPath.setObject(height, forKey: indexPath as NSCopying)
}

答案 1 :(得分:1)

扩展ZaEeM ZaFaR很棒的答案:

目标C

NSMutableDictionary *heightAtIndexPath;
heightAtIndexPath = [[NSMutableDictionary alloc] init];

- (CGFloat)tableView:(UITableView *)tableView estimatedHeightForRowAtIndexPath:(NSIndexPath *)indexPath {
    NSNumber *height = [heightAtIndexPath objectForKey:indexPath];
    if(height) {
        return height.floatValue;
    } else {
        return UITableViewAutomaticDimension;
    }
}

- (void)tableView:(UITableView *)tableView willDisplayCell:(UITableViewCell *)cell forRowAtIndexPath:(NSIndexPath *)indexPath {
    [heightAtIndexPath setObject:@(cell.frame.size.height) forKey:indexPath];
}

答案 2 :(得分:0)

我认为问题是你在tableView更新之前滚动(self.tableView.reloadData())。 尝试延迟滚动:

     dispatch_async(dispatch_get_main_queue()) {
         self.scrollTableViewToBottom()
    }

或者强制tableView更新:

    self.tableView.reloadData()
    self.tableView.layoutIfNeeded()
    self.scrollTableViewToBottom()

答案 3 :(得分:0)

此问题包含在此处:How to tell when UITableVIew has completed ReloadData?

将您的代码更改为:

self.tableView.reloadData()
self.tableView.layoutIfNeeded()
self.scrollTableViewToBottom()

layoutIfNeeded强制表重新加载立即发生。所有这些都在链接中解释。

答案 4 :(得分:0)

为了更加轻松,我会简化上面介绍的Swift 3版本:

"