使用 Three.js 创建动态粒子效果

news/2025/1/11 23:04:22/

今天,带大家使用粒子实现一个粒子飞毯的效果,我们先来看一下效果。
请添加图片描述

实现

初始化场景

首先创建一个场景,所有 3D 对象都会被添加到这个场景中。

javascript">const scene = new THREE.Scene();

相机和渲染器

配置相机和渲染器来捕捉和显示场景。

相机
  • 使用 PerspectiveCamera 提供广角视图。
渲染器
  • 创建 WebGLRenderer 渲染器,将画布设置为全屏。
javascript">const camera = new THREE.PerspectiveCamera(75,window.innerWidth / window.innerHeight,0.1,100
);
camera.position.z = 13;const renderer = new THREE.WebGLRenderer({canvas: document.querySelector("canvas"),
});
renderer.setSize(window.innerWidth, window.innerHeight);
renderer.setPixelRatio(Math.min(window.devicePixelRatio, 2));

添加粒子

我们将创建一个由数十万个小粒子组成的云,这些粒子可以模拟星空或任何动态环境效果。粒子的创建涉及两个主要组成部分:几何体和材质。

我们使用 BufferGeometry 来存储粒子的位置信息。每个粒子是通过三个坐标(x, y, z)定义的,所以对于 count 个粒子,我们需要一个长度为 count * 3 的数组来存储它们的位置。

javascript">const particlesGeometry = new THREE.BufferGeometry();
const count = 500000; // 粒子数量
const positions = new Float32Array(count * 3); // 存储位置的数组
const colors = new Float32Array(count * 3); // 存储颜色的数组for (let i = 0; i < count * 3; i += 3) {positions[i] = (Math.random() - 0.5) * 10; // x位置positions[i + 1] = (Math.random() - 0.5) * 10; // y位置positions[i + 2] = (Math.random() - 0.5) * 10; // z位置colors[i] = Math.random(); // r颜色colors[i + 1] = Math.random(); // g颜色colors[i + 2] = Math.random(); // b颜色
}

这里的随机数 Math.random() - 0.5 范围为 -0.50.5,乘以 10 来扩展粒子云的空间分布。

粒子材质

粒子材质决定了粒子的视觉效果,如大小、颜色和透明度。我们使用 PointsMaterial 并启用 vertexColors 使每个粒子可以有独立的颜色。

vertexColors: 允许每个粒子使用与其关联的顶点颜色,从而实现每个粒子拥有不同颜色的效果。

注意这里,我们使用了 load 一个透明贴图,这样可以让粒子看起来更加自然。
在这里插入图片描述

javascript">const particlesMaterial = new THREE.PointsMaterial({size: 0.02, // 粒子大小alphaMap: textureLoader.load("/point.png"), // 粒子透明贴图transparent: true, // 开启透明度vertexColors: true, // 使用顶点颜色
});

上面的几个属性,我们在更加详细的介绍一下

  • size:定义了每个粒子点的屏幕上渲染的大小,单位为像素。它控制了粒子的基本尺寸,但实际视觉大小还会受到 sizeAttenuation 属性的影响,后者决定了粒子大小是否随着与摄像机的距离而变化。如果 sizeAttenuation 设置为 true(默认值),粒子大小将随距离变小,模拟现实世界中远处对象看起来更小的透视效果。
  • alphaMap:为每个粒子指定一个纹理贴图,该贴图的透明度(alpha 通道)将被用来影响粒子的透明度。这个功能使得粒子可以具有不均匀的透明度,从而创造更加自然或复杂的视觉效果,如模拟星星或发光粒子等。使用纹理的灰度值来决定相应区域的透明度。
  • 用于开启材质的透明处理。在使用 alphaMap 或者在材质颜色中指定 alpha 值小于 1 的情况下,需要将此属性设置为 true 以确保 Three.js 正确处理透明度。没有启用 transparent 时,即使材质具有透明纹理或颜色值,也会被渲染为完全不透明。
  • vertexColors:用于开启顶点颜色处理。当设置为 true 时,粒子的颜色将由顶点颜色决定,而不是整个粒子使用单一颜色。这使得每个粒子可以具有不同的颜色,从而实现更加丰富多彩的效果。

控制器

使用 OrbitControls 允许用户通过鼠标来旋转和缩放视图。

javascript">const controls = new OrbitControls(camera, renderer.domElement);
controls.enableDamping = true;

动画循环

创建动画循环来实时更新粒子位置和渲染场景。

更新粒子位置

