背景
最近开发公司的公众号 H5,做了一个点击 icon 滚动到顶部的功能。实现功能比较简单,直接调用 window.scrollTo(0, 0),一行代码完成。但是作为一个攻城狮,怎么可能对自己要求这么低,所以我给自己加了个需求,实现 平滑滚动页面到顶部 的功能。经过调研和查看文档,有了以下三种方案。
1. 使用 CSS
完成功能的最高境界,只用 CSS 就搞定。代码如下:
html {scroll-behavior: smooth;}
该样式的作用是为有滚动条的元素指定一个滚动的行为,但是只有在当用户手动导航或者 CSSOM scrolling API 触发滚动的时候生效,不影响用户行为产生的滚动。就在我庆祝的时候,打开 can i use 看了下兼容性:
日了狗,还是老老实实用 JS 实现吧。
2. 使用 Window.scrollTo API
我们都知道 window.scrollTo(x, y),通过传入文档中的 x,y 轴坐标来实现滚动到页面某个位置的功能。这个 API 其实还可以传入一个 option,是一个对象,left 值对应坐标中的 x,top 对应坐标中的 y,还有一个值为 behavior,可以让你自定义滚动行为,然后我们这样来实现滚动到顶部:
window.scrollTo({
left: 0,
top: 0,
behavior: 'smooth'
})
真香,搞定。过了几天,产品经理扛着 5 米大刀来找我,说在 Safari 上滚动效果奇怪,体检极差。于是我默默打开了 MDN 文档,滚动到底部:
看到了这张图,虽然 API 浏览器都几乎支持,但是 option 选项在 Safari 上直接挂掉,于是我又打开了 stackoverflow,总结了终极方案。
3. 使用 requestAnimationFrame
经常能看到大名鼎鼎的 requestAnimationFrame,但是没机会用上,这次可以尝尝鲜了。我们知道 requestAnimationFrame 的作用就是告诉浏览器在下次重绘之前执行传入的回调函数,这个行为是浏览器自动帮你做的。于是有了如下代码:
const scrollToTop = () => {
let scrollTop = document.documentElement.scrollTo || document.body.scrollTop
if (scrollTop > 0) {window.requestAnimationFrame(scrollTop)
window.scrollTop(0, scrollTop - scrollTo / 8)
}
}
Done!!!Perfect!!!雀跃的同时,我还是打开了 can i use 查看了 requestAnimationFrame 的兼容性:
晚上吃鸡!!!