mirror of
https://github.com/Ttanasart-pt/Pixel-Composer.git
synced 2025-01-27 21:38:33 +01:00
313 lines
7.6 KiB
GLSL
313 lines
7.6 KiB
GLSL
// FIXME: Temporary fix!
|
|
precision highp float;
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Defines
|
|
//
|
|
|
|
// Maximum number of point lights
|
|
#define MAX_PUNCTUAL_LIGHTS 8
|
|
// Number of samples used when computing shadows
|
|
#define SHADOWMAP_SAMPLE_COUNT 12
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Varyings
|
|
//
|
|
|
|
varying vec3 v_vVertex;
|
|
|
|
varying vec2 v_vTexCoord;
|
|
varying mat3 v_mTBN;
|
|
varying vec4 v_vPosition;
|
|
|
|
varying vec4 v_vPosShadowmap;
|
|
|
|
varying vec2 v_vSplatmapCoord;
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Uniforms
|
|
//
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
// Material
|
|
|
|
// Material index
|
|
// uniform float bbmod_MaterialIndex;
|
|
|
|
// RGB: Base color, A: Opacity
|
|
#define bbmod_BaseOpacity gm_BaseTexture
|
|
|
|
// RGBA
|
|
uniform vec4 bbmod_BaseOpacityMultiplier;
|
|
|
|
// If 1.0 then the material uses roughness
|
|
uniform float bbmod_IsRoughness;
|
|
// If 1.0 then the material uses metallic workflow
|
|
uniform float bbmod_IsMetallic;
|
|
// RGB: Tangent-space normal, A: Smoothness or roughness
|
|
uniform sampler2D bbmod_NormalW;
|
|
// RGB: specular color / R: Metallic, G: ambient occlusion
|
|
uniform sampler2D bbmod_Material;
|
|
|
|
// Pixels with alpha less than this value will be discarded
|
|
uniform float bbmod_AlphaTest;
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
// Camera
|
|
|
|
// Camera's position in world space
|
|
uniform vec3 bbmod_CamPos;
|
|
// Distance to the far clipping plane
|
|
uniform float bbmod_ZFar;
|
|
// Camera's exposure value
|
|
uniform float bbmod_Exposure;
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
// Fog
|
|
|
|
// The color of the fog
|
|
uniform vec4 bbmod_FogColor;
|
|
// Maximum fog intensity
|
|
uniform float bbmod_FogIntensity;
|
|
// Distance at which the fog starts
|
|
uniform float bbmod_FogStart;
|
|
// 1.0 / (fogEnd - fogStart)
|
|
uniform float bbmod_FogRcpRange;
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
// Ambient light
|
|
|
|
// RGBM encoded ambient light color on the upper hemisphere.
|
|
uniform vec4 bbmod_LightAmbientUp;
|
|
// RGBM encoded ambient light color on the lower hemisphere.
|
|
uniform vec4 bbmod_LightAmbientDown;
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
// Directional light
|
|
|
|
// Direction of the directional light
|
|
uniform vec3 bbmod_LightDirectionalDir;
|
|
// RGBM encoded color of the directional light
|
|
uniform vec4 bbmod_LightDirectionalColor;
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
// Terrain
|
|
|
|
// Splatmap texture
|
|
uniform sampler2D bbmod_Splatmap;
|
|
// Splatmap channel to read. Use -1 for none.
|
|
uniform int bbmod_SplatmapIndex;
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Includes
|
|
//
|
|
struct Material
|
|
{
|
|
vec3 Base;
|
|
float Opacity;
|
|
vec3 Normal;
|
|
float Metallic;
|
|
float Roughness;
|
|
vec3 Specular;
|
|
float Smoothness;
|
|
float SpecularPower;
|
|
float AO;
|
|
vec3 Emissive;
|
|
vec4 Subsurface;
|
|
vec3 Lightmap;
|
|
};
|
|
|
|
Material CreateMaterial(mat3 TBN)
|
|
{
|
|
Material m;
|
|
m.Base = vec3(1.0);
|
|
m.Opacity = 1.0;
|
|
m.Normal = normalize(TBN * vec3(0.0, 0.0, 1.0));
|
|
m.Metallic = 0.0;
|
|
m.Roughness = 1.0;
|
|
m.Specular = vec3(0.0);
|
|
m.Smoothness = 0.0;
|
|
m.SpecularPower = 1.0;
|
|
m.AO = 1.0;
|
|
m.Emissive = vec3(0.0);
|
|
m.Subsurface = vec4(0.0);
|
|
m.Lightmap = vec3(0.0);
|
|
return m;
|
|
}
|
|
#define F0_DEFAULT vec3(0.04)
|
|
#define X_GAMMA 2.2
|
|
|
|
/// @desc Converts gamma space color to linear space.
|
|
vec3 xGammaToLinear(vec3 rgb)
|
|
{
|
|
return pow(rgb, vec3(X_GAMMA));
|
|
}
|
|
|
|
/// @desc Converts linear space color to gamma space.
|
|
vec3 xLinearToGamma(vec3 rgb)
|
|
{
|
|
return pow(rgb, vec3(1.0 / X_GAMMA));
|
|
}
|
|
|
|
/// @desc Gets color's luminance.
|
|
float xLuminance(vec3 rgb)
|
|
{
|
|
return (0.2126 * rgb.r + 0.7152 * rgb.g + 0.0722 * rgb.b);
|
|
}
|
|
/// @note Input color should be in gamma space.
|
|
/// @source https://graphicrants.blogspot.cz/2009/04/rgbm-color-encoding.html
|
|
vec4 xEncodeRGBM(vec3 color)
|
|
{
|
|
vec4 rgbm;
|
|
color *= 1.0 / 6.0;
|
|
rgbm.a = clamp(max(max(color.r, color.g), max(color.b, 0.000001)), 0.0, 1.0);
|
|
rgbm.a = ceil(rgbm.a * 255.0) / 255.0;
|
|
rgbm.rgb = color / rgbm.a;
|
|
return rgbm;
|
|
}
|
|
|
|
/// @source https://graphicrants.blogspot.cz/2009/04/rgbm-color-encoding.html
|
|
vec3 xDecodeRGBM(vec4 rgbm)
|
|
{
|
|
return 6.0 * rgbm.rgb * rgbm.a;
|
|
}
|
|
|
|
/// @desc Unpacks material from textures.
|
|
/// @param texBaseOpacity RGB: base color, A: opacity
|
|
/// @param isRoughness
|
|
/// @param texNormalW
|
|
/// @param isMetallic
|
|
/// @param texMaterial
|
|
/// @param texSubsurface RGB: subsurface color, A: intensity
|
|
/// @param texEmissive RGBA: RGBM encoded emissive color
|
|
/// @param texLightmap RGBA: RGBM encoded lightmap
|
|
/// @param uvLightmap Lightmap texture coordinates
|
|
/// @param TBN Tangent-bitangent-normal matrix
|
|
/// @param uv Texture coordinates
|
|
Material UnpackMaterial(
|
|
sampler2D texBaseOpacity,
|
|
float isRoughness,
|
|
sampler2D texNormalW,
|
|
float isMetallic,
|
|
sampler2D texMaterial,
|
|
mat3 TBN,
|
|
vec2 uv)
|
|
{
|
|
Material m = CreateMaterial(TBN);
|
|
|
|
// Base color and opacity
|
|
vec4 baseOpacity = texture2D(texBaseOpacity,
|
|
uv
|
|
);
|
|
m.Base = xGammaToLinear(baseOpacity.rgb);
|
|
m.Opacity = baseOpacity.a;
|
|
|
|
// Normal vector and smoothness/roughness
|
|
vec4 normalW = texture2D(texNormalW,
|
|
uv
|
|
);
|
|
m.Normal = normalize(TBN * (normalW.rgb * 2.0 - 1.0));
|
|
|
|
if (isRoughness == 1.0)
|
|
{
|
|
m.Roughness = mix(0.1, 0.9, normalW.a);
|
|
m.Smoothness = 1.0 - m.Roughness;
|
|
}
|
|
else
|
|
{
|
|
m.Smoothness = mix(0.1, 0.9, normalW.a);
|
|
m.Roughness = 1.0 - m.Smoothness;
|
|
}
|
|
|
|
// Material properties
|
|
vec4 materialProps = texture2D(texMaterial,
|
|
uv
|
|
);
|
|
|
|
if (isMetallic == 1.0)
|
|
{
|
|
m.Metallic = materialProps.r;
|
|
m.AO = materialProps.g;
|
|
m.Specular = mix(F0_DEFAULT, m.Base, m.Metallic);
|
|
m.Base *= (1.0 - m.Metallic);
|
|
}
|
|
else
|
|
{
|
|
m.Specular = materialProps.rgb;
|
|
m.SpecularPower = exp2(1.0 + (m.Smoothness * 10.0));
|
|
}
|
|
|
|
return m;
|
|
}
|
|
|
|
void Fog(float depth)
|
|
{
|
|
vec3 ambientUp = xGammaToLinear(bbmod_LightAmbientUp.rgb) * bbmod_LightAmbientUp.a;
|
|
vec3 ambientDown = xGammaToLinear(bbmod_LightAmbientDown.rgb) * bbmod_LightAmbientDown.a;
|
|
vec3 directionalLightColor = xGammaToLinear(bbmod_LightDirectionalColor.rgb) * bbmod_LightDirectionalColor.a;
|
|
vec3 fogColor = xGammaToLinear(bbmod_FogColor.rgb) * (ambientUp + ambientDown + directionalLightColor);
|
|
float fogStrength = clamp((depth - bbmod_FogStart) * bbmod_FogRcpRange, 0.0, 1.0) * bbmod_FogColor.a;
|
|
gl_FragColor.rgb = mix(gl_FragColor.rgb, fogColor, fogStrength * bbmod_FogIntensity);
|
|
}
|
|
void Exposure()
|
|
{
|
|
gl_FragColor.rgb = vec3(1.0) - exp(-gl_FragColor.rgb * bbmod_Exposure);
|
|
}
|
|
|
|
void GammaCorrect()
|
|
{
|
|
gl_FragColor.rgb = xLinearToGamma(gl_FragColor.rgb);
|
|
}
|
|
|
|
void UnlitShader(Material material, float depth)
|
|
{
|
|
gl_FragColor.rgb = material.Base;
|
|
gl_FragColor.rgb += material.Emissive;
|
|
gl_FragColor.a = material.Opacity;
|
|
// Soft particles
|
|
Fog(depth);
|
|
Exposure();
|
|
GammaCorrect();
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Main
|
|
//
|
|
void main()
|
|
{
|
|
Material material = UnpackMaterial(
|
|
bbmod_BaseOpacity,
|
|
bbmod_IsRoughness,
|
|
bbmod_NormalW,
|
|
bbmod_IsMetallic,
|
|
bbmod_Material,
|
|
v_mTBN,
|
|
v_vTexCoord);
|
|
|
|
// Splatmap
|
|
vec4 splatmap = texture2D(bbmod_Splatmap, v_vSplatmapCoord);
|
|
if (bbmod_SplatmapIndex >= 0)
|
|
{
|
|
// splatmap[bbmod_SplatmapIndex] does not work in HTML5
|
|
material.Opacity *= ((bbmod_SplatmapIndex == 0) ? splatmap.r
|
|
: ((bbmod_SplatmapIndex == 1) ? splatmap.g
|
|
: ((bbmod_SplatmapIndex == 2) ? splatmap.b
|
|
: splatmap.a)));
|
|
}
|
|
|
|
material.Base *= bbmod_BaseOpacityMultiplier.rgb;
|
|
material.Opacity *= bbmod_BaseOpacityMultiplier.a;
|
|
|
|
if (material.Opacity < bbmod_AlphaTest)
|
|
{
|
|
discard;
|
|
}
|
|
|
|
UnlitShader(material, v_vPosition.z);
|
|
|
|
}
|