Quantcast
Channel: Envato Tuts+ Game Development
Viewing all articles
Browse latest Browse all 728

Creating Toon Water for the Web: Part 1

$
0
0

In my Beginner's Guide to Shaders I focused exclusively on fragment shaders, which is enough for any 2D effect and every ShaderToy example. But there's a whole category of techniques that require vertex shaders. This tutorial will walk you through creating stylized toon water while introducing vertex shaders. I will also introduce the depth buffer and how to use that to get more information about your scene and create foam lines.

Here's what the final effect should look like. You can try a live demo here (left mouse to orbit, right mouse to pan, scroll wheel to zoom).

Kayak and lighthouse in waterKayak and lighthouse in waterKayak and lighthouse in water

Specifically, this effect is composed of:

  1. A subdivided translucent water mesh with displaced vertices to make waves.
  2. Static water lines on the surface.
  3. Fake buoyancy on the boats.
  4. Dynamic foam lines around the edge of objects in the water.
  5. A post-process distortion of everything underwater.

What I like about this effect is that it touches on a lot of different concepts in computer graphics, so it will allow us to draw on ideas from past tutorials, as well as developing techniques we can use for a variety of future effects.

I'll be using PlayCanvas for this just because it has a convenient free web-based IDE, but everything should be applicable to any environment running WebGL. You can find a Three.js version of the source code at the end. I'll be assuming you're comfortable using fragment shaders and navigating the PlayCanvas interface. You can brush up on shaders here and skim an intro to PlayCanvas here.

Environment Setup

The goal of this section is to set up our PlayCanvas project and place some environment objects to test the water against. 

If you don't already have an account with PlayCanvas, sign up for one and create a new blank project. By default, you should have a couple of objects, a camera and a light in your scene.

A blank PlayCanvas project showing the objects the scene contains A blank PlayCanvas project showing the objects the scene contains A blank PlayCanvas project showing the objects the scene contains

Inserting Models

Google's Poly project is a really great resource for 3D models for the web. Here is the boat model I used. Once you download and unzip that, you should find a .obj and a .png file.

  1. Drag both files into the asset window in your PlayCanvas project.
  2. Select the material that was automatically created, and set its diffuse map to the .png file.
Click on the diffuse tab and select the boat imageClick on the diffuse tab and select the boat imageClick on the diffuse tab and select the boat image

Now you can drag the Tugboat.json into your scene and delete the Box and Plane objects. You can scale the boat up if it looks too small (I set mine to 50).

You can scale the model up using the properties panel on the right once its selectedYou can scale the model up using the properties panel on the right once its selectedYou can scale the model up using the properties panel on the right once its selected

You can add any other models to your scene in the same way.

Orbit Camera

To set up an orbit camera, we'll copy a script from this PlayCanvas example. Go to that link, and click on Editor to enter the project.

  1. Copy the contents of mouse-input.js and orbit-camera.js from that tutorial project into the files of the same name in your own project.
  2. Add a Script component to your camera.
  3. Attach the two scripts to the camera.

Tip: You can create folders in the asset window to keep things organized. I put these two camera scripts under Scripts/Camera/, my model under Models/, and my material under Materials/.

Now, when you launch the game (play button on the top right of the scene view), you should be able to see your boat and orbit around it with the mouse. 

Subdivided Water Surface

The goal of this section is to generate a subdivided mesh to use as our water surface.

To generate the water surface, we're going to adapt some code from this terrain generation tutorial. Create a new script file called Water.js. Edit this script and create a new function called GeneratePlaneMesh that looks like this:

