1
+ Shader "Hidden/Post FX/Bloom"
2
+ {
3
+ Properties
4
+ {
5
+ _MainTex ( "" , 2D ) = "" {}
6
+ _BaseTex ( "" , 2D ) = "" {}
7
+ _AutoExposure ( "" , 2D ) = "" {}
8
+ }
9
+
10
+ CGINCLUDE
11
+ #pragma target 3.0
12
+ float4 _MainTex_ST;
13
+ float4 _MainTex_TexelSize;
14
+ sampler2D _MainTex;
15
+ ENDCG
16
+
17
+ SubShader
18
+ {
19
+ ZTest Always Cull Off ZWrite Off
20
+
21
+ Pass
22
+ {
23
+ CGPROGRAM
24
+ #pragma multi_compile __ ANTI_FLICKER
25
+ #pragma multi_compile __ UNITY_COLORSPACE_GAMMA
26
+ #pragma vertex SetVertexShader
27
+ #pragma fragment SetPixelShader
28
+
29
+ sampler2D _AutoExposure;
30
+ float3 _Curve;
31
+ float _Threshold;
32
+ float _PrefilterOffs;
33
+
34
+ void SetVertexShader (inout float4 vertex : POSITION , inout float2 texcoord : TEXCOORD0 )
35
+ {
36
+ vertex = UnityObjectToClipPos (vertex);
37
+ }
38
+
39
+ half Brightness (half3 c)
40
+ {
41
+ return max (c.x, max (c.y, c.z));
42
+ }
43
+
44
+ half3 Median (half3 a, half3 b, half3 c)
45
+ {
46
+ return a + b + c - min (min (a, b), c) - max (max (a, b), c);
47
+ }
48
+
49
+ half4 FetchAutoExposed (sampler2D tex, float2 uv)
50
+ {
51
+ float autoExposure = 1.0 ;
52
+ autoExposure = tex2D (_AutoExposure, uv).r;
53
+ return tex2D (tex, uv) * autoExposure;
54
+ }
55
+
56
+ half4 SetPixelShader (in float4 vertex : POSITION , in float2 texcoord : TEXCOORD0 ) : SV_Target
57
+ {
58
+ float2 uv = texcoord + _MainTex_TexelSize.xy * _PrefilterOffs;
59
+ #if ANTI_FLICKER
60
+ float3 d = _MainTex_TexelSize.xyx * float3 (1.0 , 1.0 , 0.0 );
61
+ half4 s0 = min (FetchAutoExposed (_MainTex, uv),65504.0 );
62
+ half3 s1 = min (FetchAutoExposed (_MainTex, uv - d.xz).rgb,65504.0 );
63
+ half3 s2 = min (FetchAutoExposed (_MainTex, uv + d.xz).rgb,65504.0 );
64
+ half3 s3 = min (FetchAutoExposed (_MainTex, uv - d.zy).rgb,65504.0 );
65
+ half3 s4 = min (FetchAutoExposed (_MainTex, uv + d.zy).rgb,65504.0 );
66
+ half3 m = Median (Median (s0.rgb, s1, s2), s3, s4);
67
+ #else
68
+ half4 s0 = min (FetchAutoExposed (_MainTex, uv),65504.0 );
69
+ half3 m = s0.rgb;
70
+ #endif
71
+ #if UNITY_COLORSPACE_GAMMA
72
+ m = m * (m * (m * 0. 305306011h + 0. 682171111h) + 0. 012522878h);
73
+ #endif
74
+ half br = Brightness (m);
75
+ half rq = clamp (br - _Curve.x, 0.0 , _Curve.y);
76
+ rq = _Curve.z * rq * rq;
77
+ m *= max (rq, br - _Threshold) / max (br, 1e-5 );
78
+ return half4 (m,0 );
79
+ }
80
+
81
+ ENDCG
82
+ }
83
+
84
+ Pass
85
+ {
86
+ CGPROGRAM
87
+ #pragma multi_compile __ ANTI_FLICKER
88
+ #pragma vertex SetVertexShader
89
+ #pragma fragment SetPixelShader
90
+
91
+ half Brightness (half3 c)
92
+ {
93
+ return max (c.x, max (c.y, c.z));
94
+ }
95
+
96
+ half3 DownsampleFilter (sampler2D tex, float2 uv, float2 texelSize)
97
+ {
98
+ float4 d = texelSize.xyxy * float4 (-1.0 , -1.0 , 1.0 , 1.0 );
99
+ half3 s;
100
+ s = tex2D (tex, uv + d.xy).rgb;
101
+ s += tex2D (tex, uv + d.zy).rgb;
102
+ s += tex2D (tex, uv + d.xw).rgb;
103
+ s += tex2D (tex, uv + d.zw).rgb;
104
+ return s * (1.0 / 4.0 );
105
+ }
106
+
107
+ half3 DownsampleAntiFlickerFilter (sampler2D tex, float2 uv, float2 texelSize)
108
+ {
109
+ float4 d = texelSize.xyxy * float4 (-1.0 , -1.0 , 1.0 , 1.0 );
110
+ half3 s1 = tex2D (tex, uv + d.xy).rgb;
111
+ half3 s2 = tex2D (tex, uv + d.zy).rgb;
112
+ half3 s3 = tex2D (tex, uv + d.xw).rgb;
113
+ half3 s4 = tex2D (tex, uv + d.zw).rgb;
114
+ half s1w = 1.0 / (Brightness (s1) + 1.0 );
115
+ half s2w = 1.0 / (Brightness (s2) + 1.0 );
116
+ half s3w = 1.0 / (Brightness (s3) + 1.0 );
117
+ half s4w = 1.0 / (Brightness (s4) + 1.0 );
118
+ half one_div_wsum = 1.0 / (s1w + s2w + s3w + s4w);
119
+ return (s1 * s1w + s2 * s2w + s3 * s3w + s4 * s4w) * one_div_wsum;
120
+ }
121
+
122
+ void SetVertexShader (inout float4 vertex : POSITION , inout float2 texcoord : TEXCOORD0 )
123
+ {
124
+ vertex = UnityObjectToClipPos (vertex);
125
+ }
126
+
127
+ half4 SetPixelShader (in float4 vertex : POSITION , in float2 texcoord : TEXCOORD0 ) : SV_Target
128
+ {
129
+ #if ANTI_FLICKER
130
+ return half4 (DownsampleAntiFlickerFilter (_MainTex, texcoord, _MainTex_TexelSize.xy),0 );
131
+ #else
132
+ return half4 (DownsampleFilter (_MainTex, texcoord, _MainTex_TexelSize.xy),0 );
133
+ #endif
134
+ }
135
+
136
+ ENDCG
137
+ }
138
+
139
+ Pass
140
+ {
141
+ CGPROGRAM
142
+ #pragma vertex SetVertexShader
143
+ #pragma fragment SetPixelShader
144
+
145
+ void SetVertexShader (inout float4 vertex : POSITION , inout float2 texcoord : TEXCOORD0 )
146
+ {
147
+ vertex = UnityObjectToClipPos (vertex);
148
+ }
149
+
150
+ half3 DownsampleFilter (sampler2D tex, float2 uv, float2 texelSize)
151
+ {
152
+ float4 d = texelSize.xyxy * float4 (-1.0 , -1.0 , 1.0 , 1.0 );
153
+ half3 s;
154
+ s = tex2D (tex, uv + d.xy).rgb;
155
+ s += tex2D (tex, uv + d.zy).rgb;
156
+ s += tex2D (tex, uv + d.xw).rgb;
157
+ s += tex2D (tex, uv + d.zw).rgb;
158
+ return s * (1.0 / 4.0 );
159
+ }
160
+
161
+ half4 SetPixelShader (in float4 vertex : POSITION , in float2 texcoord : TEXCOORD0 ) : SV_Target
162
+ {
163
+ return half4 (DownsampleFilter (_MainTex, texcoord, _MainTex_TexelSize.xy),0 );
164
+ }
165
+
166
+ ENDCG
167
+ }
168
+
169
+ Pass
170
+ {
171
+ CGPROGRAM
172
+ #pragma vertex SetVertexShader
173
+ #pragma fragment SetPixelShader
174
+
175
+ float _SampleScale;
176
+ sampler2D _BaseTex;
177
+ float2 _BaseTex_TexelSize;
178
+
179
+ struct SHADERDATA
180
+ {
181
+ float4 pos : SV_POSITION ;
182
+ float2 uvMain : TEXCOORD0 ;
183
+ float2 uvBase : TEXCOORD1 ;
184
+ };
185
+
186
+ half3 UpsampleFilter (sampler2D tex, float2 uv, float2 texelSize, float sampleScale)
187
+ {
188
+ float4 d = texelSize.xyxy * float4 (1.0 , 1.0 , -1.0 , 0.0 ) * sampleScale;
189
+ half3 s;
190
+ s = tex2D (tex, uv - d.xy).rgb;
191
+ s += tex2D (tex, uv - d.wy).rgb * 2.0 ;
192
+ s += tex2D (tex, uv - d.zy).rgb;
193
+ s += tex2D (tex, uv + d.zw).rgb * 2.0 ;
194
+ s += tex2D (tex, uv).rgb * 4.0 ;
195
+ s += tex2D (tex, uv + d.xw).rgb * 2.0 ;
196
+ s += tex2D (tex, uv + d.zy).rgb;
197
+ s += tex2D (tex, uv + d.wy).rgb * 2.0 ;
198
+ s += tex2D (tex, uv + d.xy).rgb;
199
+ return s * (1.0 / 16.0 );
200
+ }
201
+
202
+ SHADERDATA SetVertexShader (float4 vertex : POSITION , float2 texcoord : TEXCOORD0 )
203
+ {
204
+ SHADERDATA o;
205
+ o.pos = UnityObjectToClipPos (vertex);
206
+ o.uvMain = texcoord.xy;
207
+ o.uvBase = o.uvMain;
208
+ #if UNITY_UV_STARTS_AT_TOP
209
+ if (_BaseTex_TexelSize.y < 0.0 ) o.uvBase.y = 1.0 - o.uvBase.y;
210
+ #endif
211
+ return o;
212
+ }
213
+
214
+ half4 SetPixelShader (SHADERDATA i) : SV_Target
215
+ {
216
+ half3 base = tex2D (_BaseTex, i.uvBase).rgb;
217
+ half3 blur = UpsampleFilter (_MainTex, i.uvMain, _MainTex_TexelSize.xy, _SampleScale);
218
+ return half4 (base + blur,0 );
219
+ }
220
+ ENDCG
221
+ }
222
+ }
223
+ }
0 commit comments