L'aberration chromatique est causée lorsqu'un objectif ne peut pas focaliser chaque couleur sur le même point focal. Un moyen simple de simuler cet effet et de le rendre en tant que post-traitement plein écran rapide consiste à appliquer un décalage à chaque canal de couleur dans un shader de fragment.
En utilisant un décalage différent pour chaque canal, vous pouvez obtenir un fac-similé raisonnable de l'effet souhaité. Un exemple de cette technique peut être trouvé ici ; le fragment shader ressemblerait à ceci:
void main () {
// Previously, you'd have rendered your complete scene into a texture
// bound to "fullScreenTexture."
vec4 rValue = texture2D(fullscreenTexture, gl_TexCoords[0] - rOffset);
vec4 gValue = texture2D(fullscreenTexture, gl_TexCoords[0] - gOffset);
vec4 bValue = texture2D(fullscreenTexture, gl_TexCoords[0] - bOffset);
// Combine the offset colors.
gl_FragColor = vec4(rValue.r, gValue.g, bValue.b, 1.0);
}
Ce simple hack ne prend pas vraiment en compte le fait que l'aberration chromatique est un effet d'objectif, cependant: pour obtenir une meilleure simulation, vous voudriez réellement rendre quelque chose pour agir comme l'objectif. Ceci est similaire à la façon dont vous effectuez le rendu d'objets réfléchissants ou réfringents. Par conséquent, un shader de réflexion / réfraction typique peut être la base de la mise en œuvre de l'aberration chromatique.
Normalement, vous devriez calculer un seul vecteur de réfraction basé sur un vecteur de vue et certains défini indice de réfraction , en utilisant de GLSL réfractent fonction dans un vertex shader:
void main () {
// ...
// RefractionVector is a varying vec3.
// 'ratio' is the ratio of the two indices of refraction.
RefractionVector = refract(incidentVector, normalVector, ratio);
// ...
}
Ensuite, vous utiliseriez ce vecteur dans un shader de fragment pour effectuer une recherche de texture de cube (dans une carte d'environnement). En règle générale, cela se fait parallèlement à un effet de réflexion, et combiné utilisé un terme de Fresnel calculé .
Pour simuler l'aberration chromatique, vous pouvez alors effectuer trois calculs de vecteur de réfraction différents , chacun légèrement décalé via différents indices de réfraction, dans le vertex shader:
void main () {
// ...
// RefractionVector is a varying vec3, as above.
// 'ratioR,' et cetera, is the ratio of indices of refraction for
// the red, green and blue components respectively.
RedRefractionVector = refract(incidentVector, normalVector, ratioR);
GreenRefractionVector = refract(incidentVector, normalVector, ratioG);
BlueRefractionVector = refract(incidentVector, normalVector, ratioB);
// ...
}
Ces trois vecteurs différents peuvent être utilisés pour effectuer trois recherches de carte de cube différentes, qui peuvent être mélangées de la même manière que les couleurs ont été mélangées dans l'exemple simple:
void main () {
vec3 color;
color.r = vec3(textureCube(EnvironmentMap, RedRefractionVector)).r;
color.g = vec3(textureCube(EnvironmentMap, GreenRefractionVector)).g;
color.b = vec3(textureCube(EnvironmentMap, BlueRefractionVector)).b;
gl_FragColor = vec4(color, 1.0);
}
Pour plus de détails, l' OpenGL Orange Book est disponible et contient un exemple des effets de réflexion et de réfraction de base, ainsi qu'un exemple de l'effet d'aberration chromatique.