JavaScript 实现回到顶部功能

今天我们一起来用 JavaScript 实现一个回到顶部的功能。

首先,需要梳理一下我们需求:

  1. 在页面的右下角有一个"回到顶部"的按钮
  2. 点击这个按钮不是直接回到顶部而是要以渐慢的速度回到顶部
  3. 在回到顶部的滚动中如果用户滑动鼠标的滚动轮则停止回到顶部的滚动效果
  4. 只有在滚动超过页面可视窗口高度之后才显示"回到顶部"的按钮,否则隐藏"回到顶部"按钮

首先先写出最基本的 HTML 结构:

<!DOCTYPE html>
  <html>
    <head>
      <meta charset="utf-8">
      <title>js 回到顶部</title>
      <style media="screen">
        body,html {
          margin: 0;
          padding: 0;
        }
        #box {
          width: 1190px;
          margin: 0 auto;
        }
        #box > div {
          background-color: #eee;
          height: 10000px;
        }
        #btn {
          height: 30px;
          width: 100px;
          background-color: #eee;
          position: fixed;
          right: 0px;
          bottom: 40px;
          -webkit-transition: background-color 1s;
          -o-transition: background-color 1s;
          transition: background-color 1s;
        }
        #btn:hover {
          background-color: #ff0;
          cursor: pointer;
        }
      </style>
    </head>
    <body>
      <div id="box">
        <div>用来撑开空间高度</div>
      </div>
      <div id="btn">回到顶部</div>
    </body>
    <script type="text/javascript">
      /* 这里添加 js 代码 */
    </script>
  </html>

本文主要实现功能,css 这块不太好看希望谅解 _

接下来要实现点击"回到顶部"的执行事件

第一步呢,先实现点击按钮直接向上滚动 200 像素的效果

window.onload = function() {
  var obtn = document.getElementById('btn');
  obtn.onclick = function() {
    // 获取当前视图最上方距这个内容区最顶部的距离(也就是已经滚动的距离)
    var osTop = document.documentElement.scrollTop || document.body.scrollTop;
    // 其实这里写法并不好
    document.body.scrollTop = document.documentElement.scrollTop -= 200;
  }
}

第二步,给上面这个向上滚动 200 像素的动作添加一个定时器自动执行,就实现了一个缓慢向上滚动到顶部的效果,注意当滚动到顶部的时候需要清除计时器。

第三步,改变这个 200 像素的值,速度的值应该是越靠顶部越慢的,所以还需要计算这个值。

下面是这部分的代码

var obtn = document.getElementById('btn');
// 定时器
var timer = null;
obtn.onclick = function() {
  // 速度值
  var speed = 0;
  // 当前视图最上方距这个内容区最顶部的距离(也就是已经滚动的距离)
  var osTop = 0;
  timer = setInterval(function() {
    osTop = document.documentElement.scrollTop || document.body.scrollTop;
    // 改变回到顶部的速度(越来越慢)
    // 注意此处是向上取整
    speed = Math.ceil(-osTop / 2);
    document.body.scrollTop = document.documentElement.scrollTop -= (osTop + speed);
    if (speed === 0) {
      clearInterval(timer);
    }
  }, 30);
}

完成到此其实基本的回到顶部的功能就已经完成了,下面就是一些优化操作。

第四步,如果在滚动回到顶部的过程中,用户滑动了鼠标的滚轮,那就停止回到顶部的滚动操作。此处我们需要调用 onmousewheel ,暂不考虑此方法的浏览器兼容性问题,代码如下:

// 监听鼠标滑轮的滚动事件
window.onmousewheel = function() {
  clearInterval(timer);
}

最后一步,就是只有在滚动超过页面可视高度后才显示"回到顶部"按钮,否则隐藏该按钮,以下是代码:

  #btn {
    /* other code... */
    display: none;
  }
// 获取页面的可视窗口高度
var clientHeight = document.documentElement.clientHeight || document.body.clientHeight;
var obtn = document.getElementById('btn');
// 页面滚动时增加判断,超出页面可视化高度时显示回到顶部的按钮
window.onscroll = function() {
  // 此处需要重新声明,因为和上面的 osTop 不在同一个函数域,不可忽略!
  var osTop = document.documentElement.scrollTop || document.body.scrollTop;
  if (osTop > clientHeight) {
    obtn.style.display = 'block';
  } else {
    obtn.style.display = 'none';
  }
}

写到此处一个回到顶部的功能就完成啦,是不是非常简单呢~

下面是完整的页面代码:

<!DOCTYPE html>
<html>
  <head>
    <meta charset="utf-8">
    <title>js回到顶部</title>
    <style media="screen">
      body, html {
        margin: 0;
        padding: 0;
      }
      #box {
        width: 1190px;
        margin: 0 auto;
      }
      #box > div {
        background-color: #eee;
        height: 10000px;
      }
      #btn {
        height: 30px;
        line-height: 30px;
        width: 100px;
        background-color: #eee;
        position: fixed;
        right: 0px;
        bottom: 40px;
        display: none;
        -webkit-transition: background-color 1s;
        -o-transition: background-color 1s;
        transition: background-color 1s;
      }
      #btn:hover {
        background-color: #ff0;
        cursor: pointer;
      }
    </style>
  </head>
  <body>
    <div id="box">
      <div>占位</div>
    </div>
    <div id="btn">回到顶部</div>
  </body>
  <script type="text/javascript">
    window.onload = function() {
      var obtn = document.getElementById('btn');
      // 定时器
      var timer = null;
      // 获取页面的可视窗口高度
      var clientHeight = document.documentElement.clientHeight || document.body.clientHeight;
      obtn.onclick = function() {
        var osTop = 0;
        var speed = 0;
        timer = setInterval(function() {
          isScroll = true;
          // 距内容区最顶部的距离
          osTop = document.documentElement.scrollTop || document.body.scrollTop;
          // 改变回到顶部的速度(越来越慢)
          speed = Math.ceil(-osTop / 1.05);
          document.body.scrollTop = document.documentElement.scrollTop -= (osTop + speed);
          if (speed == 0) {
            clearInterval(timer);
            isScroll = false;
          }
        }, 30);
      };
      // 监听鼠标滑轮的滚动事件
      window.onmousewheel = function() {
        clearInterval(timer);
      };
      // 页面滚动时增加判断,超出页面可视化高度时显示回到顶部的按钮
      window.onscroll = function() {
        var osTop = document.documentElement.scrollTop || document.body.scrollTop;
        if (osTop > clientHeight) {
          obtn.style.display = 'block';
        } else {
          obtn.style.display = 'none';
        }
      };
    };
  </script>
</html>

其实这是一个非常简单的功能,也是提醒大家在使用框架的同时,这些 JavaScript 底层的功能也不要忘记,因为技术在不断的更新但是底层的知识是不会变的。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,504评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,434评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,089评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,378评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,472评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,506评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,519评论 3 413
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,292评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,738评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,022评论 2 329
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,194评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,873评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,536评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,162评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,413评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,075评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,080评论 2 352

推荐阅读更多精彩内容