文档章节

D3D中的粒子系统(5)

rise-worlds
 rise-worlds
发布于 2016/06/20 13:40
字数 756
阅读 0
收藏 0

14.3.2 例子程序:焰火系统

本例程实现了一个焰火例子系统,运行效果如图14.3所示:

火系统类定义如下:
    class cFirework : public cParticleSystem
    {
    public:
        cFirework(D3DXVECTOR3* origin, int num_particles);
        virtual void reset_particle(sParticleAttribute* attr);
        virtual void update(float time_delta);
        virtual void pre_render();
    };

构造函数需要提供一个点作为粒子系统中的原点,和系统中的粒子数,原点是火焰爆发的那个点。
    cFirework::cFirework(D3DXVECTOR3* origin, int num_particles)
    {
        m_origin        = *origin;
        m_size            = 0.9f;
        m_vb_num        = 2048;
        m_vb_offset     = 0;
        m_vb_batch_num    = 512;
        for(int i = 0; i < num_particles; i++)
            add_particle();
    }

reset_particle方法在原点位置初始化粒子系统,并在边界球内创建一个随机的速度,粒子系统中的每个例子有一个随机的颜色,我们定义粒子只能存活2秒。
    void cFirework::reset_particle(sParticleAttribute* attr)
    {
        attr->is_alive = true;
        attr->position = m_origin;
        D3DXVECTOR3 min = D3DXVECTOR3(-1.0f, -1.0f, -1.0f);
        D3DXVECTOR3 max = D3DXVECTOR3(1.0f, 1.0f, 1.0f);
        get_random_vector(&attr->velocity, &min, &max);
        // normalize to make spherical
            D3DXVec3Normalize(&attr->velocity, &attr->velocity);
        attr->velocity *= 100.0f;
        attr->color = D3DXCOLOR(get_random_float(0.0f, 1.0f),
                                get_random_float(0.0f, 1.0f),
                                get_random_float(0.0f, 1.0f),
                                1.0f);
        attr->age        = 0.0f;
        attr->life_time = 2.0f;    // lives for 2 seconds
    }

update方法更新每个粒子的位置,并在粒子超出自己的生活周期时杀死它。注意:这个系统不移除死掉的粒子,这么做是因为我们想产生一个新的火焰的时候,我们只要简单的重新设置已经存在的死了的火焰系统就可以了。这样我们不必频繁的去产生和释放粒子。
    void cFirework::update(float time_delta)
    {
        for(list<sParticleAttribute>::iterator iter = m_particles.begin(); iter != m_particles.end(); iter++)
        {
            // only update living particles
            if(iter->is_alive)
            {
                iter->position += iter->velocity * time_delta;
                iter->age       += time_delta;
                if(iter->age > iter->life_time)    // kill
                    iter->is_alive = false;
            }
        }
    }

重载pre_render以使绘制粒子时与地板颜色融合。
    void cFirework::pre_render()
    {
        cParticleSystem::pre_render();
        m_device->SetRenderState(D3DRS_DESTBLEND, D3DBLEND_ONE);
    }

