调用layoutIfNeeded()时,AutoLayout会中断约束

时间:2014-07-27 19:17:45

标签: uitableview swift autolayout xcode6

我正在尝试使用Swift在UITaleViewCell中使用XCode6实现动态高度UITableView

通过以图形方式设置约束,我将我的单元格布局如下(屏幕截图来自XCode5,因为XCode6上有NDA)。我还将BodyLabel的Line Break属性设置为'Word Wrap',并将Line number设置为'0'以允许多行。

enter image description here

现在如果我只是在tableView(tableView: UITableView?, cellForRowAtIndexPath indexPath: NSIndexPath?)方法中设置单元格的内容,那么我就可以正确获得动态高度行为。

但是,由于我按照在线提供的教程(特别是this one),我添加了另一种方法来确定tableView(tableView: UITableView!, heightForRowAtIndexPath indexPath: NSIndexPath!)单元格的高度。 在我关注的教程中,它告诉我添加cell.layoutIfNeeded(),所以我也加了。

override func tableView(tableView: UITableView!, heightForRowAtIndexPath indexPath: NSIndexPath!) -> CGFloat {

    var cell = tableView!.dequeueReusableCellWithIdentifier(kCellIdentifier) as GroupFeedCell

    // Configure the cell
    if let frc = self.fetchedResultsController {
        let feed = frc.objectAtIndexPath(indexPath) as GroupFeed
        cell.titleLabel.text = feed.name
        if let message = feed.message {
            cell.bodyLabel.text = message
        }
    }

    // Layout the cell
    cell.layoutIfNeeded()

    // Get the height
    var height : CGFloat = cell.contentView.systemLayoutSizeFittingSize(UILayoutFittingCompressedSize).height
    return height
}

但是,当我运行程序时,虽然表格视图仍然正确显示了单元格的动态高度,但它显示的错误如下:

2014-07-27 13:59:22.599 FBGroups[4631:979424] 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:0x17809e780 H:[UILabel:0x14fd12f50'Body Label Contents...']-(8)-|   (Names:    '|':UITableViewCellContentView:0x178185d70 )>",
    "<NSLayoutConstraint:0x17809e7d0 H:|-(8)-[UILabel:0x14fd12f50'Body Label Contents...']   (Names:    '|':UITableViewCellContentView:0x178185d70 )>",
    "<NSLayoutConstraint:0x17809ea50 'UIView-Encapsulated-Layout-Width' H:[UITableViewCellContentView:0x178185d70(0)]>"
)

Will attempt to recover by breaking constraint 
<NSLayoutConstraint:0x17809e780 H:[UILabel:0x14fd12f50'Body Label Contents...']-(8)-|   (Names:     '|':UITableViewCellContentView:0x178185d70 )>

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.  

我试图找出可能出错的地方,并花费大量时间,并且我发现每当我删除cell.layoutIfNeeded()方法时,约束错误就会消失。

似乎在相互冲突的约束中,UIView-Encapsulated-Layout-Width不是我添加的那个,除此之外,所有约束对我来说都是无辜的。我试图搜索可能产生UIView-Encapsulated-Layout-Width约束的内容,但在我的情况下我无法得到令人满意的解释。我想知道这个错误消息的原因是什么以及如何解决这个问题。

此外,有人可以解释在计算单元格高度时调用cell.layoutIfNeeded()方法的目的是什么,以及何时需要?大多数覆盖动态高度UITableViewCell的教程都使用了这种方法,虽然我的程序在没有调用方法的情况下仍能正确显示单元格,每当我尝试使用该方法时,都会导致异常。

4 个答案:

答案 0 :(得分:30)

在Xcode中,将最底层垂直约束的优先级设置为750(或任何小于1000的值)。

答案 1 :(得分:1)

可能是冲突的约束是因为前两个约束将标签的宽度定义为小于单元格宽度的16pts,但是第三个约束UIView-Encapsulated-Layout-Width想要使标签成为不同的大小。

您可能会尝试降低内容压缩阻力/拥抱属性的值,以允许标签忽略其内在大小。

答案 2 :(得分:0)

这似乎是Apple的一个错误。我在运行时使用了一个附加width约束的解决方法。

- (void)configurateBodyLabelConstraint {
    UIScreen *mainScreen = [UIScreen mainScreen];
    CGFloat viewWidth = mainScreen.bounds.size.width;
    CGFloat bodyLabelWidth = viewWidth - 76 - 8;
    NSDictionary *metric = @ {
        @"bodyLabelWidth": [NSNumber numberWithFloat:bodyLabelWidth]
    };

    UILabel *bodyLabel = self.bodyLabel;
    NSArray *constraints = [NSLayoutConstraint constraintsWithVisualFormat:@"[bodyLabel(bodyLabelWidth)]"
                                                                   options:0
                                                                   metrics:metric
                                                                     views:@ {
                                                                         @"bodyLabel": bodyLabel
                                                                     }];
    [self.contentView addConstraints:constraints];
}

答案 3 :(得分:-3)

cell.layoutIfNeeded() - 无需调用。 创建IBOutlet和NSLayoutConstraint