forked from teamnwah/openmw-tes3coop
eef63a880a
osgFX::Effect is awkward to use because of the lazy-definition of passes, no support for compileGLObjects, useless 'Technique' abstraction and having to define silly methods like 'effectAuthor()' Handling the multi-pass rendering inside the Drawable also avoids redundant culling tests against the same bounding box for each pass.
85 lines
2 KiB
C++
85 lines
2 KiB
C++
#include "terraindrawable.hpp"
|
|
|
|
#include <osgUtil/CullVisitor>
|
|
|
|
#include <components/sceneutil/lightmanager.hpp>
|
|
|
|
namespace Terrain
|
|
{
|
|
|
|
TerrainDrawable::TerrainDrawable()
|
|
{
|
|
mLightListCallback = new SceneUtil::LightListCallback;
|
|
}
|
|
|
|
TerrainDrawable::TerrainDrawable(const TerrainDrawable ©, const osg::CopyOp ©op)
|
|
: osg::Geometry(copy, copyop)
|
|
, mPasses(copy.mPasses)
|
|
, mLightListCallback(copy.mLightListCallback)
|
|
{
|
|
|
|
}
|
|
|
|
void TerrainDrawable::accept(osg::NodeVisitor &nv)
|
|
{
|
|
if (nv.getVisitorType() != osg::NodeVisitor::CULL_VISITOR)
|
|
{
|
|
osg::Geometry::accept(nv);
|
|
}
|
|
else if (nv.validNodeMask(*this))
|
|
{
|
|
nv.pushOntoNodePath(this);
|
|
cull(static_cast<osgUtil::CullVisitor*>(&nv));
|
|
nv.popFromNodePath();
|
|
}
|
|
}
|
|
|
|
inline float distance(const osg::Vec3& coord,const osg::Matrix& matrix)
|
|
{
|
|
return -((float)coord[0]*(float)matrix(0,2)+(float)coord[1]*(float)matrix(1,2)+(float)coord[2]*(float)matrix(2,2)+matrix(3,2));
|
|
}
|
|
|
|
void TerrainDrawable::cull(osgUtil::CullVisitor *cv)
|
|
{
|
|
const osg::BoundingBox& bb = getBoundingBox();
|
|
|
|
if (cv->isCulled(getBoundingBox()))
|
|
return;
|
|
|
|
osg::RefMatrix& matrix = *cv->getModelViewMatrix();
|
|
|
|
float depth = bb.valid() ? distance(bb.center(),matrix) : 0.0f;
|
|
if (osg::isNaN(depth))
|
|
return;
|
|
|
|
bool pushedLight = mLightListCallback->pushLightState(this, cv);
|
|
|
|
for (PassVector::const_iterator it = mPasses.begin(); it != mPasses.end(); ++it)
|
|
{
|
|
cv->pushStateSet(*it);
|
|
cv->addDrawableAndDepth(this, &matrix, depth);
|
|
cv->popStateSet();
|
|
}
|
|
|
|
if (pushedLight)
|
|
cv->popStateSet();
|
|
}
|
|
|
|
void TerrainDrawable::setPasses(const TerrainDrawable::PassVector &passes)
|
|
{
|
|
mPasses = passes;
|
|
}
|
|
|
|
void TerrainDrawable::compileGLObjects(osg::RenderInfo &renderInfo) const
|
|
{
|
|
for (PassVector::const_iterator it = mPasses.begin(); it != mPasses.end(); ++it)
|
|
{
|
|
osg::StateSet* stateset = *it;
|
|
stateset->compileGLObjects(*renderInfo.getState());
|
|
}
|
|
|
|
osg::Geometry::compileGLObjects(renderInfo);
|
|
}
|
|
|
|
}
|
|
|