先看效果:

再放源代码:

点击查看代码

<!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. 声明全局变量

就是这一段代码:

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

  • 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())。如此往复

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

canvas - 酷炫粒子文字效果代码解析相关推荐

  1. 一分钟搞定触手app主页酷炫滑动切换效果

    代码地址如下: http://www.demodashi.com/demo/12826.html 前言: 前几天在看手机直播的时候,自己就用上了触手app.一进到主页就看上了里面页面切换的效果,自己想 ...

  2. html5霓虹效果代码,纯CSS实现酷炫的霓虹灯效果(附demo)

    最近关注了油管上的 CSS Animation Effects Tutorial 系列,里面介绍了非常多有意思的 CSS 动效.其中第一个就是很酷炫的霓虹灯效果,这里就实现思路做一个简单的记录和分享. ...

  3. Flutter 制作一个具有酷炫液体滑动效果的酷炫入门页面

    本文主要介绍如何使用 Flutter 制作一个具有酷炫液体滑动效果的酷炫入门页面 我将向您展示如何使用 Flutter 制作一个具有酷炫液体滑动效果的酷炫入门页面,所以不用多说,让我们开始吧.在本课程 ...

  4. 赞!15个来自 CodePen 的酷炫 CSS 动画效果

    CodePen 是一个在线的前端代码编辑和展示网站,能够编写代码并即时预览效果.你在上面可以在线分享自己的 Web 作品,也可以欣赏到世界各地的优秀开发者在网页中实现的各种令人惊奇的效果. 今天这篇文 ...

  5. ae制h5文字动画_html5酷炫的文字打字动画特效

    特效描述:html5 酷炫的文字 打字动画特效.html5和css3制作键盘输入文字酷炫的打字动画特效. 代码结构 1. 引入JS 2. HTML代码 0 1 2 3 4 5 6 7 8 9 10 1 ...

  6. HTML5+CSS3小实例:酷炫的文字裂开特效

    HTML5+CSS3实现酷炫的文字裂开特效,鼠标悬停,文字上下裂开,中间显示更多说明,悬停时,其他文字模糊,不说了,看看效果有多酷炫吧! 效果: 源码: <!DOCTYPE html> & ...

  7. canvas 形状碰撞_【案例】如何用html5 制作canvas酷炫的网状图形动画特效

    点击上方[我分享我快乐]→[...]右上角→[设为星标⭐]即可第一时间获取最新设计资源 哈喽大家好,又到了每周二案例环节啦,在教学开始前先插播一条招聘信息~ 高薪招聘 杭州招聘(非技术) 公司背景: ...

  8. android 自定义酷炫ViewPager切换效果

    Android 自定义特效酷炫ViewPager切换效果,当ViewPager切换时,下面的自定义页面标识图也跟着做动画切换, 效果图如下: 具体代码如下: 自定义ViewPager适配器如下: pa ...

  9. 想实现前端酷炫的打字机效果吗

    实现酷炫的打字机效果插件typed.js 前言: 最近在写前端项目的过程中,在逛别人的优质的个人博客的时候,发现一个好玩有趣的东西.看下图,类似在打字的效果,然后又在思考的感觉,感觉整个网页都充满了思 ...

最新文章

  1. STM32低功耗模式下GPIO如何配置最节能?
  2. ICallbackEventHandler 前后台无刷新交互
  3. es6 的数组的方法
  4. nosql怎么使用_使用NoSQL实施实体服务–第5部分:使用云提高自治性
  5. 如何利用计算机英文缩写,【英文缩写】有关计算机的英文都在这儿
  6. C++提高部分_C++模板的局限性_以及用模板具体化来解决模板局限性问题---C++语言工作笔记086
  7. 企业应该了解的ISO27001体系建设指导
  8. 华南理工大学 电力电子技术(王兆安) 期末复习笔记1 第二章第九章
  9. Creo 9.0 如何快速修改CAD坐标系?
  10. 目前人工智能技术,主要有应用于哪些领域?
  11. Walking Robot Simulation
  12. GPU求解粘性不可压流体
  13. excel入门/常用的技巧
  14. 使用 OpenCV 和 Python 识别信用卡号
  15. 手把手教你通过端口映射,轻松搭建Windows远程桌面
  16. 设计模式-创建型模式:原型模式PrototypeModel
  17. java 发送客服消息,Java调用微信客服消息实现发货通知的方法详解
  18. Java中的可变参数使用语法及用途
  19. 程序员都秃顶?Python创始人笑了,养生还得学这门语言
  20. 天猫店群玩法有流量销售额却上不去?可能是转化率出了问题。

热门文章

  1. 前特斯拉核心技术专家谷俊丽加入小鹏汽车
  2. 商务个人邮箱注册申请,商务邮箱登录技巧
  3. 【案例练习】12—50 个从今天就可以开始做起来的小型Web项目
  4. LWN:5.8内核合并窗口第二部分!
  5. 手把手教你使用Ubuntu系统搭建个人不限速私有网盘
  6. 高等数学-对无穷小无穷大的理解
  7. 磁耦合谐振式无线电能传输simulink频率跟踪仿真模型,以及相关模型说明文档
  8. matlab振型分解反应谱法,结构动力分析的MATLAB实现
  9. 计算机考研学校报录比,【22考研】热门院校报录比汇总
  10. springboot内置tomcat和我们平时用的tomcat的区别