I am trying to create a transparent shader using the alpha map.
I referred to the following threads and manuals.
And I wrote a simple prototype shader.
Shader "Transparent/Diffuse ZWrite" {
Properties {
_Color ("Main Color", Color) = (1,1,1,1)
_MainTex ("Base (RGB) Trans (A)", 2D) = "white" {}
}
SubShader {
Tags {"Queue"="Transparent" "IgnoreProjector"="True" "RenderType"="Transparent"}
//Tags { "Queue"="AlphaTest" "IgnoreProjector"="true" "RenderType"="TransparentCutout" }
LOD 200
Cull Off
Blend SrcAlpha OneMinusSrcAlpha
// extra pass that renders to depth buffer only
Pass {
ZWrite On
ColorMask 0
Cull Off
}
ZWrite Off
Cull Off
CGPROGRAM
#pragma surface surf Standard fullforwardshadows alpha:blend
//#pragma surface surf Standard fullforwardshadows alphatest:_Cutout
//#pragma surface surf Standard fullforwardshadows alpha:fade
sampler2D _MainTex;
fixed4 _Color;
struct Input {
float2 uv_MainTex;
};
//void surf (Input IN, inout SurfaceOutput o) {
void surf (Input IN, inout SurfaceOutputStandard o) {
fixed4 c = tex2D(_MainTex, IN.uv_MainTex) * _Color;
o.Albedo = c.rgb;
o.Alpha = c.a;
}
ENDCG
// paste in forward rendering passes from Transparent/Diffuse
//UsePass "Transparent/Diffuse/FORWARD"
}
Fallback "Transparent/VertexLit"
}
As a result, it was able to draw the depth, there is a problem with the transparency by alpha.
Please see the attached image (also to compensate for my strange English).
The results of the rendering seem to be different from unity4.
Although polygon has become transparent by the texture, it “clip” to the object behind should look.(Shader seems to ignore the object in the back really.)
You have Zwrite enabled, meaning a pixel rendered will write its depth in the Zbuffer, and for any other pixel at the same place, it will be discarded if its depth is greater than the one in the depth buffer.
What is happening is that your plane that stick toward the cam get rendered first, and pixel write their depth (even with alpha == 0, it just make it blend with that’s already there, i.e. the background, but the pixel still rendered & write in depth buffer)
So when the plane facing the cam get rendered, its pixel behind don’t even reach the pixel shader stage, they are discarded because for the GPU, there is a pixel in front. So they don’t get blended with that’s already there. Shader DO ignore the object behind, the GPU don’t send those pixel to the pixel shader
That why you disable depth write for transparent mesh usually.
Perhaps, I was somehow able to understand.
That means, Is it that it cannot sort a transparent object?
For example, does the particle system sort it by other methods?
Seems right to me. A transparent object will get clipped correctly by opaque zwrite geometry in the right places. For transparent sorting, it’s fine if it’s a separate object, you cannot sort transparent object without 2 passes (ie one pass for alpha cutout and one pass for drawing blended colour).
Otherwise if each transparent object is an independent mesh it will be sorted by view distance or whatever. Problems arise when you combine transparent meshes.
Transparency is problematic for real-time graphics. It is technically possible to do, but requires such overhead and additional work that compromises must be made. That said there are also some tricks that can disguise the problem such as using additive blending instead of alpha blending, but that’s no use if you need alpha blending.
In general a real-time 3D engine will only sort transparent models by depth ( and then render back to front), but it will make no attempt to sort polygons within a model/mesh. The reason for not sorting per polygon is that it can break batching (i.e. if you mesh uses more than 1 material), secondly it can be an expensive process and finally intersecting polygons can never have a correct draw order unless they are divided/split (i.e The Painters Algorithm issue). Further more even if you sorted within a mesh it does not then resolve the problems of overlapping meshes. For that to work you’d need to sort against both meshes and … well you can see the madness that can quickly escalate to.
Now the Unity particle system offers two methods that may solve the sorting issue. First you can use additive blending, which is draw order independent, meaning it does matter which triangles are drawn first the result is always correct. This can be great for fx like explosions or sparks, but is no use if you need normal alpha blending. The second method is that it provides the option to self sort the triangles within the particle system, which mostly works but can be expensive on the cpu. Unfortunately this second method suffers from the same issues outlined above in terms of sorting polygons within a mesh.
So where does that leave you?
Well the simplest fix, though not a very efficient one, would be to divide up each of you three planes into four quadrants each. In other words split each plane into 4 quadrants and keep each as a separate model. That way the Unity sorting should ensure the correct draw order. If this is for a different model than the one you’ve shown then the same technique may help, but its unlikely and it will just become more in-efficient.
You could look into writing your own script for dynamic polygon sorting within a mesh. In fact i’m sure someone would have already done this, so maybe just search google.
There are other options such a blended order independent transparency, but those are generally more complex to set up.
I seemed to think about it slightly lightly…
CPU/GPU power seems to be necessary to get the result that I thought about at first than I expected it.
So I decide to think including other approach.
(Possibly I should have said this first, but I was going to make a shader for the cloud.)
hippocoder, may I ask you a question one more?(Because you like a particle of expert.)
When I attached the shader using normal map to a particle, it was like a normal direction has been strange.
Does the particle system not calculate the tangent?
(If I should make a thread separately, I will do so. sorry.)