《逐梦旅程 WINDOWS游戏编程之从零开始》笔记6——Direct3D中的顶点缓存和索引缓存

第12章 Direct3D绘制基础

1. 顶点缓存

计算机所描绘的3D图形是通过多边形网格来构成的,网网格勾勒出轮廓,然后在网格轮廓的表面上贴上相应的图片,这样就构成了一个3D模型。三角形网格是构建物体模型的基本单元,而一个三角形有3个顶点,为了能够使用大量三角形组成三角形网格来描述物体,需要首先定义号三角形的顶点(Vertex),3个顶点确定一个三角形,顶点除了定义每个顶点的坐标位置外,还还含有颜色等其他属性。

在Direct3D中,顶点的具体表现形式是顶点缓存,顶点缓存保存了顶点数据的内存空间。如果想在Direct3D中创建物体的额话,就需要创建构成物体的所有顶点结构。但Direct3D绘制图形时,将根据这些顶点的结构创建一个三角形列表,来描述物体的形状和轮廓。

在图中用了4个顶点组成了一个正方形,这4个顶点分别是V0,V1,V2,V3。为了正确描述这个正方形,需要根据这4个顶点创建两个三角形V0V1V2和V0V2V3,而这两个三角形的顶点数据会依次保存在顶点缓存中。在Direct3D中一般是顺时针来给顶点赋值的。下面开始讲解在Direct3D中使用顶点缓存的具体步骤。

顶点缓存使用四步曲之一:设计顶点缓存

第一步工作是对顶点的类型进行设计,先来介绍固定功能流水线中使用频繁的一套顶点定义格式——灵活顶点格式(Flexible Vertex Format,FVF)。灵活顶点格式可以让我们随心所欲地自定义其中锁包含的顶点属性信息。例如,顶点的三维坐标,颜色,顶点线法和纹理坐标等。

创建自定义灵活顶点格式时,根据实际的需求来定义一个包含特定定点信息的结构体。如:

struct CUSTOMVERTEX
{
    float x,y,z;    //坐标
    DWORD colot;    //颜色
};

仅定义出结构体,Direct3D是不能理解我们要做什么的,还需要一个宏来传达我们定义的顶点有哪些属性:

#define  D3DFVF_CUSTOMVERTEX  (D3DFVF_XYZRHW|D3DFVF_DIFFUSE)

注:

上面的单竖杠 “|” 是按位或的意思,没有短路功能,右边的表达式总会被计算 。但是我这里其实并不明白上面那个宏定义为什么要做个按位或运算呢?

上面的CUSTOMVERTEX和宏定义的部分中的D3DFVF_CUSTOMVERTEX不是一码事,后面的 D3DFVF_XYZRHW|D3DFVF_DIFFUSE 全是16进制数值,貌似在Windows编程中,像这些预定义好的宏其实大部分都是数值,因为单纯的数字代表不了其具体含义,所以要定义为更有语义化的宏。至于这里定义的宏D3DFVF_CUSTOMVERTEX是用来描述我们自定义的顶点结构体的,估计之后在程序中要用的,因为它是一一对应着构体中的属性。

在Direct3D中常用的FVF格式可以取一下所表示的这些值:

注意在书写灵活顶点格式的宏定义的时候要遵循一个顺序原则,优先级要这样来分:

顶点缓存使用四步曲之二:创建顶点缓存

顶点缓存有IDirect3DVertexBuffer9接口对象来表示。先定义一个指向IDirect3DVertexBuffer9接口的指针变量:

LPDIRECT3DVERTEXBUFFER9 g_pVertexBuffer = NULL;

然后用我们的Direct3D设备调用CreateVertexBuffer方法创建顶点缓存,把内存地址复制给我们创建的这个指针了。

关于CreateVertexBuffer方法:

HRESULT CreateVertexBuffer(
  [in]          UINT                   Length,  //顶点缓存的大小,以字节为单位
  [in]          DWORD                  Usage,  //指定使用缓存的一些附加属性,取0表示没有附加属性,或者取下面表格中的一个或多个值,多个值用 “|” 连接
  [in]          DWORD                  FVF,   //指定要存储在顶点缓存中的灵活顶点格式,就是之前我们定义的宏
  [in]          D3DPOOL                Pool,  //指定存储顶点缓存的内存位置实在内存中还是在显卡中
  [out, retval] IDirect3DVertexBuffer9 **ppVertexBuffer,  //调用CreateVertexBuffer方法就是对这个变量初始化
  [in]          HANDLE                 *pSharedHandle   //设置为NULL或0就可以了
);

注:

还是关于 "|",我百度了下,这个符号在C++/C中就是按位或运算啊,但是书上这里的解释是参数取多个值时用 "|" 连接,从程序上来看的确使这个作用,但是我不明白为什么这里按位或运算能实现这种效果。

所以,这一步创建顶点缓存的代码合起来就是:

LPDIRECT3DVERTEXBUFFER9 g_pVertexBuffer = NULL;    //顶点缓冲对象,记住这也是个指针
//创建顶点缓存区
g_pd3dDevice->CreateVertexBuffer(6*sizeof(CUSTOMVERTEX), 0, D3DFVF_CUSTOMVERTEX, D3DPOOL_DEFAULT, &g_pVertexBuffer, NUll);

顶点缓存使用四步曲之三:访问顶点缓存

上一步我们创建了顶点缓存的金钥匙——指向IDirect3DVertexBuffer9接口的指针。之后呢,就是利用这个指针去“指”一下IDirect3DVertexBuffer9接口的Lock以及Unlock方法,然后在Lock和Unlock方法之间访问我们的顶点缓存就好了。

加锁函数IDirect3DVertexBuffer9::Lock():

HRESULT Lock(
  [in]  UINT  OffsetToLock,  //加锁区域字存储空间的起始位置到开始锁定位置的偏移量,单位为字节
  [in]  UINT  SizeToLock,  //要锁定的存储区字节数
  [out] VOID  **ppbData,  //指向被锁定的存储区的起始地址的指针
  [in]  DWORD Flags  //锁定方式
);

有两种方式来在Lock和UnLock方法之间访问缓存的内容,第一种方式是直接在Lock和UnLock方法之间对每个顶点的数据进行赋值和修改,以Lock方法中的ppbData指针参数作为数组的首地址,例如:

g_pVertexBuf->Lock(0,0, (void**)&pVertices,0);  //加锁
pVertices[0] = CUSTOMVERTEX(-80.0f, -80.0f, 0.0f, 1.0f, D3DCOLOR_XRGB(255,0,0)); //V0
pVertices[1] = CUSTOMVERTEX(-80.0f, 80.0f, 0.0f, 1.0f, D3DCOLOR_XRGB(0,255,0)); //V1
pVertices[2] = CUSTOMVERTEX(80.0f, 80.0f, 0.0f, 1.0f, D3DCOLOR_XRGB(0,0,255)); //V2
pVertices[3] = CUSTOMVERTEX(80.0f, -80.0f, 0.0f, 1.0f, D3DCOLOR_XRGB(255,0,255)); //V3
g_pVertexBuf->UnLock();    //解锁

与上面对应的顶点结构题体定义是:

第二种方式是事先准备好顶点数据的数组,然后在加锁和解锁之间使用memcpy函数,进行数组内容的拷贝就可以了:

顶点缓存使用四步曲之四:图形的绘制

这步是利用之前我们缓存的顶点来绘制出图形来。

首先是SetStreamSource方法:

