无需使用jQuery即可平滑滚动


81

我正在编写一个页面,其中我只想将原始JavaScript代码用于UI,而不会受到插件或框架的干扰。

现在,我正在努力寻找一种无需使用jQuery即可平滑滚动页面的方法。


3
jQuery将采用相同的方式。将动画分为一系列非常小的步骤,使用间隔计时器以较小的间隔依次执行这些步骤,直到完成为止。
tvanfosson 2012年

我在一开始就想到了这种解决方案,我唯一想念的地方实际上是在下面的Kamal链接中,这是如何计算对象的Y位置。感谢tvanfosson :)


这类似于我的问题之一。查看stackoverflow.com/questions/40598955/…
Cannicide

Answers:


27

尝试使用此平滑滚动演示或类似的算法:

  1. 使用获取当前的最佳位置 self.pageYOffset
  2. 获取元素的位置,直到要滚动到的位置: element.offsetTop
  3. 进行for循环到达该位置,这将非常快,或者使用计时器进行平滑滚动,直到使用 window.scrollTo

另请参见该问题的其他流行答案


安德鲁·约翰逊的原始代码:

function currentYPosition() {
    // Firefox, Chrome, Opera, Safari
    if (self.pageYOffset) return self.pageYOffset;
    // Internet Explorer 6 - standards mode
    if (document.documentElement && document.documentElement.scrollTop)
        return document.documentElement.scrollTop;
    // Internet Explorer 6, 7 and 8
    if (document.body.scrollTop) return document.body.scrollTop;
    return 0;
}


function elmYPosition(eID) {
    var elm = document.getElementById(eID);
    var y = elm.offsetTop;
    var node = elm;
    while (node.offsetParent && node.offsetParent != document.body) {
        node = node.offsetParent;
        y += node.offsetTop;
    } return y;
}


function smoothScroll(eID) {
    var startY = currentYPosition();
    var stopY = elmYPosition(eID);
    var distance = stopY > startY ? stopY - startY : startY - stopY;
    if (distance < 100) {
        scrollTo(0, stopY); return;
    }
    var speed = Math.round(distance / 100);
    if (speed >= 20) speed = 20;
    var step = Math.round(distance / 25);
    var leapY = stopY > startY ? startY + step : startY - step;
    var timer = 0;
    if (stopY > startY) {
        for ( var i=startY; i<stopY; i+=step ) {
            setTimeout("window.scrollTo(0, "+leapY+")", timer * speed);
            leapY += step; if (leapY > stopY) leapY = stopY; timer++;
        } return;
    }
    for ( var i=startY; i>stopY; i-=step ) {
        setTimeout("window.scrollTo(0, "+leapY+")", timer * speed);
        leapY -= step; if (leapY < stopY) leapY = stopY; timer++;
    }
}

相关链接:


2
大纲不错,但是请不要创建超时字符串以继续执行,请继续创建一个匿名函数来调用实际函数。
tvanfosson 2012年

@nness这是一个耻辱。我真的很想看代码。
dzny 2014年

37

JavaScript中的本机浏览器平滑滚动是这样的:

// scroll to specific values,
// same as window.scroll() method.
// for scrolling a particular distance, use window.scrollBy().
window.scroll({
  top: 2500, 
  left: 0, 
  behavior: 'smooth' 
});

// scroll certain amounts from current position 
window.scrollBy({ 
  top: 100, // negative value acceptable
  left: 0, 
  behavior: 'smooth' 
});

// scroll to a certain element
document.querySelector('.hello').scrollIntoView({ 
  behavior: 'smooth' 
});

你可以放演示链接吗?
DV Yogesh

2
@Vishu Rana,行为支持仅适用于MDN中
Evandro Cavalcate Santos,2015年

2
@EvandroCavalcateSantos:不再了!;-) developer.mozilla.org/en-US/docs/Web/API/Element/...
佩德罗·费雷拉

1
这个现代标准很好并且可读,但是需要一个speed选择。当前滚动速度太慢,无法流畅播放。
尼尔斯·林德曼

31

编辑:此答案写于2013年。请检查以下有关requestAnimationFrame的CristianTraìna的评论

我做到了。下面的代码不依赖于任何框架。

