Skip to content

Instantly share code, notes, and snippets.

@AGresvig
Forked from Eyal-Shalev/scroller.js
Last active August 29, 2015 14:14
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save AGresvig/971518335a4615802730 to your computer and use it in GitHub Desktop.
Save AGresvig/971518335a4615802730 to your computer and use it in GitHub Desktop.
Forked for Node compatibility
'use strict';
/*
* Based on work done by Nikhil Nigade (@dezinezync) https://gist.github.com/dezinezync/5487119
*/
var DEFAULT_ELEMENT = document.documentElement.scrollTop ? document.documentElement : document.body;
var REQUEST_ANIMATION_FRAME = window.requestAnimationFrame || window.mozRequestAnimationFrame ||
window.webkitRequestAnimationFrame || window.oRequestAnimationFrame;
/**
* Creates a new Scroller object.
*
* Usage example:
var SomeHTMLContainer = document.getElementById('some-scroll-container');
var scrollTime = 1000;
var someCallbackFunction = function() {console.log('hello');};
var scroller = new Scroller(SomeHTMLContainer, scrollTime)
.easing(Scroller.EASING.easeInQuad)
.callback(someCallbackFunction);
var top = 50, left = 150;
scroller.to(top,left);
*
* @param {HTMLElement} element
* @param {Number} duration
*/
var Scroller = function Scroller(element, duration) {
element = element ? element : DEFAULT_ELEMENT;
var prefs = {
easing: Scroller.EASING.linear,
callback: null
};
var callback = function () {
if (prefs.callback) {
prefs.callback();
}
};
/**
* Sets the easing function function to use.
* @param {Function} easing The easing function to use.
* @returns {Scroller}
*/
this.easing = function (easing) {
prefs.easing = easing;
return this;
};
/**
* Sets the callback function to call at the end of the scroll.
* @param {Function} callback
* @returns {Scroller}
*/
this.callback = function (callback) {
prefs.callback = callback;
return this;
};
/**
* Scrolls the prepared element to targetX and targetY.
* @param {Number} targetX
* @param {Number} targetY
*/
this.to = function (targetX, targetY) {
var startX = element.scrollLeft,
startY = element.scrollTop;
var distanceX = (targetX - startX),
distanceY = (targetY - startY);
// Prevent Scrolling if already there.
if (startX === targetX && startY === targetY) {
return callback();
}
var startTime = Date.now();
var scroll = function (/*timestamp*/) {
var currentTime = Date.now(),
time = Math.min(1, ((currentTime - startTime) / duration)),
easedT = prefs.easing(time);
element.scrollTop = (easedT * distanceY) + startY;
element.scrollLeft = (easedT * distanceX) + startX;
if (time < 1) {
REQUEST_ANIMATION_FRAME.call(window, scroll);
}
else callback();
};
REQUEST_ANIMATION_FRAME(scroll);
};
};
/* bits and bytes of the scrollTo function inspired by the works of Benjamin DeCock */
/*
* Easing Functions - inspired from http://gizma.com/easing/
* only considering the t value for the range [0, 1] => [0, 1]
*/
Scroller.EASING = {
// no easing, no acceleration
linear: function (t) {
return t;
},
// accelerating from zero velocity
easeInQuad: function (t) {
return t * t;
},
// decelerating to zero velocity
easeOutQuad: function (t) {
return t * (2 - t);
},
// acceleration until halfway, then deceleration
easeInOutQuad: function (t) {
return t < .5 ? 2 * t * t : -1 + (4 - 2 * t) * t;
},
// accelerating from zero velocity
easeInCubic: function (t) {
return t * t * t;
},
// decelerating to zero velocity
easeOutCubic: function (t) {
return (--t) * t * t + 1;
},
// acceleration until halfway, then deceleration
easeInOutCubic: function (t) {
return t < .5 ? 4 * t * t * t : (t - 1) * (2 * t - 2) * (2 * t - 2) + 1;
},
// accelerating from zero velocity
easeInQuart: function (t) {
return t * t * t * t;
},
// decelerating to zero velocity
easeOutQuart: function (t) {
return 1 - (--t) * t * t * t;
},
// acceleration until halfway, then deceleration
easeInOutQuart: function (t) {
return t < .5 ? 8 * t * t * t * t : 1 - 8 * (--t) * t * t * t;
},
// accelerating from zero velocity
easeInQuint: function (t) {
return t * t * t * t * t;
},
// decelerating to zero velocity
easeOutQuint: function (t) {
return 1 + (--t) * t * t * t * t;
},
// acceleration until halfway, then deceleration
easeInOutQuint: function (t) {
return t < .5 ? 16 * t * t * t * t * t : 1 + 16 * (--t) * t * t * t * t;
}
};
module.exports = Scroller;
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment