6.7 渲染状态

从本质上讲,Direct3D是一个状态机(state machine)。在我们改变它的状态之前,驻留在状态机内的当前状态是不会改变的。例如,我们在6.1节、6.2节和6.3节中看到,当顶点缓冲和索引缓冲绑定到管线的输入装配阶段时,如果我们不绑定其他缓冲,那么它们就会一直驻留在那里;同样,在没有改变图元拓扑之前,当前的图元拓扑设置会一直有效。另外,Direct3D将配置信息封装在状态组中,我们可以使用如下3种状态组配置Direct3D:

1.ID3D11RasterizerState:该接口表示用于配置管线光栅化阶段的状态组。

2.ID3D11BlendState:该接口表示用于配置混合操作的状态组。我们将在有关混合的章节讨论这些状态;默认情况下,混合处于禁用状态,所以我们可以先不考虑这方面的问题。

3.ID3D11DepthStencilState:该接口表示用于配置深度测试和模板测试的状态组。我们将在有关模板缓冲的章节讨论这些状态;默认情况下,模板是禁用的,所以我们可以先不考虑这方面的问题。而默认的深度测试是我们在4.1.5节描述的标准深度测试。

目前,我们唯一需要关心的状态块接口是ID3D11RasterizerState。我们可以通过填充一个D3D11_RASTERIZER_DESC结构体并调用如下方法来创建ID3D11RasterizerState对象:

HRESULT ID3D11Device::CreateRasterizerState(
    const D3D11_RASTERIZER_DESC *pRasterizerDesc, 
    ID3D11RasterizerState **ppRasterizerState);

第1个参数是一个指向D3D11_RASTERIZER_DESC结构体的指针,该结构体用于描述所要创建的光栅化状态块;第二个参数用于返回创建后的ID3D11RasterizerState对象。

D3D11_RASTERIZER_DESC结构体的定义如下:

typedef struct D3D11_RASTERIZER_DESC{ 
    D3D11_FILL_MODE FillMode;     // Default:D3D11_FILL_SOLID
    D3D11_CULL_MODE CullMode;     // Default:D3D11_CULL_BACK
    BOOL FrontCounterClockwise;  // Default:false
    INT DepthBias;   // Default:0
    FLOAT DepthBiasClamp;   // Default:0.0f
    FLOAT SlopeScaledDepthBias;   // Default:0.0f
    BOOL DepthClipEnable;   // Default:true
    BOOL ScissorEnable;   // Default:false
    BOOL MultisampleEnable;  // Default:false
    BOOL AntialiasedLineEnable;  // Default:false
} D3D11_RASTERIZER_DESC;

这里面的大部分成员是高级选项或者不常用的选项;因此,我们在这里只讲解前3个成员的含义,其他成员的详情请参见SDK文档。

1.FillMode:当指定为D3D11_FILL_WIREFRAME时,表示以线框模式渲染几何体;当指定为D3D11_FILL_SOLID时,表示以实心模式渲染几何体,这是默认值。

2.CullMode:当指定为D3D11_CULL_NONE时,表示禁用背面消隐功能;当指定为D3D11_CULL_FRONT时,表示消隐朝前的三角形;当指定为D3D11_CULL_BACK时,表示消隐朝后的三角形,这是默认值。

3.FrontCounterClockwise:当设为false时,表示按顺时针方向环绕的三角形(相对于观察者)是朝前的,而按逆时针方向环绕的三角形(相对于观察者)是朝后的,这是默认值。当设为true时,表示按逆时针方向环绕的三角形(相对于观察者)是朝前的,而按顺时针方向环绕的三角形(相对于观察者)是朝后的。

在创建ID3D11RasterizerState对象之后,我们可以使用个新的状态块来更新设备:

void ID3D11DeviceContext::RSSetState(ID3D11RasterizerState *pRasterizerState); 

下面的代码示范了如何通过创建一个光栅化状态块来禁用背面消隐:

D3D11_RASTERIZER_DESC rsDesc; 
ZeroMemory(&rsDesc, sizeof(D3D11_RASTERIZER_DESC)); 
rsDesc.FillMode = D3D11_FILL_SOLID; 
rsDesc.CullMode = D3D11_CULL_NONE; 
rsDesc.FrontCounterClockwise = false; 
rsDesc.DepthClipEnable = true;

HR(md3dDevice->CreateRasterizerState(&rsDesc,&mNoCullRS));

注意:因为没有设置的属性的默认值是0或false,所以使用ZeroMemory可以正常初始化这些属性。但是,若有些属性默认值不是0或是true,那么你就必须显式地设置这些值。

注意,对于一个应用程序来说,你可能会用到多个不同的ID3D11RasterizerState对象。所以,你应该在初始化时把它们都创建出来,然后在应用程序的更新/绘图代码中切换些状态。例如,场景中有两个物体,你希望先以线框模式绘制第一个物体,然后再以实心模式绘制第二个物体。那么,你就应该创建两个ID3D11RasterizerState对象,当绘制物体时,切换这两种不同的状态:

// 在初始化时创建绘制状态对象 
ID3D11RasterizerState* mWireframeRS; 
ID3D11RasterizerState* mSolidRS; 
... 
// 在draw方法中切换渲染状态对象 
md3dDeviceContext->RSSetState(mSolidRS); 
DrawObject(); 
md3dDeviceContext->RSSetState(mWireframeRS); 
DrawObject(); 

注意,Direct3D不会从一种状态自动恢复到先前状态。所以,当绘制物体时,你应该根据需要手工指定状态对象。错误地假设设备的当前状态必然会导致错误的渲染结果。每个状态块都有一个默认状态。我们可以通过在调用RSSetState方法时指定空值来恢复默认状态:

md3dDeviceContext->RSSetState( 0 ); 

注意:应用程序无需在运行时创建额外的渲染状态组。所以,应该在初始化时就定义并创建所有需要用到的状态组。而且,因为无需在运行时修改状态组,你可以在渲染代码中对这些状态组提供全局只读访问。例如,你可以将所有状态组对象放置在一个静态类中,通过这个方法,你就无需创建重复的状态组,渲染代码的不同部分都能共享这个渲染状态组对象。

文件下载(已下载 915 次)

发布时间:2014/8/1 下午7:54:32  阅读次数:5189

2006 - 2024,推荐分辨率 1024*768 以上,推荐浏览器 Chrome、Edge 等现代浏览器,截止 2021 年 12 月 5 日的访问次数:1872 万 9823 站长邮箱

沪 ICP 备 18037240 号-1

沪公网安备 31011002002865 号