移动端图片操作(一)——上传

上传我们一般都是用“input[type=file]”控件。当你用此控件时,你就授权了网页和服务器访问对应的文件,就可以得到File对象。

友情提示在,在Android手机webview中,是不支持上传文件的,网上说是修改Android端的代码,但我没试过,我们这边是使用客户端提供的接口来实现上传的。

下面的示例代码可以在这里查看到。

一、accept属性

该属性表明了服务器端可接受的文件类型,可以限制你手机选择相关的文件,如果限制多个,可以用逗号分割,下面的代码就表示只能选择图片与音频相关的文件:

<input accept="image/*,audio/*" type="file"/>

在移动端,点击后会让你选择拍照或相册,还是蛮高大上的。下图是UC浏览器中:

二、change事件

一般选择文件都会使用“change”事件,下面的代码就是绑定了change事件,弹出文件大小:

var upload = document.getElementById(‘upload‘);
upload.addEventListener(‘change‘, function() {
  var file = upload.files[0];
  alert(file.size);
}, false);

1) 有些手机浏览器在点击的时候,会弹出键盘选择,我用onfocus="this.blur()",来强制失去焦点。

<input type="file" id="upload" onfocus="this.blur()"/>

2) 当选择过一次后,再次选择同一个文件,“change”事件不会触发,因为value没有改变,在网上看到个方法,我还没有在实际项目中使用,兼容性有待考证。

使用“Node.cloneNode”复制上传元素,再用“Node.replaceChild”替换节点。

这里注意下:克隆一个元素节点会拷贝它所有的属性以及属性值,但不会拷贝那些使用addEventListener()方法或者node.onclick = fn用JavaScript动态绑定的事件。

upload.addEventListener(‘change‘, function() {
  var upload = document.getElementById(‘upload‘); //每次要动态获取
  var file = upload.files[0];
  console.log(file.size);

  //解决上传相同文件不触发onchange事件
  var clone = upload.cloneNode(true);
  clone.onchange = arguments.callee; //克隆不会复制动态绑定事件
  clone.value = ‘‘;
  upload.parentNode.replaceChild(clone, upload);
}, false);

三、File对象

用户所选择的文件都存储在了一个FileList对象上,其中每个文件都对应了一个File对象

File对象负责处理那些以文件形式存在的二进制数据,也就是操作本地文件。

File对象是Blob【下面会提到】的特殊类型,即大块的二进制数据,File对象的尺寸及类型等属性都继承自Blob。

1)File对象可以通过3种方式获取:

1. <input>元素上选择文件后返回的FileList对象中的成员

2. 拖放操作【Drag或Drop】生成的 DataTransfer对象内files属性中的成员

3. HTMLCanvasElement上执行mozGetAsFile()方法后的返回结果

document.getElementById(‘upload‘).files[0]//选取第一个文件对象

2)File对象有9个属性,这里就只介绍3个:

1. name:当前File对象所引用文件的文件名,不包括路径,只读。

2. size:文件大小,单位为字节,只读的64位整数.

3. type:MIME类型,只读字符串,如果类型未知,则返回null。有些移动端的浏览器明明选择了图片,返回的却是null,非常坑。

还有3个非标准的方法:getAsBinary()、getAsDataURL()和getAsText(in DOMString encoding)。

这3个方法现在已经过时,现在用FileReader对象中的方法来取代。

四、FileReader

web应用程序可以异步的读取存储在用户计算机上的文件(或者原始数据缓冲)内容,可以使用File对象或者Blob对象来指定所要处理的文件或数据。

1) readAsArrayBuffer():在返回的result属性中将包含一个ArrayBuffer对象【缓冲数组,是一种用于呈现通用、固定长度的二进制数据的类型】以表示所读取文件的内容

Blob可以“append”,ArrayBuffer数据。ArrayBuffer存在的意义就是作为数据源提前写入在内存中,就是提前钉死在某个区域,长度也固定。

2) readAsBinaryString():result属性中将包含所读取文件的原始二进制数据

3) readAsDataURL():result属性中将包含一个data: URL格式的字符串以表示所读取文件的内容

4) readAsText():result属性中将包含一个字符串以表示所读取的文件内容

下面的代码是获取data:URL,可以将返回的result内容赋值给img的src,用于预览等操作。

var reader = new FileReader();
reader.readAsDataURL(file);
reader.onload = function(e) {
  var img = new Image();
  img.src = this.result;
  console.log(this.result);
};
console.log(this.result)内容如下:

五、URL对象

URL对象是硬盘上指向文件的URL。上面的例子中获取图片的引用,通过读取data URI,data URI是个一大串的字符。

图片原本就在硬盘上,还要转换成另一个格式再用,有点绕了,完全可以直接引用文件的URL,下面是两个方法:

