10

我试图让自动布局管理器根据超级视图的宽度调整视图的中心点。我不明白为什么这是属性的“无效配对”(如崩溃和 NSInvalidArgumentException 所述)

UIView *ac;
NSLayoutConstraint *cXloc = [NSLayoutConstraint constraintWithItem:ac 
                                                         attribute:NSLayoutAttributeCenterX 
                                                         relatedBy:NSLayoutRelationEqual 
                                                            toItem:ac.superview 
                                                         attribute:NSLayoutAttributeWidth 
                                                        multiplier:.1 
                                                          constant:x*ac.superview.frame.size.width*.2];
[ac.superview addConstraint:cXloc];

有人可以解释为什么这是“无效配对”以及我应该如何解决这个问题?谢谢

4

4 回答 4

4

这是当前实现的限制Auto Layout。但是,您可以轻松解决它,因为所有约束都是线性的并且NSLayoutAttributes 是相关的。例如,假设您想要的约束是:

subview.centerX = m * superview.width + c;

您可以将其表示为 tow 之间的关系centerX

// Since width == 2 * centerX
subview.centerX = m * 2 * superview.centerX + c;
于 2013-04-19T23:32:49.667 回答
3

如果您将 ac 的 AttributeCenterX 与其父视图的 AttributeCenterX、AttributeLeading 或 AttributeTrailing 相关联,您应该能够使用乘数和约束来表达您想要的约束。请记住,仅在创建约束时才评估常量,并且示例的常量不会随着 ac.superview 的宽度变化而更新。

如果您可以用文字表达您希望 ac 相对于其父视图的位置,我们可以建议一个约束。

编辑

这是一个带有 5 个 NSButton 的示例。它们本身和它们之间的空间扩展,使得空间的宽度为按钮的 30%,所有按钮的宽度相同,所有空间的宽度都相同。仅为间距创建 4 个不可见的 NSView 非常麻烦,尤其是考虑到您已经让它在自动布局之外工作。但如果你好奇:

// Assuming these NSViews and NSButtons exist,
//NSView* superview ;
//NSButton *buttonOne, *buttonTwo, *buttonThree, *buttonFour, *buttonFive ;


[superView removeConstraints:superView.constraints] ;

// Create empty NSViews to fill the space between the 5 buttons.
NSView* spaceOne = [NSView new] ;
NSView* spaceTwo = [NSView new] ;
NSView* spaceThree = [NSView new] ;
NSView* spaceFour = [NSView new] ;
spaceOne.translatesAutoresizingMaskIntoConstraints = NO ;
spaceTwo.translatesAutoresizingMaskIntoConstraints = NO ;
spaceThree.translatesAutoresizingMaskIntoConstraints = NO ;
spaceFour.translatesAutoresizingMaskIntoConstraints = NO ;
[superView addSubview:spaceOne] ;
[superView addSubview:spaceTwo] ;
[superView addSubview:spaceThree] ;
[superView addSubview:spaceFour] ;

NSDictionary* views = NSDictionaryOfVariableBindings(superView,buttonOne,buttonTwo,buttonThree,buttonFour,buttonFive,spaceOne,spaceTwo,spaceThree,spaceFour) ;

// Vertically align buttonOne to its superview however you like.
[superView addConstraints: [NSLayoutConstraint constraintsWithVisualFormat:@"V:|-[buttonOne]"  options:0  metrics:nil  views:views ] ] ;

// Make the "space" NSViews' widths equal and >= 10. Make the buttons' widths equal.
[superView addConstraints: [NSLayoutConstraint constraintsWithVisualFormat:@"H:|[buttonOne][spaceOne(>=10)][buttonTwo(==buttonOne)][spaceTwo(==spaceOne)][buttonThree(==buttonOne)][spaceThree(==spaceOne)][buttonFour(==buttonOne)][spaceFour(==spaceOne)][buttonFive(==buttonOne)]|"  options: NSLayoutFormatAlignAllCenterY  metrics:nil  views:views ] ] ;
// Make the "space" NSViews' widths 30% of the NSButtons' widths.
[superView addConstraint: [NSLayoutConstraint constraintWithItem: spaceOne
                                                       attribute: NSLayoutAttributeWidth
                                                       relatedBy: NSLayoutRelationEqual
                                                          toItem: buttonOne
                                                       attribute: NSLayoutAttributeWidth
                                                      multiplier: 0.3
                                                        constant: 0 ] ] ;
于 2012-12-14T14:49:46.157 回答
1

根据 an0 的回答,并假设您有一个包含按钮的 NSArray,以下内容应在超级视图中平均分配按钮:

  NSUInteger currentButton = 1;
  for (UIButton *button in self.buttons)
  {
    [self.view addConstraint:[NSLayoutConstraint constraintWithItem:button attribute:NSLayoutAttributeCenterX relatedBy:NSLayoutRelationEqual toItem:self.view attribute:NSLayoutAttributeCenterX multiplier:2.0 * (CGFloat) currentButton / (CGFloat) (self.buttons.count + 1) constant:0.0]];
    currentButton++;
  }
于 2014-01-31T01:03:15.247 回答
0

如果您希望以编程方式生成的视图适合其父视图的宽度。您可以使用和的约束NSLayoutAttributeLeading配对attribute:NSLayoutAttributeCenterX

您必须进行正确的计算才能获得正确的乘数。计算涉及要布局的视图总数和当前视图的索引。

    //Caculate constraint multiplier from parentView CenterX.
    //This sets the width of the button relative to parentView.
    // A value of 2 = Full width.

    CGFloat multiplier = 2/(arr.count/counter);

    [parentView addConstraint:[NSLayoutConstraint constraintWithItem:btn
                    attribute:NSLayoutAttributeTrailing
                    relatedBy:NSLayoutRelationEqual toItem:parentView
                    attribute:NSLayoutAttributeCenterX multiplier:multiplier constant:0]];

这将分配视图宽度以填充其父视图。

于 2016-09-18T15:27:43.873 回答