/*!
* justifiedGallery - v3.8.1
* http://miromannino.github.io/Justified-Gallery/
* Copyright (c) 2020 Miro Mannino
* Licensed under the MIT license.
*/
(function (factory) {
if (typeof define === "function" && define.amd) {
// AMD. Register as an anonymous module.
define(["jquery"], factory);
} else if (typeof module === "object" && module.exports) {
// Node/CommonJS
module.exports = function (root, jQuery) {
if (jQuery === undefined) {
// require('jQuery') returns a factory that requires window to
// build a jQuery instance, we normalize how we use modules
// that require this pattern but the window provided is a noop
// if it's defined (how jquery works)
if (typeof window !== "undefined") {
jQuery = require("jquery");
} else {
jQuery = require("jquery")(root);
}
}
factory(jQuery);
return jQuery;
};
} else {
// Browser globals
factory(jQuery);
}
})(function ($) {
/**
* Justified Gallery controller constructor
*
* @param $gallery the gallery to build
* @param settings the settings (the defaults are in JustifiedGallery.defaults)
* @constructor
*/
var JustifiedGallery = function ($gallery, settings) {
this.settings = settings;
this.checkSettings();
this.imgAnalyzerTimeout = null;
this.entries = null;
this.buildingRow = {
entriesBuff: [],
width: 0,
height: 0,
aspectRatio: 0
};
this.lastFetchedEntry = null;
this.lastAnalyzedIndex = -1;
this.yield = {
every: 2, // do a flush every n flushes (must be greater than 1)
flushed: 0 // flushed rows without a yield
};
this.border = settings.border >= 0 ? settings.border : settings.margins;
this.maxRowHeight = this.retrieveMaxRowHeight();
this.suffixRanges = this.retrieveSuffixRanges();
this.offY = this.border;
this.rows = 0;
this.spinner = {
phase: 0,
timeSlot: 150,
$el: $(
'<div class="jg-spinner"><span></span><span></span><span></span></div>'
),
intervalId: null
};
this.scrollBarOn = false;
this.checkWidthIntervalId = null;
this.galleryWidth = $gallery.width();
this.$gallery = $gallery;
};
/** @returns {String} the best suffix given the width and the height */
JustifiedGallery.prototype.getSuffix = function (width, height) {
var longestSide, i;
longestSide = width > height ? width : height;
for (i = 0; i < this.suffixRanges.length; i++) {
if (longestSide <= this.suffixRanges[i]) {
return this.settings.sizeRangeSuffixes[this.suffixRanges[i]];
}
}
return this.settings.sizeRangeSuffixes[this.suffixRanges[i - 1]];
};
/**
* Remove the suffix from the string
*
* @returns {string} a new string without the suffix
*/
JustifiedGallery.prototype.removeSuffix = function (str, suffix) {
return str.substring(0, str.length - suffix.length);
};
/**
* @returns {boolean} a boolean to say if the suffix is contained in the str or not
*/
JustifiedGallery.prototype.endsWith = function (str, suffix) {
return str.indexOf(suffix, str.length - suffix.length) !== -1;
};
/**
* Get the used suffix of a particular url
*
* @param str
* @returns {String} return the used suffix
*/
JustifiedGallery.prototype.getUsedSuffix = function (str) {
for (var si in this.settings.sizeRangeSuffixes) {
if (this.settings.sizeRangeSuffixes.hasOwnProperty(si)) {
if (this.settings.sizeRangeSuffixes[si].length === 0) continue;
if (this.endsWith(str, this.settings.sizeRangeSuffixes[si]))
return this.settings.sizeRangeSuffixes[si];
}
}
return "";
};
/**
* Given an image src, with the width and the height, returns the new image src with the
* best suffix to show the best quality thumbnail.
*
* @returns {String} the suffix to use
*/
JustifiedGallery.prototype.newSrc = function (
imageSrc,
imgWidth,
imgHeight,
image
) {
var newImageSrc;
if (this.settings.thumbnailPath) {
newImageSrc = this.settings.thumbnailPath(
imageSrc,
imgWidth,
imgHeight,
image
);
} else {
var matchRes = imageSrc.match(this.settings.extension);
var ext = matchRes !== null ? matchRes[0] : "";
newImageSrc = imageSrc.replace(this.settings.extension, "");
newImageSrc = this.removeSuffix(
newImageSrc,
this.getUsedSuffix(newImageSrc)
);
newImageSrc += this.getSuffix(imgWidth, imgHeight) + ext;
}
return newImageSrc;
};
/**
* Shows the images that is in the given entry
*
* @param $entry the entry
* @param callback the callback that is called when the show animation is finished
*/
JustifiedGallery.prototype.showImg = function ($entry, callback) {
if (this.settings.cssAnimation) {
$entry.addClass("jg-entry-visible");
if (callback) callback();
} else {
$entry
.stop()
.fadeTo(this.settings.imagesAnimationDuration, 1.0, callback);
$entry
.find(this.settings.imgSelector)
.stop()
.fadeTo(this.settings.imagesAnimationDuration, 1.0, callback);
}
};
/**
* Extract the image src form the image, looking from the 'safe-src', and if it can't be found, from the
* 'src' attribute. It saves in the image data the 'jg.originalSrc' field, with the extracted src.
*
* @param $image the image to analyze
* @returns {String} the extracted src
*/
JustifiedGallery.prototype.extractImgSrcFromImage = function ($image) {
var imageSrc = $image.data("safe-src");
var imageSrcLoc = "data-safe-src";
if (typeof imageSrc === "undefined") {
imageSrc = $image.attr("src");
imageSrcLoc = "src";
}
$image.data("jg.originalSrc", imageSrc); // this is saved for the destroy method
$image.data("jg.src", imageSrc); // this will change overtime
$image.data("jg.originalSrcLoc", imageSrcLoc); // this is saved for the destroy method
return imageSrc;
};
/** @returns {jQuery} the image in the given entry */
JustifiedGallery.prototype.imgFromEntry = function ($entry) {
var $img = $entry.find(this.settings.imgSelector);
return $img.length === 0 ? null : $img;
};
/** @returns {jQuery} the caption in the given entry */
JustifiedGallery.prototype.captionFromEntry = function ($entry) {
var $caption = $entry.find("> .jg-caption");
return $caption.length === 0 ? null : $caption;
};
/**
* Display the entry
*
* @param {jQuery} $entry the entry to display
* @param {int} x the x position where the entry must be positioned
* @param y the y position where the entry must be positioned
* @param imgWidth the image width
* @param imgHeight the image height
* @param rowHeight the row height of the row that owns the entry
*/
JustifiedGallery.prototype.displayEntry = function (
$entry,
x,
y,
imgWidth,
imgHeight,
rowHeight
) {
$entry.width(imgWidth);
$entry.height(rowHeight);
$entry.css("top", y);
$entry.css("left", x);
var $image = this.imgFromEntry($entry);
if ($image !== null) {
$image.css("width", imgWidth);
$image.css("height", imgHeight);
$image.css("margin-left", -imgWidth / 2);
$image.css("margin-top", -imgHeight / 2);
// Image reloading for an high quality of thumbnails
var imageSrc = $image.data("jg.src");
if (imageSrc) {
imageSrc = this.newSrc(imageSrc, imgWidth, imgHeight, $image[0]);
$image.one("error", function () {
this.resetImgSrc($image); //revert to the original thumbnail
});
var loadNewImage = function () {
// if (imageSrc !== newImageSrc) {
$image.attr("src", imageSrc);
// }
};
if ($entry.data("jg.loaded") === "skipped" && imageSrc) {
this.onImageEvent(
imageSrc,
function () {
this.showImg($entry, loadNewImage); //load the new image after the fadeIn
$entry.data("jg.loaded", true);
}.bind(this)
);
} else {
this.showImg($entry, loadNewImage); //load the new image after the fadeIn
}
}
} else {
this.showImg($entry);
}
this.displayEntryCaption($entry);
};
/**
* Display the entry caption. If the caption element doesn't exists, it creates the caption using the 'alt'
* or the 'title' attributes.
*
* @param {jQuery} $entry the entry to process
*/
JustifiedGallery.prototype.displayEntryCaption = function ($entry) {
var $image = this.imgFromEntry($entry);
if ($image !== null && this.settings.captions) {
var $imgCaption = this.captionFromEntry($entry);
// Create it if it doesn't exists
if ($imgCaption === null) {
var caption = $image.attr("alt");
if (!this.isValidCaption(caption)) caption = $entry.attr("title");
if (this.isValidCaption(caption)) {
// Create only we found something
$imgCaption = $('<div class="jg-caption">' + caption + "</div>");
$entry.append($imgCaption);
$entry.data("jg.createdCaption", true);
}
}
// Create events (we check again the $imgCaption because it can be still inexistent)
if ($imgCaption !== null) {
if (!this.settings.cssAnimation)
$imgCaption
.stop()
.fadeTo(0, this.settings.captionSettings.nonVisibleOpacity);
this.addCaptionEventsHandlers($entry);
}
} else {
this.removeCaptionEventsHandlers($entry);
}
};
/**
* Validates the caption
*
* @param caption The caption that should be validated
* @return {boolean} Validation result
*/
JustifiedGallery.prototype.isValidCaption = function (caption) {
return typeof caption !== "undefined" && caption.length > 0;
};
/**
* The callback for the event 'mouseenter'. It assumes that the event currentTarget is an entry.
* It shows the caption using jQuery (or using CSS if it is configured so)
*
* @param {Event} eventObject the event object
*/
JustifiedGallery.prototype.onEntryMouseEnterForCaption = function (
eventObject
) {
var $caption = this.captionFromEntry($(eventObject.currentTarget));
if (this.settings.cssAnimation) {
$caption.addClass("jg-caption-visible").removeClass("jg-caption-hidden");
} else {
$caption
.stop()
.fadeTo(
this.settings.captionSettings.animationDuration,
this.settings.captionSettings.visibleOpacity
);
}
};
/**
* The callback for the event 'mouseleave'. It assumes that the event currentTarget is an entry.
* It hides the caption using jQuery (or using CSS if it is configured so)
*
* @param {Event} eventObject the event object
*/
JustifiedGallery.prototype.onEntryMouseLeaveForCaption = function (
eventObject
) {
var $caption = this.captionFromEntry($(eventObject.currentTarget));
if (this.settings.cssAnimation) {
$caption
.removeClass("jg-caption-visible")
.removeClass("jg-caption-hidden");
} else {
$caption
.stop()
.fadeTo(
this.settings.captionSettings.animationDuration,
this.settings.captionSettings.nonVisibleOpacity
);
}
};
/**
* Add the handlers of the entry for the caption
*
* @param $entry the entry to modify
*/
JustifiedGallery.prototype.addCaptionEventsHandlers = function ($entry) {
var captionMouseEvents = $entry.data("jg.captionMouseEvents");
if (typeof captionMouseEvents === "undefined") {
captionMouseEvents = {
mouseenter: $.proxy(this.onEntryMouseEnterForCaption, this),
mouseleave: $.proxy(this.onEntryMouseLeaveForCaption, this)
};
$entry.on(
"mouseenter",
undefined,
undefined,
captionMouseEvents.mouseenter
);
$entry.on(
"mouseleave",
undefined,
undefined,
captionMouseEvents.mouseleave
);
$entry.data("jg.captionMouseEvents", captionMouseEvents);
}
};
/**
* Remove the handlers of the entry for the caption
*
* @param $entry the entry to modify
*/
JustifiedGallery.prototype.removeCaptionEventsHandlers = function ($entry) {
var captionMouseEvents = $entry.data("jg.captionMouseEvents");
if (typeof captionMouseEvents !== "undefined") {
$entry.off("mouseenter", undefined, captionMouseEvents.mouseenter);
$entry.off("mouseleave", undefined, captionMouseEvents.mouseleave);
$entry.removeData("jg.captionMouseEvents");
}
};
/**
* Clear the building row data to be used for a new row
*/
JustifiedGallery.prototype.clearBuildingRow = function () {
this.buildingRow.entriesBuff = [];
this.buildingRow.aspectRatio = 0;
this.buildingRow.width = 0;
};
/**
* Justify the building row, preparing it to
*
* @param isLastRow
* @param hiddenRow undefined or false for normal behavior. hiddenRow = true to hide the row.
* @returns a boolean to know if the row has been justified or not
*/
JustifiedGallery.prototype.prepareBuildingRow = function (
isLastRow,
hiddenRow
) {
var i,
$entry,
imgAspectRatio,
newImgW,
newImgH,
justify = true;
var minHeight = 0;
var availableWidth =
this.galleryWidth -
2 * this.border -
(this.buildingRow.entriesBuff.length - 1) * this.settings.margins;
var rowHeight = availableWidth / this.buildingRow.aspectRatio;
var defaultRowHeight = this.settings.rowHeight;
var justifiable =
this.buildingRow.width / availableWidth > this.settings.justifyThreshold;
//Skip the last row if we can't justify it and the lastRow == 'hide'
if (
hiddenRow ||
(isLastRow && this.settings.lastRow === "hide" && !justifiable)
) {
for (i = 0; i < this.buildingRow.entriesBuff.length; i++) {
$entry = this.buildingRow.entriesBuff[i];
if (this.settings.cssAnimation) $entry.removeClass("jg-entry-visible");
else {
$entry.stop().fadeTo(0, 0.1);
$entry.find("> img, > a > img").fadeTo(0, 0);
}
}
return -1;
}
// With lastRow = nojustify, justify if is justificable (the images will not become too big)
if (
isLastRow &&
!justifiable &&
this.settings.lastRow !== "justify" &&
this.settings.lastRow !== "hide"
) {
justify = false;
if (this.rows > 0) {
defaultRowHeight =
(this.offY - this.border - this.settings.margins * this.rows) /
this.rows;
justify =
(defaultRowHeight * this.buildingRow.aspectRatio) / availableWidth >
this.settings.justifyThreshold;
}
}
for (i = 0; i < this.buildingRow.entriesBuff.length; i++) {
$entry = this.buildingRow.entriesBuff[i];
imgAspectRatio = $entry.data("jg.width") / $entry.data("jg.height");
if (justify) {
newImgW =
i === this.buildingRow.entriesBuff.length - 1
? availableWidth
: rowHeight * imgAspectRatio;
newImgH = rowHeight;
} else {
newImgW = defaultRowHeight * imgAspectRatio;
newImgH = defaultRowHeight;
}
availableWidth -= Math.round(newImgW);
$entry.data("jg.jwidth", Math.round(newImgW));
$entry.data("jg.jheight", Math.ceil(newImgH));
if (i === 0 || minHeight > newImgH) minHeight = newImgH;
}
this.buildingRow.height = minHeight;
return justify;
};
/**
* Flush a row: justify it, modify the gallery height accordingly to the row height
*
* @param isLastRow
* @param hiddenRow undefined or false for normal behavior. hiddenRow = true to hide the row.
*/
JustifiedGallery.prototype.flushRow = function (isLastRow, hiddenRow) {
var settings = this.settings;
var $entry,
buildingRowRes,
offX = this.border,
i;
buildingRowRes = this.prepareBuildingRow(isLastRow, hiddenRow);
if (
hiddenRow ||
(isLastRow && settings.lastRow === "hide" && buildingRowRes === -1)
) {
this.clearBuildingRow();
return;
}
if (this.maxRowHeight) {
if (this.maxRowHeight < this.buildingRow.height)
this.buildingRow.height = this.maxRowHeight;
}
//Align last (unjustified) row
if (
isLastRow &&
(settings.lastRow === "center" || settings.lastRow === "right")
) {
var availableWidth =
this.galleryWidth -
2 * this.border -
(this.buildingRow.entriesBuff.length - 1) * settings.margins;
for (i = 0; i < this.buildingRow.entriesBuff.length; i++) {
$entry = this.buildingRow.entriesBuff[i];
availableWidth -= $entry.data("jg.jwidth");
}
if (settings.lastRow === "center") offX += Math.round(availableWidth / 2);
else if (settings.lastRow === "right") offX += availableWidth;
}
var lastEntryIdx = this.buildingRow.entriesBuff.length - 1;
for (i = 0; i <= lastEntryIdx; i++) {
$entry =
this.buildingRow.entriesBuff[this.settings.rtl ? lastEntryIdx - i : i];
this.displayEntry(
$entry,
offX,
this.offY,
$entry.data("jg.jwidth"),
$entry.data("jg.jheight"),
this.buildingRow.height
);
offX += $entry.data("jg.jwidth") + settings.margins;
}
//Gallery Height
this.galleryHeightToSet = this.offY + this.buildingRow.height + this.border;
this.setGalleryTempHeight(
this.galleryHeightToSet + this.getSpinnerHeight()
);
if (
!isLastRow ||
(this.buildingRow.height <= settings.rowHeight && buildingRowRes)
) {
//Ready for a new row
this.offY += this.buildingRow.height + settings.margins;
this.rows += 1;
this.clearBuildingRow();
this.settings.triggerEvent.call(this, "jg.rowflush");
}
};
// Scroll position not restoring: https://github.com/miromannino/Justified-Gallery/issues/221
var galleryPrevStaticHeight = 0;
JustifiedGallery.prototype.rememberGalleryHeight = function () {
galleryPrevStaticHeight = this.$gallery.height();
this.$gallery.height(galleryPrevStaticHeight);
};
// grow only
JustifiedGallery.prototype.setGalleryTempHeight = function (height) {
galleryPrevStaticHeight = Math.max(height, galleryPrevStaticHeight);
this.$gallery.height(galleryPrevStaticHeight);
};
JustifiedGallery.prototype.setGalleryFinalHeight = function (height) {
galleryPrevStaticHeight = height;
this.$gallery.height(height);
};
/**
* Checks the width of the gallery container, to know if a new justification is needed
*/
JustifiedGallery.prototype.checkWidth = function () {
this.checkWidthIntervalId = setInterval(
$.proxy(function () {
// if the gallery is not currently visible, abort.
if (!this.$gallery.is(":visible")) return;
var galleryWidth = parseFloat(this.$gallery.width());
if (
Math.abs(galleryWidth - this.galleryWidth) >
this.settings.refreshSensitivity
) {
this.galleryWidth = galleryWidth;
this.rewind();
this.rememberGalleryHeight();
// Restart to analyze
this.startImgAnalyzer(true);
}
}, this),
this.settings.refreshTime
);
};
/**
* @returns {boolean} a boolean saying if the spinner is active or not
*/
JustifiedGallery.prototype.isSpinnerActive = function () {
return this.spinner.intervalId !== null;
};
/**
* @returns {int} the spinner height
*/
JustifiedGallery.prototype.getSpinnerHeight = function () {
return this.spinner.$el.innerHeight();
};
/**
* Stops the spinner animation and modify the gallery height to exclude the spinner
*/
JustifiedGallery.prototype.stopLoadingSpinnerAnimation = function () {
clearInterval(this.spinner.intervalId);
this.spinner.intervalId = null;
this.setGalleryTempHeight(this.$gallery.height() - this.getSpinnerHeight());
this.spinner.$el.detach();
};
/**
* Starts the spinner animation
*/
JustifiedGallery.prototype.startLoadingSpinnerAnimation = function () {
var spinnerContext = this.spinner;
var $spinnerPoints = spinnerContext.$el.find("span");
clearInterval(spinnerContext.intervalId);
this.$gallery.append(spinnerContext.$el);
this.setGalleryTempHeight(
this.offY + this.buildingRow.height + this.getSpinnerHeight()
);
spinnerContext.intervalId = setInterval(function () {
if (spinnerContext.phase < $spinnerPoints.length) {
$spinnerPoints
.eq(spinnerContext.phase)
.fadeTo(spinnerContext.timeSlot, 1);
} else {
$spinnerPoints
.eq(spinnerContext.phase - $spinnerPoints.length)
.fadeTo(spinnerContext.timeSlot, 0);
}
spinnerContext.phase =
(spinnerContext.phase + 1) % ($spinnerPoints.length * 2);
}, spinnerContext.timeSlot);
};
/**
* Rewind the image analysis to start from the first entry.
*/
JustifiedGallery.prototype.rewind = function () {
this.lastFetchedEntry = null;
this.lastAnalyzedIndex = -1;
this.offY = this.border;
this.rows = 0;
this.clearBuildingRow();
};
/**
* @returns {String} `settings.selector` rejecting spinner element
*/
JustifiedGallery.prototype.getSelectorWithoutSpinner = function () {
return this.settings.selector + ", div:not(.jg-spinner)";
};
/**
* @returns {Array} all entries matched by `settings.selector`
*/
JustifiedGallery.prototype.getAllEntries = function () {
var selector = this.getSelectorWithoutSpinner();
return this.$gallery.children(selector).toArray();
};
/**
* Update the entries searching it from the justified gallery HTML element
*
* @param norewind if norewind only the new entries will be changed (i.e. randomized, sorted or filtered)
* @returns {boolean} true if some entries has been founded
*/
JustifiedGallery.prototype.updateEntries = function (norewind) {
var newEntries;
if (norewind && this.lastFetchedEntry != null) {
var selector = this.getSelectorWithoutSpinner();
newEntries = $(this.lastFetchedEntry).nextAll(selector).toArray();
} else {
this.entries = [];
newEntries = this.getAllEntries();
}
if (newEntries.length > 0) {
// Sort or randomize
if ($.isFunction(this.settings.sort)) {
newEntries = this.sortArray(newEntries);
} else if (this.settings.randomize) {
newEntries = this.shuffleArray(newEntries);
}
this.lastFetchedEntry = newEntries[newEntries.length - 1];
// Filter
if (this.settings.filter) {
newEntries = this.filterArray(newEntries);
} else {
this.resetFilters(newEntries);
}
}
this.entries = this.entries.concat(newEntries);
return true;
};
/**
* Apply the entries order to the DOM, iterating the entries and appending the images
*
* @param entries the entries that has been modified and that must be re-ordered in the DOM
*/
JustifiedGallery.prototype.insertToGallery = function (entries) {
var that = this;
$.each(entries, function () {
$(this).appendTo(that.$gallery);
});
};
/**
* Shuffle the array using the Fisher-Yates shuffle algorithm
*
* @param a the array to shuffle
* @return the shuffled array
*/
JustifiedGallery.prototype.shuffleArray = function (a) {
var i, j, temp;
for (i = a.length - 1; i > 0; i--) {
j = Math.floor(Math.random() * (i + 1));
temp = a[i];
a[i] = a[j];
a[j] = temp;
}
this.insertToGallery(a);
return a;
};
/**
* Sort the array using settings.comparator as comparator
*
* @param a the array to sort (it is sorted)
* @return the sorted array
*/
JustifiedGallery.prototype.sortArray = function (a) {
a.sort(this.settings.sort);
this.insertToGallery(a);
return a;
};
/**
* Reset the filters removing the 'jg-filtered' class from all the entries
*
* @param a the array to reset
*/
JustifiedGallery.prototype.resetFilters = function (a) {
for (var i = 0; i < a.length; i++) $(a[i]).removeClass("jg-filtered");
};
/**
* Filter the entries considering theirs classes (if a string has been passed) or using a function for filtering.
*
* @param a the array to filter
* @return the filtered array
*/
JustifiedGallery.prototype.filterArray = function (a) {
var settings = this.settings;
if ($.type(settings.filter) === "string") {
// Filter only keeping the entries passed in the string
return a.filter(function (el) {
var $el = $(el);
if ($el.is(settings.filter)) {
$el.removeClass("jg-filtered");
return true;
} else {
$el.addClass("jg-filtered").removeClass("jg-visible");
return false;
}
});
} else if ($.isFunction(settings.filter)) {
// Filter using the passed function
var filteredArr = a.filter(settings.filter);
for (var i = 0; i < a.length; i++) {
if (filteredArr.indexOf(a[i]) === -1) {
$(a[i]).addClass("jg-filtered").removeClass("jg-visible");
} else {
$(a[i]).removeClass("jg-filtered");
}
}
return filteredArr;
}
};
/**
* Revert the image src to the default value.
*/
JustifiedGallery.prototype.resetImgSrc = function ($img) {
if ($img.data("jg.originalSrcLoc") === "src") {
$img.attr("src", $img.data("jg.originalSrc"));
} else {
$img.attr("src", "");
}
};
/**
* Destroy the Justified Gallery instance.
*
* It clears all the css properties added in the style attributes. We doesn't backup the original
* values for those css attributes, because it costs (performance) and because in general one
* shouldn't use the style attribute for an uniform set of images (where we suppose the use of
* classes). Creating a backup is also difficult because JG could be called multiple times and
* with different style attributes.
*/
JustifiedGallery.prototype.destroy = function () {
clearInterval(this.checkWidthIntervalId);
this.stopImgAnalyzerStarter();
// Get fresh entries list since filtered entries are absent in `this.entries`
$.each(
this.getAllEntries(),
$.proxy(function (_, entry) {
var $entry = $(entry);
// Reset entry style
$entry.css("width", "");
$entry.css("height", "");
$entry.css("top", "");
$entry.css("left", "");
$entry.data("jg.loaded", undefined);
$entry.removeClass("jg-entry jg-filtered jg-entry-visible");
// Reset image style
var $img = this.imgFromEntry($entry);
if ($img) {
$img.css("width", "");
$img.css("height", "");
$img.css("margin-left", "");
$img.css("margin-top", "");
this.resetImgSrc($img);
$img.data("jg.originalSrc", undefined);
$img.data("jg.originalSrcLoc", undefined);
$img.data("jg.src", undefined);
}
// Remove caption
this.removeCaptionEventsHandlers($entry);
var $caption = this.captionFromEntry($entry);
if ($entry.data("jg.createdCaption")) {
// remove also the caption element (if created by jg)
$entry.data("jg.createdCaption", undefined);
if ($caption !== null) $caption.remove();
} else {
if ($caption !== null) $caption.fadeTo(0, 1);
}
}, this)
);
this.$gallery.css("height", "");
this.$gallery.removeClass("justified-gallery");
this.$gallery.data("jg.controller", undefined);
this.settings.triggerEvent.call(this, "jg.destroy");
};
/**
* Analyze the images and builds the rows. It returns if it found an image that is not loaded.
*
* @param isForResize if the image analyzer is called for resizing or not, to call a different callback at the end
*/
JustifiedGallery.prototype.analyzeImages = function (isForResize) {
for (var i = this.lastAnalyzedIndex + 1; i < this.entries.length; i++) {
var $entry = $(this.entries[i]);
if (
$entry.data("jg.loaded") === true ||
$entry.data("jg.loaded") === "skipped"
) {
var availableWidth =
this.galleryWidth -
2 * this.border -
(this.buildingRow.entriesBuff.length - 1) * this.settings.margins;
var imgAspectRatio = $entry.data("jg.width") / $entry.data("jg.height");
this.buildingRow.entriesBuff.push($entry);
this.buildingRow.aspectRatio += imgAspectRatio;
this.buildingRow.width += imgAspectRatio * this.settings.rowHeight;
this.lastAnalyzedIndex = i;
if (
availableWidth / (this.buildingRow.aspectRatio + imgAspectRatio) <
this.settings.rowHeight
) {
this.flushRow(
false,
this.settings.maxRowsCount > 0 &&
this.rows === this.settings.maxRowsCount
);
if (++this.yield.flushed >= this.yield.every) {
this.startImgAnalyzer(isForResize);
return;
}
}
} else if ($entry.data("jg.loaded") !== "error") {
return;
}
}
// Last row flush (the row is not full)
if (this.buildingRow.entriesBuff.length > 0) {
this.flushRow(
true,
this.settings.maxRowsCount > 0 &&
this.rows === this.settings.maxRowsCount
);
}
if (this.isSpinnerActive()) {
this.stopLoadingSpinnerAnimation();
}
/* Stop, if there is, the timeout to start the analyzeImages.
This is because an image can be set loaded, and the timeout can be set,
but this image can be analyzed yet.
*/
this.stopImgAnalyzerStarter();
this.setGalleryFinalHeight(this.galleryHeightToSet);
//On complete callback
this.settings.triggerEvent.call(
this,
isForResize ? "jg.resize" : "jg.complete"
);
};
/**
* Stops any ImgAnalyzer starter (that has an assigned timeout)
*/
JustifiedGallery.prototype.stopImgAnalyzerStarter = function () {
this.yield.flushed = 0;
if (this.imgAnalyzerTimeout !== null) {
clearTimeout(this.imgAnalyzerTimeout);
this.imgAnalyzerTimeout = null;
}
};
/**
* Starts the image analyzer. It is not immediately called to let the browser to update the view
*
* @param isForResize specifies if the image analyzer must be called for resizing or not
*/
JustifiedGallery.prototype.startImgAnalyzer = function (isForResize) {
var that = this;
this.stopImgAnalyzerStarter();
this.imgAnalyzerTimeout = setTimeout(function () {
that.analyzeImages(isForResize);
}, 0.001); // we can't start it immediately due to a IE different behaviour
};
/**
* Checks if the image is loaded or not using another image object. We cannot use the 'complete' image property,
* because some browsers, with a 404 set complete = true.
*
* @param imageSrc the image src to load
* @param onLoad callback that is called when the image has been loaded
* @param onError callback that is called in case of an error
*/
JustifiedGallery.prototype.onImageEvent = function (
imageSrc,
onLoad,
onError
) {
if (!onLoad && !onError) return;
var memImage = new Image();
var $memImage = $(memImage);
if (onLoad) {
$memImage.one("load", function () {
$memImage.off("load error");
onLoad(memImage);
});
}
if (onError) {
$memImage.one("error", function () {
$memImage.off("load error");
onError(memImage);
});
}
memImage.src = imageSrc;
};
/**
* Init of Justified Gallery controlled
* It analyzes all the entries starting theirs loading and calling the image analyzer (that works with loaded images)
*/
JustifiedGallery.prototype.init = function () {
var imagesToLoad = false,
skippedImages = false,
that = this;
$.each(this.entries, function (index, entry) {
var $entry = $(entry);
var $image = that.imgFromEntry($entry);
$entry.addClass("jg-entry");
if (
$entry.data("jg.loaded") !== true &&
$entry.data("jg.loaded") !== "skipped"
) {
// Link Rel global overwrite
if (that.settings.rel !== null) $entry.attr("rel", that.settings.rel);
// Link Target global overwrite
if (that.settings.target !== null)
$entry.attr("target", that.settings.target);
if ($image !== null) {
// Image src
var imageSrc = that.extractImgSrcFromImage($image);
/* If we have the height and the width, we don't wait that the image is loaded,
but we start directly with the justification */
if (that.settings.waitThumbnailsLoad === false || !imageSrc) {
var width = parseFloat($image.attr("width"));
var height = parseFloat($image.attr("height"));
if ($image.prop("tagName") === "svg") {
width = parseFloat($image[0].getBBox().width);
height = parseFloat($image[0].getBBox().height);
}
if (!isNaN(width) && !isNaN(height)) {
$entry.data("jg.width", width);
$entry.data("jg.height", height);
$entry.data("jg.loaded", "skipped");
skippedImages = true;
that.startImgAnalyzer(false);
return true; // continue
}
}
$entry.data("jg.loaded", false);
imagesToLoad = true;
// Spinner start
if (!that.isSpinnerActive()) that.startLoadingSpinnerAnimation();
that.onImageEvent(
imageSrc,
function (loadImg) {
// image loaded
$entry.data("jg.width", loadImg.width);
$entry.data("jg.height", loadImg.height);
$entry.data("jg.loaded", true);
that.startImgAnalyzer(false);
},
function () {
// image load error
$entry.data("jg.loaded", "error");
that.startImgAnalyzer(false);
}
);
} else {
$entry.data("jg.loaded", true);
$entry.data(
"jg.width",
$entry.width() | parseFloat($entry.css("width")) | 1
);
$entry.data(
"jg.height",
$entry.height() | parseFloat($entry.css("height")) | 1
);
}
}
});
if (!imagesToLoad && !skippedImages) this.startImgAnalyzer(false);
this.checkWidth();
};
/**
* Checks that it is a valid number. If a string is passed it is converted to a number
*
* @param settingContainer the object that contains the setting (to allow the conversion)
* @param settingName the setting name
*/
JustifiedGallery.prototype.checkOrConvertNumber = function (
settingContainer,
settingName
) {
if ($.type(settingContainer[settingName]) === "string") {
settingContainer[settingName] = parseFloat(settingContainer[settingName]);
}
if ($.type(settingContainer[settingName]) === "number") {
if (isNaN(settingContainer[settingName]))
throw "invalid number for " + settingName;
} else {
throw settingName + " must be a number";
}
};
/**
* Checks the sizeRangeSuffixes and, if necessary, converts
* its keys from string (e.g. old settings with 'lt100') to int.
*/
JustifiedGallery.prototype.checkSizeRangesSuffixes = function () {
if ($.type(this.settings.sizeRangeSuffixes) !== "object") {
throw "sizeRangeSuffixes must be defined and must be an object";
}
var suffixRanges = [];
for (var rangeIdx in this.settings.sizeRangeSuffixes) {
if (this.settings.sizeRangeSuffixes.hasOwnProperty(rangeIdx))
suffixRanges.push(rangeIdx);
}
var newSizeRngSuffixes = { 0: "" };
for (var i = 0; i < suffixRanges.length; i++) {
if ($.type(suffixRanges[i]) === "string") {
try {
var numIdx = parseInt(suffixRanges[i].replace(/^[a-z]+/, ""), 10);
newSizeRngSuffixes[numIdx] =
this.settings.sizeRangeSuffixes[suffixRanges[i]];
} catch (e) {
throw (
"sizeRangeSuffixes keys must contains correct numbers (" + e + ")"
);
}
} else {
newSizeRngSuffixes[suffixRanges[i]] =
this.settings.sizeRangeSuffixes[suffixRanges[i]];
}
}
this.settings.sizeRangeSuffixes = newSizeRngSuffixes;
};
/**
* check and convert the maxRowHeight setting
* requires rowHeight to be already set
* TODO: should be always called when only rowHeight is changed
* @return number or null
*/
JustifiedGallery.prototype.retrieveMaxRowHeight = function () {
var newMaxRowHeight = null;
var rowHeight = this.settings.rowHeight;
if ($.type(this.settings.maxRowHeight) === "string") {
if (this.settings.maxRowHeight.match(/^[0-9]+%$/)) {
newMaxRowHeight =
(rowHeight *
parseFloat(this.settings.maxRowHeight.match(/^([0-9]+)%$/)[1])) /
100;
} else {
newMaxRowHeight = parseFloat(this.settings.maxRowHeight);
}
} else if ($.type(this.settings.maxRowHeight) === "number") {
newMaxRowHeight = this.settings.maxRowHeight;
} else if (
this.settings.maxRowHeight === false ||
this.settings.maxRowHeight == null
) {
return null;
} else {
throw "maxRowHeight must be a number or a percentage";
}
// check if the converted value is not a number
if (isNaN(newMaxRowHeight)) throw "invalid number for maxRowHeight";
// check values, maxRowHeight must be >= rowHeight
if (newMaxRowHeight < rowHeight) newMaxRowHeight = rowHeight;
return newMaxRowHeight;
};
/**
* Checks the settings
*/
JustifiedGallery.prototype.checkSettings = function () {
this.checkSizeRangesSuffixes();
this.checkOrConvertNumber(this.settings, "rowHeight");
this.checkOrConvertNumber(this.settings, "margins");
this.checkOrConvertNumber(this.settings, "border");
this.checkOrConvertNumber(this.settings, "maxRowsCount");
var lastRowModes = [
"justify",
"nojustify",
"left",
"center",
"right",
"hide"
];
if (lastRowModes.indexOf(this.settings.lastRow) === -1) {
throw "lastRow must be one of: " + lastRowModes.join(", ");
}
this.checkOrConvertNumber(this.settings, "justifyThreshold");
if (
this.settings.justifyThreshold < 0 ||
this.settings.justifyThreshold > 1
) {
throw "justifyThreshold must be in the interval [0,1]";
}
if ($.type(this.settings.cssAnimation) !== "boolean") {
throw "cssAnimation must be a boolean";
}
if ($.type(this.settings.captions) !== "boolean")
throw "captions must be a boolean";
this.checkOrConvertNumber(
this.settings.captionSettings,
"animationDuration"
);
this.checkOrConvertNumber(this.settings.captionSettings, "visibleOpacity");
if (
this.settings.captionSettings.visibleOpacity < 0 ||
this.settings.captionSettings.visibleOpacity > 1
) {
throw "captionSettings.visibleOpacity must be in the interval [0, 1]";
}
this.checkOrConvertNumber(
this.settings.captionSettings,
"nonVisibleOpacity"
);
if (
this.settings.captionSettings.nonVisibleOpacity < 0 ||
this.settings.captionSettings.nonVisibleOpacity > 1
) {
throw "captionSettings.nonVisibleOpacity must be in the interval [0, 1]";
}
this.checkOrConvertNumber(this.settings, "imagesAnimationDuration");
this.checkOrConvertNumber(this.settings, "refreshTime");
this.checkOrConvertNumber(this.settings, "refreshSensitivity");
if ($.type(this.settings.randomize) !== "boolean")
throw "randomize must be a boolean";
if ($.type(this.settings.selector) !== "string")
throw "selector must be a string";
if (this.settings.sort !== false && !$.isFunction(this.settings.sort)) {
throw "sort must be false or a comparison function";
}
if (
this.settings.filter !== false &&
!$.isFunction(this.settings.filter) &&
$.type(this.settings.filter) !== "string"
) {
throw "filter must be false, a string or a filter function";
}
};
/**
* It brings all the indexes from the sizeRangeSuffixes and it orders them. They are then sorted and returned.
* @returns {Array} sorted suffix ranges
*/
JustifiedGallery.prototype.retrieveSuffixRanges = function () {
var suffixRanges = [];
for (var rangeIdx in this.settings.sizeRangeSuffixes) {
if (this.settings.sizeRangeSuffixes.hasOwnProperty(rangeIdx))
suffixRanges.push(parseInt(rangeIdx, 10));
}
suffixRanges.sort(function (a, b) {
return a > b ? 1 : a < b ? -1 : 0;
});
return suffixRanges;
};
/**
* Update the existing settings only changing some of them
*
* @param newSettings the new settings (or a subgroup of them)
*/
JustifiedGallery.prototype.updateSettings = function (newSettings) {
// In this case Justified Gallery has been called again changing only some options
this.settings = $.extend({}, this.settings, newSettings);
this.checkSettings();
// As reported in the settings: negative value = same as margins, 0 = disabled
this.border =
this.settings.border >= 0 ? this.settings.border : this.settings.margins;
this.maxRowHeight = this.retrieveMaxRowHeight();
this.suffixRanges = this.retrieveSuffixRanges();
};
JustifiedGallery.prototype.defaults = {
sizeRangeSuffixes: {} /* e.g. Flickr configuration
{
100: '_t', // used when longest is less than 100px
240: '_m', // used when longest is between 101px and 240px
320: '_n', // ...
500: '',
640: '_z',
1024: '_b' // used as else case because it is the last
}
*/,
thumbnailPath:
undefined /* If defined, sizeRangeSuffixes is not used, and this function is used to determine the
path relative to a specific thumbnail size. The function should accept respectively three arguments:
current path, width and height */,
rowHeight: 120, // required? required to be > 0?
maxRowHeight: false, // false or negative value to deactivate. Positive number to express the value in pixels,
// A string '[0-9]+%' to express in percentage (e.g. 300% means that the row height
// can't exceed 3 * rowHeight)
maxRowsCount: 0, // maximum number of rows to be displayed (0 = disabled)
margins: 1,
border: -1, // negative value = same as margins, 0 = disabled, any other value to set the border
lastRow: "nojustify", // ? which is the same as 'left', or can be 'justify', 'center', 'right' or 'hide'
justifyThreshold: 0.9,
/* if row width / available space > 0.90 it will be always justified
* (i.e. lastRow setting is not considered) */ waitThumbnailsLoad: true,
captions: true,
cssAnimation: true,
imagesAnimationDuration: 500, // ignored with css animations
captionSettings: {
// ignored with css animations
animationDuration: 500,
visibleOpacity: 0.7,
nonVisibleOpacity: 0.0
},
rel: null, // rewrite the rel of each analyzed links
target: null, // rewrite the target of all links
extension: /\.[^.\\/]+$/, // regexp to capture the extension of an image
refreshTime: 200, // time interval (in ms) to check if the page changes its width
refreshSensitivity: 0, // change in width allowed (in px) without re-building the gallery
randomize: false,
rtl: false, // right-to-left mode
sort: false /*
- false: to do not sort
- function: to sort them using the function as comparator (see Array.prototype.sort())
*/,
filter: false /*
- false, null or undefined: for a disabled filter
- a string: an entry is kept if entry.is(filter string) returns true
see jQuery's .is() function for further information
- a function: invoked with arguments (entry, index, array). Return true to keep the entry, false otherwise.
It follows the specifications of the Array.prototype.filter() function of JavaScript.
*/,
selector: "a", // The selector that is used to know what are the entries of the gallery
imgSelector: "> img, > a > img, > svg, > a > svg", // The selector that is used to know what are the images of each entry
triggerEvent: function (event) {
// This is called to trigger events, the default behavior is to call $.trigger
this.$gallery.trigger(event); // Consider that 'this' is this set to the JustifiedGallery object, so it can
} // access to fields such as $gallery, useful to trigger events with jQuery.
};
/**
* Justified Gallery plugin for jQuery
*
* Events
* - jg.complete : called when all the gallery has been created
* - jg.resize : called when the gallery has been resized
* - jg.rowflush : when a new row appears
*
* @param arg the action (or the settings) passed when the plugin is called
* @returns {*} the object itself
*/
$.fn.justifiedGallery = function (arg) {
return this.each(function (index, gallery) {
var $gallery = $(gallery);
$gallery.addClass("justified-gallery");
var controller = $gallery.data("jg.controller");
if (typeof controller === "undefined") {
// Create controller and assign it to the object data
if (
typeof arg !== "undefined" &&
arg !== null &&
$.type(arg) !== "object"
) {
if (arg === "destroy") return; // Just a call to an unexisting object
throw "The argument must be an object";
}
controller = new JustifiedGallery(
$gallery,
$.extend({}, JustifiedGallery.prototype.defaults, arg)
);
$gallery.data("jg.controller", controller);
} else if (arg === "norewind") {
// In this case we don't rewind: we analyze only the latest images (e.g. to complete the last unfinished row
// ... left to be more readable
} else if (arg === "destroy") {
controller.destroy();
return;
} else {
// In this case Justified Gallery has been called again changing only some options
controller.updateSettings(arg);
controller.rewind();
}
// Update the entries list
if (!controller.updateEntries(arg === "norewind")) return;
// Init justified gallery
controller.init();
});
};
});
|