1) URL.createObjectURL():接收一个文件的引用(File或Blob对象)返回一个URL对象

2) URL.revokeObjectURL():销毁创建的URL

var url = URL.createObjectURL(file);
var img = new Image();
img.src = url;
img.onload = function(e) {
  window.URL.revokeObjectURL(this.src); //销毁
}
console.log(url);

console.log(url)内容如下:

在移动端需要做个兼容性判断:

window.URL = window.URL || window.webkitURL;

六、Blob对象

Blob(binary large object)对象代表了一段二进制数据,就是一个包含只读原始数据的类文件对象。

File接口基于Blob,继承了Blob的功能,并且扩展支持了用户计算机上的本地文件。

1)创建Blob对象的4种方法:

1. 调用Blob构造函数

2. 使用一个已有Blob对象上的slice()方法切出另一个Blob对象

3. 调用canvas对象上的toBlob方法

4. 过气的方法,通过BlobBuilder接口创建,但兼容性不好,并且现有的BlobBuilder实现都是带前缀的

2)利用Blob对象,生成可下载文件

var blob = new Blob(["pwstrick"]);//数组中添加DOMString对象
var a = document.createElement("a");
a.href = URL.createObjectURL(blob);//创建URL对象
a.download = "test.txt";//HTML5新属性
a.textContent = "test";
document.getElementsByTagName(‘body‘)[0].appendChild(a);

生成一个“a”标签,并且点击这个链接,可以下载一个txt文本,内容是“pwstrick”。

3)通过slice方法,将二进制数据按照字节分块,返回一个新的Blob对象

upload.addEventListener(‘change‘, function() {
  var upload = document.getElementById(‘upload‘); //每次要动态获取
  var file = upload.files[0];
  var start = 0;
  var chunk = 1024 * 10; //10KB
  var end = start + chunk;
  var size = file.size;
  while (start < size) {
    segment(file, start, end);
    start = end;
    end = start + chunk;
    if (end > size) {
      end = size;
    }
  }
}, false);

function segment(file, start, end) {
  var reader = new FileReader();
  reader.onload = function(evt) {
    console.log([‘Read bytes: ‘, start, ‘ - ‘, end].join(‘‘));
  };
  var blob = file.slice(start, end);
  reader.readAsBinaryString(blob);
}

七、formData

XMLHttpRequest Level 2添加了一个新的接口FormData

利用FormData对象,可以使用键值对来模拟一个完整的表单,然后使用XMLHttpRequest发送这个"表单"。

使用FormData的最大优点就是我们可以异步上传一个二进制文件。

  var formData = new FormData();
  formData.append("name", "value");//普通键值对
  formData.append("blob", blob); //传递一个blob对象
  formData.append("file", file); //传递一个file对象
  var oReq = new XMLHttpRequest();
  oReq.open("POST", "http://xx.com");
  oReq.send(formData);

参考资料:

https://developer.mozilla.org/zh-CN/docs/Using_files_from_web_applications    在web应用中使用文件

http://javascript.ruanyifeng.com/htmlapi/file.html    文件和二进制数据的操作

http://www.iunbug.com/archives/2012/06/04/208.html   [译]JavaScript文件操作基础

http://www.iunbug.com/archives/2012/06/05/254.html   [译]JavaScript文件操作URL对象

http://www.html5rocks.com/zh/tutorials/file/dndfiles/     通过 File API 使用 JavaScript 读取文件

时间: 2024-08-08 09:41:33

移动端图片操作(一)——上传的相关文章

UEditor+七牛,实现图片直连上传

最近做的项目,涉及到使用富文本编辑器,我选择了百度的UEditor.同时,我们的图片放在七牛云存储上.关于这两者间的集成,我写下一些个人的经验,与大家分享. 图片上传方案 目前来说,Web端基于七牛等云存储的图片上传方式分为以下两种: 1. 上传图片至服务端,再将数据转发至七牛. 通过服务端接受用户上传的内容,同时可以对内容进行有效性审核,拒绝不合规范的内容,然后从服务端将内容上传至七牛. 这种方法可以有效控制并记录用户提交的内容,但同时也增加了服务器的运行压力. 2. 直接上传图片至七牛,然后

HTML5实现图片文件异步上传

利用HTML5的新特点做文件异步上传非常简单方便,本文主要展示JS部分,html结构.下面的代码并未使用第三发库,如果有参照,请注意一些未展现出来的代码片段.我这边的效果预览: 1.文件未选择 2.文件已选择 HTML代码部分: 思路:下面代码中我利用css的z-index属性将input="file"标签隐藏在了id=btnSelect元素下面,通过触发a标签的点击后,弹出文件选择框.下面的masklayer用于点击确认按钮后的弹出层,避免用户重复点击确认按钮. <div id

基于JSP+Servlet+JavaBean的图片或文件上传

