CSS 新特性 contain控制页面的重绘与重排问题


Posted in HTML / CSS onApril 30, 2021

CSS 新特性 contain控制页面的重绘与重排问题

在介绍新的 CSS 属性 contain 之前,需要读者已经知道什么是页面的重绘与重排。

发现之前已经描述过很多次了,可以看看这个提高 CSS 动画性能的正确姿势[1]。

OK,下面进入本文正题,

contain 为何?

contain 属性允许我们指定特定的 DOM 元素和它的子元素,让它们能够独立于整个 DOM 树结构之外。目的是能够让浏览器有能力只对部分元素进行重绘、重排,而不必每次都针对整个页面。

The contain property allows an author to indicate that an element and its contents are, as much as possible, independent of the rest of the document tree. This allows the browser to recalculate layout, style, paint, size, or any combination of them for a pmited area of the DOM and not the entire page.contain 语法

看看它的语法:

{ 
  /* No layout containment. */ 
  contain: none; 
  /* Turn on containment for layout, style, paint, and size. */ 
  contain: strict; 
  /* Turn on containment for layout, style, and paint. */ 
  contain: content; 
  /* Turn on size containment for an element. */ 
  contain: size; 
  /* Turn on layout containment for an element. */ 
  contain: layout; 
  /* Turn on style containment for an element. */ 
  contain: style; 
  /* Turn on paint containment for an element. */ 
  contain: paint; 
}

 

除去 none,取值还有 6 个,我们一个一个来看看。

contain: size

contain: size: 设定了 contain: size 的元素的渲染不会受到其子元素内容的影响。

The value turns on size containment for the element. This ensures that the containing box can be laid out without needing to examine its descendants.

我开始看到这个定义也是一头雾水,光看定义很难明白到底是什么意思。还需实践一番:

假设我们有如下简单结构:

<div class="container"> 
    
</div>
.container { 
    width: 300px; 
    padding: 10px; 
    border: 1px solid red; 
} 
 
p { 
    border: 1px solid #333; 
    margin: 5px; 
    font-size: 14px; 
}

并且,借助 jQuery 实现每次点击容器添加一个 <p>Coco</p> 结构:

$('.container').on('click', e => { 
    $('.container').append('<p>Coco</p>') 
})

 

那么会得到如下结果:

CSS 新特性 contain控制页面的重绘与重排问题

可以看到,容器 .container 的高度是会随着元素的增加而增加的,这是正常的现象。

此刻,我们给容器 .container 添加一个 contain: size,也就会出现上述说的:设定了 contain: size 的元素的渲染不会受到其子元素内容的影响。

.container { 
    width: 300px; 
    padding: 10px; 
    border: 1px solid red; 
+   contain: size 
}

再看看会发生什么:

CSS 新特性 contain控制页面的重绘与重排问题

正常而言,父元素的高度会因为子元素的增多而被撑高,而现在,子元素的变化不再影响父元素的样式布局,这就是 contain: size 的作用。

contain: style

接下来再说说 contain: style、contain: layout 、contain: paint。先看看 contain: style。

截止至本文书写的过程中,contain: style 暂时被移除了。

CSS Containment Module Level 1[2]: Drop the at-risk “style containment” feature from this specification, move it Level 2。

嗯,官方说辞是因为存在某些风险,暂时被移除,可能在规范的第二版会重新定义吧,那这个属性也暂且放一放。

contain: paint

contain: paint:设定了 contain: paint 的元素即是设定了布局限制,也就是说告知 User Agent,此元素的子元素不会在此元素的边界之外被展示,因此,如果元素不在屏幕上或以其他方式设定为不可见,则还可以保证其后代不可见不被渲染。

This value turns on paint containment for the element. This ensures that the descendants of the containing box don’t display outside its bounds, so if an element is off-screen or otherwise not visible, its descendants are also guaranteed to be not visible.

这个稍微好理解一点,先来看第一个特性:

设定了 contain: paint 的元素的子元素不会在此元素的边界之外被展示 设定了 contain: paint 的元素的子元素不会在此元素的边界之外被展示

