CSS 如何设置自动滚动定位的间距?

介绍两个和滚动定位相关的 CSS 属性:scroll-padding[1]和 scroll-margin[2]

在平时开发中,经常会碰到需要快速定位的问题,比如常见的锚点定位

html 复制代码
<ul>
  <li><a href="#语法">语法</a></li>
  <li><a href="#示例">示例</a></li>
  ...
</ul>
<article>
 <h2 id="语法">语法</h2>
  <p>...</p>
  <h2 id="示例">示例</h2>
  <p>...</p>
  ...
</article>

这样,在点击a标签时会自动定位到与之相对应的内容上,如下

但是,这种通过锚点触发的定位默认是紧贴滚动容器边缘 的,如果一些定位元素,比如fixed定位的头部,就会出现被遮挡的情况,如下

可以看到,"示例"这个标题由于紧贴顶部,导致被sticky定位的头部遮住了。

那么,如何让自动定位时让目标元素预留出足够大的间距呢

一、一行 CSS解决

没错,看似有些麻烦的问题其实可以通过一行 CSS 解决,那就是 scroll-margin[3] ,下面是 MDN 的介绍

scroll-margin 属性的值代表用于将盒元素拖拽到显示区域的拖拽滚动区域的起点。拖拽滚动区域由是由转换后边框大小的盒元素决定的,它会找到盒元素的矩形边界(在滚动的容器的坐标空间轴上),并添加指定的起点。

这个描述很官方,有点不知道在说什么😂,导致很长一段时间都不知道这个属性的真正用途。

在上面这个例子中,可以直接给目标设置scroll-margin

css 复制代码
h2{
  scroll-margin: 6rem; 
}

设置这个属性后,当自动滚动定位到h2时,会自动预留6rm的间隔(可以防止被头部遮挡),下面是演示(红框表示6rem的间隔)

是不是非常简单,最终效果如下

其实,MDN官方已经采用了这种方式,经常看看 MDN,会发现有很多非常巧妙的实现,如下

二、还有一个 scroll-padding

scroll-margin比较类似的还有一个 scroll-padding[4],功能都是一样的,只是作用对象不一样。

在前面的例子中可以看到,scroll-margin是直接设置目前元素 上的,scroll-padding不一样,它需要设置在滚动容器上,比如

css 复制代码
html{
  scroll-padding: 6rem
}

这种方式也是可以达到相同的效果的

一般情况下,两种方式都可以自行选择,如果很清楚滚动容器是哪个,可以直接选择用scroll-padding,否则就用scroll-margin

三、其他滚动定位方式

除了锚点定位以外,还有其他一些方式可以触发滚动定位

1. scrollIntoView

有时候,我们需要将指定元素滚动到视线范围之内,这时就需要用到这样一个DOM方法:scrollIntoView[5]

ini 复制代码
element.scrollIntoView();

这个定位和前面的锚点定位一样,默认也是紧贴滚动容器的,如果设置了scroll-margin或者scroll-pading,也可以实现在滚动定位时自动预留一定间距

2. focus 定位

在默认情况下,元素(比如a链接)在focus聚焦时都会自动滚动到视线范围之内。和上面几种情况一样,如果有fixed定位元素,有可能在focus时被遮挡的问题。

如果设置了scroll-margin或者scroll-pading,这样就可以避免找不到焦点 的情况,确保一直都能看到焦点,下面是通过tab键聚焦的情况

3. scroll-snap

还有一种情况是滚动捕捉:scroll-snap-type[6],这个属性可以让滚动时自动捕捉临界点。正常情况下,滚动临界点是紧贴滚动容器的,像这样

如果希望预留一定的距离如何处理呢?还是这个scroll-margin,下面给第二个元素设置了一定的scroll-margin,效果如下

可以看到,在滚动到第2个元素时,提前预留了一定的距离,而且还可以设置负值,这样在滚动到第2个元素时,可以提前看到第3个的部分内容

目前我所了解的就是这几种定位方式,有其他的可以留言告知

四、兼容性和总结

最后来看一下兼容性,一个体验增强属性,兼容性还不错(safari有些拉胯😭)

这里有个疑惑是,都出来这么久了,为啥一直不知道呢?其实我也一直被 MDN 官网误导了,首先,scroll-marginscroll-padding的官方示例中只有关于scroll-snap-type的应用场景,导致我一直误以为这个属性就是和scroll-snap-type搭配使用的,没有朝其他方向上思考。另外,从时间上来看,这个属性确实是和scroll-snap-type几乎是一同推出的(chrome上),这点从兼容性上可以看出,这样有点更坚信了前面的误导

所以,一直以来,这样一个超级好用的 CSS 属性被我忽略了,可惜😭

