Directx11教程(67) 显示模型文件

      在前面的教程中,我们都是通过在ModelClass中直接产生顶点和索引数据,简单的三角形,立方体等等还好说,毕竟比较简单,如何显示复杂的三维物体呢?特别是利用已有的3D文件,比如obj, 3ds, md2, x等格式的文件,这时,就要利用这些3D格式的解析器,本教程中,我们利用Open Asset Import Library库,来显示各种格式的3D文件(动画文件,暂时不考虑,只考虑静态的3D文件)。

      Open Asset Import Library是一个开源的模型导入库,支持很多格式,它的下载和安装就不介绍了,下面我再myTutorialD3D11_35的基础上,加入使用Assimp库导入3D文件的代码。

     主要就是增加了一个AssimpModelClass类,该类中顶点格式为:

{
D3DXVECTOR3 position;
D3DXVECTOR3 normal; //法向
D3DXVECTOR2 texture; //纹理坐标
D3DXVECTOR4 Kd;  //材质漫反射系数
D3DXVECTOR4 Ks;  //材质的高光系数
};

 

产生顶点和索引缓冲的函数为AssimpModelClass,我们通过该函数产生顶点缓冲和索引缓冲。

bool  AssimpModelClass::LoadModel(ID3D11Device* device, std::string filename)
{
    HRESULT result;

注意:首先我们会定义一个assimp导入器类,用该类读入模型文件,我们会做2趟循环,第一趟循环得到顶点和索引的数量,然后创建顶点和索引临时缓冲,用来保存顶点和索引数据,第二趟循环从模型文件中读取顶点和索引的数据,最后创建顶点和索引缓冲。
    Assimp::Importer importer;
    VertexType* vertices;
    unsigned long* indices;
    D3D11_BUFFER_DESC vertexBufferDesc, indexBufferDesc;
    D3D11_SUBRESOURCE_DATA vertexData, indexData;

    const aiScene* scene = importer.ReadFile(filename,aiProcessPreset_TargetRealtime_Quality);
   
    if(!scene)
    {
        MessageBoxA(NULL, importer.GetErrorString(), "Error", MB_OK);
        return false;
    }

    int m =0;
    //第一趟扫描,得到顶点和索引计数
    for(m=0; m<scene->mNumMeshes; ++m )
    {
        //第m个mesh
        aiMesh* aiMesh = scene->mMeshes[m];

        m_vertexCount += aiMesh->mNumVertices;
        m_indexCount  += aiMesh->mNumFaces*3;
    }

    // 创建顶点临时缓冲.
    vertices = new VertexType[m_vertexCount];
    if(!vertices)
    {
        return false;
    }

    // 创建索引临时缓冲.
    indices = new unsigned long[m_indexCount];
    if(!indices)
    {
        return false;
    }

    //临时的顶点和索引指针
    int index1 = 0;
    int index2 = 0;
    int i = 0;

    //第二趟循环,得到每个顶点和索引的值
    for(m=0; m<scene->mNumMeshes; ++m )
    {
        //第m个mesh
        aiMesh* aiMesh = scene->mMeshes[m];

        if(!aiMesh->HasNormals() || !aiMesh->HasTextureCoords(0))
        {
            MessageBox(NULL, L"模型文件中没有纹理坐标或者法向信息", L"Error", MB_OK);
            return false;
        }

        int vertexCount = aiMesh->mNumVertices;
        for(i = 0;i < vertexCount;++i)
        {
            vertices[index1].position = D3DXVECTOR3(aiMesh->mVertices[i].x, aiMesh->mVertices[i].y, aiMesh->mVertices[i].z);
            vertices[index1].normal = D3DXVECTOR3(aiMesh->mNormals[i].x, aiMesh->mNormals[i].y, aiMesh->mNormals[i].z);
            vertices[index1].texture = D3DXVECTOR2(aiMesh->mTextureCoords[0][i].x, aiMesh->mTextureCoords[0][i].y);
            vertices[index1].Kd = D3DXVECTOR4(1.0, 1.0, 1.0,1.0);
            vertices[index1].Ks = D3DXVECTOR4(0.2, 0.2, 0.2,1.0);
            index1++;
        }

        for (i = 0; i < aiMesh->mNumFaces;++i)
        {
            const aiFace& Face = aiMesh->mFaces[i];
            //assert(Face.mNumIndices == 3);
            indices[index2] = Face.mIndices[0];
            index2++;
            indices[index2] = Face.mIndices[1];
            index2++;
            indices[index2] = Face.mIndices[2];
            index2++;

        }
    }
    // 设置顶点缓冲描述
    vertexBufferDesc.Usage = D3D11_USAGE_DEFAULT;
    vertexBufferDesc.ByteWidth = sizeof(VertexType) * m_vertexCount;
    vertexBufferDesc.BindFlags = D3D11_BIND_VERTEX_BUFFER;
    vertexBufferDesc.CPUAccessFlags = 0;
    vertexBufferDesc.MiscFlags = 0;
    vertexBufferDesc.StructureByteStride = 0;

    // 指向保存顶点数据的临时缓冲.
    vertexData.pSysMem = vertices;
    vertexData.SysMemPitch = 0;
    vertexData.SysMemSlicePitch = 0;

    // 创建顶点缓冲.
    result = device->CreateBuffer(&vertexBufferDesc, &vertexData, &m_vertexBuffer);
    if(FAILED(result))
    {
        HR(result);
        return false;
    }

    // 设置索引缓冲描述.
    indexBufferDesc.Usage = D3D11_USAGE_DEFAULT;
    indexBufferDesc.ByteWidth = sizeof(unsigned long) * m_indexCount;
    indexBufferDesc.BindFlags = D3D11_BIND_INDEX_BUFFER;
    indexBufferDesc.CPUAccessFlags = 0;
    indexBufferDesc.MiscFlags = 0;
    indexBufferDesc.StructureByteStride = 0;

    // 指向存临时索引缓冲.
    indexData.pSysMem = indices;
    indexData.SysMemPitch = 0;
    indexData.SysMemSlicePitch = 0;

    // 创建索引缓冲.
    result = device->CreateBuffer(&indexBufferDesc, &indexData, &m_indexBuffer);
    if(FAILED(result))
    {
        HR(result);
        return false;
    }

    // 释放临时缓冲.
    delete [] vertices;
    vertices = 0;

    delete [] indices;
    indices = 0;

    return true;

}

     随后在GraphicsClass中,我们会增加AssimpModelClass变量,并用LightTexShader来渲染该类装入的模型,主要的代码如下:

1、初始化的代码

// 创建assimp模型对象
m_AssimpModel = new AssimpModelClass;
if(!m_AssimpModel)
    {
    return false;
    }
// 初始化坐标assimp模型对象.faerie.md2,tiny.x
result = m_AssimpModel->Initialize(m_D3D->GetDevice(), "tiny.x");
if(!result)
    {
    MessageBox(hwnd, L"Could not initialize the axis model object.", L"Error", MB_OK);
    return false;
    }

2、渲染的代码:

D3DXMatrixScaling(&worldMatrix4, 0.02, 0.02,0.02);

m_AssimpModel->Render(m_D3D->GetDeviceContext());
//用light shader渲染,faerie2.bmp,Tiny_skin.dds
result = m_LightTexShader->Render(m_D3D->GetDeviceContext(), m_AssimpModel->GetIndexCount(), worldMatrix4, viewMatrix, projectionMatrix,
             light, material, camera,m_TexManager->createTex(m_D3D->GetDevice(),string("Tiny_skin.dds")));
if(!result)
    {
    return false;
    }

程序执行后的效果图如下:

装入x格式文件tiny.x

装入md2格式文件faerie.md2

     

完整的代码请参考:

工程文件myTutorialD3D11_64

代码下载:

稍后提供

 

 

时间: 2024-09-29 09:01:51

Directx11教程(67) 显示模型文件的相关文章

NSIS使用教程(安装包制作安装文件教程,如何封装打包文件) 中文版

原文:NSIS使用教程(安装包制作安装文件教程,如何封装打包文件) 中文版 nsis中文版(Nullsoft Scriptable Install System)是一个专业的开源的可以用来封闭Windows程序的实用工具,是一个开源的 Windows 系统下安装程序制作程序.nsis它功能强,源码是直接使用C语言编写而成,并且可以直接到nsis官网下载所有nsis版本,并且提供了详细的帮助文档,方便用户制作时使用.安装页面可以使用Page自定义界面.卸载页面可以配置让用户选择是否删除用户文件.系

php scandir遍历显示所有文件与文件夹下的文件

