231 lines
6.3 KiB
HLSL
231 lines
6.3 KiB
HLSL
#ifndef SPRITE_LIGHTING_INCLUDED
|
|
#define SPRITE_LIGHTING_INCLUDED
|
|
|
|
//Check for using mesh normals
|
|
#if !defined(_FIXED_NORMALS_VIEWSPACE) && !defined(_FIXED_NORMALS_VIEWSPACE_BACKFACE) && !defined(_FIXED_NORMALS_MODELSPACE) && !defined(_FIXED_NORMALS_MODELSPACE_BACKFACE) && !defined(_FIXED_NORMALS_WORLDSPACE)
|
|
#define MESH_NORMALS
|
|
#endif
|
|
|
|
//Check for fixing backfacing tangents
|
|
#if defined(_FIXED_NORMALS_VIEWSPACE_BACKFACE) || defined(_FIXED_NORMALS_MODELSPACE_BACKFACE)
|
|
#define FIXED_NORMALS_BACKFACE_RENDERING
|
|
#endif
|
|
|
|
////////////////////////////////////////
|
|
// Vertex structs
|
|
//
|
|
|
|
struct VertexInput
|
|
{
|
|
float4 vertex : POSITION;
|
|
float4 texcoord : TEXCOORD0;
|
|
float4 color : COLOR;
|
|
#if defined(MESH_NORMALS)
|
|
float3 normal : NORMAL;
|
|
#endif // _FIXED_NORMALS
|
|
#if defined(_NORMALMAP)
|
|
float4 tangent : TANGENT;
|
|
#endif // _NORMALMAP
|
|
#if defined(_TINT_BLACK_ON)
|
|
float2 tintBlackRG : TEXCOORD1;
|
|
float2 tintBlackB : TEXCOORD2;
|
|
#endif
|
|
UNITY_VERTEX_INPUT_INSTANCE_ID
|
|
|
|
};
|
|
|
|
////////////////////////////////////////
|
|
// Normal functions
|
|
//
|
|
|
|
#if !defined(USE_LWRP) && !defined(USE_URP)
|
|
uniform float4 _FixedNormal = float4(0, 0, 1, 1);
|
|
#endif
|
|
|
|
inline float3 getFixedNormal()
|
|
{
|
|
return _FixedNormal.xyz;
|
|
}
|
|
|
|
inline float calculateBackfacingSign(float3 worldPos)
|
|
{
|
|
//If we're using fixed normals and mesh is facing away from camera, flip tangentSign
|
|
//Unity uses a left handed coordinate system so camera always looks down the negative z axis
|
|
float3 cameraForward = float3(0,0,-1);
|
|
float3 meshWorldForward = mul((float3x3)unity_ObjectToWorld, cameraForward);
|
|
float3 toCamera = _WorldSpaceCameraPos - worldPos;
|
|
return sign(dot(toCamera, meshWorldForward));
|
|
}
|
|
|
|
inline half3 calculateSpriteWorldNormal(VertexInput vertex, float backFaceSign)
|
|
{
|
|
#if defined(MESH_NORMALS)
|
|
|
|
return calculateWorldNormal(vertex.normal);
|
|
|
|
#else // !MESH_NORMALS
|
|
|
|
float3 normal = getFixedNormal();
|
|
|
|
#if defined(_FIXED_NORMALS_VIEWSPACE) || defined(_FIXED_NORMALS_VIEWSPACE_BACKFACE)
|
|
//View space fixed normal
|
|
//Rotate fixed normal by inverse view matrix to convert the fixed normal into world space
|
|
float3x3 invView = transpose((float3x3)UNITY_MATRIX_V);
|
|
return normalize(mul(invView, normal));
|
|
#elif defined (_FIXED_NORMALS_WORLDSPACE)
|
|
//World space fixed normal
|
|
return normal;
|
|
#else
|
|
//Model space fixed normal.
|
|
#if defined(FIXED_NORMALS_BACKFACE_RENDERING)
|
|
//If back face rendering is enabled and the sprite is facing away from the camera (ie we're rendering the backface) then need to flip the normal
|
|
normal *= backFaceSign;
|
|
#endif
|
|
return calculateWorldNormal(normal);
|
|
#endif
|
|
|
|
#endif // !MESH_NORMALS
|
|
}
|
|
|
|
inline half3 calculateSpriteViewNormal(VertexInput vertex, float backFaceSign)
|
|
{
|
|
#if defined(MESH_NORMALS)
|
|
|
|
return normalize(mul((float3x3)UNITY_MATRIX_IT_MV, vertex.normal));
|
|
|
|
#else // !MESH_NORMALS
|
|
|
|
float3 normal = getFixedNormal();
|
|
|
|
#if defined(_FIXED_NORMALS_VIEWSPACE) || defined(_FIXED_NORMALS_VIEWSPACE_BACKFACE)
|
|
//View space fixed normal
|
|
return normal;
|
|
#elif defined (_FIXED_NORMALS_WORLDSPACE)
|
|
//World space fixed normal
|
|
return normalize(mul((float3x3)UNITY_MATRIX_V, normal));
|
|
#else
|
|
//Model space fixed normal
|
|
#if defined(FIXED_NORMALS_BACKFACE_RENDERING)
|
|
//If back face rendering is enabled and the sprite is facing away from the camera (ie we're rendering the backface) then need to flip the normal
|
|
normal *= backFaceSign;
|
|
#endif
|
|
return normalize(mul((float3x3)UNITY_MATRIX_IT_MV, normal));
|
|
#endif
|
|
|
|
#endif // !MESH_NORMALS
|
|
}
|
|
|
|
////////////////////////////////////////
|
|
// Normal map functions
|
|
//
|
|
|
|
#if defined(_NORMALMAP)
|
|
|
|
inline half3 calculateSpriteWorldBinormal(VertexInput vertex, half3 normalWorld, half3 tangentWorld, float backFaceSign)
|
|
{
|
|
float tangentSign = vertex.tangent.w;
|
|
|
|
#if defined(FIXED_NORMALS_BACKFACE_RENDERING)
|
|
tangentSign *= backFaceSign;
|
|
#endif
|
|
|
|
return calculateWorldBinormal(normalWorld, tangentWorld, tangentSign);
|
|
}
|
|
|
|
#endif // _NORMALMAP
|
|
|
|
#if defined(_DIFFUSE_RAMP)
|
|
|
|
|
|
////////////////////////////////////////
|
|
// Diffuse ramp functions
|
|
//
|
|
|
|
uniform sampler2D _DiffuseRamp;
|
|
|
|
inline fixed3 calculateDiffuseRamp(float ramp)
|
|
{
|
|
return tex2D(_DiffuseRamp, float2(ramp, ramp)).rgb;
|
|
}
|
|
|
|
inline fixed3 calculateRampedDiffuse(fixed3 lightColor, float attenuation, float angleDot)
|
|
{
|
|
#if defined(_FULLRANGE_HARD_RAMP)
|
|
float d = angleDot;
|
|
half3 ramp = calculateDiffuseRamp(d);
|
|
return lightColor * ramp * attenuation;
|
|
#elif defined(_FULLRANGE_SOFT_RAMP)
|
|
float d = angleDot;
|
|
half3 ramp = calculateDiffuseRamp(d * attenuation);
|
|
return lightColor * ramp;
|
|
#elif defined(_OLD_SOFT_RAMP)
|
|
// for unmodified behaviour with existing projects when
|
|
// the HARD_DIFFUSE_RAMP define was disabled in this file.
|
|
// uses only the right half of the ramp texture, as
|
|
// negative angleDot is clamped to [0,1] before.
|
|
float d = angleDot * 0.5 + 0.5;
|
|
half3 ramp = calculateDiffuseRamp(d);
|
|
return lightColor * ramp * (attenuation * 2);
|
|
#else // _OLD_HARD_RAMP
|
|
// old default, for unmodified behaviour with existing projects,
|
|
// uses only the right half of the ramp texture, as
|
|
// negative angleDot is clamped to [0,1] before.
|
|
float d = angleDot * 0.5 + 0.5;
|
|
half3 ramp = calculateDiffuseRamp(d * attenuation * 2);
|
|
return lightColor * ramp;
|
|
#endif
|
|
}
|
|
#endif // _DIFFUSE_RAMP
|
|
|
|
////////////////////////////////////////
|
|
// Rim Lighting functions
|
|
//
|
|
|
|
#ifdef _RIM_LIGHTING
|
|
#if !defined(USE_LWRP) && !defined(USE_URP)
|
|
uniform float _RimPower;
|
|
uniform fixed4 _RimColor;
|
|
#endif
|
|
|
|
inline fixed3 applyRimLighting(fixed3 posWorld, fixed3 normalWorld, fixed4 pixel) : SV_Target
|
|
{
|
|
fixed3 viewDir = normalize(_WorldSpaceCameraPos - posWorld);
|
|
float invDot = 1.0 - saturate(dot(normalWorld, viewDir));
|
|
float rimPower = pow(invDot, _RimPower);
|
|
float rim = saturate(rimPower * _RimColor.a);
|
|
|
|
#if defined(_DIFFUSE_RAMP)
|
|
rim = calculateDiffuseRamp(rim).r;
|
|
#endif
|
|
|
|
return lerp(pixel.rgb, _RimColor.xyz * pixel.a, rim);
|
|
}
|
|
|
|
#endif //_RIM_LIGHTING
|
|
|
|
////////////////////////////////////////
|
|
// Emission functions
|
|
//
|
|
|
|
#ifdef _EMISSION
|
|
|
|
uniform sampler2D _EmissionMap;
|
|
|
|
#if !defined(USE_LWRP) && !defined(USE_URP)
|
|
uniform fixed4 _EmissionColor;
|
|
uniform float _EmissionPower;
|
|
#endif
|
|
|
|
|
|
#define APPLY_EMISSION(diffuse, uv) diffuse += tex2D(_EmissionMap, uv).rgb * _EmissionColor.rgb * _EmissionPower;
|
|
#define APPLY_EMISSION_SPECULAR(pixel, uv) pixel.rgb += (tex2D(_EmissionMap, uv).rgb * _EmissionColor.rgb * _EmissionPower) * pixel.a;
|
|
|
|
#else //!_EMISSION
|
|
|
|
#define APPLY_EMISSION(diffuse, uv)
|
|
#define APPLY_EMISSION_SPECULAR(pixel, uv)
|
|
|
|
#endif //!_EMISSION
|
|
|
|
#endif // SPRITE_LIGHTING_INCLUDED
|