UIScrollView添加AutoLayout约束的坑

这边想在scrollview中添加控件,比如label,button, collectionview等等。但是约束总是不对,查看资料后有了初步的认识。

 

之前在使用AutoLayout给UIScrollView进行布局的时候,总会出现点这样那样莫名其妙的问题.我也曾跳坑两次,挣扎许久最后都以放弃storyboard改为代码实现而告终.今天终得正解,遂拿出来说说.

先从最基础的开始,我们试着在storyboard上添加一个UIScrollView,并且在内部添加一个和它一样大的UIImageView.

首先,拖一个UIScrollView到storyboard,设置约束如下:


顶部距离控制器view距离为50
左侧和右侧距离控制器view距离分别为0
高度固定250(多吉利的数字)

scrollView的约束

然后拖一个UIImageView到scrollView中,我们想让它在{0,0}处并和scrollView的尺寸一样,于是设置上下左右距离父控件都为0:

imageView相对scrollView的约束

好像没什么问题,可是...

约束报错了...

没看错,Xcode告诉我们这样安排约束是错误的.
UIScrollView的子控件添加约束与普通view不同,仅仅这4个约束不足以满足它的需求.

那么,怎样才是正确的做法呢?

首先:

scrollView自身的约束(scrollView的位置和尺寸)可以像正常的UIView一样参照其父控件添加.

正如上面我们第一步所做的,在给scrollView添加子控件之前,那四个约束决定了scrollView的大小和位置,这步是没有问题的.

问题的关键在于如何给scrollView内部的子控件添加约束.

scrollView内部子控件约束的添加需要遵循两个原则:

1、scrollView内部子控件的尺寸不能以scrollView的尺寸为参照
2、scrollView内部的子控件的约束必须完整

首先,子控件的尺寸不能以scrollView的尺寸为参照,那么我们有两种选择:

  • 提供一个具体值的约束(比如200)
  • 子控件的尺寸可以参照scrollView以外其它的控件的尺寸(如控制器的view的尺寸)

其次,约束"完整"的意思是说:子控件在水平及竖直方向上的约束要把scrollView"撑满".

也就是说,在水平方向上,我们需要设置:

  • 子控件左侧与父控件的距离
  • 子控件自身的宽度
  • 子控件右侧距父控件的距离.

竖直方向上也一样,要设置:

  • 子控件顶部距父控件的距离
  • 子控件的高度
  • 子控件底部距父控件的距离.
    如图:

scrollView子视图的约束条件图1

scrollView子视图的约束条件图2

两张图片中,所有红色线条的长度都要确定(黄线表示对齐),才能保证AutoLayout不会报错.

为什么scrollView如此隔路(隔路:特殊,与众不同)呢?

这是因为,scrollView需要根据添加在其内部的子控件的宽高及与四周的距离计算出它的contentSize.

举个栗子:
一个添加在scrollView内部的imageView的宽高为{80, 50}, imageView距离上左下右的距离分别为:100, 200, 300, 400,那么不需要用代码赋值contentSize,我们就可以打印出scrollView的contentSize为{680, 450}.
如图:

IB添加约束的原理

如果理解起来还是有困难,我们可以把scrollView的contentSize的范围想象成一块UIView(上图中的蓝色区域),暂且叫它Container(实际是没有这个东西的).当我们在storyboard或xib中设置子控件与scrollView之间约束时,实际上设置的是子控件与container之间的约束.

也就是说,子控件的约束决定了container的尺寸(contentSize).
这就说明了为什么我们要在水平和竖直方向用约束"撑满".如果不撑满,container不知道它自己应该多大.

也正是因为container的尺寸由子控件的约束决定,所以子控件的尺寸不能再反过来参照container的尺寸.不然你等于我,我等于你,那到底是多少呢?如果你是Xcode你也会抓狂.(再次强调那个container不是真的,只是为了方便理解)

理论部分解释完毕,回到一开始的案例上.

imageView有了上下左右四个约束,还缺少宽高,我们再添加个固定宽高的约束(可以大一点,太小了看不到滚动效果),问题就可以解决了.

强调一下, 用代码给scrollView添加约束(包括使用Masonry的情况)是一样的.

填平一坑,心中大喜.掌握了这些,复杂一点的scrollView布局我们也不怕了.
给大家出个题目,如果是要在scrollView里面添加三张水平方向滚动的图片怎么搞? 参考答案:https://github.com/CoderAO/UIScrollViewAutoLayout

 

那么自己也 编写了例子, 发现的确需要将scrollview中的控件的上下左右和宽高约束都定义完全。而对于button, stackview和collectionview等总是定义好后上边距需要重新定义负值才可以正确到scrollview的中心位置,为什么呢?可能和设置了scrollview的bottom约束有关。下面的例子中实现后没有出现负值约束的设置。

大致步骤官方如下:

1. 添加scrollview

2.为scrollview添加位置和大小的约束

3. 为scrollview添加content view,就是子view

4.添加contenview到scrollveiw的上下左右边距。此刻contentview定义了scroll view的content area

5. (可选)为了让scrollview水平禁止滚动,可以将contentview的宽度设置为同scrollview一致的

6. (可选)为了让scrollview垂直禁止滚动,可以将contentview的高度设置为同scrollview一致的

7. 布局contentview的content内容,并且添加约束

 

 

 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值