#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