JS 图片简易压缩【实践】(js实现图片压缩上传)
myzbx 2025-06-10 16:53 6 浏览
作者:政采云前端团队
转发链接:
https://juejin.im/post/5ea574cc518825736e57fcca
前言
说起图片压缩,大家想到的或者平时用到的很多工具都可以实现,例如,客户端类的有图片压缩工具 PPDuck3, JS 实现类的有插件 compression.js ,亦或是在线处理类的 OSS 上传,文件上传后,在访问文件时中也有图片的压缩配置选项,不过,能不能自己撸一套 JS 实现的图片压缩代码呢?当然可以,那我们先来理一下思路。
压缩思路
涉及到 JS 的图片压缩,我的想法是需要用到 Canvas 的绘图能力,通过调整图片的分辨率或者绘图质量来达到图片压缩的效果,实现思路如下:
- 获取上传 Input 中的图片对象 File
- 将图片转换成 base64 格式
- base64 编码的图片通过 Canvas 转换压缩,这里会用到的 Canvas 的 drawImage 以及 toDataURL 这两个 Api,一个调节图片的分辨率的,一个是调节图片压缩质量并且输出的,后续会有详细介绍
- 转换后的图片生成对应的新图片,然后输出
优缺点介绍
不过 Canvas 压缩的方式也有着自己的优缺点:
- 优点:实现简单,参数可以配置化,自定义图片的尺寸,指定区域裁剪等等。
- 缺点:只有 jpeg 、webp 支持原图尺寸下图片质量的调整来达到压缩图片的效果,其他图片格式,仅能通过调节尺寸来实现
代码实现
<template>
<div class="container">
<input type="file" id="input-img" @change="compress" />
<a :download="fileName" :href="compressImg" >普通下载</a>
<button @click="downloadImg">兼容 IE 下载</button>
<div>
<img :src="compressImg" />
</div>
</div>
</template>
<script>
export default {
name: 'compress',
data: function() {
return {
compressImg: null,
fileName: null,
};
},
components: {},
methods: {
compress() {
// 获取文件对象
const fileObj = document.querySelector('#input-img').files[0];
// 获取文件名称,后续下载重命名
this.fileName = `${new Date().getTime()}-${fileObj.name}`;
// 获取文件后缀名
const fileNames = fileObj.name.split('.');
const type = fileNames[fileNames.length-1];
// 压缩图片
this.handleCompressImage(fileObj, type);
},
handleCompressImage(img, type) {
const vm = this;
let reader = new FileReader();
// 读取文件
reader.readAsDataURL(img);
reader.onload = function(e) {
let image = new Image(); //新建一个img标签
image.src = e.target.result;
image.onload = function() {
let canvas = document.createElement('canvas');
let context = canvas.getContext('2d');
// 定义 canvas 大小,也就是压缩后下载的图片大小
let imageWidth = image.width; //压缩后图片的大小
let imageHeight = image.height;
canvas.width = imageWidth;
canvas.height = imageHeight;
// 图片不压缩,全部加载展示
context.drawImage(image, 0, 0);
// 图片按压缩尺寸载入
// let imageWidth = 500; //压缩后图片的大小
// let imageHeight = 200;
// context.drawImage(image, 0, 0, 500, 200);
// 图片去截取指定位置载入
// context.drawImage(image,100, 100, 100, 100, 0, 0, imageWidth, imageHeight);
vm.compressImg = canvas.toDataURL(`image/${type}`);
};
};
},
// base64 图片转 blob 后下载
downloadImg() {
let parts = this.compressImg.split(';base64,');
let contentType = parts[0].split(':')[1];
let raw = window.atob(parts[1]);
let rawLength = raw.length;
let uInt8Array = new Uint8Array(rawLength);
for(let i = 0; i < rawLength; ++i) {
uInt8Array[i] = raw.charCodeAt(i);
}
const blob = new Blob([uInt8Array], {type: contentType});
this.compressImg = URL.createObjectURL(blob);
if (window.navigator.msSaveOrOpenBlob) {
// 兼容 ie 的下载方式
window.navigator.msSaveOrOpenBlob(blob, this.fileName);
}else{
const a = document.createElement('a');
a.href = this.compressImg;
a.setAttribute('download', this.fileName);
a.click();
}
},
}
};
</script>
复制代码
上面的代码是可以直接拿来看效果的,不喜欢用 Vue 的也可以把代码稍微调整一下,下面开始具体分解一下代码的实现思路。
Input 上传 File 处理
将 File 对象通过 FileReader 的 readAsDataURL 方法转换为URL格式的字符串(base64编码)。
const fileObj = document.querySelector('#input-img').files[0];
let reader = new FileReader();
// 读取文件
reader.readAsDataURL(fileObj);
复制代码
Canvas 处理 File 对象
建立一个 Image 对象,一个 canvas 画布,设定自己想要下载的图片尺寸,调用 drawImage 方法在 canvas 中绘制上传的图片。
let image = new Image(); //新建一个img标签
image.src = e.target.result;
let canvas = document.createElement('canvas');
let context = canvas.getContext('2d');
context.drawImage(image, 0, 0);
复制代码
Api 解析:drawImage
context.drawImage(img, sx, sy, sWidth, sHeight, dx, dy, dWidth, dHeight);
复制代码
img
就是图片对象,可以是页面上获取的 DOM 对象,也可以是虚拟 DOM 中的图片对象。
dx , dy , dWidth , dHeight
表示在 canvas 画布上规划出一片区域用来放置图片,dx, dy 为绘图位置在 Canvas 元素的 X 轴、Y 轴坐标,dWidth, dHeight 指在 Canvas 元素上绘制图像的宽度和高度(如果不说明, 在绘制时图片的宽度和高度不会缩放)。
sx , sy , swidth , sheight
这 4 个参数是用来裁剪源图片的,表示图片在 canvas 画布上显示的大小和位置。sx,sy 表示在原图片上裁剪位置的 X 轴、Y 轴坐标,然后以 swidth,sheight 尺寸来选择一个区域范围,裁剪出来的图片作为最终在 Canvas 上显示的图片内容( swidth,sheight 不说明的情况下,整个矩形(裁剪)从坐标的 sx 和 sy 开始,到图片的右下角结束)。
以下为图片绘制的实例:
context.drawImage(image, 0, 0, 100, 100);
context.drawImage(image, 300, 300, 200, 200);
context.drawImage(image, 0, 100, 150, 150, 300, 0, 150, 150);
复制代码
Api 中奇怪之处在于,sx,sy,swidth,sheight 为选填参数,但位置在 dx, dy, dWidth, dHeight 之前。
Canvas 输出图片
调用 canvas 的 toDataURL 方法可以输出 base64 格式的图片。
canvas.toDataURL(`image/${type}`);
复制代码
Api 解析:toDataURL
canvas.toDataURL(type, encoderOptions);
复制代码
type 可选
图片格式,默认为 image/png。
encoderOptions 可选
在指定图片格式为 image/jpeg 或 image/webp的情况下,可以从 0 到 1 的区间内选择图片的质量。如果超出取值范围,将会使用默认值 0.92。其他参数会被忽略。
a 标签的下载
调用 <a> 标签的 download 属性,即可完成图片的下载。
Api 解析:download
// href 下载必填
<a download="filename" href="href"> 下载 </a>
复制代码
filename
选填,规定作为文件名来使用的文本。
href
文件的下载地址。
非主流浏览器下载处理
到此可以解决 Chroma 、 Firefox 和 Safari(自测支持) 浏览器的下载功能,因为 IE 等浏览器不支持 download 属性,所以需要进行其他方式的下载,也就有了代码中的后续内容。
// base64 图片转 blob 后下载
downloadImg() {
let parts = this.compressImg.split(';base64,');
let contentType = parts[0].split(':')[1];
let raw = window.atob(parts[1]);
let rawLength = raw.length;
let uInt8Array = new Uint8Array(rawLength);
for(let i = 0; i < rawLength; ++i) {
uInt8Array[i] = raw.charCodeAt(i);
}
const blob = new Blob([uInt8Array], {type: contentType});
this.compressImg = URL.createObjectURL(blob);
if (window.navigator.msSaveOrOpenBlob) {
// 兼容 ie 的下载方式
window.navigator.msSaveOrOpenBlob(blob, this.fileName);
}else{
const a = document.createElement('a');
a.href = this.compressImg;
a.setAttribute('download', this.fileName);
a.click();
}
}
复制代码
- 将之前 canvas 生成的 base64 数据拆分后,通过 atob 方法解码
- 将解码后的数据转换成 Uint8Array 格式的无符号整形数组
- 转换后的数组来生成一个 Blob 数据对象,通过 URL.createObjectURL(blob) 来生成一个临时的 DOM 对象
- 之后 IE 类浏览器可以调用 window.navigator.msSaveOrOpenBlob 方法来执行下载,其他浏览器也可以继续通过 <a> 标签的 download 属性来进行下载
Api 解析:atob
base-64 解码使用方法是 atob()。
window.atob(encodedStr)
复制代码
encodedStr
必需,是一个通过 btoa() 方法编码的字符串,btoa()是 base64 编码的使用方法。
Api 解析:Uint8Array
new Uint8Array(length)
复制代码
length
创建初始化为 0 的,包含 length 个元素的无符号整型数组。
Api 解析: Blob
Blob 对象表示一个不可变、原始数据的类文件对象。
// 构造函数允许通过其它对象创建 Blob 对象
new Blob([obj],{type:createType})
复制代码
obj
字符串内容
createType
要构造的类型
兼容性 IE 10 以上
Api 解析:createObjectURL
静态方法会创建一个 DOMString。
objectURL = URL.createObjectURL(object);
复制代码
object
用于创建 URL 的 File 对象、Blob 对象或者 MediaSource 对象。
Api 解析: window.navigator
// 官方已不建议使用的文件下载方式,仅针对 ie 且兼容性 10 以上
// msSaveBlob 仅提供下载
// msSaveOrOpenBlob 支持下载和打开
window.navigator.msSaveOrOpenBlob(blob, fileName);
复制代码
blob
要下载的 blob 对象
fileName
下载后命名的文件名称。
总结
本文仅针对图片压缩介绍了一些思路,简单的使用场景可能如下介绍,当然也会引申出来更多的使用场景,这些还有待大家一起挖掘。
- 上传存储图片如果需要对文件大小格式有要求的,可以统一压缩处理图片
- 前台页面想要编辑图片,可以在 Canvas 处理图片的时候,加一些其他逻辑,例如添加文字,剪裁,拼图等等操作
当然温馨提示:因部分接口有 IE 兼容性问题,IE 浏览器方面,仅能支持 IE10 以上版本进行下载。
推荐图片优化以及移动端兼容相关文章
《手把手教你Vue解析pdf(base64)转图片【实践】》
《全平台(Vue/React/微信小程序)任意角度旋图片裁剪组件》
《手把手教你如何编写一个前端图片压缩、方向纠正、预览、上传插件》
《深入前端tree优化渲染速度从14.65s到0.49s【实战】》
《首屏时间从12.67s到1.06s,手把手教你如何做到的?》
作者:政采云前端团队
转发链接:
https://juejin.im/post/5ea574cc518825736e57fcca
相关推荐
- 搞笑句子j(搞笑句子精辟幽默专治不开心)
-
#我的搞笑生活#你的废话怎么比湖南卫视的广告还多啊2.我要瘦成一道闪电照亮所有猥琐的死胖子3.宁可胖的精致,也不要瘦的雷同4.秀发去无踪,头屑更出众5.心情郁闷时拿房东的牙刷刷马桶6.黄瓜在于拍,...
- 解决 JS 对象中继承性问题之方式一:通过原型链继承来解决继承问题
-
Ⅰ、继承问题:1、什么是继承?答:子类去继承父类的东西,称之为继承;如:子类继承父类的属性或方法等;2、常见的继承方式有哪些?答:继承方式有五种:其一、原型链继承;其二、构造函数继承(也称call...
- 西门子S7-300 SCL编程笔记(附实例)
-
1.数据类型:注:还有两种类型:P:POINTER(指针数据类型)。A:ANY(任何类型)数组的定义:M1:ARRAY[n0..m0,n1..m1]OFINT;数组的类型2.寻址绝对寻址符号寻...
- 10 个常问的 JS 面试题(js面试题目及答案)
-
作者:JoanneLee-(Vivi)译者:前端小智来源:medium1.如何理解JS中的`this`关键字?JS初学者总是对this关键字感到困惑,因为与其他现代编程语言相比,JS中...
- js 箭头函数(js 箭头函数返回值)
-
js箭头函数目录一、语法基础语法高级语法实例二、注意点正文回到顶部一、语法基础语法(参数1,参数2,…,参数N)=>{函数声明}(参数1,参数2,…,参数N)=>...
- JavaScript中关于null的一切(js中的null数据类型)
-
本文已经作者@DmitriPavluti授权翻译JavaScript有2种类型:基本类型(string,booleansnumber,symbol)和对象。对象是复杂的数据结构,JS中最简单的...
- 「JS 逆向百例」复杂的登录过程,最新WB逆向
-
声明本文章中所有内容仅供学习交流,抓包内容、敏感网址、数据接口均已做脱敏处理,严禁用于商业用途和非法用途,否则由此产生的一切后果均与作者无关,若有侵权,请联系我立即删除!逆向目标本次的逆向目标是WB的...
- 三年级语文下册自测题来啦!赶紧收藏!(附答案)
-
2020—2021学年度下学期学业水平测试小学三年级语文试题时间60分钟;总分105分,其中试题100分,卷面分5分书写规范,卷面整洁,奖励5分;书写较认真,奖励3分;书写不认真,不得分。一、基础知识...
- 爬虫基础之自动化工具 DrissionPage 的使用
-
概述前三期文章中已经介绍到了Selenium与Playwright、Pyppeteer的使用方法,它们的功能都非常强大。而本期要讲的DrissionPage更为独特,强大,而且使用更为方...
- js基础面试题92-130道题目(js面试基础知识)
-
92.说说你对作用域链的理解参考答案:作用域链的作用是保证执行环境里有权访问的变量和函数是有序的,作用域链的变量只能向上访问,变量访问到window对象即被终止,作用域链向下访问变量是不被允许的。...
- 模拟 Vue 中 JS 动态表达式在模版中被动态解析的实现
-
最近在写自己的一个web框架ref-lit.js,仅仅打算自己练练手,在这个框架中,其模版语法借助了lit-html.js,而lit-html.js是通过ES2015规范中的模版字符串...
- 怼渣男j句子(有什么怼渣男的句子)
-
各位继续答题,我提前交卷了。天是蓝的,海是深的,你说的话没有一句是真的在我的世界里,你才是配角。不爱就滚,爱我的在排队。“我就是和她聊聊,没别的意思,你干嘛发火?”“我就是想给你一巴掌,没别的意思,你...
- jcmd(检查梅毒挂什么门诊)
-
一、工具概述核心功能JDK7+提供的多功能命令行工具,用于与运行中的JVM交互,执行诊断、监控及管理操作。整合了jps、jstack、jmap等传统命令的功能,并扩展了更多诊断能力。优势特性轻量级...
- 3C1A四口配置,150W总功率,拆解一款电源厂新款大功率桌面充电器
-
前言充电头网拿到了C-SMARTLINK旭联推出的一款150W桌面充电器,这款充电器内置第三代半导体氮化镓器件,具备3C1A输出接口,USB-C1和USB-C2具备100W快充输出,USB-C3具备3...
- 爆发输出巅峰竟不是机炮!你对哪种开火机制情有独钟
-
前言在1.25.1版本当中,“极限火力”街机模式将会偕机炮坦克一起回归,在带来轻松乐趣的同时,也为后续将要登台的J系机炮轻型坦克线路做了预热。——哪里还要什么战术,配合?冲上去按住左键倾泻所有炮弹,完...
- 一周热门
- 最近发表
- 标签列表
-
- HTML 简介 (30)
- HTML 响应式设计 (31)
- HTML URL 编码 (32)
- HTML Web 服务器 (31)
- HTML 表单属性 (32)
- HTML 音频 (31)
- HTML5 支持 (33)
- HTML API (36)
- HTML 总结 (32)
- HTML 全局属性 (32)
- HTML 事件 (31)
- HTML 画布 (32)
- HTTP 方法 (30)
- 键盘快捷键 (30)
- CSS 语法 (35)
- CSS 选择器 (30)
- CSS 轮廓宽度 (31)
- CSS 谷歌字体 (33)
- CSS 链接 (31)
- CSS 定位 (31)
- CSS 图片库 (32)
- CSS 图像精灵 (31)
- SVG 文本 (32)
- 时钟启动 (33)
- HTML 游戏 (34)