182 lines
6.7 KiB
C++
182 lines
6.7 KiB
C++
/******************************************************************************
|
|
Plush Version 1.2
|
|
read_cob.c
|
|
ASCII COB Object Reader
|
|
Copyright (c) 1996-2000, Justin Frankel
|
|
******************************************************************************/
|
|
|
|
#include "plush.h"
|
|
|
|
#define PL_COB_MAX_LINELENGTH 1024
|
|
|
|
pl_Obj *plReadCOBObj(char *fn, pl_Mat *mat) {
|
|
FILE *fp = fopen(fn,"rt");
|
|
int p1,m1,p2,m2,p3,m3;
|
|
char temp_string[PL_COB_MAX_LINELENGTH];
|
|
float TransMatrix[4][4];
|
|
pl_Obj *obj;
|
|
pl_sInt32 x,i2;
|
|
int numVertices, numMappingVertices, numFaces, i;
|
|
pl_Float *MappingVertices = 0;
|
|
if (!fp) return 0;
|
|
|
|
fgets(temp_string,PL_COB_MAX_LINELENGTH,fp);
|
|
if (memcmp("Caligari",temp_string,8)) { fclose(fp); return 0; }
|
|
|
|
do {
|
|
fgets(temp_string,PL_COB_MAX_LINELENGTH,fp);
|
|
} while (!feof(fp) && memcmp("Transform",temp_string,9));
|
|
if (feof(fp)) { fclose(fp); return 0; }
|
|
fgets(temp_string,PL_COB_MAX_LINELENGTH,fp);
|
|
sscanf(temp_string,"%f %f %f %f",
|
|
&TransMatrix[0][0],&TransMatrix[0][1],&TransMatrix[0][2],&TransMatrix[0][3]);
|
|
fgets(temp_string,PL_COB_MAX_LINELENGTH,fp);
|
|
sscanf(temp_string,"%f %f %f %f",
|
|
&TransMatrix[1][0],&TransMatrix[1][1],&TransMatrix[1][2],&TransMatrix[1][3]);
|
|
fgets(temp_string,PL_COB_MAX_LINELENGTH,fp);
|
|
sscanf(temp_string,"%f %f %f %f",
|
|
&TransMatrix[2][0],&TransMatrix[2][1],&TransMatrix[2][2],&TransMatrix[2][3]);
|
|
fgets(temp_string,PL_COB_MAX_LINELENGTH,fp);
|
|
sscanf(temp_string,"%f %f %f %f",
|
|
&TransMatrix[3][0],&TransMatrix[3][1],&TransMatrix[3][2],&TransMatrix[3][3]);
|
|
|
|
do {
|
|
fgets(temp_string,PL_COB_MAX_LINELENGTH,fp);
|
|
} while (!feof(fp) && memcmp("World Vertices",temp_string,12));
|
|
if (feof(fp) || sscanf(temp_string,"World Vertices %d",&numVertices) != 1)
|
|
{ fclose(fp); return 0; }
|
|
|
|
rewind(fp);
|
|
do {
|
|
fgets(temp_string,PL_COB_MAX_LINELENGTH,fp);
|
|
} while (!feof(fp) && memcmp("Texture Vertices",temp_string,16));
|
|
if (feof(fp) ||
|
|
sscanf(temp_string,"Texture Vertices %d",&numMappingVertices) != 1) {
|
|
fclose(fp); return 0;
|
|
}
|
|
|
|
rewind(fp);
|
|
do {
|
|
fgets(temp_string,PL_COB_MAX_LINELENGTH,fp);
|
|
} while (!feof(fp) && memcmp("Faces",temp_string,5));
|
|
if (feof(fp) || sscanf(temp_string,"Faces %d",&numFaces) != 1) {
|
|
fclose(fp); return 0;
|
|
}
|
|
for (x = numFaces; x; x--) {
|
|
fgets(temp_string,PL_COB_MAX_LINELENGTH,fp);
|
|
if (feof(fp) || sscanf(temp_string+4," verts %d",&i) != 1 || i < 3) {
|
|
fclose(fp);
|
|
return 0;
|
|
}
|
|
numFaces += i-3;
|
|
fgets(temp_string,PL_COB_MAX_LINELENGTH,fp);
|
|
}
|
|
obj = new pl_Obj(numVertices,numFaces);
|
|
if (!obj) { fclose(fp); return 0; }
|
|
rewind(fp);
|
|
do {
|
|
fgets(temp_string,PL_COB_MAX_LINELENGTH,fp);
|
|
} while (!feof(fp) && memcmp("World Vertices",temp_string,12));
|
|
if (feof(fp)) { delete obj; fclose(fp); return 0; }
|
|
for (x = 0; x < numVertices; x ++) {
|
|
float xp, yp, zp;
|
|
fgets(temp_string,PL_COB_MAX_LINELENGTH,fp);
|
|
if (feof(fp) ||
|
|
sscanf(temp_string,"%f %f %f", &xp, &yp, &zp) != 3) {
|
|
delete obj; fclose(fp); return 0;
|
|
}
|
|
obj->Vertices.Get()[x].x = (TransMatrix[0][0]*xp+TransMatrix[0][1]*yp+
|
|
TransMatrix[0][2]*zp+TransMatrix[0][3]);
|
|
obj->Vertices.Get()[x].y = (TransMatrix[1][0]*xp+TransMatrix[1][1]*yp+
|
|
TransMatrix[1][2]*zp+TransMatrix[1][3]);
|
|
obj->Vertices.Get()[x].z = (TransMatrix[2][0]*xp+TransMatrix[2][1]*yp+
|
|
TransMatrix[2][2]*zp+TransMatrix[2][3]);
|
|
}
|
|
rewind(fp);
|
|
do {
|
|
fgets(temp_string,PL_COB_MAX_LINELENGTH,fp);
|
|
} while (!feof(fp) && memcmp("Texture Vertices",temp_string,16));
|
|
if (!feof(fp)) {
|
|
MappingVertices = (pl_Float *)
|
|
malloc(sizeof(pl_Float ) * numMappingVertices * 2);
|
|
if (MappingVertices) {
|
|
for (x = 0; x < numMappingVertices; x ++) {
|
|
float p1, p2;
|
|
fgets(temp_string,PL_COB_MAX_LINELENGTH,fp);
|
|
if (feof(fp) || sscanf(temp_string,"%f %f", &p1, &p2) != 2) {
|
|
free(MappingVertices); delete obj; fclose(fp); return 0;
|
|
}
|
|
MappingVertices[x*2] = p1;
|
|
MappingVertices[x*2+1] = p2;
|
|
}
|
|
}
|
|
}
|
|
rewind(fp);
|
|
do {
|
|
fgets(temp_string,PL_COB_MAX_LINELENGTH,fp);
|
|
} while (!feof(fp) && memcmp("Faces",temp_string,5));
|
|
if (feof(fp)) {
|
|
if (MappingVertices) free(MappingVertices);
|
|
delete obj; fclose(fp); return 0;
|
|
}
|
|
for (x = 0; x < numFaces; x ++) {
|
|
fgets(temp_string,PL_COB_MAX_LINELENGTH,fp);
|
|
sscanf(temp_string+4," verts %d",&i);
|
|
fgets(temp_string,PL_COB_MAX_LINELENGTH,fp);
|
|
if (i == 3) {
|
|
if (feof(fp) || sscanf(temp_string,"<%d,%d> <%d,%d> <%d,%d>",
|
|
&p3,&m3,&p2,&m2,&p1,&m1) != 6) {
|
|
if (MappingVertices) free(MappingVertices);
|
|
delete obj; fclose(fp); return 0;
|
|
}
|
|
obj->Faces.Get()[x].VertexIndices[0] = p1;
|
|
obj->Faces.Get()[x].VertexIndices[1] = p2;
|
|
obj->Faces.Get()[x].VertexIndices[2] = p3;
|
|
if (MappingVertices) {
|
|
obj->Faces.Get()[x].MappingU[0][0] = MappingVertices[m1*2];
|
|
obj->Faces.Get()[x].MappingV[0][0] = MappingVertices[m1*2+1];
|
|
obj->Faces.Get()[x].MappingU[0][1] = MappingVertices[m2*2];
|
|
obj->Faces.Get()[x].MappingV[0][1] = MappingVertices[m2*2+1];
|
|
obj->Faces.Get()[x].MappingU[0][2] = MappingVertices[m3*2];
|
|
obj->Faces.Get()[x].MappingV[0][2] = MappingVertices[m3*2+1];
|
|
}
|
|
obj->Faces.Get()[x].Material = mat;
|
|
} else {
|
|
int p[16],m[16];
|
|
if (feof(fp)) {
|
|
if (MappingVertices) free(MappingVertices);
|
|
delete obj; fclose(fp); return 0;
|
|
}
|
|
sscanf(temp_string,
|
|
"<%d,%d> <%d,%d> <%d,%d> <%d,%d> "
|
|
"<%d,%d> <%d,%d> <%d,%d> <%d,%d> "
|
|
"<%d,%d> <%d,%d> <%d,%d> <%d,%d> "
|
|
"<%d,%d> <%d,%d> <%d,%d> <%d,%d> ",
|
|
p+0,m+0,p+1,m+1,p+2,m+2,p+3,m+3,
|
|
p+4,m+4,p+5,m+5,p+6,m+6,p+7,m+7,
|
|
p+8,m+8,p+9,m+9,p+10,m+10,p+11,m+11,
|
|
p+12,m+12,p+13,m+13,p+14,m+14,p+15,m+15);
|
|
for (i2 = 1; i2 < (i-1); i2 ++) {
|
|
obj->Faces.Get()[x].VertexIndices[0] = p[0];
|
|
obj->Faces.Get()[x].VertexIndices[1] = p[i2+1];
|
|
obj->Faces.Get()[x].VertexIndices[2] = p[i2];
|
|
if (MappingVertices) {
|
|
obj->Faces.Get()[x].MappingU[0][0] = MappingVertices[m[0]*2];
|
|
obj->Faces.Get()[x].MappingV[0][0] = MappingVertices[m[0]*2+1];
|
|
obj->Faces.Get()[x].MappingU[0][1] = MappingVertices[m[i2+1]*2];
|
|
obj->Faces.Get()[x].MappingV[0][1] = MappingVertices[m[i2+1]*2+1];
|
|
obj->Faces.Get()[x].MappingU[0][2] = MappingVertices[m[i2]*2];
|
|
obj->Faces.Get()[x].MappingV[0][2] = MappingVertices[m[i2]*2+1];
|
|
}
|
|
obj->Faces.Get()[x].Material = mat;
|
|
x++;
|
|
}
|
|
x--;
|
|
}
|
|
}
|
|
if (MappingVertices) free(MappingVertices);
|
|
obj->CalculateNormals();
|
|
fclose(fp);
|
|
return obj;
|
|
}
|