如何在JavaScript中对画布图像进行灰度处理?

时间:2018-11-18 18:25:14

标签: javascript html5 canvas grayscale imagedata

我在HTML5画布中有一个图像,尺寸为28x28像素。我使用以下代码将画布的imageData作为RGBA(红色,绿色,蓝色,alpha)值的数组获取:

canvas = document.getElementById('canvas');
ctx = canvas.getContext("2d");
imgData = ctx.getImageData(0, 0, 28, 28);

现在,我想对图像进行灰度处理,以便获得784个值(28x28像素)的数组,其中每个像素都有一个值(而不是四个)。

我发现了很多不同的灰阶公式,有些正在乘以rgb值,有些只是计算平均值-我真的不知道要使用哪个...

我也一直坚持要获得784个值-始终为3136(因为有4个通道)...

谢谢!

2 个答案:

答案 0 :(得分:1)

主要思想是使颜色的红色,绿色和蓝色分量具有相同的值。为此,您需要计算每个像素的亮度。有几种计算亮度的方法。这就是其中之一。

window.onload = function() {
let canvas = document.getElementById("c");

	  let ctx = canvas.getContext("2d");
    canvas.width=50;
    canvas.height=50;

		let srcImg = document.getElementById("sof");
		ctx.drawImage(srcImg, 0, 0, ctx.canvas.width, ctx.canvas.height);
		let imgData = ctx.getImageData(0, 0, ctx.canvas.width, ctx.canvas.height);
		let pixels = imgData.data;
		for (var i = 0; i < pixels.length; i += 4) {
		  
		let lightness = parseInt((pixels[i] + pixels[i + 1] + pixels[i + 2])/3);
    
		  pixels[i] = lightness; 
		  pixels[i + 1] = lightness; 
		  pixels[i + 2] = lightness; 
		}
		ctx.putImageData(imgData, 0, 0);
	  }
<canvas id="c"></canvas>
<img src="" id="sof" />

更新

这是wikipedia

中亮度的另一个公式
let lightness = parseInt(pixels[i]*.299 + pixels[i + 1]*.587 + pixels[i + 2]*.114);

这是我在Google中找到的另一个公式:

let lightness = parseInt(3*pixels[i] + 4*pixels[i + 1] + pixels[i + 2] >>> 3);

答案 1 :(得分:1)

您可以直接为渲染上下文设置过滤器。它会比计算每个像素颜色更简单。

为此,您应该使用 ctx.filter = 'grayscale(1)';

const img = document.querySelector('img');
const canvas = document.querySelector('canvas');
const ctx = canvas.getContext('2d');

img.onload = function() {
  canvas.width = img.width;
  canvas.height = img.height;
  ctx.filter = 'grayscale(1)';
  ctx.drawImage(img, 0, 0, img.width, img.height);
}
<canvas></canvas>
<img src=""
/>