7

I have an image encoded in base64 and I need to draw it on a canvas in a synchronous fashion. Take a look at this code I had:

//convert base64 to actual Image()
var input = new Image();
input.src = dataurl;

//draw on canvas
var w = input.width, h = input.height;
canvas.width = w; canvas.height = h;
ctx.drawImage(input, 0, 0)

This used to work on Chrome. I know it is not a good practice to assuming the browser will be able to finish loading before the next instruction, but it worked. After several updates later, it finally fails.

I cannot listen to the load event because that would make it async, which is not good because that's a generator function and I need to return the canvas synchronously. The program is built around it and making it async would mean complete rewrite. That's not good.


Attempt

input.src = dataurl;

while(!input.width){ /* busy wait */ }

var w = input.width, h = input.height;

Does not work since the browser won't update it before my code finishes executing.


Right now I really can't think of any way to solve this dilemma, so any input is appreciated. (Do note that the datauri is generated from another canvas, which I do have access to.)

Derek 朕會功夫
  • 92,235
  • 44
  • 185
  • 247
  • Will [this](http://stackoverflow.com/questions/4405336/how-to-copy-contents-of-one-canvas-to-another-canvas-locally) work? – sabithpocker Oct 21 '15 at 18:32
  • 1
    I'm guessing your other code depends on the canvas having the image already drawn -- otherwise you could just return the new canvas element in it's "un-drawImage'ed" state and the drawing will be done after the image loads. Ok, how about refactoring the code that calls your generator to supply a callback function that gets executed by the generator code inside `image.onload`? – markE Oct 21 '15 at 23:41
  • 1
    @markE Thanks for the suggestion. I have thought about adding a callback function, but that would be too much work for me to rewrite other functions that are calling it, since they should be synchronous as well. Anyway I have tried @sabithpocker's suggestions and modified my code slightly to return a `` instead of the image in base64 form. – Derek 朕會功夫 Oct 21 '15 at 23:59

2 Answers2

4

It's not possible to draw an image to a canvas before it is loaded, which means waiting to a load event. This means you cannot load a data URI into an DOM image synchronously, so you will need to convert your API to an asynchronous one.

The only other way would be to have a JavaScript-based image decoder that converts the data URI into a ImageData object, which would not be trivial to do and require users to load a lot more code.

There used to be a different hack involving innerHTML that would work, but it too no longer works.

Now-Broken Example:

var base64 = '';

var el = document.createElement('p');
el.innerHTML = '<img src="'+base64+'" />';
var input = el.firstChild;

//draw on canvas
var canvas = document.getElementById('canvas');
var w = input.width, h = input.height;
canvas.width = w; canvas.height = h;
var ctx = canvas.getContext('2d');
ctx.drawImage(input, 0, 0);
<canvas id="canvas"></canvas>
Alexander O'Mara
  • 58,688
  • 18
  • 163
  • 171
  • Great solution, I will definitely test this out later. – Derek 朕會功夫 Oct 21 '15 at 23:15
  • Unfortunately this does not work. It behaves the same as a normal `image.src = dataurl` and sometimes the image will not get loaded in time. – Derek 朕會功夫 Oct 21 '15 at 23:22
  • @Derek朕會功夫 Strange, seems to be working for me. It's certainly possible there is a race condition though. – Alexander O'Mara Oct 21 '15 at 23:48
  • There's definitely a race condition which I should've thought of when I wrote that part of the code... (race condition in a single threaded language. What...) But anyway, I have modified my code slightly to pass in a `canvas` node instead and it seems to be working now, though I still need more testing to see if there's any other hidden problems. Thanks again. – Derek 朕會功夫 Oct 21 '15 at 23:54
0

Just use Promise to make it async.

 return new Promise(resolve => {

          image.onload = function () {
                ctx.drawImage(input, 0, 0)
                resolve('resolved');
             }

    });