#include
#pragma warning( disable : 4996 ) // disable deprecated warning
#include
#pragma warning( default : 4996 )
#include
bool InitD3D(HWND hWnd, bool fullscreen);
void RenderScene();
void ShutDown();
bool InitializeObjects();
LPDIRECT3D9 g_D3D = NULL;
LPDIRECT3DDEVICE9 g_D3DDevice = NULL;
LPDIRECT3DVERTEXBUFFER9 g_VertexBuffer = NULL; //顶点缓存接口
LPDIRECT3DINDEXBUFFER9 g_pIndexBuf = NULL; //索引缓存接口
struct CUSTOMVERTEX
{
FLOAT _x, _y, _z; // �������
DWORD _color; // ��������ɫ
CUSTOMVERTEX(FLOAT x, FLOAT y, FLOAT z, DWORD color)
: _x(x), _y(y), _z(z), _color(color) {}
};
#define D3DFVF_VERTEX (D3DFVF_XYZ | D3DFVF_DIFFUSE)
LRESULT WINAPI MsgProc(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam)
{
switch (msg)
{
case WM_DESTROY:
PostQuitMessage(0);
return 0;
break;
case WM_KEYUP:
if (wParam == VK_ESCAPE)
PostQuitMessage(0);
break;
default:
break;
}
return DefWindowProc(hWnd, msg, wParam, lParam);
}
int WINAPI WinMain(HINSTANCE hInst, HINSTANCE prevInst, LPSTR cmdLine, int show)
{
WNDCLASSEX wc = { sizeof(WNDCLASSEX), CS_CLASSDC, MsgProc, 0, 0, hInst, NULL, NULL, NULL, NULL, L"UGPDX", NULL };
RegisterClassEx(&wc);
HWND hWnd = CreateWindow(L"UGPDX", L"Blank D3D Window", WS_OVERLAPPEDWINDOW, 100, 100, 640, 480, NULL, NULL, wc.hInstance, NULL);
if (InitD3D(hWnd, false))
{
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);
}
else
{
RenderScene();
}
}
}
ShutDown();
UnregisterClass(L"UGPDX", wc.hInstance);
return 0;
}
bool InitD3D(HWND hWnd, bool fullscreen)
{
D3DDISPLAYMODE displayMode;
g_D3D = Direct3DCreate9(D3D9b_SDK_VERSION);
if (g_D3D == NULL)
return false;
if (FAILED(g_D3D->GetAdapterDisplayMode(D3DADAPTER_DEFAULT, &displayMode)))
return false;
D3DPRESENT_PARAMETERS d3dpp;
ZeroMemory(&d3dpp, sizeof(d3dpp));
if (fullscreen)
{
d3dpp.Windowed = FALSE;
d3dpp.BackBufferWidth = 640;
d3dpp.BackBufferHeight = 480;
}
else
d3dpp.Windowed = TRUE;
d3dpp.SwapEffect = D3DSWAPEFFECT_DISCARD;
d3dpp.BackBufferFormat = displayMode.Format;
if (FAILED(g_D3D->CreateDevice(D3DADAPTER_DEFAULT, D3DDEVTYPE_HAL, hWnd, D3DCREATE_SOFTWARE_VERTEXPROCESSING, &d3dpp, &g_D3DDevice)))
return false;
if (!InitializeObjects()) return false;
return true;
}
void RenderScene()
{
g_D3DDevice->Clear(0, NULL, D3DCLEAR_TARGET, D3DCOLOR_XRGB(0, 0, 0), 1.0f, 0);
g_D3DDevice->BeginScene();
// ���������任����
D3DXMATRIX matWorld, Rx, Ry, Rz;
D3DXMatrixIdentity(&matWorld); // �����
D3DXMatrixRotationX(&Rx, ::timeGetTime() / 1000.0f); // ��X����ת
D3DXMatrixRotationY(&Ry, ::timeGetTime() / 1000.0f); // ��Y����ת
D3DXMatrixRotationZ(&Rz, ::timeGetTime() / 1000.0f); // ��Z����ת
matWorld = Rx * Ry * Rz * matWorld; // �õ����յ����Ͼ���
g_D3DDevice->SetTransform(D3DTS_WORLD, &matWorld);
// ����ȡ���任����
D3DXMATRIX matView;
D3DXVECTOR3 vEye(0.0f, 0.0f, -30.0f);
D3DXVECTOR3 vAt(0.0f, 0.0f, 0.0f);
D3DXVECTOR3 vUp(0.0f, 1.0f, 0.0f);
D3DXMatrixLookAtLH(&matView, &vEye, &vAt, &vUp);
g_D3DDevice->SetTransform(D3DTS_VIEW, &matView);
// ����ͶӰ�任����
D3DXMATRIX matProj;
D3DXMatrixPerspectiveFovLH(&matProj, D3DX_PI / 4.0f, 1.0f, 1.0f, 1000.0f);
g_D3DDevice->SetTransform(D3DTS_PROJECTION, &matProj);
// ������Ⱦ״̬
g_D3DDevice->SetRenderState(D3DRS_LIGHTING, FALSE);
g_D3DDevice->SetRenderState(D3DRS_CULLMODE, D3DCULL_CCW);
g_D3DDevice->SetStreamSource(0, g_VertexBuffer, 0, sizeof(CUSTOMVERTEX));
g_D3DDevice->SetFVF(D3DFVF_VERTEX);
g_D3DDevice->SetIndices(g_pIndexBuf);
g_D3DDevice->DrawIndexedPrimitive(D3DPT_TRIANGLELIST, 0, 0, 8, 0, 12);
g_D3DDevice->EndScene();
g_D3DDevice->Present(NULL, NULL, NULL, NULL);
}
void ShutDown()
{
if (g_D3DDevice != NULL)
g_D3DDevice->Release();
if (g_D3D != NULL)
g_D3D->Release();
if (g_VertexBuffer != NULL)
g_VertexBuffer->Release();
if (g_pIndexBuf != NULL)
g_pIndexBuf->Release();
g_D3D = NULL;
g_D3DDevice = NULL;
g_VertexBuffer = NULL;
g_pIndexBuf = NULL;
}
bool InitializeObjects()
{
//创建顶点缓存,填充顶点数据
if (FAILED(g_D3DDevice->CreateVertexBuffer(8 * sizeof(CUSTOMVERTEX), 0, D3DFVF_VERTEX, D3DPOOL_DEFAULT, &g_VertexBuffer, NULL)))
return false;
CUSTOMVERTEX *pVertices = NULL;
if (FAILED(g_VertexBuffer->Lock(0, 0, (void**)&pVertices, 0)))
return false;
pVertices[0] = CUSTOMVERTEX(-5.0f, 5.0f, -5.0f, D3DCOLOR_XRGB(255, 0, 0));
pVertices[1] = CUSTOMVERTEX(-5.0f, 5.0f, 5.0f, D3DCOLOR_XRGB(0, 255, 0));
pVertices[2] = CUSTOMVERTEX(5.0f, 5.0f, 5.0f, D3DCOLOR_XRGB(0, 0, 255));
pVertices[3] = CUSTOMVERTEX(5.0f, 5.0f, -5.0f, D3DCOLOR_XRGB(255, 255, 0));
pVertices[4] = CUSTOMVERTEX(-5.0f, -5.0f, -5.0f, D3DCOLOR_XRGB(0, 0, 255));
pVertices[5] = CUSTOMVERTEX(-5.0f, -5.0f, 5.0f, D3DCOLOR_XRGB(255, 255, 0));
pVertices[6] = CUSTOMVERTEX(5.0f, -5.0f, 5.0f, D3DCOLOR_XRGB(255, 0, 0));
pVertices[7] = CUSTOMVERTEX(5.0f, -5.0f, -5.0f, D3DCOLOR_XRGB(0, 255, 0));
g_VertexBuffer->Unlock();
//创建索引缓存
if (FAILED(g_D3DDevice->CreateIndexBuffer(36 * sizeof(WORD), 0, D3DFMT_INDEX16, D3DPOOL_DEFAULT, &g_pIndexBuf, NULL)))
return false;
WORD *pIndices = NULL;
if (FAILED(g_pIndexBuf->Lock(0, 0, (void**)&pIndices, 0)))
return false;
pIndices[0] = 0; pIndices[1] = 1; pIndices[2] = 2;
pIndices[3] = 0; pIndices[4] = 2; pIndices[5] = 3;
pIndices[6] = 0; pIndices[7] = 3; pIndices[8] = 7;
pIndices[9] = 0; pIndices[10] = 7; pIndices[11] = 4;
pIndices[12] = 0; pIndices[13] = 4; pIndices[14] = 5;
pIndices[15] = 0; pIndices[16] = 5; pIndices[17] = 1;
pIndices[18] = 2; pIndices[19] = 6; pIndices[20] = 7;
pIndices[21] = 2; pIndices[22] = 7; pIndices[23] = 3;
pIndices[24] = 2; pIndices[25] = 5; pIndices[26] = 6;
pIndices[27] = 2; pIndices[28] = 1; pIndices[29] = 5;
pIndices[30] = 4; pIndices[31] = 6; pIndices[32] = 5;
pIndices[33] = 4; pIndices[34] = 7; pIndices[35] = 6;
g_pIndexBuf->Unlock();
return true;
}
【ZT】D3DFVF_XYZ和D3DFVF_XYZRHW的区别
D3DFVF_XYZ和D3DFVF_XYZRHW有什么区别?以前好像没有仔细思考过,只是见到Beginning DirectX9中如是说:
The RHW value, which stands for Reciprocal of Homogeneous W[1], tells Direct3D that the vertices that
are being used are already in screen coordinates. This value is normally used in fog and clipping
calculations and should be set to 1.0.
今天,做了个实验得知,在顶点结构体中没有RHW时,Direct3D将执行视、投影、世界等变换以及进行光线计算,
之后你才能在窗口中得到你所绘制的物体。当顶点结构体中有RHW时,就像上面那段英文所述,告知Direct3D使
用的顶点已经在屏幕坐标系中了,不再执行视图、投影、世界等变换和光线计算,因为D3DFVF_XYZRHW标志告诉
它顶点已经经过了这些处理,并直接将顶点进行光栅操作,任何用SetTransform进行的转换都对其无效。不过
这时的原点就在客户区的左上角了,其中x向右为正,y向下为正,而z的意义已经变为z-buffer的象素深度。
值得注意的是,D3DFVF_XYZRHW和D3DFVF_XYZ、D3DFVF_NORMAL不能共存,因为后两个标志与前一个矛盾。在使用
这种顶点时,系统需要顶点的位置已经经过变换了,也就是说x、y必须在屏幕坐标系中,z必须是z-buffer中
的象素深度,取值范围:0.0-1.0,离观察者最近的地方为0.0,观察范围内最远可见的地方为1.0。(不过我测
试的时候似乎z值不起作用。)
If you use D3DFVF_XYZ, then your vertex format needs to have 3 floats in it, for x, y and z. Those are used
to define a vertex position in 3D space.If you use D3DFVF_XYZRHW, then your vertex format needs to have 4
floats in it, for x, y, z and rhw. X and Y are used to define a vertex position in 2D space, Z is ignored
(I think, it may be used for fog and such, but I don’t recall just now - I always set it to 0.0f), and rhw
is the Reciprocal of Homogenous W - which is basically 1 / the depth of the vertex.
Usually, you use D3DFVF_XYZRHW for doing 2D, and D3DFVF_XYZ any other time. However, a lot of people just
use D3DFVF_XYZ, and use an orthoganal projection matrix to make it seem 2D.
[1] RHW表示投影空间中顶点所在的齐次点(x,y,z,w)(homogeneous point)的w坐标的倒数(reciprocal)。