2009-12-09 01:41:44 +08:00
|
|
|
// -*-c++-*-
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Draw an outline around a model.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <osg/Group>
|
2010-09-17 19:20:11 +08:00
|
|
|
#include <osg/PositionAttitudeTransform>
|
2009-12-09 01:41:44 +08:00
|
|
|
|
|
|
|
#include <osgDB/ReadFile>
|
|
|
|
#include <osgViewer/Viewer>
|
|
|
|
|
|
|
|
#include <osgFX/Outline>
|
|
|
|
|
|
|
|
|
|
|
|
int main(int argc, char** argv)
|
|
|
|
{
|
|
|
|
osg::ArgumentParser arguments(&argc,argv);
|
|
|
|
arguments.getApplicationUsage()->setCommandLineUsage(arguments.getApplicationName()+" [options] <file>");
|
2010-09-17 19:20:11 +08:00
|
|
|
arguments.getApplicationUsage()->addCommandLineOption("--testOcclusion","Test occlusion by other objects");
|
2009-12-09 01:41:44 +08:00
|
|
|
arguments.getApplicationUsage()->addCommandLineOption("-h or --help","Display this information");
|
2010-09-17 19:20:11 +08:00
|
|
|
|
|
|
|
bool testOcclusion = false;
|
|
|
|
while (arguments.read("--testOcclusion")) { testOcclusion = true; }
|
|
|
|
|
|
|
|
// load outlined object
|
|
|
|
std::string modelFilename = arguments.argc() > 1 ? arguments[1] : "dumptruck.osg";
|
|
|
|
osg::ref_ptr<osg::Node> outlineModel = osgDB::readNodeFile(modelFilename);
|
|
|
|
if (!outlineModel)
|
|
|
|
{
|
|
|
|
osg::notify(osg::FATAL) << "Unable to load model '" << modelFilename << "'\n";
|
|
|
|
return -1;
|
|
|
|
}
|
2009-12-09 01:41:44 +08:00
|
|
|
|
|
|
|
// create scene
|
|
|
|
osg::ref_ptr<osg::Group> root = new osg::Group;
|
|
|
|
|
2010-09-17 19:20:11 +08:00
|
|
|
{
|
|
|
|
// create outline effect
|
|
|
|
osg::ref_ptr<osgFX::Outline> outline = new osgFX::Outline;
|
|
|
|
root->addChild(outline.get());
|
|
|
|
|
|
|
|
outline->setWidth(8);
|
|
|
|
outline->setColor(osg::Vec4(1,1,0,1));
|
|
|
|
outline->addChild(outlineModel.get());
|
|
|
|
}
|
|
|
|
|
|
|
|
if (testOcclusion)
|
|
|
|
{
|
|
|
|
// load occluder
|
|
|
|
std::string occludedModelFilename = "cow.osg";
|
|
|
|
osg::ref_ptr<osg::Node> occludedModel = osgDB::readNodeFile(occludedModelFilename);
|
|
|
|
if (!occludedModel)
|
|
|
|
{
|
|
|
|
osg::notify(osg::FATAL) << "Unable to load model '" << occludedModelFilename << "'\n";
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// occluder offset
|
|
|
|
const osg::BoundingSphere& bsphere = outlineModel->getBound();
|
|
|
|
const osg::Vec3 occluderOffset = osg::Vec3(0,1,0) * bsphere.radius() * 1.2f;
|
|
|
|
|
|
|
|
// occluder behind outlined model
|
|
|
|
osg::ref_ptr<osg::PositionAttitudeTransform> modelTransform0 = new osg::PositionAttitudeTransform;
|
|
|
|
modelTransform0->setPosition(bsphere.center() + occluderOffset);
|
|
|
|
modelTransform0->addChild(occludedModel.get());
|
|
|
|
root->addChild(modelTransform0.get());
|
|
|
|
|
|
|
|
// occluder in front of outlined model
|
|
|
|
osg::ref_ptr<osg::PositionAttitudeTransform> modelTransform1 = new osg::PositionAttitudeTransform;
|
|
|
|
modelTransform1->setPosition(bsphere.center() - occluderOffset);
|
|
|
|
modelTransform1->addChild(occludedModel.get());
|
|
|
|
root->addChild(modelTransform1.get());
|
|
|
|
}
|
2009-12-09 01:41:44 +08:00
|
|
|
|
|
|
|
// must have stencil buffer...
|
|
|
|
osg::DisplaySettings::instance()->setMinimumNumStencilBits(1);
|
|
|
|
|
|
|
|
// construct the viewer
|
|
|
|
osgViewer::Viewer viewer;
|
2010-01-13 21:09:48 +08:00
|
|
|
viewer.setSceneData(root.get());
|
2009-12-09 01:41:44 +08:00
|
|
|
|
|
|
|
// must clear stencil buffer...
|
|
|
|
unsigned int clearMask = viewer.getCamera()->getClearMask();
|
|
|
|
viewer.getCamera()->setClearMask(clearMask | GL_STENCIL_BUFFER_BIT);
|
|
|
|
viewer.getCamera()->setClearStencil(0);
|
|
|
|
|
|
|
|
return viewer.run();
|
|
|
|
}
|