青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

天行健 君子當自強而不息

D3D中的網格模型(6)

11.4 外接體(Bounding Volumes)

有時我們需要計算mesh的外接體(邊界范圍),常用的有兩種類型:立方體和球。也有使用其它方法的,如圓柱體,橢球體,菱形體,膠囊形。圖11.4演示了對同一個mesh分別使用立方體和球體類型。

邊界盒/球常常被用來加速可見性測試,碰撞檢測等。假如一個mesh的邊界盒/球不可見,那么我們就說mesh不可見。一個盒/球可見性測試是比分別測試 mesh中的每個三角形要廉價的多。對于一個碰撞檢測例子,如果一枚導彈點火起飛,我們需要檢測它是否擊中了同一場景中的目標。由于這些物體都是由大量三角形構成,我們可以依次檢測每個對象的每個三角形,來測試導彈(可以用射線數學模型)是否碰撞到了這些三角形。這個方法需要進行多次的射線/三角形交點的運算。一個更好的方法是使用邊界盒或邊界球,計算射線與場景中的每個對象的邊界盒/邊界球的交點。如果射線與對象的邊界范圍相交,可以認為該對象被擊中了。這是一個公平的近似方法,如果需要更高的精度,可以用邊界范圍法先去除那些明顯不會相撞的對象,然后用更精確地方法檢測很可能相撞的對象。如果邊界范圍檢測發現相撞,則該對象就很有可能相撞。

D3DX庫提供了計算mesh的邊界盒和邊界球的函數。這些函數使用頂點數組作為輸入計算邊界盒/球。這些函數本來就是設計的很靈活的,它們可以使用各種頂點格式:

HRESULT D3DXComputeBoundingSphere(

       LPD3DXVECTOR3 pFirstPosition,

       DWORD NumVertices,

       DWORD dwStride,

       D3DXVECTOR3* pCenter,

       FLOAT* pRadius

);

pFirstPosition——指向在頂點數組中第一個頂點的向量,描述頂點位置。

NumVertices——在頂點數組中的的頂點數。

dwStride——每個頂點的字節大小。這很重要,因為頂點結構可能有一些額外信息如法向量和紋理坐標,函數需要知道應該跳過多少字節來得到下一個頂點的位置。

pCenter——返回邊界球的中心。

pRadius——返回邊界球的半徑。

HRESULT D3DXComputeBoundingBox(

       LPD3DXVECTOR3 pFirstPosition,

       DWORD NumVertices,

       DWORD dwStride,

       D3DXVECTOR3* pMin,

       D3DXVECTOR3* pMax

);

前三個參數和D3DXComputeBoundingSphere的前三個參數是完全一樣的。最后兩個參數分別用來返回邊界盒的最小和最大點。< /p>

 

11.4.1一些新的特殊常量

const float INFINITY = FLT_MAX;
const float EPSILON = 0.001f;

常量INFINITY是用來表示一個浮點數所能存儲的最大數。因為我們找不到一個比FLT_MAX還要大的浮點數,我們可以將它視為無窮大。常量 EPSILON是一個很小的值,我們這樣定義它,凡是比它小的數就視為0。這也是很有必要的,因為得到的浮點是不精確的。因此,讓它和0比較相等肯定會失敗。我們因此可以通過把該值與0的差值與EPSILON比較來確定是否相等:

bool Equals(float lhs, float rhs)

{

       // if lhs == rhs their difference should be zero

       return fabs(lhs - rhs) < EPSILON ? true : false;

}

 

11.4.2外接體類型

為了更容易的使用邊界盒和邊界球,我們將它們分別封裝到兩個類中。

class cBoundingBox
{
public:
    D3DXVECTOR3 m_min, m_max;
    
    cBoundingBox();
    
bool is_point_inside(D3DXVECTOR3& point);
};

class cBoundingSphere
{
public:
    D3DXVECTOR3 m_center;
    
float        m_radius;

    cBoundingSphere();
};

cBoundingBox::cBoundingBox()
{    
    m_min.x 
= INFINITY;
    m_min.y 
= INFINITY;
    m_min.z 
= INFINITY;

    m_max.x 
= -INFINITY;
    m_max.y 
= -INFINITY;
    m_max.z 
= -INFINITY;
}

