From 2a5ab374d0324da5d568d6fd5adf04541bbf5714 Mon Sep 17 00:00:00 2001
From: Shon Ferguson <shonferg@users.sourceforge.net>
Date: Wed, 27 Sep 2006 04:59:23 +0000
Subject: [PATCH] Added NiGeometry::ApplyTransforms function to automatically
 zero out the transforms of a mesh and apply them to its vertices. Added
 NiGeometryData::Transform function to apply an arbitrary transform to all
 vertices and normals of a mesh (used by above). Added
 NiNode::PropagateTransform function to apply a NiNode's local transform to
 its children and then zero it out. Fixed NiTriShapeData to properly set
 numTriangles when SetTriangles is called. NiGeometry::BindSkin function now
 automatically propagates transforms of NiNodes between the root and the
 meshes and then applies those transforms to the vertices, making the
 NiSkinData overall transform unnecessary. Removed "set root to scene root"
 option as it is no longer necessary.

---
 include/obj/NiGeometry.h     |  14 ++++-
 include/obj/NiGeometryData.h |   7 +++
 include/obj/NiNode.h         |   9 ++-
 src/obj/NiGeometry.cpp       | 114 +++++++++++++++++++----------------
 src/obj/NiGeometryData.cpp   |  19 +++---
 src/obj/NiNode.cpp           |  15 +++++
 src/obj/NiTriShapeData.cpp   |   5 +-
 7 files changed, 121 insertions(+), 62 deletions(-)

diff --git a/include/obj/NiGeometry.h b/include/obj/NiGeometry.h
index c1929af1..a1042b24 100644
--- a/include/obj/NiGeometry.h
+++ b/include/obj/NiGeometry.h
@@ -49,7 +49,7 @@ public:
 	 * NiSkinInstance and NiSkinData class. The bones must have a common
 	 * ancestor in the scenegraph.  This becomes the skeleton root.
 	 */
-	void BindSkin( vector< Ref<NiNode> > bone_nodes, bool bind_to_scene = false );
+	void BindSkin( vector< Ref<NiNode> > bone_nodes );
 	void UnbindSkin();
 	/*!
 	 * Sets the skin weights in the attached NiSkinData object.
@@ -69,8 +69,20 @@ public:
 	string GetShader() const;
 	void SetShader( const string & n );
 
+	/*
+	 * Returns the position of the verticies and values of the normals after they
+	 * have been deformed by the positions of their skin influences.
+	 * \param[out] vertices A vector that will be filled with the skin deformed position of the verticies.
+	 * \param[out] normals A vector thta will be filled with the skin deformed normal values.
+	 */
 	void GetSkinDeformation( vector<Vector3> & vertices, vector<Vector3> & normals ) const;
 
+	/*
+	 * Applies the local transform values to the vertices of the geometry and
+	 * zeros them out to the identity.
+	 */
+	void ApplyTransforms();
+
 protected:
 	list< Ref<NiNode> > ListAncestors( const Ref<NiNode> & leaf ) const;
 	NI_GEOMETRY_MEMBERS
diff --git a/include/obj/NiGeometryData.h b/include/obj/NiGeometryData.h
index 3ae06bb5..fbc03b27 100644
--- a/include/obj/NiGeometryData.h
+++ b/include/obj/NiGeometryData.h
@@ -127,6 +127,13 @@ public:
 	 */
 	void SetUVSet( int index, const vector<TexCoord> & in );
 
+	/*!
+	 * Used to apply a transformation directly to all the vertices and normals in
+	 * this mesh.
+	 * \param[in] transform The 4x4 transformation matrix to apply to the vertices and normals in this mesh.  Normals are only affected by the rotation portion of this matrix.
+	 */
+	void Transform( Matrix44 & transform );
+
 protected:
 	NI_GEOMETRY_DATA_MEMBERS
 private:
diff --git a/include/obj/NiNode.h b/include/obj/NiNode.h
index 51b72a4e..68d24b7f 100644
--- a/include/obj/NiNode.h
+++ b/include/obj/NiNode.h
@@ -72,11 +72,18 @@ public:
 	bool IsSplitMeshProxy() const;
 	   
 
-	/*! Causes all children's transforms to be changed so that all the skin
+	/*! 
+	 * Causes all children's transforms to be changed so that all the skin
 	 * pieces line up without any vertex transformations.
 	 */
 	void GoToSkeletonBindPosition();
 
