iOS回顾笔记(06) -- AutoLayout从入门到精通

随着iOS设备屏幕尺寸的增多,当下无论是纯代码开发还是Xib/StoryBoard开发,自动布局已经是必备的开发技能了。

我使用自动布局也有一段时间了,遇到了不少问题,在解决的过程中也收获了很多知识。尤其是在使用熟练之后开发速度上的提升非常明显。这里把AutoLayout的基本使用和个人使用心得汇总一下,希望能帮助到大家!

适配的概念

适配主要分两种

  • 系统适配

    系统适配主要指适配不同版本系统,如 iOS 6(拟物化) 到 iOS 7(扁平化)两个系统系统的适配,我们需要写不同的代码来保证项目在不同系统上的美观与可用。

  • 屏幕适配

    屏幕适配主要是针对不同尺寸的屏幕进行适配,同一个页面再不同尺寸屏幕上的布局,如Safari在手机横竖屏下的布局等等。

常见设备的分辨率:

Snip20170313_1.png

屏幕适配发展史

  • iPhone 4s 以前的时代

iPhone 4s 和之前设备的屏幕都是3.5英寸,可以说没有屏幕适配,所有的坐标点就是 320*480.
适配完全使用frame、bounds、center进行计算,代码基本写死。

// 直接写死
UIImageView *iv = [UIImageView new];
iv.frame = CGRectMake(50, 300, 200, 80);
[self.view addSubview:iv];
  • iPad、iPhone横屏时代

    • 出现 AutoResizing 技术

      • 优点:

        • 解决了父子控件相对位置的问题
        • 子控件可根据父控件的行为发生相对应的变化
        • 让横竖屏的适配变得简单
        • 无法处理兄弟控件相对位置的问题
      • 使用前提:

        • 关闭AutoLayout
      • 局限性:

        • 只能解决父子控件的相对关系,
        • 无法解决兄弟控件之间的相对关系


    • AutoResizing在Xib中的使用介绍

    在Xib中主要有6根线来设置AutoResizing

Snip20170313_2.png

外部四根线

外部四根线分别表示上、下、左、右四个方向,子控件相对于父控件的距离。
实线:表示固定位置
虚线:表示非固定位置

**内部两根线**

内部两根线分别表示水平和竖直方向,子控件是否根据父控件等比例缩放。
实线:该方向上跟随父控件等比缩放
虚线:该方向上不跟随父控件等比缩放

    
- **AutoResizing在代码中的使用介绍**

通常代码中子控件在添加到父控件之前设置AutoResizing对应的属性值,其代码属性值和Xib中相反,代码中设置可变部分,Xib中是选中部分为固定不变的。

    // 上下左右四个方向参数(与Xib中设置相反)
    UIViewAutoresizingNone                 = 0, 
    UIViewAutoresizingFlexibleLeftMargin   = 1 << 0,
    UIViewAutoresizingFlexibleRightMargin  = 1 << 2,
    UIViewAutoresizingFlexibleTopMargin    = 1 << 3,
    UIViewAutoresizingFlexibleBottomMargin = 1 << 5
    
    // 宽高是否根据父控件等比缩放
    UIViewAutoresizingFlexibleHeight       = 1 << 4,
    UIViewAutoresizingFlexibleWidth        = 1 << 1,
    

比如要设置一个UIView与其父控件关系为右下角对齐
示例代码如下:

XYBannerView *banner = [XYBannerView bannerView];
banner.frame = CGRectMake(80, 20, 200, 90);
banner.autoresizingMask = UIViewAutoresizingFlexibleTopMargin | UIViewAutoresizingFlexibleLeftMargin;
[self.view addSubview:banner];
  • iOS 6之后

    • 出现了AutoLayout技术

    AutoLayout弥补了AutoResizing的局限性,不仅可以解决父子控件之间的相对关系,还可以描述兄弟控件之间的相对关系,还可以描述自身关系等,功能非常强大。

    AutoLayout的两个核心概念

    • 参照
    • 约束

AutoLayout介绍和使用

Xib中常用的面板

Xib 主要有以下三个面板

  • Align : 对齐方式
Snip20170313_4.png
  • Add New Constraint: 添加新的约束
Snip20170313_5.png
  • Resolve Auto Layout Issues : 修复自动布局问题


    Snip20170313_7.png

AutoLayout的使用原则

AutoLayout是为了确定View的frame(确定View的Point和Size)。

  • 至少添加四个约束,宽高和位置来确定frame
  • 避免约束冲突
    1.如同一个View的宽设置两次,一个100,一个200,导致无法确定,形成冲突,冲突属于错误,须解决:
    Snip20170314_1.png


    2.如约束不够,无法确定View的frame时候也会报错:
