101

当我的 UIStackView “行”被压扁时,它们会AutoLayout发出警告。但是,它们显示良好,除了这些日志记录之外没有其他问题:

无法同时满足约束。以下列表中的至少一个约束可能是您不想要的。试试这个:(1)查看每个约束并尝试找出您不期望的;(2) 找到添加了一个或多个不需要的约束的代码并修复它。(注意:如果您看到NSAutoresizingMaskLayoutConstraints不明白的地方,请参阅该UIView属性的文档translatesAutoresizingMaskIntoConstraints)(

所以,我还不确定如何解决这个问题,但除了烦人之外,它似乎并没有破坏任何东西。

有谁知道如何解决它?有趣的是,布局约束经常被标记为'UISV-hiding',这表明它可能应该忽略子视图的高度最小值或在这种情况下的其他东西?

4

14 回答 14

211

您会遇到此问题,因为当将子视图从内部设置UIStackView为隐藏时,它将首先将其高度限制为零以便对其进行动画处理。

我收到以下错误:

2015-10-01 11:45:13.732 <redacted>[64455:6368084] 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. (Note: If you're seeing NSAutoresizingMaskLayoutConstraints that you don't understand, refer to the documentation for the UIView property translatesAutoresizingMaskIntoConstraints) 
(
    "<NSLayoutConstraint:0x7f7f5be18c80 V:[UISegmentedControl:0x7f7f5bec4180]-(8)-|   (Names: '|':UIView:0x7f7f5be69d30 )>",
    "<NSLayoutConstraint:0x7f7f5be508d0 V:|-(8)-[UISegmentedControl:0x7f7f5bec4180]   (Names: '|':UIView:0x7f7f5be69d30 )>",
    "<NSLayoutConstraint:0x7f7f5bdfbda0 'UISV-hiding' V:[UIView:0x7f7f5be69d30(0)]>"
)

Will attempt to recover by breaking constraint 
<NSLayoutConstraint:0x7f7f5be18c80 V:[UISegmentedControl:0x7f7f5bec4180]-(8)-|   (Names: '|':UIView:0x7f7f5be69d30 )>

Make a symbolic breakpoint at UIViewAlertForUnsatisfiableConstraints to catch this in the debugger.
The methods in the UIConstraintBasedLayoutDebugging category on UIView listed in <UIKit/UIView.h> may also be helpful.

我试图做的是UIView在我的内部放置一个在每个边缘上UIStackView包含一个UISegmentedControl8pts 的插图。

当我将它设置为隐藏时,它会尝试将容器视图约束为零高度,但因为我有一组从上到下的约束,所以发生了冲突。

为了解决这个问题,我将我的 8pt 顶部和底部约束优先级从 1000 更改为 999,以便在UISV-hiding需要时可以优先考虑约束。

于 2015-10-01T11:02:37.640 回答
57

我遇到了一个不容易解决的类似问题。就我而言,我在堆栈视图中嵌入了一个堆栈视图。内部 UIStackView 有两个标签和指定的非零间距。

当您调用 addArrangedSubview() 时,它将自动创建类似于以下内容的约束:

V:|[innerStackView]|              | = outerStackView

  V:|[label1]-(2)-[label2]|       | = innerStackView

现在,当您尝试隐藏 innerStackView 时,您会收到一个模棱两可的约束警告。

要理解为什么,让我们首先看看为什么当等于时不会发生这种情况。当你打电话时,@liamnichols 是正确的......它将神奇地拦截这个电话,并创建一个优先级为 1000 的高度UISV 隐藏约束(必需)。大概这是为了让堆栈视图中的元素在视图之外被动画化,以防您的隐藏代码在块内被调用。不幸的是,似乎没有办法阻止添加此约束。不过,您不会收到“无法同时满足约束”(USSC)警告,因为会发生以下情况:innerStackView.spacing0innerStackView.hidden = trueouterStackView0UIView.animationWithDuration()

  1. label1 的高度设置为 0
  2. 两个标签之间的间距已经定义为 0
  3. label2 的高度设置为 0
  4. innerStackView 的高度设置为 0

很明显可以满足这 4 个约束条件。堆栈视图只是将所有内容平滑到一个 0 高度的像素中。

现在回到错误的例子,如果我们将 设置spacing2,我们现在有这些约束:

  1. label1 的高度设置为 0
  2. 两个标签之间的间距由堆栈视图自动创建为 2 像素高,优先级为 1000。
  3. label2 的高度设置为 0
  4. innerStackView 的高度设置为 0

堆栈视图不能既是 0 像素高又是其内容高 2 像素。不能满足约束。

注意:您可以通过一个更简单的示例来查看此行为。只需将 UIView 作为排列的子视图添加到堆栈视图。然后在该 UIView 上设置一个高度约束,优先级为 1000。现在尝试调用 hide 。

注意:无论出于何种原因,这只发生在我的堆栈视图是 UICollectionViewCell 或 UITableViewCell 的子视图时。但是,您仍然可以通过innerStackView.systemLayoutSizeFittingSize(UILayoutFittingCompressedSize)在隐藏内部堆栈视图后调用下一个运行循环来在单元格之外重现此行为。

注意:即使您尝试在 UIView.performWithoutAnimations 中执行代码,堆栈视图仍会添加 0 高度约束,这将导致 USSC 警告。


这个问题至少有3个解决方案:

  1. 在堆栈视图中隐藏任何元素之前,请检查它是否是堆栈视图,如果是,请将其更改spacing为 0。这很烦人,因为无论何时再次显示内容,您都需要反转该过程(并记住原始间距)。
  2. 不要在堆栈视图中隐藏元素,而是调用removeFromSuperview. 这更令人讨厌,因为当您反转该过程时,您需要记住在哪里插入已删除的项目。您可以通过仅调用 removeArrangedSubview 然后隐藏来进行优化,但是仍然需要做很多簿记。
  3. spacing在 UIView 中包装嵌套的堆栈视图(非零)。将至少一个约束指定为非必需优先级(999 或以下)。这是最好的解决方案,因为您不必做任何簿记。在我的示例中,我在堆栈视图和包装器视图之间的 1000 处创建了顶部、前导和尾随约束,然后从堆栈视图的底部到包装器视图创建了 999 约束。这样,当外部堆栈视图创建零高度约束时,999 约束被破坏,您不会看到 USSC 警告。(注意:这类似于是否应该将 UICollectionViewCell 子类的 contentView.translatesAutoResizingMaskToConstraints 设置为false的解决方案)

总之,你得到这种行为的原因是:

  1. 当您将托管子视图添加到堆栈视图时,Apple 会自动为您创建 1000 个优先级约束。
  2. 当您隐藏堆栈视图的子视图时,Apple 会自动为您创建 0 高度约束。

如果 Apple (1) 允许您指定约束(尤其是间隔)的优先级,或者 (2) 允许您选择退出自动UISV 隐藏约束,那么这个问题将很容易解决。

于 2016-06-27T23:17:48.450 回答
6

大多数情况下,可以通过降低约束优先级以消除冲突来解决此错误。

于 2015-09-06T21:42:25.277 回答
2

当您将视图设置为隐藏时,UIStackview将尝试对其进行动画处理。如果你想要这种效果,你需要为约束设置正确的优先级,这样它们就不会发生冲突(正如上面许多人所建议的那样)。

但是,如果您不关心动画(也许您将其隐藏在 ViewDidLoad 中),那么您可以使用简单removeFromSuperview的方法来实现相同的效果,但不会出现任何约束问题,因为这些约束将与视图一起被删除。

于 2016-09-05T18:32:50.020 回答
1

根据@Senseful 的回答,这是一个 UIStackView 扩展,用于将堆栈视图包装在视图中并应用他或她推荐的约束:

/// wraps in a `UIView` to prevent autolayout warnings when a stack view with spacing is placed inside another stack view whose height might be zero (usually due to `hidden` being `true`).
/// See http://stackoverflow.com/questions/32428210
func wrapped() -> UIView {
    let wrapper = UIView()
    translatesAutoresizingMaskIntoConstraints = false
    wrapper.addSubview(self)

    for attribute in [NSLayoutAttribute.Top, .Left, .Right, .Bottom] {
        let constraint = NSLayoutConstraint(item: self,
                                            attribute: attribute,
                                            relatedBy: .Equal,
                                            toItem: wrapper,
                                            attribute: attribute,
                                            multiplier: 1,
                                            constant: 0)
        if attribute == .Bottom { constraint.priority = 999 }
        wrapper.addConstraint(constraint)
    }
    return wrapper
}

而不是添加你的stackView,使用stackView.wrapped().

于 2016-08-04T18:23:10.673 回答
1

首先,正如其他人所建议的那样,确保您可以控制的约束,即不是 UIStackView 固有的约束设置为优先级 999,以便在隐藏视图时可以覆盖它们。

如果您仍然遇到问题,那么问题可能是由于隐藏的 StackViews 中的间距。 我的解决方案是添加一个 UIView 作为间隔并将 UIStackView 间距设置为零。 然后将 View.height 或 View.width 约束(取决于垂直或水平堆栈)设置为 StackView 的间距。

然后调整新添加视图的内容拥抱和内容压缩阻力优先级。您可能还必须更改父 StackView 的分布。

以上所有内容都可以在 Interface Builder 中完成。您可能还必须以编程方式隐藏/取消隐藏一些新添加的视图,这样您就没有不需要的间距。

于 2016-09-11T00:26:42.607 回答
1

我最近在隐藏UIStackView. 我没有做一堆簿记和包装堆栈UIViews,而是选择为我parentStackView的孩子创建一个出口,为我想要隐藏/取消隐藏的孩子创建一个出口。

@IBOutlet weak var parentStackView: UIStackView!
@IBOutlet var stackViewNumber1: UIStackView!
@IBOutlet var stackViewNumber2: UIStackView!

在情节提要中,这是我的 parentStack 的样子:

在此处输入图像描述

它有 4 个孩子,每个孩子里面都有一堆堆栈视图。当您隐藏堆栈视图时,如果它的 UI 元素也是堆栈视图,您将看到一连串自动布局错误。我没有隐藏,而是选择删除它们。

在我的示例中,parentStackViews包含 4 个元素的数组:顶部堆栈视图、堆栈视图编号 1、堆栈视图编号 2 和停止按钮。它们的索引arrangedSubviews分别为 0、1、2 和 3。当我想隐藏一个时,我只需将其从parentStackView's arrangedSubviews数组中删除。由于它不弱,它会在内存中徘徊,您可以稍后将其放回所需的索引。我没有重新初始化它,所以它只是挂起直到需要它,但不会膨胀内存。

所以基本上,你可以...

1) 将您的父堆栈的 IBOutlets 和您想要隐藏/取消隐藏的子级拖动到情节提要。

