Divergent/mods/Parallax Reflex Sights/gamedata/shaders/r3/parallax_mark.ps

107 lines
3.8 KiB
PostScript

/*
=====================================================================
Addon : Parallax Reflex Sights
Link : https://www.moddb.com/mods/stalker-anomaly/addons/parallax-reflex-sights
Authors : LVutner, party_50
Date : 06.02.2024
Last Edit : 03.09.2024
=====================================================================
*/
#include "common.h"
#include "mark_adjust.h"
// Important:
// In perfect world OFFSET constants should be 0, but most of reflex sight lenses
// are not actually parallel to screen, so we compensate it. For PROJECT_DISTANCE=100
// offset values should be at least 0.005 even for perfect models and position configs due
// to normal vectors resolution.
//
// If you want the most realistic look, set PROJECT_DISTANCE to some high value (like 100.0),
// increase SIZE_FACTOR to something like 20.0, set OFFSET_X and OFFSET_Y to 0.005.
// Then you will have to adjust models so that mark texture point is exactly in center
// and edit aim position in configs.
#define OFFSET_X 0.004 // (default 0.004) Normal vector x coordinate max absolute value which is considered 0
#define OFFSET_Y 0.05 // (default 0.05) Normal vector y coordinate max absolute value which is considered 0
#define PROJECT_DISTANCE 20.0 // (default 20.0) Distance to projected mark
#define SIZE_FACTOR 4.0 // (default 4.0) Mark size factor
// Vertex to Pixel struct
struct vf
{
float2 tc0 : TEXCOORD0;
float3 v_pos : TEXCOORD1;
float3 v_nrm : TEXCOORD2;
};
// This gives us cotangent basis that can be used instead of TBN.
// It is useful when tangents of your mesh are broken, or not available.
// Source: http://www.thetenthplanet.de/archives/1180
float3x3 cotangent_frame(float3 N, float3 P, float2 uv)
{
// Get edge vectors of the pixel triangle
float3 dp1 = ddx(P);
float3 dp2 = ddy(P);
float2 duv1 = ddx(uv);
float2 duv2 = ddy(uv);
// Solve the linear system
float3 dp2perp = cross(dp2, N);
float3 dp1perp = cross(N, dp1);
float3 T = dp2perp * duv1.x + dp1perp * duv2.x;
float3 B = dp2perp * duv1.y + dp1perp * duv2.y;
// Construct a scale-invariant frame
float invmax = rsqrt(max(dot(T, T), dot(B, B)));
return float3x3(T * invmax, B * invmax, N);
}
// If N.xy vector is close to zero, make it zero
float3 offset_normal(float3 N)
{
if (N.x > 0)
N.x = max(N.x, OFFSET_X) - OFFSET_X;
else
N.x = min(N.x, -OFFSET_X) + OFFSET_X;
if (N.y > 0)
N.y = max(N.y, OFFSET_Y) - OFFSET_Y;
else
N.y = min(N.y, -OFFSET_Y) + OFFSET_Y;
return N;
}
float4 main(vf I): SV_Target
{
// Derive view direction from view space position
float3 V = -I.v_pos;
// Build cotangent frame
// Important: In theory, you don't need to do this. It should be possible to pass TBN straight from VS
float3x3 TBN = cotangent_frame(offset_normal(I.v_nrm), I.v_pos, I.tc0.xy);
// Transform view direction to tangent space, and normalize (Just in case)
float3 V_tangent = normalize(float3(dot(V, TBN[0]), dot(V, TBN[1]), dot(V, TBN[2])));
// Calculate texture coordinates used to fetch the mark texture
// Important: PROJECT_DISTANCE can be positive or negative, 0 = no projection at all
float2 parallax_tc = I.tc0 - V_tangent.xy * PROJECT_DISTANCE;
// Upscaling the texture
parallax_tc.x = (parallax_tc.x + (SIZE_FACTOR - 1) / 2) / SIZE_FACTOR;
parallax_tc.y = (parallax_tc.y + (SIZE_FACTOR - 1) / 2) / SIZE_FACTOR;
#ifdef MARK_ADJUST
parallax_tc = mark_adjust(parallax_tc);
#endif
// Fetch the mark texture
// Important: We do not want texture to repeat itself, so we use sampler with CLAMP address
// Important2: We do not want to sample mip levels of the mark texture, let's keep this thing sharp as fuck
float4 color = s_base.SampleLevel(smp_rtlinear, parallax_tc, 0.0);
return color;
}