靈活頂點格式
靈活頂點格式(Flexible Vertex Format, FVF)用來描述在頂點緩沖區中的頂點存儲格式中包含了哪些屬性。Direct3D應用程序可以用幾種不同的方式定義靈活頂點格式。靈活頂點格式使應用程序只使用它需要的頂點數據,排除那些它不需要的組成成分。這樣,應用程序可以節省內存空間,減少系統帶寬。通過D3DFVF的組合,可以描述圖元頂點的格式。靈活頂點格式指定的格式包括點的大小,用D3DFVF_PSIZE指定,該大小在投影機空間用來表示未經變換的頂點,在設備空間用來表示經過變換的頂點。
接口IDirect3DDevice9的渲染方法能夠接受這些標志的組合,并用它們來決定如何渲染圖元。這些標志告訴系統應用程序所使用的頂點的組成,包括頂點的位置、頂點混合權重、法向量、顏色和紋理坐標的格式和數量,以及向Direct3D申請何種渲染流水線。另外,提交或撤銷一個具體的頂點格式標志會告訴系統哪些頂點組成單元還留在系統中,哪些已經被忽略了。
紋理坐標可用不同的格式聲明,紋理可以由一個坐標尋址,也可以由多達4個紋理坐標來尋址。使用宏集合D3DFVF_TEXCOORDSIZEn可以創建位模式,定義要使用的頂點格式所需的紋理坐標格式。
為了使用索引頂點混合,標志D3DFVF_LASTBETA_UBYTE4應該追加到頂點靈活格式中去。這個標志的出現表明第5個混合權重將被當作一個DWORD值,而不是一個浮點值。
渲染狀態
設備渲染狀態控制Direct3D設備的光柵化組件的行為。通過改變渲染狀態屬性,可以控制使用何種著色模式,如何進行霧化及其他光柵化操作。Direct3D編程很大一部分工作就是設置合適的渲染狀態。
Direct3D圖形程序通過調用IDirect3DDevice9::SetRenderState()函數來設置渲染狀態。枚舉類型D3DRENDERSTATETYPE列舉出所有可能的渲染狀態。應用程序將D3DRENDERSTATETYPE類型的某一個枚舉值作為第一個參數傳遞給函數SetRenderState(),然后用第二個參數指定相應的渲染狀態。
著色模式
Direct3D中的物體表面是由許許多多的多邊形構成的。當渲染一個物體的多邊形時,不同的著色模式在其表面產生不同的效果。著色模式決定了多邊形上每個點的顏色和光照的強度。Direct3D提供兩種著色模式:平面著色模式(FLAT)和戈勞德著色模式(GOURAUD)。
(1)平面著色模式
在平面著色模式下,Direct3D渲染一個多邊形時,把多邊形第一個頂點的顏色作為整個多邊形的顏色進行著色,也就是說,在平面著色模式下,一個多邊形內的所有像素的顏色都等于該多邊形第一個頂點的顏色。如果這些多邊形不共面,用平面著色模式渲染的三維圖形將出現明顯的陡沿。平面著色模式是渲染速度最快的著色模式。
(2)戈勞德著色模式
當用戈勞德著色渲染一個多邊形時,它會先用頂點法向量和燈光參數計算每個頂點的顏色。然后,在該多邊形的表面上進行線性插值,進而得到每個像素的顏色。例如,頂點1顏色的紅色值為0.8,頂點2顏色的紅色至為0.4,使用戈勞德著色模式和RGB顏色模式,Direct3D燈光組件將使這兩點連線上中點的顏色的紅色值為0.6。
(3)著色模式比較
在平面著色模式中,相鄰兩個面之間會有明顯的邊緣,而采用戈勞德著色模式時,邊緣處的著色值會由內插運算產生,因而最后會得到一個彎曲的表面。在戈勞德著色模式下,對平面光照的處理要比在平面著色模式下更真實。在平面著色模式下,一個面的顏色是唯一的,但戈勞德著色模式可以使光線更準確地照射在每一個面上。當一個表面距離一個點光源很近時,它們的區別將會更明顯地表現出來。
戈勞德著色模式可以使在平面著色模式下多邊形間的陡沿變得平滑。然而這樣可能會導致馬赫帶效應(Mach
bands)的產生,也就是相鄰的顏色或光線帶之間不能很平滑的相互融合。對于程序開發人員來說,可以通過增加構成對象的多邊形的數目來降低馬赫帶效應,當然也可以通過提高屏幕分辨率或者增加程序的顏色深度來達到目的。
(4)設置著色模式
Direct3D一次只能選擇一種著色模式。默認情況下,選擇戈勞德著色模式。在Direct3D圖形程序中,調用IDirect3DDevice9::
SetRenderState()方法來改變著色模式。設置狀態參數為D3DRS_SHADEMODE,該狀態參數值必須是D3DSHADEMODE枚舉成員的一個。
Defines constants that describe the supported shading
modes.
typedef enum D3DSHADEMODE
{
D3DSHADE_FLAT = 1,
D3DSHADE_GOURAUD = 2,
D3DSHADE_PHONG = 3,
D3DSHADE_FORCE_DWORD = 0x7fffffff,
} D3DSHADEMODE, *LPD3DSHADEMODE;
Constants
- D3DSHADE_FLAT
- Flat shading mode. The color and specular
component of the first vertex in the triangle are used to determine the
color and specular component of the face. These colors remain constant
across the triangle; that is, they are not interpolated. The specular alpha
is interpolated. See Remarks.
- D3DSHADE_GOURAUD
- Gouraud shading mode. The color and specular
components of the face are determined by a linear interpolation between all
three of the triangle's vertices.
- D3DSHADE_PHONG
- Not supported.
- D3DSHADE_FORCE_DWORD
- Forces this enumeration to compile to 32 bits in
size. Without this value, some compilers would allow this enumeration to
compile to a size other than 32 bits. This value is not used.
Remarks
The first vertex of a triangle for flat shading mode is
defined in the following manner.
- For a triangle list, the first vertex of the
triangle i is i * 3.
- For a triangle strip, the first vertex of the
triangle i is vertex i.
- For a triangle fan, the first vertex of the
triangle i is vertex i + 1.
The members of this enumerated type define the vales
for the D3DRS_SHADEMODE render state.
多邊形填充模式
默認狀態下,Direct3D會把渲染好的多邊形面的圖像繪制出來,這種方法適合于絕大多數情況。然而,有時為了調試程序或其他特殊目的,可能只需繪制出多邊形的頂點或邊。這可以通過設置渲染狀態D3DRS_FILLMODE來實現,通過為D3DRS_FILLMODE渲染狀態指定枚舉類型D3DFILLMODE中的一個數值,來選擇多邊形填充模式。
Defines constants describing the fill mode.
typedef enum D3DFILLMODE
{
D3DFILL_POINT = 1,
D3DFILL_WIREFRAME = 2,
D3DFILL_SOLID = 3,
D3DFILL_FORCE_DWORD = 0x7fffffff,
} D3DFILLMODE, *LPD3DFILLMODE;
Constants
- D3DFILL_POINT
- Fill points.
- D3DFILL_WIREFRAME
- Fill wireframes.
- D3DFILL_SOLID
- Fill solids.
- D3DFILL_FORCE_DWORD
- Forces this enumeration to compile to 32 bits in
size. Without this value, some compilers would allow this enumeration to
compile to a size other than 32 bits. This value is not used.
Remarks
The values in this enumerated type are used by the
D3DRS_FILLMODE render state.
著色模式和填充模式示例程序
示例程序RenderState演示了著色模式和填充模式對圖形顯示結果的影響,在RenderState示例程序中用全局變量g_is_flat和g_fill_mode控制渲染圖形使用的著色模式和填充模式,通過單擊鼠標左鍵在兩種著色模式之間進行切換,單擊鼠標右鍵在三種填充模式之間進行切換。