HRESULT SetStreamSource(
  [in] UINT                   StreamNumber,  //指定与该顶点缓存建立连接的数据流,通常设为0
  [in] IDirect3DVertexBuffer9 *pStreamData,  //包含顶点数据的顶点缓存指针,就是我们定义的指向IDirect3DVertextBuffer9接口的指针变量g_pVertexBuffer
  [in] UINT                   OffsetInBytes,  //数据流中以字节为单位的偏移量,通常设为0
  [in] UINT                   Stride    //表示在顶点缓存中存储的每个顶点结构的大小,以字节为单位
);

具体的调用例子:

g_pd3Device->SetStreamSource(0, g_pVertexBuffer, 0, sizeof(CUSTOMVERTEX));

然后是调用SerFVF这个方法:

HRESULT SetFVF(
  [in] DWORD FVF
);

这个不多说了,参数直接填之前我们定义的宏就可以了。最后是DrawPrimitive:

HRESULT DrawPrimitive(
  [in] D3DPRIMITIVETYPE PrimitiveType,  //将要绘制的图元类型,在D3DPRIMITIVETYPE枚举中取值
  [in] UINT             StartVertex,   //指定从顶点缓存中读取顶点数据的其实索引位置
  [in] UINT             PrimitiveCount  //需要绘制的图元数量,通过StartVertex和PrimitiveCount可以对缓存中的某一部分进行绘制
);

来个调用实例:

g_p3dDevice->DrawPrimitive(D3DPT_TRIANGLELIST, 0, 8);


