Convert blob to base64
Asked Answered
A

16

303

This is a snippet for the code that I want to do Blob to Base64 string:

This commented part works and when the URL generated by this is set to img src it displays the image:

var blob = items[i].getAsFile();
//var URLObj = window.URL || window.webkitURL;
//var source = URLObj.createObjectURL(blob);
//console.log("image source=" + source);
                        
var reader = new FileReader();
reader.onload = function(event){
console.log(event.target.result)
}; // data url!
var source = reader.readAsBinaryString(blob);

The problem is with the lower code, the source variable generated is null

Update:

Is there an easier way to do this with JQuery to be able to create Base64 String from the Blob file as in the code above?

Alyse answered 6/9, 2013 at 4:55 Comment(1)
use btoa to convert bytes directly to base64; not sure what's up with the intermediate data urls in the answers.Explanatory
H
518
var reader = new FileReader();
reader.readAsDataURL(blob); 
reader.onloadend = function() {
  var base64data = reader.result;                
  console.log(base64data);
}

Form the docs readAsDataURL encodes to base64

As an awaitable function:

function blobToBase64(blob) {
  return new Promise((resolve, _) => {
    const reader = new FileReader();
    reader.onloadend = () => resolve(reader.result);
    reader.readAsDataURL(blob);
  });
}

Note: The blob's result cannot be directly decoded as Base64 without first removing the Data-URL declaration preceding the Base64-encoded data. To retrieve only the Base64 encoded string, first remove data:/;base64, from the result.

Housecoat answered 6/9, 2013 at 5:5 Comment(13)
The output String doesn't seem to look like base64?Alyse
@xybrek If you print read.result, you will see base64 in the string itself.Verminous
But, Concern is how to get just Base64 String from that instead of those other tags, e.g. I am getting data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAB4AAAAQ4CAYAAADo08FDAAAKoWlDQ…ApOxJ4m+EDQbZE3wR+j7XSz2unTElvZ+s/1Vw/8B6sw4fipgOycAAAAASUVORK5CYII= but I want just : iVBORw0KGgoAAAANSUhEUgAAB4AAAAQ4CAYAAADo08FDAAAKoWlDQ…ApOxJ4m+EDQbZE3wR+j7XSz2unTElvZ+s/1Vw/8B6sw4fipgOycAAAAASUVORK5CYII=Leftover
console.log( base64data.substr(base64data.indexOf(',')+1) );Photo
onloadend should come before readAsDataURL just in case something weird happens and it finishes loading before it reaches the next line of code. Obviously this would never happen but it's still good practice.Buffum
This answer is incorrect, it appends none-base64 characters to the front of the string.Hesler
combined with https://mcmap.net/q/66099/-how-to-get-a-file-or-blob-from-an-object-url for reading from a blob url, this really helped!Adamek
@Buffum that can't happen. JavaScript is single-threaded, nothing weird can happen in the while.Litch
@CristianTraìna readAsDataURL is a function call, and could, therefore, complete the entire load (including calling the event listeners) before the function returns. This is not how the function is implemented in every environment I know of, but JavaScript being single-threaded doesn't have anything to do with it. Furthermore, all it takes is for some junior developer to come along and unwittingly add an await statement or something in there, and suddenly you have a bug that is incredibly difficult to troubleshoot. As I said, "obviously this would never happen but it's still good practice."Buffum
@JoshuaSmith Yeah; I dunno, I mean, you could just use btoa(await blob.text()) and skip all this extra strangeness. Maybe btoa wasn't around in 2013?Explanatory
@Buffum No, it can't. The File API spec defines read ops; and loadend is never fired directly, as all paths to loadend queue the event task on the file reading task source, whose implied event loop is the same one that's executing the script.Explanatory
I.e. it will never fire "between lines of code"; the only opportunities it has is either during async awaits, or after the script has been fully executed. Also, on a higher level, W3C went to great lengths to build determinism into the web specs; and if it wasn't there in this case, most things just... wouldn't work -- in particular, API objects that begin firing events as soon as they are constructed would be essentially unusable.Explanatory
this is function blobToDataUrlVervain
H
57

There is a pure JavaScript way that is not depended on any stacks:

const blobToBase64 = blob => {
  const reader = new FileReader();
  reader.readAsDataURL(blob);
  return new Promise(resolve => {
    reader.onloadend = () => {
      resolve(reader.result);
    };
  });
};

For using this helper function you should set a callback, example:

blobToBase64(blobData).then(res => {
  // do what you wanna do
  console.log(res); // res is base64 now
});

I write this helper function for my problem on React Native project, I wanted to download an image and then store it as a cached image:

