在UITableViewCell上遇到AutoLayout问题

我在xcode 5项目上遇到了自动布局问题。 我正在使用导航控制器内部的普通视图控制器。 上半部分有一个MKMapView ,下半部分有一个UITableView 。 我正在使用storyboards ,并configuration了原型UITableViewCell ,但我通过代码添加约束。 我已经仔细检查了原型中的每个控件,并且没有看到任何configuration的约束。 我为UITableViewCell添加约束时发生问题。 我在单元格中有以下代码:

 -(void)updateConstraints { [super updateConstraints]; //first remove old constraints [self removeConstraints:self.constraints]; [self.nameLabel removeConstraints:self.nameLabel.constraints]; [self.addressLabel removeConstraints:self.nameLabel.constraints]; [self.rentableSquareFeetLabel removeConstraints:self.rentableSquareFeetLabel.constraints]; [self.lastSaleAmountLabel removeConstraints:self.lastSaleAmountLabel.constraints]; [self.lastSaleDateLabel removeConstraints:self.lastSaleAmountLabel.constraints]; [self.thumbnailImageView removeConstraints:self.thumbnailImageView.constraints]; //then set up constraints NSDictionary *viewsDictionary = NSDictionaryOfVariableBindings(_thumbnailImageView, _nameLabel, _rentableSquareFeetLabel, _lastSaleAmountLabel, _addressLabel, _lastSaleDateLabel); [self addConstraints:[NSLayoutConstraint constraintsWithVisualFormat:@"H:|[_thumbnailImageView(60)]-[_nameLabel(<=200)]-(>=8)-[_rentableSquareFeetLabel]-(>=8)-[_lastSaleAmountLabel]|" options:0 metrics:nil views:viewsDictionary]]; [self addConstraints:[NSLayoutConstraint constraintsWithVisualFormat:@"V:[_nameLabel]-(-4)-[_addressLabel]" options:NSLayoutFormatAlignAllLeading metrics:nil views:viewsDictionary]]; [self addConstraints:[NSLayoutConstraint constraintsWithVisualFormat:@"V:[_lastSaleAmountLabel]-(-4)-[_lastSaleDateLabel]" options:NSLayoutFormatAlignAllLeading metrics:nil views:viewsDictionary]]; } 

我在debugging控制台中得到以下内容。 该exception由第一个addConstraints行触发。 如果我只是继续这些,那么最终所有东西都会显示出来,因为看起来xcode正在select打破正确的约束:

 2013-09-25 15:07:14.169 PECProperties[32381:a0b] 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) ( "<NSIBPrototypingLayoutConstraint:0x9d56c70 'IB auto generated at build time for view with fixed frame' H:|-(0)-[UIImageView:0x9d558f0](LTR) (Names: '|':UITableViewCellContentView:0x9d55620 )>", "<NSIBPrototypingLayoutConstraint:0x9d56d20 'IB auto generated at build time for view with fixed frame' H:[UIImageView:0x9d558f0(60)]>", "<NSIBPrototypingLayoutConstraint:0x9d56d80 'IB auto generated at build time for view with fixed frame' H:|-(78)-[UILabel:0x9d559e0](LTR) (Names: '|':UITableViewCellContentView:0x9d55620 )>", "<NSLayoutConstraint:0x9d53830 H:[UIImageView:0x9d558f0]-(NSSpace(8))-[UILabel:0x9d559e0]>" ) Will attempt to recover by breaking constraint <NSIBPrototypingLayoutConstraint:0x9d56d80 'IB auto generated at build time for view with fixed frame' H:|-(78)-[UILabel:0x9d559e0](LTR) (Names: '|':UITableViewCellContentView:0x9d55620 )> Break on objc_exception_throw to catch this in the debugger. The methods in the UIConstraintBasedLayoutDebugging category on UIView listed in <UIKit/UIView.h> may also be helpful. 

第三个NSIBPrototypingLayoutConstraint在视图的边缘和标签之间显示了78个点。 这就是原型定位的粗略位置(如果我在原型中移动原型,我会看到debugging控制台中的约束发生了变化),但这与我自己的图像视图和标签之间“标准”距离的约束冲突。

我已经尝试在视图控制器的cellForRowAtIndexPath设置translatesAutoresizingMaskIntoConstraints=NO ,但这似乎并没有帮助。 我怎样才能修复布局?

这里包括几件事情:

  1. 您正在运行(以及导致exception)的NSIBPrototypingLayoutConstraint约束是由Interface Builder自动生成的,以使您的Storyboard或XIB视图布局不含糊。 这样做很偷偷摸摸,但它会自动添加所需的最小约束,以便每个模糊视图的位置和大小都被完全指定。 这是从Xcode 4的变化,因为在Xcode 4中,你不能在Interface Builder中有不明确的布局。 使用Xcode 5和更高版本,您可以,但是如果您的布局在编译时不明确,IB将自动为您生成这些约束。

    解决此问题的方法是在Interface Builder中添加所需的最小约束条件,以便完全指定每个视图的位置和大小,然后select这些不需要的约束条件,转到右侧边栏属性检查器,然后选中占位符旁边的框– 在构build时删除

    Xcode 6中的约束占位符复选框的屏幕截图。

    这个checkbox不仅删除了你添加的约束,而且最重要的是它将阻止自动生成的IB约束取代它! (可以想象,当你在IB中有许多视图并且想要pipe理你所有的代码约束时,这是非常乏味的,因此你可能希望避免完全使用IB来实现Auto的视图层次结构以编程方式布局。)

    占位符约束和卸载约束之间有什么区别? 下面是我的Adaptive Auto Layout讲座(video) ( PDF幻灯片 )中的一张幻灯片,比较两者:

    占位符约束和卸载约束之间的比较。

  2. updateConstraints ,你不想删除约束并重新添加,就像你在那里。 为什么不? 从本质上来说,这对于性能来说是非常糟糕的,而且我已经向苹果的工程师证实了这不是一个好主意。 看到我已经张贴在这里的问题/答案一些更多的细节,以及这个答案 。 为了防止约束被多次添加,使用布尔标志(例如hasSetupConstraints ),一旦你第一次设置你的约束,就设置为YES;如果再次调用updateConstraints ,你可以立即返回,如果你没有新的约束添加。 看到这个问题进一步讨论。

  3. 您用于删除约束的代码可能无法完全运行。 这是因为[view removeConstraints:view.constraints]只会移除已添加到view中的约束 – 请记住,约束可以添加到它们约束的视图的任何常见的superview中 – 并且添加到view的约束可能不是只影响view的布局! 如果您需要删除一些约束,则应该在属性中存储对每个约束的引用(例如,包含NSLayoutConstraint实例的NSArray属性),然后使用NSLayoutConstraint上的API或PureLayout开放源代码上的API停用/ 源库 。 您应该只停用/删除尽可能less的约束,因为这样做的计算量很大。 另一方面,改变任何约束的constant是非常有效和鼓励的,你不需要去除或重新添加约束来做到这一点。