mirror of https://github.com/saimn/sigal.git
19 changed files with 5999 additions and 4 deletions
@ -0,0 +1,483 @@
|
||||
/*! PhotoSwipe Default UI CSS by Dmitry Semenov | photoswipe.com | MIT license */ |
||||
/* |
||||
|
||||
Contents: |
||||
|
||||
1. Buttons |
||||
2. Share modal and links |
||||
3. Index indicator ("1 of X" counter) |
||||
4. Caption |
||||
5. Loading indicator |
||||
6. Additional styles (root element, top bar, idle state, hidden state, etc.) |
||||
|
||||
*/ |
||||
/* |
||||
|
||||
1. Buttons |
||||
|
||||
*/ |
||||
/* <button> css reset */ |
||||
.pswp__button { |
||||
width: 44px; |
||||
height: 44px; |
||||
position: relative; |
||||
background: none; |
||||
cursor: pointer; |
||||
overflow: visible; |
||||
-webkit-appearance: none; |
||||
display: block; |
||||
border: 0; |
||||
padding: 0; |
||||
margin: 0; |
||||
float: right; |
||||
opacity: 0.75; |
||||
-webkit-transition: opacity 0.2s; |
||||
transition: opacity 0.2s; |
||||
-webkit-box-shadow: none; |
||||
box-shadow: none; } |
||||
.pswp__button:focus, |
||||
.pswp__button:hover { |
||||
opacity: 1; } |
||||
.pswp__button:active { |
||||
outline: none; |
||||
opacity: 0.9; } |
||||
.pswp__button::-moz-focus-inner { |
||||
padding: 0; |
||||
border: 0; } |
||||
|
||||
/* pswp__ui--over-close class it added when mouse is over element that should close gallery */ |
||||
.pswp__ui--over-close .pswp__button--close { |
||||
opacity: 1; } |
||||
|
||||
.pswp__button, |
||||
.pswp__button--arrow--left:before, |
||||
.pswp__button--arrow--right:before { |
||||
background: url('default-skin/default-skin.png') 0 0 no-repeat; |
||||
background-size: 264px 88px; |
||||
width: 44px; |
||||
height: 44px; } |
||||
|
||||
@media (-webkit-min-device-pixel-ratio: 1.1), (-webkit-min-device-pixel-ratio: 1.09375), (min-resolution: 105dpi), (min-resolution: 1.1dppx) { |
||||
/* Serve SVG sprite if browser supports SVG and resolution is more than 105dpi */ |
||||
.pswp--svg .pswp__button, |
||||
.pswp--svg .pswp__button--arrow--left:before, |
||||
.pswp--svg .pswp__button--arrow--right:before { |
||||
background-image: url(default-skin.svg); } |
||||
.pswp--svg .pswp__button--arrow--left, |
||||
.pswp--svg .pswp__button--arrow--right { |
||||
background: none; } } |
||||
|
||||
.pswp__button--close { |
||||
background-position: 0 -44px; } |
||||
|
||||
.pswp__button--share { |
||||
background-position: -44px -44px; } |
||||
|
||||
.pswp__button--fs { |
||||
display: none; } |
||||
|
||||
.pswp--supports-fs .pswp__button--fs { |
||||
display: block; } |
||||
|
||||
.pswp--fs .pswp__button--fs { |
||||
background-position: -44px 0; } |
||||
|
||||
.pswp__button--zoom { |
||||
display: none; |
||||
background-position: -88px 0; } |
||||
|
||||
.pswp--zoom-allowed .pswp__button--zoom { |
||||
display: block; } |
||||
|
||||
.pswp--zoomed-in .pswp__button--zoom { |
||||
background-position: -132px 0; } |
||||
|
||||
/* no arrows on touch screens */ |
||||
.pswp--touch .pswp__button--arrow--left, |
||||
.pswp--touch .pswp__button--arrow--right { |
||||
visibility: hidden; } |
||||
|
||||
/* |
||||
Arrow buttons hit area |
||||
(icon is added to :before pseudo-element) |
||||
*/ |
||||
.pswp__button--arrow--left, |
||||
.pswp__button--arrow--right { |
||||
background: none; |
||||
top: 50%; |
||||
margin-top: -50px; |
||||
width: 70px; |
||||
height: 100px; |
||||
position: absolute; } |
||||
|
||||
.pswp__button--arrow--left { |
||||
left: 0; } |
||||
|
||||
.pswp__button--arrow--right { |
||||
right: 0; } |
||||
|
||||
.pswp__button--arrow--left:before, |
||||
.pswp__button--arrow--right:before { |
||||
content: ''; |
||||
top: 35px; |
||||
background-color: rgba(0, 0, 0, 0.3); |
||||
height: 30px; |
||||
width: 32px; |
||||
position: absolute; } |
||||
|
||||
.pswp__button--arrow--left:before { |
||||
left: 6px; |
||||
background-position: -138px -44px; } |
||||
|
||||
.pswp__button--arrow--right:before { |
||||
right: 6px; |
||||
background-position: -94px -44px; } |
||||
|
||||
/* |
||||
|
||||
2. Share modal/popup and links |
||||
|
||||
*/ |
||||
.pswp__counter, |
||||
.pswp__share-modal { |
||||
-webkit-user-select: none; |
||||
-moz-user-select: none; |
||||
-ms-user-select: none; |
||||
user-select: none; } |
||||
|
||||
.pswp__share-modal { |
||||
display: block; |
||||
background: rgba(0, 0, 0, 0.5); |
||||
width: 100%; |
||||
height: 100%; |
||||
top: 0; |
||||
left: 0; |
||||
padding: 10px; |
||||
position: absolute; |
||||
z-index: 1600; |
||||
opacity: 0; |
||||
-webkit-transition: opacity 0.25s ease-out; |
||||
transition: opacity 0.25s ease-out; |
||||
-webkit-backface-visibility: hidden; |
||||
will-change: opacity; } |
||||
|
||||
.pswp__share-modal--hidden { |
||||
display: none; } |
||||
|
||||
.pswp__share-tooltip { |
||||
z-index: 1620; |
||||
position: absolute; |
||||
background: #FFF; |
||||
top: 56px; |
||||
border-radius: 2px; |
||||
display: block; |
||||
width: auto; |
||||
right: 44px; |
||||
-webkit-box-shadow: 0 2px 5px rgba(0, 0, 0, 0.25); |
||||
box-shadow: 0 2px 5px rgba(0, 0, 0, 0.25); |
||||
-webkit-transform: translateY(6px); |
||||
-ms-transform: translateY(6px); |
||||
transform: translateY(6px); |
||||
-webkit-transition: -webkit-transform 0.25s; |
||||
transition: transform 0.25s; |
||||
-webkit-backface-visibility: hidden; |
||||
will-change: transform; } |
||||
.pswp__share-tooltip a { |
||||
display: block; |
||||
padding: 8px 12px; |
||||
color: #000; |
||||
text-decoration: none; |
||||
font-size: 14px; |
||||
line-height: 18px; } |
||||
.pswp__share-tooltip a:hover { |
||||
text-decoration: none; |
||||
color: #000; } |
||||
.pswp__share-tooltip a:first-child { |
||||
/* round corners on the first/last list item */ |
||||
border-radius: 2px 2px 0 0; } |
||||
.pswp__share-tooltip a:last-child { |
||||
border-radius: 0 0 2px 2px; } |
||||
|
||||
.pswp__share-modal--fade-in { |
||||
opacity: 1; } |
||||
.pswp__share-modal--fade-in .pswp__share-tooltip { |
||||
-webkit-transform: translateY(0); |
||||
-ms-transform: translateY(0); |
||||
transform: translateY(0); } |
||||
|
||||
/* increase size of share links on touch devices */ |
||||
.pswp--touch .pswp__share-tooltip a { |
||||
padding: 16px 12px; } |
||||
|
||||
a.pswp__share--facebook:before { |
||||
content: ''; |
||||
display: block; |
||||
width: 0; |
||||
height: 0; |
||||
position: absolute; |
||||
top: -12px; |
||||
right: 15px; |
||||
border: 6px solid transparent; |
||||
border-bottom-color: #FFF; |
||||
-webkit-pointer-events: none; |
||||
-moz-pointer-events: none; |
||||
pointer-events: none; } |
||||
|
||||
a.pswp__share--facebook:hover { |
||||
background: #3E5C9A; |
||||
color: #FFF; } |
||||
a.pswp__share--facebook:hover:before { |
||||
border-bottom-color: #3E5C9A; } |
||||
|
||||
a.pswp__share--twitter:hover { |
||||
background: #55ACEE; |
||||
color: #FFF; } |
||||
|
||||
a.pswp__share--pinterest:hover { |
||||
background: #CCC; |
||||
color: #CE272D; } |
||||
|
||||
a.pswp__share--download:hover { |
||||
background: #DDD; } |
||||
|
||||
/* |
||||
|
||||
3. Index indicator ("1 of X" counter) |
||||
|
||||
*/ |
||||
.pswp__counter { |
||||
position: absolute; |
||||
left: 0; |
||||
top: 0; |
||||
height: 44px; |
||||
font-size: 13px; |
||||
line-height: 44px; |
||||
color: #FFF; |
||||
opacity: 0.75; |
||||
padding: 0 10px; } |
||||
|
||||
/* |
||||
|
||||
4. Caption |
||||
|
||||
*/ |
||||
.pswp__caption { |
||||
position: absolute; |
||||
left: 0; |
||||
bottom: 0; |
||||
width: 100%; |
||||
min-height: 44px; } |
||||
.pswp__caption small { |
||||
font-size: 11px; |
||||
color: #BBB; } |
||||
|
||||
.pswp__caption__center { |
||||
text-align: left; |
||||
max-width: 420px; |
||||
margin: 0 auto; |
||||
font-size: 13px; |
||||
padding: 10px; |
||||
line-height: 20px; |
||||
color: #CCC; } |
||||
|
||||
.pswp__caption--empty { |
||||
display: none; } |
||||
|
||||
/* Fake caption element, used to calculate height of next/prev image */ |
||||
.pswp__caption--fake { |
||||
visibility: hidden; } |
||||
|
||||
/* |
||||
|
||||
5. Loading indicator (preloader) |
||||
|
||||
You can play with it here - http://codepen.io/dimsemenov/pen/yyBWoR |
||||
|
||||
*/ |
||||
.pswp__preloader { |
||||
width: 44px; |
||||
height: 44px; |
||||
position: absolute; |
||||
top: 0; |
||||
left: 50%; |
||||
margin-left: -22px; |
||||
opacity: 0; |
||||
-webkit-transition: opacity 0.25s ease-out; |
||||
transition: opacity 0.25s ease-out; |
||||
will-change: opacity; |
||||
direction: ltr; } |
||||
|
||||
.pswp__preloader__icn { |
||||
width: 20px; |
||||
height: 20px; |
||||
margin: 12px; } |
||||
|
||||
.pswp__preloader--active { |
||||
opacity: 1; } |
||||
.pswp__preloader--active .pswp__preloader__icn { |
||||
/* We use .gif in browsers that don't support CSS animation */ |
||||
background: url(preloader.gif) 0 0 no-repeat; } |
||||
|
||||
.pswp--css_animation .pswp__preloader--active { |
||||
opacity: 1; } |
||||
.pswp--css_animation .pswp__preloader--active .pswp__preloader__icn { |
||||
-webkit-animation: clockwise 500ms linear infinite; |
||||
animation: clockwise 500ms linear infinite; } |
||||
.pswp--css_animation .pswp__preloader--active .pswp__preloader__donut { |
||||
-webkit-animation: donut-rotate 1000ms cubic-bezier(0.4, 0, 0.22, 1) infinite; |
||||
animation: donut-rotate 1000ms cubic-bezier(0.4, 0, 0.22, 1) infinite; } |
||||
|
||||
.pswp--css_animation .pswp__preloader__icn { |
||||
background: none; |
||||
opacity: 0.75; |
||||
width: 14px; |
||||
height: 14px; |
||||
position: absolute; |
||||
left: 15px; |
||||
top: 15px; |
||||
margin: 0; } |
||||
|
||||
.pswp--css_animation .pswp__preloader__cut { |
||||
/* |
||||
The idea of animating inner circle is based on Polymer ("material") loading indicator |
||||
by Keanu Lee https://blog.keanulee.com/2014/10/20/the-tale-of-three-spinners.html |
||||
*/ |
||||
position: relative; |
||||
width: 7px; |
||||
height: 14px; |
||||
overflow: hidden; } |
||||
|
||||
.pswp--css_animation .pswp__preloader__donut { |
||||
-webkit-box-sizing: border-box; |
||||
box-sizing: border-box; |
||||
width: 14px; |
||||
height: 14px; |
||||
border: 2px solid #FFF; |
||||
border-radius: 50%; |
||||
border-left-color: transparent; |
||||
border-bottom-color: transparent; |
||||
position: absolute; |
||||
top: 0; |
||||
left: 0; |
||||
background: none; |
||||
margin: 0; } |
||||
|
||||
@media screen and (max-width: 1024px) { |
||||
.pswp__preloader { |
||||
position: relative; |
||||
left: auto; |
||||
top: auto; |
||||
margin: 0; |
||||
float: right; } } |
||||
|
||||
@-webkit-keyframes clockwise { |
||||
0% { |
||||
-webkit-transform: rotate(0deg); |
||||
transform: rotate(0deg); } |
||||
100% { |
||||
-webkit-transform: rotate(360deg); |
||||
transform: rotate(360deg); } } |
||||
|
||||
@keyframes clockwise { |
||||
0% { |
||||
-webkit-transform: rotate(0deg); |
||||
transform: rotate(0deg); } |
||||
100% { |
||||
-webkit-transform: rotate(360deg); |
||||
transform: rotate(360deg); } } |
||||
|
||||
@-webkit-keyframes donut-rotate { |
||||
0% { |
||||
-webkit-transform: rotate(0); |
||||
transform: rotate(0); } |
||||
50% { |
||||
-webkit-transform: rotate(-140deg); |
||||
transform: rotate(-140deg); } |
||||
100% { |
||||
-webkit-transform: rotate(0); |
||||
transform: rotate(0); } } |
||||
|
||||
@keyframes donut-rotate { |
||||
0% { |
||||
-webkit-transform: rotate(0); |
||||
transform: rotate(0); } |
||||
50% { |
||||
-webkit-transform: rotate(-140deg); |
||||
transform: rotate(-140deg); } |
||||
100% { |
||||
-webkit-transform: rotate(0); |
||||
transform: rotate(0); } } |
||||
|
||||
/* |
||||
|
||||
6. Additional styles |
||||
|
||||
*/ |
||||
/* root element of UI */ |
||||
.pswp__ui { |
||||
-webkit-font-smoothing: auto; |
||||
visibility: visible; |
||||
opacity: 1; |
||||
z-index: 1550; } |
||||
|
||||
/* top black bar with buttons and "1 of X" indicator */ |
||||
.pswp__top-bar { |
||||
position: absolute; |
||||
left: 0; |
||||
top: 0; |
||||
height: 44px; |
||||
width: 100%; } |
||||
|
||||
.pswp__caption, |
||||
.pswp__top-bar, |
||||
.pswp--has_mouse .pswp__button--arrow--left, |
||||
.pswp--has_mouse .pswp__button--arrow--right { |
||||
-webkit-backface-visibility: hidden; |
||||
will-change: opacity; |
||||
-webkit-transition: opacity 333ms cubic-bezier(0.4, 0, 0.22, 1); |
||||
transition: opacity 333ms cubic-bezier(0.4, 0, 0.22, 1); } |
||||
|
||||
/* pswp--has_mouse class is added only when two subsequent mousemove events occur */ |
||||
.pswp--has_mouse .pswp__button--arrow--left, |
||||
.pswp--has_mouse .pswp__button--arrow--right { |
||||
visibility: visible; } |
||||
|
||||
.pswp__top-bar, |
||||
.pswp__caption { |
||||
background-color: rgba(0, 0, 0, 0.5); } |
||||
|
||||
/* pswp__ui--fit class is added when main image "fits" between top bar and bottom bar (caption) */ |
||||
.pswp__ui--fit .pswp__top-bar, |
||||
.pswp__ui--fit .pswp__caption { |
||||
background-color: rgba(0, 0, 0, 0.3); } |
||||
|
||||
/* pswp__ui--idle class is added when mouse isn't moving for several seconds (JS option timeToIdle) */ |
||||
.pswp__ui--idle .pswp__top-bar { |
||||
opacity: 0; } |
||||
|
||||
.pswp__ui--idle .pswp__button--arrow--left, |
||||
.pswp__ui--idle .pswp__button--arrow--right { |
||||
opacity: 0; } |
||||
|
||||
/* |
||||
pswp__ui--hidden class is added when controls are hidden |
||||
e.g. when user taps to toggle visibility of controls |
||||
*/ |
||||
.pswp__ui--hidden .pswp__top-bar, |
||||
.pswp__ui--hidden .pswp__caption, |
||||
.pswp__ui--hidden .pswp__button--arrow--left, |
||||
.pswp__ui--hidden .pswp__button--arrow--right { |
||||
/* Force paint & create composition layer for controls. */ |
||||
opacity: 0.001; } |
||||
|
||||
/* pswp__ui--one-slide class is added when there is just one item in gallery */ |
||||
.pswp__ui--one-slide .pswp__button--arrow--left, |
||||
.pswp__ui--one-slide .pswp__button--arrow--right, |
||||
.pswp__ui--one-slide .pswp__counter { |
||||
display: none; } |
||||
|
||||
.pswp__element--disabled { |
||||
display: none !important; } |
||||
|
||||
.pswp--minimal--dark .pswp__top-bar { |
||||
background: none; } |
||||
|
After Width: | Height: | Size: 547 B |
|
After Width: | Height: | Size: 1.5 KiB |
|
After Width: | Height: | Size: 866 B |
|
After Width: | Height: | Size: 42 B |
@ -0,0 +1,126 @@
|
||||
/*! echo.js v1.6.0 | (c) 2015 @toddmotto | https://github.com/toddmotto/echo */ |
||||
(function (root, factory) { |
||||
if (typeof define === 'function' && define.amd) { |
||||
define(function() { |
||||
return factory(root); |
||||
}); |
||||
} else if (typeof exports === 'object') { |
||||
module.exports = factory; |
||||
} else { |
||||
root.echo = factory(root); |
||||
} |
||||
})(this, function (root) { |
||||
|
||||
'use strict'; |
||||
|
||||
var echo = {}; |
||||
|
||||
var callback = function () {}; |
||||
|
||||
var offset, poll, delay, useDebounce, unload; |
||||
|
||||
var inView = function (element, view) { |
||||
var box = element.getBoundingClientRect(); |
||||
return (box.right >= view.l && box.bottom >= view.t && box.left <= view.r && box.top <= view.b); |
||||
}; |
||||
|
||||
var debounceOrThrottle = function () { |
||||
if(!useDebounce && !!poll) { |
||||
return; |
||||
} |
||||
clearTimeout(poll); |
||||
poll = setTimeout(function(){ |
||||
echo.render(); |
||||
poll = null; |
||||
}, delay); |
||||
}; |
||||
|
||||
echo.init = function (opts) { |
||||
opts = opts || {}; |
||||
var offsetAll = opts.offset || 0; |
||||
var offsetVertical = opts.offsetVertical || offsetAll; |
||||
var offsetHorizontal = opts.offsetHorizontal || offsetAll; |
||||
var optionToInt = function (opt, fallback) { |
||||
return parseInt(opt || fallback, 10); |
||||
}; |
||||
offset = { |
||||
t: optionToInt(opts.offsetTop, offsetVertical), |
||||
b: optionToInt(opts.offsetBottom, offsetVertical), |
||||
l: optionToInt(opts.offsetLeft, offsetHorizontal), |
||||
r: optionToInt(opts.offsetRight, offsetHorizontal) |
||||
}; |
||||
delay = optionToInt(opts.throttle, 250); |
||||
useDebounce = opts.debounce !== false; |
||||
unload = !!opts.unload; |
||||
callback = opts.callback || callback; |
||||
echo.render(); |
||||
if (document.addEventListener) { |
||||
root.addEventListener('scroll', debounceOrThrottle, false); |
||||
root.addEventListener('load', debounceOrThrottle, false); |
||||
} else { |
||||
root.attachEvent('onscroll', debounceOrThrottle); |
||||
root.attachEvent('onload', debounceOrThrottle); |
||||
} |
||||
}; |
||||
|
||||
echo.render = function () { |
||||
var nodes = document.querySelectorAll('img[data-echo], [data-echo-background]'); |
||||
var length = nodes.length; |
||||
var src, elem; |
||||
var view = { |
||||
l: 0 - offset.l, |
||||
t: 0 - offset.t, |
||||
b: (root.innerHeight || document.documentElement.clientHeight) + offset.b, |
||||
r: (root.innerWidth || document.documentElement.clientWidth) + offset.r |
||||
}; |
||||
for (var i = 0; i < length; i++) { |
||||
elem = nodes[i]; |
||||
if (inView(elem, view)) { |
||||
|
||||
if (unload) { |
||||
elem.setAttribute('data-echo-placeholder', elem.src); |
||||
} |
||||
|
||||
if (elem.getAttribute('data-echo-background') !== null) { |
||||
elem.style.backgroundImage = "url(" + elem.getAttribute('data-echo-background') + ")"; |
||||
} |
||||
else { |
||||
elem.src = elem.getAttribute('data-echo'); |
||||
} |
||||
|
||||
if (!unload) { |
||||
elem.removeAttribute('data-echo'); |
||||
} |
||||
|
||||
callback(elem, 'load'); |
||||
} |
||||
else if (unload && !!(src = elem.getAttribute('data-echo-placeholder'))) { |
||||
|
||||
if (elem.getAttribute('data-echo-background') !== null) { |
||||
elem.style.backgroundImage = "url(" + src + ")"; |
||||
} |
||||
else { |
||||
elem.src = src; |
||||
} |
||||
|
||||
elem.removeAttribute('data-echo-placeholder'); |
||||
callback(elem, 'unload'); |
||||
} |
||||
} |
||||
if (!length) { |
||||
echo.detach(); |
||||
} |
||||
}; |
||||
|
||||
echo.detach = function () { |
||||
if (document.removeEventListener) { |
||||
root.removeEventListener('scroll', debounceOrThrottle); |
||||
} else { |
||||
root.detachEvent('onscroll', debounceOrThrottle); |
||||
} |
||||
clearTimeout(poll); |
||||
}; |
||||
|
||||
return echo; |
||||
|
||||
}); |
||||
@ -0,0 +1,2 @@
|
||||
/*! echo.js v1.6.0 | (c) 2015 @toddmotto | https://github.com/toddmotto/echo */ |
||||
!function(t,e){"function"==typeof define&&define.amd?define(function(){return e(t)}):"object"==typeof exports?module.exports=e:t.echo=e(t)}(this,function(t){"use strict";var e,n,o,r,c,a={},d=function(){},u=function(t,e){var n=t.getBoundingClientRect();return n.right>=e.l&&n.bottom>=e.t&&n.left<=e.r&&n.top<=e.b},l=function(){(r||!n)&&(clearTimeout(n),n=setTimeout(function(){a.render(),n=null},o))};return a.init=function(n){n=n||{};var u=n.offset||0,i=n.offsetVertical||u,f=n.offsetHorizontal||u,s=function(t,e){return parseInt(t||e,10)};e={t:s(n.offsetTop,i),b:s(n.offsetBottom,i),l:s(n.offsetLeft,f),r:s(n.offsetRight,f)},o=s(n.throttle,250),r=n.debounce!==!1,c=!!n.unload,d=n.callback||d,a.render(),document.addEventListener?(t.addEventListener("scroll",l,!1),t.addEventListener("load",l,!1)):(t.attachEvent("onscroll",l),t.attachEvent("onload",l))},a.render=function(){for(var n,o,r=document.querySelectorAll("img[data-echo], [data-echo-background]"),l=r.length,i={l:0-e.l,t:0-e.t,b:(t.innerHeight||document.documentElement.clientHeight)+e.b,r:(t.innerWidth||document.documentElement.clientWidth)+e.r},f=0;l>f;f++)o=r[f],u(o,i)?(c&&o.setAttribute("data-echo-placeholder",o.src),null!==o.getAttribute("data-echo-background")?o.style.backgroundImage="url("+o.getAttribute("data-echo-background")+")":o.src=o.getAttribute("data-echo"),c||o.removeAttribute("data-echo"),d(o,"load")):c&&(n=o.getAttribute("data-echo-placeholder"))&&(null!==o.getAttribute("data-echo-background")?o.style.backgroundImage="url("+n+")":o.src=n,o.removeAttribute("data-echo-placeholder"),d(o,"unload"));l||a.detach()},a.detach=function(){document.removeEventListener?t.removeEventListener("scroll",l):t.detachEvent("onscroll",l),clearTimeout(n)},a}); |
||||
@ -0,0 +1,860 @@
|
||||
/*! PhotoSwipe Default UI - 4.0.8 - 2015-05-21 |
||||
* http://photoswipe.com
|
||||
* Copyright (c) 2015 Dmitry Semenov; */ |
||||
/** |
||||
* |
||||
* UI on top of main sliding area (caption, arrows, close button, etc.). |
||||
* Built just using public methods/properties of PhotoSwipe. |
||||
*
|
||||
*/ |
||||
(function (root, factory) {
|
||||
if (typeof define === 'function' && define.amd) { |
||||
define(factory); |
||||
} else if (typeof exports === 'object') { |
||||
module.exports = factory(); |
||||
} else { |
||||
root.PhotoSwipeUI_Default = factory(); |
||||
} |
||||
})(this, function () { |
||||
|
||||
'use strict'; |
||||
|
||||
|
||||
|
||||
var PhotoSwipeUI_Default = |
||||
function(pswp, framework) { |
||||
|
||||
var ui = this; |
||||
var _overlayUIUpdated = false, |
||||
_controlsVisible = true, |
||||
_fullscrenAPI, |
||||
_controls, |
||||
_captionContainer, |
||||
_fakeCaptionContainer, |
||||
_indexIndicator, |
||||
_shareButton, |
||||
_shareModal, |
||||
_shareModalHidden = true, |
||||
_initalCloseOnScrollValue, |
||||
_isIdle, |
||||
_listen, |
||||
|
||||
_loadingIndicator, |
||||
_loadingIndicatorHidden, |
||||
_loadingIndicatorTimeout, |
||||
|
||||
_galleryHasOneSlide, |
||||
|
||||
_options, |
||||
_defaultUIOptions = { |
||||
barsSize: {top:44, bottom:'auto'}, |
||||
closeElClasses: ['item', 'caption', 'zoom-wrap', 'ui', 'top-bar'],
|
||||
timeToIdle: 4000,
|
||||
timeToIdleOutside: 1000, |
||||
loadingIndicatorDelay: 1000, // 2s
|
||||
|
||||
addCaptionHTMLFn: function(item, captionEl /*, isFake */) { |
||||
if(!item.title) { |
||||
captionEl.children[0].innerHTML = ''; |
||||
return false; |
||||
} |
||||
captionEl.children[0].innerHTML = item.title; |
||||
return true; |
||||
}, |
||||
|
||||
closeEl:true, |
||||
captionEl: true, |
||||
fullscreenEl: true, |
||||
zoomEl: true, |
||||
shareEl: true, |
||||
counterEl: true, |
||||
arrowEl: true, |
||||
preloaderEl: true, |
||||
|
||||
tapToClose: false, |
||||
tapToToggleControls: true, |
||||
|
||||
clickToCloseNonZoomable: true, |
||||
|
||||
shareButtons: [ |
||||
{id:'facebook', label:'Share on Facebook', url:'https://www.facebook.com/sharer/sharer.php?u={{url}}'}, |
||||
{id:'twitter', label:'Tweet', url:'https://twitter.com/intent/tweet?text={{text}}&url={{url}}'}, |
||||
{id:'pinterest', label:'Pin it', url:'http://www.pinterest.com/pin/create/button/'+ |
||||
'?url={{url}}&media={{image_url}}&description={{text}}'}, |
||||
{id:'download', label:'Download image', url:'{{raw_image_url}}', download:true} |
||||
], |
||||
getImageURLForShare: function( /* shareButtonData */ ) { |
||||
return pswp.currItem.src || ''; |
||||
}, |
||||
getPageURLForShare: function( /* shareButtonData */ ) { |
||||
return window.location.href; |
||||
}, |
||||
getTextForShare: function( /* shareButtonData */ ) { |
||||
return pswp.currItem.title || ''; |
||||
}, |
||||
|
||||
indexIndicatorSep: ' / ' |
||||
|
||||
}, |
||||
_blockControlsTap, |
||||
_blockControlsTapTimeout; |
||||
|
||||
|
||||
|
||||
var _onControlsTap = function(e) { |
||||
if(_blockControlsTap) { |
||||
return true; |
||||
} |
||||
|
||||
|
||||
e = e || window.event; |
||||
|
||||
if(_options.timeToIdle && _options.mouseUsed && !_isIdle) { |
||||
// reset idle timer
|
||||
_onIdleMouseMove(); |
||||
} |
||||
|
||||
|
||||
var target = e.target || e.srcElement, |
||||
uiElement, |
||||
clickedClass = target.className, |
||||
found; |
||||
|
||||
for(var i = 0; i < _uiElements.length; i++) { |
||||
uiElement = _uiElements[i]; |
||||
if(uiElement.onTap && clickedClass.indexOf('pswp__' + uiElement.name ) > -1 ) { |
||||
uiElement.onTap(); |
||||
found = true; |
||||
|
||||
} |
||||
} |
||||
|
||||
if(found) { |
||||
if(e.stopPropagation) { |
||||
e.stopPropagation(); |
||||
} |
||||
_blockControlsTap = true; |
||||
|
||||
// Some versions of Android don't prevent ghost click event
|
||||
// when preventDefault() was called on touchstart and/or touchend.
|
||||
//
|
||||
// This happens on v4.3, 4.2, 4.1,
|
||||
// older versions strangely work correctly,
|
||||
// but just in case we add delay on all of them)
|
||||
var tapDelay = framework.features.isOldAndroid ? 600 : 30; |
||||
_blockControlsTapTimeout = setTimeout(function() { |
||||
_blockControlsTap = false; |
||||
}, tapDelay); |
||||
} |
||||
|
||||
}, |
||||
_fitControlsInViewport = function() { |
||||
return !pswp.likelyTouchDevice || _options.mouseUsed || screen.width > 1200; |
||||
}, |
||||
_togglePswpClass = function(el, cName, add) { |
||||
framework[ (add ? 'add' : 'remove') + 'Class' ](el, 'pswp__' + cName); |
||||
}, |
||||
|
||||
// add class when there is just one item in the gallery
|
||||
// (by default it hides left/right arrows and 1ofX counter)
|
||||
_countNumItems = function() { |
||||
var hasOneSlide = (_options.getNumItemsFn() === 1); |
||||
|
||||
if(hasOneSlide !== _galleryHasOneSlide) { |
||||
_togglePswpClass(_controls, 'ui--one-slide', hasOneSlide); |
||||
_galleryHasOneSlide = hasOneSlide; |
||||
} |
||||
}, |
||||
_toggleShareModalClass = function() { |
||||
_togglePswpClass(_shareModal, 'share-modal--hidden', _shareModalHidden); |
||||
}, |
||||
_toggleShareModal = function() { |
||||
|
||||
_shareModalHidden = !_shareModalHidden; |
||||
|
||||
|
||||
if(!_shareModalHidden) { |
||||
_toggleShareModalClass(); |
||||
setTimeout(function() { |
||||
if(!_shareModalHidden) { |
||||
framework.addClass(_shareModal, 'pswp__share-modal--fade-in'); |
||||
} |
||||
}, 30); |
||||
} else { |
||||
framework.removeClass(_shareModal, 'pswp__share-modal--fade-in'); |
||||
setTimeout(function() { |
||||
if(_shareModalHidden) { |
||||
_toggleShareModalClass(); |
||||
} |
||||
}, 300); |
||||
} |
||||
|
||||
if(!_shareModalHidden) { |
||||
_updateShareURLs(); |
||||
} |
||||
return false; |
||||
}, |
||||
|
||||
_openWindowPopup = function(e) { |
||||
e = e || window.event; |
||||
var target = e.target || e.srcElement; |
||||
|
||||
pswp.shout('shareLinkClick', e, target); |
||||
|
||||
if(!target.href) { |
||||
return false; |
||||
} |
||||
|
||||
if( target.hasAttribute('download') ) { |
||||
return true; |
||||
} |
||||
|
||||
window.open(target.href, 'pswp_share', 'scrollbars=yes,resizable=yes,toolbar=no,'+ |
||||
'location=yes,width=550,height=420,top=100,left=' +
|
||||
(window.screen ? Math.round(screen.width / 2 - 275) : 100) ); |
||||
|
||||
if(!_shareModalHidden) { |
||||
_toggleShareModal(); |
||||
} |
||||
|
||||
return false; |
||||
}, |
||||
_updateShareURLs = function() { |
||||
var shareButtonOut = '', |
||||
shareButtonData, |
||||
shareURL, |
||||
image_url, |
||||
page_url, |
||||
share_text; |
||||
|
||||
for(var i = 0; i < _options.shareButtons.length; i++) { |
||||
shareButtonData = _options.shareButtons[i]; |
||||
|
||||
image_url = _options.getImageURLForShare(shareButtonData); |
||||
page_url = _options.getPageURLForShare(shareButtonData); |
||||
share_text = _options.getTextForShare(shareButtonData); |
||||
|
||||
shareURL = shareButtonData.url.replace('{{url}}', encodeURIComponent(page_url) ) |
||||
.replace('{{image_url}}', encodeURIComponent(image_url) ) |
||||
.replace('{{raw_image_url}}', image_url ) |
||||
.replace('{{text}}', encodeURIComponent(share_text) ); |
||||
|
||||
shareButtonOut += '<a href="' + shareURL + '" target="_blank" '+ |
||||
'class="pswp__share--' + shareButtonData.id + '"' + |
||||
(shareButtonData.download ? 'download' : '') + '>' +
|
||||
shareButtonData.label + '</a>'; |
||||
|
||||
if(_options.parseShareButtonOut) { |
||||
shareButtonOut = _options.parseShareButtonOut(shareButtonData, shareButtonOut); |
||||
} |
||||
} |
||||
_shareModal.children[0].innerHTML = shareButtonOut; |
||||
_shareModal.children[0].onclick = _openWindowPopup; |
||||
|
||||
}, |
||||
_hasCloseClass = function(target) { |
||||
for(var i = 0; i < _options.closeElClasses.length; i++) { |
||||
if( framework.hasClass(target, 'pswp__' + _options.closeElClasses[i]) ) { |
||||
return true; |
||||
} |
||||
} |
||||
}, |
||||
_idleInterval, |
||||
_idleTimer, |
||||
_idleIncrement = 0, |
||||
_onIdleMouseMove = function() { |
||||
clearTimeout(_idleTimer); |
||||
_idleIncrement = 0; |
||||
if(_isIdle) { |
||||
ui.setIdle(false); |
||||
} |
||||
}, |
||||
_onMouseLeaveWindow = function(e) { |
||||
e = e ? e : window.event; |
||||
var from = e.relatedTarget || e.toElement; |
||||
if (!from || from.nodeName === 'HTML') { |
||||
clearTimeout(_idleTimer); |
||||
_idleTimer = setTimeout(function() { |
||||
ui.setIdle(true); |
||||
}, _options.timeToIdleOutside); |
||||
} |
||||
}, |
||||
_setupFullscreenAPI = function() { |
||||
if(_options.fullscreenEl) { |
||||
if(!_fullscrenAPI) { |
||||
_fullscrenAPI = ui.getFullscreenAPI(); |
||||
} |
||||
if(_fullscrenAPI) { |
||||
framework.bind(document, _fullscrenAPI.eventK, ui.updateFullscreen); |
||||
ui.updateFullscreen(); |
||||
framework.addClass(pswp.template, 'pswp--supports-fs'); |
||||
} else { |
||||
framework.removeClass(pswp.template, 'pswp--supports-fs'); |
||||
} |
||||
} |
||||
}, |
||||
_setupLoadingIndicator = function() { |
||||
// Setup loading indicator
|
||||
if(_options.preloaderEl) { |
||||
|
||||
_toggleLoadingIndicator(true); |
||||
|
||||
_listen('beforeChange', function() { |
||||
|
||||
clearTimeout(_loadingIndicatorTimeout); |
||||
|
||||
// display loading indicator with delay
|
||||
_loadingIndicatorTimeout = setTimeout(function() { |
||||
|
||||
if(pswp.currItem && pswp.currItem.loading) { |
||||
|
||||
if( !pswp.allowProgressiveImg() || (pswp.currItem.img && !pswp.currItem.img.naturalWidth) ) { |
||||
// show preloader if progressive loading is not enabled,
|
||||
// or image width is not defined yet (because of slow connection)
|
||||
_toggleLoadingIndicator(false);
|
||||
// items-controller.js function allowProgressiveImg
|
||||
} |
||||
|
||||
} else { |
||||
_toggleLoadingIndicator(true); // hide preloader
|
||||
} |
||||
|
||||
}, _options.loadingIndicatorDelay); |
||||
|
||||
}); |
||||
_listen('imageLoadComplete', function(index, item) { |
||||
if(pswp.currItem === item) { |
||||
_toggleLoadingIndicator(true); |
||||
} |
||||
}); |
||||
|
||||
} |
||||
}, |
||||
_toggleLoadingIndicator = function(hide) { |
||||
if( _loadingIndicatorHidden !== hide ) { |
||||
_togglePswpClass(_loadingIndicator, 'preloader--active', !hide); |
||||
_loadingIndicatorHidden = hide; |
||||
} |
||||
}, |
||||
_applyNavBarGaps = function(item) { |
||||
var gap = item.vGap; |
||||
|
||||
if( _fitControlsInViewport() ) { |
||||
|
||||
var bars = _options.barsSize;
|
||||
if(_options.captionEl && bars.bottom === 'auto') { |
||||
if(!_fakeCaptionContainer) { |
||||
_fakeCaptionContainer = framework.createEl('pswp__caption pswp__caption--fake'); |
||||
_fakeCaptionContainer.appendChild( framework.createEl('pswp__caption__center') ); |
||||
_controls.insertBefore(_fakeCaptionContainer, _captionContainer); |
||||
framework.addClass(_controls, 'pswp__ui--fit'); |
||||
} |
||||
if( _options.addCaptionHTMLFn(item, _fakeCaptionContainer, true) ) { |
||||
|
||||
var captionSize = _fakeCaptionContainer.clientHeight; |
||||
gap.bottom = parseInt(captionSize,10) || 44; |
||||
} else { |
||||
gap.bottom = bars.top; // if no caption, set size of bottom gap to size of top
|
||||
} |
||||
} else { |
||||
gap.bottom = bars.bottom === 'auto' ? 0 : bars.bottom; |
||||
} |
||||
|
||||
// height of top bar is static, no need to calculate it
|
||||
gap.top = bars.top; |
||||
} else { |
||||
gap.top = gap.bottom = 0; |
||||
} |
||||
}, |
||||
_setupIdle = function() { |
||||
// Hide controls when mouse is used
|
||||
if(_options.timeToIdle) { |
||||
_listen('mouseUsed', function() { |
||||
|
||||
framework.bind(document, 'mousemove', _onIdleMouseMove); |
||||
framework.bind(document, 'mouseout', _onMouseLeaveWindow); |
||||
|
||||
_idleInterval = setInterval(function() { |
||||
_idleIncrement++; |
||||
if(_idleIncrement === 2) { |
||||
ui.setIdle(true); |
||||
} |
||||
}, _options.timeToIdle / 2); |
||||
}); |
||||
} |
||||
}, |
||||
_setupHidingControlsDuringGestures = function() { |
||||
|
||||
// Hide controls on vertical drag
|
||||
_listen('onVerticalDrag', function(now) { |
||||
if(_controlsVisible && now < 0.95) { |
||||
ui.hideControls(); |
||||
} else if(!_controlsVisible && now >= 0.95) { |
||||
ui.showControls(); |
||||
} |
||||
}); |
||||
|
||||
// Hide controls when pinching to close
|
||||
var pinchControlsHidden; |
||||
_listen('onPinchClose' , function(now) { |
||||
if(_controlsVisible && now < 0.9) { |
||||
ui.hideControls(); |
||||
pinchControlsHidden = true; |
||||
} else if(pinchControlsHidden && !_controlsVisible && now > 0.9) { |
||||
ui.showControls(); |
||||
} |
||||
}); |
||||
|
||||
_listen('zoomGestureEnded', function() { |
||||
pinchControlsHidden = false; |
||||
if(pinchControlsHidden && !_controlsVisible) { |
||||
ui.showControls(); |
||||
} |
||||
}); |
||||
|
||||
}; |
||||
|
||||
|
||||
|
||||
var _uiElements = [ |
||||
{
|
||||
name: 'caption',
|
||||
option: 'captionEl', |
||||
onInit: function(el) {
|
||||
_captionContainer = el;
|
||||
}
|
||||
}, |
||||
{
|
||||
name: 'share-modal',
|
||||
option: 'shareEl', |
||||
onInit: function(el) {
|
||||
_shareModal = el; |
||||
}, |
||||
onTap: function() { |
||||
_toggleShareModal(); |
||||
}
|
||||
}, |
||||
{
|
||||
name: 'button--share',
|
||||
option: 'shareEl', |
||||
onInit: function(el) {
|
||||
_shareButton = el; |
||||
}, |
||||
onTap: function() { |
||||
_toggleShareModal(); |
||||
}
|
||||
}, |
||||
{
|
||||
name: 'button--zoom',
|
||||
option: 'zoomEl', |
||||
onTap: pswp.toggleDesktopZoom |
||||
}, |
||||
{
|
||||
name: 'counter',
|
||||
option: 'counterEl', |
||||
onInit: function(el) {
|
||||
_indexIndicator = el; |
||||
}
|
||||
}, |
||||
{
|
||||
name: 'button--close',
|
||||
option: 'closeEl', |
||||
onTap: pswp.close |
||||
}, |
||||
{
|
||||
name: 'button--arrow--left',
|
||||
option: 'arrowEl', |
||||
onTap: pswp.prev |
||||
}, |
||||
{
|
||||
name: 'button--arrow--right',
|
||||
option: 'arrowEl', |
||||
onTap: pswp.next |
||||
}, |
||||
{
|
||||
name: 'button--fs',
|
||||
option: 'fullscreenEl', |
||||
onTap: function() {
|
||||
if(_fullscrenAPI.isFullscreen()) { |
||||
_fullscrenAPI.exit(); |
||||
} else { |
||||
_fullscrenAPI.enter(); |
||||
} |
||||
}
|
||||
}, |
||||
{
|
||||
name: 'preloader',
|
||||
option: 'preloaderEl', |
||||
onInit: function(el) {
|
||||
_loadingIndicator = el; |
||||
}
|
||||
} |
||||
|
||||
]; |
||||
|
||||
var _setupUIElements = function() { |
||||
var item, |
||||
classAttr, |
||||
uiElement; |
||||
|
||||
var loopThroughChildElements = function(sChildren) { |
||||
if(!sChildren) { |
||||
return; |
||||
} |
||||
|
||||
var l = sChildren.length; |
||||
for(var i = 0; i < l; i++) { |
||||
item = sChildren[i]; |
||||
classAttr = item.className; |
||||
|
||||
for(var a = 0; a < _uiElements.length; a++) { |
||||
uiElement = _uiElements[a]; |
||||
|
||||
if(classAttr.indexOf('pswp__' + uiElement.name) > -1 ) { |
||||
|
||||
if( _options[uiElement.option] ) { // if element is not disabled from options
|
||||
|
||||
framework.removeClass(item, 'pswp__element--disabled'); |
||||
if(uiElement.onInit) { |
||||
uiElement.onInit(item); |
||||
} |
||||
|
||||
//item.style.display = 'block';
|
||||
} else { |
||||
framework.addClass(item, 'pswp__element--disabled'); |
||||
//item.style.display = 'none';
|
||||
} |
||||
} |
||||
} |
||||
} |
||||
}; |
||||
loopThroughChildElements(_controls.children); |
||||
|
||||
var topBar = framework.getChildByClass(_controls, 'pswp__top-bar'); |
||||
if(topBar) { |
||||
loopThroughChildElements( topBar.children ); |
||||
} |
||||
}; |
||||
|
||||
|
||||
|
||||
|
||||
ui.init = function() { |
||||
|
||||
// extend options
|
||||
framework.extend(pswp.options, _defaultUIOptions, true); |
||||
|
||||
// create local link for fast access
|
||||
_options = pswp.options; |
||||
|
||||
// find pswp__ui element
|
||||
_controls = framework.getChildByClass(pswp.scrollWrap, 'pswp__ui'); |
||||
|
||||
// create local link
|
||||
_listen = pswp.listen; |
||||
|
||||
|
||||
_setupHidingControlsDuringGestures(); |
||||
|
||||
// update controls when slides change
|
||||
_listen('beforeChange', ui.update); |
||||
|
||||
// toggle zoom on double-tap
|
||||
_listen('doubleTap', function(point) { |
||||
var initialZoomLevel = pswp.currItem.initialZoomLevel; |
||||
if(pswp.getZoomLevel() !== initialZoomLevel) { |
||||
pswp.zoomTo(initialZoomLevel, point, 333); |
||||
} else { |
||||
pswp.zoomTo(_options.getDoubleTapZoom(false, pswp.currItem), point, 333); |
||||
} |
||||
}); |
||||
|
||||
// Allow text selection in caption
|
||||
_listen('preventDragEvent', function(e, isDown, preventObj) { |
||||
var t = e.target || e.srcElement; |
||||
if( |
||||
t &&
|
||||
t.className && e.type.indexOf('mouse') > -1 &&
|
||||
( t.className.indexOf('__caption') > 0 || (/(SMALL|STRONG|EM)/i).test(t.tagName) )
|
||||
) { |
||||
preventObj.prevent = false; |
||||
} |
||||
}); |
||||
|
||||
// bind events for UI
|
||||
_listen('bindEvents', function() { |
||||
framework.bind(_controls, 'pswpTap click', _onControlsTap); |
||||
framework.bind(pswp.scrollWrap, 'pswpTap', ui.onGlobalTap); |
||||
|
||||
if(!pswp.likelyTouchDevice) { |
||||
framework.bind(pswp.scrollWrap, 'mouseover', ui.onMouseOver); |
||||
} |
||||
}); |
||||
|
||||
// unbind events for UI
|
||||
_listen('unbindEvents', function() { |
||||
if(!_shareModalHidden) { |
||||
_toggleShareModal(); |
||||
} |
||||
|
||||
if(_idleInterval) { |
||||
clearInterval(_idleInterval); |
||||
} |
||||
framework.unbind(document, 'mouseout', _onMouseLeaveWindow); |
||||
framework.unbind(document, 'mousemove', _onIdleMouseMove); |
||||
framework.unbind(_controls, 'pswpTap click', _onControlsTap); |
||||
framework.unbind(pswp.scrollWrap, 'pswpTap', ui.onGlobalTap); |
||||
framework.unbind(pswp.scrollWrap, 'mouseover', ui.onMouseOver); |
||||
|
||||
if(_fullscrenAPI) { |
||||
framework.unbind(document, _fullscrenAPI.eventK, ui.updateFullscreen); |
||||
if(_fullscrenAPI.isFullscreen()) { |
||||
_options.hideAnimationDuration = 0; |
||||
_fullscrenAPI.exit(); |
||||
} |
||||
_fullscrenAPI = null; |
||||
} |
||||
}); |
||||
|
||||
|
||||
// clean up things when gallery is destroyed
|
||||
_listen('destroy', function() { |
||||
if(_options.captionEl) { |
||||
if(_fakeCaptionContainer) { |
||||
_controls.removeChild(_fakeCaptionContainer); |
||||
} |
||||
framework.removeClass(_captionContainer, 'pswp__caption--empty'); |
||||
} |
||||
|
||||
if(_shareModal) { |
||||
_shareModal.children[0].onclick = null; |
||||
} |
||||
framework.removeClass(_controls, 'pswp__ui--over-close'); |
||||
framework.addClass( _controls, 'pswp__ui--hidden'); |
||||
ui.setIdle(false); |
||||
}); |
||||
|
||||
|
||||
if(!_options.showAnimationDuration) { |
||||
framework.removeClass( _controls, 'pswp__ui--hidden'); |
||||
} |
||||
_listen('initialZoomIn', function() { |
||||
if(_options.showAnimationDuration) { |
||||
framework.removeClass( _controls, 'pswp__ui--hidden'); |
||||
} |
||||
}); |
||||
_listen('initialZoomOut', function() { |
||||
framework.addClass( _controls, 'pswp__ui--hidden'); |
||||
}); |
||||
|
||||
_listen('parseVerticalMargin', _applyNavBarGaps); |
||||
|
||||
_setupUIElements(); |
||||
|
||||
if(_options.shareEl && _shareButton && _shareModal) { |
||||
_shareModalHidden = true; |
||||
} |
||||
|
||||
_countNumItems(); |
||||
|
||||
_setupIdle(); |
||||
|
||||
_setupFullscreenAPI(); |
||||
|
||||
_setupLoadingIndicator(); |
||||
}; |
||||
|
||||
ui.setIdle = function(isIdle) { |
||||
_isIdle = isIdle; |
||||
_togglePswpClass(_controls, 'ui--idle', isIdle); |
||||
}; |
||||
|
||||
ui.update = function() { |
||||
// Don't update UI if it's hidden
|
||||
if(_controlsVisible && pswp.currItem) { |
||||
|
||||
ui.updateIndexIndicator(); |
||||
|
||||
if(_options.captionEl) { |
||||
_options.addCaptionHTMLFn(pswp.currItem, _captionContainer); |
||||
|
||||
_togglePswpClass(_captionContainer, 'caption--empty', !pswp.currItem.title); |
||||
} |
||||
|
||||
_overlayUIUpdated = true; |
||||
|
||||
} else { |
||||
_overlayUIUpdated = false; |
||||
} |
||||
|
||||
if(!_shareModalHidden) { |
||||
_toggleShareModal(); |
||||
} |
||||
|
||||
_countNumItems(); |
||||
}; |
||||
|
||||
ui.updateFullscreen = function(e) { |
||||
|
||||
if(e) { |
||||
// some browsers change window scroll position during the fullscreen
|
||||
// so PhotoSwipe updates it just in case
|
||||
setTimeout(function() { |
||||
pswp.setScrollOffset( 0, framework.getScrollY() ); |
||||
}, 50); |
||||
} |
||||
|
||||
// toogle pswp--fs class on root element
|
||||
framework[ (_fullscrenAPI.isFullscreen() ? 'add' : 'remove') + 'Class' ](pswp.template, 'pswp--fs'); |
||||
}; |
||||
|
||||
ui.updateIndexIndicator = function() { |
||||
if(_options.counterEl) { |
||||
_indexIndicator.innerHTML = (pswp.getCurrentIndex()+1) +
|
||||
_options.indexIndicatorSep +
|
||||
_options.getNumItemsFn(); |
||||
} |
||||
}; |
||||
|
||||
ui.onGlobalTap = function(e) { |
||||
e = e || window.event; |
||||
var target = e.target || e.srcElement; |
||||
|
||||
if(_blockControlsTap) { |
||||
return; |
||||
} |
||||
|
||||
if(e.detail && e.detail.pointerType === 'mouse') { |
||||
|
||||
// close gallery if clicked outside of the image
|
||||
if(_hasCloseClass(target)) { |
||||
pswp.close(); |
||||
return; |
||||
} |
||||
|
||||
if(framework.hasClass(target, 'pswp__img')) { |
||||
if(pswp.getZoomLevel() === 1 && pswp.getZoomLevel() <= pswp.currItem.fitRatio) { |
||||
if(_options.clickToCloseNonZoomable) { |
||||
pswp.close(); |
||||
} |
||||
} else { |
||||
pswp.toggleDesktopZoom(e.detail.releasePoint); |
||||
} |
||||
} |
||||
|
||||
} else { |
||||
|
||||
// tap anywhere (except buttons) to toggle visibility of controls
|
||||
if(_options.tapToToggleControls) { |
||||
if(_controlsVisible) { |
||||
ui.hideControls(); |
||||
} else { |
||||
ui.showControls(); |
||||
} |
||||
} |
||||
|
||||
// tap to close gallery
|
||||
if(_options.tapToClose && (framework.hasClass(target, 'pswp__img') || _hasCloseClass(target)) ) { |
||||
pswp.close(); |
||||
return; |
||||
} |
||||
|
||||
} |
||||
}; |
||||
ui.onMouseOver = function(e) { |
||||
e = e || window.event; |
||||
var target = e.target || e.srcElement; |
||||
|
||||
// add class when mouse is over an element that should close the gallery
|
||||
_togglePswpClass(_controls, 'ui--over-close', _hasCloseClass(target)); |
||||
}; |
||||
|
||||
ui.hideControls = function() { |
||||
framework.addClass(_controls,'pswp__ui--hidden'); |
||||
_controlsVisible = false; |
||||
}; |
||||
|
||||
ui.showControls = function() { |
||||
_controlsVisible = true; |
||||
if(!_overlayUIUpdated) { |
||||
ui.update(); |
||||
} |
||||
framework.removeClass(_controls,'pswp__ui--hidden'); |
||||
}; |
||||
|
||||
ui.supportsFullscreen = function() { |
||||
var d = document; |
||||
return !!(d.exitFullscreen || d.mozCancelFullScreen || d.webkitExitFullscreen || d.msExitFullscreen); |
||||
}; |
||||
|
||||
ui.getFullscreenAPI = function() { |
||||
var dE = document.documentElement, |
||||
api, |
||||
tF = 'fullscreenchange'; |
||||
|
||||
if (dE.requestFullscreen) { |
||||
api = { |
||||
enterK: 'requestFullscreen', |
||||
exitK: 'exitFullscreen', |
||||
elementK: 'fullscreenElement', |
||||
eventK: tF |
||||
}; |
||||
|
||||
} else if(dE.mozRequestFullScreen ) { |
||||
api = { |
||||
enterK: 'mozRequestFullScreen', |
||||
exitK: 'mozCancelFullScreen', |
||||
elementK: 'mozFullScreenElement', |
||||
eventK: 'moz' + tF |
||||
}; |
||||
|
||||
|
||||
|
||||
} else if(dE.webkitRequestFullscreen) { |
||||
api = { |
||||
enterK: 'webkitRequestFullscreen', |
||||
exitK: 'webkitExitFullscreen', |
||||
elementK: 'webkitFullscreenElement', |
||||
eventK: 'webkit' + tF |
||||
}; |
||||
|
||||
} else if(dE.msRequestFullscreen) { |
||||
api = { |
||||
enterK: 'msRequestFullscreen', |
||||
exitK: 'msExitFullscreen', |
||||
elementK: 'msFullscreenElement', |
||||
eventK: 'MSFullscreenChange' |
||||
}; |
||||
} |
||||
|
||||
if(api) { |
||||
api.enter = function() {
|
||||
// disable close-on-scroll in fullscreen
|
||||
_initalCloseOnScrollValue = _options.closeOnScroll;
|
||||
_options.closeOnScroll = false;
|
||||
|
||||
if(this.enterK === 'webkitRequestFullscreen') { |
||||
pswp.template[this.enterK]( Element.ALLOW_KEYBOARD_INPUT ); |
||||
} else { |
||||
return pswp.template[this.enterK]();
|
||||
} |
||||
}; |
||||
api.exit = function() {
|
||||
_options.closeOnScroll = _initalCloseOnScrollValue; |
||||
|
||||
return document[this.exitK]();
|
||||
|
||||
}; |
||||
api.isFullscreen = function() { return document[this.elementK]; }; |
||||
} |
||||
|
||||
return api; |
||||
}; |
||||
|
||||
|
||||
|
||||
}; |
||||
return PhotoSwipeUI_Default; |
||||
|
||||
|
||||
}); |
||||
File diff suppressed because one or more lines are too long
@ -0,0 +1,180 @@
|
||||
/*! PhotoSwipe main CSS by Dmitry Semenov | photoswipe.com | MIT license */ |
||||
/* |
||||
Styles for basic PhotoSwipe functionality (sliding area, open/close transitions) |
||||
*/ |
||||
/* pswp = photoswipe */ |
||||
.pswp { |
||||
display: none; |
||||
position: absolute; |
||||
width: 100%; |
||||
height: 100%; |
||||
left: 0; |
||||
top: 0; |
||||
overflow: hidden; |
||||
-ms-touch-action: none; |
||||
touch-action: none; |
||||
z-index: 1500; |
||||
-webkit-text-size-adjust: 100%; |
||||
/* create separate layer, to avoid paint on window.onscroll in webkit/blink */ |
||||
-webkit-backface-visibility: hidden; |
||||
outline: none; } |
||||
.pswp * { |
||||
-webkit-box-sizing: border-box; |
||||
box-sizing: border-box; } |
||||
.pswp img { |
||||
max-width: none; } |
||||
|
||||
/* style is added when JS option showHideOpacity is set to true */ |
||||
.pswp--animate_opacity { |
||||
/* 0.001, because opacity:0 doesn't trigger Paint action, which causes lag at start of transition */ |
||||
opacity: 0.001; |
||||
will-change: opacity; |
||||
/* for open/close transition */ |
||||
-webkit-transition: opacity 333ms cubic-bezier(0.4, 0, 0.22, 1); |
||||
transition: opacity 333ms cubic-bezier(0.4, 0, 0.22, 1); } |
||||
|
||||
.pswp--open { |
||||
display: block; } |
||||
|
||||
.pswp--zoom-allowed .pswp__img { |
||||
/* autoprefixer: off */ |
||||
cursor: -webkit-zoom-in; |
||||
cursor: -moz-zoom-in; |
||||
cursor: zoom-in; } |
||||
|
||||
.pswp--zoomed-in .pswp__img { |
||||
/* autoprefixer: off */ |
||||
cursor: -webkit-grab; |
||||
cursor: -moz-grab; |
||||
cursor: grab; } |
||||
|
||||
.pswp--dragging .pswp__img { |
||||
/* autoprefixer: off */ |
||||
cursor: -webkit-grabbing; |
||||
cursor: -moz-grabbing; |
||||
cursor: grabbing; } |
||||
|
||||
/* |
||||
Background is added as a separate element. |
||||
As animating opacity is much faster than animating rgba() background-color. |
||||
*/ |
||||
.pswp__bg { |
||||
position: absolute; |
||||
left: 0; |
||||
top: 0; |
||||
width: 100%; |
||||
height: 100%; |
||||
background: #000; |
||||
opacity: 0; |
||||
-webkit-backface-visibility: hidden; |
||||
will-change: opacity; } |
||||
|
||||
.pswp__scroll-wrap { |
||||
position: absolute; |
||||
left: 0; |
||||
top: 0; |
||||
width: 100%; |
||||
height: 100%; |
||||
overflow: hidden; } |
||||
|
||||
.pswp__container, |
||||
.pswp__zoom-wrap { |
||||
-ms-touch-action: none; |
||||
touch-action: none; |
||||
position: absolute; |
||||
left: 0; |
||||
right: 0; |
||||
top: 0; |
||||
bottom: 0; } |
||||
|
||||
/* Prevent selection and tap highlights */ |
||||
.pswp__container, |
||||
.pswp__img { |
||||
-webkit-user-select: none; |
||||
-moz-user-select: none; |
||||
-ms-user-select: none; |
||||
user-select: none; |
||||
-webkit-tap-highlight-color: transparent; |
||||
-webkit-touch-callout: none; } |
||||
|
||||
.pswp__zoom-wrap { |
||||
position: absolute; |
||||
width: 100%; |
||||
-webkit-transform-origin: left top; |
||||
-ms-transform-origin: left top; |
||||
transform-origin: left top; |
||||
/* for open/close transition */ |
||||
-webkit-transition: -webkit-transform 333ms cubic-bezier(0.4, 0, 0.22, 1); |
||||
transition: transform 333ms cubic-bezier(0.4, 0, 0.22, 1); } |
||||
|
||||
.pswp__bg { |
||||
will-change: opacity; |
||||
/* for open/close transition */ |
||||
-webkit-transition: opacity 333ms cubic-bezier(0.4, 0, 0.22, 1); |
||||
transition: opacity 333ms cubic-bezier(0.4, 0, 0.22, 1); } |
||||
|
||||
.pswp--animated-in .pswp__bg, |
||||
.pswp--animated-in .pswp__zoom-wrap { |
||||
-webkit-transition: none; |
||||
transition: none; } |
||||
|
||||
.pswp__container, |
||||
.pswp__zoom-wrap { |
||||
-webkit-backface-visibility: hidden; |
||||
will-change: transform; } |
||||
|
||||
.pswp__item { |
||||
position: absolute; |
||||
left: 0; |
||||
right: 0; |
||||
top: 0; |
||||
bottom: 0; |
||||
overflow: hidden; } |
||||
|
||||
.pswp__img { |
||||
position: absolute; |
||||
width: auto; |
||||
height: auto; |
||||
top: 0; |
||||
left: 0; |
||||
-webkit-transition: opacity 0.15s; |
||||
transition: opacity 0.15s; } |
||||
|
||||
/* |
||||
stretched thumbnail or div placeholder element (see below) |
||||
style is added to avoid flickering in webkit/blink when layers overlap |
||||
*/ |
||||
.pswp__img--placeholder { |
||||
-webkit-backface-visibility: hidden; } |
||||
|
||||
/* |
||||
div element that matches size of large image |
||||
large image loads on top of it |
||||
*/ |
||||
.pswp__img--placeholder--blank { |
||||
background: #222; } |
||||
|
||||
.pswp--ie .pswp__img { |
||||
width: 100% !important; |
||||
height: auto !important; |
||||
left: 0; |
||||
top: 0; } |
||||
|
||||
/* |
||||
Error message appears when image is not loaded |
||||
(JS option errorMsg controls markup) |
||||
*/ |
||||
.pswp__error-msg { |
||||
position: absolute; |
||||
left: 0; |
||||
top: 50%; |
||||
width: 100%; |
||||
text-align: center; |
||||
font-size: 14px; |
||||
line-height: 16px; |
||||
margin-top: -8px; |
||||
color: #CCC; } |
||||
|
||||
.pswp__error-msg a { |
||||
color: #CCC; |
||||
text-decoration: underline; } |
||||
File diff suppressed because it is too large
Load Diff
File diff suppressed because one or more lines are too long
@ -0,0 +1,227 @@
|
||||
html, body, div, object, iframe, h1, h2, h3, h4, h5, h6, p, blockquote, ol, ul, li, form, fieldset, legend, label, table, header, footer, nav, section, figure { |
||||
margin: 0; |
||||
padding: 0; |
||||
} |
||||
|
||||
body { |
||||
background: #EEE; |
||||
font-family: "myriad-pro","Myriad Pro","Helvetica Neue",Helvetica,Arial,sans-serif; |
||||
font-size: 18px; |
||||
line-height: 26px; |
||||
color: #282B30; |
||||
|
||||
text-rendering: optimizeLegibility; |
||||
-webkit-font-smoothing: antialiased; |
||||
-moz-osx-font-smoothing: grayscale; |
||||
-moz-font-feature-settings: "liga", "kern"; |
||||
} |
||||
|
||||
figure { |
||||
display: block; |
||||
max-width: 100%; |
||||
} |
||||
|
||||
a { |
||||
color: #3169B3; |
||||
} |
||||
a:hover { |
||||
color: #C00; |
||||
} |
||||
|
||||
img { |
||||
width: auto; |
||||
max-width: 100%; |
||||
height: auto; |
||||
border: 0; |
||||
} |
||||
|
||||
p { |
||||
margin: 0 0 12px; |
||||
} |
||||
ul { |
||||
list-style: disc; |
||||
} |
||||
ul, ol { |
||||
padding: 0; |
||||
margin: 0 0 12px 25px; |
||||
} |
||||
li { |
||||
margin: 0 0 12px 0; |
||||
} |
||||
h1, h2, h3, h4, h5, h6 { |
||||
margin: 0; |
||||
font-weight: normal; |
||||
} |
||||
|
||||
h1 { |
||||
font-size: 48px; |
||||
line-height: 1; |
||||
margin: 1em; |
||||
font-weight: 600; |
||||
text-align: center; |
||||
} |
||||
|
||||
h1 a { |
||||
text-decoration: none; |
||||
color: #282B30; |
||||
} |
||||
|
||||
strong { |
||||
font-weight: 600; |
||||
} |
||||
|
||||
figcaption { |
||||
display:none; |
||||
} |
||||
|
||||
.container { |
||||
width: 100%; |
||||
} |
||||
|
||||
.content { |
||||
max-width: 1024px; |
||||
margin: 24px auto; |
||||
padding:2em 0; |
||||
} |
||||
|
||||
.menu { |
||||
width: 100%; |
||||
background: #F7F7F7 none repeat scroll 0% 0%; |
||||
padding: 30px 0px; |
||||
} |
||||
|
||||
.menu ul { |
||||
max-width: 1024px; |
||||
margin: 0 auto; |
||||
} |
||||
|
||||
.menu li { |
||||
display: inline; |
||||
list-style: none; /* pour enlever les puces sur IE7 */ |
||||
margin: 10px; |
||||
} |
||||
|
||||
.album-list { |
||||
width: 100%; |
||||
padding: 2em 0; |
||||
margin: auto; |
||||
background: #EEE none repeat scroll 0% 0%; |
||||
float:left; |
||||
} |
||||
|
||||
.thumbnail { |
||||
text-align: center; |
||||
float:left; |
||||
} |
||||
.thumbnail img { |
||||
max-width: 85%; |
||||
height: auto; |
||||
margin: 0 2px 5px 2px; |
||||
border: 10px solid #fff; |
||||
border-radius: 2px 2px 2px 2px; |
||||
box-shadow: 0 0 3px #B1B1B1; |
||||
padding: 0 0 2px; |
||||
} |
||||
.thumbnail img:hover { |
||||
box-shadow: 0 0 5px #818181; |
||||
} |
||||
|
||||
.album_title { |
||||
display: block; |
||||
} |
||||
|
||||
.video { |
||||
width: 100%%; |
||||
margin: 0 0 24px 0; |
||||
} |
||||
.video__container { |
||||
position: relative; |
||||
width: 100%; |
||||
padding-bottom: 56.25%; |
||||
padding-top: 35px; |
||||
height: 0; |
||||
overflow: hidden; |
||||
} |
||||
.video__container iframe { |
||||
position: absolute; |
||||
top:0; |
||||
left: 0; |
||||
width: 100%; |
||||
height: 100%; |
||||
} |
||||
.video-desc { |
||||
width: 100%; |
||||
max-width: 740px; |
||||
margin: 12px auto; |
||||
} |
||||
|
||||
.additionnal-infos { |
||||
width: 100%; |
||||
height: auto; |
||||
float: left; |
||||
} |
||||
|
||||
.gallery { |
||||
width: 100%; |
||||
height: auto; |
||||
float: left; |
||||
padding-top:1em; |
||||
} |
||||
.gallery a { |
||||
line-height: 0; |
||||
} |
||||
|
||||
.gallery__img--main a{ |
||||
width: auto; |
||||
height: auto; |
||||
} |
||||
|
||||
.menu-img, |
||||
.gallery__img--secondary { |
||||
width: 25%; |
||||
float:left; |
||||
} |
||||
|
||||
.gallery__title { |
||||
line-height: 14px; |
||||
font-size: 14px; |
||||
opacity: 0.8; |
||||
margin-top: 5px; |
||||
width: 100%; |
||||
float: left; |
||||
} |
||||
|
||||
footer { |
||||
width:100%; |
||||
float:left; |
||||
} |
||||
|
||||
footer p { |
||||
text-align: center; |
||||
} |
||||
|
||||
@media screen and (max-width: 700px) { |
||||
h1 { |
||||
font-size: 40px; |
||||
} |
||||
.menu-img, |
||||
.gallery__img--secondary { |
||||
width: 33%; |
||||
} |
||||
} |
||||
|
||||
|
||||
@media screen and (max-width: 500px) { |
||||
.menu-img, |
||||
.gallery__img--secondary { |
||||
width: 50%; |
||||
} |
||||
} |
||||
|
||||
|
||||
@media screen and (max-width: 300px) { |
||||
.menu-img, |
||||
.gallery__img--secondary { |
||||
width: 100%; |
||||
} |
||||
} |
||||
File diff suppressed because one or more lines are too long
@ -0,0 +1,401 @@
|
||||
<!doctype html> |
||||
<html lang="en"> |
||||
<head> |
||||
{% block head %} |
||||
<meta charset="utf-8"> |
||||
<meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1"> |
||||
|
||||
<title>{{ album.title }}</title> |
||||
<meta name="description" content=""> |
||||
<meta name="author" content="{{ album.author }}"> |
||||
<meta name="viewport" content="width=device-width"> |
||||
|
||||
<link rel="stylesheet" href="//fonts.googleapis.com/css?family=PT+Sans"> |
||||
<link rel="stylesheet" href="{{ theme.url }}/styles.min.css"> |
||||
|
||||
<!-- Core JS file --> |
||||
<script src="{{ theme.url }}/photoswipe.min.js"></script> |
||||
|
||||
<!-- UI JS file --> |
||||
<script src="{{ theme.url }}/photoswipe-ui-default.min.js"></script> |
||||
<script src="{{ theme.url }}/echo/echo.min.js"></script> |
||||
|
||||
<!--[if lt IE 9]> |
||||
<script src="//cdnjs.cloudflare.com/ajax/libs/html5shiv/3.7.2/html5shiv.min.js"></script> |
||||
<![endif]--> |
||||
{% endblock %} |
||||
</head> |
||||
<body> |
||||
<div class="container"> |
||||
<h1><a href="{{ album.index_url }}">{{ index_title }}</a></h1> |
||||
|
||||
{% if settings.links %} |
||||
<div class="menu"> |
||||
<ul> |
||||
{% for title, link in settings.links %} |
||||
<li><a href="{{ link }}">{{ title }}</a></li> |
||||
{% endfor %} |
||||
</ul> |
||||
</div> |
||||
{% endif %} |
||||
|
||||
{% block content %} |
||||
|
||||
<div class="content"> |
||||
{% if album.breadcrumb %} |
||||
<div class="breadcrumb"> |
||||
<h2> |
||||
{% for url, title in album.breadcrumb %} |
||||
<a href="{{ url }}">{{ title }}</a>{% if not loop.last %} » {% endif %} |
||||
{% endfor -%} |
||||
</h2> |
||||
</div> |
||||
{% endif %} |
||||
|
||||
{% if album.albums %} |
||||
<div class="album-list"> |
||||
{% for alb in album.albums %} |
||||
<div class="menu-img thumbnail"> |
||||
<a href="{{ alb.url }}"> |
||||
<img src="{{ alb.thumbnail }}" class="album_thumb" alt="{{ alb.title }}" title="{{ alb.title }}" /></a> |
||||
<span class="album_title">{{ alb.title }}</span> |
||||
</div> |
||||
{% endfor %} |
||||
</div> |
||||
{% endif %} |
||||
|
||||
{% if album.zip %} |
||||
<div class="additionnal-infos"> |
||||
<p><a href="{{ album.zip }}" title="Download a zip archive with all images">Download ZIP</a></p> |
||||
</div> |
||||
{% endif %} |
||||
|
||||
{% if album.medias %} |
||||
<div class="gallery" itemscope itemtype="http://schema.org/ImageGallery"> |
||||
{% for media in album.medias %} |
||||
{% set media_title = media.title if media.title else media.filename %} |
||||
{% if media.type == "image" %} |
||||
{% if loop.first %} |
||||
<figure class="gallery__img--main thumbnail" itemprop="associatedMedia" itemscope itemtype="http://schema.org/ImageObject"> |
||||
<a href="{{ media.filename }}" itemprop="contentUrl" data-size="{{media.size.width}}x{{media.size.height}}"> |
||||
<img src="{{ theme.url }}/echo/blank.gif" data-echo="{{ media.filename }}" alt="{{ media.filename }}" itemprop="thumbnail" title="{{ media.exif.datetime }}" /> |
||||
</a> |
||||
<figcaption itemprop="caption description">{{ media_title }} - {{ media.exif.datetime }}</figcaption> |
||||
</figure> |
||||
{% else %} |
||||
<figure class="gallery__img--secondary thumbnail" itemprop="associatedMedia" itemscope itemtype="http://schema.org/ImageObject"> |
||||
<a href="{{ media.filename }}" itemprop="contentUrl" data-size="{{media.size.width}}x{{media.size.height}}"> |
||||
<img src="{{ theme.url }}/echo/blank.gif" data-echo="{{ media.thumbnail }}" alt="{{ media.filename }}" itemprop="thumbnail" title="{{ media.exif.datetime }}" /> |
||||
</a> |
||||
<figcaption itemprop="caption description">{{ media_title }} - {{ media.exif.datetime }}</figcaption> |
||||
</figure> |
||||
{% endif %} |
||||
{% endif %} |
||||
{% endfor %} |
||||
</div> |
||||
{% endif %} |
||||
|
||||
{% if album.description %} |
||||
<div id="description" class="row"> |
||||
{{ album.description }} |
||||
</div> |
||||
{% endif %} |
||||
|
||||
</div> |
||||
{% endblock %} |
||||
|
||||
{% block footer %} |
||||
<footer> |
||||
<p>{% if album.author %}© {{ album.author }} - {% endif %} |
||||
Generated by <a href="{{ sigal_link }}">sigal</a> |
||||
</p> |
||||
</footer> |
||||
{% if album.medias %} |
||||
<!-- Root element of PhotoSwipe. Must have class pswp. --> |
||||
<div class="pswp" tabindex="-1" role="dialog" aria-hidden="true"> |
||||
|
||||
<!-- Background of PhotoSwipe. |
||||
It's a separate element as animating opacity is faster than rgba(). --> |
||||
<div class="pswp__bg"></div> |
||||
|
||||
<!-- Slides wrapper with overflow:hidden. --> |
||||
<div class="pswp__scroll-wrap"> |
||||
|
||||
<!-- Container that holds slides. |
||||
PhotoSwipe keeps only 3 of them in the DOM to save memory. |
||||
Don't modify these 3 pswp__item elements, data is added later on. --> |
||||
<div class="pswp__container"> |
||||
<div class="pswp__item"></div> |
||||
<div class="pswp__item"></div> |
||||
<div class="pswp__item"></div> |
||||
</div> |
||||
|
||||
<!-- Default (PhotoSwipeUI_Default) interface on top of sliding area. Can be changed. --> |
||||
<div class="pswp__ui pswp__ui--hidden"> |
||||
|
||||
<div class="pswp__top-bar"> |
||||
|
||||
<!-- Controls are self-explanatory. Order can be changed. --> |
||||
|
||||
<div class="pswp__counter"></div> |
||||
|
||||
<button class="pswp__button pswp__button--close" title="Close (Esc)"></button> |
||||
|
||||
<button class="pswp__button pswp__button--share" title="Share"></button> |
||||
|
||||
<button class="pswp__button pswp__button--fs" title="Toggle fullscreen"></button> |
||||
|
||||
<button class="pswp__button pswp__button--zoom" title="Zoom in/out"></button> |
||||
|
||||
<!-- Preloader demo http://codepen.io/dimsemenov/pen/yyBWoR --> |
||||
<!-- element will get class pswp__preloader-active when preloader is running --> |
||||
<div class="pswp__preloader"> |
||||
<div class="pswp__preloader__icn"> |
||||
<div class="pswp__preloader__cut"> |
||||
<div class="pswp__preloader__donut"></div> |
||||
</div> |
||||
</div> |
||||
</div> |
||||
</div> |
||||
|
||||
<div class="pswp__share-modal pswp__share-modal--hidden pswp__single-tap"> |
||||
<div class="pswp__share-tooltip"></div> |
||||
</div> |
||||
|
||||
<button class="pswp__button pswp__button--arrow--left" title="Previous (arrow left)"> |
||||
</button> |
||||
|
||||
<button class="pswp__button pswp__button--arrow--right" title="Next (arrow right)"> |
||||
</button> |
||||
|
||||
<div class="pswp__caption"> |
||||
<div class="pswp__caption__center"></div> |
||||
</div> |
||||
|
||||
</div> |
||||
|
||||
</div> |
||||
|
||||
</div> |
||||
<script> |
||||
echo.init({ |
||||
offset: 100, |
||||
throttle: 250, |
||||
unload: false |
||||
}); |
||||
|
||||
var initPhotoSwipeFromDOM = function(gallerySelector) { |
||||
|
||||
// parse slide data (url, title, size ...) from DOM elements |
||||
// (children of gallerySelector) |
||||
var parseThumbnailElements = function(el) { |
||||
var thumbElements = el.childNodes, |
||||
numNodes = thumbElements.length, |
||||
items = [], |
||||
figureEl, |
||||
linkEl, |
||||
size, |
||||
item; |
||||
|
||||
for(var i = 0; i < numNodes; i++) { |
||||
|
||||
figureEl = thumbElements[i]; // <figure> element |
||||
|
||||
// include only element nodes |
||||
if(figureEl.nodeType !== 1) { |
||||
continue; |
||||
} |
||||
|
||||
linkEl = figureEl.children[0]; // <a> element |
||||
|
||||
size = linkEl.getAttribute('data-size').split('x'); |
||||
|
||||
// create slide object |
||||
item = { |
||||
src: linkEl.getAttribute('href'), |
||||
w: parseInt(size[0], 10), |
||||
h: parseInt(size[1], 10) |
||||
}; |
||||
|
||||
|
||||
|
||||
if(figureEl.children.length > 1) { |
||||
// <figcaption> content |
||||
item.title = figureEl.children[1].innerHTML; |
||||
} |
||||
|
||||
if(linkEl.children.length > 0) { |
||||
// <img> thumbnail element, retrieving thumbnail url |
||||
item.msrc = linkEl.children[0].getAttribute('data-echo'); |
||||
} |
||||
|
||||
item.el = figureEl; // save link to element for getThumbBoundsFn |
||||
items.push(item); |
||||
} |
||||
|
||||
return items; |
||||
}; |
||||
|
||||
// find nearest parent element |
||||
var closest = function closest(el, fn) { |
||||
return el && ( fn(el) ? el : closest(el.parentNode, fn) ); |
||||
}; |
||||
|
||||
// triggers when user clicks on thumbnail |
||||
var onThumbnailsClick = function(e) { |
||||
e = e || window.event; |
||||
e.preventDefault ? e.preventDefault() : e.returnValue = false; |
||||
|
||||
var eTarget = e.target || e.srcElement; |
||||
|
||||
// find root element of slide |
||||
var clickedListItem = closest(eTarget, function(el) { |
||||
return (el.tagName && el.tagName.toUpperCase() === 'FIGURE'); |
||||
}); |
||||
|
||||
if(!clickedListItem) { |
||||
return; |
||||
} |
||||
|
||||
// find index of clicked item by looping through all child nodes |
||||
// alternatively, you may define index via data- attribute |
||||
var clickedGallery = clickedListItem.parentNode, |
||||
childNodes = clickedListItem.parentNode.childNodes, |
||||
numChildNodes = childNodes.length, |
||||
nodeIndex = 0, |
||||
index; |
||||
|
||||
for (var i = 0; i < numChildNodes; i++) { |
||||
if(childNodes[i].nodeType !== 1) { |
||||
continue; |
||||
} |
||||
|
||||
if(childNodes[i] === clickedListItem) { |
||||
index = nodeIndex; |
||||
break; |
||||
} |
||||
nodeIndex++; |
||||
} |
||||
|
||||
|
||||
|
||||
if(index >= 0) { |
||||
// open PhotoSwipe if valid index found |
||||
openPhotoSwipe( index, clickedGallery ); |
||||
} |
||||
return false; |
||||
}; |
||||
|
||||
// parse picture index and gallery index from URL (#&pid=1&gid=2) |
||||
var photoswipeParseHash = function() { |
||||
var hash = window.location.hash.substring(1), |
||||
params = {}; |
||||
|
||||
if(hash.length < 5) { |
||||
return params; |
||||
} |
||||
|
||||
var vars = hash.split('&'); |
||||
for (var i = 0; i < vars.length; i++) { |
||||
if(!vars[i]) { |
||||
continue; |
||||
} |
||||
var pair = vars[i].split('='); |
||||
if(pair.length < 2) { |
||||
continue; |
||||
} |
||||
params[pair[0]] = pair[1]; |
||||
} |
||||
|
||||
if(params.gid) { |
||||
params.gid = parseInt(params.gid, 10); |
||||
} |
||||
|
||||
return params; |
||||
}; |
||||
|
||||
var openPhotoSwipe = function(index, galleryElement, disableAnimation, fromURL) { |
||||
var pswpElement = document.querySelectorAll('.pswp')[0], |
||||
gallery, |
||||
options, |
||||
items; |
||||
|
||||
items = parseThumbnailElements(galleryElement); |
||||
|
||||
// define options (if needed) |
||||
options = { |
||||
|
||||
// define gallery index (for URL) |
||||
galleryUID: galleryElement.getAttribute('data-pswp-uid'), |
||||
|
||||
getThumbBoundsFn: function(index) { |
||||
// See Options -> getThumbBoundsFn section of documentation for more info |
||||
var thumbnail = items[index].el.getElementsByTagName('img')[0], // find thumbnail |
||||
pageYScroll = window.pageYOffset || document.documentElement.scrollTop, |
||||
rect = thumbnail.getBoundingClientRect(); |
||||
|
||||
return {x:rect.left, y:rect.top + pageYScroll, w:rect.width}; |
||||
} |
||||
|
||||
}; |
||||
|
||||
// PhotoSwipe opened from URL |
||||
if(fromURL) { |
||||
if(options.galleryPIDs) { |
||||
// parse real index when custom PIDs are used |
||||
// http://photoswipe.com/documentation/faq.html#custom-pid-in-url |
||||
for(var j = 0; j < items.length; j++) { |
||||
if(items[j].pid == index) { |
||||
options.index = j; |
||||
break; |
||||
} |
||||
} |
||||
} else { |
||||
// in URL indexes start from 1 |
||||
options.index = parseInt(index, 10) - 1; |
||||
} |
||||
} else { |
||||
options.index = parseInt(index, 10); |
||||
} |
||||
|
||||
// exit if index not found |
||||
if( isNaN(options.index) ) { |
||||
return; |
||||
} |
||||
|
||||
if(disableAnimation) { |
||||
options.showAnimationDuration = 0; |
||||
} |
||||
|
||||
// Pass data to PhotoSwipe and initialize it |
||||
gallery = new PhotoSwipe( pswpElement, PhotoSwipeUI_Default, items, options); |
||||
gallery.init(); |
||||
}; |
||||
|
||||
// loop through all gallery elements and bind events |
||||
var galleryElements = document.querySelectorAll( gallerySelector ); |
||||
|
||||
for(var i = 0, l = galleryElements.length; i < l; i++) { |
||||
galleryElements[i].setAttribute('data-pswp-uid', i+1); |
||||
galleryElements[i].onclick = onThumbnailsClick; |
||||
} |
||||
|
||||
// Parse URL and open gallery if it contains #&pid=3&gid=1 |
||||
var hashData = photoswipeParseHash(); |
||||
if(hashData.pid && hashData.gid) { |
||||
openPhotoSwipe( hashData.pid , galleryElements[ hashData.gid - 1 ], true, true ); |
||||
} |
||||
}; |
||||
|
||||
// execute above function |
||||
initPhotoSwipeFromDOM('.gallery'); |
||||
|
||||
</script> |
||||
|
||||
{% endif %} |
||||
{% endblock %} |
||||
</div> |
||||
{% include 'analytics.html' %} |
||||
{% include 'piwik.html' %} |
||||
</body> |
||||
</html> |
||||
Loading…
Reference in new issue