将基于base64的png数据转换为javascript文件对象

51

我有两个以PNG格式编码的base64,需要使用Resemble.JS进行比较。

我认为最佳的方法是使用fileReaderPNG转换为文件对象。如何实现呢?


你想用哪些属性来进行比较? - Parthik Gosar
2
查看它们有多相似 - Bonik
你可以在Blob上使用FileReader,并且可以从二进制字符串或数组缓冲区中提取Blob。这里有一些解决方案。 - dandavis
3个回答

94

方式1:仅适用于dataURL,不适用于其他类型的URL。

function dataURLtoFile(dataurl, filename) {
    var arr = dataurl.split(','), mime = arr[0].match(/:(.*?);/)[1],
        bstr = atob(arr[1]), n = bstr.length, u8arr = new Uint8Array(n);
    while(n--){
        u8arr[n] = bstr.charCodeAt(n);
    }
    return new File([u8arr], filename, {type:mime});
}

//Usage example:
var file = dataURLtoFile('data:image/png;base64,......', 'a.png');
console.log(file);

第二种方法:适用于任何类型的url(http url,dataURL,blobURL等...)

//return a promise that resolves with a File instance
function urltoFile(url, filename, mimeType){
    mimeType = mimeType || (url.match(/^data:([^;]+);/)||'')[1];
    return (fetch(url)
        .then(function(res){return res.arrayBuffer();})
        .then(function(buf){return new File([buf], filename, {type:mimeType});})
    );
}

//Usage example:
urltoFile('data:image/png;base64,......', 'a.png')
.then(function(file){
    console.log(file);
})

两者在Chrome和Firefox中均可使用。


1
非常好,这应该是被接受的答案,它更简单。 - Pablo Quemé
你可以通过以下方式猜测 MIME 类型和扩展名:let matches = url.match(/^data:([^;]+);base64,(.*)$/); let mimeType = matches[1]; let extension = matches[1].substr(matches[1].indexOf('/')+1); - C Würtz
"Way 2" 在 Chrome 和 Firefox 上运行良好,但在 Safari(Cordova UIWebview)上停止工作。这是适用于所有三个浏览器的解决方案。https://dev59.com/oW445IYBdhLWcg3wLXMK#5100158 - tanner burton
你的回答非常有用,我想问一下我们是否可以进行压缩?如果可以,请告诉我如何操作。 - Chirag Joshi

43
你可以使用你的base64数据创建一个Blob,然后以asDataURL的形式读取它。
var img_b64 = canvas.toDataURL('image/png');
var png = img_b64.split(',')[1];

var the_file = new Blob([window.atob(png)],  {type: 'image/png', encoding: 'utf-8'});

var fr = new FileReader();
fr.onload = function ( oFREvent ) {
    var v = oFREvent.target.result.split(',')[1]; // encoding is messed up here, so we fix it
    v = atob(v);
    var good_b64 = btoa(decodeURIComponent(escape(v)));
    document.getElementById("uploadPreview").src = "data:image/png;base64," + good_b64;
};
fr.readAsDataURL(the_file);

完整的示例(包括垃圾代码和控制台日志):http://jsfiddle.net/tTYb8/


或者,您可以使用.readAsText,它很好用,而且更加优雅...但是由于某些原因,text听起来不太对。

fr.onload = function ( oFREvent ) {
    document.getElementById("uploadPreview").src = "data:image/png;base64,"
    + btoa(oFREvent.target.result);
};
fr.readAsText(the_file, "utf-8"); // its important to specify encoding here

完整示例:http://jsfiddle.net/tTYb8/3/


2
你可以在TypeScript中使用axios和async/await。
const dataUrlToFile = async (dataUrl: string, fileName: string, mimeType: string): Promise<File> => {
  const res = await axios(dataUrl);
  const blob: Blob = res.data;
  return new File([blob], fileName, { type: mimeType });
}

使用示例

const TEST_IMG_BASE64 = 'data:image/gif;base64,R0lGODlhAQABAIAAAAAAAP///yH5BAEAAAAALAAAAAABAAEAAAIBRAA7'
const TEST_IMG: File = await dataUrlToFile(TEST_IMG_BASE64, 'test.gif', 'image/gif')

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接