Canvas drawImage рисует ненужные пиксели при растяжении - PullRequest
0 голосов
/ 29 марта 2020

Я использую drawImage, чтобы растянуть спрайт 1x2 пикселя в горизонтальную линию 300x2.

Раньше это прекрасно работало, но недавно я заметил проблему на Chrome, Firefox и Edge, когда функция drawImage могла бы включать в себя данные спрайта за пределами определенного поля 1x2. (Между прочим, на мобильных устройствах этого не происходит)

В приведенной ниже ссылке рисование линии отлично работает при небольшой длине, но как только ее длина превысит 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 раз больше его ширины по умолчанию, нет-нет?

1 Ответ

0 голосов
/ 29 марта 2020

Это вызвано аппаратным ускорением (HWA), т. Е. Графическим процессором. Вы можете отключить его в своем браузере, и вы увидите, что он будет работать так, как задумано.

К сожалению, вы не можете попросить всех своих пользователей отключить HWA.

Один простой способ обойти эту проблему - создать ImageBitmap, который содержит только необходимую часть вашего изображения.
Это довольно просто, поскольку метод createImageBitmap() принимает версию с 5 параметрами

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

, а drawImage() - версию с 5 параметрами

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

Таким образом, вы можете переписать свою версию с 9 параметрами 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();
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;
}

А для браузеров, которые не поддерживают интерфейс ImageBitmap, по крайней мере это использование может быть пропатчено с помощью других полотен.

...