Scroll View 深刻

转载自:http://mobile.51cto.com/hot-430409.htmphp

 

可能你很难相信,UIScrollView和一个标准的UIView差别并不大,scroll view确实会多一些方法,但这些方法只是UIView一些属性的表面而已。所以,要想弄懂UIScrollView是怎么工做以前,你须要了解 UIView,特别是视图渲染过程的两步。app

光栅化和组合工具

渲染过程的第一部分是众所周知的光栅化,光栅化简单的说就是产生一组绘图指令而且生成一张图片。好比绘制一个圆角矩形、带图片、标题居中的UIButtons。这些图片并无被绘制到屏幕上去;取而代之的是,他们被本身的视图保持着留到下一个步骤用。布局

一旦每一个视图都产生了本身的光栅化图片,这些图片便被一个接一个的绘制,并产生一个屏幕大小的图片,这即是上文所说的组合。视图层级(view hierarchy)对于组合如何进行扮演了很重要的角色:一个视图的图片被组合在它父视图图片的上面。而后,组合好的图片被组合到父视图的父视图图片上 面,就这样。。。最终视图层级最顶端是窗口(window),它组合好的图片即是咱们看到的东西了。spa

概念上,依次在每一个视图上放置独立分层的图片并最终产生一个图片,单调的图像将会变得更容易理解,特别是若是你之前使用过像Photoshop这样的工具。咱们还有另一篇文章详细解释了像素是如何绘制到屏幕上去的3d

如今,回想一下,每一个视图都有一个bounds和frame。当布局一个界面时,咱们须要处理视图的frame。这容许咱们放置并设置视图的大小。 视图的frame和bounds的大小老是同样的,可是他们的origin有可能不一样。弄懂这两个工做原理是理解UIScrollView的关键。orm

在光栅化步骤中,视图并不关心即将发生的组合步骤。也就是说,它并不关心本身的frame(这是用来放置视图的图像)或本身在视图层级中的位置(这 是决定组合的顺序)。这时视图只关心一件事就是绘制它本身的content。这个绘制发生在每一个视图的drawRect:方法中。htm

在drawRect:方法被调用前,会为视图建立一个空白的图片来绘制content。这个图片的坐标系统是视图的bounds。几乎每一个视图 bounds的origin都是{0,0}。所以,当在删格化图片左上角绘制一些东西的时候,你都会在bounds的origin({x:0,y:0}) 处绘制。在一个图片右下角的地方绘制东西的时候,你都会绘制在{x:width, y:height}处。若是你的绘制超出了视图的bounds,那么超出的部分就不属于删格化图片的部分了,而且会被丢弃。blog

在组合的步骤中,每一个视图将本身光栅化图片组合到本身父视图的光栅化图片上面。视图的frame决定了本身在父视图中绘制的位置,frame的 origin代表了视图光栅化图片左上角相对父视图光栅化图片左上角的偏移量。因此,一个origin为{x:20,y:15}的frame所绘制的图片 左边距其父视图20点,上边距父视图15点。由于视图的frame和bounds矩形的大小老是同样的,因此光栅化图片组合的时候是像素对齐的。这确保了 光栅化图片不会被拉伸或缩小。图片

记住,咱们才仅仅讨论了一个视图和它父视图之间的组合操做。一旦这两个视图被组合到一块儿,组合的结果图片将会和父视图的父视图进行组合。。。这是一个雪球效应。

考虑一下组合图片背后的公式。视图图片的左上角会根据它frame的origin进行偏移,并绘制到父视图的图片上:

  1. CompositedPosition.x = View.frame.origin.x - Superview.bounds.origin.x;  
  2.    
  3. CompositedPosition.y = View.frame.origin.y - Superview.bounds.origin.y;  

咱们能够经过几个不一样的frames看一下: 

这样作是有道理的。咱们改变button的frame.origin后,它会改变本身相对紫色父视图的位置。注意,若是咱们移动button直到它 的一部分已经在紫色父视图bounds的外面,当光栅化图片被截去时这部分也将会经过一样的绘制方式被截去。然而,技术上讲,由于iOS处理组合方法的原 因,你能够将一个子视图渲染在其父视图的bounds以外,可是光栅化期间的绘制不可能超出一个视图的bounds。

Scroll View的Content Offset

如今,咱们所讲的跟UIScrollView有什么关系呢?一切都和它有关!考虑一种咱们能够实现的滚动:咱们有一个拖动时frame不断改变的视 图。这达到了相同的效果,对吗?若是我拖动个人手指到右边,那么拖动的同时我增大视图的origin.x,瞧,这货就是scroll view。

固然,在scroll view中有不少具备表明性的视图。为了实现这个平移功能,当用户移动手指时,你须要时刻改变每一个视图的frames。当咱们提出组合一个view的光栅化图片到它父视图什么地方时,记住这个公式:

  1. CompositedPosition.x = View.frame.origin.x - Superview.bounds.origin.x;  
  2.    
  3. CompositedPosition.y = View.frame.origin.y - Superview.bounds.origin.y;  