最后是国际惯例,来个完整的程序:

  1 //-----------------------------------【头文件包含部分】---------------------------------------
  2 #include <d3d9.h>
  3 #include <d3dx9.h>
  4 #include <tchar.h>
  5
  6 //-----------------------------------【库文件包含部分】---------------------------------------
  7 #pragma comment(lib,"d3d9.lib")
  8 #pragma comment(lib,"d3dx9.lib")
  9
 10 //-----------------------------------【宏定义部分】--------------------------------------------
 11 #define WINDOW_WIDTH    800                            //为窗口宽度定义的宏,以方便在此处修改窗口宽度
 12 #define WINDOW_HEIGHT    600                            //为窗口高度定义的宏,以方便在此处修改窗口高度
 13 #define WINDOW_TITLE    L"【致我们永不熄灭的游戏开发梦想】Direct3D顶点缓存的逆袭 示例程序"    //为窗口标题定义的宏
 14 #define SAFE_RELEASE(p) { if(p) { (p)->Release(); (p)=NULL; } }   //定义一个安全释放宏,便于后面COM接口指针的释放
 15
 16
 17 //------------------------------------------------------------------------------------------------
 18 // 【顶点缓存使用四步曲之一】:设计顶点格式
 19 //------------------------------------------------------------------------------------------------
 20 struct CUSTOMVERTEX
 21 {
 22     FLOAT x, y, z, rhw;
 23     DWORD color;
 24 };
 25 #define D3DFVF_CUSTOMVERTEX (D3DFVF_XYZRHW|D3DFVF_DIFFUSE)  //FVF灵活顶点格式
 26
 27 //-----------------------------------【全局变量声明部分】-------------------------------------
 28 LPDIRECT3DDEVICE9                    g_pd3dDevice = NULL; //Direct3D设备对象
 29 ID3DXFont*                                g_pFont=NULL;    //字体COM接口
 30 float                                            g_FPS = 0.0f;       //一个浮点型的变量,代表帧速率
 31 wchar_t                                        g_strFPS[50];    //包含帧速率的字符数组
 32 LPDIRECT3DVERTEXBUFFER9        g_pVertexBuffer = NULL;    //顶点缓冲区对象
 33
 34
 35 //-----------------------------------【全局函数声明部分】-------------------------------------
 36 //    描述:全局函数声明,防止“未声明的标识”系列错误
 37 //------------------------------------------------------------------------------------------------
 38 LRESULT CALLBACK    WndProc( HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam );//窗口过程函数
 39 HRESULT                    Direct3D_Init(HWND hwnd);         //在这个函数中进行Direct3D的初始化
 40 HRESULT                    Objects_Init(HWND hwnd);         //在这个函数中进行要绘制的物体的资源初始化
 41 VOID                            Direct3D_Render(HWND hwnd);     //在这个函数中进行Direct3D渲染代码的书写
 42 VOID                            Direct3D_CleanUp( );                //在这个函数中清理COM资源以及其他资源
 43 float                            Get_FPS();
 44
 45 //-----------------------------------【WinMain( )函数】--------------------------------------
 46 //    描述:Windows应用程序的入口函数,我们的程序从这里开始
 47 //------------------------------------------------------------------------------------------------
 48 int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance,LPSTR lpCmdLine, int nShowCmd)
 49 {
 50     //【1】窗口创建四步曲之一:开始设计一个完整的窗口类
 51     WNDCLASSEX wndClass = { 0 };                            //用WINDCLASSEX定义了一个窗口类
 52     wndClass.cbSize = sizeof( WNDCLASSEX ) ;            //设置结构体的字节数大小
 53     wndClass.style = CS_HREDRAW | CS_VREDRAW;    //设置窗口的样式
 54     wndClass.lpfnWndProc = WndProc;                    //设置指向窗口过程函数的指针
 55     wndClass.cbClsExtra        = 0;                                //窗口类的附加内存,取0就可以了
 56     wndClass.cbWndExtra        = 0;                            //窗口的附加内存,依然取0就行了
 57     wndClass.hInstance = hInstance;                        //指定包含窗口过程的程序的实例句柄。
 58     wndClass.hIcon=(HICON)::LoadImage(NULL,L"icon.ico",IMAGE_ICON,0,0,LR_DEFAULTSIZE|LR_LOADFROMFILE);  //本地加载自定义ico图标
 59     wndClass.hCursor = LoadCursor( NULL, IDC_ARROW );    //指定窗口类的光标句柄。
 60     wndClass.hbrBackground=(HBRUSH)GetStockObject(GRAY_BRUSH);  //为hbrBackground成员指定一个白色画刷句柄
 61     wndClass.lpszMenuName = NULL;                        //用一个以空终止的字符串,指定菜单资源的名字。
 62     wndClass.lpszClassName = L"ForTheDreamOfGameDevelop";        //用一个以空终止的字符串,指定窗口类的名字。
 63
 64     //【2】窗口创建四步曲之二:注册窗口类
 65     if( !RegisterClassEx( &wndClass ) )                //设计完窗口后,需要对窗口类进行注册,这样才能创建该类型的窗口
 66         return -1;
 67
 68     //【3】窗口创建四步曲之三:正式创建窗口
 69     HWND hwnd = CreateWindow( L"ForTheDreamOfGameDevelop",WINDOW_TITLE,                //喜闻乐见的创建窗口函数CreateWindow
 70         WS_OVERLAPPEDWINDOW, CW_USEDEFAULT, CW_USEDEFAULT, WINDOW_WIDTH,
 71         WINDOW_HEIGHT, NULL, NULL, hInstance, NULL );
 72
 73     //Direct3D资源的初始化,调用失败用messagebox予以显示
 74     if (!(S_OK==Direct3D_Init (hwnd)))
 75     {
 76         MessageBox(hwnd, _T("Direct3D初始化失败~!"), _T("浅墨的消息窗口"), 0); //使用MessageBox函数,创建一个消息窗口
 77     }
 78
 79     //【4】窗口创建四步曲之四:窗口的移动、显示与更新
 80     MoveWindow(hwnd,250,80,WINDOW_WIDTH,WINDOW_HEIGHT,true);        //调整窗口显示时的位置,使窗口左上角位于(250,80)处
 81     ShowWindow( hwnd, nShowCmd );    //调用ShowWindow函数来显示窗口
 82     UpdateWindow(hwnd);                        //对窗口进行更新,就像我们买了新房子要装修一样
 83
 84     //【5】消息循环过程
 85     MSG msg = { 0 };  //初始化msg
 86     while( msg.message != WM_QUIT )            //使用while循环
 87     {
 88         if( PeekMessage( &msg, 0, 0, 0, PM_REMOVE ) )   //查看应用程序消息队列,有消息时将队列中的消息派发出去。
 89         {
 90             TranslateMessage( &msg );        //将虚拟键消息转换为字符消息
 91             DispatchMessage( &msg );        //该函数分发一个消息给窗口程序。
 92         }
 93         else
 94         {
 95             Direct3D_Render(hwnd);   //进行渲染
 96         }
 97     }
 98     //【6】窗口类的注销
 99     UnregisterClass(L"ForTheDreamOfGameDevelop", wndClass.hInstance);  //程序准备结束,注销窗口类
100     return 0;
101 }
102
103 //-----------------------------------【WndProc( )函数】--------------------------------------
104 //    描述:窗口过程函数WndProc,对窗口消息进行处理
105 LRESULT CALLBACK WndProc( HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam )
106 {
107     switch( message )                        //switch语句开始
108     {
109     case WM_PAINT:                        // 若是客户区重绘消息
110         Direct3D_Render(hwnd);                 //调用Direct3D渲染函数
111         ValidateRect(hwnd, NULL);        // 更新客户区的显示
112         break;                                    //跳出该switch语句
113
114     case WM_KEYDOWN:                    // 若是键盘按下消息
115         if (wParam == VK_ESCAPE)    // 如果被按下的键是ESC
116             DestroyWindow(hwnd);        // 销毁窗口, 并发送一条WM_DESTROY消息
117         break;                                    //跳出该switch语句
118
119     case WM_DESTROY:                    //若是窗口销毁消息
120         Direct3D_CleanUp();            //调用自定义的资源清理函数Game_CleanUp()进行退出前的资源清理
121         PostQuitMessage( 0 );            //向系统表明有个线程有终止请求。用来响应WM_DESTROY消息
122         break;                                    //跳出该switch语句
123
124     default:                                        //若上述case条件都不符合,则执行该default语句
125         return DefWindowProc( hwnd, message, wParam, lParam );        //调用缺省的窗口过程
126     }
127
128     return 0;                                    //正常退出
129 }
130
131 //-----------------------------------【Direct3D_Init( )函数】--------------------------------------
132 //    描述:Direct3D初始化函数,进行Direct3D的初始化
133 //------------------------------------------------------------------------------------------------
134 HRESULT Direct3D_Init(HWND hwnd)
135 {
136     //--------------------------------------------------------------------------------------
137     // 【Direct3D初始化四步曲之一,创接口】:创建Direct3D接口对象, 以便用该Direct3D对象创建Direct3D设备对象
138     //--------------------------------------------------------------------------------------
139     LPDIRECT3D9  pD3D = NULL; //Direct3D接口对象的创建
140     if( NULL == ( pD3D = Direct3DCreate9( D3D_SDK_VERSION ) ) ) //初始化Direct3D接口对象,并进行DirectX版本协商
141         return E_FAIL;
142
143     //--------------------------------------------------------------------------------------
144     // 【Direct3D初始化四步曲之二,取信息】:获取硬件设备信息
145     //--------------------------------------------------------------------------------------
146     D3DCAPS9 caps; int vp = 0;
147     if( FAILED( pD3D->GetDeviceCaps( D3DADAPTER_DEFAULT, D3DDEVTYPE_HAL, &caps ) ) )
148     {
149         return E_FAIL;
150     }
151     if( caps.DevCaps & D3DDEVCAPS_HWTRANSFORMANDLIGHT )
152         vp = D3DCREATE_HARDWARE_VERTEXPROCESSING;   //支持硬件顶点运算,我们就采用硬件顶点运算,妥妥的
153     else
154         vp = D3DCREATE_SOFTWARE_VERTEXPROCESSING; //不支持硬件顶点运算,无奈只好采用软件顶点运算
155
156     //--------------------------------------------------------------------------------------
157     // 【Direct3D初始化四步曲之三,填内容】:填充D3DPRESENT_PARAMETERS结构体
158     //--------------------------------------------------------------------------------------
159     D3DPRESENT_PARAMETERS d3dpp;
160     ZeroMemory(&d3dpp, sizeof(d3dpp));
161     d3dpp.BackBufferWidth            = WINDOW_WIDTH;
162     d3dpp.BackBufferHeight           = WINDOW_HEIGHT;
163     d3dpp.BackBufferFormat           = D3DFMT_A8R8G8B8;
164     d3dpp.BackBufferCount            = 1;
165     d3dpp.MultiSampleType            = D3DMULTISAMPLE_NONE;
166     d3dpp.MultiSampleQuality         = 0;
167     d3dpp.SwapEffect                 = D3DSWAPEFFECT_DISCARD;
168     d3dpp.hDeviceWindow              = hwnd;
169     d3dpp.Windowed                   = true;
170     d3dpp.EnableAutoDepthStencil     = true;
171     d3dpp.AutoDepthStencilFormat     = D3DFMT_D24S8;
172     d3dpp.Flags                      = 0;
173     d3dpp.FullScreen_RefreshRateInHz = 0;
174     d3dpp.PresentationInterval       = D3DPRESENT_INTERVAL_IMMEDIATE;
175
176     //--------------------------------------------------------------------------------------
177     // 【Direct3D初始化四步曲之四,创设备】:创建Direct3D设备接口
178     //--------------------------------------------------------------------------------------
179     if(FAILED(pD3D->CreateDevice(D3DADAPTER_DEFAULT, D3DDEVTYPE_HAL,
180         hwnd, vp, &d3dpp, &g_pd3dDevice)))
181         return E_FAIL;
182
183     SAFE_RELEASE(pD3D) //LPDIRECT3D9接口对象的使命完成,我们将其释放掉
184
185         if(!(S_OK==Objects_Init(hwnd))) return E_FAIL;     //调用一次Objects_Init,进行渲染资源的初始化
186     return S_OK;
187 }
188
189
190 //-----------------------------------【Object_Init( )函数】--------------------------------------
191 //    描述:渲染资源初始化函数,在此函数中进行要被渲染的物体的资源的初始化
192 //--------------------------------------------------------------------------------------------------
193 HRESULT Objects_Init(HWND hwnd)
194 {
195     //创建字体
196     if(FAILED(D3DXCreateFont(g_pd3dDevice, 36, 0, 0, 1, false, DEFAULT_CHARSET,
197         OUT_DEFAULT_PRECIS, DEFAULT_QUALITY, 0, _T("微软雅黑"), &g_pFont)))
198         return E_FAIL;
199     srand(timeGetTime());      //用系统时间初始化随机种子
200
201     //--------------------------------------------------------------------------------------
202     // 【顶点缓存使用四步曲之二】:创建顶点缓存
203     //--------------------------------------------------------------------------------------
204     //创建顶点缓存
205     if( FAILED( g_pd3dDevice->CreateVertexBuffer( 6*sizeof(CUSTOMVERTEX),
206         0, D3DFVF_CUSTOMVERTEX,
207         D3DPOOL_DEFAULT, &g_pVertexBuffer, NULL ) ) )
208     {
209         return E_FAIL;
210     }
211     //--------------------------------------------------------------------------------------
212     // 【顶点缓存使用四步曲之三】:访问顶点缓存
213     //--------------------------------------------------------------------------------------
214     //顶点数据的设置,
215     CUSTOMVERTEX vertices[] =
216     {
217         //采用rand函数,给顶点以随机的颜色和位置
218         { 300.0f, 100.0f, 0.0f, 1.0f,  D3DCOLOR_XRGB(rand() % 256, rand() % 256, rand() % 256), },
219         { 500.0f, 100.0f, 0.0f, 1.0f,  D3DCOLOR_XRGB(rand() % 256, rand() % 256, rand() % 256), },
220         { 300.0f, 300.0f, 0.0f, 1.0f,  D3DCOLOR_XRGB(rand() % 256, rand() % 256, rand() % 256), },
221         { 300.0f, 300.0f, 0.0f, 1.0f,  D3DCOLOR_XRGB(rand() % 256, rand() % 256, rand() % 256), },
222         { (float)(800.0*rand()/(RAND_MAX+1.0)) , (float)(600.0*rand()/(RAND_MAX+1.0)) , 0.0f, 1.0f,  D3DCOLOR_XRGB(rand() % 256, rand() % 256, rand() % 256), },
223         { (float)(800.0*rand()/(RAND_MAX+1.0)) , (float)(600.0*rand()/(RAND_MAX+1.0)) , 0.0f, 1.0f,  D3DCOLOR_XRGB(rand() % 256, rand() % 256, rand() % 256), }
224
225     };
226
227     //填充顶点缓冲区
228     VOID* pVertices;
229     if( FAILED( g_pVertexBuffer->Lock( 0, sizeof(vertices), (void**)&pVertices, 0 ) ) )
230         return E_FAIL;
231     memcpy( pVertices, vertices, sizeof(vertices) );
232     g_pVertexBuffer->Unlock();
233
234
235     g_pd3dDevice->SetRenderState(D3DRS_CULLMODE, false);   //关掉背面消隐,无论是否顺时针,随机的那个三角形都会显示。
236     return S_OK;
237 }
238
239
240 //-----------------------------------【Direct3D_Render( )函数】-------------------------------
241 //    描述:使用Direct3D进行渲染
242 //--------------------------------------------------------------------------------------------------
243 void Direct3D_Render(HWND hwnd)
244 {
245     //--------------------------------------------------------------------------------------
246     // 【Direct3D渲染五步曲之一】:清屏操作
247     //--------------------------------------------------------------------------------------
248     g_pd3dDevice->Clear(0, NULL, D3DCLEAR_TARGET, D3DCOLOR_XRGB(0, 0, 0), 1.0f, 0);
249
250     //定义一个矩形,用于获取主窗口矩形
251     RECT formatRect;
252     GetClientRect(hwnd, &formatRect);
253
254     //--------------------------------------------------------------------------------------
255     // 【Direct3D渲染五步曲之二】:开始绘制
256     //--------------------------------------------------------------------------------------
257     g_pd3dDevice->BeginScene();                     // 开始绘制
258     //--------------------------------------------------------------------------------------
259     // 【Direct3D渲染五步曲之三】:正式绘制
260     //--------------------------------------------------------------------------------------
261     g_pd3dDevice->SetRenderState(D3DRS_SHADEMODE,D3DSHADE_GOURAUD);//设置渲染状态
262     //------------------------------------------------------------
263     // 【顶点缓存使用四步曲之四】:绘制图形
264     //------------------------------------------------------------
265     g_pd3dDevice->SetStreamSource( 0, g_pVertexBuffer, 0, sizeof(CUSTOMVERTEX) );
266     g_pd3dDevice->SetFVF( D3DFVF_CUSTOMVERTEX );
267     g_pd3dDevice->DrawPrimitive( D3DPT_TRIANGLELIST, 0, 2 );
268
269     //在窗口右上角处,显示每秒帧数
270     int charCount = swprintf_s(g_strFPS, 20, _T("FPS:%0.3f"), Get_FPS() );
271     g_pFont->DrawText(NULL, g_strFPS, charCount , &formatRect, DT_TOP | DT_RIGHT, D3DCOLOR_XRGB(255,39,136));
272
273     //--------------------------------------------------------------------------------------
274     // 【Direct3D渲染五步曲之四】:结束绘制
275     //--------------------------------------------------------------------------------------
276     g_pd3dDevice->EndScene();                       // 结束绘制
277     //--------------------------------------------------------------------------------------
278     // 【Direct3D渲染五步曲之五】:显示翻转
279     //--------------------------------------------------------------------------------------
280     g_pd3dDevice->Present(NULL, NULL, NULL, NULL);  // 翻转与显示
281 }
282
283 //-----------------------------------【Get_FPS( )函数】------------------------------------------
284 //    描述:用于计算每秒帧速率的一个函数
285 //--------------------------------------------------------------------------------------------------
286 float Get_FPS()
287 {
288     //定义四个静态变量
289     static float  fps = 0; //我们需要计算的FPS值
290     static int     frameCount = 0;//帧数
291     static float  currentTime =0.0f;//当前时间
292     static float  lastTime = 0.0f;//持续时间
293
294     frameCount++;//每调用一次Get_FPS()函数,帧数自增1
295     currentTime = timeGetTime()*0.001f;//获取系统时间,其中timeGetTime函数返回的是以毫秒为单位的系统时间,所以需要乘以0.001,得到单位为秒的时间
296
297     //如果当前时间减去持续时间大于了1秒钟,就进行一次FPS的计算和持续时间的更新,并将帧数值清零
298     if(currentTime - lastTime > 1.0f) //将时间控制在1秒钟
299     {
300         fps = (float)frameCount /(currentTime - lastTime);//计算这1秒钟的FPS值
301         lastTime = currentTime; //将当前时间currentTime赋给持续时间lastTime,作为下一秒的基准时间
302         frameCount    = 0;//将本次帧数frameCount值清零
303     }
304
305     return fps;
306 }
307
308 //-----------------------------------【Direct3D_CleanUp( )函数】--------------------------------
309 //    描述:资源清理函数,在此函数中进行程序退出前资源的清理工作
310 //---------------------------------------------------------------------------------------------------
311 void Direct3D_CleanUp()
312 {
313     //释放COM接口对象
314     SAFE_RELEASE(g_pVertexBuffer)
315     SAFE_RELEASE(g_pFont)
316     SAFE_RELEASE(g_pd3dDevice)
317 }



