火山熔岩效果渲染 - three.js lava 实例

three.js中的webgl_lava例子实现了火山熔岩效果的渲染。火山熔岩效果的渲染过程比较繁琐,借助了复杂的移位纹理和高斯模糊算法。

实现原理

首先使用火山和扰动(noise)纹理,借助跟随时间变化的特殊纹理移位shader实现熔岩的基本动态效果。
随后,使用高斯模糊算法在水平和垂直方向进行闪光模糊。
最后,将原始渲染纹理和闪光效果纹理进行合成,生成最终的渲染效果。
整个渲染过程使用了多个渲染通道。

模仿webgl_lava例子的这个过程,使用C++和OpenGL ES 3.0获得了如下的渲染效果,iOS版本实现源码可以从github上获取。

lavaeffect_dynamic_small.gif

基本熔岩动态效果渲染

第一个通道要实现熔岩的基本效果的动态渲染。我们加载一个基本的3D环形几何体(torus),在three.js例子中使用了专用的着色器进行渲染,并将这个通道的渲染结果存储为纹理。

#version 300 es
precision highp float;

uniform float time;

uniform float fogDensity;
uniform vec3 fogColor;

uniform sampler2D texture1;
uniform sampler2D texture2;

in vec2 vUv;

out vec4 fragColor;

void main( void ) {
    //把纹理坐标域更改为canonical域
    vec2 position = - 1.0 + 2.0 * vUv;
    
    //使用cloud纹理作为扰动值(noise)
    vec4 noise = texture(texture1, vUv);
 
    //生成随时间朝向不同方向运动的移位纹理坐标值
    vec2 T1 = vUv + vec2( 1.5, - 1.5 ) * time * 0.02;
    vec2 T2 = vUv + vec2( - 0.5, 2.0 ) * time * 0.01;

    //使用扰动纹理对纹理坐标进行扰动处理,对不同维度采用不同的处理方式
    T1.x += noise.x * 2.0;
    T1.y += noise.y * 2.0;
    T2.x -= noise.y * 0.2;
    T2.y += noise.z * 0.2;

    //获取扰动纹理的alpha值
    float p = texture(texture1, T1 * 2.0).a;
    
    //使用一个移位纹理坐标获取火山熔岩纹理的色彩
    vec4 color = texture(texture2, T2 * 2.0);
    //对获取的熔岩色彩各部件根据扰动纹理进行程度变化,同时根据部件本身的强度进行变化
    vec4 temp = color * ( vec4( p, p, p, p ) * 2.0 ) + ( color * color - 0.1 );

    if( temp.r > 1.0 ) { temp.bg += clamp( temp.r - 2.0, 0.0, 100.0 ); }
    if( temp.g > 1.0 ) { temp.rb += temp.g - 1.0; }
    if( temp.b > 1.0 ) { temp.rg += temp.b - 1.0; }

    fragColor = temp;

    float depth = gl_FragCoord.z;
    const float LOG2 = 1.442695;
    float fogFactor = exp2( - fogDensity * fogDensity * depth * depth * LOG2 );
    fogFactor = 1.0 - clamp( fogFactor, 0.0, 1.0 );

    fragColor = mix( fragColor, vec4( fogColor, fragColor.w ), fogFactor );
}

实际上,在使用了熔岩纹理和动态纹理移位效果的shader渲染后,初始场景已经具备了熔岩的基本外观,但是在细节上,还有点欠缺,比如熔岩核心不够亮,外围黑斑的效果不够柔和。

basic_lavaeffect_2020_0330.jpg

使用高斯模糊算法实现发光效果

第二和第三通道使用第一通道的纹理作为输入,分别在水平和垂直方向执行高斯模糊算法,高斯模糊算法减弱图像的整体锐度,柔化图像是图像整体呈现模糊的感觉。注意,这两个通道基本的渲染方式为使用正交投射方式绘制四边形,并且将渲染结果也存储为纹理。

//vertex shader
#version 300 es
//卷积内核的最大数目
const float KERNEL_SIZE_FLOAT=25.0;
uniform mat4 uModelViewProjectionMatrix;
//水平或者垂直纹理坐标增量
uniform vec2 uImageIncrement;

layout(location = 0) in vec3  position;
layout(location = 1) in vec3  normal;
layout(location = 2) in vec2  uv;

out vec2 vUv;

void main() {
    //在vertex shader中先计算出要进行混合的最大偏离值的纹理坐标
    vUv = uv - ((KERNEL_SIZE_FLOAT-1.0)/2.0) * uImageIncrement;
    gl_Position = uModelViewProjectionMatrix * vec4(position, 1.0);
}
//fragment shader
#version 300 es
precision highp float;
//卷积内核的最大数目
const int KERNEL_SIZE_INT = 25;
//使用uniform方式传入高斯方程式计算的权重值
uniform float cKernel[KERNEL_SIZE_INT];
//正常渲染图像所保存的纹理
uniform sampler2D tDiffuse;
//水平或者垂直方向纹理坐标增量值
uniform vec2 uImageIncrement;
//从vertex shader中传入的经过差值的最大偏离纹理坐标
in vec2 vUv;

out vec4 fragColor;
void main() {
    vec2 imageCoord = vUv;
    vec4 sum = vec4( 0.0, 0.0, 0.0, 0.0 );
    for( int i = 0; i < KERNEL_SIZE_INT; i ++ ) {
        //使用高斯权重对附近的像素逐个混合
        sum += texture(tDiffuse, imageCoord) * cKernel[i];
        //每次混合的纹理坐标以最大偏离值逐渐增加增量值的方式获得
        imageCoord += uImageIncrement;
    }
    
    fragColor = sum;
}
    //高斯模糊算法中权重值的计算

    //此处高斯方程计算时没有除去2.0*PI,是因为下面计算权重值会被标准化。
    float gauss(float val,float sigma){
        return exp(-(val*val)/(2.0 * sigma * sigma));
    }
    
    //生成固定数目的权重值
    std::vector<float> buildKernal(float sigma){
        float sum, halfWidth;
        int kMaxKernelSize = 25, kernelSize = 2 * ceil(sigma * 3.0) + 1;
        
        if(kernelSize > kMaxKernelSize)
            kernelSize = kMaxKernelSize;
        
        halfWidth = (kernelSize - 1) * 0.5;
        
        std::vector<float> values(kernelSize);
        
        sum = 0.0;
        for(int i = 0; i < kernelSize; ++ i){
            values[i] = gauss(i - halfWidth, sigma);
            sum += values[i];
        }
        
        // 将权重值标准化
        for (int i = 0; i < kernelSize; ++ i)
            values[i] /= sum;
        
        return values;
    }

使用convolution shader渲染时,借助高斯模糊方程式,同时使用卷积滤波方式在水平和垂直方向对临近像素进行加权混合。这两个通道渲染出的纹理如下图。

lavaeffect_gauss_2times_tex_20200330.jpg

合成原始渲染纹理和高斯模糊纹理

最后一个步骤就是合成第一通道的渲染纹理,以及第三通道的高斯模糊纹理。合成方式为将对应的像素值直接相加。这个通道的基本渲染方式和上一步骤完全相同。

合成渲染后图像的效果增强十分显著,而且亮度明显加强。threejs这个例子的算法并没有执行专门的亮度算法,但是依然获得了不错的效果。

lavaeffect_glow_20200330.jpg

注:此处,FBO和对应纹理的管理是一个问题,如果要适应窗口的变化,则需要随时清理生成的用于多通道的FBO和纹理。否则,系统资源不可避免的会超载。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,423评论 6 491
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,147评论 2 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,019评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,443评论 1 283
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,535评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,798评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,941评论 3 407
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,704评论 0 266
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,152评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,494评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,629评论 1 340
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,295评论 4 329
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,901评论 3 313
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,742评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,978评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,333评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,499评论 2 348

推荐阅读更多精彩内容