UITableViewCell 子类,错误的高度直到滚动

Posted

技术标签:

【中文标题】UITableViewCell 子类,错误的高度直到滚动【英文标题】:UITableViewCell subclass, wrong height until scroll 【发布时间】:2015-01-16 18:18:45 【问题描述】:

我有一个自定义的 uitableviewcell,我在其中使用了 UIImageView。我希望单元格能够根据图像视图的宽度,根据特定的纵横比调整其高度,但我不断遇到这些约束问题:

"<NSLayoutConstraint:0x7ffe13c14130 'UIView-Encapsulated-Layout-Height' V:[UITableViewCellContentView:0x7ffe117fb920(44)]>",
"<NSLayoutConstraint:0x7ffe13c15090 V:|-(0)-[UIView:0x7ffe13c11c10]   (Names: '|':UITableViewCellContentView:0x7ffe117fb920 )>",
"<NSLayoutConstraint:0x7ffe13c150e0 V:[UIView:0x7ffe13c11c10]-(0)-[UIButton:0x7ffe117e8110]>",
"<NSLayoutConstraint:0x7ffe13c15190 V:[UIButton:0x7ffe117e8110(30)]>",
"<NSLayoutConstraint:0x7ffe13c151e0 V:[UIButton:0x7ffe117e8110]-(0)-[UIView:0x7ffe13c11d00]>",
"<NSLayoutConstraint:0x7ffe13c15910 UIView:0x7ffe13c11d00.height == UIView:0x7ffe13c11c10.height>",
"<NSLayoutConstraint:0x7ffe13c15960 V:[UIView:0x7ffe13c11d00]-(0)-[UIButton:0x7ffe13c0b000]>",
"<NSLayoutConstraint:0x7ffe13c159b0 UIButton:0x7ffe13c0b000.height == UIButton:0x7ffe117e8110.height>",
"<NSLayoutConstraint:0x7ffe13c15a50 V:[UIButton:0x7ffe13c0b000]-(0)-[UIView:0x7ffe13c11df0]>",
"<NSLayoutConstraint:0x7ffe13c15aa0 UIView:0x7ffe13c11df0.height == UIView:0x7ffe13c11c10.height>",
"<NSLayoutConstraint:0x7ffe13c15af0 V:[UIView:0x7ffe13c11df0]-(0)-|   (Names: '|':UITableViewCellContentView:0x7ffe117fb920 )>"

我的约束是

    [self.contentView addConstraints:[NSLayoutConstraint constraintsWithVisualFormat:@"V:|[s1(>=10)][pictureView][s2(==s1)]|" options:0 metrics:nil views:dict]];
    [self.contentView addConstraints:[NSLayoutConstraint constraintsWithVisualFormat:@"V:|[s3][markAsFavorite(==30)][s4(==s3)][markForDeletion(==markAsFavorite)][s5(==s3)]|" options:0 metrics:nil views:dict]];
    [self.contentView addConstraints:[NSLayoutConstraint constraintsWithVisualFormat:@"H:|[s1][s3]|" options:0 metrics:nil views:dict]];
    [self.contentView addConstraints:[NSLayoutConstraint constraintsWithVisualFormat:@"H:|-10-[pictureView][s3]|" options:0 metrics:nil views:dict]];
    [self.contentView addConstraints:[NSLayoutConstraint constraintsWithVisualFormat:@"H:|[s2][s3]|" options:0 metrics:nil views:dict]];
    [self.contentView addConstraints:[NSLayoutConstraint constraintsWithVisualFormat:@"H:[pictureView]-10-[markAsFavorite(==30)]-10-|" options:0 metrics:nil views:dict]];
    [self.contentView addConstraints:[NSLayoutConstraint constraintsWithVisualFormat:@"H:[pictureView]-10-[markForDeletion(==30)]-10-|" options:0 metrics:nil views:dict]];
    [self.contentView addConstraints:[NSLayoutConstraint constraintsWithVisualFormat:@"H:[pictureView][s5]-10-|" options:0 metrics:nil views:dict]];
    [self.contentView addConstraints:[NSLayoutConstraint constraintsWithVisualFormat:@"H:[pictureView][s4]-10-|" options:0 metrics:nil views:dict]];


    photoHeight = [NSLayoutConstraint constraintWithItem:pictureView
                                               attribute:NSLayoutAttributeHeight
                                               relatedBy:0
                                                  toItem:nil
                                               attribute:NSLayoutAttributeNotAnAttribute
                                              multiplier:1
                                                constant:(5*pictureView.frame.size.width)/16];


    [self.contentView addConstraint:photoHeight];

这些约束只添加一次,在 cell 子类的 updateConstraints 方法中。在“drawRect”方法中,我更新了“photoHeight”常量并强制更新布局。

结果: 在第一次加载时,单元格看起来很糟糕,但在我滚动它们之后,它们会更新它们的高度并看起来像预期的那样。我认为问题在于首次加载单元格时未更新“photoHeight”常量,但考虑到它取决于图像视图的宽度,我不知道如何修复它。 你能帮忙解决这个问题吗?

我附上了一些截图: 滚动前

滚动后

正如您在第一个屏幕截图中看到的那样,图像超出了单元格内容视图,因为单元格对于该图像来说太小了......但是,在滚动后,单元格的大小被正确调整。我希望它们从一开始就调整大小,而不仅仅是在滚动之后。

【问题讨论】:

你在什么方法中添加了约束? -UITableViewCell 子类的updateConstraints 方法。 (我只在第一次调用此方法时添加它们) 我也有同样的问题。自从早期的 Xcode 6 beta 以来,我一直在以各种形式与这个相同的概念作斗争。你解决了吗? 【参考方案1】:

你需要使用:

- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath
    
    //calc height
    return height;

【讨论】:

我在 ios 8 中使用自定尺寸单元格:table.estimatedRowHeight = 120; table.rowHeight = UITableViewAutomaticDimension; 根据文档,单元格应自动评估约束并设置必要的高度 我附上了一些截图【参考方案2】:

我知道是什么导致了我的版本。

我在单元格的生命周期中分配值太晚了。当我将单元格出列时,我正在为单元格的模型设置一个属性,然后将 UIViews(在我的例子中为 UILabels)的分配推迟到layoutSubviews()。这最终导致了一个问题,即在我实际设置 UILabel 的文本之前,所有尺寸都已计算完毕。

我建议您在单元格出列时进行所有显示设置:

override func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath)

    var cell = tableView.dequeueReusableCellWithIdentifier("MyCustomCell", forIndexPath: indexPath) as MyCustomCell
    cell.setUpAllTheThings(myTableDataz[indexPath.row])
    return cell

【讨论】:

以上是关于UITableViewCell 子类,错误的高度直到滚动的主要内容,如果未能解决你的问题,请参考以下文章

动态 UITableViewCell 高度内的动态 UITextView?

出现删除按钮时重新计算自定义 UITableViewCell 的高度

如何更新 UITableViewCell 约束常量并更新单元格高度

从 nib 加载 UITableViewCell 子类时给出的类型错误

滚动时 UITableViewCell 自动布局抖动的动态单元格高度问题

UITableViewAutomaticDimension 的 UITableViewCell 高度错误