2011-12-12 04:42:39 +00:00
|
|
|
#include "animation.hpp"
|
|
|
|
|
2015-04-22 17:08:56 +00:00
|
|
|
#include <iomanip>
|
2015-04-23 18:41:31 +00:00
|
|
|
#include <limits>
|
2015-04-22 17:08:56 +00:00
|
|
|
|
2015-04-16 23:23:37 +00:00
|
|
|
#include <osg/PositionAttitudeTransform>
|
|
|
|
#include <osg/TexGen>
|
|
|
|
#include <osg/TexEnvCombine>
|
2015-04-19 13:03:08 +00:00
|
|
|
#include <osg/ComputeBoundsVisitor>
|
2015-04-23 18:41:31 +00:00
|
|
|
#include <osg/MatrixTransform>
|
2015-05-21 21:24:22 +00:00
|
|
|
#include <osg/Geode>
|
2015-04-16 23:23:37 +00:00
|
|
|
|
2015-04-12 13:34:50 +00:00
|
|
|
#include <components/nifosg/nifloader.hpp>
|
2013-08-06 07:04:39 +00:00
|
|
|
|
2015-04-12 13:34:50 +00:00
|
|
|
#include <components/resource/resourcesystem.hpp>
|
|
|
|
#include <components/resource/scenemanager.hpp>
|
2015-04-16 23:23:37 +00:00
|
|
|
#include <components/resource/texturemanager.hpp>
|
2013-08-07 02:45:07 +00:00
|
|
|
|
2015-04-23 18:41:31 +00:00
|
|
|
#include <components/nifosg/nifloader.hpp> // KeyframeHolder
|
2015-06-14 21:13:26 +00:00
|
|
|
#include <components/nifosg/controller.hpp>
|
2015-04-23 18:41:31 +00:00
|
|
|
|
|
|
|
#include <components/vfs/manager.hpp>
|
|
|
|
|
2015-04-16 23:23:37 +00:00
|
|
|
#include <components/sceneutil/statesetupdater.hpp>
|
2015-04-18 23:57:52 +00:00
|
|
|
#include <components/sceneutil/visitor.hpp>
|
2015-04-19 13:03:08 +00:00
|
|
|
#include <components/sceneutil/lightmanager.hpp>
|
|
|
|
#include <components/sceneutil/util.hpp>
|
2015-04-22 15:34:39 +00:00
|
|
|
#include <components/sceneutil/lightcontroller.hpp>
|
2015-04-23 21:30:06 +00:00
|
|
|
#include <components/sceneutil/skeleton.hpp>
|
2015-04-16 23:23:37 +00:00
|
|
|
|
|
|
|
#include "../mwbase/environment.hpp"
|
|
|
|
#include "../mwbase/world.hpp"
|
|
|
|
#include "../mwworld/esmstore.hpp"
|
|
|
|
#include "../mwworld/class.hpp"
|
2015-06-02 15:02:56 +00:00
|
|
|
#include "../mwworld/fallback.hpp"
|
|
|
|
#include "../mwworld/cellstore.hpp"
|
2015-04-16 23:23:37 +00:00
|
|
|
|
2015-05-12 15:40:42 +00:00
|
|
|
#include "../mwmechanics/character.hpp" // FIXME: for MWMechanics::Priority
|
|
|
|
|
2015-04-18 23:57:52 +00:00
|
|
|
#include "vismask.hpp"
|
2015-04-19 15:55:56 +00:00
|
|
|
#include "util.hpp"
|
2015-04-18 23:57:52 +00:00
|
|
|
|
2015-04-16 23:23:37 +00:00
|
|
|
namespace
|
|
|
|
{
|
|
|
|
|
|
|
|
class GlowUpdater : public SceneUtil::StateSetUpdater
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
GlowUpdater(osg::Vec4f color, const std::vector<osg::ref_ptr<osg::Texture2D> >& textures)
|
|
|
|
: mTexUnit(1) // FIXME: might not always be 1
|
|
|
|
, mColor(color)
|
|
|
|
, mTextures(textures)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void setDefaults(osg::StateSet *stateset)
|
|
|
|
{
|
|
|
|
stateset->setTextureMode(mTexUnit, GL_TEXTURE_2D, osg::StateAttribute::ON);
|
|
|
|
|
|
|
|
osg::TexGen* texGen = new osg::TexGen;
|
|
|
|
texGen->setMode(osg::TexGen::SPHERE_MAP);
|
|
|
|
|
|
|
|
stateset->setTextureAttributeAndModes(mTexUnit, texGen, osg::StateAttribute::ON|osg::StateAttribute::OVERRIDE);
|
|
|
|
|
|
|
|
osg::TexEnvCombine* texEnv = new osg::TexEnvCombine;
|
|
|
|
texEnv->setSource0_RGB(osg::TexEnvCombine::CONSTANT);
|
|
|
|
texEnv->setConstantColor(mColor);
|
|
|
|
texEnv->setCombine_RGB(osg::TexEnvCombine::INTERPOLATE);
|
|
|
|
texEnv->setSource2_RGB(osg::TexEnvCombine::TEXTURE);
|
|
|
|
texEnv->setOperand2_RGB(osg::TexEnvCombine::SRC_COLOR);
|
|
|
|
|
|
|
|
stateset->setTextureAttributeAndModes(mTexUnit, texEnv, osg::StateAttribute::ON);
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void apply(osg::StateSet *stateset, osg::NodeVisitor *nv)
|
|
|
|
{
|
|
|
|
float time = nv->getFrameStamp()->getSimulationTime();
|
|
|
|
int index = (int)(time*16) % mTextures.size();
|
|
|
|
stateset->setTextureAttribute(mTexUnit, mTextures[index], osg::StateAttribute::ON|osg::StateAttribute::OVERRIDE);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
int mTexUnit;
|
|
|
|
osg::Vec4f mColor;
|
|
|
|
std::vector<osg::ref_ptr<osg::Texture2D> > mTextures;
|
|
|
|
};
|
|
|
|
|
2015-04-23 18:41:31 +00:00
|
|
|
class NodeMapVisitor : public osg::NodeVisitor
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
NodeMapVisitor() : osg::NodeVisitor(osg::NodeVisitor::TRAVERSE_ALL_CHILDREN) {}
|
|
|
|
|
|
|
|
void apply(osg::MatrixTransform& trans)
|
|
|
|
{
|
|
|
|
mMap[Misc::StringUtils::lowerCase(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;
|
|
|
|
};
|
|
|
|
|
|
|
|
NifOsg::TextKeyMap::const_iterator findGroupStart(const NifOsg::TextKeyMap &keys, const std::string &groupname)
|
|
|
|
{
|
|
|
|
NifOsg::TextKeyMap::const_iterator iter(keys.begin());
|
|
|
|
for(;iter != keys.end();++iter)
|
|
|
|
{
|
|
|
|
if(iter->second.compare(0, groupname.size(), groupname) == 0 &&
|
|
|
|
iter->second.compare(groupname.size(), 2, ": ") == 0)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return iter;
|
|
|
|
}
|
|
|
|
|
|
|
|
float calcAnimVelocity(const std::multimap<float, std::string>& keys,
|
|
|
|
NifOsg::KeyframeController *nonaccumctrl, const osg::Vec3f& accum, const std::string &groupname)
|
|
|
|
{
|
|
|
|
const std::string start = groupname+": start";
|
|
|
|
const std::string loopstart = groupname+": loop start";
|
|
|
|
const std::string loopstop = groupname+": loop stop";
|
|
|
|
const std::string stop = groupname+": stop";
|
|
|
|
float starttime = std::numeric_limits<float>::max();
|
|
|
|
float stoptime = 0.0f;
|
|
|
|
|
|
|
|
// Pick the last Loop Stop key and the last Loop Start key.
|
|
|
|
// This is required because of broken text keys in AshVampire.nif.
|
|
|
|
// It has *two* WalkForward: Loop Stop keys at different times, the first one is used for stopping playback
|
|
|
|
// but the animation velocity calculation uses the second one.
|
|
|
|
// As result the animation velocity calculation is not correct, and this incorrect velocity must be replicated,
|
|
|
|
// because otherwise the Creature's Speed (dagoth uthol) would not be sufficient to move fast enough.
|
|
|
|
NifOsg::TextKeyMap::const_reverse_iterator keyiter(keys.rbegin());
|
|
|
|
while(keyiter != keys.rend())
|
|
|
|
{
|
|
|
|
if(keyiter->second == start || keyiter->second == loopstart)
|
|
|
|
{
|
|
|
|
starttime = keyiter->first;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
++keyiter;
|
|
|
|
}
|
|
|
|
keyiter = keys.rbegin();
|
|
|
|
while(keyiter != keys.rend())
|
|
|
|
{
|
|
|
|
if (keyiter->second == stop)
|
|
|
|
stoptime = keyiter->first;
|
|
|
|
else if (keyiter->second == loopstop)
|
|
|
|
{
|
|
|
|
stoptime = keyiter->first;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
++keyiter;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(stoptime > starttime)
|
|
|
|
{
|
|
|
|
osg::Vec3f startpos = osg::componentMultiply(nonaccumctrl->getTranslation(starttime), accum);
|
|
|
|
osg::Vec3f endpos = osg::componentMultiply(nonaccumctrl->getTranslation(stoptime), accum);
|
|
|
|
|
|
|
|
return (startpos-endpos).length() / (stoptime - starttime);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0.0f;
|
|
|
|
}
|
2015-05-21 21:24:22 +00:00
|
|
|
|
|
|
|
|
|
|
|
// Removes all drawables from a graph.
|
|
|
|
class RemoveDrawableVisitor : public osg::NodeVisitor
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
RemoveDrawableVisitor()
|
|
|
|
: osg::NodeVisitor(TRAVERSE_ALL_CHILDREN)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void apply(osg::Geode &node)
|
|
|
|
{
|
|
|
|
// Not safe to remove in apply(), since the visitor is still iterating the child list
|
2015-06-14 21:56:35 +00:00
|
|
|
osg::Group* parent = node.getParent(0);
|
|
|
|
// prune nodes that would be empty after the removal
|
|
|
|
if (parent->getNumChildren() == 1 && parent->getDataVariance() == osg::Object::STATIC)
|
|
|
|
mToRemove.push_back(parent);
|
|
|
|
else
|
|
|
|
mToRemove.push_back(&node);
|
2015-05-21 21:24:22 +00:00
|
|
|
traverse(node);
|
|
|
|
}
|
|
|
|
|
|
|
|
void remove()
|
|
|
|
{
|
|
|
|
for (std::vector<osg::Node*>::iterator it = mToRemove.begin(); it != mToRemove.end(); ++it)
|
|
|
|
{
|
|
|
|
osg::Node* node = *it;
|
|
|
|
if (node->getNumParents())
|
|
|
|
node->getParent(0)->removeChild(node);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
std::vector<osg::Node*> mToRemove;
|
|
|
|
};
|
|
|
|
|
2015-04-16 23:23:37 +00:00
|
|
|
}
|
2013-11-19 23:07:26 +00:00
|
|
|
|
2012-07-13 10:51:58 +00:00
|
|
|
namespace MWRender
|
|
|
|
{
|
2012-07-13 03:12:18 +00:00
|
|
|
|
2015-06-14 21:13:26 +00:00
|
|
|
struct Animation::AnimSource
|
|
|
|
{
|
|
|
|
osg::ref_ptr<const NifOsg::KeyframeHolder> mKeyframes;
|
|
|
|
|
|
|
|
typedef std::map<std::string, osg::ref_ptr<NifOsg::KeyframeController> > ControllerMap;
|
|
|
|
|
|
|
|
ControllerMap mControllerMap[Animation::sNumGroups];
|
|
|
|
|
|
|
|
const std::multimap<float, std::string>& getTextKeys();
|
|
|
|
};
|
|
|
|
|
2015-04-23 20:46:07 +00:00
|
|
|
class ResetAccumRootCallback : public osg::NodeCallback
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
virtual void operator()(osg::Node* node, osg::NodeVisitor* nv)
|
|
|
|
{
|
|
|
|
osg::MatrixTransform* transform = static_cast<osg::MatrixTransform*>(node);
|
|
|
|
|
|
|
|
osg::Matrix mat = transform->getMatrix();
|
|
|
|
osg::Vec3f position = mat.getTrans();
|
|
|
|
position = osg::componentMultiply(mResetAxes, position);
|
|
|
|
mat.setTrans(position);
|
|
|
|
transform->setMatrix(mat);
|
|
|
|
|
|
|
|
traverse(node, nv);
|
|
|
|
}
|
|
|
|
|
|
|
|
void setAccumulate(const osg::Vec3f& accumulate)
|
|
|
|
{
|
|
|
|
// anything that accumulates (1.f) should be reset in the callback to (0.f)
|
|
|
|
mResetAxes.x() = accumulate.x() != 0.f ? 0.f : 1.f;
|
|
|
|
mResetAxes.y() = accumulate.y() != 0.f ? 0.f : 1.f;
|
|
|
|
mResetAxes.z() = accumulate.z() != 0.f ? 0.f : 1.f;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
osg::Vec3f mResetAxes;
|
|
|
|
};
|
|
|
|
|
2015-04-15 20:11:38 +00:00
|
|
|
Animation::Animation(const MWWorld::Ptr &ptr, osg::ref_ptr<osg::Group> parentNode, Resource::ResourceSystem* resourceSystem)
|
2015-05-23 20:44:00 +00:00
|
|
|
: mInsert(parentNode)
|
|
|
|
, mPtr(ptr)
|
2015-04-12 13:34:50 +00:00
|
|
|
, mResourceSystem(resourceSystem)
|
2015-04-23 20:46:07 +00:00
|
|
|
, mAccumulate(1.f, 1.f, 0.f)
|
2015-05-21 22:55:43 +00:00
|
|
|
, mTextKeyListener(NULL)
|
2013-05-07 23:59:32 +00:00
|
|
|
{
|
2015-04-23 18:41:31 +00:00
|
|
|
for(size_t i = 0;i < sNumGroups;i++)
|
2015-06-04 18:33:20 +00:00
|
|
|
mAnimationTimePtr[i].reset(new AnimationTime);
|
2014-05-26 18:37:12 +00:00
|
|
|
}
|
2013-05-12 12:08:01 +00:00
|
|
|
|
2015-04-12 13:34:50 +00:00
|
|
|
Animation::~Animation()
|
2013-05-11 01:37:44 +00:00
|
|
|
{
|
2015-04-12 13:34:50 +00:00
|
|
|
if (mObjectRoot)
|
|
|
|
mInsert->removeChild(mObjectRoot);
|
2013-05-11 01:37:44 +00:00
|
|
|
}
|
|
|
|
|
2015-05-21 21:54:39 +00:00
|
|
|
MWWorld::Ptr Animation::getPtr()
|
|
|
|
{
|
|
|
|
return mPtr;
|
|
|
|
}
|
|
|
|
|
2015-04-29 21:48:08 +00:00
|
|
|
void Animation::setActive(bool active)
|
|
|
|
{
|
|
|
|
if (SceneUtil::Skeleton* skel = dynamic_cast<SceneUtil::Skeleton*>(mObjectRoot.get()))
|
|
|
|
{
|
|
|
|
skel->setActive(active);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-04-22 17:08:56 +00:00
|
|
|
void Animation::updatePtr(const MWWorld::Ptr &ptr)
|
|
|
|
{
|
|
|
|
mPtr = ptr;
|
|
|
|
}
|
|
|
|
|
2015-04-23 18:41:31 +00:00
|
|
|
void Animation::setAccumulation(const osg::Vec3f& accum)
|
|
|
|
{
|
|
|
|
mAccumulate = accum;
|
2015-04-23 20:46:07 +00:00
|
|
|
|
|
|
|
if (mResetAccumRootCallback)
|
|
|
|
mResetAccumRootCallback->setAccumulate(mAccumulate);
|
2015-04-23 18:41:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
size_t Animation::detectAnimGroup(osg::Node* node)
|
|
|
|
{
|
|
|
|
static const char sGroupRoots[sNumGroups][32] = {
|
|
|
|
"", /* Lower body / character root */
|
|
|
|
"Bip01 Spine1", /* Torso */
|
|
|
|
"Bip01 L Clavicle", /* Left arm */
|
|
|
|
"Bip01 R Clavicle", /* Right arm */
|
|
|
|
};
|
|
|
|
|
|
|
|
while(node != mObjectRoot)
|
|
|
|
{
|
|
|
|
const std::string &name = node->getName();
|
|
|
|
for(size_t i = 1;i < sNumGroups;i++)
|
|
|
|
{
|
|
|
|
if(name == sGroupRoots[i])
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
|
|
|
assert(node->getNumParents() > 0);
|
|
|
|
|
|
|
|
node = node->getParent(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
const std::multimap<float, std::string> &Animation::AnimSource::getTextKeys()
|
|
|
|
{
|
|
|
|
return mKeyframes->mTextKeys;
|
|
|
|
}
|
|
|
|
|
2015-04-22 17:08:56 +00:00
|
|
|
void Animation::addAnimSource(const std::string &model)
|
|
|
|
{
|
2015-04-23 18:41:31 +00:00
|
|
|
std::string kfname = model;
|
|
|
|
Misc::StringUtils::toLower(kfname);
|
|
|
|
|
|
|
|
if(kfname.size() > 4 && kfname.compare(kfname.size()-4, 4, ".nif") == 0)
|
|
|
|
kfname.replace(kfname.size()-4, 4, ".kf");
|
|
|
|
|
|
|
|
if(!mResourceSystem->getVFS()->exists(kfname))
|
|
|
|
return;
|
|
|
|
|
|
|
|
boost::shared_ptr<AnimSource> animsrc;
|
|
|
|
animsrc.reset(new AnimSource);
|
|
|
|
animsrc->mKeyframes = mResourceSystem->getSceneManager()->getKeyframes(kfname);
|
|
|
|
|
|
|
|
if (animsrc->mKeyframes->mTextKeys.empty() || animsrc->mKeyframes->mKeyframeControllers.empty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
for (NifOsg::KeyframeHolder::KeyframeControllerMap::const_iterator it = animsrc->mKeyframes->mKeyframeControllers.begin();
|
|
|
|
it != animsrc->mKeyframes->mKeyframeControllers.end(); ++it)
|
|
|
|
{
|
|
|
|
std::string bonename = Misc::StringUtils::lowerCase(it->first);
|
|
|
|
NodeMap::const_iterator found = mNodeMap.find(bonename);
|
|
|
|
if (found == mNodeMap.end())
|
2015-06-15 20:42:14 +00:00
|
|
|
{
|
|
|
|
std::cerr << "addAnimSource: can't find bone '" + bonename << "' in " << model << " (referenced by " << kfname << ")" << std::endl;
|
|
|
|
continue;
|
|
|
|
}
|
2015-04-23 18:41:31 +00:00
|
|
|
|
|
|
|
osg::Node* node = found->second;
|
|
|
|
|
|
|
|
size_t group = detectAnimGroup(node);
|
|
|
|
|
|
|
|
// clone the controller, because each Animation needs its own ControllerSource
|
|
|
|
osg::ref_ptr<NifOsg::KeyframeController> cloned = osg::clone(it->second.get(), osg::CopyOp::DEEP_COPY_ALL);
|
|
|
|
cloned->setSource(mAnimationTimePtr[group]);
|
|
|
|
|
|
|
|
animsrc->mControllerMap[group].insert(std::make_pair(bonename, cloned));
|
|
|
|
}
|
|
|
|
|
|
|
|
mAnimSources.push_back(animsrc);
|
2015-04-22 17:08:56 +00:00
|
|
|
|
2015-04-23 18:41:31 +00:00
|
|
|
SceneUtil::AssignControllerSourcesVisitor assignVisitor(mAnimationTimePtr[0]);
|
|
|
|
mObjectRoot->accept(assignVisitor);
|
|
|
|
|
|
|
|
if (!mAccumRoot)
|
|
|
|
{
|
|
|
|
NodeMap::const_iterator found = mNodeMap.find("root bone");
|
|
|
|
if (found == mNodeMap.end())
|
|
|
|
found = mNodeMap.find("bip01");
|
|
|
|
|
|
|
|
if (found != mNodeMap.end())
|
|
|
|
mAccumRoot = found->second;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-04-23 21:30:06 +00:00
|
|
|
void Animation::clearAnimSources()
|
|
|
|
{
|
|
|
|
mStates.clear();
|
|
|
|
|
|
|
|
for(size_t i = 0;i < sNumGroups;i++)
|
2015-05-14 14:33:41 +00:00
|
|
|
mAnimationTimePtr[i]->setTimePtr(boost::shared_ptr<float>());
|
2015-04-23 21:30:06 +00:00
|
|
|
|
|
|
|
mAccumCtrl = NULL;
|
|
|
|
|
|
|
|
mAnimSources.clear();
|
|
|
|
}
|
|
|
|
|
2015-04-23 18:41:31 +00:00
|
|
|
bool Animation::hasAnimation(const std::string &anim)
|
|
|
|
{
|
|
|
|
AnimSourceList::const_iterator iter(mAnimSources.begin());
|
|
|
|
for(;iter != mAnimSources.end();++iter)
|
|
|
|
{
|
|
|
|
const NifOsg::TextKeyMap &keys = (*iter)->getTextKeys();
|
|
|
|
if(findGroupStart(keys, anim) != keys.end())
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
float Animation::getStartTime(const std::string &groupname) const
|
|
|
|
{
|
|
|
|
for(AnimSourceList::const_iterator iter(mAnimSources.begin()); iter != mAnimSources.end(); ++iter)
|
|
|
|
{
|
|
|
|
const NifOsg::TextKeyMap &keys = (*iter)->getTextKeys();
|
|
|
|
|
|
|
|
NifOsg::TextKeyMap::const_iterator found = findGroupStart(keys, groupname);
|
|
|
|
if(found != keys.end())
|
|
|
|
return found->first;
|
|
|
|
}
|
|
|
|
return -1.f;
|
|
|
|
}
|
|
|
|
|
|
|
|
float Animation::getTextKeyTime(const std::string &textKey) const
|
|
|
|
{
|
|
|
|
for(AnimSourceList::const_iterator iter(mAnimSources.begin()); iter != mAnimSources.end(); ++iter)
|
|
|
|
{
|
|
|
|
const NifOsg::TextKeyMap &keys = (*iter)->getTextKeys();
|
|
|
|
|
|
|
|
for(NifOsg::TextKeyMap::const_iterator iterKey(keys.begin()); iterKey != keys.end(); ++iterKey)
|
|
|
|
{
|
|
|
|
if(iterKey->second.compare(0, textKey.size(), textKey) == 0)
|
|
|
|
return iterKey->first;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return -1.f;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Animation::handleTextKey(AnimState &state, const std::string &groupname, const std::multimap<float, std::string>::const_iterator &key,
|
|
|
|
const std::multimap<float, std::string>& map)
|
|
|
|
{
|
2015-04-23 20:46:07 +00:00
|
|
|
const std::string &evt = key->second;
|
|
|
|
|
|
|
|
size_t off = groupname.size()+2;
|
|
|
|
size_t len = evt.size() - off;
|
|
|
|
|
2015-05-29 21:38:31 +00:00
|
|
|
if(evt.compare(0, groupname.size(), groupname) == 0 &&
|
|
|
|
evt.compare(groupname.size(), 2, ": ") == 0)
|
|
|
|
{
|
|
|
|
if(evt.compare(off, len, "loop start") == 0)
|
|
|
|
state.mLoopStartTime = key->first;
|
|
|
|
else if(evt.compare(off, len, "loop stop") == 0)
|
|
|
|
state.mLoopStopTime = key->first;
|
|
|
|
}
|
2015-04-23 20:46:07 +00:00
|
|
|
|
2015-05-21 22:55:43 +00:00
|
|
|
if (mTextKeyListener)
|
|
|
|
mTextKeyListener->handleTextKey(groupname, key, map);
|
2015-04-23 18:41:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Animation::play(const std::string &groupname, int priority, int groups, bool autodisable, float speedmult,
|
|
|
|
const std::string &start, const std::string &stop, float startpoint, size_t loops, bool loopfallback)
|
|
|
|
{
|
|
|
|
if(!mObjectRoot || mAnimSources.empty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
if(groupname.empty())
|
|
|
|
{
|
|
|
|
resetActiveGroups();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
priority = std::max(0, priority);
|
|
|
|
|
|
|
|
AnimStateMap::iterator stateiter = mStates.begin();
|
|
|
|
while(stateiter != mStates.end())
|
|
|
|
{
|
|
|
|
if(stateiter->second.mPriority == priority)
|
|
|
|
mStates.erase(stateiter++);
|
|
|
|
else
|
|
|
|
++stateiter;
|
|
|
|
}
|
|
|
|
|
|
|
|
stateiter = mStates.find(groupname);
|
|
|
|
if(stateiter != mStates.end())
|
|
|
|
{
|
|
|
|
stateiter->second.mPriority = priority;
|
|
|
|
resetActiveGroups();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Look in reverse; last-inserted source has priority. */
|
|
|
|
AnimState state;
|
|
|
|
AnimSourceList::reverse_iterator iter(mAnimSources.rbegin());
|
|
|
|
for(;iter != mAnimSources.rend();++iter)
|
|
|
|
{
|
|
|
|
const NifOsg::TextKeyMap &textkeys = (*iter)->getTextKeys();
|
|
|
|
if(reset(state, textkeys, groupname, start, stop, startpoint, loopfallback))
|
|
|
|
{
|
|
|
|
state.mSource = *iter;
|
|
|
|
state.mSpeedMult = speedmult;
|
|
|
|
state.mLoopCount = loops;
|
2015-05-14 14:33:41 +00:00
|
|
|
state.mPlaying = (state.getTime() < state.mStopTime);
|
2015-04-23 18:41:31 +00:00
|
|
|
state.mPriority = priority;
|
|
|
|
state.mGroups = groups;
|
|
|
|
state.mAutoDisable = autodisable;
|
|
|
|
mStates[groupname] = state;
|
|
|
|
|
2015-05-14 14:33:41 +00:00
|
|
|
NifOsg::TextKeyMap::const_iterator textkey(textkeys.lower_bound(state.getTime()));
|
2015-04-23 18:41:31 +00:00
|
|
|
if (state.mPlaying)
|
|
|
|
{
|
2015-05-14 14:33:41 +00:00
|
|
|
while(textkey != textkeys.end() && textkey->first <= state.getTime())
|
2015-04-23 18:41:31 +00:00
|
|
|
{
|
|
|
|
handleTextKey(state, groupname, textkey, textkeys);
|
|
|
|
++textkey;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-05-14 14:33:41 +00:00
|
|
|
if(state.getTime() >= state.mLoopStopTime && state.mLoopCount > 0)
|
2015-04-23 18:41:31 +00:00
|
|
|
{
|
|
|
|
state.mLoopCount--;
|
2015-05-14 14:33:41 +00:00
|
|
|
state.setTime(state.mLoopStartTime);
|
2015-04-23 18:41:31 +00:00
|
|
|
state.mPlaying = true;
|
2015-05-14 14:33:41 +00:00
|
|
|
if(state.getTime() >= state.mLoopStopTime)
|
2015-04-23 18:41:31 +00:00
|
|
|
break;
|
|
|
|
|
2015-05-14 14:33:41 +00:00
|
|
|
NifOsg::TextKeyMap::const_iterator textkey(textkeys.lower_bound(state.getTime()));
|
|
|
|
while(textkey != textkeys.end() && textkey->first <= state.getTime())
|
2015-04-23 18:41:31 +00:00
|
|
|
{
|
|
|
|
handleTextKey(state, groupname, textkey, textkeys);
|
|
|
|
++textkey;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if(iter == mAnimSources.rend())
|
|
|
|
std::cerr<< "Failed to find animation "<<groupname<<" for "<<mPtr.getCellRef().getRefId() <<std::endl;
|
|
|
|
|
|
|
|
resetActiveGroups();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Animation::reset(AnimState &state, const NifOsg::TextKeyMap &keys, const std::string &groupname, const std::string &start, const std::string &stop, float startpoint, bool loopfallback)
|
|
|
|
{
|
|
|
|
// Look for text keys in reverse. This normally wouldn't matter, but for some reason undeadwolf_2.nif has two
|
|
|
|
// separate walkforward keys, and the last one is supposed to be used.
|
|
|
|
NifOsg::TextKeyMap::const_reverse_iterator groupend(keys.rbegin());
|
|
|
|
for(;groupend != keys.rend();++groupend)
|
|
|
|
{
|
|
|
|
if(groupend->second.compare(0, groupname.size(), groupname) == 0 &&
|
|
|
|
groupend->second.compare(groupname.size(), 2, ": ") == 0)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string starttag = groupname+": "+start;
|
|
|
|
NifOsg::TextKeyMap::const_reverse_iterator startkey(groupend);
|
|
|
|
while(startkey != keys.rend() && startkey->second != starttag)
|
|
|
|
++startkey;
|
|
|
|
if(startkey == keys.rend() && start == "loop start")
|
|
|
|
{
|
|
|
|
starttag = groupname+": start";
|
|
|
|
startkey = groupend;
|
|
|
|
while(startkey != keys.rend() && startkey->second != starttag)
|
|
|
|
++startkey;
|
|
|
|
}
|
|
|
|
if(startkey == keys.rend())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
const std::string stoptag = groupname+": "+stop;
|
|
|
|
NifOsg::TextKeyMap::const_reverse_iterator stopkey(groupend);
|
|
|
|
while(stopkey != keys.rend()
|
|
|
|
// We have to ignore extra garbage at the end.
|
|
|
|
// The Scrib's idle3 animation has "Idle3: Stop." instead of "Idle3: Stop".
|
|
|
|
// Why, just why? :(
|
|
|
|
&& (stopkey->second.size() < stoptag.size() || stopkey->second.substr(0,stoptag.size()) != stoptag))
|
|
|
|
++stopkey;
|
|
|
|
if(stopkey == keys.rend())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if(startkey->first > stopkey->first)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
state.mStartTime = startkey->first;
|
|
|
|
if (loopfallback)
|
|
|
|
{
|
|
|
|
state.mLoopStartTime = startkey->first;
|
|
|
|
state.mLoopStopTime = stopkey->first;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
state.mLoopStartTime = startkey->first;
|
|
|
|
state.mLoopStopTime = std::numeric_limits<float>::max();
|
|
|
|
}
|
|
|
|
state.mStopTime = stopkey->first;
|
|
|
|
|
2015-05-14 14:33:41 +00:00
|
|
|
state.setTime(state.mStartTime + ((state.mStopTime - state.mStartTime) * startpoint));
|
2015-04-23 18:41:31 +00:00
|
|
|
|
|
|
|
// mLoopStartTime and mLoopStopTime normally get assigned when encountering these keys while playing the animation
|
|
|
|
// (see handleTextKey). But if startpoint is already past these keys, we need to assign them now.
|
2015-05-14 14:33:41 +00:00
|
|
|
if(state.getTime() > state.mStartTime)
|
2015-04-23 18:41:31 +00:00
|
|
|
{
|
|
|
|
const std::string loopstarttag = groupname+": loop start";
|
|
|
|
const std::string loopstoptag = groupname+": loop stop";
|
|
|
|
|
|
|
|
NifOsg::TextKeyMap::const_reverse_iterator key(groupend);
|
|
|
|
for (; key != startkey && key != keys.rend(); ++key)
|
|
|
|
{
|
2015-05-14 14:33:41 +00:00
|
|
|
if (key->first > state.getTime())
|
2015-04-23 18:41:31 +00:00
|
|
|
continue;
|
|
|
|
|
|
|
|
if (key->second == loopstarttag)
|
|
|
|
state.mLoopStartTime = key->first;
|
|
|
|
else if (key->second == loopstoptag)
|
|
|
|
state.mLoopStopTime = key->first;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-05-21 22:55:43 +00:00
|
|
|
void Animation::setTextKeyListener(Animation::TextKeyListener *listener)
|
|
|
|
{
|
|
|
|
mTextKeyListener = listener;
|
|
|
|
}
|
|
|
|
|
2015-04-23 18:41:31 +00:00
|
|
|
void Animation::resetActiveGroups()
|
|
|
|
{
|
|
|
|
// remove all previous external controllers from the scene graph
|
2015-05-31 00:26:31 +00:00
|
|
|
for (ControllerMap::iterator it = mActiveControllers.begin(); it != mActiveControllers.end(); ++it)
|
2015-04-23 18:41:31 +00:00
|
|
|
{
|
|
|
|
osg::Node* node = it->first;
|
|
|
|
node->removeUpdateCallback(it->second);
|
2015-04-25 13:10:37 +00:00
|
|
|
|
|
|
|
// Should be no longer needed with OSG 3.4
|
|
|
|
it->second->setNestedCallback(NULL);
|
2015-04-23 18:41:31 +00:00
|
|
|
}
|
2015-05-31 00:26:31 +00:00
|
|
|
|
|
|
|
mActiveControllers.clear();
|
2015-04-23 18:41:31 +00:00
|
|
|
|
|
|
|
mAccumCtrl = NULL;
|
|
|
|
|
|
|
|
for(size_t grp = 0;grp < sNumGroups;grp++)
|
|
|
|
{
|
|
|
|
AnimStateMap::const_iterator active = mStates.end();
|
|
|
|
|
|
|
|
AnimStateMap::const_iterator state = mStates.begin();
|
|
|
|
for(;state != mStates.end();++state)
|
|
|
|
{
|
|
|
|
if(!(state->second.mGroups&(1<<grp)))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if(active == mStates.end() || active->second.mPriority < state->second.mPriority)
|
|
|
|
active = state;
|
|
|
|
}
|
|
|
|
|
2015-05-14 14:33:41 +00:00
|
|
|
mAnimationTimePtr[grp]->setTimePtr(active == mStates.end() ? boost::shared_ptr<float>() : active->second.mTime);
|
2015-04-23 18:41:31 +00:00
|
|
|
|
|
|
|
// add external controllers for the AnimSource active in this group
|
|
|
|
if (active != mStates.end())
|
|
|
|
{
|
|
|
|
boost::shared_ptr<AnimSource> animsrc = active->second.mSource;
|
|
|
|
|
|
|
|
for (AnimSource::ControllerMap::iterator it = animsrc->mControllerMap[grp].begin(); it != animsrc->mControllerMap[grp].end(); ++it)
|
|
|
|
{
|
|
|
|
osg::ref_ptr<osg::Node> node = mNodeMap.at(it->first); // this should not throw, we already checked for the node existing in addAnimSource
|
|
|
|
|
|
|
|
node->addUpdateCallback(it->second);
|
2015-05-31 00:26:31 +00:00
|
|
|
mActiveControllers.insert(std::make_pair(node, it->second));
|
2015-04-23 18:41:31 +00:00
|
|
|
|
|
|
|
if (grp == 0 && node == mAccumRoot)
|
2015-04-23 20:46:07 +00:00
|
|
|
{
|
2015-04-23 18:41:31 +00:00
|
|
|
mAccumCtrl = it->second;
|
2015-04-23 20:46:07 +00:00
|
|
|
|
2015-04-25 13:10:37 +00:00
|
|
|
// make sure reset is last in the chain of callbacks
|
2015-04-23 20:46:07 +00:00
|
|
|
if (!mResetAccumRootCallback)
|
|
|
|
{
|
|
|
|
mResetAccumRootCallback = new ResetAccumRootCallback;
|
|
|
|
mResetAccumRootCallback->setAccumulate(mAccumulate);
|
|
|
|
}
|
|
|
|
mAccumRoot->addUpdateCallback(mResetAccumRootCallback);
|
2015-05-31 00:26:31 +00:00
|
|
|
mActiveControllers.insert(std::make_pair(mAccumRoot, mResetAccumRootCallback));
|
2015-04-23 20:46:07 +00:00
|
|
|
}
|
2015-04-23 18:41:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-05-31 00:26:31 +00:00
|
|
|
addControllers();
|
2015-04-23 18:41:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Animation::changeGroups(const std::string &groupname, int groups)
|
|
|
|
{
|
|
|
|
AnimStateMap::iterator stateiter = mStates.find(groupname);
|
|
|
|
if(stateiter != mStates.end())
|
|
|
|
{
|
|
|
|
if(stateiter->second.mGroups != groups)
|
|
|
|
{
|
|
|
|
stateiter->second.mGroups = groups;
|
|
|
|
resetActiveGroups();
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Animation::stopLooping(const std::string& groupname)
|
|
|
|
{
|
|
|
|
AnimStateMap::iterator stateiter = mStates.find(groupname);
|
|
|
|
if(stateiter != mStates.end())
|
|
|
|
{
|
|
|
|
stateiter->second.mLoopCount = 0;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Animation::adjustSpeedMult(const std::string &groupname, float speedmult)
|
|
|
|
{
|
|
|
|
AnimStateMap::iterator state(mStates.find(groupname));
|
|
|
|
if(state != mStates.end())
|
|
|
|
state->second.mSpeedMult = speedmult;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Animation::isPlaying(const std::string &groupname) const
|
|
|
|
{
|
|
|
|
AnimStateMap::const_iterator state(mStates.find(groupname));
|
|
|
|
if(state != mStates.end())
|
|
|
|
return state->second.mPlaying;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Animation::getInfo(const std::string &groupname, float *complete, float *speedmult) const
|
|
|
|
{
|
|
|
|
AnimStateMap::const_iterator iter = mStates.find(groupname);
|
|
|
|
if(iter == mStates.end())
|
|
|
|
{
|
|
|
|
if(complete) *complete = 0.0f;
|
|
|
|
if(speedmult) *speedmult = 0.0f;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(complete)
|
|
|
|
{
|
|
|
|
if(iter->second.mStopTime > iter->second.mStartTime)
|
2015-05-14 14:33:41 +00:00
|
|
|
*complete = (iter->second.getTime() - iter->second.mStartTime) /
|
2015-04-23 18:41:31 +00:00
|
|
|
(iter->second.mStopTime - iter->second.mStartTime);
|
|
|
|
else
|
|
|
|
*complete = (iter->second.mPlaying ? 0.0f : 1.0f);
|
|
|
|
}
|
|
|
|
if(speedmult) *speedmult = iter->second.mSpeedMult;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
float Animation::getCurrentTime(const std::string &groupname) const
|
|
|
|
{
|
|
|
|
AnimStateMap::const_iterator iter = mStates.find(groupname);
|
|
|
|
if(iter == mStates.end())
|
|
|
|
return -1.f;
|
|
|
|
|
2015-05-14 14:33:41 +00:00
|
|
|
return iter->second.getTime();
|
2015-04-23 18:41:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Animation::disable(const std::string &groupname)
|
|
|
|
{
|
|
|
|
AnimStateMap::iterator iter = mStates.find(groupname);
|
|
|
|
if(iter != mStates.end())
|
|
|
|
mStates.erase(iter);
|
|
|
|
resetActiveGroups();
|
|
|
|
}
|
|
|
|
|
|
|
|
float Animation::getVelocity(const std::string &groupname) const
|
|
|
|
{
|
2015-04-23 21:30:06 +00:00
|
|
|
if (!mAccumRoot)
|
|
|
|
return 0.0f;
|
|
|
|
|
2015-04-23 18:41:31 +00:00
|
|
|
// Look in reverse; last-inserted source has priority.
|
|
|
|
AnimSourceList::const_reverse_iterator animsrc(mAnimSources.rbegin());
|
|
|
|
for(;animsrc != mAnimSources.rend();++animsrc)
|
|
|
|
{
|
2015-04-23 21:30:06 +00:00
|
|
|
const NifOsg::TextKeyMap &keys = (*animsrc)->getTextKeys();
|
2015-04-23 18:41:31 +00:00
|
|
|
if(findGroupStart(keys, groupname) != keys.end())
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if(animsrc == mAnimSources.rend())
|
|
|
|
return 0.0f;
|
|
|
|
|
|
|
|
float velocity = 0.0f;
|
2015-04-23 21:30:06 +00:00
|
|
|
const NifOsg::TextKeyMap &keys = (*animsrc)->getTextKeys();
|
|
|
|
|
|
|
|
const AnimSource::ControllerMap& ctrls = (*animsrc)->mControllerMap[0];
|
|
|
|
for (AnimSource::ControllerMap::const_iterator it = ctrls.begin(); it != ctrls.end(); ++it)
|
2015-04-23 18:41:31 +00:00
|
|
|
{
|
2015-04-23 21:30:06 +00:00
|
|
|
if (Misc::StringUtils::ciEqual(it->first, mAccumRoot->getName()))
|
2015-04-23 18:41:31 +00:00
|
|
|
{
|
2015-04-23 21:30:06 +00:00
|
|
|
velocity = calcAnimVelocity(keys, it->second, mAccumulate, groupname);
|
2015-04-23 18:41:31 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// If there's no velocity, keep looking
|
|
|
|
if(!(velocity > 1.0f))
|
|
|
|
{
|
|
|
|
AnimSourceList::const_reverse_iterator animiter = mAnimSources.rbegin();
|
|
|
|
while(*animiter != *animsrc)
|
|
|
|
++animiter;
|
|
|
|
|
|
|
|
while(!(velocity > 1.0f) && ++animiter != mAnimSources.rend())
|
|
|
|
{
|
2015-04-23 21:30:06 +00:00
|
|
|
const NifOsg::TextKeyMap &keys = (*animiter)->getTextKeys();
|
|
|
|
|
|
|
|
const AnimSource::ControllerMap& ctrls = (*animiter)->mControllerMap[0];
|
|
|
|
for (AnimSource::ControllerMap::const_iterator it = ctrls.begin(); it != ctrls.end(); ++it)
|
2015-04-23 18:41:31 +00:00
|
|
|
{
|
2015-04-23 21:30:06 +00:00
|
|
|
if (Misc::StringUtils::ciEqual(it->first, mAccumRoot->getName()))
|
2015-04-23 18:41:31 +00:00
|
|
|
{
|
2015-04-23 21:30:06 +00:00
|
|
|
velocity = calcAnimVelocity(keys, it->second, mAccumulate, groupname);
|
2015-04-23 18:41:31 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return velocity;
|
2015-04-22 17:08:56 +00:00
|
|
|
}
|
|
|
|
|
2015-04-23 20:46:07 +00:00
|
|
|
void Animation::updatePosition(float oldtime, float newtime, osg::Vec3f& position)
|
|
|
|
{
|
|
|
|
// Get the difference from the last update, and move the position
|
|
|
|
osg::Vec3f off = osg::componentMultiply(mAccumCtrl->getTranslation(newtime), mAccumulate);
|
|
|
|
position += off - osg::componentMultiply(mAccumCtrl->getTranslation(oldtime), mAccumulate);
|
|
|
|
}
|
|
|
|
|
2015-04-12 13:34:50 +00:00
|
|
|
osg::Vec3f Animation::runAnimation(float duration)
|
2015-01-02 01:27:05 +00:00
|
|
|
{
|
2015-04-23 18:41:31 +00:00
|
|
|
osg::Vec3f movement(0.f, 0.f, 0.f);
|
|
|
|
AnimStateMap::iterator stateiter = mStates.begin();
|
|
|
|
while(stateiter != mStates.end())
|
|
|
|
{
|
|
|
|
AnimState &state = stateiter->second;
|
|
|
|
const NifOsg::TextKeyMap &textkeys = state.mSource->getTextKeys();
|
2015-05-14 14:33:41 +00:00
|
|
|
NifOsg::TextKeyMap::const_iterator textkey(textkeys.upper_bound(state.getTime()));
|
2015-04-23 18:41:31 +00:00
|
|
|
|
|
|
|
float timepassed = duration * state.mSpeedMult;
|
|
|
|
while(state.mPlaying)
|
|
|
|
{
|
|
|
|
float targetTime;
|
|
|
|
|
2015-05-14 14:33:41 +00:00
|
|
|
if(state.getTime() >= state.mLoopStopTime && state.mLoopCount > 0)
|
2015-04-23 18:41:31 +00:00
|
|
|
goto handle_loop;
|
|
|
|
|
2015-05-14 14:33:41 +00:00
|
|
|
targetTime = state.getTime() + timepassed;
|
2015-04-23 18:41:31 +00:00
|
|
|
if(textkey == textkeys.end() || textkey->first > targetTime)
|
|
|
|
{
|
2015-05-14 14:33:41 +00:00
|
|
|
if(mAccumCtrl && state.mTime == mAnimationTimePtr[0]->getTimePtr())
|
|
|
|
updatePosition(state.getTime(), targetTime, movement);
|
|
|
|
state.setTime(std::min(targetTime, state.mStopTime));
|
2015-04-23 18:41:31 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2015-05-14 14:33:41 +00:00
|
|
|
if(mAccumCtrl && state.mTime == mAnimationTimePtr[0]->getTimePtr())
|
|
|
|
updatePosition(state.getTime(), textkey->first, movement);
|
|
|
|
state.setTime(textkey->first);
|
2015-04-23 18:41:31 +00:00
|
|
|
}
|
|
|
|
|
2015-05-14 14:33:41 +00:00
|
|
|
state.mPlaying = (state.getTime() < state.mStopTime);
|
|
|
|
timepassed = targetTime - state.getTime();
|
2015-04-23 18:41:31 +00:00
|
|
|
|
2015-05-14 14:33:41 +00:00
|
|
|
while(textkey != textkeys.end() && textkey->first <= state.getTime())
|
2015-04-23 18:41:31 +00:00
|
|
|
{
|
|
|
|
handleTextKey(state, stateiter->first, textkey, textkeys);
|
|
|
|
++textkey;
|
|
|
|
}
|
|
|
|
|
2015-05-14 14:33:41 +00:00
|
|
|
if(state.getTime() >= state.mLoopStopTime && state.mLoopCount > 0)
|
2015-04-23 18:41:31 +00:00
|
|
|
{
|
|
|
|
handle_loop:
|
|
|
|
state.mLoopCount--;
|
2015-05-14 14:33:41 +00:00
|
|
|
state.setTime(state.mLoopStartTime);
|
2015-04-23 18:41:31 +00:00
|
|
|
state.mPlaying = true;
|
|
|
|
|
2015-05-14 14:33:41 +00:00
|
|
|
textkey = textkeys.lower_bound(state.getTime());
|
|
|
|
while(textkey != textkeys.end() && textkey->first <= state.getTime())
|
2015-04-23 18:41:31 +00:00
|
|
|
{
|
|
|
|
handleTextKey(state, stateiter->first, textkey, textkeys);
|
|
|
|
++textkey;
|
|
|
|
}
|
|
|
|
|
2015-05-14 14:33:41 +00:00
|
|
|
if(state.getTime() >= state.mLoopStopTime)
|
2015-04-23 18:41:31 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(timepassed <= 0.0f)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(!state.mPlaying && state.mAutoDisable)
|
|
|
|
{
|
|
|
|
mStates.erase(stateiter++);
|
|
|
|
|
|
|
|
resetActiveGroups();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
++stateiter;
|
|
|
|
}
|
|
|
|
|
2015-04-18 23:57:52 +00:00
|
|
|
updateEffects(duration);
|
|
|
|
|
2015-04-23 18:41:31 +00:00
|
|
|
return movement;
|
2015-01-02 01:27:05 +00:00
|
|
|
}
|
2013-05-11 01:37:44 +00:00
|
|
|
|
2015-05-21 21:24:22 +00:00
|
|
|
void Animation::setObjectRoot(const std::string &model, bool forceskeleton, bool baseonly)
|
2013-05-11 01:37:44 +00:00
|
|
|
{
|
2015-04-12 13:34:50 +00:00
|
|
|
if (mObjectRoot)
|
2013-04-25 05:45:43 +00:00
|
|
|
{
|
2015-04-12 13:34:50 +00:00
|
|
|
mObjectRoot->getParent(0)->removeChild(mObjectRoot);
|
2013-04-25 05:45:43 +00:00
|
|
|
}
|
2015-04-23 20:46:07 +00:00
|
|
|
mObjectRoot = NULL;
|
2013-04-25 07:03:27 +00:00
|
|
|
|
2015-04-23 18:41:31 +00:00
|
|
|
mNodeMap.clear();
|
2015-05-31 00:26:31 +00:00
|
|
|
mActiveControllers.clear();
|
2015-04-23 20:46:07 +00:00
|
|
|
mAccumRoot = NULL;
|
|
|
|
mAccumCtrl = NULL;
|
2015-04-23 18:41:31 +00:00
|
|
|
|
2015-04-23 21:30:06 +00:00
|
|
|
if (!forceskeleton)
|
|
|
|
mObjectRoot = mResourceSystem->getSceneManager()->createInstance(model, mInsert);
|
|
|
|
else
|
|
|
|
{
|
|
|
|
osg::ref_ptr<osg::Node> newObjectRoot = mResourceSystem->getSceneManager()->createInstance(model);
|
|
|
|
if (!dynamic_cast<SceneUtil::Skeleton*>(newObjectRoot.get()))
|
|
|
|
{
|
|
|
|
osg::ref_ptr<SceneUtil::Skeleton> skel = new SceneUtil::Skeleton;
|
|
|
|
skel->addChild(newObjectRoot);
|
|
|
|
newObjectRoot = skel;
|
|
|
|
}
|
|
|
|
mInsert->addChild(newObjectRoot);
|
|
|
|
mObjectRoot = newObjectRoot;
|
|
|
|
}
|
2015-04-23 18:41:31 +00:00
|
|
|
|
2015-05-21 21:24:22 +00:00
|
|
|
if (baseonly)
|
|
|
|
{
|
|
|
|
RemoveDrawableVisitor removeDrawableVisitor;
|
|
|
|
mObjectRoot->accept(removeDrawableVisitor);
|
|
|
|
removeDrawableVisitor.remove();
|
|
|
|
}
|
|
|
|
|
2015-04-23 18:41:31 +00:00
|
|
|
NodeMapVisitor visitor;
|
|
|
|
mObjectRoot->accept(visitor);
|
|
|
|
mNodeMap = visitor.getNodeMap();
|
2015-05-01 16:21:50 +00:00
|
|
|
|
|
|
|
mObjectRoot->addCullCallback(new SceneUtil::LightListCallback);
|
2013-07-23 12:30:37 +00:00
|
|
|
}
|
2013-04-25 07:03:27 +00:00
|
|
|
|
2015-04-12 13:34:50 +00:00
|
|
|
osg::Group* Animation::getObjectRoot()
|
2014-02-04 02:46:15 +00:00
|
|
|
{
|
2015-04-12 13:34:50 +00:00
|
|
|
return static_cast<osg::Group*>(mObjectRoot.get());
|
2014-06-10 20:20:46 +00:00
|
|
|
}
|
|
|
|
|
2015-04-12 13:34:50 +00:00
|
|
|
osg::Group* Animation::getOrCreateObjectRoot()
|
2014-06-10 20:20:46 +00:00
|
|
|
{
|
2015-04-12 13:34:50 +00:00
|
|
|
if (mObjectRoot)
|
|
|
|
return static_cast<osg::Group*>(mObjectRoot.get());
|
2014-06-10 20:20:46 +00:00
|
|
|
|
2015-04-12 13:34:50 +00:00
|
|
|
mObjectRoot = new osg::Group;
|
|
|
|
mInsert->addChild(mObjectRoot);
|
|
|
|
return static_cast<osg::Group*>(mObjectRoot.get());
|
2014-02-04 02:46:15 +00:00
|
|
|
}
|
2014-06-10 20:20:46 +00:00
|
|
|
|
2015-04-16 23:23:37 +00:00
|
|
|
void Animation::addGlow(osg::ref_ptr<osg::Node> node, osg::Vec4f glowColor)
|
|
|
|
{
|
|
|
|
std::vector<osg::ref_ptr<osg::Texture2D> > textures;
|
|
|
|
for (int i=0; i<32; ++i)
|
|
|
|
{
|
|
|
|
std::stringstream stream;
|
|
|
|
stream << "textures/magicitem/caust";
|
|
|
|
stream << std::setw(2);
|
|
|
|
stream << std::setfill('0');
|
|
|
|
stream << i;
|
|
|
|
stream << ".dds";
|
|
|
|
|
|
|
|
textures.push_back(mResourceSystem->getTextureManager()->getTexture2D(stream.str(), osg::Texture2D::REPEAT, osg::Texture2D::REPEAT));
|
|
|
|
}
|
|
|
|
|
|
|
|
osg::ref_ptr<GlowUpdater> glowupdater (new GlowUpdater(glowColor, textures));
|
|
|
|
node->addUpdateCallback(glowupdater);
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: Should not be here
|
|
|
|
osg::Vec4f Animation::getEnchantmentColor(MWWorld::Ptr item)
|
|
|
|
{
|
|
|
|
osg::Vec4f result(1,1,1,1);
|
|
|
|
std::string enchantmentName = item.getClass().getEnchantment(item);
|
|
|
|
if (enchantmentName.empty())
|
|
|
|
return result;
|
|
|
|
const ESM::Enchantment* enchantment = MWBase::Environment::get().getWorld()->getStore().get<ESM::Enchantment>().find(enchantmentName);
|
|
|
|
assert (enchantment->mEffects.mList.size());
|
|
|
|
const ESM::MagicEffect* magicEffect = MWBase::Environment::get().getWorld()->getStore().get<ESM::MagicEffect>().find(
|
|
|
|
enchantment->mEffects.mList.front().mEffectID);
|
|
|
|
result.x() = magicEffect->mData.mRed / 255.f;
|
|
|
|
result.y() = magicEffect->mData.mGreen / 255.f;
|
|
|
|
result.z() = magicEffect->mData.mBlue / 255.f;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2015-04-19 13:03:08 +00:00
|
|
|
void Animation::addExtraLight(osg::ref_ptr<osg::Group> parent, const ESM::Light *esmLight)
|
|
|
|
{
|
|
|
|
SceneUtil::FindByNameVisitor visitor("AttachLight");
|
|
|
|
parent->accept(visitor);
|
|
|
|
|
|
|
|
osg::Group* attachTo = NULL;
|
|
|
|
if (visitor.mFoundNode)
|
|
|
|
{
|
|
|
|
attachTo = visitor.mFoundNode;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
osg::ComputeBoundsVisitor computeBound;
|
2015-06-17 16:22:31 +00:00
|
|
|
computeBound.setTraversalMask(~Mask_ParticleSystem);
|
2015-04-19 13:03:08 +00:00
|
|
|
parent->accept(computeBound);
|
|
|
|
|
|
|
|
// PositionAttitudeTransform seems to be slightly faster than MatrixTransform
|
|
|
|
osg::ref_ptr<osg::PositionAttitudeTransform> trans(new osg::PositionAttitudeTransform);
|
|
|
|
trans->setPosition(computeBound.getBoundingBox().center());
|
|
|
|
|
|
|
|
parent->addChild(trans);
|
|
|
|
|
|
|
|
attachTo = trans;
|
|
|
|
}
|
|
|
|
|
|
|
|
osg::ref_ptr<SceneUtil::LightSource> lightSource = new SceneUtil::LightSource;
|
|
|
|
osg::Light* light = new osg::Light;
|
|
|
|
lightSource->setLight(light);
|
|
|
|
|
2015-06-02 15:02:56 +00:00
|
|
|
const MWWorld::Fallback* fallback = MWBase::Environment::get().getWorld()->getFallback();
|
2015-04-19 13:03:08 +00:00
|
|
|
|
2015-06-02 15:02:56 +00:00
|
|
|
float radius = esmLight->mData.mRadius;
|
|
|
|
lightSource->setRadius(radius);
|
2015-04-19 13:03:08 +00:00
|
|
|
|
2015-06-02 15:02:56 +00:00
|
|
|
static bool outQuadInLin = fallback->getFallbackBool("LightAttenuation_OutQuadInLin");
|
|
|
|
static bool useQuadratic = fallback->getFallbackBool("LightAttenuation_UseQuadratic");
|
|
|
|
static float quadraticValue = fallback->getFallbackFloat("LightAttenuation_QuadraticValue");
|
|
|
|
static float quadraticRadiusMult = fallback->getFallbackFloat("LightAttenuation_QuadraticRadiusMult");
|
|
|
|
static bool useLinear = fallback->getFallbackBool("LightAttenuation_UseLinear");
|
|
|
|
static float linearRadiusMult = fallback->getFallbackFloat("LightAttenuation_LinearRadiusMult");
|
|
|
|
static float linearValue = fallback->getFallbackFloat("LightAttenuation_LinearValue");
|
|
|
|
|
|
|
|
bool exterior = mPtr.isInCell() && mPtr.getCell()->getCell()->isExterior();
|
|
|
|
|
|
|
|
SceneUtil::configureLight(light, radius, exterior, outQuadInLin, useQuadratic, quadraticValue,
|
|
|
|
quadraticRadiusMult, useLinear, linearRadiusMult, linearValue);
|
|
|
|
|
|
|
|
osg::Vec4f diffuse = SceneUtil::colourFromRGB(esmLight->mData.mColor);
|
|
|
|
if (esmLight->mData.mFlags & ESM::Light::Negative)
|
2015-06-02 15:06:55 +00:00
|
|
|
{
|
2015-06-02 15:02:56 +00:00
|
|
|
diffuse *= -1;
|
2015-06-02 15:06:55 +00:00
|
|
|
diffuse.a() = 1;
|
|
|
|
}
|
2015-06-02 15:02:56 +00:00
|
|
|
light->setDiffuse(diffuse);
|
2015-04-19 13:03:08 +00:00
|
|
|
light->setAmbient(osg::Vec4f(0,0,0,1));
|
|
|
|
light->setSpecular(osg::Vec4f(0,0,0,0));
|
|
|
|
|
2015-04-22 15:34:39 +00:00
|
|
|
osg::ref_ptr<SceneUtil::LightController> ctrl (new SceneUtil::LightController);
|
|
|
|
ctrl->setDiffuse(light->getDiffuse());
|
|
|
|
if (esmLight->mData.mFlags & ESM::Light::Flicker)
|
|
|
|
ctrl->setType(SceneUtil::LightController::LT_Flicker);
|
|
|
|
if (esmLight->mData.mFlags & ESM::Light::FlickerSlow)
|
|
|
|
ctrl->setType(SceneUtil::LightController::LT_FlickerSlow);
|
|
|
|
if (esmLight->mData.mFlags & ESM::Light::Pulse)
|
|
|
|
ctrl->setType(SceneUtil::LightController::LT_Pulse);
|
|
|
|
if (esmLight->mData.mFlags & ESM::Light::PulseSlow)
|
|
|
|
ctrl->setType(SceneUtil::LightController::LT_PulseSlow);
|
|
|
|
|
|
|
|
lightSource->addUpdateCallback(ctrl);
|
|
|
|
|
2015-04-19 13:03:08 +00:00
|
|
|
attachTo->addChild(lightSource);
|
|
|
|
}
|
|
|
|
|
2015-04-18 23:57:52 +00:00
|
|
|
void Animation::addEffect (const std::string& model, int effectId, bool loop, const std::string& bonename, std::string texture)
|
|
|
|
{
|
2015-06-07 15:09:37 +00:00
|
|
|
if (!mObjectRoot.get())
|
|
|
|
return;
|
|
|
|
|
2015-04-18 23:57:52 +00:00
|
|
|
// Early out if we already have this effect
|
|
|
|
for (std::vector<EffectParams>::iterator it = mEffects.begin(); it != mEffects.end(); ++it)
|
|
|
|
if (it->mLoop && loop && it->mEffectId == effectId && it->mBoneName == bonename)
|
|
|
|
return;
|
|
|
|
|
|
|
|
EffectParams params;
|
|
|
|
params.mModelName = model;
|
|
|
|
osg::ref_ptr<osg::Group> parentNode;
|
|
|
|
if (bonename.empty())
|
2015-06-11 14:22:09 +00:00
|
|
|
parentNode = mInsert;
|
2015-04-18 23:57:52 +00:00
|
|
|
else
|
|
|
|
{
|
2015-05-29 18:45:27 +00:00
|
|
|
NodeMap::iterator found = mNodeMap.find(Misc::StringUtils::lowerCase(bonename));
|
|
|
|
if (found == mNodeMap.end())
|
2015-04-18 23:57:52 +00:00
|
|
|
throw std::runtime_error("Can't find bone " + bonename);
|
2015-05-29 18:45:27 +00:00
|
|
|
|
|
|
|
parentNode = found->second;
|
2015-04-18 23:57:52 +00:00
|
|
|
}
|
|
|
|
osg::ref_ptr<osg::Node> node = mResourceSystem->getSceneManager()->createInstance(model, parentNode);
|
2015-06-07 02:41:55 +00:00
|
|
|
|
|
|
|
node->getOrCreateStateSet()->setMode(GL_LIGHTING, osg::StateAttribute::OFF);
|
|
|
|
|
2015-04-18 23:57:52 +00:00
|
|
|
params.mObjects = PartHolderPtr(new PartHolder(node));
|
|
|
|
|
2015-04-19 15:55:56 +00:00
|
|
|
SceneUtil::FindMaxControllerLengthVisitor findMaxLengthVisitor;
|
2015-04-18 23:57:52 +00:00
|
|
|
node->accept(findMaxLengthVisitor);
|
|
|
|
|
|
|
|
params.mMaxControllerLength = findMaxLengthVisitor.getMaxLength();
|
|
|
|
|
|
|
|
node->setNodeMask(Mask_Effect);
|
|
|
|
|
|
|
|
params.mLoop = loop;
|
|
|
|
params.mEffectId = effectId;
|
|
|
|
params.mBoneName = bonename;
|
|
|
|
|
|
|
|
params.mAnimTime = boost::shared_ptr<EffectAnimationTime>(new EffectAnimationTime);
|
|
|
|
|
|
|
|
SceneUtil::AssignControllerSourcesVisitor assignVisitor(boost::shared_ptr<SceneUtil::ControllerSource>(params.mAnimTime));
|
|
|
|
node->accept(assignVisitor);
|
|
|
|
|
2015-04-19 15:55:56 +00:00
|
|
|
overrideTexture(texture, mResourceSystem, node);
|
2015-04-18 23:57:52 +00:00
|
|
|
|
|
|
|
// TODO: in vanilla morrowind the effect is scaled based on the host object's bounding box.
|
|
|
|
|
|
|
|
mEffects.push_back(params);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Animation::removeEffect(int effectId)
|
|
|
|
{
|
|
|
|
for (std::vector<EffectParams>::iterator it = mEffects.begin(); it != mEffects.end(); ++it)
|
|
|
|
{
|
|
|
|
if (it->mEffectId == effectId)
|
|
|
|
{
|
|
|
|
mEffects.erase(it);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Animation::getLoopingEffects(std::vector<int> &out)
|
|
|
|
{
|
|
|
|
for (std::vector<EffectParams>::iterator it = mEffects.begin(); it != mEffects.end(); ++it)
|
|
|
|
{
|
|
|
|
if (it->mLoop)
|
|
|
|
out.push_back(it->mEffectId);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Animation::updateEffects(float duration)
|
|
|
|
{
|
|
|
|
for (std::vector<EffectParams>::iterator it = mEffects.begin(); it != mEffects.end(); )
|
|
|
|
{
|
|
|
|
it->mAnimTime->addTime(duration);
|
|
|
|
|
|
|
|
if (it->mAnimTime->getTime() >= it->mMaxControllerLength)
|
|
|
|
{
|
|
|
|
if (it->mLoop)
|
|
|
|
{
|
|
|
|
// Start from the beginning again; carry over the remainder
|
|
|
|
// Not sure if this is actually needed, the controller function might already handle loops
|
|
|
|
float remainder = it->mAnimTime->getTime() - it->mMaxControllerLength;
|
|
|
|
it->mAnimTime->resetTime(remainder);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
it = mEffects.erase(it);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
++it;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-05-12 15:40:42 +00:00
|
|
|
bool Animation::upperBodyReady() const
|
|
|
|
{
|
|
|
|
for (AnimStateMap::const_iterator stateiter = mStates.begin(); stateiter != mStates.end(); ++stateiter)
|
|
|
|
{
|
|
|
|
if((stateiter->second.mPriority > MWMechanics::Priority_Movement
|
|
|
|
&& stateiter->second.mPriority < MWMechanics::Priority_Torch)
|
|
|
|
|| stateiter->second.mPriority == MWMechanics::Priority_Death)
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-05-20 00:18:20 +00:00
|
|
|
const osg::Node* Animation::getNode(const std::string &name) const
|
|
|
|
{
|
|
|
|
std::string lowerName = Misc::StringUtils::lowerCase(name);
|
|
|
|
NodeMap::const_iterator found = mNodeMap.find(lowerName);
|
|
|
|
if (found == mNodeMap.end())
|
2015-05-29 23:41:38 +00:00
|
|
|
return NULL;
|
|
|
|
else
|
|
|
|
return found->second;
|
2015-05-20 00:18:20 +00:00
|
|
|
}
|
|
|
|
|
2015-04-23 18:41:31 +00:00
|
|
|
float Animation::AnimationTime::getValue(osg::NodeVisitor*)
|
|
|
|
{
|
2015-05-14 14:33:41 +00:00
|
|
|
if (mTimePtr)
|
|
|
|
return *mTimePtr;
|
|
|
|
return 0.f;
|
2015-04-23 18:41:31 +00:00
|
|
|
}
|
|
|
|
|
2015-04-19 15:55:56 +00:00
|
|
|
float EffectAnimationTime::getValue(osg::NodeVisitor*)
|
2015-04-18 23:57:52 +00:00
|
|
|
{
|
|
|
|
return mTime;
|
|
|
|
}
|
|
|
|
|
2015-04-19 15:55:56 +00:00
|
|
|
void EffectAnimationTime::addTime(float duration)
|
2015-04-18 23:57:52 +00:00
|
|
|
{
|
|
|
|
mTime += duration;
|
|
|
|
}
|
|
|
|
|
2015-04-19 15:55:56 +00:00
|
|
|
void EffectAnimationTime::resetTime(float time)
|
2015-04-18 23:57:52 +00:00
|
|
|
{
|
|
|
|
mTime = time;
|
|
|
|
}
|
|
|
|
|
2015-04-19 15:55:56 +00:00
|
|
|
float EffectAnimationTime::getTime() const
|
2015-04-18 23:57:52 +00:00
|
|
|
{
|
|
|
|
return mTime;
|
|
|
|
}
|
|
|
|
|
2015-04-12 13:34:50 +00:00
|
|
|
// --------------------------------------------------------------------------------
|
2013-05-11 05:22:39 +00:00
|
|
|
|
2015-05-28 13:44:58 +00:00
|
|
|
ObjectAnimation::ObjectAnimation(const MWWorld::Ptr &ptr, const std::string &model, Resource::ResourceSystem* resourceSystem, bool animated, bool allowLight)
|
2015-04-12 13:34:50 +00:00
|
|
|
: Animation(ptr, osg::ref_ptr<osg::Group>(ptr.getRefData().getBaseNode()), resourceSystem)
|
2012-07-21 21:41:26 +00:00
|
|
|
{
|
2015-04-12 13:34:50 +00:00
|
|
|
if (!model.empty())
|
2013-05-12 11:34:37 +00:00
|
|
|
{
|
2015-05-21 21:24:22 +00:00
|
|
|
setObjectRoot(model, false, false);
|
2015-05-28 13:44:58 +00:00
|
|
|
if (animated)
|
|
|
|
addAnimSource(model);
|
2015-04-16 23:23:37 +00:00
|
|
|
|
|
|
|
if (!ptr.getClass().getEnchantment(ptr).empty())
|
|
|
|
addGlow(mObjectRoot, getEnchantmentColor(ptr));
|
2013-05-12 11:34:37 +00:00
|
|
|
}
|
2015-06-02 15:06:55 +00:00
|
|
|
if (ptr.getTypeName() == typeid(ESM::Light).name() && allowLight)
|
|
|
|
addExtraLight(getOrCreateObjectRoot(), ptr.get<ESM::Light>()->mBase);
|
2013-08-07 02:45:07 +00:00
|
|
|
}
|
|
|
|
|
2015-06-14 21:13:26 +00:00
|
|
|
Animation::AnimState::~AnimState()
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// ------------------------------
|
|
|
|
|
|
|
|
PartHolder::PartHolder(osg::ref_ptr<osg::Node> node)
|
|
|
|
: mNode(node)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
PartHolder::~PartHolder()
|
|
|
|
{
|
|
|
|
if (mNode->getNumParents())
|
|
|
|
mNode->getParent(0)->removeChild(mNode);
|
|
|
|
}
|
|
|
|
|
2012-02-20 13:02:24 +00:00
|
|
|
}
|