2015-04-21 14:02:40 +00:00
# include "riggeometry.hpp"
2018-08-14 15:42:41 +00:00
# include <components/debug/debuglog.hpp>
2021-10-23 08:31:46 +00:00
# include <components/resource/scenemanager.hpp>
2021-10-10 16:37:34 +00:00
# include <osg/MatrixTransform>
2018-08-14 15:42:41 +00:00
2015-04-21 14:02:40 +00:00
# include "skeleton.hpp"
2015-04-27 13:41:34 +00:00
# include "util.hpp"
2015-04-21 14:02:40 +00:00
2018-08-22 08:48:05 +00:00
namespace
{
inline void accumulateMatrix ( const osg : : Matrixf & invBindMatrix , const osg : : Matrixf & matrix , const float weight , osg : : Matrixf & result )
{
osg : : Matrixf m = invBindMatrix * matrix ;
float * ptr = m . ptr ( ) ;
float * ptrresult = result . ptr ( ) ;
ptrresult [ 0 ] + = ptr [ 0 ] * weight ;
ptrresult [ 1 ] + = ptr [ 1 ] * weight ;
ptrresult [ 2 ] + = ptr [ 2 ] * weight ;
ptrresult [ 4 ] + = ptr [ 4 ] * weight ;
ptrresult [ 5 ] + = ptr [ 5 ] * weight ;
ptrresult [ 6 ] + = ptr [ 6 ] * weight ;
ptrresult [ 8 ] + = ptr [ 8 ] * weight ;
ptrresult [ 9 ] + = ptr [ 9 ] * weight ;
ptrresult [ 10 ] + = ptr [ 10 ] * weight ;
ptrresult [ 12 ] + = ptr [ 12 ] * weight ;
ptrresult [ 13 ] + = ptr [ 13 ] * weight ;
ptrresult [ 14 ] + = ptr [ 14 ] * weight ;
}
}
2015-04-21 18:30:48 +00:00
namespace SceneUtil
2015-04-21 14:02:40 +00:00
{
RigGeometry : : RigGeometry ( )
2018-10-09 06:21:12 +00:00
: mSkeleton ( nullptr )
2015-09-25 23:21:33 +00:00
, mLastFrameNumber ( 0 )
2015-04-29 21:48:08 +00:00
, mBoundsFirstFrame ( true )
2015-04-21 14:02:40 +00:00
{
2019-02-20 13:37:00 +00:00
setNumChildrenRequiringUpdateTraversal ( 1 ) ;
// update done in accept(NodeVisitor&)
2015-04-21 14:02:40 +00:00
}
RigGeometry : : RigGeometry ( const RigGeometry & copy , const osg : : CopyOp & copyop )
2017-09-01 16:27:00 +00:00
: Drawable ( copy , copyop )
2018-10-09 06:21:12 +00:00
, mSkeleton ( nullptr )
2015-04-21 14:02:40 +00:00
, mInfluenceMap ( copy . mInfluenceMap )
2019-01-11 17:20:58 +00:00
, mBone2VertexVector ( copy . mBone2VertexVector )
, mBoneSphereVector ( copy . mBoneSphereVector )
2015-09-25 23:21:33 +00:00
, mLastFrameNumber ( 0 )
2015-12-03 18:49:45 +00:00
, mBoundsFirstFrame ( true )
2015-04-21 14:02:40 +00:00
{
2016-03-22 22:28:57 +00:00
setSourceGeometry ( copy . mSourceGeometry ) ;
2019-02-20 13:37:00 +00:00
setNumChildrenRequiringUpdateTraversal ( 1 ) ;
2015-04-21 14:02:40 +00:00
}
void RigGeometry : : setSourceGeometry ( osg : : ref_ptr < osg : : Geometry > sourceGeometry )
{
2021-10-23 08:31:46 +00:00
for ( unsigned int i = 0 ; i < 2 ; + + i )
mGeometry [ i ] = nullptr ;
2016-03-22 22:28:57 +00:00
mSourceGeometry = sourceGeometry ;
2015-04-21 14:02:40 +00:00
2017-09-01 16:27:00 +00:00
for ( unsigned int i = 0 ; i < 2 ; + + i )
2016-03-11 18:08:48 +00:00
{
2017-09-01 21:02:19 +00:00
const osg : : Geometry & from = * sourceGeometry ;
2021-10-23 08:31:46 +00:00
// DO NOT COPY AND PASTE THIS CODE. Cloning osg::Geometry without also cloning its contained Arrays is generally unsafe.
// In this specific case the operation is safe under the following two assumptions:
// - When Arrays are removed or replaced in the cloned geometry, the original Arrays in their place must outlive the cloned geometry regardless. (ensured by mSourceGeometry)
// - Arrays that we add or replace in the cloned geometry must be explicitely forbidden from reusing BufferObjects of the original geometry. (ensured by vbo below)
2017-09-01 16:27:00 +00:00
mGeometry [ i ] = new osg : : Geometry ( from , osg : : CopyOp : : SHALLOW_COPY ) ;
2021-10-23 08:31:46 +00:00
mGeometry [ i ] - > getOrCreateUserDataContainer ( ) - > addUserObject ( new Resource : : TemplateRef ( mSourceGeometry ) ) ;
2017-09-01 16:27:00 +00:00
osg : : Geometry & to = * mGeometry [ i ] ;
to . setSupportsDisplayList ( false ) ;
to . setUseVertexBufferObjects ( true ) ;
to . setCullingActive ( false ) ; // make sure to disable culling since that's handled by this class
2018-08-08 22:24:57 +00:00
to . setComputeBoundingBoxCallback ( new CopyBoundingBoxCallback ( ) ) ;
to . setComputeBoundingSphereCallback ( new CopyBoundingSphereCallback ( ) ) ;
2017-09-01 16:27:00 +00:00
// vertices and normals are modified every frame, so we need to deep copy them.
// assign a dedicated VBO to make sure that modifications don't interfere with source geometry's VBO.
osg : : ref_ptr < osg : : VertexBufferObject > vbo ( new osg : : VertexBufferObject ) ;
vbo - > setUsage ( GL_DYNAMIC_DRAW_ARB ) ;
2020-05-16 13:37:00 +00:00
osg : : ref_ptr < osg : : Array > vertexArray = static_cast < osg : : Array * > ( from . getVertexArray ( ) - > clone ( osg : : CopyOp : : DEEP_COPY_ALL ) ) ;
2017-09-01 16:27:00 +00:00
if ( vertexArray )
2016-12-15 21:39:21 +00:00
{
2017-09-01 16:27:00 +00:00
vertexArray - > setVertexBufferObject ( vbo ) ;
to . setVertexArray ( vertexArray ) ;
2016-12-15 21:39:21 +00:00
}
2016-03-23 15:48:41 +00:00
2017-09-01 21:02:19 +00:00
if ( const osg : : Array * normals = from . getNormalArray ( ) )
2017-09-01 16:27:00 +00:00
{
2020-05-16 13:37:00 +00:00
osg : : ref_ptr < osg : : Array > normalArray = static_cast < osg : : Array * > ( normals - > clone ( osg : : CopyOp : : DEEP_COPY_ALL ) ) ;
2017-09-01 16:27:00 +00:00
if ( normalArray )
{
normalArray - > setVertexBufferObject ( vbo ) ;
to . setNormalArray ( normalArray , osg : : Array : : BIND_PER_VERTEX ) ;
}
}
2016-12-15 21:39:21 +00:00
2017-09-01 21:02:19 +00:00
if ( const osg : : Vec4Array * tangents = dynamic_cast < const osg : : Vec4Array * > ( from . getTexCoordArray ( 7 ) ) )
2017-09-01 16:27:00 +00:00
{
mSourceTangents = tangents ;
2020-05-16 13:37:00 +00:00
osg : : ref_ptr < osg : : Array > tangentArray = static_cast < osg : : Array * > ( tangents - > clone ( osg : : CopyOp : : DEEP_COPY_ALL ) ) ;
2017-09-01 16:27:00 +00:00
tangentArray - > setVertexBufferObject ( vbo ) ;
to . setTexCoordArray ( 7 , tangentArray , osg : : Array : : BIND_PER_VERTEX ) ;
}
else
2018-10-09 06:21:12 +00:00
mSourceTangents = nullptr ;
2016-03-23 15:48:41 +00:00
}
}
2019-06-13 13:37:00 +00:00
osg : : ref_ptr < osg : : Geometry > RigGeometry : : getSourceGeometry ( ) const
2016-03-23 15:48:41 +00:00
{
return mSourceGeometry ;
2015-04-21 14:02:40 +00:00
}
bool RigGeometry : : initFromParentSkeleton ( osg : : NodeVisitor * nv )
{
const osg : : NodePath & path = nv - > getNodePath ( ) ;
2021-10-10 16:37:34 +00:00
for ( osg : : NodePath : : const_reverse_iterator it = path . rbegin ( ) + 1 ; it ! = path . rend ( ) ; + + it )
2015-04-21 14:02:40 +00:00
{
osg : : Node * node = * it ;
2021-10-10 16:37:34 +00:00
if ( node - > asTransform ( ) )
continue ;
2015-04-21 14:02:40 +00:00
if ( Skeleton * skel = dynamic_cast < Skeleton * > ( node ) )
{
mSkeleton = skel ;
break ;
}
}
if ( ! mSkeleton )
{
2018-08-14 15:42:41 +00:00
Log ( Debug : : Error ) < < " Error: A RigGeometry did not find its parent skeleton " ;
2015-04-21 14:02:40 +00:00
return false ;
}
if ( ! mInfluenceMap )
{
2018-08-14 15:42:41 +00:00
Log ( Debug : : Error ) < < " Error: No InfluenceMap set on RigGeometry " ;
2015-04-21 14:02:40 +00:00
return false ;
}
2019-01-11 17:20:58 +00:00
mBoneNodesVector . clear ( ) ;
2020-10-08 23:24:28 +00:00
for ( auto & bonePair : mBoneSphereVector - > mData )
2015-04-21 14:02:40 +00:00
{
2020-10-08 23:24:28 +00:00
const std : : string & boneName = bonePair . first ;
Bone * bone = mSkeleton - > getBone ( boneName ) ;
2015-04-25 17:32:07 +00:00
if ( ! bone )
2015-04-21 14:02:40 +00:00
{
2019-01-11 17:20:58 +00:00
mBoneNodesVector . push_back ( nullptr ) ;
2020-10-08 23:24:28 +00:00
Log ( Debug : : Error ) < < " Error: RigGeometry did not find bone " < < boneName ;
2015-04-25 17:32:07 +00:00
continue ;
2015-04-21 14:02:40 +00:00
}
2019-01-11 17:20:58 +00:00
mBoneNodesVector . push_back ( bone ) ;
}
2015-04-25 17:32:07 +00:00
2019-01-11 17:20:58 +00:00
for ( auto & pair : mBone2VertexVector - > mData )
{
for ( auto & weight : pair . first )
2015-04-25 17:32:07 +00:00
{
2020-10-08 23:24:28 +00:00
const std : : string & boneName = weight . first . first ;
Bone * bone = mSkeleton - > getBone ( boneName ) ;
2019-01-11 17:20:58 +00:00
if ( ! bone )
{
mBoneNodesVector . push_back ( nullptr ) ;
2020-10-08 23:24:28 +00:00
Log ( Debug : : Error ) < < " Error: RigGeometry did not find bone " < < boneName ;
2019-01-11 17:20:58 +00:00
continue ;
}
2015-04-25 17:32:07 +00:00
2019-01-11 17:20:58 +00:00
mBoneNodesVector . push_back ( bone ) ;
2015-04-25 17:32:07 +00:00
}
2015-04-21 14:02:40 +00:00
}
2015-04-25 17:32:07 +00:00
2015-04-21 14:02:40 +00:00
return true ;
}
2017-09-01 16:27:00 +00:00
void RigGeometry : : cull ( osg : : NodeVisitor * nv )
2015-04-21 14:02:40 +00:00
{
2015-06-10 17:08:56 +00:00
if ( ! mSkeleton )
2015-04-21 14:02:40 +00:00
{
2018-08-14 15:42:41 +00:00
Log ( Debug : : Error ) < < " Error: RigGeometry rendering with no skeleton, should have been initialized by UpdateVisitor " ;
2015-12-03 18:49:45 +00:00
// try to recover anyway, though rendering is likely to be incorrect.
2015-06-10 17:08:56 +00:00
if ( ! initFromParentSkeleton ( nv ) )
return ;
2015-04-21 14:02:40 +00:00
}
2018-08-22 08:48:05 +00:00
unsigned int traversalNumber = nv - > getTraversalNumber ( ) ;
if ( mLastFrameNumber = = traversalNumber | | ( mLastFrameNumber ! = 0 & & ! mSkeleton - > getActive ( ) ) )
2017-09-01 16:27:00 +00:00
{
2017-09-01 22:12:40 +00:00
osg : : Geometry & geom = * getGeometry ( mLastFrameNumber ) ;
nv - > pushOntoNodePath ( & geom ) ;
nv - > apply ( geom ) ;
nv - > popFromNodePath ( ) ;
2015-09-25 23:21:33 +00:00
return ;
2017-09-01 16:27:00 +00:00
}
2018-08-22 08:48:05 +00:00
mLastFrameNumber = traversalNumber ;
2017-09-01 16:27:00 +00:00
osg : : Geometry & geom = * getGeometry ( mLastFrameNumber ) ;
2015-04-29 21:48:08 +00:00
2018-08-22 08:48:05 +00:00
mSkeleton - > updateBoneMatrices ( traversalNumber ) ;
2015-04-21 14:02:40 +00:00
// skinning
2017-09-01 16:27:00 +00:00
const osg : : Vec3Array * positionSrc = static_cast < osg : : Vec3Array * > ( mSourceGeometry - > getVertexArray ( ) ) ;
const osg : : Vec3Array * normalSrc = static_cast < osg : : Vec3Array * > ( mSourceGeometry - > getNormalArray ( ) ) ;
const osg : : Vec4Array * tangentSrc = mSourceTangents ;
2015-04-21 14:02:40 +00:00
2017-09-01 16:27:00 +00:00
osg : : Vec3Array * positionDst = static_cast < osg : : Vec3Array * > ( geom . getVertexArray ( ) ) ;
osg : : Vec3Array * normalDst = static_cast < osg : : Vec3Array * > ( geom . getNormalArray ( ) ) ;
osg : : Vec4Array * tangentDst = static_cast < osg : : Vec4Array * > ( geom . getTexCoordArray ( 7 ) ) ;
2015-04-21 14:02:40 +00:00
2019-01-11 17:20:58 +00:00
int index = mBoneSphereVector - > mData . size ( ) ;
for ( auto & pair : mBone2VertexVector - > mData )
2015-04-21 14:02:40 +00:00
{
2018-08-22 08:48:05 +00:00
osg : : Matrixf resultMat ( 0 , 0 , 0 , 0 ,
2015-04-25 17:32:07 +00:00
0 , 0 , 0 , 0 ,
0 , 0 , 0 , 0 ,
0 , 0 , 0 , 1 ) ;
2015-04-21 14:02:40 +00:00
2018-08-22 08:48:05 +00:00
for ( auto & weight : pair . first )
2015-04-21 14:02:40 +00:00
{
2019-01-11 17:20:58 +00:00
Bone * bone = mBoneNodesVector [ index ] ;
if ( bone = = nullptr )
continue ;
2020-10-08 23:24:28 +00:00
accumulateMatrix ( weight . first . second , bone - > mMatrixInSkeletonSpace , weight . second , resultMat ) ;
2019-01-11 17:20:58 +00:00
index + + ;
2015-04-25 17:32:07 +00:00
}
2018-08-22 08:48:05 +00:00
2017-02-03 01:18:44 +00:00
if ( mGeomToSkelMatrix )
resultMat * = ( * mGeomToSkelMatrix ) ;
2015-04-21 14:02:40 +00:00
2018-08-22 08:48:05 +00:00
for ( auto & vertex : pair . second )
2015-04-25 17:32:07 +00:00
{
( * positionDst ) [ vertex ] = resultMat . preMult ( ( * positionSrc ) [ vertex ] ) ;
2016-12-15 21:39:21 +00:00
if ( normalDst )
2018-08-22 08:48:05 +00:00
( * normalDst ) [ vertex ] = osg : : Matrixf : : transform3x3 ( ( * normalSrc ) [ vertex ] , resultMat ) ;
2016-03-23 15:48:41 +00:00
if ( tangentDst )
{
2018-08-22 08:48:05 +00:00
const osg : : Vec4f & srcTangent = ( * tangentSrc ) [ vertex ] ;
osg : : Vec3f transformedTangent = osg : : Matrixf : : transform3x3 ( osg : : Vec3f ( srcTangent . x ( ) , srcTangent . y ( ) , srcTangent . z ( ) ) , resultMat ) ;
2016-03-23 15:48:41 +00:00
( * tangentDst ) [ vertex ] = osg : : Vec4f ( transformedTangent , srcTangent . w ( ) ) ;
}
2015-04-21 14:02:40 +00:00
}
}
positionDst - > dirty ( ) ;
2016-12-15 21:39:21 +00:00
if ( normalDst )
normalDst - > dirty ( ) ;
2016-03-23 15:48:41 +00:00
if ( tangentDst )
tangentDst - > dirty ( ) ;
2017-09-01 16:27:00 +00:00
2021-10-12 11:47:42 +00:00
geom . osg : : Drawable : : dirtyGLObjects ( ) ;
2018-12-18 19:44:30 +00:00
2017-09-01 16:27:00 +00:00
nv - > pushOntoNodePath ( & geom ) ;
nv - > apply ( geom ) ;
nv - > popFromNodePath ( ) ;
2015-04-21 14:02:40 +00:00
}
2015-04-27 13:41:34 +00:00
void RigGeometry : : updateBounds ( osg : : NodeVisitor * nv )
{
if ( ! mSkeleton )
{
if ( ! initFromParentSkeleton ( nv ) )
return ;
}
2015-04-29 21:48:08 +00:00
if ( ! mSkeleton - > getActive ( ) & & ! mBoundsFirstFrame )
return ;
mBoundsFirstFrame = false ;
2016-07-02 17:26:11 +00:00
mSkeleton - > updateBoneMatrices ( nv - > getTraversalNumber ( ) ) ;
2015-04-27 13:41:34 +00:00
2016-07-02 17:27:19 +00:00
updateGeomToSkelMatrix ( nv - > getNodePath ( ) ) ;
2015-11-22 18:49:11 +00:00
2015-04-27 13:41:34 +00:00
osg : : BoundingBox box ;
2019-01-11 17:20:58 +00:00
int index = 0 ;
for ( auto & boundPair : mBoneSphereVector - > mData )
2015-04-27 13:41:34 +00:00
{
2019-01-11 17:20:58 +00:00
Bone * bone = mBoneNodesVector [ index ] ;
if ( bone = = nullptr )
continue ;
index + + ;
2020-10-08 23:24:28 +00:00
osg : : BoundingSpheref bs = boundPair . second ;
2017-02-03 01:18:44 +00:00
if ( mGeomToSkelMatrix )
transformBoundingSphere ( bone - > mMatrixInSkeletonSpace * ( * mGeomToSkelMatrix ) , bs ) ;
else
transformBoundingSphere ( bone - > mMatrixInSkeletonSpace , bs ) ;
2015-04-27 13:41:34 +00:00
box . expandBy ( bs ) ;
}
2016-12-16 23:23:22 +00:00
if ( box ! = _boundingBox )
{
_boundingBox = box ;
_boundingSphere = osg : : BoundingSphere ( _boundingBox ) ;
_boundingSphereComputed = true ;
for ( unsigned int i = 0 ; i < getNumParents ( ) ; + + i )
getParent ( i ) - > dirtyBound ( ) ;
2018-08-08 22:24:57 +00:00
for ( unsigned int i = 0 ; i < 2 ; + + i )
{
osg : : Geometry & geom = * mGeometry [ i ] ;
static_cast < CopyBoundingBoxCallback * > ( geom . getComputeBoundingBoxCallback ( ) ) - > boundingBox = _boundingBox ;
static_cast < CopyBoundingSphereCallback * > ( geom . getComputeBoundingSphereCallback ( ) ) - > boundingSphere = _boundingSphere ;
geom . dirtyBound ( ) ;
}
2016-12-16 23:23:22 +00:00
}
2015-04-27 13:41:34 +00:00
}
2016-07-02 17:27:19 +00:00
void RigGeometry : : updateGeomToSkelMatrix ( const osg : : NodePath & nodePath )
2015-04-27 13:41:34 +00:00
{
bool foundSkel = false ;
2021-10-10 16:37:34 +00:00
osg : : RefMatrix * geomToSkelMatrix = mGeomToSkelMatrix ;
if ( geomToSkelMatrix )
geomToSkelMatrix - > makeIdentity ( ) ;
for ( osg : : NodePath : : const_iterator it = nodePath . begin ( ) ; it ! = nodePath . end ( ) - 1 ; + + it )
2015-04-27 13:41:34 +00:00
{
2017-02-03 01:18:44 +00:00
osg : : Node * node = * it ;
2015-04-27 13:41:34 +00:00
if ( ! foundSkel )
{
2017-02-03 01:18:44 +00:00
if ( node = = mSkeleton )
2015-04-27 13:41:34 +00:00
foundSkel = true ;
}
else
2017-02-03 01:18:44 +00:00
{
if ( osg : : Transform * trans = node - > asTransform ( ) )
{
2021-10-10 16:37:34 +00:00
osg : : MatrixTransform * matrixTrans = trans - > asMatrixTransform ( ) ;
if ( matrixTrans & & matrixTrans - > getMatrix ( ) . isIdentity ( ) )
continue ;
2017-02-03 01:18:44 +00:00
if ( ! geomToSkelMatrix )
2021-10-10 16:37:34 +00:00
geomToSkelMatrix = mGeomToSkelMatrix = new osg : : RefMatrix ;
2018-10-09 06:21:12 +00:00
trans - > computeWorldToLocalMatrix ( * geomToSkelMatrix , nullptr ) ;
2017-02-03 01:18:44 +00:00
}
}
2015-04-27 13:41:34 +00:00
}
}
2015-04-21 14:02:40 +00:00
void RigGeometry : : setInfluenceMap ( osg : : ref_ptr < InfluenceMap > influenceMap )
{
mInfluenceMap = influenceMap ;
2019-01-11 17:20:58 +00:00
2020-10-08 23:24:28 +00:00
typedef std : : map < unsigned short , std : : vector < BoneWeight > > Vertex2BoneMap ;
2019-01-11 17:20:58 +00:00
Vertex2BoneMap vertex2BoneMap ;
mBoneSphereVector = new BoneSphereVector ;
mBoneSphereVector - > mData . reserve ( mInfluenceMap - > mData . size ( ) ) ;
mBone2VertexVector = new Bone2VertexVector ;
2020-10-08 23:24:28 +00:00
for ( auto & influencePair : mInfluenceMap - > mData )
2019-01-11 17:20:58 +00:00
{
2020-10-08 23:24:28 +00:00
const std : : string & boneName = influencePair . first ;
const BoneInfluence & bi = influencePair . second ;
mBoneSphereVector - > mData . emplace_back ( boneName , bi . mBoundSphere ) ;
for ( auto & weightPair : bi . mWeights )
{
std : : vector < BoneWeight > & vec = vertex2BoneMap [ weightPair . first ] ;
vec . emplace_back ( std : : make_pair ( boneName , bi . mInvBindMatrix ) , weightPair . second ) ;
}
2019-01-11 17:20:58 +00:00
}
Bone2VertexMap bone2VertexMap ;
for ( auto & vertexPair : vertex2BoneMap )
2020-10-08 23:24:28 +00:00
{
2019-01-11 17:20:58 +00:00
bone2VertexMap [ vertexPair . second ] . emplace_back ( vertexPair . first ) ;
2020-10-08 23:24:28 +00:00
}
2019-01-11 17:20:58 +00:00
mBone2VertexVector - > mData . reserve ( bone2VertexMap . size ( ) ) ;
mBone2VertexVector - > mData . assign ( bone2VertexMap . begin ( ) , bone2VertexMap . end ( ) ) ;
2015-04-21 14:02:40 +00:00
}
2017-09-01 16:27:00 +00:00
void RigGeometry : : accept ( osg : : NodeVisitor & nv )
{
if ( ! nv . validNodeMask ( * this ) )
return ;
nv . pushOntoNodePath ( this ) ;
if ( nv . getVisitorType ( ) = = osg : : NodeVisitor : : CULL_VISITOR )
cull ( & nv ) ;
else if ( nv . getVisitorType ( ) = = osg : : NodeVisitor : : UPDATE_VISITOR )
updateBounds ( & nv ) ;
else
nv . apply ( * this ) ;
nv . popFromNodePath ( ) ;
}
2017-09-03 14:02:40 +00:00
void RigGeometry : : accept ( osg : : PrimitiveFunctor & func ) const
{
getGeometry ( mLastFrameNumber ) - > accept ( func ) ;
}
2017-09-01 16:27:00 +00:00
osg : : Geometry * RigGeometry : : getGeometry ( unsigned int frame ) const
{
return mGeometry [ frame % 2 ] . get ( ) ;
}
2015-04-21 14:02:40 +00:00
}