本系列文章由zhmxy555(毛星云)编写,转载请注明出处。
文章链接:http://blog.csdn.net/zhmxy555/article/details/8744805
作者:毛星云(浅墨) 邮箱: happylifemxy@163.com
本篇文章中,我们将一起探讨三维游戏中粒子系统的方方面面,首先对粒子系统的基本概念特性做一个全面的认知,然后我们依旧是把粒子系统封装在一个C++类中,模拟了三维游戏中唯美的雪花飞扬的景象,让我们之前的综合三维游戏场景更加炫。依旧是放出一张本篇文章的配套程序的截图:
这个帅气的大天使为我们唯美的雪花飞扬示例程序平添了几分霸气有木有?
PS:示例程序的源代码在文章末尾提供下载
大家应该记得,我们之前也用GDI实现过雪花粒子系统,那个时候由于图形库GDI的限制,实现效果或多或少显得有些拙劣,这篇文章中,我们在DirectX的帮助下,专门用粒子系统重新实现了唯美雪花的飞扬景象,算是在为强大的粒子系统正名吧。
一、对粒子系统的基本认知
1983年,奇才Reeves.V.T在它发表的论文《Particle Systems A Technique for Modeling a Class of Fuzzy Objects》中首次提出了粒子系统的概念。从此,粒子系统就开始广泛运用于计算机中各种模糊景物的模拟。经常使用粒子系统模拟的现象有火焰、爆炸、烟、水流、火花、落叶、云、雾、雪、尘、流星尾迹或者像发光轨迹这样的抽象视觉效果等等。这些物体模型在计算机中往往很难用具体的形状、大小来描述,但是我们可以通过粒子系统的思想,去描述组成这些物体的每个元素和它的变化。
一般情况下,粒子的几何特征都十分的简单,可以采用一个像素或者是一个小的多边形来表示。需要注意的是,粒子系统的最大的缺陷是,当粒子数量达到很大的规模的时候,对运行时机器性能的要求会更加苛刻,如果机器的性能跟不上,就会显得达不到实时的运行效果,俗话说,就是粒子太多了,我们的电脑跑不动了,就会很卡。
在许多三维建模及渲染包内部就可以创建、修改粒子系统,如 3D Studio Max、Maya 以及 Blender 等。这些编辑程序使艺术家能够立即看到他们设定的特性或者规则下粒子系统的表现,另外还有一些插件能够提供增强的粒子系统效果,例如 AfterBurn 以及用于流体的 RealFlow。而2D的粒子特效软件中particleIllusion最为出色,因为他的渲染比一般的3D软件快较为平面化。Combustion 这样的多用途软件或者只能用于粒子系统的 Particle Studio 等都可以用来生成电影或者视频中的粒子系统。而目前,粒子系统技术被广泛用于大型3D游戏地制作中。首先看几张用粒子系统制作出来的效果图吧:
然后下图是DirectX SDK中自带sample一个和粒子系统相关的非常华丽的demo,推荐大家去运行一下,如果你的DirectX SDK安装在D盘,那么路径就是
D:\Program Files\Microsoft DirectXSDK (Jun 2010)\Samples\C++\Direct3D11\NBodyGravityCS11
放几张运行的截图:
粒子系统通常有三个要素:群体性、统一性和随机性。下面我们分别来简单看一下:
群体性:粒子系统是由大量的可见元素构成的。因此用粒子系统描述一团烟雾是合情合理的,但是我们所用粒子系统去描述一粒烟雾显然只有闹笑话了。
统一性:粒子系统的每个元素有具有相同的表现规律。比如,雪花粒子系统中的每一片雪花,都是白色无暇、轻盈灵动的。如果雪花粒子系统中出现了彩色的粒子,那显然就是异类了。
随机性:粒子系统中每个元素又随机表现出不同的特性。比如烟雾中每一个烟雾粒子的运动轨迹往往都是杂乱无章的,但是对于整个烟雾系统来说,这些烟雾粒子往往都有一个大体的运动方向。
二、粒子系统的基本原理
粒子通常都是一个带有纹理的四边形。我们通过这个使用了纹理映射的四边形,可以认为粒子实际上是一个很小的网格模型,只不过是纹理赋予了它特殊的外表罢了。绘制粒子就如果绘制多边形一样简单,因为一个粒子说白了就是一个可改变大小并映射了纹理的四边形罢了。
下图就是一个20个单位大小的粒子。
如果给出了粒子中心点的坐标和粒子的大小,不难计算出绘制粒子所需要的4个顶点坐标,这样往往比直接给4个顶点坐标来得直观和节省空间。
另外,很多情况下,因为一个例子是使用两个三角形组成的一个矩形来表示的,所以通常需要使粒子四边形始终面向观察者,这就用到了我们在Direct3D中的广告板(Billboard)技术,也叫公告版技术。公告版技术的基本原理在这里也提一下吧,后面有机会就专门用一次更新来讲解。公告版技术的基本原理就是在渲染一个多边形时,首先根据观察方向构造一个旋转矩阵,利用这个旋转矩阵旋转多边形让这个多边形始终是面向观察者的,如果观察方向是不断变化的,那么我们这个旋转矩阵也要不断进行调节。这样,我们始终看到的是这个多边形“最美好”的一面。这样先让多边形面向观察者,然后再渲染的技术,就是传说中的广告板(Billboard)技术。
我们知道,粒子系统都由大量的粒子构成,每个粒子都有一组属性,如位置、大小以及纹理,还比如颜色、透明度、运动速度、加速度、自旋周期,生命周期等等属性。一个粒子需要具有什么样的属性,当然是取决于具体的运用了。
另外,粒子属性的初始值常常都是随机值,而粒子的产生也常常是由位于空间中某个位置的粒子源产生的。
粒子系统在宏观和微观上都是随时间不断变化的,一套粒子系统在它生命周期的每一刻,一般都需完成以下的四步曲的工作:
1.产生新的粒子
这一步当中,我们会根据预定的要求,产生一定数目的新粒子。粒子的各项初始属性都可以用rand函数来在一定的范围内赋上随机的值。
2.更新现有粒子的属性
比如粒子有位置和移动速度,自旋速度等等属性,这就需要在每一帧当中根据原来的粒子的位置、移动速度和自旋速度重新进行计算和赋值更新。
3.删除已经消亡的粒子
这一步是可选的,具体情况具体分析,因为有些粒子系统中粒子是一直都存在的,没有消亡一说。在规定了粒子生命周期的一套粒子系统中,就需要判断每个粒子是否生命走到了尽头,如果是的话,那么它就game over,消亡了,得用相关代码把它从粒子系统中消除。
4.绘制出粒子
这步没有的话什么都不是,不显示出来叫什么粒子系统啊。人家可不管你在之前做了多少工作,算了多少东西,反正玩家是要看到最终的显示效果的。
在Direct3D 8.0以后,我们可以通过一种称为点精灵(Point Sprite)的特殊点元来描述粒子系统中的粒子。和一般点元不同的是,点精灵可以进行纹理映射并改变大小。点精灵的使用常常是伴随着SetRenderState中第一个参数取如下的几个值:
D3DRS_POINTSIZE = 154,D3DRS_POINTSIZE_MIN = 155,D3DRS_POINTSPRITEENABLE = 156,D3DRS_POINTSCALEENABLE = 157,D3DRS_POINTSCALE_A = 158,D3DRS_POINTSCALE_B = 159,D3DRS_POINTSCALE_C = 160,
另外,粒子系统中的一个重要要素是保存粒子的存储结构。我们可以用数组,如果需要动态插入和删除原始的话,就进一步使用链表用或者模板了。
需要注意的是,因为粒子系统中会有很多粒子需要不断地产生、消亡,如果在每个粒子产生时都分配内存,或者在每个粒子消亡时都释放内存,这显然会造成巨大的资源开销,非常不推荐。这里我们按链表这种方案来讲解。我们通常采用的做法是,未雨绸缪,预先为所有的粒子分配内存,并将这些粒子保存到一个链表当中。当需要产生新的粒子时,从这个链表中取出所需数量的粒子,并将它们加入到渲染链表中,而当一个粒子消亡后,重新将它们放回到原链表中,并从渲染链表中删除这些粒子。最后,在程序结束时,统一一次性释放所有粒子所占的内存空间。这就是比较科学的做法。
呼,不讲概念了,太闷了,开始准备动手做些东西吧。
三、雪花粒子系统的设计
我们之前已经提到过,粒子系统可以模拟很多的现象,比如火焰、爆炸、烟、水流、火花、落叶、云、雾、雪、尘、流星尾迹或者发光轨迹。对于现象的模拟,粒子的特性往往需要根据模拟的现象的属性来具体地设计。
对于我们今天要用粒子系统来模拟的雪花飞扬场景,有两个比较特殊的地方:
1.在雪花飞扬场景中,不需要用点精灵或者公告版技术来让粒子的四个顶点所在的面始终朝向观察者,因为雪花飞舞起来是非常优雅的,会悠扬地绕着不同的轴打转,用了公告板技术反而画蛇添足,显得不那么真实了。你见过圆圆的雪花始终面朝着你转圈的吗,见鬼了吧!
2. 在雪花飞扬场景中,可以不需要粒子的动态消亡与产生,可以让雪花粒子在一定区域内下落,如果下落到Y轴的临界区域,就把粒子的Y坐标设为预定的临界最高点的Y坐标,就像粒子都是从这个地方产生的一样,这样就会模拟出源源不断地下雪景象。其实,我们又在用惯用伎俩来欺骗玩家了。
就像SisMVG童鞋在浅墨上篇文章中评论的一样“其实世界上最大的骗子团体就是程序员”。我们只是其实是在让规定数量的雪花粒子不断地在跑堂,到了终点再让他们从起点重新开始跑,一遍一遍地,只要程序不停止运行,那么就永不止息。你以为你体验了无数的雪花从你眼前呼啸而过优雅下落的样子,其实也就是那么来来回回PARTICLE_NUMBER= 8000个而已,哈哈。而这个PARTICLE_NUMBER就是下面我们要封装雪花飞扬粒子系统的类中给粒子数量规定的宏。
好了,我们开始写这个雪花粒子系统的内容吧。
首先依旧是写出这个名为SnowParticleClass类的大体轮廓。
首先当头棒喝怒写4个宏,方便宏观调控。这四个宏分别用于表示雪花粒子数量,雪花飞扬区域的长度,雪花飞扬区域的宽度,雪花飞扬区域的高度。
- #define PARTICLE_NUMBER 100000 //雪花粒子数量,显卡不好、运行起来卡的的童鞋请取小一点。
- #define SNOW_SYSTEM_LENGTH_X 20000 //下雪区域的长度
- #define SNOW_SYSTEM_WIDTH_Z 20000 //下雪区域的宽度
- #define SNOW_SYSTEM_HEIGHT_Y 20000 //下雪区域的高度
这里的PARTICLE_NUMBER雪花粒子数量我们取的10万,那么后面我们写出来的游戏场景中就有10万个雪花粒子,当然,前提是你的显卡是糕富帅,才禁得住10万及以上的粒子数量。如果像浅墨这样的显卡明日黄花ATI Radeon HD 5730,取八万的雪花粒子,跑起来帧数就只有10帧左右了,卡的飞起。当然,你取20万的粒子数量,在下雪区域是在20000x20000x20000的区域中就是超级大暴雪了。。。建议这时候把长度和宽度调大一些,来让这20万的粒子的活动区域更大。
然后我们写出雪花粒子的FVF灵活顶点格式。顶点属性当然是顶点坐标加上纹理坐标了:
-
-
-
- struct POINTVERTEX
- {
- floatx, y, z;
- floatu,v ;
- };
- #define D3DFVF_POINTVERTEX(D3DFVF_XYZ|D3DFVF_TEX1)
接下来是雪花粒子的属性结构体,想一想现实生活中的雪花有哪些特定的属性呢?唯美的雪花,有特定的位置,会旋转,有下降速度,样子不同,嗯,好,那么我们就这样写:
-
-
-
- struct SNOWPARTICLE
- {
- floatx, y, z;
- floatRotationY;
- floatRotationX;
- floatFallSpeed;
- floatRotationSpeed;
- int TextureIndex;
- };
好,边角废料写完了,下面正式来设计这个类吧。首先来看一看要写哪些成员变量,LPDIRECT3DDEVICE9类型的设备接口指针m_pd3dDevice不能少吧,雪花粒子数组m_Snows要有吧,顶点缓存对象m_pVertexBuffer要有吧,保存不同雪花纹理样式的雪花纹理数组m_pTexture要有吧,嗯,成员变量就这些。
然后看看要有哪些成员函数,构造函数,析构函数先显式地写出来,然后粒子系统初始化函数InitSnowParticle,粒子系统更新函数UpdateSnowParticle,粒子系统渲染函数RenderSnowParticle,嗯,成员函数也就是这些了。整体来看,这里类的轮廓就是如下,即贴出SnowParticleClass.h的全部代码:
-
-
-
-
-
-
-
- #pragma once
- #include "D3DUtil.h"
- #define PARTICLE_NUMBER 20000 //雪花粒子数量,显卡不好、运行起来卡的童鞋请取小一点。
- #define SNOW_SYSTEM_LENGTH_X 20000 //下雪区域的长度
- #define SNOW_SYSTEM_WIDTH_Z 20000 //下雪区域的宽度
- #define SNOW_SYSTEM_HEIGHT_Y 20000 //下雪区域的高度
-
-
-
-
- struct POINTVERTEX
- {
- floatx, y, z;
- floatu,v ;
- };
- #define D3DFVF_POINTVERTEX(D3DFVF_XYZ|D3DFVF_TEX1)
-
-
-
-
-
- struct SNOWPARTICLE
- {
- floatx, y, z;
- floatRotationY;
- floatRotationX;
- floatFallSpeed;
- floatRotationSpeed;
- int TextureIndex;
- };
-
-
-
-
- class SnowParticleClass
- {
- private:
- LPDIRECT3DDEVICE9 m_pd3dDevice;
- SNOWPARTICLE m_Snows[PARTICLE_NUMBER];
- LPDIRECT3DVERTEXBUFFER9 m_pVertexBuffer;
- LPDIRECT3DTEXTURE9 m_pTexture[6];
-
- public:
- SnowParticleClass(LPDIRECT3DDEVICE9pd3dDevice);
- ~SnowParticleClass();
- HRESULTInitSnowParticle();
- HRESULTUpdateSnowParticle( float fElapsedTime);
- HRESULTRenderSnowParticle( );
- };
四、雪花粒子系统的实现
又到了做填空题的时候,对着上面我们写勾勒出来的SnowParticleClass类,我们有5个函数需要填上实现代码,还等什么,我们开始吧。
首先呢,构造函数:
-
-
-
- SnowParticleClass::SnowParticleClass(LPDIRECT3DDEVICE9pd3dDevice)
- {
-
- m_pd3dDevice=pd3dDevice;
- m_pVertexBuffer=NULL;
- for(inti=0; i<5; i++)
- m_pTexture[i]= NULL;
- }
接下来,粒子系统初始化函数InitSnowParticle()。首先呢,调用srand重新播种一下随机数种子。然后for循环为所有的雪花粒子赋予独一无二的各项属性值。接着,用讲烂了的顶点缓存使用五步曲的其中的三步为代表着所有雪花粒子属性的一个顶点缓存赋值,最后调用6次D3DXCreateTextureFromFile从文件加载6种不同的雪花纹理进来。这6种雪花纹理图是浅墨按照素材PS出来,分别导出的,效果图在下面,还不错,各有特点,非常漂亮:
经过上面的思考,InitSnowParticle()函数的实现代码我们就知道怎么写了:
-
-
-
-
- HRESULTSnowParticleClass::InitSnowParticle( )
- {
-
- srand(GetTickCount());
- for(inti=0; i<PARTICLE_NUMBER; i++)
- {
- m_Snows[i].x =float(rand()%SNOW_SYSTEM_LENGTH_X-SNOW_SYSTEM_LENGTH_X/2);
- m_Snows[i].z = float(rand()%SNOW_SYSTEM_WIDTH_Z-SNOW_SYSTEM_WIDTH_Z/2);
- m_Snows[i].y = float(rand()%SNOW_SYSTEM_HEIGHT_Y);
- m_Snows[i].RotationY = (rand()%100)/50.0f*D3DX_PI;
- m_Snows[i].RotationX = (rand()%100)/50.0f*D3DX_PI;
- m_Snows[i].FallSpeed = 300.0f + rand()%500;
- m_Snows[i].RotationSpeed = 5.0f + rand()%10/10.0f;
- m_Snows[i].TextureIndex= rand()%6;
- }
-
-
-
- m_pd3dDevice->CreateVertexBuffer(4*sizeof(POINTVERTEX), 0,
- D3DFVF_POINTVERTEX,D3DPOOL_MANAGED,&m_pVertexBuffer, NULL );
-
-
- POINTVERTEXvertices[] =
- {
- {-20.0f, 0.0f, 0.0f, 0.0f, 1.0f, },
- {-20.0f, 40.0f, 0.0f, 0.0f, 0.0f, },
- { 20.0f, 0.0f, 0.0f, 1.0f, 1.0f, },
- { 20.0f, 40.0f, 0.0f, 1.0f, 0.0f, }
- };
-
- VOID*pVertices;
- m_pVertexBuffer->Lock(0, sizeof(vertices), (void**)&pVertices, 0 );
-
- memcpy(pVertices, vertices, sizeof(vertices) );
-
- m_pVertexBuffer->Unlock();
-
-
- D3DXCreateTextureFromFile(m_pd3dDevice, L"GameMedia\\snow1.jpg", &m_pTexture[0] );
- D3DXCreateTextureFromFile(m_pd3dDevice, L"GameMedia\\snow2.jpg", &m_pTexture[1] );
- D3DXCreateTextureFromFile(m_pd3dDevice, L"GameMedia\\snow3.jpg", &m_pTexture[2] );
- D3DXCreateTextureFromFile(m_pd3dDevice, L"GameMedia\\snow4.jpg", &m_pTexture[3] );
- D3DXCreateTextureFromFile(m_pd3dDevice, L"GameMedia\\snow5.jpg", &m_pTexture[4] );
- D3DXCreateTextureFromFile(m_pd3dDevice, L"GameMedia\\snow6.jpg", &m_pTexture[5] );
-
- returnS_OK;
- }
接着我们来看一下粒子系统更新函数UpdateSnowParticle怎么实现。其实非常简单,就是一个for循环遍历所有的粒子,看有哪些需要更新的就可以了,对于我们雪花粒子系统,需要更新一下每个粒子的Y坐标,然后判断是否到了“地面”,然后还要改变其自旋角度。那么,代码写出来就是这样了:
-
-
-
-
- HRESULTSnowParticleClass::UpdateSnowParticle( float fElapsedTime)
- {
-
-
- for(inti=0; i<PARTICLE_NUMBER; i++)
- {
- m_Snows[i].y-= m_Snows[i].FallSpeed*fElapsedTime;
-
-
- if(m_Snows[i].y<0)
- m_Snows[i].y= SNOW_SYSTEM_WIDTH_Z;
-
- m_Snows[i].RotationY += m_Snows[i].RotationSpeed * fElapsedTime;
- m_Snows[i].RotationX += m_Snows[i].RotationSpeed * fElapsedTime;
- }
-
- returnS_OK;
- }
最后来看一下最关键的粒子系统渲染函数RenderSnowParticle怎么写。首先禁用照明,然后设置纹理状态,接着设置设置Alpha混合系数,设置背面消隐模式为不剔除,然后就开始渲染了。需要注意的是设置Alpha混合系数这里依旧是我们之前没有专门花一篇文章讲,这里理解它的功能为进行透明贴图就行了。就是我们在写GDI游戏小程序的时候一直在做的纠结事情,把图片背景的黑边去掉。好了,思路有了,写代码还会难吗:
-
-
-
-
- HRESULT SnowParticleClass::RenderSnowParticle( )
- {
-
- m_pd3dDevice->SetRenderState(D3DRS_LIGHTING, false );
-
-
- m_pd3dDevice->SetTextureStageState(0, D3DTSS_COLOROP, D3DTOP_SELECTARG1);
- m_pd3dDevice->SetTextureStageState(0, D3DTSS_COLORARG1, D3DTA_TEXTURE );
- m_pd3dDevice->SetSamplerState(0, D3DSAMP_MINFILTER, D3DTEXF_LINEAR );
- m_pd3dDevice->SetSamplerState(0, D3DSAMP_MAGFILTER, D3DTEXF_LINEAR );
-
-
- m_pd3dDevice->SetRenderState(D3DRS_ALPHABLENDENABLE,true);
- m_pd3dDevice->SetRenderState(D3DRS_SRCBLEND,D3DBLEND_ONE);
- m_pd3dDevice->SetRenderState(D3DRS_DESTBLEND,D3DBLEND_ONE);
-
-
- m_pd3dDevice->SetRenderState(D3DRS_CULLMODE, D3DCULL_NONE );
-
-
- for(inti=0; i<PARTICLE_NUMBER; i++)
- {
-
- staticD3DXMATRIX matYaw, matPitch, matTrans, matWorld;
- D3DXMatrixRotationY(&matYaw,m_Snows[i].RotationY);
- D3DXMatrixRotationX(&matPitch,m_Snows[i].RotationX);
- D3DXMatrixTranslation(&matTrans,m_Snows[i].x, m_Snows[i].y, m_Snows[i].z);
- matWorld= matYaw * matPitch * matTrans;
- m_pd3dDevice->SetTransform(D3DTS_WORLD, &matWorld);
-
-
- m_pd3dDevice->SetTexture(0, m_pTexture[m_Snows[i].TextureIndex] );
- m_pd3dDevice->SetStreamSource(0,m_pVertexBuffer, 0, sizeof(POINTVERTEX));
- m_pd3dDevice->SetFVF(D3DFVF_POINTVERTEX);
- m_pd3dDevice->DrawPrimitive(D3DPT_TRIANGLESTRIP,0, 2);
-
- }
-
-
- m_pd3dDevice->SetRenderState(D3DRS_ALPHABLENDENABLE,false);
- m_pd3dDevice->SetRenderState(D3DRS_CULLMODE, D3DCULL_CCW );
- m_pd3dDevice->SetRenderState(D3DRS_LIGHTING, true );
-
- returnS_OK;
- }
析构函数就很简单了,就是在收拾残局,看有什么COM接口要释放的:
-
-
-
- SnowParticleClass::~SnowParticleClass()
- {
- SAFE_RELEASE(m_pVertexBuffer);
-
- for(inti=0;i<3; i++)
- {
- SAFE_RELEASE(m_pTexture[i]);
- }
-
五,雪花飞扬粒子类的使用
与类打交道封装功能和我们经历的生活一样,也是一个先苦后甜的过程。
设计和实现这个类的时候或许是苦涩的,但是先苦后甜是必须的,写完这个类之后,用起来非常的方便,只用几行代码而已,一个唯美的雪花飞扬景象就加入到我们的游戏场景中了。。
也就是如下的三步:
Ⅰ.首先,定义一个SnowParticleClass类的全局指针实例:
- SnowParticleClass* g_pSnowParticles= NULL;
Ⅱ.然后,在初始化阶段拿着雪花飞扬类的指针对象SnowParticleClass到处“指”,创建并初始化粒子系统:
-
- g_pSnowParticles= new SnowParticleClass(g_pd3dDevice);
- g_pSnowParticles->InitSnowParticle();
Ⅲ.最后,就是在Render函数中依然是拿着天空类的指针对象g_pSnowParticles先指一下UpdateSnowParticle函数,更新粒子系统,然后再指一下RenderSnowParticle函数,进行渲染。
-
- g_pSnowParticles->UpdateSnowParticle(fTimeDelta);
- g_pSnowParticles->RenderSnowParticle();
另外需要注意上面更新粒子系统的UpdateSnowParticle函数我们用到了一个流逝时间参数fTimeDelta,所以我们就要把我们服役多年的消息循环中改成如下含有流逝时间的更加先进的消息循环体系,然后让Direct3D_Update和Direct3D_Render各增加一个代表流逝时间的fTimeDelta参数。
-
- MSGmsg = { 0 };
- while(msg.message != WM_QUIT )
- {
- staticFLOAT fLastTime =(float)::timeGetTime();
- staticFLOAT fCurrTime =(float)::timeGetTime();
- staticFLOAT fTimeDelta = 0.0f;
- fCurrTime = (float)::timeGetTime();
- fTimeDelta= (fCurrTime - fLastTime) / 1000.0f;
- fLastTime = fCurrTime;
-
- if(PeekMessage( &msg, 0, 0, 0, PM_REMOVE ) )
- {
- TranslateMessage(&msg );
- DispatchMessage(&msg );
- }
- else
- {
- Direct3D_Update(hwnd,fTimeDelta);
- Direct3D_Render(hwnd,fTimeDelta);
- }
- }<span style="font-family: 'Microsoft YaHei'; "> </span>
嗯,既然类都写完了,用起来就是这么简单。
六、详细注释的源代码欣赏
本篇文章配套的源代码在之前的基础上又增加了两个文件,也就是实现雪花飞扬粒子系统类的源文件和头文件。全部文件数量增加到了12个,它们的列表如下:
我们依旧只贴出核心代码main.cpp,其他的众多文件大家下源代码回去看就好了。
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
- #define SCREEN_WIDTH 932 //为窗口宽度定义的宏,以方便在此处修改窗口宽度
- #define SCREEN_HEIGHT 700 //为窗口高度定义的宏,以方便在此处修改窗口高度
- #define WINDOW_TITLE _T("【Visual C++】游戏开发笔记系列配套示例程序五十 浅墨DirectX教程十八 雪花飞扬:实现唯美的粒子系统") //为窗口标题定义的宏
-
-
-
-
-
-
- #include <d3d9.h>
- #include <d3dx9.h>
- #include <tchar.h>
- #include <time.h>
- #include "DirectInputClass.h"
- #include "CameraClass.h"
- #include "TerrainClass.h"
- #include "SkyBoxClass.h"
- #include "SnowParticleClass.h"
-
-
-
-
-
- #pragma comment(lib,"d3d9.lib")
- #pragma comment(lib,"d3dx9.lib")
- #pragma comment(lib, "dinput8.lib") // 使用DirectInput必须包含的库文件,注意这里有8
- #pragma comment(lib,"dxguid.lib")
- #pragma comment(lib, "winmm.lib")
-
-
-
-
-
-
- LPDIRECT3DDEVICE9 g_pd3dDevice = NULL;
- LPD3DXFONT g_pTextFPS =NULL;
- LPD3DXFONT g_pTextAdaperName = NULL;
- LPD3DXFONT g_pTextHelper = NULL;
- LPD3DXFONT g_pTextInfor= NULL;
- float g_FPS= 0.0f;
- wchar_t g_strFPS[50] ={0};
- wchar_t g_strAdapterName[60] ={0};
- D3DXMATRIX g_matWorld;
- LPD3DXMESH g_pMesh = NULL;
- D3DMATERIAL9* g_pMaterials= NULL;
- LPDIRECT3DTEXTURE9* g_pTextures = NULL;
- DWORD g_dwNumMtrls = 0;
- LPD3DXMESH g_cylinder = NULL;
- D3DMATERIAL9 g_MaterialCylinder;
- D3DLIGHT9 g_Light;
- DInputClass* g_pDInput = NULL;
- CameraClass* g_pCamera = NULL;
- TerrainClass* g_pTerrain = NULL;
- SkyBoxClass* g_pSkyBox=NULL;
- SnowParticleClass* g_pSnowParticles = NULL;
-
-
-
-
-
-
-
-
- LRESULT CALLBACK WndProc( HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam );
- HRESULT Direct3D_Init(HWND hwnd,HINSTANCE hInstance);
- HRESULT Objects_Init();
- void Direct3D_Render( HWND hwnd,FLOAT fTimeDelta);
- void Direct3D_Update( HWND hwnd,FLOAT fTimeDelta);
- void Direct3D_CleanUp( );
- float Get_FPS();
- void HelpText_Render(HWND hwnd);
-
-
-
-
-
- int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance,LPSTR lpCmdLine, int nShowCmd)
- {
-
-
- WNDCLASSEX wndClass={0} ;
- wndClass.cbSize = sizeof( WNDCLASSEX ) ;
- wndClass.style = CS_HREDRAW | CS_VREDRAW;
- wndClass.lpfnWndProc = WndProc;
- wndClass.cbClsExtra = 0;
- wndClass.cbWndExtra = 0;
- wndClass.hInstance = hInstance;
- wndClass.hIcon=(HICON)::LoadImage(NULL,_T("GameMedia\\icon.ico"),IMAGE_ICON,0,0,LR_DEFAULTSIZE|LR_LOADFROMFILE);
- wndClass.hCursor = LoadCursor( NULL, IDC_ARROW );
- wndClass.hbrBackground=(HBRUSH)GetStockObject(GRAY_BRUSH);
- wndClass.lpszMenuName = NULL;
- wndClass.lpszClassName = _T("ForTheDreamOfGameDevelop");
-
- if( !RegisterClassEx( &wndClass ) )
- return -1;
-
- HWND hwnd = CreateWindow( _T("ForTheDreamOfGameDevelop"),WINDOW_TITLE,
- WS_OVERLAPPEDWINDOW, CW_USEDEFAULT, CW_USEDEFAULT, SCREEN_WIDTH,
- SCREEN_HEIGHT, NULL, NULL, hInstance, NULL );
-
-
-
- if (!(S_OK==Direct3D_Init (hwnd,hInstance)))
- {
- MessageBox(hwnd, _T("Direct3D初始化失败~!"), _T("浅墨的消息窗口"), 0);
- }
- PlaySound(L"GameMedia\\NightElf1.wav", NULL, SND_FILENAME | SND_ASYNC|SND_LOOP);
-
-
-
- MoveWindow(hwnd,200,10,SCREEN_WIDTH,SCREEN_HEIGHT,true);
- ShowWindow( hwnd, nShowCmd );
- UpdateWindow(hwnd);
-
-
- g_pDInput = new DInputClass();
- g_pDInput->Init(hwnd,hInstance,DISCL_FOREGROUND | DISCL_NONEXCLUSIVE,DISCL_FOREGROUND | DISCL_NONEXCLUSIVE);
-
-
- MSG msg = { 0 };
- while( msg.message != WM_QUIT )
- {
- static FLOAT fLastTime = (float)::timeGetTime();
- static FLOAT fCurrTime = (float)::timeGetTime();
- static FLOAT fTimeDelta = 0.0f;
- fCurrTime = (float)::timeGetTime();
- fTimeDelta = (fCurrTime - fLastTime) / 1000.0f;
- fLastTime = fCurrTime;
-
- if( PeekMessage( &msg, 0, 0, 0, PM_REMOVE ) )
- {
- TranslateMessage( &msg );
- DispatchMessage( &msg );
- }
- else
- {
- Direct3D_Update(hwnd,fTimeDelta);
- Direct3D_Render(hwnd,fTimeDelta);
- }
- }
-
- UnregisterClass(_T("ForTheDreamOfGameDevelop"), wndClass.hInstance);
- return 0;
- }
-
-
-
-
-
-
-
- LRESULT CALLBACK WndProc( HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam )
- {
- switch( message )
- {
- case WM_PAINT:
- Direct3D_Render(hwnd,0.0f);
- ValidateRect(hwnd, NULL);
- break;
-
- case WM_KEYDOWN:
- if (wParam == VK_ESCAPE)
- DestroyWindow(hwnd);
- break;
- case WM_DESTROY:
- Direct3D_CleanUp();
- PostQuitMessage( 0 );
- break;
-
- default:
- return DefWindowProc( hwnd, message, wParam, lParam );
- }
-
- return 0;
- }
-
-
-
-
-
-
-
-
-
-
-
-
- HRESULT Direct3D_Init(HWND hwnd,HINSTANCE hInstance)
- {
-
-
-
-
- LPDIRECT3D9 pD3D = NULL;
- if( NULL == ( pD3D = Direct3DCreate9( D3D_SDK_VERSION ) ) )
- return E_FAIL;
-
-
-
-
- D3DCAPS9 caps; int vp = 0;
- if( FAILED( pD3D->GetDeviceCaps( D3DADAPTER_DEFAULT, D3DDEVTYPE_HAL, &caps ) ) )
- {
- return E_FAIL;
- }
- if( caps.DevCaps & D3DDEVCAPS_HWTRANSFORMANDLIGHT )
- vp = D3DCREATE_HARDWARE_VERTEXPROCESSING;
- else
- vp = D3DCREATE_SOFTWARE_VERTEXPROCESSING;
-
-
-
-
- D3DPRESENT_PARAMETERS d3dpp;
- ZeroMemory(&d3dpp, sizeof(d3dpp));
- d3dpp.BackBufferWidth = SCREEN_WIDTH;
- d3dpp.BackBufferHeight = SCREEN_HEIGHT;
- d3dpp.BackBufferFormat = D3DFMT_A8R8G8B8;
- d3dpp.BackBufferCount = 2;
- d3dpp.MultiSampleType = D3DMULTISAMPLE_NONE;
- d3dpp.MultiSampleQuality = 0;
- d3dpp.SwapEffect = D3DSWAPEFFECT_DISCARD;
- d3dpp.hDeviceWindow = hwnd;
- d3dpp.Windowed = true;
- d3dpp.EnableAutoDepthStencil = true;
- d3dpp.AutoDepthStencilFormat = D3DFMT_D24S8;
- d3dpp.Flags = 0;
- d3dpp.FullScreen_RefreshRateInHz = 0;
- d3dpp.PresentationInterval = D3DPRESENT_INTERVAL_IMMEDIATE;
-
-
-
-
- if(FAILED(pD3D->CreateDevice(D3DADAPTER_DEFAULT, D3DDEVTYPE_HAL,
- hwnd, vp, &d3dpp, &g_pd3dDevice)))
- return E_FAIL;
-
-
-
- wchar_t TempName[60]=L"当前显卡型号:";
- D3DADAPTER_IDENTIFIER9 Adapter;
- pD3D->GetAdapterIdentifier(0,0,&Adapter);
- int len = MultiByteToWideChar(CP_ACP,0, Adapter.Description, -1, NULL, 0);
- MultiByteToWideChar(CP_ACP, 0, Adapter.Description, -1, g_strAdapterName, len);
- wcscat_s(TempName,g_strAdapterName);
- wcscpy_s(g_strAdapterName,TempName);
-
- if(!(S_OK==Objects_Init())) return E_FAIL;
-
- SAFE_RELEASE(pD3D)
-
-
-
- return S_OK;
- }
-
-
- HRESULT Objects_Init()
- {
-
- D3DXCreateFont(g_pd3dDevice, 36, 0, 0, 1000, false, DEFAULT_CHARSET,
- OUT_DEFAULT_PRECIS, DEFAULT_QUALITY, 0, _T("Calibri"), &g_pTextFPS);
- D3DXCreateFont(g_pd3dDevice, 20, 0, 1000, 0, false, DEFAULT_CHARSET,
- OUT_DEFAULT_PRECIS, DEFAULT_QUALITY, 0, L"华文中宋", &g_pTextAdaperName);
- D3DXCreateFont(g_pd3dDevice, 23, 0, 1000, 0, false, DEFAULT_CHARSET,
- OUT_DEFAULT_PRECIS, DEFAULT_QUALITY, 0, L"微软雅黑", &g_pTextHelper);
- D3DXCreateFont(g_pd3dDevice, 26, 0, 1000, 0, false, DEFAULT_CHARSET,
- OUT_DEFAULT_PRECIS, DEFAULT_QUALITY, 0, L"黑体", &g_pTextInfor);
-
-
-
-
- LPD3DXBUFFER pAdjBuffer = NULL;
- LPD3DXBUFFER pMtrlBuffer = NULL;
-
- D3DXLoadMeshFromX(L"angle.X", D3DXMESH_MANAGED, g_pd3dDevice,
- &pAdjBuffer, &pMtrlBuffer, NULL, &g_dwNumMtrls, &g_pMesh);
-
- D3DXMATERIAL *pMtrls = (D3DXMATERIAL*)pMtrlBuffer->GetBufferPointer();
- g_pMaterials = new D3DMATERIAL9[g_dwNumMtrls];
- g_pTextures = new LPDIRECT3DTEXTURE9[g_dwNumMtrls];
- for (DWORD i=0; i<g_dwNumMtrls; i++)
- {
-
- g_pMaterials[i] = pMtrls[i].MatD3D;
- g_pMaterials[i].Ambient = g_pMaterials[i].Diffuse;
-
-
- g_pTextures[i] = NULL;
- D3DXCreateTextureFromFileA(g_pd3dDevice, pMtrls[i].pTextureFilename, &g_pTextures[i]);
- }
- SAFE_RELEASE(pAdjBuffer)
- SAFE_RELEASE(pMtrlBuffer)
-
-
-
-
- D3DXCreateCylinder(g_pd3dDevice, 280.0f, 10.0f, 3000.0f, 60, 60, &g_cylinder, 0);
- g_MaterialCylinder.Ambient = D3DXCOLOR(1.0f, 0.0f, 0.0f, 1.0f);
- g_MaterialCylinder.Diffuse = D3DXCOLOR(1.0f, 0.0f, 0.0f, 1.0f);
- g_MaterialCylinder.Specular = D3DXCOLOR(0.5f, 0.0f, 0.3f, 0.3f);
- g_MaterialCylinder.Emissive = D3DXCOLOR(0.0f, 0.0f, 0.0f, 1.0f);
-
-
- ::ZeroMemory(&g_Light, sizeof(g_Light));
- g_Light.Type = D3DLIGHT_DIRECTIONAL;
- g_Light.Ambient = D3DXCOLOR(0.7f, 0.7f, 0.7f, 1.0f);
- g_Light.Diffuse = D3DXCOLOR(1.0f, 1.0f, 1.0f, 1.0f);
- g_Light.Specular = D3DXCOLOR(0.9f, 0.9f, 0.9f, 1.0f);
- g_Light.Direction = D3DXVECTOR3(1.0f, 1.0f, 1.0f);
- g_pd3dDevice->SetLight(0, &g_Light);
- g_pd3dDevice->LightEnable(0, true);
- g_pd3dDevice->SetRenderState(D3DRS_NORMALIZENORMALS, true);
- g_pd3dDevice->SetRenderState(D3DRS_SPECULARENABLE, true);
-
-
- g_pCamera = new CameraClass(g_pd3dDevice);
- g_pCamera->SetCameraPosition(&D3DXVECTOR3(0.0f, 1400.0f, -1800.0f));
- g_pCamera->SetTargetPosition(&D3DXVECTOR3(0.0f, 1200.0f, 0.0f));
- g_pCamera->SetViewMatrix();
- g_pCamera->SetProjMatrix();
-
-
- g_pTerrain = new TerrainClass(g_pd3dDevice);
- g_pTerrain->LoadTerrainFromFile(L"GameMedia\\heighmap.raw", L"GameMedia\\terrainstone.jpg");
- g_pTerrain->InitTerrain(200, 200, 60.0f, 8.0f);
-
-
- g_pSkyBox = new SkyBoxClass( g_pd3dDevice );
- g_pSkyBox->LoadSkyTextureFromFile(L"GameMedia\\TropicalSunnyDayFront2048.png",L"GameMedia\\TropicalSunnyDayBack2048.png",L"GameMedia\\TropicalSunnyDayRight2048.png",L"GameMedia\\TropicalSunnyDayLeft2048.png", L"GameMedia\\TropicalSunnyDayUp2048.png");
- g_pSkyBox->InitSkyBox(50000);
-
-
- g_pSnowParticles = new SnowParticleClass(g_pd3dDevice);
- g_pSnowParticles->InitSnowParticle();
-
-
- return S_OK;
- }
-
- void Direct3D_Update( HWND hwnd,FLOAT fTimeDelta)
- {
-
- g_pDInput->GetInput();
-
-
- if (g_pDInput->IsKeyDown(DIK_A)) g_pCamera->MoveAlongRightVec(-3.0f);
- if (g_pDInput->IsKeyDown(DIK_D)) g_pCamera->MoveAlongRightVec( 3.0f);
- if (g_pDInput->IsKeyDown(DIK_W)) g_pCamera->MoveAlongLookVec( 3.0f);
- if (g_pDInput->IsKeyDown(DIK_S)) g_pCamera->MoveAlongLookVec(-3.0f);
- if (g_pDInput->IsKeyDown(DIK_R)) g_pCamera->MoveAlongUpVec( 3.0f);
- if (g_pDInput->IsKeyDown(DIK_F)) g_pCamera->MoveAlongUpVec(-3.0f);
-
-
- if (g_pDInput->IsKeyDown(DIK_LEFT)) g_pCamera->RotationUpVec(-0.003f);
- if (g_pDInput->IsKeyDown(DIK_RIGHT)) g_pCamera->RotationUpVec( 0.003f);
- if (g_pDInput->IsKeyDown(DIK_UP)) g_pCamera->RotationRightVec(-0.003f);
- if (g_pDInput->IsKeyDown(DIK_DOWN)) g_pCamera->RotationRightVec( 0.003f);
- if (g_pDInput->IsKeyDown(DIK_Q)) g_pCamera->RotationLookVec(0.001f);
- if (g_pDInput->IsKeyDown(DIK_E)) g_pCamera->RotationLookVec( -0.001f);
-
-
- g_pCamera->RotationUpVec(g_pDInput->MouseDX()* 0.001f);
- g_pCamera->RotationRightVec(g_pDInput->MouseDY() * 0.001f);
-
-
- static FLOAT fPosZ=0.0f;
- fPosZ += g_pDInput->MouseDZ()*0.03f;
-
-
- D3DXMATRIX matView;
- g_pCamera->CalculateViewMatrix(&matView);
- g_pd3dDevice->SetTransform(D3DTS_VIEW, &matView);
-
-
- D3DXMatrixTranslation(&g_matWorld, 0.0f, 0.0f, fPosZ);
-
-
-
-
- POINT lt,rb;
- RECT rect;
- GetClientRect(hwnd,&rect);
-
- lt.x = rect.left;
- lt.y = rect.top;
-
- rb.x = rect.right;
- rb.y = rect.bottom;
-
- ClientToScreen(hwnd,<);
- ClientToScreen(hwnd,&rb);
-
- rect.left = lt.x;
- rect.top = lt.y;
- rect.right = rb.x;
- rect.bottom = rb.y;
-
- ClipCursor(&rect);
-
- ShowCursor(false);
-
- }
-
-
-
-
-
-
-
-
-
-
-
-
-
-
- void Direct3D_Render(HWND hwnd,FLOAT fTimeDelta)
- {
-
-
-
- g_pd3dDevice->Clear(0, NULL, D3DCLEAR_TARGET|D3DCLEAR_ZBUFFER|D3DCLEAR_STENCIL, D3DCOLOR_XRGB(100, 255, 255), 1.0f, 0);
-
-
-
-
- g_pd3dDevice->BeginScene();
-
-
-
-
-
-
-
-
- D3DXMATRIX mScal,mRot2,mTrans,mFinal;
- D3DXMatrixTranslation(&mTrans,50.0f,1200.0f,0.0f);
- D3DXMatrixScaling(&mScal,3.0f,3.0f,3.0f);
- mFinal=mScal*mTrans*g_matWorld;
- g_pd3dDevice->SetTransform(D3DTS_WORLD, &mFinal);
-
- for (DWORD i = 0; i < g_dwNumMtrls; i++)
- {
- g_pd3dDevice->SetMaterial(&g_pMaterials[i]);
- g_pd3dDevice->SetTexture(0, g_pTextures[i]);
- g_pMesh->DrawSubset(i);
- }
-
-
-
- D3DXMATRIX TransMatrix, RotMatrix, FinalMatrix;
- D3DXMatrixRotationX(&RotMatrix, -D3DX_PI * 0.5f);
- g_pd3dDevice->SetLight(0, &g_Light);
- g_pd3dDevice->SetMaterial(&g_MaterialCylinder);
- g_pd3dDevice->SetTexture(0, NULL);
- for(int i = 0; i < 4; i++)
- {
- D3DXMatrixTranslation(&TransMatrix, -300.0f, 0.0f, -350.0f + (i * 500.0f));
- FinalMatrix = RotMatrix * TransMatrix ;
- g_pd3dDevice->SetTransform(D3DTS_WORLD, &FinalMatrix);
- g_cylinder->DrawSubset(0);
-
- D3DXMatrixTranslation(&TransMatrix, 300.0f, 0.0f, -350.0f + (i * 500.0f));
- FinalMatrix = RotMatrix * TransMatrix ;
- g_pd3dDevice->SetTransform(D3DTS_WORLD, &FinalMatrix);
- g_cylinder->DrawSubset(0);
- }
-
-
-
- g_pTerrain->RenderTerrain(&g_matWorld, false);
-
-
- D3DXMATRIX matSky,matTransSky,matRotSky;
- D3DXMatrixTranslation(&matTransSky,0.0f,-12000.0f,0.0f);
- D3DXMatrixRotationY(&matRotSky, -0.000005f*timeGetTime());
- matSky=matTransSky*matRotSky;
- g_pSkyBox->RenderSkyBox(&matSky, false);
-
-
- g_pSnowParticles->UpdateSnowParticle(fTimeDelta);
- g_pSnowParticles->RenderSnowParticle();
-
-
- HelpText_Render(hwnd);
-
-
-
-
-
- g_pd3dDevice->EndScene();
-
-
-
- g_pd3dDevice->Present(NULL, NULL, NULL, NULL);
-
- }
-
-
- void HelpText_Render(HWND hwnd)
- {
-
- RECT formatRect;
- GetClientRect(hwnd, &formatRect);
-
-
- formatRect.top = 5;
- int charCount = swprintf_s(g_strFPS, 20, _T("FPS:%0.3f"), Get_FPS() );
- g_pTextFPS->DrawText(NULL, g_strFPS, charCount , &formatRect, DT_TOP | DT_RIGHT, D3DCOLOR_RGBA(0,239,136,255));
-
-
- g_pTextAdaperName->DrawText(NULL,g_strAdapterName, -1, &formatRect,
- DT_TOP | DT_LEFT, D3DXCOLOR(1.0f, 0.5f, 0.0f, 1.0f));
-
-
- formatRect.left = 0,formatRect.top = 380;
- g_pTextInfor->DrawText(NULL, L"控制说明:", -1, &formatRect,
- DT_SINGLELINE | DT_NOCLIP | DT_LEFT, D3DCOLOR_RGBA(235,123,230,255));
- formatRect.top += 35;
- g_pTextHelper->DrawText(NULL, L" W:向前飞翔 S:向后飞翔 ", -1, &formatRect,
- DT_SINGLELINE | DT_NOCLIP | DT_LEFT, D3DCOLOR_RGBA(255,200,0,255));
- formatRect.top += 25;
- g_pTextHelper->DrawText(NULL, L" A:向左飞翔 D:向右飞翔", -1, &formatRect,
- DT_SINGLELINE | DT_NOCLIP | DT_LEFT, D3DCOLOR_RGBA(255,200,0,255));
- formatRect.top += 25;
- g_pTextHelper->DrawText(NULL, L" R:垂直向上飞翔 F:垂直向下飞翔", -1, &formatRect,
- DT_SINGLELINE | DT_NOCLIP | DT_LEFT, D3DCOLOR_RGBA(255,200,0,255));
- formatRect.top += 25;
- g_pTextHelper->DrawText(NULL, L" Q:向左倾斜 E:向右倾斜", -1, &formatRect,
- DT_SINGLELINE | DT_NOCLIP | DT_LEFT, D3DCOLOR_RGBA(255,200,0,255));
- formatRect.top += 25;
- g_pTextHelper->DrawText(NULL, L" 上、下、左、右方向键、鼠标移动:视角变化 ", -1, &formatRect,
- DT_SINGLELINE | DT_NOCLIP | DT_LEFT, D3DCOLOR_RGBA(255,200,0,255));
- formatRect.top += 25;
- g_pTextHelper->DrawText(NULL, L" 鼠标滚轮:人物模型Y轴方向移动", -1, &formatRect,
- DT_SINGLELINE | DT_NOCLIP | DT_LEFT, D3DCOLOR_RGBA(255,200,0,255));
- formatRect.top += 25;
- g_pTextHelper->DrawText(NULL, L" ESC键 : 退出程序", -1, &formatRect,
- DT_SINGLELINE | DT_NOCLIP | DT_LEFT, D3DCOLOR_RGBA(255,200,0,255));
- }
-
-
-
-
-
- float Get_FPS()
- {
-
-
- static float fps = 0;
- static int frameCount = 0;
- static float currentTime =0.0f;
- static float lastTime = 0.0f;
-
- frameCount++;
- currentTime = timeGetTime()*0.001f;
-
-
- if(currentTime - lastTime > 1.0f)
- {
- fps = (float)frameCount /(currentTime - lastTime);
- lastTime = currentTime;
- frameCount = 0;
- }
-
- return fps;
- }
-
-
-
-
-
-
-
- void Direct3D_CleanUp()
- {
-
-
- for (DWORD i = 0; i<g_dwNumMtrls; i++)
- SAFE_RELEASE(g_pTextures[i]);
- SAFE_DELETE(g_pTextures);
- SAFE_DELETE(g_pMaterials);
- SAFE_DELETE(g_pDInput);
- SAFE_RELEASE(g_cylinder);
- SAFE_RELEASE(g_pMesh);
- SAFE_RELEASE(g_pd3dDevice);
- SAFE_RELEASE(g_pTextAdaperName)
- SAFE_RELEASE(g_pTextHelper)
- SAFE_RELEASE(g_pTextInfor)
- SAFE_RELEASE(g_pTextFPS)
- SAFE_RELEASE(g_pd3dDevice)
- }
学到目前为止,我们用学的知识写出来的游戏场景程序已经比较炫了,一些运行截图如下:
我们操纵着摄像机在雪中飞行,那种雪花铺面而来的感觉非常地舒服。
仰望天空,任雪花飘落到“身上”:
下面这张图可以明显看到雪花被我们规定了活动区域:
冒着显卡被烧的危险拍了一张10万粒子数量时的“暴雪”画面,大家看右上角的帧数,只有可怜的8帧了。。。。
扑,一朵雪花砸脸上了。。。。