美文网首页OC基础iOS进阶iOS开发攻城狮的集散地
iOS回顾笔记(06) -- AutoLayout从入门到精通

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

作者: xiaoyouPrince | 来源:发表于2017-03-14 10:59 被阅读541次

    随着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. 自动布局非常灵活,想要熟练需要多用多练

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

    相关文章

      网友评论

        本文标题:iOS回顾笔记(06) -- AutoLayout从入门到精通

        本文链接:https://www.haomeiwen.com/subject/unbcnttx.html