这个特点有点类似与 overflow: hidden,也就是明确告知用户代理,子元素的内容不会超出元素的边界,所以超出部分无需渲染。

简单示例,假设元素结构如下:

<div class="container"> 
    <p>Coco</p> 
</div>
.container { 
    contain: paint; 
    border: 1px solid red; 
} 
 
p{ 
    left: -100px; 
}

我们来看看,设定了 contain: paint 与没设定时会发生什么:

CSS 新特性 contain控制页面的重绘与重排问题

CodePen Demo -- contain: paint Demo[3]

设定了 contain: paint 的元素在屏幕之外时不会渲染绘制

通过使用 contain: paint, 如果元素处于屏幕外,那么用户代理就会忽略渲染这些元素,从而能更快的渲染其它内容。

contain: layout

contain: layout:设定了 contain: layout 的元素即是设定了布局限制,也就是说告知 User Agent,此元素内部的样式变化不会引起元素外部的样式变化,反之亦然。

This value turns on layout containment for the element. This ensures that the containing box is totally opaque for layout purposes; nothing outside can affect its internal layout, and vice versa.

启用 contain: layout 可以潜在地将每一帧需要渲染的元素数量减少到少数,而不是重新渲染整个文档,从而为浏览器节省了大量不必要的工作,并显着提高了性能。

使用 contain:layout,开发人员可以指定对该元素任何后代的任何更改都不会影响任何外部元素的布局,反之亦然。

因此,浏览器仅计算内部元素的位置(如果对其进行了修改),而其余DOM保持不变。因此,这意味着帧渲染管道中的布局过程将加快。

存在的问题

描述很美好,但是在实际 Demo 测试的过程中(截止至2021/04/27,Chrome 90.0.4430.85),仅仅单独使用 contain:layout 并没有验证得到上述那么美好的结果。

设定了 contain: layout 的指定元素,改元素的任何后代的任何更改还是会影响任何外部元素的布局,点击红框会增加一条<p>Coco<p>元素插入到 container 中:

简单的代码如下:

<div class="container"> 
    <p>Coco</p> 
    ... 
</div> 
<div class="g-test"></div>
html, 
body { 
    width: 100%; 
    height: 100%; 
    display: flex; 
    justify-content: center; 
    align-items: center; 
    flex-direction: column; 
    gap: 10px; 
} 
 
.container { 
    width: 150px; 
    padding: 10px; 
    contain: layout; 
    border: 1px solid red; 
} 
 
.g-test { 
    width: 150px; 
    height: 150px; 
    border: 1px solid green; 
}

CSS 新特性 contain控制页面的重绘与重排问题

CodePen Demo -- contain: layout Demo[4]

Can i Use -- CSS Contain

截止至 2021-04-27,Can i Use 上的 CSS Contain 兼容性,已经可以开始使用起来:

CSS 新特性 contain控制页面的重绘与重排问题

参考文献

CSS Containment Module Level 1[5]

CSS containment[6]

CSS Containment in Chrome 52[7]

参考资料

[1]提高 CSS 动画性能的正确姿势:

https://github.com/chokcoco/iCSS/issues/11

[2]CSS Containment Module Level 1:

https://www.w3.org/TR/css-contain-1/

[3]CodePen Demo -- contain: paint Demo:

https://codepen.io/Chokcoco/pen/KKwmgmN

[4]CodePen Demo -- contain: layout Demo:

https://codepen.io/Chokcoco/pen/rNjRELL

[5]CSS Containment Module Level 1:

https://www.w3.org/TR/css-contain-1/

[6]CSS containment:

https://justmarkup.com/articles/2016-04-05-css-containment/

[7]CSS Containment in Chrome 52:

https://developers.google.com/web/updates/2016/06/css-containment

[8]Github -- iCSS:

https://github.com/chokcoco/iCSS

到此这篇关于CSS 新特性 contain控制页面的重绘与重排的文章就介绍到这了,更多相关CSS 控制页面的重绘与重排内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章,希望大家以后多多支持三水点靠木!

