原文链接
Auto Layout的可视格式化语言(以下简称VFL)允许使用者通过ASCII-art格式化字符串定义约束。
用一行简单的代码,你可以定义多个水平或垂直方向的约束。对比一个一个加约束,这样可以可以节省很多代码量。
在这个教程中,你可以用VFL做下面这些事情哦:!
注意:建议读者对Auto Layout有充分了解的情况下阅读此文,如果对于Auto Layout不是很熟悉,建议先阅读Auto Layout Tutorial Part 1: Getting Started和Auto Layout Tutorial Part 2: Constraints
首先下载事例工程便于教程使用,该工程提供了一个初级网络社交app-Grapevine的基本欢迎页面。在Xcode中运行工程;你将看到如下画面(在模拟器的Hardware\Rotate Right中旋转屏幕):
好吧,这个页面真是一团乱,为什么这种情况会发生呢?面对这种情况我们应该怎么做呢?
当前界面的所有元素都是跟界面的上边缘(top)和左边缘(left)联系的,这是因为它们没有用Auto Layout约束。通过接下来的教程你会让视图看起来更漂亮。
打开Main.storyboard观察界面元素。注意到这些元素都被设置为在编译期移除Auto Layout约束。你不应该在真实项目中这样使用,但是这会让你节省一些元素的初始化时间。
接下来,打开ViewController.swift。在顶部,你可以看到在Main.storyboard中跟Interface Builder(IB)视图元素联系的outlet和一些在runtime代替约束的属性。
这个时候没啥可以说,但是接下来有一大堆跟VFL有关的东西要学!
在你开始编写布局和约束之前,你需要有一些关于VFL格式化串的相关知识。
第一件要知道的事情:VFL格式化串可以分成如下组成:
接下来一个一个解释VFL格式化串:
VFL使用一系列符号去描述布局
H:|-[icon(==iconDate)]-20-[iconLabel(120@250)]-20@750-[iconDate(>=50)]-|
接下来一步一步解释这个串:
Apple在NSLayoutConstraint提供了类方法constraintsWithVisualFormat去创建约束。你将在Grapevine程序化的创建约束
在Xcode中打开ViewController.swift,并且添加如下代码到viewDidLoad()中:
1 2 3 4 |
appImageView.hidden = true welcomeLabel.hidden = true summaryLabel.hidden = true pageControl.hidden = true |
这些代码会隐藏除了iconImageView,appNameLabel和skipButton之外的元素。运行工程;你会看到如下:
棒!你现在已经清除了烦人的元素了,现在在viewDidLoad()添加如下代码:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 |
// 1 let views = ["iconImageView": iconImageView, "appNameLabel": appNameLabel, "skipButton": skipButton] // 2 var allConstraints = [NSLayoutConstraint]() // 3 let iconVerticalConstraints = NSLayoutConstraint.constraintsWithVisualFormat( "V:|-20-[iconImageView(30)]", options: [], metrics: nil, views: views) allConstraints += iconVerticalConstraints // 4 let nameLabelVerticalConstraints = NSLayoutConstraint.constraintsWithVisualFormat( "V:|-23-[appNameLabel]", options: [], metrics: nil, views: views) allConstraints += nameLabelVerticalConstraints // 5 let skipButtonVerticalConstraints = NSLayoutConstraint.constraintsWithVisualFormat( "V:|-20-[skipButton]", options: [], metrics: nil, views: views) allConstraints += skipButtonVerticalConstraints // 6 let topRowHorizontalConstraints = NSLayoutConstraint.constraintsWithVisualFormat( "H:|-15-[iconImageView(30)]-[appNameLabel]-[skipButton]-15-|", options: [], metrics: nil, views: views) allConstraints += topRowHorizontalConstraints // 7 NSLayoutConstraint.activateConstraints(allConstraints) |
接下来一步步解释上面的代码:
注意:在views字典中的key必须在格式化串中得view串匹配。如果没有,Auto Layout将不能找到引用并且在runtime崩溃。
运行工程,元素现在看起来怎么样?
哈哈,看看是不是已经变得好看多了?
现在把它放着,这不过是个前戏(误)。你还要有一大坨代码要写呢,但是到最后这些都是值得的。
接下来,你需要给剩下的元素布局,首先,你需要把最开始加到viewDidLoad()的代码去掉。不要有怨言,删除下面这些:
1 2 3 4 |
appImageView.hidden = true welcomeLabel.hidden = true summaryLabel.hidden = true pageControl.hidden = true |
这样最开始隐藏的元素就又出现了。
接下来,把当前的views替换成如下的代码:
1 2 3 4 5 6 7 |
let views = ["iconImageView": iconImageView, "appNameLabel": appNameLabel, "skipButton": skipButton, "appImageView": appImageView, "welcomeLabel": welcomeLabel, "summaryLabel": summaryLabel, "pageControl": pageControl] |
现在你已经为appImageView,welcomeLabel,summaryLabel和pageControl添加了视图定义,这些都可以在VFL格式化串中使用。
在activateConstraints()调用之前,在viewDidLoad()中添加如下代码:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 |
// 1 let summaryHorizontalConstraints = NSLayoutConstraint.constraintsWithVisualFormat( "H:|-15-[summaryLabel]-15-|", options: [], metrics: nil, views: views) allConstraints += summaryHorizontalConstraints let welcomeHorizontalConstraints = NSLayoutConstraint.constraintsWithVisualFormat( "H:|-15-[welcomeLabel]-15-|", options: [], metrics: nil, views: views) allConstraints += welcomeHorizontalConstraints // 2 let iconToImageVerticalConstraints = NSLayoutConstraint.constraintsWithVisualFormat( "V:[iconImageView]-10-[appImageView]", options: [], metrics: nil, views: views) allConstraints += iconToImageVerticalConstraints // 3 let imageToWelcomeVerticalConstraints = NSLayoutConstraint.constraintsWithVisualFormat( "V:[appImageView]-10-[welcomeLabel]", options: [], metrics: nil, views: views) allConstraints += imageToWelcomeVerticalConstraints // 4 let summaryLabelVerticalConstraints = NSLayoutConstraint.constraintsWithVisualFormat( "V:[welcomeLabel]-4-[summaryLabel]", options: [], metrics: nil, views: views) allConstraints += summaryLabelVerticalConstraints // 5 let summaryToPageVerticalConstraints = NSLayoutConstraint.constraintsWithVisualFormat( "V:[summaryLabel]-15-[pageControl(9)]-15-|", options: [], metrics: nil, views: views) allConstraints += summaryToPageVerticalConstraints |
接下来一步步解释上面的代码:
运行工程;这些元素看起来怎么样?
现在看起来还不错了哦。错,其中的一些元素的布局是正确的,然后,有些并没有,image和page control并没有居中!
不要害怕,下一节将会告诉你更多关于布局的工具。
Layout Options提供了一个让你在定义约束的时候对视图进行垂线方向上的约束。
使用NSLayoutFormatOptions.AlignAllCenterY是一个使用Layout Options的例子,它可以让view在创建水平约束的时候同时让垂直方向居中。
如果你不想让水平布局的时候垂直方向都居中,而是边对边的话,那就不应该用这个选项。
接下来,让我们看看Layout Options在创建约束的时候是多么有用。移除viewDidLoad()中如下的代码:
1 2 3 4 5 6 7 8 9 10 11 12 13 |
let nameLabelVerticalConstraints = NSLayoutConstraint.constraintsWithVisualFormat( "V:|-23-[appNameLabel]", options: [], metrics: nil, views: views) allConstraints += nameLabelVerticalConstraints let skipButtonVerticalConstraints = NSLayoutConstraint.constraintsWithVisualFormat( "V:|-20-[skipButton]", options: [], metrics: nil, views: views) allConstraints += skipButtonVerticalConstraints |
你刚刚移除了appNameLabel和skipButton的垂直布局。作为替代,你将用Layout Options去给它们添加垂直约束。
找到创建topRowHorizontalConstraints的代码并且设置options为[.AlignAllCenterY]。看起来是这个样子的:
1 2 3 4 5 |
let topRowHorizontalConstraints = NSLayoutConstraint.constraintsWithVisualFormat( "H:|-15-[iconImageView(30)]-[appNameLabel]-[skipButton]-15-|", options: [.AlignAllCenterY], metrics: nil, views: views) |
添加NSLayoutFormatOption .AlignAllCenterY对上面格式化串中的所有视图都有效,并且创建了一个它们垂直方向中心的约束。如果iconImageView提前创建了包含高度的垂直约束也是有效的。因此,appNameLabel和skipButton同iconImageView一样垂直居中。
如果你现在运行,布局看起来可能没有改变,但是代码变得更棒了。移除创建welcomeHorizontalConstraints和将它放进数组的代码。这样就移除了welcomeLabel的水平约束。接下来,更新创建summaryLabelVerticalConstraints的Layout Options:
1 2 3 4 |
summaryLabelVerticalConstraints = NSLayoutConstraint.constraintsWithVisualFormat("V:[welcomeLabel]-4-[summaryLabel]", options: [.AlignAllLeading, .AlignAllTrailing], metrics: nil, views: views); |
这个代码增加了NSLayoutFormatOptions的.NSLayoutFormatOptions和.AlignAllTrailing,welcomeLabel和summaryLabel’s的头边缘和尾边缘会距离它们的父视图的边缘15pt。由于提前为summaryLabel定义了水平约束,所以上述代码才会有效。虽然上面的代码带来的是同样的效果,但是实现起来更加优雅了。
接下来,更新你在创建summaryToPageVerticalConstraints时候的选项:
1 2 3 4 5 |
let pageControlVerticalConstraints = NSLayoutConstraint.constraintsWithVisualFormat( "V:[summaryLabel]-15-[pageControl(9)]-15-|", options: [.AlignAllCenterX], metrics: nil, views: views) |
这样就添加了沿x轴中心对齐。同样为imageToWelcomeVerticalConstraints添加选项:
1 2 3 4 5 |
let imageToWelcomeVerticalConstraints = NSLayoutConstraint.constraintsWithVisualFormat( "V:[appImageView]-10-[welcomeLabel]", options: [.AlignAllCenterX], metrics: nil, views: views) |
运行工程,看看发生了什么?
感觉都居中了是吧?Layout Options让你做出了一个更棒的交互界面。
###NSLayoutFormat选项快速参考
下面是在Grapevine中使用过的属性:
(译者:由于有些种类的文字是从右到左书写的,所以它们的.AlignAllLeading等价于.AlignAllRight,而对于中文来说,.AlignAllLeading等价于.AlignAllLeft)
下面是剩余的一些属性:
你同样可以在文档详细查看。
注意:为了让Layout Options有效,至少要有一个元素定义过垂直方向的约束。看下面的例子:
1 2 3 4 5 |
NSLayoutConstraints.constraintsWithVisualFormat( "V:[topView]-[middleView]-[bottomView]", options: [.AlignAllLeading], metrics: nil, views: ["topView": topView, "middleView": middleView, "bottomView":"bottomView"]) |
topView,middleView或者bottomView其中一个必须要有一个约束来布局它们的头缘,这样Auto Layout才会正确的产生正确的约束。
接下来学习新的概念!Metrics
Metrics是一个能在VFL格式化串中出现的以number为value的字典。如果你需要让距离变得标准化或者有些距离需要计算所以不能直接放在格式化串中的话,Metrics将会变得非常有用!
将如下常量声明在ViewController.swift的变量之上:
1 2 |
// MARK: - Constants private let horizontalPadding: CGFloat = 15.0 |
现在你有了一个用于padding的常量,你可以创建一个metrics字典并且将这个常量使用进去。将如下代码添加到views声明的上面:
1 2 |
let metrics = ["hp": horizontalPadding, "iconImageViewWidth": 30.0] |
上面的代码创建的字典中的key可以再格式化串中使用。
接下来,用如下代码代替topRowHorizontalConstraints和summaryHorizontalConstraints的定义:
1 2 3 4 5 6 7 8 9 10 11 |
let horizontalConstraints = NSLayoutConstraint.constraintsWithVisualFormat( "H:|-hp-[iconImageView(iconImageViewWidth)]-[appNameLabel]-[skipButton]-hp-|", options: [.AlignAllCenterY], metrics: metrics, views: views) let summaryHorizontalConstraints = NSLayoutConstraint.constraintsWithVisualFormat( "H:|-hp-[summaryLabel]-hp-|", options: [], metrics: metrics, views: views) |
现在你已经将格式化串中得硬代码用metrics字典中keys代替掉了。
Auto Layout可以进行串替换,将metrics字典中的value替换到格式化串中的key。所以最终,hp将会被替换成15pt,iconImageViewWidth将会被替换成30pt。
你将一个重复出现的莫名其妙的数字变成了一个优雅的变量。如果你想要改变padding,现在就只需要做一件事了。这不是更好吗?metrics字典并不仅限制于常量;如果你需要在runtime期间进行计算,同样可以把这种变量放到metrics中。
最后的一点小问题是如果你想把这些元素放进UINavigationController或者UITabBarController中,那该怎么办呢?
视图控制器有两个可用的Layout Guides:
它们都指定了试图控制器的视图中顶部或者底部导航栏边缘的位置,但是在Grapevine中,唯一的导航栏边缘是从状态栏开始的。
更新iconVerticalConstraints的声明代码:
1 2 3 4 5 |
let verticalConstraints = NSLayoutConstraint.constraintsWithVisualFormat( "V:|-[iconImageView(30)]", options: [], metrics: nil, views: views) |
这样你就把状态栏和iconImageView之间的20pt的距离移除了,运行代码:
现在你的状态栏覆盖掉了视图上的一些元素。在横屏模式时,iOS为了给小屏幕设备提供更多的有效空间移除状态栏,这样iconImageView会紧靠在屏幕的上方。
使用topLayoutGuide将会解决这种问题,用如下代码代替views字典:
1 2 3 4 5 6 7 8 9 |
let views: [String: AnyObject] = ["iconImageView": iconImageView, "appNameLabel": appNameLabel, "skipButton": skipButton, "appImageView": appImageView, "welcomeLabel": welcomeLabel, "summaryLabel": summaryLabel, "pageControl": pageControl, "topLayoutGuide": topLayoutGuide, "bottomLayoutGuide": bottomLayoutGuide] |
这次增加了topLayoutGuide和bottomLayoutGuide,它们继承自UILayoutSupport,比不是UIView。
接下来,就可以使用layout guides去对齐界面元素了。更新iconVerticalConstraints的声明:
1 2 3 4 5 |
let verticalConstraints = NSLayoutConstraint.constraintsWithVisualFormat( "V:[topLayoutGuide]-[iconImageView(30)]", options: [], metrics: nil, views: views) |
接下来运行工程,完美!
现在你的顶部的界面元素都依赖着topLayoutGuide布局并且无论在横屏或者竖屏模式下状态栏的展现都控制着布局。
在这一节,你已经学会了当界面存在状态栏的时候如何利用topLayoutGuide来控制界面元素的布局。如果你的视图控制器在UINavigationController中,topLayoutGuide将会包含状态栏和UINavigationBar的状态。同时,如果你的试图控制器在UITabBarController中,bottomLayoutGuide将会提供底部边缘的状态。
VFL让你用一行代码写出了多个约束,大大降低了手指的负担。但是对于当前的实现,还存在一些限制;还有一些重要的东西需要理解。
在Grapevine中,你用了.AlignAllCenterY和.AlignAllCenterX。
使用这些表示你让一些视图和其他的一些视图的垂直中心或者水平中心对齐,然而只有在这些视图中存在已经有足够约束能够确定它们的水平和垂直中心位置的时候才能变得有效。
即使现在通过VFL你可以用一些小把戏来处理中心视图,但是这也不保证在将来的版本中依然有效。
####使用约束中的Multiplier
通过Multiplier,你可以通过比例来对视图进行布局,比如你可以让一个label的宽度是它父视图的60%。由于VFL会同时创建多个没有名字的约束,所以不能通过格式化串来设置百分比系数。
注意:你可以通过constraintsWithVisualFormat返回的数组来遍历约束,但是你需要去确定它们的NSLayoutAttribute属性,这样才能正确的设定Multiplier,但即使是这样,你依然需要替换这些约束,因为约束的Multiplier是不可变的。
你可以下载完整的工程。
注意:如果你有多个工程使用相同的bundle id,Xcode可能会出现问题。所以如果你完成了这个教程并且想最后运行一下刚才下载的工程,你可以使用shift+option+command+K清空一下build目录。
现在你已经知道VFL如何工作啦,你已经可以在你的界面中使用这种布局咯。
你已经知道了如何使用layout options 来减少需要定义的约束。你也已经知道如何使用metrics来在runtime定义距离而不仅仅是编译期。最后,你也知道了VFL的一些限制,但是利大于弊,你应该好好的利用它。
如果你对该教程或者Auto Layout有什么问题或者建议的话,请留言!
转自:http://mmmmmax.wang/2015/12/11/Auto-Layout-Visual-Format-Language-Tutorial/