【笔记】《DirectX 9.0 3D游戏开发编程基础》:Direct3D初始化

Direct3D初始化大概分为4个步骤:

1.获取接口IDirect3D9的指针。(Direct3DCreate9函数调用)。

     该接口用户获取系统中物理硬件设备的信息并创建接口IDirect3DDevice9,此接口是一个C++对象,代表显示3D图形的物理硬件设备。

2.检查设备性能(D3DCAPS9结构体),判断主显卡是否支持某些特性,比如是否支持顶点运算。创建IDirect3DDevice9之前,必须确定主显卡支持的特性。

3.初始化D3DPRESENT_PARAMETER结构的一个实例,通过设置该结构体的成员变量来指定即将创建的借口IDirect3DDevice9的特性。

4.利用已经初始化的D3DPRESENT_PARAMETER结构创建IDirect3DDevice9对象。

 

以下是DirectX 9.0 3D游戏开发编程基础的基本程序框架(包含初始化的过程):

d3dUtility.h

 InitD3D:该函数对应用程序主窗口进行了初始化(windows程序设计的基础,RegisterClass,CreateWindow,ShowWindow,UpdateWindow);

             然后执行Direct3D初始化过程,返回一个指向已经创建好的IDirect3DDevice9接口的指针。

EnterMsgLoop:封装了消息循环,参数为一个指向显示函数的函数指针。空闲处理期间显示场景。

Release:模板函数是方便地释放COM接口并将其设置为NULL。

Delete:模板函数是方便地删除自由堆对象,将其指针设置为NULL。

WndProc:主窗口的窗口过程函数,针对不同类型的消息进行相对应的相应。

#ifndef D3DUTILITY_H
#define D3DUTILITY_H

#include <d3dx9.h>
#include <windows.h>
#include <tchar.h>


namespace d3d {

        bool InitD3D( HINSTANCE hInstance, int width, int height, bool windowed, D3DDEVTYPE deviceType, IDirect3DDevice9 **device);
//[in] 应用实例 [in] 后台缓存的宽 [in]后台缓存的高 [in] 是否窗口或者全屏 [in] 设备类型 HAL或REF [out] 初始化成功的IDirect3DDevice指针 int EnterMsgLoop( bool (*ptr_display)(float timeDelta) );
LRESULT CALLBACK WndProc( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam); template<class T> void Release(T t) { if( t ) { t->Release(); t = 0; } } template<class T> void Delete(T t) { if( t ) { delete t; t = 0; } } } #endif

d3dUtility.cpp

上述接口的具体实现部分。

#include "d3dUtility.h"

namespace d3d {

	bool InitD3D( HINSTANCE hInstance, int width, int height, bool windowed, D3DDEVTYPE deviceType, IDirect3DDevice9 **device) {
HWND hwnd; //窗口句柄
WNDCLASS wndclass; //窗口类别,在CreateWindow之前需要RegisterClass , 针对WNDCLASS的参数可以了解一下API wndclass.style = CS_HREDRAW | CS_VREDRAW; wndclass.lpfnWndProc = (WNDPROC)d3d::WndProc; //WndProc wndclass.cbClsExtra = 0; wndclass.cbWndExtra = 0; wndclass.hInstance = hInstance; wndclass.hIcon = LoadIcon (NULL, IDI_APPLICATION) ; wndclass.hCursor = LoadCursor (NULL, IDC_ARROW) ; wndclass.hbrBackground = (HBRUSH) GetStockObject (WHITE_BRUSH) ; wndclass.lpszMenuName = NULL ; wndclass.lpszClassName = _T("Direct3D9App"); if( !RegisterClass(&wndclass) ) { MessageBox(0, _T("This program requires windows NT!"), 0, 0); return false; } hwnd = CreateWindow(_T("Direct3D9App")/*0*/,_T("Direct3D9App")/*0*/,WS_EX_TOPMOST/*WS_OVERLAPPEDWINDOW*/,0,0,width,height,0,0,hInstance,0); ::ShowWindow(hwnd,SW_SHOW/*0*/); ::UpdateWindow(hwnd);

//完成了主窗口初始化的过程,以下部分实现Direct3D的初始化过程,并且设置IDirect3DDevice9指针 IDirect3D9 *_d3d9; _d3d9 = Direct3DCreate9(D3D_SDK_VERSION); //获取IDirect3D9的指针,进行设备的迭代,检测特性&获取IDirect3DDevice9指针准备 if( !_d3d9 ) { ::MessageBox(0, _T("Direct3DCreate9() - FAILED"), 0, 0); return false; } D3DCAPS9 caps; //检测图形设备的支持的特性 _d3d9->GetDeviceCaps(D3DADAPTER_DEFAULT, D3DDEVTYPE_HAL, &caps); int vp = 0; //是否支持硬件顶点运算 if(caps.DevCaps & D3DDEVCAPS_HWTRANSFORMANDLIGHT) { vp = D3DCREATE_HARDWARE_VERTEXPROCESSING; } else { vp = D3DCREATE_SOFTWARE_VERTEXPROCESSING; } D3DPRESENT_PARAMETERS d3dpp; //D3DPRESENT_PARAMETERS设置,设置IDirect3DDevice的特性 ::ZeroMemory(&d3dpp, sizeof(D3DPRESENT_PARAMETERS)); d3dpp.BackBufferWidth = width; d3dpp.BackBufferHeight = height; d3dpp.BackBufferFormat = D3DFMT_A8R8G8B8; d3dpp.BackBufferCount = 1; d3dpp.MultiSampleType = D3DMULTISAMPLE_NONE; d3dpp.MultiSampleQuality = 0; d3dpp.SwapEffect = D3DSWAPEFFECT_DISCARD; d3dpp.hDeviceWindow = hwnd; d3dpp.Windowed = windowed; d3dpp.EnableAutoDepthStencil = true; d3dpp.AutoDepthStencilFormat = D3DFMT_D24S8; d3dpp.Flags = 0; d3dpp.FullScreen_RefreshRateInHz = D3DPRESENT_RATE_DEFAULT; d3dpp.PresentationInterval = D3DPRESENT_INTERVAL_IMMEDIATE; HRESULT hr = _d3d9->CreateDevice(D3DADAPTER_DEFAULT, D3DDEVTYPE_HAL, hwnd, vp, &d3dpp, device); if( FAILED(hr) ) { ::MessageBox(0, _T("CreateDevice() - Failed"), 0, 0); return false; } return true; } int EnterMsgLoop( bool (*ptr_display)(float timeDelta) ) { MSG msg; ::ZeroMemory( &msg, sizeof(MSG) ); static float lastTime = (float)timeGetTime(); while(msg.message != WM_QUIT) { if(::PeekMessage(&msg, 0, 0, 0, PM_REMOVE)) { ::TranslateMessage(&msg); ::DispatchMessage(&msg); } else { //空闲的时候进行绘制 float currTime = (float)timeGetTime(); float timeDelta = (currTime - lastTime)*0.001f; ptr_display(timeDelta); lastTime = currTime; } } return msg.wParam; } LRESULT CALLBACK WndProc( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam) { switch( msg ) { case WM_DESTROY: ::PostQuitMessage(0); break; case WM_KEYDOWN: if(wParam == VK_ESCAPE) //esc退出主窗口程序 ::DestroyWindow(hwnd); break; } return ::DefWindowProc(hwnd, msg, wParam, lParam); } }

commonFunc.h

此书中的3个例程函数:

Setup:设置和初始化部分在此函数中进行,如分配资源,检查设备性能,设置应用程序状态。

Cleanup:释放Setup函数中分配的任何资源,如分配的存储单元。

Display:实现全部的绘制代码以及相邻帧之间应该执行的操作,如更新物体的位置。参数timeDelta为相邻帧的时间差,主要用于将动画与显示器的刷新频率保持同步。

后续的章节的实现部分集中在这3个函数中,其他部分的代码设计到Direct3D的初始化和Windows程序设计的基本概念,在其他章节中改动很少。

#ifndef COMMONFUNC_H
#define COMMONFUNC_H

#include "d3dUtility.h"
IDirect3DDevice9  *Device = 0;
bool windowed = true;
bool Setup() {
	return true;
}

void Cleanup() {
	
}

bool Display(float timeDelta) {
	if( Device ) {
		Device->Clear( 0, 0, D3DCLEAR_TARGET | D3DCLEAR_ZBUFFER, 0x00000000, 1.0f, 0);
		Device->Present(0, 0, 0, 0);
	}
	return true;
} 

#endif

Entry.cpp

window程序设计的入口,WinMain函数部分。

#include "d3dUtility.h"
#include "commonFunc.h"

int WINAPI WinMain( __in HINSTANCE hInstance, __in_opt HINSTANCE hPrevInstance, __in_opt LPSTR lpCmdLine, __in int nShowCmd ) {
	if( !d3d::InitD3D(hInstance, 800, 600, true, D3DDEVTYPE_HAL, &Device )) {
		::MessageBox(0,_T("InitD3D() - Failed"), 0, 0 );
		return 0;
	}
	if( !Setup() ) {
		::MessageBox(0,_T("Setup() - Failed"), 0, 0 );
		return 0;
	}
	
	d3d::EnterMsgLoop( Display );

	Cleanup();

	Device->Release();

	return 0;
}

 

程序运行效果截图:

posted @ 2014-12-03 09:23  weixliu  阅读(...)  评论(...编辑  收藏