原生js实现图片预览并上传
最近主导的PC客户端网站重构工程告一段落,下一阶段开始给公司APP开发H5页面,技术栈是react。最近碰到一个需求:需要在H5页面上添加身份证照片,预览并上传。因为要兼容安卓4.4以下版本的手机,所以连html5的新属性formData都用不了,纯原生js实现。
首先获取input输入框,并给其注册onchange事件。
uploadImage(){
let idCardFrontParams={
showID: "img-box1",
flag: "idCardFrontFileId"
};
let $this=this;
//获取页面的input标签
var idCardFrontFile = document.getElementById("idCardFrontFile");
idCardFrontFile.onchange = function () {
let that = this;
$this.preview(that, idCardFrontParams);
};
}
接下来是实现上传并预览功能,预览的关键是使用getObjectURL方法来获得所上传图片的url地址。URL.createObjectURL()方法会根据传入的参数创建一个指向该参数对象的URL,这个URL的生命仅存在于它被创建的这个文档里,新的对象URL指向执行的File对象或者是Blob对象。其语法为:objectURL = window.URL.createObjectURL(blob || file);不同浏览器有差异
获得图片url后调用表单的方法submit将其上传, 并将其地址赋给指定的div。
preview(that, options) {
//接受files数组列表
let _file = that.files,$this=this;
if(_file[]){
let addr = getObjectURL(_file[]);
let curFile=options.flag;
this.setState({isLoading:true});
document.querySelector("#"+curFile).submit();
if(curFile==="deviceFileId"){
$this.setState({deviceFile:true});
}else if(curFile==="idCardFrontFileId"){
$this.setState({idCardFrontFile:true});
}else if(curFile==="idCardbackFileId"){
$this.setState({idCardBackFile:true});
}
var fileUpIframe = document.getElementById("file_upload_iframe");
fileUpIframe.onload = fileUpIframe.onreadystatechange = function () {
try {
var data = JSON.parse(fileUpIframe.contentWindow.document.body.innerHTML);
if (data.code) {
$this.submitParams[curFile]="";
alert("上传图片失败,请刷新页面重试!");
} else {
$this.submitParams[curFile]=data.fileId;
}
$this.setState({isLoading:false});
}catch (err){
console.warn(err);
}
}
let dom =document.getElementById(options.showID);
dom.style.backgroundImage = "url("+addr+")";
}
function getObjectURL(file) {
let url = null;
if (window.createObjectURL != undefined) { // basic
url = window.createObjectURL(file);
} else if (window.URL != undefined) { // mozilla(firefox)
url = window.URL.createObjectURL(file);
} else if (window.webkitURL != undefined) { // webkit or chrome
url = window.webkitURL.createObjectURL(file);
}
return url;
}
}
在这里由于采用的是最原始的表单提交,而submit方法提交后会跳转,又不像ajax可以采用回调函数获取返回值,故需要一个iframe来承载表单提交后的返回值,用target来指向承载返回值的iframe。
<form action="/eyeplus/api/upload" method="post" encType="multipart/form-data" id="idCardbackFileId" target="file_upload_iframe">
<input id="idCardBackFile" type="file" accept="image/*" multiple="multiple" name="file" className="uploadImg"/>
<input type="text" className="ub-hidden-ele ub-token" readOnly="readOnly" name="token" value={this.submitParams.token}/>
<input type="text" className="ub-hidden-ele ub-clientId" readOnly="readOnly" name="client_id" value={this.submitParams.client_id}/>
<input type="text" className="ub-hidden-ele ub-deviceId" readOnly="readOnly" name="device_id" value={this.submitParams.device_id}/>
<input type="text" className="ub-hidden-ele ub-type" readOnly="readOnly" name="type" value="" />
</form>
<iframe style={{display: "none"}} id="file_upload_iframe" name="file_upload_iframe"></iframe>
在确认服务器得到正确上传后的,使用方法JSON.parse(fileUpIframe.contentWindow.document.body.innerHTML)来解析服务器返回值,并与其它所需要的参数组合在一起,就可以愉快的进行下一步操作了。
-----------------------------------------------------补充于2018/06/29--------------------------------------------------------------
以上方法虽然能完成需求开发,但后续测试的时候发现了一个问题:IOS手机是没问题的,只是在安卓手机上,在返回的时候因iframe的存在,导致多级页面从而使得页面标题栏变化但是页面内容没变化,故点击返回的时候并不能准确的返回。不得不说这是个大坑,经反复实践,采用了下面这种方法顺利通过测试。
preview(that, options) {
//接受files数组列表
let _file = that.files, $this = this;
if (_file[]) {
let addr = getObjectURL(_file[]);
let curFile = options.flag;
this.setState({isLoading: true});
if (curFile === "deviceFileId") {
$this.setState({deviceFile: true});
} else if (curFile === "idCardFrontFileId") {
$this.setState({idCardFrontFile: true});
} else if (curFile === "idCardbackFileId") {
$this.setState({idCardBackFile: true});
}
let dom = document.getElementById(options.showID);
dom.style.backgroundImage = "url(" + addr + ")";
let frameId = 'uploadFrame'+ new Date().getTime();
let fileUpIframe = document.createElement("iframe");
fileUpIframe.id = fileUpIframe.name = frameId;
fileUpIframe.style.display = 'none';
document.body.appendChild(fileUpIframe);
//let fileUpIframe = document.getElementById("file_upload_iframe");
fileUpIframe.onload = fileUpIframe.onreadystatechange = function () {
try {
var data = JSON.parse(fileUpIframe.contentWindow.document.body.innerHTML);
if (data.code) {
$this.submitParams[curFile] = "";
alert($this.props.locale.ub_fail_upload);
} else {
$this.submitParams[curFile] = data.fileId;
}
$this.setState({isLoading: false});
document.body.removeChild(fileUpIframe);
} catch (err) {
console.warn(err);
}
};
let form = document.querySelector("#" + curFile);
form.target = frameId;
form.submit();
}
即先去掉HTML文档中存在的iframe,然后在预览上传时create一个iframe,在上传时修改表单提交的target为此新建iframe的ID,上传完成后获取到数据移除掉这个iframe,从而避免了单页面存在多级页面的问题。
原生js实现图片预览并上传的更多相关文章
- 基于“formData批量上传的多种实现” 的多图片预览、上传的多种实现
前言 图片上传是web项目常见的需求,我基于之前的博客的代码(请戳:formData批量上传的多种实现)里的第三种方法实现多图片的预览.上传,并且支持三种方式添加图片到上传列表:选择图片.复制粘贴图片 ...
- 原生JS实现图片预览功能
html代码: <div class="album-new fr"> <div class="upload-btn btn-new container& ...
- H5-FileReader实现图片预览&Ajax上传文件
图片预览 <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF- ...
- js多图预览及上传功能
<%-- Created by IntelliJ IDEA. User: Old Zhang Date: 2018/12/27 Time: 11:17 To change this templa ...
- 前端图片预览,上传前预览,兼容IE7、8、9、10、11,Firefox,Chrome(学习到的知识)
文章地址:http://www.cnblogs.com/rubylouvre/p/4597344.html 一.window.URL 在Chrome中,window.URL和window.webkit ...
- jquery+html5+canvas实现图片 预览 压缩 上传
javascirpt (function($){ $.fn.extend({ aiiUpload:function(obj) { if(typeof obj !="object") ...
- vue组件利用formdata图片预览以及上传《转载》
转载修改 在项目中直接新建一个单文件页,复制一下代码即可 upload组件: <template> <div class="vue-uploader" ...
- vue组件利用formdata图片预览以及上传
转载修改 在项目中直接新建一个单文件页,复制一下代码即可 upload组件: <template> <div class="vue-uploader" ...
- 用js实现预览待上传的本地图片
js实现预览待上传的本地图片,代码如下: <form name="form5" id="form5" method="post" ac ...
随机推荐
- **后台怎么处理JSON数据中含有双引号?
http://bbs.csdn.net/topics/390578406?page=1 注意是后台,不是用js另外我这个json是直接取得别人的传过来的字符串,不是我自己拼写的,所以我自己不能做到转义 ...
- 如何查看K8S的网络是否完好
今天工作中遇到这个问题, 检查从以下几个方面入手. 一,查看各个POD的LOG,如果有错误,则要解决了再继续 二,登陆各个POD之间,互相要能PING通. 三,在物理节点上可以PING通SERVICE ...
- 1391: [Ceoi2008]order
有N个工作,M种机器,每种机器你可以租或者买过来. 每个工作包括若干道工序,每道工序需要某种机器来完成,你可以通过购买或租用机器来完成. 现在给出这些参数,求最大利润 Input 第一行给出 N,M( ...
- python 判断字符编码
一般情况下,需要加这个: import sys reload(sys) sys.setdefaultencoding('utf-8') 打开其他文件编码用codecs.open 读 下面的代码读取了文 ...
- ubuntu安装redis 和可视化工具
方式一: 下载地址:http://redis.io/download,下载最新文档版本. 本教程使用的最新文档版本为 2.8.17,下载并安装: $ wget http://download.redi ...
- explicit 显示的类型转换运算符
C++提供了关键字explicit,可以阻止不应该允许的经过转换构造函数进行的隐式转换的发生.声明为explicit的构造函数不能在隐式转换中使用. 调用构造函数可以分为显示调用和隐式调用,当用赋值初 ...
- 【LOJ】#2574. 「TJOI2018」智力竞赛
题解 二分答案 求最小路径点覆盖 由于这里最小路径点覆盖,点是可重的,用floyd求出传递闭包(也就是求出,哪两点之间是可达的) 最后用这个floyd求出的数组建出一个新图,在这个图上跑普通的最小路径 ...
- HDU 3530 单调队列
题目大意:给你n个数, 让你问你最长的满足要求的区间有多长,区间要求:MAX - MIN >= m && MAX - MIN <= k 思路:单调队列维护递增和递减,在加入 ...
- JSP的学习二(请求转发与 重定向)
一: 1.介绍知识点 1). 本质区别: 请求的转发只发出了一次请求, 而重定向则发出了两次请求. 具体: ①. 请求的转发: 地址栏是初次发出请求的地址. 请求的重定向: 地址栏不再是初次发出的请 ...
- PHP程序员未来路在何方
PHP 从诞生到现在已经有20多年历史,从Web时代兴起到移动互联网退潮,互联网领域各种编程语言和技术层出不穷, Node.js . GO . Python 不断地在挑战 PHP 的地位.这些技术的推 ...