限制:锚定活动未写在URL中。

代码版本:1.0 | GitHub:https : //github.com/Yappli/smooth-scroll

(function() // Code in a function to create an isolate scope
{
var speed = 500;
var moving_frequency = 15; // Affects performance !
var links = document.getElementsByTagName('a');
var href;
for(var i=0; i<links.length; i++)
{   
    href = (links[i].attributes.href === undefined) ? null : links[i].attributes.href.nodeValue.toString();
    if(href !== null && href.length > 1 && href.substr(0, 1) == '#')
    {
        links[i].onclick = function()
        {
            var element;
            var href = this.attributes.href.nodeValue.toString();
            if(element = document.getElementById(href.substr(1)))
            {
                var hop_count = speed/moving_frequency
                var getScrollTopDocumentAtBegin = getScrollTopDocument();
                var gap = (getScrollTopElement(element) - getScrollTopDocumentAtBegin) / hop_count;

                for(var i = 1; i <= hop_count; i++)
                {
                    (function()
                    {
                        var hop_top_position = gap*i;
                        setTimeout(function(){  window.scrollTo(0, hop_top_position + getScrollTopDocumentAtBegin); }, moving_frequency*i);
                    })();
                }
            }

            return false;
        };
    }
}

var getScrollTopElement =  function (e)
{
    var top = 0;

    while (e.offsetParent != undefined && e.offsetParent != null)
    {
        top += e.offsetTop + (e.clientTop != null ? e.clientTop : 0);
        e = e.offsetParent;
    }

    return top;
};

var getScrollTopDocument = function()
{
    return document.documentElement.scrollTop + document.body.scrollTop;
};
})();

2
我认为您只需使用类似的方式var _updateURL = function ( anchor, url ) { if ( (url === true || url === 'true') && history.pushState ) { history.pushState( {pos:anchor.id}, '', anchor ); } };通过Chris Fernandi的Smooth Scroll js)将锚点URL推送到历史堆栈,即可将锚点写入URL。很高兴看到人们这样做,而不仅仅是“使用JQuery”!记录下来,:target选择器也是一个很好的解决方案。
Louis Maddox 2014年

1
这个答案来自2013年,当时它已经被编写为requestAnimationFrame,这可以提高性能。我不知道为什么这里不使用。
CristianTraìna18年6

23

算法

滚动元素需要scrollTop随时间更改其值。对于给定的时间点,计算一个新scrollTop值。要平滑制作动画,请使用平滑步长算法进行插值。

计算scrollTop如下:

var point = smooth_step(start_time, end_time, now);
var scrollTop = Math.round(start_top + (distance * point));

哪里:

  • start_time 动画开始的时间;
  • end_time是动画结束的时间(start_time + duration);
  • start_topscrollTop开始时的值;和
  • distance是期望的最终值和起始值之间的差(target - start_top)

一个健壮的解决方案应该检测动画何时被中断,等等。阅读我有关无jQuery的平滑滚动的文章以了解详细信息。

演示版

参见JSFiddle

实作

编码:

/**
    Smoothly scroll element to the given target (element.scrollTop)
    for the given duration

    Returns a promise that's fulfilled when done, or rejected if
    interrupted
 */
var smooth_scroll_to = function(element, target, duration) {
    target = Math.round(target);
    duration = Math.round(duration);
    if (duration < 0) {
        return Promise.reject("bad duration");
    }
    if (duration === 0) {
        element.scrollTop = target;
        return Promise.resolve();
    }

    var start_time = Date.now();
    var end_time = start_time + duration;

    var start_top = element.scrollTop;
    var distance = target - start_top;

    // based on http://en.wikipedia.org/wiki/Smoothstep
    var smooth_step = function(start, end, point) {
        if(point <= start) { return 0; }
        if(point >= end) { return 1; }
        var x = (point - start) / (end - start); // interpolation
        return x*x*(3 - 2*x);
    }

    return new Promise(function(resolve, reject) {
        // This is to keep track of where the element's scrollTop is
        // supposed to be, based on what we're doing
        var previous_top = element.scrollTop;

        // This is like a think function from a game loop
        var scroll_frame = function() {
            if(element.scrollTop != previous_top) {
                reject("interrupted");
                return;
            }

            // set the scrollTop for this frame
            var now = Date.now();
            var point = smooth_step(start_time, end_time, now);
            var frameTop = Math.round(start_top + (distance * point));
            element.scrollTop = frameTop;

            // check if we're done!
            if(now >= end_time) {
                resolve();
                return;
            }

            // If we were supposed to scroll but didn't, then we
            // probably hit the limit, so consider it done; not
            // interrupted.
            if(element.scrollTop === previous_top
                && element.scrollTop !== frameTop) {
                resolve();
                return;
            }
            previous_top = element.scrollTop;

            // schedule next frame for execution
            setTimeout(scroll_frame, 0);
        }

        // boostrap the animation process
        setTimeout(scroll_frame, 0);
    });
}

