滚动锚定(Scroll Anchoring)- 让视口内容不再因视口上方 DOM 元素的高度变化而产生跳动
不知道你有没有经历过这样的场景:当你打开一张“多图杀猫”的页面后,正一张图一张图边滚边看,在你刚准备定睛看某一张图的时候,这张图突然被它上面的内容挤到了视口下方,然后你赶紧把滚动条往下拉,试图追赶这张没看完的图,当你刚刚追上的时候,这张图又一次被挤到了你看不见的地方。
发生这种情况的原因是因为在很多场景下(比如论坛里),你没法事先知道一张图的高度,所以你没法事先给这张图占位,在网速不理想的情况下,可能就会发生我上面描述的这种因页面靠上的图片比靠下的图片晚加载出来而导致用户当前浏览的内容被频繁挤出视口的情况。
我通过在定时器回调里向页面上方插入图片来模拟一下刚才描述的这种情况:
<style>
img {
display: block;
margin: 0 auto;
}
</style>
<img src="https://aecpm.alicdn.com/tfscom/TB1.52aPFXXXXa0XXXXXXXXXXXX.jpg">
<img src="https://aecpm.alicdn.com/tfscom/TB1_utRPVXXXXapXVXXXXXXXXXX.png">
<img src="https://static.dingtalk.com/media/lAHOuOFd_czSzQEn_295_210.gif">
<img src="https://aecpm.alicdn.com/tfscom/TB1f1xwQpXXXXXBXVXXXXXXXXXX.jpg">
<img src="https://gtms03.alicdn.com/tps/i3/TB1eSxvJVXXXXaKXFXXYoAvIXXX-220-50.png">
<img src="https://gw.alicdn.com/bao/uploaded/TB1EGvvPVXXXXX3aXXXXXXXXXXX-200-200.jpg">
<img src="https://gw.alicdn.com/tfscom/TB1CLTHNFXXXXaDXpXXXXXXXXXX">
<script>
const urls = `
https://asearch.alicdn.com/bao/uploaded/i1/1381306006414474986/TB2_gZAlNtmpuFjSZFqXXbHFpXa_!!0-saturn_solar.jpg
https://asearch.alicdn.com/bao/uploaded/i1/153360285303496277/TB2SO.Wa4vzQeBjSZFEXXbYEpXa_!!0-saturn_solar.jpg
https://asearch.alicdn.com/bao/uploaded/i1/188050339412916381/TB2geTXaypnpuFjSZFkXXc4ZpXa_!!0-saturn_solar.jpg
https://asearch.alicdn.com/bao/uploaded/i2/181720289489216985/TB2UFz6amjz11Bjy0FnXXcnxXXa_!!0-saturn_solar.jpg
https://asearch.alicdn.com/bao/uploaded/i3/108480250457898935/TB28r5osFXXXXbrXXXXXXXXXXXX_!!0-saturn_solar.jpg
https://asearch.alicdn.com/bao/uploaded/i3/111180208599309441/TB2kAsQnVXXXXXcXFXXXXXXXXXX_!!0-saturn_solar.jpg
https://asearch.alicdn.com/bao/uploaded/i3/171530328819399773/TB2rgtke9iK.eBjSZFsXXbxZpXa_!!0-saturn_solar.jpg
https://asearch.alicdn.com/bao/uploaded/i3/1880505035634435666/TB2bToNiHXlpuFjSszfXXcSGXXa_!!0-saturn_solar.jpg
https://asearch.alicdn.com/bao/uploaded/i4/1519305020726924733/TB2I2VuhNhmpuFjSZFyXXcLdFXa_!!0-saturn_solar.jpg
`.split("\n")
let i = 0
setInterval(() => {
if (i === urls.length) i = 0
let img = new Image()
img.src = urls[i++]
document.body.prepend(img)
}, 2000)
onscroll = function(argument) {
console.log("scrollY:" + scrollY)
}
</script>
上面这个 demo 里,假设我一直“追赶”的那张图是“金凯瑞摇头三人组”那张 GIF,那么在 Chrome 56 之前的版本以及在其它的浏览器中,你看到的会是下面这样的场景:
为了获得更好的用户体验,Chrome 从 56 开始,开启了一个叫做“滚动锚定(Scroll Anchoring)”的优化,效果就是,当页面在视口上方的部分突然变高了 x 像素,那么浏览器会为你自动向下滚动 x 像素,从而保证视口内容完全不变:
浏览器自动为你向下滚动 x 像素,就意味着浏览器自己会触发一次 scroll 事件,也意味着 scrollY 的值会增加 x,你可以通过上面的 demo 验证这一点。
有些同学可能会有疑问,“这种场景多吗?”、“我怎么从来没注意到?”、“有必要把事情搞复杂吗?”。 从 Chrome 官方的统计可以看到,这个特性被触发(替你滚动页面)的概率大概为 1%,并不多,但也算不上是极端情况,所以优化还是有必要的。可能因为近些年网络条件越来越好,图片加载的速度比你滚动页面的速度还要快,所以不太容易遇到因网速慢导致的这类场景了(尤其在 WIFI 网络下)。
不过这个优化的确不是个简单的改动,Chrome 从去年 3 月份开始实现这个特性,直到一年多后的今天,仍然有一些因这个优化导致的 bug 存在,这些 bug 多表现为页面异常滚动,甚至像永动机一样无限抖动,从这方面看,事情的确有一些被搞的复杂了。但幸好有一个 CSS 属性可以关掉这个优化:overflow-anchor: none,你可以把这个属性添加到发生 bug 的容器元素上,甚至加到 body 元素上也行,然后该元素及其它的所有后代节点就都不会被应用“滚动锚定”的优化了。除了作为浏览器 bug 的临时 fix,我想不到其它使用这个属性的场景了。
这个优化不仅限于看图片的时候,任何元素节点,甚至文本节点也同样适用。比如你在某新闻网站浏览一段文字的时候,视口上方突然异步插入了一个未事先占位的 iframe 广告(微博输入框下方就有这么一个广告),如果你使用了 Chrome 56 及以上版本的话,你完全察觉不到这一变化,你的阅读不会被打断。
页面在视口上方的高度增加 x 像素,浏览器会为你向下滚动 x 像素;反过来,页面在视口上方的高度减少 x 像素,浏览器也会为你向上滚动 x 像素,但这种情况更少见了。
该优化同样适用于元素级别的滚动条,我也写了一个 demo:
<style>
div {
width: 300px;
height: 300px;
} #container {
background: red;
overflow: scroll;
} #aboveViewport {
background: blue;
} #anchorNode {
background: green;
}
</style>
<div id="container">
向下滚动到底
<div id="aboveAnchorNode"></div>
<div id="anchorNode"></div>
这段文字一旦出现就会始终在视口内
</div>
<script>
let height = 100
setInterval(() => {
aboveAnchorNode.style.height = height += 10
}, 1000)
</script>
由于本文讲的是一个浏览器的优化,即便是前端开发者也没有深究的必要,所以我故意省略了一些内容,比如什么是锚定节点(anchor node )以及浏览器如何选定一个锚点节点?以及哪些样式改动会把锚定节点挤出视口但不会触发优化(Suppression Triggers),如果你想深究,可以从规范里找到答案。
滚动锚定(Scroll Anchoring)- 让视口内容不再因视口上方 DOM 元素的高度变化而产生跳动的更多相关文章
- dom元素的自上而下自动滚动
dom元素的自上而下自动滚动 <!doctype html> <html lang="en"> <head> <meta charset= ...
- 解决:mui 的 选项卡 + 下拉刷新 功能,在其中嵌入 iframe 后,在 iphone 的情况下,iframe 的内容不能滚动,只显示第一屏内容。
我所遇到的情况是,使用 mui 的 选项卡 + 下拉刷新 功能时,其中有2个页面是嵌入了别的网站的页面,而别个几个是通过 ajax 加载本网站的数据.然后 在其中嵌入 iframe 后,在 iphon ...
- [转]用 jQuery 实现页面滚动(Scroll)效果的完美方法
转自: http://zww.me/archives/25144 很多博主都写过/转载过用 jQuery 实现页面滚动(Scroll)效果的方法,但目前搜来的方法大都在 Opera 下有个小 Bug: ...
- 滚动锁定 scroll lock 键有什么用?
滚动锁定 scroll lock 键有什么用? 中文名称:滚动锁定键 按下此键后在Excel等按上.下键滚动时,会锁定光标而滚动页面:如果放开此键,则按上.下键时会滚动光标而不滚动页面. ...
- 取消设置透明状态栏,使 ContentView 内容不再覆盖状态栏
取消设置透明状态栏,使 ContentView 内容不再覆盖状态栏,在MainActivity中添加以下代码: getWindow().clearFlags(WindowManager.LayoutP ...
- JS复制DOM元素文字内容
要实现的效果:将HTML页面中的某个DOM元素例如DIV下面的文本内容进行复制. 实现过程如下: <html> <head> <title>Copy text De ...
- 第一百六十六节,jQuery,基础 DOM 和 CSS 操作,元素内容,元素属性,css和class,元素宽度高度、偏移、滚动条
jQuery,基础 DOM 和 CSS 操作,元素内容,元素属性,css和class,元素宽度高度.偏移.滚动条 学习要点: 1.DOM 简介 2.设置元素及内容 3.元素属性操作 4.元素样式操作 ...
- Android中竖线随内容高度变化而变化的问题和解决办法
项目中要求显示竖线,并且竖线高度不确定,竖线的高度要随着内容的变化而变化.不能使用match_parent 充满,也不能在布局中写死,此时使用 android:layout_height=" ...
- 如何根据搜索页面内容得到的结果生成该元素的xpath路径
如何根据搜索页面内容得到的结果生成该元素的xpath路径?
随机推荐
- 【Python 19】BMR计算器3.0(字符串分割与格式化输出)
1.案例描述 基础代谢率(BMR):我们安静状态下(通常为静卧状态)消耗的最低热量,人的其他活动都建立在这个基础上. 计算公式: BMR(男) = (13.7*体重kg)+(5.0*身高cm)-(6. ...
- 在虚拟机中,设置centos7静态ip
https://blog.csdn.net/qq_34182808/article/details/80065908
- Spring service本类中方法调用另一个方法事务不生效问题(转载)
前些日子一朋友在需要在目标对象中进行自我调用,且需要实施相应的事务定义,且网上的一种通过BeanPostProcessor的解决方案是存在问题的.因此专门写此篇帖子分析why. 1.预备知识 aop概 ...
- 在Linux命令行中以图形化窗口打开文件夹
Linux 系统中也有类似的命令.Ubuntu 发行版的命令行中,我们可以使用 nautilus 命令来打开指定目录的图形化窗口界面.类似下面命令这样使用: nautilus /home/testPr ...
- Html 改变原有标签属性
内容简要: 当标签内内容 达到某以条件的时候改变当前标签属性 例如原标签为<tr> 当tr内的值符合某一条件时把<tr>变成<a>标签 例:当订单状体编程已支付的时 ...
- Ubuntu 18.04安装MySQL指南
前言 Ubuntu18.04想要安装MySQL,只能安装MySQL8.0版本.如果你直接 apt-get install mysql-server 安装,那么恭喜踩坑! 先给出彻底删除mysql5.x ...
- Web前端知识点记录
一.HTML的加载顺序 浏览器边下载HTML,边解析HTML代码,二者是从上往下同步进行的 先解析<head>中的代码,在<head>中遇到了<script>标签, ...
- vue slot插槽的使用
slot插槽的使用场景 父组件向子组件传递dom时会用到插槽 作用域插槽:当同一个子组件想要在不同的父组件里展示不同的状态,可以使用作用域插槽.展示的状态由父组件来决定 注:想要修改父组件向子 ...
- 从HTTL模板引擎看软件设计原则
HTTL (Hyper-Text Template Language) 是一个高性能的开源JAVA模板引擎, 适用于动态HTML页面输出, 可替代JSP页面, 指令和Velocity相似.作者是阿里巴 ...
- enex 转 md 格式的几种方式(免费版/氪金版)
因为最近有读者投稿,用的是印象笔记,文件格式为 .enex ,一般发文章都用 markdown 格式,这叫我好生苦恼,于是乎,Google 搜了一下,找到了如下解决办法. 氪金版: 我只找到了一款比较 ...