2)当你想隐藏它们时,从parentStackView's arrangedSubviews数组中删除你想隐藏的堆栈。

self.view.layoutIfNeeded()3)用调用UIView.animateWithDuration

注意最后两个 stackViews 不是weak. 当您取消隐藏它们时,您需要保留它们。

假设我想隐藏 stackViewNumber2:

parentStackView.removeArrangedSubview(stackViewNumber2)
stackViewNumber2.removeFromSuperview()

然后对其进行动画处理:

UIView.animate(withDuration: 0.25,
               delay: 0,
               usingSpringWithDamping: 2.0,
               initialSpringVelocity: 10.0,
               options: [.curveEaseOut],
               animations: {
                self.view.layoutIfNeeded()
},
               completion: nil)

如果您想stackViewNumber2稍后“取消隐藏”,您可以将其插入所需的parentStackView arrangedSubViews索引并为更新设置动画。

parentStackView.removeArrangedSubview(stackViewNumber1)
stackViewNumber1.removeFromSuperview()
parentStackView.insertArrangedSubview(stackViewNumber2, at: 1)

// Then animate it
UIView.animate(withDuration: 0.25,
               delay: 0,
               usingSpringWithDamping: 2.0,
               initialSpringVelocity: 10.0,
               options: [.curveEaseOut],
               animations: {
                self.view.layoutIfNeeded()
},
               completion: nil)

