527 lines
20 KiB
Plaintext
527 lines
20 KiB
Plaintext
|
// The following comment prevents Unity from auto upgrading the shader. Please keep it to keep backward compatibility
|
||
|
// UNITY_SHADER_NO_UPGRADE
|
||
|
|
||
|
#ifndef _VOLUMETRIC_LIGHT_BEAM_SHARED_INCLUDED_
|
||
|
#define _VOLUMETRIC_LIGHT_BEAM_SHARED_INCLUDED_
|
||
|
|
||
|
/// ****************************************
|
||
|
/// SHADER INPUT / OUTPUT STRUCT
|
||
|
/// ****************************************
|
||
|
struct vlb_appdata
|
||
|
{
|
||
|
float4 vertex : POSITION;
|
||
|
float4 texcoord : TEXCOORD0;
|
||
|
|
||
|
#if VLB_INSTANCING_API_AVAILABLE && (VLB_STEREO_INSTANCING || VLB_GPU_INSTANCING)
|
||
|
UNITY_VERTEX_INPUT_INSTANCE_ID // for GPU Instancing and Single Pass Instanced rendering
|
||
|
#endif
|
||
|
};
|
||
|
|
||
|
struct v2f
|
||
|
{
|
||
|
float4 posClipSpace : SV_POSITION;
|
||
|
float3 posObjectSpace : TEXCOORD0;
|
||
|
float4 posWorldSpace : TEXCOORD1;
|
||
|
float3 posViewSpace : TEXCOORD2;
|
||
|
float3 cameraPosObjectSpace : TEXCOORD3;
|
||
|
|
||
|
float4 projPos : TEXCOORD6;
|
||
|
|
||
|
#ifdef VLB_FOG_UNITY_BUILTIN_COORDS
|
||
|
UNITY_FOG_COORDS(7)
|
||
|
#endif
|
||
|
|
||
|
#if VLB_INSTANCING_API_AVAILABLE
|
||
|
#if VLB_GPU_INSTANCING
|
||
|
UNITY_VERTEX_INPUT_INSTANCE_ID // not sure this one is useful
|
||
|
#endif
|
||
|
|
||
|
#if VLB_STEREO_INSTANCING
|
||
|
UNITY_VERTEX_OUTPUT_STEREO // for Single Pass Instanced rendering
|
||
|
#endif
|
||
|
#endif // VLB_INSTANCING_API_AVAILABLE
|
||
|
};
|
||
|
|
||
|
|
||
|
#include "ShaderUtils.cginc"
|
||
|
|
||
|
inline float ComputeFadeWithCamera(float3 posViewSpace, float enabled)
|
||
|
{
|
||
|
float distCamToPixWS = abs(posViewSpace.z); // only check Z axis (instead of length(posViewSpace.xyz)) to have smoother transition with near plane (which is not curved)
|
||
|
float camFadeDistStart = _ProjectionParams.y; // cam near place
|
||
|
float camFadeDistEnd = camFadeDistStart + _VLB_CameraBlendingDistance;
|
||
|
float fadeWhenTooClose = smoothstep(0, 1, invLerpClamped(camFadeDistStart, camFadeDistEnd, distCamToPixWS));
|
||
|
|
||
|
// fade out according to camera's near plane
|
||
|
return lerp(1, fadeWhenTooClose, enabled);
|
||
|
}
|
||
|
|
||
|
// Vector Camera to current Pixel, in object space and normalized
|
||
|
inline float3 ComputeVectorCamToPixOSN(float3 pixPosOS, float3 cameraPosOS)
|
||
|
{
|
||
|
float3 vecCamToPixOSN = normalize(pixPosOS - cameraPosOS);
|
||
|
|
||
|
// Deal with ortho camera:
|
||
|
// With ortho camera, we don't want to change the fresnel according to camera position.
|
||
|
// So instead of computing the proper vector "Camera to Pixel", we take account of the "Camera Forward" vector (which is not dependant on the pixel position)
|
||
|
float3 vecCamForwardOSN = VLB_GET_PROP(_CameraForwardOS);
|
||
|
|
||
|
return lerp(vecCamToPixOSN, vecCamForwardOSN, VLB_CAMERA_ORTHO);
|
||
|
}
|
||
|
|
||
|
v2f vertShared(vlb_appdata v)
|
||
|
{
|
||
|
v2f o;
|
||
|
|
||
|
#if VLB_INSTANCING_API_AVAILABLE && (VLB_STEREO_INSTANCING || VLB_GPU_INSTANCING)
|
||
|
UNITY_SETUP_INSTANCE_ID(v);
|
||
|
|
||
|
#if VLB_STEREO_INSTANCING
|
||
|
#ifndef VLB_SRP_API // TODO CHECK THAT WE DON'T NEED THIS WITH SRP
|
||
|
UNITY_INITIALIZE_OUTPUT(v2f, o);
|
||
|
#endif
|
||
|
UNITY_INITIALIZE_VERTEX_OUTPUT_STEREO(o);
|
||
|
#endif
|
||
|
|
||
|
#if VLB_GPU_INSTANCING
|
||
|
UNITY_TRANSFER_INSTANCE_ID(v, o);
|
||
|
#endif
|
||
|
#endif
|
||
|
|
||
|
// compute the proper cone shape, so the whole beam fits into a 2x2x1 box
|
||
|
// The model matrix (computed via the localScale from BeamGeometry.)
|
||
|
float4 vertexOS = v.vertex;
|
||
|
|
||
|
float2 coneRadius = VLB_GET_PROP(_ConeRadius);
|
||
|
float maxRadius = max(coneRadius.x, coneRadius.y);
|
||
|
float normalizedRadiusStart = coneRadius.x / maxRadius;
|
||
|
float normalizedRadiusEnd = coneRadius.y / maxRadius;
|
||
|
vertexOS.xy *= lerp(normalizedRadiusStart, normalizedRadiusEnd, vertexOS.z);
|
||
|
|
||
|
float3 scaleObjectSpace = float3(maxRadius, maxRadius, VLB_GET_PROP(_DistanceFallOff).z); // maxGeometryDistance
|
||
|
|
||
|
o.posWorldSpace = VLBObjectToWorldPos(vertexOS);
|
||
|
o.posClipSpace = VLBObjectToClipPos(vertexOS.xyz);
|
||
|
// TODO Should create and use VLBWorldToClipPos instead
|
||
|
//o.posClipSpace = VLBWorldToClipPos(o.posWorldSpace.xyz);
|
||
|
|
||
|
#if defined(VLBWorldToViewPos)
|
||
|
float3 posViewSpace = VLBWorldToViewPos(o.posWorldSpace.xyz);
|
||
|
#elif defined(VLBObjectToViewPos)
|
||
|
float3 posViewSpace = VLBObjectToViewPos(vertexOS);
|
||
|
#else
|
||
|
You_should_define_either_VLBWorldToViewPos_or_VLBObjectToViewPos
|
||
|
#endif
|
||
|
|
||
|
// apply the same scaling than we do through the localScale in BeamGeometry.ComputeLocalMatrix
|
||
|
// to get the proper transformed vertex position in object space
|
||
|
o.posObjectSpace = vertexOS.xyz * scaleObjectSpace;
|
||
|
|
||
|
|
||
|
o.projPos = Depth_VS_ComputeProjPos(posViewSpace, o.posClipSpace);
|
||
|
|
||
|
o.cameraPosObjectSpace = VLBGetCameraPositionObjectSpace(scaleObjectSpace);
|
||
|
|
||
|
o.posViewSpace = posViewSpace;
|
||
|
|
||
|
#ifdef VLB_FOG_UNITY_BUILTIN_COORDS
|
||
|
UNITY_TRANSFER_FOG(o, o.posClipSpace);
|
||
|
#endif
|
||
|
return o;
|
||
|
}
|
||
|
|
||
|
// original code from Inigo Quilez: https://www.iquilezles.org/www/articles/intersectors/intersectors.htm
|
||
|
float coneIntersect(float3 rayOrigin, float3 rayDir, float3 conePosEnd, float radiusStart, float radiusEnd)
|
||
|
{
|
||
|
float3 ba = conePosEnd;
|
||
|
float3 oa = rayOrigin;
|
||
|
float3 ob = rayOrigin - conePosEnd;
|
||
|
float m0 = dot(ba, ba);
|
||
|
float m1 = dot(oa, ba);
|
||
|
float m2 = dot(rayDir, ba);
|
||
|
float m3 = dot(rayDir, oa);
|
||
|
float m5 = dot(oa, oa);
|
||
|
float m9 = dot(ob, ba);
|
||
|
|
||
|
// caps
|
||
|
if (m1 < 0.0)
|
||
|
{
|
||
|
if (dot2(oa*m2 - rayDir * m1) < (radiusStart*radiusStart*m2*m2)) // delayed division
|
||
|
return -m1 / m2;
|
||
|
}
|
||
|
else if (m9 > 0.0)
|
||
|
{
|
||
|
float t = -m9 / m2; // NOT delayed division
|
||
|
if (dot2(ob + rayDir * t) < (radiusEnd*radiusEnd))
|
||
|
return t;
|
||
|
}
|
||
|
|
||
|
// body
|
||
|
float rr = radiusStart - radiusEnd;
|
||
|
float hy = m0 + rr * rr;
|
||
|
float k2 = m0 * m0 - m2 * m2*hy;
|
||
|
float k1 = m0 * m0*m3 - m1 * m2*hy + m0 * radiusStart*(rr*m2*1.0);
|
||
|
float k0 = m0 * m0*m5 - m1 * m1*hy + m0 * radiusStart*(rr*m1*2.0 - m0 * radiusStart);
|
||
|
float h = k1 * k1 - k2 * k0;
|
||
|
if (h < 0.0) return -1.0; // no intersection
|
||
|
|
||
|
float t = (-k1 - sqrt(h)) / k2;
|
||
|
float y = m1 + t * m2;
|
||
|
if (y<0.0 || y>m0) return -1.0; //no intersection
|
||
|
return t;
|
||
|
}
|
||
|
|
||
|
half4 fragShared(v2f i)
|
||
|
{
|
||
|
#if VLB_INSTANCING_API_AVAILABLE && VLB_GPU_INSTANCING
|
||
|
UNITY_SETUP_INSTANCE_ID(i);
|
||
|
#endif
|
||
|
|
||
|
#if VLB_INSTANCING_API_AVAILABLE && VLB_STEREO_INSTANCING
|
||
|
// This fix access to depth map on the right eye when using single pass (aka Stereo Rendering Mode Multiview) on Gear VR or Oculus Go/Quest
|
||
|
UNITY_SETUP_STEREO_EYE_INDEX_POST_VERTEX(i); // https://docs.unity3d.com/Manual/SinglePassInstancing.html
|
||
|
#endif
|
||
|
|
||
|
float3 transformScale = VLB_GET_PROP(_TransformScale);
|
||
|
float3 intersectOutOS = i.posObjectSpace * transformScale;
|
||
|
|
||
|
// compute proper ray start and dir
|
||
|
float3 realCamPosOS = i.cameraPosObjectSpace.xyz * transformScale;
|
||
|
float3 rayDirOS = ComputeVectorCamToPixOSN(intersectOutOS, realCamPosOS); // deal with ortho cam here
|
||
|
float3 realVecCamToPix = intersectOutOS - realCamPosOS;
|
||
|
float rayLength = dot(rayDirOS, realVecCamToPix);
|
||
|
float3 rayStartOS = intersectOutOS - rayLength * rayDirOS;
|
||
|
|
||
|
#if VLB_NOISE_3D
|
||
|
// only useful for noise world space
|
||
|
float3 intersectOutWS = i.posWorldSpace.xyz;
|
||
|
float3 rayDirWS, rayStartWS;
|
||
|
{
|
||
|
float3 perspRayStartWS = _WorldSpaceCameraPos.xyz;
|
||
|
float3 perspRayDirWS = normalize(intersectOutWS - perspRayStartWS);
|
||
|
|
||
|
float3 orthoRayDirWS = VLB_GET_PROP(_CameraForwardWS);
|
||
|
float3 orthoRayStartWS = intersectOutWS - rayLength * orthoRayDirWS;
|
||
|
|
||
|
rayStartWS = lerp(perspRayStartWS, orthoRayStartWS, VLB_CAMERA_ORTHO);
|
||
|
rayDirWS = lerp(perspRayDirWS, orthoRayDirWS, VLB_CAMERA_ORTHO);
|
||
|
}
|
||
|
#endif // VLB_NOISE_3D
|
||
|
|
||
|
float distanceFadeEnd = VLB_GET_PROP(_DistanceFallOff).y * transformScale.z;
|
||
|
float2 coneRadius = VLB_GET_PROP(_ConeRadius) * min(transformScale.x, transformScale.y);
|
||
|
|
||
|
float tIn = coneIntersect(rayStartOS, rayDirOS, float3(0, 0, distanceFadeEnd), coneRadius.x, coneRadius.y);
|
||
|
|
||
|
// fix artifact in VR on geometry edges when no intersection is found
|
||
|
float intensity = 1.0 - ifAnd(
|
||
|
isLower(tIn, 0.0),
|
||
|
isLower(realCamPosOS.z, 0.0) // is it ok to apply transformScale here?
|
||
|
);
|
||
|
|
||
|
tIn = max(tIn, 0); // when camera is inside the beam, tIn = -1: set tIn at 0
|
||
|
|
||
|
float tOut = length(rayStartOS - intersectOutOS);
|
||
|
float sceneZ = Depth_PS_GetSceneDepthFromEye(i.projPos, i.posViewSpace);
|
||
|
|
||
|
{ // DEBUG
|
||
|
#if DEBUG_DEPTH_MODE == DEBUG_VALUE_DEPTHBUFFER_FROMEYE
|
||
|
return Depth_PS_GetSceneDepthFromEye(i.projPos, i.posViewSpace) * _ProjectionParams.w;
|
||
|
#elif DEBUG_DEPTH_MODE == DEBUG_VALUE_DEPTHBUFFER_FROMNEARPLANE
|
||
|
return Depth_PS_GetSceneDepthFromNearPlane(i.projPos) * _ProjectionParams.w;
|
||
|
#elif DEBUG_DEPTH_MODE == DEBUG_VALUE_DEPTHSTEREOEYE
|
||
|
float depthValue = Depth_PS_GetSceneDepthFromEye(i.projPos, i.posViewSpace) * _ProjectionParams.w;
|
||
|
#if defined(USING_STEREO_MATRICES) && defined(UNITY_STEREO_MULTIVIEW_ENABLED) // used with single pass / multiview on android VR (Oculus Go/Quest, Gear VR)
|
||
|
return depthValue * lerp(float4(1, 0, 0, 1), float4(0, 1, 0, 1), unity_StereoEyeIndex);
|
||
|
#elif defined(UNITY_SINGLE_PASS_STEREO)
|
||
|
return depthValue * lerp(float4(1, 0, 0, 1), float4(0, 0, 1, 1), unity_StereoEyeIndex);
|
||
|
#elif defined(UNITY_STEREO_INSTANCING_ENABLED)
|
||
|
return depthValue * lerp(float4(0, 1, 0, 1), float4(0, 0, 1, 1), unity_StereoEyeIndex);
|
||
|
#elif defined(UNITY_STEREO_MULTIVIEW_ENABLED)
|
||
|
return depthValue * lerp(float4(1, 1, 0, 1), float4(0, 1, 1, 1), unity_StereoEyeIndex);
|
||
|
#else
|
||
|
return depthValue;
|
||
|
#endif
|
||
|
#elif DEBUG_DEPTH_MODE == DEBUG_VALUE_DEPTHBLEND
|
||
|
return float4(1 - saturate(abs(tIn - sceneZ)), 1 - saturate(abs(tOut - sceneZ)), 0, 1);
|
||
|
#endif
|
||
|
} // DEBUG
|
||
|
|
||
|
tOut = min(tOut, sceneZ);
|
||
|
|
||
|
float tInJittered = tIn;
|
||
|
|
||
|
// add jittering
|
||
|
{
|
||
|
float2 screenPos = i.projPos.xy / i.projPos.w;
|
||
|
|
||
|
float2 jitterCoord = screenPos * _ScreenParams.xy * _VLB_JitteringNoiseTex_TexelSize.xy;
|
||
|
float jitterNoise = tex2D(_VLB_JitteringNoiseTex, jitterCoord).r;
|
||
|
|
||
|
float4 jitterProps = VLB_GET_PROP(_Jittering);
|
||
|
|
||
|
// Golden Ratio Animated Noise https://blog.demofox.org/2017/10/31/animating-noise-for-integration-over-time/
|
||
|
float frameRate = jitterProps.y;
|
||
|
const float kGoldenRatio = 1.61803398875f; // (1.0f + sqrt(5.0f)) * 0.5f;
|
||
|
const float kGoldenRatioFrac = 0.61803398875f;
|
||
|
|
||
|
uint frame = uint(floor(_Time.y * frameRate));
|
||
|
jitterNoise = frac(jitterNoise + float(frame) * kGoldenRatio);
|
||
|
|
||
|
float3 intersectInOS = rayStartOS + rayDirOS * tIn;
|
||
|
float currentPosZNorm = max(intersectInOS.z, intersectOutOS.z) / distanceFadeEnd;
|
||
|
|
||
|
float jitterRatio = invLerpClamped(jitterProps.z /* range start */, jitterProps.w /* range end */, currentPosZNorm);
|
||
|
tInJittered += jitterNoise * jitterRatio * jitterProps.x;
|
||
|
}
|
||
|
|
||
|
#if VLB_COLOR_GRADIENT
|
||
|
float4 colorGradient = 0.0;
|
||
|
#endif // VLB_COLOR_GRADIENT
|
||
|
|
||
|
#if VLB_COOKIE_RGBA
|
||
|
float4 colorCookieSum = 0;
|
||
|
#endif // VLB_COOKIE_RGBA
|
||
|
|
||
|
{
|
||
|
float distanceFadeStart = VLB_GET_PROP(_DistanceFallOff).x * transformScale.z;
|
||
|
|
||
|
float sumLerp = 0.0;
|
||
|
float sumLinear = 0.0;
|
||
|
|
||
|
float sideSoftness = VLB_GET_PROP(_SideSoftness);
|
||
|
|
||
|
const int raymarchSteps = VLB_RAYMARCHING_STEP_COUNT;
|
||
|
|
||
|
int stepCountLinear = 0;
|
||
|
float stepLinear = 1.1 * distanceFadeEnd / raymarchSteps;
|
||
|
float tLinear = tIn; // using the non jittered version of tIn for linear sampling helps a bit smoothing the noise
|
||
|
|
||
|
#if VLB_SHADOW
|
||
|
// cache some values useful for shadow computing
|
||
|
float4 shw_props = VLB_GET_PROP(_ShadowProps);
|
||
|
const float shw_isPersp = shw_props.w;
|
||
|
const float shw_apexDist = VLB_GET_PROP(_ConeGeomProps).x * shw_isPersp;
|
||
|
const float kMinNearClipPlane = 0.1f * shw_isPersp; // should be the same than in VolumetricShadowHD.cs
|
||
|
|
||
|
const float shw_nearUnscaled = max(shw_apexDist, kMinNearClipPlane / transformScale.z);
|
||
|
const float shw_nearScaled = max(shw_apexDist, kMinNearClipPlane);
|
||
|
const float shw_farUnscaled = shw_nearUnscaled + distanceFadeEnd / transformScale.z;
|
||
|
const float shw_farScaled = shw_nearScaled + distanceFadeEnd;
|
||
|
|
||
|
// handle scale X & Y
|
||
|
float2 shw_ratioScale;
|
||
|
{
|
||
|
float ratioScale = transformScale.x / transformScale.y;
|
||
|
if (ratioScale >= 1) shw_ratioScale = float2(1.0 / ratioScale, 1.0);
|
||
|
else shw_ratioScale = float2(1.0, ratioScale);
|
||
|
}
|
||
|
|
||
|
#if DEBUG_DEPTH_MODE == DEBUG_VALUE_SHADOW_DEPTH
|
||
|
{
|
||
|
float shadowDepthRaw = tex2D(_ShadowDepthTexture, i.projPos.xy / i.projPos.w).r;
|
||
|
float shadowDepthLinearPersp = VLB_ZBufferToLinear(shadowDepthRaw, shw_nearUnscaled, shw_farUnscaled);
|
||
|
shadowDepthLinearPersp = fromABtoCD_Clamped(shadowDepthLinearPersp, shw_nearUnscaled, shw_farUnscaled, shw_nearScaled, shw_farScaled);
|
||
|
const float shadowDepthLinearOrtho = shadowDepthRaw * (shw_farScaled - shw_nearScaled);
|
||
|
float shadowDepthLinear = lerp(shadowDepthLinearOrtho, shadowDepthLinearPersp, shw_isPersp) - shw_apexDist;
|
||
|
return shadowDepthLinear;
|
||
|
}
|
||
|
#endif // DEBUG_DEPTH_MODE
|
||
|
#endif // VLB_SHADOW
|
||
|
|
||
|
#if DEBUG_DEPTH_MODE == DEBUG_VALUE_LINEAR_OVERFLOW
|
||
|
bool linearStepsOverflow = false;
|
||
|
#endif // DEBUG_VALUE_LINEAR_OVERFLOW
|
||
|
for (int i = 0; i < raymarchSteps; i++)
|
||
|
{
|
||
|
float t = saturate(float(i+1) / (raymarchSteps + 1));
|
||
|
float tLerp = lerp(tInJittered, tOut, t); // use the jittered version of tIn to apply noise on raymarching
|
||
|
|
||
|
float3 posOSLinear = rayStartOS + rayDirOS * tLinear;
|
||
|
float3 posOSLerp = rayStartOS + rayDirOS * tLerp;
|
||
|
|
||
|
//if (length(posOSLerp - rayStartOS) > tOut) { break; } // Fix cookie sampling artifacts when a geometry is placed between the camera and the beam. Useless when disabling mipmaps on cookie texture
|
||
|
|
||
|
float att = ComputeAttenuationHD(posOSLerp.z, distanceFadeStart, distanceFadeEnd);
|
||
|
float widthAtThisZ = fromABtoCD_Clamped(posOSLerp.z, 0.0, distanceFadeEnd, coneRadius.x, coneRadius.y);
|
||
|
|
||
|
float fresnel = (widthAtThisZ - length(posOSLerp.xy)) / (widthAtThisZ * sideSoftness);
|
||
|
fresnel = saturate(fresnel);
|
||
|
|
||
|
float powerAtThisStepLerp = att * fresnel;
|
||
|
float powerAtThisStepLinear = 1.0;
|
||
|
|
||
|
#if VLB_COLOR_GRADIENT
|
||
|
colorGradient += ApplyAlphaToColor(ComputeColorGradient(posOSLerp.z / transformScale.z));
|
||
|
#endif // VLB_COLOR_GRADIENT
|
||
|
|
||
|
#if VLB_NOISE_3D
|
||
|
{
|
||
|
//#define VLB_NOISE_3D_LINEAR 1
|
||
|
#if VLB_NOISE_3D_LINEAR
|
||
|
float3 posWSLinear = rayStartWS + rayDirWS * tLinear;
|
||
|
float3 noiseUVWLinear = Noise3D_GetUVW(posWSLinear, posOSLinear);
|
||
|
float noiseFactorLinear = Noise3D_GetFactorFromUVW(noiseUVWLinear);
|
||
|
powerAtThisStepLinear *= noiseFactorLinear;
|
||
|
#else
|
||
|
float3 posWSLerp = rayStartWS + rayDirWS * tLerp;
|
||
|
float3 noiseUVWLerp = Noise3D_GetUVW(posWSLerp, posOSLerp);
|
||
|
float noiseFactorLerp = Noise3D_GetFactorFromUVW(noiseUVWLerp);
|
||
|
powerAtThisStepLerp *= noiseFactorLerp;
|
||
|
#endif
|
||
|
}
|
||
|
#endif // VLB_NOISE_3D
|
||
|
|
||
|
#if VLB_COOKIE_1CHANNEL || VLB_COOKIE_RGBA
|
||
|
{
|
||
|
float4 posAndScale = VLB_GET_PROP(_CookiePosAndScale);
|
||
|
float4 props = VLB_GET_PROP(_CookieProperties); // contrib + negative, texture channel, cos(rot), sin(rot)
|
||
|
float2x2 rotMatrix = float2x2(props.z, -props.w, props.w, props.z);
|
||
|
|
||
|
float2 posOSXY = posOSLerp.xy / widthAtThisZ; // [-0.5 ; 0.5]
|
||
|
posOSXY += posAndScale.xy; // translate
|
||
|
posOSXY = mul(posOSXY, rotMatrix); // rotate
|
||
|
posOSXY *= posAndScale.zw; // scale
|
||
|
posOSXY = posOSXY * 0.5 + 0.5; // transform coord to [0.0 ; 1.0]
|
||
|
|
||
|
#if VLB_COOKIE_1CHANNEL
|
||
|
float cookie = tex2D(_CookieTexture, posOSXY)[(int)props.y];
|
||
|
float negative = max(props.x, 0); // props.x also store contribution as negative value if negative
|
||
|
float contrib = abs(props.x);
|
||
|
powerAtThisStepLerp *= lerp(1.0, lerp(cookie, 1.0 - cookie, negative), contrib);
|
||
|
#endif
|
||
|
|
||
|
#if VLB_COOKIE_RGBA
|
||
|
colorCookieSum += tex2D(_CookieTexture, posOSXY);
|
||
|
#endif
|
||
|
}
|
||
|
#endif // VLB_COOKIE_1CHANNEL || VLB_COOKIE_RGBA
|
||
|
|
||
|
#if VLB_SHADOW
|
||
|
{
|
||
|
float3 posOSRef = posOSLerp;
|
||
|
|
||
|
float width = lerp(coneRadius.x, widthAtThisZ, shw_isPersp);
|
||
|
float2 posOSXYNormalized = posOSRef.xy / width;
|
||
|
|
||
|
// handle scale X & Y
|
||
|
posOSXYNormalized *= shw_ratioScale;
|
||
|
|
||
|
posOSXYNormalized = posOSXYNormalized * 0.5 + 0.5;
|
||
|
posOSXYNormalized.x = flipUV(posOSXYNormalized.x, shw_props.x);
|
||
|
posOSXYNormalized.y = flipUV(posOSXYNormalized.y, shw_props.y);
|
||
|
|
||
|
float shadowDepthRaw = tex2D(_ShadowDepthTexture, posOSXYNormalized).r;
|
||
|
shadowDepthRaw = lerp(shadowDepthRaw, 1.0f - shadowDepthRaw, _VLB_UsesReversedZBuffer);
|
||
|
|
||
|
// Compte perspective linear depth and handle Z scaling
|
||
|
float shadowDepthLinearPersp = VLB_ZBufferToLinear(shadowDepthRaw, shw_nearUnscaled, shw_farUnscaled); // decode depth value using unscaled near/far distance
|
||
|
shadowDepthLinearPersp = fromABtoCD_Clamped(shadowDepthLinearPersp, shw_nearUnscaled, shw_farUnscaled, shw_nearScaled, shw_farScaled); // scale the linear depth value according to the scaled near/far distance
|
||
|
|
||
|
// Compute ortho linear depth
|
||
|
const float shadowDepthLinearOrtho = shadowDepthRaw * (shw_farScaled - shw_nearScaled);
|
||
|
|
||
|
// get either the perspective or ortho depth
|
||
|
float shadowDepthLinear = lerp(shadowDepthLinearOrtho, shadowDepthLinearPersp, shw_isPersp) - shw_apexDist;
|
||
|
|
||
|
float factor = isEqualOrGreater(shadowDepthLinear, posOSRef.z);
|
||
|
|
||
|
float dimmer = shw_props.z;
|
||
|
factor = (1 - dimmer) + factor * dimmer; // lerp(1 - dimmer, 1, factor)
|
||
|
powerAtThisStepLerp *= factor;
|
||
|
}
|
||
|
#endif // VLB_SHADOW
|
||
|
|
||
|
sumLerp += powerAtThisStepLerp;
|
||
|
|
||
|
if (tLinear < tOut)
|
||
|
{
|
||
|
tLinear += stepLinear;
|
||
|
sumLinear += powerAtThisStepLinear;
|
||
|
stepCountLinear++;
|
||
|
}
|
||
|
#if DEBUG_DEPTH_MODE == DEBUG_VALUE_LINEAR_OVERFLOW
|
||
|
else
|
||
|
{
|
||
|
linearStepsOverflow = true;
|
||
|
}
|
||
|
#endif // DEBUG_VALUE_LINEAR_OVERFLOW
|
||
|
}
|
||
|
// for loop end
|
||
|
|
||
|
#if VLB_COLOR_GRADIENT
|
||
|
colorGradient /= raymarchSteps;
|
||
|
#endif // VLB_COLOR_GRADIENT
|
||
|
|
||
|
#if VLB_COOKIE_RGBA
|
||
|
colorCookieSum /= raymarchSteps;
|
||
|
#endif // VLB_COOKIE_RGBA
|
||
|
|
||
|
|
||
|
intensity *= (sumLerp / raymarchSteps);
|
||
|
|
||
|
if (stepCountLinear > 0)
|
||
|
{
|
||
|
float meanLinear = (sumLinear / stepCountLinear);
|
||
|
#if DEBUG_DEPTH_MODE == DEBUG_VALUE_LINEAR_OVERFLOW
|
||
|
{
|
||
|
if (!linearStepsOverflow)
|
||
|
return float4(0, 0, 1, 1);
|
||
|
}
|
||
|
#endif // DEBUG_VALUE_LINEAR_OVERFLOW
|
||
|
intensity *= meanLinear;
|
||
|
}
|
||
|
|
||
|
intensity *= (tOut - tIn); // prevent from having a darker circle at end cap
|
||
|
intensity *= VLB_GET_PROP(_Intensity);
|
||
|
}
|
||
|
|
||
|
{
|
||
|
float fadeWithCameraEnabled = 1 - VLB_CAMERA_ORTHO; // fading according to camera eye position doesn't make sense with ortho camera
|
||
|
intensity *= ComputeFadeWithCamera(i.posViewSpace, fadeWithCameraEnabled);
|
||
|
}
|
||
|
|
||
|
{
|
||
|
float factorInsideGeom = isEqualOrGreater(tOut, tIn);
|
||
|
intensity *= factorInsideGeom;
|
||
|
}
|
||
|
|
||
|
#if VLB_COLOR_GRADIENT
|
||
|
float4 color = colorGradient;
|
||
|
#elif VLB_COLOR_FLAT
|
||
|
float4 color = ApplyAlphaToColor(ComputeColorFlat());
|
||
|
#endif // VLB_COLOR_GRADIENT / VLB_COLOR_FLAT
|
||
|
|
||
|
#if VLB_COOKIE_RGBA
|
||
|
{
|
||
|
float4 props = VLB_GET_PROP(_CookieProperties); // contrib + negative, texture channel, cos(rot), sin(rot)
|
||
|
float contrib = abs(props.x);
|
||
|
color *= lerp(1, colorCookieSum, contrib);
|
||
|
}
|
||
|
#endif // VLB_COOKIE_RGBA
|
||
|
|
||
|
#if VLB_DITHERING
|
||
|
{
|
||
|
float2 screenPos = i.projPos.xy / i.projPos.w;
|
||
|
float2 ditherCoord = screenPos * _ScreenParams.xy * _VLB_DitheringNoiseTex_TexelSize.xy;
|
||
|
float dither = tex2D(_VLB_DitheringNoiseTex, ditherCoord).r - 0.5;
|
||
|
color += (1 - saturate(intensity)) * _VLB_DitheringFactor * dither;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
ApplyPipelineSpecificIntensityModifier(/* inout */ intensity);
|
||
|
|
||
|
#if VLB_ALPHA_AS_BLACK
|
||
|
color *= intensity;
|
||
|
#else
|
||
|
color.a *= intensity;
|
||
|
#endif
|
||
|
|
||
|
#ifdef VLB_FOG_APPLY
|
||
|
VLB_FOG_APPLY(color);
|
||
|
#endif
|
||
|
|
||
|
return color;
|
||
|
}
|
||
|
|
||
|
#endif
|