mirror of
				https://github.com/Laupetin/OpenAssetTools.git
				synced 2025-10-25 15:55:53 +00:00 
			
		
		
		
	feat: dump and load xmodels for IW5 via template
This commit is contained in:
		| @@ -545,8 +545,8 @@ namespace IW5 | ||||
|         for (auto i = 0u; i < originalKnotCount; i++) | ||||
|         { | ||||
|             auto& knot = graph.knots[i]; | ||||
|             knot.x = originalKnots[i][0]; | ||||
|             knot.y = originalKnots[i][1]; | ||||
|             knot.x = originalKnots[i].x; | ||||
|             knot.y = originalKnots[i].y; | ||||
|         } | ||||
|  | ||||
|         return graph; | ||||
|   | ||||
| @@ -1,506 +1,9 @@ | ||||
| #include "AssetDumperXModel.h" | ||||
|  | ||||
| #include "Game/IW5/CommonIW5.h" | ||||
| #include "ObjWriting.h" | ||||
| #include "Utils/DistinctMapper.h" | ||||
| #include "Utils/HalfFloat.h" | ||||
| #include "Utils/QuatInt16.h" | ||||
| #include "XModel/Export/XModelExportWriter.h" | ||||
| #include "XModel/Gltf/GltfBinOutput.h" | ||||
| #include "XModel/Gltf/GltfTextOutput.h" | ||||
| #include "XModel/Gltf/GltfWriter.h" | ||||
| #include "XModel/Obj/ObjWriter.h" | ||||
| #include "XModel/XModelWriter.h" | ||||
|  | ||||
| #include <cassert> | ||||
| #include <format> | ||||
| #include "Game/IW5/XModel/XModelDumperIW5.h" | ||||
|  | ||||
| using namespace IW5; | ||||
|  | ||||
| namespace | ||||
| { | ||||
|     GfxImage* GetMaterialColorMap(const Material* material) | ||||
|     { | ||||
|         std::vector<MaterialTextureDef*> potentialTextureDefs; | ||||
|  | ||||
|         for (auto textureIndex = 0u; textureIndex < material->textureCount; textureIndex++) | ||||
|         { | ||||
|             MaterialTextureDef* def = &material->textureTable[textureIndex]; | ||||
|  | ||||
|             if (def->semantic == TS_COLOR_MAP) | ||||
|                 potentialTextureDefs.push_back(def); | ||||
|         } | ||||
|  | ||||
|         if (potentialTextureDefs.empty()) | ||||
|             return nullptr; | ||||
|         if (potentialTextureDefs.size() == 1) | ||||
|             return potentialTextureDefs[0]->u.image; | ||||
|  | ||||
|         for (const auto* def : potentialTextureDefs) | ||||
|         { | ||||
|             if (def->nameStart == 'c' && def->nameEnd == 'p') | ||||
|                 return def->u.image; | ||||
|         } | ||||
|  | ||||
|         return potentialTextureDefs[0]->u.image; | ||||
|     } | ||||
|  | ||||
|     GfxImage* GetMaterialNormalMap(const Material* material) | ||||
|     { | ||||
|         std::vector<MaterialTextureDef*> potentialTextureDefs; | ||||
|  | ||||
|         for (auto textureIndex = 0u; textureIndex < material->textureCount; textureIndex++) | ||||
|         { | ||||
|             MaterialTextureDef* def = &material->textureTable[textureIndex]; | ||||
|  | ||||
|             if (def->semantic == TS_NORMAL_MAP) | ||||
|                 potentialTextureDefs.push_back(def); | ||||
|         } | ||||
|  | ||||
|         if (potentialTextureDefs.empty()) | ||||
|             return nullptr; | ||||
|         if (potentialTextureDefs.size() == 1) | ||||
|             return potentialTextureDefs[0]->u.image; | ||||
|  | ||||
|         for (const auto* def : potentialTextureDefs) | ||||
|         { | ||||
|             if (def->nameStart == 'n' && def->nameEnd == 'p') | ||||
|                 return def->u.image; | ||||
|         } | ||||
|  | ||||
|         return potentialTextureDefs[0]->u.image; | ||||
|     } | ||||
|  | ||||
|     GfxImage* GetMaterialSpecularMap(const Material* material) | ||||
|     { | ||||
|         std::vector<MaterialTextureDef*> potentialTextureDefs; | ||||
|  | ||||
|         for (auto textureIndex = 0u; textureIndex < material->textureCount; textureIndex++) | ||||
|         { | ||||
|             MaterialTextureDef* def = &material->textureTable[textureIndex]; | ||||
|  | ||||
|             if (def->semantic == TS_SPECULAR_MAP) | ||||
|                 potentialTextureDefs.push_back(def); | ||||
|         } | ||||
|  | ||||
|         if (potentialTextureDefs.empty()) | ||||
|             return nullptr; | ||||
|         if (potentialTextureDefs.size() == 1) | ||||
|             return potentialTextureDefs[0]->u.image; | ||||
|  | ||||
|         for (const auto* def : potentialTextureDefs) | ||||
|         { | ||||
|             if (def->nameStart == 's' && def->nameEnd == 'p') | ||||
|                 return def->u.image; | ||||
|         } | ||||
|  | ||||
|         return potentialTextureDefs[0]->u.image; | ||||
|     } | ||||
|  | ||||
|     void AddXModelBones(XModelCommon& out, const AssetDumpingContext& context, const XModel* model) | ||||
|     { | ||||
|         for (auto boneNum = 0u; boneNum < model->numBones; boneNum++) | ||||
|         { | ||||
|             XModelBone bone; | ||||
|             if (model->boneNames[boneNum] < context.m_zone->m_script_strings.Count()) | ||||
|                 bone.name = context.m_zone->m_script_strings[model->boneNames[boneNum]]; | ||||
|             else | ||||
|                 bone.name = "INVALID_BONE_NAME"; | ||||
|  | ||||
|             if (boneNum >= model->numRootBones) | ||||
|                 bone.parentIndex = boneNum - static_cast<unsigned int>(model->parentList[boneNum - model->numRootBones]); | ||||
|             else | ||||
|                 bone.parentIndex = std::nullopt; | ||||
|  | ||||
|             bone.scale[0] = 1.0f; | ||||
|             bone.scale[1] = 1.0f; | ||||
|             bone.scale[2] = 1.0f; | ||||
|  | ||||
|             bone.globalOffset[0] = model->baseMat[boneNum].trans[0]; | ||||
|             bone.globalOffset[1] = model->baseMat[boneNum].trans[1]; | ||||
|             bone.globalOffset[2] = model->baseMat[boneNum].trans[2]; | ||||
|             bone.globalRotation = { | ||||
|                 model->baseMat[boneNum].quat[0], | ||||
|                 model->baseMat[boneNum].quat[1], | ||||
|                 model->baseMat[boneNum].quat[2], | ||||
|                 model->baseMat[boneNum].quat[3], | ||||
|             }; | ||||
|  | ||||
|             if (boneNum < model->numRootBones) | ||||
|             { | ||||
|                 bone.localOffset[0] = 0; | ||||
|                 bone.localOffset[1] = 0; | ||||
|                 bone.localOffset[2] = 0; | ||||
|                 bone.localRotation = {0, 0, 0, 1}; | ||||
|             } | ||||
|             else | ||||
|             { | ||||
|                 bone.localOffset[0] = model->trans[boneNum - model->numRootBones][0]; | ||||
|                 bone.localOffset[1] = model->trans[boneNum - model->numRootBones][1]; | ||||
|                 bone.localOffset[2] = model->trans[boneNum - model->numRootBones][2]; | ||||
|                 bone.localRotation = { | ||||
|                     QuatInt16::ToFloat(model->quats[boneNum - model->numRootBones][0]), | ||||
|                     QuatInt16::ToFloat(model->quats[boneNum - model->numRootBones][1]), | ||||
|                     QuatInt16::ToFloat(model->quats[boneNum - model->numRootBones][2]), | ||||
|                     QuatInt16::ToFloat(model->quats[boneNum - model->numRootBones][3]), | ||||
|                 }; | ||||
|             } | ||||
|  | ||||
|             out.m_bones.emplace_back(std::move(bone)); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     const char* AssetName(const char* input) | ||||
|     { | ||||
|         if (input && input[0] == ',') | ||||
|             return &input[1]; | ||||
|  | ||||
|         return input; | ||||
|     } | ||||
|  | ||||
|     void AddXModelMaterials(XModelCommon& out, DistinctMapper<Material*>& materialMapper, const XModel* model) | ||||
|     { | ||||
|         for (auto surfaceMaterialNum = 0; surfaceMaterialNum < model->numsurfs; surfaceMaterialNum++) | ||||
|         { | ||||
|             Material* material = model->materialHandles[surfaceMaterialNum]; | ||||
|             if (materialMapper.Add(material)) | ||||
|             { | ||||
|                 XModelMaterial xMaterial; | ||||
|                 xMaterial.ApplyDefaults(); | ||||
|  | ||||
|                 xMaterial.name = AssetName(material->info.name); | ||||
|                 const auto* colorMap = GetMaterialColorMap(material); | ||||
|                 if (colorMap) | ||||
|                     xMaterial.colorMapName = AssetName(colorMap->name); | ||||
|  | ||||
|                 const auto* normalMap = GetMaterialNormalMap(material); | ||||
|                 if (normalMap) | ||||
|                     xMaterial.normalMapName = AssetName(normalMap->name); | ||||
|  | ||||
|                 const auto* specularMap = GetMaterialSpecularMap(material); | ||||
|                 if (specularMap) | ||||
|                     xMaterial.specularMapName = AssetName(specularMap->name); | ||||
|  | ||||
|                 out.m_materials.emplace_back(std::move(xMaterial)); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     void AddXModelObjects(XModelCommon& out, const XModelSurfs* modelSurfs, const DistinctMapper<Material*>& materialMapper, const int baseSurfaceIndex) | ||||
|     { | ||||
|         for (auto surfIndex = 0u; surfIndex < modelSurfs->numsurfs; surfIndex++) | ||||
|         { | ||||
|             XModelObject object; | ||||
|             object.name = std::format("surf{}", surfIndex); | ||||
|             object.materialIndex = static_cast<int>(materialMapper.GetDistinctPositionByInputPosition(surfIndex + baseSurfaceIndex)); | ||||
|  | ||||
|             out.m_objects.emplace_back(std::move(object)); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     void AddXModelVertices(XModelCommon& out, const XModelSurfs* modelSurfs) | ||||
|     { | ||||
|         for (auto surfIndex = 0u; surfIndex < modelSurfs->numsurfs; surfIndex++) | ||||
|         { | ||||
|             const auto& surface = modelSurfs->surfs[surfIndex]; | ||||
|  | ||||
|             for (auto vertexIndex = 0u; vertexIndex < surface.vertCount; vertexIndex++) | ||||
|             { | ||||
|                 const auto& v = surface.verts0.packedVerts0[vertexIndex]; | ||||
|  | ||||
|                 XModelVertex vertex{}; | ||||
|                 vertex.coordinates[0] = v.xyz[0]; | ||||
|                 vertex.coordinates[1] = v.xyz[1]; | ||||
|                 vertex.coordinates[2] = v.xyz[2]; | ||||
|                 Common::Vec3UnpackUnitVec(v.normal, vertex.normal); | ||||
|                 Common::Vec4UnpackGfxColor(v.color, vertex.color); | ||||
|                 Common::Vec2UnpackTexCoords(v.texCoord, vertex.uv); | ||||
|  | ||||
|                 out.m_vertices.emplace_back(vertex); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     void AllocateXModelBoneWeights(const XModelSurfs* modelSurfs, XModelVertexBoneWeightCollection& weightCollection) | ||||
|     { | ||||
|         auto totalWeightCount = 0u; | ||||
|         for (auto surfIndex = 0u; surfIndex < modelSurfs->numsurfs; surfIndex++) | ||||
|         { | ||||
|             const auto& surface = modelSurfs->surfs[surfIndex]; | ||||
|  | ||||
|             if (surface.vertList) | ||||
|             { | ||||
|                 totalWeightCount += surface.vertListCount; | ||||
|             } | ||||
|  | ||||
|             if (surface.vertInfo.vertsBlend) | ||||
|             { | ||||
|                 totalWeightCount += surface.vertInfo.vertCount[0] * 1; | ||||
|                 totalWeightCount += surface.vertInfo.vertCount[1] * 2; | ||||
|                 totalWeightCount += surface.vertInfo.vertCount[2] * 3; | ||||
|                 totalWeightCount += surface.vertInfo.vertCount[3] * 4; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         weightCollection.weights.resize(totalWeightCount); | ||||
|     } | ||||
|  | ||||
|     float BoneWeight16(const uint16_t value) | ||||
|     { | ||||
|         return static_cast<float>(value) / static_cast<float>(std::numeric_limits<uint16_t>::max()); | ||||
|     } | ||||
|  | ||||
|     void AddXModelVertexBoneWeights(XModelCommon& out, const XModelSurfs* modelSurfs) | ||||
|     { | ||||
|         auto& weightCollection = out.m_bone_weight_data; | ||||
|         size_t weightOffset = 0u; | ||||
|  | ||||
|         for (auto surfIndex = 0u; surfIndex < modelSurfs->numsurfs; surfIndex++) | ||||
|         { | ||||
|             const auto& surface = modelSurfs->surfs[surfIndex]; | ||||
|             auto handledVertices = 0u; | ||||
|  | ||||
|             if (surface.vertList) | ||||
|             { | ||||
|                 for (auto vertListIndex = 0u; vertListIndex < surface.vertListCount; vertListIndex++) | ||||
|                 { | ||||
|                     const auto& vertList = surface.vertList[vertListIndex]; | ||||
|                     const auto boneWeightOffset = weightOffset; | ||||
|  | ||||
|                     weightCollection.weights[weightOffset++] = XModelBoneWeight{vertList.boneOffset / sizeof(DObjSkelMat), 1.0f}; | ||||
|  | ||||
|                     for (auto vertListVertexOffset = 0u; vertListVertexOffset < vertList.vertCount; vertListVertexOffset++) | ||||
|                     { | ||||
|                         out.m_vertex_bone_weights.emplace_back(boneWeightOffset, 1); | ||||
|                     } | ||||
|                     handledVertices += vertList.vertCount; | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             auto vertsBlendOffset = 0u; | ||||
|             if (surface.vertInfo.vertsBlend) | ||||
|             { | ||||
|                 // 1 bone weight | ||||
|                 for (auto vertIndex = 0; vertIndex < surface.vertInfo.vertCount[0]; vertIndex++) | ||||
|                 { | ||||
|                     const auto boneWeightOffset = weightOffset; | ||||
|                     const auto boneIndex0 = surface.vertInfo.vertsBlend[vertsBlendOffset + 0] / sizeof(DObjSkelMat); | ||||
|                     weightCollection.weights[weightOffset++] = XModelBoneWeight{boneIndex0, 1.0f}; | ||||
|  | ||||
|                     vertsBlendOffset += 1; | ||||
|  | ||||
|                     out.m_vertex_bone_weights.emplace_back(boneWeightOffset, 1); | ||||
|                 } | ||||
|  | ||||
|                 // 2 bone weights | ||||
|                 for (auto vertIndex = 0; vertIndex < surface.vertInfo.vertCount[1]; vertIndex++) | ||||
|                 { | ||||
|                     const auto boneWeightOffset = weightOffset; | ||||
|                     const auto boneIndex0 = surface.vertInfo.vertsBlend[vertsBlendOffset + 0] / sizeof(DObjSkelMat); | ||||
|                     const auto boneIndex1 = surface.vertInfo.vertsBlend[vertsBlendOffset + 1] / sizeof(DObjSkelMat); | ||||
|                     const auto boneWeight1 = BoneWeight16(surface.vertInfo.vertsBlend[vertsBlendOffset + 2]); | ||||
|                     const auto boneWeight0 = 1.0f - boneWeight1; | ||||
|  | ||||
|                     weightCollection.weights[weightOffset++] = XModelBoneWeight{boneIndex0, boneWeight0}; | ||||
|                     weightCollection.weights[weightOffset++] = XModelBoneWeight{boneIndex1, boneWeight1}; | ||||
|  | ||||
|                     vertsBlendOffset += 3; | ||||
|  | ||||
|                     out.m_vertex_bone_weights.emplace_back(boneWeightOffset, 2); | ||||
|                 } | ||||
|  | ||||
|                 // 3 bone weights | ||||
|                 for (auto vertIndex = 0; vertIndex < surface.vertInfo.vertCount[2]; vertIndex++) | ||||
|                 { | ||||
|                     const auto boneWeightOffset = weightOffset; | ||||
|                     const auto boneIndex0 = surface.vertInfo.vertsBlend[vertsBlendOffset + 0] / sizeof(DObjSkelMat); | ||||
|                     const auto boneIndex1 = surface.vertInfo.vertsBlend[vertsBlendOffset + 1] / sizeof(DObjSkelMat); | ||||
|                     const auto boneWeight1 = BoneWeight16(surface.vertInfo.vertsBlend[vertsBlendOffset + 2]); | ||||
|                     const auto boneIndex2 = surface.vertInfo.vertsBlend[vertsBlendOffset + 3] / sizeof(DObjSkelMat); | ||||
|                     const auto boneWeight2 = BoneWeight16(surface.vertInfo.vertsBlend[vertsBlendOffset + 4]); | ||||
|                     const auto boneWeight0 = 1.0f - boneWeight1 - boneWeight2; | ||||
|  | ||||
|                     weightCollection.weights[weightOffset++] = XModelBoneWeight{boneIndex0, boneWeight0}; | ||||
|                     weightCollection.weights[weightOffset++] = XModelBoneWeight{boneIndex1, boneWeight1}; | ||||
|                     weightCollection.weights[weightOffset++] = XModelBoneWeight{boneIndex2, boneWeight2}; | ||||
|  | ||||
|                     vertsBlendOffset += 5; | ||||
|  | ||||
|                     out.m_vertex_bone_weights.emplace_back(boneWeightOffset, 3); | ||||
|                 } | ||||
|  | ||||
|                 // 4 bone weights | ||||
|                 for (auto vertIndex = 0; vertIndex < surface.vertInfo.vertCount[3]; vertIndex++) | ||||
|                 { | ||||
|                     const auto boneWeightOffset = weightOffset; | ||||
|                     const auto boneIndex0 = surface.vertInfo.vertsBlend[vertsBlendOffset + 0] / sizeof(DObjSkelMat); | ||||
|                     const auto boneIndex1 = surface.vertInfo.vertsBlend[vertsBlendOffset + 1] / sizeof(DObjSkelMat); | ||||
|                     const auto boneWeight1 = BoneWeight16(surface.vertInfo.vertsBlend[vertsBlendOffset + 2]); | ||||
|                     const auto boneIndex2 = surface.vertInfo.vertsBlend[vertsBlendOffset + 3] / sizeof(DObjSkelMat); | ||||
|                     const auto boneWeight2 = BoneWeight16(surface.vertInfo.vertsBlend[vertsBlendOffset + 4]); | ||||
|                     const auto boneIndex3 = surface.vertInfo.vertsBlend[vertsBlendOffset + 5] / sizeof(DObjSkelMat); | ||||
|                     const auto boneWeight3 = BoneWeight16(surface.vertInfo.vertsBlend[vertsBlendOffset + 6]); | ||||
|                     const auto boneWeight0 = 1.0f - boneWeight1 - boneWeight2 - boneWeight3; | ||||
|  | ||||
|                     weightCollection.weights[weightOffset++] = XModelBoneWeight{boneIndex0, boneWeight0}; | ||||
|                     weightCollection.weights[weightOffset++] = XModelBoneWeight{boneIndex1, boneWeight1}; | ||||
|                     weightCollection.weights[weightOffset++] = XModelBoneWeight{boneIndex2, boneWeight2}; | ||||
|                     weightCollection.weights[weightOffset++] = XModelBoneWeight{boneIndex3, boneWeight3}; | ||||
|  | ||||
|                     vertsBlendOffset += 7; | ||||
|  | ||||
|                     out.m_vertex_bone_weights.emplace_back(boneWeightOffset, 4); | ||||
|                 } | ||||
|  | ||||
|                 handledVertices += | ||||
|                     surface.vertInfo.vertCount[0] + surface.vertInfo.vertCount[1] + surface.vertInfo.vertCount[2] + surface.vertInfo.vertCount[3]; | ||||
|             } | ||||
|  | ||||
|             for (; handledVertices < surface.vertCount; handledVertices++) | ||||
|             { | ||||
|                 out.m_vertex_bone_weights.emplace_back(0, 0); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     void AddXModelFaces(XModelCommon& out, const XModelSurfs* modelSurfs) | ||||
|     { | ||||
|         for (auto surfIndex = 0u; surfIndex < modelSurfs->numsurfs; surfIndex++) | ||||
|         { | ||||
|             const auto& surface = modelSurfs->surfs[surfIndex]; | ||||
|             auto& object = out.m_objects[surfIndex]; | ||||
|             object.m_faces.reserve(surface.triCount); | ||||
|  | ||||
|             for (auto triIndex = 0u; triIndex < surface.triCount; triIndex++) | ||||
|             { | ||||
|                 const auto& tri = surface.triIndices[triIndex]; | ||||
|  | ||||
|                 XModelFace face{}; | ||||
|                 face.vertexIndex[0] = tri[0] + surface.baseVertIndex; | ||||
|                 face.vertexIndex[1] = tri[1] + surface.baseVertIndex; | ||||
|                 face.vertexIndex[2] = tri[2] + surface.baseVertIndex; | ||||
|                 object.m_faces.emplace_back(face); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     void PopulateXModelWriter(XModelCommon& out, const AssetDumpingContext& context, const unsigned lod, const XModel* model) | ||||
|     { | ||||
|         const auto* modelSurfs = model->lodInfo[lod].modelSurfs; | ||||
|  | ||||
|         DistinctMapper<Material*> materialMapper(model->numsurfs); | ||||
|         AllocateXModelBoneWeights(modelSurfs, out.m_bone_weight_data); | ||||
|  | ||||
|         out.m_name = modelSurfs->name; | ||||
|         AddXModelBones(out, context, model); | ||||
|         AddXModelMaterials(out, materialMapper, model); | ||||
|         AddXModelObjects(out, modelSurfs, materialMapper, model->lodInfo[lod].surfIndex); | ||||
|         AddXModelVertices(out, modelSurfs); | ||||
|         AddXModelVertexBoneWeights(out, modelSurfs); | ||||
|         AddXModelFaces(out, modelSurfs); | ||||
|     } | ||||
|  | ||||
|     void DumpObjMtl(const XModelCommon& common, const AssetDumpingContext& context, const XAssetInfo<XModel>* asset) | ||||
|     { | ||||
|         const auto* model = asset->Asset(); | ||||
|         const auto mtlFile = context.OpenAssetFile(std::format("model_export/{}.mtl", model->name)); | ||||
|  | ||||
|         if (!mtlFile) | ||||
|             return; | ||||
|  | ||||
|         const auto writer = obj::CreateMtlWriter(*mtlFile, context.m_zone->m_game->GetShortName(), context.m_zone->m_name); | ||||
|         DistinctMapper<Material*> materialMapper(model->numsurfs); | ||||
|  | ||||
|         writer->Write(common); | ||||
|     } | ||||
|  | ||||
|     void DumpObjLod(const XModelCommon& common, const AssetDumpingContext& context, const XAssetInfo<XModel>* asset, const unsigned lod) | ||||
|     { | ||||
|         const auto* model = asset->Asset(); | ||||
|         const auto* modelSurfs = model->lodInfo[lod].modelSurfs; | ||||
|  | ||||
|         if (modelSurfs->name[0] == ',' || modelSurfs->surfs == nullptr) | ||||
|             return; | ||||
|  | ||||
|         const auto assetFile = context.OpenAssetFile(std::format("model_export/{}.obj", modelSurfs->name)); | ||||
|  | ||||
|         if (!assetFile) | ||||
|             return; | ||||
|  | ||||
|         const auto writer = | ||||
|             obj::CreateObjWriter(*assetFile, std::format("{}.mtl", model->name), context.m_zone->m_game->GetShortName(), context.m_zone->m_name); | ||||
|         DistinctMapper<Material*> materialMapper(model->numsurfs); | ||||
|  | ||||
|         writer->Write(common); | ||||
|     } | ||||
|  | ||||
|     void DumpXModelExportLod(const XModelCommon& common, const AssetDumpingContext& context, const XAssetInfo<XModel>* asset, const unsigned lod) | ||||
|     { | ||||
|         const auto* model = asset->Asset(); | ||||
|         const auto* modelSurfs = model->lodInfo[lod].modelSurfs; | ||||
|         const auto assetFile = context.OpenAssetFile(std::format("model_export/{}.XMODEL_EXPORT", modelSurfs->name)); | ||||
|  | ||||
|         if (!assetFile) | ||||
|             return; | ||||
|  | ||||
|         const auto writer = xmodel_export::CreateWriterForVersion6(*assetFile, context.m_zone->m_game->GetShortName(), context.m_zone->m_name); | ||||
|         writer->Write(common); | ||||
|     } | ||||
|  | ||||
|     template<typename T> | ||||
|     void DumpGltfLod( | ||||
|         const XModelCommon& common, const AssetDumpingContext& context, const XAssetInfo<XModel>* asset, const unsigned lod, const std::string& extension) | ||||
|     { | ||||
|         const auto* model = asset->Asset(); | ||||
|         const auto* modelSurfs = model->lodInfo[lod].modelSurfs; | ||||
|         const auto assetFile = context.OpenAssetFile(std::format("model_export/{}{}", modelSurfs->name, extension)); | ||||
|  | ||||
|         if (!assetFile) | ||||
|             return; | ||||
|  | ||||
|         const auto output = std::make_unique<T>(*assetFile); | ||||
|         const auto writer = gltf::Writer::CreateWriter(output.get(), context.m_zone->m_game->GetShortName(), context.m_zone->m_name); | ||||
|  | ||||
|         writer->Write(common); | ||||
|     } | ||||
|  | ||||
|     void DumpXModelSurfs(const AssetDumpingContext& context, const XAssetInfo<XModel>* asset) | ||||
|     { | ||||
|         const auto* model = asset->Asset(); | ||||
|  | ||||
|         for (auto currentLod = 0u; currentLod < model->numLods; currentLod++) | ||||
|         { | ||||
|             XModelCommon common; | ||||
|             PopulateXModelWriter(common, context, currentLod, asset->Asset()); | ||||
|  | ||||
|             switch (ObjWriting::Configuration.ModelOutputFormat) | ||||
|             { | ||||
|             case ObjWriting::Configuration_t::ModelOutputFormat_e::OBJ: | ||||
|                 DumpObjLod(common, context, asset, currentLod); | ||||
|                 if (currentLod == 0u) | ||||
|                     DumpObjMtl(common, context, asset); | ||||
|                 break; | ||||
|  | ||||
|             case ObjWriting::Configuration_t::ModelOutputFormat_e::XMODEL_EXPORT: | ||||
|                 DumpXModelExportLod(common, context, asset, currentLod); | ||||
|                 break; | ||||
|  | ||||
|             case ObjWriting::Configuration_t::ModelOutputFormat_e::GLTF: | ||||
|                 DumpGltfLod<gltf::TextOutput>(common, context, asset, currentLod, ".gltf"); | ||||
|                 break; | ||||
|  | ||||
|             case ObjWriting::Configuration_t::ModelOutputFormat_e::GLB: | ||||
|                 DumpGltfLod<gltf::BinOutput>(common, context, asset, currentLod, ".glb"); | ||||
|                 break; | ||||
|  | ||||
|             default: | ||||
|                 assert(false); | ||||
|                 break; | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } // namespace | ||||
|  | ||||
| bool AssetDumperXModel::ShouldDump(XAssetInfo<XModel>* asset) | ||||
| { | ||||
|     return !asset->m_name.empty() && asset->m_name[0] != ','; | ||||
| @@ -508,5 +11,5 @@ bool AssetDumperXModel::ShouldDump(XAssetInfo<XModel>* asset) | ||||
|  | ||||
| void AssetDumperXModel::DumpAsset(AssetDumpingContext& context, XAssetInfo<XModel>* asset) | ||||
| { | ||||
|     DumpXModelSurfs(context, asset); | ||||
|     DumpXModel(context, asset); | ||||
| } | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| #options GAME (T5, T6) | ||||
| #options GAME (IW5, T5, T6) | ||||
|  | ||||
| #filename "Game/" + GAME + "/XModel/XModelDumper" + GAME + ".cpp" | ||||
|  | ||||
| @@ -6,7 +6,9 @@ | ||||
| #set COMMON_HEADER "\"Game/" + GAME + "/Common" + GAME + ".h\"" | ||||
| #set JSON_HEADER "\"Game/" + GAME + "/XModel/JsonXModel" + GAME + ".h\"" | ||||
|  | ||||
| #if GAME == "T5" | ||||
| #if GAME == "IW5" | ||||
| #define FEATURE_IW5 | ||||
| #elif GAME == "T5" | ||||
| #define FEATURE_T5 | ||||
| #elif GAME == "T6" | ||||
| #define FEATURE_T6 | ||||
| @@ -54,8 +56,13 @@ namespace GAME | ||||
|         { | ||||
|             MaterialTextureDef* def = &material->textureTable[textureIndex]; | ||||
|  | ||||
| #ifdef FEATURE_IW5             | ||||
|             if (def->semantic == TS_COLOR_MAP) | ||||
|                 potentialTextureDefs.push_back(def); | ||||
| #else | ||||
|             if (def->semantic == TS_COLOR_MAP || def->semantic >= TS_COLOR0_MAP && def->semantic <= TS_COLOR15_MAP) | ||||
|                 potentialTextureDefs.push_back(def); | ||||
| #endif | ||||
|         } | ||||
|  | ||||
|         if (potentialTextureDefs.empty()) | ||||
| @@ -136,15 +143,33 @@ namespace GAME | ||||
|         return GetImageFromTextureDef(*potentialTextureDefs[0]); | ||||
|     } | ||||
|  | ||||
|     bool GetSurfaces(const XModel* model, const unsigned lod, XSurface*& surfs, unsigned& surfCount) | ||||
|     { | ||||
| #ifdef FEATURE_IW5 | ||||
|         if (!model->lodInfo[lod].modelSurfs || !model->lodInfo[lod].modelSurfs->surfs) | ||||
|             return false; | ||||
|  | ||||
|         surfs = model->lodInfo[lod].modelSurfs->surfs; | ||||
|         surfCount = model->lodInfo[lod].modelSurfs->numsurfs; | ||||
| #else | ||||
|         if (!model->surfs) | ||||
|             return false; | ||||
|  | ||||
|         surfs = &model->surfs[model->lodInfo[lod].surfIndex]; | ||||
|         surfCount = model->lodInfo[lod].numsurfs; | ||||
| #endif | ||||
|  | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
|     bool HasDefaultArmature(const XModel* model, const unsigned lod) | ||||
|     { | ||||
|         if (model->numRootBones != 1 || model->numBones != 1) | ||||
|             return false; | ||||
|  | ||||
|         const auto* surfs = &model->surfs[model->lodInfo[lod].surfIndex]; | ||||
|         const auto surfCount = model->lodInfo[lod].numsurfs; | ||||
|  | ||||
|         if (!surfs) | ||||
|         XSurface* surfs; | ||||
|         unsigned surfCount; | ||||
|         if (!GetSurfaces(model, lod, surfs, surfCount)) | ||||
|             return true; | ||||
|  | ||||
|         for (auto surfIndex = 0u; surfIndex < surfCount; surfIndex++) | ||||
| @@ -284,10 +309,9 @@ namespace GAME | ||||
|  | ||||
|     void AddXModelVertices(XModelCommon& out, const XModel* model, const unsigned lod) | ||||
|     { | ||||
|         const auto* surfs = &model->surfs[model->lodInfo[lod].surfIndex]; | ||||
|         const auto surfCount = model->lodInfo[lod].numsurfs; | ||||
|  | ||||
|         if (!surfs) | ||||
|         XSurface* surfs; | ||||
|         unsigned surfCount; | ||||
|         if (!GetSurfaces(model, lod, surfs, surfCount)) | ||||
|             return; | ||||
|  | ||||
|         for (auto surfIndex = 0u; surfIndex < surfCount; surfIndex++) | ||||
| @@ -313,10 +337,9 @@ namespace GAME | ||||
|  | ||||
|     void AllocateXModelBoneWeights(const XModel* model, const unsigned lod, XModelVertexBoneWeightCollection& weightCollection) | ||||
|     { | ||||
|         const auto* surfs = &model->surfs[model->lodInfo[lod].surfIndex]; | ||||
|         const auto surfCount = model->lodInfo[lod].numsurfs; | ||||
|  | ||||
|         if (!surfs) | ||||
|         XSurface* surfs; | ||||
|         unsigned surfCount; | ||||
|         if (!GetSurfaces(model, lod, surfs, surfCount)) | ||||
|             return; | ||||
|  | ||||
|         auto totalWeightCount = 0u; | ||||
| @@ -348,13 +371,12 @@ namespace GAME | ||||
|  | ||||
|     void AddXModelVertexBoneWeights(XModelCommon& out, const XModel* model, const unsigned lod) | ||||
|     { | ||||
|         const auto* surfs = &model->surfs[model->lodInfo[lod].surfIndex]; | ||||
|         const auto surfCount = model->lodInfo[lod].numsurfs; | ||||
|         auto& weightCollection = out.m_bone_weight_data; | ||||
|  | ||||
|         if (!surfs) | ||||
|         XSurface* surfs; | ||||
|         unsigned surfCount; | ||||
|         if (!GetSurfaces(model, lod, surfs, surfCount)) | ||||
|             return; | ||||
|  | ||||
|         auto& weightCollection = out.m_bone_weight_data; | ||||
|         size_t weightOffset = 0u; | ||||
|  | ||||
|         for (auto surfIndex = 0u; surfIndex < surfCount; surfIndex++) | ||||
| @@ -467,10 +489,9 @@ namespace GAME | ||||
|  | ||||
|     void AddXModelFaces(XModelCommon& out, const XModel* model, const unsigned lod) | ||||
|     { | ||||
|         const auto* surfs = &model->surfs[model->lodInfo[lod].surfIndex]; | ||||
|         const auto surfCount = model->lodInfo[lod].numsurfs; | ||||
|  | ||||
|         if (!surfs) | ||||
|         XSurface* surfs; | ||||
|         unsigned surfCount; | ||||
|         if (!GetSurfaces(model, lod, surfs, surfCount)) | ||||
|             return; | ||||
|  | ||||
|         for (auto surfIndex = 0u; surfIndex < surfCount; surfIndex++) | ||||
| @@ -671,8 +692,15 @@ namespace GAME | ||||
|             if (xmodel.physPreset && xmodel.physPreset->name) | ||||
|                 jXModel.physPreset = AssetName(xmodel.physPreset->name); | ||||
|  | ||||
| #ifdef FEATURE_IW5 | ||||
|             if (xmodel.physCollmap && xmodel.physCollmap->name) | ||||
|                 jXModel.physCollmap = AssetName(xmodel.physCollmap->name); | ||||
| #endif | ||||
|  | ||||
| #if defined(FEATURE_T5) || defined(FEATURE_T6) | ||||
|             if (xmodel.physConstraints && xmodel.physConstraints->name) | ||||
|                 jXModel.physConstraints = AssetName(xmodel.physConstraints->name); | ||||
| #endif | ||||
|  | ||||
|             jXModel.flags = xmodel.flags; | ||||
|  | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| #options GAME (T5, T6) | ||||
| #options GAME (IW5, T5, T6) | ||||
|  | ||||
| #filename "Game/" + GAME + "/XModel/XModelDumper" + GAME + ".h" | ||||
|  | ||||
|   | ||||
		Reference in New Issue
	
	Block a user