#ifndef OPENMW_COMPONENTS_NIFOSG_RIGGEOMETRY_H #define OPENMW_COMPONENTS_NIFOSG_RIGGEOMETRY_H #include #include namespace SceneUtil { class Skeleton; class Bone; /// @brief Mesh skinning implementation. /// @note A RigGeometry may be attached directly to a Skeleton, or somewhere below a Skeleton. /// Note though that the RigGeometry ignores any transforms below the Skeleton, so the attachment point is not that important. /// @note To avoid race conditions, the rig geometry needs to be double buffered. This can be done /// using a FrameSwitch node that has two RigGeometry children. In the future we may want to consider implementing /// the double buffering inside RigGeometry. class RigGeometry : public osg::Geometry { public: RigGeometry(); RigGeometry(const RigGeometry& copy, const osg::CopyOp& copyop); META_Object(SceneUtil, RigGeometry) struct BoneInfluence { osg::Matrixf mInvBindMatrix; osg::BoundingSpheref mBoundSphere; // std::map mWeights; }; struct InfluenceMap : public osg::Referenced { std::map mMap; }; void setInfluenceMap(osg::ref_ptr influenceMap); /// Initialize this geometry from the source geometry. /// @note The source geometry will not be modified. /// @note The source geometry's normal and vertex arrays must be an osg::Vec3Array. void setSourceGeometry(osg::ref_ptr sourceGeom); // Called automatically by our CullCallback void update(osg::NodeVisitor* nv); // Called automatically by our UpdateCallback void updateBounds(osg::NodeVisitor* nv); private: osg::ref_ptr mSourceVertices; osg::ref_ptr mSourceNormals; Skeleton* mSkeleton; osg::NodePath mSkelToGeomPath; osg::Matrixf mGeomToSkelMatrix; osg::ref_ptr mInfluenceMap; typedef std::pair BoneBindMatrixPair; typedef std::pair BoneWeight; typedef std::vector VertexList; typedef std::map, VertexList> Bone2VertexMap; Bone2VertexMap mBone2VertexMap; typedef std::map BoneSphereMap; BoneSphereMap mBoneSphereMap; unsigned int mLastFrameNumber; bool mBoundsFirstFrame; bool initFromParentSkeleton(osg::NodeVisitor* nv); void updateGeomToSkelMatrix(osg::NodeVisitor* nv); }; } #endif