2

I have the same image and the same size of canvas, but the output is different. I want the same output, how should I do it?

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;
Pang
  • 9,564
  • 146
  • 81
  • 122
Zan
  • 23
  • 4
  • 1
    you could specify format and compression instead of leaving it to defaults. even that might not give you bit-perfect matching outputs though, as there are many ways to encode the same image in a given format. – dandavis Mar 29 '16 at 01:46
  • Possible duplicate of [Is canvas getImageData method machine/browser dependent?](http://stackoverflow.com/q/26615580/1529630) – Oriol Mar 29 '16 at 02:16
  • @Oriol, while clearly related, I don't think it's a dupe since here OP is asking for a way to get the same output. (I already had upvoted your answer there so I can't do it anymore...) – Kaiido Mar 29 '16 at 02:19
  • 1
    @Kaiido True, I meant "closely related", and didn't vote to close. My answer there explains why canvas fails, but your answer here explains how it can be done without canvas, +1 – Oriol Mar 29 '16 at 02:32

1 Answers1

3

Images drawn onto a canvas are decoded before being drawn, then reencoded when the toDataURL method is called.
This process will produce some variations in every browser (e.g some will be able to decode color-profiles embedded in the image while others won't), and even every machine (look at canvas fingerprinting and this post by @Oriol which concern images with transparency). Add to that that every browser will use different encoders/settings for a different tradeoff between speed, size and quality and you arrive at a situation where you can't expect two users to produce the same result from the same input image.

But since all you do with that canvas is to draw an image, you should rather use a FileReader and its method readAsDataURL(). For external files, you can still use it by first fetching the resource as a Blob.

This will work directly on the binary data of the file, encoding each byte to its base64 representation, and thus you will be sure to have the same result in every browser.

Here is a snippet which will test your browser's conversion against mine's.

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 = ""

However for the ones using the canvas for real drawing, as we said above, you can't really have the same results between different UAs. Every method from drawing ones to export ones may produce different results and you would have to actually rewrite all these methods entirely in order to have the exact same results.

For the ones that really need this, a project like node-canvas could help, though it would obviously be a lot less performant than native implementations.

Kaiido
  • 123,334
  • 13
  • 219
  • 285