Stylized Shader Robot Scene
Features Implemented
Shaders
- Stylized Toon Surface Shader
- Wobbly Shader (Vertex Deformation)
Post-Processing Effects
- Wobbly Depth-Based Outlines
- Static Color-Based Outlines
- Cohesive Color Saturation Filter
- Black & White Mode
- TV Pixelization Static Filter
Shaders
- Switching between two color palettes
- Toggling Black & White mode + TV Pixelization Static Filter
Stylized toon shader with half tone shading, post processing effects, and interactivity.
1. Inspiration
Artist: Katawoka
Ghost Girl Illustration
My inspiration for my stylized shader was this comic style illustration to the left. There were a few visual elements from this piece I wanted to incorporate into my toon shader:
- Half tone shading
- Outlined shadows
- Minimal color palette
Credit: Robot Model by Tasha Lime
Robot Scene Assets
- Because the main focus of this project was to create an interesting stylized shader, I chose to import my main asset from SketchFab.
- I chose this sad looking robot character because I thought it replicated the melancholy expression of the girl in the illustration.
- The other assets in the scene are simple cubes and spheres made in Unity.
2. Interesting Shaders
I. STYLIZED TOON SHADER
My stylized toon shader has the following visual characteristics:
- Three tone shading
- Outlined shadows
- Half tone shading
- Multiple light support
- Specular highlight
- Minimalistic color palette
Three Tone Shading
- This was achieved by using three threshold values to bucket the colors in the scene. These values control the amount of Highlight, Midtone, and Shadow seen on the meshes and can be customized by the user.
Specular Highlight
- The specular highlight appears on objects in the scene as a white dot at bright spots of light.
- This was created by using the view vector from the current fragment to the camera and the reflected light vector. The reflected light vector can be calculated using the normal vector and the vector from the current vertex to the light position.
- The angle between the reflection vector and the view vector can be found with a dot product. This angle is then used to determine the amount of specular reflection the current fragment gets.
- This is added to the diffuse color value and creates the bright spot of white!
Outlined Shadows
- I thought the outlines on the shadows of my inspiration piece were really interesting! I achieved a similar look in my scene, exemplified well in the image below.
- We can see the outline on the shadows on both the ground and the robot.
- The shadow outline on the robot separates the light teal color from the darker green color on the front of the robot and follows the shape of the boundary between the Highlight and the Shadow tones
- Shadow Outline color and width can be customized by the user
- This was a simple effect done by using the logic of the Three Tone Shader but decreasing the size of the Midtone so it acted as an outline instead of a tone. (The reason the robot still looks like it has multiple tones — more than a 2 tone shader — is because of the point lights creating a bucketed coloring effect!)
Half Tone Shading
To implement half tones, I first created a custom half tone texture.
- I used a brush tool in Procreate to draw the dots to make it look hand-drawn.
- I sampled the UVs of the half tone texture, making sure to first tile + offset and rotate the UV to make it more visually interesting.
- To be able to customize the color of the dots on the shadow, I take the minimum of the Shadow color and the sampled texture color. Thus, if we are sampling one of the dots, then the minimum will be black (the dot color) and the color of the Shadow otherwise.
- We then ensure the color of the dots is colored correctly with the Shadow Dots color (rather than fixing the color of the dots to be black from the sample texture). We do this by taking the maximum of the sampled color and the Shadow Dots color, which will return Shadow Dots if we originally sampled a dot or Shadow if we originally sampled empty space around the dots.
- The user can customize Shadow, Shadow Dots, and Shadow Tiling, which control the color of the shadow, the color of the dots, and the size of the dots.
Screenshot of back view to demonstrate half-tone shading
Multiple Light Support
To implement half tones, I first created a custom half tone texture.
- I used a brush tool in Procreate to draw the dots to make it look hand-drawn.
- The intersection of the lights can be seen in the scene below. I increased the intensity of the second light here and made it purple to exaggerate the visual effect.
- We can see the interaction of the two lights at areas where they overlap — note the brighter spots
- The lights also change the appearance of the shadows and help overall bring more complexity to the scene
Example scene to illustrate additional lights
II. WOBBLY SURFACE SHADER
- I created a second shader based on above, with some added vertex deformation.
- For each vertex, I interpolate between its original position and an animated offset position.
- The animated position is created by offsetting the x position of the vertex using a sine curve, with time and several exposed parameters (like speed and frequency) as an input. This creates a nice wiggly-wobbly effect!
- The user can tweak the Wobble Speed, Wobble Frequency, Wobble Distance, and Wobble Amount, to customize the intensity of the wobbly animation.
- The intersection of the lights can be seen in the scene below. I increased the intensity of the second light here and made it purple to exaggerate the visual effect.
- We can see the interaction of the two lights at areas where they overlap — note the brighter spots
- The lights also change the appearance of the shadows and help overall bring more complexity to the scene
3. Outlines
Left: Example scene with Depth-based outlines; Right: Example wobble effect on outlines
I. DEPTH-BASED OUTLINES
In order to create depth based outlines, I used Unity’s URP to get access to the Depth buffer. Then, I used a Sobel outline implementation to detect large differences in depth.
- There are several adjustable parameters, including outline Width, ColorThreshold (adjusts amount of outlines), and Strength (adjusts the strength/opacity of the outlines)
- To make the outlines more interesting, I decided to add a wobble effect using similar logic for the wobble shader.
- The intensity of wobble can also be adjusted by the user, similar to the shader wobble effect described above.
II. NORMAL-BASED (COLOR) OUTLINES
Because my inspiration piece included outlines between different tones, I also chose to implement normal based outlines which achieved the outlines between colors.
- I used a Normal Shader feature and added a material made from it to the render pipeline. I then connected a Normal Buffer texture as the destination target for this render feature.
- Using this Normal Buffer texture, I was able to detect large difference in normals across the screen, which I used to create color-based outlines.
- These outlines had the same adjustable parameters as the Depth-based ones.
4. Cohesive Color Saturation Filter
I wanted the colors in my scene to complement each other well, like the inspiration piece. To do this, I implemented a filter that can easily tweak the saturation of the image and the color palette applied to it.
- To adjust saturation, I interpolated between the original color and and a grayscale color defined based on the original input color. The user can tweak the amount of saturation by adjusting the xyz components of the Sat parameter (which is the input for interpolation).
- To tweak the color palette, the user can individually change each of the xyz components of the Sat parameter. Because they correspond to the amount of rgb mixing, doing so will vary the colors shown in the final output.
- Below are some the results that can be achieve with this post-processing effect:
Scene with no color saturation post-processing effect applied
Scene with color saturation post-processing effect applied
As you can see, on the left, the colors are a bit jarring and don’t complement each other very well. The scene doesn’t feel as cohesive. In contrast, the scene on the right feels a lot more cohesive. The orange gets mapped to a less intense peach color and the bold green becomes a calmer teal.
More variations of the color saturation filter!
5. Interactivity
I. KEY CONTROLS
I implemented several interactive elements with the following key controls:
- Pressing Space: Switches between two color palettes (see below)
- Pressing B: Toggles Black & White mode on/off (see Extra Polish for more details)
- Pressing S: Toggles a TV Pixelization Static Filter on/off (see Extra Polish for more details)
II. TWO COLOR PALETTES: Green/Orange + Gray/Purple
- I started out with a high contrast color palette — the Green/Orange — and really liked how the colors complemented each other. I thought it would be interesting to have more neutral colors as a second palette, to achieve a higher contrast between the outlines and the colors.
- The switch in colors was achieved by attaching a material switching script to each mesh in the scene and assigning the two sets of colors to each mesh.
Scene with colors from Green/Orange color palette attached
Scene with colors from Gray/Purple color palette attached
6. Extra Polish
I. BLACK & WHITE
- Since my main inspiration was comic style art, I thought it was fitting to make a Black & White mode for the scene. A lot of comic styles use minimal color palettes, and the most indicative example of this is a simple black and white color scheme.
- The Black & White mode is a post-processing effect, achieved by mapping colors above a certain threshold to black and all other colors to white.
- The Threshold value can be customized by the user. Higher threshold create more black in the scene and lower thresholds yield more white:
Scene with Black & White mode applied, with a Threshold value of 1.0 (left), 1.41 (middle), and 1.79 (right)
II. TV Pixelization Static Filter
- I wanted to experiment with bucketing the colors to create a pixelization post-processing effect.
- I noticed this made a glitch-like effect when rotating the camera which ultimately inspired the static effect!
Scene with TV Pixelization Static Filter applied (Pixel Amount: 100, Distortion Amount: 0.03, Distortion Offset: 0.62, Color Distortion: 4, Dot Density: 108)
There are a few visual elements that, combined, create the static filter. Each of these can be customized by the user.
- Pixelization — achieved by bucketing the UV sampling of the scene’s main color texture
- Dots Overlay — overlays a grid of dots on top of the main texture to mimic a TV screen. This was done by using the output of a Voronoi texture to sample a three tone color gradient. Decreasing the opacity of the gradient and then subtracting it from the main color achieved the dotted texture.
- Static Distortion Effect — creates the glitchy warping effect that can be seen best along the edges of the objects in the scene
- Color Distortion — distorts the colors of the scene to achieve different kinds of retro color styles
Left: After Pixelization; Middle: After Dots Overlay; Right: After Static Distortion & Color Distortion
The final result: