canvas2d到DataURL()在不同的浏览器上有不同的输出

canvas2d toDataURL() different output on different browser

本文关键字:浏览器 输出 DataURL canvas2d      更新时间:2023-09-26

我有相同的图像和相同大小的画布,但输出不同。我想要相同的输出,我应该怎么做?

var canvas = document.createElement('canvas'),
    ctx = canvas.getContext('2d'),
    img = new Image;
img.crossOrigin = 'Anonymous';
img.onload = function(){
    canvas.height = img.height;
    canvas.width = img.width;
    ctx.drawImage(img, 0, 0);
    var dataURL = canvas.toDataURL();
    setBreakpoint(dataURL);
    callback.call(this, dataURL);
    canvas = null;
};
img.src = url;

绘制到画布上的图像在绘制之前会进行解码,然后在调用toDataURL方法时重新编码
这个过程会在每个浏览器中产生一些变化(例如,有些浏览器能够解码嵌入图像中的颜色配置文件,而另一些浏览器则无法解码),甚至在每台机器中(看看画布指纹和@Oriol的这篇关于透明图像的帖子)。此外,每个浏览器都会使用不同的编码器/设置来在速度、大小和质量之间进行不同的权衡,这样你就无法指望两个用户从同一输入图像中产生相同的结果。

但是,由于您使用画布所做的只是绘制图像,因此您应该使用FileReader及其方法readAsDataURL()。对于外部文件,您仍然可以通过首先以Blob的形式获取资源来使用它。

这将直接作用于文件的二进制数据,将每个字节编码为其base64表示,因此您将确保在每个浏览器中都有相同的结果。

这里有一个片段,将测试你的浏览器与我的浏览器的转换。

fetch("https://dl.dropboxusercontent.com/s/4e90e48s5vtmfbd/aaa.png")
  .then((resp) => resp.ok && resp.blob())
  .then((blob) => {
    const reader = new FileReader();
    reader.onload = (evt) => {
      if (reader.result === imageDataURL) {
        console.log("same result");
      }
      else {
        console.log("please post a comment stating which browser has such a bug");
      }
    };
    reader.readAsDataURL(blob);  
  });
var imageDataURL = ""


然而,对于那些使用画布进行真实绘制的用户,正如我们上面所说的,在不同的UA之间不可能有相同的结果。从绘制方法到导出方法,每种方法都可能产生不同的结果,你必须完全重写所有这些方法才能获得完全相同的结果。

对于真正需要这一点的人来说,像node canvas这样的项目可能会有所帮助,尽管它的性能显然比本机实现低得多。