15

I have an image like this:

enter image description here

I want it so that when I click the transparent parts, the click should go through to the underlying element, but when I click a non-transparent part, this image element should get the click.

Is this even possible?

Roko C. Buljan
  • 196,159
  • 39
  • 305
  • 313
Nick
  • 153
  • 1
  • 5

2 Answers2

18
  1. Use a temporary canvas element.
  2. Get the overlay image mousedown coordinates.
  3. Populate your canvas with that image data.
  4. Read from canvas the 1px image data (from the same coordinates).
    If the retrieved Alpha channel data is 0 means it's transparent.
  5. If pixel is transparent set pointerEvents to "none" and retrieve the underneath element using Document.elementFromPoint
    1 Trigger "click" event on the underneath element
  6. Reset pointerEvents back to "auto" for all overlays

var ctx = document.createElement("canvas").getContext("2d");

$('#logo').on("mousedown", function(event) {
  
  // Get click coordinates
  var x = event.pageX - this.offsetLeft,
      y = event.pageY - this.offsetTop,
      w = ctx.canvas.width = this.width,
      h = ctx.canvas.height = this.height,
      alpha;

  // Draw image to canvas
  // and read Alpha channel value
  ctx.drawImage(this, 0, 0, w, h);
  alpha = ctx.getImageData(x, y, 1, 1).data[3]; // [0]R [1]G [2]B [3]A

  // If pixel is transparent,
  // retrieve the element underneath and trigger it's click event
  if( alpha===0 ) {
    this.style.pointerEvents = "none";
    $(document.elementFromPoint(event.clientX, event.clientY)).trigger("click");
    this.style.pointerEvents = "auto";
  } else {
    console.log("LOGO clicked!");
  }
});


$("#green").on("click", function(){
  console.log("Green image clicked!");
});
img{border:1px solid #000;}
img + img{position:absolute;top:0; left:0;}
<script src="https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js"></script>
<img id="green" src="//placehold.it/200x140/cf5">
<img id="logo"  src="
">
Roko C. Buljan
  • 196,159
  • 39
  • 305
  • 313
  • 2
    Wow, this is great. I wish I could accept the other answer also, but this is more elegant and easy to use. Thanks. – Nick Jul 20 '16 at 18:58
  • In the case of a transparent pixel, where does the click event propagate to? I'm trying to catch it from an underlying image but it doesn't fire. – Nick Jul 20 '16 at 19:26
  • 2
    If your parent element has a click listener than clicking on the transparent part will propagate. Else... the click will be stopped from propagation. – Roko C. Buljan Jul 20 '16 at 19:30
  • I see, so in my case it goes to the parent div. How can I capture that click on an underlying image instead of the parent div? – Nick Jul 20 '16 at 19:33
  • Get the ElementFromPoint – Roko C. Buljan Jul 20 '16 at 19:36
  • im on mobile currently. Will expand my answer in 30min or so. – Roko C. Buljan Jul 20 '16 at 19:37
  • Then perhaps I'm missing something. Because the on() handler is assigned to all images of the same class in this div, but the transparent click directly goes to the div, not the underlying image. – Nick Jul 20 '16 at 19:44
  • 2
    @Nick As I've said, use `elementFromPoint`. The trick was to quickly hide the logo, get the element from point (the element underneath), trigger it's click event, and than finally show the logo. The hide/show is not visible to the eye, but allows to get the right element from coordinates: https://developer.mozilla.org/en-US/docs/Web/API/Document/elementFromPoint. Anyways, updated my answer if this comment seems a bit fuzzy :) – Roko C. Buljan Jul 20 '16 at 22:22
  • 2
    Excellent answer. Thank you. – Nick Jul 20 '16 at 23:05
6

For your scenario, would it be sufficient to capture click events on the non-transparent area of the image?

This can be done using a map and some JavaScript:

<img src="https://i.stack.imgur.com/Ab6aW.jpg" alt="" usemap="#Map" />
<map name="Map" id="Map">
<area
  href="javascript:alert('You clicked on the image!')"
  shape="poly" 
  coords="193,148,50,193,254,292,461,191,302,141,332,99,285,27,220,16,170,89"
/>
</map>

(The coordinates were derived using this site.)

Richard Ev
  • 52,939
  • 59
  • 191
  • 278
  • Yes, this will do. Would this be the preferred way to do it, or would it be easier to get the alpha value of the clicked pixel and check for transparency? Because with this I need to generate maps for each image. – Nick Jul 20 '16 at 18:16
  • 1
    If you can figure out how to get the alpha value of the clicked pixel, go for it! Otherwise, it may take less time to just manually create `map`s for each image (depending on the number of images, and the complexity of their transparencies). It only took me a few seconds to create the map in my answer, but it is pretty rough around the edges. – Richard Ev Jul 20 '16 at 18:19
  • 1
    Tracing the image non transparent area is a hell of a job. – Roko C. Buljan Jul 20 '16 at 18:37