基于JSP+Servlet+JavaBean的图片或文件上传 一.概述 现在不管是博客论坛还是企业办公,都离不开资源的共享.通过文件上传的方式,与大家同分享,从而达到大众间广泛的沟通和交流,我们既可以从中获得更多的知识和经验,也能通过他人的反馈达到自我改进和提升的目的. 下面我就为大家介绍 web项目中的这一上传功能,那么文件是如何从本地发送到服务器的呢?大家可以在在线视频课程进修学习<基于JSP+Servlet+JavaBean的人力资源管理系统开发>中第22课-项目开发-其它功能完善-图片

基于h5的图片无刷新上传(uploadifive)

基于h5的图片无刷新上传(uploadifive) uploadifive简介 了解uploadify之前,首先了解来一下什么是uploadify,uploadfy官网,uploadify和uploadifive是一家的,他们都是基于jquery的插件,都支持多文件异步上传,支持显示上传进度,不同的是uploadify基于swfUpload这一开源无刷新上传插件开发,基于flash,而uploadifive则是基于html5,不依赖于flash. 基于他们的不同点,我们可以根据自己的需求来进行选

Web 本地图片 canvas 截取上传

我做了一个 Web 本地图片 canvas 截取上传 的demo.发现了几个问题,记录下: 1.  Canvas 元素大小 (css width height) 和表面大小(canvas 自身的 width height 属性)两个概念是不一样的,当两个大小不一致时,坐标需要进行转换计算. // 其中 x, y 是视口坐标 function windowToCanvas(canvas, x, y) { var bbox = canvas.getBoundingClientRect(); retu

CANVAS运用-对图片的压缩上传(仅针对移动浏览器)

最近在移动端设计头像上传功能时,原本是以<input type="file">直接通过formData上传,然而实际使用情况是:对于过大的图片(高像素手机所拍摄的照片等)上传时间过长会导致上传失败,而每次都上传原始大小的图片(后台处理压缩)十分影响用户体验,所以研究了一下通过canvas压缩图片并上传的方法,以下是整理的一些思路和心得: 一.<input type="file">获取本地图片,并将图片绘制到画布中.此处的难点在于:由于浏览器的

前端获取图片压缩后上传给后台

 此前有同事跟我聊过关于移动端用canvas压缩图片后再上传的功能,最近有了点空闲时间,所以就实践了一下.demo效果链接在文章底部贴出. 在做移动端图片上传的时候,用户传的都是手机本地图片,而本地图片一般都相对比较大,拿iphone6来说,平时拍很多图片都是一两M的,如果直接这样上传,那图片就太大了,如果用户用的是移动流量,完全把图片上传显然不是一个好办法. 目前来说,HTML5的各种新API都在移动端的webkit上得到了较好的 实现.根据查看caniuse,本demo里使用到的FileRe

从web编辑器 UEditor 中单独提取图片上传,包含多图片单图片上传以及在线涂鸦功能

UEditor是由百度web前端研发部开发所见即所得富文本web编辑器,具有轻量,可定制,注重用户体验等特点,开源基于MIT协议,允许自由使用和修改代码.(抄的...) UEditor是非常好用的富文本web编辑器,而且全中文API和注释,方便学习和使用.特别是图片上传查看及涂鸦功能极为喜欢,但是有很多情况我们并不需要Web编辑器,而只需要图片上传.那么问题来了,提取图片上传哪家强..... 网上有很多图片上传的控件.插件.但都不是那么的完美,有的只有一张图片上传不包含批量上传,有的没有图片查看

JSP+SpringMVC框架使用WebUploader插件实现注册时候头像图片的异步上传功能

一.去官网下载webuploader文件上传插件 https://fex.baidu.com/webuploader/ 下载好后把它放到Javaweb项目的文件夹中(我放到了webcontent下面的static里面) 二.复制前端的样式 把这段代码放到你想要放到的位置(刷新页面和示例中不一样?不用担心因为你还没有初始化{就是还没有导入swf文件},指定路径后刷新应该就好了) 三.复制实现文件异步上传的js代码(这里我们只复制图片上传的部分) js源码的中文是乱码,应该是缺少谷歌改编码格式的插件

IE8升级新版Flash Player ActiveX14导致的discuz图片附件无法上传 解决方法

之前发的这篇文章被编辑之后丢失了,无奈从百度快照找来重新发布,不知道csdn抽啥风 架不住sb adobe的频繁升级提示,手欠升级到了了flash player 14,结果IE8下所有discuz论坛中都无法看到上传图片的按钮了 没办法,遇到问题就解决吧 刚好在解决IE11遇到编辑器不显示问题的时候看到discuz编辑器文件上传有非flash解决方案 所以这个问题看上去就不难了,把普通上传给打开就行了 编辑discuz文件/template/default/forum/editor_menu_f