Directx11教程(56) 建立一个skydome

      本章建立一个skydome(天空穹),主要学习如何使用cube mapping。

     cube map就是把六张纹理当作一个cube的六个面,而cube的中心,则是坐标轴,而六个面则是垂直于坐标轴某个轴,如下图所示,在cube mapping中,我们不在使用二维纹理坐标,而是用(u,v,w)三维纹理坐标,用这个坐标产生一个查询向量,这个向量和cube 纹理的交点,即为该顶点对应的纹理texel。

    

       可以通过微软的Directx Texture tool制作cube map纹理,前提是要准备好6张无缝过度的图片,这可能需要专业的工具。如下图,就是用6张图片生成的cube map纹理图,每章图片对应cube的一个面。

     我们建立的skydome是一个球形体,并不是skybox,但使用cube map的方法是一样的。

     首先是建立一个SkyDomeModelClass类,这个类中的顶点结构比较简单,只有一个参数:position,至于skydome贴图用的纹理坐标,我们则是在vs中生成。

struct VertexType
    {
    D3DXVECTOR3 position;

    };

    SkyDome模型是通过SkyDomeModelClass中的函数BuildGeoSphere得到,它通过细分一个20面体,得到一个近似的球形模型,具体的细分算法是把一个三角形细分成四个三角形(取每条边的中点),如下图所示:

生成SkyDome模型的代码如下:

//通过一个20面体细分,近似得到一个球体
void SkyDomeModelClass::BuildGeoSphere(    int numSubdivisions,    float radius,    VertexList& vertices, IndexList& indices)
    {
    // 最小的细分数量.
    numSubdivisions = min(numSubdivisions, 5);

    const float X = 0.525731f;
    const float Z = 0.850651f;

    D3DXVECTOR3 pos[12] =
        {
        D3DXVECTOR3(-X, 0.0f, Z),  D3DXVECTOR3(X, 0.0f, Z), 
        D3DXVECTOR3(-X, 0.0f, -Z), D3DXVECTOR3(X, 0.0f, -Z),   
        D3DXVECTOR3(0.0f, Z, X),   D3DXVECTOR3(0.0f, Z, -X),
        D3DXVECTOR3(0.0f, -Z, X),  D3DXVECTOR3(0.0f, -Z, -X),   
        D3DXVECTOR3(Z, X, 0.0f),   D3DXVECTOR3(-Z, X, 0.0f),
        D3DXVECTOR3(Z, -X, 0.0f),  D3DXVECTOR3(-Z, -X, 0.0f)
        };

    DWORD k[60] =
        {
        1,4,0,  4,9,0,  4,5,9,  8,5,4,  1,8,4,   
        1,10,8, 10,3,8, 8,3,5,  3,2,5,  3,7,2,   
        3,10,7, 10,6,7, 6,11,7, 6,0,11, 6,1,0,
        10,1,6, 11,0,9, 2,11,9, 5,2,9,  11,2,7
        };

    vertices.resize(12);
    indices.resize(60);

    for(int i = 0; i < 12; ++i)
        vertices[i] = pos[i];

    for(int i = 0; i < 60; ++i)
        indices[i] = k[i];

    for(int i = 0; i < numSubdivisions; ++i)
        Subdivide(vertices, indices);

    //投影顶点到球面上,然后缩放顶点到球心的距离
    for(int i = 0; i < vertices.size(); ++i)
        {
        D3DXVec3Normalize(&vertices[i], &vertices[i]);
        vertices[i] *= radius;
        }
    }
//细分输入三角形,为四个面积相等的三角形
void SkyDomeModelClass::Subdivide(VertexList& vertices, IndexList& indices)
    {
    VertexList vin = vertices;
    IndexList  iin = indices;

    vertices.resize(0);
    indices.resize(0);

    int numTris = (int)iin.size()/3;
    for(int i = 0; i < numTris; ++i)
        {
        D3DXVECTOR3 v0 = vin[ iin[i*3+0] ];
        D3DXVECTOR3 v1 = vin[ iin[i*3+1] ];
        D3DXVECTOR3 v2 = vin[ iin[i*3+2] ];

        D3DXVECTOR3 m0 = 0.5f*(v0 + v1);
        D3DXVECTOR3 m1 = 0.5f*(v1 + v2);
        D3DXVECTOR3 m2 = 0.5f*(v0 + v2);

        vertices.push_back(v0); // 0
        vertices.push_back(v1); // 1
        vertices.push_back(v2); // 2
        vertices.push_back(m0); // 3
        vertices.push_back(m1); // 4
        vertices.push_back(m2); // 5

        //索引出四个三角形
        indices.push_back(i*6+0);

        indices.push_back(i*6+4);
        }
    }

     另外我们新建一个SkyDomeShaderClass, 用来渲染SkyDome,它调用的vs,ps shader文件为cubetex.vs, cubetex.ps

     在cubeTex.vs中,我们要注意两点:

