当前位置: 首页 > news >正文

使用HTML5 Canvas 实现呼吸粒子球动画效果的原理

在网页开发领域,动画效果能够极大地提升用户体验,让页面变得更加生动有趣。今天,我们深入剖析一个基于 HTML5 Canvas 的 3D 粒子动画 —— 呼吸粒子球。通过详细解读其代码实现,我们将全面了解如何运用 HTML5 的强大功能构建出如此迷人的视觉效果。
效果展示:

用 HTML5 Canvas 和 JavaScript 实现粒子恒星脉动特效

1,整体架构

整个呼吸粒子球效果是 HTML、CSS 和 JavaScript 协同工作的成果。HTML 搭建基础页面结构,CSS 负责样式与背景设置,而 JavaScript 则实现核心动画逻辑。

1.1,HTML结构

<!DOCTYPE html>
<html>
<head><title>呼吸粒子球</title><style>body {margin: 0;overflow: hidden;background: radial-gradient(circle at center, #1a1a2e 0%, #16213e 50%, #0f172a 100%);}canvas {display: block;}</style>
</head>
<body><canvas id="canvas"></canvas><script>// 动画相关的JavaScript代码</script>
</body>
</html>

在 HTML 部分,我们创建了一个canvas元素,它是动画的绘制区域。通过 CSS 设置页面背景为径向渐变,营造深邃空间感,同时将canvas设为块级元素并使其充满整个页面,保证动画能全屏展示。

1.2,JavaScript初始化

const canvas = document.getElementById('canvas');
const ctx = canvas.getContext('2d');// 设置canvas尺寸为全屏
canvas.width = window.innerWidth;
canvas.height = window.innerHeight;

JavaScript 代码首先获取canvas元素,通过getContext(‘2d’)获取 2D 绘图上下文。然后将canvas尺寸设置为当前窗口的宽高,确保动画全屏显示。

2、粒子类的实现

粒子是构成呼吸粒子球的基本元素,通过定义Particle类来管理每个粒子的属性和行为。

2.1,粒子属性初始化

class Particle {constructor() {this.x = 0;this.y = 0;this.z = 0;this.radius = Math.random() * 1.2 + 0.3;this.theta = Math.random() * Math.PI * 2;this.phi = Math.acos((Math.random() * 2) - 1);this.baseR = 200;this.speed = Math.random() * 0.001 + 0.0005;// 根据深度设置颜色this.baseColor = {r: Math.random() * 100 + 155,g: Math.random() * 100 + 155,b: 255};}

在Particle类的构造函数中:

  • x、y、z表示粒子在三维空间中的位置,初始值为 0 ,后续会在动画过程中更新。
  • radius是粒子半径,通过Math.random()函数生成一个随机小数,再乘以 1.2 并加上 0.3,这样粒子半径范围在 0.3 到
    1.5 之间,使得粒子大小呈现多样化。
  • theta和phi是将三维空间点转换为笛卡尔坐标的角度值。theta通过Math.random()生成一个 0(包括)到 2 *
    Math.PI(不包括)之间的随机数,决定粒子在水平方向上的旋转角度;phi通过Math.acos((Math.random() * 2)- 1)生成,Math.random()生成 0(包括)到 1(不包括)之间的随机数,乘以 2 再减 1 得到 - 1(包括)到 1(不包括)之间的值,然后通过Math.acos函数得到 0(包括)到Math.PI(不包括)之间的角度,决定粒子在垂直方向上的角度,确保粒子在球面上均匀分布。
  • baseR是粒子围绕的基准半径,值为 200,用于控制粒子整体分布范围,所有粒子大致围绕这个半径形成一个球体。
  • speed是粒子旋转速度,通过Math.random()生成随机小数,乘以 0.001 再加上 0.0005,速度范围在 0.0005 到
    0.0015 之间,使粒子运动速度有差异。
  • baseColor是粒子基础颜色,红色和绿色分量通过Math.random()生成随机数,范围在 155 到 255 之间,蓝色分量固定为
    255,从而创建出不同蓝色调的粒子。

2.2,粒子位置更新

update(breathe) {this.theta += this.speed;const r = this.baseR * (1 + breathe * 0.2);this.x = r * Math.sin(this.phi) * Math.cos(this.theta);this.y = r * Math.sin(this.phi) * Math.sin(this.theta);this.z = r * Math.cos(this.phi);
}

update方法用于更新粒子位置。breathe参数控制粒子球呼吸效果,它是通过Math.sin函数生成的在 -1 到 1 之间波动的值。随着时间推移,theta不断增加(增加量为this.speed),使粒子围绕球体中心旋转。r的值根据breathe调整,breathe为正时,r增大,粒子向外扩张;breathe为负时,r减小,粒子向内收缩,实现粒子球缩放效果。最后通过三角函数将极坐标转换为笛卡尔坐标,更新粒子的x、y、z位置。

2.3,粒子绘制

draw() {// 增强透视效果const perspective = 800;const scale = perspective / (perspective + this.z);const x2d = canvas.width/2 + this.x * scale;const y2d = canvas.height/2 + this.y * scale;// 根据深度调整颜色和大小const depth = (this.z + this.baseR) / (this.baseR * 2);const alpha = 0.8 + (depth * 0.2);const radiusScale = this.radius * scale * (0.8 + depth * 0.4);// 计算深度相关的颜色const depthFactor = 0.3 + depth * 0.7;const color = {r: this.baseColor.r * depthFactor,g: this.baseColor.g * depthFactor,b: this.baseColor.b * depthFactor};ctx.beginPath();ctx.arc(x2d, y2d, radiusScale, 0, Math.PI * 2);ctx.fillStyle = `rgba(${color.r}, ${color.g}, ${color.b}, ${alpha})`;ctx.fill();// 为前景粒子添加发光效果if (depth > 0.8) {ctx.shadowBlur = 5;ctx.shadowColor = `rgba(${color.r}, ${color.g}, ${color.b}, 0.5)`;} else {ctx.shadowBlur = 0;}
}

draw方法负责将粒子绘制到canvas上。首先,通过透视变换公式计算粒子在 2D 画布上的位置x2d和y2d。这里perspective值设为 800,它模拟人眼到屏幕的距离,scale通过perspective / (perspective + this.z)计算得出,z值越大(粒子越远),scale越小,实现近大远小的透视效果。x2d和y2d通过画布中心坐标加上this.x和this.y乘以scale得到,确定粒子在画布上的实际绘制位置。
然后,根据粒子深度depth调整粒子透明度alpha、半径缩放radiusScale以及颜色。depth通过(this.z + this.baseR) / (this.baseR * 2)计算,范围在 0 到 1 之间,越接近 1 表示粒子越靠近观察者。alpha从 0.8 开始,随着depth增加而增大,使远处粒子更透明;radiusScale不仅考虑了scale,还根据depth进一步缩放,让近处粒子更大;depthFactor用于调整颜色,使近处粒子颜色更鲜艳。
最后,使用ctx.arc方法绘制粒子,并根据depth判断是否为粒子添加发光效果。当depth > 0.8时,为粒子添加模糊半径为 5、颜色为当前粒子颜色半透明的阴影,增强视觉层次感。

3、动画循环与管理

3.1,粒子数组创建与初始化

const particles = [];
const particleCount = 1500;
for(let i = 0; i < particleCount; i++) {particles.push(new Particle());
}

这里创建了一个包含 1500 个粒子的数组particles,通过循环实例化每个粒子并添加到数组中,这些粒子将共同构成呼吸粒子球。

3.2,动画循环函数

let time = 0;
function animate() {// 使用渐变背景增强深度感ctx.fillStyle = 'rgba(0, 0, 0, 0.3)';ctx.fillRect(0, 0, canvas.width, canvas.height);// 按照z轴深度排序粒子particles.sort((a, b) => b.z - a.z);const breathe = Math.sin(time * 0.01);// 更新和绘制所有粒子particles.forEach(particle => {particle.update(breathe);particle.draw();});time++;requestAnimationFrame(animate);
}animate();

animate函数是动画的核心循环。每次循环:

  • 首先使用ctx.fillRect方法填充一个半透明黑色背景(rgba(0, 0, 0,
    0.3)),模拟深度感并清除上一帧绘制内容,为新的一帧绘制做准备。
  • 然后对粒子数组按z轴深度排序,particles.sort((a, b) => b.z -
    a.z)确保距离观察者近的粒子在上面绘制,实现正确遮挡效果,符合现实中物体的遮挡逻辑。
  • 通过Math.sin(time * 0.01)计算breathe的值,time每次循环自增 1,time *0.01作为Math.sin函数的参数,使breathe的值在 -1 到 1 之间周期性变化,控制粒子球呼吸节奏。
  • 遍历粒子数组,依次调用每个粒子的update和draw方法,更新粒子位置并绘制到画布上,让每个粒子都能按照设定的逻辑进行运动和显示。
  • 最后,time自增,并使用requestAnimationFrame方法请求浏览器在下一次重绘前调用animate函数,该方法会根据浏览器的刷新频率来优化动画性能,实现流畅动画效果,通常浏览器刷新频率为60Hz,即每秒 60 帧。

3.3,窗口大小响应

window.addEventListener('resize', () => {canvas.width = window.innerWidth;canvas.height = window.innerHeight;
});

为确保动画在不同窗口大小下正常显示,添加窗口大小变化监听器。当窗口大小改变时,重新设置canvas的宽度和高度为当前窗口的宽高,使动画始终适应屏幕尺寸,保证用户在不同设备和窗口状态下都能获得良好的视觉体验。

4、总结

通过对呼吸粒子球动画效果实现原理的详细分析,我们清晰看到如何利用 HTML5 Canvas 的 2D 绘图功能,结合 JavaScript 的数学计算和动画控制,创建出具有立体感和动态效果的粒子动画。从粒子属性初始化、位置更新、绘制,到动画循环管理以及窗口大小响应,每个环节紧密配合,共同打造出迷人的呼吸粒子球效果。希望本文能助您更好地理解和运用 HTML5 Canvas 进行动画开发,为网页增添更多精彩视觉效果。

完整的源码获取、技术问题讨论,请大家在评论区留言!

相关文章:

使用HTML5 Canvas 实现呼吸粒子球动画效果的原理

在网页开发领域&#xff0c;动画效果能够极大地提升用户体验&#xff0c;让页面变得更加生动有趣。今天&#xff0c;我们深入剖析一个基于 HTML5 Canvas 的 3D 粒子动画 —— 呼吸粒子球。通过详细解读其代码实现&#xff0c;我们将全面了解如何运用 HTML5 的强大功能构建出如此…...

Java 中实体类与操作类分离

目录 一、为啥要把实体类和操作类分开 二、实体类长啥样&#xff0c;怎么用 三、操作类的使命与实现 四、实战演练&#xff1a;实体类与操作类协同工作 五、拓展思考&#xff1a;这种分离带来的好处与进一步优化 六、总结与展望 家人们&#xff0c;今天我想跟你们唠唠我在…...

【STM32HAL-----GPIO】

1. 什么是GPIO&#xff1f;&#xff08;了解&#xff09; 2. STM32 GPIO简介 2.1. GPIO特点 2.2. GPIO电气特性 2.3. GPIO引脚分布图 IO引脚分布特点&#xff1a;按组存在、组数视芯片而定、每组最多16个IO引脚。 3. IO端口基本结构介绍 4. GPIO八种工作模式 4.1. 输入浮空 特…...

Java Web开发高级——单元测试与集成测试

测试是软件开发的重要环节&#xff0c;确保代码质量和功能的正确性。在Spring Boot项目中&#xff0c;单元测试和集成测试是常用的两种测试类型&#xff1a; 单元测试&#xff1a;测试单个模块&#xff08;如类或方法&#xff09;是否按预期工作。集成测试&#xff1a;测试多个…...

编译chromium笔记

编译环境&#xff1a; windows10 powershell7.2.24 git 2.47.1 https://storage.googleapis.com/chrome-infra/depot_tools.zip 配置git git config --global user.name "John Doe" git config --global user.email "jdoegmail.com" git config --global …...

Web开发 -前端部分-CSS3新特性

1 CSS概述 2 CSS3私有前缀 3 CSS3的长度单位 代码实现&#xff1a; <!DOCTYPE html> <html lang"zh-CN"><head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"…...

【基础篇】什么是SQL注入,如何防止?

什么是 SQL 注入&#xff0c;如何防止&#xff1f; SQL 注入&#xff08;SQL Injection&#xff09;是一种常见的网络安全漏洞&#xff0c;它发生在 Web 应用程序中&#xff0c;当恶意用户在输入数据时&#xff0c;将恶意的 SQL 代码插入到输入中&#xff0c;从而导致应用程序…...

Swift语言的数据结构

Swift语言的数据结构 Swift是一种现代化的编程语言&#xff0c;它以安全性、性能和简洁性著称。尽管Swift通常被视为面向对象的语言&#xff0c;但它也支持函数式编程的特性&#xff0c;使得开发者可以以多种方式构建应用程序。在Swift中&#xff0c;数据结构是编程的基础&…...

牛客周赛 Round 77

题目链接&#xff1a;牛客周赛 Round 77 A. 时间表 tag&#xff1a;签到 B. 数独数组 tag&#xff1a;签到 Description&#xff1a;给定n个数&#xff0c;每个数的范围为1-9&#xff0c;问能否经过排列&#xff0c;使其每个长度为9的连续子数组都包含1-9这9个数字。 Sol…...

浅谈云端编辑器,分析其亮点与不足

浅谈云端编辑器&#xff0c;分析其亮点与不足 这个云端编辑器界面可以分为左侧题目筛选栏、中间题目描述与代码编辑区域、右侧AI提示功能三部分。以下是详细的分析&#xff1a; 1. 左侧题目筛选栏 层次结构清晰&#xff1a;左侧栏展示了一个层级结构&#xff0c;题目按主题分…...

web应用引入cookie机制的用途和cookie技术主要包括的内容

web应用引入cookie机制&#xff0c;用于用户跟踪。 &#xff08;1&#xff09;HTTP响应报文中的Cookie头行&#xff1a;set-Cookie &#xff08;2&#xff09;用户浏览器在本地存储、维护和管理的Cookie文件 &#xff08;3&#xff09;HTTP请求报文中的Cookie头行&#xff1a;…...

【HTML+CSS】使用HTML与后端技术连接数据库

目录 一、概述 1.1 HTML前端 1.2 后端技术 1.3 数据库 二、HTML表单示例 三、PHP后端示例 3.1 连接数据库 3.2 接收数据并插入数据库 四、安全性 4.1 防止SQL注入 4.2 数据验证与清洗 五、优化 5.1 索引优化 5.2 查询优化 六、现代Web开发中的最佳实践 6.1 使用…...

「2024·我的成长之路」:年终反思与展望

文章目录 1. 前言2.创作历程2.1 摆烂期2.2 转变期3. 上升期 2. 个人收获3.经验分享4. 展望未来 1. 前言 2025年1月16日&#xff0c;2024年博客之星入围公布&#xff0c;很荣幸获得了这次入围的机会。2024年对我个人是里程碑的一年&#xff0c;是意义非凡的一年&#xff0c;是充…...

C#PaddleOCRSharp使用

using PaddleOCRSharp;namespace PaddleOCRSharpDemo {internal class Program{static void Main(string[] args){//中英文模型V3模型OCRModelConfig config null;//OCR参数OCRParameter oCRParameter new OCRParameter();oCRParameter.cpu_math_library_num_threads 6;//预…...

【Excel】【VBA】Reaction超限点筛选与散点图可视化

【Excel】【VBA】Reaction超限点筛选与散点图可视化 功能概述 这段代码实现了以下功能&#xff1a; 从SAFE输出的结果worksheet通过datalink获取更新数据从指定工作表中读取数据检测超过阈值的数据点生成结果表格并添加格式化创建可视化散点图显示执行时间 流程图 #mermaid-…...

京华春梦,守岁这方烟火人间

文章目录 准备篇温度公共交通人流情况年货采购 文化体验传统庙会博物馆与展览烟花灯会祈福仪式民俗集市现代氛围其他活动 美食盛宴传统美食与特色小吃传统老字号京城新宠特色小吃街多元美食街 准备篇 温度 北京春节期间气温较低&#xff0c;室外通常在零下几度到零上几度之间…...

学Python的人…

学Python的人… 一、Python能干什么&#xff1f; 1.爬虫&#xff1a;前几年&#xff0c;深度学习还没发展起来的时候&#xff0c;书店里Python就和爬虫挂钩&#xff0c;因为Python写爬虫确实方便。 2.数据分析&#xff1a;Python有各种的数据分析库可以方便使用&#xff0…...

WebSocket 和 Socket 的区别

一、协议层次和工作方式 1.1 &#xff09;Socket 1.1.1&#xff09;Socket位于传输层&#xff0c;通常使用TCP或UDP协议 1.1.2&#xff09;提供了一个通用的网络编程接口&#xff0c;允许应用程序通过它发送和接收数据 1.1.3&#xff09;一般需要手动管理连接&#xff0c;错…...

学习ASP.NET Core的身份认证(基于JwtBearer的身份认证6)

重新创建WebApi项目&#xff0c;安装Microsoft.AspNetCore.Authentication.JwtBearer包&#xff0c;将之前JwtBearer测试项目中的初始化函数&#xff0c;jwt配置类、token生成类全部挪到项目中。   重新编写login函数&#xff0c;之前测试Cookie和Session认证时用的函数适合m…...

【SpringBoot】SpringBoot中分页插件(PageHelper)的使用

目录 1.分页概念 2.原生写法 3.PageHelper插件分页查询 3.1 介绍 3.2?使用 3.3 Page对象和PageInf对象 1.分页概念 用户查询的数据不可能一次性全部展示给用户&#xff08;如果用户有一万条数据呢&#xff09;&#xff0c;而是分页展示给用户&#xff0c;这就是分页查询…...

接口测试中缓存处理策略

在接口测试中&#xff0c;缓存处理策略是一个关键环节&#xff0c;直接影响测试结果的准确性和可靠性。合理的缓存处理策略能够确保测试环境的一致性&#xff0c;避免因缓存数据导致的测试偏差。以下是接口测试中常见的缓存处理策略及其详细说明&#xff1a; 一、缓存处理的核…...

理解 MCP 工作流:使用 Ollama 和 LangChain 构建本地 MCP 客户端

&#x1f31f; 什么是 MCP&#xff1f; 模型控制协议 (MCP) 是一种创新的协议&#xff0c;旨在无缝连接 AI 模型与应用程序。 MCP 是一个开源协议&#xff0c;它标准化了我们的 LLM 应用程序连接所需工具和数据源并与之协作的方式。 可以把它想象成你的 AI 模型 和想要使用它…...

【Zephyr 系列 10】实战项目:打造一个蓝牙传感器终端 + 网关系统(完整架构与全栈实现)

🧠关键词:Zephyr、BLE、终端、网关、广播、连接、传感器、数据采集、低功耗、系统集成 📌目标读者:希望基于 Zephyr 构建 BLE 系统架构、实现终端与网关协作、具备产品交付能力的开发者 📊篇幅字数:约 5200 字 ✨ 项目总览 在物联网实际项目中,**“终端 + 网关”**是…...

大模型多显卡多服务器并行计算方法与实践指南

一、分布式训练概述 大规模语言模型的训练通常需要分布式计算技术,以解决单机资源不足的问题。分布式训练主要分为两种模式: 数据并行:将数据分片到不同设备,每个设备拥有完整的模型副本 模型并行:将模型分割到不同设备,每个设备处理部分模型计算 现代大模型训练通常结合…...

工业自动化时代的精准装配革新:迁移科技3D视觉系统如何重塑机器人定位装配

AI3D视觉的工业赋能者 迁移科技成立于2017年&#xff0c;作为行业领先的3D工业相机及视觉系统供应商&#xff0c;累计完成数亿元融资。其核心技术覆盖硬件设计、算法优化及软件集成&#xff0c;通过稳定、易用、高回报的AI3D视觉系统&#xff0c;为汽车、新能源、金属制造等行…...

AspectJ 在 Android 中的完整使用指南

一、环境配置&#xff08;Gradle 7.0 适配&#xff09; 1. 项目级 build.gradle // 注意&#xff1a;沪江插件已停更&#xff0c;推荐官方兼容方案 buildscript {dependencies {classpath org.aspectj:aspectjtools:1.9.9.1 // AspectJ 工具} } 2. 模块级 build.gradle plu…...

Xen Server服务器释放磁盘空间

disk.sh #!/bin/bashcd /run/sr-mount/e54f0646-ae11-0457-b64f-eba4673b824c # 全部虚拟机物理磁盘文件存储 a$(ls -l | awk {print $NF} | cut -d. -f1) # 使用中的虚拟机物理磁盘文件 b$(xe vm-disk-list --multiple | grep uuid | awk {print $NF})printf "%s\n"…...

【无标题】路径问题的革命性重构:基于二维拓扑收缩色动力学模型的零点隧穿理论

路径问题的革命性重构&#xff1a;基于二维拓扑收缩色动力学模型的零点隧穿理论 一、传统路径模型的根本缺陷 在经典正方形路径问题中&#xff08;图1&#xff09;&#xff1a; mermaid graph LR A((A)) --- B((B)) B --- C((C)) C --- D((D)) D --- A A -.- C[无直接路径] B -…...

华为OD最新机试真题-数组组成的最小数字-OD统一考试(B卷)

题目描述 给定一个整型数组,请从该数组中选择3个元素 组成最小数字并输出 (如果数组长度小于3,则选择数组中所有元素来组成最小数字)。 输入描述 行用半角逗号分割的字符串记录的整型数组,0<数组长度<= 100,0<整数的取值范围<= 10000。 输出描述 由3个元素组成…...

【大模型】RankRAG:基于大模型的上下文排序与检索增强生成的统一框架

文章目录 A 论文出处B 背景B.1 背景介绍B.2 问题提出B.3 创新点 C 模型结构C.1 指令微调阶段C.2 排名与生成的总和指令微调阶段C.3 RankRAG推理&#xff1a;检索-重排-生成 D 实验设计E 个人总结 A 论文出处 论文题目&#xff1a;RankRAG&#xff1a;Unifying Context Ranking…...