2015-01-21 16:03:20 +01:00
/ * ! r e s p i m a g e - v 1 . 2 . 1 - 2 0 1 4 - 1 2 - 2 3
2014-11-05 21:25:18 +01:00
Licensed MIT * /
! function ( window , document , undefined ) {
2014-07-17 16:45:12 +02:00
"use strict" ;
2014-11-05 21:25:18 +01:00
function trim ( str ) {
return str . trim ? str . trim ( ) : str . replace ( /^\s+|\s+$/g , "" ) ;
2014-07-17 16:45:12 +02:00
}
2014-11-25 23:27:31 +01:00
function updateMetrics ( ) {
var dprM ;
2015-01-21 16:03:20 +01:00
isVwDirty = ! 1 , DPR = window . devicePixelRatio , cssCache = { } , sizeLengthCache = { } ,
dprM = ( DPR || 1 ) * cfg . xQuant , cfg . uT || ( cfg . maxX = Math . max ( 1.3 , cfg . maxX ) , dprM = Math . min ( dprM , cfg . maxX ) ,
ri . DPR = dprM ) , units . width = Math . max ( window . innerWidth || 0 , docElem . clientWidth ) ,
units . height = Math . max ( window . innerHeight || 0 , docElem . clientHeight ) , units . vw = units . width / 100 ,
units . vh = units . height / 100 , units . em = ri . getEmValue ( ) , units . rem = units . em ,
lazyFactor = cfg . lazyFactor / 2 , lazyFactor = lazyFactor * dprM + lazyFactor , substractCurRes = . 1 * dprM ,
lowTreshHold = . 5 + . 2 * dprM , partialLowTreshHold = . 5 + . 25 * dprM , tMemory = dprM + 1.3 ,
( isLandscape = units . width > units . height ) || ( lazyFactor *= . 9 ) , supportAbort && ( lazyFactor *= . 9 ) ;
2014-11-05 21:25:18 +01:00
}
function chooseLowRes ( lowRes , diff , dpr ) {
2015-01-21 16:03:20 +01:00
var add = diff * Math . pow ( lowRes , 2 ) ;
return isLandscape || ( add /= 1.3 ) , lowRes += add , lowRes > dpr ;
2014-11-05 21:25:18 +01:00
}
function inView ( el ) {
if ( ! el . getBoundingClientRect ) return ! 0 ;
var bottom , right , left , top , rect = el . getBoundingClientRect ( ) ;
2014-11-25 23:27:31 +01:00
return ! ! ( ( bottom = rect . bottom ) >= - 9 && ( top = rect . top ) <= units . height + 9 && ( right = rect . right ) >= - 9 && ( left = rect . left ) <= units . height + 9 && ( bottom || right || left || top ) ) ;
2014-11-05 21:25:18 +01:00
}
function applyBestCandidate ( img ) {
var srcSetCandidates , matchingSet = ri . getSet ( img ) , evaluated = ! 1 ;
"pending" != matchingSet && ( evaluated = ! 0 , matchingSet && ( srcSetCandidates = ri . setRes ( matchingSet ) ,
evaluated = ri . applySetCandidate ( srcSetCandidates , img ) ) ) , img [ ri . ns ] . evaled = evaluated ;
}
function ascendingSort ( a , b ) {
return a . res - b . res ;
}
function setSrcToCur ( img , src , set ) {
var candidate ;
return ! set && src && ( set = img [ ri . ns ] . sets , set = set && set [ set . length - 1 ] ) ,
candidate = getCandidateForSrc ( src , set ) , candidate && ( src = ri . makeUrl ( src ) , img [ ri . ns ] . curSrc = src ,
2014-11-25 23:27:31 +01:00
img [ ri . ns ] . curCan = candidate , candidate . res || setResolution ( candidate , candidate . set . sizes ) ) ,
2014-11-05 21:25:18 +01:00
candidate ;
}
function getCandidateForSrc ( src , set ) {
var i , candidate , candidates ;
if ( src && set ) for ( candidates = ri . parseSet ( set ) , src = ri . makeUrl ( src ) , i = 0 ; i < candidates . length ; i ++ ) if ( src == ri . makeUrl ( candidates [ i ] . url ) ) {
candidate = candidates [ i ] ;
break ;
2014-07-20 02:43:13 +02:00
}
2014-11-05 21:25:18 +01:00
return candidate ;
}
function getAllSourceElements ( picture , candidates ) {
var i , len , source , srcset , sources = picture . getElementsByTagName ( "source" ) ;
for ( i = 0 , len = sources . length ; len > i ; i ++ ) source = sources [ i ] , source [ ri . ns ] = ! 0 ,
srcset = source . getAttribute ( "srcset" ) , srcset && candidates . push ( {
srcset : srcset ,
media : source . getAttribute ( "media" ) ,
type : source . getAttribute ( "type" ) ,
sizes : source . getAttribute ( "sizes" )
} ) ;
}
document . createElement ( "picture" ) ;
2015-01-21 16:03:20 +01:00
var lowTreshHold , partialLowTreshHold , isLandscape , lazyFactor , tMemory , substractCurRes , eminpx , alwaysCheckWDescriptor , resizeThrottle , ri = { } , noop = function ( ) { } , image = document . createElement ( "img" ) , getImgAttr = image . getAttribute , setImgAttr = image . setAttribute , removeImgAttr = image . removeAttribute , docElem = document . documentElement , types = { } , cfg = {
2014-11-05 21:25:18 +01:00
xQuant : 1 ,
2015-01-21 16:03:20 +01:00
lazyFactor : . 4 ,
maxX : 2
} , srcAttr = "data-risrc" , srcsetAttr = srcAttr + "set" , reflowBug = "webkitBackfaceVisibility" in docElem . style , ua = navigator . userAgent , supportNativeLQIP = /AppleWebKit/i . test ( ua ) , supportAbort = /rident/ . test ( ua ) || /ecko/ . test ( ua ) && ua . match ( /rv\:(\d+)/ ) && RegExp . $1 > 35 , imgAbortCount = 0 , curSrcProp = "currentSrc" , regWDesc = /\s+\+?\d+(e\d+)?w/ , regSize = /(\([^)]+\))?\s*(.+)/ , regDescriptor = /^([\+eE\d\.]+)(w|x)$/ , regHDesc = /\s*\d+h\s*/ , setOptions = window . respimgCFG , baseStyle = ( "https:" == location . protocol ,
"position:absolute;left:0;visibility:hidden;display:block;padding:0;border:none;font-size:1em;width:1em;overflow:hidden;clip:rect(0px, 0px, 0px, 0px)" ) , fsCss = "font-size:100%!important;" , isVwDirty = ! 0 , cssCache = { } , sizeLengthCache = { } , DPR = window . devicePixelRatio , units = {
2014-11-25 23:27:31 +01:00
px : 1 ,
"in" : 96
2015-01-21 16:03:20 +01:00
} , anchor = document . createElement ( "a" ) , alreadyRun = ! 1 , on = function ( obj , evt , fn , capture ) {
obj . addEventListener ? obj . addEventListener ( evt , fn , capture || ! 1 ) : obj . attachEvent && obj . attachEvent ( "on" + evt , fn ) ;
} , off = function ( obj , evt , fn , capture ) {
obj . removeEventListener ? obj . removeEventListener ( evt , fn , capture || ! 1 ) : obj . detachEvent && obj . detachEvent ( "on" + evt , fn ) ;
} , memoize = function ( fn ) {
var cache = { } ;
return function ( input ) {
return input in cache || ( cache [ input ] = fn ( input ) ) , cache [ input ] ;
} ;
} , evalCSS = function ( ) {
var regLength = /^([\d\.]+)(em|vw|px)$/ , replace = function ( ) {
2014-11-25 23:27:31 +01:00
for ( var args = arguments , index = 0 , string = args [ 0 ] ; ++ index in args ; ) string = string . replace ( args [ index ] , args [ ++ index ] ) ;
return string ;
2015-01-21 16:03:20 +01:00
} , buidlStr = memoize ( function ( css ) {
return "return " + replace ( ( css || "" ) . toLowerCase ( ) , /\band\b/g , "&&" , /,/g , "||" , /min-([a-z-\s]+):/g , "e.$1>=" , /max-([a-z-\s]+):/g , "e.$1<=" , /calc([^)]+)/g , "($1)" , /(\d+[\.]*[\d]*)([a-z]+)/g , "($1 * e.$2)" , /^(?!(e.[a-z]|[0-9\.&=|><\+\-\*\(\)\/])).*/gi , "" ) ;
} ) ;
2014-11-25 23:27:31 +01:00
return function ( css , length ) {
var parsedLength ;
if ( ! ( css in cssCache ) ) if ( cssCache [ css ] = ! 1 , length && ( parsedLength = css . match ( regLength ) ) ) cssCache [ css ] = parsedLength [ 1 ] * units [ parsedLength [ 2 ] ] ; else try {
cssCache [ css ] = new Function ( "e" , buidlStr ( css ) ) ( units ) ;
} catch ( e ) { }
return cssCache [ css ] ;
} ;
2015-01-21 16:03:20 +01:00
} ( ) , setResolution = function ( candidate , sizesattr ) {
return candidate . w ? ( candidate . cWidth = ri . calcListLength ( sizesattr || "100vw" ) ,
candidate . res = candidate . w / candidate . cWidth ) : candidate . res = candidate . x , candidate ;
} , respimage = function ( opt ) {
var elements , i , plen , options = opt || { } ;
if ( options . elements && 1 == options . elements . nodeType && ( "IMG" == options . elements . nodeName . toUpperCase ( ) ? options . elements = [ options . elements ] : ( options . context = options . elements ,
options . elements = null ) ) , elements = options . elements || ri . qsa ( options . context || document , options . reevaluate || options . reparse ? ri . sel : ri . selShort ) ,
plen = elements . length ) {
for ( ri . setupRun ( options ) , alreadyRun = ! 0 , i = 0 ; plen > i ; i ++ ) imgAbortCount ++ ,
6 > imgAbortCount && ! elements [ i ] . complete && imgAbortCount ++ , ri . fillImg ( elements [ i ] , options ) ;
ri . teardownRun ( options ) , imgAbortCount ++ ;
}
} , reevaluateAfterLoad = function ( ) {
var onload = function ( ) {
off ( this , "load" , onload ) , off ( this , "error" , onload ) , ri . fillImgs ( {
elements : [ this ]
} ) ;
} ;
return function ( img ) {
off ( img , "load" , onload ) , off ( img , "error" , onload ) , on ( img , "error" , onload ) , on ( img , "load" , onload ) ;
} ;
} ( ) , parseDescriptor = memoize ( function ( descriptor ) {
var descriptorObj = [ 1 , "x" ] , parsedDescriptor = trim ( descriptor || "" ) ;
return parsedDescriptor && ( parsedDescriptor = parsedDescriptor . replace ( regHDesc , "" ) ,
descriptorObj = parsedDescriptor . match ( regDescriptor ) ? [ 1 * RegExp . $1 , RegExp . $2 ] : ! 1 ) ,
descriptorObj ;
} ) ;
curSrcProp in image || ( curSrcProp = "src" ) , types [ "image/jpeg" ] = ! 0 , types [ "image/gif" ] = ! 0 ,
types [ "image/png" ] = ! 0 , types [ "image/svg+xml" ] = document . implementation . hasFeature ( "http://wwwindow.w3.org/TR/SVG11/feature#Image" , "1.1" ) ,
ri . ns = ( "ri" + new Date ( ) . getTime ( ) ) . substr ( 0 , 9 ) , ri . supSrcset = "srcset" in image ,
ri . supSizes = "sizes" in image , ri . selShort = "picture>img,img[srcset]" , ri . sel = ri . selShort ,
ri . cfg = cfg , ri . supSrcset && ( ri . sel += ",img[" + srcsetAttr + "]" ) , ri . DPR = DPR || 1 ,
ri . u = units , ri . types = types , alwaysCheckWDescriptor = ri . supSrcset && ! ri . supSizes ,
ri . setSize = noop , ri . makeUrl = memoize ( function ( src ) {
return anchor . href = src , anchor . href ;
} ) , ri . qsa = function ( context , sel ) {
return context . querySelectorAll ( sel ) ;
} , ri . matchesMedia = function ( ) {
return ri . matchesMedia = window . matchMedia && ( matchMedia ( "(min-width: 0.1em)" ) || { } ) . matches ? function ( media ) {
return ! media || matchMedia ( media ) . matches ;
} : ri . mMQ , ri . matchesMedia . apply ( this , arguments ) ;
} , ri . mMQ = function ( media ) {
return media ? evalCSS ( media ) : ! 0 ;
} , ri . calcLength = function ( sourceSizeValue ) {
2014-11-25 23:27:31 +01:00
var value = evalCSS ( sourceSizeValue , ! 0 ) || ! 1 ;
return 0 > value && ( value = ! 1 ) , value ;
2015-01-21 16:03:20 +01:00
} , ri . supportsType = function ( type ) {
return type ? types [ type ] : ! 0 ;
} , ri . parseSize = memoize ( function ( sourceSizeStr ) {
var match = ( sourceSizeStr || "" ) . match ( regSize ) ;
return {
media : match && match [ 1 ] ,
length : match && match [ 2 ]
2014-11-05 21:25:18 +01:00
} ;
2015-01-21 16:03:20 +01:00
} ) , ri . parseSet = function ( set ) {
2014-11-05 21:25:18 +01:00
if ( ! set . cands ) {
2015-01-21 16:03:20 +01:00
var pos , url , descriptor , last , descpos , can , srcset = set . srcset ;
2014-11-05 21:25:18 +01:00
for ( set . cands = [ ] ; srcset ; ) srcset = srcset . replace ( /^\s+/g , "" ) , pos = srcset . search ( /\s/g ) ,
descriptor = null , - 1 != pos ? ( url = srcset . slice ( 0 , pos ) , last = url . charAt ( url . length - 1 ) ,
"," != last && url || ( url = url . replace ( /,+$/ , "" ) , descriptor = "" ) , srcset = srcset . slice ( pos + 1 ) ,
null == descriptor && ( descpos = srcset . indexOf ( "," ) , - 1 != descpos ? ( descriptor = srcset . slice ( 0 , descpos ) ,
srcset = srcset . slice ( descpos + 1 ) ) : ( descriptor = srcset , srcset = "" ) ) ) : ( url = srcset ,
2015-01-21 16:03:20 +01:00
srcset = "" ) , url && ( descriptor = parseDescriptor ( descriptor ) ) && ( can = {
2014-11-05 21:25:18 +01:00
url : url . replace ( /^,+/ , "" ) ,
set : set
2015-01-21 16:03:20 +01:00
} , can [ descriptor [ 1 ] ] = descriptor [ 0 ] , "x" == descriptor [ 1 ] && 1 == descriptor [ 0 ] && ( set . has1x = ! 0 ) ,
set . cands . push ( can ) ) ;
2014-07-17 16:45:12 +02:00
}
2014-11-05 21:25:18 +01:00
return set . cands ;
2015-01-21 16:03:20 +01:00
} , ri . getEmValue = function ( ) {
2014-11-05 21:25:18 +01:00
var body ;
if ( ! eminpx && ( body = document . body ) ) {
var div = document . createElement ( "div" ) , originalHTMLCSS = docElem . style . cssText , originalBodyCSS = body . style . cssText ;
div . style . cssText = baseStyle , docElem . style . cssText = fsCss , body . style . cssText = fsCss ,
body . appendChild ( div ) , eminpx = div . offsetWidth , body . removeChild ( div ) , eminpx = parseFloat ( eminpx , 10 ) ,
docElem . style . cssText = originalHTMLCSS , body . style . cssText = originalBodyCSS ;
2014-07-17 16:45:12 +02:00
}
2014-11-05 21:25:18 +01:00
return eminpx || 16 ;
2014-11-25 23:27:31 +01:00
} , ri . calcListLength = function ( sourceSizeListStr ) {
2014-11-05 21:25:18 +01:00
if ( ! ( sourceSizeListStr in sizeLengthCache ) || cfg . uT ) {
var sourceSize , parsedSize , length , media , i , len , sourceSizeList = trim ( sourceSizeListStr ) . split ( /\s*,\s*/ ) , winningLength = ! 1 ;
for ( i = 0 , len = sourceSizeList . length ; len > i && ( sourceSize = sourceSizeList [ i ] ,
parsedSize = ri . parseSize ( sourceSize ) , length = parsedSize . length , media = parsedSize . media ,
! length || ! ri . matchesMedia ( media ) || ( winningLength = ri . calcLength ( length ) ) === ! 1 ) ; i ++ ) ;
2014-11-25 23:27:31 +01:00
sizeLengthCache [ sourceSizeListStr ] = winningLength ? winningLength : units . width ;
2014-07-17 16:45:12 +02:00
}
2014-11-05 21:25:18 +01:00
return sizeLengthCache [ sourceSizeListStr ] ;
} , ri . setRes = function ( set ) {
2015-01-21 16:03:20 +01:00
var candidates ;
2014-11-05 21:25:18 +01:00
if ( set ) {
candidates = ri . parseSet ( set ) ;
2015-01-21 16:03:20 +01:00
for ( var i = 0 , len = candidates . length ; len > i ; i ++ ) setResolution ( candidates [ i ] , set . sizes ) ;
2014-07-17 16:45:12 +02:00
}
return candidates ;
2015-01-21 16:03:20 +01:00
} , ri . setRes . res = setResolution , ri . applySetCandidate = function ( candidates , img ) {
2014-11-05 21:25:18 +01:00
if ( candidates . length ) {
2015-01-21 16:03:20 +01:00
var candidate , dpr , i , j , diff , length , bestCandidate , curSrc , curCan , isSameSet , candidateSrc , abortCurSrc , oldRes , imageData = img [ ri . ns ] , evaled = ! 0 , lazyF = lazyFactor , sub = substractCurRes ;
2014-11-05 21:25:18 +01:00
if ( curSrc = imageData . curSrc || img [ curSrcProp ] , curCan = imageData . curCan || setSrcToCur ( img , curSrc , candidates [ 0 ] . set ) ,
2015-01-21 16:03:20 +01:00
dpr = ri . DPR , oldRes = curCan && curCan . res , ! bestCandidate && curSrc && ( abortCurSrc = supportAbort && ! img . complete && curCan && oldRes > dpr ,
abortCurSrc || curCan && ! ( tMemory > oldRes ) || ( curCan && dpr > oldRes && oldRes > lowTreshHold && ( partialLowTreshHold > oldRes && ( lazyF *= . 87 ,
sub += . 04 * dpr ) , curCan . res += lazyF * Math . pow ( oldRes - sub , 2 ) ) , isSameSet = ! imageData . pic || curCan && curCan . set == candidates [ 0 ] . set ,
curCan && isSameSet && curCan . res >= dpr ? bestCandidate = curCan : supportNativeLQIP || img . complete || ! getImgAttr . call ( img , "src" ) || img . lazyload || supportAbort && ! ( 5 > imgAbortCount ) || ! isSameSet && inView ( img ) || ( bestCandidate = curCan ,
candidateSrc = curSrc , evaled = "L" , reevaluateAfterLoad ( img ) ) ) ) , ! bestCandidate ) for ( oldRes && ( curCan . res = curCan . res - ( curCan . res - oldRes ) / 2 ) ,
candidates . sort ( ascendingSort ) , length = candidates . length , bestCandidate = candidates [ length - 1 ] ,
i = 0 ; length > i ; i ++ ) if ( candidate = candidates [ i ] , candidate . res >= dpr ) {
j = i - 1 , bestCandidate = candidates [ j ] && ( diff = candidate . res - dpr ) && ( abortCurSrc || curSrc != ri . makeUrl ( candidate . url ) ) && chooseLowRes ( candidates [ j ] . res , diff , dpr ) ? candidates [ j ] : candidate ;
2014-07-17 16:45:12 +02:00
break ;
}
2014-11-25 23:27:31 +01:00
return oldRes && ( curCan . res = oldRes ) , bestCandidate && ( candidateSrc = ri . makeUrl ( bestCandidate . url ) ,
2015-01-21 16:03:20 +01:00
imageData . curSrc = candidateSrc , imageData . curCan = bestCandidate , candidateSrc != curSrc && ri . setSrc ( img , bestCandidate ) ,
ri . setSize ( img ) ) , evaled ;
2014-07-17 16:45:12 +02:00
}
2015-01-21 16:03:20 +01:00
} , ri . setSrc = function ( img , bestCandidate ) {
var origStyle ;
img . src = bestCandidate . url , reflowBug && ( origStyle = img . style . zoom , img . style . zoom = "0.999" ,
img . style . zoom = origStyle ) ;
} , ri . getSet = function ( img ) {
2014-11-25 23:27:31 +01:00
var i , set , supportsType , match = ! 1 , sets = img [ ri . ns ] . sets ;
for ( i = 0 ; i < sets . length && ! match ; i ++ ) if ( set = sets [ i ] , set . srcset && ri . matchesMedia ( set . media ) && ( supportsType = ri . supportsType ( set . type ) ) ) {
"pending" == supportsType && ( set = supportsType ) , match = set ;
break ;
}
return match ;
2015-01-21 16:03:20 +01:00
} , ri . parseSets = function ( element , parent ) {
var srcsetAttribute , imageSet , isWDescripor , srcsetParsed , hasPicture = "PICTURE" == parent . nodeName . toUpperCase ( ) , imageData = element [ ri . ns ] ;
2014-11-05 21:25:18 +01:00
imageData . src === undefined && ( imageData . src = getImgAttr . call ( element , "src" ) ,
imageData . src ? setImgAttr . call ( element , srcAttr , imageData . src ) : removeImgAttr . call ( element , srcAttr ) ) ,
imageData . srcset === undefined && ( srcsetAttribute = getImgAttr . call ( element , "srcset" ) ,
2014-11-25 23:27:31 +01:00
imageData . srcset = srcsetAttribute , srcsetParsed = ! 0 ) , imageData . sets = [ ] , hasPicture && ( imageData . pic = ! 0 ,
2015-01-21 16:03:20 +01:00
getAllSourceElements ( parent , imageData . sets ) ) , imageData . srcset ? ( imageSet = {
2014-11-25 23:27:31 +01:00
srcset : imageData . srcset ,
sizes : getImgAttr . call ( element , "sizes" )
2015-01-21 16:03:20 +01:00
} , imageData . sets . push ( imageSet ) , isWDescripor = ( alwaysCheckWDescriptor || imageData . src ) && regWDesc . test ( imageData . srcset || "" ) ,
isWDescripor || ! imageData . src || getCandidateForSrc ( imageData . src , imageSet ) || imageSet . has1x || ( imageSet . srcset += ", " + imageData . src ,
imageSet . cands . push ( {
url : imageData . src ,
x : 1 ,
set : imageSet
} ) ) ) : imageData . src && imageData . sets . push ( {
2014-11-25 23:27:31 +01:00
srcset : imageData . src ,
sizes : null
2015-01-21 16:03:20 +01:00
} ) , imageData . curCan = null , imageData . supported = ! ( hasPicture || imageSet && ! ri . supSrcset || isWDescripor ) ,
2014-11-05 21:25:18 +01:00
srcsetParsed && ri . supSrcset && ! imageData . supported && ( srcsetAttribute ? ( setImgAttr . call ( element , srcsetAttr , srcsetAttribute ) ,
2014-11-25 23:27:31 +01:00
element . srcset = "" ) : removeImgAttr . call ( element , srcsetAttr ) ) , imageData . supported && ! imageData . srcset && ( ! imageData . src && element . src || element . src != ri . makeUrl ( imageData . src ) ) && ( null == imageData . src ? element . removeAttribute ( "src" ) : element . src = imageData . src ) ,
imageData . parsed = ! 0 ;
2015-01-21 16:03:20 +01:00
} , ri . fillImg = function ( element , options ) {
2014-11-05 21:25:18 +01:00
var parent , imageData , extreme = options . reparse || options . reevaluate ;
2014-11-25 23:27:31 +01:00
if ( element [ ri . ns ] || ( element [ ri . ns ] = { } ) , imageData = element [ ri . ns ] , "L" == imageData . evaled && element . complete && ( imageData . evaled = ! 1 ) ,
2014-11-05 21:25:18 +01:00
extreme || ! imageData . evaled ) {
if ( ! imageData . parsed || options . reparse ) {
if ( parent = element . parentNode , ! parent ) return ;
ri . parseSets ( element , parent , options ) ;
2014-07-17 16:45:12 +02:00
}
2014-11-05 21:25:18 +01:00
imageData . supported ? imageData . evaled = ! 0 : applyBestCandidate ( element ) ;
2014-07-17 16:45:12 +02:00
}
2015-01-21 16:03:20 +01:00
} , ri . setupRun = function ( options ) {
2014-11-25 23:27:31 +01:00
( ! alreadyRun || options . reevaluate || isVwDirty ) && ( updateMetrics ( ) , options . elements || options . context || clearTimeout ( resizeThrottle ) ) ;
2015-01-21 16:03:20 +01:00
} , window . HTMLPictureElement ? ( respimage = noop , ri . fillImg = noop ) : ! function ( ) {
var isDomReady , regReady = window . attachEvent ? /d$|^c/ : /d$|^c|^i/ , run = function ( ) {
2014-11-25 23:27:31 +01:00
var readyState = document . readyState || "" ;
2015-01-21 16:03:20 +01:00
timerId = setTimeout ( run , "loading" == readyState ? 200 : 999 ) , document . body && ( isDomReady = isDomReady || regReady . test ( readyState ) ,
ri . fillImgs ( ) , isDomReady && ( imgAbortCount += 6 , clearTimeout ( timerId ) ) ) ;
2014-11-05 21:25:18 +01:00
} , resizeEval = function ( ) {
ri . fillImgs ( {
reevaluate : ! 0
} ) ;
} , onResize = function ( ) {
clearTimeout ( resizeThrottle ) , isVwDirty = ! 0 , resizeThrottle = setTimeout ( resizeEval , 99 ) ;
} , timerId = setTimeout ( run , document . body ? 9 : 99 ) ;
on ( window , "resize" , onResize ) , on ( document , "readystatechange" , run ) ;
2015-01-21 16:03:20 +01:00
} ( ) , ri . respimage = respimage , ri . fillImgs = respimage , ri . teardownRun = noop , respimage . _ = ri ,
window . respimage = respimage , window . respimgCFG = {
ri : ri ,
push : function ( args ) {
var name = args . shift ( ) ;
"function" == typeof ri [ name ] ? ri [ name ] . apply ( ri , args ) : ( cfg [ name ] = args [ 0 ] ,
alreadyRun && ri . fillImgs ( {
reevaluate : ! 0
} ) ) ;
}
} ;
for ( ; setOptions && setOptions . length ; ) window . respimgCFG . push ( setOptions . shift ( ) ) ;
2014-11-05 21:25:18 +01:00
} ( window , document ) ;
2015-01-21 16:03:20 +01:00
( function ( factory ) {
"use strict" ;
var interValId ;
var intervalIndex = 0 ;
var run = function ( ) {
if ( window . respimage ) {
factory ( window . respimage ) ;
}
if ( window . respimage || intervalIndex > 9999 ) {
clearInterval ( interValId ) ;
}
intervalIndex ++ ;
} ;
interValId = setInterval ( run , 8 ) ;
run ( ) ;
} ( function ( respimage ) {
"use strict" ;
var ri = respimage . _ ;
var runningTests = 0 ;
var setTypeValue = function ( types , value ) {
var i ;
for ( i = 0 ; i < types . length ; i ++ ) {
ri . types [ types [ i ] ] = value ;
}
} ;
if ( window . HTMLPictureElement && ! ri . cfg . uT ) {
respimage . testTypeSupport = function ( ) { } ;
return ;
}
ri . types [ "image/bmp" ] = true ;
ri . types [ "image/x-bmp" ] = true ;
respimage . testTypeSupport = function ( types , url , width , useCanvas ) {
if ( typeof types == "string" ) {
types = types . split ( /\s*\,*\s+/g ) ;
}
var canvas ;
var supports = "pending" ;
var img = document . createElement ( 'img' ) ;
var onComplete = function ( ) {
runningTests -- ;
setTypeValue ( types , supports ) ;
if ( runningTests < 1 ) {
respimage ( { reevaluate : true } ) ;
}
} ;
if ( useCanvas ) {
canvas = document . createElement ( 'canvas' ) ;
if ( ! canvas . getContext ) {
setTypeValue ( types , false ) ;
return ;
}
}
img . onload = function ( ) {
var ctx ;
supports = true ;
if ( width ) {
supports = img . width == width ;
}
if ( useCanvas ) {
ctx = canvas . getContext ( '2d' ) ;
ctx . drawImage ( img , 0 , 0 ) ;
supports = ctx . getImageData ( 0 , 0 , 1 , 1 ) . data [ 3 ] === 0 ;
}
onComplete ( ) ;
} ;
img . onerror = function ( ) {
supports = false ;
onComplete ( ) ;
} ;
runningTests ++ ;
setTypeValue ( types , "pending" ) ;
img . src = url ;
} ;
respimage . testTypeSupport ( "image/webp" , "" , 1 ) ;
respimage . testTypeSupport ( "image/jp2 image/jpx image/jpm" , "" , 1 ) ;
respimage . testTypeSupport ( "image/vnd.ms-photo" , "" , 1 ) ;
respimage . testTypeSupport ( "video/png video/apng video/x-mng video/x-png" , "" , false , true ) ;
} ) ) ;
2014-11-25 23:27:31 +01:00
( function ( factory ) {
"use strict" ;
var interValId ;
var intervalIndex = 0 ;
var run = function ( ) {
if ( window . respimage ) {
factory ( window . respimage ) ;
}
if ( window . respimage || intervalIndex > 9999 ) {
clearInterval ( interValId ) ;
}
intervalIndex ++ ;
} ;
interValId = setInterval ( run , 8 ) ;
run ( ) ;
} ( function ( respimage , undefined ) {
"use strict" ;
var ri = respimage . _ ;
var knownWidths = { } ;
var cfg = ri . cfg ;
var curSrcProp = "currentSrc" ;
var setSize = function ( width , img , data ) {
var curCandidate = data . curCan ;
if ( width ) {
2015-01-21 16:03:20 +01:00
img . setAttribute ( "width" , parseInt ( width / curCandidate . res , 10 ) ) ;
2014-11-25 23:27:31 +01:00
}
} ;
var loadBg = function ( url , img , data ) {
2015-01-21 16:03:20 +01:00
var bgImg , curCandidate , clear ;
2014-11-25 23:27:31 +01:00
if ( knownWidths [ url ] ) {
setSize ( knownWidths [ url ] , img , data ) ;
} else {
2015-01-21 16:03:20 +01:00
clear = function ( ) {
data . pendingURLSize = null ;
bgImg . onload = null ;
bgImg . onerror = null ;
img = null ;
bgImg = null ;
} ;
data . pendingURLSize = url ;
2014-11-25 23:27:31 +01:00
curCandidate = data . curCan ;
2015-01-21 16:03:20 +01:00
if ( curCandidate . w ) {
setSize ( curCandidate . w , img , data ) ;
2014-11-25 23:27:31 +01:00
}
bgImg = document . createElement ( 'img' ) ;
2015-01-21 16:03:20 +01:00
2014-11-25 23:27:31 +01:00
bgImg . onload = function ( ) {
knownWidths [ url ] = bgImg . naturalWidth || bgImg . width ;
if ( url == img [ curSrcProp ] ) {
setSize ( knownWidths [ url ] , img , data ) ;
}
2015-01-21 16:03:20 +01:00
clear ( ) ;
2014-11-25 23:27:31 +01:00
} ;
2015-01-21 16:03:20 +01:00
bgImg . onerror = clear ;
2014-11-25 23:27:31 +01:00
bgImg . src = url ;
if ( bgImg && bgImg . complete ) {
bgImg . onload ( ) ;
}
}
} ;
var reeval = ( function ( ) {
var running , timer ;
2015-01-21 16:03:20 +01:00
2014-11-25 23:27:31 +01:00
var run = function ( ) {
2015-01-21 16:03:20 +01:00
var i , len , imgData ;
var imgs = document . getElementsByTagName ( 'img' ) ;
var options = { elements : [ ] } ;
ri . setupRun ( options ) ;
2014-11-25 23:27:31 +01:00
running = false ;
clearTimeout ( timer ) ;
2015-01-21 16:03:20 +01:00
for ( i = 0 , len = imgs . length ; i < len ; i ++ ) {
imgData = imgs [ i ] [ ri . ns ] ;
if ( imgData && imgData . curCan ) {
ri . setRes . res ( imgData . curCan , imgData . curCan . set . sizes ) ;
ri . setSize ( imgs [ i ] ) ;
}
}
ri . teardownRun ( options ) ;
2014-11-25 23:27:31 +01:00
} ;
return function ( ) {
if ( ! running && cfg . addSize ) {
running = true ;
clearTimeout ( timer ) ;
2015-01-21 16:03:20 +01:00
timer = setTimeout ( run ) ;
2014-11-25 23:27:31 +01:00
}
} ;
} ) ( ) ;
if ( ! ( curSrcProp in document . createElement ( "img" ) ) ) {
curSrcProp = "src" ;
}
ri . setSize = function ( img ) {
var url ;
var data = img [ ri . ns ] ;
var curCandidate = data . curCan ;
if ( data . dims === undefined ) {
data . dims = img . getAttribute ( "height" ) && img . getAttribute ( "width" ) ;
}
if ( ! cfg . addSize || ! curCandidate || data . dims ) { return ; }
url = ri . makeUrl ( curCandidate . url ) ;
2015-01-21 16:03:20 +01:00
if ( url == img [ curSrcProp ] && url !== data . pendingURLSize ) {
2014-11-25 23:27:31 +01:00
loadBg ( url , img , data ) ;
}
} ;
if ( window . addEventListener ) {
addEventListener ( 'resize' , reeval , false ) ;
}
2014-07-17 16:45:12 +02:00
2015-01-21 16:03:20 +01:00
if ( ! ( 'addSize' in cfg ) ) {
cfg . addSize = true ;
} else {
cfg . addSize = ! ! cfg . addSize ;
2014-11-05 21:25:18 +01:00
}
2014-07-17 16:45:12 +02:00
2015-01-21 16:03:20 +01:00
reeval ( ) ;
2014-11-05 21:25:18 +01:00
} ) ) ;
2014-08-01 01:38:33 +02:00
2014-11-25 23:27:31 +01:00
2014-11-05 21:25:18 +01:00
( function ( ) {
2014-07-17 16:45:12 +02:00
2014-11-05 21:25:18 +01:00
webshim . isReady ( 'picture' , true ) ;
var sel = 'picture, img[srcset]' ;
webshim . addReady ( function ( context ) {
if ( context . querySelector ( sel ) ) {
window . respimage ( ) ;
}
} ) ;
} ) ( ) ;
2014-07-17 16:45:12 +02:00
2014-07-20 02:43:13 +02:00
2015-01-21 16:03:20 +01:00
/*! matchMedia() polyfill - Test a CSS media type/query in JS. Authors & copyright (c) 2012: Scott Jehl, Paul Irish, Nicholas Zakas, David Knight. Dual MIT/BSD license */
2014-07-17 16:45:12 +02:00
window . matchMedia || ( window . matchMedia = function ( ) {
"use strict" ;
// For browsers that support matchMedium api such as IE 9 and webkit
var styleMedia = ( window . styleMedia || window . media ) ;
// For those that don't support matchMedium
if ( ! styleMedia ) {
var style = document . createElement ( 'style' ) ,
script = document . getElementsByTagName ( 'script' ) [ 0 ] ,
info = null ;
style . type = 'text/css' ;
style . id = 'matchmediajs-test' ;
script . parentNode . insertBefore ( style , script ) ;
// 'style.currentStyle' is used by IE <= 8 and 'window.getComputedStyle' for all other browsers
info = ( 'getComputedStyle' in window ) && window . getComputedStyle ( style , null ) || style . currentStyle ;
styleMedia = {
matchMedium : function ( media ) {
var text = '@media ' + media + '{ #matchmediajs-test { width: 1px; } }' ;
// 'style.styleSheet' is used by IE <= 8 and 'style.textContent' for all other browsers
if ( style . styleSheet ) {
style . styleSheet . cssText = text ;
} else {
style . textContent = text ;
}
// Test if media query is true or false
return info . width === '1px' ;
}
} ;
}
return function ( media ) {
return {
matches : styleMedia . matchMedium ( media || 'all' ) ,
media : media || 'all'
} ;
} ;
} ( ) ) ;
/*! matchMedia() polyfill addListener/removeListener extension. Author & copyright (c) 2012: Scott Jehl. Dual MIT/BSD license */
( function ( ) {
// Bail out for browsers that have addListener support
if ( window . matchMedia && window . matchMedia ( 'all' ) . addListener ) {
return false ;
}
var localMatchMedia = window . matchMedia ,
hasMediaQueries = localMatchMedia ( 'only all' ) . matches ,
isListening = false ,
timeoutID = 0 , // setTimeout for debouncing 'handleChange'
queries = [ ] , // Contains each 'mql' and associated 'listeners' if 'addListener' is used
handleChange = function ( evt ) {
// Debounce
clearTimeout ( timeoutID ) ;
timeoutID = setTimeout ( function ( ) {
for ( var i = 0 , il = queries . length ; i < il ; i ++ ) {
var mql = queries [ i ] . mql ,
listeners = queries [ i ] . listeners || [ ] ,
matches = localMatchMedia ( mql . media ) . matches ;
// Update mql.matches value and call listeners
// Fire listeners only if transitioning to or from matched state
if ( matches !== mql . matches ) {
mql . matches = matches ;
for ( var j = 0 , jl = listeners . length ; j < jl ; j ++ ) {
listeners [ j ] . call ( window , mql ) ;
}
}
}
} , 30 ) ;
} ;
window . matchMedia = function ( media ) {
var mql = localMatchMedia ( media ) ,
listeners = [ ] ,
index = 0 ;
mql . addListener = function ( listener ) {
// Changes would not occur to css media type so return now (Affects IE <= 8)
if ( ! hasMediaQueries ) {
return ;
}
// Set up 'resize' listener for browsers that support CSS3 media queries (Not for IE <= 8)
// There should only ever be 1 resize listener running for performance
if ( ! isListening ) {
isListening = true ;
window . addEventListener ( 'resize' , handleChange , true ) ;
}
// Push object only if it has not been pushed already
if ( index === 0 ) {
index = queries . push ( {
mql : mql ,
listeners : listeners
} ) ;
}
listeners . push ( listener ) ;
} ;
mql . removeListener = function ( listener ) {
for ( var i = 0 , il = listeners . length ; i < il ; i ++ ) {
if ( listeners [ i ] === listener ) {
listeners . splice ( i , 1 ) ;
}
}
} ;
return mql ;
} ;
} ( ) ) ;
webshim . isReady ( 'matchMedia' , true ) ;