likes
comments
collection
share

来吧,烟花走起!

作者站长头像
站长
· 阅读数 51

需求

今天我们来实现一个烟花自动绽放的效果场景,包括从地面发射升空和到最高点绽放两部分,实现效果图如下:

来吧,烟花走起!

实现

HTML代码

HTML 页面声明如下:

<!DOCTYPE html>
<html>

<head lang="en">
	<meta charset="GBK">
	<title>烟花</title>
	<style>
	</style>
</head>

<body>
	<canvas id='canvas'></canvas>
	<script type="text/javascript">
	</script>
</body>

</html>

CSS代码

接下来,我们看 CSS 样式,具体内容如下:

    body {
            margin: 0;
            padding: 0;
            /* background-color: black; */
    }

JavaScript代码

JS 相关的代码如下,具体的方法就不一一介绍了,有必要的地方都添加了注释:

var canvas = document.getElementById('canvas');
		canvas.width = document.documentElement.clientWidth;
		canvas.height = document.documentElement.clientHeight;
		var ctx = canvas.getContext("2d");

		var arr = [];//存取烟花的数组
		var list = [];//存取碎片的数组

		//定义烟花
		function Fire() {
			this.x = canvas.width / 2, //烟花的x坐标
				this.y = canvas.height, //烟花的y坐标
				this.radius = 2, //烟花的半径
				this.speed = getRandom(6, 8), //运行的速度
				this.angle = getRandom(1.1 * Math.PI, 1.9 * Math.PI), //发射的范围
				//this.color='white';//颜色
				this.color = Math.random() > 0.5 ? 'white' : ('#' + Math.random().toString(16).substr(2, 6).toUpperCase());
		}
		Fire.prototype.draw = function () {
			ctx.beginPath();
			ctx.arc(this.x, this.y, this.radius, 0, 2 * Math.PI);//画圆
			ctx.fillStyle = this.color;
			ctx.fill();
			ctx.closePath();
		}
		//爆炸后重置烟花的位置、速度等参数,以便再次发射
		Fire.prototype.restore = function () {
			this.x = canvas.width / 2;
			this.y = canvas.height;
			this.speed = getRandom(6, 8);
			this.angle = getRandom(1.1 * Math.PI, 1.9 * Math.PI);
		}

		Fire.prototype.update = function () {
			//更新X坐标
			this.x += Math.cos(this.angle) * this.speed;
			//更新Y坐标
			this.y += Math.sin(this.angle) * this.speed;
			//控制飞行的最大位置
			if (this.y < canvas.height * 0.2 || this.x < canvas.width * 0.1 || this.x > canvas.width * 0.9) {
				//达到最大位置后
				buildChildFire(this);//生成烟花碎片
				this.restore();//将烟花重置
			}
		}
		//定义烟花碎片
		function ChildFire() {
			this.x = 0,//烟花碎片x位置
				this.y = 0,//烟花碎片y位置
				this.radius = 1,//烟花碎片半径
				this.n = 6,//烟花碎片次数
				this.speed = getRandom(1, 10) / 5,//烟花碎片速度
				this.g = 0.98,//重力
				this.mocha = 0.96,//摩擦力
				//this.color='#' + Math.random().toString(16).substr(2, 6).toUpperCase(),

				//一半是白色的碎片,一半是五颜六色的碎片
				this.color = Math.random() > 0.5 ? 'white' : ('#' + Math.random().toString(16).substr(2, 6).toUpperCase()),
				//碎片的角度
				this.angle = getRandom(0, 2 * Math.PI);//0-2π之间随机,这样就各个方向都有了
		}
		//绘制碎片
		ChildFire.prototype.draw = function () {
			ctx.beginPath();
			ctx.arc(this.x, this.y, this.radius, 0, 2 * Math.PI);
			ctx.fillStyle = this.color;
			ctx.fill();
			ctx.closePath();
		}
		ChildFire.prototype.update = function (index) {
			//更新X坐标
			this.x += Math.cos(this.angle) * this.speed * this.mocha;
			//更新Y坐标
			this.y += Math.sin(this.angle) * this.speed * this.mocha * this.g;

			//执行一次加1
			this.n++;
			//50次后清除烟花碎片
			if (this.n >= 50) {
				//清除当前碎片的动画函数
				window.cancelAnimationFrame(this.raf);
				//清除当前烟花碎片,否则会导致卡死
				list.splice(index, 1);
			} else {
				this.raf = window.requestAnimationFrame(this.draw);
			}
		}

		//创建烟花对象(这里默认2个,多了可能会卡)
		var fire = new Fire();
		arr.push(fire);
		var fire2 = new Fire();
		arr.push(fire2);

		//清洗画布
		function clear() {
			ctx.fillStyle = 'rgba(0,0,0,0.3)';
			ctx.fillRect(0, 0, canvas.width, canvas.height);
		}
		//创建碎片
		function buildChildFire(fire) {
			for (var i = 0; i < 200; i++) {
				var cFire = new ChildFire();
				//定义烟花碎片的x、y轴位置
				cFire.x = fire.x;
				cFire.y = fire.y;
				list.push(cFire);
			}
		}

		function draw() {
			clear();
			// ctx.drawImage(img, 0, 0, canvas.width, canvas.height);
			//循环存烟花碎片数组
			list.forEach(function (c, i) {
				c.draw();
				c.update(i);
			})
			//循环存烟花数组
			arr.forEach(function (c, i) {
				c.draw();
				c.update();
			})
			window.requestAnimationFrame(draw);
		}
		//获取一定范围内的随机数
		function getRandom(min, max) {
			return Math.random() * (max - min) + min;
		}
		//执行动画函数
		window.requestAnimationFrame(draw)

内容有些多,不过终于介绍完了,赶紧让大家看看项目的展示效果吧,项目地址如下:


作者简介:😄大家好,我是 Data-Mining(liuzhen007),是一位典型的音视频技术爱好者,前后就职于传统广电巨头和音视频互联网公司,具有丰富的音视频直播和点播相关经验,对 WebRTC、FFmpeg 和 Electron 有非常深入的了解,😄公众号:玩转音视频。同时也是 CSDN 博客专家、华为云享专家(共创编辑)、InfoQ 签约作者,欢迎关注我分享更多干货!😄

转载自:https://juejin.cn/post/7088332540682960904
评论
请登录