DirectXGuide入门
本教学主要介绍DXGuide类库2D部分的一些入门知识,主要包括如何用DXGDD向导
创建一个DirectX程序、翻页、渲染页面等,不管你是高手还是新手,包管你尽快
上手。
1.第一个DirectX程序
2.让你的页面动起来
3.渲染页面
4.从文件中读入位图
5.将你的文件打包
6.......
第一个DirectX程序
在开始之前,首先你要下载DXGuid类库的库文件和头文件(注意有调试版和发行版
哦!),这里暂且假定你已经将DXGuid的头文件放在D:\DXGuid\include,库文件
放在D:\DXGuid\lib。然后还要下载一个工程向导DXGDD.awx,将它放在
..\Microsoft Visual Studio\Common\MSDev98\Template中(VC++6.0)。最后是设
定DXGuid头文件和库文件的路径,从Tools菜单依次进入Options...\Directorys
,分别在Include files和Library files中加入头文件和库文件的路径。
准备工作完成之后,你就可以开始你的第一个DirectX程序了。进入VC,从File菜
单选择New,在Pogjects表单中选择DirectX Guide's DirectDraw AppWizard,并
在Project name中输入你的工程文件名,如DDTest。
单击OK按钮你就可以生成一个DirectX程序了。
在编译连接之前按Alt+F7进入工程参数设置,在General中将Microsoft
Foundation Classes设置为Use MFC in a static Library。
好了到这里,你的DirectX程序就可以编译运行了,怎么样,还不错吧?程序可以
通过Alt+Enter键在窗口和全屏模式下进行切换。
2)让你的页面动起来
从显屏中我们看到的是正面显存(Front buffer)中的内容,我们同时还可创建一
个或多个反面显存(Backbuffer),就象一张纸有正面和背面一样,我们可以将纸
翻过来看。同样我们可以在Front buffer和Backbuffer之间进行翻页操作。一般
是在Back buffer中进行渲染,渲染好后将它翻到正页即可,翻页后Back
buffer就成了Front buffer,而原来的Front buffer则成了Back buffer,其实
Front buffer 和Back buffer构成的是一个环路,就象立在我们面前的一个立方
柱,正面朝向我们的是正面,其它三个面称为背面。不过编程时这些细节都不需
要我们参与处理。全部过程只需一个函数Flip()就可完成。翻页有什么好处?您
想想看如果我们直接在Front buffer中做渲染,眼睛是可以看到渲染过程的,就
象在黑板上画画,先要将黑板擦干净,再将图一个一个画上去。也就是说每一显
示帧中间都有一个能被眼睛察觉的间隙,看起来会觉得很不舒服。下面我们来看
看如何翻页。
利用Class Wizard转到CDDTest的UpdateFrame()
bool CDDTestApp::UpdateFrame(void)
{
// add your code here !
m_pDDDevice->GetRenderSurface()->Fill(0);
return CDirectXApp::UpdateFrame();
}
将m_pDDDevice->GetRenderSurface()->Fill(0);用以下代码替换:
. . . . . .
static BOOL bOdd = FALSE;
if (bOdd)
m_pDDDevice->GetRenderSurface()->Fill(0);
else
m_pDDDevice->GetRenderSurface()->Fill(0xFFFFFFFF);
bOdd = !bOdd;
. . . . . .
运行后会怎么样?眼睛是不是很不舒服?不过不要紧,下一个教程会好受些。
好象忘了什么?Flip()?哪有翻页动作?我怎么没看见?其实基本的动作DXGuide
都给你封装到基类里了,看到CDirectXApp::UpdateFrame()了没有?你将它改成
m_pDirectDraw->Flip()再运行看看,真对不起,又让你眼花了,下不为例。
渲染页面
前面讲到我们的渲染都是在Back buffer里完成的,渲染后再Flip到Front buffer
。那么如何取到Backbuffer呢?我们回到前面一个教程来看看,在前面有这样一
句:m_pDDDevice->GetRenderSurface()->Fill(0);这里的
(CDDDevice*)m_pDDDevice是你当前使用的显卡设备,
CDDDevice主要用于枚举显示设备,也就是说DDGuide支持多显屏,这里暂不多加
介绍,知道m_pDDDevice是什么就可以了。GetRenderSurface()返回的就是此显卡
的Back buffer,返回值类型是CDDSurface*,Fill(DWORD)是CDDSurface是一个成
员函数,用于在buffer里填充颜色。所以Fill(0)是在buffer里填充黑色,
Fill(0xFFFFFFFF)是填充白色。看到这里大家应该知道,其实前一教程就已经做
渲染操作了。下面我们来看看如何在buffer上做一些简单的渲染。
还是用第一个例子中生成的工程来做一些小小的改动,在
CDDTestApp::UpdateFrame()中,将
m_pDDDevice->GetRenderSurface()->Fill(0);
删除,并在此位置插入以下行:
m_pDDDevice->GetRenderSurface()->Fill(0xFFFFFFFF);
CDC dc;
m_pDDDevice->GetRenderSurface()->GetDC(dc);
dc.SetBkColor(0x00FFFFFF);
dc.SetTextColor(0x0000FF00);
dc.TextOut(10,10,"Welcome to DirectX Guide!");
dc.Arc(30,30, 100,100, 30, 30, 40, 40);
dc.MoveTo(30,30);
dc.LineTo(40,40);
m_pDDDevice->GetRenderSurface()->ReleaseDC(dc);
熟悉GDI编程的朋友一定觉得眼熟,不错,这里几乎是一样的,如果你不在乎速度
的话,尽可使用DC,不过建议还是少用为好。好了,编译运行看看,你不觉得帧
数比第一个例子减少了很多吗?
3)从文件中读入位图
这一节要教你如何将位图画到buffer上去。前面介绍过Front buffer和Back
buffer这两种buffer大小与你初始化的分辨率是一样的,还有一种buffer可以为
任意大小,称为Off screen buffer,我们可以将位图预先读到这样的Off screen
buffer中,然后在用到时直接将它Blit到Back buffer,Blit意为位图的块传送
。DXGuide中的CDDDIBSurface类就是专门用于处理位图的,下面我们来看看
CDDDIBSurface的基本用法,仍是利用例一中的工程。
在DDTestApp.h中加入行
#include "DDDIBSurface.h"
利用ClassWizard在CDDTestApp中添加一个成员变量
CDDDIBSurface *m_pDib;
利用ClassWizard转到CDDTestApp::InitDXObjects(void)中,在行
// add your init code here !
后插入下列行
m_pDib = new CDDDIBSurface;
m_pDib->Create(m_pDDDevice, "DXGuide.bmp");
转到CDDTestApp::UpdateFrame(void)中,在行
m_pDDDevice->GetRenderSurface()->Fill(0);
后插入下列行
CRect rect(0, 0, m_pDib->GetWidth(), m_pDib->GetHeight());
m_pDDDevice->GetRenderSurface()->Blt(rect, m_pDib, rect,
DDBLT_WAIT, NULL);
m_pDDDevice->GetRenderSurface()->BltFast(100, 100, m_pDib, rect,
DDBLTFAST_WAIT);
转到CDDTestApp::DestroyDXObjects(void)中,在行
// add your destroy code here !
后插入下行
delete m_pDib; m_pDib = NULL;
在DDTest目录下创建一个名为DXGuide.bmp
Blt和BltFast的用法参照SDK帮助中IDirectDrawSurface::Blt和
IDirectDrawSurface::BltFast的用法。
注意:到这里必须将CDDTestApp.cpp的结构搞清楚,构造函数中的
m_dwAppInitFlags = CDirectXApp::DXAPPIF_DD;
| CDirectXApp::DXAPPIF_DI
| CDirectXApp::DXAPPIF_DS
| CDirectXApp::DXAPPIF_GUIMANAGER;
决定使用哪些DirectX部件,如你不想使用DirectInput,将
CDirectXApp::DXAPPIF_DI拿掉就行了(注意:要
使用GUIManager的话一定要使用DirectInput)。
CDDTestApp::GetDXInitSettings(void)是用于获取初始信息,如选取哪个设备来
显示画面或播放声音。
CDDTestApp::InitDXObjects(void)用于初始化成员变量。
CDDTestApp::DestroyDXObjects(void)用于释放为成员变量申请的内存空间。
CDDTestApp::UpdateFrame(void)用于渲染页面。
你自己加入的成员变量的初始化和释放应该都放在InitDXObjects()和
DestroyDXObjects()中完成。
4)将你的文件打包
为什么要将文件打包?首先是加快文件的读取速度,如果你有几十个甚至几千个
文件放在同一个目录内,打开文件时文件查找是一种费时的操作,如果包装成一
个大文件的话,不管你要读多少个文件,打开文件的操作只有一次;别外还有一
种作用是让目录看起来整洁一些,打包后文件也不会那么直观,别人不会一目了
然看出你的文件都是些什么。
记得前面的读位图语句吧?m_pDib->Create(m_pDDDevice, "DXGuide.bmp"),
CDDDIBSurface的Create可接受四个参数,第一个是CDDevice*, 第二个是位图的
文件名,第三个是CPackFileManager*,第四个是DIB文件类型。其中第三个参数
CPackFileManager*指向的就是一个文件包,如果这项为NULL(缺省),则直接从当
前目录中读位图文件,否则从文件包内读取位图。第四个参数用于告诉函数读取
的DIB是位图(缺省)还是其它格式文件。
我们再回头看看CDDTest的构造函数中有一SetPackFileName(_T(""),
_T("gui.ff"))语句,这是指定包文件
名,第一个参数用于指定一般数据文件包的文件名,第二个参数是GUI相关数据文
件包的文件名,如果指定的文件名是一个目录名,则此目录与文件包同等看待。
第一个文件名指定后,会生成一个CPackFileManager,指针放在成员变量
m_pPackFileManager里。
所以我们把前一个例子中的位图打成包,放于文件包PackFile.ff中,且CDDTest
构造函数中的语句
SetPackFileName(_T(""), _T("gui.ff"))改成
SetPackFileName(_T("PackFile.ff"), _T("gui.ff")),函数
CDDTestApp::InitDXObjects(void)中的语句m_pDib->Create(m_pDDDevice,
"DXGuide.bmp")改为m_pDib->Create(m_pDDDevice, "DXGuide.bmp",
m_pPackFileManager)。再运行看看,是不是一样的?
[url=http://apexliu.top263.net/game/content.htm][/url]