このatob
関数は、Base64でエンコードされた文字列を、バイナリデータの各バイトの文字を含む新しい文字列にデコードします。
const byteCharacters = atob(b64Data);
各文字のコードポイント(charCode)は、バイトの値になります。.charCodeAt
文字列の各文字にメソッドを使用してこれを適用することにより、バイト値の配列を作成できます。
const byteNumbers = new Array(byteCharacters.length);
for (let i = 0; i < byteCharacters.length; i++) {
byteNumbers[i] = byteCharacters.charCodeAt(i);
}
このバイト値の配列をUint8Array
コンストラクターに渡すことにより、実際の型付きバイト配列に変換できます。
const byteArray = new Uint8Array(byteNumbers);
これは、配列にラップしてBlob
コンストラクタに渡すことにより、BLOBに変換できます。
const blob = new Blob([byteArray], {type: contentType});
上記のコードは機能します。ただし、byteCharacters
一度にすべてではなく、より小さなスライスでを処理することにより、パフォーマンスを少し向上させることができます。私の大まかなテストでは、512バイトが適切なスライスサイズのようです。これにより、次の機能が得られます。
const b64toBlob = (b64Data, contentType='', sliceSize=512) => {
const byteCharacters = atob(b64Data);
const byteArrays = [];
for (let offset = 0; offset < byteCharacters.length; offset += sliceSize) {
const slice = byteCharacters.slice(offset, offset + sliceSize);
const byteNumbers = new Array(slice.length);
for (let i = 0; i < slice.length; i++) {
byteNumbers[i] = slice.charCodeAt(i);
}
const byteArray = new Uint8Array(byteNumbers);
byteArrays.push(byteArray);
}
const blob = new Blob(byteArrays, {type: contentType});
return blob;
}
const blob = b64toBlob(b64Data, contentType);
const blobUrl = URL.createObjectURL(blob);
window.location = blobUrl;
完全な例:
const b64toBlob = (b64Data, contentType='', sliceSize=512) => {
const byteCharacters = atob(b64Data);
const byteArrays = [];
for (let offset = 0; offset < byteCharacters.length; offset += sliceSize) {
const slice = byteCharacters.slice(offset, offset + sliceSize);
const byteNumbers = new Array(slice.length);
for (let i = 0; i < slice.length; i++) {
byteNumbers[i] = slice.charCodeAt(i);
}
const byteArray = new Uint8Array(byteNumbers);
byteArrays.push(byteArray);
}
const blob = new Blob(byteArrays, {type: contentType});
return blob;
}
const contentType = 'image/png';
const b64Data = 'iVBORw0KGgoAAAANSUhEUgAAAAUAAAAFCAYAAACNbyblAAAAHElEQVQI12P4//8/w38GIAXDIBKE0DHxgljNBAAO9TXL0Y4OHwAAAABJRU5ErkJggg==';
const blob = b64toBlob(b64Data, contentType);
const blobUrl = URL.createObjectURL(blob);
const img = document.createElement('img');
img.src = blobUrl;
document.body.appendChild(img);