执行程序:
    #include "d3dUtility.h"
    #include "camera.h"
    #include "ParticleSystem.h"
    #include <cstdlib>
    #include <ctime>
    #pragma warning(disable : 4100)
    const int WIDTH  = 640;
    const int HEIGHT = 480;
    IDirect3DDevice9*    g_device;
    cParticleSystem*    g_exploder;
    cCamera                g_camera(AIR_CRAFT);
        ////////////////////////////////////////////////////////////////////////////////////////////////////
    bool setup()
    {   
        srand((unsigned int)time(NULL));
        // create firwworlk system
            D3DXVECTOR3 origin(0.0f, 10.0f, 50.0f);
        g_exploder = new cFirework(&origin, 6000);
        g_exploder->init(g_device, "flare.bmp");
        // setup a basic scnen, the scene will be created the first time this function is called.
            draw_basic_scene(g_device, 1.0f);
        // set the projection matrix
        D3DXMATRIX proj;
        D3DXMatrixPerspectiveFovLH(&proj, D3DX_PI/4.0f, (float)WIDTH/HEIGHT, 1.0f, 1000.0f);
        g_device->SetTransform(D3DTS_PROJECTION, &proj);
        return true;
    }
        ///////////////////////////////////////////////////////////////////////////////////////////////////////
    void cleanup()
    {   
        delete g_exploder;
        // pass NULL for the first parameter to instruct cleanup
        draw_basic_scene(NULL, 0.0f);
    }
        ///////////////////////////////////////////////////////////////////////////////////////////////////////
    bool display(float time_delta)
    {
        // update the camera
        if( GetAsyncKeyState(VK_UP) & 0x8000f )
            g_camera.walk(4.0f * time_delta);
        if( GetAsyncKeyState(VK_DOWN) & 0x8000f )
            g_camera.walk(-4.0f * time_delta);
        if( GetAsyncKeyState(VK_LEFT) & 0x8000f )
            g_camera.yaw(-1.0f * time_delta);
        if( GetAsyncKeyState(VK_RIGHT) & 0x8000f )
            g_camera.yaw(1.0f * time_delta);
        if( GetAsyncKeyState('N') & 0x8000f )
            g_camera.strafe(-4.0f * time_delta);
        if( GetAsyncKeyState('M') & 0x8000f )
            g_camera.strafe(4.0f * time_delta);
        if( GetAsyncKeyState('W') & 0x8000f )
            g_camera.pitch(1.0f * time_delta);
        if( GetAsyncKeyState('S') & 0x8000f )
            g_camera.pitch(-1.0f * time_delta);   
        // update the view matrix representing the camera's new position/orientation
        D3DXMATRIX view_matrix;
        g_camera.get_view_matrix(&view_matrix);
        g_device->SetTransform(D3DTS_VIEW, &view_matrix);   
        g_exploder->update(time_delta);
        if(g_exploder->is_dead())
            g_exploder->reset();
        // render now
        g_device->Clear(0, NULL, D3DCLEAR_TARGET | D3DCLEAR_ZBUFFER, 0x00000000, 1.0f, 0);
        g_device->BeginScene();
        D3DXMATRIX identity_matrix;
        D3DXMatrixIdentity(&identity_matrix);
        g_device->SetTransform(D3DTS_WORLD, &identity_matrix);
        draw_basic_scene(g_device, 1.0f);
        // order important, render firework last.
            g_device->SetTransform(D3DTS_WORLD, &identity_matrix);
        g_exploder->render();
        g_device->EndScene();
        g_device->Present(NULL, NULL, NULL, NULL);
        return true;
    }
        ///////////////////////////////////////////////////////////////////////////////////////////////////////
    LRESULT CALLBACK wnd_proc(HWND hwnd, UINT msg, WPARAM word_param, LPARAM long_param)
    {
        switch(msg)
        {
        case WM_DESTROY:
            PostQuitMessage(0);
            break;
        case WM_KEYDOWN:
            if(word_param == VK_ESCAPE)
                DestroyWindow(hwnd);
            break;
        }
        return DefWindowProc(hwnd, msg, word_param, long_param);
    }
        ///////////////////////////////////////////////////////////////////////////////////////////////////////
    int WINAPI WinMain(HINSTANCE inst, HINSTANCE, PSTR cmd_line, int cmd_show)
    {
        if(! init_d3d(inst, WIDTH, HEIGHT, true, D3DDEVTYPE_HAL, &g_device))
        {
            MessageBox(NULL, "init_d3d() - failed.", 0, MB_OK);
            return 0;
        }
        if(! setup())
        {
            MessageBox(NULL, "Steup() - failed.", 0, MB_OK);
            return 0;
        }
        enter_msg_loop(display);
        cleanup();
        g_device->Release();
        return 0;
    }

