pourmesomecode
pourmesomecode

Reputation: 4318

Inset-shadow on HTML5 canvas image

I've seen this question before but the answers given are for canvas images that have been drawn on via path however, i'm drawing an image.

Is it possible to create an inset-shadow?

context.shadowOffsetX = 0;
context.shadowOffsetY = 0;
context.shadowBlur = 10;
context.shadowColor = 'rgba(30,30,30, 0.4)';

var imgOne = new Image();
imgOne.onload = function() {
    context.drawImage(imgOne, 0, 0);
};
imgOne.src = "./public/circle.png";

So I draw the circle picture on. I've now at the moment got a slight shadow on the outside of the circle, how can I get this inset instead of offset?

Upvotes: 5

Views: 12776

Answers (5)

user11403032
user11403032

Reputation:

const width = 100 * devicePixelRatio;
const height = 100 * devicePixelRatio;

// original canvas
const c = document.getElementById('canvas');
c.width = 300 * devicePixelRatio;
c.height = 300 * devicePixelRatio;
c.style.width = '300px';
c.style.height = '300px';
const cctx = c.getContext('2d');
cctx.fillStyle = 'rgb(20,205,75)';
cctx.arc(150 * devicePixelRatio, 150 * devicePixelRatio, 50 * devicePixelRatio, 0, Math.PI * 2);
cctx.fill();

// temporary canvas
const canvas = document.createElement('canvas');
canvas.width = width;
canvas.height = height;
canvas.style.width = `${width / devicePixelRatio}px`;
canvas.style.height = `${height / devicePixelRatio}px`;
document.body.appendChild(canvas);
var ctx = canvas.getContext('2d');

// original object on temporary canvas
ctx.arc(50 * devicePixelRatio, 50 * devicePixelRatio, 50 * devicePixelRatio, 0, Math.PI * 2);
ctx.fill();

// shadow cutting
ctx.globalCompositeOperation = 'xor';
ctx.arc(50 * devicePixelRatio, 50 * devicePixelRatio, 50 * devicePixelRatio, 0, Math.PI * 2);
ctx.fill();

// shadow props
ctx.shadowBlur = 50;
ctx.shadowOffsetX = 0;
ctx.shadowOffsetY = -25;
ctx.shadowColor = '#000';
ctx.arc(50 * devicePixelRatio, 50 * devicePixelRatio, 50 * devicePixelRatio, 0, Math.PI * 2);
ctx.fill();

// shadow color
ctx.globalCompositeOperation = 'source-in';
ctx.fillStyle = 'blue';
ctx.fillRect(0, 0, canvas.width, canvas.height);

// object cutting
ctx.globalCompositeOperation = 'destination-in';
ctx.arc(50 * devicePixelRatio, 50 * devicePixelRatio, 50 * devicePixelRatio, 0, Math.PI * 2);
ctx.fill();

// shadow opacity
cctx.globalAlpha = .4

// inserting shadow into original canvas
cctx.drawImage(canvas, 200, 200);

Colored shadow /w opacity

Upvotes: 0

Tom Lecoz
Tom Lecoz

Reputation: 85

Inspired by markE's answer , I made my own version based on a png instead of vector-graphics.

Additionnaly, I made possible to choose the true alpha of the shadow (because the default shadow strength is a way too soft in my opinion)

enter image description here

var img = document.getElementById("myImage");
img.onload = function(){
   createInnerShadow(this,5,1);
}

function createInnerShadow(img,distance,alpha){
  
   //the size of the shadow depends on the size of the target,
   //then I will create extra "walls" around the picture to be sure
   //tbat the shadow will be correctly filled (with the same intensity everywhere)
   //(it's not obvious with this image, but it is when there is no space at all between the image and its border)
   var offset = 50 + distance;
   var hole = document.createElement("canvas");
   var holeContext = hole.getContext("2d");
   hole.width = img.width + offset*2;
   hole.height = img.height + offset*2;
   
    //first, I draw a big black rect
   holeContext.fillStyle = "#000000";
   holeContext.fillRect(0,0,hole.width,hole.height);
   
   //then I use the image to make an hole in it
   holeContext.globalCompositeOperation = "destination-out";
   holeContext.drawImage(img,offset,offset);
   
   //I create a new canvas that will contains the shadow of the hole only
   var shadow = document.createElement("canvas");
   var shadowContext = shadow.getContext("2d");
   shadow.width = img.width;
   shadow.height = img.height;
   shadowContext.filter = "drop-shadow(0px 0px "+distance+"px #000000 ) ";
   shadowContext.drawImage(hole,-offset,-offset);
   shadowContext.globalCompositeOperation = "destination-out";
   shadowContext.drawImage(hole,-offset,-offset);
  
   //now, because the default-shadow filter is really to soft, I normalize the shadow 
   //then I will be sure that the alpha-gradient of the shadow will start at "alpha" and end at 0
   normalizeAlphaShadow(shadow,alpha);
   
   
   //Finally, I create another canvas that will contain the image and the shadow over it
   var result = document.createElement("canvas");
   result.width = img.width;
   result.height = img.height;
   var context = result.getContext("2d");
   context.drawImage(img,0,0)
   context.drawImage(shadow,0,0);
   
   //and that's it !
   document.body.appendChild(result);
   
}

