"use strict"; /* * jQuery plugin to show images and pdf files on preview * https://github.com/guillermodiazga/EZView * * Copyright 2017, Guillermo Diaz * * Licensed under the MIT license: * https://opensource.org/licenses/MIT */ /** * Check if jQuery is loaded */ if (!window.jQuery) { throw Error('jQuery is required'); } /** * Plugin Definition * @param {string} collectionName [This galery name in use to create a set of images and navigate through them] * * Viewer will create inside the body a container by each collection of images * <.EZView #EZView[auto-numeric-collection]> * <.image-collection> * <img [index-content = EZView[auto-numeric-collection][auto-numeric-img]]> * </.image-collection> * </.EZView> * * e.g. * * <body> * <.EZView #EZView0> * <.image-collection> * <img [index-content = EZView00]> * <img [index-content = EZView01]> * <img [index-content = EZView02]> * </.image-collection> * </.EZView> * <.EZView #EZView1> * <.image-collection> * <img [index-content = EZView10]> * <img [index-content = EZView11]> * </.image-collection> * </.EZView> * </body> * */ $.fn.EZView = function(collectionName) { var self = this; if (!$(self).length) { throw Error('No jQuery elements found'); } /** * Initial actions * @return void */ self.constructor = function() { var $thumbnails = $(this); self.$EZView = null; self.arContent = []; self.index = []; self.icons = { 'next': '', 'back': '', 'download': '', 'zoomIn': '', 'zoomOut': '', 'close': '', }; // If we have a collection set it otherwise set an default name collectionName = collectionName || 'EZView' + (parseInt($('.EZView').length)); // Create container by collection self.createContainer(); /** * index is the current image showing by collection * @type {Array} */ self.index[collectionName] = 0; /** * arContent allow to have all the images information to be show by collection * @type {Array} */ self.arContent[collectionName] = []; // Iterate each element and set the click event $thumbnails.each(function(i, thumbnail) { let $thumbnail = $(thumbnail); // Check if the element exists if (!$thumbnail.length) { return; } try { // Set all the data requiered to navigate through elements self.arContent[collectionName][i] = []; self.arContent[collectionName][i].href = $thumbnail.attr('href') || $thumbnail.attr('src'); self.arContent[collectionName][i].name = $thumbnail.attr('title') ? $thumbnail.attr('title').substring(0, 30) : ''; self.arContent[collectionName][i].isRender = false; self.arContent[collectionName][i].isImg = true; } catch (e) { console.error(e); console.error(el); } // Set $thumbnail properties $thumbnail // Set index as data property on each $thumbnail element .data('index', i) // Add cursor pointer .css({ cursor: 'pointer' }) // Add events to each element .off().click(function(e) { e.preventDefault(); e.stopPropagation(); self.runEZView(e); }); }); } /** * Methods Definition: */ /** * Return image to original measures * @param {int} index [description] * @return void */ self.returnImageToOriginalMeasures = function(index) { if (!self.isImg(index)) { return; } var $img = $('[index-content=' + collectionName + index + ']', self.$EZView), hasOriginalSizes = self.arContent[collectionName][index]['original-height']; if (hasOriginalSizes) { $img.animate({ 'height': self.arContent[collectionName][index]['original-height'] + 'px', 'width': self.arContent[collectionName][index]['original-width'] + 'px' }, 200); // Center img over container $img.css({ top: 'auto', left: 'auto' }); } }; /** * Create the collection container * @return void */ self.createContainer = function() { if (!$('#EZView' + collectionName).length) { var tools = '<div class="tools-container">' + '<spam class="name"><b></b></spam>' + '<spam class="tools">' + '<a class="download EZViewHighlight" download="FileName" target="_blank">' + '<img src="' + self.icons.download + '" title="Download" /></a>' + '<a class="zoomIn EZViewHighlight" href="">' + '<img src="' + self.icons.zoomIn + '" title="Zoom in [hotkey +]" /></a>' + '<a class="zoomOut EZViewHighlight" href="">' + '<img src="' + self.icons.zoomOut + '" title="Zoom out [hotkey -]" /></a>' + '</spam>' + '<a class="close EZViewHighlight" href="">' + '<img src="' + self.icons.close + '"/></a>' + '</div>', container = '<div class="pdf-preview-container"/>' + '<a class="back EZViewHighlight"><img src="' + self.icons.back + '" title="Back [hotkey <]" /></a>' + '<a class="next EZViewHighlight"><img src="' + self.icons.next + '" title="Next [hotkey >]" /></a>' + tools, template = '<div id="' + collectionName + '" ' + 'class="EZView EZView-container hide" style="display: flex; align-items: center; justify-content: center">' + container + '</div>'; try { $('body').append(template); } catch (e) { console.error('Error creating container: ' + e); } self.$EZView = $('#' + collectionName).hide(); self.addEvents(); } }; /** * Check if the index belongs to any img * @param {int} imgIndex index to be checked * @return {boolean} To know if the index belongs to any img */ self.isImg = function(imgIndex) { return self.arContent[collectionName][imgIndex].isImg; }; /** * Show controlls acording of the current image * @param {int} imgIndex index image to be checked * @return void */ self.showOrHideControls = function(imgIndex = self.index[collectionName]) { var href = self.arContent[collectionName][imgIndex].href, name = self.arContent[collectionName][imgIndex].name; if (self.isImg(imgIndex)) { $('.zoomIn, .zoomOut').fadeIn(200); } else { $('.zoomIn, .zoomOut').fadeOut(200); } $('.download').attr('href', href); $('.name>b').html(name); if (imgIndex > 0) { $('.back').fadeIn(200); } else { $('.back').fadeOut(200); } if (imgIndex < self.arContent[collectionName].length - 1) { $('.next').fadeIn(200); } else { $('.next').fadeOut(200); } }; self.buildHtmlContent = function() { var imgIndex = self.index[collectionName], src = self.arContent[collectionName][imgIndex].href, isPdf = src.match('.pdf'); // Content to show var htmlContent = '<img index-content="' + collectionName + imgIndex + '" src="' + src + '" class="content" />'; // To show pdf files if (isPdf) { htmlContent = '<iframe class="content" frameborder="0" index-content="' + collectionName + imgIndex + '" height="' + $(window).height() * 0.95 + '" width="' + $(window).width() * 0.9 + '" src="' + src + '" type="application/pdf"><p>Your browser does not support iframes.</p>' + '<script type="text/javascript">alert("Your browser does not support iframes.")</script><iframe/>'; self.arContent[collectionName][imgIndex].isImg = false; } return htmlContent; }; /** * Set img on view container * @param string htmlContent */ self.setContentOnViewer = function(htmlContent) { var newIndex = self.index[collectionName]; // Append object to body self.$EZView.find('.pdf-preview-container') // Add img in container .append(htmlContent) // Stop propagation .find('.download, img').click(function(e) { // Avoid trigger remove action e.stopPropagation(); }); // Check If href isn't exists and show unsuported msg $('[index-content=' + collectionName + newIndex + ']').on("error", function() { var style = 'padding: 10px; border-radius: 10px; background-color: #fff'; $(this).replaceWith('<h1 class = "preview-title" index-content="' + collectionName + self.index[collectionName] + '" style="' + style + '">Unsupported preview</h1>'); self.arContent[collectionName][self.index[collectionName]].isImg = false; }); self.setStyles(); self.showOrHideControls(newIndex); }; self.setStyles = function() { // Set Styles self.$EZView .css({ 'background-color': '#fff!important', 'height': '100%', 'width': '100%', 'z-index': '10000', 'position': 'fixed', 'top': '0', 'left': '0', 'cursor': 'default', 'aling-items': 'center', 'margin': 'auto', }) .find('.content').css({ 'max-width': $(window).width() * 0.7, 'max-height': $(window).height() * 0.9, 'width': $(window).width() * 0.5, // Inicial image width 'z-index': '10001', }).end() .find('.name').css({ 'display': 'flex', 'align-items': 'center', 'justify-content': 'center', 'padding': '7px', }).end() .find('.close>img').css({ 'top': '0', 'right': '0', 'position': 'fixed', }).end() .find('.back>img').css({ 'top': $(window).height() * 0.5, 'left': '0', 'position': 'fixed', }).end() .find('.next>img').css({ 'top': $(window).height() * 0.5, 'right': '0', 'position': 'fixed', }).end() .find('.tools-container').css({ 'background-color': 'rgba(255,255,255,0.5)', 'height': '35px', 'width': '100%', 'position': 'fixed', 'top': '0', 'left': '0', 'padding': '3px', 'padding-top': '0px', 'z-index': '100000' }).end() .find('.tools-container>a').css({ 'margin': '10px', }).end() .find('.tools').css({ 'top': '0', 'left': '0', 'position': 'fixed', }).end() .find('.EZViewHighlight>img').css({ 'padding': '10px', 'height': 'auto', 'width': 'auto', 'z-index': '100000', 'cursor': 'pointer', }).end() .find('.EZViewHighlight>img').hover( function() { $(this).css({ 'border-radius': '100%', 'background-color': 'rgba(255,255,255,0.5)', 'z-index': '100000' }); }, function() { $(this).css({ 'background-color': '', 'border-radius': '', }); } ); }; /** * Set keyboard events * @param {object} e Window object * @return void */ self.keyupEvents = function(e) { var keyCode = e.keyCode; switch (keyCode) { // Esc case 27: self.close(); break; // Next case 39: self.next(); break; // Back case 37: self.back(); break; // zoomn case 107: self.zoom(true); break; // zoomOut case 109: self.zoom(); break; } }; /** * Hide collection * @return void */ self.close = function() { // Hide EZView elements self.$EZView.hide(200) self.returnImageToOriginalMeasures(self.index[collectionName]); // Remove keyup events $(window).off('keyup', null, self.keyupEvents); }; /** * zoom in or out on current img * @param {boolean} increment When is true do zoom in * @return void */ self.zoom = function(increment) { if (!self.isImg(self.index[collectionName])) { return; } var $img = $('[index-content=' + collectionName + self.index[collectionName] + ']', self.$EZView), height = parseInt($img.css('height')), width = parseInt($img.css('width')); // Allow the image to return to its original measures when is closed or the focus is lost if (!self.arContent[collectionName][self.index[collectionName]]['original-height']) { self.arContent[collectionName][self.index[collectionName]]['original-height'] = height; } if (!self.arContent[collectionName][self.index[collectionName]]['original-width']) { self.arContent[collectionName][self.index[collectionName]]['original-width'] = width; } // Avoid lose img proportions on zoom in $img.css({ 'max-width': '', 'max-height': '', }); // Enable drag if we have the pluging available if ($.fn.draggable) { $img .draggable() .css({ 'cursor': 'move' }); } // Increase or decrease sizes if (increment) { height += height * 0.2; width += width * 0.2; } else { height -= height * 0.2; width -= width * 0.2; } // Do the zoom $img.animate({ 'height': (height) + 'px', 'width': (width) + 'px' }, 200); }; self.addEvents = function() { // Main container self.$EZView // Add close Event .click(function(e) { e.stopPropagation(); e.preventDefault(); self.close(); }) // Add close Event .find('.close>img').click(function(e) { e.preventDefault(); e.stopPropagation(); self.close(); }).end() // Stop propagation .find('.download, img, .tools').click(function(e) { // Avoid trigger remove action e.stopPropagation(); }).end() // Add back event .find('.back>img').click(function(e) { e.stopPropagation(); e.preventDefault(); self.back(); }).end() // Add next event .find('.next>img').click(function(e) { e.stopPropagation(); e.preventDefault(); self.next(); }).end() // Add prevent default .find('.zoomIn, .zoomOut, .close').click(function(e) { e.stopPropagation(); e.preventDefault(); }).end() // Add zoomIn event .find('.zoomIn>img').click(function(e) { e.stopPropagation(); e.preventDefault(); self.zoom(true); }).end() // Add zoomOut event .find('.zoomOut>img').click(function(e) { e.stopPropagation(); e.preventDefault(); self.zoom(); }); }; /** * Calculate the new index and call to show that img * @return void */ self.next = function() { var newIndex = self.index[collectionName] + 1; // Check if the link element is visible over the DOM if (!self.arContent[collectionName][newIndex]) { return; } // Check if the element exists var hrefExist = $('[href="' + self.arContent[collectionName][newIndex].href + '"]:visible').length, srcExist = $('[src="' + self.arContent[collectionName][newIndex].href + '"]:visible').length, goToNext = hrefExist || srcExist; // If the element does not exists on view, go to the next img if (!goToNext) { self.index[collectionName] += 1; return self.next(); } self.goTo(newIndex); }; /** * Calculate the new index and call to show that img * @return void */ self.back = function() { var newIndex = self.index[collectionName] - 1; // Check if exist next image if (!self.arContent[collectionName][newIndex]) { return; } // Check if the link element is visible over the DOM var hrefExist = $('[href="' + self.arContent[collectionName][newIndex].href + '"]:visible').length, srcExist = $('[src="' + self.arContent[collectionName][newIndex].href + '"]:visible').length, goToBack = hrefExist || srcExist; // If the element does not exists on view, go to the next back img if (!goToBack) { self.index[collectionName] -= 1; return self.back(); } self.goTo(newIndex); }; /** * Set image on view container collection * @param {int} newIndex image index to be show * @return void */ self.goTo = function(newIndex) { // Hide all imgs $('.content', self.$EZView).hide(); self.index[collectionName] = newIndex; self.returnImageToOriginalMeasures(newIndex); if (self.arContent[collectionName][newIndex]) { if (self.arContent[collectionName][newIndex].isRender) { $('[index-content=' + collectionName + self.index[collectionName] + ']').slideUp(); $('[index-content=' + collectionName + newIndex + ']').slideDown(); } else { $('[index-content=' + collectionName + self.index[collectionName] + ']').slideUp(); self.setContentOnViewer(self.buildHtmlContent()); $('[index-content=' + collectionName + newIndex + ']').slideDown(); self.arContent[collectionName][newIndex].isRender = true; } self.showOrHideControls(); } }; /** * Open collection with the image thumbnail clicked * @param {object} e event imag thumbnail clicked * @return void */ self.runEZView = function(e) { var $thumbnail = $(e.target); // Set index of collection img self.index[collectionName] = parseInt($thumbnail.data('index')); self.showOrHideControls(); // Set img on collection self.goTo(self.index[collectionName]); // Add keyup events $(window).off('keyup', null, self.keyupEvents).keyup(self.keyupEvents); // Show EZView self.$EZView.show(200); }; // Return constructor execution return self.constructor(); };