bool cBoundingBox::is_point_inside(D3DXVECTOR3 &point)
{
    
return (point.x >= m_min.x && point.y >= m_min.y && point.z >= m_min.z &&
            point.x 
<= m_max.x && point.y <= m_max.y && point.z <= m_max.z);
}

cBoundingSphere::cBoundingSphere()
{
    m_radius 
= 0.0f;
}

 

11.4.3實例程序:外接體

實例程序主要演示使用D3DXComputeBoundingSphere和 D3DXComputeBoundingBox。程序讀取一個X文件并且計算該mesh的邊界球,它創建兩個ID3DXMesh對象,一個用來作為邊界球模型一個用來作為邊界盒模型(如下圖所示)。你能夠通過敲空格鍵在邊界球和邊界盒之間切換。



主程序:

/**************************************************************************************
  Demonstrates how to use D3DXComputeBoundingSphere and D3DXComputeBoundingBox.
  The spacebar key switches between rendering the mesh's bounding sphere and box.  
 *************************************************************************************
*/

#include 
<vector>
#include 
"d3dUtility.h"

#pragma warning(disable : 
4100)

using namespace std;

const int WIDTH  = 640;
const int HEIGHT = 480;

IDirect3DDevice9
*            g_device;
ID3DXMesh
*                    g_mesh;
ID3DXMesh
*                    g_sphere_mesh;
ID3DXMesh
*                    g_box_mesh;
vector
<D3DMATERIAL9>        g_materials;
vector
<IDirect3DTexture9*>    g_textures;

bool g_is_render_bounding_sphere = true;

bool compute_bounding_sphere(ID3DXMesh* mesh, cBoundingSphere* sphere);
bool compute_bounding_box(ID3DXMesh* mesh, cBoundingBox* box);

////////////////////////////////////////////////////////////////////////////////////////////////////

