mirror of
https://gitlab.com/OpenMW/openmw.git
synced 2025-01-25 06:35:30 +00:00
29556a1802
A Warning indicates a potential problem in the content file(s) that the user told OpenMW to load. E.g. this might cause an object to not display at all or as intended, however the rest of the game will run fine. An Error, however, is more likely to be a bug with the engine itself - it means that basic assumptions have been violated and the engine might not run correctly anymore. The above mostly applies to errors/warnings during game-play; startup issues are handled differently: when a file is completely invalid/corrupted to the point that the engine can not start, that might cause messages that are worded as Error due to the severity of the issue but are not necessarily the engine's fault. Hopefully, being a little more consistent here will alleviate confusion among users as to when a log message should be reported and to whom.
206 lines
4.7 KiB
C++
206 lines
4.7 KiB
C++
#include "skeleton.hpp"
|
|
|
|
#include <osg/Transform>
|
|
#include <osg/MatrixTransform>
|
|
|
|
#include <components/misc/stringops.hpp>
|
|
|
|
#include <iostream>
|
|
|
|
namespace SceneUtil
|
|
{
|
|
|
|
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;
|
|
|
|
mCache[Misc::StringUtils::lowerCase(bone->getName())] = std::make_pair(getNodePath(), bone);
|
|
|
|
traverse(node);
|
|
}
|
|
private:
|
|
std::map<std::string, std::pair<osg::NodePath, osg::MatrixTransform*> >& mCache;
|
|
};
|
|
|
|
Skeleton::Skeleton()
|
|
: mBoneCacheInit(false)
|
|
, mNeedToUpdateBoneMatrices(true)
|
|
, mActive(true)
|
|
, mLastFrameNumber(0)
|
|
, mTraversedEvenFrame(false)
|
|
, mTraversedOddFrame(false)
|
|
{
|
|
|
|
}
|
|
|
|
Skeleton::Skeleton(const Skeleton ©, const osg::CopyOp ©op)
|
|
: osg::Group(copy, copyop)
|
|
, mBoneCacheInit(false)
|
|
, mNeedToUpdateBoneMatrices(true)
|
|
, mActive(copy.mActive)
|
|
, mLastFrameNumber(0)
|
|
, mTraversedEvenFrame(false)
|
|
, mTraversedOddFrame(false)
|
|
{
|
|
|
|
}
|
|
|
|
Bone* Skeleton::getBone(const std::string &name)
|
|
{
|
|
if (!mBoneCacheInit)
|
|
{
|
|
InitBoneCacheVisitor visitor(mBoneCache);
|
|
accept(visitor);
|
|
mBoneCacheInit = true;
|
|
}
|
|
|
|
BoneCache::iterator found = mBoneCache.find(Misc::StringUtils::lowerCase(name));
|
|
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;
|
|
}
|
|
|
|
void Skeleton::updateBoneMatrices(unsigned int traversalNumber)
|
|
{
|
|
if (traversalNumber != mLastFrameNumber)
|
|
mNeedToUpdateBoneMatrices = true;
|
|
|
|
mLastFrameNumber = traversalNumber;
|
|
|
|
if (mLastFrameNumber % 2 == 0)
|
|
mTraversedEvenFrame = true;
|
|
else
|
|
mTraversedOddFrame = true;
|
|
|
|
if (mNeedToUpdateBoneMatrices)
|
|
{
|
|
if (mRootBone.get())
|
|
{
|
|
for (unsigned int i=0; i<mRootBone->mChildren.size(); ++i)
|
|
mRootBone->mChildren[i]->update(NULL);
|
|
}
|
|
else
|
|
std::cerr << "Error: no root bone" << std::endl;
|
|
|
|
mNeedToUpdateBoneMatrices = false;
|
|
}
|
|
}
|
|
|
|
void Skeleton::setActive(bool active)
|
|
{
|
|
mActive = active;
|
|
}
|
|
|
|
bool Skeleton::getActive() const
|
|
{
|
|
return mActive;
|
|
}
|
|
|
|
void Skeleton::markDirty()
|
|
{
|
|
mTraversedEvenFrame = false;
|
|
mTraversedOddFrame = false;
|
|
mBoneCache.clear();
|
|
mBoneCacheInit = false;
|
|
}
|
|
|
|
void Skeleton::traverse(osg::NodeVisitor& nv)
|
|
{
|
|
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
|
|
&& mLastFrameNumber != 0 && mTraversedEvenFrame && mTraversedOddFrame)
|
|
return;
|
|
osg::Group::traverse(nv);
|
|
}
|
|
|
|
void Skeleton::childInserted(unsigned int)
|
|
{
|
|
markDirty();
|
|
}
|
|
|
|
void Skeleton::childRemoved(unsigned int, unsigned int)
|
|
{
|
|
markDirty();
|
|
}
|
|
|
|
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)
|
|
{
|
|
std::cerr << "Error: Bone without node " << std::endl;
|
|
return;
|
|
}
|
|
if (parentMatrixInSkeletonSpace)
|
|
mMatrixInSkeletonSpace = mNode->getMatrix() * (*parentMatrixInSkeletonSpace);
|
|
else
|
|
mMatrixInSkeletonSpace = mNode->getMatrix();
|
|
|
|
for (unsigned int i=0; i<mChildren.size(); ++i)
|
|
{
|
|
mChildren[i]->update(&mMatrixInSkeletonSpace);
|
|
}
|
|
}
|
|
|
|
}
|