fetch(imageAddressAsStringValue)
  .then(res => res.blob())
  .then(blobToBase64)
  .then(finalResult => { 
    storeOnMyLocalDatabase(finalResult);
  });
Hilarity answered 15/4, 2020 at 10:6 Comment(0)
L
54

this worked for me:

var blobToBase64 = function(blob, callback) {
    var reader = new FileReader();
    reader.onload = function() {
        var dataUrl = reader.result;
        var base64 = dataUrl.split(',')[1];
        callback(base64);
    };
    reader.readAsDataURL(blob);
};
Lew answered 27/10, 2016 at 16:19 Comment(6)
What is the argument cb?Fokine
@FellowStranger commonly callback, used like blobToBase64(myBlob, function(base64String) {/*use the base64String*/}), because it's asyncLemmy
@yeahdixon, Seems I need your help. Look at this : #46192569Frontiersman
How do you wait for the execution? I always get undefinedEffortful
the callback is meant to be a function that gets passed that fires when completeLew
i got the error "edit:1 Not allowed to load local resource: blob". Anyone know how I can fix this?Artful
B
11
var audioURL = window.URL.createObjectURL(blob);
audio.src = audioURL;

var reader = new window.FileReader();
reader.readAsDataURL(blob);
reader.onloadend = function () {
     base64data = reader.result;
     console.log(base64data);
}
Brandie answered 2/2, 2017 at 15:16 Comment(1)
Nice, @Vemonus because FileReader expose into result attribute as base64, works perfectly.Cupro
Z
9
function bufferToBinaryString(arrayBuffer){
    return String.fromCharCode(...new Uint8Array(arrayBuffer));
}
(async () => console.log(btoa(bufferToBinaryString(await new Response(blob).arrayBuffer()))))();

or

function bufferToBinaryString(arrayBuffer){
    return String.fromCharCode(...new Uint8Array(arrayBuffer));
}
new Response(blob).arrayBuffer().then(arr_buf => console.log(btoa(bufferToBinaryString(arr_buf)))))

see Response's constructor, you can turn [blob, buffer source form data, readable stream, etc.] into Response, which can then be turned into [json, text, array buffer, blob] with async method/callbacks.

