gpt4 book ai didi

ios - 自定义TableViewCell触发NSLayoutConstraint错误 'UIView-Encapsulated-Layout-Height'

转载 作者:搜寻专家 更新时间:2023-10-31 08:22:31 27 4
gpt4 key购买 nike

我对单元格使用自动高度。每次我想用新数据(这里是模型变量)更新我的单元格时,我更新了我的自动布局约束,但我得到了一个错误。这里只是为了说明问题,我什至没有更改约束。我只是要求重新计算布局。

在单元格的第一个初始化:没有警告,没问题。

错误:

<NSLayoutConstraint:0x174285d70 'UIView-Encapsulated-Layout-Height'
UITableViewCellContentView:0x1030f8a50.height == 500 (active)>

代码:

tableView.rowHeight = UITableViewAutomaticDimension

override func tableView(_ tableView: UITableView, estimatedHeightForRowAt indexPath: IndexPath) -> CGFloat {
return 500
}

class TestTableViewCell: UITableViewCell {

var model: X? {
didSet {
setNeedsLayout()
layoutIfNeeded()
}
}

override func awakeFromNib() {
super.awakeFromNib()
// Initialization code
}

override func setSelected(_ selected: Bool, animated: Bool) {
super.setSelected(selected, animated: animated)

// Configure the view for the selected state
}

override init(style: UITableViewCellStyle, reuseIdentifier: String?) {
super.init(style: style, reuseIdentifier: reuseIdentifier)

let viewtop = UIView()
let viewbottom = UIView()

viewtop.backgroundColor = UIColor.yellow
viewbottom.backgroundColor = UIColor.red
contentView.backgroundColor = UIColor.blue

contentView.addSubview(viewtop)
contentView.addSubview(viewbottom)

viewtop.snp.makeConstraints { (make) in
make.top.equalTo(contentView)
make.left.right.equalTo(contentView)
make.height.equalTo(50)
}

viewbottom.snp.makeConstraints { (make) in
make.top.equalTo(viewtop.snp.bottom)
make.left.right.equalTo(contentView)
make.bottom.equalTo(contentView)
make.height.equalTo(120)
}
}

required init?(coder aDecoder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}

}

问题:为什么在询问相同约束的重新布局后,我会收到错误消息?

编辑:这里是另一个更好理解的例子。

var botViewHeightConstraint:Constraint!
class TestTableViewCell: UITableViewCell {

var model: Int? {
didSet {
if model == 1 {
botViewHeightConstraint.update(offset:200)
}else{
botViewHeightConstraint.update(offset:120)
}
}
}

override func awakeFromNib() {
super.awakeFromNib()
// Initialization code
}

override func setSelected(_ selected: Bool, animated: Bool) {
super.setSelected(selected, animated: animated)

// Configure the view for the selected state
}

override init(style: UITableViewCellStyle, reuseIdentifier: String?) {
super.init(style: style, reuseIdentifier: reuseIdentifier)

let viewtop = UIView()
let viewbottom = UIView()

viewtop.backgroundColor = UIColor.yellow
viewbottom.backgroundColor = UIColor.red
contentView.backgroundColor = UIColor.blue

contentView.addSubview(viewtop)
contentView.addSubview(viewbottom)

viewtop.snp.makeConstraints { (make) in
make.top.equalTo(contentView)
make.left.right.equalTo(contentView)
make.height.equalTo(50)
}

viewbottom.snp.makeConstraints { (make) in
make.top.equalTo(viewtop.snp.bottom)
make.left.right.equalTo(contentView)
make.bottom.equalTo(contentView)
botViewHeightConstraint = make.height.equalTo(120).constraint
}
}

required init?(coder aDecoder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}

}

CellForRow代码:

 override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
if let post = fetchedResultsController?.fetchedObjects?[(indexPath as NSIndexPath).section] {
let cell = tableView.dequeueReusableCell(withIdentifier: imagePostCellId) as! TestTableViewCell!
cell.model = 1
return cell
}
}

最佳答案

首先,您总是忽略将 subview 的 translatesAutoresizingMaskIntoConstraints 设置为 false。这很重要。 (不过,也许 snapkit 会为您做到这一点。我不知道。)

其次——这是重点——你所做的并不是从内到外调整可变高度单元格的大小。您不会像您的代码那样更改绝对高度约束。大小最终基于 subview 的内部内容大小。当然,某些 subview 可能具有绝对高度,但最终必须至少有一个具有固有内容大小。 这是您可以动态更改以确定单元格高度的大小。

这就是为什么,例如,包含 UILabel 的单元格很容易与动态行高一起使用。它具有固有的内容大小。

固有内容大小与单元格的内置高度(控制台转储中的 UIView-Encapsulated-Layout-Height)冲突;它补充它(当运行时在幕后调用 systemLayoutSizeFitting(UILayoutFittingCompressedSize) 时,这就是自动可变行高的工作方式)。

如果您使用带有 intrinsicContentSize 实现的自定义 subview ,并且如果在单元格中设置模型值会触发对 invalidateIntrinsicContentSize 的调用,则您的示例将完美运行,无需控制台中的投诉。

以下是此类自定义 View 的示例:

class MyView : UIView {
var h : CGFloat = 200 {
didSet {
self.invalidateIntrinsicContentSize()
}
}
override var intrinsicContentSize: CGSize {
return CGSize(width:300, height:self.h)
}
}

当这是单元格内容 View 的 subview 时,在 cellForRow 中设置此 View 的 h 可正确调整单元格的高度。

例如,假设我们单元格的内容 View 只有一个 subview v,它是一个 MyView。然后:

override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
let cell = tableView.dequeueReusableCell(withIdentifier: "Cell", for: indexPath) as! MyCell
let even = indexPath.row % 2 == 0
cell.v.backgroundColor = even ? .red : .green
cell.v.h = even ? 40 : 80 // triggers layout!
return cell
}

关于ios - 自定义TableViewCell触发NSLayoutConstraint错误 'UIView-Encapsulated-Layout-Height',我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39656281/

27 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com