搞清Image加载事件(onload)、加载状态(complete)后,实现图片的本地预览,并自适应于父元素内(完成)
onload与complete介绍
complete只是HTMLImageElement对象的一个属性,可以判断图片加载完成,不管图片是不是有缓存;而onload则是这个Image对象的load事件回调,当图片加载完成后执行onload绑定的函数。
给下面一个例子,解释下:
document.getElementById('load').onclick = function() {
var img = new Image();
img.src="data:images/avatar.png";
if(img.complete) {
console.log('dd');
}
img.onload = function() {
console.log('ff');
}
}
打印结果:
第一次点击,谷歌浏览器结果为:dd;IE浏览器结果为:ff。
第二次点击,谷歌浏览器结果为:dd,ff;IE浏览器结果为:ff。
第三次点击,谷歌浏览器结果为:dd,ff;IE浏览器结果为:dd,ff。
...
例2:
document.getElementById('load').onclick = function() {
var img = new Image();
if(img.complete) {
console.log('dd');
}
img.onload = function() {
console.log('ff')
}
img.src="data:images/avatar.png";
}
打印结果:
第一次点击,谷歌浏览器结果为:dd,ff;IE浏览器结果为:ff。
第二次点击,谷歌浏览器结果为:dd,ff;IE浏览器结果为:ff。
第二次点击,谷歌浏览器结果为:dd,ff;IE浏览器结果为:ff。
...
例3:
document.getElementById('load').onclick = function() {
var img = new Image();
if(img.complete) {
console.log('dd');
}
img.onload = function() {
console.log('ff')
}
img.src="";
}
打印结果:
第一次点击,谷歌浏览器结果为:dd;IE浏览器结果为:空。
第二次点击,谷歌浏览器结果为:dd;IE浏览器结果为:空。
第二次点击,谷歌浏览器结果为:dd;IE浏览器结果为:空。
...
根据结果得出:对于 complete 属性来讲,IE是根据图片是否显示过来判断,就是说当加载的图片显示出来后,complete 属性的值才为 true ,否则一直是 false ,和以前是否加载过该张图片没有关系,即和缓存没有关系!但是其它浏览器表现出来的确不一样,只要以前加载过该图,浏览器有缓存,也无论src是否有值,成功与否,只要获取到image,就可以执行,complete 就为 true。所以这个complete在不同浏览器中结果是不一样的。
本地图片预览
首先先写下布局,html代码:
<div class="centerView">
<div class="localPreview">
<img id="showViewImg"/>
</div>
<a class="inputParent" href="javascript:void(0)">
<i>点击上传文件</i>
<input type="file" id="filePath" onchange="getCurrFile()"/>
</a>
</div>
css代码:
.centerView{
width:150px;
}
.localPreview{
position:relative;
width:150px;
height:150px;
line-height:150px;
text-align:center;
background:#ccc;
}
.localPreview img{
position: relative;
vertical-align: middle;
}
.inputParent{
position:relative;
display:block;
margin:10px auto;
cursor:pointer;
width:80px;
height:30px;
line-height:30px;
background:#27bb6e;
text-align: center;
font-size:12px;
color:#fff;
}
.inputParent i{
font-style: normal;
color:#fff;
}
.inputParent #filePath{
position:absolute;
width:100%;
height:100%;
top:;
left:;
filter:alpha(opacity=0);
opacity:;
}
静态页面的效果如图所示:

