| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828 |
- /*!
- * Swipe 2.2.14
- *
- * Brad Birdsall
- * Copyright 2013, MIT License
- *
- */
- // if the module has no dependencies, the above pattern can be simplified to
- // eslint-disable-next-line no-extra-semi
- ; (function (root, factory) {
- // eslint-disable-next-line no-undef
- if (typeof define === 'function' && define.amd) {
- // AMD. Register as an anonymous module.
- // eslint-disable-next-line no-undef
- define([], function () {
- root.Swipe = factory();
- return root.Swipe;
- });
- } else if (typeof module === 'object' && module.exports) {
- // Node. Does not work with strict CommonJS, but
- // only CommonJS-like environments that support module.exports,
- // like Node.
- module.exports = factory();
- } else {
- // Browser globals
- root.Swipe = factory();
- }
- }(this, function () {
- // Establish the root object, `window` (`self`) in the browser, `global`
- // on the server, or `this` in some virtual machines. We use `self`
- // instead of `window` for `WebWorker` support.
- var root = typeof self == 'object' && self.self === self && self ||
- typeof global == 'object' && global.global === global && global ||
- this;
- var _document = root.document;
- function Swipe(container, options) {
- 'use strict';
- options = options || {};
- // setup initial vars
- var start = {};
- var delta = {};
- var isScrolling;
- // setup auto slideshow
- var delay = options.auto || 0;
- var interval;
- var disabled = false;
- // utilities
- // simple no operation function
- var noop = function () { };
- // offload a functions execution
- var offloadFn = function (fn) { setTimeout(fn || noop, 0); };
- // Returns a function, that, as long as it continues to be invoked, will not
- // be triggered.
- var throttle = function (fn, threshhold) {
- threshhold = threshhold || 100;
- var timeout = null;
- function cancel() {
- if (timeout) clearTimeout(timeout);
- }
- function throttledFn() {
- var context = this;
- var args = arguments;
- cancel();
- timeout = setTimeout(function () {
- timeout = null;
- fn.apply(context, args);
- }, threshhold);
- }
- // allow remove throttled timeout
- throttledFn.cancel = cancel;
- return throttledFn;
- };
- function getScrollbarWidth() {
- return root.innerWidth - _document.documentElement.clientWidth;
- }
- // check whether event is cancelable
- var isCancelable = function (event) {
- if (!event) return false;
- return typeof event.cancelable !== 'boolean' || event.cancelable;
- };
- // check browser capabilities
- var browser = {
- addEventListener: !!root.addEventListener,
- passiveEvents: (function () {
- // Test via a getter in the options object to see if the passive property is accessed
- var supportsPassive = false;
- try {
- var opts = Object.defineProperty({}, 'passive', {
- // eslint-disable-next-line getter-return
- get: function () {
- supportsPassive = true;
- }
- });
- root.addEventListener('testEvent', null, opts);
- root.removeEventListener('testEvent', null, opts);
- }
- catch (e) {
- supportsPassive = false;
- }
- return supportsPassive;
- })(),
- // eslint-disable-next-line no-undef
- touch: ('ontouchstart' in root) || root.DocumentTouch && _document instanceof DocumentTouch,
- transitions: (function (temp) {
- var props = ['transitionProperty', 'WebkitTransition', 'MozTransition', 'OTransition', 'msTransition'];
- for (var i in props) {
- if (temp.style[props[i]] !== undefined) {
- return true;
- }
- }
- return false;
- })(_document.createElement('swipe'))
- };
- // quit if no root element
- if (!container) return;
- var element = container.children[0];
- var slides, slidePos, width, length;
- var index = parseInt(options.startSlide, 10) || 0;
- var speed = options.speed || 300;
- options.continuous = options.continuous !== undefined ? options.continuous : true;
- // check text direction
- var slideDir = (function (el, prop, dir) {
- if (el.currentStyle) {
- dir = el.currentStyle[prop];
- } else if (root.getComputedStyle) {
- dir = root.getComputedStyle(el, null).getPropertyValue(prop);
- }
- return 'rtl' === dir ? 'right' : 'left';
- })(container, 'direction');
- // AutoRestart option: auto restart slideshow after user's touch event
- options.autoRestart = options.autoRestart !== undefined ? options.autoRestart : false;
- // throttled setup
- var throttledSetup = throttle(setup);
- // setup event capturing
- var events = {
- handleEvent: function (event) {
- if (disabled) return;
- switch (event.type) {
- case 'mousedown':
- case 'touchstart': this.start(event); break;
- case 'mousemove':
- case 'touchmove': this.move(event); break;
- case 'mouseup':
- case 'mouseleave':
- case 'touchend': this.end(event); break;
- case 'webkitTransitionEnd':
- case 'msTransitionEnd':
- case 'oTransitionEnd':
- case 'otransitionend':
- case 'transitionend': this.transitionEnd(event); break;
- case 'resize': throttledSetup(); break;
- }
- if (options.stopPropagation) {
- event.stopPropagation();
- }
- },
- start: function (event) {
- var touches;
- if (isMouseEvent(event)) {
- touches = event;
- event.preventDefault(); // For desktop Safari drag
- } else {
- touches = event.touches[0];
- }
- // measure start values
- start = {
- // get initial touch coords
- x: touches.pageX,
- y: touches.pageY,
- // store time to determine touch duration
- time: +new Date()
- };
- // used for testing first move event
- isScrolling = undefined;
- // reset delta and end measurements
- delta = {};
- // attach touchmove and touchend listeners
- if (isMouseEvent(event)) {
- element.addEventListener('mousemove', this, false);
- element.addEventListener('mouseup', this, false);
- element.addEventListener('mouseleave', this, false);
- } else {
- element.addEventListener('touchmove', this, browser.passiveEvents ? { passive: false } : false);
- element.addEventListener('touchend', this, false);
- }
- },
- move: function (event) {
- var touches;
- if (isMouseEvent(event)) {
- touches = event;
- } else {
- // ensure swiping with one touch and not pinching
- if (event.touches.length > 1 || event.scale && event.scale !== 1) {
- return;
- }
- // we can disable scrolling unless it is already in progress
- if (options.disableScroll && isCancelable(event)) {
- event.preventDefault();
- }
- touches = event.touches[0];
- }
- // measure change in x and y
- delta = {
- x: touches.pageX - start.x,
- y: touches.pageY - start.y
- };
- // determine if scrolling test has run - one time test
- if (typeof isScrolling === 'undefined') {
- isScrolling = !!(isScrolling || Math.abs(delta.x) < Math.abs(delta.y));
- }
- // if user is not trying to scroll vertically
- if (!isScrolling) {
- // if it is not already scrolling
- if (isCancelable(event)) {
- // prevent native scrolling
- event.preventDefault();
- }
- // stop slideshow
- stop();
- // increase resistance if first or last slide
- if (options.continuous) { // we don't add resistance at the end
- translate(circle(index - 1), delta.x + slidePos[circle(index - 1)], 0);
- translate(index, delta.x + slidePos[index], 0);
- translate(circle(index + 1), delta.x + slidePos[circle(index + 1)], 0);
- } else {
- delta.x =
- delta.x /
- ((!index && delta.x > 0 || // if first slide and sliding left
- index === slides.length - 1 && // or if last slide and sliding right
- delta.x < 0 // and if sliding at all
- ) ?
- (Math.abs(delta.x) / width + 1) // determine resistance level
- : 1); // no resistance if false
- // translate 1:1
- translate(index - 1, delta.x + slidePos[index - 1], 0);
- translate(index, delta.x + slidePos[index], 0);
- translate(index + 1, delta.x + slidePos[index + 1], 0);
- }
- }
- },
- end: function (event) {
- // measure duration
- var duration = +new Date() - start.time;
- // determine if slide attempt triggers next/prev slide
- var isValidSlide =
- Number(duration) < 250 && // if slide duration is less than 250ms
- Math.abs(delta.x) > 20 || // and if slide amt is greater than 20px
- Math.abs(delta.x) > width / 2; // or if slide amt is greater than half the width
- // determine if slide attempt is past start and end
- var isPastBounds =
- !index && delta.x > 0 || // if first slide and slide amt is greater than 0
- index === slides.length - 1 && delta.x < 0; // or if last slide and slide amt is less than 0
- if (options.continuous) {
- isPastBounds = false;
- }
- // OLD determine direction of swipe (true:right, false:left)
- // determine direction of swipe (1: backward, -1: forward)
- var direction = Math.abs(delta.x) / delta.x;
- // if not scrolling vertically
- if (!isScrolling) {
- if (isValidSlide && !isPastBounds) {
- // if we're moving right
- if (direction < 0) {
- if (options.continuous) { // we need to get the next in this direction in place
- move(circle(index - 1), -width, 0);
- move(circle(index + 2), width, 0);
- } else {
- move(index - 1, -width, 0);
- }
- move(index, slidePos[index] - width, speed);
- move(circle(index + 1), slidePos[circle(index + 1)] - width, speed);
- index = circle(index + 1);
- } else {
- if (options.continuous) { // we need to get the next in this direction in place
- move(circle(index + 1), width, 0);
- move(circle(index - 2), -width, 0);
- } else {
- move(index + 1, width, 0);
- }
- move(index, slidePos[index] + width, speed);
- move(circle(index - 1), slidePos[circle(index - 1)] + width, speed);
- index = circle(index - 1);
- }
- runCallback(getPos(), slides[index], direction);
- } else {
- if (options.continuous) {
- move(circle(index - 1), -width, speed);
- move(index, 0, speed);
- move(circle(index + 1), width, speed);
- } else {
- move(index - 1, -width, speed);
- move(index, 0, speed);
- move(index + 1, width, speed);
- }
- }
- }
- // kill touchmove and touchend event listeners until touchstart called again
- if (isMouseEvent(event)) {
- element.removeEventListener('mousemove', events, false);
- element.removeEventListener('mouseup', events, false);
- element.removeEventListener('mouseleave', events, false);
- } else {
- element.removeEventListener('touchmove', events, browser.passiveEvents ? { passive: false } : false);
- element.removeEventListener('touchend', events, false);
- }
- },
- transitionEnd: function (event) {
- var currentIndex = parseInt(event.target.getAttribute('data-index'), 10);
- if (currentIndex === index) {
- if (delay || options.autoRestart) restart();
- runTransitionEnd(getPos(), slides[index]);
- }
- }
- };
- // trigger setup
- setup();
- // start auto slideshow if applicable
- begin();
- // Expose the Swipe API
- return {
- // initialize
- setup: setup,
- // go to slide
- slide: function (to, speed) {
- stop();
- slide(to, speed);
- },
- // move to previous
- prev: function () {
- stop();
- prev();
- },
- // move to next
- next: function () {
- stop();
- next();
- },
- // Restart slideshow
- restart: restart,
- // cancel slideshow
- stop: stop,
- // return current index position
- getPos: getPos,
- // disable slideshow
- disable: disable,
- // enable slideshow
- enable: enable,
- // return total number of slides
- getNumSlides: function () { return length; },
- // completely remove swipe
- kill: kill
- };
- // remove all event listeners
- function detachEvents() {
- if (browser.addEventListener) {
- // remove current event listeners
- element.removeEventListener('touchstart', events, browser.passiveEvents ? { passive: true } : false);
- element.removeEventListener('mousedown', events, false);
- element.removeEventListener('webkitTransitionEnd', events, false);
- element.removeEventListener('msTransitionEnd', events, false);
- element.removeEventListener('oTransitionEnd', events, false);
- element.removeEventListener('otransitionend', events, false);
- element.removeEventListener('transitionend', events, false);
- root.removeEventListener('resize', events, false);
- } else {
- root.onresize = null;
- }
- }
- // add event listeners
- function attachEvents() {
- if (browser.addEventListener) {
- // set touchstart event on element
- if (browser.touch) {
- element.addEventListener('touchstart', events, browser.passiveEvents ? { passive: true } : false);
- }
- if (options.draggable) {
- element.addEventListener('mousedown', events, false);
- }
- if (browser.transitions) {
- element.addEventListener('webkitTransitionEnd', events, false);
- element.addEventListener('msTransitionEnd', events, false);
- element.addEventListener('oTransitionEnd', events, false);
- element.addEventListener('otransitionend', events, false);
- element.addEventListener('transitionend', events, false);
- }
- // set resize event on window
- root.addEventListener('resize', events, false);
- } else {
- root.onresize = throttledSetup; // to play nice with old IE
- }
- }
- // clone nodes when there is only two slides
- function cloneNode(el) {
- var clone = el.cloneNode(true);
- element.appendChild(clone);
- // tag these slides as clones (to remove them on kill)
- clone.setAttribute('data-cloned', true);
- // Remove id from element
- clone.removeAttribute('id');
- }
- function setup(opts) {
- // Overwrite options if necessary
- if (opts != null) {
- for (var prop in opts) {
- options[prop] = opts[prop];
- }
- }
- // cache slides
- slides = element.children;
- length = slides.length;
- // slides length correction, minus cloned slides
- for (var i = 0; i < slides.length; i++) {
- if (slides[i].getAttribute('data-cloned')) length--;
- }
- // set continuous to false if only one slide
- if (slides.length < 2) {
- options.continuous = false;
- }
- // special case if two slides
- if (browser.transitions && options.continuous && slides.length < 3) {
- cloneNode(slides[0]);
- cloneNode(slides[1]);
- slides = element.children;
- }
- // adjust style on rtl
- if ('right' === slideDir) {
- for (var j = 0; j < slides.length; j++) {
- slides[j].style.float = 'right';
- }
- }
- // create an array to store current positions of each slide
- slidePos = new Array(slides.length);
- // determine width of each slide
- //width = container.getBoundingClientRect().width || container.offsetWidth;
- width = root.innerWidth - getScrollbarWidth();
- element.style.width = (slides.length * width * 2) + 'px';
- // stack elements
- var pos = slides.length;
- while (pos--) {
- var slide = slides[pos];
- slide.style.width = width + 'px';
- slide.setAttribute('data-index', pos);
- if (browser.transitions) {
- slide.style[slideDir] = (pos * -width) + 'px';
- move(pos, index > pos ? -width : (index < pos ? width : 0), 0);
- }
- }
- // reposition elements before and after index
- if (options.continuous && browser.transitions) {
- move(circle(index - 1), -width, 0);
- move(circle(index + 1), width, 0);
- }
- if (!browser.transitions) {
- element.style[slideDir] = (index * -width) + 'px';
- }
- container.style.visibility = 'visible';
- // reinitialize events
- detachEvents();
- attachEvents();
- }
- function prev() {
- if (disabled) return;
- if (options.continuous) {
- slide(index - 1);
- } else if (index) {
- slide(index - 1);
- }
- }
- function next() {
- if (disabled) return;
- if (options.continuous) {
- slide(index + 1);
- } else if (index < slides.length - 1) {
- slide(index + 1);
- }
- }
- function runCallback(pos, index, dir) {
- if (options.callback) {
- options.callback(pos, index, dir);
- }
- }
- function runTransitionEnd(pos, index) {
- if (options.transitionEnd) {
- options.transitionEnd(pos, index);
- }
- }
- function circle(index) {
- // a simple positive modulo using slides.length
- return (slides.length + (index % slides.length)) % slides.length;
- }
- function getPos() {
- // Fix for the clone issue in the event of 2 slides
- var currentIndex = index;
- if (currentIndex >= length) {
- currentIndex = currentIndex - length;
- }
- return currentIndex;
- }
- function slide(to, slideSpeed) {
- // ensure to is of type 'number'
- to = typeof to !== 'number' ? parseInt(to, 10) : to;
- // do nothing if already on requested slide
- if (index === to) return;
- if (browser.transitions) {
- var direction = Math.abs(index - to) / (index - to); // 1: backward, -1: forward
- // get the actual position of the slide
- if (options.continuous) {
- var natural_direction = direction;
- direction = -slidePos[circle(to)] / width;
- // if going forward but to < index, use to = slides.length + to
- // if going backward but to > index, use to = -slides.length + to
- if (direction !== natural_direction) {
- to = -direction * slides.length + to;
- }
- }
- var diff = Math.abs(index - to) - 1;
- // move all the slides between index and to in the right direction
- while (diff--) {
- move(circle((to > index ? to : index) - diff - 1), width * direction, 0);
- }
- to = circle(to);
- move(index, width * direction, slideSpeed || speed);
- move(to, 0, slideSpeed || speed);
- if (options.continuous) { // we need to get the next in place
- move(circle(to - direction), -(width * direction), 0);
- }
- } else {
- to = circle(to);
- animate(index * -width, to * -width, slideSpeed || speed);
- // no fallback for a circular continuous if the browser does not accept transitions
- }
- index = to;
- offloadFn(function () {
- runCallback(getPos(), slides[index], direction);
- });
- }
- function move(index, dist, speed) {
- translate(index, dist, speed);
- slidePos[index] = dist;
- }
- function translate(index, dist, speed) {
- var slide = slides[index];
- var style = slide && slide.style;
- if (!style) return;
- style.webkitTransitionDuration =
- style.MozTransitionDuration =
- style.msTransitionDuration =
- style.OTransitionDuration =
- style.transitionDuration = speed + 'ms';
- style.webkitTransform =
- style.msTransform =
- style.MozTransform =
- style.OTransform =
- style.transform = 'translateX(' + dist + 'px)';
- }
- function animate(from, to, speed) {
- // if not an animation, just reposition
- if (!speed) {
- element.style[slideDir] = to + 'px';
- return;
- }
- var start = +new Date();
- var timer = setInterval(function () {
- var timeElap = +new Date() - start;
- if (timeElap > speed) {
- element.style[slideDir] = to + 'px';
- if (delay || options.autoRestart) restart();
- runTransitionEnd(getPos(), slides[index]);
- clearInterval(timer);
- return;
- }
- element.style[slideDir] = (((to - from) * (Math.floor((timeElap / speed) * 100) / 100)) + from) + 'px';
- }, 4);
- }
- function begin() {
- delay = options.auto || 0;
- if (delay) interval = setTimeout(next, delay);
- }
- function stop() {
- delay = 0;
- clearTimeout(interval);
- }
- function restart() {
- stop();
- begin();
- }
- function disable() {
- stop();
- disabled = true;
- }
- function enable() {
- disabled = false;
- restart();
- }
- function isMouseEvent(e) {
- return /^mouse/.test(e.type);
- }
- function kill() {
- // cancel slideshow
- stop();
- // remove inline styles
- container.style.visibility = '';
- // reset element
- element.style.width = '';
- element.style[slideDir] = '';
- // reset slides
- var pos = slides.length;
- while (pos--) {
- if (browser.transitions) {
- translate(pos, 0, 0);
- }
- var slide = slides[pos];
- // if the slide is tagged as clone, remove it
- if (slide.getAttribute('data-cloned')) {
- var _parent = slide.parentElement;
- _parent.removeChild(slide);
- }
- // remove styles
- slide.style.width = '';
- slide.style[slideDir] = '';
- slide.style.webkitTransitionDuration =
- slide.style.MozTransitionDuration =
- slide.style.msTransitionDuration =
- slide.style.OTransitionDuration =
- slide.style.transitionDuration = '';
- slide.style.webkitTransform =
- slide.style.msTransform =
- slide.style.MozTransform =
- slide.style.OTransform = '';
- // remove custom attributes (?)
- // slide.removeAttribute('data-index');
- }
- // remove all events
- detachEvents();
- // remove throttled function timeout
- throttledSetup.cancel();
- }
- }
- if (root.jQuery || root.Zepto) {
- (function ($) {
- $.fn.Swipe = function (params) {
- return this.each(function () {
- $(this).data('Swipe', new Swipe($(this)[0], params));
- });
- };
- })(root.jQuery || root.Zepto);
- }
- return Swipe;
- }));
|