这似乎是一个非常好的解决方案。但是如何在窗口对象而不是像Fiddle中的元素上实现它呢?
Mudlabs

没关系,我想通了。不得不使用pageYOffsetscrollTo()element替代scrollTop
Mudlabs

6

我在这里做了一个没有jQuery的示例:http : //codepen.io/sorinnn/pen/ovzdq

/**
    by Nemes Ioan Sorin - not an jQuery big fan 
    therefore this script is for those who love the old clean coding style  
    @id = the id of the element who need to bring  into view

    Note : this demo scrolls about 12.700 pixels from Link1 to Link3
*/
(function()
{
      window.setTimeout = window.setTimeout; //
})();

      var smoothScr = {
      iterr : 30, // set timeout miliseconds ..decreased with 1ms for each iteration
        tm : null, //timeout local variable
      stopShow: function()
      {
        clearTimeout(this.tm); // stopp the timeout
        this.iterr = 30; // reset milisec iterator to original value
      },
      getRealTop : function (el) // helper function instead of jQuery
      {
        var elm = el; 
        var realTop = 0;
        do
        {
          realTop += elm.offsetTop;
          elm = elm.offsetParent;
        }
        while(elm);
        return realTop;
      },
      getPageScroll : function()  // helper function instead of jQuery
      {
        var pgYoff = window.pageYOffset || document.body.scrollTop || document.documentElement.scrollTop;
        return pgYoff;
      },
      anim : function (id) // the main func
      {
        this.stopShow(); // for click on another button or link
        var eOff, pOff, tOff, scrVal, pos, dir, step;

        eOff = document.getElementById(id).offsetTop; // element offsetTop

        tOff =  this.getRealTop(document.getElementById(id).parentNode); // terminus point 

        pOff = this.getPageScroll(); // page offsetTop

        if (pOff === null || isNaN(pOff) || pOff === 'undefined') pOff = 0;

        scrVal = eOff - pOff; // actual scroll value;

        if (scrVal > tOff) 
        {
          pos = (eOff - tOff - pOff); 
          dir = 1;
        }
        if (scrVal < tOff)
        {
          pos = (pOff + tOff) - eOff;
          dir = -1; 
        }
        if(scrVal !== tOff) 
        {
          step = ~~((pos / 4) +1) * dir;

          if(this.iterr > 1) this.iterr -= 1; 
          else this.itter = 0; // decrease the timeout timer value but not below 0
          window.scrollBy(0, step);
          this.tm = window.setTimeout(function()
          {
             smoothScr.anim(id);  
          }, this.iterr); 
        }  
        if(scrVal === tOff) 
        { 
          this.stopShow(); // reset function values
          return;
        }
    }
 }

1
请在此处发布相关代码段或有关其工作方式的说明。您的答案应该可以理解,而无需我单击任何内容。谢谢!
爱德华

谢谢-但是没有-codeopen是为了共享示例-这里不是放置所有CSS / js / html代码的正确地方-util stackoverflow将添加必要的位-大多数人会将您发送到codepen / jsfiddle /或其他地方..
2014年

1
以下是一些有关我们为何要求这样做的对话:meta.stackexchange.com/q/149890
Edward

2
我不明白window.setTimeout = window.setTimeout;
pmrotule '17

6

现代浏览器支持CSS“滚动行为:平滑”属性。因此,我们甚至根本不需要任何Javascript。只需将其添加到body元素,然后使用常规锚点和链接即可。 滚动行为MDN文档


Edge和Safari均不支持此属性。我希望他们会很快。
6754534367 '19

它不能很好地工作..即使在Firefox中。因此,这不是解决方案。
huseyin39 '19

5

我最近开始着手解决无法使用jQuery的问题,因此我将解决方案记录在后,以供后代参考。

var scroll = (function() {

    var elementPosition = function(a) {
        return function() {
            return a.getBoundingClientRect().top;
        };
    };

    var scrolling = function( elementID ) {

        var el = document.getElementById( elementID ),
            elPos = elementPosition( el ),
            duration = 400,
            increment = Math.round( Math.abs( elPos() )/40 ),
            time = Math.round( duration/increment ),
            prev = 0,
            E;

        function scroller() {
            E = elPos();

            if (E === prev) {
                return;
            } else {
                prev = E;
            }

            increment = (E > -20 && E < 20) ? ((E > - 5 && E < 5) ? 1 : 5) : increment;

            if (E > 1 || E < -1) {

                if (E < 0) {
                    window.scrollBy( 0,-increment );
                } else {
                    window.scrollBy( 0,increment );
                }

                setTimeout(scroller, time);

            } else {

                el.scrollTo( 0,0 );

            }
        }

        scroller();
    };

    return {
        To: scrolling
    }

})();

/* usage */
scroll.To('elementID');

scroll()函数使用显露模块模式通过传递目标元素的ID到其scrolling()函数scroll.To('id'),后者设置函数使用的值scroller()

分解

scrolling()

  • el :目标DOM对象
  • elPos :通过返回函数 elememtPosition()该,每次调用时都会给出目标元素相对于页面顶部的位置。
  • duration :转换时间(以毫秒为单位)。
  • increment :将目标元素的起始位置分为40步。
  • time :设置每个步骤的时间。
  • prev:目标元素在中的先前位置scroller()
  • E:保留目标元素在中的位置scroller()

实际工作由scroller()继续调用自身的函数完成(通过setTimeout())直到目标元素位于页面顶部或页面不再滚动为止。

每次scroller()调用时,它都会检查目标元素的当前位置(保留在变量中E),如果是> 1OR < -1,或者页面仍可滚动,则将窗口按increment像素移动-上下移动,取决于E是正值还是负值。当E不是> 1OR< -1E===时prev,函数将停止。我DOMElement.scrollTo()在完成时添加了该方法,只是为了确保目标元素在窗口顶部爆炸(不是您会注意到它只是一个像素点!)。

if2行的语句scroller()通过检查E其先前位置来检查页面是否正在滚动(如果目标可能朝页面底部,并且页面无法进一步滚动)prev))。

其下的三元条件将increment值降低为E接近零。这将阻止页面以一种方式超调,然后弹跳以另一种方式超调,然后弹回以另一种方式超调,例如乒乓球样式,直至无穷远。

如果您的页面高度超过c.4000px,则可能要增加三元表达式的第一个条件的值(此处为+/- 20)和/或设置该increment值的除数(此处为40)。

duration,设置的除数increment以及的三元条件下的值一起玩scroller()应该可以使您定制适合您页面的功能。

  • JSFiddle

  • NB在Lubuntu的最新版本Firefox和Chrome以及Windows8上的Firefox,Chrome和IE中进行了测试。



2

我做了这样的事情。我不知道它是否可以在IE8中使用。在IE9,Mozilla,Chrome,Edge中进行了测试。

function scroll(toElement, speed) {
  var windowObject = window;
  var windowPos = windowObject.pageYOffset;
  var pointer = toElement.getAttribute('href').slice(1);
  var elem = document.getElementById(pointer);
  var elemOffset = elem.offsetTop;

  var counter = setInterval(function() {
    windowPos;

    if (windowPos > elemOffset) { // from bottom to top
      windowObject.scrollTo(0, windowPos);
      windowPos -= speed;

      if (windowPos <= elemOffset) { // scrolling until elemOffset is higher than scrollbar position, cancel interval and set scrollbar to element position
        clearInterval(counter);
        windowObject.scrollTo(0, elemOffset);
      }
    } else { // from top to bottom
      windowObject.scrollTo(0, windowPos);
      windowPos += speed;

      if (windowPos >= elemOffset) { // scroll until scrollbar is lower than element, cancel interval and set scrollbar to element position
        clearInterval(counter);
        windowObject.scrollTo(0, elemOffset);
      }
    }

  }, 1);
}