梳理一下思路,我们要实现图片的本地预览,需要如下几点:
1.点击file上传文件按钮后,选中图片后,获得图片的路径。
2.根据图片实例一个new Image()得到图片的实际的大小。
3.得到图片的实际大小,再根据显示区域的宽高来处理图片的宽高,让其自适应于父元素区域中。
4.在IE9以及低版本浏览器中需要使用滤镜来实现图片的预览。
根据以上几点我们就写如下代码,首先我们先创建一个构造函数。
function DealPic(width,height){
this.oriWidth = width;
this.oriHeight = height;
}
这个oriWidth与oriHeight指的是父区域的宽高,也就是图片要跟该宽高进行比较的值。
接下来实现一个getObjectURL,干嘛的呢,如果支持file对象支持files,就返回只包含url的一个对象,如果是IE9以及低版本浏览器返回的对象中还包括滤镜图片的原始大小。
DealPic.prototype.getObjectURL = function(fileObj){
var result = {} ;
var file;
if(fileObj.files){
file = fileObj.files[0];
if (window.createObjectURL!=undefined) { // basic
result.url = window.createObjectURL(file) ;
}else if (window.URL!=undefined) { // mozilla(firefox)
result.url = window.URL.createObjectURL(file) ;
}else if (window.webkitURL!=undefined) { // webkit or chrome
result.url = window.webkitURL.createObjectURL(file) ;
}
}else{
var hiddenAlphaImageWidth,hiddenAlphaImageHeight;
var hiddenAlphaImage = document.createElement('img');
document.body.appendChild(hiddenAlphaImage);
fileObj.select();
fileObj.blur();
result.url = document.selection.createRange().text;
hiddenAlphaImage.style.filter = "progid:DXImageTransform.Microsoft.AlphaImageLoader(sizingMethod=image)";
hiddenAlphaImage.filters.item("DXImageTransform.Microsoft.AlphaImageLoader").src = result.url;
//但是当滤镜使用的图片超过10M大小,使用上面的代码页面会报错,说hiddenAlphaImage出现未指明的错误;
//解决办法就是使用下面的注释的方式,注释上面的两行代码
//使用下面代码滤镜图片超过10M后本地预览不了,通过这个滤镜得到的图片的宽高始终是28*30
//hiddenAlphaImage.style.filter = "progid:DXImageTransform.Microsoft.AlphaImageLoader(sizingMethod='image',src=\"" + result.url + "\")";
result.width = hiddenAlphaImage.offsetWidth;
result.height = hiddenAlphaImage.offsetHeight;
if(hiddenAlphaImage.parentNode){
hiddenAlphaImage.parentNode.removeChild(hiddenAlphaImage);
}
}
return result;
}
在IE低版本浏览器为什么要这样处理呢,如果我们要得到滤镜图片的元素大小,首先得创建一个img元素,然后通过IE浏览器的document.selection.createRange().text得到图片路径,然后给这个img元素进行设置,这儿关键得用到filter的sizingMethod属性。
sizingMethod属性:可选值,设置或检索的方式来显示一个图像在对象边界显示方式。有三个值:
- crop:裁剪图像以适应对象的尺寸;
- image:默认值,扩大或减少对象的边界,以适应图像的尺寸;
- scale:伸展或收缩图像填充对象的边界;
这儿使用image才能得到滤镜图片的原始大小。然后返回。
如果一开始只是把这个url返回回去,没有返回滤镜图片的实际大小,就不能达到自适应的效果。
当然上面获取图片的url用到的是window.createObjectURL,也可以用FileReader.readAsDataURL读取指定Blob或File的内容。
简单实现一下:
if (input.files && input.files[0]) {
var reader = new FileReader();
reader.onload = function (e) {
var showImg = document.getElementById('showViewImg');
showImg.src = e.target.result;
showImg.style.width = '150px';
showImg.style.height = '80px';
};
reader.readAsDataURL(input.files[0]);
}
这儿就不详细介绍了,只是这儿得到的url是base64编码的字符串,所以我一般还是选中上面第一种方式。
接下来就是图片自适应的比较方法:
DealPic.prototype.getPicResult = function(targetWidth,targetHeight,callback){
if(this.oriWidth / this.oriHeight > targetWidth / targetHeight){
var th = this.oriHeight;
var tw = this.oriHeight / targetHeight * targetWidth;
}else{
var tw = this.oriWidth;
var th = this.oriWidth / targetWidth * targetHeight;
}
if(callback){
callback(tw,th);
}
}
这儿就不细说了。
最后就是绑定到file按钮上的change事件的方法了。
function getCurrFile(){
var fileObj = document.getElementById('filePath');
var showImgObj = document.getElementById('showViewImg');
var newPicObj = new DealPic(150,150);
var resultFileObj = newPicObj.getObjectURL(fileObj);
if(fileObj.files){
var newImg = new Image();
newImg.onload = function(){
newPicObj.getPicResult(newImg.width,newImg.height,function(tw,th){
showImgObj.style.width = tw + 'px';
showImgObj.style.height = th + 'px';
});
}
newImg.src = resultFileObj.url;
showImgObj.setAttribute('src',resultFileObj.url);
}else{
showImgObj.style.filter = "progid:DXImageTransform.Microsoft.AlphaImageLoader(sizingMethod=scale)";
showImgObj.filters.item("DXImageTransform.Microsoft.AlphaImageLoader").src = resultFileObj.url;
//IE9低版本不设置图片src会显示裂图,所以设置一个透明图片或者base64的透明图片
showImgObj.setAttribute('src','./images/transparent.png');
//showImgObj.src = 'data:image/gif;base64,R0lGODlhAQABAIAAAP///wAAACH5BAEAAAAALAAAAAABAAEAAAICRAEAOw==';
newPicObj.getPicResult(resultFileObj.width,resultFileObj.height,function(resw,resh){
showImgObj.style.width = resw + 'px';
showImgObj.style.height = resh + 'px';
});
}
}
最后的js代码总结:
function DealPic(width,height){
this.oriWidth = width;
this.oriHeight = height;
}
DealPic.prototype.getObjectURL = function(fileObj){
var result = {} ;
var file;
if(fileObj.files){
file = fileObj.files[0];
if (window.createObjectURL!=undefined) { // basic
result.url = window.createObjectURL(file) ;
}else if (window.URL!=undefined) { // mozilla(firefox)
result.url = window.URL.createObjectURL(file) ;
}else if (window.webkitURL!=undefined) { // webkit or chrome
result.url = window.webkitURL.createObjectURL(file) ;
}
}else{
var hiddenAlphaImageWidth,hiddenAlphaImageHeight;
var hiddenAlphaImage = document.createElement('img');
document.body.appendChild(hiddenAlphaImage);
fileObj.select();
fileObj.blur();
result.url = document.selection.createRange().text;
hiddenAlphaImage.style.filter = "progid:DXImageTransform.Microsoft.AlphaImageLoader(sizingMethod=image)";
hiddenAlphaImage.filters.item("DXImageTransform.Microsoft.AlphaImageLoader").src = result.url;
//但是当滤镜使用的图片超过10M大小,使用上面的代码页面会报错,说hiddenAlphaImage出现未指明的错误;
//解决办法就是使用下面的注释的方式,注释上面的两行代码
//使用下面代码滤镜图片超过10M后本地预览不了,通过这个滤镜得到的图片的宽高始终是28*30
//hiddenAlphaImage.style.filter = "progid:DXImageTransform.Microsoft.AlphaImageLoader(sizingMethod='image',src=\"" + result.url + "\")";
result.width = hiddenAlphaImage.offsetWidth;
result.height = hiddenAlphaImage.offsetHeight;
if(hiddenAlphaImage.parentNode){
hiddenAlphaImage.parentNode.removeChild(hiddenAlphaImage);
}
}
return result;
}
DealPic.prototype.getPicResult = function(targetWidth,targetHeight,callback){
if(this.oriWidth / this.oriHeight > targetWidth / targetHeight){
var th = this.oriHeight;
var tw = this.oriHeight / targetHeight * targetWidth;
}else{
var tw = this.oriWidth;
var th = this.oriWidth / targetWidth * targetHeight;
}
if(callback){
callback(tw,th);
}
}
function getCurrFile(){
var fileObj = document.getElementById('filePath');
var showImgObj = document.getElementById('showViewImg');
var newPicObj = new DealPic(150,150);
var resultFileObj = newPicObj.getObjectURL(fileObj);
if(fileObj.files){
var newImg = new Image();
newImg.onload = function(){
newPicObj.getPicResult(newImg.width,newImg.height,function(tw,th){
showImgObj.style.width = tw + 'px';
showImgObj.style.height = th + 'px';
});
}
newImg.src = resultFileObj.url;
showImgObj.setAttribute('src',resultFileObj.url);
}else{
showImgObj.style.filter = "progid:DXImageTransform.Microsoft.AlphaImageLoader(sizingMethod=scale)";
showImgObj.filters.item("DXImageTransform.Microsoft.AlphaImageLoader").src = resultFileObj.url;
//IE9低版本不设置图片src会显示裂图,所以设置一个透明图片或者base64的透明图片
showImgObj.setAttribute('src','./images/transparent.png');
//showImgObj.src = 'data:image/gif;base64,R0lGODlhAQABAIAAAP///wAAACH5BAEAAAAALAAAAAABAAEAAAICRAEAOw==';
newPicObj.getPicResult(resultFileObj.width,resultFileObj.height,function(resw,resh){
showImgObj.style.width = resw + 'px';
showImgObj.style.height = resh + 'px';
});
}
}
最后本地预览的效果如图所示:

