;/** * Owl Carousel v2.3.4 * Copyright 2013-2018 David Deutsch * Licensed under: SEE LICENSE IN https://github.com/OwlCarousel2/OwlCarousel2/blob/master/LICENSE */ /** * Owl carousel * * @version 2.3.4 * @author Bartosz Wojciechowski * @author David Deutsch * @license The MIT License (MIT) * @todo Lazy Load Icon * @todo prevent animationend bubling * @todo itemsScaleUp * @todo Test Zepto * @todo stagePadding calculate wrong active classes */(function($, window, document, undefined) { /** * Creates a carousel. * * @class The Owl Carousel. * @public * @param {HTMLElement|jQuery} element - The element to create the carousel for. * @param {Object} [options] - The options */ function Owl(element, options) { /** * Current settings for the carousel. * * @public */ this.settings = null; /** * Current options set by the caller including defaults. * * @public */ this.options = $.extend( {}, Owl.Defaults, options ); /** * Plugin element. * * @public */ this.$element = $( element ); /** * Proxied event handlers. * * @protected */ this._handlers = {}; /** * References to the running plugins of this carousel. * * @protected */ this._plugins = {}; /** * Currently suppressed events to prevent them from being retriggered. * * @protected */ this._supress = {}; /** * Absolute current position. * * @protected */ this._current = null; /** * Animation speed in milliseconds. * * @protected */ this._speed = null; /** * Coordinates of all items in pixel. * * @todo The name of this member is missleading. * @protected */ this._coordinates = []; /** * Current breakpoint. * * @todo Real media queries would be nice. * @protected */ this._breakpoint = null; /** * Current width of the plugin element. */ this._width = null; /** * All real items. * * @protected */ this._items = []; /** * All cloned items. * * @protected */ this._clones = []; /** * Merge values of all items. * * @todo Maybe this could be part of a plugin. * @protected */ this._mergers = []; /** * Widths of all items. */ this._widths = []; /** * Invalidated parts within the update process. * * @protected */ this._invalidated = {}; /** * Ordered list of workers for the update process. * * @protected */ this._pipe = []; /** * Current state information for the drag operation. * * @todo #261 * @protected */ this._drag = { time: null, target: null, pointer: null, stage: { start: null, current: null }, direction: null }; /** * Current state information and their tags. * * @type {Object} * @protected */ this._states = { current: {}, tags: { 'initializing': [ 'busy' ], 'animating': [ 'busy' ], 'dragging': [ 'interacting' ] } }; $.each( [ 'onResize', 'onThrottledResize' ], $.proxy( function(i, handler) { this._handlers[handler] = $.proxy( this[handler], this ); }, this ) ); $.each( Owl.Plugins, $.proxy( function(key, plugin) { this._plugins[key.charAt( 0 ).toLowerCase() + key.slice( 1 )] = new plugin( this ); }, this ) ); $.each( Owl.Workers, $.proxy( function(priority, worker) { this._pipe.push( { 'filter': worker.filter, 'run': $.proxy( worker.run, this ) } ); }, this ) ); this.setup(); this.initialize(); } /** * Default options for the carousel. * * @public */ Owl.Defaults = { items: 3, loop: false, center: false, rewind: false, checkVisibility: true, mouseDrag: true, touchDrag: true, pullDrag: true, freeDrag: false, margin: 0, stagePadding: 0, merge: false, mergeFit: true, autoWidth: false, startPosition: 0, rtl: false, smartSpeed: 250, fluidSpeed: false, dragEndSpeed: false, responsive: {}, responsiveRefreshRate: 200, responsiveBaseElement: window, fallbackEasing: 'swing', slideTransition: '', info: false, nestedItemSelector: false, itemElement: 'div', stageElement: 'div', refreshClass: 'owl-refresh', loadedClass: 'owl-loaded', loadingClass: 'owl-loading', rtlClass: 'owl-rtl', responsiveClass: 'owl-responsive', dragClass: 'owl-drag', itemClass: 'owl-item', stageClass: 'owl-stage', stageOuterClass: 'owl-stage-outer', grabClass: 'owl-grab' }; /** * Enumeration for width. * * @public * @readonly * @enum {String} */ Owl.Width = { Default: 'default', Inner: 'inner', Outer: 'outer' }; /** * Enumeration for types. * * @public * @readonly * @enum {String} */ Owl.Type = { Event: 'event', State: 'state' }; /** * Contains all registered plugins. * * @public */ Owl.Plugins = {}; /** * List of workers involved in the update process. */ Owl.Workers = [ { filter: [ 'width', 'settings' ], run: function() { this._width = this.$element.width(); } }, { filter: [ 'width', 'items', 'settings' ], run: function(cache) { cache.current = this._items && this._items[this.relative( this._current )]; } }, { filter: [ 'items', 'settings' ], run: function() { this.$stage.children( '.cloned' ).remove(); } }, { filter: [ 'width', 'items', 'settings' ], run: function(cache) { var margin = this.settings.margin || '', grid = ! this.settings.autoWidth, rtl = this.settings.rtl, css = { 'width': 'auto', 'margin-left': rtl ? margin : '', 'margin-right': rtl ? '' : margin }; ! grid && this.$stage.children().css( css ); cache.css = css; } }, { filter: [ 'width', 'items', 'settings' ], run: function(cache) { var width = (this.width() / this.settings.items).toFixed( 3 ) - this.settings.margin, merge = null, iterator = this._items.length, grid = ! this.settings.autoWidth, widths = []; cache.items = { merge: false, width: width }; while (iterator--) { merge = this._mergers[iterator]; merge = this.settings.mergeFit && Math.min( merge, this.settings.items ) || merge; cache.items.merge = merge > 1 || cache.items.merge; widths[iterator] = ! grid ? this._items[iterator].width() : width * merge; } this._widths = widths; } }, { filter: [ 'items', 'settings' ], run: function() { var clones = [], items = this._items, settings = this.settings, // TODO: Should be computed from number of min width items in stage. view = Math.max( settings.items * 2, 4 ), size = Math.ceil( items.length / 2 ) * 2, repeat = settings.loop && items.length ? settings.rewind ? view : Math.max( view, size ) : 0, append = '', prepend = ''; repeat /= 2; while (repeat > 0) { // Switch to only using appended clones. clones.push( this.normalize( clones.length / 2, true ) ); append = append + items[clones[clones.length - 1]][0].outerHTML; clones.push( this.normalize( items.length - 1 - (clones.length - 1) / 2, true ) ); prepend = items[clones[clones.length - 1]][0].outerHTML + prepend; repeat -= 1; } this._clones = clones; $( append ).addClass( 'cloned' ).appendTo( this.$stage ); $( prepend ).addClass( 'cloned' ).prependTo( this.$stage ); } }, { filter: [ 'width', 'items', 'settings' ], run: function() { var rtl = this.settings.rtl ? 1 : -1, size = this._clones.length + this._items.length, iterator = -1, previous = 0, current = 0, coordinates = []; while (++iterator < size) { previous = coordinates[iterator - 1] || 0; current = this._widths[this.relative( iterator )] + this.settings.margin; coordinates.push( previous + current * rtl ); } this._coordinates = coordinates; } }, { filter: [ 'width', 'items', 'settings' ], run: function() { var padding = this.settings.stagePadding, coordinates = this._coordinates, css = { 'width': Math.ceil( Math.abs( coordinates[coordinates.length - 1] ) ) + padding * 2, 'padding-left': padding || '', 'padding-right': padding || '' }; this.$stage.css( css ); } }, { filter: [ 'width', 'items', 'settings' ], run: function(cache) { var iterator = this._coordinates.length, grid = ! this.settings.autoWidth, items = this.$stage.children(); if (grid && cache.items.merge) { while (iterator--) { cache.css.width = this._widths[this.relative( iterator )]; items.eq( iterator ).css( cache.css ); } } else if (grid) { cache.css.width = cache.items.width; items.css( cache.css ); } } }, { filter: [ 'items' ], run: function() { this._coordinates.length < 1 && this.$stage.removeAttr( 'style' ); } }, { filter: [ 'width', 'items', 'settings' ], run: function(cache) { cache.current = cache.current ? this.$stage.children().index( cache.current ) : 0; cache.current = Math.max( this.minimum(), Math.min( this.maximum(), cache.current ) ); this.reset( cache.current ); } }, { filter: [ 'position' ], run: function() { this.animate( this.coordinates( this._current ) ); } }, { filter: [ 'width', 'position', 'items', 'settings' ], run: function() { var rtl = this.settings.rtl ? 1 : -1, padding = this.settings.stagePadding * 2, begin = this.coordinates( this.current() ) + padding, end = begin + this.width() * rtl, inner, outer, matches = [], i, n; for (i = 0, n = this._coordinates.length; i < n; i++) { inner = this._coordinates[i - 1] || 0; outer = Math.abs( this._coordinates[i] ) + padding * rtl; if ((this.op( inner, '<=', begin ) && (this.op( inner, '>', end ))) || (this.op( outer, '<', begin ) && this.op( outer, '>', end ))) { matches.push( i ); } } this.$stage.children( '.active' ).removeClass( 'active' ); this.$stage.children( ':eq(' + matches.join( '), :eq(' ) + ')' ).addClass( 'active' ); this.$stage.children( '.center' ).removeClass( 'center' ); if (this.settings.center) { this.$stage.children().eq( this.current() ).addClass( 'center' ); } } } ]; /** * Create the stage DOM element */ Owl.prototype.initializeStage = function() { this.$stage = this.$element.find( '.' + this.settings.stageClass ); // if the stage is already in the DOM, grab it and skip stage initialization. if (this.$stage.length) { return; } this.$element.addClass( this.options.loadingClass ); // create stage. this.$stage = $( '<' + this.settings.stageElement + '>', { "class": this.settings.stageClass } ).wrap( $( '
', { "class": this.settings.stageOuterClass } ) ); // append stage. this.$element.append( this.$stage.parent() ); }; /** * Create item DOM elements */ Owl.prototype.initializeItems = function() { var $items = this.$element.find( '.owl-item' ); // if the items are already in the DOM, grab them and skip item initialization. if ($items.length) { this._items = $items.get().map( function(item) { return $( item ); } ); this._mergers = this._items.map( function() { return 1; } ); this.refresh(); return; } // append content. this.replace( this.$element.children().not( this.$stage.parent() ) ); // check visibility. if (this.isVisible()) { // update view. this.refresh(); } else { // invalidate width. this.invalidate( 'width' ); } this.$element .removeClass( this.options.loadingClass ) .addClass( this.options.loadedClass ); }; /** * Initializes the carousel. * * @protected */ Owl.prototype.initialize = function() { this.enter( 'initializing' ); this.trigger( 'initialize' ); this.$element.toggleClass( this.settings.rtlClass, this.settings.rtl ); if (this.settings.autoWidth && ! this.is( 'pre-loading' )) { var imgs, nestedSelector, width; imgs = this.$element.find( 'img' ); nestedSelector = this.settings.nestedItemSelector ? '.' + this.settings.nestedItemSelector : undefined; width = this.$element.children( nestedSelector ).width(); if (imgs.length && width <= 0) { this.preloadAutoWidthImages( imgs ); } } this.initializeStage(); this.initializeItems(); // register event handlers. this.registerEventHandlers(); this.leave( 'initializing' ); this.trigger( 'initialized' ); }; /** * @returns {Boolean} visibility of $element * if you know the carousel will always be visible you can set `checkVisibility` to `false` to * prevent the expensive browser layout forced reflow the $element.is(':visible') does */ Owl.prototype.isVisible = function() { return this.settings.checkVisibility ? this.$element.is( ':visible' ) : true; }; /** * Setups the current settings. * * @todo Remove responsive classes. Why should adaptive designs be brought into IE8? * @todo Support for media queries by using `matchMedia` would be nice. * @public */ Owl.prototype.setup = function() { var viewport = this.viewport(), overwrites = this.options.responsive, match = -1, settings = null; if ( ! overwrites) { settings = $.extend( {}, this.options ); } else { $.each( overwrites, function(breakpoint) { if (breakpoint <= viewport && breakpoint > match) { match = Number( breakpoint ); } } ); settings = $.extend( {}, this.options, overwrites[match] ); if (typeof settings.stagePadding === 'function') { settings.stagePadding = settings.stagePadding(); } delete settings.responsive; // responsive class. if (settings.responsiveClass) { this.$element.attr( 'class', this.$element.attr( 'class' ).replace( new RegExp( '(' + this.options.responsiveClass + '-)\\S+\\s', 'g' ), '$1' + match ) ); } } this.trigger( 'change', { property: { name: 'settings', value: settings } } ); this._breakpoint = match; this.settings = settings; this.invalidate( 'settings' ); this.trigger( 'changed', { property: { name: 'settings', value: this.settings } } ); }; /** * Updates option logic if necessery. * * @protected */ Owl.prototype.optionsLogic = function() { if (this.settings.autoWidth) { this.settings.stagePadding = false; this.settings.merge = false; } }; /** * Prepares an item before add. * * @todo Rename event parameter `content` to `item`. * @protected * @returns {jQuery|HTMLElement} - The item container. */ Owl.prototype.prepare = function(item) { var event = this.trigger( 'prepare', { content: item } ); if ( ! event.data) { event.data = $( '<' + this.settings.itemElement + '/>' ) .addClass( this.options.itemClass ).append( item ) } this.trigger( 'prepared', { content: event.data } ); return event.data; }; /** * Updates the view. * * @public */ Owl.prototype.update = function() { var i = 0, n = this._pipe.length, filter = $.proxy( function(p) { return this[p] }, this._invalidated ), cache = {}; while (i < n) { if (this._invalidated.all || $.grep( this._pipe[i].filter, filter ).length > 0) { this._pipe[i].run( cache ); } i++; } this._invalidated = {}; ! this.is( 'valid' ) && this.enter( 'valid' ); }; /** * Gets the width of the view. * * @public * @param {Owl.Width} [dimension=Owl.Width.Default] - The dimension to return. * @returns {Number} - The width of the view in pixel. */ Owl.prototype.width = function(dimension) { dimension = dimension || Owl.Width.Default; switch (dimension) { case Owl.Width.Inner: case Owl.Width.Outer: return this._width; default: return this._width - this.settings.stagePadding * 2 + this.settings.margin; } }; /** * Refreshes the carousel primarily for adaptive purposes. * * @public */ Owl.prototype.refresh = function() { this.enter( 'refreshing' ); this.trigger( 'refresh' ); this.setup(); this.optionsLogic(); this.$element.addClass( this.options.refreshClass ); this.update(); this.$element.removeClass( this.options.refreshClass ); this.leave( 'refreshing' ); this.trigger( 'refreshed' ); }; /** * Checks window `resize` event. * * @protected */ Owl.prototype.onThrottledResize = function() { window.clearTimeout( this.resizeTimer ); this.resizeTimer = window.setTimeout( this._handlers.onResize, this.settings.responsiveRefreshRate ); }; /** * Checks window `resize` event. * * @protected */ Owl.prototype.onResize = function() { if ( ! this._items.length) { return false; } if (this._width === this.$element.width()) { return false; } if ( ! this.isVisible()) { return false; } this.enter( 'resizing' ); if (this.trigger( 'resize' ).isDefaultPrevented()) { this.leave( 'resizing' ); return false; } this.invalidate( 'width' ); this.refresh(); this.leave( 'resizing' ); this.trigger( 'resized' ); }; /** * Registers event handlers. * * @todo Check `msPointerEnabled` * @todo #261 * @protected */ Owl.prototype.registerEventHandlers = function() { if ($.support.transition) { this.$stage.on( $.support.transition.end + '.owl.core', $.proxy( this.onTransitionEnd, this ) ); } if (this.settings.responsive !== false) { this.on( window, 'resize', this._handlers.onThrottledResize ); } if (this.settings.mouseDrag) { this.$element.addClass( this.options.dragClass ); this.$stage.on( 'mousedown.owl.core', $.proxy( this.onDragStart, this ) ); this.$stage.on( 'dragstart.owl.core selectstart.owl.core', function() { return false } ); } if (this.settings.touchDrag) { this.$stage.on( 'touchstart.owl.core', $.proxy( this.onDragStart, this ) ); this.$stage.on( 'touchcancel.owl.core', $.proxy( this.onDragEnd, this ) ); } }; /** * Handles `touchstart` and `mousedown` events. * * @todo Horizontal swipe threshold as option * @todo #261 * @protected * @param {Event} event - The event arguments. */ Owl.prototype.onDragStart = function(event) { var stage = null; if (event.which === 3) { return; } if ($.support.transform) { stage = this.$stage.css( 'transform' ).replace( /.*\(|\)| /g, '' ).split( ',' ); stage = { x: stage[stage.length === 16 ? 12 : 4], y: stage[stage.length === 16 ? 13 : 5] }; } else { stage = this.$stage.position(); stage = { x: this.settings.rtl ? stage.left + this.$stage.width() - this.width() + this.settings.margin : stage.left, y: stage.top }; } if (this.is( 'animating' )) { $.support.transform ? this.animate( stage.x ) : this.$stage.stop() this.invalidate( 'position' ); } this.$element.toggleClass( this.options.grabClass, event.type === 'mousedown' ); this.speed( 0 ); this._drag.time = new Date().getTime(); this._drag.target = $( event.target ); this._drag.stage.start = stage; this._drag.stage.current = stage; this._drag.pointer = this.pointer( event ); $( document ).on( 'mouseup.owl.core touchend.owl.core', $.proxy( this.onDragEnd, this ) ); $( document ).one( 'mousemove.owl.core touchmove.owl.core', $.proxy( function(event) { var delta = this.difference( this._drag.pointer, this.pointer( event ) ); $( document ).on( 'mousemove.owl.core touchmove.owl.core', $.proxy( this.onDragMove, this ) ); if (Math.abs( delta.x ) < Math.abs( delta.y ) && this.is( 'valid' )) { return; } event.preventDefault(); this.enter( 'dragging' ); this.trigger( 'drag' ); }, this ) ); }; /** * Handles the `touchmove` and `mousemove` events. * * @todo #261 * @protected * @param {Event} event - The event arguments. */ Owl.prototype.onDragMove = function(event) { var minimum = null, maximum = null, pull = null, delta = this.difference( this._drag.pointer, this.pointer( event ) ), stage = this.difference( this._drag.stage.start, delta ); if ( ! this.is( 'dragging' )) { return; } event.preventDefault(); if (this.settings.loop) { minimum = this.coordinates( this.minimum() ); maximum = this.coordinates( this.maximum() + 1 ) - minimum; stage.x = (((stage.x - minimum) % maximum + maximum) % maximum) + minimum; } else { minimum = this.settings.rtl ? this.coordinates( this.maximum() ) : this.coordinates( this.minimum() ); maximum = this.settings.rtl ? this.coordinates( this.minimum() ) : this.coordinates( this.maximum() ); pull = this.settings.pullDrag ? -1 * delta.x / 5 : 0; stage.x = Math.max( Math.min( stage.x, minimum + pull ), maximum + pull ); } this._drag.stage.current = stage; this.animate( stage.x ); }; /** * Handles the `touchend` and `mouseup` events. * * @todo #261 * @todo Threshold for click event * @protected * @param {Event} event - The event arguments. */ Owl.prototype.onDragEnd = function(event) { var delta = this.difference( this._drag.pointer, this.pointer( event ) ), stage = this._drag.stage.current, direction = delta.x > 0 ^ this.settings.rtl ? 'left' : 'right'; $( document ).off( '.owl.core' ); this.$element.removeClass( this.options.grabClass ); if (delta.x !== 0 && this.is( 'dragging' ) || ! this.is( 'valid' )) { this.speed( this.settings.dragEndSpeed || this.settings.smartSpeed ); this.current( this.closest( stage.x, delta.x !== 0 ? direction : this._drag.direction ) ); this.invalidate( 'position' ); this.update(); this._drag.direction = direction; if (Math.abs( delta.x ) > 3 || new Date().getTime() - this._drag.time > 300) { this._drag.target.one( 'click.owl.core', function() { return false; } ); } } if ( ! this.is( 'dragging' )) { return; } this.leave( 'dragging' ); this.trigger( 'dragged' ); }; /** * Gets absolute position of the closest item for a coordinate. * * @todo Setting `freeDrag` makes `closest` not reusable. See #165. * @protected * @param {Number} coordinate - The coordinate in pixel. * @param {String} direction - The direction to check for the closest item. Ether `left` or `right`. * @return {Number} - The absolute position of the closest item. */ Owl.prototype.closest = function(coordinate, direction) { var position = -1, pull = 30, width = this.width(), coordinates = this.coordinates(); if ( ! this.settings.freeDrag) { // check closest item. $.each( coordinates, $.proxy( function(index, value) { // on a left pull, check on current index. if (direction === 'left' && coordinate > value - pull && coordinate < value + pull) { position = index; // on a right pull, check on previous index. // to do so, subtract width from value and set position = index + 1. } else if (direction === 'right' && coordinate > value - width - pull && coordinate < value - width + pull) { position = index + 1; } else if (this.op( coordinate, '<', value ) && this.op( coordinate, '>', coordinates[index + 1] !== undefined ? coordinates[index + 1] : value - width )) { position = direction === 'left' ? index + 1 : index; } return position === -1; }, this ) ); } if ( ! this.settings.loop) { // non loop boundries. if (this.op( coordinate, '>', coordinates[this.minimum()] )) { position = coordinate = this.minimum(); } else if (this.op( coordinate, '<', coordinates[this.maximum()] )) { position = coordinate = this.maximum(); } } return position; }; /** * Animates the stage. * * @todo #270 * @public * @param {Number} coordinate - The coordinate in pixels. */ Owl.prototype.animate = function(coordinate) { var animate = this.speed() > 0; this.is( 'animating' ) && this.onTransitionEnd(); if (animate) { this.enter( 'animating' ); this.trigger( 'translate' ); } if ($.support.transform3d && $.support.transition) { this.$stage.css( { transform: 'translate3d(' + coordinate + 'px,0px,0px)', transition: (this.speed() / 1000) + 's' + ( this.settings.slideTransition ? ' ' + this.settings.slideTransition : '' ) } ); } else if (animate) { this.$stage.animate( { left: coordinate + 'px' }, this.speed(), this.settings.fallbackEasing, $.proxy( this.onTransitionEnd, this ) ); } else { this.$stage.css( { left: coordinate + 'px' } ); } }; /** * Checks whether the carousel is in a specific state or not. * * @param {String} state - The state to check. * @returns {Boolean} - The flag which indicates if the carousel is busy. */ Owl.prototype.is = function(state) { return this._states.current[state] && this._states.current[state] > 0; }; /** * Sets the absolute position of the current item. * * @public * @param {Number} [position] - The new absolute position or nothing to leave it unchanged. * @returns {Number} - The absolute position of the current item. */ Owl.prototype.current = function(position) { if (position === undefined) { return this._current; } if (this._items.length === 0) { return undefined; } position = this.normalize( position ); if (this._current !== position) { var event = this.trigger( 'change', { property: { name: 'position', value: position } } ); if (event.data !== undefined) { position = this.normalize( event.data ); } this._current = position; this.invalidate( 'position' ); this.trigger( 'changed', { property: { name: 'position', value: this._current } } ); } return this._current; }; /** * Invalidates the given part of the update routine. * * @param {String} [part] - The part to invalidate. * @returns {Array.} - The invalidated parts. */ Owl.prototype.invalidate = function(part) { if ($.type( part ) === 'string') { this._invalidated[part] = true; this.is( 'valid' ) && this.leave( 'valid' ); } return $.map( this._invalidated, function(v, i) { return i } ); }; /** * Resets the absolute position of the current item. * * @public * @param {Number} position - The absolute position of the new item. */ Owl.prototype.reset = function(position) { position = this.normalize( position ); if (position === undefined) { return; } this._speed = 0; this._current = position; this.suppress( [ 'translate', 'translated' ] ); this.animate( this.coordinates( position ) ); this.release( [ 'translate', 'translated' ] ); }; /** * Normalizes an absolute or a relative position of an item. * * @public * @param {Number} position - The absolute or relative position to normalize. * @param {Boolean} [relative=false] - Whether the given position is relative or not. * @returns {Number} - The normalized position. */ Owl.prototype.normalize = function(position, relative) { var n = this._items.length, m = relative ? 0 : this._clones.length; if ( ! this.isNumeric( position ) || n < 1) { position = undefined; } else if (position < 0 || position >= n + m) { position = ((position - m / 2) % n + n) % n + m / 2; } return position; }; /** * Converts an absolute position of an item into a relative one. * * @public * @param {Number} position - The absolute position to convert. * @returns {Number} - The converted position. */ Owl.prototype.relative = function(position) { position -= this._clones.length / 2; return this.normalize( position, true ); }; /** * Gets the maximum position for the current item. * * @public * @param {Boolean} [relative=false] - Whether to return an absolute position or a relative position. * @returns {Number} */ Owl.prototype.maximum = function(relative) { var settings = this.settings, maximum = this._coordinates.length, iterator, reciprocalItemsWidth, elementWidth; if (settings.loop) { maximum = this._clones.length / 2 + this._items.length - 1; } else if (settings.autoWidth || settings.merge) { iterator = this._items.length; if (iterator) { reciprocalItemsWidth = this._items[--iterator].width(); elementWidth = this.$element.width(); while (iterator--) { reciprocalItemsWidth += this._items[iterator].width() + this.settings.margin; if (reciprocalItemsWidth > elementWidth) { break; } } } maximum = iterator + 1; } else if (settings.center) { maximum = this._items.length - 1; } else { maximum = this._items.length - settings.items; } if (relative) { maximum -= this._clones.length / 2; } return Math.max( maximum, 0 ); }; /** * Gets the minimum position for the current item. * * @public * @param {Boolean} [relative=false] - Whether to return an absolute position or a relative position. * @returns {Number} */ Owl.prototype.minimum = function(relative) { return relative ? 0 : this._clones.length / 2; }; /** * Gets an item at the specified relative position. * * @public * @param {Number} [position] - The relative position of the item. * @return {jQuery|Array.} - The item at the given position or all items if no position was given. */ Owl.prototype.items = function(position) { if (position === undefined) { return this._items.slice(); } position = this.normalize( position, true ); return this._items[position]; }; /** * Gets an item at the specified relative position. * * @public * @param {Number} [position] - The relative position of the item. * @return {jQuery|Array.} - The item at the given position or all items if no position was given. */ Owl.prototype.mergers = function(position) { if (position === undefined) { return this._mergers.slice(); } position = this.normalize( position, true ); return this._mergers[position]; }; /** * Gets the absolute positions of clones for an item. * * @public * @param {Number} [position] - The relative position of the item. * @returns {Array.} - The absolute positions of clones for the item or all if no position was given. */ Owl.prototype.clones = function(position) { var odd = this._clones.length / 2, even = odd + this._items.length, map = function(index) { return index % 2 === 0 ? even + index / 2 : odd - (index + 1) / 2 }; if (position === undefined) { return $.map( this._clones, function(v, i) { return map( i ) } ); } return $.map( this._clones, function(v, i) { return v === position ? map( i ) : null } ); }; /** * Sets the current animation speed. * * @public * @param {Number} [speed] - The animation speed in milliseconds or nothing to leave it unchanged. * @returns {Number} - The current animation speed in milliseconds. */ Owl.prototype.speed = function(speed) { if (speed !== undefined) { this._speed = speed; } return this._speed; }; /** * Gets the coordinate of an item. * * @todo The name of this method is missleanding. * @public * @param {Number} position - The absolute position of the item within `minimum()` and `maximum()`. * @returns {Number|Array.} - The coordinate of the item in pixel or all coordinates. */ Owl.prototype.coordinates = function(position) { var multiplier = 1, newPosition = position - 1, coordinate; if (position === undefined) { return $.map( this._coordinates, $.proxy( function(coordinate, index) { return this.coordinates( index ); }, this ) ); } if (this.settings.center) { if (this.settings.rtl) { multiplier = -1; newPosition = position + 1; } coordinate = this._coordinates[position]; coordinate += (this.width() - coordinate + (this._coordinates[newPosition] || 0)) / 2 * multiplier; } else { coordinate = this._coordinates[newPosition] || 0; } coordinate = Math.ceil( coordinate ); return coordinate; }; /** * Calculates the speed for a translation. * * @protected * @param {Number} from - The absolute position of the start item. * @param {Number} to - The absolute position of the target item. * @param {Number} [factor=undefined] - The time factor in milliseconds. * @returns {Number} - The time in milliseconds for the translation. */ Owl.prototype.duration = function(from, to, factor) { if (factor === 0) { return 0; } return Math.min( Math.max( Math.abs( to - from ), 1 ), 6 ) * Math.abs( (factor || this.settings.smartSpeed) ); }; /** * Slides to the specified item. * * @public * @param {Number} position - The position of the item. * @param {Number} [speed] - The time in milliseconds for the transition. */ Owl.prototype.to = function(position, speed) { var current = this.current(), revert = null, distance = position - this.relative( current ), direction = (distance > 0) - (distance < 0), items = this._items.length, minimum = this.minimum(), maximum = this.maximum(); if (this.settings.loop) { if ( ! this.settings.rewind && Math.abs( distance ) > items / 2) { distance += direction * -1 * items; } position = current + distance; revert = ((position - minimum) % items + items) % items + minimum; if (revert !== position && revert - distance <= maximum && revert - distance > 0) { current = revert - distance; position = revert; this.reset( current ); } } else if (this.settings.rewind) { maximum += 1; position = (position % maximum + maximum) % maximum; } else { position = Math.max( minimum, Math.min( maximum, position ) ); } this.speed( this.duration( current, position, speed ) ); this.current( position ); if (this.isVisible()) { this.update(); } }; /** * Slides to the next item. * * @public * @param {Number} [speed] - The time in milliseconds for the transition. */ Owl.prototype.next = function(speed) { speed = speed || false; this.to( this.relative( this.current() ) + 1, speed ); }; /** * Slides to the previous item. * * @public * @param {Number} [speed] - The time in milliseconds for the transition. */ Owl.prototype.prev = function(speed) { speed = speed || false; this.to( this.relative( this.current() ) - 1, speed ); }; /** * Handles the end of an animation. * * @protected * @param {Event} event - The event arguments. */ Owl.prototype.onTransitionEnd = function(event) { // if css2 animation then event object is undefined. if (event !== undefined) { event.stopPropagation(); // Catch only owl-stage transitionEnd event. if ((event.target || event.srcElement || event.originalTarget) !== this.$stage.get( 0 )) { return false; } } this.leave( 'animating' ); this.trigger( 'translated' ); }; /** * Gets viewport width. * * @protected * @return {Number} - The width in pixel. */ Owl.prototype.viewport = function() { var width; if (this.options.responsiveBaseElement !== window) { width = $( this.options.responsiveBaseElement ).width(); } else if (window.innerWidth) { width = window.innerWidth; } else if (document.documentElement && document.documentElement.clientWidth) { width = document.documentElement.clientWidth; } else { console.warn( 'Can not detect viewport width.' ); } return width; }; /** * Replaces the current content. * * @public * @param {HTMLElement|jQuery|String} content - The new content. */ Owl.prototype.replace = function(content) { this.$stage.empty(); this._items = []; if (content) { content = (content instanceof jQuery) ? content : $( content ); } if (this.settings.nestedItemSelector) { content = content.find( '.' + this.settings.nestedItemSelector ); } content.filter( function() { return this.nodeType === 1; } ).each( $.proxy( function(index, item) { item = this.prepare( item ); this.$stage.append( item ); this._items.push( item ); this._mergers.push( item.find( '[data-merge]' ).addBack( '[data-merge]' ).attr( 'data-merge' ) * 1 || 1 ); }, this ) ); this.reset( this.isNumeric( this.settings.startPosition ) ? this.settings.startPosition : 0 ); this.invalidate( 'items' ); }; /** * Adds an item. * * @todo Use `item` instead of `content` for the event arguments. * @public * @param {HTMLElement|jQuery|String} content - The item content to add. * @param {Number} [position] - The relative position at which to insert the item otherwise the item will be added to the end. */ Owl.prototype.add = function(content, position) { var current = this.relative( this._current ); position = position === undefined ? this._items.length : this.normalize( position, true ); content = content instanceof jQuery ? content : $( content ); this.trigger( 'add', { content: content, position: position } ); content = this.prepare( content ); if (this._items.length === 0 || position === this._items.length) { this._items.length === 0 && this.$stage.append( content ); this._items.length !== 0 && this._items[position - 1].after( content ); this._items.push( content ); this._mergers.push( content.find( '[data-merge]' ).addBack( '[data-merge]' ).attr( 'data-merge' ) * 1 || 1 ); } else { this._items[position].before( content ); this._items.splice( position, 0, content ); this._mergers.splice( position, 0, content.find( '[data-merge]' ).addBack( '[data-merge]' ).attr( 'data-merge' ) * 1 || 1 ); } this._items[current] && this.reset( this._items[current].index() ); this.invalidate( 'items' ); this.trigger( 'added', { content: content, position: position } ); }; /** * Removes an item by its position. * * @todo Use `item` instead of `content` for the event arguments. * @public * @param {Number} position - The relative position of the item to remove. */ Owl.prototype.remove = function(position) { position = this.normalize( position, true ); if (position === undefined) { return; } this.trigger( 'remove', { content: this._items[position], position: position } ); this._items[position].remove(); this._items.splice( position, 1 ); this._mergers.splice( position, 1 ); this.invalidate( 'items' ); this.trigger( 'removed', { content: null, position: position } ); }; /** * Preloads images with auto width. * * @todo Replace by a more generic approach * @protected */ Owl.prototype.preloadAutoWidthImages = function(images) { images.each( $.proxy( function(i, element) { this.enter( 'pre-loading' ); element = $( element ); $( new Image() ).one( 'load', $.proxy( function(e) { element.attr( 'src', e.target.src ); element.css( 'opacity', 1 ); this.leave( 'pre-loading' ); ! this.is( 'pre-loading' ) && ! this.is( 'initializing' ) && this.refresh(); }, this ) ).attr( 'src', element.attr( 'src' ) || element.attr( 'data-src' ) || element.attr( 'data-src-retina' ) ); }, this ) ); }; /** * Destroys the carousel. * * @public */ Owl.prototype.destroy = function() { this.$element.off( '.owl.core' ); this.$stage.off( '.owl.core' ); $( document ).off( '.owl.core' ); if (this.settings.responsive !== false) { window.clearTimeout( this.resizeTimer ); this.off( window, 'resize', this._handlers.onThrottledResize ); } for (var i in this._plugins) { this._plugins[i].destroy(); } this.$stage.children( '.cloned' ).remove(); this.$stage.unwrap(); this.$stage.children().contents().unwrap(); this.$stage.children().unwrap(); this.$stage.remove(); this.$element .removeClass( this.options.refreshClass ) .removeClass( this.options.loadingClass ) .removeClass( this.options.loadedClass ) .removeClass( this.options.rtlClass ) .removeClass( this.options.dragClass ) .removeClass( this.options.grabClass ) .attr( 'class', this.$element.attr( 'class' ).replace( new RegExp( this.options.responsiveClass + '-\\S+\\s', 'g' ), '' ) ) .removeData( 'owl.carousel' ); }; /** * Operators to calculate right-to-left and left-to-right. * * @protected * @param {Number} [a] - The left side operand. * @param {String} [o] - The operator. * @param {Number} [b] - The right side operand. */ Owl.prototype.op = function(a, o, b) { var rtl = this.settings.rtl; switch (o) { case '<': return rtl ? a > b : a < b; case '>': return rtl ? a < b : a > b; case '>=': return rtl ? a <= b : a >= b; case '<=': return rtl ? a >= b : a <= b; default: break; } }; /** * Attaches to an internal event. * * @protected * @param {HTMLElement} element - The event source. * @param {String} event - The event name. * @param {Function} listener - The event handler to attach. * @param {Boolean} capture - Wether the event should be handled at the capturing phase or not. */ Owl.prototype.on = function(element, event, listener, capture) { if (element.addEventListener) { element.addEventListener( event, listener, capture ); } else if (element.attachEvent) { element.attachEvent( 'on' + event, listener ); } }; /** * Detaches from an internal event. * * @protected * @param {HTMLElement} element - The event source. * @param {String} event - The event name. * @param {Function} listener - The attached event handler to detach. * @param {Boolean} capture - Wether the attached event handler was registered as a capturing listener or not. */ Owl.prototype.off = function(element, event, listener, capture) { if (element.removeEventListener) { element.removeEventListener( event, listener, capture ); } else if (element.detachEvent) { element.detachEvent( 'on' + event, listener ); } }; /** * Triggers a public event. * * @todo Remove `status`, `relatedTarget` should be used instead. * @protected * @param {String} name - The event name. * @param {*} [data=null] - The event data. * @param {String} [namespace=carousel] - The event namespace. * @param {String} [state] - The state which is associated with the event. * @param {Boolean} [enter=false] - Indicates if the call enters the specified state or not. * @returns {Event} - The event arguments. */ Owl.prototype.trigger = function(name, data, namespace, state, enter) { var status = { item: { count: this._items.length, index: this.current() } }, handler = $.camelCase( $.grep( [ 'on', name, namespace ], function(v) { return v } ) .join( '-' ).toLowerCase() ), event = $.Event( [ name, 'owl', namespace || 'carousel' ].join( '.' ).toLowerCase(), $.extend( { relatedTarget: this }, status, data ) ); if ( ! this._supress[name]) { $.each( this._plugins, function(name, plugin) { if (plugin.onTrigger) { plugin.onTrigger( event ); } } ); this.register( { type: Owl.Type.Event, name: name } ); this.$element.trigger( event ); if (this.settings && typeof this.settings[handler] === 'function') { this.settings[handler].call( this, event ); } } return event; }; /** * Enters a state. * * @param name - The state name. */ Owl.prototype.enter = function(name) { $.each( [ name ].concat( this._states.tags[name] || [] ), $.proxy( function(i, name) { if (this._states.current[name] === undefined) { this._states.current[name] = 0; } this._states.current[name]++; }, this ) ); }; /** * Leaves a state. * * @param name - The state name. */ Owl.prototype.leave = function(name) { $.each( [ name ].concat( this._states.tags[name] || [] ), $.proxy( function(i, name) { this._states.current[name]--; }, this ) ); }; /** * Registers an event or state. * * @public * @param {Object} object - The event or state to register. */ Owl.prototype.register = function(object) { if (object.type === Owl.Type.Event) { if ( ! $.event.special[object.name]) { $.event.special[object.name] = {}; } if ( ! $.event.special[object.name].owl) { var _default = $.event.special[object.name]._default; $.event.special[object.name]._default = function(e) { if (_default && _default.apply && ( ! e.namespace || e.namespace.indexOf( 'owl' ) === -1)) { return _default.apply( this, arguments ); } return e.namespace && e.namespace.indexOf( 'owl' ) > -1; }; $.event.special[object.name].owl = true; } } else if (object.type === Owl.Type.State) { if ( ! this._states.tags[object.name]) { this._states.tags[object.name] = object.tags; } else { this._states.tags[object.name] = this._states.tags[object.name].concat( object.tags ); } this._states.tags[object.name] = $.grep( this._states.tags[object.name], $.proxy( function(tag, i) { return $.inArray( tag, this._states.tags[object.name] ) === i; }, this ) ); } }; /** * Suppresses events. * * @protected * @param {Array.} events - The events to suppress. */ Owl.prototype.suppress = function(events) { $.each( events, $.proxy( function(index, event) { this._supress[event] = true; }, this ) ); }; /** * Releases suppressed events. * * @protected * @param {Array.} events - The events to release. */ Owl.prototype.release = function(events) { $.each( events, $.proxy( function(index, event) { delete this._supress[event]; }, this ) ); }; /** * Gets unified pointer coordinates from event. * * @todo #261 * @protected * @param {Event} - The `mousedown` or `touchstart` event. * @returns {Object} - Contains `x` and `y` coordinates of current pointer position. */ Owl.prototype.pointer = function(event) { var result = { x: null, y: null }; event = event.originalEvent || event || window.event; event = event.touches && event.touches.length ? event.touches[0] : event.changedTouches && event.changedTouches.length ? event.changedTouches[0] : event; if (event.pageX) { result.x = event.pageX; result.y = event.pageY; } else { result.x = event.clientX; result.y = event.clientY; } return result; }; /** * Determines if the input is a Number or something that can be coerced to a Number * * @protected * @param {Number|String|Object|Array|Boolean|RegExp|Function|Symbol} - The input to be tested * @returns {Boolean} - An indication if the input is a Number or can be coerced to a Number */ Owl.prototype.isNumeric = function(number) { return ! isNaN( parseFloat( number ) ); }; /** * Gets the difference of two vectors. * * @todo #261 * @protected * @param {Object} - The first vector. * @param {Object} - The second vector. * @returns {Object} - The difference. */ Owl.prototype.difference = function(first, second) { return { x: first.x - second.x, y: first.y - second.y }; }; /** * The jQuery Plugin for the Owl Carousel * * @todo Navigation plugin `next` and `prev` * @public */ $.fn.owlCarousel = function(option) { var args = Array.prototype.slice.call( arguments, 1 ); return this.each( function() { var $this = $( this ), data = $this.data( 'owl.carousel' ); if ( ! data) { data = new Owl( this, typeof option == 'object' && option ); $this.data( 'owl.carousel', data ); $.each( [ 'next', 'prev', 'to', 'destroy', 'refresh', 'replace', 'add', 'remove' ], function(i, event) { data.register( { type: Owl.Type.Event, name: event } ); data.$element.on( event + '.owl.carousel.core', $.proxy( function(e) { if (e.namespace && e.relatedTarget !== this) { this.suppress( [ event ] ); data[event].apply( this, [].slice.call( arguments, 1 ) ); this.release( [ event ] ); } }, data ) ); } ); } if (typeof option == 'string' && option.charAt( 0 ) !== '_') { data[option].apply( data, args ); } } ); }; /** * The constructor for the jQuery Plugin * * @public */ $.fn.owlCarousel.Constructor = Owl; })( window.Zepto || window.jQuery, window, document ); /** * AutoRefresh Plugin * * @version 2.3.4 * @author Artus Kolanowski * @author David Deutsch * @license The MIT License (MIT) */ ;(function($, window, document, undefined) { /** * Creates the auto refresh plugin. * * @class The Auto Refresh Plugin * @param {Owl} carousel - The Owl Carousel */ var AutoRefresh = function(carousel) { /** * Reference to the core. * * @protected * @type {Owl} */ this._core = carousel; /** * Refresh interval. * * @protected * @type {number} */ this._interval = null; /** * Whether the element is currently visible or not. * * @protected * @type {Boolean} */ this._visible = null; /** * All event handlers. * * @protected * @type {Object} */ this._handlers = { 'initialized.owl.carousel': $.proxy( function(e) { if (e.namespace && this._core.settings.autoRefresh) { this.watch(); } }, this ) }; // set default options. this._core.options = $.extend( {}, AutoRefresh.Defaults, this._core.options ); // register event handlers. this._core.$element.on( this._handlers ); }; /** * Default options. * * @public */ AutoRefresh.Defaults = { autoRefresh: true, autoRefreshInterval: 500 }; /** * Watches the element. */ AutoRefresh.prototype.watch = function() { if (this._interval) { return; } this._visible = this._core.isVisible(); this._interval = window.setInterval( $.proxy( this.refresh, this ), this._core.settings.autoRefreshInterval ); }; /** * Refreshes the element. */ AutoRefresh.prototype.refresh = function() { if (this._core.isVisible() === this._visible) { return; } this._visible = ! this._visible; this._core.$element.toggleClass( 'owl-hidden', ! this._visible ); this._visible && (this._core.invalidate( 'width' ) && this._core.refresh()); }; /** * Destroys the plugin. */ AutoRefresh.prototype.destroy = function() { var handler, property; window.clearInterval( this._interval ); for (handler in this._handlers) { this._core.$element.off( handler, this._handlers[handler] ); } for (property in Object.getOwnPropertyNames( this )) { typeof this[property] != 'function' && (this[property] = null); } }; $.fn.owlCarousel.Constructor.Plugins.AutoRefresh = AutoRefresh; })( window.Zepto || window.jQuery, window, document ); /** * Lazy Plugin * * @version 2.3.4 * @author Bartosz Wojciechowski * @author David Deutsch * @license The MIT License (MIT) */ ;(function($, window, document, undefined) { /** * Creates the lazy plugin. * * @class The Lazy Plugin * @param {Owl} carousel - The Owl Carousel */ var Lazy = function(carousel) { /** * Reference to the core. * * @protected * @type {Owl} */ this._core = carousel; /** * Already loaded items. * * @protected * @type {Array.} */ this._loaded = []; /** * Event handlers. * * @protected * @type {Object} */ this._handlers = { 'initialized.owl.carousel change.owl.carousel resized.owl.carousel': $.proxy( function(e) { if ( ! e.namespace) { return; } if ( ! this._core.settings || ! this._core.settings.lazyLoad) { return; } if ((e.property && e.property.name == 'position') || e.type == 'initialized') { var settings = this._core.settings, n = (settings.center && Math.ceil( settings.items / 2 ) || settings.items), i = ((settings.center && n * -1) || 0), position = (e.property && e.property.value !== undefined ? e.property.value : this._core.current()) + i, clones = this._core.clones().length, load = $.proxy( function(i, v) { this.load( v ) }, this ); // TODO: Need documentation for this new option. if (settings.lazyLoadEager > 0) { n += settings.lazyLoadEager; // If the carousel is looping also preload images that are to the "left". if (settings.loop) { position -= settings.lazyLoadEager; n++; } } while (i++ < n) { this.load( clones / 2 + this._core.relative( position ) ); clones && $.each( this._core.clones( this._core.relative( position ) ), load ); position++; } } }, this ) }; // set the default options. this._core.options = $.extend( {}, Lazy.Defaults, this._core.options ); // register event handler. this._core.$element.on( this._handlers ); }; /** * Default options. * * @public */ Lazy.Defaults = { lazyLoad: false, lazyLoadEager: 0 }; /** * Loads all resources of an item at the specified position. * * @param {Number} position - The absolute position of the item. * @protected */ Lazy.prototype.load = function(position) { var $item = this._core.$stage.children().eq( position ), $elements = $item && $item.find( '.owl-lazy' ); if ( ! $elements || $.inArray( $item.get( 0 ), this._loaded ) > -1) { return; } $elements.each( $.proxy( function(index, element) { var $element = $( element ), image, url = (window.devicePixelRatio > 1 && $element.attr( 'data-src-retina' )) || $element.attr( 'data-src' ) || $element.attr( 'data-srcset' ); this._core.trigger( 'load', { element: $element, url: url }, 'lazy' ); if ($element.is( 'img' )) { $element.one( 'load.owl.lazy', $.proxy( function() { $element.css( 'opacity', 1 ); this._core.trigger( 'loaded', { element: $element, url: url }, 'lazy' ); }, this ) ).attr( 'src', url ); } else if ($element.is( 'source' )) { $element.one( 'load.owl.lazy', $.proxy( function() { this._core.trigger( 'loaded', { element: $element, url: url }, 'lazy' ); }, this ) ).attr( 'srcset', url ); } else { image = new Image(); image.onload = $.proxy( function() { $element.css( { 'background-image': 'url("' + url + '")', 'opacity': '1' } ); this._core.trigger( 'loaded', { element: $element, url: url }, 'lazy' ); }, this ); image.src = url; } }, this ) ); this._loaded.push( $item.get( 0 ) ); }; /** * Destroys the plugin. * * @public */ Lazy.prototype.destroy = function() { var handler, property; for (handler in this.handlers) { this._core.$element.off( handler, this.handlers[handler] ); } for (property in Object.getOwnPropertyNames( this )) { typeof this[property] != 'function' && (this[property] = null); } }; $.fn.owlCarousel.Constructor.Plugins.Lazy = Lazy; })( window.Zepto || window.jQuery, window, document ); /** * AutoHeight Plugin * * @version 2.3.4 * @author Bartosz Wojciechowski * @author David Deutsch * @license The MIT License (MIT) */ ;(function($, window, document, undefined) { /** * Creates the auto height plugin. * * @class The Auto Height Plugin * @param {Owl} carousel - The Owl Carousel */ var AutoHeight = function(carousel) { /** * Reference to the core. * * @protected * @type {Owl} */ this._core = carousel; this._previousHeight = null; /** * All event handlers. * * @protected * @type {Object} */ this._handlers = { 'initialized.owl.carousel refreshed.owl.carousel': $.proxy( function(e) { if (e.namespace && this._core.settings.autoHeight) { this.update(); } }, this ), 'changed.owl.carousel': $.proxy( function(e) { if (e.namespace && this._core.settings.autoHeight && e.property.name === 'position') { this.update(); } }, this ), 'loaded.owl.lazy': $.proxy( function(e) { if (e.namespace && this._core.settings.autoHeight && e.element.closest( '.' + this._core.settings.itemClass ).index() === this._core.current()) { this.update(); } }, this ) }; // set default options. this._core.options = $.extend( {}, AutoHeight.Defaults, this._core.options ); // register event handlers. this._core.$element.on( this._handlers ); this._intervalId = null; var refThis = this; // These changes have been taken from a PR by gavrochelegnou proposed in #1575. // and have been made compatible with the latest jQuery version. $( window ).on( 'load', function() { if (refThis._core.settings.autoHeight) { refThis.update(); } } ); // Autoresize the height of the carousel when window is resized. // When carousel has images, the height is dependent on the width. // and should also change on resize. $( window ).resize( function() { if (refThis._core.settings.autoHeight) { if (refThis._intervalId != null) { clearTimeout( refThis._intervalId ); } refThis._intervalId = setTimeout( function() { refThis.update(); }, 250 ); } } ); }; /** * Default options. * * @public */ AutoHeight.Defaults = { autoHeight: false, autoHeightClass: 'owl-height' }; /** * Updates the view. */ AutoHeight.prototype.update = function() { var start = this._core._current, end = start + this._core.settings.items, lazyLoadEnabled = this._core.settings.lazyLoad, visible = this._core.$stage.children().toArray().slice( start, end ), heights = [], maxheight = 0; $.each( visible, function(index, item) { heights.push( $( item ).height() ); } ); maxheight = Math.max.apply( null, heights ); if (maxheight <= 1 && lazyLoadEnabled && this._previousHeight) { maxheight = this._previousHeight; } this._previousHeight = maxheight; this._core.$stage.parent() .height( maxheight ) .addClass( this._core.settings.autoHeightClass ); }; AutoHeight.prototype.destroy = function() { var handler, property; for (handler in this._handlers) { this._core.$element.off( handler, this._handlers[handler] ); } for (property in Object.getOwnPropertyNames( this )) { typeof this[property] !== 'function' && (this[property] = null); } }; $.fn.owlCarousel.Constructor.Plugins.AutoHeight = AutoHeight; })( window.Zepto || window.jQuery, window, document ); /** * Video Plugin * * @version 2.3.4 * @author Bartosz Wojciechowski * @author David Deutsch * @license The MIT License (MIT) */ ;(function($, window, document, undefined) { /** * Creates the video plugin. * * @class The Video Plugin * @param {Owl} carousel - The Owl Carousel */ var Video = function(carousel) { /** * Reference to the core. * * @protected * @type {Owl} */ this._core = carousel; /** * Cache all video URLs. * * @protected * @type {Object} */ this._videos = {}; /** * Current playing item. * * @protected * @type {jQuery} */ this._playing = null; /** * All event handlers. * * @todo The cloned content removale is too late * @protected * @type {Object} */ this._handlers = { 'initialized.owl.carousel': $.proxy( function(e) { if (e.namespace) { this._core.register( { type: 'state', name: 'playing', tags: [ 'interacting' ] } ); } }, this ), 'resize.owl.carousel': $.proxy( function(e) { if (e.namespace && this._core.settings.video && this.isInFullScreen()) { e.preventDefault(); } }, this ), 'refreshed.owl.carousel': $.proxy( function(e) { if (e.namespace && this._core.is( 'resizing' )) { this._core.$stage.find( '.cloned .owl-video-frame' ).remove(); } }, this ), 'changed.owl.carousel': $.proxy( function(e) { if (e.namespace && e.property.name === 'position' && this._playing) { this.stop(); } }, this ), 'prepared.owl.carousel': $.proxy( function(e) { if ( ! e.namespace) { return; } var $element = $( e.content ).find( '.owl-video' ); if ($element.length) { $element.css( 'display', 'none' ); this.fetch( $element, $( e.content ) ); } }, this ) }; // set default options. this._core.options = $.extend( {}, Video.Defaults, this._core.options ); // register event handlers. this._core.$element.on( this._handlers ); this._core.$element.on( 'click.owl.video', '.owl-video-play-icon', $.proxy( function(e) { this.play( e ); }, this ) ); }; /** * Default options. * * @public */ Video.Defaults = { video: false, videoHeight: false, videoWidth: false }; /** * Gets the video ID and the type (YouTube/Vimeo/vzaar only). * * @protected * @param {jQuery} target - The target containing the video data. * @param {jQuery} item - The item containing the video. */ Video.prototype.fetch = function(target, item) { var type = (function() { if (target.attr( 'data-vimeo-id' )) { return 'vimeo'; } else if (target.attr( 'data-vzaar-id' )) { return 'vzaar' } else { return 'youtube'; } })(), id = target.attr( 'data-vimeo-id' ) || target.attr( 'data-youtube-id' ) || target.attr( 'data-vzaar-id' ), width = target.attr( 'data-width' ) || this._core.settings.videoWidth, height = target.attr( 'data-height' ) || this._core.settings.videoHeight, url = target.attr( 'href' ); if (url) { /* Parses the id's out of the following urls (and probably more): https://www.youtube.com/watch?v=:id https://youtu.be/:id https://vimeo.com/:id https://vimeo.com/channels/:channel/:id https://vimeo.com/groups/:group/videos/:id https://app.vzaar.com/videos/:id Visual example: https://regexper.com/#(http%3A%7Chttps%3A%7C)%5C%2F%5C%2F(player.%7Cwww.%7Capp.)%3F(vimeo%5C.com%7Cyoutu(be%5C.com%7C%5C.be%7Cbe%5C.googleapis%5C.com)%7Cvzaar%5C.com)%5C%2F(video%5C%2F%7Cvideos%5C%2F%7Cembed%5C%2F%7Cchannels%5C%2F.%2B%5C%2F%7Cgroups%5C%2F.%2B%5C%2F%7Cwatch%5C%3Fv%3D%7Cv%5C%2F)%3F(%5BA-Za-z0-9._%25-%5D*)(%5C%26%5CS%2B)%3F */ id = url.match( /(http:|https:|)\/\/(player.|www.|app.)?(vimeo\.com|youtu(be\.com|\.be|be\.googleapis\.com|be\-nocookie\.com)|vzaar\.com)\/(video\/|videos\/|embed\/|channels\/.+\/|groups\/.+\/|watch\?v=|v\/)?([A-Za-z0-9._%-]*)(\&\S+)?/ ); if (id[3].indexOf( 'youtu' ) > -1) { type = 'youtube'; } else if (id[3].indexOf( 'vimeo' ) > -1) { type = 'vimeo'; } else if (id[3].indexOf( 'vzaar' ) > -1) { type = 'vzaar'; } else { throw new Error( 'Video URL not supported.' ); } id = id[6]; } else { throw new Error( 'Missing video URL.' ); } this._videos[url] = { type: type, id: id, width: width, height: height }; item.attr( 'data-video', url ); this.thumbnail( target, this._videos[url] ); }; /** * Creates video thumbnail. * * @protected * @param {jQuery} target - The target containing the video data. * @param {Object} info - The video info object. * @see `fetch` */ Video.prototype.thumbnail = function(target, video) { var tnLink, icon, path, dimensions = video.width && video.height ? 'width:' + video.width + 'px;height:' + video.height + 'px;' : '', customTn = target.find( 'img' ), srcType = 'src', lazyClass = '', settings = this._core.settings, create = function(path) { icon = '
'; if (settings.lazyLoad) { tnLink = $( '
', { "class": 'owl-video-tn ' + lazyClass, "srcType": path } ); } else { tnLink = $( '
', { "class": "owl-video-tn", "style": 'opacity:1;background-image:url(' + path + ')' } ); } target.after( tnLink ); target.after( icon ); }; // wrap video content into owl-video-wrapper div. target.wrap( $( '
', { "class": "owl-video-wrapper", "style": dimensions } ) ); if (this._core.settings.lazyLoad) { srcType = 'data-src'; lazyClass = 'owl-lazy'; } // custom thumbnail. if (customTn.length) { create( customTn.attr( srcType ) ); customTn.remove(); return false; } if (video.type === 'youtube') { path = "//img.youtube.com/vi/" + video.id + "/hqdefault.jpg"; create( path ); } else if (video.type === 'vimeo') { $.ajax( { type: 'GET', url: '//vimeo.com/api/v2/video/' + video.id + '.json', jsonp: 'callback', dataType: 'jsonp', success: function(data) { path = data[0].thumbnail_large; create( path ); } } ); } else if (video.type === 'vzaar') { $.ajax( { type: 'GET', url: '//vzaar.com/api/videos/' + video.id + '.json', jsonp: 'callback', dataType: 'jsonp', success: function(data) { path = data.framegrab_url; create( path ); } } ); } }; /** * Stops the current video. * * @public */ Video.prototype.stop = function() { this._core.trigger( 'stop', null, 'video' ); this._playing.find( '.owl-video-frame' ).remove(); this._playing.removeClass( 'owl-video-playing' ); this._playing = null; this._core.leave( 'playing' ); this._core.trigger( 'stopped', null, 'video' ); }; /** * Starts the current video. * * @public * @param {Event} event - The event arguments. */ Video.prototype.play = function(event) { var target = $( event.target ), item = target.closest( '.' + this._core.settings.itemClass ), video = this._videos[item.attr( 'data-video' )], width = video.width || '100%', height = video.height || this._core.$stage.height(), html, iframe; if (this._playing) { return; } this._core.enter( 'playing' ); this._core.trigger( 'play', null, 'video' ); item = this._core.items( this._core.relative( item.index() ) ); this._core.reset( item.index() ); html = $( '' ); html.attr( 'height', height ); html.attr( 'width', width ); if (video.type === 'youtube') { html.attr( 'src', '//www.youtube.com/embed/' + video.id + '?autoplay=1&rel=0&v=' + video.id ); } else if (video.type === 'vimeo') { html.attr( 'src', '//player.vimeo.com/video/' + video.id + '?autoplay=1' ); } else if (video.type === 'vzaar') { html.attr( 'src', '//view.vzaar.com/' + video.id + '/player?autoplay=true' ); } iframe = $( html ).wrap( '
' ).insertAfter( item.find( '.owl-video' ) ); this._playing = item.addClass( 'owl-video-playing' ); }; /** * Checks whether an video is currently in full screen mode or not. * * @todo Bad style because looks like a readonly method but changes members. * @protected * @returns {Boolean} */ Video.prototype.isInFullScreen = function() { var element = document.fullscreenElement || document.mozFullScreenElement || document.webkitFullscreenElement; return element && $( element ).parent().hasClass( 'owl-video-frame' ); }; /** * Destroys the plugin. */ Video.prototype.destroy = function() { var handler, property; this._core.$element.off( 'click.owl.video' ); for (handler in this._handlers) { this._core.$element.off( handler, this._handlers[handler] ); } for (property in Object.getOwnPropertyNames( this )) { typeof this[property] != 'function' && (this[property] = null); } }; $.fn.owlCarousel.Constructor.Plugins.Video = Video; })( window.Zepto || window.jQuery, window, document ); /** * Animate Plugin * * @version 2.3.4 * @author Bartosz Wojciechowski * @author David Deutsch * @license The MIT License (MIT) */ ;(function($, window, document, undefined) { /** * Creates the animate plugin. * * @class The Navigation Plugin * @param {Owl} scope - The Owl Carousel */ var Animate = function(scope) { this.core = scope; this.core.options = $.extend( {}, Animate.Defaults, this.core.options ); this.swapping = true; this.previous = undefined; this.next = undefined; this.handlers = { 'change.owl.carousel': $.proxy( function(e) { if (e.namespace && e.property.name == 'position') { this.previous = this.core.current(); this.next = e.property.value; } }, this ), 'drag.owl.carousel dragged.owl.carousel translated.owl.carousel': $.proxy( function(e) { if (e.namespace) { this.swapping = e.type == 'translated'; } }, this ), 'translate.owl.carousel': $.proxy( function(e) { if (e.namespace && this.swapping && (this.core.options.animateOut || this.core.options.animateIn)) { this.swap(); } }, this ) }; this.core.$element.on( this.handlers ); }; /** * Default options. * * @public */ Animate.Defaults = { animateOut: false, animateIn: false }; /** * Toggles the animation classes whenever an translations starts. * * @protected * @returns {Boolean|undefined} */ Animate.prototype.swap = function() { if (this.core.settings.items !== 1) { return; } if ( ! $.support.animation || ! $.support.transition) { return; } this.core.speed( 0 ); var left, clear = $.proxy( this.clear, this ), previous = this.core.$stage.children().eq( this.previous ), next = this.core.$stage.children().eq( this.next ), incoming = this.core.settings.animateIn, outgoing = this.core.settings.animateOut; if (this.core.current() === this.previous) { return; } if (outgoing) { left = this.core.coordinates( this.previous ) - this.core.coordinates( this.next ); previous.one( $.support.animation.end, clear ) .css( { 'left': left + 'px' } ) .addClass( 'animated owl-animated-out' ) .addClass( outgoing ); } if (incoming) { next.one( $.support.animation.end, clear ) .addClass( 'animated owl-animated-in' ) .addClass( incoming ); } }; Animate.prototype.clear = function(e) { $( e.target ).css( { 'left': '' } ) .removeClass( 'animated owl-animated-out owl-animated-in' ) .removeClass( this.core.settings.animateIn ) .removeClass( this.core.settings.animateOut ); this.core.onTransitionEnd(); }; /** * Destroys the plugin. * * @public */ Animate.prototype.destroy = function() { var handler, property; for (handler in this.handlers) { this.core.$element.off( handler, this.handlers[handler] ); } for (property in Object.getOwnPropertyNames( this )) { typeof this[property] != 'function' && (this[property] = null); } }; $.fn.owlCarousel.Constructor.Plugins.Animate = Animate; })( window.Zepto || window.jQuery, window, document ); /** * Autoplay Plugin * * @version 2.3.4 * @author Bartosz Wojciechowski * @author Artus Kolanowski * @author David Deutsch * @author Tom De Caluwé * @license The MIT License (MIT) */ ;(function($, window, document, undefined) { /** * Creates the autoplay plugin. * * @class The Autoplay Plugin * @param {Owl} scope - The Owl Carousel */ var Autoplay = function(carousel) { /** * Reference to the core. * * @protected * @type {Owl} */ this._core = carousel; /** * The autoplay timeout id. * * @type {Number} */ this._call = null; /** * Depending on the state of the plugin, this variable contains either * the start time of the timer or the current timer value if it's * paused. Since we start in a paused state we initialize the timer * value. * * @type {Number} */ this._time = 0; /** * Stores the timeout currently used. * * @type {Number} */ this._timeout = 0; /** * Indicates whenever the autoplay is paused. * * @type {Boolean} */ this._paused = true; /** * All event handlers. * * @protected * @type {Object} */ this._handlers = { 'changed.owl.carousel': $.proxy( function(e) { if (e.namespace && e.property.name === 'settings') { if (this._core.settings.autoplay) { this.play(); } else { this.stop(); } } else if (e.namespace && e.property.name === 'position' && this._paused) { // Reset the timer. This code is triggered when the position // of the carousel was changed through user interaction. this._time = 0; } }, this ), 'initialized.owl.carousel': $.proxy( function(e) { if (e.namespace && this._core.settings.autoplay) { this.play(); } }, this ), 'play.owl.autoplay': $.proxy( function(e, t, s) { if (e.namespace) { this.play( t, s ); } }, this ), 'stop.owl.autoplay': $.proxy( function(e) { if (e.namespace) { this.stop(); } }, this ), 'mouseover.owl.autoplay': $.proxy( function() { if (this._core.settings.autoplayHoverPause && this._core.is( 'rotating' )) { this.pause(); } }, this ), 'mouseleave.owl.autoplay': $.proxy( function() { if (this._core.settings.autoplayHoverPause && this._core.is( 'rotating' )) { this.play(); } }, this ), 'touchstart.owl.core': $.proxy( function() { if (this._core.settings.autoplayHoverPause && this._core.is( 'rotating' )) { this.pause(); } }, this ), 'touchend.owl.core': $.proxy( function() { if (this._core.settings.autoplayHoverPause) { this.play(); } }, this ) }; // register event handlers. this._core.$element.on( this._handlers ); // set default options. this._core.options = $.extend( {}, Autoplay.Defaults, this._core.options ); }; /** * Default options. * * @public */ Autoplay.Defaults = { autoplay: false, autoplayTimeout: 5000, autoplayHoverPause: false, autoplaySpeed: false }; /** * Transition to the next slide and set a timeout for the next transition. * * @private * @param {Number} [speed] - The animation speed for the animations. */ Autoplay.prototype._next = function(speed) { this._call = window.setTimeout( $.proxy( this._next, this, speed ), this._timeout * (Math.round( this.read() / this._timeout ) + 1) - this.read() ); if (this._core.is( 'interacting' ) || document.hidden) { return; } this._core.next( speed || this._core.settings.autoplaySpeed ); } /** * Reads the current timer value when the timer is playing. * * @public */ Autoplay.prototype.read = function() { return new Date().getTime() - this._time; }; /** * Starts the autoplay. * * @public * @param {Number} [timeout] - The interval before the next animation starts. * @param {Number} [speed] - The animation speed for the animations. */ Autoplay.prototype.play = function(timeout, speed) { var elapsed; if ( ! this._core.is( 'rotating' )) { this._core.enter( 'rotating' ); } timeout = timeout || this._core.settings.autoplayTimeout; // Calculate the elapsed time since the last transition. If the carousel. // wasn't playing this calculation will yield zero. elapsed = Math.min( this._time % (this._timeout || timeout), timeout ); if (this._paused) { // Start the clock. this._time = this.read(); this._paused = false; } else { // Clear the active timeout to allow replacement. window.clearTimeout( this._call ); } // Adjust the origin of the timer to match the new timeout value. this._time += this.read() % timeout - elapsed; this._timeout = timeout; this._call = window.setTimeout( $.proxy( this._next, this, speed ), timeout - elapsed ); }; /** * Stops the autoplay. * * @public */ Autoplay.prototype.stop = function() { if (this._core.is( 'rotating' )) { // Reset the clock. this._time = 0; this._paused = true; window.clearTimeout( this._call ); this._core.leave( 'rotating' ); } }; /** * Pauses the autoplay. * * @public */ Autoplay.prototype.pause = function() { if (this._core.is( 'rotating' ) && ! this._paused) { // Pause the clock. this._time = this.read(); this._paused = true; window.clearTimeout( this._call ); } }; /** * Destroys the plugin. */ Autoplay.prototype.destroy = function() { var handler, property; this.stop(); for (handler in this._handlers) { this._core.$element.off( handler, this._handlers[handler] ); } for (property in Object.getOwnPropertyNames( this )) { typeof this[property] != 'function' && (this[property] = null); } }; $.fn.owlCarousel.Constructor.Plugins.autoplay = Autoplay; })( window.Zepto || window.jQuery, window, document ); /** * Navigation Plugin * * @version 2.3.4 * @author Artus Kolanowski * @author David Deutsch * @license The MIT License (MIT) */ ;(function($, window, document, undefined) { 'use strict'; /** * Creates the navigation plugin. * * @class The Navigation Plugin * @param {Owl} carousel - The Owl Carousel. */ var Navigation = function(carousel) { /** * Reference to the core. * * @protected * @type {Owl} */ this._core = carousel; /** * Indicates whether the plugin is initialized or not. * * @protected * @type {Boolean} */ this._initialized = false; /** * The current paging indexes. * * @protected * @type {Array} */ this._pages = []; /** * All DOM elements of the user interface. * * @protected * @type {Object} */ this._controls = {}; /** * Markup for an indicator. * * @protected * @type {Array.} */ this._templates = []; /** * The carousel element. * * @type {jQuery} */ this.$element = this._core.$element; /** * Overridden methods of the carousel. * * @protected * @type {Object} */ this._overrides = { next: this._core.next, prev: this._core.prev, to: this._core.to }; /** * All event handlers. * * @protected * @type {Object} */ this._handlers = { 'prepared.owl.carousel': $.proxy( function(e) { if (e.namespace && this._core.settings.dotsData) { this._templates.push( '
' + $( e.content ).find( '[data-dot]' ).addBack( '[data-dot]' ).attr( 'data-dot' ) + '
' ); } }, this ), 'added.owl.carousel': $.proxy( function(e) { if (e.namespace && this._core.settings.dotsData) { this._templates.splice( e.position, 0, this._templates.pop() ); } }, this ), 'remove.owl.carousel': $.proxy( function(e) { if (e.namespace && this._core.settings.dotsData) { this._templates.splice( e.position, 1 ); } }, this ), 'changed.owl.carousel': $.proxy( function(e) { if (e.namespace && e.property.name == 'position') { this.draw(); } }, this ), 'initialized.owl.carousel': $.proxy( function(e) { if (e.namespace && ! this._initialized) { this._core.trigger( 'initialize', null, 'navigation' ); this.initialize(); this.update(); this.draw(); this._initialized = true; this._core.trigger( 'initialized', null, 'navigation' ); } }, this ), 'refreshed.owl.carousel': $.proxy( function(e) { if (e.namespace && this._initialized) { this._core.trigger( 'refresh', null, 'navigation' ); this.update(); this.draw(); this._core.trigger( 'refreshed', null, 'navigation' ); } }, this ) }; // set default options. this._core.options = $.extend( {}, Navigation.Defaults, this._core.options ); // register event handlers. this.$element.on( this._handlers ); }; /** * Default options. * * @public * @todo Rename `slideBy` to `navBy` */ Navigation.Defaults = { nav: false, navText: [ '', '' ], navSpeed: false, navElement: 'button type="button" role="presentation"', navContainer: false, navContainerClass: 'owl-nav', navClass: [ 'owl-prev', 'owl-next' ], slideBy: 1, dotClass: 'owl-dot', dotsClass: 'owl-dots', dots: true, dotsEach: false, dotsData: false, dotsSpeed: false, dotsContainer: false }; /** * Initializes the layout of the plugin and extends the carousel. * * @protected */ Navigation.prototype.initialize = function() { var override, settings = this._core.settings; // create DOM structure for relative navigation this._controls.$relative = (settings.navContainer ? $( settings.navContainer ) : $( '
' ).addClass( settings.navContainerClass ).appendTo( this.$element )).addClass( 'disabled' ); this._controls.$previous = $( '<' + settings.navElement + '>' ) .addClass( settings.navClass[0] ) .html( settings.navText[0] ) .prependTo( this._controls.$relative ) .on( 'click', $.proxy( function(e) { this.prev( settings.navSpeed ); }, this ) ); this._controls.$next = $( '<' + settings.navElement + '>' ) .addClass( settings.navClass[1] ) .html( settings.navText[1] ) .appendTo( this._controls.$relative ) .on( 'click', $.proxy( function(e) { this.next( settings.navSpeed ); }, this ) ); // create DOM structure for absolute navigation. if ( ! settings.dotsData) { this._templates = [ $( '