自动布局约束警告

Posted

技术标签:

【中文标题】自动布局约束警告【英文标题】:Autolayout constraint warnings 【发布时间】:2018-10-09 14:00:23 【问题描述】:

所以我以编程方式将视图添加到 tableviewcell 的内容视图中,并使用 tableview 提供的动态高度,我似乎收到以下自动布局警告,我是否应该担心,因为布局似乎没有搞砸但是有一个充满这些警告的控制台很烦人。

Make a symbolic breakpoint at UIViewAlertForUnsatisfiableConstraints to catch this in the debugger.
The methods in the UIConstraintBasedLayoutDebugging category on UIView listed in <UIKitCore/UIView.h> may also be helpful.
2018-10-09 14:51:46.748606+0100 GenericForms[78197:5088471] [LayoutConstraints] Unable to simultaneously satisfy constraints.
    Probably at least one of the constraints in the following list is one you don't want. 
    Try this: 
        (1) look at each constraint and try to figure out which you don't expect; 
        (2) find the code that added the unwanted constraint or constraints and fix it. 
(
    "<NSLayoutConstraint:0x600001005fe0 UIView:0x7fa582d162e0.height == 55   (active)>",
    "<NSLayoutConstraint:0x600001018960 V:|-(20)-[UIView:0x7fa582d162e0]   (active, names: '|':UITableViewCellContentView:0x7fa582d15800 )>",
    "<NSLayoutConstraint:0x60000101c910 UIView:0x7fa582d162e0.bottom == UITableViewCellContentView:0x7fa582d15800.bottom - 20   (active)>",
    "<NSLayoutConstraint:0x60000103d950 'UIView-Encapsulated-Layout-Height' UITableViewCellContentView:0x7fa582d15800.height == 95   (active)>"
)

Will attempt to recover by breaking constraint 
<NSLayoutConstraint:0x600001005fe0 UIView:0x7fa582d162e0.height == 55   (active)>

这是我用来在 UITableViewCell 内容视图内容中添加视图的代码。

if view == nil 
    view = UIView(frame: .zero)
    view?.backgroundColor = .green
    view?.translatesAutoresizingMaskIntoConstraints = false

    contentView.addSubview(view!)

    NSLayoutConstraint.activate([
        view!.heightAnchor.constraint(equalToConstant: 55),
        view!.topAnchor.constraint(equalTo: contentView.topAnchor, constant: edgeInsets.top),
        view!.bottomAnchor.constraint(equalTo: contentView.bottomAnchor, constant: -edgeInsets.bottom),
        view!.leftAnchor.constraint(equalTo: contentView.leftAnchor, constant: edgeInsets.left),
        view!.rightAnchor.constraint(equalTo: contentView.rightAnchor, constant: -edgeInsets.right)
        ])

在我的视图控制器中,这是我用来设置表格视图的代码。

class SelfSizedTableView: UITableView 
    var maxHeight: CGFloat = UIScreen.main.bounds.size.height

    override func reloadData() 
        super.reloadData()
        self.invalidateIntrinsicContentSize()
        self.layoutIfNeeded()
    

    override var intrinsicContentSize: CGSize 
        let height = min(contentSize.height, maxHeight)
        return CGSize(width: contentSize.width, height: height)
    

class FormViewController: UIViewController 

    private let tableView: SelfSizedTableView = 
       let tv = SelfSizedTableView()
        tv.isScrollEnabled = false
        tv.translatesAutoresizingMaskIntoConstraints = false
        tv.tableFooterView = UIView()
        tv.register(TestTableViewCell.self, forCellReuseIdentifier: "cellId")
        return tv
    ()

    override func viewDidLoad() 
        super.viewDidLoad()

        self.view.backgroundColor = .white
        tableView.dataSource = self
        view.addSubview(tableView)
    

    override func viewWillLayoutSubviews() 
        super.viewWillLayoutSubviews()

        centerTableViewContent()
    

    private func centerTableViewContent() 

        tableView.removeConstraints(tableView.constraints)

        if tableView.intrinsicContentSize.height > view.safeAreaLayoutGuide.layoutFrame.size.height 

            NSLayoutConstraint.activate([

                tableView.topAnchor.constraint(equalTo: view.safeAreaLayoutGuide.topAnchor),
                tableView.bottomAnchor.constraint(equalTo: view.safeAreaLayoutGuide.bottomAnchor),
                tableView.leftAnchor.constraint(equalTo: view.safeAreaLayoutGuide.leftAnchor),
                tableView.rightAnchor.constraint(equalTo: view.safeAreaLayoutGuide.rightAnchor)
                ])

         else 

            NSLayoutConstraint.activate([

                tableView.heightAnchor.constraint(equalToConstant: tableView.intrinsicContentSize.height),
                tableView.widthAnchor.constraint(equalToConstant: view.frame.width),
                tableView.centerXAnchor.constraint(equalTo: view.centerXAnchor),
                tableView.centerYAnchor.constraint(equalTo: view.centerYAnchor)
                ])
        
    


extension FormViewController: UITableViewDataSource 

    func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int 
        return 5
    

    func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell 

        let cell = tableView.dequeueReusableCell(withIdentifier: "cellId", for: indexPath) as! TestTableViewCell

        switch indexPath.row 
        case 0:
            cell.backgroundColor = .red
            cell.configure(with: "Item at: \(indexPath.row)", edgeInsets: UIEdgeInsets(top: 10, left: 10, bottom: 10, right: 10))
        case 1:
            cell.backgroundColor = .yellow
            cell.configure(with: "Item at: \(indexPath.row)", edgeInsets: UIEdgeInsets(top: 20, left: 10, bottom: 20, right: 10))
        default:
            cell.backgroundColor = .blue
            cell.configure(with: "Item at: \(indexPath.row)", edgeInsets: UIEdgeInsets(top: 70, left: 10, bottom: 70, right: 10))

        

        return cell
    

【问题讨论】:

似乎单元格的高度有冲突。您是否在其他任何地方设置单元格高度? @RakeshaShastri 不 如果你使用的是动态高度,那为什么要设置为 55?如果您希望将其作为最小值或最大值,可以尝试(a)将事物设置为大于/小于而不是等于或(b)在等于 55 约束上放置较低的优先级。 (我通常更喜欢后者,因为所有警告都在说 - 引擎不知道要打破哪个约束。) @dfd 我设置的高度是针对内容视图中的视图,而不是实际的内容视图本身 @Tunds 冲突发生在UITableViewCell 中的heightAnchortopAnchor bottomAnchor 。从这三个约束中删除一个 【参考方案1】:

只是为了分享:有一个great online tool(顺便说一句,我与它没有关联)有助于理解记录的错误消息。只需复制冲突约束列表(包括周围的括号,见下文)并将它们粘贴到工具的文本区域中。然后它会给你一个很好的约束的可视化,通常有助于理解什么是错的。试试看!

复制粘贴:

(
    "<NSLayoutConstraint:0x600001005fe0 UIView:0x7fa582d162e0.height == 55   (active)>",
    "<NSLayoutConstraint:0x600001018960 V:|-(20)-[UIView:0x7fa582d162e0]   (active, names: '|':UITableViewCellContentView:0x7fa582d15800 )>",
    "<NSLayoutConstraint:0x60000101c910 UIView:0x7fa582d162e0.bottom == UITableViewCellContentView:0x7fa582d15800.bottom - 20   (active)>",
    "<NSLayoutConstraint:0x60000103d950 'UIView-Encapsulated-Layout-Height' UITableViewCellContentView:0x7fa582d15800.height == 95   (active)>"
)

【讨论】:

