iOS UITableView 向上滚动时滚动位置跳跃

Posted

技术标签:

【中文标题】iOS UITableView 向上滚动时滚动位置跳跃【英文标题】:iOS UITableView scroll position jumps when scrolling up 【发布时间】:2015-03-31 07:40:18 【问题描述】:

这就是我想要做的。 我有一个UITableViewCell,可以说固定高度为 300(它实际上是一个可变大小的高度,但我试图简化示例)

我想要实现的是,当我向上滚动时 - 我将有一个“缩略图”版本的单元格 - 高度为 75

我设法做到了,但现在的问题是,当我向上滚动时,之前的单元格高度会被调整,一旦单元格尺寸变小,滚动位置就会“跳跃”,这会导致视图“向下跳跃”当他向上滚动时。

如何调整?

代码:

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath

    UITableViewCell *cell;
    if (indexPath.row < lastViewedChapter)
    
        cell = [self generateChapterCell:tableView indexPath:indexPath collapsed:YES];
    
    else
    
        cell = [self generateChapterCell:tableView indexPath:indexPath collapsed:NO];
        if (indexPath.row > lastViewedChapter)
        
            lastViewedChapter = indexPath.row;
        
    
    return cell;


- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath

    if (indexPath.row < lastViewedChapter)
    
        return 73;
    
    else
    
        return 300;  //actually here is a code that calculates the height
    

【问题讨论】:

这是一段时间以来我在 SO 上看到的最酷的东西之一。因此,当您说视图“跳回”时,您的意思是 tableview 的内容,即单元格突然向上跳?因为那是我认为应该发生的事情。 我认为我们在谈论同样的事情。当您向上滚动时,您会到达可见单元格的边缘,然后重新生成顶部单元格,导致滚动返回到可见单元格中间的某个位置。用户体验真的很差——怎么办? 我操纵了委托函数来获得你需要的东西。检查一下,让我知道它是否适合你。 【参考方案1】:

您已经降低了上部单元格的高度,然后其他单元格向上移动以填充该空间,而您仍在向右滚动?

当你改变单元格的高度时,尝试设置新的 tableView.contentOffset。

在您的情况下,当您将单元格的高度返回为 73 时, contentOffset.y 应该是 (old contentOffset.y - (300 - 73))。

我没有对此进行测试,但我认为它可能会有所帮助,并且您也必须为其他情况计算新的 contentOffset(向下滚动时,当表格重新加载数据时)。

static NSInteger _lastRow = -1;

- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath 
    if (_lastRow == -1) 
        _lastRow = indexPath.row;
        return 300;
     else 
        if (_lastRow > indexPath.row) 
            _lastRow = indexPath.row;
            if ([tableView rectForRowAtIndexPath:indexPath].size.height == 300) 
                [tableView setContentOffset:CGPointMake(tableView.contentOffset.x, (tableView.contentOffset.y - (300 - 73)))];
            
            return 73;
         else 
            _lastRow = indexPath.row;
            return 300;
        
    

这段代码工作正常,但仍有一些错误(第一次加载数据时的第一行高度就像你向上滚动一次,当你快速向上滚动到顶部时它不会正常反弹)但我希望这对你有帮助.

【讨论】:

OMG...所以我必须保存“何时”折叠单元格的状态? (因为到目前为止我只需要最后一个没有折叠的单元格索引) 你的意思是旧的内容偏移吗?我的旧 contentOffset 是这样的当前 contentOffset。 tableView.contentOffset = CGPointMake(tableView.contentOffset.x, (tableView.contentOffset.y - (300 - 73)));这段代码应该放在return 73之前;我认为在你的代码中。 我的意思是我需要保存从 300 切换到 73 的状态(点击时反之亦然)。到目前为止,我只有一个单元格的状态(折叠/未折叠) - 但它可以重绘多次 - 所以这还不够 滚动是跳转的,因为在调整单元格大小时 contentOffset 发生了变化。所以你应该设置新的 contentOffset 来强制滚动跳回正确的位置,但是你不会看到它上下跳动,因为它太快了。 好的 - 我会尝试描述它 - 我向上滚动 - 单元格被缩小了。凉爽的。现在我向下滚动,然后再次向上滚动 - 它们保持相同的大小(缩小)。在这种情况下,我不能碰 contentOffset... 你明白我的意思了吗?我需要每个单元格的“增量状态”【参考方案2】:

这是肯定会发生的,因为您更改了单元格高度。

问题是如何减轻这种糟糕的用户体验。

UITableViewUIScrollView 的子类。 UIScrollViews 提供在 UITableView 类中也可用的委托。

执行以下操作。

self.tableView.delegate = self;

然后实现下面的函数。在下文中,位置是在您的标头中定义的CGPoint 变量。

-(void)tableView:(UITableView *)tableView willDisplayCell:(UITableViewCell *)cell forRowAtIndexPath:(NSIndexPath *)indexPath

    location = tableView.contentOffset;


-(void)tableView:(UITableView *)tableView didEndDisplayingCell:(UITableViewCell *)cell forRowAtIndexPath:(NSIndexPath *)indexPath

    CGPoint newLocation = tableView.contentOffset;
    if (CGPointEqualToPoint(location, newLocation))
    
        NSLog(@"are equal");
        tableView.contentOffset = CGPointMake(location.x, location.y-227);
    

【讨论】:

以上是关于iOS UITableView 向上滚动时滚动位置跳跃的主要内容,如果未能解决你的问题,请参考以下文章

在 UITableView 中滚动时缩放图像

iOS Swift ScrollView、UITableView、滚动区域

滚动时禁用 UITableView 垂直反弹

iOS开发之视差滚动视图

UITableview 向上滚动 UISectionHeader 的一半

UITableView 仅在向上滚动时更新,而不是向下滚动