2. 索引缓存

三角形是Direct3D中绘制图形的基本单元,我们绘制的任何图形都是用大量的三角形组合起来的。一个正方形有4个顶点,但是要绘制它要需要2个三角形,6个顶点。说白了,当物体模型复杂,顶点数量很大时,仅使用顶点缓存绘制图形会使重复的顶点大大增加,为了解决这个问题才出现了索引缓存。索引用于记录缓存中每一个顶点的索引位置。通过索引查找对应的顶点,以完成图形的绘制。以上一节中的正方形为例子:

可以看到这里用了4个顶点和6个索引来描述一个正方形。

使用顶点缓存配合索引缓存绘制图形时所用到的函数是 IDirect3DDevice9::DrawIndexedPrimitive:

HRESULT DrawIndexedPrimitive(
  [in] D3DPRIMITIVETYPE Type,        //和DrawPrimitive方法中的第一个参数一样,表示图元类型
  [in] INT              BaseVertexIndex,  //将要进行绘制的索引缓存的起始顶点的索引位置
  [in] UINT             MinIndex,      //数组中最小的索引值,通常为0
  [in] UINT             NumVertices,    //需要的顶点个数
  [in] UINT             StartIndex,    //从索引的第几个索引处开始绘制我们的图元
  [in] UINT             PrimitiveCount  //要绘制的图元个数
);

来张图说明一下:

顶点缓存和索引缓存使用四步曲之一:设计顶点格式

顶点缓存和索引缓存使用四步曲之二:创建顶点缓存以及索引缓存

