// BigPicture.js | license MIT | henrygd.me/bigpicture
(function () {
const // assign window object to variable
global = window;
let // trigger element used to open popup
el,
// set to true after first interaction
initialized,
// container element holding html needed for script
container,
// currently active display element (image, video, youtube / vimeo iframe container)
displayElement,
// popup image element
displayImage,
// popup video element
displayVideo,
// popup audio element
displayAudio,
// container element to hold youtube / vimeo iframe
iframeContainer,
// iframe to hold youtube / vimeo player
iframeSiteVid,
// store requested image source
imgSrc,
// button that closes the container
closeButton,
// youtube / vimeo video id
siteVidID,
// keeps track of loading icon display state
isLoading,
// timeout to check video status while loading
checkMediaTimeout,
// loading icon element
loadingIcon,
// caption element
caption,
// caption content element
captionText,
// store caption content
captionContent,
// hide caption button element
captionHideButton,
// open state for container element
isOpen,
// gallery open state
galleryOpen,
// used during close animation to avoid triggering timeout twice
isClosing;
const // array of prev viewed image urls to check if cached before showing loading icon
imgCache = [];
let // store whether image requested is remote or local
remoteImage,
// store animation opening callbacks
animationStart,
animationEnd,
// gallery left / right icons
rightArrowBtn,
leftArrowBtn,
// position of gallery
galleryPosition,
// hold active gallery els / image src
galleryEls,
// counter element
galleryCounter,
// store images in gallery that are being loaded
preloadedImages = {},
// whether device supports touch events
supportsTouch,
// options object
opts;
const // Save bytes in the minified version
doc = document,
appendEl = 'appendChild',
createEl = 'createElement',
removeEl = 'removeChild',
htmlInner = 'innerHTML',
pointerEventsAuto = 'pointer-events:auto',
cHeight = 'clientHeight',
cWidth = 'clientWidth',
listenFor = 'addEventListener',
timeout = global.setTimeout,
clearTimeout = global.clearTimeout;
global.BigPicture = function (options) {
// initialize called on initial open to create elements / style / event handlers
initialized || initialize();
// clear currently loading stuff
if (isLoading) {
clearTimeout(checkMediaTimeout);
removeContainer()
}
opts = options;
// store video id if youtube / vimeo video is requested
siteVidID = options.ytSrc || options.vimeoSrc;
// store optional callbacks
animationStart = options.animationStart;
animationEnd = options.animationEnd;
// set trigger element
el = options.el;
// wipe existing remoteImage state
remoteImage = false;
// set caption if provided
captionContent = el.getAttribute('data-caption');
if (options.gallery) {
makeGallery(options.gallery)
} else if (siteVidID || options.iframeSrc) {
// if vimeo, youtube, or iframe video
toggleLoadingIcon(true);
displayElement = iframeContainer;
createIframe();
} else if (options.imgSrc) {
// if remote image
remoteImage = true;
imgSrc = options.imgSrc;
!~imgCache.indexOf(imgSrc) && toggleLoadingIcon(true);
displayElement = displayImage;
displayElement.src = imgSrc
} else if (options.audio) {
// if direct video link
toggleLoadingIcon(true);
displayElement = displayAudio;
displayElement.src = options.audio;
checkMedia('audio file')
} else if (options.vidSrc) {
// if direct video link
toggleLoadingIcon(true);
makeVidSrc(options.vidSrc);
checkMedia('video')
} else {
// local image / background image already loaded on page
displayElement = displayImage;
// get img source or element background image
displayElement.src =
el.tagName === 'IMG'
? el.src
: global
.getComputedStyle(el)
.backgroundImage.replace(/^url|[(|)|'|"]/g, '')
}
// add container to page
container[appendEl](displayElement);
doc.body[appendEl](container)
};
// create all needed methods / store dom elements on first use
function initialize() {
let startX;
// return close button elements
function createCloseButton(className) {
const el = doc[createEl]('button');
el.className = className;
el[htmlInner] = '';
return el
}
function createArrowSymbol(direction, style) {
const el = doc[createEl]('button');
el.className = 'bp-lr';
el[htmlInner] =
'';
changeCSS(el, style);
el.onclick = function (e) {
e.stopPropagation();
updateGallery(direction)
};
return el
}
// add style - if you want to tweak, run through beautifier
const style = doc[createEl]('STYLE');
style[htmlInner] =
'#bp_caption,#bp_container{bottom:0;left:0;right:0;position:fixed;opacity:0}#bp_container>*,#bp_loader{position:absolute;right:0;z-index:10}#bp_container,#bp_caption,#bp_container svg{pointer-events:none}#bp_container{top:0;z-index:9999;background:rgba(0,0,0,.7);opacity:0;transition:opacity .35s}#bp_loader{top:0;left:0;bottom:0;display:flex;margin:0;cursor:wait;z-index:9;background:0 0}#bp_loader svg{width:50%;max-width:300px;max-height:50%;margin:auto;animation:bpturn 1s infinite linear}#bp_aud,#bp_container img,#bp_sv,#bp_vid{user-select:none;max-height:96%;max-width:96%;top:0;bottom:0;left:0;margin:auto;box-shadow:0 0 3em rgba(0,0,0,.4);z-index:-1}#bp_sv{height:0;padding-bottom:54%;background-color:#000;width:96%}#bp_caption{font-size:.9em;padding:1.3em;background:rgba(15,15,15,.94);color:#fff;text-align:center;transition:opacity .3s}#bp_aud{width:650px;top:calc(50% - 20px);bottom:auto;box-shadow:none}#bp_count{left:0;right:auto;padding:14px;color:rgba(255,255,255,.7);font-size:22px;cursor:default}#bp_container button{position:absolute;border:0;outline:0;background:0 0;cursor:pointer;transition:all .1s}#bp_container>.bp-x{height:41px;width:41px;border-radius:100%;top:8px;right:14px;opacity:.8}#bp_container>.bp-x:focus,#bp_container>.bp-x:hover{background:rgba(255,255,255,.2)}.bp-x svg,.bp-xc svg{height:21px;width:20px;fill:#fff;vertical-align:top;}.bp-xc svg{width:16px}#bp_container .bp-xc{left:2%;bottom:100%;padding:9px 20px 7px;background:#d04444;border-radius:2px 2px 0 0;opacity:.85}#bp_container .bp-xc:focus,#bp_container .bp-xc:hover{opacity:1}.bp-lr{top:50%;top:calc(50% - 130px);padding:99px 0;width:6%;background:0 0;border:0;opacity:.4;transition:opacity .1s}.bp-lr:focus,.bp-lr:hover{opacity:.8}@keyframes bpf{50%{transform:translatex(15px)}100%{transform:none}}@keyframes bpl{50%{transform:translatex(-15px)}100%{transform:none}}@keyframes bpfl{0%{opacity:0;transform:translatex(70px)}100%{opacity:1;transform:none}}@keyframes bpfr{0%{opacity:0;transform:translatex(-70px)}100%{opacity:1;transform:none}}@keyframes bpfol{0%{opacity:1;transform:none}100%{opacity:0;transform:translatex(-70px)}}@keyframes bpfor{0%{opacity:1;transform:none}100%{opacity:0;transform:translatex(70px)}}@keyframes bpturn{0%{transform:none}100%{transform:rotate(360deg)}}@media (max-width:600px){.bp-lr{font-size:15vw}}@media (min-aspect-ratio:9/5){#bp_sv{height:98%;width:170.6vh;padding:0}}';
doc.head[appendEl](style);
// create container element
container = doc[createEl]('DIV');
container.id = 'bp_container';
container.onclick = close;
closeButton = createCloseButton('bp-x');
container[appendEl](closeButton);
// gallery swipe listeners
if ('ontouchstart' in global) {
supportsTouch = true;
container.ontouchstart = function (e) {
startX = e.changedTouches[0].pageX
};
container.ontouchmove = function (e) {
e.preventDefault()
};
container.ontouchend = function (e) {
if (!galleryOpen) {
return
}
const touchobj = e.changedTouches[0];
const distX = touchobj.pageX - startX;
// swipe right
distX < -30 && updateGallery(1);
// swipe left
distX > 30 && updateGallery(-1)
}
}
// create display image element
displayImage = doc[createEl]('IMG');
// create display video element
displayVideo = doc[createEl]('VIDEO');
displayVideo.id = 'bp_vid';
displayVideo.setAttribute('playsinline', true);
displayVideo.controls = true;
displayVideo.loop = true;
// create audio element
displayAudio = doc[createEl]("audio");
displayAudio.id = "bp_aud";
displayAudio.controls = true;
displayAudio.loop = true;
// create gallery counter
galleryCounter = doc[createEl]('span');
galleryCounter.id = 'bp_count';
// create caption elements
caption = doc[createEl]('DIV');
caption.id = 'bp_caption';
captionHideButton = createCloseButton('bp-xc');
captionHideButton.onclick = toggleCaption.bind(null, false);
caption[appendEl](captionHideButton);
captionText = doc[createEl]('SPAN');
caption[appendEl](captionText);
container[appendEl](caption);
// left / right arrow icons
rightArrowBtn = createArrowSymbol(1, 'transform:scalex(-1)');
leftArrowBtn = createArrowSymbol(-1, 'left:0;right:auto');
// create loading icon element
loadingIcon = doc[createEl]('DIV');
loadingIcon.id = 'bp_loader';
loadingIcon[htmlInner] =
'';
// create youtube / vimeo container
iframeContainer = doc[createEl]('DIV');
iframeContainer.id = 'bp_sv';
// create iframe to hold youtube / vimeo player
iframeSiteVid = doc[createEl]('IFRAME');
iframeSiteVid.setAttribute('allowfullscreen', true);
iframeSiteVid.allow = 'autoplay; fullscreen';
iframeSiteVid.onload = open;
changeCSS(iframeSiteVid, 'border:0;position:absolute;height:100%;width:100%;left:0;top:0');
iframeContainer[appendEl](iframeSiteVid);
// display image bindings for image load and error
displayImage.onload = open;
displayImage.onerror = open.bind(null, 'image');
// adjust loader position on window resize
global[listenFor]('resize', function () {
galleryOpen || (isLoading && toggleLoadingIcon(true))
});
// close container on escape key press and arrow buttons for gallery
doc[listenFor]('keyup', function (e) {
const key = e.keyCode;
key === 27 && isOpen && close(container);
if (galleryOpen) {
key === 39 && updateGallery(1);
key === 37 && updateGallery(-1);
key === 38 && updateGallery(10);
key === 40 && updateGallery(-10)
}
});
// prevent scrolling with arrow keys if gallery open
doc[listenFor]('keydown', function (e) {
const usedKeys = [37, 38, 39, 40];
if (galleryOpen && ~usedKeys.indexOf(e.keyCode)) {
e.preventDefault()
}
});
// trap focus within container while open
doc[listenFor](
'focus',
function (e) {
if (isOpen && !container.contains(e.target)) {
e.stopPropagation();
closeButton.focus()
}
},
true
);
// all done
initialized = true
}
// return transform style to make full size display el match trigger el size
function getRect() {
const rect = el.getBoundingClientRect();
const leftOffset = rect.left - (container[cWidth] - rect.width) / 2;
const centerTop = rect.top - (container[cHeight] - rect.height) / 2;
const scaleWidth = el[cWidth] / displayElement[cWidth];
const scaleHeight = el[cHeight] / displayElement[cHeight];
return 'transform:translate3D(' +
leftOffset +
'px, ' +
centerTop +
'px, 0) scale3D(' +
scaleWidth +
', ' +
scaleHeight +
', 0)'
}
function makeVidSrc(source) {
if (Array.isArray(source)) {
displayElement = displayVideo.cloneNode();
source.forEach(function (src) {
const source = doc[createEl]('SOURCE');
source.src = src;
source.type = 'video/' + src.match(/.(\w+)$/)[1];
displayElement[appendEl](source)
})
} else {
displayElement = displayVideo;
displayElement.src = source
}
}
function makeGallery(gallery) {
if (Array.isArray(gallery)) {
// is array of images
galleryPosition = 0;
galleryEls = gallery;
captionContent = gallery[0].caption
} else {
// is element selector or nodelist
galleryEls = [].slice.call(typeof gallery === 'string' ? doc.querySelectorAll(gallery + ' [data-bp]') : gallery);
// find initial gallery position
const elIndex = galleryEls.indexOf(el);
galleryPosition = elIndex !== -1 ? elIndex : 0;
// make gallery object w/ els / src / caption
galleryEls = galleryEls.map(function (el) {
return {
el: el,
src: el.getAttribute('data-bp'),
caption: el.getAttribute('data-caption')
}
})
}
// show loading icon if needed
remoteImage = true;
// set initial src to imgSrc so it will be cached in open func
imgSrc = galleryEls[galleryPosition].src;
!~imgCache.indexOf(imgSrc) && toggleLoadingIcon(true);
if (galleryEls.length > 1) {
// if length is greater than one, add gallery stuff
container[appendEl](galleryCounter);
galleryCounter[htmlInner] = galleryPosition + 1 + '/' + galleryEls.length;
if (!supportsTouch) {
// add arrows if device doesn't support touch
container[appendEl](rightArrowBtn);
container[appendEl](leftArrowBtn)
}
} else {
// gallery is one, just show without clutter
galleryEls = false
}
displayElement = displayImage;
// set initial image src
displayElement.src = imgSrc
}
function updateGallery(movement) {
const galleryLength = galleryEls.length - 1;
let isEnd;
// only allow one change at a time
if (isLoading) {
return
}
// return if requesting out of range image
if (movement > 0) {
if (galleryPosition === galleryLength) {
isEnd = true
}
} else if (galleryPosition === 0) {
isEnd = true
}
if (isEnd) {
// if beginning or end of gallery, run end animation
changeCSS(displayImage, '');
timeout(changeCSS, 9, displayImage, 'animation:' + (movement > 0 ? 'bpl' : 'bpf') + ' .3s;transition:transform .35s');
return
}
// normalize position
galleryPosition = Math.max(
0,
Math.min(galleryPosition + movement, galleryLength)
)
// load images before and after for quicker scrolling through pictures
;[galleryPosition - 1, galleryPosition, galleryPosition + 1].forEach(
function (position) {
// normalize position
position = Math.max(0, Math.min(position, galleryLength));
// cancel if image has already been preloaded
if (preloadedImages[position]) return;
const src = galleryEls[position].src;
// create image for preloadedImages
const img = doc[createEl]('IMG');
img[listenFor]('load', addToImgCache.bind(null, src));
img.src = src;
preloadedImages[position] = img
}
);
// if image is loaded, show it
if (preloadedImages[galleryPosition].complete) {
return changeGalleryImage(movement)
}
// if not, show loading icon and change when loaded
isLoading = true;
changeCSS(loadingIcon, 'opacity:.4;');
container[appendEl](loadingIcon);
preloadedImages[galleryPosition].onload = function () {
galleryOpen && changeGalleryImage(movement)
};
// if error, store error object in el array
preloadedImages[galleryPosition].onerror = function () {
galleryEls[galleryPosition] = {
error: 'Error loading image'
};
galleryOpen && changeGalleryImage(movement)
}
}
function changeGalleryImage(movement) {
if (isLoading) {
container[removeEl](loadingIcon);
isLoading = false
}
const activeEl = galleryEls[galleryPosition];
if (activeEl.error) {
// show alert if error
alert(activeEl.error)
} else {
// add new image, animate images in and out w/ css animation
const oldimg = container.querySelector('img:last-of-type');
displayImage = displayElement = preloadedImages[galleryPosition];
changeCSS(displayImage, 'animation:' + (movement > 0 ? 'bpfl' : 'bpfr') + ' .35s;transition:transform .35s');
changeCSS(oldimg, 'animation:' + (movement > 0 ? 'bpfol' : 'bpfor') + ' .35s both');
container[appendEl](displayImage);
// update el for closing animation
if (activeEl.el) {
el = activeEl.el
}
}
// update counter
galleryCounter[htmlInner] = galleryPosition + 1 + '/' + galleryEls.length;
// show / hide caption
toggleCaption(galleryEls[galleryPosition].caption)
}
// create video iframe
function createIframe() {
let url;
const prefix = 'https://';
const suffix = 'autoplay=1';
// create appropriate url
if (opts.ytSrc) {
url = prefix + 'www.youtube.com/embed/' + siteVidID + '?html5=1&rel=0&playsinline=1&' + suffix;
} else if (opts.vimeoSrc) {
url = prefix + 'player.vimeo.com/video/' + siteVidID + '?' + suffix;
} else if (opts.iframeSrc) {
url = opts.iframeSrc;
}
// set iframe src to url
iframeSiteVid.src = url;
}
// timeout to check video status while loading
function checkMedia(errMsg) {
if (~[1, 4].indexOf(displayElement.readyState)) {
open();
// short timeout to to make sure controls show in safari 11
timeout(function () {
displayElement.play()
}, 99)
} else if (displayElement.error) open(errMsg);
else checkMediaTimeout = timeout(checkMedia, 35, errMsg)
}
// hide / show loading icon
function toggleLoadingIcon(bool) {
// don't show loading icon if noLoader is specified
if (opts.noLoader) return;
// bool is true if we want to show icon, false if we want to remove
// change style to match trigger element dimensions if we want to show
bool &&
changeCSS(
loadingIcon,
'top:' +
el.offsetTop +
'px;left:' +
el.offsetLeft +
'px;height:' +
el[cHeight] +
'px;width:' +
el[cWidth] +
'px'
);
// add or remove loader from DOM
el.parentElement[bool ? appendEl : removeEl](loadingIcon);
isLoading = bool
}
// hide & show caption
function toggleCaption(captionContent) {
if (captionContent) {
captionText[htmlInner] = captionContent
}
changeCSS(
caption,
'opacity:' + (captionContent ? '1;' + pointerEventsAuto : '0')
)
}
function addToImgCache(url) {
!~imgCache.indexOf(url) && imgCache.push(url)
}
// animate open of image / video; display caption if needed
function open(err) {
// hide loading spinner
isLoading && toggleLoadingIcon();
// execute animationStart callback
animationStart && animationStart();
// check if we have an error string instead of normal event
if (typeof err === 'string') {
removeContainer();
return opts.onError ? opts.onError() : alert('Error: The requested ' + err + ' could not be loaded.')
}
// if remote image is loaded, add url to imgCache array
remoteImage && addToImgCache(imgSrc);
// transform displayEl to match trigger el
changeCSS(displayElement, getRect());
// fade in container
changeCSS(container, 'opacity:1;' + pointerEventsAuto);
// set animationEnd callback to run after animation ends (cleared if container closed)
animationEnd = timeout(animationEnd, 410);
isOpen = true;
galleryOpen = !!galleryEls;
// enlarge displayEl, fade in caption if hasCaption
timeout(function () {
changeCSS(displayElement, 'transition:transform .35s;transform:none');
captionContent && timeout(toggleCaption, 250, captionContent)
}, 60)
}
// close active display element
function close(e) {
const target = e.target;
const clickEls = [
caption,
captionHideButton,
displayVideo,
displayAudio,
captionText,
leftArrowBtn,
rightArrowBtn,
loadingIcon
];
// blur to hide close button focus style
target && target.blur();
// don't close if one of the clickEls was clicked or container is already closing
if (isClosing || ~clickEls.indexOf(target)) {
return
}
// animate closing
displayElement.style.cssText += getRect();
changeCSS(container, pointerEventsAuto);
// timeout to remove els from dom; use variable to avoid calling more than once
timeout(removeContainer, 350);
// clear animationEnd timeout
clearTimeout(animationEnd);
isOpen = false;
isClosing = true
}
// remove container / display element from the DOM
function removeContainer() {
// remove container from DOM & clear inline style
doc.body[removeEl](container);
container[removeEl](displayElement);
changeCSS(container, '')
// clear src of displayElement (or iframe if display el is iframe container)
;(displayElement === iframeContainer
? iframeSiteVid
: displayElement
).removeAttribute('src');
// remove caption
toggleCaption(false);
if (galleryOpen) {
// remove all gallery stuff
const images = container.querySelectorAll('img');
for (let i = 0; i < images.length; i++) {
container[removeEl](images[i])
}
isLoading && container[removeEl](loadingIcon);
container[removeEl](galleryCounter);
galleryOpen = galleryEls = false;
preloadedImages = {};
supportsTouch || container[removeEl](rightArrowBtn);
supportsTouch || container[removeEl](leftArrowBtn);
// in case displayimage changed, we need to update event listeners
displayImage.onload = open;
displayImage.onerror = open.bind(null, 'image')
}
// run close callback
opts.onClose && opts.onClose();
isClosing = isLoading = false
}
// style helper functions
function changeCSS(element, newStyle) {
element.style.cssText = newStyle
}
})();