glowing shader article splash

Writing a Unity Shader script for a glowing material — A deeper look

Subtle background animations always help to set the mood of the game. It might only be something simple as a slowly pulsing glow, but it adds to the ambience of the digital city that Terminus is set in.

This article takes a deeper look into some parts of my animated shader tutorial on Youtube, where I went through the creation of a shader used in one of my team’s recent projects.

  1. SubShader tags
    1. Queue
    2. RenderType
  2. Vertex and fragment
  3. Time
  4. Emission formula

1. SubShader tags

Subshaders use tags to tell how and when they expect to be rendered to the rendering engine.

Unity manual – ShaderLab: SubShader Tags

In our Shader, we used 2 types of tags: Queue and RenderType.

a. Queue

Queue handles the “when” in the rendering process, telling the engine which layer that the object should be rendered on. The “layer” I am referring to is similar to how 2D sorting layers work, where sprites on the top layer [default layer] will appear behind sprites in layers below it (with sprites in the midground layer appearing at the forefront of the screen).

SpriteRenderer sorting layer

Similarly, the offset value in “queue” is similar to the “order in layer”, with the higher value rendered last in that layer. In my case, I used “Queue = Geometry”, also known as the default Queue type. It only renders opaque objects in the scene. Here are the other types of queue provided in the unity manual:

queue tag types
Source: Unity manual, page ShaderLab: assigning tags to a SubShader.

Article continues after the advertisement:


To provide an illustration of how the different queues come into play, I am using the 2 images below. In Hitman 3, the non-interactable background is most likely rendered using the “background” queue or equivalent.

Hitman 3 environment
Image source: Hilde Hjønnevåg Sunde portfolio

While in Valorant, these are the queue types:

  1. The opaque interactable building and environment — “geometry”
  2. Glass panes — “transparent”
  3. Enemy outline — “overlay” 
valorant gameplay screenshot
Image source: Imgur

b. RenderType

The other tag is RenderType, which handles the “how” in rendering. However, unlike Queue, RenderType does not have a definite list of types to choose from, instead it depends on the individual source file of the shader. However, there are 4 that will always work in unity, and you also see them in the rendering mode drop down list whenever you create a default unity material.

material rendering mode

In our shader, we utilised Opaque as the base rendering type, since our object is opaque. However, we used Fade as part of the “Pass” block. Pass blocks are considered one of the fundamental blocks of shader creation and are used for rendering things that are not affected by light; it usually appears when you create an unlit shader template. In my case, I utilised the pass block to generate an overlay that blocks the material’s emission.

We used “Fade” instead of “transparent” for one main reason. “Fade” only deals with the transparency value of the object, with the ability to “entirely fade an object out” (quoted from unity manual). This means that unlike the transparent render type, it does not take into consideration the highlights and reflections (transparent render type is typically used to render objects like glass). Thus, by using the gradient map that we had, it fades out the emission.

For transparency, black is opaque and white is transparent. The opaque part of the gradient map blocks all emission, with the textures lined up with the centre portion of the map having a brighter glow.

ScrollingMask
The scrolling mask used in the video.

Article continues after the advertisement:


2. Vertex and fragment shader

This is linked to the above section, in the way it is animated.

vertex and fragment shader

Taking a look at the script, you might be wondering:

 Why are we moving the UV under the fragment instead of the vertex shader?

This is because the vertex shader is “executed for each vertex in the scene”. If you had done 3D modelling before, you might be familiar with this, where vertices are the edges of a face in a model. This shader does not handle each pixel on the face, only the shape of the object.

Vertices in 3D modelling
The vertices are the purple and yellow dots; image source: Intro to Maya [Youtube]

This leads to the fragment shader, also known as a pixel shader. This is the shader that handles each pixel colour on the object. Thus when we move the uv in this shader, it matches the overlay pixel to a different texture pixel each time to make the animated effect, similar to the example below.

Texture movement

3. _Time

In both the animation above and the emission formula, I used _Time.y to calculate the timing. _Time refers to the time since the level was loaded and is one of Unity’s built-in shader values. We used Time.y because it is the position of the unaltered value that was provided to us.

The Time value was defined as such in the manual: Time (t/20, t, t*2, t*3)

4. Emission formula

Lastly, I will be explaining how the pulsation works, which is handled by the highlighted portion in the formula.

mission.rgb * _EmissionColor * (_EmissionIntensity + abs(frac(_Time.y * (1 / _EmissionGlowDuration)) - 0.5) * _EmissionGlow)

As mentioned in the video, _EmissionIntensity is the base intensity of the emission. The pulsating comes from the additional value that was added to this base value. For this example, I will be taking _EmissionGlowDuration as 1. Here is a simplified table of what is happening at each part of the formula:

_Time.y * (1/_EmissionGlowDuration)frac(_Time.y * (1/_EmissionGlowDuration))frac(_Time.y * (1/_EmissionGlowDuration)) - 0.5abs(frac(_Time.y * (1/_EmissionGlowDuration)) - 0.5)
1.00.0-0.50.5
1.10.1-0.40.4
1.20.2-0.30.3
1.30.3-0.20.2
1.40.4-0.10.1
1.50.500
1.60.60.10.1
1.70.70.20.2
1.80.80.30.3
1.90.90.40.4
2.00.00.50.5

The explanation for each column is as follow:

  1. We find the point in the cycle where we are at
  2. Used frac(), a shader function that returns only the fractional (i.e. the numbers after the decimal point) in the range of 0 – 1
  3. Take the fractional – 0.5 to get the recurring values from -0.5 to 0.5
  4. Obtained the absolute values [non-negative value].

This is to get values that are always in a curve, so that there will always be a gradual change in the brightness of the glow. If the values have been linear, from 0.0 – 0.9, there will always be an abrupt reset of the glow to 0 once it reaches the brightest point.

Below is a comparison between the curve and linear glows:

curve glow
Curve glow
linear glow
Linear glow (the sudden drop to no emission)

5. Conclusion

In conclusion, these are some parts of the shader that I did not go in depth during the tutorial. I hope this helps you gain a deeper understanding behind the choices made while creating the shader.


Article continues after the advertisement:


Leave a Reply

Your email address will not be published. Required fields are marked *

Note: You can use Markdown to format your comments.

This site uses Akismet to reduce spam. Learn how your comment data is processed.