使用剪辑区域减慢画布性能

时间:2014-06-30 12:20:04

标签: canvas html5-canvas

我在Canvas中使用粒子发射器创建了一些烟雾。一切都很好但设计要求我在画布上添加一个遮罩,这样我们就可以看到页面背景了。当我添加剪辑区域时,我的性能极差,尤其是在Safari中。在使用剪辑区域时,我可以做些什么来优化性能?

这是没有面具的烟雾

http://codepen.io/fatlinesofcode/pen/AcLim

并使用面具

http://codepen.io/fatlinesofcode/pen/GLFqf

我每次都在渲染循环中绘制一个矢量区域。

ctx.beginPath();
ctx.moveTo(0, 0);
ctx.lineTo(960, 00);
ctx.lineTo(960, 500);
ctx.lineTo(960 / 2, 550);
ctx.lineTo(0, 500);
ctx.closePath();
ctx.clip();
ctx.restore();

1 个答案:

答案 0 :(得分:2)

是的,context.clip很贵。

也许使用覆盖蒙版而不是剪贴蒙版。

由于您只是将叠加蒙版blitting到画布上,因此它比计算剪切路径中的像素要快得多。

第1步:抽烟

enter image description here

步骤2:绘制隐藏不需要的部分烟雾的覆盖遮罩

左:覆盖蒙版。
右:带覆盖面罩的烟雾。

enter image description here enter image description here

示例代码和演示:http://jsfiddle.net/m1erickson/uHg4E/

<!doctype html>
<html>
<head>
<link rel="stylesheet" type="text/css" media="all" href="css/reset.css" /> <!-- reset css -->
<script type="text/javascript" src="http://code.jquery.com/jquery.min.js"></script>
<style>
    body{ background-color: ivory; }
    canvas{border:1px solid red;}
</style>
<script>
$(function(){

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

    var maskCanvas=document.createElement("canvas");
    var maskCtx=maskCanvas.getContext('2d');

    var img=new Image();
    img.onload=start;
    img.src="https://dl.dropboxusercontent.com/u/139992952/multple/mountain.png";
    function start(){

        makeOverlayMask();

        var smokeImage = new Image();
        smokeImage.src = "";
        smokeImage.onload = function () {

            ctx.clearRect(0,0,canvas.width,canvas.height);
            for(var y=0;y<canvas.height;y+=smokeImage.height){
                ctx.drawImage(smokeImage,canvas.width/2-15,y);
            }
            ctx.drawImage(maskCanvas,0,0);

        }
    }

    function makeOverlayMask(){
        maskCanvas.width=img.width;
        maskCanvas.height=img.height+166;
        //
        maskCtx.drawImage(img,0,166);
        //
        maskCtx.save();
        maskCtx.globalCompositeOperation="destination-out";    
        maskCtx.beginPath();
        maskCtx.moveTo(0, 0);
        maskCtx.lineTo(320, 00);
        maskCtx.lineTo(320, 166);
        maskCtx.lineTo(320 / 2, 183);
        maskCtx.lineTo(0, 166);
        maskCtx.closePath();
        maskCtx.fill();
        maskCtx.restore();
    }

}); // end $(function(){});
</script>
</head>
<body>
    <canvas id="canvas" width=320 height=300></canvas>
</body>
</html>