不过,现在了解也不晚,下面来总结一下

  1. 默认情况下自动滚动定位都是与滚动容器贴边的,有时候并不美好
  2. scroll-paddingscroll-margin可以在自动滚动定位时预留指定的间距
  3. scroll-margin作用对象是目标元素,scroll-padding作用对象是滚动容器
  4. 滚动定位方式有锚点定位、scrollIntoView定位、focus定位、还有Scroll-snap定位
  5. 体验增强属性,兼容性还不错,主要是safari拖后腿

参考资料

1\] scroll-padding: *[developer.mozilla.org/en-US/docs/...](https://link.juejin.cn?target=https%3A%2F%2Fdeveloper.mozilla.org%2Fen-US%2Fdocs%2FWeb%2FCSS%2Fscroll-padding "https://developer.mozilla.org/en-US/docs/Web/CSS/scroll-padding")* \[2\] scroll-margin: *[developer.mozilla.org/zh-CN/docs/...](https://link.juejin.cn?target=https%3A%2F%2Fdeveloper.mozilla.org%2Fzh-CN%2Fdocs%2FWeb%2FCSS%2Fscroll-margin "https://developer.mozilla.org/zh-CN/docs/Web/CSS/scroll-margin")* \[3\] scroll-margin: *[developer.mozilla.org/zh-CN/docs/...](https://link.juejin.cn?target=https%3A%2F%2Fdeveloper.mozilla.org%2Fzh-CN%2Fdocs%2FWeb%2FCSS%2Fscroll-margin "https://developer.mozilla.org/zh-CN/docs/Web/CSS/scroll-margin")* \[4\] scroll-padding: *[developer.mozilla.org/en-US/docs/...](https://link.juejin.cn?target=https%3A%2F%2Fdeveloper.mozilla.org%2Fen-US%2Fdocs%2FWeb%2FCSS%2Fscroll-padding "https://developer.mozilla.org/en-US/docs/Web/CSS/scroll-padding")* \[5\] scrollIntoView: *[developer.mozilla.org/zh-CN/docs/...](https://link.juejin.cn?target=https%3A%2F%2Fdeveloper.mozilla.org%2Fzh-CN%2Fdocs%2FWeb%2FAPI%2FElement%2FscrollIntoView "https://developer.mozilla.org/zh-CN/docs/Web/API/Element/scrollIntoView")* \[6\] scroll-snap-type: *[developer.mozilla.org/zh-CN/docs/...](https://link.juejin.cn?target=https%3A%2F%2Fdeveloper.mozilla.org%2Fzh-CN%2Fdocs%2FWeb%2FCSS%2Fscroll-snap-type "https://developer.mozilla.org/zh-CN/docs/Web/CSS/scroll-snap-type")* 转载:[前端侦探:CSS 如何设置自动滚动定位的间距?](https://link.juejin.cn?target=https%3A%2F%2Fmp.weixin.qq.com%2Fs%3F__biz%3DMzIyMDc1NTYxNg%3D%3D%26mid%3D2247486316%26idx%3D1%26sn%3D2a54ec347f72cd91d61a68d6e232f370%26chksm%3D97c668a3a0b1e1b56f501e2d7267606af48cc406769359e3867ac53149ecb52bf9c86f95cc43%26scene%3D21%23wechat_redirect "https://mp.weixin.qq.com/s?__biz=MzIyMDc1NTYxNg==&mid=2247486316&idx=1&sn=2a54ec347f72cd91d61a68d6e232f370&chksm=97c668a3a0b1e1b56f501e2d7267606af48cc406769359e3867ac53149ecb52bf9c86f95cc43&scene=21#wechat_redirect")

相关推荐
萌萌哒草头将军4 小时前
⚡⚡⚡尤雨溪宣布开发 Vite Devtools,这两个很哇塞 🚀 Vite 的插件,你一定要知道!
前端·vue.js·vite
小彭努力中4 小时前
7.Three.js 中 CubeCamera详解与实战示例
开发语言·前端·javascript·vue.js·ecmascript
浪裡遊5 小时前
跨域问题(Cross-Origin Problem)
linux·前端·vue.js·后端·https·sprint
LinDaiuuj5 小时前
判断符号??,?. ,! ,!! ,|| ,&&,?: 意思以及举例
开发语言·前端·javascript
敲厉害的燕宝5 小时前
Pinia——Vue的Store状态管理库
前端·javascript·vue.js
Aphasia3116 小时前
react必备JavaScript知识点(二)——类
前端·javascript
玖玖passion6 小时前
数组转树:数据结构中的经典问题
前端
呼Lu噜6 小时前
WPF-遵循MVVM框架创建图表的显示【保姆级】
前端·后端·wpf
珠峰下的沙砾6 小时前
Vue3 里 CSS 深度作用选择器 :global
前端·javascript·css
航Hang*6 小时前
WEBSTORM前端 —— 第2章:CSS —— 第3节:背景属性与显示模式
前端·css·css3·html5·webstorm