2015-04-21 14:02:40 +00:00
|
|
|
#ifndef OPENMW_COMPONENTS_NIFOSG_RIGGEOMETRY_H
|
|
|
|
#define OPENMW_COMPONENTS_NIFOSG_RIGGEOMETRY_H
|
|
|
|
|
|
|
|
#include <osg/Geometry>
|
|
|
|
#include <osg/Matrixf>
|
|
|
|
|
2015-04-21 18:30:48 +00:00
|
|
|
namespace SceneUtil
|
2015-04-21 14:02:40 +00:00
|
|
|
{
|
|
|
|
class Skeleton;
|
|
|
|
class Bone;
|
|
|
|
|
2021-10-23 08:31:46 +00:00
|
|
|
// TODO: This class has a lot of issues.
|
|
|
|
// - We require too many workarounds to ensure safety.
|
|
|
|
// - mSourceGeometry should be const, but can not be const because of a use case in shadervisitor.cpp.
|
|
|
|
// - We create useless mGeometry clones in template RigGeometries.
|
|
|
|
// - We do not support compileGLObjects.
|
2022-09-22 18:26:05 +00:00
|
|
|
// - We duplicate some code in MorphGeometry.
|
2021-10-23 08:31:46 +00:00
|
|
|
|
2015-04-21 18:42:50 +00:00
|
|
|
/// @brief Mesh skinning implementation.
|
|
|
|
/// @note A RigGeometry may be attached directly to a Skeleton, or somewhere below a Skeleton.
|
2022-09-22 18:26:05 +00:00
|
|
|
/// Note though that the RigGeometry ignores any transforms below the Skeleton, so the attachment point is not that
|
|
|
|
/// important.
|
|
|
|
/// @note The internal Geometry used for rendering is double buffered, this allows updates to be done in a thread
|
|
|
|
/// safe way while not compromising rendering performance. This is crucial when using osg's default threading model
|
|
|
|
/// of DrawThreadPerContext.
|
2017-09-01 16:27:00 +00:00
|
|
|
class RigGeometry : public osg::Drawable
|
2015-04-21 14:02:40 +00:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
RigGeometry();
|
|
|
|
RigGeometry(const RigGeometry& copy, const osg::CopyOp& copyop);
|
|
|
|
|
2015-12-06 14:27:43 +00:00
|
|
|
META_Object(SceneUtil, RigGeometry)
|
2015-04-21 14:02:40 +00:00
|
|
|
|
2022-09-22 18:26:05 +00:00
|
|
|
// Currently empty as this is difficult to implement. Technically we would need to compile both internal
|
|
|
|
// geometries in separate frames but this method is only called once. Alternatively we could compile just the
|
|
|
|
// static parts of the model.
|
2020-10-16 18:18:54 +00:00
|
|
|
void compileGLObjects(osg::RenderInfo& renderInfo) const override {}
|
2017-09-01 16:27:00 +00:00
|
|
|
|
2015-04-21 14:02:40 +00:00
|
|
|
struct BoneInfluence
|
|
|
|
{
|
|
|
|
osg::Matrixf mInvBindMatrix;
|
2015-04-27 13:41:34 +00:00
|
|
|
osg::BoundingSpheref mBoundSphere;
|
2020-10-08 23:24:28 +00:00
|
|
|
// <vertex index, weight>
|
|
|
|
std::vector<std::pair<unsigned short, float>> mWeights;
|
2015-04-21 14:02:40 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
struct InfluenceMap : public osg::Referenced
|
|
|
|
{
|
2020-10-08 23:24:28 +00:00
|
|
|
std::vector<std::pair<std::string, BoneInfluence>> mData;
|
2015-04-21 14:02:40 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
void setInfluenceMap(osg::ref_ptr<InfluenceMap> influenceMap);
|
|
|
|
|
2016-03-11 18:18:51 +00:00
|
|
|
/// Initialize this geometry from the source geometry.
|
|
|
|
/// @note The source geometry will not be modified.
|
2015-04-21 14:02:40 +00:00
|
|
|
void setSourceGeometry(osg::ref_ptr<osg::Geometry> sourceGeom);
|
|
|
|
|
2019-06-13 13:37:00 +00:00
|
|
|
osg::ref_ptr<osg::Geometry> getSourceGeometry() const;
|
2016-03-23 15:48:41 +00:00
|
|
|
|
2022-09-22 18:26:05 +00:00
|
|
|
void accept(osg::NodeVisitor& nv) override;
|
|
|
|
bool supports(const osg::PrimitiveFunctor&) const override { return true; }
|
2020-10-16 18:18:54 +00:00
|
|
|
void accept(osg::PrimitiveFunctor&) const override;
|
2015-04-21 14:02:40 +00:00
|
|
|
|
2018-08-08 22:24:57 +00:00
|
|
|
struct CopyBoundingBoxCallback : osg::Drawable::ComputeBoundingBoxCallback
|
|
|
|
{
|
|
|
|
osg::BoundingBox boundingBox;
|
|
|
|
|
2020-10-16 18:18:54 +00:00
|
|
|
osg::BoundingBox computeBound(const osg::Drawable&) const override { return boundingBox; }
|
2018-08-08 22:24:57 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
struct CopyBoundingSphereCallback : osg::Node::ComputeBoundingSphereCallback
|
|
|
|
{
|
|
|
|
osg::BoundingSphere boundingSphere;
|
|
|
|
|
2020-10-16 18:18:54 +00:00
|
|
|
osg::BoundingSphere computeBound(const osg::Node&) const override { return boundingSphere; }
|
2018-08-08 22:24:57 +00:00
|
|
|
};
|
|
|
|
|
2017-09-01 16:27:00 +00:00
|
|
|
private:
|
|
|
|
void cull(osg::NodeVisitor* nv);
|
2015-04-27 13:41:34 +00:00
|
|
|
void updateBounds(osg::NodeVisitor* nv);
|
2015-04-21 14:02:40 +00:00
|
|
|
|
2017-09-01 16:27:00 +00:00
|
|
|
osg::ref_ptr<osg::Geometry> mGeometry[2];
|
|
|
|
osg::Geometry* getGeometry(unsigned int frame) const;
|
|
|
|
|
2016-03-22 22:28:57 +00:00
|
|
|
osg::ref_ptr<osg::Geometry> mSourceGeometry;
|
2017-09-01 21:02:19 +00:00
|
|
|
osg::ref_ptr<const osg::Vec4Array> mSourceTangents;
|
2015-05-26 17:12:29 +00:00
|
|
|
Skeleton* mSkeleton;
|
2015-04-21 14:02:40 +00:00
|
|
|
|
2017-02-03 05:44:14 +00:00
|
|
|
osg::ref_ptr<osg::RefMatrix> mGeomToSkelMatrix;
|
2015-11-22 18:49:11 +00:00
|
|
|
|
2015-04-21 14:02:40 +00:00
|
|
|
osg::ref_ptr<InfluenceMap> mInfluenceMap;
|
|
|
|
|
2020-10-08 23:24:28 +00:00
|
|
|
typedef std::pair<std::string, osg::Matrixf> BoneBindMatrixPair;
|
|
|
|
|
|
|
|
typedef std::pair<BoneBindMatrixPair, float> BoneWeight;
|
|
|
|
|
|
|
|
typedef std::vector<unsigned short> VertexList;
|
2015-04-25 17:32:07 +00:00
|
|
|
|
2020-10-08 23:24:28 +00:00
|
|
|
typedef std::map<std::vector<BoneWeight>, VertexList> Bone2VertexMap;
|
2015-04-21 14:02:40 +00:00
|
|
|
|
2019-01-11 17:20:58 +00:00
|
|
|
struct Bone2VertexVector : public osg::Referenced
|
|
|
|
{
|
2020-10-08 23:24:28 +00:00
|
|
|
std::vector<std::pair<std::vector<BoneWeight>, VertexList>> mData;
|
2019-01-11 17:20:58 +00:00
|
|
|
};
|
|
|
|
osg::ref_ptr<Bone2VertexVector> mBone2VertexVector;
|
2015-04-27 13:41:34 +00:00
|
|
|
|
2019-01-11 17:20:58 +00:00
|
|
|
struct BoneSphereVector : public osg::Referenced
|
|
|
|
{
|
2020-10-08 23:24:28 +00:00
|
|
|
std::vector<std::pair<std::string, osg::BoundingSpheref>> mData;
|
2019-01-11 17:20:58 +00:00
|
|
|
};
|
|
|
|
osg::ref_ptr<BoneSphereVector> mBoneSphereVector;
|
|
|
|
std::vector<Bone*> mBoneNodesVector;
|
2015-04-27 13:41:34 +00:00
|
|
|
|
2015-09-25 23:21:33 +00:00
|
|
|
unsigned int mLastFrameNumber;
|
2015-04-29 21:48:08 +00:00
|
|
|
bool mBoundsFirstFrame;
|
|
|
|
|
2015-04-21 14:02:40 +00:00
|
|
|
bool initFromParentSkeleton(osg::NodeVisitor* nv);
|
2015-04-27 13:41:34 +00:00
|
|
|
|
2016-07-02 17:27:19 +00:00
|
|
|
void updateGeomToSkelMatrix(const osg::NodePath& nodePath);
|
2015-04-21 14:02:40 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|