-4

I need to know when my mouse pointer is over a certain image, and while in that image, it goes over a very specific colour. Is this possible and how can I do it?

Yu Hao
  • 119,891
  • 44
  • 235
  • 294
ProfK
  • 49,207
  • 121
  • 399
  • 775

3 Answers3

1

Just wrote this :

Basically, it does what @nepeo told you to do.
+ it cleans up the canvas and reset the original image after calculations.
+ I also added a customized cursor that you can disable.


Usage : Set your own callback function in colorPicker.callback, then attach the colorPicker.init function on the mouseover event of desired image.

N.B : You will have to host the images on your own server or to convert them to base64 in order to use this script.


var colorPicker = function() {};
//Settings

colorPicker.radius = 1, //square px

colorPicker.customCursor = true; 

colorPicker.cursorColor = '#FFF';

//the fomatting for color return : "hex" || "rgba" || "array"
// !! hex is upperCase and returns "transparent" if opacity == 0; rgba is 'rgba(r,g,b,a)';  array is '[r,g,b,a]'
colorPicker.colorFormat = "hex";

//What to do with that color? Edit this to whatever you want
colorPicker.callback = function (color) {
    var res = document.getElementById('ColorPickerResult');
    res.style.backgroundColor = color;
    res.style.color = invertHex(color);
    res.innerHTML = color;
    //Here I'm searching for some Hex color
    if (color == '#FC771F') {
        res.innerHTML = '!!! Found some orange !!!';
    }
}

//What to do when we leave the image (Additionally to reset the original image)
colorPicker.callbackOut = function(){
    var res = document.getElementById('ColorPickerResult');
    res.innerHTML = '';
  }

//END Settings

//This is optional
colorPicker.setCursor = function (canvas) {
    if (!colorPicker.setCursor) return;
    var can = document.createElement('canvas');
    can.width = colorPicker.radius;
    can.height = colorPicker.radius;
    var ctx = can.getContext('2d');
    ctx.strokeStyle = colorPicker.cursorColor;
    ctx.rect(0, 0, colorPicker.radius, colorPicker.radius);
    ctx.stroke();
    canvas.style.cursor = 'url("' + can.toDataURL("image/png") + '"),default';
}
colorPicker.Init = function () {
    var img = this;
    var can = document.createElement('canvas');
    can.id = 'ColorPicker';
    can.width = img.width;
    can.height = img.height;
    can.ori = img;
    can.oridisp = img.style.display;
    var ctx = can.getContext('2d');
    ctx.drawImage(img, 0, 0);
    can.addEventListener('mousemove', colorPicker.pick);
    can.addEventListener('mouseout', colorPicker.out);
    colorPicker.setCursor(can);
    img.parentNode.insertBefore(can, img);
    img.style.display = 'none';
};
colorPicker.pick = function (evt) {
    var rect = this.getBoundingClientRect();
    ctx = this.getContext('2d'),
    imageData = ctx.getImageData(evt.clientX - rect.left, evt.clientY - rect.top, colorPicker.radius, colorPicker.radius);
    data = imageData.data;
    r = 0, g = 0, b = 0, a = 0;
    for (i = 0; i < data.length; i++) {
        r += data[i];
        g += data[++i];
        b += data[++i];
        a += data[++i];
    }
    var p = data.length / 4;

    var mR = function (i) {
        return Math.round(i);
    };
    r = mR(r/p);
    g = mR(g/p);
    b = mR(b/p);
    a = (mR((a/p)/255 * 100)/100).toFixed(2);
  
    if(colorPicker.colorFormat === "hex"){
      var color = colorPicker.toHex(r,g,b,a);
      }
    else if(colorPicker.colorFormat === "rgba"){
      var color = 'rgba('+r+', '+g+', '+b+', '+a+')';
      }
    else if(colorPicker.colorFormat === "array"){
      var color = [r, g, b, a];
      }
 
    colorPicker.callback(color);
}
colorPicker.out = function () {
    this.ori.style.display = this.oridisp;
    this.parentNode.removeChild(this);
    colorPicker.callbackOut();
}

//toHex function originally posted by http://stackoverflow.com/users/10474/felipec in http://stackoverflow.com/a/19765382/3702797
colorPicker.toHex= function(r,g,b,a){
        if(a=="0.00") return "transparent";
        var rgb = b | (g << 8) | (r << 16);
        return '#' + (0x1000000 + rgb).toString(16).slice(1).toUpperCase()
  }
