2015-07-26 21:39:49 +00:00
|
|
|
#include "CCollisionLoader.h"
|
|
|
|
#include <Core/Log.h>
|
|
|
|
#include <iostream>
|
|
|
|
|
|
|
|
CCollisionLoader::CCollisionLoader()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2015-09-26 22:55:14 +00:00
|
|
|
CCollisionMesh::CCollisionOctree* CCollisionLoader::ParseOctree(CInputStream&)
|
2015-07-26 21:39:49 +00:00
|
|
|
{
|
2015-09-26 22:55:14 +00:00
|
|
|
// Not using: Parameter 1 (CInputStream& - src)
|
|
|
|
return nullptr;
|
|
|
|
}
|
2015-07-26 21:39:49 +00:00
|
|
|
|
2015-09-26 22:55:14 +00:00
|
|
|
CCollisionMesh::CCollisionOctree::SBranch* CCollisionLoader::ParseOctreeBranch(CInputStream&)
|
|
|
|
{
|
|
|
|
// Not using: Parameter 1 (CInputStream& - src)
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
CCollisionMesh::CCollisionOctree::SLeaf* CCollisionLoader::ParseOctreeLeaf(CInputStream&)
|
|
|
|
{
|
|
|
|
// Not using: Parameter 1 (CInputStream& - src)
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CCollisionLoader::ParseOBBNode(CInputStream& DCLN)
|
|
|
|
{
|
|
|
|
bool b = false;
|
|
|
|
|
|
|
|
while (b == false)
|
2015-07-26 21:39:49 +00:00
|
|
|
{
|
2015-09-26 22:55:14 +00:00
|
|
|
DCLN.Seek(0x3C, SEEK_CUR);
|
|
|
|
b = (DCLN.ReadByte() == 1);
|
|
|
|
if (!b) ParseOBBNode(DCLN);
|
2015-07-26 21:39:49 +00:00
|
|
|
}
|
|
|
|
|
2015-09-26 22:55:14 +00:00
|
|
|
u32 numFaces = DCLN.ReadLong();
|
|
|
|
DCLN.Seek(numFaces * 2, SEEK_CUR);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CCollisionLoader::ReadPropertyFlags(CInputStream& src)
|
|
|
|
{
|
|
|
|
CCollisionMesh::SCollisionProperties property;
|
|
|
|
|
|
|
|
if (mVersion == ePrime)
|
2015-07-26 21:39:49 +00:00
|
|
|
{
|
2015-09-26 22:55:14 +00:00
|
|
|
u32 flag = src.ReadLong();
|
|
|
|
property.Invert = (flag >> 25) & 0x1;
|
2015-07-26 21:39:49 +00:00
|
|
|
}
|
|
|
|
|
2015-09-26 22:55:14 +00:00
|
|
|
if (mVersion == eEchoes)
|
|
|
|
{
|
|
|
|
u64 flag = src.ReadLongLong();
|
|
|
|
property.Invert = (flag >> 24) & 0x1;
|
|
|
|
}
|
2015-07-26 21:39:49 +00:00
|
|
|
|
2015-09-26 22:55:14 +00:00
|
|
|
mProperties.push_back(property);
|
|
|
|
}
|
2015-07-26 21:39:49 +00:00
|
|
|
|
2015-09-26 22:55:14 +00:00
|
|
|
void CCollisionLoader::LoadCollisionIndices(CInputStream &file, bool buildAABox)
|
|
|
|
{
|
2015-07-26 21:39:49 +00:00
|
|
|
// Properties
|
2015-09-26 22:55:14 +00:00
|
|
|
u32 propSetCount = file.ReadLong();
|
|
|
|
for (u32 iProp = 0; iProp < propSetCount; iProp++)
|
|
|
|
ReadPropertyFlags(file);
|
2015-07-26 21:39:49 +00:00
|
|
|
|
|
|
|
// Property indices for vertices/lines/faces
|
2015-09-26 22:55:14 +00:00
|
|
|
u32 vtxIndexCount = file.ReadLong();
|
2015-07-26 21:39:49 +00:00
|
|
|
std::vector<u8> vtxIndices(vtxIndexCount);
|
2015-09-26 22:55:14 +00:00
|
|
|
file.ReadBytes(vtxIndices.data(), vtxIndices.size());
|
2015-07-26 21:39:49 +00:00
|
|
|
|
2015-09-26 22:55:14 +00:00
|
|
|
u32 lineIndexCount = file.ReadLong();
|
2015-07-26 21:39:49 +00:00
|
|
|
std::vector<u8> lineIndices(lineIndexCount);
|
2015-09-26 22:55:14 +00:00
|
|
|
file.ReadBytes(lineIndices.data(), lineIndices.size());
|
2015-07-26 21:39:49 +00:00
|
|
|
|
2015-09-26 22:55:14 +00:00
|
|
|
u32 faceIndexCount = file.ReadLong();
|
2015-07-26 21:39:49 +00:00
|
|
|
std::vector<u8> faceIndices(faceIndexCount);
|
2015-09-26 22:55:14 +00:00
|
|
|
file.ReadBytes(faceIndices.data(), faceIndices.size());
|
2015-07-26 21:39:49 +00:00
|
|
|
|
|
|
|
// Lines
|
2015-09-26 22:55:14 +00:00
|
|
|
mpMesh->mLineCount = file.ReadLong();
|
|
|
|
mpMesh->mCollisionLines.resize(mpMesh->mLineCount);
|
|
|
|
for (u32 iLine = 0; iLine < mpMesh->mLineCount; iLine++)
|
2015-07-26 21:39:49 +00:00
|
|
|
{
|
2015-09-26 22:55:14 +00:00
|
|
|
CCollisionMesh::CCollisionLine *pLine = &mpMesh->mCollisionLines[iLine];
|
|
|
|
pLine->Vertices[0] = file.ReadShort();
|
|
|
|
pLine->Vertices[1] = file.ReadShort();
|
|
|
|
pLine->Properties = mProperties[lineIndices[iLine]];
|
2015-07-26 21:39:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Faces
|
2015-09-26 22:55:14 +00:00
|
|
|
mpMesh->mFaceCount = file.ReadLong() / 3; // Not sure why they store it this way. It's inconsistent.
|
|
|
|
mpMesh->mCollisionFaces.resize(mpMesh->mFaceCount);
|
|
|
|
|
|
|
|
for (u32 iFace = 0; iFace < mpMesh->mFaceCount; iFace++)
|
2015-07-26 21:39:49 +00:00
|
|
|
{
|
2015-09-26 22:55:14 +00:00
|
|
|
CCollisionMesh::CCollisionFace *pFace = &mpMesh->mCollisionFaces[iFace];
|
|
|
|
pFace->Lines[0] = file.ReadShort();
|
|
|
|
pFace->Lines[1] = file.ReadShort();
|
|
|
|
pFace->Lines[2] = file.ReadShort();
|
|
|
|
pFace->Properties = mProperties[faceIndices[iFace]];
|
2015-07-26 21:39:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Echoes introduces a new data chunk; don't know what it is yet, skipping for now
|
2015-09-26 22:55:14 +00:00
|
|
|
if (mVersion == eEchoes)
|
2015-07-26 21:39:49 +00:00
|
|
|
{
|
2015-09-26 22:55:14 +00:00
|
|
|
u32 unknownCount = file.ReadLong();
|
|
|
|
file.Seek(unknownCount * 2, SEEK_CUR);
|
2015-07-26 21:39:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Vertices
|
2015-09-26 22:55:14 +00:00
|
|
|
mpMesh->mVertexCount = file.ReadLong();
|
|
|
|
mpMesh->mCollisionVertices.resize(mpMesh->mVertexCount);
|
|
|
|
CAABox bounds;
|
|
|
|
|
|
|
|
for (u32 iVtx = 0; iVtx < mpMesh->mVertexCount; iVtx++)
|
2015-07-26 21:39:49 +00:00
|
|
|
{
|
2015-09-26 22:55:14 +00:00
|
|
|
CCollisionMesh::CCollisionVertex *pVtx = &mpMesh->mCollisionVertices[iVtx];
|
|
|
|
pVtx->Pos = CVector3f(file);
|
|
|
|
pVtx->Properties = mProperties[vtxIndices[iVtx]];
|
|
|
|
if (buildAABox) bounds.ExpandBounds(pVtx->Pos);
|
2015-07-26 21:39:49 +00:00
|
|
|
}
|
2015-09-26 22:55:14 +00:00
|
|
|
if (buildAABox) mpMesh->mAABox = bounds;
|
2015-07-26 21:39:49 +00:00
|
|
|
}
|
|
|
|
|
2015-09-26 22:55:14 +00:00
|
|
|
// ************ STATIC ************
|
|
|
|
CCollisionMeshGroup* CCollisionLoader::LoadAreaCollision(CInputStream& MREA)
|
2015-07-26 21:39:49 +00:00
|
|
|
{
|
2015-09-26 22:55:14 +00:00
|
|
|
if (!MREA.IsValid()) return nullptr;
|
|
|
|
CCollisionLoader loader;
|
2015-07-26 21:39:49 +00:00
|
|
|
|
2015-09-26 22:55:14 +00:00
|
|
|
MREA.Seek(0x8, SEEK_CUR);
|
|
|
|
u32 deafbabe = MREA.ReadLong();
|
|
|
|
if (deafbabe != 0xDEAFBABE)
|
|
|
|
{
|
|
|
|
Log::FileError(MREA.GetSourceString(), MREA.Tell() - 4, "Invalid collision magic: " + StringUtil::ToHexString(deafbabe));
|
|
|
|
return nullptr;
|
|
|
|
}
|
2015-07-26 21:39:49 +00:00
|
|
|
|
2015-09-26 22:55:14 +00:00
|
|
|
u32 version = MREA.ReadLong();
|
|
|
|
loader.mVersion = GetFormatVersion(version);
|
|
|
|
if ((loader.mVersion != ePrime) && (loader.mVersion != eEchoes))
|
|
|
|
{
|
|
|
|
Log::FileError(MREA.GetSourceString(), MREA.Tell() - 4, "Unsupported collision version: " + StringUtil::ToHexString(version));
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
loader.mpGroup = new CCollisionMeshGroup;
|
|
|
|
loader.mpMesh = new CCollisionMesh;
|
|
|
|
|
|
|
|
// Octree - structure is known, but not coding this right now
|
|
|
|
loader.mpMesh->mAABox = CAABox(MREA);
|
|
|
|
MREA.Seek(0x4, SEEK_CUR);
|
|
|
|
u32 octreeSize = MREA.ReadLong();
|
|
|
|
MREA.Seek(octreeSize, SEEK_CUR); // Skipping the octree for now
|
|
|
|
loader.mpMesh->mOctreeLoaded = false;
|
|
|
|
|
|
|
|
// Read collision indices and return
|
|
|
|
loader.LoadCollisionIndices(MREA, false);
|
|
|
|
loader.mpGroup->AddMesh(loader.mpMesh);
|
|
|
|
return loader.mpGroup;
|
2015-07-26 21:39:49 +00:00
|
|
|
}
|
|
|
|
|
2015-09-26 22:55:14 +00:00
|
|
|
CCollisionMeshGroup* CCollisionLoader::LoadDCLN(CInputStream &DCLN)
|
2015-07-26 21:39:49 +00:00
|
|
|
{
|
2015-09-26 22:55:14 +00:00
|
|
|
if (!DCLN.IsValid()) return nullptr;
|
2015-07-26 21:39:49 +00:00
|
|
|
|
2015-09-26 22:55:14 +00:00
|
|
|
CCollisionLoader loader;
|
|
|
|
loader.mpGroup = new CCollisionMeshGroup;
|
|
|
|
|
|
|
|
u32 numMeshes = DCLN.ReadLong();
|
|
|
|
|
|
|
|
for (u32 iMesh = 0; iMesh < numMeshes; iMesh++)
|
2015-07-26 21:39:49 +00:00
|
|
|
{
|
2015-09-26 22:55:14 +00:00
|
|
|
u32 deafbabe = DCLN.ReadLong();
|
|
|
|
|
|
|
|
if (deafbabe != 0xDEAFBABE)
|
|
|
|
{
|
|
|
|
Log::FileError(DCLN.GetSourceString(), DCLN.Tell() - 4, "Invalid collision magic: " + StringUtil::ToHexString(deafbabe));
|
|
|
|
delete loader.mpGroup;
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
u32 version = DCLN.ReadLong();
|
|
|
|
loader.mVersion = GetFormatVersion(version);
|
|
|
|
|
|
|
|
if ((loader.mVersion != ePrime) && (loader.mVersion != eEchoes))
|
|
|
|
{
|
|
|
|
Log::FileError(DCLN.GetSourceString(), DCLN.Tell() - 4, "Unsupported collision version: " + StringUtil::ToHexString(version));
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
loader.mpMesh = new CCollisionMesh;
|
|
|
|
loader.mpMesh->mOctreeLoaded = false;
|
|
|
|
|
|
|
|
// Read indices and return
|
|
|
|
DCLN.Seek(0x4, SEEK_CUR);
|
|
|
|
loader.LoadCollisionIndices(DCLN, true);
|
|
|
|
loader.mpGroup->AddMesh(loader.mpMesh);
|
|
|
|
|
|
|
|
// Parse OBB tree
|
|
|
|
loader.ParseOBBNode(DCLN);
|
2015-07-26 21:39:49 +00:00
|
|
|
}
|
2015-09-26 22:55:14 +00:00
|
|
|
return loader.mpGroup;
|
|
|
|
}
|
2015-07-26 21:39:49 +00:00
|
|
|
|
2015-09-26 22:55:14 +00:00
|
|
|
EGame CCollisionLoader::GetFormatVersion(u32 version)
|
|
|
|
{
|
|
|
|
switch (version)
|
2015-07-26 21:39:49 +00:00
|
|
|
{
|
2015-09-26 22:55:14 +00:00
|
|
|
case 0x2: return ePrime;
|
|
|
|
case 0x3: return ePrime;
|
|
|
|
case 0x4: return eEchoes;
|
|
|
|
case 0x5: return eReturns;
|
|
|
|
default: return eUnknownVersion;
|
2015-07-26 21:39:49 +00:00
|
|
|
}
|
|
|
|
}
|