如何使用JavaScript将画布图像转为灰度?

10

我有一张28x28像素大小的图片在HTML5画布中。我使用以下代码将画布的imageData作为RGBA(红色,绿色,蓝色,透明度)值的数组获取:

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

现在我想将图像灰度化,以便获得一个由784个值(28x28像素)组成的数组,其中每个像素只有一个值(而不是四个)。
我找到了很多不同的灰度化公式,有些是乘以RGB值,有些只是计算平均值 - 我真的不知道该使用哪个...
我也卡在获取784个值上了 - 它总是3136(因为有4个通道)...
提前谢谢!

是的,有多个公式可供选择。没有一个单一的真正公式。 - Bergi
好的,谢谢。但是在将每个像素的四个值转换为灰度后,我如何获得一个包含784个值的数组? - Tom
在你的for循环中,只需要有两个计数器 - 一个是i += 1,另一个是每次迭代都加4的j += 4 - Bergi
太好了!使用两个计数器的for循环完美地解决了问题(我不知道为什么我自己没有想到)- 谢谢! - Tom
这并不是 OP 所追求的,因此不是对这个问题主体的回答,但是对于标题来说,请注意 CSS 滤镜已经添加到 Canvas2D API 中,在支持的浏览器中,您可以通过在绘制带有“copy”全局组合操作模式的画布之前将上下文的滤镜设置为“grayscale(100%)”来实现灰度效果。 - Kaiido
2个回答

16

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

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" />

更新:
替代的lightness计算方法:

  • 维基百科 (Luma):
    let lightness = parseInt(pixels[i]*.299 + pixels[i + 1]*.587 + pixels[i + 2]*.114);

  • 其他来源(未知):
    let lightness = parseInt(3*pixels[i] + 4*pixels[i + 1] + pixels[i + 2] >>> 3);

  • 维基百科 (线性亮度):
    let lightness = 0.2126 * pixels[i] + 0.715 * pixels[i+1] + 0.0722 * pixels[i+2];


11
你可以直接为渲染上下文设置滤镜。这比计算每个像素的颜色更简单。
为此,您应该使用 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=""
/>


1
不是Safari :( https://developer.mozilla.org/zh-CN/docs/Web/API/CanvasRenderingContext2D/filter - murkle
!#$%& Safari!!! - Bron Davies
已经有一份关于此事的Pull Request https://github.com/WebKit/WebKit/pull/3793,很快可能就会在那里了。 - tzi

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接