/*! * Ladda 0.8.0 * http://lab.hakim.se/ladda * MIT licensed * * Copyright (C) 2013 Hakim El Hattab, http://hakim.se */ (function( root, factory ) { // CommonJS if( typeof exports === 'object' ) { module.exports = factory(); } // AMD module else if( typeof define === 'function' && define.amd ) { define( [ 'spin' ], factory ); } // Browser global else { root.Ladda = factory( root.Spinner ); } } (this, function( Spinner ) { 'use strict'; // All currently instantiated instances of Ladda var ALL_INSTANCES = []; /** * Creates a new instance of Ladda which wraps the * target button element. * * @return An API object that can be used to control * the loading animation state. */ function create( button ) { if( typeof button === 'undefined' ) { console.warn( "Ladda button target must be defined." ); return; } // The text contents must be wrapped in a ladda-label // element, create one if it doesn't already exist if( !button.querySelector( '.ladda-label' ) ) { button.innerHTML = ''+ button.innerHTML +''; } // Create the spinner var spinner = createSpinner( button ); // Wrapper element for the spinner var spinnerWrapper = document.createElement( 'span' ); spinnerWrapper.className = 'ladda-spinner'; button.appendChild( spinnerWrapper ); // Timer used to delay starting/stopping var timer; var instance = { /** * Enter the loading state. */ start: function() { button.setAttribute( 'disabled', '' ); button.setAttribute( 'data-loading', '' ); clearTimeout( timer ); spinner.spin( spinnerWrapper ); this.setProgress( 0 ); return this; // chain }, /** * Enter the loading state, after a delay. */ startAfter: function( delay ) { clearTimeout( timer ); timer = setTimeout( function() { instance.start(); }, delay ); return this; // chain }, /** * Exit the loading state. */ stop: function() { button.removeAttribute( 'disabled' ); button.removeAttribute( 'data-loading' ); // Kill the animation after a delay to make sure it // runs for the duration of the button transition clearTimeout( timer ); timer = setTimeout( function() { spinner.stop(); }, 1000 ); return this; // chain }, /** * Toggle the loading state on/off. */ toggle: function() { if( this.isLoading() ) { this.stop(); } else { this.start(); } return this; // chain }, /** * Sets the width of the visual progress bar inside of * this Ladda button * * @param {Number} progress in the range of 0-1 */ setProgress: function( progress ) { // Cap it progress = Math.max( Math.min( progress, 1 ), 0 ); var progressElement = button.querySelector( '.ladda-progress' ); // Remove the progress bar if we're at 0 progress if( progress === 0 && progressElement && progressElement.parentNode ) { progressElement.parentNode.removeChild( progressElement ); } else { if( !progressElement ) { progressElement = document.createElement( 'div' ); progressElement.className = 'ladda-progress'; button.appendChild( progressElement ); } progressElement.style.width = ( ( progress || 0 ) * button.offsetWidth ) + 'px'; } }, enable: function() { this.stop(); return this; // chain }, disable: function () { this.stop(); button.setAttribute( 'disabled', '' ); return this; // chain }, isLoading: function() { return button.hasAttribute( 'data-loading' ); } }; ALL_INSTANCES.push( instance ); return instance; } /** * Binds the target buttons to automatically enter the * loading state when clicked. * * @param target Either an HTML element or a CSS selector. * @param options * - timeout Number of milliseconds to wait before * automatically cancelling the animation. */ function bind( target, options ) { options = options || {}; var targets = []; if( typeof target === 'string' ) { targets = toArray( document.querySelectorAll( target ) ); } else if( typeof target === 'object' && typeof target.nodeName === 'string' ) { targets = [ target ]; } for( var i = 0, len = targets.length; i < len; i++ ) { (function() { var element = targets[i]; // Make sure we're working with a DOM element if( typeof element.addEventListener === 'function' ) { var instance = create( element ); var timeout = -1; element.addEventListener( 'click', function() { // This is asynchronous to avoid an issue where setting // the disabled attribute on the button prevents forms // from submitting instance.startAfter( 1 ); // Set a loading timeout if one is specified if( typeof options.timeout === 'number' ) { clearTimeout( timeout ); timeout = setTimeout( instance.stop, options.timeout ); } // Invoke callbacks if( typeof options.callback === 'function' ) { options.callback.apply( null, [ instance ] ); } }, false ); } })(); } } /** * Stops ALL current loading animations. */ function stopAll() { for( var i = 0, len = ALL_INSTANCES.length; i < len; i++ ) { ALL_INSTANCES[i].stop(); } } function createSpinner( button ) { var height = button.offsetHeight, spinnerColor; // If the button is tall we can afford some padding if( height > 32 ) { height *= 0.8; } // Prefer an explicit height if one is defined if( button.hasAttribute( 'data-spinner-size' ) ) { height = parseInt( button.getAttribute( 'data-spinner-size' ), 10 ); } // Allow buttons to specify the color of the spinner element if (button.hasAttribute('data-spinner-color' ) ) { spinnerColor = button.getAttribute( 'data-spinner-color' ); } var lines = 12, radius = height * 0.2, length = radius * 0.6, width = radius < 7 ? 2 : 3; return new Spinner( { color: spinnerColor || '#fff', lines: lines, radius: radius, length: length, width: width, zIndex: 'auto', top: 'auto', left: 'auto', className: '' } ); } function toArray( nodes ) { var a = []; for ( var i = 0; i < nodes.length; i++ ) { a.push( nodes[ i ] ); } return a; } // Public API return { bind: bind, create: create, stopAll: stopAll }; }));