2017-07-16 76 views
0

我想检测下图中图标的宽度(右手指) 。如何使用javascript检测图像内图标的宽度?

enter image description here

我假设可能的解决方案可能涉及color removal illustrated in this approach,其可以被用于去除在这种情况下,lightblue背景颜色。但是我不确定去除颜色后的下一步。

+0

我想我发现[A溶液](https://stackoverflow.com/a/12178531/1091014)。我将尝试使用该解决方案来实现它,但欢迎任何更好的解决方案。 –

回答

1

我想出了一些东西!

我是这样做的:

  1. 使用<img>标记的图像添加到文档(无论是通过普通的OLE-HTML或JS)
  2. 创建2D canvas元素,使它的尺寸相同大小的IMG
  3. 绘制图像投影到使用context.drawImage
  4. 使用context.getImageData获得的像素的颜色在画布
  5. TRANSFO画布RM此原始数的阵列,以十六进制字符串颜色
  6. 组的颜色由图像
  7. 抓取的图像的第一颜色作为colorToFilter
  8. 的行到行创建minXminYmaxX,和maxY变量保持不属于colorToFilter
  9. 迭代穿过pixelMatrix每个像素检查是否当前rowcolumn或比色小于或大于不属于colorToFilter
  10. 像素的轨道
  11. maxX - minXmaxY - minY的差值,这就是您的尺寸!

仅供参考:由于交叉来源问题,我使用图像src的数据uri。

让我知道你是否需要更多的解释。这个解决方案可能不是最有效的,但它的工作原理!

/** 
 
* converts a number from 0-255 to a hex number padded with a zero if necessary 
 
* @param {number} number 
 
*/ 
 
function toHex(number) { 
 
    const numberAsString = Number(number).toString(16); 
 
    return numberAsString.length === 1 ? '0' + numberAsString : numberAsString; 
 
} 
 

 
/** 
 
* Grabs the pixel colors from an image and stores it in a matrix 
 
* @param {HTMLImageElement} img 
 
*/ 
 
function convertImageToPixelData(img) { 
 
    // credit for getting the color of pixel comes from here: 
 
    // https://stackoverflow.com/questions/8751020/how-to-get-a-pixels-x-y-coordinate-color-from-an-image 
 
    const canvas = document.createElement('canvas'); 
 
    const imgWidth = img.clientWidth; 
 
    const imgHeight = img.clientHeight; 
 
    canvas.width = imgWidth; 
 
    canvas.height = imgHeight; 
 
    const context = canvas.getContext('2d'); 
 
    context.drawImage(img, 0, 0, imgWidth, imgHeight); 
 

 
    const rawPixelData = context.getImageData(0, 0, imgWidth, imgHeight).data; 
 

 
    // convert to hex string 
 
    const pixelColors = rawPixelData.reduce((groups, pixelColor, index) => { 
 
    if (index % 4 === 3) { // skip the alpha channel 
 
     return groups; 
 
    } 
 
    if (groups[groups.length - 1].length === 6) { 
 
     groups.push(''); 
 
    } 
 
    groups[groups.length - 1] += toHex(pixelColor); 
 
    return groups; 
 
    }, ['']); 
 

 
    // convert to matrix 
 
    /** 
 
    * @type {string[][]} 
 
    */ 
 
    const pixelMatrix = pixelColors.reduce((matrix, pixel) => { 
 
    const currentRow = matrix[matrix.length - 1]; 
 
    if (currentRow.length < imgWidth) { 
 
     currentRow.push(pixel); 
 
    } else { 
 
     matrix.push([pixel]); 
 
    } 
 
    return matrix; 
 
    }, [[]]); 
 
    return pixelMatrix; 
 
} 
 

 
/** 
 
* finds the dimensions of an icon inside a single colored image 
 
* @param {HTMLImageElement} img 
 
*/ 
 
function findBox(img) { 
 
    const pixelMatrix = convertImageToPixelData(img); 
 
    const colorToFilter = pixelMatrix[0][0]; 
 

 
    const imgWidth = img.clientWidth; 
 
    const imgHeight = img.clientHeight; 
 

 
    let maxX = 0; 
 
    let minX = imgWidth; 
 
    let maxY = 0; 
 
    let minY = imgHeight; 
 

 
    // for each pixel calculate if the index of the pixel is greater than the current max or min 
 
    for (let row = 0; row < imgHeight; row += 1) { 
 
    for (let column = 0; column < imgWidth; column += 1) { 
 
     const pixel = pixelMatrix[row][column]; 
 
     if (pixel !== colorToFilter) { 
 
     if (column < minX) { 
 
      minX = column; 
 
     } 
 
     if (column > maxX) { 
 
      maxX = column; 
 
     } 
 
     if (row < minY) { 
 
      minY = row; 
 
     } 
 
     if (row > maxY) { 
 
      maxY = row; 
 
     } 
 
     } 
 
    } 
 
    } 
 

 
    const width = maxX - minX + 1; 
 
    const height = maxY - minY + 1; 
 

 
    return { width, height }; 
 
} 
 

 
const img = document.querySelector('#my-image'); 
 

 
const box = findBox(img); 
 
console.log(box);
<img id="my-image" src="">

+0

看起来很棒!你的方法更清洁,更有组织。非常感谢! –

相关问题