GLES drivers adapted, but only did make compile-tests. git-svn-id: svn://svn.code.sf.net/p/irrlicht/code/branches/ogl-es@6038 dfc29bdd-3216-0410-991c-e03cc46cb475
		
			
				
	
	
		
			313 lines
		
	
	
		
			7.1 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			313 lines
		
	
	
		
			7.1 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-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
 | |
| 
 |