javascript - 在图像中查找特定颜色

标签 javascript jquery html image

我有这样一张图片:

Image 1

并喜欢在图像中找到特定颜色 (#ffa400/#ffffff) 并获取它们的出现百分比值(橙色可能是:75%,白色:25%)

NOTE: Keep in mind that I don't like to get the average color (in this case: orange) like Color Thief does, but I'D like to find a SPECIFIC color.


下面是一个片段,所以这就是我到目前为止所得到的全部内容,但代码似乎无法正常工作,而且我的想法似乎也不是一种高效且快速的方法。

function extract_colors(img) {
    var canvas = document.createElement("canvas");
    var c = canvas.getContext('2d');
    c.width = canvas.width = img.width;
    c.height = canvas.height = img.height;
    c.clearRect(0, 0, c.width, c.height);
    c.drawImage(img, 0, 0, img.width, img.height);
    return getColors(c);

}

function getColors(c) {
    var col, colors = {};
    var pixels, r, g, b, a;
    r = g = b = a = 0;
    pixels = c.getImageData(0, 0, c.width, c.height);
    for (var i = 0, data = pixels.data; i < data.length; i + = 4) {
        r = data[i];
        g = data[i + 1];
        b = data[i + 2];
        a = data[i + 3];
        if (a < (255 / 2))
            continue;
        col = rgbToHex(r, g, b);
        if (col == 'ffa400') { // find color #ffa400
            if (!colors[col])
                colors[col] = 0;
            colors[col] + +;
        }
    }
    return colors;
}


function rgbToHex(r, g, b) {
    return ((r << 16) | (g << 8) | b).toString(16);
}

var img = document.getElementById('img');
out.innerHTML = extract_colors(img)
<img id='img' width=100 src='/image/EPDlQ.png'>

<p id='out'>...</p>

EDIT 1.0

我正在尝试使用这样的代码,但它似乎也不起作用。

var orangeMatches=0, whiteMatches=0
        Jimp.read("/image/EPDlQ.png").then(function (image) {
            image.scan(0, 0, image.bitmap.width, image.bitmap.height, function (x, y, idx) {
              var red   = this.bitmap.data[ idx + 0 ];
              var green = this.bitmap.data[ idx + 1 ];
              var blue  = this.bitmap.data[ idx + 2 ];
              var alpha = this.bitmap.data[ idx + 3 ];

              if (red == 255 && green == 164 && blue == 0 && alpha == 1){
                orangeMatches++;
              }
              if (red == 255 && green == 255 && blue == 255 && alpha == 1){
                whiteMatches++;
              }
          });
          console.log(orangeMatches, whiteMatches)
        });

Edit 2.0

我想要某种容忍值 (10%),这样就不会计算每种颜色,而是将匹配在一起的颜色(例如 #ffa400 #ffa410)计算在一起。

最佳答案

一些注意事项

你图片的主色其实是ffa500,不是ffa400。实际上根本没有出现 ffa400。

另请记住,这是一个消除锯齿的位图图形。在视觉上我们只能看到两种颜色,但是为了平滑从一种颜色到另一种颜色的过渡,生成了一组“中间色”。换句话说,ffffff 和 ffa500 的总和不会正好是 100%。 (这就是为什么我下面的函数的输出将显示比所讨论的两个更多的颜色)


解决方案一

该解决方案的基本思想是获取图片中的每种颜色,将它们存储在一个公共(public)对象中,在该对象中计算每种颜色的出现次数。返回此对象后,我们可以稍后计算输出中某些特定颜色的出现次数。

这将返回如下内容:

{
    ffa500: 7802,
    ffa501: 4,
    ffa502: 2,
    ...,
    ffffff: 1919,
    total: 10000
}

现在您可以通过这种方式访问​​给定颜色的出现:

var imageData = extract_colors(img),
    white = imageData.ffffff,
    total = imageData.total;

要显示给定颜色覆盖图像的百分比,请执行以下操作(toFixed() 只是将值限制为两位小数):

whitePct = (white / total * 100).toFixed(2);

工作样本

function getColors(ctx) {

		// Get the canvas data
    var pixels = ctx.getImageData(0, 0, ctx.width, ctx.height),
        data = pixels.data,
        
        // Set up our output object to collect color data
        output = {};
    
    // For each color we encounter, check the
    // output object. If the color already exists
    // there, simply increase its counted value.
    // If it does not, create a new key.
    for (var i = 0; i < data.length; i+=4) {
        var r = data[i],
            g = data[i + 1],
            b = data[i + 2],
            col = rgbToHex(r, g, b);

        if( output[col] )
        	output[col]++
        else
        	output[col] = 1
    }
    
    // Count total
    var total = 0;
    for(var key in output) {
    	total = total + parseInt(output[key])
    }
    output.total = total;
    
    // Return the color data as an object
    return output;
}

// Our elements
var img = document.getElementById('img'),
	  out = document.getElementById('out');

// Retrieve the image data
var imageData = extract_colors(img),

    // Count our given colors
    white = imageData.ffffff,
    orange = imageData.ffa500,
    total = imageData.total,

    // Calculate percentage value
    whitePct = (white / total * 100).toFixed(2),
    orangePct = (orange / total * 100).toFixed(2);
  
out.innerHTML = `White: ${whitePct}%<br/>Orange: ${orangePct}%`;

// See the console for all colors identified
console.log(extract_colors(img))

// ----- These functions are left untouched ----- \\

function extract_colors(img) {
    var canvas = document.createElement("canvas");
    var c = canvas.getContext('2d');
    c.width = canvas.width = img.width;
    c.height = canvas.height = img.height;
    c.clearRect(0, 0, c.width, c.height);
    c.drawImage(img, 0, 0, img.width, img.height);
    return getColors(c);
}

function rgbToHex(r, g, b) {
    return ((r << 16) | (g << 8) | b).toString(16);
}
p, img {
 margin: 10px 5px;
 float: left
}
<!-- Image converted to base64 to keep it locally availible -->
<img id='img' width=100 src=''>
<p id='out'>...</p>


方案二

(将此解决方案视为草稿。如果要用于实时网站,可能需要进行一些调整,但至少它应该可以工作!)

这里的想法是调用 extract_colors() 并将我们想要映射的颜色作为参数。构建了一个通用对象来保存输出,并为每种颜色填充了一个对象。颜色对象由一个计数器和一个 rgb 值组成。使用 isNeighborColor() 函数,rgb 值不仅用于匹配特定颜色,还用于匹配任何接近它的颜色。根据需要调整容差级别。 (理想情况下,我猜这将是另一个论点)。

调用指定图像和计算出现次数的颜色的函数:

var imageData = extract_colors(img, 'ffffff', 'ffa500'),

这将为每种颜色输出一个对象:

{
  "ffffff": {
    "counter": 1979,
    "rgb": {
      "r": 255,
      "g": 255,
      "b": 255
    }
  },
  "ffa500": {
    "counter": 7837,
    "rgb": {
      "r": 255,
      "g": 165,
      "b": 0
    }
  },
  "total": 9816
}

我们想要的是访问计数器值:

var white = imageData.ffffff.counter;

工作样本

// Added a rest parameter for unlimited color input
function extract_colors(img, ...colors) {
    var canvas = document.createElement("canvas");
    var c = canvas.getContext('2d');
    c.width = canvas.width = img.width;
    c.height = canvas.height = img.height;
    c.clearRect(0, 0, c.width, c.height);
    c.drawImage(img, 0, 0, img.width, img.height);
    return getColors(c, ...colors);
}

// Unchanged
function rgbToHex(r, g, b) {
    return ((r << 16) | (g << 8) | b).toString(16);
}

// From: https://stackoverflow.com/a/5624139/2311559
function hexToRgb(hex) {
    var result = /^#?([a-f\d]{2})([a-f\d]{2})([a-f\d]{2})$/i.exec(hex);
    return result ? {
        r: parseInt(result[1], 16),
        g: parseInt(result[2], 16),
        b: parseInt(result[3], 16)
    } : null;
}

// From: https://stackoverflow.com/a/11506531/2311559
// Slightly modified
function isNeighborColor(color1, color2, tolerance) {
    tolerance = tolerance || 32;
    return Math.abs(color1.r - color2.r) <= tolerance
        && Math.abs(color1.g - color2.g) <= tolerance
        && Math.abs(color1.b - color2.b) <= tolerance;
}

function getColors(ctx, ...colors) {

		var pixels = ctx.getImageData(0, 0, ctx.width, ctx.height),
        data = pixels.data,
        output = {};
    
    // Build an 'output' object. This will hold one object for
    // each color we want to check. The color objects will 
    // each consist of a counter and an rgb value. The counter 
    // is used to count the occurrence of each color. The rgb 
    // value is used to match colors with a certain tolerance 
    // using the 'isNeighborColor()' function above.
    for (var i = 0; i < colors.length; i++) {
        output[colors[i]] = {
        	counter: 0,
          rgb: hexToRgb(colors[i])
        };
    }

    // For each pixel, match its color against the colors in our
    // 'output' object. Using the 'isNeighborColor()' function 
    // we will also match colors close to our input, given the 
    // tolerance defined.
    for (var i = 0; i < data.length; i+=4) {
        var r = data[i],
            g = data[i + 1],
            b = data[i + 2],
            colobj = {r, g, b},
            col = rgbToHex(r, g, b);

        Object.keys(output).map(function(objectKey, index) {
            var rgb = output[objectKey].rgb,
            		count = output[objectKey].counter;
            if(isNeighborColor(rgb, colobj)) {
            	output[objectKey].counter = count+1;
            }
        });

    }
    
    // Count total
    var total = 0;
    for(var key in output) {
    	total = total + parseInt(output[key].counter)
    }
    output.total = total;
    
    // Return the color data as an object
    return output;
}

// Our elements
var img = document.getElementById('img'),
	  out = document.getElementById('out');

// Retrieve the image data
var imageData = extract_colors(img,'ffffff','ffa500'),

    // Count our given colors
    white = imageData.ffffff.counter,
    orange = imageData.ffa500.counter,
    total = imageData.total,

    // Calculate percentage value
    whitePct = (white / total * 100).toFixed(2),
    orangePct = (orange / total * 100).toFixed(2);

out.innerHTML = `White: ${whitePct}%<br/>Orange: ${orangePct}%`;

// Optional console output
console.log(extract_colors(img,'ffffff','ffa500'));
p, img {
 margin: 10px 5px;
 float: left
}
<!-- Image converted to base64 to keep it locally availible -->
<img id='img' width=100 src=''>
<p id='out'>...</p>

关于javascript - 在图像中查找特定颜色,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47608864/

相关文章:

javascript - 访问变量的数组/范围

javascript - Microsoft Edge 的打印对话框在子窗口中打开

javascript - 从更改时的选择中检索值和数据属性

javascript - 我需要对 HTML 和 JavaScript 代码进行哪些更改才能使 character_count 正常工作?

javascript - 如何防止我的文本在更改背景后褪色? CSS\HTML

javascript - 如何在剑道组合框中更改/设置值

jquery - 如何使div可点击?

javascript - 如何——回调函数

html - 页脚和背景图像之间的大空白

javascript - 具有随机结果的组合生成器