1
0
Fork 0
mirror of https://github.com/OpenMW/openmw.git synced 2025-01-22 04:53:52 +00:00
openmw/components/resource/keyframemanager.cpp

259 lines
10 KiB
C++
Raw Normal View History

#include "keyframemanager.hpp"
2023-07-28 22:29:03 +00:00
#include <array>
#include <components/vfs/manager.hpp>
2020-11-18 23:11:56 +00:00
#include <osgAnimation/Animation>
#include <osgAnimation/BasicAnimationManager>
#include <osgAnimation/Channel>
2022-07-21 11:51:34 +00:00
#include <components/debug/debuglog.hpp>
2022-09-22 18:26:05 +00:00
#include <components/misc/pathhelpers.hpp>
#include <components/misc/strings/algorithm.hpp>
2023-03-18 09:30:48 +00:00
#include <components/misc/strings/conversion.hpp>
#include <components/nifosg/nifloader.hpp>
2020-11-18 23:11:56 +00:00
#include <components/sceneutil/keyframe.hpp>
#include <components/sceneutil/osgacontroller.hpp>
2023-05-31 21:11:03 +00:00
#include <components/vfs/pathutil.hpp>
2020-11-18 23:11:56 +00:00
#include "animation.hpp"
#include "objectcache.hpp"
#include "scenemanager.hpp"
namespace Resource
{
2022-09-22 18:26:05 +00: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)
2023-07-29 08:58:27 +00:00
, mAnimationManager(std::move(animationManager))
2022-09-22 18:26:05 +00:00
, mNormalized(normalized)
, mVFS(vfs)
{
}
2022-10-14 11:32:20 +00:00
bool RetrieveAnimationsVisitor::belongsToLeftUpperExtremity(const std::string& name)
{
static const std::array boneNames = { "bip01 l clavicle", "left clavicle", "bip01 l upperarm", "left upper arm",
"bip01 l forearm", "bip01 l hand", "left hand", "left wrist", "shield bone", "bip01 l pinky1",
"bip01 l pinky2", "bip01 l pinky3", "bip01 l ring1", "bip01 l ring2", "bip01 l ring3", "bip01 l middle1",
"bip01 l middle2", "bip01 l middle3", "bip01 l pointer1", "bip01 l pointer2", "bip01 l pointer3",
"bip01 l thumb1", "bip01 l thumb2", "bip01 l thumb3", "left forearm" };
2022-10-14 11:32:20 +00:00
2022-12-30 22:52:05 +00:00
if (std::find(boneNames.begin(), boneNames.end(), name) != boneNames.end())
return true;
2022-10-14 11:32:20 +00:00
return false;
}
2022-10-14 11:32:20 +00:00
bool RetrieveAnimationsVisitor::belongsToRightUpperExtremity(const std::string& name)
2020-11-18 23:11:56 +00:00
{
static const std::array boneNames = { "bip01 r clavicle", "right clavicle", "bip01 r upperarm",
"right upper arm", "bip01 r forearm", "bip01 r hand", "right hand", "right wrist", "bip01 r thumb1",
"bip01 r thumb2", "bip01 r thumb3", "weapon bone", "bip01 r pinky1", "bip01 r pinky2", "bip01 r pinky3",
"bip01 r ring1", "bip01 r ring2", "bip01 r ring3", "bip01 r middle1", "bip01 r middle2", "bip01 r middle3",
"bip01 r pointer1", "bip01 r pointer2", "bip01 r pointer3", "right forearm" };
2022-10-14 11:32:20 +00:00
2022-12-30 22:52:05 +00:00
if (std::find(boneNames.begin(), boneNames.end(), name) != boneNames.end())
return true;
2022-10-14 11:32:20 +00:00
return false;
}
2020-11-18 23:11:56 +00:00
bool RetrieveAnimationsVisitor::belongsToTorso(const std::string& name)
{
2022-11-14 12:49:35 +00:00
static const std::array boneNames
= { "bip01 spine1", "bip01 spine2", "bip01 neck", "bip01 head", "head", "neck", "chest", "groin" };
2022-10-14 11:32:20 +00:00
2022-12-30 22:52:05 +00:00
if (std::find(boneNames.begin(), boneNames.end(), name) != boneNames.end())
return true;
2022-10-14 11:32:20 +00:00
return false;
}
2020-11-18 23:11:56 +00:00
void RetrieveAnimationsVisitor::addKeyframeController(const std::string& name, const osg::Node& node)
{
osg::ref_ptr<SceneUtil::OsgAnimationController> callback = new SceneUtil::OsgAnimationController();
callback->setName(name);
std::vector<SceneUtil::EmulatedAnimation> emulatedAnimations;
for (const auto& animation : mAnimationManager->getAnimationList())
{
if (animation)
2022-09-22 18:26:05 +00:00
{
2022-12-30 22:33:35 +00:00
//"Default" is osg dae plugin's default naming scheme for unnamed animations
if (animation->getName() == "Default")
{
2022-12-30 22:33:35 +00:00
animation->setName(std::string("idle"));
}
osg::ref_ptr<Resource::Animation> mergedAnimationTrack = new Resource::Animation;
const std::string animationName = animation->getName();
mergedAnimationTrack->setName(animationName);
2022-09-22 18:26:05 +00:00
const osgAnimation::ChannelList& channels = animation->getChannels();
for (const auto& channel : channels)
2022-10-14 11:32:20 +00:00
{
// Replace channel target name to match the renamed bones/transforms
channel->setTargetName(Misc::StringUtils::underscoresToSpaces(channel->getTargetName()));
if (name == "Bip01 R Clavicle")
{
if (!belongsToRightUpperExtremity(channel->getTargetName()))
continue;
}
else if (name == "Bip01 L Clavicle")
{
if (!belongsToLeftUpperExtremity(channel->getTargetName()))
continue;
}
else if (name == "Bip01 Spine1")
{
if (!belongsToTorso(channel->getTargetName()))
continue;
}
else if (belongsToRightUpperExtremity(channel->getTargetName())
|| belongsToLeftUpperExtremity(channel->getTargetName())
|| belongsToTorso(channel->getTargetName()))
2022-10-14 11:32:20 +00:00
continue;
2022-09-22 18:26:05 +00:00
2022-12-30 22:33:35 +00:00
mergedAnimationTrack->addChannel(channel.get()->clone());
}
2022-09-22 18:26:05 +00:00
2024-01-21 09:48:33 +00:00
callback->addMergedAnimationTrack(std::move(mergedAnimationTrack));
2022-09-22 18:26:05 +00:00
float startTime = animation->getStartTime();
float stopTime = startTime + animation->getDuration();
2022-09-22 18:26:05 +00:00
SceneUtil::EmulatedAnimation emulatedAnimation;
emulatedAnimation.mStartTime = startTime;
emulatedAnimation.mStopTime = stopTime;
emulatedAnimation.mName = animationName;
emulatedAnimations.emplace_back(emulatedAnimation);
2022-09-22 18:26:05 +00:00
}
}
// mTextKeys is a nif-thing, used by OpenMW's animation system
// Format is likely "AnimationName: [Keyword_optional] [Start OR Stop]"
// AnimationNames are keywords like idle2, idle3... AiPackages and various mechanics control which
// 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
// value e.g. idle: start 0.0333
try
{
Files::IStreamPtr textKeysFile = mVFS->get(changeFileExtension(mNormalized, "txt"));
std::string line;
while (getline(*textKeysFile, line))
2022-09-22 18:26:05 +00:00
{
mTarget.mTextKeys.emplace(parseTimeSignature(line), parseTextKey(line));
2020-11-18 23:11:56 +00:00
}
}
catch (std::exception&)
{
Log(Debug::Warning) << "No textkey file found for " << mNormalized;
}
callback->setEmulatedAnimations(emulatedAnimations);
mTarget.mKeyframeControllers.emplace(name, callback);
}
void RetrieveAnimationsVisitor::apply(osg::Node& node)
{
if (node.libraryName() == std::string_view("osgAnimation") && node.className() == std::string_view("Bone")
&& Misc::StringUtils::lowerCase(node.getName()) == std::string_view("bip01"))
{
addKeyframeController("bip01", node); /* Character root */
addKeyframeController("Bip01 Spine1", node); /* Torso */
addKeyframeController("Bip01 L Clavicle", node); /* Left arm */
addKeyframeController("Bip01 R Clavicle", node); /* Right arm */
2022-09-22 18:26:05 +00:00
}
2020-11-18 23:11:56 +00:00
traverse(node);
}
std::string RetrieveAnimationsVisitor::parseTextKey(const std::string& line)
{
size_t spacePos = line.find_last_of(' ');
if (spacePos != std::string::npos)
2022-09-22 18:26:05 +00:00
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())
2023-03-18 09:30:48 +00:00
time = Misc::StringUtils::toNumeric<double>(line.substr(spacePos + 1), time);
return time;
}
std::string RetrieveAnimationsVisitor::changeFileExtension(const std::string& file, const std::string& ext)
{
size_t extPos = file.find_last_of('.');
2022-09-22 18:26:05 +00:00
if (extPos != std::string::npos && extPos + 1 < file.size())
{
return file.substr(0, extPos + 1) + ext;
}
return file;
}
}
namespace Resource
{
2024-01-17 17:10:42 +00:00
KeyframeManager::KeyframeManager(const VFS::Manager* vfs, SceneManager* sceneManager, double expiryDelay,
const ToUTF8::StatelessUtf8Encoder* encoder)
: ResourceManager(vfs, expiryDelay)
, mSceneManager(sceneManager)
2024-01-16 19:56:58 +00:00
, mEncoder(encoder)
{
}
2022-09-22 18:26:05 +00:00
osg::ref_ptr<const SceneUtil::KeyframeHolder> KeyframeManager::get(const std::string& name)
{
2023-05-31 21:11:03 +00:00
const std::string normalized = VFS::Path::normalizeFilename(name);
osg::ref_ptr<osg::Object> obj = mCache->getRefFromObjectCache(normalized);
if (obj)
return osg::ref_ptr<const SceneUtil::KeyframeHolder>(static_cast<SceneUtil::KeyframeHolder*>(obj.get()));
else
{
2022-09-22 18:26:05 +00:00
osg::ref_ptr<SceneUtil::KeyframeHolder> loaded(new SceneUtil::KeyframeHolder);
if (Misc::getFileExtension(normalized) == "kf")
{
auto file = std::make_shared<Nif::NIFFile>(normalized);
2024-01-16 19:56:58 +00:00
Nif::Reader reader(*file, mEncoder);
reader.parse(mVFS->getNormalized(normalized));
NifOsg::Loader::loadKf(*file, *loaded.get());
}
else
{
2022-09-22 18:26:05 +00: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 17:46:08 +00:00
if (bam)
{
2024-01-21 09:48:33 +00:00
Resource::RetrieveAnimationsVisitor rav(*loaded.get(), std::move(bam), normalized, mVFS);
2020-11-20 17:46:08 +00:00
scene->accept(rav);
}
}
2015-12-14 14:11:06 +00:00
mCache->addEntryToObjectCache(normalized, loaded);
return loaded;
}
}
2022-09-22 18:26:05 +00:00
void KeyframeManager::reportStats(unsigned int frameNumber, osg::Stats* stats) const
{
2023-12-21 23:23:49 +00:00
Resource::reportStats("Keyframe", frameNumber, mCache->getStats(), *stats);
}
}