1
Water.prototype.GeneratePlaneMesh=function(options){
2
// 1 - Set default options if none are provided 
3
if(options===undefined)
4
options={subdivisions:100,width:10,height:10};
5
// 2 - Generate points, uv's and indices 
6
varpositions=[];
7
varuvs=[];
8
varindices=[];
9
varrow,col;
10
varnormals;
11
12
for(row=0;row<=options.subdivisions;row++){
13
for(col=0;col<=options.subdivisions;col++){
14
varposition=newpc.Vec3((col*options.width)/options.subdivisions-(options.width/2.0),0,((options.subdivisions-row)*options.height)/options.subdivisions-(options.height/2.0));
15
16
positions.push(position.x,position.y,position.z);
17
18
uvs.push(col/options.subdivisions,1.0-row/options.subdivisions);
19
}
20
}
21
22
for(row=0;row<options.subdivisions;row++){
23
for(col=0;col<options.subdivisions;col++){
24
indices.push(col+row*(options.subdivisions+1));
25
indices.push(col+1+row*(options.subdivisions+1));
26
indices.push(col+1+(row+1)*(options.subdivisions+1));
27
28
indices.push(col+row*(options.subdivisions+1));
29
indices.push(col+1+(row+1)*(options.subdivisions+1));
30
indices.push(col+(row+1)*(options.subdivisions+1));
31
}
32
}
33
34
// Compute the normals 
35
normals=pc.calculateNormals(positions,indices);
36
37
38
// Make the actual model
39
varnode=newpc.GraphNode();
40
varmaterial=newpc.StandardMaterial();
41
42
// Create the mesh 
43
varmesh=pc.createMesh(this.app.graphicsDevice,positions,{
44
normals:normals,
45
uvs:uvs,
46
indices:indices
47
});
48
49
varmeshInstance=newpc.MeshInstance(node,mesh,material);
50
51
// Add it to this entity 
52
varmodel=newpc.Model();
53
model.graph=node;
54
model.meshInstances.push(meshInstance);
55
56
this.entity.addComponent('model');
57
this.entity.model.model=model;
58
this.entity.model.castShadows=false;// We don't want the water surface itself to cast a shadow 
59
};

Now you can call this in the initialize function:

1
Water.prototype.initialize=function(){
2
this.GeneratePlaneMesh({subdivisions:100,width:10,height:10});
3
};

You should see just a flat plane when you launch the game now. But this is not just a flat plane. It's a mesh composed of a thousand vertices. As a challenge, try to verify this (it's a good excuse to read through the code you just copied). 

Challenge #1: Displace the Y coordinate of each vertex by a random amount to get the plane to look something like the image below.
A subdivided plane with displaced verticesA subdivided plane with displaced verticesA subdivided plane with displaced vertices

Waves

The goal of this section is to give the water surface a custom material and create animated waves.

To get the effects we want, we need to set up a custom material. Most 3D engines will have some pre-defined shaders for rendering objects and a way to override them. Here's a good reference for doing this in PlayCanvas.

Attaching a Shader

Let's create a new function called CreateWaterMaterial that defines a new material with a custom shader and returns it:

1
Water.prototype.CreateWaterMaterial=function(){
2
// Create a new blank material  
3
varmaterial=newpc.Material();
4
// A name just makes it easier to identify when debugging 
5
material.name="DynamicWater_Material";
6
7
// Create the shader definition 
8
// dynamically set the precision depending on device.
9
vargd=this.app.graphicsDevice;
10
varfragmentShader="precision "+gd.precision+" float;\n";
11
fragmentShader=fragmentShader+this.fs.resource;
12
13
varvertexShader=this.vs.resource;
14
15
// A shader definition used to create a new shader.
16
varshaderDefinition={
17
attributes:{
18
aPosition:pc.gfx.SEMANTIC_POSITION,
19
aUv0:pc.SEMANTIC_TEXCOORD0,
20
},
21
vshader:vertexShader,
22
fshader:fragmentShader
23
};
24
25
// Create the shader from the definition
26
this.shader=newpc.Shader(gd,shaderDefinition);
27
28
// Apply shader to this material 
29
material.setShader(this.shader);
30
31
returnmaterial;
32
};

This function grabs the vertex and fragment shader code from the script attributes. So let's define those at the top of the file (after the pc.createScript line):

1
Water.attributes.add('vs',{
2
type:'asset',
3
assetType:'shader',
4
title:'Vertex Shader'
5
});
6
7
Water.attributes.add('fs',{
8
type:'asset',
9
assetType:'shader',
10
title:'Fragment Shader'
11
});

Now we can create these shader files and attach them to our script. Go back to the editor, and create two new shader files: Water.frag and Water.vert. Attach these shaders to your script as shown below.

Watervert and Waterfrag are attached to WaterInitWatervert and Waterfrag are attached to WaterInitWatervert and Waterfrag are attached to WaterInit

If the new attributes don't show up in the editor, click the Parse button to refresh the script.

Now put this basic shader in Water.frag:

1
voidmain(void)
2
{
3
vec4color=vec4(0.0,0.0,1.0,0.5);
4
gl_FragColor=color;
5
}

And this in Water.vert:

1
attributevec3aPosition;
2
3
uniformmat4matrix_model;
4
uniformmat4matrix_viewProjection;
5
6
voidmain(void)
7
{
8
gl_Position=matrix_viewProjection*matrix_model*vec4(aPosition,1.0);
9
}

Finally, go back to Water.js and make it use our new custom material instead of the standard material. So instead of:

1
varmaterial=newpc.StandardMaterial();

Do:

1
varmaterial=this.CreateWaterMaterial();

