2014-12-19 16:24:29 +01:00
|
|
|
#include <cstdio>
|
|
|
|
#include <cstdlib>
|
|
|
|
#include <cstring>
|
|
|
|
|
|
|
|
#include <iostream>
|
|
|
|
#include <fstream>
|
|
|
|
|
|
|
|
#include "rwbase.h"
|
|
|
|
#include "rwplugin.h"
|
|
|
|
#include "rw.h"
|
|
|
|
|
|
|
|
using namespace std;
|
|
|
|
using namespace Rw;
|
|
|
|
|
|
|
|
static void*
|
|
|
|
createNodeName(void *object, int32 offset, int32)
|
|
|
|
{
|
|
|
|
char *name = PLUGINOFFSET(char, object, offset);
|
|
|
|
name[0] = '\0';
|
|
|
|
return object;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void*
|
|
|
|
copyNodeName(void *dst, void *src, int32 offset, int32)
|
|
|
|
{
|
|
|
|
char *dstname = PLUGINOFFSET(char, dst, offset);
|
|
|
|
char *srcname = PLUGINOFFSET(char, src, offset);
|
|
|
|
strncpy(dstname, srcname, 17);
|
|
|
|
return dst;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void*
|
|
|
|
destroyNodeName(void *object, int32, int32)
|
|
|
|
{
|
|
|
|
return object;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
readNodeName(istream &stream, Rw::int32 len, void *object, int32 offset, int32)
|
|
|
|
{
|
|
|
|
char *name = PLUGINOFFSET(char, object, offset);
|
|
|
|
stream.read(name, len);
|
|
|
|
name[len] = '\0';
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
writeNodeName(ostream &stream, Rw::int32 len, void *object, int32 offset, int32)
|
|
|
|
{
|
|
|
|
char *name = PLUGINOFFSET(char, object, offset);
|
|
|
|
stream.write(name, len);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int32
|
|
|
|
getSizeNodeName(void *object, int32 offset)
|
|
|
|
{
|
|
|
|
char *name = PLUGINOFFSET(char, object, offset);
|
|
|
|
int32 len = strlen(name);
|
|
|
|
return len > 0 ? len : -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
registerNodeNamePlugin(void)
|
|
|
|
{
|
|
|
|
Rw::Frame::registerPlugin(18, 0x253f2fe, (Constructor)createNodeName,
|
|
|
|
(Destructor)destroyNodeName,
|
|
|
|
(CopyConstructor)copyNodeName);
|
|
|
|
Rw::Frame::registerPluginStream(0x253f2fe, (StreamRead)readNodeName,
|
|
|
|
(StreamWrite)writeNodeName,
|
|
|
|
(StreamGetSize)getSizeNodeName);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
readMesh(istream &stream, Rw::int32, void *object, int32, int32)
|
|
|
|
{
|
|
|
|
Geometry *geo = (Geometry*)object;
|
|
|
|
uint32 buf[3];
|
|
|
|
stream.read((char*)buf, 12);
|
|
|
|
geo->meshHeader = new MeshHeader;
|
|
|
|
geo->meshHeader->flags = buf[0];
|
|
|
|
geo->meshHeader->numMeshes = buf[1];
|
|
|
|
geo->meshHeader->totalIndices = buf[2];
|
|
|
|
geo->meshHeader->mesh = new Mesh[geo->meshHeader->numMeshes];
|
|
|
|
Mesh *mesh = geo->meshHeader->mesh;
|
|
|
|
for(uint32 i = 0; i < geo->meshHeader->numMeshes; i++){
|
|
|
|
stream.read((char*)buf, 8);
|
|
|
|
mesh->numIndices = buf[0];
|
|
|
|
mesh->material = geo->materialList[buf[1]];
|
|
|
|
mesh->indices = NULL;
|
|
|
|
if(geo->geoflags & Geometry::NATIVE)
|
|
|
|
// TODO: compressed indices in OpenGL
|
|
|
|
;
|
|
|
|
else{
|
|
|
|
mesh->indices = new uint16[mesh->numIndices];
|
|
|
|
for(uint32 j = 0; j < mesh->numIndices; j++)
|
|
|
|
mesh->indices[j] = readUInt32(stream);
|
|
|
|
}
|
|
|
|
mesh++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
writeMesh(ostream &stream, Rw::int32, void *object, int32, int32)
|
|
|
|
{
|
|
|
|
Geometry *geo = (Geometry*)object;
|
|
|
|
uint32 buf[3];
|
|
|
|
buf[0] = geo->meshHeader->flags;
|
|
|
|
buf[1] = geo->meshHeader->numMeshes;
|
|
|
|
buf[2] = geo->meshHeader->totalIndices;
|
|
|
|
stream.write((char*)buf, 12);
|
|
|
|
Mesh *mesh = geo->meshHeader->mesh;
|
|
|
|
for(uint32 i = 0; i < geo->meshHeader->numMeshes; i++){
|
|
|
|
buf[0] = mesh->numIndices;
|
|
|
|
buf[1] = findPointer((void*)mesh->material,
|
|
|
|
(void**)geo->materialList,
|
|
|
|
geo->numMaterials);
|
|
|
|
stream.write((char*)buf, 8);
|
|
|
|
if(geo->geoflags & Geometry::NATIVE)
|
|
|
|
// TODO: compressed indices in OpenGL
|
|
|
|
;
|
|
|
|
else
|
|
|
|
for(uint32 j = 0; j < mesh->numIndices; j++)
|
|
|
|
writeUInt32(mesh->indices[j], stream);
|
|
|
|
mesh++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int32
|
|
|
|
getSizeMesh(void *object, int32)
|
|
|
|
{
|
|
|
|
Geometry *geo = (Geometry*)object;
|
|
|
|
if(geo->meshHeader == NULL)
|
|
|
|
return -1;
|
|
|
|
int32 size = 12 + geo->meshHeader->numMeshes*8;
|
|
|
|
if(geo->geoflags & Geometry::NATIVE)
|
|
|
|
// TODO: compressed indices in OpenGL
|
|
|
|
;
|
|
|
|
else{
|
|
|
|
size += geo->meshHeader->totalIndices*4;
|
|
|
|
}
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
registerMeshPlugin(void)
|
|
|
|
{
|
2014-12-19 22:58:10 +01:00
|
|
|
Rw::Geometry::registerPlugin(0, 0x50E, NULL, NULL, NULL);
|
2014-12-19 16:24:29 +01:00
|
|
|
Rw::Geometry::registerPluginStream(0x50E, (StreamRead)readMesh,
|
|
|
|
(StreamWrite)writeMesh,
|
|
|
|
(StreamGetSize)getSizeMesh);
|
|
|
|
}
|
|
|
|
|
|
|
|
|