在Direct3D中,顶点缓存由IDirect3DVertexBuffer9接口对象来表示,而索引对象则由IDirect3DIndexBuffer来表示。所以这一步要做的是定义指向这两个接口的指针:

然后在CreateVertexBuffer方法和CreateIndexBuffer方法中创建点缓存和索引缓存,把内存地址分别复制给我们创建的这连个指针。关于CreateIndexBuffer:

HRESULT CreateIndexBuffer(
  [in]          UINT                  Length,  //索引缓存的大小,以字节为单位
  [in]          DWORD                 Usage,   //指定使用缓存的一些附加属性
  [in]          D3DFORMAT             Format,  //指定索引缓存中存储每个索引的大小
  [in]          D3DPOOL               Pool,   //指定索引缓存是存储在内存中还是显存中
  [out, retval] IDirect3DIndexBuffer9 **ppIndexBuffer,  //对这个变量初始化,得到创建好的索引缓存的地址,我们拿这个指针来操作索引缓存
  [in]          HANDLE                *pSharedHandle   //设为NULL或0即可
);

(吐槽:看到这里,我发现Windows游戏编程中的函数设计和我学java中的函数设计思路真的是完全不一样,前者在传入的参数方面分为输入和输出,根据输入的参数([in])进行加工得到的结果赋给输出([out])参数,以供我们后续使用,函数的返回值往往表示这个方法是否执行成功。而java中的函数,参数大部分都是作为纯输入,然后取执行的返回值作为执行结果,很少把执行结果直接传给函数参数的,至于这其中的设计差异原因现在还不是很明白,希望有一天自己能到设计层面上的那个水平)。

顶点缓存和索引缓存使用四步曲之三:访问顶点缓存以及索引缓存

先加锁,然后取缓存首地址,通过该地址向索引缓存中写入索引信息,最后解锁。在这里,和访问顶点缓存一样,也有来年各种方式。第一种是直接在Lock和UnLock之间对每个索引的数据进行赋值和修改:

第二种方式是实现准备好索引数据的数组,然后在Lock和UnLock之间用下memcpy函数进行数组内容拷贝就可以了:

顶点缓存和索引缓存使用四步曲之四:绘制图形

步骤是:调用IDirect3DDevice9::SetStreamSource把包含几何体信息的顶点缓存和渲染流水线相关联 ——> IDirect3DDevice9::SetFVF指定我们使用的灵活顶点格式 ——>  IDirect3DDevice9::SetIndices设置索引缓存 ——> 绘制函数DrawIndexedPrimitive

SetIndices函数:

HRESULT SetIndices(
  [in] IDirect3DIndexBuffer9 *pIndexData
);

DrawIndexedPrimitive函数:

HRESULT DrawIndexedPrimitive(
  [in] D3DPRIMITIVETYPE Type,        //和DrawPrimitive方法中的第一个参数一样,表示图元类型
  [in] INT              BaseVertexIndex,  //将要进行绘制的索引缓存的起始顶点的索引位置
  [in] UINT             MinIndex,      //数组中最小的索引值,通常为0
  [in] UINT             NumVertices,    //需要的顶点个数
  [in] UINT             StartIndex,    //从索引的第几个索引处开始绘制我们的图元
  [in] UINT             PrimitiveCount  //要绘制的图元个数
);