function normalizeAlphaShadow(canvas,alpha){
   var imageData = canvas.getContext("2d").getImageData(0,0,canvas.width,canvas.height);
   var pixelData = imageData.data;
   var i,len = pixelData.length;
   var max = 0;
   for(i=3;i<len;i+=4) if(pixelData[i]>max) max = pixelData[i];
   
   max = (255/max) * alpha;
   for(i=3;i<len;i+=4) pixelData[i] *= max;
   
   canvas.getContext("2d").putImageData(imageData,0,0)
   
}
<html>
<body>
<img id="myImage" src="" />
</body>
</html>

the jsfiddle is here : https://jsfiddle.net/jrekw5og/141/

Upvotes: 4

Pat Lillis
Pat Lillis

Reputation: 1180

Inspired by K3N's answer, I've created Inset.js for this exact situation!

Inset.js

Only requires setting ctx.shadowInset = true;

For example: http://codepen.io/patlillis/pen/ryoWey

var ctx = canvas.getContext('2d');
var img = new Image;
img.onload = function() {
  ctx.shadowInset = true;
  ctx.shadowBlur = 25;
  ctx.shadowColor = "#000";
  ctx.drawImage(this, 0, 0);
}
img.src = "http://i.imgur.com/Qrfga2b.png";

Inset circle

Upvotes: 2

markE
markE

Reputation: 105015

Canvas will shadow where an image changes from opaque to transparent so, as K3N shows in his correct answer, you can turn the image inside out (opaque becomes transparent & visa-versa) so the shadows are drawn inside the circle.

If you know your circle's centerpoint and radius, you can use a stroked-path to create an inset circle shadow. Here's an example:

var canvas=document.getElementById("canvas");
var context=canvas.getContext("2d");
var cw=canvas.width;
var ch=canvas.height;

context.beginPath();
context.arc(cw/2,ch/2,75,0,Math.PI*2);
context.fillStyle='lightcyan';
context.fill();

context.globalCompositeOperation='source-atop';

context.shadowOffsetX = 500;
context.shadowOffsetY = 0;
context.shadowBlur = 15;
context.shadowColor = 'rgba(30,30,30,1)';

context.beginPath();
context.arc(cw/2-500,ch/2,75,0,Math.PI*2);
context.stroke();
context.stroke();
context.stroke();

context.globalCompositeOperation='source-over';
<canvas id="canvas" width=300 height=300></canvas>

If your path is irregular or hard to define mathematically, you can also use edge-path detection algorithms. One common edge-path algorithm is Marching Squares. Stackoverflow's K3N has coded a nice Marching Squares algorithm.

Upvotes: 6

user1693593
user1693593

Reputation:

Composition chain

Use a series of composite + draw operation to obtain inset shadow.

Note: the solution require exclusive access to the canvas element when created so either do this on an off-screen canvas and draw back to main, or if possible, plan secondary graphics to be drawn after this has been generated.

The needed steps:

  • Draw in original image
  • Invert alpha channel filling the canvas with a solid using xor composition
  • Define shadow and draw itself back in
  • Deactivate shadow and draw in original image (destination-atop)

Result

var ctx = c.getContext("2d"), img = new Image;
img.onload = function() {

  // draw in image to main canvas
  ctx.drawImage(this, 0, 0);

  // invert alpha channel
  ctx.globalCompositeOperation = "xor";
  ctx.fillRect(0, 0, c.width, c.height);

  // draw itself again using drop-shadow filter
  ctx.shadowBlur = 7*2;  // use double of what is in CSS filter (Chrome x4)
  ctx.shadowOffsetX = ctx.shadowOffsetY = 5;
  ctx.shadowColor = "#000";
  ctx.drawImage(c, 0, 0);

  // draw original image with background mixed on top
  ctx.globalCompositeOperation = "destination-atop";
  ctx.shadowColor = "transparent";                  // remove shadow !
  ctx.drawImage(this, 0, 0);
}
img.src = "http://i.imgur.com/Qrfga2b.png";
<canvas id=c height=300></canvas>

Upvotes: 8

Related Questions