咱们减小Superview.bounds.origin的值(由于他们老是0)。可是若是他们不为0呢?咱们用和前一个图例相同的frames,可是咱们改变了紫色视图bounds的origin为{-30,-30}。获得下图:

如今,巧妙的是经过改变这个紫色视图的bounds,它每个单独的子视图都被移动了。事实上,这正是一个scroll view工做的原理。当你设置它的contentOffset属性时:它改变scroll view.bounds的origin。事实上,contentOffset甚至不是实际存在的。代码看起来像这样:

  1. - (void)setContentOffset:(CGPoint)offset  
  2. {  
  3.     CGRect bounds = [self bounds];  
  4.     bounds.origin = offset;  
  5.     [self setBounds:bounds];  
  6. }  

注意:前一个图例,只要足够的改变bounds的origin,button将会超出紫色视图和button组合成的图片的范围。这也是当你足够的移动scroll view时,一个视图会消失!

世界之窗:Content Size

如今,最难的部分已通过去了,咱们再看看UIScrollView另外一个属性:contentSize。scroll view的content size并不会改变其bounds的任何东西,因此这并不会影响scroll view如何组合本身的子视图。反而,content size定义了可滚动区域。scroll view的默认content size为{w:0,h:0}。既然没有可滚动区域,用户是不能够滚动的,可是scroll view任然会显示其bounds范围内全部的子视图。

当content size设置为比bounds大的时候,用户就能够滚动视图了。你能够认为scroll view的bounds为可滚动区域上的一个窗口:

当content offset为{x:0,y:0}时,可见窗口的左上角在可滚动区域的左上角处。这也是content offset的最小值;用户不能再往可滚动区域的左边或上边移动了。那儿没啥,别滚了!

content offset的最大值是content size和scroll view size的差。这也在情理之中:从左上角一直滚动到右下角,用户中止时,滚动区域右下角边缘和滚动视图bounds的右下角边缘是齐平的。你能够像这样记 下content offset的最大值:

  1. contentOffset.x = contentSize.width - bounds.size.width;  
  2. contentOffset.y = contentSize.height - bounds.size.height;  

用Content Insets对窗口稍做调整

contentInset属性能够改变content offset的最大和最小值,这样即可以滚动出可滚动区域。它的类型为UIEdgeInsets,包含四个值: {top,left,bottom,right}。当你引进一个inset时,你改变了content offset的范围。好比,设置content inset顶部值为10,则容许content offset的y值达到10。这介绍了可滚动区域周围的填充。

这咋一看好像没什么用。实际上,为何不只仅增长content size呢?除非没办法,不然你须要避免改变scroll view的content size。想要知道为何?想一想一个table view(UItableView是UIScrollView的子类,因此它有全部相同的属性),table view为了适应每个cell,它的可滚动区域是经过精心计算的。当你滚动通过table view的第一个或最后一个cell的边界时,table view将content offset弹回并复位,因此cells又一次恰到好处的紧贴scroll view的bounds。

当你想要使用UIRefreshControl实现拉动刷新时发生了什么?你不能在table view的可滚动区域内放置UIRefreshControl,不然,table view将会容许用户经过refresh control中途中止滚动,而且将refresh control的顶部弹回到视图的顶部。所以,你必须将refresh control放在可滚动区域上方。这将容许首先将content offset弹回第一行,而不是refresh control。

可是等等,若是你经过滚动足够多的距离初始化pull-to-refresh机制,由于table view设置了content inset,这将容许content offset将refresh control弹回到可滚动区域。当刷新动做被初始化时,content inset已经被校订过,因此content offset的最小值包含了完整的refresh control。当刷新完成后,content inset恢复正常,content offset也跟着适应大小,这里并不须要为content size作数学计算。(这里可能比较难理解,建议看看EGOTableViewPullRefresh这样的类库就应该明白了)

如何在本身的代码中使用content inset?当键盘在屏幕上时,有一个很好的用途:你想要设置一个紧贴屏幕的用户界面。当键盘出如今屏幕上时,你损失了几百个像素的空间,键盘下面的东西全都被挡住了。

如今,scroll view的bounds并无改变,content size也并无改变(也不须要改变)。可是用户不能滚动scroll view。考虑一下以前一个公式:content offset的最大值并不一样于content size和bounds的大小。若是他们相等,如今content offset的最大值是{x:0,y:0}.

如今开始出绝招,将界面放入一个scroll view。scroll view的content size仍然和scroll view的bounds同样大。当键盘出如今屏幕上时,你设置content inset的底部等于键盘的高度。

这容许在content offset的最大值下显示滚动区域外的区域。可视区域的顶部在scroll view bounds的外面,所以被截取了(虽然它在屏幕以外了,但这并无什么)。

希望这能让你理解一些滚动视图内部工做的原理,你对缩放感兴趣?好吧,咱们今天不会谈论它,可是这儿有一个有趣的小窍门:检查 viewForZoomingInScrollView:方法返回视图的transform属性。你将再次发现scroll view只是聪明的利用了UIView已经存在的属性。

相关文章
相关标签/搜索