javascript实现超好看的3D烟花特效


Posted in Javascript onJanuary 01, 2020

本文实例为大家分享了超好看3D烟花的具体代码,供大家参考,具体内容如下

javascript实现超好看的3D烟花特效

<!doctype html>
<html>
<head>
<meta charset="utf-8">
<title>3D烟花</title>

<style>
html,body{
 margin:0px;
 width:100%;
 height:100%;
 overflow:hidden;
 background:#000;
}

#canvas{
 width:100%;
 height:100%;
}
</style>
</head>
<body>
<canvas id="canvas"></canvas><script>
function initVars(){

 pi=Math.PI;
 ctx=canvas.getContext("2d");
 canvas.width=canvas.clientWidth;
 canvas.height=canvas.clientHeight;
 cx=canvas.width/2;
 cy=canvas.height/2;
 playerZ=-25;
 playerX=playerY=playerVX=playerVY=playerVZ=pitch=yaw=pitchV=yawV=0;
 scale=600;
 seedTimer=0;seedInterval=5,seedLife=100;gravity=.02;
 seeds=new Array();
 sparkPics=new Array();
 s="https://cantelope.org/NYE/";
 for(i=1;i<=10;++i){
 sparkPic=new Image();
 sparkPic.src=s+"spark"+i+".png";
 sparkPics.push(sparkPic);
 }
 sparks=new Array();
 pow1=new Audio(s+"pow1.ogg");
 pow2=new Audio(s+"pow2.ogg");
 pow3=new Audio(s+"pow3.ogg");
 pow4=new Audio(s+"pow4.ogg");
 frames = 0;
}

function rasterizePoint(x,y,z){

 var p,d;
 x-=playerX;
 y-=playerY;
 z-=playerZ;
 p=Math.atan2(x,z);
 d=Math.sqrt(x*x+z*z);
 x=Math.sin(p-yaw)*d;
 z=Math.cos(p-yaw)*d;
 p=Math.atan2(y,z);
 d=Math.sqrt(y*y+z*z);
 y=Math.sin(p-pitch)*d;
 z=Math.cos(p-pitch)*d;
 var rx1=-1000,ry1=1,rx2=1000,ry2=1,rx3=0,ry3=0,rx4=x,ry4=z,uc=(ry4-ry3)*(rx2-rx1)-(rx4-rx3)*(ry2-ry1);
 if(!uc) return {x:0,y:0,d:-1};
 var ua=((rx4-rx3)*(ry1-ry3)-(ry4-ry3)*(rx1-rx3))/uc;
 var ub=((rx2-rx1)*(ry1-ry3)-(ry2-ry1)*(rx1-rx3))/uc;
 if(!z)z=.000000001;
 if(ua>0&&ua<1&&ub>0&&ub<1){
 return {
  x:cx+(rx1+ua*(rx2-rx1))*scale,
  y:cy+y/z*scale,
  d:Math.sqrt(x*x+y*y+z*z)
 };
 }else{
 return {
  x:cx+(rx1+ua*(rx2-rx1))*scale,
  y:cy+y/z*scale,
  d:-1
 };
 }
}

function spawnSeed(){
 
 seed=new Object();
 seed.x=-50+Math.random()*100;
 seed.y=25;
 seed.z=-50+Math.random()*100;
 seed.vx=.1-Math.random()*.2;
 seed.vy=-1.5;//*(1+Math.random()/2);
 seed.vz=.1-Math.random()*.2;
 seed.born=frames;
 seeds.push(seed);
}

