mirror of https://github.com/aap/librw.git
408 lines
9.6 KiB
C++
408 lines
9.6 KiB
C++
#include <cstdio>
|
|
#include <cstdlib>
|
|
#include <cstring>
|
|
|
|
#include <iostream>
|
|
#include <fstream>
|
|
|
|
#include "rwbase.h"
|
|
#include "rwplugin.h"
|
|
#include "rw.h"
|
|
|
|
using namespace std;
|
|
|
|
namespace Rw {
|
|
|
|
Geometry::Geometry(void)
|
|
{
|
|
geoflags = 0;
|
|
numTriangles = 0;
|
|
numVertices = 0;
|
|
numMorphTargets = 0;
|
|
numTexCoordSets = 0;
|
|
triangles = 0;
|
|
colors = 0;
|
|
for(int i = 0; i < 8; i++)
|
|
texCoords[i] = 0;
|
|
morphTargets = NULL;
|
|
constructPlugins();
|
|
}
|
|
|
|
Geometry::Geometry(Geometry *g)
|
|
{
|
|
// TODO
|
|
geoflags = g->geoflags;
|
|
numTriangles = g->numTriangles;
|
|
numVertices = g->numVertices;
|
|
numMorphTargets = g->numMorphTargets;
|
|
numTexCoordSets = g->numTexCoordSets;
|
|
copyPlugins(g);
|
|
}
|
|
|
|
Geometry::~Geometry(void)
|
|
{
|
|
destructPlugins();
|
|
}
|
|
|
|
struct GeoStreamData
|
|
{
|
|
uint32 flags;
|
|
int32 numTriangles;
|
|
int32 numVertices;
|
|
int32 numMorphTargets;
|
|
};
|
|
|
|
Geometry*
|
|
Geometry::streamRead(istream &stream)
|
|
{
|
|
uint32 version;
|
|
GeoStreamData buf;
|
|
if(!FindChunk(stream, ID_STRUCT, NULL, &version))
|
|
return NULL;
|
|
Geometry *geo = new Geometry;
|
|
stream.read((char*)&buf, sizeof(buf));
|
|
geo->geoflags = buf.flags & 0xFF00FFFF;
|
|
geo->numTexCoordSets = (buf.flags & 0xFF0000) >> 16;
|
|
if(geo->numTexCoordSets == 0 && (geo->geoflags & TEXTURED))
|
|
geo->numTexCoordSets = 1;
|
|
geo->numTriangles = buf.numTriangles;
|
|
geo->numVertices = buf.numVertices;
|
|
geo->numMorphTargets = buf.numMorphTargets;
|
|
// skip surface properties
|
|
if(version < 0x34000)
|
|
stream.seekg(12, ios::cur);
|
|
|
|
if(!(geo->geoflags & 0xFF000000)){
|
|
if(geo->geoflags & PRELIT){
|
|
geo->colors = new uint8[4*geo->numVertices];
|
|
stream.read((char*)geo->colors, 4*geo->numVertices);
|
|
}
|
|
if((geo->geoflags & TEXTURED) || (geo->geoflags & TEXTURED2))
|
|
for(int32 i = 0; i < geo->numTexCoordSets; i++){
|
|
geo->texCoords[i] =
|
|
new float32[2*geo->numVertices];
|
|
stream.read((char*)geo->texCoords[i],
|
|
2*geo->numVertices*4);
|
|
}
|
|
geo->triangles = new uint16[4*geo->numTriangles];
|
|
stream.read((char*)geo->triangles, 4*geo->numTriangles*2);
|
|
}
|
|
|
|
geo->morphTargets = new MorphTarget[geo->numMorphTargets];
|
|
for(int32 i = 0; i < geo->numMorphTargets; i++){
|
|
MorphTarget *m = &geo->morphTargets[i];
|
|
stream.read((char*)m->boundingSphere, 4*4);
|
|
int32 hasVertices = readInt32(stream);
|
|
int32 hasNormals = readInt32(stream);
|
|
if(hasVertices){
|
|
m->vertices = new float32[3*geo->numVertices];
|
|
stream.read((char*)m->vertices, 3*geo->numVertices*4);
|
|
}
|
|
if(hasNormals){
|
|
m->normals = new float32[3*geo->numVertices];
|
|
stream.read((char*)m->normals, 3*geo->numVertices*4);
|
|
}
|
|
}
|
|
|
|
if(!FindChunk(stream, ID_MATLIST, NULL, NULL))
|
|
return geo;
|
|
if(!FindChunk(stream, ID_STRUCT, NULL, NULL))
|
|
return geo;
|
|
geo->numMaterials = readInt32(stream);
|
|
geo->materialList = new Material*[geo->numMaterials];
|
|
stream.seekg(geo->numMaterials*4, ios::cur); // unused (-1)
|
|
for(int32 i = 0; i < geo->numMaterials; i++){
|
|
if(!FindChunk(stream, ID_MATERIAL, NULL, NULL))
|
|
return geo;
|
|
geo->materialList[i] = Material::streamRead(stream);
|
|
}
|
|
|
|
geo->streamReadPlugins(stream);
|
|
|
|
return geo;
|
|
}
|
|
|
|
static uint32
|
|
geoStructSize(Geometry *geo)
|
|
{
|
|
uint32 size = 0;
|
|
size += sizeof(GeoStreamData);
|
|
if(Version < 0x34000)
|
|
size += 12; // surface properties
|
|
if(!(geo->geoflags & 0xFF000000)){
|
|
if(geo->geoflags&geo->PRELIT)
|
|
size += 4*geo->numVertices;
|
|
if((geo->geoflags & geo->TEXTURED) ||
|
|
(geo->geoflags & geo->TEXTURED2))
|
|
for(int32 i = 0; i < geo->numTexCoordSets; i++)
|
|
size += 2*geo->numVertices*4;
|
|
size += 4*geo->numTriangles*2;
|
|
}
|
|
for(int32 i = 0; i < geo->numMorphTargets; i++){
|
|
MorphTarget *m = &geo->morphTargets[i];
|
|
size += 4*4 + 2*4; // bounding sphere and bools
|
|
if(m->vertices)
|
|
size += 3*geo->numVertices*4;
|
|
if(m->normals)
|
|
size += 3*geo->numVertices*4;
|
|
}
|
|
return size;
|
|
}
|
|
|
|
bool
|
|
Geometry::streamWrite(ostream &stream)
|
|
{
|
|
GeoStreamData buf;
|
|
uint32 size;
|
|
static float32 fbuf[3] = { 1.0f, 1.0f, 1.0f };
|
|
|
|
WriteChunkHeader(stream, ID_GEOMETRY, this->streamGetSize());
|
|
WriteChunkHeader(stream, ID_STRUCT, geoStructSize(this));
|
|
|
|
buf.flags = this->geoflags | this->numTexCoordSets << 16;
|
|
buf.numTriangles = this->numTriangles;
|
|
buf.numVertices = this->numVertices;
|
|
buf.numMorphTargets = this->numMorphTargets;
|
|
stream.write((char*)&buf, sizeof(buf));
|
|
if(Version < 0x34000)
|
|
stream.write((char*)fbuf, sizeof(fbuf));
|
|
|
|
if(!(this->geoflags & 0xFF000000)){
|
|
if(this->geoflags & PRELIT)
|
|
stream.write((char*)this->colors, 4*this->numVertices);
|
|
if((this->geoflags & TEXTURED) || (this->geoflags & TEXTURED2))
|
|
for(int32 i = 0; i < this->numTexCoordSets; i++)
|
|
stream.write((char*)this->texCoords[i],
|
|
2*this->numVertices*4);
|
|
stream.write((char*)this->triangles, 4*this->numTriangles*2);
|
|
}
|
|
|
|
for(int32 i = 0; i < this->numMorphTargets; i++){
|
|
MorphTarget *m = &this->morphTargets[i];
|
|
stream.write((char*)m->boundingSphere, 4*4);
|
|
writeInt32(m->vertices != NULL, stream);
|
|
writeInt32(m->normals != NULL, stream);
|
|
if(m->vertices)
|
|
stream.write((char*)m->vertices, 3*this->numVertices*4);
|
|
if(m->normals)
|
|
stream.write((char*)m->normals, 3*this->numVertices*4);
|
|
}
|
|
|
|
size = 12 + 4;
|
|
for(int32 i = 0; i < this->numMaterials; i++)
|
|
size += 4 + 12 + this->materialList[i]->streamGetSize();
|
|
WriteChunkHeader(stream, ID_MATLIST, size);
|
|
WriteChunkHeader(stream, ID_STRUCT, 4 + this->numMaterials*4);
|
|
writeInt32(this->numMaterials, stream);
|
|
for(int32 i = 0; i < this->numMaterials; i++)
|
|
writeInt32(-1, stream);
|
|
for(int32 i = 0; i < this->numMaterials; i++)
|
|
this->materialList[i]->streamWrite(stream);
|
|
|
|
this->streamWritePlugins(stream);
|
|
return true;
|
|
}
|
|
|
|
uint32
|
|
Geometry::streamGetSize(void)
|
|
{
|
|
uint32 size = 0;
|
|
size += 12 + geoStructSize(this);
|
|
size += 12 + 12 + 4;
|
|
for(int32 i = 0; i < this->numMaterials; i++)
|
|
size += 4 + 12 + this->materialList[i]->streamGetSize();
|
|
size += 12 + this->streamGetPluginSize();
|
|
return size;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Material::Material(void)
|
|
{
|
|
this->texture = NULL;
|
|
color[0] = color[1] = color[2] = color[3] = 0xFF;
|
|
surfaceProps[0] = surfaceProps[1] = surfaceProps[2] = 1.0f;
|
|
constructPlugins();
|
|
}
|
|
|
|
Material::Material(Material *m)
|
|
{
|
|
m->color[0] = this->color[0];
|
|
m->color[1] = this->color[1];
|
|
m->color[2] = this->color[2];
|
|
m->color[3] = this->color[3];
|
|
m->surfaceProps[0] = this->surfaceProps[0];
|
|
m->surfaceProps[1] = this->surfaceProps[1];
|
|
m->surfaceProps[2] = this->surfaceProps[2];
|
|
copyPlugins(m);
|
|
}
|
|
|
|
Material::~Material(void)
|
|
{
|
|
destructPlugins();
|
|
}
|
|
|
|
struct MatStreamData
|
|
{
|
|
int32 flags; // unused according to RW
|
|
uint8 color[4];
|
|
int32 unused;
|
|
int32 textured;
|
|
float32 surfaceProps[3];
|
|
};
|
|
|
|
Material*
|
|
Material::streamRead(istream &stream)
|
|
{
|
|
uint32 length;
|
|
MatStreamData buf;
|
|
if(!FindChunk(stream, ID_STRUCT, NULL, NULL))
|
|
return NULL;
|
|
stream.read((char*)&buf, sizeof(buf));
|
|
Material *mat = new Material;
|
|
mat->color[0] = buf.color[0];
|
|
mat->color[1] = buf.color[1];
|
|
mat->color[2] = buf.color[2];
|
|
mat->color[3] = buf.color[3];
|
|
mat->surfaceProps[0] = buf.surfaceProps[0];
|
|
mat->surfaceProps[1] = buf.surfaceProps[1];
|
|
mat->surfaceProps[2] = buf.surfaceProps[2];
|
|
|
|
if(buf.textured){
|
|
if(!FindChunk(stream, ID_TEXTURE, &length, NULL))
|
|
return NULL;
|
|
mat->texture = Texture::streamRead(stream);
|
|
}
|
|
|
|
mat->streamReadPlugins(stream);
|
|
|
|
return mat;
|
|
}
|
|
|
|
bool
|
|
Material::streamWrite(ostream &stream)
|
|
{
|
|
MatStreamData buf;
|
|
|
|
WriteChunkHeader(stream, ID_MATERIAL, this->streamGetSize());
|
|
WriteChunkHeader(stream, ID_STRUCT, sizeof(MatStreamData));
|
|
|
|
buf.color[0] = this->color[0];
|
|
buf.color[1] = this->color[1];
|
|
buf.color[2] = this->color[2];
|
|
buf.color[3] = this->color[3];
|
|
buf.surfaceProps[0] = this->surfaceProps[0];
|
|
buf.surfaceProps[1] = this->surfaceProps[1];
|
|
buf.surfaceProps[2] = this->surfaceProps[2];
|
|
buf.flags = 0;
|
|
buf.unused = 0;
|
|
buf.textured = this->texture != NULL;
|
|
stream.write((char*)&buf, sizeof(buf));
|
|
|
|
if(this->texture)
|
|
this->texture->streamWrite(stream);
|
|
|
|
this->streamWritePlugins(stream);
|
|
return true;
|
|
}
|
|
|
|
uint32
|
|
Material::streamGetSize(void)
|
|
{
|
|
uint32 size = 0;
|
|
size += 12 + sizeof(MatStreamData);
|
|
if(this->texture)
|
|
size += 12 + this->texture->streamGetSize();
|
|
size += 12 + this->streamGetPluginSize();
|
|
return size;
|
|
}
|
|
|
|
|
|
|
|
Texture::Texture(void)
|
|
{
|
|
memset(this->name, 0, 32);
|
|
memset(this->mask, 0, 32);
|
|
this->filterAddressing = 0;
|
|
constructPlugins();
|
|
}
|
|
|
|
Texture::Texture(Texture *t)
|
|
{
|
|
memcpy(this->name, t->name, 32);
|
|
memcpy(this->mask, t->name, 32);
|
|
this->filterAddressing = t->filterAddressing;
|
|
copyPlugins(t);
|
|
}
|
|
|
|
Texture::~Texture(void)
|
|
{
|
|
destructPlugins();
|
|
}
|
|
|
|
Texture*
|
|
Texture::streamRead(istream &stream)
|
|
{
|
|
uint32 length;
|
|
if(!FindChunk(stream, ID_STRUCT, NULL, NULL))
|
|
return NULL;
|
|
Texture *tex = new Texture;
|
|
tex->filterAddressing = readUInt16(stream);
|
|
stream.seekg(2, ios::cur);
|
|
|
|
if(!FindChunk(stream, ID_STRING, &length, NULL))
|
|
return NULL;
|
|
stream.read(tex->name, length);
|
|
|
|
if(!FindChunk(stream, ID_STRING, &length, NULL))
|
|
return NULL;
|
|
stream.read(tex->mask, length);
|
|
|
|
tex->streamReadPlugins(stream);
|
|
|
|
return tex;
|
|
}
|
|
|
|
bool
|
|
Texture::streamWrite(ostream &stream)
|
|
{
|
|
int size;
|
|
WriteChunkHeader(stream, ID_TEXTURE, this->streamGetSize());
|
|
WriteChunkHeader(stream, ID_STRUCT, 4);
|
|
writeUInt32(this->filterAddressing, stream);
|
|
|
|
size = strnlen(this->name, 32)+3 & ~3;
|
|
if(size < 4)
|
|
size = 4;
|
|
WriteChunkHeader(stream, ID_STRING, size);
|
|
stream.write(this->name, size);
|
|
|
|
size = strnlen(this->mask, 32)+3 & ~3;
|
|
if(size < 4)
|
|
size = 4;
|
|
WriteChunkHeader(stream, ID_STRING, size);
|
|
stream.write(this->mask, size);
|
|
|
|
this->streamWritePlugins(stream);
|
|
return true;
|
|
}
|
|
|
|
uint32
|
|
Texture::streamGetSize(void)
|
|
{
|
|
uint32 size = 0;
|
|
int strsize;
|
|
size += 12 + 4;
|
|
size += 12 + 12;
|
|
strsize = strnlen(this->name, 32)+3 & ~3;
|
|
size += strsize < 4 ? 4 : strsize;
|
|
strsize = strnlen(this->mask, 32)+3 & ~3;
|
|
size += strsize < 4 ? 4 : strsize;
|
|
size += 12 + this->streamGetPluginSize();
|
|
return size;
|
|
}
|
|
|
|
}
|