bool setup()
{    
    
// load the XFile data

    ID3DXBuffer
*    adjacency_buffer = NULL;
    ID3DXBuffer
*    material_buffer  = NULL;
    DWORD            num_material     
= 0;

    HRESULT hr 
= D3DXLoadMeshFromX("bigship1.x", D3DXMESH_MANAGED, g_device, &adjacency_buffer, &material_buffer,
                                   NULL, 
&num_material, &g_mesh);

    
if(FAILED(hr))
    {
        MessageBox(NULL, 
"D3DXLoadMeshFromX() - FAILED""ERROR", MB_OK);
        
return false;
    }

    
// extract the materials, and load textures.
    if(material_buffer != NULL && num_material != 0)
    {
        D3DXMATERIAL
* materials = (D3DXMATERIAL*) material_buffer->GetBufferPointer();

        
for(DWORD i = 0; i < num_material; i++)
        {
            
// the MatD3D property doesn't have an ambient value set when it load, so set it now.
            materials[i].MatD3D.Ambient = materials[i].MatD3D.Diffuse;

            
// save the ith material
            g_materials.push_back(materials[i].MatD3D);

            
// check if the ith material has an associative texture
            if(materials[i].pTextureFilename != NULL)
            {
                
// yes, load the texture for the ith subset.
                IDirect3DTexture9* texture;
                D3DXCreateTextureFromFile(g_device, materials[i].pTextureFilename, 
&texture);

                
// save the loaded texture
                g_textures.push_back(texture);
            }
            
else
            {
                
// no texture for the ith subset
                g_textures.push_back(NULL);
            }
        }
    }

    safe_release
<ID3DXBuffer*>(material_buffer);
    
    
// optimize the mesh
    
    hr 
= g_mesh->OptimizeInplace(D3DXMESHOPT_ATTRSORT | D3DXMESHOPT_COMPACT | D3DXMESHOPT_VERTEXCACHE,
        (DWORD
*) adjacency_buffer->GetBufferPointer(), NULL, NULL, NULL);

    safe_release
<ID3DXBuffer*>(adjacency_buffer);

    
if(FAILED(hr))
    {
        MessageBox(NULL, 
"OptimizeInplace() - FAILED""ERROR", MB_OK);
        
return false;
    }

    
// compute bounding sphere and bounding box

    cBoundingSphere bounding_sphere;
    cBoundingBox    bounding_box;

    compute_bounding_sphere(g_mesh, 
&bounding_sphere);
    compute_bounding_box(g_mesh, 
&bounding_box);

    D3DXCreateSphere(g_device, bounding_sphere.m_radius, 
2020&g_sphere_mesh, NULL);

    
float width  = bounding_box.m_max.x - bounding_box.m_min.x;
    
float height = bounding_box.m_max.y - bounding_box.m_min.y;
    
float depth  = bounding_box.m_max.z - bounding_box.m_min.z;

    D3DXCreateBox(g_device, width, height, depth, 
&g_box_mesh, NULL);

    
// set texture filters    
    g_device->SetSamplerState(0, D3DSAMP_MAGFILTER, D3DTEXF_LINEAR);
    g_device
->SetSamplerState(0, D3DSAMP_MINFILTER, D3DTEXF_LINEAR);
    g_device
->SetSamplerState(0, D3DSAMP_MIPFILTER, D3DTEXF_LINEAR);

    
// set lights

    D3DXVECTOR3 dir(
1.0f-1.0f1.0f);
    D3DXCOLOR color(
1.0f1.0f1.0f1.0f);
    D3DLIGHT9 light 
= init_directional_light(&dir, &color);

    g_device
->SetLight(0&light);
    g_device
->LightEnable(0, TRUE);
    g_device
->SetRenderState(D3DRS_NORMALIZENORMALS, TRUE);
    g_device
->SetRenderState(D3DRS_SPECULARENABLE, TRUE);

    
// set camera

    D3DXVECTOR3 pos(
4.0f12.0f-20.0f);
    D3DXVECTOR3 target(
0.0f0.0f0.0f);
    D3DXVECTOR3 up(
0.0f1.0f0.0f);

    D3DXMATRIX view_matrix;
    D3DXMatrixLookAtLH(
&view_matrix, &pos, &target, &up);
    g_device
->SetTransform(D3DTS_VIEW, &view_matrix);

    
// set the projection matrix
    D3DXMATRIX proj;
    D3DXMatrixPerspectiveFovLH(
&proj, D3DX_PI * 0.5f, (float)WIDTH/HEIGHT, 1.0f1000.0f);
    g_device
->SetTransform(D3DTS_PROJECTION, &proj);
    
    
return true;
}

///////////////////////////////////////////////////////////////////////////////////////////////////////

void cleanup()
{    
    safe_release
<ID3DXMesh*>(g_mesh);
    safe_release
<ID3DXMesh*>(g_sphere_mesh);
    safe_release
<ID3DXMesh*>(g_box_mesh);
    
    
for(DWORD i = 0; i < g_textures.size(); i++)
        safe_release
<IDirect3DTexture9*>(g_textures[i]);
}

///////////////////////////////////////////////////////////////////////////////////////////////////////

bool display(float time_delta)
{
    
// update: rotate the mesh

    
static float y = 0.0f;

    D3DXMATRIX y_rot_matrix;    
    D3DXMatrixRotationY(
&y_rot_matrix, y);
    g_device
->SetTransform(D3DTS_WORLD, &y_rot_matrix);

    y 
+= time_delta;
    
if(y >= 6.28f)
        y 
= 0.0f;

    
// render now

    g_device
->Clear(0, NULL, D3DCLEAR_TARGET | D3DCLEAR_ZBUFFER, 0x000000001.0f0);

    g_device
->BeginScene();

    
for(DWORD i = 0; i < g_materials.size(); i++)
    {
        g_device
->SetMaterial(&g_materials[i]);
        g_device
->SetTexture(0, g_textures[i]);
        g_mesh
->DrawSubset(i);
    }

    
// draw bounding volume in yellow and at 30% opacity
    D3DMATERIAL9 blue_material = YELLOW_MATERIAL;
    blue_material.Diffuse.a 
= 0.30f;    // 30% opacity
    g_device->SetMaterial(&blue_material);

    g_device
->SetRenderState(D3DRS_ALPHABLENDENABLE, TRUE);
    g_device
->SetRenderState(D3DRS_SRCBLEND,  D3DBLEND_SRCALPHA);
    g_device
->SetRenderState(D3DRS_DESTBLEND, D3DBLEND_INVSRCALPHA);

    
if(g_is_render_bounding_sphere)
        g_sphere_mesh
->DrawSubset(0);
    
else
        g_box_mesh
->DrawSubset(0);

    g_device
->SetRenderState(D3DRS_ALPHABLENDENABLE, FALSE);

    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);

        