1、设置skydome顶点的世界坐标系z=w,这样,skydome总会在远裁剪平面上。

2、用顶点local坐标做为cubemap的纹理坐标。

// set z = w so that z/w = 1 (i.e., skydome always on far plane).
//设置z=w
output.position =  output.position.xyww;

//用local坐标做为cubemap查询向量.
output.tex = input.position;

     cubeTex.ps中,我们通过cube sample函数得到cube纹理:

TextureCube gCubeMap;

float4 CubePixelShader(PixelInputType input) : SV_TARGET
{

     return gCubeMap.Sample(SampleType, input.tex);
}

     在GraphicsClass类中渲染SkyDome时,我们要关掉cull,再就是建立一个skydome专用depthstencil状态,在该状态中深度比较函数是: D3D11_COMPARISON_LESS_EQUAL

//skyome 顶点和索引数据放入缓冲区,准备渲染
m_SkydomeModel->Render(m_D3D->GetDeviceContext());

m_D3D->EnableCubeDepthStencil();
m_D3D->ChangeNoCullMode(true);

result = m_SkydomeShader->Render(m_D3D->GetDeviceContext(), m_SkydomeModel->GetIndexCount(), worldMatrix, viewMatrix, projectionMatrix,
    m_TexManager->createCubeTex(m_D3D->GetDevice(),string("grassenvmap1024.dds")));

if(!result)
    {
    return false;
    }
m_D3D->ChangeNoCullMode(false);
m_D3D->EnableDefaultDepthStencil();

     另外,在textureManagerClass类中,我们增加了函数createCubeTex,专门用来读入cube texture

ID3D11ShaderResourceView* TexManagerClass::createCubeTex(ID3D11Device* device,string filename)
    {

   // 如果纹理资源已经存在,则返回,否则创建
    for(int i = 0; i < m_TextureRVs.size(); ++i)
        if(! m_TextureNames[i].compare(filename) )
            return m_TextureRVs[i];

     HRESULT result;
     D3DX11_IMAGE_LOAD_INFO loadInfo;
     loadInfo.MiscFlags = D3D11_RESOURCE_MISC_TEXTURECUBE;

     ID3D11Texture2D* tex = 0;
     result = D3DX11CreateTextureFromFile(device, stringToLPCWSTR(filename), &loadInfo, 0, (ID3D11Resource**)&tex, 0) ;

     D3D11_TEXTURE2D_DESC texDesc;
    tex->GetDesc(&texDesc);

    D3D11_SHADER_RESOURCE_VIEW_DESC viewDesc;
    viewDesc.Format = texDesc.Format;
    viewDesc.ViewDimension = D3D11_SRV_DIMENSION_TEXTURECUBE;
    viewDesc.TextureCube.MipLevels = texDesc.MipLevels;
    viewDesc.TextureCube.MostDetailedMip = 0;

    ID3D11ShaderResourceView* rv = 0;
    result = device->CreateShaderResourceView(tex, &viewDesc, &rv);
    if(FAILED(result))
        {
        HR(result);
        return false;
        }

    m_TextureNames.push_back(filename);
    m_TextureRVs.push_back(rv);

    return rv;
    }

程序执行后界面如下,你可以旋转摄像机,看看能不能超过skydome的包围:

完整的代码请参考:

工程文件myTutorialD3D11_51

代码下载:

http://files.cnblogs.com/mikewolf2002/d3d1150-58.zip

http://files.cnblogs.com/mikewolf2002/pictures.zip

时间: 2024-09-03 19:11:23

Directx11教程(56) 建立一个skydome的相关文章

Directx11教程(5) 画一个简单的三角形(1)

      在本篇教程中,我们将通过D3D11画一个简单的三角形.在D3D11中,GPU的渲染主要通过shader来操作(当然还有一些操作是由GPU固定管线完成,比如光栅化操作),最常用的shader操作是顶点shader(vertex shader)和像素shader(pixel shader).其实shader就是在GPU中执行的代码,这些代码被driver编译成硬件依赖的机器码,最终被GPU中shader pipe执行,从而完成3D渲染.D3D11中shader是用一种类C的语言HLSL编