Gouraud著色模式
、
FLAT著色模式

點模式

線模式
完整源代碼:
#include <d3d9.h>
#define CLASS_NAME "GameApp"
#define release_com(p) do { if(p) { (p)->Release(); (p) = NULL; } } while(0)
IDirect3D9* g_d3d;
IDirect3DDevice9* g_device;
IDirect3DVertexBuffer9* g_vertex_buffer;
bool g_is_flat = false;
DWORD g_fill_mode = D3DFILL_SOLID;
struct sCustomVertex
{
float x, y, z, rhw;
DWORD color;
};
#define D3DFVF_CUSTOM_VERTEX (D3DFVF_XYZRHW | D3DFVF_DIFFUSE)
void init_vertices()
{
sCustomVertex vertices[] =
{
{ 100.0f, 400.0f, 0.5f, 1.0f, 0xffff0000, },
{ 300.0f, 50.0f, 0.5f, 1.0f, 0xff00ff00, },
{ 500.0f, 400.0f, 0.5f, 1.0f, 0xff0000ff, },
};
// push vertex data into vertex buffer
g_device->CreateVertexBuffer(sizeof(vertices), 0, D3DFVF_CUSTOM_VERTEX, D3DPOOL_DEFAULT, &g_vertex_buffer, NULL);
void* ptr;
g_vertex_buffer->Lock(0, sizeof(vertices), (void**)&ptr, 0);
memcpy(ptr, vertices, sizeof(vertices));
g_vertex_buffer->Unlock();
}
bool init_d3d(HWND hwnd)
{
g_d3d = Direct3DCreate9(D3D_SDK_VERSION);
if(g_d3d == NULL)
return false;
D3DPRESENT_PARAMETERS d3dpp;
ZeroMemory(&d3dpp, sizeof(d3dpp));
d3dpp.Windowed = TRUE;
d3dpp.SwapEffect = D3DSWAPEFFECT_DISCARD;
d3dpp.BackBufferFormat = D3DFMT_UNKNOWN;
if(FAILED(g_d3d->CreateDevice(D3DADAPTER_DEFAULT, D3DDEVTYPE_HAL, hwnd, D3DCREATE_SOFTWARE_VERTEXPROCESSING,
&d3dpp, &g_device)))
{
return false;
}
init_vertices();
return true;
}
void cleanup()
{
release_com(g_vertex_buffer);
release_com(g_device);
release_com(g_d3d);
}
void render()
{
g_device->SetRenderState(D3DRS_SHADEMODE, g_is_flat ? D3DSHADE_FLAT : D3DSHADE_GOURAUD);
g_device->SetRenderState(D3DRS_FILLMODE, g_fill_mode);
g_device->Clear(0, NULL, D3DCLEAR_TARGET, D3DCOLOR_XRGB(5, 5, 5), 1.0f, 0);
g_device->BeginScene();
g_device->SetStreamSource(0, g_vertex_buffer, 0, sizeof(sCustomVertex));
g_device->SetFVF(D3DFVF_CUSTOM_VERTEX);
g_device->DrawPrimitive(D3DPT_TRIANGLELIST, 0, 1);
g_device->EndScene();
g_device->Present(NULL, NULL, NULL, NULL);
}
LRESULT WINAPI WinProc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam)
{
switch(msg)
{
case WM_LBUTTONDOWN:
g_is_flat = !g_is_flat;
break;
case WM_RBUTTONDOWN:
if(g_fill_mode == D3DFILL_POINT)
g_fill_mode = D3DFILL_WIREFRAME;
else if(g_fill_mode == D3DFILL_WIREFRAME)
g_fill_mode = D3DFILL_SOLID;
else if(g_fill_mode == D3DFILL_SOLID)
g_fill_mode = D3DFILL_POINT;
break;
case WM_KEYDOWN:
switch(wParam)
{
case VK_ESCAPE:
DestroyWindow(hwnd);
break;
}
break;
case WM_DESTROY:
PostQuitMessage(0);
return 0;
}
return DefWindowProc(hwnd, msg, wParam, lParam);
}
int WINAPI WinMain(HINSTANCE inst, HINSTANCE, LPSTR, INT)
{
WNDCLASSEX wc;
wc.cbSize = sizeof(WNDCLASSEX);
wc.style = CS_CLASSDC;
wc.lpfnWndProc = WinProc;
wc.cbClsExtra = 0;
wc.cbWndExtra = 0;
wc.hInstance = inst;
wc.hIcon = NULL;
wc.hCursor = NULL;
wc.hbrBackground = NULL;
wc.lpszMenuName = NULL;
wc.lpszClassName = CLASS_NAME;
wc.hIconSm = NULL;
if(! RegisterClassEx(&wc))
return -1;
HWND hwnd = CreateWindow(CLASS_NAME, "Direct3D App", WS_OVERLAPPEDWINDOW, 200, 100, 600, 500,
NULL, NULL, wc.hInstance, NULL);
if(hwnd == NULL)
return -1;
if(init_d3d(hwnd))
{
ShowWindow(hwnd, SW_SHOWDEFAULT);
UpdateWindow(hwnd);
MSG msg;
ZeroMemory(&msg, sizeof(msg));
while(msg.message != WM_QUIT)
{
if(PeekMessage(&msg, NULL, 0, 0, PM_REMOVE))
{
TranslateMessage(&msg);
DispatchMessage(&msg);
}
render();
}
}
cleanup();
UnregisterClass(CLASS_NAME, wc.hInstance);
return 0;
}