HTML / CSS 相关文章推荐
CSS3教程(10):CSS3 HSL声明设置颜色
Apr 02 HTML / CSS
CSS3控制HTML元素动画效果
Feb 08 HTML / CSS
CSS3中线性颜色渐变的一些实现方法
Jul 14 HTML / CSS
纯CSS3单页切换导航菜单界面设计的简单实现
Aug 16 HTML / CSS
CSS3绘制六边形的简单实现
Aug 25 HTML / CSS
css3实现蒙版弹幕功能
Jun 18 HTML / CSS
HTML5 Canvas像素处理使用接口介绍
Dec 02 HTML / CSS
浅谈html5标签css3的常用样式
Oct 20 HTML / CSS
纯HTML5+CSS3制作生日蛋糕(代码易懂)
Nov 16 HTML / CSS
Canvas与图片压缩的示例代码
Nov 28 HTML / CSS
Canvas制作的下雨动画的示例
Mar 06 HTML / CSS
HTML5超文本标记语言的实现方法
Sep 24 HTML / CSS
CSS3新特性详解(五):多列columns column-count和flex布局
css position fixed 左右双定位的实现代码
Apr 29 #HTML / CSS
css3 利用transform-origin 实现圆点分布在大圆上布局及旋转特效
CSS3实现的侧滑菜单
CSS完成视差滚动效果
Html5通过数据流方式播放视频的实现
Apr 27 #HTML / CSS
CSS3实现的3D隧道效果
You might like
DC这些乐高系列动画电影你看过几部?
2020/04/09 欧美动漫
PHP实现微信网页授权开发教程
2016/01/19 PHP
php mysqli查询语句返回值类型实例分析
2016/06/29 PHP
JavaScript 字符编码规则
2009/05/04 Javascript
Javascript中获取出错代码所在文件及行数的代码
2010/09/23 Javascript
ExtJS中文乱码之GBK格式编码解决方案及代码
2013/01/20 Javascript
jquery实现带单选按钮的表格行选中时高亮显示
2013/08/01 Javascript
jquery each的几种常用的使用方法示例
2014/01/21 Javascript
JS获得浏览器版本和操作系统版本的例子
2014/05/13 Javascript
jquery左边浮动到一定位置时显示返回顶部按钮
2014/06/05 Javascript
jquery实现人性化的有选择性禁用鼠标右键
2014/06/30 Javascript
常用的JS验证和函数汇总
2014/12/23 Javascript
JavaScript实现Java中StringBuffer的方法
2015/02/09 Javascript
jquery实现Li滚动时滚动条自动添加样式的方法
2015/08/10 Javascript
限制复选框最多选择项的实现代码
2016/05/30 Javascript
xmlplus组件设计系列之分隔框(DividedBox)(8)
2017/05/02 Javascript
微信小程序实现城市列表选择
2018/06/05 Javascript
[39:07]LGD vs VP 2018国际邀请赛淘汰赛BO3 第二场 8.21
2018/08/22 DOTA
python通过线程实现定时器timer的方法
2015/03/16 Python
Python中常用操作字符串的函数与方法总结
2016/02/04 Python
Python+OpenCV实现车牌字符分割和识别
2018/03/31 Python
Python-OpenCV基本操作方法详解
2018/04/02 Python
如何使用VSCode愉快的写Python于调试配置步骤
2018/04/06 Python
基于Python爬虫采集天气网实时信息
2020/06/05 Python
python 中的命名空间,你真的了解吗?
2020/08/19 Python
浅析Python 责任链设计模式
2020/09/11 Python
在HTML5 canvas里用卷积核进行图像处理的方法
2018/05/02 HTML / CSS
瑞士隐形眼镜和护理产品网上商店:Linsenklick
2019/10/21 全球购物
PHP如何删除一个Cookie值
2012/11/15 面试题
体育教育毕业生自荐信
2013/11/21 职场文书
教育课题研究自我鉴定范文
2013/12/28 职场文书
总结表彰大会主持词
2014/03/26 职场文书
小学校园文化建设汇报材料
2014/08/19 职场文书
医药销售自我评价200字
2014/09/11 职场文书
自主招生自荐信怎么写
2015/03/24 职场文书
最新动漫情报:2022年7月新番定档超过30部, OVERLORD骨王第四季也在其中噢
2022/05/04 日漫