Directx11教程(19) 画一个简单的地形

      通常我们在xz平面定义一个二维的网格,然后y的值根据一定的函数计算得到,比如正弦.余弦函数的组合等等,可以得到一个看似不错的地形或者水面的效果. 在本教程中我们修改ModelClass.h和ModelClass.cpp,得到一个近似的地形.         在本章代码中,我们定义300*300=90000个顶点,共(300-1)(300-1)*2个三角形,每个网格的大小都为1.     我们得到y值的函数为: float ModelClass::getHeight(float x,

Directx11教程(7) 画一个颜色立方体

      前面教程我们通过D3D11画了一个三角形,本章我们将画一个颜色立方体,它的立体感更强.主要的变动是ModelClass类,在ModelClass中定义一个立方体需要的顶点信息,然后创建顶点缓冲和索引缓冲.      在ModelClass.h中,我们定义一些宏来表示颜色,以便后面给顶点颜色属性赋值时用. ModelClass.h代码如下: #pragma once #include <d3d11.h> #include <d3dx10math.h> //定义一些常用颜色

Directx11教程(9) 增加一个TimerClass类

     在上篇教程代码的基础上,我们增加一个TimerClass类,这个类的功能很简单,就是可以计算相邻2帧的时间差.利用这个时间差值,可以实现平滑的动画,使得动画不会因为不同机器fps不同,从而动画效果变快或者变慢.     我们主要是利用QueryPerformanceCounter函数来查询定时器的计数值.     如果硬件里有定时器,它会启动这个定时器,之后会不断获取定时器的值,这样的定时器精度,就跟硬件时钟的晶振一样精确的. TimerClass.h代码如下: #pragma onc

Directx11教程(6) 画一个简单的三角形(2)

     在上篇教程中,我们实现了在D3D11中画一个简单的三角形,但是,当我们改变窗口大小时候,三角形形状却随着窗口高宽比例改变而改变,如下图所示:           这是因为我们改变了窗口大小,但后缓冲大小在程序初始化时候,已经被指定,不随着窗口改变而改变,这样在视口映射下,我们所渲染的三角形就改变了形状. 下面我们将对程序进行一些小的改动,从而实现改变窗口大小,而渲染的图形形状不变.     首先是SystemClass.cpp,主要是增加了WM_SIZE消息的处理,当窗口大小改变时候,

Directx11教程(55) 建立球形和锥形物体

本教程中,我们新建2个model class,SphereModelClass以及CylinderModelClass,分别用来表示球形和锥形物体. 程序执行后的界面如下: 线框模式界面如下: 从线框模式可以看出,球形是由三个因素决定:半径.经度线.纬度线.        在SphereModelClass.cpp中,我们看到,初始化顶点缓冲和索引缓冲的函数为:InitializeBuffers(ID3D11Device* device,  float radius, int numSlices

Ultradev实例教程:2.1 建立一个简单的access数据库

access|教程|数据|数据库 第二章:设置数据库 第一节.建立一个简单的access数据库 这一节我们将学习有关数据库创建的内容,用过Access创建数据库的朋友可以略过不看. 打开Microsoft Access,选择 文件 -〉新建数据库,然后选择一个位置保存你的数据库文件,这里我们按它默认的db1.mdb命名保存. 保存数据库后我们看到的是下面的界面,双击使用设计器创建表,开始创建一个数据表.我们这里的应用你只需照着步骤一步步做下去,即使你以前从没接触过数据库也没有问题,如果你想做深层

如何建立一个网站?咱们弄点简单点教程

如何去建立一个好的网站,想到这样问题的朋友大都是建站小白,咱们就针对个人弄点简单的建站流程一. 1.怎样建网站-最简单就用建站工具 网站建设是一个简单又是一个复杂的过程,一般简单的展示产品类型的网站,我们要学最少两套软件才可以完成网站建设. 首先,网站的排版软件工具,一般我们最多人用,又是最常用的就是Macromedia Dreamweaver网页排版软件了,它有着简单.清析.操作方便的 优点.学完网页的排版我们可以做出一些简单的网页.第二步我们要学习PhotoShop作图软件,初学的一般学会如

Directx11教程(20) 一个简单的水面

nnd,以前发的这篇教程怎么没有了?是我自己误删除了,还是被系统删除了? 找不到存稿了,没有心情再写一遍了.      简单说一下,本篇教程就是实现一个水面的动画,主要是利用动态顶点缓冲,在每一帧都改变顶点的值,从而实现水面的动画.主要参考的是<introduction to 3D game programming with d3d10>这本书,但D3D10和D3D11的动态缓冲实现有所改变.具体算法就不再说了,大家可以参考: Mathematics for 3D Game Programmi