Three.js中实现Bloom效果及完整示例

目录
  • 在 Three.js 中实现 Bloom 效果
  • 准备工作
  • 实现 Bloom 效果
  • 完整代码
  • 结论

在 Three.js 中实现 Bloom 效果

Bloom 是一种常用于游戏和电影场景中的后期特效,用于模拟相机透镜光晕的效果。它可以使图像看起来更加真实、生动,并且增强视觉体验。在 Three.js 中,我们可以使用 UnrealBloomPassShaderPass 来实现这个效果。本文将介绍如何实现一个简单的 Bloom 效果。

准备工作

首先,我们需要引入 Three.js 库:

<script src="./build/three.min.js"></script>

然后,我们需要在画布中添加一个渲染器,以便能够看到 Three.js 场景:

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

接下来,我们需要创建一个场景和一个相机:

const scene = new THREE.Scene();
const camera = new THREE.PerspectiveCamera(75, window.innerWidth / window.innerHeight, 0.1, 1000);
camera.position.z = 5;

然后,我们需要在场景中添加几何体和材质:

const geometry = new THREE.BoxGeometry(1, 1, 1);
const material = new THREE.MeshBasicMaterial({ color: 0xffffff });
const mesh = new THREE.Mesh(geometry, material);
scene.add(mesh);

实现 Bloom 效果

接下来,我们需要添加 UnrealBloomPassShaderPass 来实现 Bloom 效果。这两个 pass 都是从 EffectComposer 类中派生的。EffectComposer 是 Three.js 中用于渲染后期特效的类。

首先,我们需要引入 UnrealBloomPass 和 ShaderPass:

<script src="./examples/jsm/postprocessing/UnrealBloomPass.js"></script>
<script src="./examples/jsm/postprocessing/ShaderPass.js"></script>

然后,在渲染循环中创建一个 EffectComposer 对象:

const composer = new THREE.EffectComposer(renderer);
const renderPass = new THREE.RenderPass(scene, camera);
composer.addPass(renderPass);
const bloomPass = new THREE.UnrealBloomPass(new THREE.Vector2(window.innerWidth, window.innerHeight), 1.5, 0.4, 0.85);
composer.addPass(bloomPass);
const finalPass = new THREE.ShaderPass(
  new THREE.ShaderMaterial({
    uniforms: {
      baseTexture: { value: null },
      bloomTexture: { value: composer.renderTarget2.texture },
    },
    vertexShader: `
      varying vec2 vUv;
      void main() {
        vUv = uv;
        gl_Position = projectionMatrix * modelViewMatrix * vec4(position, 1.0);
      }
    `,
    fragmentShader: `
      uniform sampler2D baseTexture;
      uniform sampler2D bloomTexture;
      varying vec2 vUv;
      void main() {
        gl_FragColor = texture2D(baseTexture, vUv) + texture2D(bloomTexture, vUv);
      }
    `,
    defines: {},
  }),
  "baseTexture"
);
finalPass.needsSwap = true;
composer.addPass(finalPass);

在上述代码中,我们首先创建了一个 RenderPass 对象,并将其添加到 composer 中。然后,我们创建了一个 UnrealBloomPass 对象,并将其添加到 composer 中。UnrealBloomPass 用于生成 Bloom 纹理。参数 new THREE.Vector2(window.innerWidth, window.innerHeight) 是指 Bloom 纹理的分辨率大小,1.5 是 Bloom 效果的强度,0.4 是阈值,0.85 是模糊程度。

接着,我们创建了一个 ShaderPass 对象,并将其添加到 composer 中。它用于将 Bloom 纹理和场景纹理相加,以生成最终的图像。ShaderMaterial 是用于渲染 ShaderPass 的材质。在这里,我们定义了两个 uniform 变量:baseTexturebloomTexture 分别表示场景纹理和 Bloom 纹理。然后,在顶点着色器中,我们将 vUv 声明为一个 varying 类型的变量,并将其赋值为当前顶点的纹理坐标。在片段着色器中,我们使用 texture2D() 函数获取 baseTexturebloomTexture 的颜色值,并相加起来。最后,我们将 finalPass 添加到 composer 中,并指定需要将结果渲染到屏幕上。

完整代码

