1 Introdução
Esse é o seguimento de Blinn Phong com Textures de Metallic e esperadamente, o último, tenho que dizer, eu nunca escrevi artigos antes, eu sei que algumas coisas aqui pode ser um pouco confusas ou estranhas mas em minha opinião é melhor escrever um artigo ruim sobre algo interessante que você descobriu do que dizer no chat do discord e deixar que se perca para sempre.
2 Efeito de Fresnel
O Efeito de Fresnel é uma daquelas coisas que uma vez que você perceber que ele existe você sempre vai ver ele em todo lugar, de plásticos brilhantes a pisos cerâmicos ou grandes corpos de água, fresnel normalmente não é visível em metais e em renderização 3D normalmente se assume que metais não tem fresnel.
Ao invés de te mostrar um gráfico chato, irei the mostrar fotos reais do efeito.
Você entendeu? se não, vou dizer então:
O efeito de fresnel é um efeito de luz que você normalmente vê em superfície não metálicas onde o reflexo se torna mais intenso quanto maior o ângulo entre o normal da superfície e a direção de view se torna, como pode ver na primeira foto, nós quase não conseguimos ver um reflexo, mas ainda tem um, fresnel não remove completamente os reflexos quando se olha de cima, isso é normalmente assumido de ser próximo a 5% para grande parte das coisas mas você pode encontrar tabelas com mais valores na internet e como pode ver na segunda e terceira foto, o reflexo fica mais intenso conforme o ângulo aumenta.
3 Em código
Se você sabe o que é um produto escalar (dot product) você provavelmente já adivinhou como isso vai se parecer.
#define FRESNEL_MIN 0.05
#define FRESNEL_MAX 0.95
float fresnelFactor(vec3 viewDirection, vec3 normal) {
float fresnelDot = 1.0 - max(dot(-viewDirection, normal), 0.0);
return FRESNEL_MIN + (FRESNEL_MAX * pow(fresnelDot, 5.0));
}
Aqui nós criamos uma função para calcular o fator de fresnel para a direção de view (na minha configuração, isso aponta da câmera para o fragmento, eu acho mais intuitivo desse jeito) e o normal da superfície.
Nós então calculamos o produto escalar entre o oposto da direção de view e o normal da superfície (isso nos dá o quão "próximo" a direção de view está do normal) e então limitamos o valor para que não se torne negativo e espelhamos ele pois queremos um valor que fique maior conforme o ângulo aumenta.
Como fresnel não é linear (só vemos reflexos fortes quando o ângulo está bem grande), nós precisamos aplicar uma função exponencial (pow) para ele, que normalmente é 5.0, nós então multiplicamos pelo valor máximo do efeito de fresnel e adicionamos o valor mínimo.
Já que o mínimo e máximo sempre somam para um, pode ser simplificado para uma única constante.
#define FRESNEL_BALANCE 0.05
float fresnelFactor(vec3 viewDirection, vec3 normal) {
float fresnelDot = 1.0 - max(dot(-viewDirection, normal), 0.0);
return FRESNEL_BALANCE + ((1.0 - FRESNEL_BALANCE) * pow(fresnelDot, 5.0));
}
Aqui nós chamamos de "FRESNEL_BALANCE" mas é normalmente chamado de "F0" (Fresnel no ângulo 0).
Agora nós só precisamos adicionar ele para a nossa função de reflexo e também multiplicar pela roughness para que o reflexo diminua conforme a roughness aumente.
vec3 computeReflection(
samplerCube cube,
vec3 viewDirection,
vec3 normal,
vec3 diffuseColor,
float metallic,
float roughness
) {
ivec2 cubemapSize = textureSize(cube, 0);
float mipLevels = 1.0 + floor(log2(max(float(cubemapSize.x), float(cubemapSize.y))));
float lodLevel = mipLevels * sqrt(roughness);
vec3 reflectedColor = textureLod(cube, reflect(viewDirection, normal), lodLevel).rgb;
return mix(
reflectedColor * fresnelFactor(viewDirection, normal) * pow(1.0 - roughness, 2.0),
reflectedColor * diffuseColor,
metallic
);
}
E é assim que se parece:
4 Blinn Phong
Depois de pensar sobre o que eu fiz, eu eventualmente simplifiquei toda a conversão de roughness para shininess de blinn phong em uma única função que converte um material de PBR de Metallic Roughness para um Material de Blinn Phong no shader, você só vai precisar usar isso como material nos seus cálculos de luz.
#define PI 3.14159265359
#define MAX_SHININESS 2048.0
#define DIFFUSE_BALANCE 0.50
#define FRESNEL_BALANCE 0.05
struct BlinnPhongMaterial {
float shininess;
vec3 diffuse;
vec3 specular;
vec3 ambient;
};
float fresnelFactor(vec3 viewDirection, vec3 normal) {
float fresnelDot = 1.0 - max(dot(-viewDirection, normal), 0.0);
return FRESNEL_BALANCE + ((1.0 - FRESNEL_BALANCE) * pow(fresnelDot, 5.0));
}
BlinnPhongMaterial convertPBRMaterialToBlinnPhong(
vec3 viewDirection, vec3 normal,
vec3 color, float metallic, float roughness, float ambientOcclusion
) {
float shininess = pow(MAX_SHININESS, 1.0 - roughness);
float specular = ((shininess + 2.0) * (shininess + 4.0)) / (8.0 * PI * (pow(2.0, -shininess * 0.5) + shininess));
float fresnel = fresnelFactor(viewDirection, normal);
return BlinnPhongMaterial(
shininess,
mix(
(color * DIFFUSE_BALANCE) / PI,
vec3(0.0),
metallic
),
mix(
vec3(max(specular - 0.3496155267919281, 0.0)) * (1.0 - DIFFUSE_BALANCE) * fresnel * PI,
vec3(specular) * color,
metallic
),
mix(
vec3(ambientOcclusion) * color,
vec3(0.0),
metallic
)
);
}
vec3 directionalLight(
vec3 lightDirection,
vec3 viewDirection,
vec3 normal,
BlinnPhongMaterial material
) {
vec3 halfwayDirection = -normalize(lightDirection + viewDirection);
float diffuseFactor = max(dot(normal, -lightDirection), 0.0);
float specularFactor = pow(max(dot(normal, halfwayDirection), 0.0), material.shininess) * diffuseFactor;
float lightDiffuse = 1.90;
float lightSpecular = 1.0;
float lightAmbient = 0.10;
vec3 diffuse = lightDiffuse * diffuseFactor * material.diffuse;
vec3 specular = lightSpecular * specularFactor * material.specular;
vec3 ambient = lightAmbient * material.ambient;
return diffuse + specular + ambient;
}
Dessa vez eu dividi o diffuse por PI então vai se parecer escuro, eu também adicionei um fresnel nele, então deve parecer mais realista quando olhando de altos ângulos, é assim que se parece naquela textura de couro sem reflexos:
Eu também dividi a função de reflexo em três funções mais genéricas.
float cubemapMipLevels(samplerCube cube) {
ivec2 cubemapSize = textureSize(cube, 0);
float mipLevels = 1.0 + floor(log2(max(float(cubemapSize.x), float(cubemapSize.y))));
return mipLevels;
}
vec4 cubemapRoughnessColor(samplerCube cube, float roughness, vec3 direction) {
float mipLevels = cubemapMipLevels(cube);
float lodLevel = mipLevels * sqrt(roughness);
return textureLod(cube, direction, lodLevel);
}
vec3 cubemapReflection(
vec3 cubemapColor,
vec3 viewDirection, vec3 normal,
vec3 color, float metallic, float roughness
) {
return mix(
cubemapColor * fresnelFactor(viewDirection, normal) * pow(1.0 - roughness, 2.0),
cubemapColor * color,
metallic
);
}
5 Resultados Finais
Dessa vez essas texturas vem de múltiplos lugares.