//call example

var navPointer = document.getElementsByClassName('nav__anchor');

for (i = 0; i < navPointer.length; i++) {
  navPointer[i].addEventListener('click', function(e) {
    scroll(this, 18);
    e.preventDefault();
  });
}

描述

  • pointer—如果元素具有属性“ href”,则获取元素和chceck,如果是,则摆脱“#”
  • elem—没有“#”的指针变量
  • elemOffset—页面顶部的“滚动到”元素的偏移量

2

您可以使用

document.querySelector('your-element').scrollIntoView({behavior: 'smooth'});

如果要滚动到页面顶部,可以在顶部放置一个空元素,然后平滑滚动到该元素。


0

您可以将for循环与window.scrollTo和setTimeout配合使用,以使用纯Javascript平滑滚动。用我的scrollToSmoothly函数滚动到一个元素:(scrollToSmoothly(elem.offsetTop)假设elem是一个DOM元素)。您可以使用它来平滑滚动到文档中的任何y位置。

function scrollToSmoothly(pos, time){
/*Time is only applicable for scrolling upwards*/
/*Code written by hev1*/
/*pos is the y-position to scroll to (in pixels)*/
     if(isNaN(pos)){
      throw "Position must be a number";
     }
     if(pos<0){
     throw "Position can not be negative";
     }
    var currentPos = window.scrollY||window.screenTop;
    if(currentPos<pos){
    var t = 10;
       for(let i = currentPos; i <= pos; i+=10){
       t+=10;
        setTimeout(function(){
        window.scrollTo(0, i);
        }, t/2);
      }
    } else {
    time = time || 2;
       var i = currentPos;
       var x;
      x = setInterval(function(){
         window.scrollTo(0, i);
         i -= 10;
         if(i<=pos){
          clearInterval(x);
         }
     }, time);
      }
}

演示:


0
<script>
var set = 0;

function animatescroll(x, y) {
    if (set == 0) {
        var val72 = 0;
        var val73 = 0;
        var setin = 0;
        set = 1;

        var interval = setInterval(function() {
            if (setin == 0) {
                val72++;
                val73 += x / 1000;
                if (val72 == 1000) {
                    val73 = 0;
                    interval = clearInterval(interval);
                }
                document.getElementById(y).scrollTop = val73;
            }
        }, 1);
    }
}
</script>

x = scrollTop
y =用于滚动的div的ID

注意: 要使主体滚动,请为主体指定一个ID。


0

这是我的解决方案。适用于大多数浏览器

document.getElementById("scrollHere").scrollIntoView({behavior: "smooth"});

文件

document.getElementById("end").scrollIntoView({behavior: "smooth"});
body {margin: 0px; display: block; height: 100%; background-image: linear-gradient(red, yellow);}
.start {display: block; margin: 100px 10px 1000px 0px;}
.end {display: block; margin: 0px 0px 100px 0px;}
<div class="start">Start</div>
<div class="end" id="end">End</div>



-1

这是对我有用的代码。

`$('a[href*="#"]')

    .not('[href="#"]')
    .not('[href="#0"]')
    .click(function(event) {
     if (
       location.pathname.replace(/^\//, '') == this.pathname.replace(/^\//, '')
       &&
       location.hostname == this.hostname
        ) {

  var target = $(this.hash);
  target = target.length ? target : $('[name=' + this.hash.slice(1) + ']');
  if (target.length) {

    event.preventDefault();
    $('html, body').animate({
      scrollTop: target.offset().top
    }, 1000, function() {

      var $target = $(target);
      $target.focus();
      if ($target.is(":focus")) { 
        return false;
      } else {
        $target.attr('tabindex','-1'); 
        $target.focus(); 
      };
    });
    }
   }
  });

`


您能否进一步解释一下代码的作用?
rhavelka

欢迎使用stackoverflow,感谢您的回答,但问题是关于不使用jquery。
斯蒂芬·L
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.