mirror of
https://github.com/Laupetin/OpenAssetTools.git
synced 2025-10-23 23:05:52 +00:00
chore: implement vertex weights
This commit is contained in:
@@ -20,6 +20,7 @@
|
||||
#include <format>
|
||||
#include <iostream>
|
||||
#include <nlohmann/json.hpp>
|
||||
#include <numeric>
|
||||
#include <vector>
|
||||
|
||||
using namespace nlohmann;
|
||||
@@ -412,13 +413,166 @@ namespace
|
||||
vertex.tangent = Common::Vec3PackUnitVec(wrongTangent); // TODO: Fill with actual value
|
||||
}
|
||||
|
||||
static size_t GetRigidBoneForVertex(const size_t vertexIndex, const XModelCommon& common)
|
||||
{
|
||||
return common.m_bone_weight_data.weights[common.m_vertex_bone_weights[vertexIndex].weightOffset].boneIndex;
|
||||
}
|
||||
|
||||
static std::vector<std::optional<size_t>>
|
||||
GetRigidBoneIndicesForTris(const std::vector<size_t>& vertexIndices, XSurface& surface, const XModelCommon& common)
|
||||
{
|
||||
std::vector<std::optional<size_t>> rigidBoneIndexForTri;
|
||||
rigidBoneIndexForTri.reserve(surface.triCount);
|
||||
for (auto triIndex = 0u; triIndex < surface.triCount; triIndex++)
|
||||
{
|
||||
const auto& tri = surface.triIndices[triIndex];
|
||||
const auto vert0Bone = GetRigidBoneForVertex(vertexIndices[tri.i[0]], common);
|
||||
const auto vert1Bone = GetRigidBoneForVertex(vertexIndices[tri.i[1]], common);
|
||||
const auto vert2Bone = GetRigidBoneForVertex(vertexIndices[tri.i[2]], common);
|
||||
|
||||
const auto hasSameBone = vert0Bone == vert1Bone && vert1Bone == vert2Bone;
|
||||
if (hasSameBone)
|
||||
rigidBoneIndexForTri.emplace_back(vert0Bone);
|
||||
else
|
||||
rigidBoneIndexForTri.emplace_back(std::nullopt);
|
||||
}
|
||||
|
||||
return rigidBoneIndexForTri;
|
||||
}
|
||||
|
||||
static void ReorderRigidTrisByBoneIndex(const std::vector<size_t>& vertexIndices, XSurface& surface, const XModelCommon& common)
|
||||
{
|
||||
const auto rigidBoneIndexForTri = GetRigidBoneIndicesForTris(vertexIndices, surface, common);
|
||||
|
||||
std::vector<size_t> triSortList(surface.triCount);
|
||||
std::ranges::iota(triSortList, 0);
|
||||
|
||||
std::ranges::sort(triSortList,
|
||||
[&rigidBoneIndexForTri](const size_t triIndex0, const size_t triIndex1)
|
||||
{
|
||||
const auto rigidBone0 = rigidBoneIndexForTri[triIndex0];
|
||||
const auto rigidBone1 = rigidBoneIndexForTri[triIndex1];
|
||||
|
||||
if (rigidBone0.has_value() != rigidBone1.has_value())
|
||||
return rigidBone0.has_value();
|
||||
if (!rigidBone0.has_value())
|
||||
return true;
|
||||
|
||||
return *rigidBone0 < *rigidBone1;
|
||||
});
|
||||
|
||||
std::vector<std::remove_pointer_t<decltype(XSurface::triIndices)>> sortedTris(surface.triCount);
|
||||
for (auto i = 0u; i < surface.triCount; i++)
|
||||
memcpy(&sortedTris[i], &surface.triIndices[triSortList[i]], sizeof(std::remove_pointer_t<decltype(XSurface::triIndices)>));
|
||||
memcpy(surface.triIndices, sortedTris.data(), sizeof(std::remove_pointer_t<decltype(XSurface::triIndices)>) * surface.triCount);
|
||||
}
|
||||
|
||||
void CreateVertListData(XSurface& surface, const std::vector<size_t>& vertexIndices, const XModelCommon& common)
|
||||
{
|
||||
ReorderRigidTrisByBoneIndex(vertexIndices, surface, common);
|
||||
const auto rigidBoneIndexForTri = GetRigidBoneIndicesForTris(vertexIndices, surface, common);
|
||||
|
||||
std::vector<XRigidVertList> vertLists;
|
||||
|
||||
auto currentVertexTail = 0u;
|
||||
auto currentTriTail = 0u;
|
||||
|
||||
const auto vertexCount = vertexIndices.size();
|
||||
const auto triCount = static_cast<size_t>(surface.triCount);
|
||||
const auto boneCount = common.m_bones.size();
|
||||
for (auto boneIndex = 0u; boneIndex < boneCount; boneIndex++)
|
||||
{
|
||||
XRigidVertList boneVertList{};
|
||||
boneVertList.boneOffset = static_cast<uint16_t>(boneIndex * sizeof(DObjSkelMat));
|
||||
|
||||
auto currentVertexHead = currentVertexTail;
|
||||
while (currentVertexHead < vertexCount && GetRigidBoneForVertex(currentVertexHead, common) == boneIndex)
|
||||
currentVertexHead++;
|
||||
|
||||
auto currentTriHead = currentTriTail;
|
||||
while (currentTriHead < triCount && rigidBoneIndexForTri[currentTriHead] && *rigidBoneIndexForTri[currentTriHead] == boneIndex)
|
||||
currentTriHead++;
|
||||
|
||||
boneVertList.vertCount = static_cast<uint16_t>(currentVertexHead - currentVertexTail);
|
||||
boneVertList.triOffset = static_cast<uint16_t>(currentTriTail);
|
||||
boneVertList.triCount = static_cast<uint16_t>(currentTriHead - currentTriTail);
|
||||
|
||||
if (boneVertList.triCount > 0 || boneVertList.vertCount > 0)
|
||||
{
|
||||
boneVertList.collisionTree = nullptr; // TODO
|
||||
vertLists.emplace_back(boneVertList);
|
||||
|
||||
currentVertexTail = currentVertexHead;
|
||||
currentTriTail = currentTriHead;
|
||||
}
|
||||
}
|
||||
|
||||
if (!vertLists.empty())
|
||||
{
|
||||
surface.vertListCount = static_cast<unsigned char>(vertLists.size());
|
||||
surface.vertList = m_memory.Alloc<XRigidVertList>(surface.vertListCount);
|
||||
|
||||
memcpy(surface.vertList, vertLists.data(), sizeof(XRigidVertList) * surface.vertListCount);
|
||||
}
|
||||
}
|
||||
|
||||
void CreateVertsBlendData(XSurface& surface, const std::vector<size_t>& vertexIndices, const XModelCommon& common)
|
||||
{
|
||||
// TODO
|
||||
assert(false);
|
||||
}
|
||||
|
||||
static void ReorderVerticesByWeightCount(std::vector<size_t>& vertexIndices, XSurface& surface, const XModelCommon& common)
|
||||
{
|
||||
if (common.m_vertex_bone_weights.empty())
|
||||
return;
|
||||
|
||||
const auto vertexCount = vertexIndices.size();
|
||||
std::vector<size_t> reorderLookup(vertexCount);
|
||||
std::ranges::iota(reorderLookup, 0);
|
||||
|
||||
std::ranges::sort(reorderLookup,
|
||||
[&common, &vertexIndices](const size_t& i0, const size_t& i1)
|
||||
{
|
||||
const auto& weights0 = common.m_vertex_bone_weights[vertexIndices[i0]];
|
||||
const auto& weights1 = common.m_vertex_bone_weights[vertexIndices[i1]];
|
||||
|
||||
if (weights0.weightCount < weights1.weightCount)
|
||||
return true;
|
||||
|
||||
// If there is only one weight, make sure all vertices of the same bone follow another
|
||||
if (weights0.weightCount == 1)
|
||||
{
|
||||
const auto bone0 = common.m_bone_weight_data.weights[weights0.weightOffset].boneIndex;
|
||||
const auto bone1 = common.m_bone_weight_data.weights[weights1.weightOffset].boneIndex;
|
||||
return bone0 < bone1;
|
||||
}
|
||||
|
||||
return false;
|
||||
});
|
||||
|
||||
for (auto triIndex = 0u; triIndex < surface.triCount; triIndex++)
|
||||
{
|
||||
auto& triIndices = surface.triIndices[triIndex];
|
||||
|
||||
triIndices.i[0] = static_cast<uint16_t>(reorderLookup[triIndices.i[0]]);
|
||||
triIndices.i[1] = static_cast<uint16_t>(reorderLookup[triIndices.i[1]]);
|
||||
triIndices.i[2] = static_cast<uint16_t>(reorderLookup[triIndices.i[2]]);
|
||||
}
|
||||
|
||||
for (auto& entry : reorderLookup)
|
||||
entry = vertexIndices[entry];
|
||||
|
||||
vertexIndices = std::move(reorderLookup);
|
||||
}
|
||||
|
||||
bool CreateXSurface(XSurface& surface, const XModelObject& commonObject, const XModelCommon& common)
|
||||
{
|
||||
std::vector<GfxPackedVertex> verts;
|
||||
std::vector<size_t> vertexIndices;
|
||||
std::unordered_map<size_t, size_t> usedVertices;
|
||||
|
||||
surface.triCount = static_cast<uint16_t>(commonObject.m_faces.size());
|
||||
surface.triIndices = m_memory.Alloc<r_index16_t[3]>(commonObject.m_faces.size());
|
||||
surface.triIndices = m_memory.Alloc<XSurfaceTri>(commonObject.m_faces.size());
|
||||
|
||||
const auto faceCount = commonObject.m_faces.size();
|
||||
for (auto faceIndex = 0u; faceIndex < faceCount; faceIndex++)
|
||||
@@ -432,25 +586,46 @@ namespace
|
||||
const auto existingVertex = usedVertices.find(vertexIndex);
|
||||
if (existingVertex == usedVertices.end())
|
||||
{
|
||||
const auto newVertexIndex = verts.size();
|
||||
tris[faceVertexIndex] = static_cast<r_index16_t>(newVertexIndex);
|
||||
const auto newVertexIndex = vertexIndices.size();
|
||||
tris.i[faceVertexIndex] = static_cast<uint16_t>(newVertexIndex);
|
||||
|
||||
const auto& commonVertex = common.m_vertices[vertexIndex];
|
||||
GfxPackedVertex vertex{};
|
||||
|
||||
CreateVertex(vertex, commonVertex);
|
||||
|
||||
verts.emplace_back(vertex);
|
||||
vertexIndices.emplace_back(vertexIndex);
|
||||
usedVertices.emplace(vertexIndex, newVertexIndex);
|
||||
}
|
||||
else
|
||||
tris[faceVertexIndex] = static_cast<r_index16_t>(existingVertex->second);
|
||||
tris.i[faceVertexIndex] = static_cast<uint16_t>(existingVertex->second);
|
||||
}
|
||||
}
|
||||
|
||||
surface.vertCount = static_cast<uint16_t>(verts.size());
|
||||
surface.verts0 = m_memory.Alloc<GfxPackedVertex>(verts.size());
|
||||
memcpy(surface.verts0, verts.data(), sizeof(GfxPackedVertex) * verts.size());
|
||||
ReorderVerticesByWeightCount(vertexIndices, surface, common);
|
||||
|
||||
const auto vertexCount = vertexIndices.size();
|
||||
surface.vertCount = static_cast<uint16_t>(vertexCount);
|
||||
surface.verts0 = m_memory.Alloc<GfxPackedVertex>(vertexCount);
|
||||
|
||||
for (auto vertexIndex = 0u; vertexIndex < vertexCount; vertexIndex++)
|
||||
{
|
||||
const auto& commonVertex = common.m_vertices[vertexIndex];
|
||||
CreateVertex(surface.verts0[vertexIndex], commonVertex);
|
||||
}
|
||||
|
||||
if (!common.m_vertex_bone_weights.empty())
|
||||
{
|
||||
// Since bone weights are sorted by weight count, the last must have the highest weight count
|
||||
const auto hasVertsBlend = common.m_vertex_bone_weights[vertexIndices[vertexIndices.size() - 1]].weightCount > 1;
|
||||
if (!hasVertsBlend)
|
||||
CreateVertListData(surface, vertexIndices, common);
|
||||
else
|
||||
CreateVertsBlendData(surface, vertexIndices, common);
|
||||
}
|
||||
|
||||
const auto boneCount = common.m_bones.size();
|
||||
for (auto boneIndex = 0u; boneIndex < boneCount; boneIndex++)
|
||||
{
|
||||
const auto partBitsIndex = boneIndex / 32u;
|
||||
const auto shiftValue = 31u - (boneIndex % 32u);
|
||||
surface.partBits[partBitsIndex] = 1 << (31u - shiftValue);
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
Reference in New Issue
Block a user