16/shaders/gshader.frag

105 lines
4.4 KiB
GLSL
Raw Permalink Normal View History

2023-02-02 15:22:33 +00:00
#version 420 core
layout(std140, binding = 1) uniform Material
{
vec3 ka;
vec3 kd;
vec3 ks;
float p;
2023-02-03 12:30:42 +00:00
bool normalmapped;
2023-02-09 07:21:02 +00:00
bool parallaxmapped;
2023-02-02 15:22:33 +00:00
};
layout (location = 0) out vec3 gPosition;
layout (location = 1) out vec3 gNormal;
layout (location = 2) out vec4 gDiffuseP;
layout (location = 3) out vec4 gAmbientSpecular;
in vec3 vertex; // Позиция вершины в пространстве
in vec3 N; // Нормаль трансформированноая
in vec2 texCoord; // Текстурные координаты
2023-02-03 12:11:08 +00:00
in vec3 T; // Касательный вектор
in vec3 B; // Бикасательный вектор
2023-02-09 07:21:02 +00:00
in vec3 view; // Вектор от поверхности к камере
2023-02-02 15:22:33 +00:00
uniform sampler2D tex_diffuse;
uniform sampler2D tex_ambient;
uniform sampler2D tex_specular;
2023-02-03 12:11:08 +00:00
uniform sampler2D tex_heights;
uniform sampler2D tex_normal;
2023-02-02 15:22:33 +00:00
2023-02-13 07:12:34 +00:00
uniform float parallax_heightScale = 0.1;
2023-02-09 07:21:02 +00:00
2023-02-02 15:22:33 +00:00
void main()
{
2023-02-03 12:11:08 +00:00
// Сформируем TBN матрицу
mat3 TBN = mat3(T, B, N);
2023-02-09 07:21:02 +00:00
// Перевод вектора в касательное пространство
vec3 viewTBN = normalize(transpose(TBN) * view);
// Измененные текстурные координаты
vec2 new_texCoord = texCoord;
2023-02-02 15:22:33 +00:00
// Сохранение позиции фрагмента в G-буфере
gPosition = vertex;
2023-02-09 07:21:02 +00:00
if (parallaxmapped)
{
2023-02-13 07:12:34 +00:00
// Число слоев
float layersCount = 32;
// Вычислим размер каждого слоя
float layerDepth = 1.0 / layersCount;
// Глубина текущего слоя
float currentLayerDepth = 0.0;
// Величина сдвига между слоями
vec2 deltaTexCoords = (parallax_heightScale * viewTBN.xy / viewTBN.z) / layersCount;
// Переменные для вычислений
vec2 currentTexCoords = texCoord;
float currentDepthMapValue = 1.0 - texture(tex_heights, currentTexCoords).r;
// Пока глубина текущего слоя меньше текущего значения глубины из текстуры
while(currentLayerDepth < currentDepthMapValue)
{
// Сдвигаем координаты
currentTexCoords -= deltaTexCoords;
// Обновляем значение глубины из текстуры
currentDepthMapValue = 1.0 - texture(tex_heights, currentTexCoords).r;
// Сдвигаем глубину на следующий слой
currentLayerDepth += layerDepth;
}
// Получим значение текстурных координат с предыдущего шага
vec2 prevTexCoords = currentTexCoords + deltaTexCoords;
2023-02-09 07:21:02 +00:00
2023-02-13 07:12:34 +00:00
// Значения глубины до и после пересечения
float afterDepth = currentDepthMapValue - currentLayerDepth;
float beforeDepth = 1.0 - texture(tex_heights, prevTexCoords).r - currentLayerDepth + layerDepth;
// Интерполяция текстурных координат
float weight = afterDepth / (afterDepth - beforeDepth);
new_texCoord = prevTexCoords * weight + currentTexCoords * (1.0 - weight);
// Проверка диапазона [0;1]
2023-02-09 07:21:02 +00:00
if(new_texCoord.x > 1.0 || new_texCoord.y > 1.0 || new_texCoord.x < 0.0 || new_texCoord.y < 0.0)
discard;
}
2023-02-02 15:22:33 +00:00
// Сохранение нормали в G-буфере
gNormal = N;
2023-02-03 12:30:42 +00:00
// Если используется карта нормалей
if (normalmapped)
{
// Получим значение из карты нормалей и приведем их к диапазону [-1;1]
2023-02-09 07:21:02 +00:00
gNormal = texture(tex_normal, new_texCoord).rgb * 2 - 1.0f;
2023-02-03 12:30:42 +00:00
gNormal = normalize(TBN * gNormal); // Из касательного пространства в мировые координаты
}
2023-02-02 15:22:33 +00:00
// Сохранение диффузного цвета
2023-02-09 07:21:02 +00:00
gDiffuseP.rgb = texture(tex_diffuse, new_texCoord).rgb * kd;
2023-02-02 15:22:33 +00:00
// Сохранение глянцевости
gDiffuseP.a = p;
// Сохранение фоновой составляющей
2023-02-09 07:21:02 +00:00
gAmbientSpecular.rgb = texture(tex_ambient, new_texCoord).rgb * ka;
2023-02-02 15:22:33 +00:00
// Сохранение зеркальной составляющей
2023-02-09 07:21:02 +00:00
gAmbientSpecular.a = texture(tex_specular, new_texCoord).r * ks.r;
2023-02-02 15:22:33 +00:00
}