#include "animation.hpp" #include #include #include #include #include #include #include "../mwbase/environment.hpp" #include "../mwbase/world.hpp" namespace MWRender { Animation::Animation(const MWWorld::Ptr &ptr) : mPtr(ptr) , mInsert(NULL) , mTime(0.0f) , mAnimState(NULL) , mSkipFrame(false) , mAccumRoot(NULL) , mNonAccumRoot(NULL) , mLastPosition(0.0f) { mCurGroup.mStart = mCurGroup.mLoopStart = 0.0f; mCurGroup.mLoopStop = mCurGroup.mStop = 0.0f; mNextGroup.mStart = mNextGroup.mLoopStart = 0.0f; mNextGroup.mLoopStop = mNextGroup.mStop = 0.0f; } Animation::~Animation() { if(mInsert) { Ogre::SceneManager *sceneMgr = mInsert->getCreator(); for(size_t i = 0;i < mEntityList.mEntities.size();i++) sceneMgr->destroyEntity(mEntityList.mEntities[i]); } mEntityList.mEntities.clear(); mEntityList.mSkelBase = NULL; } void Animation::createEntityList(Ogre::SceneNode *node, const std::string &model) { mInsert = node; assert(mInsert); mEntityList = NifOgre::NIFLoader::createEntities(mInsert, model); if(mEntityList.mSkelBase) { Ogre::AnimationStateSet *aset = mEntityList.mSkelBase->getAllAnimationStates(); Ogre::AnimationStateIterator as = aset->getAnimationStateIterator(); while(as.hasMoreElements()) { Ogre::AnimationState *state = as.getNext(); state->setEnabled(true); state->setLoop(false); if(!mAnimState) mAnimState = state; } Ogre::SkeletonInstance *skelinst = mEntityList.mSkelBase->getSkeleton(); // Would be nice if Ogre::SkeletonInstance allowed access to the 'master' Ogre::SkeletonPtr. Ogre::SkeletonManager &skelMgr = Ogre::SkeletonManager::getSingleton(); Ogre::SkeletonPtr skel = skelMgr.getByName(skelinst->getName()); Ogre::Skeleton::BoneIterator iter = skel->getBoneIterator(); while(iter.hasMoreElements()) { Ogre::Bone *bone = iter.getNext(); const Ogre::Any &data = bone->getUserObjectBindings().getUserAny(NifOgre::sTextKeyExtraDataID); if(!data.isEmpty()) { mTextKeys = Ogre::any_cast(data); mAccumRoot = skelinst->getRootBone(); mAccumRoot->setManuallyControlled(true); mNonAccumRoot = skelinst->getBone(bone->getHandle()); mLastPosition = mNonAccumRoot->getPosition(); break; } } } } void Animation::updatePosition(float time) { mAnimState->setTimePosition(time); if(mNonAccumRoot) { /* Update the animation and get the non-accumulation root's difference from the * last update. */ mEntityList.mSkelBase->getSkeleton()->setAnimationState(*mAnimState->getParent()); Ogre::Vector3 posdiff = mNonAccumRoot->getPosition() - mLastPosition; /* Translate the accumulation root back to compensate for the move. */ mAccumRoot->translate(-posdiff); mLastPosition += posdiff; /* Finally, move the object based on how much the non-accumulation root moved. */ Ogre::Vector3 newpos(mPtr.getRefData().getPosition().pos); newpos += mInsert->getOrientation() * posdiff; MWBase::World *world = MWBase::Environment::get().getWorld(); world->moveObject(mPtr, newpos.x, newpos.y, newpos.z); } } void Animation::resetPosition(float time) { mAnimState->setTimePosition(time); if(mNonAccumRoot) { mEntityList.mSkelBase->getSkeleton()->setAnimationState(*mAnimState->getParent()); mLastPosition = mNonAccumRoot->getPosition(); /* FIXME: This should be set to -mLastPosition, but without proper collision the * model gets placed halfway into the ground. */ mAccumRoot->setPosition(0.0f, 0.0f, 0.0f); } } struct checklow { bool operator()(const char &a, const char &b) const { return ::tolower(a) == ::tolower(b); } }; bool Animation::findGroupTimes(const std::string &groupname, Animation::GroupTimes *times) { const std::string &start = groupname+": start"; const std::string &startloop = groupname+": loop start"; const std::string &stop = groupname+": stop"; const std::string &stoploop = groupname+": loop stop"; NifOgre::TextKeyMap::const_iterator iter; for(iter = mTextKeys.begin();iter != mTextKeys.end();iter++) { if(times->mStart >= 0.0f && times->mLoopStart >= 0.0f && times->mLoopStop >= 0.0f && times->mStop >= 0.0f) return true; std::string::const_iterator strpos = iter->second.begin(); std::string::const_iterator strend = iter->second.end(); size_t strlen = strend-strpos; if(start.size() <= strlen && std::mismatch(strpos, strend, start.begin(), checklow()).first == strend) { times->mStart = iter->first; times->mLoopStart = iter->first; } else if(startloop.size() <= strlen && std::mismatch(strpos, strend, startloop.begin(), checklow()).first == strend) { times->mLoopStart = iter->first; } else if(stoploop.size() <= strlen && std::mismatch(strpos, strend, stoploop.begin(), checklow()).first == strend) { times->mLoopStop = iter->first; } else if(stop.size() <= strlen && std::mismatch(strpos, strend, stop.begin(), checklow()).first == strend) { times->mStop = iter->first; if(times->mLoopStop < 0.0f) times->mLoopStop = iter->first; break; } } return (times->mStart >= 0.0f && times->mLoopStart >= 0.0f && times->mLoopStop >= 0.0f && times->mStop >= 0.0f); } void Animation::playGroup(std::string groupname, int mode, int loops) { GroupTimes times; times.mLoops = loops; if(groupname == "all") { times.mStart = times.mLoopStart = 0.0f; times.mLoopStop = times.mStop = 0.0f; NifOgre::TextKeyMap::const_reverse_iterator iter = mTextKeys.rbegin(); if(iter != mTextKeys.rend()) times.mLoopStop = times.mStop = iter->first; } else if(!findGroupTimes(groupname, ×)) throw std::runtime_error("Failed to find animation group "+groupname); if(mode == 0 && mCurGroup.mLoops > 0) mNextGroup = times; else { mCurGroup = times; mNextGroup = GroupTimes(); mTime = ((mode==2) ? mCurGroup.mLoopStart : mCurGroup.mStart); resetPosition(mTime); } } void Animation::skipAnim() { mSkipFrame = true; } void Animation::runAnimation(float timepassed) { if(mAnimState && !mSkipFrame) { mTime += timepassed; recheck: if(mTime >= mCurGroup.mLoopStop) { if(mCurGroup.mLoops > 1) { mCurGroup.mLoops--; updatePosition(mCurGroup.mLoopStop); mTime = mTime - mCurGroup.mLoopStop + mCurGroup.mLoopStart; resetPosition(mCurGroup.mLoopStart); goto recheck; } else if(mTime >= mCurGroup.mStop) { if(mNextGroup.mLoops > 0) { updatePosition(mCurGroup.mStop); mTime = mTime - mCurGroup.mStop + mNextGroup.mStart; resetPosition(mNextGroup.mStart); mCurGroup = mNextGroup; mNextGroup = GroupTimes(); goto recheck; } mTime = mCurGroup.mStop; } } updatePosition(mTime); } mSkipFrame = false; } }