function splode(x,y,z){
 
 t=5+parseInt(Math.random()*150);
 sparkV=1+Math.random()*2.5;
 type=parseInt(Math.random()*3);
 switch(type){
 case 0:
  pic1=parseInt(Math.random()*10);
  break;
 case 1:
  pic1=parseInt(Math.random()*10);
  do{ pic2=parseInt(Math.random()*10); }while(pic2==pic1);
  break;
 case 2:
  pic1=parseInt(Math.random()*10);
  do{ pic2=parseInt(Math.random()*10); }while(pic2==pic1);
  do{ pic3=parseInt(Math.random()*10); }while(pic3==pic1 || pic3==pic2);
  break;
 }
 for(m=1;m<t;++m){
 spark=new Object();
 spark.x=x; spark.y=y; spark.z=z;
 p1=pi*2*Math.random();
 p2=pi*Math.random();
 v=sparkV*(1+Math.random()/6)
 spark.vx=Math.sin(p1)*Math.sin(p2)*v;
 spark.vz=Math.cos(p1)*Math.sin(p2)*v;
 spark.vy=Math.cos(p2)*v;
 switch(type){
  case 0: spark.img=sparkPics[pic1]; break;
  case 1:
  spark.img=sparkPics[parseInt(Math.random()*2)?pic1:pic2];
  break;
  case 2:
  switch(parseInt(Math.random()*3)){
   case 0: spark.img=sparkPics[pic1]; break;
   case 1: spark.img=sparkPics[pic2]; break;
   case 2: spark.img=sparkPics[pic3]; break;
  }
  break;
 }
 spark.radius=25+Math.random()*50;
 spark.alpha=1;
 spark.trail=new Array();
 sparks.push(spark);
 }
 switch(parseInt(Math.random()*4)){
 case 0: pow=new Audio(s+"pow1.ogg"); break;
 case 1: pow=new Audio(s+"pow2.ogg"); break;
 case 2: pow=new Audio(s+"pow3.ogg"); break;
 case 3: pow=new Audio(s+"pow4.ogg"); break;
 }
 d=Math.sqrt((x-playerX)*(x-playerX)+(y-playerY)*(y-playerY)+(z-playerZ)*(z-playerZ));
 pow.volume=1.5/(1+d/10);
 pow.play();
}

function doLogic(){
 
 if(seedTimer<frames){
 seedTimer=frames+seedInterval*Math.random()*10;
 spawnSeed();
 }
 for(i=0;i<seeds.length;++i){
 seeds[i].vy+=gravity;
 seeds[i].x+=seeds[i].vx;
 seeds[i].y+=seeds[i].vy;
 seeds[i].z+=seeds[i].vz;
 if(frames-seeds[i].born>seedLife){
  splode(seeds[i].x,seeds[i].y,seeds[i].z);
  seeds.splice(i,1);
 }
 }
 for(i=0;i<sparks.length;++i){
 if(sparks[i].alpha>0 && sparks[i].radius>5){
  sparks[i].alpha-=.01;
  sparks[i].radius/=1.02;
  sparks[i].vy+=gravity;
  point=new Object();
  point.x=sparks[i].x;
  point.y=sparks[i].y;
  point.z=sparks[i].z;
  if(sparks[i].trail.length){
  x=sparks[i].trail[sparks[i].trail.length-1].x;
  y=sparks[i].trail[sparks[i].trail.length-1].y;
  z=sparks[i].trail[sparks[i].trail.length-1].z;
  d=((point.x-x)*(point.x-x)+(point.y-y)*(point.y-y)+(point.z-z)*(point.z-z));
  if(d>9){
   sparks[i].trail.push(point);
  }
  }else{
  sparks[i].trail.push(point);
  }
  if(sparks[i].trail.length>5)sparks[i].trail.splice(0,1);  
  sparks[i].x+=sparks[i].vx;
  sparks[i].y+=sparks[i].vy;
  sparks[i].z+=sparks[i].vz;
  sparks[i].vx/=1.075;
  sparks[i].vy/=1.075;
  sparks[i].vz/=1.075;
 }else{
  sparks.splice(i,1);
 }
 }
 p=Math.atan2(playerX,playerZ);
 d=Math.sqrt(playerX*playerX+playerZ*playerZ);
 d+=Math.sin(frames/80)/1.25;
 t=Math.sin(frames/200)/40;
 playerX=Math.sin(p+t)*d;
 playerZ=Math.cos(p+t)*d;
 yaw=pi+p+t;
}