if(word_param == VK_SPACE)
            g_is_render_bounding_sphere 
= !g_is_render_bounding_sphere;

        
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;
}

///////////////////////////////////////////////////////////////////////////////////////////////////////

bool compute_bounding_sphere(ID3DXMesh* mesh, cBoundingSphere* sphere)
{
    BYTE
* v;

    mesh
->LockVertexBuffer(0, (void**)&v);

    HRESULT hr 
= D3DXComputeBoundingSphere((D3DXVECTOR3*)v, mesh->GetNumVertices(),
        D3DXGetFVFVertexSize(mesh
->GetFVF()), &sphere->m_center, &sphere->m_radius);

    mesh
->UnlockVertexBuffer();

    
if(FAILED(hr))
        
return false;

    
return true;
}

bool compute_bounding_box(ID3DXMesh* mesh, cBoundingBox* box)
{
    BYTE
* v;

    mesh
->LockVertexBuffer(0, (void**)&v);

    HRESULT hr 
= D3DXComputeBoundingBox((D3DXVECTOR3*)v, mesh->GetNumVertices(),
        D3DXGetFVFVertexSize(mesh
->GetFVF()), &box->m_min, &box->m_max);

    mesh
->UnlockVertexBuffer();

    
if(FAILED(hr))
        
return false;

    
return true;
}

 


下載源代碼

posted on 2008-03-29 19:16 lovedday 閱讀(2379) 評論(0)  編輯 收藏 引用


只有注冊用戶登錄后才能發表評論。
網站導航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


公告

導航

統計

常用鏈接

隨筆分類(178)

3D游戲編程相關鏈接

搜索