+	/*!
+	 * Applies the local transforms of this node to its children,
+	 * causing itself to be cleared to identity transforms.
+	 */
+	void PropagateTransform();
+
 	/*! 
 	 * Should only be called by NiTriBasedGeom
 	 * Adds a new SkinInstance to the specified mesh.
diff --git a/src/obj/NiGeometry.cpp b/src/obj/NiGeometry.cpp
index ced04547..8ffc6b4d 100644
--- a/src/obj/NiGeometry.cpp
+++ b/src/obj/NiGeometry.cpp
@@ -74,7 +74,7 @@ Ref<NiSkinInstance> NiGeometry::GetSkinInstance() const {
 	return skinInstance;
 }
 
-void NiGeometry::BindSkin( vector< Ref<NiNode> > bone_nodes, bool bind_to_scene ) {
+void NiGeometry::BindSkin( vector< Ref<NiNode> > bone_nodes ) {
 	//Ensure skin is not aleady bound
 	if ( skinInstance != 0 ) {
 		throw runtime_error("You have attempted to re-bind a skin that is already bound.  Unbind it first.");
@@ -110,68 +110,55 @@ void NiGeometry::BindSkin( vector< Ref<NiNode> > bone_nodes, bool bind_to_scene
 		throw runtime_error("Shape and all skin influence bones must be part of the same tree before skin bind can take place.");
 	}
 
-	NiNodeRef skeleton_root;
-   if (bind_to_scene) {
-      // Just parent to the scene
-      NiNodeRef parent = GetParent();
-      while (parent != NULL) {
-         skeleton_root = parent;
-         parent = parent->GetParent();
-      }
-
-   } else {
-      skeleton_root = ancestors[0].front();
-
-	   //Make sure bone and shapes are part of the same tree
-	   for ( int i = 1; i < num_lists; ++i ) {
-		   if ( ancestors[i].size() == 0 ) {
-			   throw runtime_error("Shape and all skin influence bones must be part of the same tree before skin bind can take place.");
-		   }
-		   if ( ancestors[i].front() != skeleton_root ) {
-			   throw runtime_error("Shape and all skin influence bones must be part of the same tree before skin bind can take place.");
-		   }
+	NiNodeRef skeleton_root = ancestors[0].front();
+   //Make sure bone and shapes are part of the same tree
+   for ( int i = 1; i < num_lists; ++i ) {
+	   if ( ancestors[i].size() == 0 ) {
+		   throw runtime_error("Shape and all skin influence bones must be part of the same tree before skin bind can take place.");
 	   }
-
-	   //Since the first items have been shown to match, pop all the stacks
-	   for ( int i = 0; i < num_lists; ++i ) {
-		   ancestors[i].pop_front();
+	   if ( ancestors[i].front() != skeleton_root ) {
+		   throw runtime_error("Shape and all skin influence bones must be part of the same tree before skin bind can take place.");
 	   }
+   }
 
-	   //Now search for the common ancestor
+   //Since the first items have been shown to match, pop all the stacks
+   for ( int i = 0; i < num_lists; ++i ) {
+	   ancestors[i].pop_front();
+   }
 
-	   while(true) {
-		   bool all_same = true;
-		   if ( ancestors[0].size() == 0 ) {
+   //Now search for the common ancestor
+   while(true) {
+	   bool all_same = true;
+	   if ( ancestors[0].size() == 0 ) {
+		   //This list is over, so the last top is the common ancestor
+		   //break out of the loop
+		   break;
+	   }
+	   NiNodeRef first_ancestor = ancestors[0].front();
+	   for ( int i = 1; i < num_lists; ++i ) {
+		   if ( ancestors[i].size() == 0 ) {
 			   //This list is over, so the last top is the common ancestor
 			   //break out of the loop
+			   all_same = false;
 			   break;
 		   }
-		   NiNodeRef first_ancestor = ancestors[0].front();
-		   for ( int i = 1; i < num_lists; ++i ) {
-			   if ( ancestors[i].size() == 0 ) {
-				   //This list is over, so the last top is the common ancestor
-				   //break out of the loop
-				   all_same = false;
-				   break;
-			   }
-			   if ( ancestors[i].front() != first_ancestor ) {
-				   all_same = false;
-			   }
+		   if ( ancestors[i].front() != first_ancestor ) {
+			   all_same = false;
 		   }
+	   }
 
-		   if ( all_same == true ) {
-			   //They're all the same, so set the top, pop all the stacks
-			   //and look again
-   			
-			   skeleton_root = ancestors[0].front();
-			   for ( int i = 0; i < num_lists; ++i ) {
-				   ancestors[i].pop_front();
-			   }
-		   } else {
-			   //One is different, so the last top is the common ancestor.
-			   //break out of the loop
-			   break;
+	   if ( all_same == true ) {
+		   //They're all the same, so set the top, pop all the stacks
+		   //and look again
+		
+		   skeleton_root = ancestors[0].front();
+		   for ( int i = 0; i < num_lists; ++i ) {
+			   ancestors[i].pop_front();
 		   }
+	   } else {
+		   //One is different, so the last top is the common ancestor.
+		   //break out of the loop
+		   break;
 	   }
    }
 
@@ -179,6 +166,15 @@ void NiGeometry::BindSkin( vector< Ref<NiNode> > bone_nodes, bool bind_to_scene
 		throw runtime_error("Failed to find suitable skeleton root.");
 	}
 
+	//Ancestors[bone_nodes.size()] should now hold all nodes between (but not including) the
+	//skeleton root and this mesh.  Cycle through and propogate their transforms
+	for ( list<NiNodeRef>::iterator it = ancestors[bone_nodes.size()].begin(); it != ancestors[bone_nodes.size()].end(); ++it ) {
+		(*it)->PropagateTransform();
+	}
+
+	//Now apply the transforms to the vertices and normals of this mesh
+	this->ApplyTransforms();
+
 	//Create a skin instance using the bone and root data
 	skinInstance = new NiSkinInstance( skeleton_root, bone_nodes );
 
@@ -287,6 +283,20 @@ void NiGeometry::GetSkinDeformation( vector<Vector3> & vertices, vector<Vector3>
 	}
 }
 
+void NiGeometry::ApplyTransforms() {
+	//Get Data
+	NiGeometryDataRef geom_data = this->GetData();
+	if ( geom_data == NULL ) {
+		throw runtime_error( "Called ApplyTransform on a NiGeometry object that has no NiGeometryData attached.");
+	}
+
+	//Transform the vertices by the local transform of this mesh
+	geom_data->Transform( this->GetLocalTransform() );
+
+	//Now that the transforms have been applied, clear them to the identity
+	this->SetLocalTransform( Matrix44::IDENTITY );
+}
+
 // Calculate bounding sphere using minimum-volume axis-align bounding box.  Its fast but not a very good fit.
 static void CalcAxisAlignedBox(const vector<SkinWeight> & n, const vector<Vector3>& vertices, Vector3& center, float& radius)
 {
diff --git a/src/obj/NiGeometryData.cpp b/src/obj/NiGeometryData.cpp
index 82c58494..b2992d00 100644
--- a/src/obj/NiGeometryData.cpp
+++ b/src/obj/NiGeometryData.cpp
@@ -119,17 +119,22 @@ void NiGeometryData::SetUVSet( int index, const vector<TexCoord> & in ) {
 	uvSets[index] = in;
 }
 
-/*! Returns the 3D center of the mesh.
- * \return The center of this mesh.
- */
 Vector3 NiGeometryData::GetCenter() const {
 	return center;
 }
 