Snip20170314_3.png
  • AutoLayout的警告:警告一般是Xib中添加好了约束,但是对应View没有移动到对应位置导致。这种情况程序运行起来之后会是正确约束的样子,而不是Xib中的样子。
    • 警告解决办法:
      1.选中View,update frame 或使用快捷键‘command’ + ‘option’ + ‘=’

      2.如图:
      Snip20170314_2.png
  • 约束的修改:
    约束的修改有很多种方式,下面列举一种我常用的方式!
    Snip20170314_4.png

注意:上面说的Add New Constaints面板只能添加新的约束,修改不能在那里。

AutoLayout案例练习

AutoLayout这种灵活的实用技术最直接的学习办法就是实战练习,下面几个小案例来展示一下。

练习1

1.在控制器底部添加两个View,一个红色,一个蓝色
2.两个View的高度、宽度永远相等
3.距离父控件左边、右边、两者中间和距底边的距离相等
示意如图:


Snip20170314_5.png

根据:添加四个约束确定frame,避免冲突和警告的原则。我们按照要求添加约束
1.两者等高等宽:使用Add New Constraints面板

Snip20170314_6.png

或者可以直接拖线:选中红色 按住‘control’键拖线到蓝色

Snip20170314_8.png

2.设置红色View约束
选中红色View打开Add New Contraint面板设置对应约束

Snip20170314_9.png

3.设置蓝色View约束
选中红色View打开Add New Contraint面板设置对应约束,其中蓝色View的约束同上图,只需要设置蓝色的右边距同红色相等 为 20 即可(其他约束在设置红色的时候已经有了)

4.设置红蓝色View等高/底
设置等高/底 和上面设置两者等高等宽步骤一致,选择 Top/Bottom即可。

效果:


练习1.gif


练习2

同样两个View 一蓝一红
1.两个View的高度相等
2.红色View和蓝色View的右边对齐
3.蓝色View距离父控件的左右相等,且距离红色View的间距相等
4.红色View的左边和蓝色View中点对齐


Snip20170314_10.png

1.设置蓝色View约束
蓝色View约束:距离父控件边距和红色View的边距

Snip20170314_11.png

2.设置红色View和蓝色View等高 和边距

1.设置两者等高,直接拖线即可
2.设置红色view的边距直:距右 20 和 距底边 20。(参考上图)

3.设置红色View与蓝色View的中心对齐

红色View和蓝色View的中心对齐可转化为 红色View长度为蓝色一半。可先设置等宽再修改等宽约束。


Snip20170314_14.png

效果图:

练习2.gif

练习3

四个相同的View均分占据屏幕的四个角,如图


Snip20170314_15.png

1.四个view是等宽等高
直接分别设置四个的等宽等高,拖线就很方便

2.四个view互相之间的间距为零
使用Add New Contraint面板分别添加每个View的四边距 为0即可

最终效果如图:


练习3.gif

AutoLayout 中的UILabel

UILabel相对比较特殊一点,需要单独说一下。

在不使用 AutoLayout的时候 UILabel 内部的文字默认是居中显示的,如果设置的Size较大,而内部文字较少就会造成上下留白,从而造成资源的浪费。

在实际的使过程中,需求往往是UILabel正好包裹住内部的文字。

有了AutoLayout之后的UILabel在添加约束的时候可以不用添加高度,系统会自动计算内部文字高度来自适应UILabel的高度!

Snip20170314_18.png

实际应用中经常需要设置UILabel的根据文字多少来自动适应高度,并且UILabel.width <= 某个值.
这种情况需要给UILabel添加宽度约束,比例关系设置为Less Than Or Equal


Snip20170314_19.png

练习4

1.设置两个View,两者间距为0,一红一绿,
2.红色View内部有一个UILabel,
3.根据Label内部的文字自适应高度,
4.点击屏幕修改Labe内部文字,让其父控件的frame也自动适应


Snip20170314_20.png

1.设置两个View一红一绿,分别设置边距等约束
2.红色View中添加UILabel,设置Label的文字和约束,设置Label高度自适应

最终效果图:


练习4.gif

这只是几个简单的小练习,若想使用熟练AutoLayout还需要认真练习

AutoLayout在代码中的使用

以上讲了AutoLayout的可视化使用,但是项目中有很多页面是动态生成的,需要我们用代码实现,所以下面讲讲AutoLayout的代码实现

代码实现的特点:繁琐、技术含量低

Xib中的每一条拖线对应代码中一个 NSLayoutConstraint 对象,从NSLayoutConstraint头文件中可以查看其对象的创建方法:


    NSLayoutConstraint *constraint = [NSLayoutConstraint constraintWithItem:view1 
    attribute:attr1 relatedBy:NSLayoutRelation toItem:view2 attribute:attr2 
    multiplier:multiplier constant:c];
    
    参数含义:
    
    view1:                  约束的第1个View
    attr1:                  第1个View的属性
    NSLayoutRelation:       两个View的属性之间的关系
    view2:                  约束的第2个View
    attr2:                  第2个View的属性
    multiplier:             倍数关系
    c:                      需要增加的常量
    

