我希望子视图在超级视图的后沿(具有固定宽度)和自动'换行'当超级视图的后沿到达了。 我正在寻找一种解决方案,使Auto Layout能够为我完成整个工作 - without(重新)计算子视图的大小动态变化时的行结构(例如,以其内容为增长/缩小的文本字段)和添加或删除约束。AutoLayout(> iOS 6)子视图的自动和动态换行
我已经有一个(工作)解决方案,根据子视图的预计算(重新)计算的行结构创建约束。
所以我的问题是关于是否可以找到一个解决方案,在这样的星座中使用约束(灵活性,优先级aso),使Auto Layout能够在需要时自动在子视图上执行换行(在第一次加载和运行时)。
我创建了一个示例项目UILabels
作为'子视图'。请玩弄这个 - 或试图说服我,我的想法是有远见的 - 但目前还没有实现。
实际屏幕截图代码如下:
这是测试动态新行自动布局的方法,包括:延伸NSLayoutConstraint
- (void) addConstraintsForLabels : (NSArray*) labels
superview : (UIView*) superview {
for (int i = 0; i < labels.count; i++) {
UILabel* precedingLabel = i == 0 ? nil : labels[i - 1];
UILabel* label = labels[i];
[label setTranslatesAutoresizingMaskIntoConstraints: NO];
//// set all contentHuggingPriorities to no growing and no compression
[label setContentHuggingPriority: UILayoutPriorityRequired forAxis: UILayoutConstraintAxisHorizontal];
[label setContentHuggingPriority: UILayoutPriorityRequired forAxis: UILayoutConstraintAxisVertical];
[label setContentCompressionResistancePriority: UILayoutPriorityRequired forAxis: UILayoutConstraintAxisHorizontal];
[label setContentCompressionResistancePriority: UILayoutPriorityRequired forAxis: UILayoutConstraintAxisVertical];
//// constraints affecting superview
// top to superview
NSLayoutConstraint* topToSup = [NSLayoutConstraint alignEdge: NSLayoutAttributeTop ofViews: @[label, superview] relation: i == 0 ? NSLayoutRelationEqual : NSLayoutRelationGreaterThanOrEqual
spacing: 0];
// leading to superview
NSLayoutConstraint* leadingToSup = [NSLayoutConstraint alignEdge: NSLayoutAttributeLeading
ofViews: @[label, superview]
relation: i == 0 ? NSLayoutRelationEqual : NSLayoutRelationGreaterThanOrEqual
spacing: 0];
// bottom to superview
NSLayoutConstraint* bottomToSup = [NSLayoutConstraint alignEdge: NSLayoutAttributeBottom
ofViews: @[label, superview]
relation: i == labels.count - 1 ? NSLayoutRelationEqual : NSLayoutRelationLessThanOrEqual
spacing: 0];
// trailing to superview
NSLayoutConstraint* trailingToSup = [NSLayoutConstraint alignEdge: NSLayoutAttributeTrailing
ofViews: @[label, superview]
relation: NSLayoutRelationLessThanOrEqual
spacing: 0];
//// example constraints affecting preceding label
// leading to preceding label
NSLayoutConstraint* leadingToPrec = precedingLabel == nil ? nil : [NSLayoutConstraint horizontalSpacing: 0
betweenViews: @[precedingLabel, label]
flexible: NO
inMaximum: NO];
// y position to preceding label
NSLayoutConstraint* centerYToPrec = precedingLabel == nil ? nil : [NSLayoutConstraint alignEdge: NSLayoutAttributeCenterY
ofViews: @[label, precedingLabel]
relation: NSLayoutRelationEqual
spacing: 0];
// top to preceding view
NSLayoutConstraint* topToPrec = precedingLabel == nil ? nil : [NSLayoutConstraint verticalSpacing: 0
ofViews: @[precedingLabel, label]
flexible: NO
inMaximum: NO];
//// priorities
// affecting the superview
topToSup.priority = UILayoutPriorityRequired; // is either flexible or non-flexible for the first
leadingToSup.priority = UILayoutPriorityRequired; // is either flexible or non-flexible for the first
bottomToSup.priority = UILayoutPriorityRequired; // is either flexible or non-flexible for the last
trailingToSup.priority = UILayoutPriorityRequired; // it is required, that the view does not exceed the right edge of its superview
[superview addConstraints: @[trailingToSup, topToSup, leadingToSup, bottomToSup]];
// affecting the preceding view
if (i > 0) {
leadingToPrec.priority = UILayoutPriorityDefaultHigh;
centerYToPrec.priority = UILayoutPriorityDefaultHigh;
topToPrec.priority = UILayoutPriorityDefaultHigh;
[superview addConstraints: @[leadingToPrec, centerYToPrec, topToPrec]];
}
}
}
导入的类别:
@implementation NSLayoutConstraint (ConstructorAdditions)
// align edges of two views
// if one view is the superview, put it at the second position in the array
+ (NSLayoutConstraint*) alignEdge : (NSLayoutAttribute) edge
ofViews : (NSArray*) /*UIView*/ views
relation : (NSLayoutRelation) relation
spacing : (CGFloat) spacing {
NSLayoutConstraint* constraint = nil;
if (views.count == 2) {
if (edge == NSLayoutAttributeBaseline || edge == NSLayoutAttributeTrailing || edge == NSLayoutAttributeBottom || edge == NSLayoutAttributeRight) {
spacing = -spacing;
}
constraint = [NSLayoutConstraint constraintWithItem : [views objectAtIndex: 0]
attribute : edge
relatedBy : relation
toItem : [views objectAtIndex: 1]
attribute : edge
multiplier : 1.0
constant : spacing];
}
return constraint;
}
// vertical spcacing between views
// the method assumes, that the first view in the array is above the second view
+ (NSLayoutConstraint*) verticalSpacing : (CGFloat) spacing
ofViews : (NSArray*) /*UIView*/ views
flexible : (BOOL) flexible
inMaximum: (BOOL) inMax {
NSLayoutConstraint* constraint = nil;
NSLayoutRelation relation = flexible ? (inMax ? NSLayoutRelationLessThanOrEqual : NSLayoutRelationGreaterThanOrEqual) : NSLayoutRelationEqual;
if (views.count == 2) {
constraint = [NSLayoutConstraint constraintWithItem : [views objectAtIndex: 0]
attribute : NSLayoutAttributeBottom
relatedBy : relation
toItem : [views objectAtIndex: 1]
attribute : NSLayoutAttributeTop
multiplier : 1.0
constant : -spacing];
}
return constraint;
}
// horizontal spacing between views
// the method assumes, that the first view in the array is left of the second view
+ (NSLayoutConstraint*) horizontalSpacing : (CGFloat) spacing
betweenViews : (NSArray*) /*UIView*/ views
flexible : (BOOL) flexible
inMaximum: (BOOL) inMax {
NSLayoutConstraint* constraint = nil;
NSLayoutRelation relation = flexible ? (inMax ? NSLayoutRelationLessThanOrEqual : NSLayoutRelationGreaterThanOrEqual) : NSLayoutRelationEqual;
if (views.count == 2) {
constraint = [NSLayoutConstraint constraintWithItem : [views objectAtIndex: 0]
attribute : NSLayoutAttributeTrailing
relatedBy : relation
toItem : [views objectAtIndex: 1]
attribute : NSLayoutAttributeLeading
multiplier : 1.0
constant : -spacing];
}
return constraint;
}
+ (NSArray*) equalWidthOfViews : (NSArray*) views
toView : (UIView*) view
distance : (CGFloat) distance
relation : (NSLayoutRelation) relation
multiplier : (CGFloat) multiplier {
NSMutableArray* constraints = [[NSMutableArray alloc] initWithCapacity: views.count];
for (UIView* aView in views) {
[constraints addObject: [NSLayoutConstraint constraintWithItem : aView
attribute : NSLayoutAttributeWidth
relatedBy : relation
toItem : view
attribute : NSLayoutAttributeWidth
multiplier : multiplier
constant : distance]];
}
return constraints;
}
请将您提供的代码缩减为包含您问题的部分。您应该自己找出调用错误的原因,而不是委托给社区! – 2014-08-27 18:30:14
我既没有错误问题,也不知道上面的截图结果来自哪里。不过,为了更清晰起见,我会稍微减少一些代码。谢谢。 – anneblue 2014-08-28 08:14:29