Html canvas、像素操作和 alpha 通道

Html canvas, pixel manipulation and alpha channel

在htmlcanvas中,我想实现的是图片上的白噪声效果。所以我得到了图片,得到它的 ImageData,修改它的 alpha 并把它画回来。

var bat = new Image();
bat.src = ""

var canvas = document.getElementById('canvas');
var ctx = canvas.getContext('2d');

canvas.width = 250;
canvas.height = 250;

function animate() {
  ctx.clearRect(0, 0, canvas.width, canvas.height);

  ctx.drawImage(bat, 0, 0);

  var imageData = ctx.getImageData(0, 0, canvas.width, canvas.height);

  var data = imageData.data;

  for (let i = 0; i < data.length; i += 4) {
    if (data[i] != 0 || data[i + 1] != 0 || data[i + 2] != 0) {
      data[i + 3] = Math.floor(Math.random() * 255);
    }
  }

  ctx.putImageData(imageData, 0, 0);

  requestAnimationFrame(animate);
}



animate();
<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <meta http-equiv="X-UA-Compatible" content="IE=edge" />
    <meta name="viewport" content="width=device-width, initial-scale=1.0" />
    <title>Bat</title>
    <link rel="stylesheet" href="style.css" />
    <style>
      #main {
        display: flex;
        flex-direction: row;
      }
      #canvas {
        border: 1px solid black;
        width: 150px;
        height: 150px;
      }
    
    </style>
  </head>

  <body>
    <div id="main">
      <canvas id="canvas"></canvas>
      <br />
      
    </div>

    <script src="script.js"></script>
  </body>
</html>

结果就是我需要的。但问题是当你使用putImageData时,即使alpha为0,它也不会透明。

所以我用了一个临时的canvas,在上面画图,修改一下再画回canvas。

var bat = new Image();
bat.src = "";

var canvas2 = document.getElementById('canvas2');
var ctx2 = canvas2.getContext('2d');

var tempCanvas = document.createElement('canvas');
var tempContext = tempCanvas.getContext('2d');

canvas2.width = 250;
canvas2.height = 250;

tempCanvas.width = 250;
tempCanvas.height = 250;

var img = new Image();

var alpha = 0;

function animate2() {
  alpha += 0.1;

  ctx2.clearRect(0, 0, canvas.width, canvas.height);

  ctx2.fillStyle = 'blue';
  ctx2.fillRect(0, 0, 250, 250);

  tempContext.drawImage(bat, 0, 0);

  var imageData = tempContext.getImageData(
    0,
    0,
    tempCanvas.width,
    tempCanvas.height
  );

  var data = imageData.data;

  for (let i = 0; i < data.length; i += 4) {
    if (data[i] != 0 || data[i + 1] != 0 || data[i + 2] != 0) {
      //data[i + 3] = Math.floor(Math.random() * 255);
      data[i + 3] = 123;
      //data[i + 3] = alpha;
    }
  }

  tempContext.putImageData(imageData, 0, 0);

  img.src = tempCanvas.toDataURL('image/png');

  ctx2.drawImage(img, 0, 0);

  requestAnimationFrame(animate2);
}

animate2();
<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <meta http-equiv="X-UA-Compatible" content="IE=edge" />
    <meta name="viewport" content="width=device-width, initial-scale=1.0" />
    <title>Bat</title>
    <link rel="stylesheet" href="style.css" />
    <style>
      #main {
        display: flex;
        flex-direction: row;
      }
      
      #canvas2 {
        border: 1px solid black;
        width: 150px;
        height: 150px;
      }
    </style>
  </head>

  <body>
    <div id="main">
      <canvas id="canvas"></canvas>
      <br />
      <canvas id="canvas2"></canvas>
    </div>

    <script src="script.js"></script>
  </body>
</html>

但是每当我想随机更改 alpha 时:

data[i + 3] = Math.floor(Math.random() * 255);

