如何将图像同步绘制到画布上

7

我有一张以base64编码的图片,我需要同步地在画布上绘制它。看一下我之前写的代码:

//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)

这在Chrome上曾经可以工作。我知道假设浏览器能够在下一个指令之前完成加载并不是一个好的实践,但它确实可以工作。在几次更新之后,它最终失败了。
我无法监听“load”事件,因为那会使其变成异步的。由于这是一个生成器函数,并且我需要同步返回canvas,所以这并不好。程序是围绕它构建的,将其变成异步将意味着完全重写。那不好。
input.src = dataurl;

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

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

由于浏览器在我的代码执行完成之前不会更新它,所以这种方法无法使用。


目前我真的想不出任何解决这个困境的办法,因此非常感谢您的建议。(请注意,datauri是从另一个画布生成的,我可以访问该画布。)


这个能行吗?链接 - sabithpocker
1
我猜你的其他代码依赖于画布已经绘制好图像 - 否则,你可以只返回新的画布元素,处于“未绘制图像”的状态,绘制将在图像加载后完成。好的,那么重构调用生成器的代码,提供一个回调函数,由生成器代码在image.onload内执行? - markE
1
@markE 感谢您的建议。我考虑过添加回调函数,但是这将需要我重写调用它的其他函数,因为它们也应该是同步的,这太麻烦了。无论如何,我已经尝试了@sabithpocker的建议,并稍微修改了我的代码,以返回一个<canvas>而不是base64格式的图像。 - Derek 朕會功夫
2个回答

4
在画布加载完成之前无法将图像绘制到画布上,这意味着需要等待load事件。这意味着您无法同步地将数据URI加载到DOM图像中,因此您需要将API转换为异步API,以便进行操作。
另一种方法是拥有一个基于JavaScript的图像解码器,可以将数据URI转换为ImageData对象,但这并不容易实现,还需要用户加载更多代码。
曾经有一种使用innerHTML的hack方法可以解决该问题,但现在已经失效了。 现在已失效的示例:

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>


很棒的解决方案,我肯定以后会测试一下。 - Derek 朕會功夫
很遗憾,这并不起作用。它的行为与普通的 image.src = dataurl 相同,有时图像无法及时加载。 - Derek 朕會功夫
@Derek朕会功夫,奇怪,對我而言似乎可以運作。然而可能存在競態條件。 - Alexander O'Mara
当我编写代码的时候,我应该考虑到一个竞态条件...(单线程语言中的竞态条件。什么...)但无论如何,我稍微修改了我的代码,传入了一个“canvas”节点,现在似乎可以工作了,尽管我仍需要进行更多测试,以查看是否存在其他隐藏问题。再次感谢。 - Derek 朕會功夫

0

只需使用 Promise 使其异步化。

 return new Promise(resolve => {

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

    });

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