在之前的DirectX例子里我用的模型是.x文件,DirectX有一个方法D3DXLoadMeshFromX可以加载.x模型,但是这里有个问题,.x文件是没法用文本编辑器打开查看结构的,这里我来演示一下如何解析.obj模型.
首先让我们看一下.obj模型的组成部分以及结构,一个完整的obj模型一共分为三个部分:obj模型文件,mtl材质文件,纹理贴图;其中obj文件和mtl文件是可以用文本编辑器打开的,先打开obj文件,可以看到这样的内容:
v -3.000767 2.993211 2.014205
v -3.000767 -0.006789 2.014205
v -2.750767 2.993211 2.014205
v -2.750767 -0.006789 2.014205
v -2.750767 2.993211 2.014205
v -2.750767 -0.006789 2.014205
v -2.750767 2.993211 -1.985795
v -2.750767 -0.006789 -1.985795
v -2.750767 2.993211 -1.985795
vt 0.948633 0.500977
vt 0.948633 0.000977
vt 0.998633 0.500977
vt 0.998633 0.000977
vt 0.000000 0.500000
vt 0.000000 0.000000
vt 1.000000 0.500000
vt 1.000000 0.000000
vt 1.000000 0.501343
vt 0.000000 0.501343
vt 1.000000 0.438843
vn 0.000000 0.000000 1.000000
vn 1.000000 0.000000 0.000000
vn 0.000000 0.000000 -1.000000
vn -1.000000 0.000000 0.000000
vn 0.000000 1.000000 0.000000
vn 0.000000 -1.000000 0.000000
vn -0.000000 -0.707107 -0.707107
vn 0.000000 0.707107 0.707107
vn -0.000000 0.707107 -0.707106
g Left
usemtl wood
s 1
f 1/1/1 2/2/1 3/3/1
f 2/2/1 4/4/1 3/3/1
s 2
f 5/5/2 6/6/2 7/7/2
f 6/6/2 8/8/2 7/7/2
s 1
f 9/3/3 10/4/3 11/1/3
f 10/4/3 12/2/3 11/1/3
s 2
f 13/7/4 14/8/4 15/5/4
f 14/8/4 16/6/4 15/5/4
s 3
f 17/9/5 18/10/5 19/11/5
f 18/10/5 20/12/5 19/11/5
f 21/10/6 22/9/6 23/12/6
f 22/9/6 24/11/6 23/12/6
首先v和其后三个值表示一个顶点的xyz坐标值;
vt和其后两个或者三个值表示顶点的纹理坐标uv(w);
vn和其后三个值表示顶点的法向量;
g表示一组面;
usemtl表示这个组用的mtl文件里那个材质的名称;
f及其后三组值表示一个面的三个 顶点/纹理/法线 在之前v,vt,vn集合里边的索引值.
打开mtl文件就会看到:
newmtl wood
illum 2
Kd 0.800000 0.800000 0.800000
Ka 0.200000 0.200000 0.200000
Ks 0.000000 0.000000 0.000000
Ke 0.000000 0.000000 0.000000
Ns 0.000000
map_Ka house/house.bmp
map_Kd house/house.bmp
为了简单代码里只用到了map_Kd,它表示漫反射所使用的纹理名称;
其他的是光照属性,代码里采用默认材质的光照属性;
为了解析模型,首先要把材质文件给解析出来,把材质名称与纹理名称放入数组,这样解析obj的时候通过材质名称就能够找到对应材质数组的下标就能找到对应的纹理:
void MtlObj::getLineNum() {
ifstream infile(path.c_str()); //打开指定文件
string sline;//每一行
while(getline(infile,sline)) {//从指定文件逐行读取
if(sline[0]=='n'&&sline[1]=='e')//newmtl
mtlNum++;
}
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
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++;
else
vNum++;
}
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(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>s1>>f2>>f3>>f4;
vnArr[ii][0]=f2;
vnArr[ii][1]=f3;
vnArr[ii][2]=f4;
ii++;
} else if(sline[1]=='t') {//vt
istringstream ins(sline);
ins>>s1>>f2>>f3>>f4;
vtArr[tt][0]=f2;
vtArr[tt][1]=1-f3;
vtArr[tt][2]=f4;
tt++;
} else {//v
istringstream 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;//去掉f
int 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;
然后通过各种索引组装三角形:
void ModelObj::initTriangles() {
for (int i=0;i
组装结束,清除读取文件用的内存:
void ModelObj::clearTriangles() {
for(int i=0;i
ObjModel::ObjModel(ModelObj* obj) {
objLoader=obj;
D3DXCreateMeshFVF(objLoader->fNum,objLoader->fNum*3,D3DXMESH_MANAGED,normalTexVertFvf,d3d,&mesh);
initVertices();
initTextures();
DWORD* aAdjacency=new DWORD[objLoader->fNum*3];
mesh->GenerateAdjacency(0.001,aAdjacency);
mesh->OptimizeInplace(D3DXMESHOPT_ATTRSORT|D3DXMESHOPT_VERTEXCACHE,aAdjacency,NULL,NULL,NULL);
delete[] aAdjacency;
}
void ObjModel::initVertices() {
NormalTexVertex* vertices=NULL;
mesh->LockVertexBuffer(0,(void**)&vertices);
for(int i=0;ifNum*3;i++)
vertices[i]=objLoader->vertices[i];
mesh->UnlockVertexBuffer();
WORD* indices=NULL;
mesh->LockIndexBuffer(0,(void**)&indices);
for(int i=0;ifNum*3;i++)
indices[i]=objLoader->indices[i];
mesh->UnlockIndexBuffer();
DWORD* attributes=NULL;
mesh->LockAttributeBuffer(0,&attributes);
for(int i=0;ifNum;i++)
attributes[i]=objLoader->groupArr[i]-1;
mesh->UnlockAttributeBuffer();
}
void ObjModel::initTextures() {
objLoader->mtl->getLength(mtlNum);
textures=new LPDIRECT3DTEXTURE9[mtlNum];
for(int i=0;imtl->textures[i];
D3DXCreateTextureFromFile(d3d,texFile.c_str(),&textures[i]);
}
}
void ObjModel::render() {
for(DWORD i=0;i<(DWORD)objLoader->groupNum;i++) {
map::iterator itor=objLoader->groupMtlMap.find((int)(i+1));
int mtlId=itor->second;
d3d->SetTexture(1,textures[mtlId]);
mesh->DrawSubset(i);
}
}
void initObjModel() {
objLoader=new ModelObj(MODEL_TANK,MTL_TANK);
objModel=new ObjModel(objLoader);
}
void renderObjModel() {
objModel->render();
}
void releaseObjModel() {
delete objModel;
delete objLoader;
}
模型加载器的代码已经写好,下载地址: 点击下载