zoukankan      html  css  js  c++  java
  • water Caustics

      水下环境的模拟在现阶段的计算能力下,Jos Stam(https://www.opengl.org/archives/resources/code/samples/mjktips/caustics/) 提出的方法是预先计算多张caustics图,根据时间选取不同的图片,然后与水下模型进行混合。在underWater.c中,

    glEnable(GL_BLEND); 

    glBlendFunc(GL_ZERO, GL_SRC_COLOR);

    上面这两行代表绘制时与现有模型进行颜色混合。

    GLfloat sPlane[4] = { 0.05, 0.03, 0.0, 0.0 };

    GLfloat tPlane[4] = { 0.0, 0.03, 0.05, 0.0 };

    glTexGeni(GL_S, GL_TEXTURE_GEN_MODE, GL_OBJECT_LINEAR);

    glTexGeni(GL_T, GL_TEXTURE_GEN_MODE, GL_OBJECT_LINEAR);

    glTexGenfv(GL_S, GL_OBJECT_PLANE, sPlane);

    glTexGenfv(GL_T, GL_OBJECT_PLANE, tPlane);

    glEnable(GL_TEXTURE_GEN_S);

    glEnable(GL_TEXTURE_GEN_T);

    glEnable(GL_TEXTURE_2D);

    代表着caustics纹理坐标的计算,可以看到sPlane和tPlane的Y值并不为0,首先我们设置Y0时cube的效果,这时候cube的竖直面没有ripple效果,这是因为纹理坐标在Y值没有偏移时,X,Z一致值取得的caustics纹理坐标完全一样。左图是Y值为0的情况,右图是Y值不为0的情况。

                                                                      

      在GpuGem中,文章用一种简单的构造折射向量的方式来计算纹理的映射,用来映射的纹理就是 sun map,方法分为三步:

    1.构造入射向量。

    2.根据波当前处理的顶点位置和顶点法线来计算折射向量。

    3.根据折射向量来计算映射的纹理坐标。

    在计算光照折射效果之前,需要利用波公式来构建波,这里所用的wave function 如下:

    下面就是该函数的实现部分:

    float func(float x,float z)
    {
    float y=0;
    
    float factor=1;
    float d=sqrt(x*x+z*z);
    d=d/40;
    if (d>1.5) d=1.5;
    if (d<0) d=0;
    for (int i=0;i<octaves;i++)
        {
        y-=    (factor)*VTXSIZE*d*cos(((float)timer*SPEED)+(1/factor)*x*z*WAVESIZE)+
                (factor)*VTXSIZE*d*cos(((float)timer*SPEED)+(1/factor)*(x*z)*WAVESIZE) 
                ;
        //y-=factor*VTXSIZE*d*noise((float)(x*(1/factor))/5,SPEED*timer,(float)z*(1/factor)/5);
        factor=factor/1.7;        
        }
    
    return y;
    }

    而在波纹理的构造过程中,主要使用下面的公式进行纹理坐标的计算。

    double plane::testline(point r,point vec,point &pres)
    // return value is distance along the line to the collision point
    {
    // a*x +b*y+c*z+d =0为平面构造公式,r为波的顶点坐标,vec为顶点的法线,该方法主要
    //是求pres,也就是平面与顶点和法线构造的直线相交点。
    double fres;
    double t;
    
    //fres = |vec|*|n|cos(Q) 
    fres=vec*n;
    pres=r;
    if ((fres)!=0)
        {
    // r到平面的距离公式= |a*r.x + b*r.y+c*r.z +d|/(sqrt(a*a+b*b+c*c)) = |n*r+d|
    // cos(Q) = (nr+d)/t = fres/(|vec|*|n|)
        t=-(n*r + d)/fres;
    //pres相交点。
        pres=r+vec*t;
        }
    // hack... arreglar
    else t=0;
    return t;
    }

     最后计算的是折射光线的效果,原始的snell’s law计算方法不太适用于计算机计算,Foley等在1996提出了下面一个便于计算的公式。

    其中n代表面的法线方向,E为入射向量, h 1/ h 2  是折射率,计算的函数如下。

    void sample_caustic(float xi,float zi,float &u,float &v)
    // here's the main stuff for the caustic
    {
    point p(xi,0,zi);p.y=func(xi,zi);
    point q(xi+QUADSIZE,0,zi);q.y=func(xi+1,zi);
    point r(xi,0,zi+QUADSIZE);r.y=func(xi,zi+1);
            
    point e1=q-p; point e2=r-p;    point n=e1^e2;    // the normal above the sampling point 
    u=0; v=0;
    for (int i=0;i<numrays;i++)
        {
        float alpha=6.28*(float)i/numrays;
        float rad=(float)i/numrays;
        float xf=rad*cos(alpha)/2.0;
        float zf=rad*sin(alpha)/2.0;
    
        //构造入射向量
        point incident(xf,1,zf);
    
        float c=incident*n;
        float sq=1+(IOR*IOR*(c*c-1));
        if (sq>0) sq=sqrt(sq); else sq=0;
    
        //折射向量的计算。
        point transmitted=incident*IOR + n*(IOR*c-sq);
        transmitted.normalize();
    
        u+=transmitted.x;
        v+=transmitted.z;
        }
    u=4*(u/numrays) + 0.5;
    v=4*(v/numrays) + 0.5;    
    }

      最后比较几张wave function 中octaves参数从1到3的情况。

  • 相关阅读:
    浅析NetFilter和iptables
    关于skb_header_pointer函数
    Linux kernel 绝对路径之d_path篇
    几个内核函数:filp_open、filp_read、IS_ERR、ERR_PTR、PTR_ERR
    一文读懂数字签名
    Nginx配置文件nginx.conf中文详解(总结)
    nginx 重写 rewrite 基础及实例
    最完美解决Nginx部署ThinkPHP项目的办法
    nginx中的try_files指令解释
    js电话号码正则校验--座机和手机号
  • 原文地址:https://www.cnblogs.com/VanHu/p/4901558.html
Copyright © 2011-2022 走看看