function rgb(col){
 
 var r = parseInt((.5+Math.sin(col)*.5)*16);
 var g = parseInt((.5+Math.cos(col)*.5)*16);
 var b = parseInt((.5-Math.sin(col)*.5)*16);
 return "#"+r.toString(16)+g.toString(16)+b.toString(16);
}

function draw(){
 
 ctx.clearRect(0,0,cx*2,cy*2);
 
 ctx.fillStyle="#ff8";
 for(i=-100;i<100;i+=3){
 for(j=-100;j<100;j+=4){
  x=i;z=j;y=25;
  point=rasterizePoint(x,y,z);
  if(point.d!=-1){
  size=250/(1+point.d);
  d = Math.sqrt(x * x + z * z);
  a = 0.75 - Math.pow(d / 100, 6) * 0.75;
  if(a>0){
   ctx.globalAlpha = a;
   ctx.fillRect(point.x-size/2,point.y-size/2,size,size);  
  }
  }
 }
 }
 ctx.globalAlpha=1;
 for(i=0;i<seeds.length;++i){
 point=rasterizePoint(seeds[i].x,seeds[i].y,seeds[i].z);
 if(point.d!=-1){
  size=200/(1+point.d);
  ctx.fillRect(point.x-size/2,point.y-size/2,size,size);
 }
 }
 point1=new Object();
 for(i=0;i<sparks.length;++i){
 point=rasterizePoint(sparks[i].x,sparks[i].y,sparks[i].z);
 if(point.d!=-1){
  size=sparks[i].radius*200/(1+point.d);
  if(sparks[i].alpha<0)sparks[i].alpha=0;
  if(sparks[i].trail.length){
  point1.x=point.x;
  point1.y=point.y;
  switch(sparks[i].img){
   case sparkPics[0]:ctx.strokeStyle="#f84";break;
   case sparkPics[1]:ctx.strokeStyle="#84f";break;
   case sparkPics[2]:ctx.strokeStyle="#8ff";break;
   case sparkPics[3]:ctx.strokeStyle="#fff";break;
   case sparkPics[4]:ctx.strokeStyle="#4f8";break;
   case sparkPics[5]:ctx.strokeStyle="#f44";break;
   case sparkPics[6]:ctx.strokeStyle="#f84";break;
   case sparkPics[7]:ctx.strokeStyle="#84f";break;
   case sparkPics[8]:ctx.strokeStyle="#fff";break;
   case sparkPics[9]:ctx.strokeStyle="#44f";break;
  }
  for(j=sparks[i].trail.length-1;j>=0;--j){
   point2=rasterizePoint(sparks[i].trail[j].x,sparks[i].trail[j].y,sparks[i].trail[j].z);
   if(point2.d!=-1){
   ctx.globalAlpha=j/sparks[i].trail.length*sparks[i].alpha/2;
   ctx.beginPath();
   ctx.moveTo(point1.x,point1.y);
   ctx.lineWidth=1+sparks[i].radius*10/(sparks[i].trail.length-j)/(1+point2.d);
   ctx.lineTo(point2.x,point2.y);
   ctx.stroke();
   point1.x=point2.x;
   point1.y=point2.y;
   }
  }
  }
  ctx.globalAlpha=sparks[i].alpha;
  ctx.drawImage(sparks[i].img,point.x-size/2,point.y-size/2,size,size);
 }
 }
}

function frame(){

 if(frames>100000){
 seedTimer=0;
 frames=0;
 }
 frames++;
 draw();
 doLogic();
 requestAnimationFrame(frame);
}

window.addEventListener("resize",()=>{
 canvas.width=canvas.clientWidth;
 canvas.height=canvas.clientHeight;
 cx=canvas.width/2;
 cy=canvas.height/2;
});

