Inset-shadow on HTML5 canvas image

后端 未结 5 1887
遥遥无期
遥遥无期 2021-01-03 15:19

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 creat

相关标签:
5条回答
  • 2021-01-03 15:37

    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";
    

    0 讨论(0)
  • 2021-01-03 15:39

    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.

    0 讨论(0)
  • 2021-01-03 15:40

    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)

    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/

    0 讨论(0)
  • 2021-01-03 15:48
    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

    0 讨论(0)
  • 2021-01-03 15:55

    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>

    0 讨论(0)
提交回复
热议问题