views:

54

answers:

1

Hi.

I've got a problem with canvas context drawImage() method if an image is drawn on a canvas that already has not-integer scale factor. It seems that such images are clipped in a weird way (sometimes the most right part of an image is clipped, sometimes the most bottom side, sometimes both). This problem appears at least in Google Chrome 6 (at least stable), Chromium 6, perhaps even latest (dev-)builds too, and even Safari 5. Firefox has not this bug. Obviously, it's a Webkit problem if I'm not wrong. Let's look at the following code (provided entire code for a demo):

<html>
<head>
  <style type="text/css">
    canvas {
      border: solid 1px black;
    }
  </style>
  <script type="text/javascript">    
    window.onload = function() {
      var canvas = document.getElementById("canvas");
      var increase = document.getElementById("increase");
      var decrease = document.getElementById("decrease");
      var scale = document.getElementById("scale");
      var context = canvas.getContext("2d");
      var image = new Image();
      image.src = "image-3x5.png";
      var scaleX = 1;
      var scaleY = 1;
      var repaint = function() {
        scale.innerHTML = scaleX + "; " + scaleY;
        context.fillStyle = "#FFF";
        context.fillRect(0, 0, 1000, 750);
        context.drawImage(image, 0, 0, 3, 5);
      };
      var scaleXF = 1.2; // change both to 2 and it will be fixed
      var scaleYF = 1.2;
      decrease.onclick = function() {
        scaleX /= scaleXF;
        scaleY /= scaleYF;
        context.scale(1 / scaleXF, 1 / scaleYF);
        repaint();
      };
      increase.onclick = function() {
        scaleX *= scaleXF;
        scaleY *= scaleYF;
        context.scale(scaleXF, scaleYF);
        repaint();
      };
      repaint();
    };
  </script>
</head>
<body>
  <div>
    <span id="scale"></span>
  </div>
  <div>
    <canvas id="canvas" width="1000" height="750"></canvas>
  </div>
  <div>    
    <input id="decrease" type="button" value="decrease" />    
    <input id="increase" type="button" value="increase" />    
  </div>
</body>
</html>

So is it a real bug? Or are there any workarounds?

Thanks in advance.

UPD:

Let's consider we have the following BASE64-encoded 3x5 image:


A: 

You're right, it's a bug.

The difference between

context.setTransform(1*55.206143891243606, 0, 0, 1*55.206143891243606, 0, 0);

and

context.setTransform(1*55, 0, 0, 1*55, 0, 0);

Is massive. But only when drawing images. When filling a rectangle it works correctly.

I did a bit of testing and made this page to illustrate. The top two are with drawImage and a scale factor of 55 and 55.206143891243606. They clearly look very different.

Below are two more canvases with the same respective scale factors and a fillRect command. They look fine.

http://simonsarris.com/misc/badscale.html

This should be reported to the chromium issue tracker. Let me know if you want to do it, otherwise I will do it.

Good catch, by the way :D

Simon Sarris
I've reported it here - http://code.google.com/p/chromium/issues/detail?id=61198
Lyubomyr Shaydariv