2015-04-21 14:02:40 +00:00
|
|
|
#ifndef OPENMW_COMPONENTS_NIFOSG_SKELETON_H
|
|
|
|
#define OPENMW_COMPONENTS_NIFOSG_SKELETON_H
|
|
|
|
|
|
|
|
#include <osg/Group>
|
|
|
|
|
|
|
|
#include <memory>
|
2021-10-09 09:14:22 +00:00
|
|
|
#include <unordered_map>
|
2015-04-21 14:02:40 +00:00
|
|
|
|
2015-04-21 18:30:48 +00:00
|
|
|
namespace SceneUtil
|
2015-04-21 14:02:40 +00:00
|
|
|
{
|
|
|
|
|
2015-04-21 18:42:50 +00:00
|
|
|
/// @brief Defines a Bone hierarchy, used for updating of skeleton-space bone matrices.
|
|
|
|
/// @note To prevent unnecessary updates, only bones that are used for skinning will be added to this hierarchy.
|
2015-04-21 14:02:40 +00:00
|
|
|
class Bone
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Bone();
|
|
|
|
|
2015-04-22 15:34:39 +00:00
|
|
|
osg::Matrixf mMatrixInSkeletonSpace;
|
2015-04-21 14:02:40 +00:00
|
|
|
|
|
|
|
osg::MatrixTransform* mNode;
|
|
|
|
|
2022-07-17 10:12:17 +00:00
|
|
|
std::vector<std::unique_ptr<Bone>> mChildren;
|
2015-04-21 14:02:40 +00:00
|
|
|
|
2015-04-21 18:42:50 +00:00
|
|
|
/// Update the skeleton-space matrix of this bone and all its children.
|
2015-04-21 14:02:40 +00:00
|
|
|
void update(const osg::Matrixf* parentMatrixInSkeletonSpace);
|
|
|
|
};
|
|
|
|
|
2015-04-21 18:42:50 +00:00
|
|
|
/// @brief Handles the bone matrices for any number of child RigGeometries.
|
|
|
|
/// @par Bones should be created as osg::MatrixTransform children of the skeleton.
|
|
|
|
/// To be a referenced by a RigGeometry, a bone needs to have a unique name.
|
2015-04-21 14:02:40 +00:00
|
|
|
class Skeleton : public osg::Group
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Skeleton();
|
|
|
|
Skeleton(const Skeleton& copy, const osg::CopyOp& copyop);
|
|
|
|
|
2015-12-06 14:27:43 +00:00
|
|
|
META_Node(SceneUtil, Skeleton)
|
2015-04-21 14:02:40 +00:00
|
|
|
|
2022-10-05 22:50:36 +00:00
|
|
|
/// Retrieve a bone by name.
|
|
|
|
Bone* getBone(const std::string& name);
|
2015-04-21 18:42:50 +00:00
|
|
|
|
|
|
|
/// Request an update of bone matrices. May be a no-op if already updated in this frame.
|
2016-07-02 17:26:11 +00:00
|
|
|
void updateBoneMatrices(unsigned int traversalNumber);
|
2015-04-21 14:02:40 +00:00
|
|
|
|
2018-01-11 01:49:35 +00:00
|
|
|
enum ActiveType
|
|
|
|
{
|
|
|
|
Inactive = 0,
|
|
|
|
SemiActive, /// Like Active, but don't bother with Update (including new bounding box) if we're off-screen
|
|
|
|
Active
|
|
|
|
};
|
|
|
|
|
2015-04-29 21:48:08 +00:00
|
|
|
/// Set the skinning active flag. Inactive skeletons will not have their child rigs updated.
|
|
|
|
/// You should set this flag to false if you know that bones are not currently moving.
|
2018-01-11 01:49:35 +00:00
|
|
|
void setActive(ActiveType active);
|
2015-04-29 21:48:08 +00:00
|
|
|
|
|
|
|
bool getActive() const;
|
|
|
|
|
2020-10-16 18:18:54 +00:00
|
|
|
void traverse(osg::NodeVisitor& nv) override;
|
2016-12-16 19:09:20 +00:00
|
|
|
|
2016-07-02 18:19:55 +00:00
|
|
|
void markDirty();
|
|
|
|
|
2020-10-16 18:18:54 +00:00
|
|
|
void childInserted(unsigned int) override;
|
|
|
|
void childRemoved(unsigned int, unsigned int) override;
|
2015-06-30 01:25:30 +00:00
|
|
|
|
2015-04-21 14:02:40 +00:00
|
|
|
private:
|
|
|
|
// The root bone is not a "real" bone, it has no corresponding node in the scene graph.
|
|
|
|
// As far as the scene graph goes we support multiple root bones.
|
2017-04-28 15:30:26 +00:00
|
|
|
std::unique_ptr<Bone> mRootBone;
|
2015-04-21 14:02:40 +00:00
|
|
|
|
2021-10-09 09:14:22 +00:00
|
|
|
typedef std::unordered_map<std::string, std::vector<osg::MatrixTransform*>> BoneCache;
|
2015-04-21 14:02:40 +00:00
|
|
|
BoneCache mBoneCache;
|
|
|
|
bool mBoneCacheInit;
|
|
|
|
|
|
|
|
bool mNeedToUpdateBoneMatrices;
|
2015-04-21 18:22:32 +00:00
|
|
|
|
2018-01-11 01:49:35 +00:00
|
|
|
ActiveType mActive;
|
2015-04-29 21:48:08 +00:00
|
|
|
|
2015-04-21 18:22:32 +00:00
|
|
|
unsigned int mLastFrameNumber;
|
2018-01-11 01:49:35 +00:00
|
|
|
unsigned int mLastCullFrameNumber;
|
2015-04-21 14:02:40 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|