搞清Image加载事件(onload)、加载状态(complete)后,实现图片的本地预览,并自适应于父元素内(完成)的更多相关文章
- image的加载事件(onload)和加载状态(complete)
之前做过这样一个需求,要让商家页的商家图片按照图片外面box的大小等比例缩放.之前的想法是在页面中先输出图片的src,然后在页面底部初始化js,然后在js中写相应的可以使图片按照box的大小等比例缩放 ...
- JS---BOM基本知识 (顶级对象,系统对话框,加载事件,location对象, history对象, navigator对象)
BOM JavaScript分三个部分: 1. ECMAScript标准---基本语法 2. DOM--->Document Object Model 文档对象模型,操作页面元素的 3. BOM ...
- 利用js加载本地图片预览功能
直接上代码: 经测试,除safari6包括6以下不支持,其他均可正常显示. 原因:safari6不支持filereader,同时不能使用IE滤镜导致失效. fix: 可以利用canvas,解决safa ...
- JS事件 加载事件(onload)注意:1. 加载页面时,触发onload事件,事件写在<body>标签内。 2. 此节的加载页面,可理解为打开一个新页面时。
加载事件(onload) 事件会在页面加载完成后,立即发生,同时执行被调用的程序. 注意:1. 加载页面时,触发onload事件,事件写在<body>标签内. 2. 此节的加载页面,可理解 ...
- jquery mobile在页面加载时添加加载中效果 document.ready 和window.onload执行顺序比较
想要添加这个效果,先来弄明白页面的加载和事件执行顺序,看这个简单例子: <html xmlns="http://www.w3.org/1999/xhtml"> < ...
- 深入理解DOM事件类型系列第六篇——加载事件
前面的话 提到加载事件,可能想到了window.onload,但实际上,加载事件是一大类事件,本文将详细介绍加载事件 load load事件是最常用的一个事件,当页面完全加载后(包括所有图像.java ...
- 从零开始学 Web 之 jQuery(一)jQuery的概念,页面加载事件
大家好,这里是「 从零开始学 Web 系列教程 」,并在下列地址同步更新...... github:https://github.com/Daotin/Web 微信公众号:Web前端之巅 博客园:ht ...
- 支持xhr浏览器:超时设定、加载事件、进度事件
各个浏览器虽然都支持xhr,但还是有些差异. 1.超时设定 IE8为xhr对象添加了一个timeout属性,表示请求在等待响应多少毫秒后就终止.再给timeout这只一个数值后,如果在规定的时间内浏览 ...
- js 动态加载事件的几种方法总结
本篇文章主要是对js 动态加载事件的几种方法进行了详细的总结介绍,需要的朋友可以过来参考下,希望对大家有所帮助 有些时候需要动态加载javascript事件的一些方法往往我们需要在 JS 中动态添 ...
随机推荐
- 第41节:Java当中的类集框架
Java当中的类集框架 类集框架,那么什么是类集框架,集合的种类有哪些,类集框架的基础结构. 类集框架是一组类和接口的集合,位于java.util包当中,是用来用户存储和管理对象的,在这个类集合框架中 ...
- Kali学习笔记12:服务扫描
关于什么是服务扫描不多介绍,通俗来看: 我已经扫描到目标机器某个端口开放,接下来我需要知道开放这个端口的是什么应用 情景: 我的Kali机器IP地址:192.168.22.130 我要扫描的Metas ...
- Spring详解(八)------事务管理
PS:本篇博客源码下载链接:http://pan.baidu.com/s/1mi3NhX2 密码:3io2 1.事务介绍 事务(Transaction),一般是指要做的或所做的事情.在计算机术语中是指 ...
- Python - 使用Pylint检查分析代码
1-简介 Home Page : https://www.pylint.org/ 检查语法错误,是否遵守编码风格标准.潜在的问题等: 支持自定义配置:例如显示或隐藏特定的警告,并且可以通过编写插件来添 ...
- python(leetcode)-14最长公共前缀
编写一个函数来查找字符串数组中的最长公共前缀. 如果不存在公共前缀,返回空字符串 "". 示例 1: 输入: ["flower","flow" ...
- 将本地代码上传到github走过的坑
1.因为github是服务端,需要自己在自己的电脑上安装一个客户端git 2.配置SSH(配置一次就好了) github是不能随便上传代码上去的,而是通过一种网络协议---SSH授权的.SSH是一种网 ...
- mysql 下 计算 两点 经纬度 之间的距离(转)
公式如下,单位米: 第一点经纬度:lng1 lat1 第二点经纬度:lng2 lat2 round(6378.138*2*asin(sqrt(pow(sin( (lat1*pi()/180-lat2* ...
- Salesforce Sales Cloud 零基础学习(一) Product 和 Price Book
以前的博客大部分都是基于force.com以及lightning展开的自定义开发,其实salesforce提供了很多的标准的功能以及平台, Sales Cloud便是作为Salesforce核心的平台 ...
- ③JSP经典回顾
jsp概述 jsp实际就是一个高级servlet,比servlet容易很多.jsp/servlet在jsp容器中运行.例如,Tomcat就是一个Servlet/jsp容器. 关于tomcat:[传送门 ...
- 减少重复代码的书写--Lombok
本文版权归 远方的风lyh和博客园共有,欢迎转载,但须保留此段声明,并给出原文链接,谢谢合作. 在开发中,比如我们要写一个modol 实体类 要编写 get.set 这些获取和填值的方法,这样写也没错 ...