Fancy Sprite Lighting (Normals and Specular)

EDIT: Turns out I need to learn more. To get lighting working as it should, you just need to add a tangent line to your vertex shader, giving you something like:

		void vert (inout appdata_full v, out Input o) {
			#if defined(PIXELSNAP_ON)  !defined(SHADER_API_FLASH)
			v.vertex = UnityPixelSnap (v.vertex);
			#endif

			v.normal = float3(0, 0, -1);
			v.tangent = float4(1, 0, 0, 1);
		}

Thanks to PurulentExudate on reddit for their comment about this. As such, please ignore the code below. It’s bad and, well, bad.

When you’re working with shaders on sprites, the provided lightDir doesn’t work so you have to calculate it yourself.

I’ve got a little unity web player with my test project at http://stendec.me/LightDir/LightDir.html and you can download the project from http://stendec.me/LightDir/LightDirBug.7z

I might be doing something wrong (and if I am, please let me know), but lightDir is just flat for a sprite. My first attempt at normals and specular lighting was this:

half4 LightingSpriteNormal (SurfaceOutputCustom s, half3 lightDir, half3 viewDir, half atten) {
	half3 h = normalize(lightDir + viewDir);
	fixed diff = max(0, dot(s.Normal, lightDir));
	
	float nh = max(0, dot(s.Normal, h));
	float spec = pow(nh, s.Specular * 128.0) * s.Gloss;
	
	fixed4 c;
	c.rgb = (s.Albedo * _LightColor0.rgb * diff + _LightColor0.rgb * _SpecColor.rgb * spec) * (atten * 2);
	c.a = s.Alpha + _LightColor0.a * _SpecColor.a * spec * atten;
	return c;
}

Some of you may notice that that’s exactly the same as LightingBlinnPhong. It doesn’t work, because it relies on lightDir to be correct, which it’s not.

In order to make normals and specular work, I had to call ObjSpaceLightDir with code that ended up looking like:

void vert (inout appdata_full v, out Input o) {
	v.normal = float3(0, 0, 1);
	UNITY_INITIALIZE_OUTPUT(Input, o);
	o.lightDir = ObjSpaceLightDir(v.vertex);
}

void surf (Input IN, inout SurfaceOutputCustom o) {
	half4 c = tex2D (_MainTex, IN.uv_MainTex);
	o.Albedo = c.rgb;
	o.Alpha = c.a;
	o.Normal = UnpackNormal(tex2D(_NormTex, IN.uv_NormTex));
	o.lightDir = IN.lightDir;	
	o.Gloss = 1;
	o.Specular = _SpecPower;
}
		
half4 LightingSpriteNormal (SurfaceOutputCustom s, half3 lightDir, half3 viewDir, half atten) {
	half3 h = normalize(s.lightDir + viewDir);
	fixed diff = max(0, dot(s.Normal, s.lightDir));
	
	float nh = max(0, dot(s.Normal, h));
	float spec = pow(nh, s.Specular * 128.0) * s.Gloss;
	
	fixed4 c;
	c.rgb = (s.Albedo * _LightColor0.rgb * diff + _LightColor0.rgb * _SpecColor.rgb * spec) * (atten * 2);
	c.a = s.Alpha + _LightColor0.a * _SpecColor.a * spec * atten;
	return c;
}

I’ve attached the full shader file to this post.

1420598–74768–$Sprite Normals.shader (2.25 KB)

Yeah, UT has said that SpriteRenderer is not meant to be affected by dynamic lights -.- So thanks for putting your findings up here :slight_smile: