Learning to make a Simple Procedural Skybox in Unity URP

March 14, 2025

Recently I've been asked to implement a day/night cycle for an upcoming game I've been working on. In practice, this is easily achievable using Unity's built-in procedural skybox shader.

Plenty of resources on YouTube, for example, here's one that shows you how to get one up and running in 6 minutes: https://www.youtube.com/watch?v=m9hj9PdO328

To sum things up, adjusting the rotation of the directional light, changing the ambient lighting and fog color according to the current time of day is enough to achieve a believable day/night cycle in Unity.

However, what if we wanted to add stars, or clouds?

What I tried

  1. Create a separate dome mesh, and apply a material to it, making it transparent to allow it to blend with the background skybox.
  2. Attempt to modify the built-in Unity Procedural Skybox shader.
  3. Made my own procedural skybox shader.

Method 1: Separate Dome Mesh

I followed this amazing tutorial by Digvijaysinh Gohil to create a material for stars.

Aside from the stars shader, this video is a great introduction to how to begin using ShaderGraph to create your own shaders in Unity.

It briefly touches on UVs, procedural noise, useful math nodes, colors, gradients, and simple animation techniques to make the stars twinkle.

I made a simple dome mesh in blender, imported it into Unity, and then wasted a day or two struggling to get it to work properly with the camera. TLDR: Far plane clipping, temporal aliasing, camera layer stack, culling masks, etc.

This method is actually perfectly fine, but I wasn't too satisfied with the end result because of how finnicky the setup felt. So I decided to explore other options for more fine-grained control over the skybox.

Method 2: Modifying the built-in shader.

I don't know how to, so I decided to roll my own in the end.

Method 3: Time to learn ShaderLab

This time, I decided to go at it from scratch, learning how to write HLSL and ShaderLab.

Here's great resource to get started with shader programming: Book of Shaders It introduces a whole bunch of core concepts for drawing things in shaders.

See also this great article by Jannik Boysen on creating a procedural skybox shader in Shader Graph.

Sky

I started with a simple Unlit surface surface (URP), and began by filling in the sky color.

Below is some of the simplest code possible for a primitive procedural skybox. All it has is a simple sun disc, and some color interpolation between the horizon and the sky.

Shader "Unlit/ProceduralSkybox"
{
    Properties
    {
        _SunDirection ("Sun Direction", Vector) = (0., 0., 0., 0.)
        _SunColor ("Sun Color", Color) = (1., 1., 1., 1.)
        _SkyColor ("Sky Color", Color) = (1., 1., 1., 1.)
        _HorizonColor ("Horizon Color", Color) = (1., 1., 1., 1.)
    }

    SubShader
    {
        Tags { "Queue"="Background" "RenderType"="Background" "PreviewType"="Skybox" }
        Cull Off
        ZWrite Off

        Pass
        {
            CGPROGRAM
            #pragma vertex vert
            #pragma fragment frag

            #include "UnityCG.cginc"

            struct appdata
            {
                float4 vertex : POSITION;
                float3 uv : TEXCOORD0;
            };

            struct v2f
            {
                float3 uv : TEXCOORD0;
                float4 vertex : SV_POSITION;
            };

            float3 _SunDirection;
            half4 _SunColor;
            half4 _SkyColor;
            half4 _HorizonColor;

            v2f vert (appdata v)
            {
                v2f o;

                o.vertex = UnityObjectToClipPos(v.vertex);
                o.uv = v.uv;

                return o;
            }

            fixed4 frag (v2f i) : SV_Target
            {
                float3 pos = normalize(i.uv);
                float horizon = smoothstep(-0.1, 0.1, pos.y);

                half4 skyColor = lerp(_HorizonColor, _SkyColor, horizon);
                float sunFactor = dot(normalize(_SunDirection), pos);
                float sunMask = 1 - smoothstep(1.0, 0.99, sunFactor);

                skyColor += sunMask * _SunColor;
                return skyColor;
            }
            ENDCG
        }
    }
}

Result

Primitive Sky

Stars

We can use some layered value noise to generate a star field.

// Pseudo-random function
float rand(float3 co)
{
    return frac(sin(dot(co.xyz, float3(12.9898, 78.233, 45.5432))) * 43758.5453);
}

// Value noise function
float valueNoise(float3 p)
{
    float3 ip = floor(p);
    float3 fp = frac(p);
    
    float3 smooth = fp * fp * (3.0 - 2.0 * fp);
    
    float n000 = rand(ip);
    float n001 = rand(ip + float3(0, 0, 1));
    float n010 = rand(ip + float3(0, 1, 0));
    float n011 = rand(ip + float3(0, 1, 1));
    float n100 = rand(ip + float3(1, 0, 0));
    float n101 = rand(ip + float3(1, 0, 1));
    float n110 = rand(ip + float3(1, 1, 0));
    float n111 = rand(ip + float3(1, 1, 1));
    
    // Interpolate along x
    float n00 = lerp(n000, n100, smooth.x);
    float n01 = lerp(n001, n101, smooth.x);
    float n10 = lerp(n010, n110, smooth.x);
    float n11 = lerp(n011, n111, smooth.x);
    
    // Interpolate along y
    float n0 = lerp(n00, n10, smooth.y);
    float n1 = lerp(n01, n11, smooth.y);
    
    // Interpolate along z
    return lerp(n0, n1, smooth.z);
}

half4 generateStars(float3 viewDir) {
    float3 dir = viewDir;
    
    // Scale the direction to control star density
    float3 p = dir * _StarDensity;
    
    // Initialize color
    float4 col = float4(0, 0, 0, 0);
    
    // Star field generation
    for (int layer = 0; layer < _StarLayers; layer++)
    {
        // Add different scale noise layers
        float3 layerPos = p * (1.0 + layer * 0.5);
        
        // Get base star value
        float star = valueNoise(layerPos);
        
        // Threshold to create stars (adjust threshold for density)
        float threshold = 0.97 - (layer * 0.01);
        
        if (star > threshold)
        {
            // Remap star value to 0-1 range for this threshold
            star = (star - threshold) / (1.0 - threshold);
            
            // Apply star size adjustment (make smaller for more realism)
            star = smoothstep(0, _StarSize, star);
            
            // Create twinkle effect
            float twinkle = sin(_Time.y * _TwinkleSpeed + layerPos.x * 10 + layerPos.y * 10 + layerPos.z * 10);
            twinkle = lerp(1, (twinkle * 0.5 + 0.5), _TwinkleAmount);
            
            // Apply different colors based on noise
            float starTemperature = rand(floor(layerPos * 100));
            float4 starColor = lerp(_StarColor1, _StarColor2, starTemperature);
            
            // Adjust brightness and color
            float brightness = star * _StarIntensity * twinkle;
            col += starColor * brightness;
        }
    }
    
    // Apply overall visibility from day-night cycle
    col.rgba *= _Visibility;
    
    return col;
}

WIP

Clouds

Clouds can be achieved by sampling a noise or otherwise other cloudy texture.

Playing around with Math

Lerp

Smoothstep

Noise Erosion

To add more depth to our clouds, we can use a separate noise texture to manipulate the cloud texels.

Color Minus

One Minus

WIP

Cloud Mask

Scrolling

Common Pitfalls

1. Not using Seamless textures

  • Problem: My clouds are ugly and don't wrap properly!
  • Solution: Use a seamless texture.

2. Wrong mipmapping settings

  • Problem: There's a weird line in the middle of my skybox!
  • Solution: Disable mipmapping.

3. Not correctly mapping UVs to spherical coordinates

  • Problem: My clouds are all stretched and weird!
  • Solution: Properly map your UVs.

4. Texture Compression

  • Problem: My clouds look kinda ugly/jagged.
  • Solution: Make sure your textures are configured correctly, for the sharpest results, use Point Filtering with no compression.

Additional Resources

Results

Day Sky Night Sky

WIP

  • Color Math and operations (One-minus, Saturate, Smoothstep, Lerp, Masking)
  • Noise, erosion, scrolling UVs, texture wrapping
  • Post Processing (Bloom, Color Filters, Tonemapping, etc.)
  • Volumetric Clouds (Raymarching)