Merge pull request #3721 from vfxgordon/fbxBlendshapes
Added Blendshape Support to FBX Exportpull/3727/head
commit
3880cd225c
|
@ -812,6 +812,18 @@ void FBXExporter::WriteDefinitions ()
|
||||||
// Geometry / FbxMesh
|
// Geometry / FbxMesh
|
||||||
// <~~ aiMesh
|
// <~~ aiMesh
|
||||||
count = mScene->mNumMeshes;
|
count = mScene->mNumMeshes;
|
||||||
|
|
||||||
|
// Blendshapes are considered Geometry
|
||||||
|
int32_t bsDeformerCount=0;
|
||||||
|
for (size_t mi = 0; mi < mScene->mNumMeshes; ++mi) {
|
||||||
|
aiMesh* m = mScene->mMeshes[mi];
|
||||||
|
if (m->mNumAnimMeshes > 0) {
|
||||||
|
count+=m->mNumAnimMeshes;
|
||||||
|
bsDeformerCount+=m->mNumAnimMeshes; // One deformer per blendshape
|
||||||
|
bsDeformerCount++; // Plus one master blendshape deformer
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (count) {
|
if (count) {
|
||||||
n = FBX::Node("ObjectType", "Geometry");
|
n = FBX::Node("ObjectType", "Geometry");
|
||||||
n.AddChild("Count", count);
|
n.AddChild("Count", count);
|
||||||
|
@ -978,7 +990,7 @@ void FBXExporter::WriteDefinitions ()
|
||||||
}
|
}
|
||||||
|
|
||||||
// Deformer
|
// Deformer
|
||||||
count = int32_t(count_deformers(mScene));
|
count = int32_t(count_deformers(mScene))+bsDeformerCount;
|
||||||
if (count) {
|
if (count) {
|
||||||
n = FBX::Node("ObjectType", "Deformer");
|
n = FBX::Node("ObjectType", "Deformer");
|
||||||
n.AddChild("Count", count);
|
n.AddChild("Count", count);
|
||||||
|
@ -1363,6 +1375,7 @@ void FBXExporter::WriteObjects ()
|
||||||
n.End(outstream, binary, indent, true);
|
n.End(outstream, binary, indent, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// aiMaterial
|
// aiMaterial
|
||||||
material_uids.clear();
|
material_uids.clear();
|
||||||
for (size_t i = 0; i < mScene->mNumMaterials; ++i) {
|
for (size_t i = 0; i < mScene->mNumMaterials; ++i) {
|
||||||
|
@ -1697,6 +1710,100 @@ void FBXExporter::WriteObjects ()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Blendshapes, if any
|
||||||
|
for (size_t mi = 0; mi < mScene->mNumMeshes; ++mi) {
|
||||||
|
const aiMesh* m = mScene->mMeshes[mi];
|
||||||
|
if (m->mNumAnimMeshes == 0) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
// make a deformer for this mesh
|
||||||
|
int64_t deformer_uid = generate_uid();
|
||||||
|
FBX::Node dnode("Deformer");
|
||||||
|
dnode.AddProperties(deformer_uid, m->mName.data + FBX::SEPARATOR + "Blendshapes", "BlendShape");
|
||||||
|
dnode.AddChild("Version", int32_t(101));
|
||||||
|
dnode.Dump(outstream, binary, indent);
|
||||||
|
// connect it
|
||||||
|
connections.emplace_back("C", "OO", deformer_uid, mesh_uids[mi]);
|
||||||
|
std::vector<int32_t> vertex_indices = vVertexIndice[mi];
|
||||||
|
|
||||||
|
for (unsigned int am = 0; am < m->mNumAnimMeshes; ++am) {
|
||||||
|
aiAnimMesh *pAnimMesh = m->mAnimMeshes[am];
|
||||||
|
std::string blendshape_name = pAnimMesh->mName.data;
|
||||||
|
|
||||||
|
// start the node record
|
||||||
|
FBX::Node bsnode("Geometry");
|
||||||
|
int64_t blendshape_uid = generate_uid();
|
||||||
|
mesh_uids.push_back(blendshape_uid);
|
||||||
|
bsnode.AddProperty(blendshape_uid);
|
||||||
|
bsnode.AddProperty(blendshape_name + FBX::SEPARATOR + "Blendshape");
|
||||||
|
bsnode.AddProperty("Shape");
|
||||||
|
bsnode.AddChild("Version", int32_t(100));
|
||||||
|
bsnode.Begin(outstream, binary, indent);
|
||||||
|
bsnode.DumpProperties(outstream, binary, indent);
|
||||||
|
bsnode.EndProperties(outstream, binary, indent);
|
||||||
|
bsnode.BeginChildren(outstream, binary, indent);
|
||||||
|
indent++;
|
||||||
|
if (pAnimMesh->HasPositions()) {
|
||||||
|
std::vector<int32_t>shape_indices;
|
||||||
|
std::vector<double>pPositionDiff;
|
||||||
|
std::vector<double>pNormalDiff;
|
||||||
|
|
||||||
|
for (unsigned int vt = 0; vt < vertex_indices.size(); ++vt) {
|
||||||
|
aiVector3D pDiff = (pAnimMesh->mVertices[vertex_indices[vt]] - m->mVertices[vertex_indices[vt]]);
|
||||||
|
if(pDiff.Length()>1e-8){
|
||||||
|
shape_indices.push_back(vertex_indices[vt]);
|
||||||
|
pPositionDiff.push_back(pDiff[0]);
|
||||||
|
pPositionDiff.push_back(pDiff[1]);
|
||||||
|
pPositionDiff.push_back(pDiff[2]);
|
||||||
|
|
||||||
|
if (pAnimMesh->HasNormals()) {
|
||||||
|
aiVector3D nDiff = (pAnimMesh->mNormals[vertex_indices[vt]] - m->mNormals[vertex_indices[vt]]);
|
||||||
|
pNormalDiff.push_back(nDiff[0]);
|
||||||
|
pNormalDiff.push_back(nDiff[1]);
|
||||||
|
pNormalDiff.push_back(nDiff[2]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
FBX::Node::WritePropertyNode(
|
||||||
|
"Indexes", shape_indices, outstream, binary, indent
|
||||||
|
);
|
||||||
|
|
||||||
|
FBX::Node::WritePropertyNode(
|
||||||
|
"Vertices", pPositionDiff, outstream, binary, indent
|
||||||
|
);
|
||||||
|
|
||||||
|
if (pNormalDiff.size()>0) {
|
||||||
|
FBX::Node::WritePropertyNode(
|
||||||
|
"Normals", pNormalDiff, outstream, binary, indent
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
indent--;
|
||||||
|
bsnode.End(outstream, binary, indent, true);
|
||||||
|
|
||||||
|
// Add blendshape Channel Deformer
|
||||||
|
FBX::Node sdnode("Deformer");
|
||||||
|
const int64_t blendchannel_uid = generate_uid();
|
||||||
|
sdnode.AddProperties(
|
||||||
|
blendchannel_uid, blendshape_name + FBX::SEPARATOR + "SubDeformer", "BlendShapeChannel"
|
||||||
|
);
|
||||||
|
sdnode.AddChild("Version", int32_t(100));
|
||||||
|
sdnode.AddChild("DeformPercent", int32_t(100));
|
||||||
|
FBX::Node p("Properties70");
|
||||||
|
p.AddP70numberA("DeformPercent", 100.);
|
||||||
|
sdnode.AddChild(p);
|
||||||
|
// TODO: Normally just one weight per channel, adding stub for later development
|
||||||
|
std::vector<float>fFullWeights;
|
||||||
|
fFullWeights.push_back(0.);
|
||||||
|
sdnode.AddChild("FullWeights", fFullWeights);
|
||||||
|
sdnode.Dump(outstream, binary, indent);
|
||||||
|
|
||||||
|
connections.emplace_back("C", "OO", blendchannel_uid, deformer_uid);
|
||||||
|
connections.emplace_back("C", "OO", blendshape_uid, blendchannel_uid);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// bones.
|
// bones.
|
||||||
//
|
//
|
||||||
// output structure:
|
// output structure:
|
||||||
|
|
Loading…
Reference in New Issue