Beautiful and mind-bending effects with WebGL Render Targets
March 14, 2023 / 23 min read
Last Updated: June 8, 2023As I'm continuously working on sharpening my skills for creating beautiful and interactive 3D experiences on the web, I get constantly exposed to new tools and techniques, from the diverse React Three Fiber scenes to gorgeous shader materials. However, there's one Three.js tool that's often a common denominator to all these fascinating projects that I keep running into: Render Targets.
This term may seem scary, boring, or a bit opaque at first, but once you get familiar with it, it becomes one of the most versatile tools for achieving greatness in any 3D scene! You may have already seen render targets mentioned or used in some of my writing/projects for a large range of use cases, such as making a transparent shader materials or rendering 100's of thousands of particles without your computer breaking a sweat. There's, however, a lot more they can do, and since learning about render targets, I've used them in countless other scenes to create truly beautiful โจ yet mind-bending ๐ effects.
In this article, I want to share some of my favorite use cases for render targets. Whether it's for building infinity mirrors, portals to other scenes, optical illusions, and post-processing/transition effects, you'll learn how to leverage this amazing tool to push your React Three projects above and beyond. ๐ช
Render Targets in React Three Fiber
Before we jump into some of those mind-bending 3D scenes I mentioned above, let's get us up to speed on what a render target or WebGLRenderTarget
is:
A tool to render a scene into an offscreen buffer.
That is the simplest definition I could come up with ๐ . In a nutshell, it lets you take a snapshot of a scene by rendering it and gives you access to its pixel content as a texture for later use. Here are some of the use cases where you may encounter render targets:
- Post-processing effects: we can modify that texture using a shader material and GLSL's
texture2D
function. - Apply scenes as textures: we can take a snapshot of a 3D scene and render it as a texture on a mesh. That lets us create different kinds of effects like the transparency of glass or a portal to another world.
- Store data: we can store anything in a render target, even if not necessarily meant to be "rendered" visually on the screen. For example, in particle-based scenes, we can use a render target to store the position of an entire set of particles that require large amounts of computation, as it's been showcased in The magical world of Particles with React Three Fiber and Shaders.
Whether it's for a Three.js or React Three Fiber, creating a render target is straightforward: all you need to use is the WebGLRenderTarget
class.
1const renderTarget = new THREE.WebGLRenderTarget();
However, if you use the @react-three/drei
library in combination with React Three Fiber, you also get a hook named useFBO
that gives you a more concise way to create a WebGLRenderTarget
, with things like a better interface, smart defaults, and memoization set up out of the box. All the demos and code snippets throughout this article will feature this hook.
Now, when it comes to using render targets in a React Three Fiber scene, it's luckily pretty simple. Once instantiated, every render target-related manipulation will be done within the useFrame
hook i.e. the render loop.
Using a render target within the useFrame hook
1const mainRenderTarget = useFBO();23useFrame((state) => {4const { gl, scene, camera } = state;56gl.setRenderTarget(mainRenderTarget);7gl.render(scene, camera);89mesh.current.material.map = mainRenderTarget.texture;1011gl.setRenderTarget(null);12});
This first code snippet features four main steps to set up and use our first render target:
- We create our render target with
useFBO
. - We set the render target to
mainRenderTarget
. - We render our scene with the current camera onto the render target.
- We map the content of the render target as a texture on a plane.
- We set the render target back to
null
, the default render target.
The code playground below showcases what a scene using the code snippet above would look like:
After seeing this first example you might ask yourself Why is our scene mapped onto the plane moving along as we rotate the camera? or Why do we see an empty plane inside the mapped texture?. Here's a diagram that explains it all:
- We set our render target.
- We first take a snapshot of our scene including the empty plane with its default camera.
- We map the render target's texture, pixel by pixel, onto the plane. Notice that we have a snapshot of the scene before we filled the plane with our texture, hence why it appears empty.
- We do that for every frame ๐คฏ 60 times per second (or 120 if you have a top-of-the-line display).
That means that any updates to the scene, like the position of the camera, will be taken into account and reproduced onto the plane since we run this loop of taking a snapshot and mapping it continuously.
Now, what if we wanted to fix the empty plane issue? There's actually an easy fix to this! We just need to introduce a second render target that takes a snapshot of the first one and maps the resulting texture of this up-to-date snapshot onto the plane. The result is an infinite mirror of the scene rendering itself over and over again inside itself โจ.
Offscreen scenes
We now know how to use render targets to take snapshots of the current scene and use the result as a texture. If we leverage this technique with the very neat createPortal
utility function from @react-three/fiber
, we can start creating some effects that are close to magic ๐ช!
A quick introduction to createPortal
Much like its counterpart in React that lets you render components into a different part of the DOM, with @react-three/fiber
's createPortal
, we can render 3D objects into another scene.
Example usage of the createPortal function
1import { Canvas, createPortal } from '@react-three/fiber';2import * as THREE from 'three';34const Scene = () => {5const otherScene = new THREE.Scene();67return (8<>9<mesh>10<planeGeometry args={[2, 2]} />11<meshBasicMaterial />12</mesh>13{createPortal(14<mesh>15<sphereGeometry args={[1, 64]} />16<meshBasicMaterial />17</mesh>,18otherScene19)}20</>21);22};
With the code snippet above, we render a sphere in a dedicated scene object called otherScene
within createPortal
. This scene is rendered offscreen and thus will not be visible in the final render of this component. This has many potential applications, like rendering HUDs or rendering different views angles of the same scene.
createPortal in combination with render target
You may be wondering why createPortal
has anything to do with render targets. We can use them together to render any portalled/offscreen scene to get its texture without even having it visible on the screen in the first place! With this combination, we can achieve one of the first mind-bending use cases (or at least it was to me when I first encountered it ๐
) of render targets: render a scene, within a scene.
Using createPortal in combination with render targets
1const Scene = () => {2const mesh = useRef();3const otherCamera = useRef();4const otherScene = new THREE.Scene();56const renderTarget = useFBO();78useFrame((state) => {9const { gl } = state;1011gl.setRenderTarget(renderTarget);12gl.render(otherScene, otherCamera.current);1314mesh.current.material.map = renderTarget.texture;1516gl.setRenderTarget(null);17});1819return (20<>21<PerspectiveCamera manual ref={otherCamera} aspect={1.5 / 1} />22<mesh ref={mesh}>23<planeGeometry args={[3, 2]} />24<meshBasicMaterial />25</mesh>26{createPortal(27<mesh>28<sphereGeometry args={[1, 64]} />29<meshBasicMaterial />30</mesh>,31otherScene32)}33</>34);35};
There's only one change in the render loop of this example compared to the first one: instead of calling gl.render(scene, camera)
, we called gl.render(otherScene, otherCamera)
. Thus the resulting texture of this render target is of the portalled scene and its custom camera!
This code playground uses the same combination of render target and createPortal
we just saw. You will also notice that if you drag the scene left or right, we notice a little parallax effect that gives a sense of depth to our portal.
This is a nice trick I discovered looking at some of @0xca0a's demos that only requires a single line of code:
1otherCamera.current.matrixWorldInverse.copy(camera.matrixWorldInverse);
Here, we copy the matrix representing the position and orientation of our default camera onto our portalled camera. That has the result of mimicking any camera movement of the default scene in the portalled scene while having the ability to have a custom camera with different settings.
Using shader materials
In all the examples we've seen so far, we used meshBasicMaterial
's map
prop to map our render target's texture onto a mesh for simplicity. However, there are some use cases where you'd want to have more control over how to map this texture (we'll see some examples for that in the last part of this article), and for that, knowing the shaderMaterial
equivalent can be handy.
The only thing that this map
prop is doing is mapping each pixel of the 2D render target's texture onto our 3D mesh which is what we generally refer to as UV mapping.
The GLSL code to do that is luckily concise and can serve as a base for many shader materials requiring mapping a scene as a texture:
Simple fragment shader to sample a texture with uv mapping
1// uTexture is our render target's texture2uniform sample2D uTexture;3varying vec2 vUv;45void main() {6vec2 uv = vUv;7vec4 color = texture2D(uTexture, uv);89gl_FragColor = color;10}
Using Drei's RenderTexture component
Now that we've seen the manual way of achieving this effect, I can actually tell you a secret: there's a shorter/faster way to do all this using @react-three/drei
's RenderTexture component.
Using RenderTexture to map a portalled scene onto a mesh
1const Scene = () => {2return (3<mesh>4<planeGeometry />5<meshBasicMaterial>6<RenderTexture attach="map">7<PerspectiveCamera makeDefault manual aspect={1.5 / 1} />8<mesh>9<sphereGeometry args={[1, 64]} />10<meshBasicMaterial />11</mesh>12</RenderTexture>13</meshBasicMaterial>14</mesh>15);16};
I know we could have taken this route from the get-go, but at least now, you can tell that you know the inner workings of this very well-designed component ๐.
Render targets with screen coordinates
So far, we only looked at examples using render targets for applying textures to a 3D object using UV mapping, i.e. mapping all the pixels of the portalled scene onto a mesh. Moreover, if you visualized the last code playground with the renderBox
options turned on, you probably noticed that the portalled scene was rendered on each face of the cube: our mapping was following the UV coordinates of that specific geometry.
We may have use cases where we don't want those effects, and that's where using screen coordinates instead of uv coordinates comes into play.
UV coordinates VS screen coordinates.
To illustrate the nuance between these two, let's take a look at the diagram below:
You can see that the main difference here lies in how we're sampling our texture onto our 3D object: instead of making the entire scene fit within the mesh, we render it "in place" as if the mesh was some "mask" revealing another scene behind the scene. Code-wise, there's only two lines to modify in our shader to use screen coordinates:
Simple fragment shader to sample a texture using screen coordinates
1// winResolution contains the width/height of the canvas2uniform vec2 winResolution;3uniform sampler2D uTexture;45void main() {6vec2 uv = gl_FragCoord.xy / winResolution.xy;7vec4 color = texture2D(uTexture, uv);89gl_FragColor = color;10}
which yields the following result if we update our last example with this change:
If you've read my article Refraction, dispersion, and other shader light effects this is the technique I used to make a mesh transparent: by mapping the scene surrounding the mesh onto it as a texture with screen coordinates.
Now you may wonder what are other applications for using screen coordinates with render targets ๐; that's why the following two parts feature two of my favorite scenes to show you what can be achieved when combining these two concepts!
Swapping materials and geometries in the render loop
Since all our render target magic happens within the useFrame
hook that executes its callback on every frame, it's easy to swap materials or even geometries between render targets to create optical illusions for the viewer.
The code snippet bellow details how you can achieve this effect:
Hidding/showing mesh and swapping materials in the render loop
1//...23const mesh1 = useRef();4const mesh2 = useRef();5const mesh3 = useRef();6const mesh4 = useRef();7const lens = useRef();8const renderTarget = useFBO();910useFrame((state) => {11const { gl } = state;1213const oldMaterialMesh3 = mesh3.current.material;14const oldMaterialMesh4 = mesh4.current.material;1516// Hide mesh 117mesh1.current.visible = false;18// Show mesh 219mesh2.current.visible = true;2021// Swap the materials of mesh3 and mesh422mesh3.current.material = new THREE.MeshBasicMaterial();23mesh3.current.material.color = new THREE.Color("#000000");24mesh3.current.material.wireframe = true;2526mesh4.current.material = new THREE.MeshBasicMaterial();27mesh4.current.material.color = new THREE.Color("#000000");28mesh4.current.material.wireframe = true;2930gl.setRenderTarget(renderTarget);31gl.render(scene, camera);3233// Pass the resulting texture to a shader material34lens.current.material.uniforms.uTexture.value = renderTarget.texture;3536// Show mesh 137mesh1.current.visible = true;38// Hide mesh 239mesh2.current.visible = false;4041// Restore the materials of mesh3 and mesh442mesh3.current.material = oldMaterialMesh3;43mesh3.current.material.wireframe = false;4445mesh4.current.material = oldMaterialMesh4;46mesh4.current.material.wireframe = false;4748gl.setRenderTarget(null);49}5051//...
- There are four meshes in this scene: one dodecahedron (mesh1), one torus (mesh2), and two spheres (mesh3 and mesh4).
- At the beginning of the render loop, we hide mesh1 and set the material of mesh3 and mesh4 to render as a wireframe
- We take a snapshot of that scene in our render target.
- We pass the texture of the render target to another mesh called
lens
. - We then show mesh1, hide mesh2, and restore the material of mesh3 and mesh4.
If we use this render loop and add a few lines of code to make the lens
follow the viewer's cursor, we get a pretty sweet optical illusion: through the lens we can see an alternative version of our scene ๐คฏ.
I absolutely love this example of using render targets:
- it's clever.
- it doesn't introduce a lot of code.
- it is not very complicated if you are familiar with the basics we've covered early.
This scene is originally inspired by the stunning website of Monopo London, which has been sitting in my "inspiration bookmarks" for over a year now. I can't confirm they used this technique, but by adjusting our demo to use @react-three/drei
's MeshTransmissionMaterial
we can get pretty close to the original effect ๐:
A more advanced Portal scene
Yet another portal! I know, I'm not very original with my examples in this article, but bear with me... this one is pretty magical ๐ช. A few months ago @jesper*vos came up with a scene where a mesh moves through a frame and comes out on the other side with a completely different geometry! It looked as if the mesh transformed itself while going through this "portal":
Of course, I immediately got obsessed with this so I had to give it a try to reproduce it. The diagram below showcases how we can achieve this scene by using what we learned so far about render targets and screen coordinates:
You can see the result in the code playground below๐. I'll let you explore this one on your own time ๐
An alternative to EffectComposer for post-processing effects
This part focuses on a peculiar use case (or at least it was to me the first time I encountered it) for render targets: using them to build a simple post-processing effect pipeline.
I first encountered render targets in the wild through this use case, perhaps not the best way to get started with them ๐
, in @pschroen's beautiful hologram scene. In it, he uses a series of layers, render targets, and custom shader materials in the render loop to create a post-processing effect pipeline. I reached out to ask why would one use that rather than the standard EffectComposer
and among the reasons were:
- it can be easier to understand
- some render passes when using
EffectComposer
may have unnecessary bloat, thus you may see some performance improvements by using render targets instead
As for the implementation of such pipeline, it works as follows:
Post-processing pipeline using render targets and custom materials
1const PostProcessing = () => {2const screenMesh = useRef();3const screenCamera = useRef();4const MaterialWithEffect1Ref = useRef();5const MaterialWithEffect2Ref = useRef();6const magicScene = new THREE.Scene();78const renderTargetA = useFBO();9const renderTargetB = useFBO();1011useFrame((state) => {12const { gl, scene, camera, clock } = state;1314// First pass15gl.setRenderTarget(renderTargetA);16gl.render(magicScene, camera);1718MaterialWithEffect1Ref.current.uniforms.uTexture.value =19renderTargetA.texture;20screenMesh.current.material = MaterialWithEffect1Ref.current;2122// Second pass23gl.setRenderTarget(renderTargetB);24gl.render(screenMesh.current, camera);2526MaterialWithEffect2Ref.current.uniforms.uTexture.value =27renderTargetB.texture;28screenMesh.current.material = MaterialWithEffect2Ref.current;2930gl.setRenderTarget(null);31});3233return (34<>35{createPortal(36<mesh ref={sphere} position={[2, 0, 0]}>37<sphereGeometry args={[1, 64]} />38<meshBasicMaterial />39</mesh>40magicScene41)}42<OrthographicCamera ref={screenCamera} args={[-1, 1, 1, -1, 0, 1]} />43<MaterialWithEffect1 ref={MaterialWithEffect1Ref} />44<MaterialWithEffect2 ref={MaterialWithEffect2Ref} />45<mesh46ref={screenMesh}47geometry={getFullscreenTriangle()}48frustumCulled={false}49/>50</>51)52}
- We render our main scene in a portal.
- The default scene only contains a fullscreen triangle (
screenMesh
) and an orthographic camera with specific settings to ensure this triangle fills the screen. - We render our main scene in a render target and then use its texture onto the fullscreen triangle.
- We apply a series of post-processing effects by using a dedicated render target for each of them and rendering only the fullscreen triangle mesh to that render target.
For the fullscreen triangle, we can use a BufferGeometry
and manually add the position attributes and uv coordinates to achieve the desired geometry.
Function returning a triangle geometry
1import { BufferGeometry, Float32BufferAttribute } from 'three';23const getFullscreenTriangle = () => {4const geometry = new BufferGeometry();5geometry.setAttribute(6'position',7new Float32BufferAttribute([-1, -1, 3, -1, -1, 3], 2)8);9geometry.setAttribute(10'uv',11new Float32BufferAttribute([0, 0, 2, 0, 0, 2], 2)12);1314return geometry;15};
The code playground below showcases an implementation of a small post-processing pipeline using this technique:
- We first apply a Chromatic Aberration pass on the scene.
- We then add a "Video Glitch" effect before displaying the final render.
As you can see, the result is indistinguishable from an equivalent scene that would use EffectComposer
๐.
Transition gracefully between scenes
This part is a last-minute addition to this article. I was reading the case study of one of my favorite Three.js-based websites โจ Atmos โจ and in it, there's a section dedicated to the transition effect that occurs towards the bottom of the website: a smooth, noisy reveal of a different scene.
Implementation-wise, the author mentions the following:
You can really go crazy with transitions, but in order to not break the calm and slow pace of the experience, we wanted something smooth and gradual. We ended up with a rather simple transition, achieved by mixing 2 render passes together using a good olโ Perlin 2D noise.
Lucky us, we know how to do all this now thanks to our newly acquired render target knowledge and shader skills โก! So I thought this would be a great addition as a last examples to all the amazing use cases we've seen so far!
Let's first break down the problem. To achieve this kind of transition we need to:
- Build two portalled scenes: the original one
sceneA
and the target onesceneB
. - Declare two render targets, one for each scene.
- In our
useFrame
loop, we render each scene in its respective render target. - We can pass both render targets' textures to the shader material of a full-screen triangle (identical to the one we used in our post-processing examples) along with a
uProgress
uniform representing the progress of the transition: towards-1
we rendersceneA
, whereas towards1
we rendersceneB
.
The shader code itself is actually not that complicated (which was a surprise to me at first, not going to lie). The only line we should really pay attention to here is line 17
where we mix our textures based on the value of noise
. That value depends on the Perlin noise we applied to the UV coordinates and also the uProgress
uniform along with some other tweaks:
Fragment shader mixing 2 textures
1varying vec2 vUv;23uniform sampler2D textureA;4uniform sampler2D textureB;5uniform float uProgress;67//...89void main() {10vec2 uv = vUv;11vec4 colorA = texture2D(textureA, uv);12vec4 colorB = texture2D(textureB, uv);1314// clamp the value between 0 and 1 to make sure the colors don't get messed up15float noise = clamp(cnoise(vUv * 2.5) + uProgress * 2.0, 0.0, 1.0);1617vec4 color = mix(colorA, colorB, noise);18gl_FragColor = color;19}
Once we stitch everything together in the scene, we achieve a similar transition to the one the author of Atmos built on their project. It did not require any insane amount of codes or obscure shader knowledge, just a few render target manipulations in the render loop and some beautiful Peril noise ๐ช
Conclusion
You now know pretty much everything I know about render targets, my favorite tool to create beautiful and mind-bending scenes with React Three Fiber! There are obviously many other use cases for them, but the ones we saw through the many examples featured in this blog post are the ones that I felt represented the best what is possible to achieve with them.
When used in combination with useFrame
and createPortal
a new world of possibilities opens up for your scenes, which lets you create and experiment with new interactions or new ways to implement some stunning shader materials and post-processing effects โจ.
Render targets have been challenging to learn for me at first, but since understanding how they work, they have been the secret tool behind some of my favorite creations, and I have had a lot of fun working with them so far. I really hope this article will make these concepts click for you to enable you to push your next React Three Fiber/Three.js project even further.
Liked this article? Share it with a friend on Twitter or support me to take on more ambitious projects to write about. Have a question, feedback or simply wish to contact me privately? Shoot me a DM and I'll do my best to get back to you.
Have a wonderful day.
โ Maxime
A deep dive into WebGL Render Targets and how to leverage their capabilities in combination with the render loop to create scenes with post-processing effects, transition, and many types of optical illusions to trick the viewer's eyes.