// END colorPicker \\

// Just an example of external function beeing called 
function invertHex(c){
  if(c==="transparent") return "#000000";
  c = c.replace('#', '');
  var reqHex = "#";
    for(var i=0;i<6;i++){
        reqHex = reqHex + (15-parseInt(c[i],16)).toString(16);
    }
    return reqHex;
}

//Attach the function to whatever images you want
document.getElementsByTagName('img')[0].addEventListener('mouseover', colorPicker.Init);
body {
    background: #175;
}
<img src="" />
<span id="ColorPickerResult"></span>
Kaiido
  • 123,334
  • 13
  • 219
  • 285
  • In your script, how do you give an indication of what colour to look for, like `var colorChecker = new ColorChecker({ searchedColor: "#FF0000", });` in @Jonas's script? – ProfK Oct 12 '14 at 07:09
  • It's in the `colorPicker.callback`. But the color is formatted in rgba(), in the example, I'm looking for `rgba(252, 119, 31, 1.00)` (#fc771f). you could add an rgba to hex function call in the callback too. – Kaiido Oct 12 '14 at 11:27
0

Use a canvas instead of the image, resize the canvas to the size of the image and draw the image into the canvas.

Then you can get the pixel location of the cursor using the canvas.onmousemove event. Canvas may have something in it's api for querying the colour at a location(can't remember at this moment) but otherwise you can use getImageData to get the rgba data for every pixel and do a look up during the event. Ta da!

nepeo
  • 509
  • 2
  • 9
0

Here I made an attempt. In this example the script searches for the color #FF0000, as passed in the options object. http://jsfiddle.net/2sw3pzs4/

var ColorChecker = function( options ) {
    var canvas, ctx, output;
    this.init = function() {
        canvas = document.getElementById('canvas');
        output = document.getElementById('output');
        ctx = canvas.getContext("2d");
        color = this.hexToRGB(options.searchedColor);
        this.drawOnCanvas();

        canvas.addEventListener('mousemove', function(e) {
            if(e.offsetX) {
                mouseX = e.offsetX;
                mouseY = e.offsetY;
            }
            else if(e.layerX) {
                mouseX = e.layerX;
                mouseY = e.layerY;
            }
            var c = ctx.getImageData(mouseX, mouseY, 1, 1).data;
            var foundColor = [c[0], c[1], c [2]];
            var is_identical = foundColor.join() === color.join();
            var message = is_identical == true ? 'red found' : 'not found';
            output.innerHTML = message;
        });
    }

    this.drawOnCanvas = function() {
        ctx.fillStyle = "#FF0000";
        ctx.fillRect(0,0, 20, 20);
    }

    this.hexToRGB = function( hex ) {
        var r = parseInt((this.cutHex(hex)).substring(0,2),16);
        var g = parseInt((this.cutHex(hex)).substring(2,4),16);
        var b = parseInt((this.cutHex(hex)).substring(4,6),16);

        return [r,g,b];
    }

    this.cutHex = function( hex ) {
         return hex.replace("#",'');
    }
    this.init();
}

var colorChecker = new ColorChecker({
    searchedColor: "#FF0000",
});

If you want to use an image just use putImageData() to draw the image on the canvas. You can use the function drawOnCanvas in my code to draw whatever you want inside the canvas, or resize it or other stuff.

Jonas Grumann
  • 10,438
  • 2
  • 22
  • 40
  • It's working great, thanks, but outside Visual Studio, my canvas +image just appear as a small red square the top left of the screen. In Visual Studio it works a charm. – ProfK Oct 12 '14 at 15:40
  • Sorry I don't understand what you're saying. Could you be more specific? – Jonas Grumann Oct 13 '14 at 06:44
  • Sorry, I had another problem that was drawing a small red square on my canvas after I drew my image. It wasn't my code and I was unaware of it, so I thought it was a problem with my code. – ProfK Oct 13 '14 at 10:50
  • Oh yes the red square is just me drawing a test square on the canvas, just remove the ctx.fillStyle and ctx.fillRect parts in my code – Jonas Grumann Oct 13 '14 at 12:01