2001-10-22 05:27:40 +08:00
|
|
|
#include <osg/NodeVisitor>
|
2002-02-11 06:35:22 +08:00
|
|
|
#include <osg/Transform>
|
2001-01-11 00:32:10 +08:00
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
using namespace osg;
|
|
|
|
|
|
|
|
NodeVisitor::NodeVisitor(TraversalMode tm)
|
|
|
|
{
|
2001-09-22 10:42:08 +08:00
|
|
|
_traversalNumber = -1;
|
|
|
|
|
2001-09-20 05:08:56 +08:00
|
|
|
_traversalVisitor = NULL;
|
|
|
|
_traversalMode = tm;
|
2001-09-22 10:42:08 +08:00
|
|
|
_traversalMask = 0xffffffff;
|
|
|
|
_nodeMaskOverride = 0x0;
|
2001-01-11 00:32:10 +08:00
|
|
|
}
|
|
|
|
|
2001-09-20 05:08:56 +08:00
|
|
|
|
2001-01-11 00:32:10 +08:00
|
|
|
NodeVisitor::~NodeVisitor()
|
|
|
|
{
|
2001-09-20 05:08:56 +08:00
|
|
|
// if (_traversalVisitor) detach from _traversalVisitor;
|
2001-01-11 00:32:10 +08:00
|
|
|
}
|
|
|
|
|
2001-09-20 05:08:56 +08:00
|
|
|
|
|
|
|
void NodeVisitor::setTraversalMode(const TraversalMode mode)
|
2001-01-11 00:32:10 +08:00
|
|
|
{
|
2001-09-20 05:08:56 +08:00
|
|
|
if (_traversalMode==mode) return;
|
2001-01-11 00:32:10 +08:00
|
|
|
if (mode==TRAVERSE_VISITOR)
|
|
|
|
{
|
2001-09-20 05:08:56 +08:00
|
|
|
if (_traversalVisitor==NULL) _traversalMode = TRAVERSE_NONE;
|
|
|
|
else _traversalMode = TRAVERSE_VISITOR;
|
2001-01-11 00:32:10 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2001-09-20 05:08:56 +08:00
|
|
|
if (_traversalVisitor.valid()) _traversalVisitor=NULL;
|
|
|
|
_traversalMode = mode;
|
2001-01-11 00:32:10 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2001-09-20 05:08:56 +08:00
|
|
|
|
|
|
|
void NodeVisitor::setTraversalVisitor(NodeVisitor* nv)
|
2001-01-11 00:32:10 +08:00
|
|
|
{
|
2001-09-20 05:08:56 +08:00
|
|
|
if (_traversalVisitor==nv) return;
|
|
|
|
_traversalVisitor = nv;
|
|
|
|
if (_traversalVisitor.valid()) _traversalMode = TRAVERSE_VISITOR;
|
|
|
|
else _traversalMode = TRAVERSE_NONE;
|
2001-01-11 00:32:10 +08:00
|
|
|
}
|
2002-02-06 05:54:46 +08:00
|
|
|
|
2002-02-11 06:35:22 +08:00
|
|
|
|
|
|
|
class TransformVisitor : public NodeVisitor
|
2002-02-06 05:54:46 +08:00
|
|
|
{
|
2002-02-11 06:35:22 +08:00
|
|
|
public:
|
|
|
|
|
|
|
|
enum CoordMode
|
|
|
|
{
|
|
|
|
WORLD_TO_LOCAL,
|
|
|
|
LOCAL_TO_WORLD
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
MatrixMode _matrixMode;
|
|
|
|
CoordMode _coordMode;
|
|
|
|
Matrix& _matrix;
|
|
|
|
NodeVisitor* _nodeVisitor;
|
|
|
|
|
|
|
|
TransformVisitor(Matrix& matrix,MatrixMode matrixMode,CoordMode coordMode,NodeVisitor* nv):
|
|
|
|
NodeVisitor(),
|
|
|
|
_matrixMode(matrixMode),
|
|
|
|
_coordMode(coordMode),
|
|
|
|
_matrix(matrix),
|
|
|
|
_nodeVisitor(nv)
|
|
|
|
{}
|
|
|
|
|
|
|
|
virtual void apply(Transform& transform)
|
|
|
|
{
|
|
|
|
bool applyTransform =
|
|
|
|
(_matrixMode==transform.getMatrixMode()) ||
|
|
|
|
(_matrixMode==MODELVIEW && (transform.getMatrixMode()==MODEL || transform.getMatrixMode()==VIEW));
|
|
|
|
|
|
|
|
if (applyTransform)
|
|
|
|
{
|
|
|
|
if (_coordMode==LOCAL_TO_WORLD)
|
|
|
|
{
|
|
|
|
osg::Matrix localToWorldMat;
|
|
|
|
transform.getLocalToWorldMatrix(localToWorldMat,_nodeVisitor);
|
|
|
|
_matrix.preMult(localToWorldMat);
|
|
|
|
}
|
|
|
|
else // worldToLocal
|
|
|
|
{
|
|
|
|
osg::Matrix worldToLocalMat;
|
|
|
|
transform.getWorldToLocalMatrix(worldToLocalMat,_nodeVisitor);
|
|
|
|
_matrix.postMult(worldToLocalMat);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
const bool NodeVisitor::getLocalToWorldMatrix(Matrix& matrix, MatrixMode mode, Node* node)
|
|
|
|
{
|
|
|
|
TransformVisitor tv(matrix,mode,TransformVisitor::LOCAL_TO_WORLD,this);
|
|
|
|
for(NodePath::iterator itr=_nodePath.begin();
|
|
|
|
itr!=_nodePath.end();
|
|
|
|
++itr)
|
|
|
|
{
|
|
|
|
if (*itr==node) return true; // don't account for matrix attached to specofied node
|
|
|
|
(*itr)->accept(tv);
|
|
|
|
}
|
|
|
|
return true;
|
2002-02-06 05:54:46 +08:00
|
|
|
}
|
|
|
|
|
2002-02-11 06:35:22 +08:00
|
|
|
const bool NodeVisitor::getWorldToLocalMatrix(Matrix& matrix, MatrixMode mode, Node* node)
|
2002-02-06 05:54:46 +08:00
|
|
|
{
|
2002-02-11 06:35:22 +08:00
|
|
|
TransformVisitor tv(matrix,mode,TransformVisitor::WORLD_TO_LOCAL,this);
|
|
|
|
for(NodePath::iterator itr=_nodePath.begin();
|
|
|
|
itr!=_nodePath.end();
|
|
|
|
++itr)
|
|
|
|
{
|
|
|
|
if (*itr==node) return true; // don't account for matrix attached to specofied node
|
|
|
|
(*itr)->accept(tv);
|
|
|
|
}
|
|
|
|
return true;
|
2002-02-06 05:54:46 +08:00
|
|
|
}
|