canvas - 酷炫粒子文字的代码解析
先看效果:
再放源代码:
点击查看代码
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<meta http-equiv="X-UA-Compatible" content="IE=edge" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>粒子文字</title>
<style>
html,
body {
height: 100%;
}
body {
background: black;
overflow: hidden;
}
canvas {
max-width: 100%;
}
</style>
</head>
<body>
<canvas id="c"></canvas>
</body>
<script>
/* 兼容什么的 */
window.requestAnimationFrame =
window.requestAnimationFrame ||
window.webkitRequestAnimationFrame ||
window.mozRequestAnimationFrame;
var canvas = document.getElementById('c'),
ctx = canvas.getContext('2d'),
w = (canvas.width = window.innerWidth),
h = (canvas.height = window.innerHeight),
logoParticles = [],
particleIndex = 0,
logo = new Image(),
hue = 0;
function Particle(x, y) {
this.origX = this.x = x;
this.origY = this.y = y;
this.color = 'white';
this.maxLife = this.random(20);
this.life = 0;
this.vx = this.random(-1, 1);
this.vy = this.random(-1, 1);
this.grav = 0.04;
this.index = particleIndex;
logoParticles[particleIndex] = this;
particleIndex++;
}
Particle.prototype = {
constructor: Particle,
draw: function () {
ctx.fillStyle = this.color;
ctx.fillRect(this.x, this.y, 2, 2);
this.update();
},
update: function () {
if (this.life >= this.maxLife) {
logoParticles[this.index].reset();
}
this.x += this.vx;
this.y += this.vy;
this.vy += this.grav;
this.life++;
},
reset: function () {
this.x = this.origX;
this.y = this.origY;
this.life = 0;
this.color = 'hsl(' + hue + ', 100%, 50%)';
this.vx = this.random(-1, 1);
this.vy = this.random(-1, 1);
},
random: function () {
var min = arguments.length == 1 ? 0 : arguments[0],
max = arguments.length == 1 ? arguments[0] : arguments[1];
return Math.random() * (max - min) + min;
},
};
logo.src =
'';
logo.onload = function () {
var posX = (w - this.width) / 2,
posY = (h - this.height) / 2;
ctx.drawImage(this, posX, posY);
var imgData = ctx.getImageData(0, 0, w, h),
pixels = imgData.data;
for (var y = 0; y < imgData.height; y += 3) {
for (var x = 0; x < imgData.width; x += 3) {
var alpha = pixels[(imgData.width * y + x) * 4 + 3];
if (alpha > 0) {
logoParticles.push(new Particle(x, y));
}
}
}
setTimeout(function () {
animate();
}, 800);
};
function animate() {
ctx.fillStyle = 'rgba(0,0,0,.1)';
ctx.fillRect(0, 0, w, h);
for (var i in logoParticles) {
logoParticles[i].draw();
}
hue += 1;
window.requestAnimationFrame(animate);
}
</script>
</html>
以上代码复制到一个html文件中即可预览。
接下来从程序运行顺序开始解读:
1. 声明全局变量
就是这一段代码:
其中canvas
、ctx
、w
、h
、lolo
四个变量没什么说的,说下其他三个:
logoParticles
:这是canvas元素所有的像素实例的对象,这个数组包含这个canvas所有的像素点,通过遍历每个像素点对其进行粒子偏移的操作。particleIndex
:这是当前遍历的像素点的索引hue
:这个是粒子跳动时的颜色,这里使用的是hsl
颜色表示法。这个表示色相。
不懂什么意思的就先放这,继续往下看。
先不看下面的构造函数Particle()
。声明完全局变量后就是加载图片并绘制到canvas上了:
2. 图片加载完成
就是下面一段代码:
在图片在页面上加载完成以后,就要把图片绘制到canvas上了:
-
上面的
posX
,posY
是将屏幕宽度和高度减去图片宽度后除以二得到的canvas的左上角的x,y轴坐标位置,其目的是让canvas位于整个窗口的正中间。然后就是使用drawImage方法将图片绘制到canvas画布上。 -
imgData
是这个canvas画布的所有像素点数据,然后通过下面的两个for循环遍历所有的像素点数据,遍历每个像素点的alpha值(即透明度)。如果不是透明的(非背景:alpha>0)。就继续接下来的操作。(关于两个for循环是如何遍历出所有像素点数据的可以见我另一篇博文) -
logoParticles.push(new Particle(x, y));
这一步操作是把所有处于canvas像素点x,y位置的像素点作为一个实例对象存到所有需要操作的数组中;便于后续操作。
在所有的像素点数据保存完毕以后。接下来开始执行动画操作(animate()
)。
3. 每个关键帧里重绘画布
4. 重绘完画布以后再重绘canvas每个需要重绘的像素点
重绘每个像素点的操作都封装在Particle()
构造函数中了。就是这两段:
-
this.draw() 每个像素点先绘制一个以该像素点为基本地址的有偏移量的宽高为2的方块
-
this.update() 再把偏移量增减一个随机值(就是vx、vy值)。当随机值(this.life)达到最大值(this.maxLife)时。回到这个像素点的初始位置继续画,并重新获取偏移量(this.reset())。如此往复
其中maxLife
和vx
,vy
都是通过random()
方法获取到的。
5. Particle()构造函数解析
这里的构造函数及其原型是es6之前的写法,为什么写一个constructor:Particle
,本来其实Particle.prototype.constructor
就是指向Particle
的,这么写应该是为了省事,不然的话,这个原型上4个方法就要这么写了:
Particle.prototype.draw = function(){ ... }
Particle.prototype.update = function(){ ... }
Particle.prototype.reset = function(){ ... }
Particle.prototype.random = function(){ ... }
其实这里用ES6的class写法可以改写成下面这样,功能一致:
点击查看代码
class Particle {
constructor(x, y) {
this.origX = this.x = x; // this.origX:像素点原始位置;this.x:像素点当前位置
this.origY = this.y = y;
this.color = 'white'; // 当前像素点颜色
this.maxLife = this.random(20); // 当前像素点跳动的次数
this.life = 0; // 当前像素点已经偏移的距离
this.vx = this.random(-1, 1); // 每次x轴偏移的距离,可以是负的
this.vy = this.random(-1, 1);
this.grav = 0.04;
this.index = particleIndex; // 当前像素点在所有像素点数组(logoParticles)的索引
logoParticles[particleIndex] = this; // 把当前像素点的实例保存到数组中
particleIndex++; // 继续遍历下一个像素点
}
draw() {
ctx.fillStyle = this.color;
ctx.fillRect(this.x, this.y, 2, 2);
this.update();
}
update() {
if (this.life >= this.maxLife) {
logoParticles[this.index].reset();
};
this.x += this.vx;
this.y += this.vy;
// this.vy += this.grav;
this.life++;
}
reset() {
this.x = this.origX;
this.y = this.origY;
this.life = 0;
this.color = 'hsl(' + hue + ', 100%, 50%)';
this.vx = this.random(-1, 1);
this.vy = this.random(-1, 1);
}
/**
* 取最大值最小值中间的一个数(或0-最大值的那个值)
*/
random() {
var min = arguments.length == 1 ? 0 : arguments[0],
max = arguments.length == 1 ? arguments[0] : arguments[1];
return Math.random() * (max - min) + min;
}
}
6.小结
其实整个代码的关于canvas的知识并不多,就用到了drawImage,getImageData,fillRect
三个canvas的api,用法很简单。
主要还是每个像素点的实例对象的理解上。logoParticles
这个数组保存的是每个像素点的实例对象,每个实例对象里面包含了当前像素点的初始位置,最大偏移量,已经偏移量,当前x,y轴位置。通过遍历这个实例对象的数组,并调用其draw()方法,即可实现单个像素点的循环重绘。