<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>Canvas烟花</title>
<meta name="description" content="">
<meta name="keywords" content="">
<link href="" rel="stylesheet">
<style type="text/css">
body{
padding: 0;
margin: 0;
background: #000;
}
</style>
</head>
<body>
<canvas id="canvas">Canvas is not supported in your browser.</canvas>
<script type="text/javascript">
window.requestAnimationFrame = (function(){
return window.requestAnimationFrame ||
window.webkitRequestAnimationFrame ||
window.mozRequestAnimationFrame ||
function(callback){
window.setTimeout(callback,1000/60);
}
})();
var canvas = document.getElementById("canvas"),
ctx = canvas.getContext("2d"),
cw = window.innerWidth,
ch = window.innerHeight,
fireworks = [],
particles = [],
hue = 120,
timerTotal = 80,
timerTick = 0;
canvas.width = cw;
canvas.height = ch;
function random(min,max){
return Math.random()*(max-min)+min;
}
function caculateDistance(sx,sy,tx,ty){
var xDistance = sx-tx,
yDistance = sy-ty;
return Math.sqrt(Math.pow(xDistance,2)+Math.pow(yDistance,2));
}
function Firework(sx,sy,tx,ty){
this.x = sx;
this.y = sy;
this.sx = sx;
this.sy = sy;
this.tx= tx;
this.ty = ty;
this.distanceToTarget = caculateDistance(sx,sy,tx,ty);
this.distanceTraveled = 0;
this.coordinates = [];
this.coordinatesCount = 3;
while(this.coordinatesCount--){
this.coordinates.push([this.x,this.y]);
}
this.angle = Math.atan2(ty-sy,tx-sx);
this.speed = 2;
this.acceleration = 1.05
this.brightness = random(50,70);
this.targetRadius = 1;
}
Firework.prototype.update = function(index){
this.coordinates.pop();
this.coordinates.unshift([this.x,this.y]);
if(this.targetRadius<8) {
this.targetRadius += 0.3;
} else {
this.targetRadius = 1;
}
this.speed *= this.acceleration;
var vx = Math.cos(this.angle)*this.speed,
vy = Math.sin(this.angle)*this.speed;
this.distanceTraveled = caculateDistance(this.sx,this.sy,this.x+vx,this.y+vy);
if(this.distanceTraveled>=this.distanceToTarget){
createParticles(this.tx,this.ty);
fireworks.splice(index,1);
}else{
this.x += vx;
this.y += vy;
}
}
Firework.prototype.draw = function(){
ctx.beginPath();
ctx.moveTo(this.coordinates[this.coordinates.length-1][0],this.coordinates[this.coordinates.length-1][1]);
ctx.lineTo(this.x,this.y);
ctx.strokeStyle = 'hsl('+hue+',100%,'+this.brightness+'%)';
ctx.stroke();
ctx.beginPath();
ctx.arc(this.tx,this.ty,this.targetRadius,0,Math.PI*2);
ctx.stroke();
}
function Particle(x,y){
this.x = x;
this.y = y;
this.coordinates = [];
this.coordinatesCount = 5;
while(this.coordinatesCount--){
this.coordinates.push([this.x,this.y]);
}
this.angle = random(0,2*Math.PI);
this.speed = random(1,10);
this.friction = 0.95
this.gravity = 1;
this.hue = random(hue-20,hue+20);
this.brightness = random(50,80);
this.alpha = 1;
this.decay = random(0.015,0.03);
}
Particle.prototype.update = function(index){
this.coordinates.pop();
this.coordinates.unshift([this.x,this.y]);
this.speed *= this.friction;
this.x += Math.cos(this.angle) * this.speed;
this.y += Math.sin(this.angle) * this.speed + this.gravity;
this.alpha -= this.decay;
if(this.alpha<=this.decay){
particles.splice(index,1);
}
}
Particle.prototype.draw = function(){
ctx.beginPath();
ctx.moveTo(this.coordinates[this.coordinates.length-1][0],this.coordinates[this.coordinates.length-1][1]);
ctx.lineTo(this.x,this.y);
ctx.strokeStyle = 'hsla('+this.hue+',100%,'+this.brightness+'%,'+this.alpha+')';
ctx.stroke();
}
function createParticles(x,y){
var particleCount = 30;
while(particleCount--) {
particles.push(new Particle(x,y));
}
}
function loop(){
requestAnimationFrame(loop);
hue += 0.5;
ctx.globalCompositeOperation = 'destination-out';
ctx.fillStyle = 'rgba(0, 0, 0, 0.5)';
ctx.fillRect( 0, 0, cw, ch );
ctx.globalCompositeOperation = 'lighter';
var i = fireworks.length;
while(i--) {
fireworks[i].draw();
fireworks[i].update(i);
}
var i = particles.length;
while(i--) {
particles[i].draw();
particles[i].update(i);
}
if(timerTick >= timerTotal) {
fireworks.push(new Firework(cw/2,ch,random(0,cw),random(0,ch/2)));
timerTick = 0;
} else {
timerTick++;
}
}
window.onload = loop;
</script>
</body>
</html>
这个效果之前是在runjs上看到的,看了一段时间才看懂,根据自己的理解,较少了一些效果。
window.requestAnimationFrame = (function(){
return window.requestAnimationFrame ||
window.webkitRequestAnimationFrame ||
window.mozRequestAnimationFrame ||
function(callback){
window.setTimeout(callback,1000/60);//每秒60帧
}
})();var canvas = document.getElementById(“canvas”),
ctx = canvas.getContext(“2d”),
cw = window.innerWidth,
ch = window.innerHeight,
fireworks = [], //烟花数组
particles = [], //烟花爆炸屑数字
hue = 120, //初始色调 0/360 红色 120 绿色 240 蓝色
timerTotal = 80, //每隔80下释放一次烟花
timerTick = 0;
//设置canvas画布的宽高
canvas.width = cw;
canvas.height = ch;//生成min到max之间的随机数
function random(min,max){
return Math.random()*(max-min)+min;
}//计算点(sx,sy)到点(tx,ty)之间的距离
function caculateDistance(sx,sy,tx,ty){
var xDistance = sx-tx,
yDistance = sy-ty;
return Math.sqrt(Math.pow(xDistance,2)+Math.pow(yDistance,2));
}//烟花对象 (sx,sy)初始位置 (tx,ty)目标位置
function Firework(sx,sy,tx,ty){
this.x = sx; //实时运行坐标x
this.y = sy; //实时运行坐标y
this.sx = sx; //初始位置坐标x
this.sy = sy; //初始位置坐标Y
this.tx= tx; //目标位置坐标x
this.ty = ty; //目标位置坐标Y
this.distanceToTarget = caculateDistance(sx,sy,tx,ty); //计算初始位置到目标位置之间的距离
this.distanceTraveled = 0; //已运行距离
this.coordinates = []; //我理解是一个辅助变量 用于生成实时运动轨迹
this.coordinatesCount = 3;
while(this.coordinatesCount–){
this.coordinates.push([this.x,this.y]);
}
this.angle = Math.atan2(ty-sy,tx-sx); //初始位置和目标位置之间的角度
this.speed = 2; //初始速度
this.acceleration = 1.05 //加速度
this.brightness = random(50,70); //明度
this.targetRadius = 1; //目标位置标示圆圈的初始半径
}//更新烟花的位置
Firework.prototype.update = function(index){
this.coordinates.pop();
this.coordinates.unshift([this.x,this.y]);
//上面是一个技巧吧 先将数组最后一个移除,然后将当前烟花位置插入到第一个,那数组最后一个元素和更新之后的坐标之间就形成了一条轨迹
//让目标标示处圆圈动起来
if(this.targetRadius<8) {
this.targetRadius += 0.3;
} else {
this.targetRadius = 1;
}
this.speed *= this.acceleration; //根据加速度变换速度
var vx = Math.cos(this.angle)*this.speed,//计算水瓶方向速度
vy = Math.sin(this.angle)*this.speed; //计算垂直方向速度
this.distanceTraveled = caculateDistance(this.sx,this.sy,this.x+vx,this.y+vy); //重新计算烟花已运行距离
//如果烟花运行距离大于或等于初始位置到目标位置之间的距离,生成新烟花并移除当前烟花,否则更新烟花位置
if(this.distanceTraveled>=this.distanceToTarget){
createParticles(this.tx,this.ty);
fireworks.splice(index,1);
}else{
this.x += vx;
this.y += vy;
}
}Firework.prototype.draw = function(){
//画出烟花运行轨迹
ctx.beginPath();
ctx.moveTo(this.coordinates[this.coordinates.length-1][0],this.coordinates[this.coordinates.length-1][1]);
ctx.lineTo(this.x,this.y);
ctx.strokeStyle = ‘hsl(‘+hue+’,100%,’+this.brightness+’%)’;
ctx.stroke();//画出目标位置标示小圆圈
ctx.beginPath();
ctx.arc(this.tx,this.ty,this.targetRadius,0,Math.PI*2);
ctx.stroke();}
//烟花爆炸屑对象
function Particle(x,y){
this.x = x;
this.y = y;
this.coordinates = [];
this.coordinatesCount = 5;
while(this.coordinatesCount–){
this.coordinates.push([this.x,this.y]);
}
this.angle = random(0,2*Math.PI); //生成任意方向的碎屑
this.speed = random(1,10); //随机速度
this.friction = 0.95 //摩擦力
this.gravity = 1; //重力
this.hue = random(hue-20,hue+20); //生成与烟花色彩相近的碎屑
this.brightness = random(50,80); //随机明度
this.alpha = 1; //初始透明度
this.decay = random(0.015,0.03); //碎屑消失时间
}Particle.prototype.update = function(index){
this.coordinates.pop();
this.coordinates.unshift([this.x,this.y]);
//上面是一个技巧吧 先将数组最后一个移除,然后将当前烟花位置插入到第一个,那数组最后一个元素和更新之后的坐标之间就形成了一条轨迹
this.speed *= this.friction;
this.x += Math.cos(this.angle) * this.speed;
this.y += Math.sin(this.angle) * this.speed + this.gravity;
this.alpha -= this.decay;
if(this.alpha<=this.decay){
particles.splice(index,1);
}
}Particle.prototype.draw = function(){
ctx.beginPath();
ctx.moveTo(this.coordinates[this.coordinates.length-1][0],this.coordinates[this.coordinates.length-1][1]);
ctx.lineTo(this.x,this.y);
ctx.strokeStyle = ‘hsla(‘+this.hue+’,100%,’+this.brightness+’%,’+this.alpha+’)’;
ctx.stroke();}
function createParticles(x,y){
//生成30个烟花碎屑
var particleCount = 30;
while(particleCount–) {
particles.push(new Particle(x,y));
}
}function loop(){
//流畅动画过程
requestAnimationFrame(loop);
hue += 0.5;
ctx.globalCompositeOperation = ‘destination-out’;
ctx.fillStyle = ‘rgba(0, 0, 0, 0.5)’;
ctx.fillRect( 0, 0, cw, ch );
ctx.globalCompositeOperation = ‘lighter’;
var i = fireworks.length;
while(i–) {
fireworks[i].draw();
fireworks[i].update(i);
}
var i = particles.length;
while(i–) {
particles[i].draw();
particles[i].update(i);
}
if(timerTick >= timerTotal) {
fireworks.push(new Firework(cw/2,ch,random(0,cw),random(0,ch/2)));
timerTick = 0;
} else {
timerTick++;
}
}
<!doctype html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="Generator" content="EditPlusR">
<meta name="Author" content="">
<meta name="Keywords" content="">
<meta name="Description" content="">
<title>Document</title>
<style>
body{margin:0;padding:0;overflow: hidden;}
.city{width:100%;position:fixed;bottom: 0px;z-index: 100;}
.city img{width: 100%;}
</style>
</head>
<body>
<canvas id='cas' style="background-color:rgba(0,5,24,1)">canvas</canvas>
<div class="city"><img src="city.png" alt="" /></div>
<img src="moon.png" alt="" id="moon" style="visibility: hidden;"/>
<div style="display:none">
<div class="shape">HAPPY</div>
<div class="shape">HAPPY</div>
<div class="shape">HAPPY</div>
<div class="shape">HAPPY</div>
<div class="shape">HAPPY</div>
</div>
<audio src="boom.mp3" preload="auto"></audio>
<audio src="boom.mp3" preload="auto"></audio>
<audio src="boom.mp3" preload="auto"></audio>
<audio src="boom.mp3" preload="auto"></audio>
<audio src="boom.mp3" preload="auto"></audio>
<audio src="boom.mp3" preload="auto"></audio>
<audio src="shotfire.mp3" preload="auto"></audio>
<audio src="shotfire.mp3" preload="auto"></audio>
<audio src="shotfire.mp3" preload="auto"></audio>
</body>
<script>
var cas=document.getElementById('cas'),
ctx=cas.getContext('2d'),
ocas=document.createElement('canvas'),
octx=ocas.getContext('2d'),
cw=document.compatMode=='BackCompat'?document.body.clientWidth:document.documentElement.clientWidth,
ch=document.compatMode=='BackCompat'?document.body.clientHeight:document.documentElement.clientHeight,
last,
now,
stars=[],
booms=[],
moon=document.getElementById('moon'),
shape=document.querySelectorAll('.shape'),
audio=document.getElementsByTagName('audio');
cas.width=ocas.width=cw;
cas.height=ocas.height=ch;
window.onload=function(){
createStars();
last=+new Date();
animate();
}
function animate(){
ctx.save();
ctx.globalCompositeOperation='destination-out';
ctx.globalAlpha=0.1;
ctx.fillRect(0,0,cw,ch);
ctx.restore();
stars.forEach(function(){
this.paint();
});
createMoon();
now=+new Date();
if(now-last>1000){
var rand=getRand(0,100)>50?true:false,num=Math.floor(getRand(0,shape.length)),
x=getRand(cw/3,cw*2/3),y=ch,r=2,color='#fff',tx=getRand(cw/5,cw*4/5),ty=getRand(50,ch-300);
if(rand){
booms.push(new Firework(x,y,r,color,tx,ty));
}else{
booms.push(new Firework(x,y,r,color,tx,ty,shape[num]));
}
for(var i=0;i<audio.length;i++){
if(audio[i].src.indexOf('shotfire')>=0 && (audio[i].paused || audio[i].ended)){
audio[i].play();
break;
}
}
last=now;
}
booms.forEach(function(){
var that=this;
if(!this.dead){
this.update();
this.smoke();
}else{
this.booms.forEach(function(index){
if(!this.dead){
this.update();
}else{
if(index==that.booms.length){
booms.splice(booms.indexOf(that),1);
}
}
});
}
});
requestAnimationFrame(animate);
}
function Particle(x,y,r,color,tx,ty){
this.x=x;
this.y=y;
this.r=r;
this.color=color;
this.tx=tx;
this.ty=ty;
this.dead=false;
}
Particle.prototype={
paint:function(){
ctx.beginPath();
ctx.fillStyle='rgba('+this.color.a+','+this.color.b+','+this.color.c+',1)';
ctx.fillRect(this.x-this.r,this.y-this.r,2*this.r,2*this.r);
ctx.closePath();
},
update:function(){
this.ty+=0.3;
var dx=this.tx-this.x,dy=this.ty-this.y;
if(Math.abs(dx)<0.1 && Math.abs(dy)<20){
this.dead=true;
}else{
this.x=Math.abs(dx)<0.1?this.tx:this.x+dx*0.1;
this.y=Math.abs(dy)<0.1?this.ty:this.y+dy*0.1;
this.paint();
}
}
}
function Firework(x,y,r,color,tx,ty,shape){
this.x=x;
this.y=y;
this.r=r;
this.color=color;
this.tx=tx;
this.ty=ty;
this.shape=shape || false;
this.dead=false;
this.ba=getRand(30,50);
this.booms=[];
}
Firework.prototype={
paint:function(){
ctx.save();
ctx.beginPath();
ctx.fillStyle=this.color;
ctx.arc(this.x,this.y,this.r,0,2*Math.PI);
ctx.fill();
ctx.closePath();
ctx.restore();
},
smoke:function(){
ctx.save();
ctx.beginPath();
ctx.fillStyle='rgba(255,228,150,0.3)';
ctx.arc(this.x,this.y,this.r+Math.random()*3,0,2*Math.PI);
ctx.fill();
ctx.closePath();
ctx.restore();
},
update:function(){
var dx=this.tx-this.x,dy=this.ty-this.y;
if(Math.abs(dx)<this.ba && Math.abs(dy)<this.ba){
this.dead=true;
if(this.shape){
this.shapeBoom();
}else{
this.boom();
}
for(var i=0;i<audio.length;i++){
if(audio[i].src.indexOf('boom')>=0 && (audio[i].paused || audio[i].ended)){
audio[i].play();
break;
}
}
}else{
this.x+=dx*0.03;
this.y+=dy*0.03;
this.paint();
}
},
boom:function(){
var count=getRand(100,300),r,color,tx,ty,angle,distance;
for(var i=0;i<count;i++){
r=getRand(1,3);
angle=getRand(-Math.PI,Math.PI);
color={
a:parseInt(getRand(0,255)),
b:parseInt(getRand(0,255)),
c:parseInt(getRand(0,255))
},
distance=getRand(0,count);
tx=Math.cos(angle)*distance+this.x;
ty=Math.sin(angle)*distance+this.y;
this.booms.push(new Particle(this.x,this.y,r,color,tx,ty));
}
},
shapeBoom:function(){
var that=this,step=4,dx=cw/2-this.tx,dy=ch/2-this.ty;
getData(this.shape,step,function(dots){
dots.forEach(function(){
that.booms.push(new Particle(that.x,that.y,1,this.color,this.x-dx,this.y-dy));
});
});
}
}
function getData(dom,step,callback){
var text=dom.innerHTML,imgData=[],dots=[],i;
octx.clearRect(0,0,cw,ch);
octx.save();
octx.font='120px 微?雅黑 bold';
octx.textAlign='center';
octx.textBaseline='middle';
octx.fillStyle='rgba('+parseInt(getRand(168,255))+','+parseInt(getRand(168,255))+','+parseInt(getRand(168,255))+',1)';
octx.fillText(text,cw/2,ch/2);
octx.restore();
imgData=octx.getImageData(0,0,cw,ch);
for(var x=0;x<imgData.width;x+=step){
for(var y=0;y<imgData.height;y+=step){
i=(y*imgData.width+x)*4;
if(imgData.data[i+3]>167){
dots.push({x:x,y:y,color:{
a:imgData.data[i],
b:imgData.data[i+1],
c:imgData.data[i+2]
}});
}
}
}
callback(dots);
}
function createMoon(){
var sx=cw-200,sy=100,r=40,add=2;
if(moon.complete){
ctx.drawImage(moon,sx,sy,2*r,2*r);
}else{
moon.onload=function(){
ctx.drawImage(moon,sx,sy,2*r,2*r);
}
}
ctx.save();
ctx.beginPath();
ctx.fillStyle='rgba(240,219,120,0.05)';
for(var i=0;i<5;i++){
ctx.arc(sx+r,sy+r,r+add,0,2*Math.PI);
ctx.fill();
add+=2;
}
ctx.closePath();
ctx.restore();
}
function createStars(){
for(var i=0;i<100;i++){
stars.push(new Star(getRand(0,cw),getRand(0,ch-200),Math.random(),'#fff'));
}
}
function Star(x,y,r,color){
this.x=x;
this.y=y;
this.r=r;
this.color=color;
}
Star.prototype={
paint:function(){
ctx.save();
ctx.beginPath();
ctx.fillStyle=this.color;
ctx.arc(this.x,this.y,this.r,0,2*Math.PI);
ctx.fill();
ctx.closePath();
ctx.restore();
}
}
function getRand(min,max){
return Math.random()*(max-min)+min;
}
Array.prototype.forEach=function(callback){
for(var i=0;i<this.length;i++){
if(this[i]!=null){
callback.call(this[i],i);
}
}
}
</script>
</html>
鼠标点击实现
<!doctype html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="Generator" content="EditPlusR">
<meta name="Author" content="">
<meta name="Keywords" content="">
<meta name="Description" content="">
<title>?日快?</title>
<style>
*{margin:0;padding:0;list-style-type:none;}
a,img{border:0;}
body{font:12px/180% Arial, Helvetica, sans-serif, "新宋体";background:#000 url("pic.jpg") no-repeat center center;background-size: 100% 100%;}
#canvas{cursor:crosshair;display:block;color:black;}
</style>
</head>
<body>
<canvas id='canvas' width='100%' height='100%'></canvas>
<script type="text/javascript" src="lib/jquery-1.11.3.js"></script>
<script type="text/javascript">
var canvas=document.getElementById('canvas'),
ctx=canvas.getContext('2d'),
cw=window.innerWidth,
ch=window.innerHeight,
mousedown=!1,
mx,
my,
last,
now,
timerTick=0,
timerTotal=80,
timeWait=0,
timeWaitTotal=10,
fireworks=[],
particles=[],
fl,
pl,
hue=120,
accelerate=1.05,
img=new Image();
canvas.width=cw;
canvas.height=ch;
img.src='pic.jpg';
document.onmousedown=function(e){
mousedown=!0;
}
document.onmouseup=function(e){
mousedown=!1;
}
document.onmousemove=function(e){
mx=e.pageX-canvas.offsetLeft;
my=e.pageY-canvas.offsetTop;
}
window.onload=function(){
last=+new Date();
animate();
}
function animate(){
now=+new Date();
if(now-last>30){
hue=getRand(50,80);
ctx.globalCompositeOperation='destination-out';
ctx.globalAlpha=0.5;
ctx.fillRect(0,0,cw,ch);
ctx.fill();
ctx.globalCompositeOperation='lighter';
ctx.drawImage(img,0,0,cw,ch);
/*??和更新*/
fl=fireworks.length-1;
while(fl>=0){
fireworks[fl].paint();
fireworks[fl].update(fl);
fl--;
}
pl=particles.length-1;
while(pl>=0){
particles[pl].paint();
particles[pl].update(pl);
pl--;
}
/*添加?象*/
if(timerTick>timerTotal){
if(!mousedown){
fireworks.push(new Firework(cw/2,ch,getRand(0,cw),getRand(50,ch/2)));
timerTick=0;
}
}else{
timerTick++;
}
if(timeWait>timeWaitTotal){
if(mousedown){
fireworks.push(new Firework(cw/2,ch,mx,my));
timeWait=0;
}
}else{
timeWait++;
}
}
if('requestAnimationFrame' in window){
requestAnimationFrame(animate);
}else if('mozRequestAnimationFrame' in window){
mozRequestAnimationFrame(animate);
}
}
function Firework(sx,sy,tx,ty){
this.x=sx;
this.y=sy;
this.tx=tx;
this.ty=ty;
this.radius=1;
this.brightness=getRand(50,70);
this.angle=Math.atan2(this.ty-this.y,this.tx-this.x);
this.speed=5;
this.xadd=Math.cos(this.angle);
this.yadd=Math.sin(this.angle);
this.coord=[];
for(var i=0;i<3;i++){
this.coord.push({x:this.x,y:this.y});
}
}
Firework.prototype={
paint:function(){
ctx.save();
ctx.beginPath();
ctx.moveTo(this.coord[this.coord.length-1].x,this.coord[this.coord.length-1].y);
ctx.lineTo(this.x,this.y);
ctx.strokeStyle='hsl('+hue+',100%,'+this.brightness+'%)';
ctx.lineWidth=1.5;
ctx.stroke();
ctx.beginPath();
ctx.arc(this.tx,this.ty,this.radius,0,2*Math.PI);
ctx.stroke();
ctx.closePath();
ctx.restore();
},
update:function(index){
var dx=this.tx-this.x,dy=this.ty-this.y;
if(Math.abs(dx)<20 && Math.abs(dy)<20){
fireworks.splice(index,1);
createParticles(this.x,this.y);
}else{
this.x+=this.speed*this.xadd;
this.y+=this.speed*this.yadd;
this.speed*=accelerate;
this.coord.pop();
this.coord.unshift({x:this.x,y:this.y});
if(this.radius>8){
this.radius=1;
}else{
this.radius+=0.3;
}
}
}
}
function createParticles(x,y){
for(var i=0;i<20;i++){
particles.push(new Particle(x,y));
}
}
function Particle(x,y){
this.x=x;
this.y=y;
this.hue=getRand(hue-50,hue+50);
this.brightness=getRand(50,80);
this.alpha=1;
this.decay=getRand(0.015,0.03);
this.angle=getRand(-Math.PI,Math.PI);
this.speed=getRand(5,10);
this.friction=0.95;
this.gravity=2;
this.xadd=Math.cos(this.angle);
this.yadd=Math.sin(this.angle);
this.coord=[];
for(var i=0;i<5;i++){
this.coord.push({x:this.x,y:this.y});
}
}
Particle.prototype={
paint:function(){
ctx.beginPath();
ctx.moveTo(this.coord[this.coord.length-1].x,this.coord[this.coord.length-1].y);
ctx.lineTo(this.x,this.y);
ctx.strokeStyle='hsla('+this.hue+',100%,'+this.brightness+'%,'+this.alpha+')';
ctx.lineWidth=2;
ctx.stroke();
ctx.closePath();
},
update:function(index){
this.alpha-=this.decay;
if(Math.abs(this.alpha)<=this.decay){
particles.splice(index,1);
}else{
this.x+=this.speed*this.xadd;
this.y+=this.speed*this.yadd+this.gravity;
this.speed*=this.friction;
this.coord.pop();
this.coord.unshift({x:this.x,y:this.y});
}
}
}
function getRand(min,max){
return Math.random()*(max-min)+min;
}
</script>
</body>
</html>