canvas - 酷炫粒子文字的代码解析

先看效果:

image

再放源代码:

点击查看代码
<!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. 声明全局变量

就是这一段代码:

image

其中canvasctxwhlolo四个变量没什么说的,说下其他三个:

  • logoParticles:这是canvas元素所有的像素实例的对象,这个数组包含这个canvas所有的像素点,通过遍历每个像素点对其进行粒子偏移的操作。
  • particleIndex:这是当前遍历的像素点的索引
  • hue:这个是粒子跳动时的颜色,这里使用的是hsl颜色表示法。这个表示色相。

不懂什么意思的就先放这,继续往下看。

先不看下面的构造函数Particle()。声明完全局变量后就是加载图片并绘制到canvas上了:

2. 图片加载完成

就是下面一段代码:

image

在图片在页面上加载完成以后,就要把图片绘制到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. 每个关键帧里重绘画布

image

4. 重绘完画布以后再重绘canvas每个需要重绘的像素点

重绘每个像素点的操作都封装在Particle()构造函数中了。就是这两段:

image

  • this.draw() 每个像素点先绘制一个以该像素点为基本地址的有偏移量的宽高为2的方块

  • this.update() 再把偏移量增减一个随机值(就是vx、vy值)。当随机值(this.life)达到最大值(this.maxLife)时。回到这个像素点的初始位置继续画,并重新获取偏移量(this.reset())。如此往复

其中maxLifevx,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()方法,即可实现单个像素点的循环重绘。

posted @ 2022-07-01 17:20  俄罗斯方块  阅读(277)  评论(0编辑  收藏  举报