Now, if you launch the game, the plane should now be blue.

The shader we wrote renders the plane as blueThe shader we wrote renders the plane as blueThe shader we wrote renders the plane as blue

Hot Reloading

So far, we've just set up some dummy shaders on our new material. Before we get to writing the real effects, one last thing I want to set up is automatic code reloading.

Uncommenting the swap function in any script file (such as Water.js) enables hot-reloading. We'll see how to use this later to maintain the state even as we update the code in real time. But for now we just want to re-apply the shaders once we've detected a change. Shaders get compiled before they are run in WebGL, so we'll need to recreate the custom material to trigger this.

We're going to check if the contents of our shader code have been updated and, if so, recreate the material. First, save the current shaders in the initialize:

1
// initialize code called once per entity
2
Water.prototype.initialize=function(){
3
this.GeneratePlaneMesh();
4
5
// Save the current shaders 
6
this.savedVS=this.vs.resource;
7
this.savedFS=this.fs.resource;
8
9
};

And in the update, check if there have been any changes:

1
// update code called every frame
2
Water.prototype.update=function(dt){
3
4
if(this.savedFS!=this.fs.resource||this.savedVS!=this.vs.resource){
5
// Re-create the material so the shaders can be recompiled 
6
varnewMaterial=this.CreateWaterMaterial();
7
// Apply it to the model 
8
varmodel=this.entity.model.model;
9
model.meshInstances[0].material=newMaterial;
10
11
// Save the new shaders
12
this.savedVS=this.vs.resource;
13
this.savedFS=this.fs.resource;
14
}
15
16
};

Now, to confirm this works, launch the game and change the color of the plane in Water.frag to a more tasteful blue. Once you save the file, it should update without having to refresh or relaunch! This was the color I chose:

1
vec4color=vec4(0.0,0.7,1.0,0.5);

Vertex Shaders

To create waves, we need to move every vertex in our mesh every frame. This sounds as if it's going to be very inefficient, but every vertex of every model already gets transformed on each frame we render. This is what the vertex shader does. 

If you think of a fragment shader as a function that runs on every pixel, takes a position, and returns a color, then a vertex shader is a function that runs on every vertex, takes a position, and returns a position.

The default vertex shader will take the world position of a given model, and return the screen position. Our 3D scene is defined in terms of x, y, and z, but your monitor is a flat two-dimensional plane, so we project our 3D world onto our 2D screen. This projection is what the view, projection, and model matrices take care of and is outside of the scope of this tutorial, but if you want to learn exactly what happens at this step, here's a very nice guide.

So this line:

1
gl_Position=matrix_viewProjection*matrix_model*vec4(aPosition,1.0);

Takes aPosition as the 3D world position of a particular vertex and transforms it into gl_Position, which is the final 2D screen position. The 'a' prefix on aPosition is to signify that this value is an attribute. Remember that a uniformvariable is a value we can define on the CPU to pass to a shader that retains the same value across all pixels/vertices. An attribute's value, on the other hand, comes from an array defined on the CPU. The vertex shader is called once for each value in that attribute array.

You can see these attributes are set up in the shader definition we set up in Water.js:

1
varshaderDefinition={
2
attributes:{
3
aPosition:pc.gfx.SEMANTIC_POSITION,
4
aUv0:pc.SEMANTIC_TEXCOORD0,
5
},
6
vshader:vertexShader,
7
fshader:fragmentShader
8
};

PlayCanvas takes care of setting up and passing an array of vertex positions for aPosition when we pass this enum, but in general you could pass any array of data to the vertex shader.

Moving the Vertices

Let's say you want to squish the plane by multiplying all x values by half. Should you change aPositionor gl_Position?

Let's try aPosition first. We can't modify an attribute directly, but we can make a copy:

1
attributevec3aPosition;
2
3
uniformmat4matrix_model;
4
uniformmat4matrix_viewProjection;
5
6
voidmain(void)
7
{
8
vec3pos=aPosition;
9
pos.x*=0.5;
10
11
gl_Position=matrix_viewProjection*matrix_model*vec4(pos,1.0);
12
}

The plane should now look more rectangular. Nothing strange there. Now what happens if we instead try modifying gl_Position?

1
attributevec3aPosition;
2
3
uniformmat4matrix_model;
4
uniformmat4matrix_viewProjection;
5
6
voidmain(void)
7
{
8
vec3pos=aPosition;
9
//pos.x *= 0.5;
10
11
gl_Position=matrix_viewProjection*matrix_model*vec4(pos,1.0);
12
gl_Position.x*=0.5;
13
}