initVars();
frame();</script>
<div style="text-align:center;">
</div>
</body>
</html>

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
javascript 多浏览器 事件大全
Mar 23 Javascript
js使用eval解析json(js中使用json)
Jan 17 Javascript
javascript基于DOM实现省市级联下拉框的方法
May 14 Javascript
javascript中this指向详解
Apr 23 Javascript
JavaScript浏览器对象之一Window对象详解
Jun 03 Javascript
vue.js的双向数据绑定Object.defineProperty方法的神奇之处
Jan 18 Javascript
Vue 刷新当前路由的实现代码
Sep 26 Javascript
Vue3.x源码调试的实现方法
Oct 13 Javascript
javascript+css实现俄罗斯方块小游戏
Jun 28 Javascript
区分vue-router的hash和history模式
Oct 03 Javascript
vue中是怎样监听数组变化的
Oct 24 Javascript
JavaScript实现alert弹框效果
Nov 19 Javascript
JavaScript工具库MyTools详解
Jan 01 #Javascript
javascript Canvas动态粒子连线
Jan 01 #Javascript
小程序新版订阅消息模板消息
Dec 31 #Javascript
小程序实现长按保存图片的方法
Dec 31 #Javascript
使用webpack搭建vue环境的教程详解
Dec 31 #Javascript
Vue 实现显示/隐藏层的思路(加全局点击事件)
Dec 31 #Javascript
TypeScript之调用栈的实现
Dec 31 #Javascript
You might like
一些php技巧与注意事项分析
2011/02/03 PHP
memcache命令启动参数中文解释
2014/01/13 PHP
PHP获取mysql数据表的字段名称和详细信息的方法
2014/09/27 PHP
两千行代码的PHP学习笔记汇总
2014/10/05 PHP
php将HTML表格每行每列转为数组实现采集表格数据的方法
2015/04/03 PHP
使用jquery插件实现图片延迟加载技术详细说明
2011/03/12 Javascript
用js小类库获取浏览器的高度和宽度信息
2012/01/15 Javascript
JavaScript面向对象知识串结(读JavaScript高级程序设计(第三版))
2012/07/17 Javascript
js同比例缩放图片的小例子
2013/10/30 Javascript
JavaScript加入收藏夹功能(兼容IE、firefox、chrome)
2014/05/05 Javascript
jQuery移除tr无效的解决方法(tr是动态添加)
2014/09/22 Javascript
jquery中object对象循环遍历的方法
2015/12/18 Javascript
Bootstrap入门书籍之(四)菜单、按钮及导航
2016/02/17 Javascript
Nodejs如何搭建Web服务器
2016/03/28 NodeJs
AngularJS入门教程之AngularJS模型
2016/04/18 Javascript
老生常谈javascript的类型转换
2016/10/12 Javascript
Angular2平滑升级到Angular4的步骤详解
2017/03/29 Javascript
Angular 2 利用Router事件和Title实现动态页面标题的方法
2017/08/23 Javascript
Vue-router 类似Vuex实现组件化开发的示例
2017/09/15 Javascript
实例详解vue.js浅度监听和深度监听及watch用法
2018/08/16 Javascript
angularjs模态框的使用代码实例
2019/12/20 Javascript
jQuery使用ajax传递json对象到服务端及contentType的用法示例
2020/03/12 jQuery
Python实现文件按照日期命名的方法
2015/07/09 Python
Python中生成Epoch的方法
2017/04/26 Python
Python3安装Scrapy的方法步骤
2017/11/23 Python
Python基于递归算法实现的汉诺塔与Fibonacci数列示例
2018/04/18 Python
python实现简单日志记录库glog的使用
2019/12/13 Python
Python中的xlrd模块使用原理解析
2020/05/21 Python
Python设计密码强度校验程序
2020/07/30 Python
python利用xlsxwriter模块 操作 Excel
2020/10/14 Python
简单介绍HTML5中audio标签的使用
2015/09/24 HTML / CSS
英国最大的在线时尚眼镜店:Eyewearbrands
2019/03/12 全球购物
工程安全员岗位职责
2014/03/09 职场文书
公司中层管理培训心得体会
2016/01/11 职场文书
go语言基础 seek光标位置os包的使用
2021/05/09 Golang
MySQL 分页查询的优化技巧
2021/05/12 MySQL