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

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

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

适配的概念

适配主要分两种

  • 系统适配

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

  • 屏幕适配

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

常见设备的分辨率:

屏幕适配发展史

  • 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


外部四根线

外部四根线分别表示上、下、左、右四个方向,子控件相对于父控件的距离。
实线:表示固定位置
虚线:表示非固定位置 **内部两根线** 内部两根线分别表示水平和竖直方向,子控件是否根据父控件等比例缩放。
实线:该方向上跟随父控件等比缩放
虚线:该方向上不跟随父控件等比缩放 - **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 : 对齐方式

  • Add New Constraint: 添加新的约束

  • Resolve Auto Layout Issues : 修复自动布局问题

AutoLayout的使用原则

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

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


    2.如约束不够,无法确定View的frame时候也会报错:

  • AutoLayout的警告:警告一般是Xib中添加好了约束,但是对应View没有移动到对应位置导致。这种情况程序运行起来之后会是正确约束的样子,而不是Xib中的样子。

    • 警告解决办法:
      1.选中View,update frame 或使用快捷键‘command’ + ‘option’ + ‘=’
      2.如图:
  • 约束的修改:
    约束的修改有很多种方式,下面列举一种我常用的方式!

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

AutoLayout案例练习

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

练习1

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

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

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

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

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

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

效果:

练习2

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

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

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

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

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

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

效果图:

练习3

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

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

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

最终效果如图:

AutoLayout 中的UILabel

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

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

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

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

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

练习4

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

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

最终效果图:

这只是几个简单的小练习,若想使用熟练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. 两个父子控件之间的约束要添加到父控件上

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

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

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

