forked from przemyslawzaworski/Unity3D-CG-programming
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathpom.shader
124 lines (118 loc) · 4.01 KB
/
pom.shader
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
Shader "Diffuse Parallax Occlusion Mapping "
{
Properties
{
_MainTex ("Diffuse map (RGB)", 2D) = "white" {}
_HeightMap ("Height map (R)", 2D) = "white" {}
_Parallax ("Height scale", Range (0.005, 0.1)) = 0.08
_ParallaxSamples ("Parallax samples", Range (10, 100)) = 40
}
SubShader
{
Pass
{
Tags { "LightMode" = "ForwardBase" }
CGPROGRAM
#pragma vertex vertex_shader
#pragma fragment pixel_shader
#pragma target 3.0
sampler2D _MainTex;
sampler2D _HeightMap;
float _Parallax;
float _ParallaxSamples;
uniform float4 _LightColor0;
struct vertexInput
{
float4 vertex: POSITION;
float3 normal: NORMAL;
float2 texcoord: TEXCOORD0;
float4 tangent : TANGENT;
};
struct vertexOutput
{
float4 pos: SV_POSITION;
float2 tex: TEXCOORD0;
float4 posWorld: TEXCOORD1;
float3 tSpace0 : TEXCOORD2;
float3 tSpace1 : TEXCOORD3;
float3 tSpace2 : TEXCOORD4;
float3 normal : TEXCOORD5;
};
vertexOutput vertex_shader( vertexInput v )
{
vertexOutput o;
o.posWorld = mul(unity_ObjectToWorld, v.vertex);
fixed3 worldNormal = mul(v.normal.xyz, (float3x3)unity_WorldToObject);
fixed3 worldTangent = normalize(mul((float3x3)unity_ObjectToWorld,v.tangent.xyz ));
fixed3 worldBinormal = cross(worldNormal, worldTangent) * v.tangent.w;
o.tSpace0 = float3(worldTangent.x, worldBinormal.x, worldNormal.x);
o.tSpace1 = float3(worldTangent.y, worldBinormal.y, worldNormal.y);
o.tSpace2 = float3(worldTangent.z, worldBinormal.z, worldNormal.z);
o.pos = UnityObjectToClipPos( v.vertex );
o.tex = v.texcoord;
o.normal=v.normal;
return o;
}
float4 pixel_shader( vertexOutput i ): SV_TARGET
{
float3 normalDirection = normalize(i.normal);
fixed3 worldViewDir = normalize(_WorldSpaceCameraPos.xyz - i.posWorld.xyz);
fixed3 viewDir = i.tSpace0.xyz * worldViewDir.x + i.tSpace1.xyz * worldViewDir.y + i.tSpace2.xyz * worldViewDir.z;
float2 vParallaxDirection = normalize( viewDir.xy );
float fLength = length( viewDir );
float fParallaxLength = sqrt( fLength * fLength - viewDir.z * viewDir.z ) / viewDir.z;
float2 vParallaxOffsetTS = vParallaxDirection * fParallaxLength * _Parallax ;
float nMinSamples = 6;
float nMaxSamples = min(_ParallaxSamples, 100);
int nNumSamples = (int)(lerp( nMinSamples, nMaxSamples, 1-dot(worldViewDir , i.normal ) ));
float fStepSize = 1.0 / (float)nNumSamples;
int nStepIndex = 0;
float fCurrHeight = 0.0;
float fPrevHeight = 1.0;
float2 vTexOffsetPerStep = fStepSize * vParallaxOffsetTS;
float2 vTexCurrentOffset = i.tex.xy;
float fCurrentBound = 1.0;
float fParallaxAmount = 0.0;
float2 pt1 = 0;
float2 pt2 = 0;
float2 dx = ddx(i.tex.xy);
float2 dy = ddy(i.tex.xy);
for (nStepIndex = 0; nStepIndex < nNumSamples; nStepIndex++)
{
vTexCurrentOffset -= vTexOffsetPerStep;
fCurrHeight = tex2D( _HeightMap, vTexCurrentOffset,dx,dy).r;
fCurrentBound -= fStepSize;
if ( fCurrHeight > fCurrentBound )
{
pt1 = float2( fCurrentBound, fCurrHeight );
pt2 = float2( fCurrentBound + fStepSize, fPrevHeight );
nStepIndex = nNumSamples + 1; //Exit loop
fPrevHeight = fCurrHeight;
}
else
{
fPrevHeight = fCurrHeight;
}
}
float fDelta2 = pt2.x - pt2.y;
float fDelta1 = pt1.x - pt1.y;
float fDenominator = fDelta2 - fDelta1;
if ( fDenominator == 0.0f )
{
fParallaxAmount = 0.0f;
}
else
{
fParallaxAmount = (pt1.x * fDelta2 - pt2.x * fDelta1 ) / fDenominator;
}
i.tex.xy -= vParallaxOffsetTS * (1 - fParallaxAmount );
float3 lightDirection = normalize( _WorldSpaceLightPos0.xyz );
float3 diffuseReflection = _LightColor0.rgb * saturate( dot( normalDirection, lightDirection ) );
float3 color = diffuseReflection + UNITY_LIGHTMODEL_AMBIENT.rgb;
float4 tex = tex2D( _MainTex, i.tex.xy );
return float4( tex.xyz * color , 1.0);
}
ENDCG
}
}
}