DirectX的OBJ模型加载与渲染

在之前的DirectX例子里我用的模型是.x文件,DirectX有一个方法D3DXLoadMeshFromX可以加载.x模型,但是这里有个问题,.x文件是没法用文本编辑器打开查看结构的,这里我来演示一下如何解析.obj模型.

首先让我们看一下.obj模型的组成部分以及结构,一个完整的obj模型一共分为三个部分:obj模型文件,mtl材质文件,纹理贴图;其中obj文件和mtl文件是可以用文本编辑器打开的,先打开obj文件,可以看到这样的内容:

v -3.000767 2.993211 2.014205v -3.000767 -0.006789 2.014205v -2.750767 2.993211 2.014205v -2.750767 -0.006789 2.014205v -2.750767 2.993211 2.014205v -2.750767 -0.006789 2.014205v -2.750767 2.993211 -1.985795v -2.750767 -0.006789 -1.985795v -2.750767 2.993211 -1.985795vt 0.948633 0.500977vt 0.948633 0.000977vt 0.998633 0.500977vt 0.998633 0.000977vt 0.000000 0.500000vt 0.000000 0.000000vt 1.000000 0.500000vt 1.000000 0.000000vt 1.000000 0.501343vt 0.000000 0.501343vt 1.000000 0.438843vn 0.000000 0.000000 1.000000vn 1.000000 0.000000 0.000000vn 0.000000 0.000000 -1.000000vn -1.000000 0.000000 0.000000vn 0.000000 1.000000 0.000000vn 0.000000 -1.000000 0.000000vn -0.000000 -0.707107 -0.707107vn 0.000000 0.707107 0.707107vn -0.000000 0.707107 -0.707106g Leftusemtl woods 1f 1/1/1 2/2/1 3/3/1f 2/2/1 4/4/1 3/3/1s 2f 5/5/2 6/6/2 7/7/2f 6/6/2 8/8/2 7/7/2s 1f 9/3/3 10/4/3 11/1/3f 10/4/3 12/2/3 11/1/3s 2f 13/7/4 14/8/4 15/5/4f 14/8/4 16/6/4 15/5/4s 3f 17/9/5 18/10/5 19/11/5f 18/10/5 20/12/5 19/11/5f 21/10/6 22/9/6 23/12/6f 22/9/6 24/11/6 23/12/6让我来解释一下字段:

首先v和其后三个值表示一个顶点的xyz坐标值;

vt和其后两个或者三个值表示顶点的纹理坐标uv(w);

vn和其后三个值表示顶点的法向量;

g表示一组面;

usemtl表示这个组用的mtl文件里那个材质的名称;

f及其后三组值表示一个面的三个 顶点/纹理/法线 在之前v,vt,vn集合里边的索引值.

打开mtl文件就会看到:

newmtl woodillum 2Kd 0.800000 0.800000 0.800000Ka 0.200000 0.200000 0.200000Ks 0.000000 0.000000 0.000000Ke 0.000000 0.000000 0.000000Ns 0.000000map_Ka house/house.bmpmap_Kd house/house.bmpnewmtl后面的是材质名对应obj文件usemtl后面跟的值;

为了简单代码里只用到了map_Kd,它表示漫反射所使用的纹理名称;

其他的是光照属性,代码里采用默认材质的光照属性;

为了解析模型,首先要把材质文件给解析出来,把材质名称与纹理名称放入数组,这样解析obj的时候通过材质名称就能够找到对应材质数组的下标就能找到对应的纹理:

void MtlObj::getLineNum() {ifstream infile(path.c_str()); //打开指定文件string sline;//每一行while(getline(infile,sline)) {//从指定文件逐行读取if(sline[0]=='n'&&sline[1]=='e')//newmtlmtlNum++;}infile.close();}void MtlObj::readfile() {getLineNum();names=new string[mtlNum];textures=new string[mtlNum];int n=0;int t=0;ifstream infile(path.c_str()); //打开指定文件string sline;//每一行string value,name,texture;while(getline(infile,sline)) {//从指定文件逐行读取if(sline!="") {istringstream ins(sline);ins>>value;if(value=="newmtl") {ins>>name;names[n]=name;n++;} else if(value=="map_Kd") {ins>>texture;textures[t]=texture;t++;}}}infile.close();}int MtlObj::getIndexByName(string name) {int index=-1;for(int i=0;i<mtlNum;i++) {if(names[i]==name) {index=i;break;}}return index;}准备好了材质信息之后开始解析obj文件:

void ModelObj::getLineNum() {ifstream infile(path.c_str()); //打开指定文件string sline;//每一行while(getline(infile,sline)) {//从指定文件逐行读取if(sline[0]=='v') {if(sline[1]=='n')vnNum++;else if(sline[1]=='t')vtNum++;elsevNum++;}if(sline[0]=='f')fNum++;}infile.close();ifstream ifile(path.c_str());string value,um,group,face;mtArr=new string[fNum];groupArr=new int[fNum];groupNum=0;int fi=0;while(getline(ifile,sline)) {istringstream ins(sline);ins>>value;if(value=="usemtl") {ins>>um;int mtlId=mtl->getIndexByName(um);groupMtlMap.insert(pair<int,int>(groupNum,mtlId));} else if(value=="g") {ins>>group;groupNum++;} else if(value=="f") {ins>>face;mtArr[fi]=um;groupArr[fi]=groupNum;fi++;}}ifile.close();}通过材质名字查找到该材质在之前的材质数组中的id,这边需要groupMtlMap保存面组id与材质id,那样在渲染时就可以通过面组id找到对应的纹理贴图.

取得基本信息后读入文件的详细内容:

void ModelObj::readfile() {getLineNum();vertices=new NormalTexVertex[fNum*3];indices=new int[fNum*3];//new二维数组vArr=new float*[vNum];for (int i=0;i<vNum;i++)vArr[i]=new float[3];vnArr=new float*[vnNum];for (int i=0;i<vnNum;i++)vnArr[i]=new float[3];vtArr=new float*[vtNum];for (int i=0;i<vtNum;i++)vtArr[i]=new float[3];fvArr=new int*[fNum];ftArr=new int*[fNum];fnArr=new int*[fNum];for (int i=0;i<fNum;i++) {fvArr[i]=new int[3];ftArr[i]=new int[3];fnArr[i]=new int[3];}ifstream infile(path.c_str());string sline;//每一行int ii=0,tt=0,jj=0,kk=0;std::string s1;float f2,f3,f4;while(getline(infile,sline)) {if(sline[0]=='v') {if(sline[1]=='n') {//vnistringstream ins(sline);ins>>s1>>f2>>f3>>f4;vnArr[ii][0]=f2;vnArr[ii][1]=f3;vnArr[ii][2]=f4;ii++;} else if(sline[1]=='t') {//vtistringstream ins(sline);ins>>s1>>f2>>f3>>f4;vtArr[tt][0]=f2;vtArr[tt][1]=1-f3;vtArr[tt][2]=f4;tt++;} else {//vistringstream ins(sline);ins>>s1>>f2>>f3>>f4;vArr[jj][0]=f2;vArr[jj][1]=f3;vArr[jj][2]=f4;jj++;}}if (sline[0]=='f') { //存储面istringstream in(sline);float a;in>>s1;//去掉fint i,k;for(i=0;i<3;i++) {in>>s1;//取出第一个顶点和法线索引a=0;for(k=0;s1[k]!='/';k++)a=a*10+(s1[k]-48);fvArr[kk][i]=a;a=0;for(k=k+1;s1[k]!='/';k++)a=a*10+(s1[k]-48);ftArr[kk][i]=a;a=0;for(k=k+1;s1[k];k++)a=a*10+(s1[k]-48);fnArr[kk][i]=a;}kk++;}}infile.close();}这里需要面数3倍的顶点,因为有n个三角形就有n*3个顶点,由于顶点之间可能不是共享法线和纹理坐标数据,因此不同三角形在同一个位置的顶点要分开来存放;

由于DirectX的纹理坐标轴v是朝下而模型的v坐标轴朝上,那么读取的纹理坐标v必须变成1-v才能有DirectX正确渲染,于是就有类似vtArr[tt][1]=1-f3这样的做法;

渲染的时候需要索引指针,大小是模型面的数量*3;

然后通过各种索引组装三角形:

可以有一个人陪着你,也可以你一个人,总之那一刻,

DirectX的OBJ模型加载与渲染

相关文章:

你感兴趣的文章:

标签云: