Shell texturing not work

I’m following a Shell texturing tutorial to create fur, and apply it to unity.
However, you are not creating the layers correctly.

Tutorial: Generating Fur in DirectX or OpenGL Easily - Tutorials made easy!

My Shader:

#ifndef LIT_PASS_INCLUDED
#define LIT_PASS_INCLUDED

#include "../../ShaderLibrary/Surface.hlsl"
#include "../../ShaderLibrary/Shadows.hlsl"
#include "../../ShaderLibrary/Light.hlsl"
#include "../../ShaderLibrary/BRDF.hlsl"
#include "../../ShaderLibrary/GI.hlsl"
#include "../../ShaderLibrary/Lighting.hlsl"
#include "../../ShaderLibrary/Simplex3D.hlsl"
#include "../../ShaderLibrary/ClearCoat.hlsl"

float4 _Time;

TEXTURE2D(_FurColor);
SAMPLER(sampler_FurColor);

UNITY_INSTANCING_BUFFER_START(UnityPerMaterial)
UNITY_DEFINE_INSTANCED_PROP(float, _FurLength)
UNITY_DEFINE_INSTANCED_PROP(float, _FurThinness)
UNITY_DEFINE_INSTANCED_PROP(float3, _Gravity)
UNITY_DEFINE_INSTANCED_PROP(float, _Cutoff)
UNITY_INSTANCING_BUFFER_END(UnityPerMaterial)

struct Attributes {
    float3 positionOS : POSITION;
    float3 normalOS : NORMAL;
    float4 tangentOS : TANGENT;
    float2 baseUV : TEXCOORD0;
    float2 lightmapUV: TEXCOORD1;
    float2 dynamicLightmapUV : TEXCOORD2;
    UNITY_VERTEX_INPUT_INSTANCE_ID
};

struct Varyings {
    float4 positionCS : SV_POSITION;
    float3 positionWS : VAR_POSITION;
    float3 normalWS : VAR_NORMAL;
    float4 tangentWS : VAR_TANGENT;
    float2 baseUV : VAR_BASE_UV;

    UNITY_VERTEX_INPUT_INSTANCE_ID
};

Varyings LitPassVertex (Attributes input) {
UNITY_SETUP_INSTANCE_ID(input);
    Varyings output;

  float3  P = input.positionOS + (input.normalOS * UNITY_ACCESS_INSTANCED_PROP(UnityPerMaterial, _FurLength)); // <--- Layers

    //float3 Gravity = UNITY_ACCESS_INSTANCED_PROP(UnityPerMaterial, _Gravity);

    //float k = pow(1, 3);

    //Gravity = TransformObjectToWorld(Gravity);
    //P = P + Gravity * k;

    output.positionWS = TransformObjectToWorld(float4(P, 1.0f));
    //output.positionCS = mul(mul(output.positionWS,normalize(_WorldSpaceCameraPos - output.positionWS)), _ProjectionParams);
    output.positionCS = TransformWorldToHClip(float4(output.positionWS,1));
    output.normalWS = TransformObjectToWorldNormal(input.normalOS);
    output.tangentWS = float4(TransformObjectToWorldDir(input.tangentOS.xyz), input.tangentOS.w);
    output.baseUV = input.baseUV * UNITY_ACCESS_INSTANCED_PROP(UnityPerMaterial, _FurThinness);

    return output;
}

float4 LitPassFragment(Varyings input) : SV_TARGET{
UNITY_SETUP_INSTANCE_ID(input);

    //float4 finalColor = UNITY_ACCESS_INSTANCED_PROP(UnityPerMaterial,_FurColor);
    float4 FurColour = SAMPLE_TEXTURE2D(_FurColor, sampler_FurColor, input.baseUV);

    clip(FurColour.a - UNITY_ACCESS_INSTANCED_PROP(UnityPerMaterial, _Cutoff));

return float4(FurColour);
}

#endif


It does not replicate the noise layers to create the desired effect.

You’re missing one very important aspect of that tutorial. The shader it’s presenting is for a single layer. It’s expecting the user to rendering the individual layers manually. In Unity this would be equivalent to having multiple game objects & materials each with slightly different fur lengths. That shader does not itself handle the creation of multiple layers.

I got it, Thanks!

How could I do this on unity?
And how could I get the layers to be generated by the shader?

Sorry I started a short time ago at HLSL so I still have a lot of difficulties.:face_with_spiral_eyes:

The options are:

  1. Actually have multiple spheres & materials. Most versatile. Also most annoying to use. Also only option if you want point lights to look correct while using alpha blending instead of alpha testing.

  2. Assign the fur material to the sphere multiple times. The Material array will re render the mesh for ever material in the list. Then adjust the fur length for each using MaterialPropertyBlocks and rend.SetPropertyBlock(matBlock, i) for each index of the array. Also versatile, allows for a dynamic number of layers, but does require c# coding. Also potentially supports instancing.
    Unity - Scripting API: MaterialPropertyBlock
    Unity - Scripting API: Renderer.SetPropertyBlock

  3. Use a shader with multiple passes, each with a hard coded multiplier on the fur length. Easiest to use once setup since it’s all in-shader. But you have a fixed number of layers, and guaranteed won’t instance. Also won’t support instancing, at all.

  4. Use a geometry shader stage to amplify the geometry. Also only requires the shader to work, and does support a dynamic number of layers! Doesn’t work on a lot of mobile devices, including anything recent from Apple.

1 Like
  1. like 1 but only one material, encode the length on the vertex data, the shader read it and adjust length, allow painting length locally.

This would either need to be done by having multiple otherwise identical models with slightly different vertex data, or use a c# script to update the duplicate mesh’s vertex data.

Once you’re going down this route you might as well just model the shells into the mesh manually.