Canvas drawImage拉伸时绘制不需要的像素

问题描述 投票:0回答:1

我正在使用drawImage将1x2像素的精灵拉伸到300x2水平线。

这曾经可以很好地工作,但是最近我注意到在Chrome,Firefox和Edge上存在问题,其中drawImage函数将开始在1x2定义的框之外包括sprite数据。(顺便说一句,不会在移动设备上发生)

在下面的链接中,对于较小的长度,绘制线条效果很好,但是一旦长度超过255像素,它将开始绘制不需要的颜色(蓝色)。

[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();

任何人都不知道是什么原因造成的?这可能是已知错误还是通过优化添加的意外功能?

是否尝试将图像拉伸为其默认宽度的256倍,不可以吗?

javascript html canvas drawimage
1个回答
0
投票

这是由硬件加速(HWA),即GPU引起的。您可以在自己的浏览器上将其禁用,然后它将按预期运行。

很遗憾,您不能要求所有用户禁用HWA。

解决此确切问题的一种简单方法是创建一个ImageBitmap,它仅包含图像的必需部分。这很简单,因为createImageBitmap()方法接受5个参数版本

createImageBitmap()

[createImageBitmap( source, sourceX, sourceY, sourceWidth, sourceHeight ); 接受5参数版本

drawImage()

因此您可以将9参数版本drawImage( source, destinationX, destinationY, destinationWidth, destinationHeight ); 重写为

drawImage()

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();

对于不支持ImageBitmap接口的浏览器,至少可以通过使用其他画布来修补此用法。

© www.soinside.com 2019 - 2024. All rights reserved.