File: /home/rockyroadprintin/www/wp-content/plugins/imagify/assets/js/imagify-gulp.min.js
/**
 * Library that handles the bulk optimization processes.
 *
 * @requires jQuery
 */
window.imagify = window.imagify || {};
/* eslint-disable no-underscore-dangle, consistent-this */
(function($, d, w) {
	/**
	 * Construct the optimizer.
	 *
	 * @param {object} settings {
	 *     Optimizer settings:
	 *
	 *     @type {string} groupID         Group ID, like 'library' or 'custom-folders'.
	 *     @type {string} context         Context within this group, like 'wp' or 'custom-folders' (yes, again).
	 *     @type {int}    level           Optimization level: 0 to 2.
	 *     @type {int}    bufferSize      Number of parallel optimizations: usually 4.
	 *     @type {string} ajaxUrl         URL to request to optimize.
	 *     @type {object} files           Files to optimize: media ID as key (prefixed with an underscore), file URL as value.
	 *     @type {string} defaultThumb    A default thumbnail URL.
	 *     @type {string} doneEvent       Name of the event to listen to know when optimizations end.
	 *     @type {array}  imageExtensions A list of supported image extensions (only images).
	 * }
	 */
	w.imagify.Optimizer = function ( settings ) {
		// Settings.
		this.groupID      = settings.groupID;
		this.context      = settings.context;
		this.level        = settings.level;
		this.bufferSize   = settings.bufferSize || 1;
		this.ajaxUrl      = settings.ajaxUrl;
		this.files        = settings.files;
		this.defaultThumb = settings.defaultThumb;
		this.doneEvent    = settings.doneEvent;
		if ( settings.imageExtensions ) {
			this.imageExtensions = settings.imageExtensions;
		} else {
			this.imageExtensions = [ 'jpg', 'jpeg', 'jpe', 'png', 'gif' ];
		}
		/**
		 * An array of media IDs (prefixed with an underscore).
		 */
		this.prefixedMediaIDs = Object.keys( this.files );
		/**
		 * An array of medias currently being optimized: {
		 *     @type {int}    mediaID   The media ID.
		 *     @type {string} filename  The file name.
		 *     @type {string} thumbnail The file thumbnail URL.
		 * }
		 */
		this.currentItems = [];
		// Internal counters.
		this.totalMedia     = this.prefixedMediaIDs.length;
		this.processedMedia = 0;
		// Global stats.
		this.globalOriginalSize  = 0;
		this.globalOptimizedSize = 0;
		this.globalGain          = 0;
		this.globalPercent       = 0;
		// Callbacks.
		this._before = function () {};
		this._each   = function () {};
		this._done   = function () {};
		// Listen to the "optimization done" event.
		if ( this.totalMedia && this.doneEvent ) {
			$( w ).on( this.doneEvent, { _this: this }, this.processedCallback );
		}
	};
	/**
	 * Callback to trigger before each media optimization.
	 *
	 * @param  {callable} fnc A callback.
	 * @return this
	 */
	w.imagify.Optimizer.prototype.before = function( fnc ) {
		this._before = fnc;
		return this;
	};
	/**
	 * Callback to trigger after each media optimization.
	 *
	 * @param  {callable} fnc A callback.
	 * @return this
	 */
	w.imagify.Optimizer.prototype.each = function( fnc ) {
		this._each = fnc;
		return this;
	};
	/**
	 * Callback to trigger all media optimizations have been done.
	 *
	 * @param  {callable} fnc A callback.
	 * @return this
	 */
	w.imagify.Optimizer.prototype.done = function( fnc ) {
		this._done = fnc;
		return this;
	};
	/**
	 * Launch optimizations.
	 *
	 * @return this
	 */
	w.imagify.Optimizer.prototype.run = function() {
		var chunkLength = this.prefixedMediaIDs.length > this.bufferSize ? this.bufferSize : this.prefixedMediaIDs.length,
			i;
		for ( i = 0; i < chunkLength; i++ ) {
			this.processNext();
		}
		return this;
	};
	/**
	 * Launch next optimization.
	 *
	 * @return this
	 */
	w.imagify.Optimizer.prototype.processNext = function() {
		if ( this.prefixedMediaIDs.length ) {
			this.process( this.prefixedMediaIDs.shift() );
		}
		return this;
	};
	/**
	 * Launch an optimization.
	 *
	 * @param  {string} prefixedId A media ID, prefixed with an underscore.
	 * @return this
	 */
	w.imagify.Optimizer.prototype.process = function( prefixedId ) {
		var _this     = this,
			fileURL   = this.files[ prefixedId ],
			data      = {
				mediaID:    parseInt( prefixedId.toString().substr( 1 ), 10 ),
				filename:  this.files[ prefixedId ].split( '/' ).pop(),
				thumbnail: this.defaultThumb
			},
			extension = data.filename.split( '.' ).pop().toLowerCase(),
			regexp    = new RegExp( '^' + this.imageExtensions.join( '|' ).toLowerCase() + '$' ),
			image;
		if ( ! extension.match( regexp ) ) {
			// Not an image.
			this.currentItems.push( data );
			this._before( data );
			this.send( data );
			return this;
		}
		// Create a thumbnail and send the ajax request.
		image = new Image();
		image.onerror = function () {
			_this.currentItems.push( data );
			_this._before( data );
			_this.send( data );
		};
		image.onload = function () {
			var maxWidth    = 33,
				maxHeight   = 33,
				imageWidth  = image.width,
				imageHeight = image.height,
				newHeight   = 0,
				newWidth    = 0,
				topOffset   = 0,
				leftOffset  = 0,
				canvas      = null,
				ctx         = null;
			if ( imageWidth < imageHeight ) {
				// Portrait.
				newWidth  = maxWidth;
				newHeight = newWidth * imageHeight / imageWidth;
				topOffset = ( maxHeight - newHeight ) / 2;
			} else {
				// Landscape.
				newHeight  = maxHeight;
				newWidth   = newHeight * imageWidth / imageHeight;
				leftOffset = ( maxWidth - newWidth ) / 2;
			}
			canvas = d.createElement( 'canvas' );
			canvas.width  = maxWidth;
			canvas.height = maxHeight;
			ctx = canvas.getContext( '2d' );
			ctx.drawImage( this, leftOffset, topOffset, newWidth, newHeight );
			try {
				data.thumbnail = canvas.toDataURL( 'image/jpeg' );
			} catch ( e ) {
				data.thumbnail = _this.defaultThumb;
			}
			canvas = null;
			ctx    = null;
			image  = null;
			_this.currentItems.push( data );
			_this._before( data );
			_this.send( data );
		};
		image.src = fileURL;
		return this;
	};
	/**
	 * Do the ajax request.
	 *
	 * @param  {object} data {
	 *     The data:
	 *
	 *     @type {int}    mediaID   The media ID.
	 *     @type {string} filename  The file name.
	 *     @type {string} thumbnail The file thumbnail URL.
	 * }
	 * @return this
	 */
	w.imagify.Optimizer.prototype.send = function( data ) {
		var _this           = this,
			defaultResponse = {
				success:   false,
				mediaID:   data.mediaID,
				groupID:   this.groupID,
				context:   this.context,
				filename:  data.filename,
				thumbnail: data.thumbnail,
				status:    'error',
				error:     ''
			};
		$.post( {
			url:      this.ajaxUrl,
			data:     {
				media_id:           data.mediaID,
				context:            this.context,
				optimization_level: this.level
			},
			dataType: 'json'
		} )
			.done( function( response ) {
				if ( response.success ) {
					return;
				}
				defaultResponse.error = response.data.error;
				_this.processed( defaultResponse );
			} )
			.fail( function( jqXHR ) {
				defaultResponse.error = jqXHR.statusText;
				_this.processed( defaultResponse );
			} );
		return this;
	};
	/**
	 * Callback triggered when an optimization is complete.
	 *
	 * @param {object} e    jQuery's Event object.
	 * @param {object} item {
	 *     The response:
	 *
	 *     @type {int}    mediaID The media ID.
	 *     @type {string} context The context.
	 * }
	 */
	w.imagify.Optimizer.prototype.processedCallback = function( e, item ) {
		var _this = e.data._this;
		if ( item.context !== _this.context ) {
			return;
		}
		if ( ! item.mediaID || typeof _this.files[ '_' + item.mediaID ] === 'undefined' ) {
			return;
		}
		item.groupID = _this.groupID;
		if ( ! _this.currentItems.length ) {
			// Trouble.
			_this.processed( item );
			return;
		}
		$.each( _this.currentItems, function( i, mediaData ) {
			if ( item.mediaID === mediaData.mediaID ) {
				item.filename  = mediaData.filename;
				item.thumbnail = mediaData.thumbnail;
				return false;
			}
		} );
		_this.processed( item );
	};
	/**
	 * After a media has been processed.
	 *
	 * @param  {object} response {
	 *     The response:
	 *
	 *     @type {bool}   success   Whether the optimization succeeded or not ("already optimized" is a success).
	 *     @type {int}    mediaID   The media ID.
	 *     @type {string} groupID   The group ID.
	 *     @type {string} context   The context.
	 *     @type {string} filename  The file name.
	 *     @type {string} thumbnail The file thumbnail URL.
	 *     @type {string} status    The status, like 'optimized', 'already-optimized', 'over-quota', 'error'.
	 *     @type {string} error     The error message.
	 * }
	 * @return this
	 */
	w.imagify.Optimizer.prototype.processed = function( response ) {
		var currentItems = this.currentItems;
		if ( currentItems.length ) {
			// Remove this media from the "current" list.
			$.each( currentItems, function( i, mediaData ) {
				if ( response.mediaID === mediaData.mediaID ) {
					currentItems.splice( i, 1 );
					return false;
				}
			} );
			this.currentItems = currentItems;
		}
		// Update stats.
		if ( response.success && 'already-optimized' !== response.status ) {
			this.globalOriginalSize  += response.originalOverallSize;
			this.globalOptimizedSize += response.newOverallSize;
			this.globalGain          += response.overallSaving;
			this.globalPercent        = ( 100 - this.globalOptimizedSize / this.globalOptimizedSize * 100 ).toFixed( 2 );
		}
		++this.processedMedia;
		response.progress = Math.floor( this.processedMedia / this.totalMedia * 100 );
		this._each( response );
		if ( this.prefixedMediaIDs.length ) {
			this.processNext();
		} else if ( this.totalMedia === this.processedMedia ) {
			this._done( {
				globalOriginalSize:  this.globalOriginalSize,
				globalOptimizedSize: this.globalOptimizedSize,
				globalGain:          this.globalGain
			} );
		}
		return this;
	};
	/**
	 * Stop the process.
	 *
	 * @return this
	 */
	w.imagify.Optimizer.prototype.stopProcess = function() {
		this.files            = {};
		this.prefixedMediaIDs = [];
		this.currentItems     = [];
		if ( this.doneEvent ) {
			$( w ).off( this.doneEvent, this.processedCallback );
		}
		return this;
	};
} )(jQuery, document, window);