2015-12-12 18:52:06 +01:00
|
|
|
#include "keyframemanager.hpp"
|
|
|
|
|
|
|
|
#include <components/vfs/manager.hpp>
|
|
|
|
|
2021-10-05 12:21:12 +00:00
|
|
|
#include <osg/Stats>
|
2020-11-19 01:11:56 +02:00
|
|
|
#include <osgAnimation/Animation>
|
|
|
|
#include <osgAnimation/BasicAnimationManager>
|
|
|
|
#include <osgAnimation/Channel>
|
|
|
|
|
2022-07-21 15:51:34 +04:00
|
|
|
#include <components/debug/debuglog.hpp>
|
2021-09-11 15:49:47 +02:00
|
|
|
#include <components/misc/pathhelpers.hpp>
|
2022-08-03 00:00:54 +02:00
|
|
|
#include <components/misc/strings/algorithm.hpp>
|
2020-11-18 22:48:47 +02:00
|
|
|
#include <components/nifosg/nifloader.hpp>
|
2020-11-19 01:11:56 +02:00
|
|
|
#include <components/sceneutil/keyframe.hpp>
|
|
|
|
#include <components/sceneutil/osgacontroller.hpp>
|
2020-11-18 22:48:47 +02:00
|
|
|
|
2020-11-19 01:11:56 +02:00
|
|
|
#include "animation.hpp"
|
2015-12-12 18:52:06 +01:00
|
|
|
#include "objectcache.hpp"
|
2020-11-18 22:48:47 +02:00
|
|
|
#include "scenemanager.hpp"
|
|
|
|
|
2020-11-28 15:03:10 +02:00
|
|
|
namespace Resource
|
2020-11-18 22:48:47 +02:00
|
|
|
{
|
|
|
|
|
2021-02-04 23:14:21 +02:00
|
|
|
RetrieveAnimationsVisitor::RetrieveAnimationsVisitor(SceneUtil::KeyframeHolder& target,
|
|
|
|
osg::ref_ptr<osgAnimation::BasicAnimationManager> animationManager, const std::string& normalized,
|
|
|
|
const VFS::Manager* vfs)
|
|
|
|
: osg::NodeVisitor(TRAVERSE_ALL_CHILDREN)
|
|
|
|
, mTarget(target)
|
|
|
|
, mAnimationManager(animationManager)
|
|
|
|
, mNormalized(normalized)
|
|
|
|
, mVFS(vfs)
|
|
|
|
{
|
|
|
|
}
|
2020-11-18 22:48:47 +02:00
|
|
|
|
2020-11-19 01:11:56 +02:00
|
|
|
void RetrieveAnimationsVisitor::apply(osg::Node& node)
|
|
|
|
{
|
2022-08-02 23:57:09 +02:00
|
|
|
if (node.libraryName() == std::string_view("osgAnimation") && node.className() == std::string_view("Bone")
|
|
|
|
&& Misc::StringUtils::lowerCase(node.getName()) == std::string_view("bip01"))
|
2020-11-19 01:11:56 +02:00
|
|
|
{
|
2020-11-28 15:03:10 +02:00
|
|
|
osg::ref_ptr<SceneUtil::OsgAnimationController> callback = new SceneUtil::OsgAnimationController();
|
2020-11-19 01:11:56 +02:00
|
|
|
|
2020-11-28 15:03:10 +02:00
|
|
|
std::vector<SceneUtil::EmulatedAnimation> emulatedAnimations;
|
2020-11-19 01:11:56 +02:00
|
|
|
|
2021-08-29 15:15:45 +02:00
|
|
|
for (const auto& animation : mAnimationManager->getAnimationList())
|
2022-09-22 21:26:05 +03:00
|
|
|
{
|
2020-11-19 01:11:56 +02:00
|
|
|
if (animation)
|
|
|
|
{
|
|
|
|
if (animation->getName()
|
|
|
|
== "Default") //"Default" is osg dae plugin's default naming scheme for unnamed animations
|
|
|
|
{
|
|
|
|
animation->setName(
|
|
|
|
std::string("idle")); // animation naming scheme "idle: start" and "idle: stop" is the
|
|
|
|
// default idle animation that OpenMW seems to want to play
|
|
|
|
}
|
2021-02-04 23:14:21 +02:00
|
|
|
|
|
|
|
osg::ref_ptr<Resource::Animation> mergedAnimationTrack = new Resource::Animation;
|
|
|
|
const std::string animationName = animation->getName();
|
|
|
|
mergedAnimationTrack->setName(animationName);
|
2022-09-22 21:26:05 +03:00
|
|
|
|
2021-02-04 23:14:21 +02:00
|
|
|
const osgAnimation::ChannelList& channels = animation->getChannels();
|
|
|
|
for (const auto& channel : channels)
|
|
|
|
{
|
2021-02-12 19:36:03 +01:00
|
|
|
mergedAnimationTrack->addChannel(channel.get()->clone()); // is ->clone needed?
|
2021-02-04 23:14:21 +02:00
|
|
|
}
|
2022-09-22 21:26:05 +03:00
|
|
|
|
2020-11-19 01:11:56 +02:00
|
|
|
callback->addMergedAnimationTrack(mergedAnimationTrack);
|
2022-09-22 21:26:05 +03:00
|
|
|
|
2020-11-19 01:11:56 +02:00
|
|
|
float startTime = animation->getStartTime();
|
|
|
|
float stopTime = startTime + animation->getDuration();
|
2022-09-22 21:26:05 +03:00
|
|
|
|
2020-11-28 15:03:10 +02:00
|
|
|
SceneUtil::EmulatedAnimation emulatedAnimation;
|
2020-11-19 01:11:56 +02:00
|
|
|
emulatedAnimation.mStartTime = startTime;
|
|
|
|
emulatedAnimation.mStopTime = stopTime;
|
|
|
|
emulatedAnimation.mName = animationName;
|
|
|
|
emulatedAnimations.emplace_back(emulatedAnimation);
|
2021-02-04 23:14:21 +02:00
|
|
|
}
|
2022-09-22 21:26:05 +03:00
|
|
|
}
|
|
|
|
|
2021-04-19 15:43:00 +04:00
|
|
|
// mTextKeys is a nif-thing, used by OpenMW's animation system
|
|
|
|
// Format is likely "AnimationName: [Keyword_optional] [Start OR Stop]"
|
2021-02-04 23:14:21 +02:00
|
|
|
// AnimationNames are keywords like idle2, idle3... AiPackages and various mechanics control which
|
2021-04-19 15:43:00 +04:00
|
|
|
// animations are played Keywords can be stuff like Loop, Equip, Unequip, Block, InventoryHandtoHand,
|
|
|
|
// InventoryWeaponOneHand, PickProbe, Slash, Thrust, Chop... even "Slash Small Follow" osgAnimation formats
|
|
|
|
// should have a .txt file with the same name, each line holding a textkey and whitespace separated time
|
2021-02-04 23:14:21 +02:00
|
|
|
// value e.g. idle: start 0.0333
|
2022-09-22 21:26:05 +03:00
|
|
|
try
|
|
|
|
{
|
2021-02-04 23:14:21 +02:00
|
|
|
Files::IStreamPtr textKeysFile = mVFS->get(changeFileExtension(mNormalized, "txt"));
|
2021-04-19 15:43:00 +04:00
|
|
|
std::string line;
|
|
|
|
while (getline(*textKeysFile, line))
|
2021-02-04 23:14:21 +02:00
|
|
|
{
|
|
|
|
mTarget.mTextKeys.emplace(parseTimeSignature(line), parseTextKey(line));
|
|
|
|
}
|
2022-09-22 21:26:05 +03:00
|
|
|
}
|
2020-11-19 01:11:56 +02:00
|
|
|
catch (std::exception&)
|
2022-09-22 21:26:05 +03:00
|
|
|
{
|
2020-11-19 01:11:56 +02:00
|
|
|
Log(Debug::Warning) << "No textkey file found for " << mNormalized;
|
|
|
|
}
|
2020-11-18 22:48:47 +02:00
|
|
|
|
2020-11-19 01:11:56 +02:00
|
|
|
callback->setEmulatedAnimations(emulatedAnimations);
|
|
|
|
mTarget.mKeyframeControllers.emplace(node.getName(), callback);
|
2022-09-22 21:26:05 +03:00
|
|
|
}
|
|
|
|
|
2020-11-19 01:11:56 +02:00
|
|
|
traverse(node);
|
|
|
|
}
|
2021-02-04 23:14:21 +02:00
|
|
|
|
|
|
|
std::string RetrieveAnimationsVisitor::parseTextKey(const std::string& line)
|
|
|
|
{
|
|
|
|
size_t spacePos = line.find_last_of(' ');
|
|
|
|
if (spacePos != std::string::npos)
|
|
|
|
return line.substr(0, spacePos);
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
|
|
|
double RetrieveAnimationsVisitor::parseTimeSignature(const std::string& line)
|
|
|
|
{
|
|
|
|
size_t spacePos = line.find_last_of(' ');
|
|
|
|
double time = 0.0;
|
|
|
|
if (spacePos != std::string::npos && spacePos + 1 < line.size())
|
|
|
|
time = std::stod(line.substr(spacePos + 1));
|
|
|
|
return time;
|
|
|
|
}
|
|
|
|
|
2021-06-23 21:56:08 +02:00
|
|
|
std::string RetrieveAnimationsVisitor::changeFileExtension(const std::string& file, const std::string& ext)
|
2021-02-04 23:14:21 +02:00
|
|
|
{
|
|
|
|
size_t extPos = file.find_last_of('.');
|
|
|
|
if (extPos != std::string::npos && extPos + 1 < file.size())
|
|
|
|
{
|
|
|
|
return file.substr(0, extPos + 1) + ext;
|
|
|
|
}
|
|
|
|
return file;
|
|
|
|
}
|
|
|
|
|
2020-11-18 22:48:47 +02:00
|
|
|
}
|
2015-12-12 18:52:06 +01:00
|
|
|
|
|
|
|
namespace Resource
|
|
|
|
{
|
|
|
|
|
2020-11-18 22:48:47 +02:00
|
|
|
KeyframeManager::KeyframeManager(const VFS::Manager* vfs, SceneManager* sceneManager)
|
2016-02-06 16:57:54 +01:00
|
|
|
: ResourceManager(vfs)
|
2020-11-18 22:48:47 +02:00
|
|
|
, mSceneManager(sceneManager)
|
2015-12-12 18:52:06 +01:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
KeyframeManager::~KeyframeManager() {}
|
|
|
|
|
2020-11-18 22:48:47 +02:00
|
|
|
osg::ref_ptr<const SceneUtil::KeyframeHolder> KeyframeManager::get(const std::string& name)
|
2015-12-12 18:52:06 +01:00
|
|
|
{
|
2021-09-06 22:01:41 +02:00
|
|
|
const std::string normalized = mVFS->normalizeFilename(name);
|
2015-12-12 18:52:06 +01:00
|
|
|
|
|
|
|
osg::ref_ptr<osg::Object> obj = mCache->getRefFromObjectCache(normalized);
|
|
|
|
if (obj)
|
2020-11-18 22:48:47 +02:00
|
|
|
return osg::ref_ptr<const SceneUtil::KeyframeHolder>(static_cast<SceneUtil::KeyframeHolder*>(obj.get()));
|
2015-12-12 18:52:06 +01:00
|
|
|
else
|
|
|
|
{
|
2020-11-18 22:48:47 +02:00
|
|
|
osg::ref_ptr<SceneUtil::KeyframeHolder> loaded(new SceneUtil::KeyframeHolder);
|
2021-09-11 15:49:47 +02:00
|
|
|
if (Misc::getFileExtension(normalized) == "kf")
|
2020-11-18 22:48:47 +02:00
|
|
|
{
|
2022-09-17 19:24:42 +02:00
|
|
|
auto file = std::make_shared<Nif::NIFFile>(normalized);
|
|
|
|
Nif::Reader reader(*file);
|
|
|
|
reader.parse(mVFS->getNormalized(normalized));
|
|
|
|
NifOsg::Loader::loadKf(std::move(file), *loaded.get());
|
2020-11-18 22:48:47 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-11-19 01:11:56 +02:00
|
|
|
osg::ref_ptr<osg::Node> scene = const_cast<osg::Node*>(mSceneManager->getTemplate(normalized).get());
|
|
|
|
osg::ref_ptr<osgAnimation::BasicAnimationManager> bam
|
|
|
|
= dynamic_cast<osgAnimation::BasicAnimationManager*>(scene->getUpdateCallback());
|
2020-11-20 19:46:08 +02:00
|
|
|
if (bam)
|
|
|
|
{
|
2021-02-04 23:14:21 +02:00
|
|
|
Resource::RetrieveAnimationsVisitor rav(*loaded.get(), bam, normalized, mVFS);
|
2020-11-20 19:46:08 +02:00
|
|
|
scene->accept(rav);
|
|
|
|
}
|
2020-11-18 22:48:47 +02:00
|
|
|
}
|
2015-12-14 15:11:06 +01:00
|
|
|
mCache->addEntryToObjectCache(normalized, loaded);
|
2015-12-12 18:52:06 +01:00
|
|
|
return loaded;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-07 04:02:06 +01:00
|
|
|
void KeyframeManager::reportStats(unsigned int frameNumber, osg::Stats* stats) const
|
2017-02-22 02:18:18 +01:00
|
|
|
{
|
|
|
|
stats->setAttribute(frameNumber, "Keyframe", mCache->getCacheSize());
|
|
|
|
}
|
|
|
|
|
2015-12-12 18:52:06 +01:00
|
|
|
}
|