php教程 scandir遍历显示所有文件与文件夹下的文件,方法很简单我们只要利用is_dir判断再递归查找一次,这样就可以把遍历目录下所有文件了. 目录遍历 <?php   function numfilesindir ($thedir){     if (is_dir ($thedir)){       $scanarray = scandir ($thedir);       for ($i = 0; $i < count ($scanarray); $i++){         if

ASP.NET MVC 5 学习教程:添加模型

原文 ASP.NET MVC 5 学习教程:添加模型 起飞网 ASP.NET MVC 5 学习教程目录: 添加控制器 添加视图 修改视图和布局页 控制器传递数据给视图 添加模型 创建连接字符串 通过控制器访问模型的数据 生成的代码详解 使用 SQL Server LocalDB Edit方法和Edit视图详解 添加查询 Entity Framework 数据迁移之添加字段 添加验证 Details 和 Delete 方法详解 在本节中,我们将添加一些管理电影数据库的类,这些类在ASP.NET M

在C++中用ActiveX控件显示PDF文件

PDF介绍 PDF是"Portable Document Format 可移植文档格式"的缩写,它是Adobe公司开发的一种电子文档格式,已经成为国际通用的电子文档交换事实标准.PDF文件可以在各种平台下阅读.编辑.发布.该文件格式支持字体.图像.甚至任何附件的嵌入. Adobe公司提供了两种软件用于处理PDF文档,分别是Acrobat及Acrobat Reader:前者功能强大,不仅能阅读,还可以创建或编辑PDF文档,但要收费:后者只具备阅读功能,为免费软件. 用ActiveX控件显

串接样式表(CSS)来显示XML文件

css|xml|显示|样式表     在本章中,你将学习显示XML 文件于Microsoft Internet Explorer 5 中的第一种方法:串接样式表(CSS).串接样式表是一个包含安排XML 文件中元素相关指令的档案.因为你已经利用XML创造了自己的元素,浏览器并不知道如何适当地显示这些元素.建立串接样式表并将它链接到XML 文件中便是一种告诉浏览器如何显示文件中每个元素的方法.附加串接样式表的XML 文件可以直接在Internet Explorer 5 中被开启.你不需要使用HTM

CB环境中实现在菜单中显示历史文件列表

本文介绍了如何在Windows环境下实现菜单中显示历史文件列表,同时概要介绍了Tregistry类的使用方法. 现在,在许多Windows应用程序都具有这样一个功能:可以在文件菜单下面显示列出最近访问过的文件,这使用户很容易再次访问曾经访问过的文件.在过去几年中,这项技术以成为拥有文件菜单项的应用程序的共同特色:如Wps系列和Office系列.在以前的DOS环境中,程序员一般通过创建一个文件用来记录文件列表:那么在Windows环境中,还有其他的方法吗?最近笔者利用C++ Builder5.0

不让tomcat显示目录文件列表的配置方法

 这篇文章主要介绍了不让tomcat显示目录文件列表的配置方法,这里需要修改conf/web.xml文件,需要的朋友可以参考下 修改conf/web.xml文件(把listings的参数改为false)    代码如下: <servlet> <servlet-name>default</servlet-name> <servlet-class>org.apache.catalina.servlets.DefaultServlet</servlet-cl

如何让Windows只显示某些文件扩展名

Windows的"资源管理器"有一个显示或隐藏文件扩展名的选项.如果选择了显示扩展名,一旦邮箱收到了"图片.jpg.exe"之类的恶意文件,完整的扩展名将使这类文件的把戏无所遁形;如果隐藏扩展名,"图片.jpg.exe"将显示为"图片.jpg",很容易上当.但另一方面,总是显示所有文件的扩展名看起来不够整洁,而且增加了无意之中改变文件类型的风险.最理想的方案是只显示某些可能隐含风险的文件扩展名,与此同时,将其余文件的扩展名隐藏

苹果Mac怎么显示资源库文件夹

  苹果Mac电脑怎么显示资源库文件夹 1.打开 Finder,然后进入"个人"(图标是小房子)文件夹, 按快捷键 Command + J 显示简介.然后里面就有能够显示资源库文件夹的选项了,勾选即可. 2.有些朋友发现没有这个"个人"(图标是小房子)文件夹怎么办?OK,没关系,现在小编再教你怎么找到"个人"(图标是小房子)文件夹. 1)打开 Finder,左上角"偏好设置". 2)打开"偏好设置"后,选择