小编典典

UITableView布局在推送和返回时弄乱了。(iOS 8,Xcode beta 5,Swift)

swift

tldr; 自动约束似乎在推送选择时中断,并返回查看自定义单元格

编辑 :我提供了一个github示例项目,展示了发生的错误 https://github.com/Matthew-
Kempson/TableViewExample.git

我正在创建一个需要自定义UITableCell的标题标签的应用程序,以允许根据帖子标题的长度来更改行。单元格可以正确加载到视图中,但是如果我按某个单元格以推送方式将帖子加载到包含WKWebView的视图中,如屏幕截图所示,单元格会立即移动到错误的位置。通过UINavigationController的后退按钮将视图加载回视图时,也可以查看此视图​​。

在这个特定示例中,我按下了最后一个单元格,标题为“我在巴黎拍了照片的两个哥们”,然后正确加载了所有内容。然后,如下面的屏幕快照所示,在加载第二个视图的背景下,所有单元都出于未知原因向上移动。然后,当我重新加载视图时,您可以看到屏幕略微向上移动,而我实际上无法滚动到比显示的更低的水平。当视图加载回去时,这似乎是随机的,与其他测试一样,底部单元格下方有空白不会消失。

我还提供了一张包含单元格约束的图片。

图像(显然,在这个问题上提供图像,我需要更高的声誉,因此它们在imgur相册中):http
://imgur.com/a/gY87E

我的代码:

自定义单元格中的方法,允许该单元格在旋转时正确调整视图的大小:

override func layoutSubviews() {
    super.layoutSubviews()

    self.contentView.layoutIfNeeded()

    // Update the label constaints
    self.titleLabel.preferredMaxLayoutWidth = self.titleLabel.frame.width
    self.detailsLabel.preferredMaxLayoutWidth = self.detailsLabel.frame.width
}

表格视图中的代码

override func viewDidLoad() {
    super.viewDidLoad()

    // Create and register the custom cell
    self.tableView.estimatedRowHeight = 56
    self.tableView.rowHeight = UITableViewAutomaticDimension
}

创建单元的代码

    override func tableView(tableView: UITableView!, cellForRowAtIndexPath indexPath: NSIndexPath!) -> UITableViewCell! {
    if let cell = tableView.dequeueReusableCellWithIdentifier("LinkCell", forIndexPath: indexPath) as? LinkTableViewCell {

        // Retrieve the post and set details
        let link: Link = self.linksArray.objectAtIndex(indexPath.row) as Link

        cell.titleLabel.text = link.title
        cell.scoreLabel.text = "\(link.score)"
        cell.detailsLabel.text = link.stringCreatedTimeIntervalSinceNow() + " ago by " + link.author + " to /r/" + link.subreddit

        return cell
    }

    return nil
}

如果您需要更多代码或信息,请询问,我将提供必要的信息

谢谢你的帮助!


阅读 275

收藏
2020-07-07

共1个答案

小编典典

此行为的问题是,当您按顺序时,tableView将调用estimatedHeightForRowAtIndexPath可见单元格并将单元格高度重置为默认值。这发生在viewWillDisappear通话后。如果返回TableView,则所有可见的单元都被弄乱了。

我用来解决了这个问题estimatedCellHeightCache。我只是将此代码片段添加到cellForRowAtIndexPath方法中:

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
    ...
    // put estimated cell height in cache if needed
    if (![self isEstimatedRowHeightInCache:indexPath]) {
        CGSize cellSize = [cell systemLayoutSizeFittingSize:CGSizeMake(self.view.frame.size.width, 0) withHorizontalFittingPriority:1000.0 verticalFittingPriority:50.0];
        [self putEstimatedCellHeightToCache:indexPath height:cellSize.height];
    }
    ...
}

现在,您必须实现estimatedHeightForRowAtIndexPath以下内容:

-(CGFloat)tableView:(UITableView *)tableView estimatedHeightForRowAtIndexPath:(NSIndexPath *)indexPath {
    return [self getEstimatedCellHeightFromCache:indexPath defaultHeight:41.5];
}

配置缓存

将此属性添加到您的.h文件中:

@property NSMutableDictionary *estimatedRowHeightCache;

实现放置/获取/重置缓存的方法:

#pragma mark - estimated height cache methods

// put height to cache
- (void) putEstimatedCellHeightToCache:(NSIndexPath *) indexPath height:(CGFloat) height {
    [self initEstimatedRowHeightCacheIfNeeded];
    [self.estimatedRowHeightCache setValue:[[NSNumber alloc] initWithFloat:height] forKey:[NSString stringWithFormat:@"%d", indexPath.row]];
}

// get height from cache
- (CGFloat) getEstimatedCellHeightFromCache:(NSIndexPath *) indexPath defaultHeight:(CGFloat) defaultHeight {
    [self initEstimatedRowHeightCacheIfNeeded];
    NSNumber *estimatedHeight = [self.estimatedRowHeightCache valueForKey:[NSString stringWithFormat:@"%d", indexPath.row]];
    if (estimatedHeight != nil) {
        //NSLog(@"cached: %f", [estimatedHeight floatValue]);
        return [estimatedHeight floatValue];
    }
    //NSLog(@"not cached: %f", defaultHeight);
    return defaultHeight;
}

// check if height is on cache
- (BOOL) isEstimatedRowHeightInCache:(NSIndexPath *) indexPath {
    if ([self getEstimatedCellHeightFromCache:indexPath defaultHeight:0] > 0) {
        return YES;
    }
    return NO;
}

// init cache
-(void) initEstimatedRowHeightCacheIfNeeded {
    if (self.estimatedRowHeightCache == nil) {
        self.estimatedRowHeightCache = [[NSMutableDictionary alloc] init];
    }
}

// custom [self.tableView reloadData]
-(void) tableViewReloadData {
    // clear cache on reload
    self.estimatedRowHeightCache = [[NSMutableDictionary alloc] init];
    [self.tableView reloadData];
}
2020-07-07