edit: as @Ralph mentioned, turning everything into utf-8 string causes problems (unfortunately Response API doesn't provide a way converting to binary string), so array buffer is use as intermediate instead, which requires two more steps (converting it to byte array THEN to binary string), if you insist on using native btoa method.

Zachariah answered 20/5, 2019 at 9:30 Comment(2)
.text() decodes using UTF8, what happens if the response has binary code? I believe this will fail for non text dataStephanystephen
I see your point, amended the answer (it becomes quite lengthy though). At this point it would be better to not insist on btoaZachariah
R
7
async function blobToBase64(blob) {
  return new Promise((resolve, _) => {
    const reader = new FileReader();
    reader.onloadend = () => resolve(reader.result);
    reader.readAsDataURL(blob);
  });
}

let blob = null; // <= your blob object goes here

blobToBase64(blob)
  .then(base64String => console.log(base64String));

See also:

Reprehensible answered 11/2, 2021 at 16:51 Comment(0)
G
7

async TypeScript variation:

async function blobToBase64Async(blob: Blob): Promise<string> {
  return new Promise((resolve, reject) => {
    const fileReader = new FileReader();
    fileReader.onerror = (e) => reject(fileReader.error);
    fileReader.onloadend = (e) => {
      const dataUrl = fileReader.result as string;
      // remove "data:mime/type;base64," prefix from data url
      const base64 = dataUrl.substring(dataUrl.indexOf(',') + 1);
      resolve(base64);
    };
    fileReader.readAsDataURL(blob);
  });
}

Sample usage:

async function fetchToBase64Async(url: string, init?: RequestInit): Promise<string> {
  try {
    const response = await fetch(url, init);
    if (!response.ok) {
      const responseText = await response.text();
      throw new Error("server status: " + response.status + "\n" + "server response:" + "\n" + responseText);
    }
    const blob = await response.blob();
    const base64 = await blobToBase64Async(blob);
    return base64;
  } catch (e) {
    throw new Error("failed to fetch: " + url + "\n" + "caused by: " + e);
  }
}

async function demoUsage() {
  const base64 = await fetchToBase64Async("https://httpstat.us/200", {
    method: "POST",
    headers: {
      "Accept": "*/*",
      "Authorization": "Bearer ...",
    }
  });
  console.log(base64);
}

Notes:

  • I don't understand why some answers use the load instead of the loadend event
  • I don't understand why some answers call readAsDataURL before setting the event handler
Gybe answered 4/6, 2021 at 14:5 Comment(1)
You should use load instead of loadend because loadend fires in case of both success and error. That is, you have 2 error handlers, which is ambiguous (it's unclear who will handle the error first).Anisotropic
S
7

The answer from @Arun Killu is a good snippet if you know what is going on, but nobody has explained what was the error on the original code. For people using async and Promise calls this is error is soo obvious but for people learning or without experience it's not so clear.

Here a simple explanation.

The Bad code enter image description here

var blob = items[i].getAsFile();

var reader = new FileReader();
reader.onload = function(event){
console.log(event.target.result)
}; // data url!
var source = reader.readAsBinaryString(blob);

Above code is trying to capture a binary string on source variable, however, FileReader.readAsBinaryString() returns undefined. This is because the result will be available whenever the event onload will be triggered. As we can see, he was trying to console.log the event.target.result value, which is a wrong approach.

The Good code enter image description here

Here is a step by step implementation:

// 1. Create a FileReader instance
const reader = new FileReader()
// 2. Add a handler for the 'onload' event
reader.onload = (e) => {
    // 5. Get the result when the 'onload' event is triggered.
    const base64data = reader.result               
    console.log({base64data})
}
// 3. Add a handler for the 'onerror' event
reader.onerror = () => {                
    console.log('error')
}
// 4. Call 'readAsDataURL' method
reader.readAsDataURL(imageBlob) 

As you can see the last step is 5 and it is because it is an asynchronous call.

Here is a working example:

const url = 'https://i.stack.imgur.com/RRuCp.png'

const fetchImage = async url => {
  const response = await fetch(url, {mode: 'no-cors',})
  const blob = await response.blob()
  
  return blob
}

const loadImage = async () => {
  const imageBlob = await fetchImage(url)

  const reader = new FileReader()
  reader.onload = () => {
    const base64data = reader.result               
    console.log({base64data})
  }
  reader.onerror = () => {                
    console.log('error')
  }
  reader.readAsDataURL(imageBlob) 
}

loadImage()

Teo, what means asynchronous?

Well young Padawan, asynchronous means that we don't know when the result will be ready, it can be different in each system and depends on how heavy or complex is the process and also it can find some errors that will not produce any result at all.

So if a process is asynchronous is a good practice to encapsulate it using an async method and returning a Promise like this:

const blobToBase64 = async blob => {
  return new Promise((resolve, reject) => {
    const reader = new FileReader();
    reader.onload = () => resolve(reader.result)
    reader.onerror = err => reject(err)
    reader.readAsDataURL(blob)
  })
}

Ah okay Teo, but what is a Promise?

Good question my young fella. A Promise object represents the eventual completion (or failure) of an asynchronous operation and its resulting value. In other words, will tell us if the result is ready and will give us its value, otherwise will return an error.

Above code shows a function blobToBase64 that will return a Promise This means that this function will return reader.result when it ready.

How can we integrate it into our code?

Super easy, just replace all the FileReader with the function blobToBase64 defined above and call it like this imageBase64 = await blobToBase64(imageBlob)

Check this snippet:

const url = 'https://i.stack.imgur.com/RRuCp.png'

const fetchImage = async url => {
  const response = await fetch(url, {
    mode: 'no-cors',
  })
  const blob = await response.blob()

  return blob
}

const blobToBase64 = async blob => {
  return new Promise((resolve, reject) => {
    const reader = new FileReader();
    reader.onload = () => resolve(reader.result)
    reader.onerror = err => reject(err)
    reader.readAsDataURL(blob)
  })
}

const loadImage = async() => {
  const imageBlob = await fetchImage(url)
  const imageBase64 = await blobToBase64(imageBlob)
  
  console.log({imageBase64})
}

loadImage()
Skitter answered 7/10, 2022 at 10:47 Comment(3)
I don't understand your question, usually you get octet-stream from ajax calls or $http service from angular. can you give an example of your code or share the image file you are using?Skitter
https://mcmap.net/q/67183/-convert-svg-to-image-jpeg-png-etc-in-the-browser check this answer I make about converting svg into a png formatSkitter
Neither of your two examples appear to work, no image is displayed, and "imageBase64" is just "data:". I have tested in my environment as well with the same, if not worse, results. Am I missing something ?Disesteem
S
6

So the problem is that you want to upload a base 64 image and you have a blob url. Now the answer that will work on all html 5 browsers is: Do:

  var fileInput = document.getElementById('myFileInputTag');
  var preview = document.getElementById('myImgTag');

  fileInput.addEventListener('change', function (e) {
      var url = URL.createObjectURL(e.target.files[0]);
      preview.setAttribute('src', url);
  });
function Upload()
{
     // preview can be image object or image element
     var myCanvas = document.getElementById('MyCanvas');
     var ctx = myCanvas.getContext('2d');
     ctx.drawImage(preview, 0,0);
     var base64Str = myCanvas.toDataURL();
     $.ajax({
         url: '/PathToServer',
         method: 'POST',
         data: {
             imageString: base64Str
         },
     success: function(data) { if(data && data.Success) {}},
     error: function(a,b,c){alert(c);}
     });
 }
Swab answered 4/2, 2017 at 8:50 Comment(0)
D
5

you can fix problem by:

var canvas = $('#canvas'); 
var b64Text = canvas.toDataURL();
b64Text = b64Text.replace('data&colon;image/png;base64,','');
var base64Data = b64Text;

I hope this help you

Dugas answered 15/12, 2016 at 12:36 Comment(0)
D
5

Another way is to use a simple wrapper around FileReader returning Observable (snippet is in TypeScript):

  function toBase64(blob: Blob): Observable<string> {
    const reader = new FileReader();
    reader.readAsDataURL(blob);
    return fromEvent(reader, 'load')
      .pipe(map(() => (reader.result as string).split(',')[1]));
  }

Usage:

toBase64(blob).subscribe(base64 => console.log(base64));
De answered 6/12, 2020 at 11:27 Comment(0)
R
3

Typescript version :

const blob2Base64 = (blob:Blob):Promise<string> => {
      return new Promise<string> ((resolve,reject)=> {
           const reader = new FileReader();
           reader.readAsDataURL(blob);
           reader.onload = () => resolve(reader.result.toString());
           reader.onerror = error => reject(error);
       })
      }

usage:

blob2Base64(blob).then(res=>console.log(res))
Redstart answered 21/6, 2021 at 16:57 Comment(1)
The onerror callback argument is not an error. You should use reader.onerror = () => reject(reader.error) instead.Anisotropic
E
2

Maybe I'm missing something but

let encoded = btoa(await myblob.text());

... is all you need to do to encode a Blob's data to base64. See Blob.text() and btoa().

Or if you want the whole thing as a promise:

let encode = myblob.text().then(btoa);

PS: To decode back to a Blob: new Blob([atob(encoded)])

Explanatory answered 30/5, 2022 at 17:18 Comment(2)
May fail when the blob has special "chars": DOMException: Failed to execute 'btoa' on 'Window': The string to be encoded contains characters outside of the Latin1 range.Deery
@RoyShilkrot Addressed in docs, with solutions (also mentioned here in btoa doc link in post).Explanatory
M
0

I wanted something where I have access to base64 value to store into a list and for me adding event listener worked. You just need the FileReader which will read the image blob and return the base64 in the result.

createImageFromBlob(image: Blob) {
    const reader = new FileReader();
    const supportedImages = []; // you can also refer to some global variable
    reader.addEventListener(
      'load',
      () => {
        // reader.result will have the required base64 image
        const base64data = reader.result;
        supportedImages.push(base64data); // this can be a reference to global variable and store the value into that global list so as to use it in the other part
      },
      false
    );
    // The readAsDataURL method is used to read the contents of the specified Blob or File.
    if (image) {
      reader.readAsDataURL(image);
    }
 }

Final part is the readAsDataURL which is very important is being used to read the content of the specified Blob

Mundt answered 31/7, 2020 at 5:21 Comment(0)
I
0

If your "blob" is an actual Blob Object and not a blob url, the conversion is pretty simple:

const reader = new FileReader()
reader.readAsDataURL(blob)
reader.onload = () => resolve(reader.result)

example of Blob Object:

console.log(blob)

output:

Blob {lastModified: 1658039931443, lastModifiedDate: Sun Jul 17 2022 08:38:51 GMT+0200 (Central European Summer Time), name: '1.jpg', size: 35493, type: 'image/jpeg'}
    lastModified: 1658039931443
    lastModifiedDate: Sun Jul 17 2022 08:38:51 GMT+0200 (Central European Summer Time) {}
    name: "1.jpg"
    size: 35493
    type: "image/jpeg"
    [[Prototype]]: Blob

In my case, the blob was produced by Compressorjs (should you need image compression).

Irremovable answered 17/7, 2022 at 6:43 Comment(0)
O
0

In angular you can request the file as an array buffer then map it to base64

this.http.get('./101.jpg', {responseType: 'arraybuffer'}).pipe(
    map(q => String.fromCharCode(...new Uint8Array(q))),
    map(q => btoa(q)),
).subscribe(q => {
    console.log(q);
})
Oina answered 5/12, 2023 at 15:7 Comment(0)

© 2022 - 2024 — McMap. All rights reserved.