const renderer = new THREE.WebGLRenderer({ antialias: true });
renderer.setSize(window.innerWidth, window.innerHeight);
document.body.appendChild(renderer.domElement);
const scene = new THREE.Scene();
const camera = new THREE.PerspectiveCamera(75, window.innerWidth / window.innerHeight, 0.1, 1000);
camera.position.z = 5;
const geometry = new THREE.BoxGeometry(1, 1, 1);
const material = new THREE.MeshBasicMaterial({ color: 0xffffff });
const mesh = new THREE.Mesh(geometry, material);
scene.add(mesh);
const composer = new THREE.EffectComposer(renderer);
const renderPass = new THREE.RenderPass(scene, camera);
composer.addPass(renderPass);
const bloomPass = new THREE.UnrealBloomPass(new THREE.Vector2(window.innerWidth, window.innerHeight), 1.5, 0.4, 0.85);
composer.addPass(bloomPass);
const finalPass = new THREE.ShaderPass(
  new THREE.ShaderMaterial({
    uniforms: {
      baseTexture: { value: null },
      bloomTexture: { value: composer.renderTarget2.texture },
    },
    vertexShader: `
      varying vec2 vUv;
      void main() {
        vUv = uv;
        gl_Position = projectionMatrix * modelViewMatrix * vec4(position, 1.0);
      }
    `,
    fragmentShader: `
      uniform sampler2D baseTexture;
      uniform sampler2D bloomTexture;
      varying vec2 vUv;
      void main() {
        gl_FragColor = texture2D(baseTexture, vUv) + texture2D(bloomTexture, vUv);
      }
    `,
    defines: {},
  }),
  "baseTexture"
);
finalPass.needsSwap = true;
composer.addPass(finalPass);
function animate() {
  requestAnimationFrame(animate);
  mesh.rotation.x += 0.01;
  mesh.rotation.y += 0.02;
  composer.render();
}
animate();

结论

通过添加 UnrealBloomPassShaderPass,我们可以在 Three.js 中实现 Bloom 效果。这种摄影技术能够显著提高场景的真实感和视觉体验。

同时,我们还学习了如何创建 EffectComposer 以及添加和组合不同的 Pass。在这个过程中,我们理解了 Bloom 效果是如何实现的,以及如何使用 GLSL 编写自定义着色器和 Uniform 变量来扩展 Three.js 的渲染功能。

最后,需要注意的是,在使用 Bloom 效果时,我们需要计算和处理额外的像素信息,因此可能会对性能产生一定的影响。通常情况下,我们可以通过调整 resolutionstrengthradius 等参数来平衡效果和性能。

以上就是Three.js中实现Bloom效果及完整示例的详细内容,更多关于Three.js实现Bloom效果的资料请关注我们其它相关文章!

(0)

