Преобразование данных base64 png в объекты файла javascript

У меня есть два base64, закодированных в PNG, и мне нужно сравнить их с помощью Resemble.JS

Я думаю, что лучший способ сделать это - преобразовать PNG в файловые объекты с помощью fileReader. Как я могу это сделать?


person Bonik    schedule 06.06.2013    source источник
comment
по каким свойствам вы хотите их сравнить?   -  person Parthik Gosar    schedule 06.06.2013
comment
чтобы увидеть, насколько они похожи   -  person Bonik    schedule 06.06.2013
comment
вы можете использовать FileReader для больших двоичных объектов, и вы можете передать большой двоичный объект из буфера двоичной строки или массива. для этого есть решения.   -  person dandavis    schedule 06.06.2013


Ответы (4)


Вы можете создать Blob из ваших данных base64, а затем прочитать его 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, он работает нормально и более элегантно ... но по какой-то причине текст звучит неправильно;)

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/

person c69    schedule 06.06.2013

Способ 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);

Способ 2: работает для любого типа 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.

person cuixiping    schedule 13.08.2016
comment
Отлично работает, это должен быть принятый ответ, это намного проще - person Pablo Quemé; 29.05.2017
comment
Вы можете угадать тип MIME и расширение с помощью: `` let matches = url.match (/ ^ data: ([^;] +); base64, (. *) $ /); пусть mimeType = соответствует [1]; пусть extension = соответствует [1] .substr (соответствует [1] .indexOf ('/') + 1); `` '' - person C Würtz; 06.12.2017
comment
Way 2 отлично работает в Chrome и Firefox, но перестал работать для меня в сафари (Cordova UIWebview). Это то, что работает на всех трех. stackoverflow.com/a/5100158/9909941 - person tanner burton; 03.12.2019
comment
Ваш ответ действительно полезен, я просто хочу спросить, можем ли мы выполнить сжатие? если да, то скажите, пожалуйста, как. - person Chirag Joshi; 10.04.2020

Предыдущий ответ у меня не сработал.

Но это сработало отлично. Преобразовать URI данных в файл, а затем добавить в FormData

person love2code    schedule 23.06.2016

Вы можете использовать axios, async / await с TypeScript.

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')
person Fatih Bulut    schedule 28.06.2021