前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >网页在Safari快速滚动和回弹的原理: -webkit-overflow-scrolling : touch;的实现

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

作者头像
李维亮
发布2021-07-09 11:33:19
1.2K0
发布2021-07-09 11:33:19
举报
文章被收录于专栏:李维亮的博客

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

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

代码语言:javascript
复制
-webkit-overflow-scrolling : touch;  

可用以下网页测试:

代码语言:javascript
复制
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">  
<html>  
    <head>  
        <meta charset="utf-8" />  
        <title>scroll</title>  
        <style type="text/css">  
            .container {  
                width : 300px;  
                height : 50%;  
                -webkit-box-sizing : border-box;  
                position : relative;  
                overflow-y : auto;  
                background-color : cyan;  
                -webkit-overflow-scrolling : touch;  /* liuhx:可以把这整行注释掉对比差别 */  
            }  
            ul {  
                height: 50px;  
            }  
        </style>  
    </head>  
<body>  
    <div align="center">  
        <nav class="container">  
            <ul>1</ul>  
            <ul>2</ul>  
            <ul>3</ul>  
            <ul>4</ul>  
            <ul>5</ul>  
            <ul>6</ul>  
            <ul>7</ul>  
            <ul>8</ul>  
            <ul>9</ul>  
            <ul>10</ul>  
            <ul>11</ul>  
            <ul>12</ul>  
            <ul>13</ul>  
            <ul>14</ul>  
            <ul>15</ul>  
            <ul>16</ul>  
            <ul>17</ul>  
            <ul>18</ul>  
            <ul>19</ul>  
            <ul>20</ul>  
        </nav>  
    </div>  
</body>  
</html>  

可以用手指滑动中间的蓝色区域,会发现回弹效果以及滚动得很快:

webkit_overflow_scrolling
webkit_overflow_scrolling

如果把-webkit-overflow-scrolling那行注释掉,就会发现滚动得很慢。

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

代码语言:javascript
复制
Thread 1, Queue : com.apple.main-thread  
#0  0x00086723 in -[UIScrollView initWithFrame:] ()  
#1  0x004ec3bd in -[UIWebOverflowScrollView initWithLayer:node:webDocumentView:] ()  
#2  0x001f1769 in -[UIWebDocumentView webView:didCreateOrUpdateScrollingLayer:withContentsLayer:scrollSize:forNode:allowHorizontalScrollbar:allowVerticalScrollbar:] ()  
#3  0x01d571bd in __invoking___ ()  
#4  0x01d570d6 in -[NSInvocation invoke] ()  
#5  0x01d5724a in -[NSInvocation invokeWithTarget:] ()  
#6  0x027fb6a1 in -[_WebSafeForwarder forwardInvocation:] ()  
#7  0x027fb8ab in __44-[_WebSafeAsyncForwarder forwardInvocation:]_block_invoke_0 ()  
#8  0x04ac753f in _dispatch_call_block_and_release ()  
#9  0x04ad9014 in _dispatch_client_callout ()  
#10 0x04ac97d5 in _dispatch_main_queue_callback_4CF ()  
#11 0x01d09af5 in __CFRunLoopRun ()  
#12 0x01d08f44 in CFRunLoopRunSpecific ()  
#13 0x01d08e1b in CFRunLoopRunInMode ()  
#14 0x01cbd7e3 in GSEventRunModal ()  
#15 0x01cbd668 in GSEventRun ()  
#16 0x00032ffc in UIApplicationMain ()  
#17 0x00002ae2 in main at /Users/liuhx/Desktop/UIWebView_Research/WebViewResearch/main.mm:16  

实际创建的是UIWebOverflowScrollView,它继承自UIScrollView,声明为:

代码语言:javascript
复制
@class DOMNode, UIWebDocumentView, UIWebOverflowContentView, UIWebOverflowScrollListener;  
  
@interface UIWebOverflowScrollView : UIScrollView  
{  
    UIWebDocumentView *_webDocumentView;  
    UIWebOverflowScrollListener *_scrollListener;  
    UIWebOverflowContentView *_overflowContentView;  
    DOMNode *_node;  
    BOOL _beingRemoved;  
}  
  
@property(nonatomic, getter=isBeingRemoved) BOOL beingRemoved; // @synthesize beingRemoved=_beingRemoved;  
@property(retain, nonatomic) DOMNode *node; // @synthesize node=_node;  
@property(retain, nonatomic) UIWebOverflowContentView *overflowContentView; // @synthesize overflowContentView=_overflowContentView;  
@property(retain, nonatomic) UIWebOverflowScrollListener *scrollListener; // @synthesize scrollListener=_scrollListener;  
@property(nonatomic) UIWebDocumentView *webDocumentView; // @synthesize webDocumentView=_webDocumentView;  
- (void)setContentOffset:(struct CGPoint)arg1;  
- (void)_replaceLayer:(id)arg1;  
- (void)prepareForRemoval;  
- (void)fixUpViewAfterInsertion;  
- (id)superview;  
- (void)dealloc;  
- (id)initWithLayer:(id)arg1 node:(id)arg2 webDocumentView:(id)arg3;  
  
@end  

其还有一个子View作为ContentView,是给WebCore真正用作渲染overflow型内容的layer的容器。 UIWebOverflowContentView的声明为:

代码语言:javascript
复制
@interface UIWebOverflowContentView : UIView  
{  
}  
  
- (void)_setCachedSubviews:(id)arg1;  
- (void)_replaceLayer:(id)arg1;  
- (void)fixUpViewAfterInsertion;  
- (id)superview;  
- (id)initWithLayer:(id)arg1;  
  
@end  

再往底层跟,都是CALayer的操作。

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

ACCELERATED_COMPOSITING

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

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

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2016-02-29,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
相关产品与服务
容器服务
腾讯云容器服务(Tencent Kubernetes Engine, TKE)基于原生 kubernetes 提供以容器为核心的、高度可扩展的高性能容器管理服务,覆盖 Serverless、边缘计算、分布式云等多种业务部署场景,业内首创单个集群兼容多种计算节点的容器资源管理模式。同时产品作为云原生 Finops 领先布道者,主导开源项目Crane,全面助力客户实现资源优化、成本控制。
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档