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

Three.js 粒子泡泡教程

粒子泡泡 ·Bubble· ▶ 在线运行案例

  • 案例合集:三维可视化功能案例(threehub.cn)
  • 开源仓库github地址:https://github.com/z2586300277/three-cesium-examples
  • 400个案例代码:网盘链接

你将学到什么

  • ShaderMaterial 自定义着色器实现核心视觉效果
  • OrbitControls 相机轨道交互
  • BufferGeometry 自定义顶点/索引数据
  • requestAnimationFrame渲染循环与resize自适应

效果说明

本案例演示粒子泡泡效果:基于 WebGL 实现「粒子泡泡」可视化效果,附完整可运行源码;核心用到 ShaderMaterial、OrbitControls、BufferGeometry。建议先打开文首在线案例查看动态画面,再对照下方源码逐步理解。

核心概念

  • Scene / Camera / WebGLRenderer构成最小渲染闭环;大场景可开logarithmicDepthBuffer缓解 Z-fighting。
  • ShaderMaterial通过uniforms+ 自定义 GLSL 控制逐像素/逐点效果;透明粒子常配合depthTest: false
  • OrbitControls提供轨道旋转/缩放;开启enableDamping后需在 animate 中controls.update()

实现步骤

  • 搭建 Scene、PerspectiveCamera、WebGLRenderer,挂载 canvas 并处理resize
  • 定义 uniforms / onBeforeCompile 或 ShaderMaterial,编写 GLSL 与材质参数
  • 创建 OrbitControls(及 Raycaster 等交互控件,若源码包含)
  • requestAnimationFrame循环中更新状态并 render(Cesium 为viewer.render或自动渲染)
  • 代码要点

    import * as THREE from "three";

    import { OrbitControls } from "three/addons/controls/OrbitControls.js"; import { GUI } from "three/addons/libs/lil-gui.module.min.js"; const { Color, ShaderMaterial, BufferGeometry, Points, Vector3, Float32BufferAttribute, } = THREE

    class Bubble extends Points {

    _count = 0;

    _size = 10;

    _color = "#ff0000";

    _speed = 0.8;

    _maxHeight = 10;

    _radius = 10;

    _radius2 = 10;

    isBubble = true;

    _emitter = "cone"

    _emitters = [ "cone", "cylinder", "box", "sphere" ];

    type = "Bubble";

    /**

    • * @param emitter {string}
    */ set emitter(emitter) { if (this._emitters.indexOf(emitter) !== -1) { this._emitter = emitter; this.setPoints(); this.uniforms.emitter.value = this._emitters.indexOf(this._emitter); } }

    set radius(radius) { this._radius = radius; this.setPoints(); }

    set radius2(radius) { this._radius2 = radius; this.setPoints(); }

    set count(count) { this._count = count; this.setPoints(); }

    set maxHeight(maxHeight) { this._maxHeight = maxHeight; this.uniforms.maxHeight.value = maxHeight || 10; this.setPoints(); }

    set speed(speed) { this._speed = speed; this.uniforms.speed.value = speed; }

    set size(size) { this._size = size; this.uniforms.size.value = size; }

    set color(color) { this._color = color; this.uniforms.color.value = new Color(color); }

    get radius() { return this._radius; }

    get emitter() { return this._emitter; }

    get emitters() { return this._emitters; }

    get count() { return this._count; }

    get speed() { return this._speed; }

    get size() { return this._size; }

    get maxHeight() { return this._maxHeight; }

    vertexShader =varying vec2 vUv; //创建uv变量,用于给片元着色器传递uv uniform float u_time; //从前端接收u_time uniform float speed; //从前端接收speed uniform float size; //从前端接收size uniform float emitter;//发射器类型 uniform float maxHeight;//从前端接收maxHeight attribute float data1; attribute float data2; void main(){ //从顶点着色器取uv给片元着色器 vUv = vec2(uv.x,uv.y); //用一个变量复制当前位置,用于计算最终位置 vec3 u_position = position; if(emitter < 3.0){ //粒子y轴变化 float t = fract( u_time * speed + position.y/maxHeight); //对u_time取小数,使得数据一直在0~1按顺序变化,减y轴是用于移动图像 u_position.y = t * maxHeight; //圆锥模式 if(emitter == 0.0){ u_position.x = cos(data2)data1u_position.y/maxHeight; u_position.z = sin(data2)data1u_position.y/maxHeight; } //圆柱模式 if(emitter == 1.0){ u_position.x = cos(data2) * data1; u_position.z = sin(data2) * data1; } //立方体模式 if(emitter == 2.0){ u_position.x = data1; u_position.z = data2; } }else{ //球体模式 if(emitter == 3.0){ float r = length(u_position); float t = fract(u_time * speed + r/maxHeight); r = t * maxHeight; u_position.x = rsin(data1)cos(data2); u_position.y = r * sin(data2); u_position.z = rcos(data1)cos(data2); } } //设定粒子大小 gl_PointSize = size; //固定写法,将最终计算完成的顶点位置传递给显卡并交由显卡计算 gl_Position = projectionMatrixmodelViewMatrixvec4( u_position, 1.0 ); };

    fragmentShader =uniform vec3 color;//从前端接收颜色 varying vec2 vUv; //获取从顶点着色器传递过来的uv void main(){ //气泡计算公式, 根据中心到边缘的距离设定透明度 float dis = pow( distance( gl_PointCoord , vec2(0.5,0.5) ) ,2.0); //透明度高于0.2的部分舍弃,用于舍弃边缘方形的区域 if(dis > 0.2){ discard; } //固定写法,将计算后的颜色渲染出来 gl_FragColor = vec4(color,dis * 2.0); };

    uniforms = { u_time: { value: 0 }, speed: { value: 0.8 }, size: { value: 10 }, color: { value: new Color("#2acdf9") }, maxHeight: { value: 10 }, emitter: { value: this._emitters.indexOf(this._emitter) } };

    /**

    • * @param config {Object}
    • count:粒子数量
    • xArea:x随机范围
    • zArea:z随机范围
    • maxHeight:最大升腾高度
    • speed:升腾速度
    • size:气泡大小
    • color:气泡颜色
    */ constructor(config) { super(); this.material = this.initLineMaterial(); this.geometry = new BufferGeometry(); config = config || { color: "#2acdf9" }; this.uniforms.speed.value = config.speed || this.uniforms.speed.value this.uniforms.size.value = config.size || this.uniforms.size.value this.uniforms.maxHeight.value = config.maxHeight || this.uniforms.maxHeight.value this.uniforms.color.value = new Color(config.color); this._count = config.count || 100; this.setPoints(); }

    initLineMaterial = () => { return new ShaderMaterial({ uniforms: this.uniforms, vertexShader: this.vertexShader, fragmentShader: this.fragmentShader, transparent: true, }); }

    setPoints = () => { let points = []; let data1Array = []; let data2Array = []; for (let i = 0; i < this._count; i++) {

    //圆柱模式下,生成的数据用于半径和角度 if (this._emitter === "cone" || this._emitter === "cylinder") { let data1 = Math.random() * this._radius; let data2 = Math.PI2Math.random(); data1Array.push(data1); data2Array.push(data2); } if (this._emitter === "box") { let data1 = Math.random() * this._radius - this._radius / 2; let data2 = Math.random() * this._radius2 - this._radius2 / 2; data1Array.push(data1); data2Array.push(data2); } if (this._emitter === "sphere") { let data1 = Math.PI2Math.random(); let data2 = Math.PI2Math.random(); data1Array.push(data1); data2Array.push(data2); } let y = i / this._count * this.uniforms.maxHeight.value; points.push(new Vector3(0, y, 0)); } this.geometry.setFromPoints(points); this.geometry.setAttribute('data1', new Float32BufferAttribute(data1Array, 1)); this.geometry.setAttribute('data2', new Float32BufferAttribute(data2Array, 1)); console.log(this.geometry); }

    onBeforeRender = () => { this.uniforms.u_time.value += 0.01; } }

    window.addEventListener('load', e => { init(); addMesh(); render(); })

    let scene, renderer, camera; let orbit;

    let gui = new GUI();

    let bubble;

    function init() {

    scene = new THREE.Scene(); renderer = new THREE.WebGLRenderer({ antialias: true }); renderer.setSize(window.innerWidth, window.innerHeight); document.body.append(renderer.domElement);

    camera = new THREE.PerspectiveCamera(50, window.innerWidth / window.innerHeight, 0.1, 2000); camera.add(new THREE.PointLight()); camera.position.set(10, 10, 10); scene.add(camera);

    orbit = new OrbitControls(camera, renderer.domElement); orbit.enableDamping = true;

    scene.add(new THREE.AxesHelper(10)); }

    function addMesh() { let size = 1; let geometry = new THREE.BoxGeometry(size, size, size).translate(0, size / 2, 0); let material = new THREE.MeshBasicMaterial({ color: "#00ffff", transparent: true, opacity: 0.5 }); let mesh = new THREE.Mesh(geometry, material); scene.add(mesh);

    bubble = new Bubble({ speed: 0.8, size: 30, maxHeight: 10, color: "#ff0000" }); scene.add(bubble);

    let params = { speed: 0.8, size: 30, maxHeight: 10, color: "#1acdf9", count: 100, radius: 10, rotateSpeed: 0.01, backgroundColor: "#000000", emitter: "cone", emitterOptions: ["cone", "cylinder", "box", "sphere"] };

    gui.add(params, "speed", -2, 2).step(0.01).onChange(v => bubble.speed = v); gui.add(params, "size").onChange(v => bubble.size = v); gui.add(params, "maxHeight").onChange(v => bubble.maxHeight = v); gui.addColor(params, "color").onChange(v => bubble.color = v); gui.add(params, "count").onChange(v => bubble.count = v); gui.add(params, "radius").onChange(v => bubble.radius = v); gui.add(params, "rotateSpeed").onChange(v => bubble.rotateSpeed = v); gui.addColor(params, 'backgroundColor').name('背景色').onChange(v => { scene.background = new THREE.Color(v); }); gui.add(params, 'emitter', params.emitterOptions).name('粒子发射方式').onChange(v => { bubble.emitter = v; }); }

    function render() { renderer.render(scene, camera); orbit.update(); requestAnimationFrame(render); }

    完整源码:GitHub

    小结

    • 本文提供粒子泡泡完整 Three.js 源码与在线 Demo,建议先运行案例再改 uniform/参数做二次实验
    • 更多 Three.js 实战案例见 three-cesium-examples 合集 与 GitHub 开源仓库
http://www.cnnetsun.cn/news/3121476.html

相关文章:

  • 01-GitHub基础认识
  • ROS 2 的发布/订阅通信验证
  • 二维码批量扫码设备硬件选型与并行解码技术方案研究
  • 未来展望:BiSheng JDK 17路线图与OpenJDK社区贡献计划终极指南
  • 特斯拉Cybercab无方向盘路测曝光!20个月落地,成本优势能否弯道超车?
  • SPI EEPROM与PIC微控制器的嵌入式存储方案设计
  • 项目进度实时监控与资源优化:项目制服务解决方案落地方法论
  • 【沈阳师范大学本科毕业论文】基于Spring boot的青少年 研学网站的设计与实现
  • 超市小程序制作,线上超市小程序开发超市小程序制作
  • 用GPT-5.5重构遗留项目:一套可复用的迁移脚本分享(附避坑指南与教程)
  • USB3.0总线高速数据采集卡,8通道、16位分辨率、5MHz同步采样,程控增益±10V、±5V、±2V、±1V
  • 2026楼宇自控品牌推荐 这些楼宇自控厂家实力太赞了!
  • Gemini 3.5 如何辅助写代码?生成代码、解释逻辑与调试思路使用指南
  • 基于STM32单片机智能书桌 坐姿提醒 智能台灯语音识别控制系统1(设计源文件+万字报告+讲解)(支持资料、图片参考_相关定制)_
  • 自建商城还是上 SaaS?企业电商系统选型的真实成本账(一篇讲清 TCO)
  • 缠论量化交易实战指南:从理论到策略的完整实现
  • 打造你的终极数字伙伴:用DyberPet桌面宠物框架重新定义桌面互动体验
  • 解锁Python :2025-2026出版新书的《人月神话》引用(9)
  • 如何在Windows上轻松安装虚拟游戏控制器驱动:ViGEmBus完整指南
  • Windows主题缓存
  • Appium会话启动失败:系统性排查与解决方案全解析
  • 自动驾驶三条技术路线的本质差异与场景适配
  • 5 分钟上手 Kimi Work:安装、配置、跑通第一个任务
  • 10个免费Adobe Illustrator自动化脚本:设计师必备的效率革命指南 [特殊字符]
  • 2026年7月北京家具回收机构哪家靠谱?大红酸枝/黄花梨/缅甸花梨实木家具回收服务商甄选
  • 3步解锁iOS设备潜能:palera1n越狱工具终极指南
  • 这个神器让你秒变黑客(非常详细),零基础入门到精通,看这一篇就够了
  • 【文献速递】Cre-LoxP+TurboID:研究动物体内分泌蛋白组邻近标记新工具
  • 量子-经典混合Benders分解算法在电力系统优化中的应用
  • Kimi K2.6真实开发测评:国产AI编程能力实战深度解析