Skip to content

HTTPS clone URL

Subversion checkout URL

You can clone with HTTPS or Subversion.

Download ZIP
Browse files

Handle scripts that tries to get src from lazyloaded images

Ex: Slideshow tries to get src using $('img').attr('src'), we will now
give him the real src if it exists (stored in lazyAttr, 'data-src').

This overrides HTMLImgElement.prototype.getAttribute (scary), does not
works for ie <8 (also this is a double ice cream).
  • Loading branch information...
commit 17966dcd8ae2280e44882b5fb8b6edd7fcb46541 1 parent 9347635
vvo authored
View
19 lazyload.js
@@ -44,6 +44,11 @@ if (!window['lzld']) {
saveViewportT = throttle(viewport, 20),
showImagesT = throttle(showImages, 20);
+ // Override image element .getAttribute globally so that we give the real src
+ // does not works for ie < 8: http://perfectionkills.com/whats-wrong-with-extending-the-dom/
+ // Internet Explorer 7 (and below) [...] does not expose global Node, Element, HTMLElement, HTMLParagraphElement
+ window['HTMLImageElement'] && overrideGetattribute();
+
// Called from every lazy <img> onload event
window['lzld'] = onDataSrcImgLoad;
@@ -244,6 +249,20 @@ if (!window['lzld']) {
addEvent(window, 'scroll', showImagesT);
}
+ function overrideGetattribute() {
+ var original = HTMLImageElement.prototype.getAttribute;
+ HTMLImageElement.prototype.getAttribute = function(name) {
+ if(name === 'src') {
+ var realSrc = original.call(this, lazyAttr);
+ return realSrc || original.call(this, name);
+ } else {
+ // our own lazyloader will go through theses lines
+ // because we use getAttribute(lazyAttr)
+ return original.call(this, name);
+ }
+ }
+ }
+
// https://github.com/jquery/sizzle/blob/3136f48b90e3edc84cbaaa6f6f7734ef03775a07/sizzle.js#L708
var contains = document.documentElement.compareDocumentPosition ?
function( a, b ) {
View
12 lazyload.min.js
@@ -1,6 +1,6 @@
-/*! lazyload v0.8.4 fasterize.com | github.com/fasterize/lazyload#licence */
-window.lzld||function(e,d){function l(){m=!0;g();setTimeout(g,25)}function n(a,b){var c=0;return function(){var d=+new Date;d-c<b||(c=d,a.apply(this,arguments))}}function h(a,b,c){a.attachEvent?a.attachEvent&&a.attachEvent("on"+b,c):a.addEventListener(b,c,!1)}function i(a,b,c){a.detachEvent?a.detachEvent&&a.detachEvent("on"+b,c):a.removeEventListener(b,c,!1)}function o(a,b){return p(d.documentElement,a)&&a.getBoundingClientRect().top<v+w?(a.onload=null,a.removeAttribute("onload"),a.onerror=null,a.removeAttribute("onerror"),
-a.src=a.getAttribute(j),a.removeAttribute(j),f[b]=null,!0):!1}function q(){return 0>d.documentElement.clientHeight?d.body&&0<=d.body.clientHeight?d.body.clientHeight:0>e.innerHeight?0:e.innerHeight:d.documentElement.clientHeight}function r(){var a,b,c=f.length,d=!0;for(a=0;a<c;a++)b=f[a],null!==b&&!o(b,a)&&(d=!1);d&&m&&(k=!0,i(e,"resize",s),i(e,"scroll",g),i(e,"load",l))}function t(){k=!1;h(e,"resize",s);h(e,"scroll",g)}function u(a,b,c){var d;if(b){if(Array.prototype.indexOf)return Array.prototype.indexOf.call(b,
-a,c);d=b.length;for(c=c?0>c?Math.max(0,d+c):c:0;c<d;c++)if(c in b&&b[c]===a)return c}return-1}var p,w=200,j="data-src",v=q(),f=[],m=!1,k=!1,s=n(q,20),g=n(r,20);e.lzld=function(a){-1===u(a,f)&&(k&&t(),o(a,f.push(a)-1))};(function(a){function b(c){"readystatechange"===c.type&&"complete"!==d.readyState||(i("load"===c.type?e:d,c.type,b),f||(f=!0,a()))}function c(){try{d.documentElement.doScroll("left")}catch(a){setTimeout(c,50);return}b("poll")}var f=!1,g=!0;if("complete"===d.readyState)a();else{if(d.createEventObject&&
-d.documentElement.doScroll){try{g=!e.frameElement}catch(j){}g&&c()}h(d,"DOMContentLoaded",b);h(d,"readystatechange",b);h(e,"load",b)}})(function(){var a,b,c,e=d.getElementsByTagName("img");b=0;for(c=e.length;b<c;b+=1)a=e[b],a.getAttribute(j)&&-1===u(a,f)&&f.push(a);r();setTimeout(g,25)});h(e,"load",l);t();p=d.documentElement.compareDocumentPosition?function(a,b){return!!(a.compareDocumentPosition(b)&16)}:d.documentElement.contains?function(a,b){return a!==b&&(a.contains?a.contains(b):!1)}:function(a,
-b){for(;b=b.parentNode;)if(b===a)return!0;return!1}}(this,document);
+/*! lazyload v0.8.5 fasterize.com | github.com/fasterize/lazyload#licence */
+window.lzld||function(e,d){function n(){o=!0;h();setTimeout(h,25)}function p(a,b){var c=0;return function(){var d=+new Date;d-c<b||(c=d,a.apply(this,arguments))}}function g(a,b,c){a.attachEvent?a.attachEvent&&a.attachEvent("on"+b,c):a.addEventListener(b,c,!1)}function j(a,b,c){a.detachEvent?a.detachEvent&&a.detachEvent("on"+b,c):a.removeEventListener(b,c,!1)}function q(a,b){return A(d.documentElement,a)&&a.getBoundingClientRect().top<B+C?(a.onload=null,a.removeAttribute("onload"),a.onerror=null,a.removeAttribute("onerror"),
+a.src=a.getAttribute(k),a.removeAttribute(k),f[b]=null,!0):!1}function r(){return 0<=d.documentElement.clientHeight?d.documentElement.clientHeight:d.body&&0<=d.body.clientHeight?d.body.clientHeight:0<=e.innerHeight?e.innerHeight:0}function s(){var a=f.length,b,c=!0;for(b=0;b<a;b++){var d=f[b];null!==d&&!q(d,b)&&(c=!1)}c&&o&&(l=!0,j(e,"resize",t),j(e,"scroll",h),j(e,"load",n))}function u(){l=!1;g(e,"resize",t);g(e,"scroll",h)}function v(a,b,c){var d;if(b){if(Array.prototype.indexOf)return Array.prototype.indexOf.call(b,
+a,c);d=b.length;for(c=c?0>c?Math.max(0,d+c):c:0;c<d;c++)if(c in b&&b[c]===a)return c}return-1}var C=200,k="data-src",B=r(),f=[],o=!1,l=!1,t=p(r,20),h=p(s,20);if(e.HTMLImageElement){var m=HTMLImageElement.prototype.getAttribute;HTMLImageElement.prototype.getAttribute=function(a){return"src"===a?m.call(this,k)||m.call(this,a):m.call(this,a)}}e.lzld=function(a){-1===v(a,f)&&(l&&u(),q(a,f.push(a)-1))};var w=function(){for(var a=d.getElementsByTagName("img"),b,c=0,e=a.length;c<e;c+=1)b=a[c],b.getAttribute(k)&&
+-1===v(b,f)&&f.push(b);s();setTimeout(h,25)},i=function(a){"readystatechange"===a.type&&"complete"!==d.readyState||(j("load"===a.type?e:d,a.type,i),x||(x=!0,w()))},y=function(){try{d.documentElement.doScroll("left")}catch(a){setTimeout(y,50);return}i("poll")},x=!1,z=!0;if("complete"===d.readyState)w();else{if(d.createEventObject&&d.documentElement.doScroll){try{z=!e.frameElement}catch(D){}z&&y()}g(d,"DOMContentLoaded",i);g(d,"readystatechange",i);g(e,"load",i)}g(e,"load",n);u();var A=d.documentElement.compareDocumentPosition?
+function(a,b){return!!(a.compareDocumentPosition(b)&16)}:d.documentElement.contains?function(a,b){return a!==b&&(a.contains?a.contains(b):!1)}:function(a,b){for(;b=b.parentNode;)if(b===a)return!0;return!1}}(this,document);
View
2  package.json
@@ -1,7 +1,7 @@
{
"name": "lazyload",
"description": "Image lazy loading",
- "version": "0.8.4",
+ "version": "0.8.5",
"repository": {
"type": "git",
"url": "http://github.com/fasterize/lazyload.git"
View
65 test/getattribute.html
@@ -0,0 +1,65 @@
+<!doctype html>
+<html>
+<head>
+ <title>getAttribute test</title>
+ <script src="../lazyload.js"></script>
+ <script src="./jquery-1.7.2.js"></script>
+ <style>html,body{margin:0;padding:0;background:#00f}</style>
+</head>
+<body>
+<script>
+ var srcs = [];
+
+ function done() {
+ var res = 'success';
+ for (var i = 0, max = srcs.length; i < max; i++) {
+ if (srcs[i].indexOf('imgs/1.jpg') === -1) {
+ res = 'error ('+srcs[i]+')';
+ break;
+ }
+ }
+
+ $('#result').text(res);
+ }
+
+ // domready
+ $(function() {
+ srcs.push(document.getElementsByTagName('img')[0].getAttribute('src'));
+ srcs.push(document.getElementById('img1').getAttribute('src'))
+ srcs.push($('#img2').attr('src'));
+ });
+
+ // onload
+ $(window).load(function() {
+ // fails in ie < 8
+ srcs.push($('#img4').attr('src'));
+ done();
+ });
+
+</script>
+
+<p>We checks that in any situation when a script requests an img src, we give real src, not
+ the fake blank image.</p>
+
+<p>A lot of slideshows are trying to get src attributes. Attention, we cannot handle img['src']</p>
+
+<p style="background: #fff">It passes if it says `success`: <span id="result"></span></p>
+
+<img id="img1" style="border:10px solid #000;display:block" width=200 height=200 data-src="imgs/1.jpg" src= onload=lzld(this) onerror=lzld(this) />
+
+<img id="img2" style="border:10px solid #000;display:block" width=200 height=200 data-src="imgs/1.jpg" src= onload=lzld(this) onerror=lzld(this) />
+
+<div style="height:900px"></div>
+
+<img id="img3" style="border:10px solid #000;display:block" width=200 height=200 data-src="imgs/1.jpg" src= onload=lzld(this) onerror=lzld(this) />
+
+<img id="img4" style="border:10px solid #000;display:block" width=200 height=200 data-src="imgs/1.jpg" src= onload=lzld(this) onerror=lzld(this) />
+
+<script>
+ // before dom ready
+ // fails in ie < 8
+ srcs.push(document.getElementById('img3').getAttribute('src'));
+</script>
+
+</body>
+</html>
Please sign in to comment.
Something went wrong with that request. Please try again.