Direct3D学习手记十一:网格二【从.x文件中加载网格】

2024-03-07 20:08

本文主要是介绍Direct3D学习手记十一:网格二【从.x文件中加载网格】,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

上篇我们介绍了自己创建网格的方法,此次我们介绍从.x文件中加载3D模型的方法

.x文件:

.x文件是一种存储3D模型数据的一种文本文件,其存储格式有两种:文本方式和二进制方式,

其中文本方式易于查看,.x文件有其特定的格式

.x文件中存储了顶点数据,材质信息和纹理贴图信息等待,可以通过特定的方式读取其中的内容。

从.x文件中加载网格:

HRESULT D3DXLoadMeshFromX(  LPCTSTR pFilename,  DWORD Options,LPDIRECT3DDEVICE9 pD3DDevice,  LPD3DXBUFFER * ppAdjacency,LPD3DXBUFFER * ppMaterials,  LPD3DXBUFFER * ppEffectInstances,DWORD * pNumMaterials,  LPD3DXMESH * ppMesh);


pFilename为.x文件的名称(注意目录结构)

Options为网格创建标志,有D3DXMESH_MANAGED、D3DXMESH_WRITEONLY、D3DXMESH_32BIT等待

pD3DDevice为设备指针

ppAdjacency保存网格的邻接信息

ppMaterials保存网格的材质信息

ppEffectInstances保存网格的特效信息,暂时不使用置为NULL即可

pNumMaterials保存网格所使用的材质的数目

ppMesh保存创建的网格

例如:

//从.x文件中加载网格HRESULT hr=E_FAIL;LPD3DXBUFFER pAdjBuffer=NULL,pMtrlBuffer=NULL;hr=D3DXLoadMeshFromX(TEXT("Demo_11_Media\\baodao.x"),D3DXMESH_MANAGED,g_pd3dDevice,&pAdjBuffer,&pMtrlBuffer,NULL,&g_dwMtrlNum,&g_pMesh);


其中LPD3DXBUFFER为ID3DXBuffer接口指针,该类型是一种泛型数据,类型为void,可以通过函数:

LPVOID GetBufferPointer();
得到内存地址,并通过强制类型转换为想要的数据格式,

如存储邻接信息(ppAdjacency)的是DWORD类型的数组可使用如下方式转换:

(DWORD *)pAdjBuffer->GetBufferPointer()
另外,保存材质信息的是D3DXMATERIAL结构类型的数组,该结构如下:
typedef struct D3DXMATERIAL {D3DMATERIAL9 MatD3D;LPSTR pTextureFilename;
} D3DXMATERIAL, *LPD3DXMATERIAL;

其中MatD3D是之前介绍过的材质,pTextureFilename指向该材质使用的纹理贴图文件名称字符串(使用时注意目录结构)

可以使用如下方式获得其首地址:

LPD3DXMATERIAL	pMtrl=(LPD3DXMATERIAL)pMtrlBuffer->GetBufferPointer();

获取材质和纹理信息:

从.x文件中加载并创建完网格后,就可以读取其中的材质和纹理信息,并保存下来

全局变量用于保存:

std::vector<D3DMATERIAL9>	g_vecMaterials(0);//材质
DWORD						g_dwMtrlNum=0;//材质数量
std::vector<LPDIRECT3DTEXTURE9>	g_vecPTextures(0);//纹理

读取信息:

//获取材质和纹理CHAR szTextureFile[MAX_PATH];if(NULL!=pMtrlBuffer && 0!=g_dwMtrlNum){LPD3DXMATERIAL	pMtrl=(LPD3DXMATERIAL)pMtrlBuffer->GetBufferPointer();for(DWORD i=0;i<g_dwMtrlNum;i++){pMtrl[i].MatD3D.Ambient=pMtrl[i].MatD3D.Diffuse;//设置材质对环境光的反射g_vecMaterials.push_back(pMtrl[i].MatD3D);//添加材质if(NULL!=pMtrl[i].pTextureFilename)//纹理文件名不为空{ZeroMemory(szTextureFile,sizeof(szTextureFile));sprintf(szTextureFile,"Demo_11_Media\\%s",pMtrl[i].pTextureFilename);LPDIRECT3DTEXTURE9 pTexture=NULL;D3DXCreateTextureFromFileA(g_pd3dDevice,szTextureFile,&pTexture);//创建新纹理g_vecPTextures.push_back(pTexture);//添加纹理}else{//贴图文件为空,则设置当前纹理为NULLg_vecPTextures.push_back(NULL);}}}SAFE_RELEASE(pMtrlBuffer);//释放材质缓存ID3DXBuffer

