316 lines
7.2 KiB
C++
316 lines
7.2 KiB
C++
|
// Copyright (C) 2002-2012 Nikolaus Gebhardt
|
||
|
// This file is part of the "Irrlicht Engine".
|
||
|
// For conditions of distribution and use, see copyright notice in irrlicht.h
|
||
|
|
||
|
#include "IrrCompileConfig.h"
|
||
|
|
||
|
#ifdef _IRR_COMPILE_WITH_IRR_WRITER_
|
||
|
|
||
|
#include "CIrrMeshWriter.h"
|
||
|
#include "os.h"
|
||
|
#include "IWriteFile.h"
|
||
|
#include "IXMLWriter.h"
|
||
|
#include "IMesh.h"
|
||
|
#include "IAttributes.h"
|
||
|
|
||
|
namespace irr
|
||
|
{
|
||
|
namespace scene
|
||
|
{
|
||
|
|
||
|
|
||
|
CIrrMeshWriter::CIrrMeshWriter(video::IVideoDriver* driver,
|
||
|
io::IFileSystem* fs)
|
||
|
: FileSystem(fs), VideoDriver(driver), Writer(0)
|
||
|
{
|
||
|
#ifdef _DEBUG
|
||
|
setDebugName("CIrrMeshWriter");
|
||
|
#endif
|
||
|
|
||
|
if (VideoDriver)
|
||
|
VideoDriver->grab();
|
||
|
|
||
|
if (FileSystem)
|
||
|
FileSystem->grab();
|
||
|
}
|
||
|
|
||
|
|
||
|
CIrrMeshWriter::~CIrrMeshWriter()
|
||
|
{
|
||
|
if (VideoDriver)
|
||
|
VideoDriver->drop();
|
||
|
|
||
|
if (FileSystem)
|
||
|
FileSystem->drop();
|
||
|
}
|
||
|
|
||
|
|
||
|
//! Returns the type of the mesh writer
|
||
|
EMESH_WRITER_TYPE CIrrMeshWriter::getType() const
|
||
|
{
|
||
|
return EMWT_IRR_MESH;
|
||
|
}
|
||
|
|
||
|
|
||
|
//! writes a mesh
|
||
|
bool CIrrMeshWriter::writeMesh(io::IWriteFile* file, scene::IMesh* mesh, s32 flags)
|
||
|
{
|
||
|
if (!file)
|
||
|
return false;
|
||
|
|
||
|
Writer = FileSystem->createXMLWriter(file);
|
||
|
|
||
|
if (!Writer)
|
||
|
{
|
||
|
os::Printer::log("Could not write file", file->getFileName());
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
os::Printer::log("Writing mesh", file->getFileName());
|
||
|
|
||
|
// write IRR MESH header
|
||
|
|
||
|
Writer->writeXMLHeader();
|
||
|
|
||
|
Writer->writeElement(L"mesh", false,
|
||
|
L"xmlns", L"http://irrlicht.sourceforge.net/IRRMESH_09_2007",
|
||
|
L"version", L"1.0");
|
||
|
Writer->writeLineBreak();
|
||
|
|
||
|
// add some informational comment. Add a space after and before the comment
|
||
|
// tags so that some braindead xml parsers (AS anyone?) are able to parse this too.
|
||
|
|
||
|
core::stringw infoComment = L" This file contains a static mesh in the Irrlicht Engine format with ";
|
||
|
infoComment += core::stringw(mesh->getMeshBufferCount());
|
||
|
infoComment += L" materials.";
|
||
|
|
||
|
Writer->writeComment(infoComment.c_str());
|
||
|
Writer->writeLineBreak();
|
||
|
|
||
|
// write mesh bounding box
|
||
|
|
||
|
writeBoundingBox(mesh->getBoundingBox());
|
||
|
Writer->writeLineBreak();
|
||
|
|
||
|
// write mesh buffers
|
||
|
|
||
|
for (int i=0; i<(int)mesh->getMeshBufferCount(); ++i)
|
||
|
{
|
||
|
scene::IMeshBuffer* buffer = mesh->getMeshBuffer(i);
|
||
|
if (buffer)
|
||
|
{
|
||
|
writeMeshBuffer(buffer);
|
||
|
Writer->writeLineBreak();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
Writer->writeClosingTag(L"mesh");
|
||
|
|
||
|
Writer->drop();
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
|
||
|
void CIrrMeshWriter::writeBoundingBox(const core::aabbox3df& box)
|
||
|
{
|
||
|
Writer->writeElement(L"boundingBox", true,
|
||
|
L"minEdge", getVectorAsStringLine(box.MinEdge).c_str(),
|
||
|
L"maxEdge", getVectorAsStringLine(box.MaxEdge).c_str());
|
||
|
}
|
||
|
|
||
|
|
||
|
core::stringw CIrrMeshWriter::getVectorAsStringLine(const core::vector3df& v) const
|
||
|
{
|
||
|
core::stringw str;
|
||
|
|
||
|
str = core::stringw(v.X);
|
||
|
str += L" ";
|
||
|
str += core::stringw(v.Y);
|
||
|
str += L" ";
|
||
|
str += core::stringw(v.Z);
|
||
|
|
||
|
return str;
|
||
|
}
|
||
|
|
||
|
|
||
|
core::stringw CIrrMeshWriter::getVectorAsStringLine(const core::vector2df& v) const
|
||
|
{
|
||
|
core::stringw str;
|
||
|
|
||
|
str = core::stringw(v.X);
|
||
|
str += L" ";
|
||
|
str += core::stringw(v.Y);
|
||
|
|
||
|
return str;
|
||
|
}
|
||
|
|
||
|
|
||
|
void CIrrMeshWriter::writeMeshBuffer(const scene::IMeshBuffer* buffer)
|
||
|
{
|
||
|
Writer->writeElement(L"buffer", false);
|
||
|
Writer->writeLineBreak();
|
||
|
|
||
|
// write bounding box
|
||
|
|
||
|
writeBoundingBox(buffer->getBoundingBox());
|
||
|
Writer->writeLineBreak();
|
||
|
|
||
|
// write material
|
||
|
|
||
|
writeMaterial(buffer->getMaterial());
|
||
|
|
||
|
// write vertices
|
||
|
|
||
|
const core::stringw vertexTypeStr = video::sBuiltInVertexTypeNames[buffer->getVertexType()];
|
||
|
|
||
|
Writer->writeElement(L"vertices", false,
|
||
|
L"type", vertexTypeStr.c_str(),
|
||
|
L"vertexCount", core::stringw(buffer->getVertexCount()).c_str());
|
||
|
|
||
|
Writer->writeLineBreak();
|
||
|
|
||
|
u32 vertexCount = buffer->getVertexCount();
|
||
|
|
||
|
switch(buffer->getVertexType())
|
||
|
{
|
||
|
case video::EVT_STANDARD:
|
||
|
{
|
||
|
video::S3DVertex* vtx = (video::S3DVertex*)buffer->getVertices();
|
||
|
for (u32 j=0; j<vertexCount; ++j)
|
||
|
{
|
||
|
core::stringw str = getVectorAsStringLine(vtx[j].Pos);
|
||
|
str += L" ";
|
||
|
str += getVectorAsStringLine(vtx[j].Normal);
|
||
|
|
||
|
char tmp[12];
|
||
|
sprintf(tmp, " %02x%02x%02x%02x ", vtx[j].Color.getAlpha(), vtx[j].Color.getRed(), vtx[j].Color.getGreen(), vtx[j].Color.getBlue());
|
||
|
str += tmp;
|
||
|
|
||
|
str += getVectorAsStringLine(vtx[j].TCoords);
|
||
|
|
||
|
Writer->writeText(str.c_str());
|
||
|
Writer->writeLineBreak();
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
case video::EVT_2TCOORDS:
|
||
|
{
|
||
|
video::S3DVertex2TCoords* vtx = (video::S3DVertex2TCoords*)buffer->getVertices();
|
||
|
for (u32 j=0; j<vertexCount; ++j)
|
||
|
{
|
||
|
core::stringw str = getVectorAsStringLine(vtx[j].Pos);
|
||
|
str += L" ";
|
||
|
str += getVectorAsStringLine(vtx[j].Normal);
|
||
|
|
||
|
char tmp[12];
|
||
|
sprintf(tmp, " %02x%02x%02x%02x ", vtx[j].Color.getAlpha(), vtx[j].Color.getRed(), vtx[j].Color.getGreen(), vtx[j].Color.getBlue());
|
||
|
str += tmp;
|
||
|
|
||
|
str += getVectorAsStringLine(vtx[j].TCoords);
|
||
|
str += L" ";
|
||
|
str += getVectorAsStringLine(vtx[j].TCoords2);
|
||
|
|
||
|
Writer->writeText(str.c_str());
|
||
|
Writer->writeLineBreak();
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
case video::EVT_TANGENTS:
|
||
|
{
|
||
|
video::S3DVertexTangents* vtx = (video::S3DVertexTangents*)buffer->getVertices();
|
||
|
for (u32 j=0; j<vertexCount; ++j)
|
||
|
{
|
||
|
core::stringw str = getVectorAsStringLine(vtx[j].Pos);
|
||
|
str += L" ";
|
||
|
str += getVectorAsStringLine(vtx[j].Normal);
|
||
|
|
||
|
char tmp[12];
|
||
|
sprintf(tmp, " %02x%02x%02x%02x ", vtx[j].Color.getAlpha(), vtx[j].Color.getRed(), vtx[j].Color.getGreen(), vtx[j].Color.getBlue());
|
||
|
str += tmp;
|
||
|
|
||
|
str += getVectorAsStringLine(vtx[j].TCoords);
|
||
|
str += L" ";
|
||
|
str += getVectorAsStringLine(vtx[j].Tangent);
|
||
|
str += L" ";
|
||
|
str += getVectorAsStringLine(vtx[j].Binormal);
|
||
|
|
||
|
Writer->writeText(str.c_str());
|
||
|
Writer->writeLineBreak();
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
Writer->writeClosingTag(L"vertices");
|
||
|
Writer->writeLineBreak();
|
||
|
|
||
|
// write indices
|
||
|
|
||
|
Writer->writeElement(L"indices", false,
|
||
|
L"indexCount", core::stringw(buffer->getIndexCount()).c_str());
|
||
|
|
||
|
Writer->writeLineBreak();
|
||
|
|
||
|
int indexCount = (int)buffer->getIndexCount();
|
||
|
|
||
|
video::E_INDEX_TYPE iType = buffer->getIndexType();
|
||
|
|
||
|
const u16* idx16 = buffer->getIndices();
|
||
|
const u32* idx32 = (u32*) buffer->getIndices();
|
||
|
const int maxIndicesPerLine = 25;
|
||
|
|
||
|
for (int i=0; i<indexCount; ++i)
|
||
|
{
|
||
|
if(iType == video::EIT_16BIT)
|
||
|
{
|
||
|
core::stringw str((int)idx16[i]);
|
||
|
Writer->writeText(str.c_str());
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
core::stringw str((int)idx32[i]);
|
||
|
Writer->writeText(str.c_str());
|
||
|
}
|
||
|
|
||
|
if (i % maxIndicesPerLine != maxIndicesPerLine)
|
||
|
{
|
||
|
if (i % maxIndicesPerLine == maxIndicesPerLine-1)
|
||
|
Writer->writeLineBreak();
|
||
|
else
|
||
|
Writer->writeText(L" ");
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ((indexCount-1) % maxIndicesPerLine != maxIndicesPerLine-1)
|
||
|
Writer->writeLineBreak();
|
||
|
|
||
|
Writer->writeClosingTag(L"indices");
|
||
|
Writer->writeLineBreak();
|
||
|
|
||
|
// close buffer tag
|
||
|
|
||
|
Writer->writeClosingTag(L"buffer");
|
||
|
}
|
||
|
|
||
|
|
||
|
void CIrrMeshWriter::writeMaterial(const video::SMaterial& material)
|
||
|
{
|
||
|
// simply use irrlichts built-in attribute serialization capabilities here:
|
||
|
|
||
|
io::IAttributes* attributes =
|
||
|
VideoDriver->createAttributesFromMaterial(material);
|
||
|
|
||
|
if (attributes)
|
||
|
{
|
||
|
attributes->write(Writer, false, L"material");
|
||
|
attributes->drop();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
} // end namespace
|
||
|
} // end namespace
|
||
|
|
||
|
#endif
|
||
|
|