• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            天行健 君子當自強而不息

            D3D中的粒子系統(2)

            新建網頁 1

            14.2粒子系統的組成

            粒子系統是粒子的集合,用來保存和顯示這些粒子。粒子系統維護所有粒子的全部屬性,影響系統中的所有粒子:粒子的尺寸,起始的位置及應用在粒子上的紋理等。粒子系統的方法負責更新、顯示、殺死和創建粒子。

            雖然不同的具體(與抽象是相對的)粒子系統有不同的行為,我們歸納并找到一些所有的粒子系統共有的基本屬性,我們把這些公共的屬性放到一個抽象的cParticleSystem基類,它是我們所有的具體粒子系統的父類,現在讓我們看一下cParticleSystem類:

                class cParticleSystem
                {
               
            protected:
                    IDirect3DDevice9*            m_device;
                    D3DXVECTOR3                    m_origin;
                    cBoundingBox                m_bounding_box;
                    
            float                        m_emit_rate;        // rate new particles are added to system
               
                float                        m_size;                // size of particles
               
                    IDirect3DTexture9*            m_texture;
                    IDirect3DVertexBuffer9*     m_vertex_buffer;
                    list<sParticleAttribute>    m_particles;
                    
            int                            m_max_particles;    // max allowed particles system can have
               
                    // following three data elements used for rendering the particle system efficiently
               

                    DWORD    m_vb_num;        
            // particle number in vertex buffer
               
                DWORD    m_vb_offset;    // offset in vertex buffer to lock
               
                DWORD    m_vb_batch_num;    // number of vertices to lock starting at m_vb_offset
               

               
            public:
                    cParticleSystem();
                    
            virtual ~cParticleSystem();
               
                    
            virtual bool init(IDirect3DDevice9*    device, const char* texture_filename);
                    
            virtual void reset();
               
                    
            // sometimes we don't want to free the memory of a dead particle, but rather respawn it instead.
               
                virtual void reset_particle(sParticleAttribute* particl_attr) = 0;
                    
            virtual void add_particle();
               
                    
            virtual void update(float time_delta) = 0;
               
                    
            virtual void pre_render();
                    
            virtual void render();
                    
            virtual void post_render();
               
                    
            bool is_empty();
                    
            bool is_dead();
               
               
            protected:
                    
            virtual void remove_dead_particles();
                };

            一些數據成員:

            ·        m_origin粒子系統的原點, 這是粒子系統產生時的位置。

            ·        m_bounding_box創建粒子系統使用的邊界盒,用于限制粒子的活動范圍。例如,假如我們讓雪系統只落在一個圍繞高山的峰頂的體積內,我們會定義一個包括這個體積的邊界盒, 出界的粒子將會被殺死。

            ·        m_emit_rate新增加到系統中的粒子的速度。通常的標準是每秒。

            ·        m_size系統中所有粒子的尺寸。

            ·        m_particles系統中粒子屬性的一個列表。 我們用這個列表創建,釋放及更新粒子。 當我們準備畫粒子時, 我們COPY列表節點的一部分到頂點緩存并畫粒子,同時我們COPY另外一批粒子,然后重復這一過程直到繪制完所有粒子。

            ·        m_max_particles在給定的時間內,系統中允許的粒子最大數。例如,如果創建粒子的速度比釋放快的話, 隨著時間的增長粒子的數量將會是巨大的,這個成員將避免出現這樣的問題。

            ·        m_vb_num在給定的時間內頂點緩存中能夠保存的粒子的數量,這個值與實際的粒子系統中的粒子數量無關。

            注意:m_vb_offsetm_vb_batch_num數據成員在渲染粒子系統時使用,我們在稍后討論。

            方法:

            cParticleSystem/ ~cParticleSystem用來初始化默認值和用來釋放設備接口 (vertex buffer, texture)

                cParticleSystem::cParticleSystem()
                {
                    m_device        = NULL;
                    m_vertex_buffer = NULL;
                    m_texture        = NULL;
                }
               
                cParticleSystem::~cParticleSystem()
                {
                    safe_release<IDirect3DVertexBuffer9*>(m_vertex_buffer);
                    safe_release<IDirect3DTexture9*>(m_texture);
                }

            init這個方法做與設備無關的初始化工作,比如創建用來保存點精靈的頂點緩存或創建紋理。
                bool cParticleSystem::init(IDirect3DDevice9* device, const char* texture_filename)
                {
                    
            // Vertex buffer's number does not equal the number of particles in our system.  
                    // We use the vertex buffer to draw a portion of our particles at a time.  
                    // The arbitrary number we choose for the vertex buffer is specified by the m_vb_num variable.
               

                    m_device = device;
               
                    HRESULT hr;
               
                    hr = device->CreateVertexBuffer(
                            m_vb_num * 
            sizeof(sParticle),
                            D3DUSAGE_DYNAMIC | D3DUSAGE_POINTS | D3DUSAGE_WRITEONLY,
                            PARTICLE_FVF,
                            D3DPOOL_DEFAULT,    
            // D3DPOOL_MANAGED can't be used with D3DUSAGE_DYNAMIC
               
                        &m_vertex_buffer,
                            NULL);
               
                    
            if(FAILED(hr))
                    {
                        MessageBox(NULL, "CreateVertexBuffer() - FAILED", "ParticleSystem", MB_OK);
                        
            return false;
                    }
               
                    hr = D3DXCreateTextureFromFile(device, texture_filename, &m_texture);
               
                    
            if(FAILED(hr))
                    {
                        MessageBox(NULL, "D3DXCreateTextureFromFile() - FAILED", "ParticleSystem", MB_OK);
                        
            return false;
                    }
               
                    
            return true;
                }

            o       注意: 我們使用動態的頂點緩存(D3DUSAGE DYNAMIC)。 因為我們需要在每幀中更新我們的粒子,意思是我們將會去存取頂點緩存的內存,回想一下,訪問一個靜態的頂點緩存慢得不可接受, 所以我們使用動態的頂點緩存。

            o       查看我們用過的 D3DUSAGE_POINTS標記,它說明頂點緩存將保存點精靈。

            o       頂點緩存的尺寸是由m_vb_num預先確定的,而且與系統中粒子的數量無關。 也就是說, m_vb_num將小于等于系統中粒子的數量。 這是因為渲染粒子系統是一批一批的,不是一次渲染全部。

            o       我們使用默認的內存池(pool)代替通常使用的托管內存池,因為動態頂點緩存不能用在托管內存池中。

            reset這個方法重新設置系統中每個粒子的屬性:

                void cParticleSystem::reset()
                {
                    
            for(list<sParticleAttribute>::iterator iter = m_particles.begin(); iter != m_particles.end(); iter++)
                        reset_particle(&(*iter));
                }

            reset_particle這個方法重新設置粒子的屬性。如何重設粒子的屬性,這依賴于具體粒子系統的特性。因此我們定義這個方法為虛擬的,等待子類去實現。

             add_particle這個方法用來在系統中增加一個粒子。在增加它到粒子列表之前,使用reset_particle方法先初始化粒子:

                void cParticleSystem::add_particle()
                {
                    sParticleAttribute attr;
                    reset_particle(&attr);
               
                    m_particles.push_back(attr);
                }

            update這個方法更新系統中所有的粒子。因為這個的方法的執行取決于具體粒子系統的特性,因此我們定義這個方法為抽象的,等待子類去實現。

            render這個方法用來顯示系統中所有的粒子。

            pre_render用它來初始化渲染狀態,在渲染前設置。 因為系統與系統之間是不同的,所以我們定義它為虛擬的。 默認將執行下列代碼:

                void cParticleSystem::pre_render()
                {
                    m_device->SetRenderState(D3DRS_LIGHTING,          FALSE);
                    m_device->SetRenderState(D3DRS_POINTSPRITEENABLE, TRUE);
                    m_device->SetRenderState(D3DRS_POINTSCALEENABLE,  TRUE);
                    m_device->SetRenderState(D3DRS_POINTSIZE,          float_to_dword(m_size));
                    m_device->SetRenderState(D3DRS_POINTSIZE_MIN,      float_to_dword(0.0f));
               
                    
            // control the size of the particle relative to distance
               
                    m_device->SetRenderState(D3DRS_POINTSCALE_A,    float_to_dword(0.0f));
                    m_device->SetRenderState(D3DRS_POINTSCALE_B,    float_to_dword(0.0f));
                    m_device->SetRenderState(D3DRS_POINTSCALE_C,    float_to_dword(1.0f));
               
                    
            // use alpha from texture
               
                    m_device->SetTextureStageState(0, D3DTSS_ALPHAARG1, D3DTA_TEXTURE);
                    m_device->SetTextureStageState(0, D3DTSS_ALPHAOP,    D3DTOP_SELECTARG1);
               
                    m_device->SetRenderState(D3DRS_ALPHABLENDENABLE, TRUE);
                    m_device->SetRenderState(D3DRS_SRCBLEND,  D3DBLEND_SRCALPHA);
                    m_device->SetRenderState(D3DRS_DESTBLEND, D3DBLEND_INVSRCALPHA);
                }

              注意:我們使用alpha混合渲染,以便設置紋理的alpha通道,來設置紋理像素的透明,用它產生多種效果。一種特殊的情況是:獲得象紋理那樣的非矩形的粒子。例如,獲得一個圓形“雪球形”的粒子,我們使用一個簡單的帶有alpha通道的紋理,它看上去是背景為黑色的帶有白色圓形的樣子。因此,顯示出來時只是一個白圓,這比白色的矩形紋理要好。

             post_render用它去保存所有渲染狀態。因為系統與系統間是不同的,所以我們定義它為虛擬的。默認將執行下列代碼:

                void cParticleSystem::post_render()
                {
                    m_device->SetRenderState(D3DRS_LIGHTING, TRUE);
                    m_device->SetRenderState(D3DRS_POINTSPRITEENABLE, FALSE);
                    m_device->SetRenderState(D3DRS_POINTSCALEENABLE,  FALSE);
                    m_device->SetRenderState(D3DRS_ALPHABLENDENABLE,  FALSE);
                }

             is_empty如果為True則在當前的系統中沒有粒子, 否則為false.

                bool cParticleSystem::is_empty()
                {
                    
            return m_particles.empty();
                }

            is_dead如果為True則系統中的所有粒子都是死的,否則為false
                bool cParticleSystem::is_dead()
                {
                    
            for(list<sParticleAttribute>::iterator iter = m_particles.begin(); iter != m_particles.end(); iter++)
                    {
                        
            // Is there at least one living particle? If yes, the system is not dead.
               
                    if(iter->is_alive)
                            
            return false;
                    }
               
                    
            // No living particles found, the system must be dead.
               
                return true;
                }

            remove_dead_particles搜索屬_particle性表,從表中殺死并刪除粒子。

                void cParticleSystem::remove_dead_particles()
                {
                    list<sParticleAttribute>::iterator iter = m_particles.begin();
               
                    
            while(iter != m_particles.end())
                    {        
                        
            if(! iter->is_alive)
                            
            // erase returns the next iterator, so no need to increment to the next one ourseleves.
               
                            iter = m_particles.erase(iter);
                        
            else
                            iter++;    
            // next in list
               
                }
                }

            posted on 2008-04-03 17:24 lovedday 閱讀(1056) 評論(0)  編輯 收藏 引用

            公告

            導航

            統計

            常用鏈接

            隨筆分類(178)

            3D游戲編程相關鏈接

            搜索

            最新評論

            观看 国产综合久久久久鬼色 欧美 亚洲 一区二区 | 午夜人妻久久久久久久久| 性欧美丰满熟妇XXXX性久久久| 亚洲国产另类久久久精品黑人| 97久久精品无码一区二区 | 久久人人爽人人爽人人片AV东京热| 欧美成人免费观看久久| 99久久er这里只有精品18| 久久人人爽人人澡人人高潮AV| 久久久国产精华液| 99久久www免费人成精品| 久久AV无码精品人妻糸列| 国产福利电影一区二区三区久久老子无码午夜伦不 | 久久永久免费人妻精品下载| 欧美精品一区二区精品久久| 99久久免费国产精品特黄| 亚洲国产成人久久综合碰碰动漫3d| 久久精品aⅴ无码中文字字幕不卡| 99久久国产综合精品五月天喷水 | 久久综合精品国产二区无码| 久久这里只有精品视频99| 久久se精品一区精品二区| 亚洲综合精品香蕉久久网| 久久久久久A亚洲欧洲AV冫 | 波多野结衣中文字幕久久| 欧美一级久久久久久久大片| 久久国产精品成人影院| 久久99热这里只有精品66| 久久久久久极精品久久久| 93精91精品国产综合久久香蕉| 国内精品人妻无码久久久影院 | 超级碰碰碰碰97久久久久| 久久无码一区二区三区少妇| 天天爽天天爽天天片a久久网| 国产婷婷成人久久Av免费高清| 亚洲中文字幕无码久久2020| 国产成人无码精品久久久性色| 中文无码久久精品| 久久久久亚洲AV无码网站| 欧美亚洲色综久久精品国产 | 久久久久久av无码免费看大片|