header{font-size:1em;padding-top:1.5em;padding-bottom:1.5em}
.markdown-body{overflow:hidden}
.markdown-body>div,.markdown-body>article{width:100%}
aside.sidebar{float:none;padding:0 18px 1px;background-color:#f7f7f7;border-top:1px solid #e0e0e0}
.flex-content,article img,article video,article .flash-video,aside.sidebar img{max-width:100%;height:auto}
.basic-alignment.left,article img.left,article video.left,article .left.flash-video,aside.sidebar img.left{float:left;margin-right:1.5em}
.basic-alignment.right,article img.right,article video.right,article .right.flash-video,aside.sidebar img.right{float:right;margin-left:1.5em}
.basic-alignment.center,article img.center,article video.center,article .center.flash-video,aside.sidebar img.center{display:block;margin:0 auto 1.5em}
.basic-alignment.left,article img.left,article video.left,article .left.flash-video,aside.sidebar img.left,.basic-alignment.right,article img.right,article video.right,article .right.flash-video,aside.sidebar img.right{margin-bottom:.8em}
.toggle-sidebar,.no-sidebar .toggle-sidebar{display:none}

.markdown-body img,.markdown-body video,.markdown-body .flash-video{ -webkit-border-radius:0.3em;-moz-border-radius:0.3em;-ms-border-radius:0.3em;-o-border-radius:0.3em;border-radius:0.3em;-webkit-box-shadow:rgba(0,0,0,0.15) 0 1px 4px;-moz-box-shadow:rgba(0,0,0,0.15) 0 1px 4px;box-shadow:rgba(0,0,0,0.15) 0 1px 4px;-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box;border:#fff 0.5em solid;}

.markdown-body img,.markdown-body video{max-width: 100%;}
.markdown-body video,.markdown-body .flash-video{margin:0 auto 1.5em}
.markdown-body video{display:block;width:100%}
.markdown-body .flash-video>div{position:relative;display:block;padding-bottom:56.25%;padding-top:1px;height:0;overflow:hidden}
.markdown-body .flash-video>div iframe,.markdown-body .flash-video>div object,.markdown-body .flash-video>div embed{position:absolute;top:0;left:0;width:100%;height:100%}
.markdown-body>footer{padding-bottom:2.5em;margin-top:2em}
.markdown-body>footer p.meta{margin-bottom:.8em;font-size:.85em;clear:both;overflow:hidden}

body,pre{ background:#fdf6e3 url('') top left;}
body{ background-color: #f8f8f8;}
pre{-webkit-border-radius:0.4em;-moz-border-radius:0.4em;-ms-border-radius:0.4em;-o-border-radius:0.4em;border-radius:0.4em;border:1px solid #e7dec3;line-height:1.45em;font-size:13px;margin-bottom:2.1em;padding:.8em 1em;color:#586e75;overflow:auto}
.markdown-body code{background: none;}
h3.filename+pre{-moz-border-radius-topleft:0px;-webkit-border-top-left-radius:0px;border-top-left-radius:0px;-moz-border-radius-topright:0px;-webkit-border-top-right-radius:0px;border-top-right-radius:0px}
p code,li code{display:inline-block;white-space:no-wrap;background:#fff;font-size:.8em;line-height:1.5em;color:#555;border:1px solid #ddd;-webkit-border-radius:0.4em;-moz-border-radius:0.4em;-ms-border-radius:0.4em;-o-border-radius:0.4em;border-radius:0.4em;padding:0 .3em;margin:-1px 0}
p pre code,li pre code{font-size:1em !important;background:none;border:none}

/*

Orginal Style from ethanschoonover.com/solarized (c) Jeremy Hull

*/

.hljs {
display: block;
overflow-x: auto;
padding: 0.5em;
background: #fdf6e3;
color: #657b83;
-webkit-text-size-adjust: none;
}

.hljs-comment,
.diff .hljs-header,
.hljs-doctype,
.hljs-pi,
.lisp .hljs-string {
color: #93a1a1;
}

/* Solarized Green */
.hljs-keyword,
.hljs-winutils,
.method,
.hljs-addition,
.css .hljs-tag,
.hljs-request,
.hljs-status,
.nginx .hljs-title {
color: #859900;
}

/* Solarized Cyan */
.hljs-number,
.hljs-command,
.hljs-string,
.hljs-tag .hljs-value,
.hljs-rule .hljs-value,
.hljs-doctag,
.tex .hljs-formula,
.hljs-regexp,
.hljs-hexcolor,
.hljs-link_url {
color: #2aa198;
}

/* Solarized Blue */
.hljs-title,
.hljs-localvars,
.hljs-chunk,
.hljs-decorator,
.hljs-built_in,
.hljs-identifier,
.vhdl .hljs-literal,
.hljs-id,
.css .hljs-function,
.hljs-name {
color: #268bd2;
}

/* Solarized Yellow */
.hljs-attribute,
.hljs-variable,
.lisp .hljs-body,
.smalltalk .hljs-number,
.hljs-constant,
.hljs-class .hljs-title,
.hljs-parent,
.hljs-type,
.hljs-link_reference {
color: #b58900;
}

/* Solarized Orange */
.hljs-preprocessor,
.hljs-preprocessor .hljs-keyword,
.hljs-pragma,
.hljs-shebang,
.hljs-symbol,
.hljs-symbol .hljs-string,
.diff .hljs-change,
.hljs-special,
.hljs-attr_selector,
.hljs-subst,
.hljs-cdata,
.css .hljs-pseudo,
.hljs-header {
color: #cb4b16;
}

/* Solarized Red */
.hljs-deletion,
.hljs-important {
color: #dc322f;
}

/* Solarized Violet */
.hljs-link_label {
color: #6c71c4;
}

.tex .hljs-formula {
background: #eee8d5;
}
-->

iOS回顾笔记(06) -- AutoLayout从入门到精通的更多相关文章

  1. iOS回顾笔记(08) -- 自定义Cell的类型和创建步骤总结

    iOS回顾笔记(08) -- 自定义Cell的类型和创建步骤总结 项目中我们常见的自定义cell主要分为两种 等高cell:如应用列表.功能列表 非等高cell:如微博列表.QQ聊天页面 下面对这 ...

  2. iOS回顾笔记( 01 )

    html,body,div,span,applet,object,iframe,h1,h2,h3,h4,h5,h6,p,blockquote,pre,a,abbr,acronym,address,bi ...

  3. iOS回顾笔记( 02 ) -- 由九宫格布局引发的一系列“惨案”

    html,body,div,span,applet,object,iframe,h1,h2,h3,h4,h5,h6,p,blockquote,pre,a,abbr,acronym,address,bi ...

  4. iOS回顾笔记(03) -- 自定义View的封装和xib文件的使用详解

    iOS回顾笔记(03) -- 自定义View的封装和xib文件的使用详解 iOS开发中,我们常常将一块View封装起来,以便于统一管理内部的子控件.如iOS回顾笔记(02)中的"书" ...

  5. iOS回顾笔记(04) -- UIScrollView的基本使用详解

    html,body,div,span,applet,object,iframe,h1,h2,h3,h4,h5,h6,p,blockquote,pre,a,abbr,acronym,address,bi ...

  6. iOS回顾笔记(05) -- 手把手教你封装一个广告轮播图框架

    html,body,div,span,applet,object,iframe,h1,h2,h3,h4,h5,h6,p,blockquote,pre,a,abbr,acronym,address,bi ...

  7. iOS回顾笔记(07) -- UITableView的使用和性能优化

    iOS回顾笔记(07) -- UITableView的使用和性能优化 如果问iOS中最重要的最常用的UI控件是什么,我觉得UITableView当之无愧!似乎所有常规APP都使用到了UITableVi ...

  8. iOS回顾笔记(09) -- Cell的添加、删除、更新、批量操作

    iOS回顾笔记(09) -- Cell的添加.删除.更新.批量操作 项目中经常有对UITableViewCell做各种操作的需求: 添加一个新的cell 删除某行cell 刷新cell上某行数据(如修 ...

  9. iOS学习笔记06—Category和Extension

    iOS学习笔记06—Category和Extension 一.概述 类别是一种为现有的类添加新方法的方式. 利用Objective-C的动态运行时分配机制,Category提供了一种比继承(inher ...

随机推荐

  1. SVGEditor

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  2. java算法 蓝桥杯 扶老奶奶街

    一共有5个红领巾,编号分别为A.B.C.D.E,老奶奶被他们其中一个扶过了马路. 五个红领巾各自说话: A :我和E都没有扶老奶奶 B :老奶奶是被C和E其中一个扶过大街的 C :老奶奶是被我和D其中 ...

  3. ubuntu linux 设置环境变量

    添加环境变量 1.添加临时变量 终端中输入: PATH="$PATH:yourpath" :yourpath是要添加的环境变量(即要添加目录的绝对路径,例:/home/myprog ...

  4. 关于a.b和a[b]的区别

    在写代码的过程中,我们经常可以看到a.b或啊a[b],但是他们有什么区别呢: 简单说一下吧,有自己的还用群友的大力支持! 在js的对象中 var arr = {a:"b",b:&q ...

  5. [2017.02.07] Lua入门学习记录

    #!/home/auss/Projects/Qt/annotated/lua -- 这是第一次系统学习Lua语言 --[[ 参考资料: 1. [Lua简明教程](http://coolshell.cn ...

  6. 源码(08) -- java.util.ListIterator<E>

    java.util.ListIterator<E> 源码分析(JDK1.7) ------------------------------------------------------- ...

  7. BOM之history对象

    前面的话 history对象保存着用户上网的历史记录,从窗口被打开的那一刻算起.由于安全方面的考虑,开发人员无法得到用户浏览器的URL,但借由用户访问过的页面列表,可以在不知道实际URL的情况下实现后 ...

  8. MyEclipse - 解决 MyEclipse build workspace慢,validation javascript更慢的问题

    在这个过程中对.projet文件进行了跟踪比对,总算发现这个Build的时候进行Validation是从哪里定义的了.似乎因为我的项目是基于ExtJS2.0.2的web project,所以会提示打开 ...

  9. substring和substr、$.extend()、$.fn.extend()、(function($){….})(jQuery)的简易讲解

    1.    JS中substring与substr的区别 Substring: 该方法可以有一个参数也可以有两个参数. l  一个参数: 示例: var str="Olive": ...

  10. Android之RecyclerView轻松实现下拉刷新和加载更多

    今天研究了下RecyclerView的滑动事件,特别是下拉刷新和加载更多事件,在现在几乎所有的APP显示数据列表时都用到了.自定义RecyclerView下拉刷新和加载更多听上去很复杂,实际上并不难, ...