图片格式转化之 webp 转为 png

0 阅读3分钟

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 URLBase64 → Blob → 下载
内存使用高,Base64 是冗长的字符串,占用较大内存。base64 编码的体积比原始二进制大 约 33% 【原因:每 3 个字节(24 bit)编码成 4 个字符(每个字符 6 bit)】低,Blob 直接存储二进制,内存开销更小
文件大小限制受限,浏览器对 URL 长度有限制Blob URL 本质上不是存放文件内容的“字符串”,而是浏览器内部为那个二进制数据对象创建的一个“指针”或“句柄”。它只是一个很短的标识符,指向浏览器内存(或者磁盘)中的真实文件数据,所以它不受 URL 长度限制
兼容性较差,部分老旧浏览器和环境支持不佳较好,主流浏览器和环境均支持
控制粒度低,只能用于简单下载高,支持内存释放、分片、上传、文件操作等功能
安全性低,Base64 数据直接暴露在 URL 中,有一定XSS风险高,Blob URL 是临时生成,外部无法直接访问内容
<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>