2014-06-30 3 views
1

Я создаю дым, используя излучатель частиц в холсте. Все работает отлично, но для дизайна требуется добавить маску к холсту, чтобы мы могли видеть фон страницы. Когда я добавляю область отсечения, я получаю крайне низкую производительность, особенно в 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(); 

ответ

2

Да, context.clip стоит дорого.

Возможно, используйте маску наложения вместо обтравочной маски.

Поскольку вы просто blitting маску наложения на холст, это намного быстрее, чем вычисление, какие пиксели находятся в обтравочном контуре.

Шаг # 1: рисовать ваш дым

enter image description here!

Шаг # 2: нарисовать маску наложения, которая скрывает ненужную часть дыма

Слева: наложения маски.
Справа: дым с наложенной маской.

enter image description hereenter 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> 
+1

+1 также всего головы, если вы не знаете, Imgur поддерживает CORS хех его намного лучше быть в состоянии загрузить, а затем используйте изображение из imgur, а затем работайте с dataURL для демонстрационных/ответных скриптов. – Loktar

+1

@Loktar Спасибо. :-) Да, imgur отлично подходит для доставки изображений, совместимых с CORS. Я сам использую dropbox.com для доставки изображений, совместимых с CORS. Я оставил «дым» dataURL из кода опроса в демо - нет особой причины, кроме лени (!). – markE

+1

Спасибо, оверлейная маска намного быстрее, вот моя обновленная ручка. http://codepen.io/fatlinesofcode/pen/GLFqf – fatlinesofcode

Смежные вопросы