File Manager
/**
* 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);
File Manager Version 1.0, Coded By Lucas
Email: hehe@yahoo.com