网格优化:

完成上面的步骤就可以绘制网格了,但是在绘制之前,我们可以对其进行优化来提高绘制效率:
//对网格进行优化,可有可无g_pMesh->OptimizeInplace(D3DXMESHOPT_COMPACT//优化标志,移除无用的顶点和索引|D3DXMESHOPT_ATTRSORT//根据属性ID对三角形图元进行排序|D3DXMESHOPT_VERTEXCACHE,//提高顶点高速缓存的命中率(DWORD *)pAdjBuffer->GetBufferPointer(),//指向尚未优化的网格的邻接数组的指针NULL,//存储优化后的网格的邻接信息NULL,NULL);SAFE_RELEASE(pAdjBuffer);//释放存储邻接信息的缓存

克隆网格:

另外,由于.x文件存储的模型可能顶点结构中不包含顶点法向量,此时要想使用光照效果就必须为其添加顶点法向量,

为此我们可以克隆一分网格,使克隆的网格具有顶点法向量,然后通过计算得到其顶点法向量,可使用如下方式完成:

	//判断顶点是否具有法向量,可有可无if(!(g_pMesh->GetFVF() & D3DFVF_NORMAL)){//没有法向量,则克隆一分LPD3DXMESH	pTempMesh=NULL;g_pMesh->CloneMeshFVF(D3DXMESH_MANAGED,g_pMesh->GetFVF()|D3DFVF_NORMAL,g_pd3dDevice,&pTempMesh);//克隆网格,使其具有法向量if(NULL!=pTempMesh){D3DXComputeNormals(pTempMesh,NULL);//计算顶点法向量SAFE_RELEASE(g_pMesh);g_pMesh=pTempMesh;}}


绘制网格:

//绘制网格for(DWORD i=0;i<g_dwMtrlNum;i++){g_pd3dDevice->SetMaterial(&g_vecMaterials[i]);//设置材质g_pd3dDevice->SetTexture(0,g_vecPTextures[i]);//设置纹理g_pMesh->DrawSubset(i);//绘制子集}


释放纹理:

在程序退出时使用Cleanup函数释放资源,注意要释放所有的纹理:

for(DWORD i=0;i<g_dwMtrlNum;i++){SAFE_RELEASE(g_vecPTextures[i]);//释放纹理}g_vecPTextures.clear();g_vecMaterials.clear();


Setup函数:

Setup函数实现了从.x文件创建网格和信息读取的关键步骤

/****************************************************************
*函数名	:	Setup
*功能		:	创建与初始化资源、缓存、变换等
*输入		:	hWnd:窗口句柄
*输出		:	无
*返回值	:	成功:TRUE		失败:FALSE
****************************************************************/
BOOL			Setup(HWND hWnd)
{if(NULL==hWnd)return FALSE;//创建字体//方式一D3DXCreateFont(g_pd3dDevice,20,14,600,D3DX_DEFAULT,FALSE,DEFAULT_CHARSET,0,0,0,TEXT("微软雅黑"),&g_pHelpFont);//方式二D3DXFONT_DESC fd;ZeroMemory(&fd,sizeof(D3DXFONT_DESC));fd.Height=20;fd.Width=14;fd.Weight=600;fd.MipLevels=D3DX_DEFAULT;fd.Italic=TRUE;fd.CharSet=DEFAULT_CHARSET;fd.OutputPrecision=0;fd.PitchAndFamily=0;fd.Quality=0;_tcscpy_s(fd.FaceName,TEXT("Times New Roman"));D3DXCreateFontIndirect(g_pd3dDevice,&fd,&g_pTipFont);//从.x文件中加载网格HRESULT hr=E_FAIL;LPD3DXBUFFER pAdjBuffer=NULL,pMtrlBuffer=NULL;hr=D3DXLoadMeshFromX(TEXT("Demo_11_Media\\baodao.x"),D3DXMESH_MANAGED,g_pd3dDevice,&pAdjBuffer,&pMtrlBuffer,NULL,&g_dwMtrlNum,&g_pMesh);if(FAILED(hr))return FALSE;//获取材质和纹理CHAR szTextureFile[MAX_PATH];if(NULL!=pMtrlBuffer && 0!=g_dwMtrlNum){LPD3DXMATERIAL	pMtrl=(LPD3DXMATERIAL)pMtrlBuffer->GetBufferPointer();for(DWORD i=0;i<g_dwMtrlNum;i++){pMtrl[i].MatD3D.Ambient=pMtrl[i].MatD3D.Diffuse;//设置材质对环境光的反射g_vecMaterials.push_back(pMtrl[i].MatD3D);//添加材质if(NULL!=pMtrl[i].pTextureFilename)//纹理文件名不为空{ZeroMemory(szTextureFile,sizeof(szTextureFile));sprintf(szTextureFile,"Demo_11_Media\\%s",pMtrl[i].pTextureFilename);LPDIRECT3DTEXTURE9 pTexture=NULL;D3DXCreateTextureFromFileA(g_pd3dDevice,szTextureFile,&pTexture);//创建新纹理g_vecPTextures.push_back(pTexture);//添加纹理}else{//贴图文件为空,则设置当前纹理为NULLg_vecPTextures.push_back(NULL);}}}SAFE_RELEASE(pMtrlBuffer);//释放材质缓存ID3DXBuffer//对网格进行优化,可有可无g_pMesh->OptimizeInplace(D3DXMESHOPT_COMPACT//优化标志,移除无用的顶点和索引|D3DXMESHOPT_ATTRSORT//根据属性ID对三角形图元进行排序|D3DXMESHOPT_VERTEXCACHE,//提高顶点高速缓存的命中率(DWORD *)pAdjBuffer->GetBufferPointer(),//指向尚未优化的网格的邻接数组的指针NULL,//存储优化后的网格的邻接信息NULL,NULL);SAFE_RELEASE(pAdjBuffer);//释放存储邻接信息的缓存//判断顶点是否具有法向量,可有可无if(!(g_pMesh->GetFVF() & D3DFVF_NORMAL)){//没有法向量,则克隆一分LPD3DXMESH	pTempMesh=NULL;g_pMesh->CloneMeshFVF(D3DXMESH_MANAGED,g_pMesh->GetFVF()|D3DFVF_NORMAL,g_pd3dDevice,&pTempMesh);//克隆网格,使其具有法向量if(NULL!=pTempMesh){D3DXComputeNormals(pTempMesh,NULL);//计算顶点法向量SAFE_RELEASE(g_pMesh);g_pMesh=pTempMesh;}}//设置取景变换矩阵D3DXMATRIX matView;D3DXVECTOR3 vEye(0.0f,0.0f,-300.0f);D3DXVECTOR3 vAt(0.0f,0.0f,0.0f);D3DXVECTOR3 vUp(0.0f,1.0f,0.0f);D3DXMatrixLookAtLH(&matView,&vEye,&vAt,&vUp);g_pd3dDevice->SetTransform(D3DTS_VIEW,&matView);//设置投影变换矩阵D3DXMATRIX matProjection;::D3DXMatrixPerspectiveFovLH(&matProjection,D3DX_PI/4.0F,(FLOAT)g_nWidth/(FLOAT)g_nHeight,1.0F,1000.0F);g_pd3dDevice->SetTransform(D3DTS_PROJECTION,&matProjection);//设置光照SetLight(2,g_pd3dDevice);//设置纹理过滤方式g_pd3dDevice->SetSamplerState(0,D3DSAMP_MAGFILTER,D3DTEXF_LINEAR);g_pd3dDevice->SetSamplerState(0,D3DSAMP_MINFILTER,D3DTEXF_LINEAR);g_pd3dDevice->SetSamplerState(0,D3DSAMP_MIPFILTER,D3DTEXF_LINEAR);//显示窗口ShowWindow(hWnd,SW_SHOW);UpdateWindow(hWnd);return TRUE;
}


程序运行结果:



源代码与工程文件下载地址:

百度网盘


这篇关于Direct3D学习手记十一:网格二【从.x文件中加载网格】的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/784669

相关文章

最好用的WPF加载动画功能

《最好用的WPF加载动画功能》当开发应用程序时,提供良好的用户体验(UX)是至关重要的,加载动画作为一种有效的沟通工具,它不仅能告知用户系统正在工作,还能够通过视觉上的吸引力来增强整体用户体验,本文给... 目录前言需求分析高级用法综合案例总结最后前言当开发应用程序时,提供良好的用户体验(UX)是至关重要

MyBatis延迟加载的处理方案

《MyBatis延迟加载的处理方案》MyBatis支持延迟加载(LazyLoading),允许在需要数据时才从数据库加载,而不是在查询结果第一次返回时就立即加载所有数据,延迟加载的核心思想是,将关联对... 目录MyBATis如何处理延迟加载?延迟加载的原理1. 开启延迟加载2. 延迟加载的配置2.1 使用

Android WebView的加载超时处理方案

《AndroidWebView的加载超时处理方案》在Android开发中,WebView是一个常用的组件,用于在应用中嵌入网页,然而,当网络状况不佳或页面加载过慢时,用户可能会遇到加载超时的问题,本... 目录引言一、WebView加载超时的原因二、加载超时处理方案1. 使用Handler和Timer进行超

HarmonyOS学习(七)——UI(五)常用布局总结

自适应布局 1.1、线性布局(LinearLayout) 通过线性容器Row和Column实现线性布局。Column容器内的子组件按照垂直方向排列,Row组件中的子组件按照水平方向排列。 属性说明space通过space参数设置主轴上子组件的间距,达到各子组件在排列上的等间距效果alignItems设置子组件在交叉轴上的对齐方式,且在各类尺寸屏幕上表现一致,其中交叉轴为垂直时,取值为Vert

Ilya-AI分享的他在OpenAI学习到的15个提示工程技巧

Ilya(不是本人,claude AI)在社交媒体上分享了他在OpenAI学习到的15个Prompt撰写技巧。 以下是详细的内容: 提示精确化:在编写提示时,力求表达清晰准确。清楚地阐述任务需求和概念定义至关重要。例:不用"分析文本",而用"判断这段话的情感倾向:积极、消极还是中性"。 快速迭代:善于快速连续调整提示。熟练的提示工程师能够灵活地进行多轮优化。例:从"总结文章"到"用

服务器集群同步时间手记

1.时间服务器配置(必须root用户) (1)检查ntp是否安装 [root@node1 桌面]# rpm -qa|grep ntpntp-4.2.6p5-10.el6.centos.x86_64fontpackages-filesystem-1.41-1.1.el6.noarchntpdate-4.2.6p5-10.el6.centos.x86_64 (2)修改ntp配置文件 [r

【前端学习】AntV G6-08 深入图形与图形分组、自定义节点、节点动画(下)

【课程链接】 AntV G6:深入图形与图形分组、自定义节点、节点动画(下)_哔哩哔哩_bilibili 本章十吾老师讲解了一个复杂的自定义节点中,应该怎样去计算和绘制图形,如何给一个图形制作不间断的动画,以及在鼠标事件之后产生动画。(有点难,需要好好理解) <!DOCTYPE html><html><head><meta charset="UTF-8"><title>06

学习hash总结

2014/1/29/   最近刚开始学hash,名字很陌生,但是hash的思想却很熟悉,以前早就做过此类的题,但是不知道这就是hash思想而已,说白了hash就是一个映射,往往灵活利用数组的下标来实现算法,hash的作用:1、判重;2、统计次数;

零基础学习Redis(10) -- zset类型命令使用

zset是有序集合,内部除了存储元素外,还会存储一个score,存储在zset中的元素会按照score的大小升序排列,不同元素的score可以重复,score相同的元素会按照元素的字典序排列。 1. zset常用命令 1.1 zadd  zadd key [NX | XX] [GT | LT]   [CH] [INCR] score member [score member ...]

【机器学习】高斯过程的基本概念和应用领域以及在python中的实例

引言 高斯过程(Gaussian Process,简称GP)是一种概率模型,用于描述一组随机变量的联合概率分布,其中任何一个有限维度的子集都具有高斯分布 文章目录 引言一、高斯过程1.1 基本定义1.1.1 随机过程1.1.2 高斯分布 1.2 高斯过程的特性1.2.1 联合高斯性1.2.2 均值函数1.2.3 协方差函数(或核函数) 1.3 核函数1.4 高斯过程回归(Gauss