上面的方法可以整合成一个自动布局的核心计算公式

obj1.property = (obj2.property) * multiplier + c

代码添加AutoLayout的步骤

  • 利用NSLayoutConstraint类创建具体的约束对象
  • 添加约束到对应的View上
- (void)addConstraint:(NSLayoutConstraint *)constraint; 
- (void)addConstraints:(NSArray<__kindof NSLayoutConstraint *> *)constraints;

代码添加AutoLayout的注意点

  • 需要先禁止AutoResizing功能,设置如下
view.translatesAutoresizingMaskIntoConstraints = NO;
  • 添加约束之前,确保所有View已经添加到父控件上
  • 设置AutoLayout之后无需再设置frame

代码添加约束的规则

  1. 对于兄弟控件之间的约束要添加到共同的父控件上
  2. 对于不同层级的'兄弟'控件的约束要添加到最近的‘父控件’上
  3. 两个父子控件之间的约束要添加到父控件上

下面代码实现一个如图自动布局


Snip20170314_21.png
    UIView * view = [UIView new];
    view.translatesAutoresizingMaskIntoConstraints = NO;
    view.backgroundColor  = [UIColor redColor];
//    view.frame = CGRectMake(10, 10, 100, 100);   ---> 不再设置frame
    [self.view addSubview:view];
    
    
    //设置底边约束
    NSLayoutConstraint * bottomConstraint = [NSLayoutConstraint constraintWithItem:view attribute:NSLayoutAttributeBottom relatedBy:NSLayoutRelationEqual toItem:self.view attribute:NSLayoutAttributeBottom multiplier:1.0 constant:-10];
    
    //设置右边约束
    NSLayoutConstraint * rightConstraint = [NSLayoutConstraint constraintWithItem:view attribute:NSLayoutAttributeRight relatedBy:NSLayoutRelationEqual toItem:self.view attribute:NSLayoutAttributeRight multiplier:1.0 constant:-10];
    
    //设置width约束
    NSLayoutConstraint * widthConstraint = [NSLayoutConstraint constraintWithItem:view attribute:NSLayoutAttributeWidth relatedBy:NSLayoutRelationEqual toItem:nil attribute:NSLayoutAttributeNotAnAttribute multiplier:0 constant:100];
    
    //设置height约束
    NSLayoutConstraint * heightConstraint = [NSLayoutConstraint constraintWithItem:view attribute:NSLayoutAttributeHeight relatedBy:NSLayoutRelationEqual toItem:nil attribute:NSLayoutAttributeNotAnAttribute multiplier:0 constant:100];
    
    
    [self.view addConstraint:bottomConstraint]; ----> 与父控件相关的约束添加到父控件上
    [self.view addConstraint:rightConstraint];
    [view addConstraint:widthConstraint];
    [view addConstraint:heightConstraint];

代码实现AutoLayout相对比较繁琐,但是如果懂得了原理还是能很好实现出来的。

AutoLayout三方框架

由于AutoLayout技术代码实现起来特别繁琐,并且技术含量不高、代码冗余等问题。有一些大牛开源了一些自己写的三方自动布局框架,使用起来非常简单。

目前最流行的是:Masnory

至于使用方法请自行搜索、学习。

小结

  1. AutoLayout功能强大、是现在屏幕适配的首选
  2. AutoLayout在Xib上使用非常简单灵活,代码实现非常繁琐
  3. 自动布局非常灵活,想要熟练需要多用多练

重要的事情多说一遍:自动布局非常灵活,想要熟练需要多用多练

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,271评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,275评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,151评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,550评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,553评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,559评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,924评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,580评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,826评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,578评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,661评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,363评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,940评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,926评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,156评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,872评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,391评论 2 342

推荐阅读更多精彩内容

  • 前言 iPhone自诞生以来,随着其屏幕尺寸不断的多样化,屏幕适配的技术一直在发展更新。目前,iOS系统版本已经更...
    VV木公子阅读 15,348评论 24 170
  • 发现 关注 消息 iOS 第三方库、插件、知名博客总结 作者大灰狼的小绵羊哥哥关注 2017.06.26 09:4...
    肇东周阅读 12,016评论 4 62
  • 剽悍晨读:怎样才能把事情做到最好,获得想要的结果? 在平台期保持进步 晨读中提到了三种类型: ①浅尝辄止者会萌生退...
    超少阅读 295评论 4 6
  • 我分手了有人问我为什么其实也没什么异地恋和他背叛这两个理由够不够 总有一天我们分离的时间会超过我们在一起的时间 会...
    旅人无氧阅读 170评论 0 0
  • 姓名:杨忠诚 公司:慧友冠源科技&272期六项精进努力二组&广东盛和塾稻牙二组 【日精进打卡第105天】' 【知~...
    杨忠诚阅读 223评论 0 0