1
0
Fork 0
mirror of https://github.com/OpenMW/openmw.git synced 2025-01-16 16:29:55 +00:00
openmw/components/sceneutil/osgacontroller.cpp

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

233 lines
8.3 KiB
C++
Raw Normal View History

2020-11-18 23:11:56 +00:00
#include <components/sceneutil/osgacontroller.hpp>
#include <osg/MatrixTransform>
2020-11-18 23:11:56 +00:00
#include <osg/Node>
#include <osg/NodeVisitor>
#include <osg/ref_ptr>
#include <osgAnimation/Animation>
#include <osgAnimation/Channel>
#include <osgAnimation/Sampler>
#include <osgAnimation/UpdateMatrixTransform>
#include <components/misc/strings/algorithm.hpp>
#include <components/misc/strings/lower.hpp>
2020-11-18 23:11:56 +00:00
#include <components/resource/animation.hpp>
#include <components/sceneutil/controller.hpp>
#include <components/sceneutil/keyframe.hpp>
namespace SceneUtil
2020-11-18 23:11:56 +00:00
{
LinkVisitor::LinkVisitor()
: osg::NodeVisitor(TRAVERSE_ALL_CHILDREN)
{
mAnimation = nullptr;
}
void LinkVisitor::link(osgAnimation::UpdateMatrixTransform* umt)
{
// If osgAnimation had underscores, we should update the umt name also
// otherwise the animation channel and updates wont be applied
umt->setName(Misc::StringUtils::underscoresToSpaces(umt->getName()));
2020-11-18 23:11:56 +00:00
const osgAnimation::ChannelList& channels = mAnimation->getChannels();
2020-12-12 20:23:20 +00:00
for (const auto& channel : channels)
2020-11-18 23:11:56 +00:00
{
const std::string& channelName = channel->getName();
const std::string& channelTargetName = channel->getTargetName();
if (channelTargetName != umt->getName())
continue;
// check if we can link a StackedTransformElement to the current Channel
for (const auto& stackedTransform : umt->getStackedTransforms())
2020-11-18 23:11:56 +00:00
{
osgAnimation::StackedTransformElement* element = stackedTransform.get();
if (element && !element->getName().empty() && channelName == element->getName())
{
osgAnimation::Target* target = element->getOrCreateTarget();
if (target)
{
channel->setTarget(target);
}
}
}
}
}
void LinkVisitor::setAnimation(Resource::Animation* animation)
{
mAnimation = animation;
}
void LinkVisitor::apply(osg::Node& node)
{
osg::Callback* cb = node.getUpdateCallback();
while (cb)
{
osgAnimation::UpdateMatrixTransform* umt = dynamic_cast<osgAnimation::UpdateMatrixTransform*>(cb);
if (umt)
2021-02-02 15:34:02 +00:00
if (Misc::StringUtils::lowerCase(node.getName()) != "bip01")
link(umt);
2020-11-18 23:11:56 +00:00
cb = cb->getNestedCallback();
}
if (node.getNumChildrenRequiringUpdateTraversal())
traverse(node);
2020-11-18 23:11:56 +00:00
}
OsgAnimationController::OsgAnimationController(const OsgAnimationController& copy, const osg::CopyOp& copyop)
: osg::Object(copy, copyop)
, SceneUtil::KeyframeController(copy)
, SceneUtil::NodeCallback<OsgAnimationController>(copy, copyop)
, mEmulatedAnimations(copy.mEmulatedAnimations)
2020-11-18 23:11:56 +00:00
{
mLinker = nullptr;
2021-01-30 13:27:47 +00:00
for (const auto& mergedAnimationTrack : copy.mMergedAnimationTracks)
{
2021-02-02 16:00:14 +00:00
Resource::Animation* copiedAnimationTrack
= static_cast<Resource::Animation*>(mergedAnimationTrack.get()->clone(copyop));
2021-01-30 13:27:47 +00:00
mMergedAnimationTracks.emplace_back(copiedAnimationTrack);
}
2020-11-18 23:11:56 +00:00
}
osg::Vec3f OsgAnimationController::getTranslation(float time) const
2020-11-18 23:11:56 +00:00
{
osg::Vec3f translationValue;
std::string animationName;
float newTime = time;
// Find the correct animation based on time
2020-11-20 17:41:01 +00:00
for (const EmulatedAnimation& emulatedAnimation : mEmulatedAnimations)
2020-11-18 23:11:56 +00:00
{
2020-12-15 11:50:19 +00:00
if (time >= emulatedAnimation.mStartTime && time <= emulatedAnimation.mStopTime)
2020-11-18 23:11:56 +00:00
{
newTime = time - emulatedAnimation.mStartTime;
animationName = emulatedAnimation.mName;
}
}
// Find the root transform track in animation
2020-12-12 20:23:20 +00:00
for (const auto& mergedAnimationTrack : mMergedAnimationTracks)
2020-11-18 23:11:56 +00:00
{
if (mergedAnimationTrack->getName() != animationName)
continue;
const osgAnimation::ChannelList& channels = mergedAnimationTrack->getChannels();
2020-12-12 20:23:20 +00:00
for (const auto& channel : channels)
2020-11-18 23:11:56 +00:00
{
2020-12-15 11:51:49 +00:00
if (channel->getTargetName() != "bip01" || channel->getName() != "transform")
continue;
2020-11-18 23:11:56 +00:00
if (osgAnimation::MatrixLinearSampler* templateSampler
= dynamic_cast<osgAnimation::MatrixLinearSampler*>(channel->getSampler()))
{
osg::Matrixf matrix;
templateSampler->getValueAt(newTime, matrix);
translationValue = matrix.getTrans();
return osg::Vec3f(translationValue[0], translationValue[1], translationValue[2]);
}
}
}
return osg::Vec3f();
}
osg::Matrixf OsgAnimationController::getTransformForNode(float time, const std::string& name) const
{
std::string animationName;
float newTime = time;
// Find the correct animation based on time
for (const EmulatedAnimation& emulatedAnimation : mEmulatedAnimations)
{
if (time >= emulatedAnimation.mStartTime && time <= emulatedAnimation.mStopTime)
{
newTime = time - emulatedAnimation.mStartTime;
animationName = emulatedAnimation.mName;
}
}
// Find the bone's transform track in animation
for (const auto& mergedAnimationTrack : mMergedAnimationTracks)
{
if (mergedAnimationTrack->getName() != animationName)
continue;
const osgAnimation::ChannelList& channels = mergedAnimationTrack->getChannels();
for (const auto& channel : channels)
{
if (!Misc::StringUtils::ciEqual(name, channel->getTargetName()) || channel->getName() != "transform")
continue;
if (osgAnimation::MatrixLinearSampler* templateSampler
= dynamic_cast<osgAnimation::MatrixLinearSampler*>(channel->getSampler()))
{
osg::Matrixf matrix;
templateSampler->getValueAt(newTime, matrix);
return matrix;
}
}
}
return osg::Matrixf::identity();
}
2021-06-23 21:13:59 +00:00
void OsgAnimationController::update(float time, const std::string& animationName)
2020-11-18 23:11:56 +00:00
{
2020-12-12 20:23:20 +00:00
for (const auto& mergedAnimationTrack : mMergedAnimationTracks)
2020-11-18 23:11:56 +00:00
{
if (mergedAnimationTrack->getName() == animationName)
mergedAnimationTrack->update(time);
}
}
void OsgAnimationController::operator()(osg::Node* node, osg::NodeVisitor* nv)
2020-11-18 23:11:56 +00:00
{
if (hasInput())
{
if (mNeedToLink)
{
2020-12-12 20:23:20 +00:00
for (const auto& mergedAnimationTrack : mMergedAnimationTracks)
2020-11-18 23:11:56 +00:00
{
if (!mLinker.valid())
mLinker = new LinkVisitor();
mLinker->setAnimation(mergedAnimationTrack);
node->accept(*mLinker);
}
mNeedToLink = false;
}
float time = getInputValue(nv);
2020-11-20 17:41:01 +00:00
for (const EmulatedAnimation& emulatedAnimation : mEmulatedAnimations)
2020-11-18 23:11:56 +00:00
{
if (time > emulatedAnimation.mStartTime && time < emulatedAnimation.mStopTime)
{
update(time - emulatedAnimation.mStartTime, emulatedAnimation.mName);
}
}
// Reset the transform of this node to whats in the animation
// we force this here because downstream some code relies on the bone having a non-modified transform
// as this is how the NIF controller behaves. RotationController is a good example of this.
// Without this here, it causes osgAnimation skeletons to spin wildly
static_cast<osg::MatrixTransform*>(node)->setMatrix(getTransformForNode(time, node->getName()));
2020-11-18 23:11:56 +00:00
}
traverse(node, nv);
}
2021-06-23 21:13:59 +00:00
void OsgAnimationController::setEmulatedAnimations(const std::vector<EmulatedAnimation>& emulatedAnimations)
2020-11-18 23:11:56 +00:00
{
mEmulatedAnimations = emulatedAnimations;
}
void OsgAnimationController::addMergedAnimationTrack(osg::ref_ptr<Resource::Animation> animationTrack)
2020-11-18 23:11:56 +00:00
{
mMergedAnimationTracks.emplace_back(animationTrack);
}
}