下载源程序

本文转载自:http://www.cnblogs.com/flying_bat/archive/2008/04/04/1137675.html

rise-worlds

rise-worlds

粉丝 3
博文 1755
码字总数 0
作品 0
深圳
程序员
私信 提问
盈趣上海众多游戏技术岗位火热招募中

职位一:C/C++客户端开发工程师 工作职责: 1.负责MTK平台游戏客户端主要模块的开发 2.负责游戏客户端与服务器端的交互 3.负责客户端代码的整合及优化 职位要求: 1. 1年以上MTK平台游戏开发...

yuer150
2011/04/26
375
3
盈趣信息技术上海有限公司众多技术岗位招募中

职位一:C/C++客户端开发工程师 工作职责: 1.负责MTK平台游戏客户端主要模块的开发 2.负责游戏客户端与服务器端的交互 3.负责客户端代码的整合及优化 职位要求: 1. 1年以上MTK平台游戏开发经验...

yuer150
2011/05/02
1K
0
3D游戏引擎--Irrlicht

Irrlicht是一个3D游戏引擎。它具有高效,实时等特点,是个完全跨平台的引擎,使用D3D、OpenGL和它的自己的渲染程序。支持动态的阴影,粒子系统,角色动画,室内和室外技术以及碰撞检测等。 更多...

匿名
2008/12/06
41.6K
0
Unreal中的烟雾火焰流体模拟

版权声明:本文所有文章均为原创,原文链接:blog.uwa4d.com,如需转载请联系support@uwa4d.com https://blog.csdn.net/UWA4D/article/details/86649505 【博物纳新】是UWA旨在为开发者推荐新...

UWA
01/29
0
0
Loon 3D游戏引擎 开发心得 1

本人一直想着自己开发一个比较高质量游戏引擎出来。虽然总被别人说怎么可能事情,一个人是不可能。但我心里暗暗的笑着,怎么不可能呢!好吧,心动不如行动。借助xna框架下开发3d引擎不是很难...

单条裤
2013/03/15
468
0

没有更多内容

加载失败,请刷新页面

加载更多

安全组和云防火墙的区别

前言 熟悉云平台的朋友可能都会注意到这样一个事情:无论公有云还是私有云,创建虚拟机的时候都需要选择安全组,来对虚拟机进行安全防护;有的云平台在VPC里,还能选择防火墙,ZStack在3.6版...

ZStack社区版
31分钟前
2
0
教育性app开发的重要性和好处

在这个精通技术的世界中,流行的app主导着无聊的教育系统。当我们将技术和教育结合在一起时,它将带来当代以及强大的学习资源。因此,将教育移动app集成到您的学习过程中,并根据自己的信念把...

a429011717
31分钟前
3
0
IE6/7/8如何兼容CSS3属性

本文转载于:专业的前端网站➩IE6/7/8如何兼容CSS3属性 最近在工作中总是要求IE8兼容CSS3属性,在网上搜了搜主要是引入了一个htc文件(ie-css3.htc或者PIE.htc。个人认为这两个文件的作用差不...

前端老手
47分钟前
4
0
手把手教你ALLEGRO的约束规则的设置教程!

约束规则的设置 分三步, 定义规则(一、基本约束规则设置:1、线间距设置;2、线宽设置;3、设置过孔;4、区域约束规则设置;5、设置阻抗;6、设置走线的长度范围;7、设置等长:7.1、不过电阻的NET 等...

demyar
48分钟前
5
0
完美解决H5滚动滑动穿透方案:不使用系统滚动

网上有很多黑科技解决这个问题,都不是从根本去解决,例如通过js控制弹出时html加上position:fixed; 弹窗关闭后再去掉该样式,总觉得不太对,像是打补丁。 今天终于找到了滚动穿透的原因和完...

未来cc
53分钟前
6
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部