我在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个通道)...
谢谢!
答案 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" />
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=""
/>