图片不会显示。如果我想在其他时间更新 alpha:

var alpha = 0;
alpha += 0.1
data[i + 3] = alpha;

画面闪烁...

获得透明的“白噪声”图片以便我可以透过它看到背景的正确方法是什么?

这是 stackblitz with the demo in it 的 link。

首先,不要每帧都调用getImageData。您只需要知道图像中黑色像素的位置,这些不会改变,因此您可以一直保持相同的 ImageData 对象,避免 GPU 变慢 read-backs。

然后,你可以直接drawImage()一个canvas。无需通过带有 toDataURL() 的新图像,这将异步加载您的图像并确实使您的动画闪烁。

所以这会给出:

var bat = new Image();
bat.src = "";

var canvas2 = document.getElementById('canvas2');
var ctx2 = canvas2.getContext('2d');

var tempCanvas = document.createElement('canvas');
var tempContext = tempCanvas.getContext('2d');

canvas2.width = 250;
canvas2.height = 250;

tempCanvas.width = 250;
tempCanvas.height = 250;
// store the ImageData in a way it's accessible at every frame
var imageData;
bat.onload = (evt) => { // always wait for the assets to load
  ctx2.drawImage(bat, 0, 0);
  imageData = ctx2.getImageData(0, 0, 250, 250);
  animate2();
}

var alpha = 0;

function animate2() {
  ctx2.clearRect(0, 0, canvas2.width, canvas2.height);

  ctx2.fillStyle = 'blue';
  ctx2.fillRect(0, 0, 250, 250);

  var data = imageData.data;

  for (let i = 0; i < data.length; i += 4) {
    if (data[i] != 0 || data[i + 1] != 0 || data[i + 2] != 0) {
      data[i + 3] = Math.floor(Math.random() * 255);
    }
  }

  tempContext.putImageData(imageData, 0, 0);
  // drawImage the tempCanvas directly
  ctx2.drawImage(tempCanvas, 0, 0);

  requestAnimationFrame(animate2);
}
#main {
    display: flex;
    flex-direction: row;
  }

  #canvas2 {
    border: 1px solid black;
    width: 150px;
    height: 150px;
  }
<div id="main">
  <canvas id="canvas2"></canvas>
</div>

但是在这里您甚至不需要第二个 canvas,由于 "destination-over" 合成模式,您可以在 canvas 上绘制当前绘图的“后面”。您甚至可以使用更多合成操作来裁剪背景,以便只有图像是“彩色的”:

var bat = new Image();
bat.src = "";

var canvas2 = document.getElementById('canvas2');
var ctx2 = canvas2.getContext('2d');

canvas2.width = 250;
canvas2.height = 250;

// store the ImageData in a way it's accessible at every frame
var imageData;
bat.onload = (evt) => { // always wait for the assets to load
  ctx2.drawImage(bat, 0, 0);
  imageData = ctx2.getImageData(0, 0, 250, 250);
  animate2();
}

function animate2() {
  var data = imageData.data;

  for (let i = 0; i < data.length; i += 4) {
    if (data[i] != 0 || data[i + 1] != 0 || data[i + 2] != 0) {
      data[i + 3] = Math.floor(Math.random() * 255);
    }
  }
  ctx2.putImageData(imageData, 0, 0);

  ctx2.globalCompositeOperation = "destination-over"; // draw behind
  ctx2.fillStyle = 'blue';
  ctx2.fillRect(0, 0, 250, 250);
  ctx2.globalCompositeOperation = "destination-in"; // keep only where new pixels are opaque
  ctx2.drawImage(bat, 0, 0);
  ctx2.globalCompositeOperation = "source-over"; // default
  requestAnimationFrame(animate2);
}
#main {
    display: flex;
    flex-direction: row;
  }

  #canvas2 {
    border: 1px solid black;
    width: 150px;
    height: 150px;
    background: yellow
  }
<div id="main">
  <canvas id="canvas2"></canvas>
</div>