相关推荐

  • Three.js材质Material类型示例详解

    目录 1. 什么是材质Material? 2. 常见材质类型 2.1 基础材质(MeshBasicMaterial) 2.2 标准材质(MeshStandardMaterial) 2.3 Lambert 材质(MeshLambertMaterial) 2.4 Phong 材质(MeshPhongMaterial) 2.5 可编程着色器材质(ShaderMaterial) 3. 纹理映射 4.材质的主要属性 5. 总结 1. 什么是材质Material? 在 Three.js 中,材质(Mater

  • Three.js 中的屏幕空间环境光遮蔽SSAO

    目录 简介 原理 实现 总结 简介 在计算机图形学中,环境光遮蔽(Ambient Occlusion)是一种可以增强场景深度感.模拟阴影效果的技术.在 Three.js 中,我们可以使用屏幕空间环境光遮蔽(Screen Space Ambient Occlusion,简称 SSAO)来实现这个效果.本篇文章将详细介绍 SSAO 技术的原理和在 Three.js 中的应用. 原理 SSAO 技术是基于屏幕空间的,即只需要对当前帧进行处理,而不需要重新渲染场景.它的基本思想是根据场景中各个像素点周围

  • Three.js概述和基础知识学习

    目录 1.Three.js简介 2.Three.js的历史 3.Three.js的应用 4.基础知识 4.1 场景(Scene) 4.2 相机(Camera) 4.3 渲染器(Renderer) 4.4 几何体(Geometry) 4.5 材质(Material) 4.6 网格(Mesh) 4.7 光源(Light) 5.总结 1.Three.js简介 Three.js是一个基于JavaScript编写的开源3D图形库,利用WebGL技术在网页上渲染3D图形.它提供了许多高级功能,如几何体.纹理

  • Three.js PBR物理渲染属性及使用介绍

    目录 详解 Three.js PBR 物理渲染 什么是 PBR? 如何启用 PBR? PBR 的属性 颜色(color) 金属度(metalness) 粗糙度(roughness) 折射率(refractionRatio) 透明度和透明度映射 环境映射(environmentMap) 总结 详解 Three.js PBR 物理渲染 Three.js 是一个流行的基于 WebGL 的 JavaScript 库,专门用于创建和运行三维动画和游戏.其中很关键的一部分是物理渲染(PBR).本文将深入探讨

  • js 实现div拖拽拉伸完整示例

    目录 前言 HTML CSS JS 前言 今天和大家分享一下.如何用js实现div拖拽拉伸等功能.功能比较简单,我就不赘述了.直接上代码. HTML <div class="resize" data-key="drag"> <img src="../images/liya.jpg" alt=""> <div class="line line-n" data-key="

  • JS实现TITLE悬停长久显示效果完整示例

    本文实例讲述了JS实现TITLE悬停长久显示效果.分享给大家供大家参考,具体如下: <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd"> <html> <head> <meta http-equiv="Content-Type" content="

  • js中创建对象的几种方式示例介绍

    JavaScript中的所有事物都是对象:字符串.数组.数值.函数等.JS中并没有类的概念, 但我们可以利用JS的语法特征,以类的思想来创建对象. 原始方法 复制代码 代码如下: <script type="text/javascript"> var obj = new Object(); obj.name = "Koji"; //为对象添加属性 obj.age = 21; obj.showName = function(){ //为对象添加方法 ale

  • 在vue.js中抽出公共代码的方法示例

    前言 当我们在使用vue构建中大型项目时,通常会遇到某些经常用的方法以及属性,比如说搭建一个员工管理系统,请求的url需要一个共同的前缀,或者在某几个view中需要用到时间,这个时间是通过某方法格式化之后的等等,如果每次用到都写共同的代码,那样如果之后有变动的话维护起来会非常麻烦. 所以我们就得想办法抽出公共代码,因为vue是组件化开发,我们就会很自然的与es6的module模块化联系到一起.其实当我们在搭建项目结构时就应该先提前埋下伏笔,有一个util文件夹,里面放的就是我们要写的公共代码,其

  • vue.js中实现登录控制的方法示例

    本文实例讲述了vue.js中实现登录控制的方法.分享给大家供大家参考,具体如下: vue中使用vue-router实现登录的控制在做后台管理系统中很常见,但是不想之前熟悉的流程那样,不过只要大家理解vue-router的使用也是很好实现的. 首先我们需要编写登录页面和主页面: <template> <div class="login"> <table width="100%" height="100%"> &l

  • node.js中 redis 的安装和基本操作示例

    本文实例讲述了node.js中 redis 的安装和基本操作.分享给大家供大家参考,具体如下: 一.win下安装redis https://github.com/MicrosoftArchive/redis/releases 下载Redis-x64-3.2.100.zip,然后解压,放到自定义目录. 然后打开命令行工具,进入到该目录下,运行安装redis服务. redis-server.exe --service-install redis.windows-service.conf --logl

  • JS document内容及样式操作完整示例

    本文实例讲述了JS document内容及样式操作.分享给大家供大家参考,具体如下: <html> <head> <title>js-documnet元素内容和样式操作</title> <meta charset="UTF-8"/> <script type="text/javascript"> // 单标签属性操作 // 固定属性的操作 function testOper1(){ // 获取对

  • js中onload与onunload的使用示例

    引言: 今天周末没事,就想起前面自己做的一个B2C的电子商务平台,还有些一些地方没有完善,就想着完善,嗯,问题是这样的,在电子商务平台中有这样一个场景,我将购物车放置在Session中,使其在整个购物的过程中都能从Session中得到购物车模型,我在购物车某型中有的商品都会在数据库中减少其购物车中购买的数量,但是如果我关闭窗口,怎样让Session中的购物车模型中的商品数量添加到数据库中,于是我查找了GOOGLE.百度,得到的第一个提示,就是:关闭窗口自动清除Session,于是找到的第一个方法

  • JS动态图片的实现方法完整示例

    本文实例讲述了JS动态图片的实现方法.分享给大家供大家参考,具体如下: <html> <head> <meta charset="UTF-8"> <title>js-是动态实现图片显示</title> <script type="text/javascript"> // 设置图片的动态效果 function photoOper(ph,photo){ // 获取图片对象 // var ph=doc

  • JS实现滑动拼图验证功能完整示例

    本文实例讲述了JS实现滑动拼图验证功能.分享给大家供大家参考,具体如下: 先看一下效果图: 设置画布滑块属性 const l = 42, // 滑块边长 r = 10, // 滑块半径 w = 310, // canvas宽度 h = 155, // canvas高度 PI = Math.PI const L = l + r * 2 // 滑块实际边长 设置背景图片: 图片链接地址可以自行更换 function getRandomImg() { return 'https://picsum.ph

随机推荐