图片格式转化之 webp 转为 png

WebP 是现代图像格式中的"通用选手",在多数情况下比 PNG、JPEG、GIF 都更节省空间,同时功能更全面。但是某些网站只支持上传 png 或者 jpg,所以要将 webp 转为 png。

1.base64ToBlob 中要转为 8位字节

base64ToBlob 中要转为 8位字节是为了还原出原始的二进制文件数据(字节流),因为只有 JS 字符串的每个字符是 16 位(UTF-16),而一组真实的 8 位字节流 才能被浏览器或文件系统识别成 图片、音频、视频、压缩包等二进制文件。 Base64 解码后的字符都是 0~255 的字节(低8位有效,高8位都是0),可以用 charCodeAt 去获取 真实8位字节值

2.Base64 => blob => 下载

维度 直接 Base64 Data URL Base64 → Blob → 下载
内存使用 高,Base64 是冗长的字符串,占用较大内存。base64 编码的体积比原始二进制大 约 33% 【原因:每 3 个字节(24 bit)编码成 4 个字符(每个字符 6 bit)】 低,Blob 直接存储二进制,内存开销更小
文件大小限制 受限,浏览器对 URL 长度有限制 Blob URL 本质上不是存放文件内容的"字符串",而是浏览器内部为那个二进制数据对象创建的一个"指针"或"句柄"。它只是一个很短的标识符,指向浏览器内存(或者磁盘)中的真实文件数据,所以它不受 URL 长度限制
兼容性 较差,部分老旧浏览器和环境支持不佳 较好,主流浏览器和环境均支持
控制粒度 低,只能用于简单下载 高,支持内存释放、分片、上传、文件操作等功能
安全性 低,Base64 数据直接暴露在 URL 中,有一定XSS风险 高,Blob URL 是临时生成,外部无法直接访问内容
js 复制代码
<div class="input-area">  
    <input type="file" id="webp-input" accept=".webp" onchange="javascript:preview()" />
    <div id="img-preview">
        
    </div>
    <button onclick="convertWebp('image/png')">转换成png</button>
    <button onclick="convertWebp('image/jpg')">转换成jpg</button>
</div>  
  
<div class="output-area" id="output-area">  
    <!-- 下载链接将在这里动态生成 -->  
</div>  
  
<script>  
function preview(){
    const imgContent = document.getElementById('img-preview');
    const file = document.getElementById("webp-input").files[0];
    imgContent.innerHTML = '';
    const img = document.createElement('img');
    img.src = URL.createObjectURL(file);	// 将上传的文件对象转为临时URL(blob URL)
    imgContent.appendChild(img);
    img.onload = function() {
        URL.revokeObjectURL(this.src);	// 图片加载完成后,释放上面创建的临时URL占用的内存资源
    }
}
function base64ToBlob(base64, mimeType) {
    const byteString = atob(base64.split(',')[1]);	// 拿到base6内容部分,将其解码为字符串
    const ab = new ArrayBuffer(byteString.length);	// 创建ArrayBuffer
    const ia = new Uint8Array(ab);	// 创建字节数组视图,可以像数组一样操作每个字节
    for (let i = 0; i < byteString.length; i++) {
        ia[i] = byteString.charCodeAt(i);	// 转换每个字符为字节
    }
    return new Blob([ab], { type: mimeType });
}
function downloadBlob(blob, filename) {
    const url = URL.createObjectURL(blob);		// 生成 blob URL
    const link = document.createElement('a');
    link.href = url;
    link.download = filename;			
    document.body.appendChild(link);
    link.click();
    document.body.removeChild(link);
    URL.revokeObjectURL(url);
}
function convertWebp(mimeType) {  
    const inputElement = document.getElementById('webp-input');  
    const file = inputElement.files[0];  
    if (!file) {  
        alert('Please select a WebP file first.');  
        return;  
    }  
    const reader = new FileReader();  
    reader.onload = function(event) {  
        const img = new Image();  
        img.onload = function() {  
            const canvas = document.createElement('canvas');  
            canvas.width = img.width;  
            canvas.height = img.height;  
            const ctx = canvas.getContext('2d');  // 获取canvas的2D渲染上下文对象ctx,用于绘图
            ctx.drawImage(img, 0, 0, img.width, img.height);  	// 将图片绘制到canvas上(从左上角开始,宽高与图片一致)
            const base64 = canvas.toDataURL(mimeType); // 将canvas按指定格式导出成base64数据URL,如:...
            const blob = base64ToBlob(base64, mimeType);
            const filename = mimeType === 'image/png' ? 'converted.png' : 'converted.jpg';
            downloadBlob(blob, filename);
        };  
        img.src = event.target.result;  
    };  
    reader.readAsDataURL(file);  
}  
</script>  
相关推荐
面向星辰35 分钟前
html各种常用标签
前端·javascript·html
梦65037 分钟前
HTML新属性
前端
东风西巷3 小时前
PDFgear:免费全能的PDF处理工具
前端·pdf·软件需求
森之鸟3 小时前
Mac电脑上如何打印出字体图标
前端·javascript·macos
mCell4 小时前
GSAP 入门指南
前端·javascript·动效
gnip4 小时前
组件循环引用依赖问题处理
前端·javascript
Aotman_5 小时前
el-input textarea 禁止输入中文字符,@input特殊字符实时替换,光标位置保持不变
前端·javascript·vue.js·前端框架·es6
Nan_Shu_6145 小时前
Web前端面试题(1)
前端·面试·职场和发展
EveryPossible5 小时前
选择数据展示
javascript
lypzcgf5 小时前
Coze源码分析-资源库-创建知识库-前端源码-核心组件
前端·typescript·react·coze·coze源码分析·ai应用平台·agent开发平台