How To Make:
A Simple Toon Shader
(Using Amplify Shader Editor)
The Idea
Toon shaders are really cool, and they can serve as the basis for creating a variety of interesting materials for our models. The basic idea of a toon shader is that it converts regular lighting information into something that looks more like cell shading like animators use in a cartoon. Regular light in the world smoothly and gradually falls off as an object is angled away from the light source, but in a cartoon, this transition is much more abrupt.
Regular
Toon
The image below is a screenshot taken from Critical Role’s Vox Machina, an anime-like show that uses cell shading for its lighting. You can see there’s 3 very noticeable changes in light. There’s other stuff happening in the image, but we’ll focus on this. There are two light sources in the shot: there’s a gentle bluish light coming from the left and an orange light source on the right. As you let your eye travel from left to right, you can see the lighter shades of the character’s armor abruptly change darker, then abruptly change again to an intense orange. We’ll try to create this effect in a shader in Unity.
Credit: Critical Role
Getting Started
If you don’t have Amplify Shader Editor for Unity installed, you can get it from the Unity asset store. I believe there are free and paid versions. I’m using the paid version in this tutorial. Once you have that installed, create a new Amplify Shader Editor surface shader and call it “Toon”. Then right click on the shader file and select create > material. If you would like to visualize what we’re doing as we go in a scene, you can apply this material to a cube and a sphere. To start with, it will look black. I won’t be referring back to these object again until much later in the tutorial, but you can use them to kind of get another perspective on how the shader works as we go through each step.
Open up the shader and set the lighting model to “custom lighting” - we’re going to be manipulating the way light interacts with this material in order to create our effect.
Light Direction
First, we need the angle that the light is hitting our surface at. To get this, we’ll use the ASE World Normal node to get our surface normal in world space. This is a vector perpendicular to the surface face (noted in blue in the image below).
Next we’ll use the World Space Light Direction node to get the light direction in the world space. This node outputs a normalized direction vector and has the added bonus of handling multiple light sources for us, including different kinds of built-in unity lights like directional and point. The end result is that it saves us the trouble of writing a lot of math in order to get a normalized light direction.
Since what we need is the relationship between the vector of the surface normal (blue) and the light direction (red) that explains to us how much each vector is aligned with the other, kind of like the angle. To get this, we get the dot product of the two vectors. If that’s complicated, don’t worry too much about it, just observe the result of this formula gives us a smooth gradient of light cast on a sphere at the direction a light source is coming from.
You can see the node setup below. To keep things organized, I’ll register the result of this formula as a variable and call it “NormalLightDirection” for later use.
Shadows
Now we need to change that gradient shadow we see into sections. It can help to think of the gradient mathematically in order to understand how we’re going to manipulate it to get our toon effect. If black is represented by the number 0 and White is represented by the number 1, then all of the shades of grey exist between 0 and 1. Ex. 0.5 is a half-grey, halfway between black (0) and white (1). A black to white gradient is like we’re slowly moving forward on a line starting from 0, and for every step forward we take, we’re adding a small value to 0. By the end o the line, we’ve reached 1, and the color of our line has changed gradually from black to white.
The smaller you amount you add for each step you take, the more smooth the gradient becomes. So what we need to do to achieve our toon effect is prevent that gradual step from happening. Instead of slowly changing from 0 to 1 over 100 steps, we want to change from 0 to 1 over maybe 1 or 2 steps.
There are multiple ways to do this, one way is use the Step function, which takes in two numbers: a value and a threshold. For each value entered, the function determines if it should return 0 or 1 depending on whether value is higher or lower than threshold. If we apply this to our NormalLightDirection we created in the last step (seen below), and set the threshold to 0 you can see how it changes the gradient we see into an abrupt change from black to white.
What’s really important to remember here is that the blackest part of our NormalLightDirection data is not 0, its actually less than 0 and the gradient transitions from that negative unknown number to 1. Remember, the Step function will return white (1) for every pixel equal to or above our threshold of 0 and black for anything less than 0. If our lighting data started at 0, Step would turn every pixel to white. There’s a node in amplify that can help us visualize this called “saturate” which restricts all data points to be no higher than one and no lower than 0. If we run our data from NormalLightDirection through saturate before going to our Step node, you’ll notice the visual for the Step node turns entirely white. This is because everything coming out of saturate is greater than or equal to 0.
Using the Step function illustrates the principles of how the toon shader works, but using the step node to achieve this effect can get messy quickly. What if we want to have multiple shades of grey instead of just black to white? We’ll need more Step nodes. What if we want each layer of shadow to be different sizes? Well then we’ll need a bunch of different parameters for adjusting the threshold of each of our Step nodes — you can see this could easily get a little cumbersome.
Instead, we can use a process called sampling to make our lives easier. Here’s how it works, remember that line from earlier when we were talking about how a gradient works? What if, instead of adding a small amount to 0 for each unit step we take along the line, we simply grabbed the coordinate for where we are along the line and then assigned some value between 0 and 1 determined by our distance from the start of the line? This idea is illustrated below showing how the same X coordinate along a line can yield different values of grey depending on which greyscale line we choose.
These different versions of scaling from black to white are commonly called ramps (because they ramp up from black to white). The process of selecting a value based on coordinates is called sampling. This is how UV maps work when your applying a texture to a 3d model. The shader reads the U and V coordinates stored on the model, then samples the pixel on a 2d texture at coordinates (X,Y) where X = U and Y = V. We’re going to use the same process to determine how far along our gradient created by light we’ve traveled, then use that as the X coordinate to sample from a ramp in order to determine how dark our shadow should be at this level of lighting.
Below is our ramp texture we’ll use. Notice that the color doesn’t change along the Y axis of the texture, only the X axis. I made this in paint. Make something similar and drop it into your project directory. When you bring it into Unity, be sure that the box “Alpha is Transparency” is unchecked and the wrap mode is set to “clamp” in your import settings.
Back in ASE, we’ll create a Texture Sample node, then select our texture and run our NormalLightDirection data into the UV input for the Texture Sample node. Now our toon effect is starting to take shape! Before we move on, lets also make sure the properties of our texture sample node are named correctly and set to “property” so that we can change out the ramp texture from the material settings screen when we need to.
Scaling & Offset
One downside to this method is the inability to change the scale of our shadow, or controlling where our shadow begins and ends - but we can fix that easy!
We can adjust the size and location of the shadow by changing the scale and offset of the coordinates we pass into the TextureSample node. To do this, create a Scale and Offset node. Then create two float parameters, make them “properties” instead of “constants” so that we can access them from the material inspector later, and name them ShadowScale and ShadowOffset respectively. Set them to a default 0.5 for now and plug them into their respective inputs in the Scale and Offset node. Finally, plug in the NormalLightDirection variable and run everything back into the UV input of the sample node. We’ll also register this output as a new variable.
Go ahead and plug our new Shadow variable into the custom lighting input of our shader’s output, then look in your scene, open the Toon material and play with the ShadowScale and ShadowOffset properties to get a feel for how they behave.
Additional Light Information
If you add a point light into your scene, then move the point light around, you might notice is behaves strangely. There’s no color even though there’s shadows. Also, if you move the light far enough away, you’ll see there’s a distance at which the shadows for the point light just instantly turn off instead of gradually fading out. Now try changing the color of the directional light that controls the “sun” in your scene. No change on our objects.
We’re missing some important information about our lights: color and attenuation. Color is, obviously, the color of the light source. Attenuation is the decrease in light intensity as an object moves away from its source. Let’s get these set up in our shader.
Red sun and blue point light, but no color or attenuation on the cube or sphere.
Color
Adding light color is simple enough. ASE provides us with a Light Color node that will one again save us a lot of trouble by handling information from multiple light sources for us. Lets add that in, then run it into a node for registering a variable called “ColorAndAttenuation” Next, let’s use the GetVariable node to grab our Shadow variable and our ColorAndAttenuation variable, then multiply them together and run the result into our custom lighting output. Compile the shader, then, back in our scene, notice that we now have light color on our objects!
Attenuation
For attenuation, ASE also has an attenuation node. All we need to do is bring that in and multiply it with our color node information.
If you want to use a normal map as part of your material, you can sample the normal map using the standard method, then run that into an Indirect Diffuse Light node, then add it with the Light Attenuation before multiplying with Light Color.
Albedo
As the final step, we’re going to allow our material to take in a base texture and a base material. We’ll create a BaseColor property and a BaseTexture texture sample property, multiply them together, and register them as our Albedo variable.
Then, we’ll multiply our Albedo variable with the product of our Shadow and Lighting variables before sending the result into our Custom Lighting output. All done!
——— // ———
That concludes the tutorial! There’s a lot you can do with this shader as a base. You can add an outline like some cartoons, or you can use the techniques here as a jumping off point for making your objects look like different drawing and painting styles!