Alain Doe
Alain Doe

Reputation: 530

Canvas drawImage drawing unwanted pixels when stretching

I'm using drawImage to stretch a 1x2 pixel sprite into a 300x2 horizontal line.

This used to work perfectly fine, but recently I noticed a problem on Chrome, Firefox and Edge where the drawImage function would start including sprite data outside of the 1x2 defined box. ( Doesn't happen on mobile by the way )

In the below link, drawing the line works fine for small lengths, but once it goes over 255 pixels in length it will start drawing unwanted colours ( blue ).

JSFiddle; https://jsfiddle.net/y6bo1zfu/1/

var rawImage = false;

var testCanvas = document.createElement('canvas');
testCanvas.width = 300;
testCanvas.height = 20;
var testContext = testCanvas.getContext("2d");
testContext.imageSmoothingEnabled = false;
document.body.appendChild( testCanvas );

function loadImage()
{
    rawImage = new Image();
    rawImage.onload = function(){
        // Create the sprite image
        createSpriteImage();
        document.body.appendChild( rawImage );
    }
    rawImage.src = "";
}

function createSpriteImage()
{
    testContext.drawImage(
        rawImage,
        2,      // Source X for this part of the Sprite
        1,      // Source Y for this part of the Sprite
        1,      // Width of this part of the Sprite
        2,      // Height of this part of the Sprite

        0,          // The X position where to draw
        0,          // The Y position where to draw
        300,        // Width of the line
        2           // Height of the line
    );

    testContext.drawImage(
        rawImage,
        2,
        1,
        1,
        2,

        0,
        3,
        298,
        2,
    );

    testContext.drawImage(
        rawImage,
        2,
        1,
        1,
        2,

        0,
        6,
        264,
        2,
    );

    testContext.drawImage(
        rawImage,
        2,
        1,
        1,
        2,

        0,
        9,
        256,
        2
    );

    testContext.drawImage(
        rawImage,
        2,
        1,
        1,
        2,

        0,
        12,
        255,
        2
    );

    testContext.drawImage(
        rawImage,
        2,
        1,
        1,
        2,

        0,
        15,
        100,
        2
    );

    testContext.drawImage(
        rawImage,
        2,
        1,
        1,
        2,

        0,
        18,
        10,
        2
    );
}

//
loadImage();

Anyone have any idea what could be causing this? Could this be a known bug or an accidental feature added through optimization?

Is trying to stretch an image 256 times its default width a no-no?

Upvotes: 0

Views: 153

Answers (1)

Kaiido
Kaiido

Reputation: 136638

This is caused by Hardware Acceleration (HWA) i.e by the GPU. You can disable it on your own browser and you'll see it will work as intended.

Unfortunately you can't ask all your users to disable HWA.

One simple way to circumvent this exact issue is to create an ImageBitmap which contains only the required part of your image.
This is quite straight forward since the createImageBitmap() method accepts a 5 params version

createImageBitmap( source, sourceX, sourceY, sourceWidth, sourceHeight );

and the drawImage() one accepts a 5 params version

drawImage( source, destinationX, destinationY, destinationWidth, destinationHeight );

You can thus rewrite your 9 params version drawImage() to

drawImage( 
  await createImageBitmap( source, sourceX, sourceY, sourceWidth, sourceHeight ),
  destinationX, destinationY, destinationWidth, destinationHeight
)

var rawImage = false;

var testCanvas = document.createElement('canvas');
testCanvas.width = 300;
testCanvas.height = 20;
var testContext = testCanvas.getContext("2d");
testContext.imageSmoothingEnabled = false;
document.body.appendChild( testCanvas );

function loadImage()
{
  rawImage = new Image();
  rawImage.onload = function(){
    // Create the sprite image
    createSpriteImage();
    
    document.body.appendChild( rawImage );
  }
  rawImage.src = "";
}

async function createSpriteImage()
{
  testContext.drawImage(
    await createImageBitmap(rawImage,
      2,     // Source X for this part of the Sprite
      1 ,    // Source Y for this part of the Sprite
      1 ,    // Width of this part of the Sprite
      2      // Height of this part of the Sprite
    ),
    0,        // The X position where to draw
    0,        // The Y position where to draw
    300 ,     // Width of the line
    2      // Height of the line
  );
  
  testContext.drawImage(
    await createImageBitmap(rawImage,
      2,    // Source X for this part of the Sprite
      1 ,   // Source Y for this part of the Sprite
      1 ,   // Width of this part of the Sprite
      2     // Height of this part of the Sprite
    ) ,  
    0 ,       // The X position where to draw
    3 ,       // The Y position where to draw
    298 ,     // Width of the line
    2    // Height of the line
  );
  
  testContext.drawImage(
    await createImageBitmap(rawImage,
      2,    // Source X for this part of the Sprite
      1 ,   // Source Y for this part of the Sprite
      1 ,   // Width of this part of the Sprite
      2     // Height of this part of the Sprite
    ) ,
    0,        // The X position where to draw
    6,        // The Y position where to draw
    264 ,     // Width of the line
    2         // Height of the line
  );
  
  testContext.drawImage(
    await createImageBitmap(rawImage,
      2,    // Source X for this part of the Sprite
      1 ,   // Source Y for this part of the Sprite
      1 ,   // Width of this part of the Sprite
      2     // Height of this part of the Sprite
    ),
    0,        // The X position where to draw
    9,        // The Y position where to draw
    256 ,     // Width of the line
    2         // Height of the line
  );
  
  testContext.drawImage(
    await createImageBitmap(rawImage,
      2,     // Source X for this part of the Sprite
      1 ,    // Source Y for this part of the Sprite
      1 ,    // Width of this part of the Sprite
      2      // Height of this part of the Sprite
    ),
    0,       // The X position where to draw
    12,      // The Y position where to draw
    255 ,    // Width of the line
    2 ,      // Height of the line
  );
  
  testContext.drawImage(
    await createImageBitmap(rawImage,
      2,     // Source X for this part of the Sprite
      1 ,    // Source Y for this part of the Sprite
      1 ,    // Width of this part of the Sprite
      2      // Height of this part of the Sprite
    ),
    0,       // The X position where to draw
    15,      // The Y position where to draw
    100 ,    // Width of the line
    2 ,      // Height of the line
  );
  
  testContext.drawImage(
    await createImageBitmap(rawImage,
      2,     // Source X for this part of the Sprite
      1 ,    // Source Y for this part of the Sprite
      1 ,    // Width of this part of the Sprite
      2      // Height of this part of the Sprite
    ),
    0,       // The X position where to draw
    18,      // The Y position where to draw
    10 ,     // Width of the line
    2 ,      // Height of the line
  );
}

//
loadImage();
body,html {
	margin: 0px;
	background-color: #4f4f4f;
}
img
{
	display: block;
	width: 50px;
	height: 50px;
	margin: 0px auto;
  image-rendering: pixelated;
}
canvas
{
  display: block;
	width: 300px;
	height: 20px;
	box-shadow: 1px 1px 4px black;
	margin: 15px auto;
	background-color: white;
	border: 1px solid white;
}

And for browsers that do not support the ImageBitmap interface, at least this usage can be monkey-patched by using other canvases.

Upvotes: 1

Related Questions