最新評論

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            久久天堂成人| 欧美一级黄色录像| 欧美日韩国产一区二区三区地区| 久久久国产一区二区| 久久精品1区| 久久成人综合视频| 免费久久99精品国产自在现线| 久热国产精品| 欧美日韩国产天堂| 国产精品免费福利| 红桃视频成人| 一区二区三区福利| 久久精品欧美日韩| 亚洲高清不卡在线| 一区二区成人精品| 久久国产精品久久精品国产| 欧美不卡在线视频| 国产欧美精品日韩区二区麻豆天美| 在线免费观看日本一区| 一本色道久久综合一区| 欧美日韩国产片| 国产精品久久一卡二卡| 狠狠久久亚洲欧美专区| 一本久道久久综合狠狠爱| 欧美一区免费视频| 亚洲黄色在线看| 欧美一级成年大片在线观看| 欧美精品一区三区| 黄色成人精品网站| 午夜精品久久久久久久99樱桃| 久久亚洲综合色一区二区三区| 日韩视频在线永久播放| 久久久噜噜噜久久中文字免| 国产精品极品美女粉嫩高清在线 | 欧美制服第一页| 欧美人妖另类| 亚洲国产精品123| 欧美一区成人| 亚洲美女毛片| 欧美成人69av| 一区二区三区在线免费播放| 亚洲嫩草精品久久| 亚洲精品免费在线| 久久久久久香蕉网| 国产欧美欧美| 亚洲免费婷婷| 99国产精品久久久久久久成人热| 久久裸体艺术| 激情久久五月| 久久久欧美精品sm网站| 亚洲一区二区三区在线播放| 欧美日韩国产免费观看| 亚洲精品乱码久久久久久黑人| 免费观看日韩av| 久久国产精品久久久| 国产日韩欧美综合精品| 午夜精品久久久久久99热| 亚洲美女在线视频| 欧美另类综合| 在线视频日韩精品| 亚洲激情二区| 欧美日韩精品一区二区| 亚洲免费精彩视频| 亚洲国产一区视频| 欧美久久电影| 亚洲综合日韩| 欧美一区二区三区视频在线| 国产亚洲一区精品| 麻豆精品在线观看| 免费中文日韩| 在线亚洲精品| 亚洲欧美美女| 在线不卡亚洲| 亚洲国产精品综合| 欧美日韩久久不卡| 欧美亚洲在线观看| 久久精品99无色码中文字幕| 一区二区三区在线视频播放| 欧美国产精品中文字幕| 久久大综合网| 国产精品视频yy9299一区| 一本久道久久综合狠狠爱| 欧美视频在线一区| 亚洲免费在线看| 久久成人这里只有精品| 亚洲成色999久久网站| 亚洲高清av| 国产精品人人爽人人做我的可爱| 欧美在线观看日本一区| 久久婷婷av| 亚洲综合久久久久| 久久精品国产第一区二区三区最新章节| 国产欧美 在线欧美| 欧美成人情趣视频| 国产精品地址| 亚洲成色777777女色窝| 国产精品视频一区二区三区| 母乳一区在线观看| 国产精品豆花视频| 欧美v日韩v国产v| 国产精品麻豆欧美日韩ww| 久久久久久久综合| 欧美日韩国产小视频在线观看| 久久精品国产亚洲aⅴ| 欧美大片一区二区| 久久精品国产亚洲高清剧情介绍| 欧美成人精品1314www| 欧美在线影院| 欧美日韩一区在线视频| 久久综合国产精品| 欧美性猛交99久久久久99按摩| 久久影视三级福利片| 欧美日韩国内| 亚洲电影第三页| 国产综合欧美| 亚洲欧美成人一区二区三区| 一区二区免费在线播放| 开心色5月久久精品| 久久精品日产第一区二区三区| 欧美日韩一本到| 亚洲国产精品电影在线观看| 国内精品免费午夜毛片| 亚洲综合激情| 亚洲欧美www| 欧美日韩一区二区三区高清| 亚洲黄色一区| 亚洲精品在线观看免费| 老色批av在线精品| 免费视频一区| 99国产麻豆精品| 久久永久免费| 美女国产一区| 伊人精品成人久久综合软件| 久久成人精品电影| 久久久精品日韩欧美| 国产日本精品| 午夜精品区一区二区三| 午夜精品国产精品大乳美女| 欧美视频在线不卡| 一区二区免费在线播放| 亚洲午夜高清视频| 国产精品久久久久久久久久久久久久 | 一区二区三区精品国产| 欧美成人一区二区三区片免费 | 亚洲一区二区三区在线| 亚洲欧美影音先锋| 国产精品你懂的| 午夜精品久久久久久久男人的天堂| 亚洲男人的天堂在线| 国产精品网站在线| 久久精品亚洲热| 亚洲国产乱码最新视频| 日韩视频精品| 国产精品久久久久久一区二区三区 | 亚洲图片欧美日产| 国产精品日韩| 久久久欧美精品| 亚洲高清二区| 亚洲一级特黄| 国产亚洲精品久久久| 久久阴道视频| 一本色道久久综合亚洲精品小说| 亚洲欧美一区二区在线观看| 韩国精品久久久999| 欧美高清在线一区| 亚洲影院色在线观看免费| 久久这里有精品15一区二区三区| 亚洲人成在线影院| 国产精品视频一| 欧美成人免费观看| 亚洲午夜一区二区三区| 免费不卡在线视频| 一区二区三区精品国产| 国内精品久久久久影院薰衣草| 欧美激情国产高清| 午夜在线电影亚洲一区| 亚洲国产国产亚洲一二三| 午夜欧美理论片| 亚洲美女少妇无套啪啪呻吟| 国产欧美日韩不卡免费| 欧美国产日本韩| 小辣椒精品导航| 亚洲精品欧美专区| 老司机午夜免费精品视频| 亚洲视频网站在线观看| 一区三区视频| 国产日本亚洲高清| 欧美日韩精品欧美日韩精品一 | 一区二区精品国产| 欧美激情综合| 久久理论片午夜琪琪电影网| 亚洲一区二区三区免费视频| 91久久精品美女| 狠狠色综合日日| 国产欧美亚洲日本| 国产精品美女xx| 欧美性猛交99久久久久99按摩| 欧美第十八页| 久久久www免费人成黑人精品| 亚洲一区二区精品视频|