2003-03-15 04:35:45 +08:00
# include <osgProducer/Viewer>
# include <osg/Transform>
# include <osg/Billboard>
# include <osg/Geode>
# include <osg/Group>
# include <osg/Notify>
# include <osgDB/Registry>
# include <osgDB/ReadFile>
# include <osgGA/TrackballManipulator>
# include <osgGA/FlightManipulator>
# include <osgGA/DriveManipulator>
# include <osgUtil/Optimizer>
class UpdateCallback : public osg : : NodeCallback
{
virtual void operator ( ) ( osg : : Node * node , osg : : NodeVisitor * nv )
{
std : : cout < < " update callback - pre traverse " < < node < < std : : endl ;
traverse ( node , nv ) ;
std : : cout < < " update callback - post traverse " < < node < < std : : endl ;
}
} ;
class CullCallback : public osg : : NodeCallback
{
virtual void operator ( ) ( osg : : Node * node , osg : : NodeVisitor * nv )
{
std : : cout < < " cull callback - pre traverse " < < node < < std : : endl ;
traverse ( node , nv ) ;
std : : cout < < " cull callback - post traverse " < < node < < std : : endl ;
}
} ;
class DrawableDrawCallback : public osg : : Drawable : : DrawCallback
{
virtual void drawImplementation ( osg : : State & state , const osg : : Drawable * drawable ) const
{
std : : cout < < " draw call back - pre drawImplementation " < < drawable < < std : : endl ;
drawable - > drawImplementation ( state ) ;
std : : cout < < " draw call back - post drawImplementation " < < drawable < < std : : endl ;
}
} ;
struct DrawableUpdateCallback : public osg : : Drawable : : UpdateCallback
{
virtual void update ( osg : : NodeVisitor * , osg : : Drawable * drawable )
{
std : : cout < < " Drawable update callback " < < drawable < < std : : endl ;
}
} ;
struct DrawableCullCallback : public osg : : Drawable : : CullCallback
{
/** do customized cull code.*/
virtual bool cull ( osg : : NodeVisitor * , osg : : Drawable * drawable , osg : : State * /*state*/ ) const
{
std : : cout < < " Drawable cull callback " < < drawable < < std : : endl ;
return false ;
}
} ;
class InsertCallbacksVisitor : public osg : : NodeVisitor
{
public :
InsertCallbacksVisitor ( ) : osg : : NodeVisitor ( osg : : NodeVisitor : : TRAVERSE_ALL_CHILDREN )
{
}
virtual void apply ( osg : : Node & node )
{
node . setUpdateCallback ( new UpdateCallback ( ) ) ;
node . setCullCallback ( new CullCallback ( ) ) ;
traverse ( node ) ;
}
virtual void apply ( osg : : Geode & geode )
{
geode . setUpdateCallback ( new UpdateCallback ( ) ) ;
//note, it makes no sense to attach a cull callback to the node
//at there are no nodes to traverse below the geode, only
//drawables, and as such the Cull node callbacks is ignored.
//If you wish to control the culling of drawables
//then use a drawable cullback...
for ( unsigned int i = 0 ; i < geode . getNumDrawables ( ) ; + + i )
{
geode . getDrawable ( i ) - > setUpdateCallback ( new DrawableUpdateCallback ( ) ) ;
geode . getDrawable ( i ) - > setCullCallback ( new DrawableCullCallback ( ) ) ;
geode . getDrawable ( i ) - > setDrawCallback ( new DrawableDrawCallback ( ) ) ;
}
}
virtual void apply ( osg : : Transform & node )
{
apply ( ( osg : : Node & ) node ) ;
}
} ;
2004-04-11 00:11:56 +08:00
class MyReadFileCallback : public osgDB : : Registry : : ReadFileCallback
{
public :
virtual osgDB : : ReaderWriter : : ReadResult readNode ( const std : : string & fileName , osgDB : : Registry : : CacheHintOptions useObjectCache )
{
std : : cout < < " before readNode " < < std : : endl ;
// note when calling the Registry to do the read you have to call readNodeImplementation NOT readNode, as this will
// cause on infinite recusive loop.
osgDB : : ReaderWriter : : ReadResult result = osgDB : : Registry : : instance ( ) - > readNodeImplementation ( fileName , useObjectCache ) ;
std : : cout < < " after readNode " < < std : : endl ;
return result ;
}
} ;
2003-03-15 04:35:45 +08:00
int main ( int argc , char * * argv )
{
// use an ArgumentParser object to manage the program arguments.
osg : : ArgumentParser arguments ( & argc , argv ) ;
// set up the usage document, in case we need to print out how to use this program.
2003-04-09 19:44:32 +08:00
arguments . getApplicationUsage ( ) - > setDescription ( arguments . getApplicationName ( ) + " is the example which demonstrates use of the range of different types of callbacks supported in the OpenSceneGraph. " ) ;
2003-04-07 05:32:44 +08:00
arguments . getApplicationUsage ( ) - > setCommandLineUsage ( arguments . getApplicationName ( ) + " [options] filename ... " ) ;
2003-03-15 04:35:45 +08:00
arguments . getApplicationUsage ( ) - > addCommandLineOption ( " -h or --help " , " Display this information " ) ;
2004-04-11 00:11:56 +08:00
// set the osgDB::Registy the read file callback to catch all requests for reading files.
osgDB : : Registry : : instance ( ) - > setReadFileCallback ( new MyReadFileCallback ( ) ) ;
2003-03-15 04:35:45 +08:00
// initialize the viewer.
osgProducer : : Viewer viewer ( arguments ) ;
// set up the value with sensible default event handlers.
viewer . setUpViewer ( osgProducer : : Viewer : : STANDARD_SETTINGS ) ;
// get details on keyboard and mouse bindings used by the viewer.
viewer . getUsage ( * arguments . getApplicationUsage ( ) ) ;
// if user request help write it out to cout.
if ( arguments . read ( " -h " ) | | arguments . read ( " --help " ) )
{
arguments . getApplicationUsage ( ) - > write ( std : : cout ) ;
return 1 ;
}
// any option left unread are converted into errors to write out later.
arguments . reportRemainingOptionsAsUnrecognized ( ) ;
// report any errors if they have occured when parsing the program aguments.
if ( arguments . errors ( ) )
{
arguments . writeErrorMessages ( std : : cout ) ;
return 1 ;
}
2003-04-07 05:32:44 +08:00
if ( arguments . argc ( ) < = 1 )
{
arguments . getApplicationUsage ( ) - > write ( std : : cout , osg : : ApplicationUsage : : COMMAND_LINE_OPTION ) ;
return 1 ;
}
2003-03-15 04:35:45 +08:00
// load the nodes from the commandline arguments.
osg : : Node * rootnode = osgDB : : readNodeFiles ( arguments ) ;
if ( ! rootnode )
{
// write_usage(osg::notify(osg::NOTICE),argv[0]);
return 1 ;
}
// run optimization over the scene graph
osgUtil : : Optimizer optimzer ;
optimzer . optimize ( rootnode ) ;
// insert all the callbacks
InsertCallbacksVisitor icv ;
rootnode - > accept ( icv ) ;
// set the scene to render
viewer . setSceneData ( rootnode ) ;
// create the windows and run the threads.
2003-04-08 23:18:45 +08:00
viewer . realize ( ) ;
2003-03-15 04:35:45 +08:00
while ( ! viewer . done ( ) )
{
// wait for all cull and draw threads to complete.
viewer . sync ( ) ;
// update the scene by traversing it with the the update visitor which will
// call all node update callbacks and animations.
viewer . update ( ) ;
// fire off the cull and draw traversals of the scene.
viewer . frame ( ) ;
}
2003-03-25 18:05:09 +08:00
// wait for all cull and draw threads to complete before exit.
viewer . sync ( ) ;
2003-03-15 04:35:45 +08:00
return 0 ;
}