如何将图像同步绘制到 canvas
How to synchronously draw an image to a canvas
我有一张用 base64 编码的图像,我需要以同步方式在 canvas 上绘制它。看看我的这段代码:
//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
事件,因为那样会使它异步,这不好,因为这是一个生成器函数,我需要同步 return 和 canvas。该程序是围绕它构建的,使其异步意味着完全重写。这样不好。
尝试
input.src = dataurl;
while(!input.width){ /* busy wait */ }
var w = input.width, h = input.height;
不起作用,因为浏览器不会在我的代码执行完成之前更新它。
现在我真的想不出任何方法来解决这个难题,所以欢迎任何意见。 (请注意,datauri 是从另一个 canvas 生成的,我可以访问它。)
无法在加载图像之前将图像绘制到 canvas,这意味着要等待 load
事件。这意味着您无法将数据 URI 同步加载到 DOM 图像中,因此您需要将 API 转换为异步 URI。
唯一的其他方法是使用基于 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>
只需使用 Promise 使其异步。
return new Promise(resolve => {
image.onload = function () {
ctx.drawImage(input, 0, 0)
resolve('resolved');
}
});
我有一张用 base64 编码的图像,我需要以同步方式在 canvas 上绘制它。看看我的这段代码:
//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
事件,因为那样会使它异步,这不好,因为这是一个生成器函数,我需要同步 return 和 canvas。该程序是围绕它构建的,使其异步意味着完全重写。这样不好。
尝试
input.src = dataurl;
while(!input.width){ /* busy wait */ }
var w = input.width, h = input.height;
不起作用,因为浏览器不会在我的代码执行完成之前更新它。
现在我真的想不出任何方法来解决这个难题,所以欢迎任何意见。 (请注意,datauri 是从另一个 canvas 生成的,我可以访问它。)
无法在加载图像之前将图像绘制到 canvas,这意味着要等待 load
事件。这意味着您无法将数据 URI 同步加载到 DOM 图像中,因此您需要将 API 转换为异步 URI。
唯一的其他方法是使用基于 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>
只需使用 Promise 使其异步。
return new Promise(resolve => {
image.onload = function () {
ctx.drawImage(input, 0, 0)
resolve('resolved');
}
});