图片处理

图片流

前端所说的图片流就是读取本地图片,并在页面使用文件流的方式显示出来。

首先,我们简单说下文件上传的几种方式

上传方式

input

通过用户点击,创建<input type="file" accept="image/*"/>,并监听change事件获取file对象,大体如下

click = () => {
    let input = document.createElement('input')
    input.setAttribute('type', 'file')
    input.setAttribute('accept', 'image/*')
    
    input.onchange = event => {
        let file = event.target.files[0]
    }
    
    input.click()
    
} 

Drag && drop

使用HTML5的拖放API,监听元素的drop事件,同样是获取file对象

会创建一个DataTransfer对象,下面我们还会遇到它,稍后再说

dragover = event => {
   event.preventDefault()
}

drop = event => {
    event.preventDefault()
    let files =  event.dataTransfer.files
}

paste

给元素绑定粘贴事件,得益于contenteditable我们可以给所有元素添加,涛声依旧,获取event中包含的file

paste = (e) => {
    e.preventDefault()
    let file = e.clipboardData.files[0]
}

clipboardData

paste事件提供了一个clipboardData属性,是一个DataTransfer类型的对象,前面我们说到,拖放会产生一个DataTransfer对象,没错,粘贴也是它。

来来来,掀起了她的盖头来。

01.jpg

上面可以看到,clipboardData有如下属性

  • dropEffect 默认是node
  • effectAllowed 默认是uninitialized
  • files 本地文件列表
  • items 剪切板中的各项数据
  • types 剪切板中的各项数据类型

我们只需要使用files即可,图片文件在它里面

文件格式

file

通常情况下, File 对象是来自用户在一个<input>元素上选择文件后返回的 FileList 对象,也可以是来自由拖放操作生成的DataTransfer对象,继承于Blob

庐山真面目,诺,就是这个样子。

02.jpg

可以看到有如下属性:

  • name:文件名,该属性只读。
  • size:文件大小,单位为字节,该属性只读。
  • type:文件的 MIME 类型,如果分辨不出类型,则为空字符串,该属性只读。
  • lastModified:文件的上次修改时间,格式为时间戳。
  • lastModifiedDate:文件的上次修改时间,格式为 Date 对象实例。

我们不去深究file对象,只需要知道通过它可以访问本地的文件。

blob

一个 Blob对象表示一个不可变的, 原始数据的类似文件对象。Blob表示的数据不一定是一个JavaScript原生格式。 File 接口基于Blob,继承 blob功能并将其扩展为支持用户系统上的文件。

创建blob对象

var aBlob = new Blob( array, options );
  • array 是一个由ArrayBuffer, ArrayBufferView, Blob, DOMString 等对象构成的 Array ,或者其他类似对象的混合体,它将会被放进 Blob.

  • options 是一个可选的Blob熟悉字典,它可能会指定如下两种属性

    • type,默认值为 “”,它代表了将会被放入到blob中的数组内容的MIME类型。

    • endings,默认值为”transparent”,它代表包含行结束符\n的字符串如何被输出。

var a = ["hello", "world"];
var myBlob = new Blob(a, { "type" : "text/xml" });
console.log(myBlob);
03.jpg

通过动态创建blob,我们可以实现纯前端下载

const foo = {hello: "world"};
const blob = new Blob([JSON.stringify(foo)], {type: "text/plain"});
const fileName = `${new Date().valueOf()}.doc`;
const link = document.createElement('a');
link.href = window.URL.createObjectURL(blob);
link.download = fileName;
link.click();
window.URL.revokeObjectURL(link.href);

Blob URL

Blob URL是blob协议的URL,格式如下

blob:http://localhost:1234/946644c4-ca98-405e-918c-759e790d0330

Blob URL可以通过URL.createObjectURL(blob)创建, 在每次调用createObjectURL()方法时,都会创建一个新的 URL 对象,即使你已经用相同的对象作为参数创建过。

在不需要这些URL对象的时候, 通过URL.revokeObjectURL(objectURL) 释放URL对象

使用Blob URL进行显示本地图片,我们只需要把创建的URL赋值给img的src属性就可以了。

05.jpg

FileReader

FileReader用来读取file或blob文件数据,基于文件大小不同,读取的过程为异步。

let render = new FileReader()
render.onload = () => {
    let src = render.result
}
render.readAsDataURL(file)

FileReader读取文件方法

  • readAsBinaryString file 将文件读取为二进制编码

  • readAsBinaryArray file 将文件读取为二进制数组

  • readAsText file[, encoding] 按照格式将文件读取为文本,encode默认为UTF-8

  • readAsDataURL file 将文件读取为DataUrl

base64

使用FileReader进行文件的读取,就可以将图片读取成base64格式的了。

04.jpg

直接在FileReader实例的onload函数里面将result赋值给src即可

格式差异

其实主要是两种格式base64和blob,它们之间的差异如下

  • Blob URL的长度一般比较短

  • Blob URL可以方便的使用XMLHttpRequest获取源数据, base64不是所有浏览器都支持

  • Blob URL 只能在当前应用内部使用

格式之间转换

canvas转为blob对象

canvas.toBlob(function (blobObj) {
    console.log(blobObj)
})

canvas转为base64

let imgSrc = canvas.toDataURL('image/png')

base64转为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 });
}

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,294评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,493评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,790评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,595评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,718评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,906评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,053评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,797评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,250评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,570评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,711评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,388评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,018评论 3 316
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,796评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,023评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,461评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,595评论 2 350

推荐阅读更多精彩内容