It might look the same until you start to rotate the camera. We're modifying screen space coordinates, which means it's going to look different depending on how you're looking at it.

So that's how you can move the vertices, and it's important to make this distinction between whether you're in world or screen space.

Challenge #2: Can you move the whole plane surface a few units up (along the Y axis) in the vertex shader without distorting its shape?
Challenge #3: I said gl_Position is 2D, but gl_Position.z does exist. Can you run some tests to determine if this value affects anything, and if so, what it's used for?

Adding Time

One last thing we need before we can create moving waves is a uniform variable to use as time. Declare a uniform in your vertex shader:

1
uniformfloatuTime;

Then, to pass this to our shader, go back to Water.js and define a time variable in the initialize:

1
Water.prototype.initialize=function(){
2
this.time=0;///// First define the time here 
3
4
this.GeneratePlaneMesh();
5
6
// Save the current shaders 
7
this.savedVS=this.vs.resource;
8
this.savedFS=this.fs.resource;
9
};

Now, to pass this to our shader, we use material.setParameter. First we set an initial value at the end of the CreateWaterMaterial function:

1
// Create the shader from the definition
2
this.shader=newpc.Shader(gd,shaderDefinition);
3
4
////////////// The new part
5
material.setParameter('uTime',this.time);
6
this.material=material;// Save a reference to this material
7
////////////////
8
9
// Apply shader to this material 
10
material.setShader(this.shader);
11
12
returnmaterial;

Now in the updatefunction we can increment time and access the material using the reference we created for it:

1
this.time+=0.1;
2
this.material.setParameter('uTime',this.time);

As a final step, in the swap function, copy over the old value of time, so that even if you change the code it'll continue incrementing without resetting to 0.

1
Water.prototype.swap=function(old){
2
this.time=old.time;
3
};

Now everything is ready. Launch the game to make sure there are no errors. Now let's move our plane by a function of time in Water.vert:

1
pos.y+=cos(uTime)

And your plane should be moving up and down now! Because we have a swap function now, you can also update Water.js without having to relaunch. Try making time increment faster or slower to confirm this works.

Moving the plane up and down with a vertex shaderMoving the plane up and down with a vertex shaderMoving the plane up and down with a vertex shader
Challenge #4: Can you move the vertices so it looks like the wave below? 

As a hint, I talked in depth about different ways to create waves here. That was in 2D, but the same math applies here. If you'd rather just peek at the solution, here's the gist.

Translucency

The goal of this section is to make the water surface translucent.

You might have noticed that the color we're returning in Water.frag does have an alpha value of 0.5, but the surface is still completely opaque. Transparency in many ways is still an open problem in computer graphics. One cheap way to achieve it is to use blending.

Normally, when a pixel is about to be drawn, it checks the value in the depth buffer against its own depth value (its position along the Z axis) to determine whether to overwrite the current pixel on the screen or discard itself. This is what allows you to render a scene correctly without having to sort objects back to front. 

With blending, instead of simply discarding or overwriting, we can combine the color of the pixel that's already drawn (the destination) with the pixel that's about to be drawn (the source). You can see all the available blending functions in WebGL here.

To make the alpha work the way we expect it, we want the combined color of the result to be the source multiplied by the alpha plus the destination multiplied by one minus the alpha. In other words, if the alpha is 0.4, the final color should be:

1
finalColor=source*0.4+destination*0.6;

In PlayCanvas, the pc.BLEND_NORMAL option does exactly this.

To enable this, just set the property on the material inside CreateWaterMaterial:

1
material.blendType=pc.BLEND_NORMAL;

If you launch the game now, the water will be translucent! This isn't perfect, though. A problem arises if the translucent surface overlaps with itself, as shown below.

Artifacts arise when a translucent surface overlaps with itself Artifacts arise when a translucent surface overlaps with itself Artifacts arise when a translucent surface overlaps with itself

We can fix this by using alpha to coverage, which is a multi-sampling technique to achieve transparencyinstead of blending:

1
//material.blendType = pc.BLEND_NORMAL;
2
material.alphaToCoverage=true;

But this is only available in WebGL 2. For the remainder of this tutorial, I'll be using blending to keep it simple.

Summary

So far we've set up our environment and created our translucent water surface with animated waves from our vertex shader. The second part will cover applying buoyancy on objects, adding water lines to the surface, and creating the foam lines around the edges of objects that intersect the surface. 

The final part will cover applying the underwater post-process distortion effect and some ideas for where to go next.

Source Code

You can find the finished hosted PlayCanvas project here. A Three.js port is also available in this repository.


Viewing all articles
Browse latest Browse all 728

Trending Articles