使用智能 CSS 基于用户滚动位置应用样式

所属分类: 网页制作 / CSS 阅读数: 498
收藏 0 赞 0 分享

通过将当前滚动偏移量添加到到 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,并可以使用它来为页面上的元素添加样式。

Live Demo on CodePen

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

更多精彩内容其他人还在看

CSS入门教程:计算CSS盒模型宽和高

 出处:当我们布局一个网页的时候,经常会遇到这样的一种情况,那就是最终网页成型的宽度或是高度会超出我们预先的计算,其实就就是所谓的CSS的盒模型造成的。 #test{margin:10px;padding:10px;width:100px;height:100px;}
收藏 0 赞 0 分享

在IE流览器中正确显示PNG透明图片

  png图片有很好的品质。阴影效果也不会有杂边,很流畅。如果插入网页的话可以给网站内容增色不少!更重要的是在不增加图片容量大小的情况下提高了页面的图片的质量。对于有复杂背景,如:在有颜色过度背景上插入不规则边框的图片带来极大很便利!   但目前IE中对于插入
收藏 0 赞 0 分享

CSS教程:DIV底部放置文字

  css对文字的布局上没有靠容器底部对齐的参数,目前使用的一个不错的方法也比较好.就是用position属性来解决,看下面的代码,用position的相对和绝对定位功能也轻松的实现了,文字靠近div低部对齐,并且靠近的距离还可以精确到像素,自己可以调节,是不是很不错呢?
收藏 0 赞 0 分享

如何用CSS让文字居于div的底部

  这个问题是别人提出的,因为css对文字的布局上没有靠容器底部对齐的参数,(或许有但是我没有发现)不过目前我使用的一个不错的方法也比较好.就是用position属性来解决,看下面的代码,我用position的相对和绝对定位功能也轻松的实现了,文字靠近div低部对齐,并且靠近
收藏 0 赞 0 分享

从A页面连接到B页面后并直接把B页面的隐藏层显示

  这个效果实现的是,在B页面里有两个层,一个显示层,我们暂定名c层,一个是隐藏层,我们暂定名d层,单独进B页面的时候,c层显示,d层隐藏,然而从A页面连接到B页面的时候,则是让d层显示,c层隐藏,我觉得这个效果对网页设计者以后会有很大帮助,现在把代码发出来,
收藏 0 赞 0 分享

CSS样式表定义标签li前面样式

定义LI前面的小点样式 view plaincopy to clipboardprint? 语法: list-style-type : disc | circle | square | decimal | lower-roman | upper-roman | lowe
收藏 0 赞 0 分享

符合标准的div css制作的弹出菜单

本文介绍了五款符合标准的div css制作的弹出菜单,而且不含有js的. NO.1最基本的:二级dropdown弹出菜单 <!DOCTYPE html PUB
收藏 0 赞 0 分享

CSS实现在文章每段后面加入带连接的隐藏文字

代码主要理解3个参数:createElement、createTextNode、appendChild。这3个js参数分别是创建元素、创建字符、追加节点。代码原理:循环页面段落标签<p>,创建连接元素<a>,创建要显示的连接字符,用SetAttribute
收藏 0 赞 0 分享

CSS:浏览器特定选择器介绍

当你想在一个浏览器里改变样式而不像在其他浏览器中改变时,这些选择器很有用。 IE6以下 *html{} IE 7 以下 *:first-child html {} * html {} 只对IE 7 *:first-child html {} 只对IE 7
收藏 0 赞 0 分享

WEB标准学习,认识两种网页声明的含义

即网页标准推出来以后,我们时常会看到两种不同的网页的声明,一个是Dhtml,一个是Xhtml。如下所示: <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "ht
收藏 0 赞 0 分享
查看更多