我们在这里使用简单的三角函数来模拟粒子的垂直移动,创建类似波浪的效果。通过三角函数 Math.sin,我们可以为每个粒子创建一个周期性变化的动画效果,使其在 y 轴方向上波动。

javascript">const tick = () => {const elapsedTime = clock.getElapsedTime(); // 获取从开始到现在的时间// 使粒子在y轴方向上根据时间变化for (let i = 0; i < count; i++) {const i3 = i * 3;const x = positions[i3]; // 获取粒子当前x位置positions[i3 + 1] = Math.sin(elapsedTime + x) * 0.3; // 根据时间和x位置计算新的y位置}particlesGeometry.attributes.position.needsUpdate = true; // 通知Three.js位置数据已更新
};

这里,Math.sin(elapsedTime + x) 的使用将每个粒子的动画周期与其 x 位置关联起来,创建更自然的波动效果。乘以 0.3 是为了控制波动的幅度,不让粒子移动得太极端。

渲染场景

在每次动画循环的最后,我们调用 renderer.render 方法来绘制更新后的场景:

javascript">renderer.render(scene, camera);
window.requestAnimationFrame(tick); // 请求下一帧继续动画

代码

github

https://github.com/calmound/threejs-demo/tree/main/feitan

gitee

https://gitee.com/calmound/threejs-demo/tree/main/feitan

学习threejs:www.threejs3d.com


http://www.ppmy.cn/news/1562348.html

相关文章

【如何使用JVM工具进行性能分析?】

如何使用JVM工具进行性能分析&#xff1f; 使用JVM工具进行性能分析可以帮助开发者监控Java应用程序的运行状况&#xff0c;识别潜在的性能瓶颈&#xff0c;并据此进行优化。以下是几种常用的JVM性能分析工具及其详细使用方法&#xff1a; 1. jstat jstat是一个命令行工具&a…

事务,事务的特点,事务并发带来的问题,实现事务管理

1.什么是事务 1、事务管理是企业级应用程序开发中必不可少的技术&#xff0c;用来确保数据的完整性和一致性 2、事务是一系列动作&#xff0c;它们被当作一个独立的工作单元&#xff0c;这些动作要么全部完成&#xff0c;要么全不起作用。 2.事务的特点 ACID 1.原子性&#xf…

51单片机——中断(重点)

学习51单片机的重点及难点主要有中断、定时器、串口等内容&#xff0c;这部分内容一定要认真掌握&#xff0c;这部分没有学好就不能说学会了51单片机 1、中断系统 1.1 概念 中断是为使单片机具有对外部或内部随机发生的事件实时处理而设置的&#xff0c;中断功能的存在&#…

Vue.js 中父组件与子组件通信指南

Vue.js 中父组件与子组件通信指南 引言 在构建复杂的用户界面时&#xff0c;Vue.js 的组件化开发模式允许我们将应用分解为多个小而独立的组件。这些组件之间往往需要进行数据交换或事件触发等交互行为。本文将探讨如何在Vue.js中实现父组件与子组件之间的通信。 使用Props从…

【Docker项目实战】使用Docker部署gallery轻量级图片管理系统

【Docker项目实战】使用Docker部署gallery轻量级图片管理系统 一、SFPG介绍1.1 应用简介1.2 主要特点1.3 主要使用场景二、本次实践规划2.1 本地环境规划2.2 本次实践介绍三、本地环境检查3.1 检查Docker服务状态3.2 检查Docker版本3.3 检查docker compose 版本四、下载gallery…

Spring Cloud服务降级与隔离

Spring Cloud 是一个为微服务架构提供工具集和框架的集合。它在处理微服务之间的通信、配置管理、服务注册与发现等方面提供了许多有用的功能。在实际应用中&#xff0c;为了提高系统的稳定性和可靠性&#xff0c;服务降级&#xff08;fallback&#xff09;和服务隔离&#xff…

2025年01月09日Github流行趋势

1. 项目名称&#xff1a;khoj 项目地址url&#xff1a;https://github.com/khoj-ai/khoj项目语言&#xff1a;Python历史star数&#xff1a;22750今日star数&#xff1a;1272项目维护者&#xff1a;debanjum, sabaimran, MythicalCow, aam-at, eltociear项目简介&#xff1a;你…

ffmpeg7.0 aac转pcm

#pragma once #define __STDC_CONSTANT_MACROS #define _CRT_SECURE_NO_WARNINGSextern "C" { #include "libavcodec/avcodec.h" }//缓冲区大小&#xff08;缓存5帧数据&#xff09; #define AUDIO_INBUF_SIZE 40960 /*name depthu8 8s16 …