Skip to content
Snippets Groups Projects
Commit 2a5ab374 authored by Shon Ferguson's avatar Shon Ferguson
Browse files

Added NiGeometry::ApplyTransforms function to automatically zero out the...

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.
parent 9447d624
No related branches found
No related tags found
No related merge requests found
......@@ -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
......
......@@ -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:
......
......@@ -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.
......
......@@ -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)
{
......
......@@ -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
......@@ -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.
......
......@@ -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;
}
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment