OpenGL ES3.0实现简单粒子火焰效果

通过粒子系统来实现火焰效果,基本思想是把一团火焰看成是由一颗颗有其生命期的粒子组成,粒子在不停的产生直至消亡从而产生升腾的火焰效果。通过生成每个粒子的坐标,每颗火焰粒子是一个矩形,而这个矩形由两个三角形构成,包含6个顶点,把坐标传入渲染管线进行GPU的渲染生成最终效果。本文章效果是修改自《OpenGL ES 3.x游戏开发》中的火焰效果。

最终的效果

1、火焰产生的基本原理

1.1、每个粒子本质上是一个矩形,通过矩形图案中的不透明区域来确定粒子的形状。这里实现效果使用的是不透明区域为圆形的矩形。



1.2、产生火焰效果的粒子的位置不是固定的,是在一定区域内随机产生的,通过控制粒子的运动方向和生命周期产生火焰的形状。


粒子生命周期中的运动区域

1.3、粒子运动过程中还需要动态改变颜色值,着色器接收渲染管线传入的起始颜色、终止颜色、总衰减因子。然后根据当前片元距离粒子纹理矩形中心点的距离、总衰减因子、片元纹理采样颜色的透明度通道值、起始/终止颜色计算出当前片元的颜色,结合矩形中的透明度展示最后的颜色。
粒子中片元颜色值的计算策略

2、具体的实现

2.1、Java层主要的构成对象
Java层总体结构
  • 2.1.1 CustomGLSurfaceView:继承自GLSurfaceView,专门负责OpenGL ES3.0的渲染和触摸事件的处理;
  • 2.1.2 SceneRender:CustomGLSurfaceView的内部类,实现GLSurfaceView.Renderer,CustomGLSurfaceView的渲染就是通过其实现的;
  • 2.1.3 ParticleSystem:火焰粒子系统的总控制类 ParticleSystem,实现了对所有粒子位置的计算以及该位置所对应的 6 个顶点坐标值的计算,同时还实现了定时更新粒子位置以及根据摄像机位置计算火焰朝向等;
  • 2.1.4 ParticleForDraw:实现粒子的绘制;
  • 2.1.5 ParticleDataConstant:常量类,封装了四个火焰效果的位置,颜色,生命周期等信息;
  • 2.1.6 MatrixState:工具类,实现矩阵变换;
  • 2.1.7 LoadUtil:工具类,用于3D模型文件加载,因为本案例没有使用到obj格式的3D模型,所以是没有实现的;
  • 2.1.8 ShaderUtil:工具类,用于加载顶点着色器和片元着色器(OpenGL ES 2.0之后已经摈弃固定管线,现今是通过着色语言实现可编程管线功能);
  • 2.1.9 TextureUtil:工具类,用于加载图片资源转化为纹理供渲染管线使用;
2.2、渲染管线着色器的构成

从下图中可以看出顶点着色器和片元着色器所处的位置。


OpenGL ES 3.0 可编程渲染管线

在这个效果实现中也得必须实现这两个这色器。


着色器

着色器是使用着色语言编写的,在GPU中实现,是可编程管线中的重要组成部分,主要分为顶点着色器和片元着色器。
  • 2.2.1 vertex.sh:顶点着色器,是一个可编程的处理单元,其功能是执行顶点的变换、光照、材质的应用与计算等顶点的相关操作,每顶点执行一次。
#version 300 es
uniform mat4 uMVPMatrix; //总变换矩阵
uniform float maxLifeSpan;//最大允许生命期
in vec4 aPosition;  //从渲染管线接收的顶点位置属性
in vec2 aTexCoor;    //从渲染管线接收的纹理坐标
out vec2 vTextureCoord;  //用于传递给片元着色器的纹理坐标

out vec4 vPosition;//用于传递给片元着色器的顶点位置属性
out float sjFactor;//用于传递给片元着色器的总衰减因子
void main()
{ //主函数
   gl_Position = uMVPMatrix * vec4(aPosition.x,aPosition.y,0.0,1); //根据总变换矩阵计算此次绘制此顶点位置
   vTextureCoord = aTexCoor;//将接收的纹理坐标传递给片元着色器
   vPosition=vec4(aPosition.x,aPosition.y,0.0,aPosition.w);//计算顶点位置属性,并将其传递给片元着色器
   sjFactor=(maxLifeSpan-aPosition.w)/maxLifeSpan;//计算总衰减因子,并将其传递给片元着色器
}
  • 2.2.2 frag.sh:片元着色器是用于处理片元值及其相关数据的可编程单元,其可以执行纹理的采样、颜色的汇总、计算雾颜色等操作,每片元执行一次。
#version 300 es
precision mediump float;//给出默认浮点精度
uniform vec4 startColor;//起始颜色
uniform vec4 endColor;//终止颜色
uniform float bj;//纹理矩形半径
uniform sampler2D sTexture;//纹理内容数据

in vec2 vTextureCoord; //接收从顶点着色器传过来的纹理坐标
in vec4 vPosition;//接收从顶点着色器传过来的片元位置属性
in float sjFactor;//接收从顶点着色器传过来的总衰减因子

out vec4 fragColor; //输出到的片元颜色
void main()
{ //主函数
    if(vPosition.w==10.0)
    {//该片元的生命期为10.0时,处于未激活状态,不绘制
        fragColor=vec4(0.0,0.0,0.0,0.0);//舍弃此片元
    }else
    {//该片元的生命期不为10.0时,处于活跃状态,绘制
        vec4 colorTL = texture(sTexture, vTextureCoord);//进行纹理采样
        vec4 colorT;//颜色变量
        float disT=distance(vPosition.xyz,vec3(0.0,0.0,0.0));//计算当前片元与中心点的距离
        float tampFactor=(1.0-disT/bj)*sjFactor;//计算片元颜色插值因子
        vec4 factor4=vec4(tampFactor,tampFactor,tampFactor,tampFactor);
        colorT=clamp(factor4,endColor,startColor);//进行颜色插值
        colorT=colorT*colorTL.a;  //结合采样出的透明度计算最终颜色
        fragColor=colorT; //将计算出来的片元颜色传给渲染管线
    }
}

3、总结

效果看起来挺简单,但是需要的知识点特别多,示例中有一些地方我理解得也还不是很充分,因为其实现过程比较繁琐,粒子的顶点需要一个个绘制,幸亏是原有的示例,只是在其基础上进行一些简化和修改,添加了一些注释,示例源码可以拿来学习一下ParticleFireProject,在项目中是不能直接使用,因为其还有很多优化点。

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

推荐阅读更多精彩内容