diff --git a/include/obj/NiGeometry.h b/include/obj/NiGeometry.h index 54ae3313ed4a79848194c09e6166b877215c8cb9..d000d0ca8d30402b9bbe6510dce959d828f3f4ff 100644 --- a/include/obj/NiGeometry.h +++ b/include/obj/NiGeometry.h @@ -92,6 +92,8 @@ public: */ void NiGeometry::ApplySkinOffset(); + void NormalizeSkinWeights(); + /* * Used to determine whether this mesh is influenced by bones as a skin. * \return true if this mesh is a skin, false otherwise. diff --git a/include/obj/NiSkinData.h b/include/obj/NiSkinData.h index 247f26f0ea7421e2771f133794374d45b1e4ea5d..a0900716a11a78f9a568ddfc408f249af0f54881 100644 --- a/include/obj/NiSkinData.h +++ b/include/obj/NiSkinData.h @@ -53,10 +53,12 @@ public: uint GetBoneCount() const; Matrix44 GetBoneTransform( uint bone_index ) const; vector<SkinWeight> GetBoneWeights( uint bone_index ) const; - void SetBoneWeights( uint bone_index, const vector<SkinWeight> & n, Vector3 center, float radius ); + void SetBoneWeights( uint bone_index, const vector<SkinWeight> & n, Vector3 center, float radius ); - Ref<NiSkinPartition> GetSkinPartition() const; - void SetSkinPartition(Ref<NiSkinPartition> skinPartition); + void NormalizeWeights( unsigned numVertices ); + + Ref<NiSkinPartition> GetSkinPartition() const; + void SetSkinPartition(Ref<NiSkinPartition> skinPartition); protected: NI_SKIN_DATA_MEMBERS diff --git a/src/ComplexShape.cpp b/src/ComplexShape.cpp index 8f81dbe068108d5294d36ad474ccfab6f6b93831..b27c1d077fd4b8df5d347f18cb112ec29f8d4991 100644 --- a/src/ComplexShape.cpp +++ b/src/ComplexShape.cpp @@ -827,7 +827,7 @@ Ref<NiAVObject> ComplexShape::Split( Ref<NiNode> & parent, Matrix44 & transform, SkinWeight sk; for ( map<NiNodeRef, float>::iterator wt = cv->weights.begin(); wt != cv->weights.end(); ++wt ) { //Only record influences that make a noticable contribution - if ( wt->second > 0.0f ) { + if ( wt->second > 0.1f ) { sk.index = vert_index; sk.weight = wt->second; if ( shapeWeights.find( wt->first ) == shapeWeights.end() ) { @@ -871,6 +871,8 @@ Ref<NiAVObject> ComplexShape::Split( Ref<NiNode> & parent, Matrix44 & transform, shapes[shape_num]->SetBoneWeights( inf, shapeWeights[ shapeInfluences[inf] ] ); } + shapes[shape_num]->NormalizeSkinWeights(); + if ( max_bones_per_partition > 0 ) { shapes[shape_num]->GenHardwareSkinInfo( max_bones_per_partition ); } diff --git a/src/obj/NiGeometry.cpp b/src/obj/NiGeometry.cpp index 244df7935247fdf9440db458e52d7aeedec2a577..fd441e219d84aecc377a8ab1d4c27871f81849b2 100644 --- a/src/obj/NiGeometry.cpp +++ b/src/obj/NiGeometry.cpp @@ -199,7 +199,6 @@ void NiGeometry::ApplySkinOffset() { continue; } if ( below_root ) { - cout << "Propogating transform of " << *it << endl; (*it)->PropagateTransform(); } } @@ -214,6 +213,20 @@ void NiGeometry::ApplySkinOffset() { skinInstance->GetSkinData()->ResetOffsets( this ); } +void NiGeometry::NormalizeSkinWeights() { + if ( IsSkin() == false ) { + throw runtime_error( "NormalizeSkinWeights called on a mesh that is not a skin." ); + } + NiSkinDataRef niSkinData = this->GetSkinInstance()->GetSkinData(); + + NiGeometryDataRef niGeomData = this->GetData(); + if ( niGeomData == NULL ) { + throw runtime_error( "NormalizeSkinWeights called on a mesh with no geometry data." ); + } + + niSkinData->NormalizeWeights( niGeomData->GetVertexCount() ); +} + bool NiGeometry::IsSkin() { //Determine whether this is a skin by looking for a skin instance and //skin data diff --git a/src/obj/NiNode.cpp b/src/obj/NiNode.cpp index 7bdd567cdadfe27a08afcc7c2efd3a58d4df744f..b20bbe5f4814c3f26e95c9b3c1e56a7d43124866 100644 --- a/src/obj/NiNode.cpp +++ b/src/obj/NiNode.cpp @@ -18,13 +18,13 @@ NiNode::NiNode() NI_NODE_CONSTRUCT { } NiNode::~NiNode() { - //Clear Children - ClearChildren(); - - //Unbind any attached skins + //Unbind any attached skins - must happen before children are cleared for ( list<NiSkinInstance*>::iterator it = skins.begin(); it != skins.end(); ++it ) { (*it)->SkeletonLost(); } + + //Clear Children + ClearChildren(); } void NiNode::Read( istream& in, list<uint> & link_stack, unsigned int version, unsigned int user_version ) { diff --git a/src/obj/NiObject.cpp b/src/obj/NiObject.cpp index d2f8b3e23539a82eb123f63996ad3160b3478fa2..fa1e626343b45ce5cef81c7fa3a618627bd20846 100644 --- a/src/obj/NiObject.cpp +++ b/src/obj/NiObject.cpp @@ -41,8 +41,10 @@ void NiObject::AddRef() const { void NiObject::SubtractRef() const { _ref_count--; if ( _ref_count < 1 ) { - //cout << this->GetIDString() << " died." << endl; + //string id = this->GetIDString(); + //cout << id << " is being destroyed." << endl; delete this; + //cout << "Destruction of " << id << " complete." << endl; } } diff --git a/src/obj/NiPixelData.cpp b/src/obj/NiPixelData.cpp index 9437fe7850fa2505ec64a6494bdec8bd4410abbd..3f171c787ea3a20174f0bc1f8f83f372a45de964 100644 --- a/src/obj/NiPixelData.cpp +++ b/src/obj/NiPixelData.cpp @@ -241,8 +241,6 @@ void NiPixelData::SetColors( const vector<Color4> & new_pixels, bool generate_mi //Pack pixel data for (uint i = 0; i < mipmaps.size(); ++i ) { - cout << "Width: " << mipmaps[i].width << " Height: " << mipmaps[i].height << " Offset: " << mipmaps[i].offset << endl; - if ( i > 0 ) { //Allocate space to store re-sized image. Color4 * resized = new Color4[ mipmaps[i].width * mipmaps[i].height ]; @@ -250,7 +248,6 @@ void NiPixelData::SetColors( const vector<Color4> & new_pixels, bool generate_mi //Visit every other pixel in each row and column of the previous image for ( uint w = 0; w < mipmaps[i-1].width; w+=2 ) { for ( uint h = 0; h < mipmaps[i-1].height; h+=2 ) { - //cout << "w: " << w << " h: " << h << endl; Color4 & av = resized[(h/2) * mipmaps[i].width + (w/2)]; //Start with the value of the current pixel diff --git a/src/obj/NiSkinData.cpp b/src/obj/NiSkinData.cpp index a684e3fa9ad7c583179164bdadb2ea6040a94e4c..efc60f9a85373de6f6fd9d02eb0d782286d83016 100644 --- a/src/obj/NiSkinData.cpp +++ b/src/obj/NiSkinData.cpp @@ -83,6 +83,42 @@ NiSkinData::NiSkinData( const Ref<NiGeometry> & owner ) NI_SKIN_DATA_CONSTRUCT { ResetOffsets( owner ); } +void NiSkinData::NormalizeWeights( unsigned numVertices ) { + vector<float> totals(numVertices); + vector<int> counts(numVertices); + + //Set all totals to 1.0 and all counts to 0 + for ( unsigned v = 0; v < numVertices; ++v ) { + totals[v] = 1.0f; + counts[v] = 0; + } + + //Calculate the total error for each vertex by subtracting the weight from + //each bone from the starting total of 1.0 + //Also count the number of bones affecting each vertex + for ( unsigned b = 0; b < boneList.size(); ++b ) { + for ( unsigned w = 0; w < boneList[b].vertexWeights.size(); ++w ) { + SkinWeight & sw = boneList[b].vertexWeights[w]; + totals[sw.index] -= sw.weight; + counts[sw.index]++; + } + } + + //Divide all error amounts by the number of bones affecting that vertex to + //get the amount of error that should be distributed to each bone. + for ( unsigned v = 0; v < numVertices; ++v ) { + totals[v] = totals[v] / float(counts[v]); + } + + //Distribute the calculated error to each weight + for ( unsigned b = 0; b < boneList.size(); ++b ) { + for ( unsigned w = 0; w < boneList[b].vertexWeights.size(); ++w ) { + SkinWeight & sw = boneList[b].vertexWeights[w]; + sw.weight += totals[sw.index]; + } + } +} + void NiSkinData::ResetOffsets( const Ref<NiGeometry> & owner ) { //Get skin instance diff --git a/src/obj/NiSkinInstance.cpp b/src/obj/NiSkinInstance.cpp index 697609ae531ec38bab8ebd1a68a34a9ad61fee17..67e15094275a62aeefbb6da2936bf4ad5d678dc8 100644 --- a/src/obj/NiSkinInstance.cpp +++ b/src/obj/NiSkinInstance.cpp @@ -47,17 +47,15 @@ NiSkinInstance::NiSkinInstance( Ref<NiNode> skeleton_root, vector< Ref<NiNode> > } NiSkinInstance::~NiSkinInstance() { - //Probably not necessary, and not very safe - ////Unflag any bones that were part of this skin instance - //for ( uint i = 0; i < bones.size(); ++i ) { - // cout << "Bone " << i << ":"; - // cout << bones[i]->GetIDString() << endl; - // bones[i]->SetSkinFlag(false); - //} + //Unflag any bones that were part of this skin instance + for ( uint i = 0; i < bones.size(); ++i ) { + bones[i]->SetSkinFlag(false); + } //Inform Skeleton Root of detatchment and clear it. if ( skeletonRoot != NULL ) { skeletonRoot->RemoveSkin( this ); + skeletonRoot = NULL; } }