示例程序:

  1 //-----------------------------------【头文件包含部分】---------------------------------------
  2 #include <d3d9.h>
  3 #include <d3dx9.h>
  4 #include <tchar.h>
  5
  6 //-----------------------------------【库文件包含部分】---------------------------------------
  7 #pragma comment(lib,"d3d9.lib")
  8 #pragma comment(lib,"d3dx9.lib")
  9
 10 //-----------------------------------【宏定义部分】--------------------------------------------
 11 #define WINDOW_WIDTH    800                            //为窗口宽度定义的宏,以方便在此处修改窗口宽度
 12 #define WINDOW_HEIGHT    600                            //为窗口高度定义的宏,以方便在此处修改窗口高度
 13 #define WINDOW_TITLE    L"【致我们永不熄灭的游戏开发梦想】Direct3D索引缓存的故事 示例程序"    //为窗口标题定义的宏
 14 #define SAFE_RELEASE(p) { if(p) { (p)->Release(); (p)=NULL; } }   //定义一个安全释放宏,便于后面COM接口指针的释放
 15
 16
 17 //------------------------------------------------------------------------------------------------
 18 // 【顶点缓存使用四步曲之一】:设计顶点格式
 19 //------------------------------------------------------------------------------------------------
 20 struct CUSTOMVERTEX
 21 {
 22     FLOAT x, y, z, rhw;
 23     DWORD color;
 24 };
 25 #define D3DFVF_CUSTOMVERTEX (D3DFVF_XYZRHW|D3DFVF_DIFFUSE)  //FVF灵活顶点格式
 26
 27 //-----------------------------------【全局变量声明部分】-------------------------------------
 28 //    描述:全局变量的声明
 29 //------------------------------------------------------------------------------------------------
 30 LPDIRECT3DDEVICE9                    g_pd3dDevice = NULL; //Direct3D设备对象
 31 ID3DXFont*                                g_pFont=NULL;    //字体COM接口
 32 float                                            g_FPS = 0.0f;       //一个浮点型的变量,代表帧速率
 33 wchar_t                                        g_strFPS[50];    //包含帧速率的字符数组
 34 LPDIRECT3DVERTEXBUFFER9        g_pVertexBuffer = NULL;    //顶点缓冲区对象
 35 LPDIRECT3DINDEXBUFFER9        g_pIndexBuffer  = NULL;    // 索引缓存对象
 36
 37
 38 //-----------------------------------【全局函数声明部分】-------------------------------------
 39 LRESULT CALLBACK    WndProc( HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam );//窗口过程函数
 40 HRESULT                    Direct3D_Init(HWND hwnd);         //在这个函数中进行Direct3D的初始化
 41 HRESULT                    Objects_Init(HWND hwnd);         //在这个函数中进行要绘制的物体的资源初始化
 42 VOID                            Direct3D_Render(HWND hwnd);     //在这个函数中进行Direct3D渲染代码的书写
 43 VOID                            Direct3D_CleanUp( );                //在这个函数中清理COM资源以及其他资源
 44 float                            Get_FPS();                                //计算帧数的函数
 45
 46 //-----------------------------------【WinMain( )函数】--------------------------------------
 47 int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance,LPSTR lpCmdLine, int nShowCmd)
 48 {
 49     //【1】窗口创建四步曲之一:开始设计一个完整的窗口类
 50     WNDCLASSEX wndClass = { 0 };                            //用WINDCLASSEX定义了一个窗口类
 51     wndClass.cbSize = sizeof( WNDCLASSEX ) ;            //设置结构体的字节数大小
 52     wndClass.style = CS_HREDRAW | CS_VREDRAW;    //设置窗口的样式
 53     wndClass.lpfnWndProc = WndProc;                    //设置指向窗口过程函数的指针
 54     wndClass.cbClsExtra        = 0;                                //窗口类的附加内存,取0就可以了
 55     wndClass.cbWndExtra        = 0;                            //窗口的附加内存,依然取0就行了
 56     wndClass.hInstance = hInstance;                        //指定包含窗口过程的程序的实例句柄。
 57     wndClass.hIcon=(HICON)::LoadImage(NULL,L"icon.ico",IMAGE_ICON,0,0,LR_DEFAULTSIZE|LR_LOADFROMFILE);  //本地加载自定义ico图标
 58     wndClass.hCursor = LoadCursor( NULL, IDC_ARROW );    //指定窗口类的光标句柄。
 59     wndClass.hbrBackground=(HBRUSH)GetStockObject(GRAY_BRUSH);  //为hbrBackground成员指定一个白色画刷句柄
 60     wndClass.lpszMenuName = NULL;                        //用一个以空终止的字符串,指定菜单资源的名字。
 61     wndClass.lpszClassName = L"ForTheDreamOfGameDevelop";        //用一个以空终止的字符串,指定窗口类的名字。
 62
 63     //【2】窗口创建四步曲之二:注册窗口类
 64     if( !RegisterClassEx( &wndClass ) )                //设计完窗口后,需要对窗口类进行注册,这样才能创建该类型的窗口
 65         return -1;
 66
 67     //【3】窗口创建四步曲之三:正式创建窗口
 68     HWND hwnd = CreateWindow( L"ForTheDreamOfGameDevelop",WINDOW_TITLE,                //喜闻乐见的创建窗口函数CreateWindow
 69         WS_OVERLAPPEDWINDOW, CW_USEDEFAULT, CW_USEDEFAULT, WINDOW_WIDTH,
 70         WINDOW_HEIGHT, NULL, NULL, hInstance, NULL );
 71
 72     //Direct3D资源的初始化,调用失败用messagebox予以显示
 73     if (!(S_OK==Direct3D_Init (hwnd)))
 74     {
 75         MessageBox(hwnd, _T("Direct3D初始化失败~!"), _T("浅墨的消息窗口"), 0); //使用MessageBox函数,创建一个消息窗口
 76     }
 77
 78     //【4】窗口创建四步曲之四:窗口的移动、显示与更新
 79     MoveWindow(hwnd,250,80,WINDOW_WIDTH,WINDOW_HEIGHT,true);        //调整窗口显示时的位置,使窗口左上角位于(250,80)处
 80     ShowWindow( hwnd, nShowCmd );    //调用ShowWindow函数来显示窗口
 81     UpdateWindow(hwnd);                        //对窗口进行更新,就像我们买了新房子要装修一样
 82
 83     PlaySound(L"グラン=パルスのルシ.wav", NULL, SND_FILENAME | SND_ASYNC|SND_LOOP); //循环播放背景音乐
 84
 85     //【5】消息循环过程
 86     MSG msg = { 0 };  //初始化msg
 87     while( msg.message != WM_QUIT )            //使用while循环
 88     {
 89         if( PeekMessage( &msg, 0, 0, 0, PM_REMOVE ) )   //查看应用程序消息队列,有消息时将队列中的消息派发出去。
 90         {
 91             TranslateMessage( &msg );        //将虚拟键消息转换为字符消息
 92             DispatchMessage( &msg );        //该函数分发一个消息给窗口程序。
 93         }
 94         else
 95         {
 96             Direct3D_Render(hwnd);   //进行渲染
 97         }
 98     }
 99     //【6】窗口类的注销
100     UnregisterClass(L"ForTheDreamOfGameDevelop", wndClass.hInstance);  //程序准备结束,注销窗口类
101     return 0;
102 }
103
104 //-----------------------------------【WndProc( )函数】--------------------------------------
105 //    描述:窗口过程函数WndProc,对窗口消息进行处理
106 //------------------------------------------------------------------------------------------------
107 LRESULT CALLBACK WndProc( HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam )
108 {
109     switch( message )                        //switch语句开始
110     {
111     case WM_PAINT:                        // 若是客户区重绘消息
112         Direct3D_Render(hwnd);                 //调用Direct3D渲染函数
113         ValidateRect(hwnd, NULL);        // 更新客户区的显示
114         break;                                    //跳出该switch语句
115
116     case WM_KEYDOWN:                    // 若是键盘按下消息
117         if (wParam == VK_ESCAPE)    // 如果被按下的键是ESC
118             DestroyWindow(hwnd);        // 销毁窗口, 并发送一条WM_DESTROY消息
119         break;                                    //跳出该switch语句
120
121     case WM_DESTROY:                    //若是窗口销毁消息
122         Direct3D_CleanUp();            //调用自定义的资源清理函数Game_CleanUp()进行退出前的资源清理
123         PostQuitMessage( 0 );            //向系统表明有个线程有终止请求。用来响应WM_DESTROY消息
124         break;                                    //跳出该switch语句
125
126     default:                                        //若上述case条件都不符合,则执行该default语句
127         return DefWindowProc( hwnd, message, wParam, lParam );        //调用缺省的窗口过程
128     }
129
130     return 0;                                    //正常退出
131 }
132
133 //-----------------------------------【Direct3D_Init( )函数】--------------------------------------
134 //    描述:Direct3D初始化函数,进行Direct3D的初始化
135 //------------------------------------------------------------------------------------------------
136 HRESULT Direct3D_Init(HWND hwnd)
137 {
138     //--------------------------------------------------------------------------------------
139     // 【Direct3D初始化四步曲之一,创接口】:创建Direct3D接口对象, 以便用该Direct3D对象创建Direct3D设备对象
140     //--------------------------------------------------------------------------------------
141     LPDIRECT3D9  pD3D = NULL; //Direct3D接口对象的创建
142     if( NULL == ( pD3D = Direct3DCreate9( D3D_SDK_VERSION ) ) ) //初始化Direct3D接口对象,并进行DirectX版本协商
143         return E_FAIL;
144
145     //--------------------------------------------------------------------------------------
146     // 【Direct3D初始化四步曲之二,取信息】:获取硬件设备信息
147     //--------------------------------------------------------------------------------------
148     D3DCAPS9 caps; int vp = 0;
149     if( FAILED( pD3D->GetDeviceCaps( D3DADAPTER_DEFAULT, D3DDEVTYPE_HAL, &caps ) ) )
150     {
151         return E_FAIL;
152     }
153     if( caps.DevCaps & D3DDEVCAPS_HWTRANSFORMANDLIGHT )
154         vp = D3DCREATE_HARDWARE_VERTEXPROCESSING;   //支持硬件顶点运算,我们就采用硬件顶点运算,妥妥的
155     else
156         vp = D3DCREATE_SOFTWARE_VERTEXPROCESSING; //不支持硬件顶点运算,无奈只好采用软件顶点运算
157
158     //--------------------------------------------------------------------------------------
159     // 【Direct3D初始化四步曲之三,填内容】:填充D3DPRESENT_PARAMETERS结构体
160     //--------------------------------------------------------------------------------------
161     D3DPRESENT_PARAMETERS d3dpp;
162     ZeroMemory(&d3dpp, sizeof(d3dpp));
163     d3dpp.BackBufferWidth            = WINDOW_WIDTH;
164     d3dpp.BackBufferHeight           = WINDOW_HEIGHT;
165     d3dpp.BackBufferFormat           = D3DFMT_A8R8G8B8;
166     d3dpp.BackBufferCount            = 1;
167     d3dpp.MultiSampleType            = D3DMULTISAMPLE_NONE;
168     d3dpp.MultiSampleQuality         = 0;
169     d3dpp.SwapEffect                 = D3DSWAPEFFECT_DISCARD;
170     d3dpp.hDeviceWindow              = hwnd;
171     d3dpp.Windowed                   = true;
172     d3dpp.EnableAutoDepthStencil     = true;
173     d3dpp.AutoDepthStencilFormat     = D3DFMT_D24S8;
174     d3dpp.Flags                      = 0;
175     d3dpp.FullScreen_RefreshRateInHz = 0;
176     d3dpp.PresentationInterval       = D3DPRESENT_INTERVAL_IMMEDIATE;
177
178     //--------------------------------------------------------------------------------------
179     // 【Direct3D初始化四步曲之四,创设备】:创建Direct3D设备接口
180     //--------------------------------------------------------------------------------------
181     if(FAILED(pD3D->CreateDevice(D3DADAPTER_DEFAULT, D3DDEVTYPE_HAL,
182         hwnd, vp, &d3dpp, &g_pd3dDevice)))
183         return E_FAIL;
184
185     SAFE_RELEASE(pD3D) //LPDIRECT3D9接口对象的使命完成,我们将其释放掉
186
187         if(!(S_OK==Objects_Init(hwnd))) return E_FAIL;     //调用一次Objects_Init,进行渲染资源的初始化
188
189     return S_OK;
190 }
191
192
193
194 //-----------------------------------【Object_Init( )函数】--------------------------------------
195 //    描述:渲染资源初始化函数,在此函数中进行要被渲染的物体的资源的初始化
196 //--------------------------------------------------------------------------------------------------
197 HRESULT Objects_Init(HWND hwnd)
198 {
199     //创建字体
200     if(FAILED(D3DXCreateFont(g_pd3dDevice, 36, 0, 0, 1, false, DEFAULT_CHARSET,
201         OUT_DEFAULT_PRECIS, DEFAULT_QUALITY, 0, _T("微软雅黑"), &g_pFont)))
202         return E_FAIL;
203     srand(timeGetTime());      //用系统时间初始化随机种子
204
205     //--------------------------------------------------------------------------------------
206     // 【顶点缓存、索引缓存绘图四步曲之二】:创建顶点缓存和索引缓存
207     //--------------------------------------------------------------------------------------
208     //创建顶点缓存
209     if( FAILED( g_pd3dDevice->CreateVertexBuffer( 18*sizeof(CUSTOMVERTEX),
210         0, D3DFVF_CUSTOMVERTEX,
211         D3DPOOL_DEFAULT, &g_pVertexBuffer, NULL ) ) )
212     {
213         return E_FAIL;
214     }
215     // 创建索引缓存
216     if( FAILED(     g_pd3dDevice->CreateIndexBuffer(48 * sizeof(WORD), 0,
217         D3DFMT_INDEX16, D3DPOOL_DEFAULT, &g_pIndexBuffer, NULL)) )
218     {
219         return E_FAIL;
220
221     }
222     //--------------------------------------------------------------------------------------
223     // 【顶点缓存、索引缓存绘图四步曲之三】:访问顶点缓存和索引缓存
224     //--------------------------------------------------------------------------------------
225     //顶点数据的设置,
226     CUSTOMVERTEX Vertices[17];
227     Vertices[0].x = 400;
228     Vertices[0].y = 300;
229     Vertices[0].z = 0.0f;
230     Vertices[0].rhw = 1.0f;
231     Vertices[0].color = D3DCOLOR_XRGB(rand() % 256, rand() % 256, rand() % 256);
232     for(int i=0; i<16; i++)
233     {
234         Vertices[i+1].x =  (float)(250*sin(i*3.14159/8.0)) + 400;
235         Vertices[i+1].y = -(float)(250*cos(i*3.14159/8.0)) + 300;
236         Vertices[i+1].z = 0.0f;
237         Vertices[i+1].rhw = 1.0f;
238         Vertices[i+1].color =  D3DCOLOR_XRGB(rand() % 256, rand() % 256, rand() % 256);
239     }
240
241     //填充顶点缓冲区
242     VOID* pVertices;
243     if( FAILED( g_pVertexBuffer->Lock( 0, sizeof(Vertices), (void**)&pVertices, 0 ) ) )
244         return E_FAIL;
245     memcpy( pVertices, Vertices, sizeof(Vertices) );
246     g_pVertexBuffer->Unlock();
247
248     //索引数组的设置
249     WORD Indices[] ={ 0,1,2, 0,2,3, 0,3,4, 0,4,5, 0,5,6, 0,6,7, 0,7,8, 0,8,9, 0,9,10, 0,10,11 ,0,11,12, 0,12,13 ,0,13,14 ,0,14,15 ,0,15,16, 0, 16,1 };
250
251     // 填充索引数据
252     WORD *pIndices = NULL;
253     g_pIndexBuffer->Lock(0, 0, (void**)&pIndices, 0);
254     memcpy( pIndices, Indices, sizeof(Indices) );
255     g_pIndexBuffer->Unlock();
256
257     return S_OK;
258 }
259
260
261 //-----------------------------------【Direct3D_Render( )函数】-------------------------------
262 //    描述:使用Direct3D进行渲染
263 //--------------------------------------------------------------------------------------------------
264 void Direct3D_Render(HWND hwnd)
265 {
266     //--------------------------------------------------------------------------------------
267     // 【Direct3D渲染五步曲之一】:清屏操作
268     //--------------------------------------------------------------------------------------
269     g_pd3dDevice->Clear(0, NULL, D3DCLEAR_TARGET, D3DCOLOR_XRGB(0, 0, 0), 1.0f, 0);
270
271     //定义一个矩形,用于获取主窗口矩形
272     RECT formatRect;
273     GetClientRect(hwnd, &formatRect);
274
275     //--------------------------------------------------------------------------------------
276     // 【Direct3D渲染五步曲之二】:开始绘制
277     //--------------------------------------------------------------------------------------
278     g_pd3dDevice->BeginScene();                     // 开始绘制
279     //--------------------------------------------------------------------------------------
280     // 【Direct3D渲染五步曲之三】:正式绘制
281     //--------------------------------------------------------------------------------------
282     // 设置渲染状态
283     g_pd3dDevice->SetRenderState(D3DRS_SHADEMODE,D3DSHADE_GOURAUD);//这句代码可省略,因为高洛德着色模式为D3D默认的着色模式
284     //-------------------------------------------------------------------
285     // 【顶点缓存、索引缓存绘图四步曲之四】:绘制图形
286     //--------------------------------------------------------------------
287     g_pd3dDevice->SetStreamSource( 0, g_pVertexBuffer, 0, sizeof(CUSTOMVERTEX) );//把包含的几何体信息的顶点缓存和渲染流水线相关联
288     g_pd3dDevice->SetFVF( D3DFVF_CUSTOMVERTEX );//指定我们使用的灵活顶点格式的宏名称
289     g_pd3dDevice->SetIndices(g_pIndexBuffer);//设置索引缓存
290     g_pd3dDevice->DrawIndexedPrimitive(D3DPT_TRIANGLELIST, 0, 0, 17, 0, 16);//利用索引缓存配合顶点缓存绘制图形
291
292     //在窗口右上角处,显示每秒帧数
293     int charCount = swprintf_s(g_strFPS, 20, _T("FPS:%0.3f"), Get_FPS() );
294     g_pFont->DrawText(NULL, g_strFPS, charCount , &formatRect, DT_TOP | DT_RIGHT, D3DCOLOR_XRGB(255,39,136));
295
296     //--------------------------------------------------------------------------------------
297     // 【Direct3D渲染五步曲之四】:结束绘制
298     //--------------------------------------------------------------------------------------
299     g_pd3dDevice->EndScene();                       // 结束绘制
300     //--------------------------------------------------------------------------------------
301     // 【Direct3D渲染五步曲之五】:显示翻转
302     //--------------------------------------------------------------------------------------
303     g_pd3dDevice->Present(NULL, NULL, NULL, NULL);  // 翻转与显示
304 }
305
306 //-----------------------------------【Get_FPS( )函数】------------------------------------------
307 //    描述:用于计算每秒帧速率的一个函数
308 //--------------------------------------------------------------------------------------------------
309 float Get_FPS()
310 {
311     //定义四个静态变量
312     static float  fps = 0; //我们需要计算的FPS值
313     static int     frameCount = 0;//帧数
314     static float  currentTime =0.0f;//当前时间
315     static float  lastTime = 0.0f;//持续时间
316
317     frameCount++;//每调用一次Get_FPS()函数,帧数自增1
318     currentTime = timeGetTime()*0.001f;//获取系统时间,其中timeGetTime函数返回的是以毫秒为单位的系统时间,所以需要乘以0.001,得到单位为秒的时间
319
320     //如果当前时间减去持续时间大于了1秒钟,就进行一次FPS的计算和持续时间的更新,并将帧数值清零
321     if(currentTime - lastTime > 1.0f) //将时间控制在1秒钟
322     {
323         fps = (float)frameCount /(currentTime - lastTime);//计算这1秒钟的FPS值
324         lastTime = currentTime; //将当前时间currentTime赋给持续时间lastTime,作为下一秒的基准时间
325         frameCount    = 0;//将本次帧数frameCount值清零
326     }
327
328     return fps;
329 }
330
331 //-----------------------------------【Direct3D_CleanUp( )函数】--------------------------------
332 //    描述:资源清理函数,在此函数中进行程序退出前资源的清理工作
333 //---------------------------------------------------------------------------------------------------
334 void Direct3D_CleanUp()
335 {
336     //释放COM接口对象
337     SAFE_RELEASE(g_pIndexBuffer)
338     SAFE_RELEASE(g_pVertexBuffer)
339     SAFE_RELEASE(g_pFont)
340     SAFE_RELEASE(g_pd3dDevice)
341 }

时间: 2024-10-13 06:24:56

《逐梦旅程 WINDOWS游戏编程之从零开始》笔记6——Direct3D中的顶点缓存和索引缓存的相关文章

《逐梦旅程 WINDOWS游戏编程之从零开始》笔记5——Direct3D编程基础

第11章 Direct3D编程基础 2D游戏是贴图的艺术,3D游戏是渲染的艺术.这句话在我学过了之前的GDI编程之后,前一句算是有所体会,现在是来理解后一句的时候了. 安装DirectX SDK配置啥的就不说了,直接进入正题,先来个典型的Direct3D程序框架图: 主要分为5个部分: 创建一个Windows窗口 Direct3D的初始化 消息循环 渲染图形 结束应用程序,清除在初始化阶段锁创建的COM对象,退出程序 至于COM (Component Object Model, 组件对象模型)

《逐梦旅程 WINDOWS游戏编程之从零开始》笔记7——四大变换

第13章 世界变换,取景变换,投影变换,视口变换 在Direct3D中,如果为进行任何空间坐标变换而直接绘图的话,图形将始终处于应用程序窗口的中心位置,默认这个位置就成为世界坐标系的原点(0,0,0).而且我们也不能改变观察图形的视角方向.默认情况下的观察方向是世界坐标系的z轴正向方向. 世界变换运算是为了能在世界空间中的指定位置来绘制图形 取景变换运算是为了以不同的视角观察图形 投影变换为了将相对较远的图形投影到同一个平面上并体现出"近大远小"的真实视觉效果 视口变换是为了控制显示图

