Use the new skinning system in OpenMW

c++11
scrawl 10 years ago
parent 111e3eb6db
commit bd88758962

@ -39,7 +39,8 @@ Skeleton::Skeleton()
} }
Skeleton::Skeleton(const Skeleton &copy, const osg::CopyOp &copyop) Skeleton::Skeleton(const Skeleton &copy, const osg::CopyOp &copyop)
: mBoneCacheInit(false) : osg::Group(copy, copyop)
, mBoneCacheInit(false)
, mNeedToUpdateBoneMatrices(true) , mNeedToUpdateBoneMatrices(true)
{ {

@ -8,155 +8,55 @@
#include <osg/Geode> #include <osg/Geode>
#include <osg/FrontFace> #include <osg/FrontFace>
#include <osg/PositionAttitudeTransform> #include <osg/PositionAttitudeTransform>
#include <osg/MatrixTransform>
#include <osgAnimation/BoneMapVisitor>
#include <osgAnimation/Skeleton>
#include <components/misc/stringops.hpp> #include <components/misc/stringops.hpp>
#include <components/nifosg/skeleton.hpp>
#include "visitor.hpp" #include "visitor.hpp"
namespace SceneUtil namespace SceneUtil
{ {
class NodeMapVisitor : public osg::NodeVisitor class CopyRigVisitor : public osg::NodeVisitor
{
public:
NodeMapVisitor() : osg::NodeVisitor(osg::NodeVisitor::TRAVERSE_ALL_CHILDREN) {}
void apply(osg::MatrixTransform& trans)
{
mMap[trans.getName()] = &trans;
traverse(trans);
}
typedef std::map<std::string, osg::ref_ptr<osg::MatrixTransform> > NodeMap;
const NodeMap& getNodeMap() const
{
return mMap;
}
private:
NodeMap mMap;
};
/// Copy the matrix of a "source" node to a "dest" node (the node that the callback is attached to).
/// Must be set on a MatrixTransform.
class CopyController : public osg::NodeCallback
{
public:
CopyController(osg::MatrixTransform* copyFrom)
: mCopyFrom(copyFrom)
{
}
CopyController(const CopyController& copy, const osg::CopyOp& copyop)
: osg::NodeCallback(copy, copyop)
, mCopyFrom(copy.mCopyFrom)
{
}
CopyController()
: mCopyFrom(NULL)
{
}
virtual void operator()(osg::Node* node, osg::NodeVisitor* nv)
{
osgAnimation::Bone* bone = static_cast<osgAnimation::Bone*>(node);
if (mCopyFrom)
{
bone->setMatrix(mCopyFrom->getMatrix());
}
traverse(node, nv);
}
private:
const osg::MatrixTransform* mCopyFrom;
};
class AddCopyControllerVisitor : public osg::NodeVisitor
{ {
public: public:
AddCopyControllerVisitor(const NodeMapVisitor::NodeMap& boneMap) CopyRigVisitor(osg::ref_ptr<osg::Group> parent, const std::string& filter)
: osg::NodeVisitor(TRAVERSE_ALL_CHILDREN)
, mNodeMap(boneMap)
{
}
virtual void apply(osg::MatrixTransform &node)
{
if (osgAnimation::Bone* bone = dynamic_cast<osgAnimation::Bone*>(&node))
{
NodeMapVisitor::NodeMap::const_iterator found = mNodeMap.find(bone->getName());
if (found != mNodeMap.end())
{
// add the CopyController at position 0 so it's executed before UpdateBone
osg::ref_ptr<osg::NodeCallback> old = bone->getUpdateCallback();
bone->setUpdateCallback(new CopyController(found->second.get()));
bone->addUpdateCallback(old);
}
}
traverse(node);
}
private:
const NodeMapVisitor::NodeMap& mNodeMap;
};
// FIXME: would be more efficient to copy only the wanted nodes instead of deleting unwanted ones later
// copying is kinda cheap though, so don't bother for now
class FilterVisitor : public osg::NodeVisitor
{
public:
FilterVisitor(const std::string& filter)
: osg::NodeVisitor(TRAVERSE_ALL_CHILDREN) : osg::NodeVisitor(TRAVERSE_ALL_CHILDREN)
, mParent(parent)
, mFilter(Misc::StringUtils::lowerCase(filter)) , mFilter(Misc::StringUtils::lowerCase(filter))
{ {
} }
virtual void apply(osg::Geode &node) virtual void apply(osg::Node& node)
{ {
std::string lowerName = Misc::StringUtils::lowerCase(node.getName()); std::string lowerName = Misc::StringUtils::lowerCase(node.getName());
if (lowerName.find(mFilter) == std::string::npos) if (lowerName.find(mFilter) != std::string::npos)
{ {
mToRemove.push_back(&node); mParent->addChild(&node);
}
}
void removeFilteredParts()
{
for (std::vector<osg::Geode*>::iterator it = mToRemove.begin(); it != mToRemove.end(); ++it)
{
osg::Geode* geode = *it;
geode->getParent(0)->removeChild(geode);
} }
else
traverse(node);
} }
private: private:
std::vector<osg::Geode*> mToRemove; osg::ref_ptr<osg::Group> mParent;
std::string mFilter; std::string mFilter;
}; };
osg::ref_ptr<osg::Node> attach(osg::ref_ptr<osg::Node> toAttach, osg::Node *master, const std::string &filter, const std::string &attachNode) osg::ref_ptr<osg::Node> attach(osg::ref_ptr<osg::Node> toAttach, osg::Node *master, const std::string &filter, const std::string &attachNode)
{ {
if (osgAnimation::Skeleton* skel = dynamic_cast<osgAnimation::Skeleton*>(toAttach.get())) if (dynamic_cast<NifOsg::Skeleton*>(toAttach.get()))
{ {
NodeMapVisitor nodeMapVisitor; osg::ref_ptr<osg::Group> handle = new osg::Group;
master->accept(nodeMapVisitor);
// would be more efficient if we could attach the RigGeometry directly to the master skeleton, but currently not possible CopyRigVisitor copyVisitor(handle, filter);
// due to a difference in binding pose of the two skeletons toAttach->accept(copyVisitor);
AddCopyControllerVisitor visitor(nodeMapVisitor.getNodeMap());
toAttach->accept(visitor);
FilterVisitor filterVisitor(filter); master->asGroup()->addChild(handle);
toAttach->accept(filterVisitor);
filterVisitor.removeFilteredParts();
master->asGroup()->addChild(skel); return handle;
return skel;
} }
else else
{ {

@ -17,15 +17,14 @@ namespace SceneUtil
{ {
} }
virtual void apply(osg::Node &node) virtual void apply(osg::Group& group)
{ {
osg::Group* group = node.asGroup(); if (group.getName() == mNameToFind)
if (group && node.getName() == mNameToFind)
{ {
mFoundNode = group; mFoundNode = &group;
return; return;
} }
traverse(node); traverse(group);
} }
std::string mNameToFind; std::string mNameToFind;

Loading…
Cancel
Save