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

相关文章

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

【学习笔记】 陈强-机器学习-Python-Ch15 人工神经网络(1)sklearn

系列文章目录 监督学习:参数方法 【学习笔记】 陈强-机器学习-Python-Ch4 线性回归 【学习笔记】 陈强-机器学习-Python-Ch5 逻辑回归 【课后题练习】 陈强-机器学习-Python-Ch5 逻辑回归(SAheart.csv) 【学习笔记】 陈强-机器学习-Python-Ch6 多项逻辑回归 【学习笔记 及 课后题练习】 陈强-机器学习-Python-Ch7 判别分析 【学

系统架构师考试学习笔记第三篇——架构设计高级知识(20)通信系统架构设计理论与实践

本章知识考点:         第20课时主要学习通信系统架构设计的理论和工作中的实践。根据新版考试大纲,本课时知识点会涉及案例分析题(25分),而在历年考试中,案例题对该部分内容的考查并不多,虽在综合知识选择题目中经常考查,但分值也不高。本课时内容侧重于对知识点的记忆和理解,按照以往的出题规律,通信系统架构设计基础知识点多来源于教材内的基础网络设备、网络架构和教材外最新时事热点技术。本课时知识

线性代数|机器学习-P36在图中找聚类

文章目录 1. 常见图结构2. 谱聚类 感觉后面几节课的内容跨越太大,需要补充太多的知识点,教授讲得内容跨越较大,一般一节课的内容是书本上的一章节内容,所以看视频比较吃力,需要先预习课本内容后才能够很好的理解教授讲解的知识点。 1. 常见图结构 假设我们有如下图结构: Adjacency Matrix:行和列表示的是节点的位置,A[i,j]表示的第 i 个节点和第 j 个