最近主导的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实现图片预览并上传的更多相关文章

  1. 基于“formData批量上传的多种实现” 的多图片预览、上传的多种实现

    前言 图片上传是web项目常见的需求,我基于之前的博客的代码(请戳:formData批量上传的多种实现)里的第三种方法实现多图片的预览.上传,并且支持三种方式添加图片到上传列表:选择图片.复制粘贴图片 ...

  2. 原生JS实现图片预览功能

    html代码: <div class="album-new fr"> <div class="upload-btn btn-new container& ...

  3. H5-FileReader实现图片预览&Ajax上传文件

    图片预览 <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF- ...

  4. js多图预览及上传功能

    <%-- Created by IntelliJ IDEA. User: Old Zhang Date: 2018/12/27 Time: 11:17 To change this templa ...

  5. 前端图片预览,上传前预览,兼容IE7、8、9、10、11,Firefox,Chrome(学习到的知识)

    文章地址:http://www.cnblogs.com/rubylouvre/p/4597344.html 一.window.URL 在Chrome中,window.URL和window.webkit ...

  6. jquery+html5+canvas实现图片 预览 压缩 上传

    javascirpt (function($){ $.fn.extend({ aiiUpload:function(obj) { if(typeof obj !="object") ...

  7. vue组件利用formdata图片预览以及上传《转载》

    转载修改 在项目中直接新建一个单文件页,复制一下代码即可       upload组件: <template> <div class="vue-uploader" ...

  8. vue组件利用formdata图片预览以及上传

    转载修改 在项目中直接新建一个单文件页,复制一下代码即可       upload组件: <template> <div class="vue-uploader" ...

  9. 用js实现预览待上传的本地图片

    js实现预览待上传的本地图片,代码如下: <form name="form5" id="form5" method="post" ac ...

随机推荐

  1. ubuntu 安装chrome 和chromedriver

    1. chromedriver 下载地址:  https://npm.taobao.org/mirrors/chromedriver 在这里找到对应的驱动 2. 安装谷歌浏览器 2.1 安装依赖 ap ...

  2. [USACO16OPEN]262144

    传送门啦 其实大家可以先看一下这个题 [USACO16OPEN]248 分析: 数据范围很奇特:n特别,a[i]特别——如果O(N^3)能接受就直接区间DP水过了,但是不行,于是考虑设计一个状态囊括a ...

  3. Unix IPC之基于共享内存的计数器

    目的 本文主要实现一个基于共享内存的计数器,通过父子进程对其访问. 本文程序需基于<<Unix网络编程-卷2>>的环境才能运行.程序中大写开头的函数为其小写同名函数的包裹函数, ...

  4. hdu 1028 整数划分 (母函数)

    假如输入44 = 4;4 = 3 + 1;4 = 2 + 2;4 = 2 + 1 + 1;4 = 1 + 1 + 1 + 1;一共5种 假如输入3 用母函数的方法就是写成(1+X+X2+X3)(1+X ...

  5. 《精通Python设计模式》学习之抽象工厂

    这种工厂模式用得少, 可能在游戏类的编程中用得比较多吧. 这个思路清晰一定要OK的. class Frog: def __init__(self, name): self.name = name de ...

  6. 【51nod】1565 模糊搜索

    题解 这个字符集很小,我们可以把每个字符拿出来做一次匹配,把第一个字符串处理每个出现过的该字符处理成一个区间加,即最后变成第一个字符串的该位置能够匹配某字符 例如对于样例 10 4 1 AGCAATT ...

  7. 最小圆覆盖 gym-102006 I

    #include<bits/stdc++.h> #define LL long long #define fi first #define se second #define mk mak ...

  8. 三 oracle 用户管理一

    一.创建用户概述:在oracle中要创建一个新的用户使用create user语句,一般是具有dba(数据库管理员)的权限才能使用.create user 用户名 identified by 密码; ...

  9. HBase集群的搭建

    HBase集群的搭建(在<HBase伪分布式安装>基础上搭建) 1 集群结构,主节点(hmaster)是hadoop0,从节点(region server)是hadoop1和hadoop2 ...

  10. Mongoose关于当天日期的查询

    参考:https://blog.csdn.net/difffate/article/details/70312894 Ant Design Pro中,有关于日期的查询条件,但日期是以数字表示的 Req ...