Space-Smash-Out/Assets/Models/Materials/FesnelRipple.shader

200 lines
5.9 KiB
GLSL

Shader "CustomShader/FesnelRipple"
{
Properties
{
_base_alpha ("base alpha", Range(0.0, 1.0)) = 0.5
_shield_color ("shield color", Color) = (0, 0, 1, 1)
_shield_color_brightness ("shield color brightness", Range(0.25, 20)) = 10.0
_shield_intensity ("shield intensity", Range(0.1,20)) = 2.0
_rotation ("rotation", Vector) = (0.1, 0.0, 0.0)
_shield_size ("shield size", Range(0.0,0.5)) = 0.0
_shield_pulse_frequency ("shield pulse frequency", float)= 1.0
_shield_pulse_density ("shield pulse density", float) = 1.0
_shield_pulse_amplitude ("shield pulse amplitude", float) = 1.0
_shield_pulse_blend ("shield pulse blend", float) = 1.0
_shield_pulse_radius ("shield pulse radius", float) = 1.0
_impact_origin ("impact origin", Vector) = (1.0, 0.0, 0.0)
_impact_frequency ("impact frequency", float) = 5.0
_impact_density ("impact density", float) = 5.0
_impact_amplitude ("impact amplitude", float) = 6.0
_impact_blend ("impact blend", float) = 0
_impact_radius ("impact radius", float) = 1.1
_impact_anim ("impact anim", Range(0.0,0.1)) = 0.0
}
HLSLINCLUDE
#pragma vertex vert
#pragma fragment frag
#include "UnityCG.cginc"
// properties
float _base_alpha;
float4 _shield_color;
float _shield_color_brightness;
float _shield_intensity;
float3 _rotation;
float _shield_size;
float _shield_pulse_frequency;
float _shield_pulse_density;
float _shield_pulse_amplitude;
float _shield_pulse_blend;
float _shield_pulse_radius;
float3 _impact_origin;
float _impact_frequency;
float _impact_amplitude;
float _impact_blend;
float _impact_density;
float _impact_radius;
float _impact_anim;
// Impact functions
float _FadeRipple(float dist, float blend, float radius) {
// create a radius
return smoothstep(radius, 0, dist);
}
float _ComputeRipple(float3 vert, float3 orig, float blend, float radius, float freq, float dens, float ampl, float anim) {
// calculate the intensity of the impact
float dist = distance(vert, orig);
float i = sin((freq * anim - dist * dens)) * ampl;
return i * _FadeRipple(dist, blend, radius);
}
// Shield functions
float ComputeFresnel(float3 norm, float3 view_dir, float intensity) {
float width = 0.95;
float softness = 0.96;
float fresnel = 1.0 - saturate(dot(norm, view_dir));
fresnel = pow(fresnel, intensity);
fresnel = lerp(1, smoothstep(width, softness, fresnel), step(0, width));
return fresnel;
}
float4 Rotate(float3 vert, float3 speed) {
// build the 3 rotation matrices
speed = speed * _Time * 5.0;
float4x4 xrot = float4x4(
float4(1.0, 0.0, 0.0, 0.0),
float4(0.0, cos(speed.x), -sin(speed.x), 0.0),
float4(0.0, sin(speed.x), cos(speed.x), 0.0),
float4(0.0, 0.0, 0.0, 1.0));
float4x4 yrot = float4x4(
float4(cos(speed.y), 0.0, -sin(speed.y), 0.0),
float4(0.0, 1.0, 0.0, 0.0),
float4(sin(speed.y), 0.0, cos(speed.y), 0.0),
float4(0.0, 0.0, 0.0, 1.0));
float4x4 zrot = float4x4(
float4(cos(speed.z), -sin(speed.z), 0.0, 0.0),
float4(sin(speed.z), cos(speed.z), 0.0, 0.0),
float4(0.0, 0.0, 1.0, 0.0),
float4(0.0, 0.0, 0.0, 1.0));
float4 output = mul(xrot, mul(yrot, mul(zrot, float4(vert.xyz, 1.0))));
return output;
}
// vertex shader inputs
struct appdata
{
float4 vertex : POSITION;
float4 normal : NORMAL;
float2 uv : TEXCOORD0;
};
// vertex shader outputs to fragment
struct v2f
{
float2 uv : TEXCOORD0; // texture coordinate
float4 vertex : SV_POSITION; // clips space position
float3 normal : NORMAL;
float3 viewDirection : TEXCOORD2;
};
// vertex shader
v2f vert (appdata v)
{
v.vertex = Rotate(v.vertex, _rotation);
v.normal = Rotate(v.normal, _rotation);
float4 vert_world_pos = mul(unity_ObjectToWorld, v.vertex);
float impact_ripple = _ComputeRipple(v.vertex.xyz,
_impact_origin.xyz,
_impact_blend,
_impact_radius,
_impact_frequency,
_impact_density,
_impact_amplitude,
_impact_anim);
v.vertex += v.normal * (impact_ripple + _shield_size);
v2f o;
// transform position to clip space
// (multiply with model*view*projection matrix)
o.vertex = UnityObjectToClipPos(v.vertex);
// just pass the texture coordinate
o.uv = v.uv;
o.normal = v.normal.xyz;
o.viewDirection = normalize(WorldSpaceViewDir(vert_world_pos));
return o;
}
ENDHLSL
SubShader
{
Tags { "RenderType"="Transparent"
"IgnoreProjector"="True"
"Queue"="Transparent"
"ForceNoShadowCasting"="True"
}
Blend SrcAlpha OneMinusSrcAlpha
Pass
{
Tags
{
"LightMode" = "UniversalForward"
}
Name "BackCullingPass"
Cull Back
HLSLPROGRAM
fixed4 frag (v2f i) : SV_Target
{
float f = ComputeFresnel(i.normal, i.viewDirection, _shield_intensity);
fixed4 col = 0;
fixed3 normalToGray = (0.299 * i.normal.r) + (0.587 * i.normal.g) + (0.114 * i.normal.b);
col.rgb = f * _shield_color_brightness * _shield_color + (abs(normalToGray) - 0.3);
col.a = f * _base_alpha;
return col;
}
ENDHLSL
}
Pass
{
Tags
{
"LightMode" = "SRPDefaultUnlit"
}
Name "FrontCullingPass"
Cull Front
HLSLPROGRAM
fixed4 frag (v2f i) : SV_Target
{
float f = ComputeFresnel(i.normal, i.viewDirection, _shield_intensity);
fixed4 col = 0;
fixed3 normalToGray = (0.299 * i.normal.r) + (0.587 * i.normal.g) + (0.114 * i.normal.b);
col.rgb = f * _shield_color_brightness * _shield_color + (abs(normalToGray) * 0.5 - 0.5);
col.a = f * _base_alpha * 0.5;
return col;
}
ENDHLSL
}
}
}