简体   繁体   English

在图像中查找特定颜色

[英]Find specific color in image

I have an image like this: 我有这样的图像:

图片1

and like to find a specific color in the image ( #ffa400 / #ffffff ) and get their percentage occurrence value (orange maybe: 75% , white: 25% ) 并想找到图像中的特定颜色( #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. 注意:请记住,我不喜欢像Color Thief那样获得平均颜色(在这种情况下: 橙色 ),但我想找到一种SPECIFIC颜色。


Below is a snippet, so this is all that I got so far, but the code doesn't seem working & my idea also doesn't seem like an efficient & fast way. 下面是一个片段,所以这就是我到目前为止所做的一切,但代码似乎不起作用,我的想法似乎也不是一种有效而快速的方法。

 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='https://i.stack.imgur.com/EPDlQ.png'> <p id='out'>...</p> 

EDIT 1.0 编辑1.0

I'm trying to use a code like this, but it doesn't seems to work either. 我正在尝试使用这样的代码,但它似乎也没有用。

var orangeMatches=0, whiteMatches=0
        Jimp.read("https://i.stack.imgur.com/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 编辑2.0

I'd like to have some kind of tolerant value (10%) so that there is not every color to be counted, but the colors (eg #ffa400 #ffa410) that are matching together are counted together. 我希望有一些容忍值(10%),这样就不会计算每种颜色,但是匹配在一起的颜色(例如#ffa400#ffa410)会被计算在一起。

A few notes 几点说明

The main color in your picture is actually ffa500, not ffa400. 图片中的主色实际上是ffa500,而不是ffa400。 There are in fact no occurances of ffa400 at all. 实际上根本没有ffa400的出现。

Also keep in mind that this is an anti-aliased bitmap graphic. 另请注意,这是一个消除锯齿的位图图形。 Visually we see only two colors, but to smoothen the transition from one color to another a set of "in-between-colors" are generated. 在视觉上我们只看到两种颜色,但是为了使从一种颜色到另一种颜色的过渡平滑,产生了一组“中间颜色”。 In other words, the sum of ffffff and ffa500 will not be exactly 100%. 换句话说,ffffff和ffa500的总和不会完全是100%。 (This is why the output from my function below will display a lot more colors than the two in question) (这就是为什么我下面的函数输出会显示比所讨论的两个颜色更多的颜色)


Solution 1 解决方案1

The basic idea of this solution is to get every color in the picture, store these in one common object where the occurrence of each color is counted. 该解决方案的基本思想是获取图片中的每种颜色,将这些颜色存储在一个共同的对象中,在该对象中计算每种颜色的出现。 When this object is returned, we can later count the occurrence of some specific color from the output. 返回此对象后,我们可以稍后从输出中计算某些特定颜色的出现次数。

This will return something like: 这将返回如下内容:

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

Now you can access the occurences of a given color this way: 现在,您可以通过以下方式访问给定颜色的出现:

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

And to display how many percent of the image a given color covers, do something like this (toFixed() is simply to limit the value to two decimals): 要显示给定颜色覆盖的图像的百分比,请执行以下操作(toFixed()只是将值限制为两位小数):

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

Working sample 工作样本

 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> 


Solution 2 解决方案2

(Consider this solution a draft. It might need some tweaking if to be used for a live website, but at least it should work!) (考虑这个解决方案草案。如果要用于实时网站,可能需要一些调整,但至少它应该工作!)

The idea here is to call extract_colors() with the colors we want to map as arguments. 这里的想法是使用我们想要映射为参数的颜色调用extract_colors() A common object is built to hold the output, and this is populated with one object for each color. 构建一个公共对象来保存输出,并为每种颜色填充一个对象。 The color objects consists of a counter and an rgb value. 颜色对象由计数器和rgb值组成。 Using the isNeighborColor() function, the rgb value is used to match not only the specific color but also any color close to it. 使用isNeighborColor()函数,rgb值不仅用于匹配特定颜色,还用于匹配与其相近的任何颜色。 Adjust the tolerance level as you want. 根据需要调整公差等级。 (Ideally this would be another argument I guess). (理想情况下,这将是我猜的另一个论点)。

Call the function specifying the image and what colors to count the occurrence of: 调用指定图像的函数以及计算出现次数的颜色:

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

This will output an object for each color: 这将输出每种颜色的对象:

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

What we want is to access the counter values: 我们想要的是访问counter值:

var white = imageData.ffffff.counter;

Working sample 工作样本

 // 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 = /^#?([af\\d]{2})([af\\d]{2})([af\\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> 

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM