zoukankan      html  css  js  c++  java
  • Direct3D11学习:(五)演示程序框架

    转载请注明出处:http://www.cnblogs.com/Ray1024

    一、概述

    在此系列最开始的文章Direct3D11学习:(一)开发环境配置中,我们运行了一个例子BoxDemo,看过这个例子源码的朋友都会发现,代码量比较大,但是Win32窗口初始化和Direct3D11初始化工作占用了很多大一部分代码,然而,我们真正关心的绘制代码并不是这些。

    为了避免以后每次创建演示程序都需要重复的初始化工作,把我们的注意力集中在演示程序度所要表达的特定细节上,我们把重复的初始化代码封装到一个简单的程序框架D3D11App中,位于D3D11App.h和D3D11App.cpp文件中。D3D11App.h和D3D11App.cpp文件包含了窗口初始化和D3D11初始化的核心代码,可以在新的演示程序中包含它,导入该框架,直接编写我们的核心代码就可以了。

    二、演示程序框架

    2.1 D3D11Util.h文件

    在介绍演示程序框架之前,我们需要介绍一组文件D3D11Util.h和D3D11Util.cpp。

    这组文件包含了一些有用的工具代码,都是程序中常用的工具。比如COM对象安全释放的宏ReleaseCOM()、HRESULT值的错误处理宏HR()和常用颜色值定义等等。

    这里解释一下HRESULT值的错误处理宏HR():

    #ifndef HR
    #define HR(x)                                              
    {                                                          
    	HRESULT hr = (x);                                      
    	if(FAILED(hr))                                         
    	{                                                      
    		DXTrace(__FILE__, (DWORD)__LINE__, hr, L#x, true); 
    	}                                                      
    }
    #endif
    

    在这个宏中,使用了DX的错误处理库dxerr.lib库中的函数DXTrace。当函数的返回值表明调用失败时,我们把返回值传递给DXTrace函数。如果hr=S_FALSE时,弹出消息框显示错误信息。

    HR()宏使用方法:

    HR(m_pD3DDevice->CheckMultisampleQualityLevels(
    		DXGI_FORMAT_R8G8B8A8_UNORM, 4, &m_4xMsaaQuality));
    

    D3D11Util.h和D3D11Util.cpp文件中其他的工具在这里就不介绍了,有兴趣的朋友可以在文章最后的下载源码处下载浏览源码。

    2.2 核心类D3D11App

    D3D11App是我们学习D3D11中所有应用程序类的基类,它提供了用于创建主应用程序窗口、运行应用程序消息循环、处理窗口消息和初始化D3D11的函数。另外,这个类还定义了一些框架函数。所有的D3D11应用程序类都继承于D3D11App类,重载它的virtual框架函数,并创建一个D3D11App派生类的单例对象。D3D11App类的定义如下: 

    class D3D11App
    {
    public:
    	D3D11App(HINSTANCE hInstance);
    	virtual ~D3D11App();
    	
    	// 获取应用程序实例句柄
    	HINSTANCE AppInst()const;
    	// 获取主窗口句柄
    	HWND MainWnd()const;
    	// 后台缓存区的长宽比
    	float AspectRatio()const;
    	// 应用程序消息循环
    	int Run();
     
    	// 框架方法
    	// 派生类需要重载这些方法实现所需的功能
    
    	virtual bool Init();
    	virtual void OnResize(); 
    	virtual void UpdateScene(float dt)=0;
    	virtual void DrawScene()=0; 
    	virtual LRESULT MsgProc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam);
    
    	// 处理鼠标输入事件的便捷重载函数
    	virtual void OnMouseDown(WPARAM btnState, int x, int y){ }
    	virtual void OnMouseUp(WPARAM btnState, int x, int y)  { }
    	virtual void OnMouseMove(WPARAM btnState, int x, int y){ }
    
    protected:
    
    	// 创建窗口
    	bool InitMainWindow();
    
    	// 初始化D3D
    	bool InitDirect3D();
    
    	// 计算帧率
    	void CalculateFrameStats();
    
    protected:
    
    	HINSTANCE m_hAppInst;		// 应用程序实例句柄
    	HWND      m_hMainWnd;		// 主窗口句柄
    	bool      m_appPaused;		// 程序是否处在暂停状态
    	bool      m_minimized;		// 程序是否最小化
    	bool      m_maximized;		// 程序是否最大化
    	bool      m_Resizing;		// 程序是否处在改变大小的状态
    	UINT      m_4xMsaaQuality;	// 4X MSAA质量等级
    
    	GameTimer m_timer;			// 用于记录deltatime和游戏时间
    
    	ID3D11Device*			m_pD3DDevice;			// D3D11设备
    	ID3D11DeviceContext*	m_pD3DImmediateContext;	// 上下文
    	IDXGISwapChain*			m_pSwapChain;			// 交换链
    	ID3D11Texture2D*		m_pDepthStencilBuffer;	// 深度缓冲区
    	ID3D11RenderTargetView* m_pRenderTargetView;	// 渲染目标视图
    	ID3D11DepthStencilView* m_pDepthStencilView;	// 深度缓冲视图	
    	D3D11_VIEWPORT			m_screenViewport;		// 视口
    
    
    	std::wstring	m_mainWndCaption;		// 窗口标题
    	D3D_DRIVER_TYPE m_D3DDriverType;		// 是否使用硬件加速
    	int				m_clientWidth;			// 窗口大小
    	int				m_clientHeight;			// 窗口大小
    	bool			m_enable4xMsaa;			// 是否使用4XMSAA
    };
    

    上面的代码中有注释,从注释中我们可以看到成员函数和变量的作用。

    下面的部分我们将详细介绍D3D11App一部分成员函数。

    2.3 D3D11App部分成员函数介绍

    在这节中我们主要介绍D3D11App类的成员函数中的框架方法。所谓框架方法,就是D3D11App中的虚函数,在之后学习的每个演示程序中,我们可以重载这些方法来实现特定示例中的代码细节。D3D11App类实现的这种结构可以将所有的初始化代码、消息处理代码和其他代码安排得井井有条,使派生类专注于实现演示程序的特定代码。下面是对这些框架方法的描述:

    (1)Init:该方法包含应用程序的初始化代码,比如分配资源、初始化对象和设置灯光。该方法在D3D11App的实现中包含InitMainWindow和InitDirect3D方法的调用语句;所以,当在派生类中重载该方法时,应首先调用该方法的D3D11App版本,就像下面这样:

    void TestApp::Init() 
    {
        if(!D3D11App::Init())
            return false;
        /* 剩下的初始化代码从这里开始 */
    }
    

    (2) OnResize:该方法在D3D11App::MsgProc收到WM_SIZE消息时调用。当窗口的尺寸改变时,一些与客户区大小相关的Direct3D属性也需要改变。尤其是需要重新创建后台缓冲区和深度/模板缓冲区,使它们与窗口客户区的大小一致。后台缓冲区的大小可以通过调用IDXGISwapChain::ResizeBuffers方法来进行调整。而深度/模板缓冲区必须被销毁,然后根据新的大小重新创建。另外,渲染目标视图和深度/模板视图也必须重新创建。OnResize方法在D3D11App的实现中包含了调整后台缓冲区和深度/模板缓冲区的代码;详情请直接参见源代码。除缓冲区外,依赖于客户区大小的其他属性(例如,投影矩阵)也必须重新创建。我们把该方法作为框架的一部分是因为当窗口大小改变时,客户代码可能需要执行一些它自己的逻辑。

    (3)UpdateScene:该抽象方法每帧都会调用,用于随着时间更新3D应用程序(例如,实现动画和碰撞检测、检查用户输入、计算每秒帧数等等)。

    (4)DrawScene:该抽象方法每帧都会调用,用于将3D场景的当前帧绘制到后台缓冲区。当绘制当前帧时,我们调用了IDXGISwapChain::Present方法将后台缓冲区的内容呈现在屏幕上。

    (5)MsgProc:该方法是主应用程序窗口的消息处理函数。通常,当你只需重载该方法,就可以处理未由D3D11App::MsgProc处理(或者没按照你所希望的方式处理)的消息。如果你重载了这个方法,那么那些你没有处理的消息都会送到D3D11App::MsgProc中进行处理。

    另外,除了上述的五个框架方法之外,为了使用起来更方便,我们还提供了三个虚函数,用于处理鼠标点击、释放和移动的事件。

    virtual void OnMouseDown(WPARAM btnState, int x, int y){ }
    virtual void OnMouseUp(WPARAM btnState, int x, int y)  { }
    virtual void OnMouseMove(WPARAM btnState, int x, int y){ }
    

    你可以重载这些方法处理鼠标事件,而用不着重载MsgProc方法。这些方法的第一个参数WPARAM都是相同的,保存了鼠标按键的状态(例如,哪个鼠标按键被按下),第二、三个参数是光标在客户区域的(x,y)坐标。

    2.4 将程序框架导入新建演示程序

    这个框架还需要另外一个部分:游戏计时器,上一篇文章我们介绍过了游戏计时器的实现,这里就不提了。

    我们将上面提到的三组文件(D3D11Util.h和D3D11Util.cpp、D3D11App.h和D3D11App.cpp、GameTimer.h和GameTimer.cpp)放到一个文件夹Common中,这样每个演示程序都可以使用,避免多次复制了。我们的演示程序需要使用程序框架时,需要导入框架,即把Common文件夹中的文件添加到演示程序中,并把Common文件夹的目录添加到演示程序项目属性的包含目录中。

    这样,程序框架就成功地导入到新建演示程序中了,我们接下来就可以使用程序框架编写演示程序了。

    2.5 使用框架编写演示程序

    (1)创建一个继承自主框架类D3D11App的类TestApp:

    class TestApp : public D3D11App
    {
    public:
    	TestApp(HINSTANCE hInstance);
    
    	void UpdateScene(float dt);
    	void DrawScene(); 
    };
    

    (2)添加TestApp类的成员函数实现

    //
    // TestApp Implement
    //
    
    TestApp::TestApp(HINSTANCE hInstance)
    	: D3D11App(hInstance)
    {
    	m_mainWndCaption = L"2_D3DTimingAndAnimation";
    }
    
    void TestApp::UpdateScene(float dt)
    {
    
    }
    
    void TestApp::DrawScene()
    {
    	assert(m_pD3DImmediateContext);
    	assert(m_pSwapChain);
    
    	m_pD3DImmediateContext->ClearRenderTargetView(m_pRenderTargetView, reinterpret_cast<const float*>(&Colors::LightSteelBlue));
    	m_pD3DImmediateContext->ClearDepthStencilView(m_pDepthStencilView, D3D11_CLEAR_DEPTH|D3D11_CLEAR_STENCIL, 1.0f, 0);
    
    	HR(m_pSwapChain->Present(0, 0));
    }
    

    (3)添加程序入口

    // 程序入口
    int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE prevInstance,
    	PSTR cmdLine, int showCmd)
    {
    #if defined(DEBUG) | defined(_DEBUG)
    	_CrtSetDbgFlag( _CRTDBG_ALLOC_MEM_DF | _CRTDBG_LEAK_CHECK_DF );
    #endif
    
    	TestApp theApp(hInstance);
    
    	if( !theApp.Init() )
    		return 0;
    
    	return theApp.Run();
    }
    

    到此为止,我们已经利用程序框架成功地生成了一个演示程序。

    有兴趣看源码的朋友可以点击这里下载,源码为3_D3DFrame文件夹。

    三、结语

    基本上,我们用不着做任何实际工作就可以实现这个程序,因为基类D3D11App已经实现了它所需要的大部分功能。

    有了程序框架之后,我们在学习过程中的演示程序的编写就简单多了,直接导入程序框架,只写我们关心的核心代码就可以了。

  • 相关阅读:
    6 15种对抗攻击的防御方法
    5 12种生成对抗样本的方法
    4 基于优化的攻击——CW
    3 基于梯度的攻击——MIM
    Hibernate 5 Maven 仓库的 Artifacts
    Hibernate 5 发行组件下载
    Hibernate 5 的模块/包(modules/artifacts)
    Hibernate 5 开始使用指南前言
    Git 如何针对项目修改本地提交提交人的信息
    Spring Batch 4.2 新特性
  • 原文地址:https://www.cnblogs.com/Ray1024/p/6087921.html
Copyright © 2011-2022 走看看