爱程序网

iOS开发-UI篇-AutoLayout

来源: 阅读:

1.概述:随着iOS设备的更新迭代,屏幕适配问题也变得愈发重要,应运而生了Autolayout。在Xcode4.1OS X10.7之后才起效。

2.简介:Apple提供了2种方式布局Autolayout,xib来布局;另一种利用代码进行布局,Applesdk中提供了2个类库:NSLayoutManagerNSLayoutConstraintNSLayoutManager类主要是配合NSTextStorage,NSTextContainer实现NSCoding协议用来布局text视图,NSLayoutConstraint类则是布局整个视图结构(着重介绍)

3.原理:Autolayout的主要思想就是约束。在NSLayoutConstraint类中,有一个最重要的类方法

+(instancetype)constraintWithItem:(id)view1 attribute:(NSLayoutAttribute)attr1 relatedBy:(NSLayoutRelation)relation toItem:(id)view2 attribute:(NSLayoutAttribute)attr2 multiplier:(CGFloat)multiplier constant:(CGFloat)c;

实现的公式是"view1.attr1 = view2.attr2 * multiplier + constant".建立2个视图之间的约束联系(第二视图可以为nil)。但是用这个方法的简单明了,缺点就是太过繁琐复杂.但是我们推荐使用VFL进行布局。

4.1内容:

1).当我们使用Autolayout,我们要摒弃Xcode自动适应设置,translatesAutoresizingMaskIntoConstraintsNO;不然会造成视图混乱。

2).遵从高内聚的原则,UIView.h类中系统定义了- (void)updateConstraints;UIViewController.h类中系统定义了- (void)updateViewConstraints;Overrides must call super or send -updateConstraints to the view=>[super updateViewConstraints];

3).最基本的添加约束的方法:创建一个view对象,创建一个NSLayoutConstraint约束对象,constraintWithItem进行约束设置,view对象中添加约束。

4.2内容:

1).Autolayout还提供了VFL(Visual Format Language)来约束,+ (NSArray *)constraintsWithVisualFormat:(NSString *)format options:(NSLayoutFormatOptions)opts metrics:(NSDictionary *)metrics views:(NSDictionary *)views;

//format:传入单独每个view的约束语句;查询xcodeapiVisual Format Syntax

2).Autolayout步骤:创建一个view,并且添加到父视图中去显示->[父视图 addConstraints:(约束数组)]->传入参数,format具体约束语句,opts可置为nil,metrics传入距离参数(用参数代替魔数),views传入相应的视图控件,metricsviews是为format服务的。

3).1:

- (void)updateViewConstraints{
    [super updateViewConstraints];//override初始化
    
    UIView *view1 = [[UIView alloc] init];
    view1.translatesAutoresizingMaskIntoConstraints = NO;//取消自动布局
    view1.backgroundColor = [UIColor blueColor];
    [self.view addSubview:view1];
    
    NSDictionary *views_Dic = @{@"view1":view1};//约束的控件
    
    NSString *hLayoutStr = @"H:|-(==100)-[view1(==100)]";//H:代表横向,|代表相对于父视图,-代表约束距离,()里面的所有表示约束距离,[]里面表示约束的视图对象,这句话表示:横向从左到右,相对于父视图距离100像素,view1的宽为100
    NSString *vLayoutStr = @"V:|-(==100)-[view1(==50)]";//V:代表纵向,|代表相对于父视图,-代表约束距离,()里面的所有表示约束距离,[]里面表示约束的视图对象,这句话表示:纵向从上到下,相对于父视图距离100像素,view1的高为50
    
    NSArray *hLayoutArr =  [NSLayoutConstraint constraintsWithVisualFormat:hLayoutStr options:0 metrics:nil views:views_Dic];
    NSArray *vLayoutArr =  [NSLayoutConstraint constraintsWithVisualFormat:vLayoutStr options:0 metrics:nil views:views_Dic];
    [self.view addConstraints:hLayoutArr];
    [self.view addConstraints:vLayoutArr];
}

2:

- (void)updateViewConstraints{
    [super updateViewConstraints];//override初始化
    
    UIView *view1 = [[UIView alloc] init];
    view1.translatesAutoresizingMaskIntoConstraints = NO;
    view1.backgroundColor = [UIColor blueColor];
    [self.view addSubview:view1];
    
    UIView *view2 = [[UIView alloc] init];
    view2.translatesAutoresizingMaskIntoConstraints = NO;
    view2.backgroundColor = [UIColor greenColor];
    [self.view addSubview:view2];
    
    NSDictionary *views_Dic = @{@"view1":view1,@"view2":view2};
    NSDictionary *views_metrics = @{@"Top":@50,@"Right":@20};
    
    NSString *hLayoutStr = @"H:|-(==100)-[view1(==100)]-[view2(view1)]-Right-|";//表示:从父视图的左边开始,约束100像素,约束view1的宽为100像素,约束view2的宽与view1等宽,约束距离父视图右边的距离为20像素。
    NSString *vLayoutStr = @"V:|-(100)-[view1(==100)]-[view2(==100)]";
    
    
    NSArray *hLayoutArr =  [NSLayoutConstraint constraintsWithVisualFormat:hLayoutStr options:0 metrics:views_metrics views:views_Dic];
    NSArray *vLayoutArr =  [NSLayoutConstraint constraintsWithVisualFormat:vLayoutStr options:0 metrics:views_metrics views:views_Dic];
    [self.view addConstraints:hLayoutArr];//横向约束
    [self.view addConstraints:vLayoutArr];//纵向约束
}

 例3:

- (void)viewDidLoad {
UIView *view_nav = [[UIView alloc] init];
view_nav.translatesAutoresizingMaskIntoConstraints = NO;
view_nav.backgroundColor = [UIColorFactory colorNav];
[self.view addSubview:view_nav];

self.mUAccounts_table = [[UITableView alloc] init];
self.mUAccounts_table.delegate = self;
self.mUAccounts_table.dataSource = self;
self.mUAccounts_table.translatesAutoresizingMaskIntoConstraints = NO;
[self.view addSubview:self.mUAccounts_table];

//AutoLayout进行约束
NSDictionary *views_dic = @{@"view_nav":view_nav,@"mUAccounts_table":self.mUAccounts_table};
NSDictionary *metrics_dic = @{@"Top":@0,@"Bottom":@0,@"Left":@0,@"Right":@0,
                              @"ScreenWidth":@((float)ScreenWidth),
                              @"tabBar_height":@(self.tabBarController.tabBar.frame.size.height),
                              @"Table_height":@((float)ScreenHeight-64-49)};
//当等宽或者等高时,将相同方向的两个视图拆分成同个方向的2个约束
[self.view addConstraints:[NSLayoutConstraint constraintsWithVisualFormat:@"H:|-Left-[view_nav(ScreenWidth)]" options:NSLayoutFormatAlignAllLeft metrics:metrics_dic views:views_dic]];
[self.view addConstraints:[NSLayoutConstraint constraintsWithVisualFormat:@"V:|-Top-[view_nav(64)]-[mUAccounts_table(Table_height)]" options:0 metrics:metrics_dic views:views_dic]];
[self.view addConstraints:[NSLayoutConstraint constraintsWithVisualFormat:@"H:|-0-[mUAccounts_table(ScreenWidth)]" options:0 metrics:metrics_dic views:views_dic]];
}

 

5.1注意:

1).有可能出现-(void)updateViewConstraints;不自动调用的情况,可以直接在viewDidLoad中添加[self updateViewConstraints];进行调用。

2).VFL的结构1:-距离-对象(尺寸)-距离,不能出现-距离-距离的情况。

3).当出现Unable to simultaneously satisfy constraints时意味着约束之间有冲突,不能满足所有约束。

4).限定一个视图的约束一般通过与其他视图的位置和自身宽高总共4个约束,或者全部与其他视图约束,自身不做宽高约束。(4个约束灵活形成一个适配的视图)

5).约束对象注意,父视图添加约束,自身宽高约束则是自身当做父视图来约束。

6).当出现等宽和等高的情况时,可将同一个方向的两个视图约束拆分成同个方向的2个约束。

关于爱程序网 - 联系我们 - 广告服务 - 友情链接 - 网站地图 - 版权声明 - 人才招聘 - 帮助