280 lines
8.3 KiB
HLSL
280 lines
8.3 KiB
HLSL
#ifndef SPRITE_PIXEL_LIGHTING_INCLUDED
|
|
#define SPRITE_PIXEL_LIGHTING_INCLUDED
|
|
|
|
#include "ShaderShared.cginc"
|
|
#include "SpriteLighting.cginc"
|
|
#include "SpriteSpecular.cginc"
|
|
#include "AutoLight.cginc"
|
|
#if defined(_ALPHAPREMULTIPLY_ON)
|
|
#undef _STRAIGHT_ALPHA_INPUT
|
|
#else
|
|
#define _STRAIGHT_ALPHA_INPUT
|
|
#endif
|
|
#include "../../CGIncludes/Spine-Skeleton-Tint-Common.cginc"
|
|
|
|
////////////////////////////////////////
|
|
// Defines
|
|
//
|
|
|
|
////////////////////////////////////////
|
|
// Vertex output struct
|
|
//
|
|
|
|
#if defined(_NORMALMAP)
|
|
#define _VERTEX_LIGHTING_INDEX TEXCOORD5
|
|
#define _LIGHT_COORD_INDEX_0 6
|
|
#define _LIGHT_COORD_INDEX_1 7
|
|
#define _FOG_COORD_INDEX 8
|
|
#define _DARKCOLOR_TEXCOORD_INDEX TEXCOORD9
|
|
#else
|
|
#define _VERTEX_LIGHTING_INDEX TEXCOORD3
|
|
#define _LIGHT_COORD_INDEX_0 4
|
|
#define _LIGHT_COORD_INDEX_1 5
|
|
#define _FOG_COORD_INDEX 6
|
|
#define _DARKCOLOR_TEXCOORD_INDEX TEXCOORD7
|
|
#endif // _NORMALMAP
|
|
|
|
struct VertexOutput
|
|
{
|
|
float4 pos : SV_POSITION;
|
|
fixed4 color : COLOR;
|
|
float2 texcoord : TEXCOORD0;
|
|
float4 posWorld : TEXCOORD1;
|
|
half3 normalWorld : TEXCOORD2;
|
|
#if defined(_NORMALMAP)
|
|
half3 tangentWorld : TEXCOORD3;
|
|
half3 binormalWorld : TEXCOORD4;
|
|
#endif // _NORMALMAP
|
|
fixed3 vertexLighting : _VERTEX_LIGHTING_INDEX;
|
|
LIGHTING_COORDS(_LIGHT_COORD_INDEX_0, _LIGHT_COORD_INDEX_1)
|
|
#if defined(_FOG)
|
|
UNITY_FOG_COORDS(_FOG_COORD_INDEX)
|
|
#endif // _FOG
|
|
|
|
#if defined(_TINT_BLACK_ON)
|
|
float3 darkColor : _DARKCOLOR_TEXCOORD_INDEX;
|
|
#endif
|
|
|
|
UNITY_VERTEX_OUTPUT_STEREO
|
|
};
|
|
|
|
////////////////////////////////////////
|
|
// Light calculations
|
|
//
|
|
|
|
uniform fixed4 _LightColor0;
|
|
|
|
inline fixed3 calculateLightDiffuse(VertexOutput input, float3 normalWorld, inout fixed4 albedo)
|
|
{
|
|
//For directional lights _WorldSpaceLightPos0.w is set to zero
|
|
float3 lightWorldDirection = normalize(_WorldSpaceLightPos0.xyz - input.posWorld.xyz * _WorldSpaceLightPos0.w);
|
|
|
|
float attenuation = LIGHT_ATTENUATION(input);
|
|
float angleDot = max(0, dot(normalWorld, lightWorldDirection));
|
|
|
|
#if defined(_DIFFUSE_RAMP)
|
|
fixed3 lightDiffuse = calculateRampedDiffuse(_LightColor0.rgb, attenuation, angleDot);
|
|
#else
|
|
fixed3 lightDiffuse = _LightColor0.rgb * (attenuation * angleDot);
|
|
#endif // _DIFFUSE_RAMP
|
|
|
|
return lightDiffuse;
|
|
}
|
|
|
|
inline float3 calculateNormalWorld(VertexOutput input)
|
|
{
|
|
#if defined(_NORMALMAP)
|
|
return calculateNormalFromBumpMap(input.texcoord, input.tangentWorld, input.binormalWorld, input.normalWorld);
|
|
#else
|
|
return input.normalWorld;
|
|
#endif
|
|
}
|
|
|
|
fixed3 calculateVertexLighting(float3 posWorld, float3 normalWorld)
|
|
{
|
|
fixed3 vertexLighting = fixed3(0,0,0);
|
|
|
|
#ifdef VERTEXLIGHT_ON
|
|
//Get approximated illumination from non-important point lights
|
|
vertexLighting = Shade4PointLights ( unity_4LightPosX0, unity_4LightPosY0, unity_4LightPosZ0,
|
|
unity_LightColor[0].rgb, unity_LightColor[1].rgb, unity_LightColor[2].rgb, unity_LightColor[3].rgb,
|
|
unity_4LightAtten0, posWorld, normalWorld) * 0.5;
|
|
#endif
|
|
|
|
return vertexLighting;
|
|
}
|
|
|
|
fixed3 calculateAmbientLight(half3 normalWorld)
|
|
{
|
|
#if defined(_SPHERICAL_HARMONICS)
|
|
fixed3 ambient = ShadeSH9(half4(normalWorld, 1.0));
|
|
#else
|
|
fixed3 ambient = unity_AmbientSky.rgb;
|
|
#endif
|
|
return ambient;
|
|
}
|
|
|
|
#if defined(SPECULAR)
|
|
|
|
fixed4 calculateSpecularLight(SpecularCommonData s, float3 viewDir, float3 normal, float3 lightDir, float3 lightColor, half3 ambient)
|
|
{
|
|
SpecularLightData data = calculatePhysicsBasedSpecularLight (s.specColor, s.oneMinusReflectivity, s.smoothness, normal, viewDir, lightDir, lightColor, ambient, unity_IndirectSpecColor.rgb);
|
|
fixed4 pixel = calculateLitPixel(fixed4(s.diffColor, s.alpha), data.lighting);
|
|
pixel.rgb += data.specular * s.alpha;
|
|
return pixel;
|
|
}
|
|
|
|
fixed4 calculateSpecularLightAdditive(SpecularCommonData s, float3 viewDir, float3 normal, float3 lightDir, float3 lightColor)
|
|
{
|
|
SpecularLightData data = calculatePhysicsBasedSpecularLight (s.specColor, s.oneMinusReflectivity, s.smoothness, normal, viewDir, lightDir, lightColor, half3(0,0,0), half3(0,0,0));
|
|
fixed4 pixel = calculateAdditiveLitPixel(fixed4(s.diffColor, s.alpha), data.lighting);
|
|
pixel.rgb += data.specular * s.alpha;
|
|
return pixel;
|
|
}
|
|
|
|
#endif //SPECULAR
|
|
|
|
////////////////////////////////////////
|
|
// Vertex program
|
|
//
|
|
|
|
VertexOutput vert(VertexInput v)
|
|
{
|
|
VertexOutput output;
|
|
|
|
UNITY_SETUP_INSTANCE_ID(v);
|
|
UNITY_INITIALIZE_VERTEX_OUTPUT_STEREO(output);
|
|
|
|
output.pos = calculateLocalPos(v.vertex);
|
|
output.color = calculateVertexColor(v.color);
|
|
#if defined(_TINT_BLACK_ON)
|
|
output.darkColor = GammaToTargetSpace(half3(v.tintBlackRG.r, v.tintBlackRG.g, v.tintBlackB.r))
|
|
+ (_Black.rgb * v.color.a);
|
|
#endif
|
|
output.texcoord = calculateTextureCoord(v.texcoord);
|
|
output.posWorld = calculateWorldPos(v.vertex);
|
|
|
|
float backFaceSign = 1;
|
|
#if defined(FIXED_NORMALS_BACKFACE_RENDERING)
|
|
backFaceSign = calculateBackfacingSign(output.posWorld.xyz);
|
|
#endif
|
|
|
|
output.normalWorld = calculateSpriteWorldNormal(v, backFaceSign);
|
|
output.vertexLighting = calculateVertexLighting(output.posWorld, output.normalWorld);
|
|
|
|
#if defined(_NORMALMAP)
|
|
output.tangentWorld = calculateWorldTangent(v.tangent);
|
|
output.binormalWorld = calculateSpriteWorldBinormal(v, output.normalWorld, output.tangentWorld, backFaceSign);
|
|
#endif
|
|
|
|
TRANSFER_VERTEX_TO_FRAGMENT(output)
|
|
|
|
#if defined(_FOG)
|
|
UNITY_TRANSFER_FOG(output,output.pos);
|
|
#endif // _FOG
|
|
|
|
return output;
|
|
}
|
|
|
|
////////////////////////////////////////
|
|
// Fragment programs
|
|
//
|
|
fixed4 fragBase(VertexOutput input) : SV_Target
|
|
{
|
|
fixed4 texureColor = calculateTexturePixel(input.texcoord);
|
|
RETURN_UNLIT_IF_ADDITIVE_SLOT_TINT(texureColor, input.color, input.darkColor, _Color.a, _Black.a) // shall be called before ALPHA_CLIP
|
|
ALPHA_CLIP(texureColor, input.color)
|
|
|
|
#if defined(_TINT_BLACK_ON)
|
|
texureColor = fragTintedColor(texureColor, input.darkColor, input.color, _Color.a, _Black.a);
|
|
#endif
|
|
|
|
//Get normal direction
|
|
fixed3 normalWorld = calculateNormalWorld(input);
|
|
|
|
//Get Ambient diffuse
|
|
fixed3 ambient = calculateAmbientLight(normalWorld);
|
|
|
|
|
|
#if defined(SPECULAR)
|
|
|
|
//For directional lights _WorldSpaceLightPos0.w is set to zero
|
|
float3 lightWorldDirection = normalize(_WorldSpaceLightPos0.xyz - input.posWorld.xyz * _WorldSpaceLightPos0.w);
|
|
float attenuation = LIGHT_ATTENUATION(input);
|
|
|
|
//Returns pixel lit by light, texture color should inlcluded alpha
|
|
half3 viewDir = normalize(_WorldSpaceCameraPos - input.posWorld.xyz);
|
|
fixed4 pixel = calculateSpecularLight(getSpecularData(input.texcoord.xy, texureColor, input.color), viewDir, normalWorld, lightWorldDirection, _LightColor0.rgb * attenuation, ambient + input.vertexLighting);
|
|
|
|
APPLY_EMISSION_SPECULAR(pixel, input.texcoord)
|
|
|
|
#else
|
|
|
|
//Get primary pixel light diffuse
|
|
fixed3 diffuse = calculateLightDiffuse(input, normalWorld, texureColor);
|
|
|
|
//Combine along with vertex lighting for the base lighting pass
|
|
fixed3 lighting = ambient + diffuse + input.vertexLighting;
|
|
|
|
APPLY_EMISSION(lighting, input.texcoord)
|
|
|
|
fixed4 pixel = calculateLitPixel(texureColor, input.color, lighting);
|
|
|
|
#endif
|
|
|
|
#if defined(_RIM_LIGHTING)
|
|
pixel.rgb = applyRimLighting(input.posWorld, normalWorld, pixel);
|
|
#endif
|
|
|
|
COLORISE(pixel)
|
|
APPLY_FOG(pixel, input)
|
|
|
|
return pixel;
|
|
}
|
|
|
|
fixed4 fragAdd(VertexOutput input) : SV_Target
|
|
{
|
|
fixed4 texureColor = calculateTexturePixel(input.texcoord);
|
|
|
|
#if defined(_COLOR_ADJUST)
|
|
texureColor = adjustColor(texureColor);
|
|
#endif // _COLOR_ADJUST
|
|
|
|
ALPHA_CLIP(texureColor, input.color)
|
|
|
|
// previous fragBase pass was zwrite pass, so overlapping regions require
|
|
// full alpha applied since they are applied only once.
|
|
#if defined(_ALPHAPREMULTIPLY_ON)
|
|
texureColor.rgb /= texureColor.a == 0 ? 1 : texureColor.a;
|
|
#endif
|
|
texureColor.a = 1.0;
|
|
|
|
//Get normal direction
|
|
fixed3 normalWorld = calculateNormalWorld(input);
|
|
|
|
#if defined(SPECULAR)
|
|
|
|
//For directional lights _WorldSpaceLightPos0.w is set to zero
|
|
float3 lightWorldDirection = normalize(_WorldSpaceLightPos0.xyz - input.posWorld.xyz * _WorldSpaceLightPos0.w);
|
|
float attenuation = LIGHT_ATTENUATION(input);
|
|
|
|
half3 viewDir = normalize(_WorldSpaceCameraPos - input.posWorld.xyz);
|
|
fixed4 pixel = calculateSpecularLightAdditive(getSpecularData(input.texcoord.xy, texureColor, input.color), viewDir, normalWorld, lightWorldDirection, _LightColor0.rgb * attenuation);
|
|
|
|
#else
|
|
|
|
//Get light diffuse
|
|
fixed3 lighting = calculateLightDiffuse(input, normalWorld, texureColor);
|
|
fixed4 pixel = calculateAdditiveLitPixel(texureColor, input.color, lighting);
|
|
|
|
#endif
|
|
|
|
COLORISE_ADDITIVE(pixel)
|
|
APPLY_FOG_ADDITIVE(pixel, input)
|
|
|
|
return pixel;
|
|
}
|
|
|
|
|
|
#endif // SPRITE_PIXEL_LIGHTING_INCLUDED
|