网页在Safari快速滚动和回弹的原理: -webkit-overflow-scrolling : touch;的实现

现在很多for Mobile的HTML5网页内都有快速滚动和回弹的效果,看上去和原生app的效率都有得一拼。

要实现这个效果很简单,只需要加一行css代码即可:

[css]  view plain  copy
  1. -webkit-overflow-scrolling : touch;  
可用以下网页测试:

[html]  view plain  copy
  1. <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">  
  2. <html>  
  3.     <head>  
  4.         <meta charset="utf-8" />  
  5.         <title>scroll</title>  
  6.         <style type="text/css">  
  7.             .container {  
  8.                 width : 300px;  
  9.                 height : 50%;  
  10.                 -webkit-box-sizing : border-box;  
  11.                 position : relative;  
  12.                 overflow-y : auto;  
  13.                 background-color : cyan;  
  14.                 -webkit-overflow-scrolling : touch;  /* liuhx:可以把这整行注释掉对比差别 */  
  15.             }  
  16.             ul {  
  17.                 height: 50px;  
  18.             }  
  19.         </style>  
  20.     </head>  
  21. <body>  
  22.     <div align="center">  
  23.         <nav class="container">  
  24.             <ul>1</ul>  
  25.             <ul>2</ul>  
  26.             <ul>3</ul>  
  27.             <ul>4</ul>  
  28.             <ul>5</ul>  
  29.             <ul>6</ul>  
  30.             <ul>7</ul>  
  31.             <ul>8</ul>  
  32.             <ul>9</ul>  
  33.             <ul>10</ul>  
  34.             <ul>11</ul>  
  35.             <ul>12</ul>  
  36.             <ul>13</ul>  
  37.             <ul>14</ul>  
  38.             <ul>15</ul>  
  39.             <ul>16</ul>  
  40.             <ul>17</ul>  
  41.             <ul>18</ul>  
  42.             <ul>19</ul>  
  43.             <ul>20</ul>  
  44.         </nav>  
  45.     </div>  
  46. </body>  
  47. </html>  
可以用手指滑动中间的蓝色区域,会发现回弹效果以及滚动得很快:

网页在Safari快速滚动和回弹的原理: -webkit-overflow-scrolling : touch;的实现_第1张图片
网页在Safari快速滚动和回弹的原理: -webkit-overflow-scrolling : touch;的实现_第2张图片(点击图片查看大图)
如果把-webkit-overflow-scrolling那行注释掉,就会发现滚动得很慢。


实际上,Safari真的用了原生控件来实现,对于有-webkit-overflow-scrolling的网页,会创建一个UIScrollView,提供子layer给渲染模块使用。创建时的堆栈如下:

[plain]  view plain  copy
  1. Thread 1, Queue : com.apple.main-thread  
  2. #0  0x00086723 in -[UIScrollView initWithFrame:] ()  
  3. #1  0x004ec3bd in -[UIWebOverflowScrollView initWithLayer:node:webDocumentView:] ()  
  4. #2  0x001f1769 in -[UIWebDocumentView webView:didCreateOrUpdateScrollingLayer:withContentsLayer:scrollSize:forNode:allowHorizontalScrollbar:allowVerticalScrollbar:] ()  
  5. #3  0x01d571bd in __invoking___ ()  
  6. #4  0x01d570d6 in -[NSInvocation invoke] ()  
  7. #5  0x01d5724a in -[NSInvocation invokeWithTarget:] ()  
  8. #6  0x027fb6a1 in -[_WebSafeForwarder forwardInvocation:] ()  
  9. #7  0x027fb8ab in __44-[_WebSafeAsyncForwarder forwardInvocation:]_block_invoke_0 ()  
  10. #8  0x04ac753f in _dispatch_call_block_and_release ()  
  11. #9  0x04ad9014 in _dispatch_client_callout ()  
  12. #10 0x04ac97d5 in _dispatch_main_queue_callback_4CF ()  
  13. #11 0x01d09af5 in __CFRunLoopRun ()  
  14. #12 0x01d08f44 in CFRunLoopRunSpecific ()  
  15. #13 0x01d08e1b in CFRunLoopRunInMode ()  
  16. #14 0x01cbd7e3 in GSEventRunModal ()  
  17. #15 0x01cbd668 in GSEventRun ()  
  18. #16 0x00032ffc in UIApplicationMain ()  
  19. #17 0x00002ae2 in main at /Users/liuhx/Desktop/UIWebView_Research/WebViewResearch/main.mm:16  
实际创建的是UIWebOverflowScrollView,它继承自UIScrollView,声明为:

[cpp]  view plain  copy
  1. @class DOMNode, UIWebDocumentView, UIWebOverflowContentView, UIWebOverflowScrollListener;  
  2.   
  3. @interface UIWebOverflowScrollView : UIScrollView  
  4. {  
  5.     UIWebDocumentView *_webDocumentView;  
  6.     UIWebOverflowScrollListener *_scrollListener;  
  7.     UIWebOverflowContentView *_overflowContentView;  
  8.     DOMNode *_node;  
  9.     BOOL _beingRemoved;  
  10. }  
  11.   
  12. @property(nonatomic, getter=isBeingRemoved) BOOL beingRemoved; // @synthesize beingRemoved=_beingRemoved;  
  13. @property(retain, nonatomic) DOMNode *node; // @synthesize node=_node;  
  14. @property(retain, nonatomic) UIWebOverflowContentView *overflowContentView; // @synthesize overflowContentView=_overflowContentView;  
  15. @property(retain, nonatomic) UIWebOverflowScrollListener *scrollListener; // @synthesize scrollListener=_scrollListener;  
  16. @property(nonatomic) UIWebDocumentView *webDocumentView; // @synthesize webDocumentView=_webDocumentView;  
  17. - (void)setContentOffset:(struct CGPoint)arg1;  
  18. - (void)_replaceLayer:(id)arg1;  
  19. - (void)prepareForRemoval;  
  20. - (void)fixUpViewAfterInsertion;  
  21. - (id)superview;  
  22. - (void)dealloc;  
  23. - (id)initWithLayer:(id)arg1 node:(id)arg2 webDocumentView:(id)arg3;  
  24.   
  25. @end  
其还有一个子View作为ContentView,是给WebCore真正用作渲染overflow型内容的layer的容器。

UIWebOverflowContentView的声明为:

[cpp]  view plain  copy
  1. @interface UIWebOverflowContentView : UIView  
  2. {  
  3. }  
  4.   
  5. - (void)_setCachedSubviews:(id)arg1;  
  6. - (void)_replaceLayer:(id)arg1;  
  7. - (void)fixUpViewAfterInsertion;  
  8. - (id)superview;  
  9. - (id)initWithLayer:(id)arg1;  
  10.   
  11. @end  
再往底层跟,都是CALayer的操作。


以上两个类都是UIKit层的实现,需要WebCore有硬件加速的支持才有实际意义,相关的逻辑被包含在

ACCELERATED_COMPOSITING

这个宏里。

从SVN log看,在WebKit 108400版本左右才支持,所以iOS Safari应该是需要5.0。Android只在4.0以上支持。


从前端开发的角度讲,只需要知道CSS的属性-webkit-overflow-scrolling是真的创建了带有硬件加速的系统级控件,所以效率很高。但是这相对是耗更多内存的,最好在产生了非常大面积的overflow时才应用。


转载请注明出处:http://blog.csdn.net/hursing

你可能感兴趣的:(网页在Safari快速滚动和回弹的原理: -webkit-overflow-scrolling : touch;的实现)