You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
269 lines
12 KiB
269 lines
12 KiB
4 years ago
|
|
||
|
vec3 foregroundColor = vec3(.0841, .5329, .9604);
|
||
|
vec3 groundColor = vec3(.2, .3, .5);
|
||
|
vec4 groundSpecular = vec4(.71, .71, .71, 10.);
|
||
|
uniform float uTime;// TIME, IN SECONDS
|
||
|
uniform int f_tex, f_rt, f_moved;
|
||
|
uniform vec4 rot; //ROTATION VALUES USED TO CALCULATE TRANSFORMATION MATRIX
|
||
|
uniform float dFL; //DELTA on FOCAL LENGTH
|
||
|
uniform mat3 transformation, invTr;
|
||
|
uniform vec3 Ambient[ns], Diffuse[ns];
|
||
|
uniform vec4 Specular[ns];
|
||
|
uniform float ks[ns], kr[ns], kf[ns];
|
||
|
uniform vec4 Sph[ns];
|
||
|
uniform sampler2D uSampler[ns];
|
||
|
|
||
|
const float kf_air = 1.000293;
|
||
|
varying vec3 vPos;
|
||
|
float fl=3.;//ORIGINAL FOCAL LENGTH
|
||
|
const float pi=3.14159265359;
|
||
|
const float _2pi=2.*pi;
|
||
|
const int n_ref=15; //2^(hits) - 1 because each hit now spawn 2 rays.
|
||
|
const int max_stack = (n_ref+1)/4;
|
||
|
|
||
|
vec3 scolor = vec3(0,0,0); //Actually 2^n_ref
|
||
|
struct Ray{
|
||
|
vec3 V;
|
||
|
vec3 W;
|
||
|
float kf, cumulativeK;
|
||
|
} stack1[max_stack], stack2[max_stack];
|
||
|
bool modulo2(int n){
|
||
|
return n-2*(n/2) == 1;
|
||
|
}
|
||
|
vec3 getRefraction(vec3 N, vec3 W, float nextkr, float eta, float c1){
|
||
|
float c2 = (1.-eta*eta*(1.-c1*c1));
|
||
|
c2 = sqrt(abs(c2));
|
||
|
return normalize(eta*W + (eta*c1 - c2)*N);
|
||
|
}
|
||
|
void main(){
|
||
|
vec3 LDir=vec3(.5,.5,.5);
|
||
|
vec3 LCol=vec3(1.,1.,1.);
|
||
|
float currKf = kf_air;
|
||
|
vec3 color=vec3(.2, .3, .5);
|
||
|
vec3 trPos = transformation*((dFL+fl+1.)/(fl+1.))*vec3(vPos.xy, -1);
|
||
|
vec3 V0=transformation*vec3(0.,0.,fl+dFL), V = V0;
|
||
|
vec3 W=(trPos-V);
|
||
|
bool rtxoff = false, showtexture = true, moved = false;
|
||
|
float currentK = 1.;
|
||
|
int curr_ptr = 0, curr_top = 0, next_top = 0;
|
||
|
bool final = false, stackswap = false;
|
||
|
for(int j=0;j<n_ref;j++)
|
||
|
{
|
||
|
for(int curr = 0; curr < max_stack; ++curr){
|
||
|
if(curr == curr_ptr){
|
||
|
bool outward = false;
|
||
|
bool skip = false;
|
||
|
if(j > 0){
|
||
|
Ray currR;
|
||
|
if(stackswap)
|
||
|
currR = stack1[curr];
|
||
|
else
|
||
|
currR = stack2[curr];
|
||
|
currKf = currR.kf;
|
||
|
currentK = currR.cumulativeK;
|
||
|
if(currKf <= 0.001 || currentK <= 0.001)
|
||
|
skip = true;
|
||
|
V = currR.V;
|
||
|
W = currR.W;
|
||
|
}
|
||
|
else
|
||
|
W = normalize(W);
|
||
|
if(!skip){
|
||
|
float tMin=10000.;
|
||
|
int iMin = -1;
|
||
|
for(int i=0;i<cns;i++){
|
||
|
vec3 Vp=V-Sph[i].xyz;
|
||
|
float B=dot(W,Vp);
|
||
|
float C=dot(Vp,Vp)-Sph[i].w*Sph[i].w;
|
||
|
float D=B*B-C;
|
||
|
if(D>0.){
|
||
|
float t=-B-sqrt(D);
|
||
|
if(t >= 0.01 && t < tMin){
|
||
|
tMin = t; // This is an optimization, we don't have to do lighting/tex
|
||
|
iMin = i; // for objects that are occuluded, which is expensive!
|
||
|
outward = false;
|
||
|
}
|
||
|
else if (t >= -0.01 && t <0.01){
|
||
|
t = -(t + 2.*B);
|
||
|
if(t > 0.01 && t < tMin){
|
||
|
tMin = t;
|
||
|
iMin = i;
|
||
|
outward = true;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if(iMin >= 0){
|
||
|
float t = tMin;
|
||
|
vec3 S=V+t*W;
|
||
|
for(int i = 0; i < cns; ++ i)
|
||
|
if(i == iMin)
|
||
|
{
|
||
|
vec3 texture_color;
|
||
|
if(showtexture)
|
||
|
{
|
||
|
vec3 tex_sph = (S-Sph[i].xyz);
|
||
|
if(moved)
|
||
|
;//tex_sph=invTr*tex_sph;
|
||
|
float R=Sph[i].w;
|
||
|
float tex_x=acos(abs(tex_sph.x)/sqrt(R*R-tex_sph.y*tex_sph.y));
|
||
|
if(tex_sph.x>0.)
|
||
|
tex_x=pi-tex_x;
|
||
|
tex_x*=1.5708;//*Correct aspect ratio of texture 2:1 -> 2pir:2r
|
||
|
tex_x=tex_x+float(uTime);
|
||
|
float quo=float(int(tex_x/_2pi));
|
||
|
tex_x=tex_x/_2pi - quo;
|
||
|
texture_color=texture2D(uSampler[i],vec2(tex_x,((R-tex_sph.y)/(2.*R)))).xyz;
|
||
|
}
|
||
|
else texture_color = foregroundColor;
|
||
|
|
||
|
vec3 N=normalize(S-Sph[i].xyz);
|
||
|
vec3 realLDir=normalize(LDir-S);
|
||
|
if(outward){
|
||
|
float c1 = dot(N, W);
|
||
|
if(c1 > 0.)
|
||
|
{
|
||
|
c1 = -c1;
|
||
|
N = -N;
|
||
|
outward = true;
|
||
|
}
|
||
|
else outward = false;
|
||
|
color=(Ambient[i]+Diffuse[i]*max(0.,dot(N,realLDir))*LCol)*texture_color;
|
||
|
if(rtxoff || final) //if it's the last hit
|
||
|
{
|
||
|
color += Specular[i].xyz*pow(max(0.,dot(-2.*c1*N-realLDir,realLDir)),Specular[i].w);
|
||
|
scolor += color * currentK;
|
||
|
}
|
||
|
else{
|
||
|
float eta =kf[i]/currKf;
|
||
|
if(outward) eta = 1./eta;
|
||
|
float nextks = currentK * ks[i], nextkr = currentK * kr[i];
|
||
|
bool refl = nextks > 0.001, refr = nextkr > 0.001;
|
||
|
if(refl || refr)
|
||
|
for(int k = 0; k < max_stack; ++k)
|
||
|
if(k == next_top){
|
||
|
if(stackswap){
|
||
|
if(refl)
|
||
|
{
|
||
|
stack2[k] = Ray(S, getRefraction(N, W, nextkr, eta, c1), currKf, nextks); //reflection
|
||
|
currentK -= nextks;
|
||
|
next_top ++;
|
||
|
}
|
||
|
if(refr)
|
||
|
{
|
||
|
if(refl)
|
||
|
stack2[k+1] = Ray(S, getRefraction(N, W, nextkr, eta, c1), kf[i], nextkr); //refraction
|
||
|
else
|
||
|
stack2[k] = Ray(S, getRefraction(N, W, nextkr, eta, c1), kf[i], nextkr); //refraction
|
||
|
currentK -= nextkr;
|
||
|
next_top ++;
|
||
|
}
|
||
|
}else{
|
||
|
if(refl)
|
||
|
{
|
||
|
stack1[k] = Ray(S, (-(2. * c1 * N - W)), currKf, nextks); //reflection
|
||
|
currentK -= nextks;
|
||
|
next_top ++;
|
||
|
}
|
||
|
if(refr)
|
||
|
{
|
||
|
if(refl)
|
||
|
stack1[k+1] = Ray(S, getRefraction(N, W, nextkr, eta, c1), kf[i], nextkr); //refraction
|
||
|
else
|
||
|
stack1[k] = Ray(S, getRefraction(N, W, nextkr, eta, c1), kf[i], nextkr); //refraction
|
||
|
currentK -= nextkr;
|
||
|
next_top ++;
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
scolor += currentK * color;
|
||
|
}
|
||
|
}
|
||
|
else{
|
||
|
float c1 = (dot(N, W));
|
||
|
float ita = kf_air/currKf;
|
||
|
if(c1<0.)
|
||
|
{
|
||
|
c1 = -c1;
|
||
|
N = - N;
|
||
|
}
|
||
|
float c2 = (1.-ita*ita*(1.-c1*c1));
|
||
|
if(c2 >= 0.)
|
||
|
c2 = sqrt(c2);
|
||
|
else c2 = -sqrt(-c2);
|
||
|
for(int k = 0; k < max_stack; ++k)
|
||
|
if(k == next_top){
|
||
|
if(stackswap)
|
||
|
{
|
||
|
//stack2[k] = Ray(S, ((2. * c1 * N - W)), currKf, currentK*ks[i]); //reflection
|
||
|
//if(c2>=0.)
|
||
|
{
|
||
|
stack2[k] = Ray(S, normalize(ita*W + (ita*c1 - c2)*N), kf_air, currentK*kr[i]); //refraction
|
||
|
next_top ++;
|
||
|
}
|
||
|
}else{
|
||
|
//stack1[k] = Ray(S, ((2. * c1 * N - W)), currKf, currentK*ks[i]); //reflection
|
||
|
//if(c2 >= 0.)
|
||
|
{
|
||
|
stack1[k] = Ray(S, normalize(ita*W + (ita*c1 - c2)*N), kf_air, currentK*kr[i]); //refraction
|
||
|
next_top ++;
|
||
|
}
|
||
|
}
|
||
|
// next_top ++;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
else {
|
||
|
float t = -(.2+V.y)/W.y;
|
||
|
float sx = V.x + t* W.x, sz = V.z + t * W.z;
|
||
|
|
||
|
if(t >= 0. && abs(sx) < 1.5 && abs(sz) < 3.)
|
||
|
{
|
||
|
vec3 S = vec3(sx, -.2, sz);
|
||
|
vec3 realLDir=normalize(LDir - S);
|
||
|
color=(0.5+0.5*max(0.,realLDir.y)*LCol)*texture2D(uSampler[4],vec2((sx+2.)/3., (sz+1.)/6.)).xyz;
|
||
|
if(rtxoff || final&&abs(sx)<1.5 && abs(sz+.6)<3.)
|
||
|
{
|
||
|
color += groundSpecular.xyz* //specular for ground.
|
||
|
pow(max(0., dot(vec3(-realLDir.x, realLDir.y,-realLDir.z),-W)),groundSpecular.w);
|
||
|
scolor += currentK * color;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
for(int k = 0; k < max_stack; ++k)
|
||
|
if(k == next_top){
|
||
|
if(stackswap)
|
||
|
stack2[k] = Ray(S, vec3(W.x, -W.y, W.z), kf_air, currentK * 0.15); //reflection
|
||
|
else
|
||
|
stack1[k] = Ray(S, vec3(W.x, -W.y, W.z), kf_air, currentK * 0.15); //reflection
|
||
|
next_top ++;
|
||
|
break;
|
||
|
}
|
||
|
scolor += (currentK*.85)*color;
|
||
|
}
|
||
|
}
|
||
|
else{
|
||
|
if(j > 0)
|
||
|
scolor += currentK * pow(max(0.,dot(W, normalize(LDir - V))), 10.) * vec3(1.,1.,1.);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
if(++curr_ptr >= curr_top){
|
||
|
curr_top = next_top;
|
||
|
curr_ptr = 0;
|
||
|
if(next_top * 2 > max_stack)
|
||
|
final = true;
|
||
|
stackswap = !stackswap;
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
gl_FragColor=vec4(sqrt(scolor),1.);
|
||
|
}
|