我发现这比记录约束、摆弄优先级等要容易得多。

如果你有一些你想默认隐藏的东西,你可以把它放在故事板上,然后在viewDidLoad不使用动画的情况下将其删除并更新view.layoutIfNeeded()

于 2017-03-17T15:31:33.757 回答
1

我在使用嵌入式堆栈视图时遇到了同样的错误,尽管在运行时一切正常。

isHidden = true我通过在隐藏父堆栈视图之前先隐藏所有子堆栈视图(设置)解决了约束错误。

这样做并没有删除子排列视图的所有复杂性,在需要添加它们时维护一个索引。

希望这可以帮助。

于 2017-04-23T19:09:46.477 回答
1

Senseful 已经为上述问题的根源提供了一个很好的答案,所以我将直接找到解决方案。

您需要做的就是将所有 stackView 约束优先级设置为低于 1000(999 将完成工作)。例如,如果 stackView 被限制在其父视图的左、右、上和下,那么所有 4 个约束的优先级都应该低于 1000。

于 2019-02-25T08:24:57.850 回答
0

您可能在使用某个尺寸类(例如:wCompact hRegular)时创建了一个约束,然后在切换到另一个尺寸类(例如:wAny hAny)时创建了一个副本。检查不同尺寸类别的 UI 对象的约束,看看约束是否存在异常。您应该看到指示碰撞约束的红线。在我获得 10 点声望点之前我不能放图片抱歉:/

于 2015-09-06T22:28:45.353 回答
0

我想一次隐藏整个 UIStackView,但我遇到了与 OP 相同的错误,这为我修复了它:

for(UIView *currentView in self.arrangedSubviews){
    for(NSLayoutConstraint *currentConstraint in currentView.constraints){
        [currentConstraint setPriority:999];
    }
}
于 2016-03-14T15:15:50.280 回答
0

我有一排带有高度限制的按钮。当一个按钮被隐藏时会发生这种情况。将该按钮高度约束的优先级设置为 999 已解决该问题。

于 2016-10-07T12:25:37.287 回答
-2

此错误与 UIStackView 无关。当您有具有相同优先级的冲突约束时,就会发生这种情况。例如,如果您有一个约束声明视图的宽度为 100,并且您同时有另一个约束声明视图的宽度是其容器的 25%。显然有两个相互冲突的约束。解决方案是删除它们。

于 2015-09-14T14:49:20.047 回答
-3

NOP 与 [mySubView removeFromSuperview]。我希望它可以帮助某人:)

于 2016-08-09T04:45:08.773 回答