#pragma once #include "athena/Types.hpp" #include "DataSpec/DNACommon/DeafBabe.hpp" #include "DataSpec/DNACommon/PAK.hpp" #include "DataSpec/DNACommon/OBBTreeBuilder.hpp" #include "DNAMP1.hpp" #include "DeafBabe.hpp" #define DCLN_DUMP_OBB 0 namespace DataSpec::DNAMP1 { struct DCLN : BigDNA { using Mesh = hecl::blender::ColMesh; AT_DECL_DNA Value<atUint32> colCount; struct Collision : BigDNA { using Material = DeafBabe::Material; using Edge = DeafBabe::Edge; using Triangle = DeafBabe::Triangle; AT_DECL_DNA Value<atUint32> magic; Value<atUint32> version; Value<atUint32> memSize; Value<atUint32> materialCount; Vector<Material, AT_DNA_COUNT(materialCount)> materials; Value<atUint32> vertMatsCount; Vector<atUint8, AT_DNA_COUNT(vertMatsCount)> vertMats; Value<atUint32> edgeMatsCount; Vector<atUint8, AT_DNA_COUNT(edgeMatsCount)> edgeMats; Value<atUint32> triMatsCount; Vector<atUint8, AT_DNA_COUNT(triMatsCount)> triMats; Value<atUint32> edgeVertsCount; Vector<Edge, AT_DNA_COUNT(edgeVertsCount)> edgeVertConnections; Value<atUint32> triangleEdgesCount; Vector<Triangle, AT_DNA_COUNT(triangleEdgesCount / 3)> triangleEdgeConnections; Value<atUint32> vertCount; Vector<atVec3f, AT_DNA_COUNT(vertCount)> verts; struct Node : BigDNA { AT_DECL_EXPLICIT_DNA struct LeafData : BigDNA { AT_DECL_DNA Value<atUint32> triangleIndexCount; Vector<atUint16, AT_DNA_COUNT(triangleIndexCount)> triangleIndices; size_t getMemoryUsage() const { return (((triangleIndices.size() * 2) + 16) + 3) & ~3; } }; Value<atVec4f> xf[3]; Value<atVec3f> halfExtent; Value<bool> isLeaf; std::unique_ptr<LeafData> leafData; std::unique_ptr<Node> left; std::unique_ptr<Node> right; size_t getMemoryUsage() const { size_t ret = 80; if (isLeaf) ret += leafData->getMemoryUsage(); else { ret += left->getMemoryUsage(); ret += right->getMemoryUsage(); } return (ret + 3) & ~3; } #if DCLN_DUMP_OBB void sendToBlender(hecl::blender::PyOutStream& os) const; #endif }; Node root; size_t getMemoryUsage() const { return root.getMemoryUsage(); } /* Dummy MP2 member */ void insertNoClimb(hecl::blender::PyOutStream&) const {} }; Vector<Collision, AT_DNA_COUNT(colCount)> collision; void sendToBlender(hecl::blender::Connection& conn, std::string_view entryName); static bool Extract(const SpecBase& dataSpec, PAKEntryReadStream& rs, const hecl::ProjectPath& outPath, PAKRouter<PAKBridge>& pakRouter, const PAK::Entry& entry, bool force, hecl::blender::Token& btok, std::function<void(const hecl::SystemChar*)> fileChanged); static bool Cook(const hecl::ProjectPath& outPath, const std::vector<Mesh>& meshes); }; template <class Op> void DCLN::Collision::Node::Enumerate(typename Op::StreamT& s) { Do<Op>(athena::io::PropId{"xf[0]"}, xf[0], s); Do<Op>(athena::io::PropId{"xf[1]"}, xf[1], s); Do<Op>(athena::io::PropId{"xf[2]"}, xf[2], s); Do<Op>(athena::io::PropId{"halfExtent"}, halfExtent, s); Do<Op>(athena::io::PropId{"isLeaf"}, isLeaf, s); if (isLeaf) { if (!leafData) { leafData = std::make_unique<LeafData>(); } Do<Op>(athena::io::PropId{"leafData"}, *leafData, s); } else { if (!left) { left = std::make_unique<Node>(); } Do<Op>(athena::io::PropId{"left"}, *left, s); if (!right) { right = std::make_unique<Node>(); } Do<Op>(athena::io::PropId{"right"}, *right, s); } } AT_SPECIALIZE_DNA(DCLN::Collision::Node) } // namespace DataSpec::DNAMP1