#include "CINF.hpp" #include "hecl/Blender/Connection.hpp" #include "DataSpec/DNAMP3/DNAMP3.hpp" namespace DataSpec::DNAMP2 { atUint32 CINF::getInternalBoneIdxFromId(atUint32 id) const { atUint32 idx = 0; for (const Bone& b : bones) { if (b.id == id) return idx; ++idx; } return -1; } atUint32 CINF::getBoneIdxFromId(atUint32 id) const { atUint32 idx = 0; for (atUint32 bid : boneIds) { if (bid == id) return idx; ++idx; } return 0; } const std::string* CINF::getBoneNameFromId(atUint32 id) const { for (const Name& name : names) if (id == name.boneId) return &name.name; return nullptr; } void CINF::sendVertexGroupsToBlender(hecl::blender::PyOutStream& os) const { for (atUint32 bid : boneIds) { for (const Name& name : names) { if (name.boneId == bid) { os.format(FMT_STRING("obj.vertex_groups.new(name='{}')\n"), name.name); break; } } } } template <class PAKBridge> void CINF::sendCINFToBlender(hecl::blender::PyOutStream& os, const typename PAKBridge::PAKType::IDType& cinfId) const { DNAANIM::RigInverter<CINF> inverter(*this); os.format(FMT_STRING( "arm = bpy.data.armatures.new('CINF_{}')\n" "arm_obj = bpy.data.objects.new(arm.name, arm)\n" "bpy.context.scene.collection.objects.link(arm_obj)\n" "bpy.context.view_layer.objects.active = arm_obj\n" "bpy.ops.object.mode_set(mode='EDIT')\n" "arm_bone_table = {{}}\n"), cinfId); for (const DNAANIM::RigInverter<CINF>::Bone& bone : inverter.getBones()) { zeus::simd_floats originF(bone.m_origBone.origin.simd); zeus::simd_floats tailF(bone.m_tail.mSimd); os.format(FMT_STRING( "bone = arm.edit_bones.new('{}')\n" "bone.head = ({},{},{})\n" "bone.tail = ({},{},{})\n" "bone.use_inherit_scale = False\n" "arm_bone_table[{}] = bone\n"), *getBoneNameFromId(bone.m_origBone.id), originF[0], originF[1], originF[2], tailF[0], tailF[1], tailF[2], bone.m_origBone.id); } if constexpr (std::is_same_v<PAKBridge, DNAMP3::PAKBridge>) { if (bones.size()) { atUint32 nullId = bones[0].parentId; for (const Bone& bone : bones) if (bone.parentId != nullId) os.format(FMT_STRING("arm_bone_table[{}].parent = arm_bone_table[{}]\n"), bone.id, bone.parentId); } } else { for (const Bone& bone : bones) if (bone.parentId != 97) os.format(FMT_STRING("arm_bone_table[{}].parent = arm_bone_table[{}]\n"), bone.id, bone.parentId); } os << "bpy.ops.object.mode_set(mode='OBJECT')\n"; for (const DNAANIM::RigInverter<CINF>::Bone& bone : inverter.getBones()) os.format(FMT_STRING("arm_obj.pose.bones['{}'].rotation_mode = 'QUATERNION'\n"), *getBoneNameFromId(bone.m_origBone.id)); } template void CINF::sendCINFToBlender<PAKBridge>(hecl::blender::PyOutStream& os, const UniqueID32& cinfId) const; template void CINF::sendCINFToBlender<DNAMP3::PAKBridge>(hecl::blender::PyOutStream& os, const UniqueID64& cinfId) const; template <class UniqueID> std::string CINF::GetCINFArmatureName(const UniqueID& cinfId) { return fmt::format(FMT_STRING("CINF_{}"), cinfId); } template std::string CINF::GetCINFArmatureName(const UniqueID32& cinfId); template std::string CINF::GetCINFArmatureName(const UniqueID64& cinfId); int CINF::RecursiveAddArmatureBone(const Armature& armature, const BlenderBone* bone, int parent, int& curId, std::unordered_map<std::string, atInt32>& idMap, std::map<std::string, int>& nameMap) { int selId; auto search = idMap.find(bone->name); if (search == idMap.end()) { selId = curId++; idMap.emplace(std::make_pair(bone->name, selId)); } else selId = search->second; bones.emplace_back(); Bone& boneOut = bones.back(); nameMap[bone->name] = selId; boneOut.id = selId; boneOut.parentId = parent; boneOut.origin = bone->origin; boneOut.linkedCount = bone->children.size() + 1; boneOut.linked.reserve(boneOut.linkedCount); const BlenderBone* child; boneOut.linked.push_back(parent); for (size_t i = 0; (child = armature.getChild(bone, i)); ++i) boneOut.linked.push_back(RecursiveAddArmatureBone(armature, child, boneOut.id, curId, idMap, nameMap)); return boneOut.id; } CINF::CINF(const Armature& armature, std::unordered_map<std::string, atInt32>& idMap) { idMap.reserve(armature.bones.size()); bones.reserve(armature.bones.size()); std::map<std::string, int> nameMap; const BlenderBone* bone = armature.getRoot(); if (bone) { if (bone->children.size()) { int curId = 4; const BlenderBone* child; for (size_t i = 0; (child = armature.getChild(bone, i)); ++i) RecursiveAddArmatureBone(armature, child, 3, curId, idMap, nameMap); } bones.emplace_back(); Bone& boneOut = bones.back(); nameMap[bone->name] = 3; boneOut.id = 3; boneOut.parentId = 2; boneOut.origin = bone->origin; idMap.emplace(std::make_pair(bone->name, 3)); if (bone->children.size()) { boneOut.linkedCount = 2; boneOut.linked = {2, 4}; } else { boneOut.linkedCount = 1; boneOut.linked = {2}; } } boneCount = bones.size(); names.reserve(nameMap.size()); nameCount = nameMap.size(); for (const auto& name : nameMap) { names.emplace_back(); Name& nameOut = names.back(); nameOut.name = name.first; nameOut.boneId = name.second; } boneIdCount = boneCount; boneIds.reserve(boneIdCount); for (auto it = bones.crbegin(); it != bones.crend(); ++it) boneIds.push_back(it->id); } template <class PAKBridge> bool CINF::Extract(const SpecBase& dataSpec, PAKEntryReadStream& rs, const hecl::ProjectPath& outPath, PAKRouter<PAKBridge>& pakRouter, const typename PAKBridge::PAKType::Entry& entry, bool force, hecl::blender::Token& btok, std::function<void(const hecl::SystemChar*)> fileChanged) { if (!force && outPath.isFile()) return true; auto& conn = btok.getBlenderConnection(); if (!conn.createBlend(outPath, hecl::blender::BlendType::Armature)) return false; auto os = conn.beginPythonOut(true); os.format(FMT_STRING("import bpy\n" "from mathutils import Vector\n" "bpy.context.scene.name = 'CINF_{}'\n" "bpy.context.scene.hecl_arm_obj = bpy.context.scene.name\n" "\n" "# Clear Scene\n" "if len(bpy.data.collections):\n" " bpy.data.collections.remove(bpy.data.collections[0])\n" "\n"), entry.id); CINF cinf; cinf.read(rs); cinf.sendCINFToBlender<PAKBridge>(os, entry.id); os.centerView(); os.close(); return conn.saveBlend(); } template bool CINF::Extract(const SpecBase& dataSpec, PAKEntryReadStream& rs, const hecl::ProjectPath& outPath, PAKRouter<PAKBridge>& pakRouter, const typename PAKBridge::PAKType::Entry& entry, bool force, hecl::blender::Token& btok, std::function<void(const hecl::SystemChar*)> fileChanged); template bool CINF::Extract(const SpecBase& dataSpec, PAKEntryReadStream& rs, const hecl::ProjectPath& outPath, PAKRouter<DNAMP3::PAKBridge>& pakRouter, const typename DNAMP3::PAKBridge::PAKType::Entry& entry, bool force, hecl::blender::Token& btok, std::function<void(const hecl::SystemChar*)> fileChanged); bool CINF::Cook(const hecl::ProjectPath& outPath, const hecl::ProjectPath& inPath, const hecl::blender::Armature& armature) { std::unordered_map<std::string, atInt32> boneIdMap; CINF cinf(armature, boneIdMap); /* Write out CINF resource */ athena::io::TransactionalFileWriter w(outPath.getAbsolutePath()); cinf.write(w); return true; } } // namespace DataSpec::DNAMP2