What I want to make is a shader that renders only the lighting (shadows/highlights/etc) generated by a bump map. I want the rest to blend with what is behind it. I want to use this for tire tracks on dirt kind of like the picture below.
There’s no simple way with just a surface shader unless you do a somewhat expensive method, which is to do a GrabPass, set that as your output color, as well as sample the DepthNormals texture and blend your normal with that and set it as output. But an issue with this is that the final normals output are going to be adjusted by the object you’re rendering this shader on…
Another option, if using Deferred rendering, is to simply use a vert/frag shader to output directly to the Normals GBuffer and everything works simply, no grab pass needed.
Okay, got it, thanks Invertex. Unfortunately both of those options won’t work for the project. I guess I was hoping to be able to toss out the Albedo output data and leave behind only the lighting generated by the Normals.
Well, technically that’s what a surface shader in Transparent mode does when it’s at 100% transparency. All you see is the specular highlights. If that’s all you want that could work. Just output your o.Normal as you want and full transparency alpha.
Only problem with that approach would be even the flat areas of this overlaid normals mesh are going to affect lighting if you ever look at the surface off-angle instead of perfectly straight-on at all times. If this is an orthographic view you’re working with then it shouldn’t be a big issue, but some bright lights could still show even looking directly at the surface unless the Smoothness is 0 or 1
You could get around this somewhat by discard/clip()'ing the pixels away where you normal map is a flat color though (but that wouldn’t look great in many cases either as specular might end up being missing at the peaks of bumped areas for example). The other option would be to add your own lightingModel to the shader as well (since you’re working with Forward rendering right?), that way you can control how the lighting is applied.
When I set the alpha to 0, the bumpmap/normal highlights and shadows are set to the same alpha, so the tire tracks disappear. I’m not using any specular since the surface is very rough (dirt). It seems that Alpha effects not just the diffuse but all other outputs.
Well, without any specular, normal maps don’t do much. Even dirt has some specular, it’s just rough, but it’s there enough that bumps would affect the rough lighting of it to make some shape visible.
Bump maps don’t cast shadows, they’ll simply be darker the greater the angle away from any light sources is. And making an object transparent will indeed lose that part of it, which is why I suggested making your own lightingModel function. This would allow you to output the bright and dark shading and do a multiplicative shader blending. All a surf() function is doing is providing basic inputs to a given lighting model to then process how it sees fit. You can take control of that next step with a custom lightingModel. The lightingModel will only be specific to that shader, it won’t affect the rest of the scene.
Another option would be to just make it a vert/frag shader, multiplicative blend mode, and calculate a black/white output based on light direction/intensity and environment lighting.