在浏览器中操做文件,多数状况下用到的是 File
对象,从 <input type='file' />
元素获取,进而继续操做(例如将选择的图片展现在页面上,用ajax将文件上传至服务器等)。这里介绍在浏览器中操做文件的相关API.javascript
File
对象继承自 Blob
对象,先看看 Blob
对象。html
Blob
对象表示一个不可变、原始数据的类文件对象。Blob 表示的不必定是JavaScript原生格式的数据。java
MIME
类型。示例:web
var content1 = ['This is my firt trip to an island'];
var blob1 = new Blob(content, {type: 'text/plain'});
var content2 = {name: 'Alice', age: 23};
var blob2 = new Blob([JSON.stringify(content2, null, 2)], {type: 'application/json'});
复制代码
属性名称 | 读/写 | 描述 |
---|---|---|
size | 只读 | Blob 对象中所包含数据的大小(字节)。 |
type | 只读 | 一个字符串,代表该Blob对象所包含数据的MIME 类型。若是类型未知,则该值为空字符串。例如 "image/png". |
示例:ajax
var content = ['<div id="box"><p class="pra">a paragraph</p></div>'];
var blob = new Blob(content, {type: 'text/html'});
console.log(blob.size); // 50
console.log(blob.type); // text/html
复制代码
slice
方法接收三个可选参数,start
和 end
都是数值,表示截取的范围,contentType
指定截取的内容的 MIME
类型。返回一个新的 Blob
对象。json
var blob = new Blob(['This is an example of Blob slice method'], {type: 'text/plain'});
console.log(blob.size); // 39
var newBlob = blob.slice(10, 20, 'text/plain');
console.log(newBlob.size); // 10
复制代码
从 Blob
对象中读取内容可使用 FileReader
. 下文会介绍。canvas
咱们接触的多数关于 File
的操做都是读取,js也为咱们提供了手动建立 File
对象的构造函数:File(bits, name[, options])
。数组
bits (required) ArrayBuffer,ArrayBufferView,Blob,或者 Array[string] — 或者任何这些对象的组合。这是 UTF-8 编码的文件内容。。浏览器
name [String] (required) 文件名称,或者文件路径.安全
options [Object] (optional) 选项对象,包含文件的可选属性。可用的选项以下:
type
: string, 表示将要放到文件中的内容的MIME
类型。默认值为 '' 。lastModified
: 数值,表示文件最后修改时间的 Unix 时间戳(毫秒)。默认值为 Date.now()。示例:
var file1 = new File(['text1', 'text2'], 'test.txt', {type: 'text/plain'});
复制代码
根据已有的 blob
对象建立 File
对象:
var file2 = new File([blob], 'test.png', {type: 'image/png'});
复制代码
File
对象的实例内容不可见,可是有如下属性能够访问:
属性名称 | 读/写 | 描述 |
---|---|---|
name | 只读 | 返回文件的名称.因为安全缘由,返回的值并不包含文件路径 。 |
type | 只读 | 返回 File 对象所表示文件的媒体类型(MIME)。例如 PNG 图像是 "image/png". |
lastModified | 只读 | number, 返回所引用文件最后修改日期,自 1970年1月1日0:00 以来的毫秒数。 |
lastModifiedDate | 只读 | Date, 返回当前文件的最后修改日期,若是没法获取到文件的最后修改日期,则使用当前日期来替代。 |
示例:
<input type="file" id='file'>
复制代码
document.getElementById('file').addEventListener('change', function(event){
const file = this.files[0];
if (file) {
console.log(file.name);
console.log(file.size);
console.log(file.lastModified);
console.log(file.lastModifiedDate);
}
});
复制代码
备注: 基于当前的实现,浏览器不会实际读取文件的字节流,来判断它的媒体类型。它基于文件扩展来假设;将PNG 图像文件的后缀名重命名为 .txt
,那么读取的该文件的 type
属性值为 "text/plain", 而不是 "image/png" 。并且,file.type
仅仅对常见文件类型可靠。例如图像、文档、音频和视频。不常见的文件扩展名会返回空字符串。开发者最好不要依靠这个属性,做为惟一的验证方案。
File
对象没有定义额外的方法,因为继承了 Blob
对象,也就继承了 slice
方法,用法同上文 Blob
的 slice
方法。
FileReader, URL.createObjectURL(), createImageBitmap(), 及 XMLHttpRequest.send() 都能处理 Blob 和 File。
FileReader
对象容许Web应用程序异步读取存储在用户计算机上的文件(或原始数据缓冲区)的内容,使用File
或Blob
对象指定要读取的文件或数据。
其中 File
对象能够是来自用户在一个 <input>
元素上选择文件后返回的 FileList
, 也能够来自拖放操做生成的 DataTransfer
对象,还能够是来自在一个 HTMLCanvasElement
上执行 mozGetAsFile()
方法后返回结果。
var reader = new FileReader()
构造函数不须要传入参数,返回一个 FileReader
的实例。FileReader
继承 EventTarget
对象。
属性名称 | 读/写 | 描述 |
---|---|---|
error | 只读 | DOMException 的实例,表示在读取文件时发生的错误 。 |
result | 只读 | 文件的内容,该属性仅在读取操做完成后(load)后才有效,格式取决于读取方法 |
readyState | 只读 | 表示读取文件时状态的数字 |
备注: readeyState
的取值以下:
值 | 常量名 | 描述 |
---|---|---|
0 | EMPTY | 尚未加载任何数据 |
1 | LOADING | 数据正在被加载 |
2 | DONE | 已完成所有的读取请求. |
使用示例:
var reader = new FileReader();
console.log(reader.error); // null
console.log(reader.result); // null
console.log(reader.readyState); // 0
console.log(reader.EMPTY); // 0
console.log(reader.LOADING); // 1
console.log(reader.DONE); // 2
复制代码
EMPTY
、LOADING
、DONE
这三个属性同时存在于 FileReader
和它的的原型对象上,所以实例上有这三个属性,FileReader
对象自己也有这三个属性:
console.log(FileReader.EMPTY); // 0
console.log(FileReader.LOADING); // 1
console.log(FileReader.DONE); // 2
复制代码
文件的读取是一个异步的过程,和 XMLHttpRequest
对象同样,在读取操做过程当中会触发一系列事件。
事件名称 | 描述 | 使用示例 |
---|---|---|
abort | 读取操做被中断时触发。 | reader.onabort = function(event) {} |
error | 在读取操做发生错误时触发。 | reader.onerror = function(event) {} |
load | 读取操做完成时触发。 | reader.addEventListener('load', function(event) {}) |
loadstart | 读取操做开始时触发。 | reader.onloadstart = function(event) {} |
loadend | 读取操做结束时(要么成功,要么失败)触发。 | reader.onloadend = function(event) {} |
progress | 在读取Blob时触发。 | reader.onprogress = function(event) {} |
FileReader
的实例具备如下可操做的方法:
方法名称 | 描述 | 使用示例 |
---|---|---|
abort() | 手动终止读取操做,只有当 readyState 为 1 时才能调用,调用后,readyState 值为 2 |
reader.abort() |
readAsArrayBuffer(blob) | 读取指定的 Blob 或 File 对象。读取操做完成后(触发loadend 事件),result 属性将包含一个 ArrayBuffer 对象表示所读取的文件的数据。 |
reader.readAsArrayBuffer(blob) |
readAsDataURL(blob) | 读取指定的 Blob 或 File 对象。读取操做完成后(触发loadend 事件),result 属性将包含一个 data:URL 格式的字符串(base64编码) |
reader.readAsArrayBuffer(file) |
readAsBinaryString(blob) | 已废弃,用 readAsArrayBuffer 代替 |
-- |
readAsText(blob[, encoding]) | 将 Blob 或者 File 对象转根据特殊的编码格式转化为内容(字符串形式), 默认编码是 utf-8 |
reader.readAsArrayBuffer(blob) |
读取本地图片示例:
<input type="file" id='file' accept="image/png, image/jpg, image/jpeg, image/gif" />><br />>
<img src="" alt="Image preview...">
复制代码
var preview = document.querySelector('img');
var reader = new FileReader();
reader.addEventListener("load", function () {
preview.src = reader.result;
}, false);
document.getElementById('file').addEventListener('change', function (event) {
var file = this.files[0];
if (file) {
reader.readAsDataURL(file);
}
});
复制代码
dataURL是base64
编码的数据格式,展现类型为字符串,形如: ...
将 dataURL
转为 blob
对象:
function dataURLToBlob (dataurl) {
let arr = dataurl.split(',');
let mime = arr[0].match(/:(.*?);/)[1];
let bstr = atob(arr[1]);
let n = bstr.length;
let u8arr = new Uint8Array(n);
while (n--) {
u8arr[n] = bstr.charCodeAt(n);
}
return new Blob([u8arr], { type: mime });
}
复制代码
结合上例,根据已有的 <img>
对象建立一个 File
对象:
reader.addEventListener("load", function () {
preview.src = reader.result;
var blob = dataURLToBlob(reader.result);
var newFile = new File([blob], 'test.jpeg', {type: blob.type});
console.log(newFile.name); // test.jpeg
console.log(newFile.type);
console.log(newFile.size);
}, false);
复制代码
将图片文件转换成 data:URL
格式供 <img>
元素展现,除了使用 fileReader.readAsDataURL
外,还可使用 URL.createObjectURL
方法。 URL.createObjectURL(blob)
方法返回一个 blob:
开头的字符串,指向文件在内存中的地址。
<input type="file" id='file' accept="image/png, image/jpg, image/jpeg, image/gif" /><br />
<img src="" alt="Image preview...">
复制代码
var preview = document.querySelector('img');
document.getElementById('file').addEventListener('change', function (event) {
var file = this.files[0];
if (file) {
preview.src = URL.createObjectURL(file);
}
});
复制代码
综合以上,能够实现一个简单的粘贴图片并显示的功能。HTML5提供的内容编辑功能,能够粘贴部分图片,例如从网页上复制的图片。可是使用截图工具截取的图片没法粘贴显示,并且从网页上复制的图片会带有原来的样式,其图片url也是原来图片的指向。咱们使用如下代码能够统一这两种粘贴操做,实现统一的效果。
<div id="comment" contenteditable></div>
复制代码
#comment{
border: 1px solid #ccc;
min-height: 500px;
padding: 10px;
}
#comment:focus {
border-color: #ccc;
outline: none;
}
.img-paste {
max-width: 100%;
}
复制代码
var comment = document.getElementById('comment');
comment.addEventListener('paste', function(event) {
console.log(event);
var item = event.clipboardData.files[0];
if (item && /image/.test(item.type)) {
var img = new Image();
img.src = URL.createObjectURL(item);
img.className = 'img-paste';
this.appendChild(img);
event.preventDefault();
}
}, false);
复制代码
实例效果能够查看这里。