2015-04-21 14:02:40 +00:00
|
|
|
#include "skeleton.hpp"
|
|
|
|
|
|
|
|
#include <osg/Transform>
|
|
|
|
#include <osg/MatrixTransform>
|
|
|
|
|
2015-12-05 14:02:35 +00:00
|
|
|
#include <components/misc/stringops.hpp>
|
|
|
|
|
2015-04-21 14:02:40 +00:00
|
|
|
#include <iostream>
|
|
|
|
|
2015-04-21 18:30:48 +00:00
|
|
|
namespace SceneUtil
|
2015-04-21 14:02:40 +00:00
|
|
|
{
|
|
|
|
|
|
|
|
class InitBoneCacheVisitor : public osg::NodeVisitor
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
InitBoneCacheVisitor(std::map<std::string, std::pair<osg::NodePath, osg::MatrixTransform*> >& cache)
|
|
|
|
: osg::NodeVisitor(TRAVERSE_ALL_CHILDREN)
|
|
|
|
, mCache(cache)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void apply(osg::Transform &node)
|
|
|
|
{
|
|
|
|
osg::MatrixTransform* bone = node.asMatrixTransform();
|
|
|
|
if (!bone)
|
|
|
|
return;
|
|
|
|
|
2015-12-05 14:02:35 +00:00
|
|
|
mCache[Misc::StringUtils::lowerCase(bone->getName())] = std::make_pair(getNodePath(), bone);
|
2015-04-21 14:02:40 +00:00
|
|
|
|
|
|
|
traverse(node);
|
|
|
|
}
|
|
|
|
private:
|
|
|
|
std::map<std::string, std::pair<osg::NodePath, osg::MatrixTransform*> >& mCache;
|
|
|
|
};
|
|
|
|
|
|
|
|
Skeleton::Skeleton()
|
|
|
|
: mBoneCacheInit(false)
|
|
|
|
, mNeedToUpdateBoneMatrices(true)
|
2015-04-29 21:48:08 +00:00
|
|
|
, mActive(true)
|
2015-05-23 20:44:00 +00:00
|
|
|
, mLastFrameNumber(0)
|
2016-02-29 17:20:56 +00:00
|
|
|
, mTraversedEvenFrame(false)
|
|
|
|
, mTraversedOddFrame(false)
|
2015-04-21 14:02:40 +00:00
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
Skeleton::Skeleton(const Skeleton ©, const osg::CopyOp ©op)
|
2015-04-21 16:52:13 +00:00
|
|
|
: osg::Group(copy, copyop)
|
|
|
|
, mBoneCacheInit(false)
|
2015-04-21 14:02:40 +00:00
|
|
|
, mNeedToUpdateBoneMatrices(true)
|
2015-04-29 21:48:08 +00:00
|
|
|
, mActive(copy.mActive)
|
2015-05-23 20:44:00 +00:00
|
|
|
, mLastFrameNumber(0)
|
2016-02-29 17:20:56 +00:00
|
|
|
, mTraversedEvenFrame(false)
|
|
|
|
, mTraversedOddFrame(false)
|
2015-04-21 14:02:40 +00:00
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
Bone* Skeleton::getBone(const std::string &name)
|
|
|
|
{
|
|
|
|
if (!mBoneCacheInit)
|
|
|
|
{
|
|
|
|
InitBoneCacheVisitor visitor(mBoneCache);
|
|
|
|
accept(visitor);
|
|
|
|
mBoneCacheInit = true;
|
|
|
|
}
|
|
|
|
|
2015-12-05 14:02:35 +00:00
|
|
|
BoneCache::iterator found = mBoneCache.find(Misc::StringUtils::lowerCase(name));
|
2015-04-21 14:02:40 +00:00
|
|
|
if (found == mBoneCache.end())
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
// find or insert in the bone hierarchy
|
|
|
|
|
|
|
|
if (!mRootBone.get())
|
|
|
|
{
|
|
|
|
mRootBone.reset(new Bone);
|
|
|
|
}
|
|
|
|
|
|
|
|
const osg::NodePath& path = found->second.first;
|
|
|
|
Bone* bone = mRootBone.get();
|
|
|
|
for (osg::NodePath::const_iterator it = path.begin(); it != path.end(); ++it)
|
|
|
|
{
|
|
|
|
osg::MatrixTransform* matrixTransform = dynamic_cast<osg::MatrixTransform*>(*it);
|
|
|
|
if (!matrixTransform)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
Bone* child = NULL;
|
|
|
|
for (unsigned int i=0; i<bone->mChildren.size(); ++i)
|
|
|
|
{
|
|
|
|
if (bone->mChildren[i]->mNode == *it)
|
|
|
|
{
|
|
|
|
child = bone->mChildren[i];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!child)
|
|
|
|
{
|
|
|
|
child = new Bone;
|
|
|
|
bone->mChildren.push_back(child);
|
|
|
|
mNeedToUpdateBoneMatrices = true;
|
|
|
|
}
|
|
|
|
bone = child;
|
|
|
|
|
|
|
|
bone->mNode = matrixTransform;
|
|
|
|
}
|
|
|
|
|
|
|
|
return bone;
|
|
|
|
}
|
|
|
|
|
2016-07-02 17:26:11 +00:00
|
|
|
void Skeleton::updateBoneMatrices(unsigned int traversalNumber)
|
2015-04-21 14:02:40 +00:00
|
|
|
{
|
2016-07-02 17:26:11 +00:00
|
|
|
if (traversalNumber != mLastFrameNumber)
|
2015-04-21 18:22:32 +00:00
|
|
|
mNeedToUpdateBoneMatrices = true;
|
|
|
|
|
2016-07-02 17:26:11 +00:00
|
|
|
mLastFrameNumber = traversalNumber;
|
2015-04-21 14:02:40 +00:00
|
|
|
|
2016-02-29 17:20:56 +00:00
|
|
|
if (mLastFrameNumber % 2 == 0)
|
|
|
|
mTraversedEvenFrame = true;
|
|
|
|
else
|
|
|
|
mTraversedOddFrame = true;
|
|
|
|
|
2015-04-21 18:22:32 +00:00
|
|
|
if (mNeedToUpdateBoneMatrices)
|
|
|
|
{
|
2015-04-21 14:02:40 +00:00
|
|
|
if (mRootBone.get())
|
|
|
|
{
|
|
|
|
for (unsigned int i=0; i<mRootBone->mChildren.size(); ++i)
|
|
|
|
mRootBone->mChildren[i]->update(NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
mNeedToUpdateBoneMatrices = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-04-29 21:48:08 +00:00
|
|
|
void Skeleton::setActive(bool active)
|
|
|
|
{
|
|
|
|
mActive = active;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Skeleton::getActive() const
|
|
|
|
{
|
|
|
|
return mActive;
|
|
|
|
}
|
|
|
|
|
2016-07-02 18:19:55 +00:00
|
|
|
void Skeleton::markDirty()
|
|
|
|
{
|
|
|
|
mTraversedEvenFrame = false;
|
|
|
|
mTraversedOddFrame = false;
|
2016-12-16 19:09:20 +00:00
|
|
|
mBoneCache.clear();
|
|
|
|
mBoneCacheInit = false;
|
2016-07-02 18:19:55 +00:00
|
|
|
}
|
|
|
|
|
2015-06-30 01:25:30 +00:00
|
|
|
void Skeleton::traverse(osg::NodeVisitor& nv)
|
|
|
|
{
|
2015-12-03 18:49:45 +00:00
|
|
|
if (!getActive() && nv.getVisitorType() == osg::NodeVisitor::UPDATE_VISITOR
|
|
|
|
// need to process at least 2 frames before shutting off update, since we need to have both frame-alternating RigGeometries initialized
|
|
|
|
// this would be more naturally handled if the double-buffering was implemented in RigGeometry itself rather than in a FrameSwitch decorator node
|
2016-02-29 17:20:56 +00:00
|
|
|
&& mLastFrameNumber != 0 && mTraversedEvenFrame && mTraversedOddFrame)
|
2015-06-30 01:25:30 +00:00
|
|
|
return;
|
|
|
|
osg::Group::traverse(nv);
|
|
|
|
}
|
|
|
|
|
2016-12-16 19:09:20 +00:00
|
|
|
void Skeleton::childInserted(unsigned int)
|
|
|
|
{
|
|
|
|
markDirty();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Skeleton::childRemoved(unsigned int, unsigned int)
|
|
|
|
{
|
|
|
|
markDirty();
|
|
|
|
}
|
|
|
|
|
2015-04-21 14:02:40 +00:00
|
|
|
Bone::Bone()
|
|
|
|
: mNode(NULL)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
Bone::~Bone()
|
|
|
|
{
|
|
|
|
for (unsigned int i=0; i<mChildren.size(); ++i)
|
|
|
|
delete mChildren[i];
|
|
|
|
mChildren.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Bone::update(const osg::Matrixf* parentMatrixInSkeletonSpace)
|
|
|
|
{
|
|
|
|
if (!mNode)
|
|
|
|
{
|
2017-03-02 21:07:43 +00:00
|
|
|
std::cerr << "Error: Bone without node " << std::endl;
|
2015-12-08 15:48:01 +00:00
|
|
|
return;
|
2015-04-21 14:02:40 +00:00
|
|
|
}
|
|
|
|
if (parentMatrixInSkeletonSpace)
|
|
|
|
mMatrixInSkeletonSpace = mNode->getMatrix() * (*parentMatrixInSkeletonSpace);
|
|
|
|
else
|
|
|
|
mMatrixInSkeletonSpace = mNode->getMatrix();
|
|
|
|
|
|
|
|
for (unsigned int i=0; i<mChildren.size(); ++i)
|
|
|
|
{
|
|
|
|
mChildren[i]->update(&mMatrixInSkeletonSpace);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|