OpenSceneGraph/include/osgUtil/StateGraph

320 lines
10 KiB
Plaintext
Raw Normal View History

2006-07-18 23:21:48 +08:00
/* -*-c++-*- OpenSceneGraph - Copyright (C) 1998-2006 Robert Osfield
*
* This library is open source and may be redistributed and/or modified under
* the terms of the OpenSceneGraph Public License (OSGPL) version 0.0 or
* (at your option) any later version. The full license is in LICENSE file
* included with this distribution, and on the openscenegraph.org website.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* OpenSceneGraph Public License for more details.
*/
#ifndef OSGUTIL_STATEGRAPH
#define OSGUTIL_STATEGRAPH 1
2001-09-20 05:19:47 +08:00
#include <osg/Matrix>
#include <osg/Drawable>
#include <osg/StateSet>
#include <osg/State>
#include <osg/Light>
#include <osgUtil/RenderLeaf>
#include <set>
#include <vector>
#include <algorithm>
2001-09-20 05:19:47 +08:00
namespace osgUtil {
struct LessDepthSortFunctor
{
bool operator() (const osg::ref_ptr<RenderLeaf>& lhs,const osg::ref_ptr<RenderLeaf>& rhs)
{
return (lhs->_depth < rhs->_depth);
}
};
2001-09-20 05:19:47 +08:00
/** StateGraph - contained in a renderBin, defines the scene to be drawn.
*/
class OSGUTIL_EXPORT StateGraph : public osg::Referenced
2001-09-20 05:19:47 +08:00
{
public:
typedef std::map< const osg::StateSet*, osg::ref_ptr<StateGraph> > ChildList;
2001-09-20 05:19:47 +08:00
typedef std::vector< osg::ref_ptr<RenderLeaf> > LeafList;
StateGraph* _parent;
2001-09-20 05:19:47 +08:00
#ifdef OSGUTIL_RENDERBACKEND_USE_REF_PTR
osg::ref_ptr<const osg::StateSet> _stateset;
#else
const osg::StateSet* _stateset;
#endif
2001-09-20 05:19:47 +08:00
int _depth;
ChildList _children;
LeafList _leaves;
mutable float _averageDistance;
mutable float _minimumDistance;
osg::ref_ptr<osg::Referenced> _userData;
2001-09-20 05:19:47 +08:00
bool _dynamic;
2001-09-20 05:19:47 +08:00
StateGraph():
osg::Referenced(false),
2007-09-07 23:03:56 +08:00
_parent(NULL),
_stateset(NULL),
_depth(0),
_averageDistance(0),
_minimumDistance(0),
_userData(NULL),
_dynamic(false)
2001-09-20 05:19:47 +08:00
{
}
StateGraph(StateGraph* parent,const osg::StateSet* stateset):
osg::Referenced(false),
2007-09-07 23:03:56 +08:00
_parent(parent),
_stateset(stateset),
_depth(0),
_averageDistance(0),
_minimumDistance(0),
_userData(NULL),
_dynamic(false)
2001-09-20 05:19:47 +08:00
{
if (_parent) _depth = _parent->_depth + 1;
if (_parent && _parent->_dynamic) _dynamic = true;
else _dynamic = stateset->getDataVariance()==osg::Object::DYNAMIC;
2001-09-20 05:19:47 +08:00
}
~StateGraph() {}
2001-09-20 05:19:47 +08:00
StateGraph* cloneType() const { return new StateGraph; }
2002-02-19 07:01:09 +08:00
void setUserData(osg::Referenced* obj) { _userData = obj; }
osg::Referenced* getUserData() { return _userData.get(); }
const osg::Referenced* getUserData() const { return _userData.get(); }
void setStateSet(const osg::StateSet* stateset) { _stateset = stateset; }
#ifdef OSGUTIL_RENDERBACKEND_USE_REF_PTR
const osg::StateSet* getStateSet() const { return _stateset.get(); }
#else
const osg::StateSet* getStateSet() const { return _stateset; }
#endif
2001-09-20 05:19:47 +08:00
2001-09-28 20:36:40 +08:00
/** return true if all of drawables, lights and children are empty.*/
inline bool empty() const
2001-09-20 05:19:47 +08:00
{
return _leaves.empty() && _children.empty();
}
inline bool leaves_empty() const
2001-09-20 05:19:47 +08:00
{
return _leaves.empty();
}
inline float getAverageDistance() const
{
if (_averageDistance==FLT_MAX && !_leaves.empty())
{
_averageDistance = 0.0f;
for(LeafList::const_iterator itr=_leaves.begin();
itr!=_leaves.end();
++itr)
{
_averageDistance += (*itr)->_depth;
}
_averageDistance /= (float)_leaves.size();
}
return _averageDistance;
}
inline float getMinimumDistance() const
{
if (_minimumDistance==FLT_MAX && !_leaves.empty())
{
LeafList::const_iterator itr=_leaves.begin();
_minimumDistance = (*itr)->_depth;
++itr;
for(;
itr!=_leaves.end();
++itr)
{
if ((*itr)->_depth<_minimumDistance) _minimumDistance=(*itr)->_depth;
}
}
return _minimumDistance;
}
inline void sortFrontToBack()
{
std::sort(_leaves.begin(),_leaves.end(),LessDepthSortFunctor());
}
/** Reset the internal contents of a StateGraph, including deleting all children.*/
2001-09-20 05:19:47 +08:00
void reset();
/** Recursively clean the StateGraph of all its drawables, lights and depths.
2001-09-20 05:19:47 +08:00
* Leaves children intact, and ready to be populated again.*/
void clean();
/** Recursively prune the StateGraph of empty children.*/
2001-09-20 05:19:47 +08:00
void prune();
inline StateGraph* find_or_insert(const osg::StateSet* stateset)
2001-09-20 05:19:47 +08:00
{
// search for the appropriate state group, return it if found.
ChildList::iterator itr = _children.find(stateset);
if (itr!=_children.end()) return itr->second.get();
2001-09-28 20:36:40 +08:00
// create a state group and insert it into the children list
2001-09-20 05:19:47 +08:00
// then return the state group.
StateGraph* sg = new StateGraph(this,stateset);
2001-09-20 05:19:47 +08:00
_children[stateset] = sg;
return sg;
}
/** add a render leaf.*/
inline void addLeaf(RenderLeaf* leaf)
{
if (leaf)
{
_averageDistance = FLT_MAX; // signify dirty.
_minimumDistance = FLT_MAX; // signify dirty.
2001-09-20 05:19:47 +08:00
_leaves.push_back(leaf);
leaf->_parent = this;
if (_dynamic) leaf->_dynamic = true;
2001-09-20 05:19:47 +08:00
}
}
static inline void moveStateGraph(osg::State& state,StateGraph* sg_curr,StateGraph* sg_new)
2001-09-20 05:19:47 +08:00
{
if (sg_new==sg_curr || sg_new==NULL) return;
if (sg_curr==NULL)
{
// use return path to trace back steps to sg_new.
std::vector<StateGraph*> return_path;
2001-09-20 05:19:47 +08:00
// need to pop back root render graph.
do
{
return_path.push_back(sg_new);
sg_new = sg_new->_parent;
} while (sg_new);
for(std::vector<StateGraph*>::reverse_iterator itr=return_path.rbegin();
2001-09-20 05:19:47 +08:00
itr!=return_path.rend();
++itr)
{
StateGraph* rg = (*itr);
if (rg->getStateSet()) state.pushStateSet(rg->getStateSet());
2001-09-20 05:19:47 +08:00
}
return;
}
// first handle the typical case which is two state groups
// are neighbours.
if (sg_curr->_parent==sg_new->_parent)
{
// state has changed so need to pop old state.
if (sg_curr->getStateSet()) state.popStateSet();
2001-09-20 05:19:47 +08:00
// and push new state.
if (sg_new->getStateSet()) state.pushStateSet(sg_new->getStateSet());
2001-09-20 05:19:47 +08:00
return;
}
2001-09-28 20:36:40 +08:00
// need to pop back up to the same depth as the new state group.
2001-09-20 05:19:47 +08:00
while (sg_curr->_depth>sg_new->_depth)
{
if (sg_curr->getStateSet()) state.popStateSet();
2001-09-20 05:19:47 +08:00
sg_curr = sg_curr->_parent;
}
// use return path to trace back steps to sg_new.
std::vector<StateGraph*> return_path;
2001-09-20 05:19:47 +08:00
2001-09-28 20:36:40 +08:00
// need to pop back up to the same depth as the curr state group.
2001-09-20 05:19:47 +08:00
while (sg_new->_depth>sg_curr->_depth)
{
return_path.push_back(sg_new);
sg_new = sg_new->_parent;
}
// now pop back up both parent paths until they agree.
// DRT - 10/22/02
// should be this to conform with above case where two StateGraph
// nodes have the same parent
while (sg_curr != sg_new)
2001-09-20 05:19:47 +08:00
{
if (sg_curr->getStateSet()) state.popStateSet();
2001-09-20 05:19:47 +08:00
sg_curr = sg_curr->_parent;
return_path.push_back(sg_new);
sg_new = sg_new->_parent;
}
for(std::vector<StateGraph*>::reverse_iterator itr=return_path.rbegin();
2001-09-20 05:19:47 +08:00
itr!=return_path.rend();
++itr)
{
StateGraph* rg = (*itr);
if (rg->getStateSet()) state.pushStateSet(rg->getStateSet());
2001-09-20 05:19:47 +08:00
}
}
inline static void moveToRootStateGraph(osg::State& state,StateGraph* sg_curr)
2001-09-20 05:19:47 +08:00
{
// need to pop back all statesets and matrices.
while (sg_curr)
{
if (sg_curr->getStateSet()) state.popStateSet();
2001-09-20 05:19:47 +08:00
sg_curr = sg_curr->_parent;
}
}
2007-04-25 18:32:28 +08:00
inline static int numToPop(StateGraph* sg_curr)
{
int numToPop = 0;
// need to pop back all statesets and matrices.
while (sg_curr)
{
if (sg_curr->getStateSet()) ++numToPop;
sg_curr = sg_curr->_parent;
}
return numToPop;
}
2001-09-20 05:19:47 +08:00
private:
/// disallow copy construction.
StateGraph(const StateGraph&):osg::Referenced() {}
2001-09-20 05:19:47 +08:00
/// disallow copy operator.
StateGraph& operator = (const StateGraph&) { return *this; }
2001-09-20 05:19:47 +08:00
};
}
2001-09-20 05:19:47 +08:00
#endif