解决获取图片实际尺寸(宽高)的bug
需求:获取图片的宽高其实是为了预先做好排版样式布局做准备。
可以利用图片onload事件监听获取图片的宽高属性值。在IE9以下版本只能使用图片的width与height属性,HTMl5中新加入了naturalHeight与naturalWidth属性,用来标记图片的实际尺寸。 代码如下:
1 //图片地址 2 var imgSrc = 'https://farm4.staticflickr.com/3530/3273073729_4cc8a5c665_z.jpg?zz=1'; 3 4 var img = new Image(); 5 img.onload = function(){ 6 console.log('img onload'); 7 8 if ( typeof img.naturalWidth != 'undefined') { 9 originalImgW = img.naturalWidth; 10 originalImgH = img.naturalHeight; 11 console.log('natural -> ', originalImgH, '~~', originalImgW); 12 } else { 13 originalImgW = img.width; 14 originalImgH = img.height; 15 console.log('IE8 -> ', originalImgH, '~~', originalImgW); 16 } 17 } 18 19 img.src = imgSrc;
但是,在实际项目中使用上述代码却出现了bug.
当浏览器为IE8,并且开启了IE8兼容模式后,在无缓存的情况下,img.width 与 img.height属性获取的值为0,缓存后width与height属性值正常。
在浏览器IE8,不开启IE8兼容模式,关闭浏览器缓存,上述现象几率性出现。
根据现象分析,图片如果缓存了,就可以正确获取图片的尺寸,因此是否可以在无缓存时加载两次图片,获取第二次的尺寸。
将代码更改如下:
1 //图片地址 2 var imgSrc = 'https://farm4.staticflickr.com/3530/3273073729_4cc8a5c665_z.jpg?zz=1'; 3 4 var img = new Image(); 5 6 img.onload = function(){ 7 console.log('img onload'); 8 9 if ( typeof img.naturalWidth != 'undefined') { 10 originalImgW = img.naturalWidth; 11 originalImgH = img.naturalHeight; 12 console.log('natural -> ', originalImgH, '~~', originalImgW); 13 } else { 14 var img4Fix = new Image(); 15 img4Fix.src = imgSrc; 16 originalImgW = img4Fix.width; 17 originalImgH = img4Fix.height; 18 console.log('IE8 with fix -> ', originalImgH, '~~', originalImgW); 19 } 20 } 21 22 img.src = imgSrc;
当img图片onload后,缓存中其实已经有了图片文件。
img4Fix 向同一个地址请求图片,浏览器判断该地址已经存在缓存中,则不再向服务器发送请求,是从缓存中获取图片,获取的缓存图片的尺寸是正确的。
由于第二次请求同一个图片不再向服务器发送请求,所以也无需再监听img4Fix的onload事件。