这能回答 OP 的问题吗? @DonMag 有些东西不是 cmets/ 也不是答案。把它们放在哪里的决定是棘手的。在我看来,这个例子最好写在答案部分。尽管如此,它也可以导致一起解决问题...... @Honey - 有些事情可能是真的可以被认为是 cmets 答案,我想说在这种情况下这不是真的。如果答案还包括“这是您的解决方案,基于使用该工具的结果”,那么我会说,是的,这是一个答案。这里不是这样。 @DonMag 你可能是对的。我应该把它作为评论发布。 它可能不会直接回答 OP 的问题,但这个答案对我和其他在布局约束问题上苦苦挣扎的人来说更有用。具体的答案对 OP 来说非常有用,但对于有类似但不同问题的人来说通常不太有用。对于这些人来说,为他们自己的问题提供答案的有用工具非常棒。将此作为评论不会赋予它相同的可见性。【参考方案2】:

如果您能正确阅读警告,您就会明白您在单元格的 contentView 中的视图上设置了显式高度 (55),并且您将其限制在 contentView 的顶部和底部。因此,您基本上是在其内容上使用自动布局指定单元格的高度。

为此,您应该将rowHeight 设置为UITableViewAutomaticDimension,并将estimatedRowHeight 设置为一个可以最佳估计其高度的常数。

然后警告中的UIView-Encapsulated-Layout-Height 约束告诉您,冲突的出现只是因为tableView 基于rowHeight/estimatedRowHeight 设置的单元格的高度 - 有关此主题的更多信息,请阅读this question。

现在您必须决定是要使用通过tableView.rowHeight 设置的行高,还是使用单元格内容的自动布局约束指定的高度。如果您想使用自动布局,我假设基于该约束,请使用以下代码。

尝试使用此代码设置tableView

private let tableView: SelfSizedTableView = 
    let tv = SelfSizedTableView()
    tv.isScrollEnabled = false
    tv.translatesAutoresizingMaskIntoConstraints = false
    tv.tableFooterView = UIView()

    // explicitly set the row height and its estimated row height
    tv.estimatedRowHeight = 55 // ideally use 55 + edgeInsets.top + edgeInsets.bottom
    tv.rowHeight = UITableViewAutomaticDimension // automatic height based on content

    tv.register(TestTableViewCell.self, forCellReuseIdentifier: "cellId")
    return tv
()

然后使用此代码设置单元格的内容视图:

if view == nil 
    view = UIView(frame: .zero)
    view?.backgroundColor = .green
    view?.translatesAutoresizingMaskIntoConstraints = false

    contentView.addSubview(view!)

    // set priority on one of the constraints to 999:
    let heightConstraint = view!.heightAnchor.constraint(equalToConstant: 55)
    heightConstraint.priority = UILayoutPriority(rawValue: 999)
    // for reasoning behind this, read https://***.com/q/44651241/2912282

    NSLayoutConstraint.activate([
        heightConstraint,
        view!.topAnchor.constraint(equalTo: contentView.topAnchor, constant: edgeInsets.top),
        view!.bottomAnchor.constraint(equalTo: contentView.bottomAnchor, constant: -edgeInsets.bottom),
        view!.leftAnchor.constraint(equalTo: contentView.leftAnchor, constant: edgeInsets.left),
        view!.rightAnchor.constraint(equalTo: contentView.rightAnchor, constant: -edgeInsets.right)
        ])

【讨论】:

一位很好的伙伴整理了警告。

以上是关于自动布局约束警告的主要内容,如果未能解决你的问题,请参考以下文章

自动布局约束警告“将尝试通过打破约束来恢复”

自动布局警告:将尝试通过在 CollectionView 中打破约束 <NSLayoutConstraint 来恢复

使用自动布局的 UIToolBar 布局约束问题

为按钮提供相等宽度约束时,自动布局设置不正确

ScrollView 自动布局约束打破设备旋转

抑制 Xcode 中的运行时自动布局警告