使用智能 CSS 基于用户滚动位置应用样式
通过将当前滚动偏移量添加到到 html 元素的属性上,我们可以根据当前滚动位置设置页面上的元素样式。我们可以使用它来构建一个浮动在页面顶部的导航组件。
这是我们将使用的 HTML,<header> 组件是我们希望当我们向下滚动时,始终浮动在页面顶部的一个组件。
<header>I'm the page header</header> <p>Lot's of content here...</p> <p>More beautiful content...</p> <p>Content...</p>
首先,我们将监听 document 上的 'scroll' 事件,并且每次用户滚动时我们都会取出当前的 scrollY 值。
document.addEventListener('scroll', () => {
document.documentElement.dataset.scroll = window.scrollY;
});
我们将滚动位置存储在 html 元素的数据属性中。如果您使用开发工具查看 DOM,它将如下所示:<html data-scroll="0">
现在我们可以使用此属性来设置页面上的元素样式。
/* 保证 header标签始终高于 3em */
header {
min-height: 3em;
width: 100%;
background-color: #fff;
}
/* 在页面顶部保留与 header 的 min-height 相同的高度 */
html:not([data-scroll='0']) body {
padding-top: 3em;
}
/* 将 header 标签切换成 fixed 定位模式,并且将它固定在页面顶部 */
html:not([data-scroll='0']) header {
position: fixed;
top: 0;
z-index: 1;
/* box-shadow 属性能够增强浮动的效果 */
box-shadow: 0 0 .5em rgba(0, 0, 0, .5);
}
基本上就是这样,当用户向下滚动时,header 标签将自动从页面中分离并浮动在内容之上。JavaScript 代码并不关心这一点,它的任务就是将滚动偏移量放在数据属性中。这很完美,因为 JavaScript 和 CSS 之间没有紧密耦合。
但仍有一些可以改进的地方,主要是在性能方面。
首先,我们必须修改 JavaScript 脚本,以适应页面加载时滚动位置不在顶部的情况。在这样的情况下,header 标签将呈现错误的样式。
页面加载时,我们必须快速获取当前的滚动偏移量,这样确保了我们始终与当前的页面的状态同步。
// 读出当前页面的滚动位置并将其存入 document 的 data 属性中
// 因此我们就可以在我们的样式表中使用它
const storeScroll = () => {
document.documentElement.dataset.scroll = window.scrollY;
}
// 监听滚动事件
document.addEventListener('scroll', storeScroll);
// 第一次打开页面时就更新滚动位置
storeScroll();
接下来我们将看一些性能方面改进。如果我们想要获取 scrollY 滚动位置,浏览器将必须计算页面上每个元素的位置,以确保它返回正确的位置。如果我们不强制它每次滚动都取值才是最好的做法。
要做到这一点,我们需要一个 debounce(防抖动)方法,这个方法会将我们的取值请求加入一个队列中,在浏览器准备好绘制下一帧之前都不会重新取值,此时它已经计算出了页面上所有元素的位置,所以它不会不断重复相同的工作。
// 防抖动函数接受一个我们自定义的函数作为参数
const debounce = (fn) => {
// 这包含了对 requestAnimationFrame 的引用,所以我们可以在我们希望的任何时候停止它
let frame;
// 防抖动函数将返回一个可以接受多个参数的新函数
return (...params) => {
// 如果 frame 的值存在,那就清除对应的回调
if (frame) {
cancelAnimationFrame(frame);
}
// 使我们的回调在浏览器下一帧刷新时执行
frame = requestAnimationFrame(() => {
// 执行我们的自定义函数并传递我们的参数
fn(...params);
});
}
};
// Reads out the scroll position and stores it in the data attribute
// so we can use it in our stylesheets
const storeScroll = () => {
document.documentElement.dataset.scroll = window.scrollY;
}
// Listen for new scroll events, here we debounce our `storeScroll` function
document.addEventListener('scroll', debounce(storeScroll));
// Update scroll position for first time
storeScroll();
通过标记事件为 passive 状态,我们可以告诉浏览器我们的滚动事件不会被触摸交互阻止(例如与谷歌地图等插件交互时)。这允许浏览器立即滚动页面,因为它现在知道该事件不会被阻止。
document.addEventListener('scroll', debounce(storeScroll), { passive: true });
解决了性能问题后,我们现在可以通过稳定的方式使用 JavaScript 将获取的数据提供给 CSS,并可以使用它来为页面上的元素添加样式。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。
相关文章
- CSS Grid 是一种二维布局系统,可以同时控制行和列,相比 Flex(一维布局),更适合用在整体页面布局或复杂模块结构中,这篇文章主要介绍了前端CSS Grid 布局详解,需要的朋2025-04-16
- CSS 中的 padding 和 margin 是两个非常基础且重要的属性,它们用于控制元素周围的空白区域,本文将详细介绍 padding 和 margin 的概念、区别以及如何在实际项目中使用它们2025-04-07
- will-change 是一个 CSS 属性,用于告诉浏览器某个元素在未来可能会发生哪些变化,本文给大家介绍CSS will-change 属性详解,感兴趣的朋友一起看看吧2025-04-07
- 本文给大家分享在 CSS 中,去除a标签(超链接)的下划线的几种方法,本文给大家介绍的非常详细,感兴趣的朋友一起看看吧2025-04-07
在前端开发中,CSS(层叠样式表)不仅是用来控制网页的外观和布局,更是实现复杂交互和动态效果的关键技术之一,随着前端技术的不断发展,CSS的用法也日益丰富和高级,本文将2025-04-07css中的 vertical-align与line-height作用详解
文章详细介绍了CSS中的`vertical-align`和`line-height`属性,包括它们的作用、适用元素、属性值、常见使用场景、常见问题及解决方案,感兴趣的朋友跟随小编一起看看吧2025-03-26浅析CSS 中z - index属性的作用及在什么情况下会失效
z-index属性用于控制元素的堆叠顺序,值越大,元素越显示在上层,它需要元素具有定位属性(如relative、absolute、fixed或sticky),本文给大家介绍CSS 中z - index属性的作用2025-03-21- 文章详细介绍了CSS中的打印媒体查询@mediaprint包括基本语法、常见使用场景和代码示例,如隐藏非必要元素、调整字体和颜色、处理链接的URL显示、分页控制、调整边距和背景等2025-03-18

CSS模拟 html 的 title 属性(鼠标悬浮显示提示文字效果)
本文介绍了如何使用CSS模拟HTML的title属性,通过鼠标悬浮显示提示文字效果,通过设置`.tipBox`和`.tipBox.tipContent`的样式,实现了提示内容的隐藏和显示,感兴趣的朋友一起2025-03-10
前端 CSS 动态设置样式::class、:style 等技巧(推荐)
本文介绍了Vue.js中动态绑定类名和内联样式的两种方法:对象语法和数组语法,通过对象语法,可以根据条件动态切换类名或样式;通过数组语法,可以同时绑定多个类名或样式,此外2025-02-26





最新评论