《逐梦旅程 WINDOWS游戏编程之从零开始》笔记10——三维天空的构建&amp;三维粒子的实现&amp;多游戏模型的载入

第23章 三维天空的构建 目前描述三维天空的技术主要包括三种类型,直接来介绍使用最广泛的模拟技术,详细的描述可以见作者的博文. 天空盒(Sky Box),即放到场景的是一个立方体.它是目前使用最广泛的三维天空模拟技术,网络上素材丰富,所以这次就用教大家用天空盒来模拟三维天空.天空盒经常是由24个顶点.六个面组成的立方体(或者直接从做好的X模型文件载入天空盒),并经常会随着视点的移动而移动,来刻画极远处玩家无法达到位置的天空 天空盒的设计 1.准备天空盒纹理素材 天空盒的纹理自然就是我们这个天空盒

《逐梦旅程 WINDOWS游戏编程之从零开始》笔记8——载入三维模型&amp;Alpha混合技术&amp;深度测试与Z缓存

第17章 三维游戏模型的载入 主要是如何从3ds max中导出.X文件,以及如何从X文件加载三维模型到DirextX游戏程序里.因为复杂的3D物体,要用代码去实现,那太反人类了,所以我们需要一些建模软件. 对于3ds max,要到出.X文件,要装个Panda插件.然后就是作者推荐的一个3D模型资源网站:http://www.cgmodel.com/. 网格模型接口ID3DXMesh 这个接口表示网格,继承自ID3DXBaseMesh.ID3DXMesh接口中的D3DXCreateMesh()可用

《逐梦旅程 WINDOWS游戏编程之从零开始》笔记7——DirectInput&amp;纹理映射

第15章 DirectInput接口 DirectInput作为DirectX的组件之一,依然是一些COM对象的集合.DirectInput由IDirectinput8.IDirectInputDevice8和IDirectInputEffect这3个接口组成.其中IDirectInput8作为DirectInput API中最主要的接口,用于初始化系统以及创建输入设备接口,DirectInput中其他所有接口都需要依赖于我们的IDirectInput8之上,都是通过这个接口进行查询的.而Dir

《逐梦旅程 WINDOWS游戏编程之从零开始》笔记8——光照与材质

第14章 绘制出质感的世界--光照与材质 1. 光照与光源 在Direct3D中的光源类型和光照类型是不同的两个概念,光照模型描述的是光线的反射特征,而光源类型主要强调的是能够产生这些光照模型的方式以及光线的位置,方向,强度等特征. 四大光照类型 环境光:基于整个自然界环境的整体亮度,称为环境光或者背景光,没有位置或者方向上的特征,只有一个颜色亮度值,不会衰减,在所有方向和所有物体表面上投射的环境光的数量是恒定不变的(有点像我们白天的自然光).在Direc3D中设置环境光可以直接使用setRen

《逐梦旅程 WINDOWS游戏编程之从零开始》笔记9——游戏摄像机&amp;三维地形的构建

第21章 游戏摄像机的构建 之前的程序示例,都是通过封装的DirectInput类来处理键盘和鼠标的输入,对应地改变我们人物模型的世界矩阵来达到移动物体,改变观察点的效果.其实我们的观察方向乃至观察点都是没有变的,变的只是我们3D人物的位置.说白了就是用D3DXMatrixLookAtLH在资源初始化时固定住视角,在程序运行过程中接收到消息并改变三维人物模型的世界矩阵而已.这章的主要内容就是创建出一个可以在三维空间中自由移动的摄像机类,我们准备给这个摄像机类取名为CameraClass. 设计摄

《逐梦旅程 WINDOWS游戏编程之从零开始》读书笔记1——创建窗口

步骤: 窗口类的设计 窗口类的注册 窗口的正式创建 窗口的显示与更新 1. 设计:使用WNDCLASSEX结构体,这里注意的是C++中的结构体中的成员默认是共有的,所以可以直接通过 . 来调用. typedef struct tagWNDCLASSEX { UINT cbSize; //UINT类型的cbSize,表示该结构体的字节数大小 UINT style; //指定窗口的风格样式 WNDPROC lpfnWndProc; //指向窗口过程函数的函数指针 int cbClsExtra; //

《逐梦旅程 WINDOWS游戏编程之从零开始》源码分析2——GDI

GDI: 图形设备接口 1. 取得设备环境的句柄(如屏幕) 使用BeginPaint和EndPaint这两个函数,或者使用GetDC和ReleaseDC这两个函数.关于函数的具体说明可以参考mdsn文档. 一个GDI程序通用框架: 1 #include <windows.h> 2 3 #define WINDOW_WIDTH 800 //为窗口宽度定义的宏,以方便在此处修改窗口宽度 4 #define WINDOW_HEIGHT 600 //为窗口高度定义的宏,以方便在此处修改窗口高度 5 #