-/*! Returns the radius of the mesh.  That is the distance from the center to
- * the farthest point from the center.
- * \return The radius of this mesh.
- */
 float NiGeometryData::GetRadius() const {
 	return radius;
 }
+
+void NiGeometryData::Transform( Matrix44 & transform ) {
+	Matrix44 rotation = Matrix44( transform.GetRotation() );
+
+	//Apply the transformations
+	for ( uint i = 0; i < vertices.size(); ++i ) {
+		vertices[i] = transform * vertices[i];
+	}
+	for ( uint i = 0; i < normals.size(); ++i ) {
+		normals[i] = rotation * normals[i];
+	}
+}
\ No newline at end of file
diff --git a/src/obj/NiNode.cpp b/src/obj/NiNode.cpp
index 82768908..a1c92d8e 100644
--- a/src/obj/NiNode.cpp
+++ b/src/obj/NiNode.cpp
@@ -213,6 +213,21 @@ void NiNode::GoToSkeletonBindPosition() {
 	}
 }
 
+void NiNode::PropagateTransform() {
+
+	Matrix44 par_trans = this->GetLocalTransform();
+
+	//Loop through each child and apply this node's transform to it
+	for ( unsigned i = 0; i < children.size(); ++i ) {
+		children[i]->SetLocalTransform(
+			children[i]->GetLocalTransform() * par_trans
+		);
+	}
+
+	//Nowthat the transforms have been propogated, clear them out
+	this->SetLocalTransform( Matrix44::IDENTITY );
+}
+
 bool NiNode::IsSplitMeshProxy() const {
 	//Let us guess that a node is a split mesh proxy if:
 	// 1)  All its children are NiTriBasedGeom derived objects.
diff --git a/src/obj/NiTriShapeData.cpp b/src/obj/NiTriShapeData.cpp
index b67fad8b..b8464bc5 100644
--- a/src/obj/NiTriShapeData.cpp
+++ b/src/obj/NiTriShapeData.cpp
@@ -78,7 +78,10 @@ void NiTriShapeData::SetTriangles( const vector<Triangle> & in ) {
 
 	hasTriangles = ( triangles.size() != 0 );
 
+	//Set nuber of triangles
+	numTriangles = uint(triangles.size());
+
 	//Set number of trianble points to the number of triangles times 3
-	numTrianglePoints = uint(triangles.size()) * 3;
+	numTrianglePoints = numTriangles * 3;
 }
 
-- 
GitLab