2003-01-22 00:45:36 +08:00
|
|
|
/* -*-c++-*- OpenSceneGraph - Copyright (C) 1998-2003 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.
|
|
|
|
*/
|
2002-06-04 01:49:28 +08:00
|
|
|
|
|
|
|
#ifndef OSG_CullingSet
|
|
|
|
#define OSG_CullingSet 1
|
|
|
|
|
|
|
|
#include <osg/Polytope>
|
2002-06-12 21:54:14 +08:00
|
|
|
#include <osg/ShadowVolumeOccluder>
|
2002-06-04 01:49:28 +08:00
|
|
|
#include <osg/Referenced>
|
|
|
|
|
|
|
|
namespace osg {
|
|
|
|
|
2004-08-31 22:49:33 +08:00
|
|
|
/** A CullingSet class which contains a frustum and a list of occluders. */
|
2002-06-04 01:49:28 +08:00
|
|
|
class SG_EXPORT CullingSet : public Referenced
|
|
|
|
{
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
|
|
|
|
CullingSet();
|
2003-09-26 05:54:33 +08:00
|
|
|
|
|
|
|
CullingSet(const CullingSet& cs):
|
|
|
|
Referenced(),
|
|
|
|
_mask(cs._mask),
|
|
|
|
_frustum(cs._frustum),
|
|
|
|
_occluderList(cs._occluderList),
|
|
|
|
_pixelSizeVector(cs._pixelSizeVector),
|
|
|
|
_smallFeatureCullingPixelSize(cs._smallFeatureCullingPixelSize)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2002-06-04 23:21:24 +08:00
|
|
|
CullingSet(const CullingSet& cs,const Matrix& matrix, const Vec4& pixelSizeVector):
|
2002-06-04 01:49:28 +08:00
|
|
|
_mask(cs._mask),
|
2002-06-10 19:21:21 +08:00
|
|
|
_frustum(cs._frustum),
|
2002-06-04 01:49:28 +08:00
|
|
|
_occluderList(cs._occluderList),
|
2002-06-04 23:21:24 +08:00
|
|
|
_pixelSizeVector(pixelSizeVector),
|
|
|
|
_smallFeatureCullingPixelSize(cs._smallFeatureCullingPixelSize)
|
2002-06-04 01:49:28 +08:00
|
|
|
{
|
2002-06-10 19:21:21 +08:00
|
|
|
_frustum.transformProvidingInverse(matrix);
|
2002-06-04 01:49:28 +08:00
|
|
|
for(OccluderList::iterator itr=_occluderList.begin();
|
|
|
|
itr!=_occluderList.end();
|
|
|
|
++itr)
|
|
|
|
{
|
|
|
|
itr->transformProvidingInverse(matrix);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-09-28 18:15:10 +08:00
|
|
|
CullingSet& operator = (const CullingSet& cs)
|
|
|
|
{
|
|
|
|
if (this==&cs) return *this;
|
|
|
|
|
|
|
|
_mask = cs._mask;
|
|
|
|
_frustum = cs._frustum;
|
|
|
|
_occluderList = cs._occluderList;
|
|
|
|
_pixelSizeVector = cs._pixelSizeVector;
|
|
|
|
_smallFeatureCullingPixelSize = cs._smallFeatureCullingPixelSize;
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-09-26 05:54:33 +08:00
|
|
|
inline void set(const CullingSet& cs)
|
|
|
|
{
|
|
|
|
_mask = cs._mask;
|
|
|
|
_frustum = cs._frustum;
|
|
|
|
_occluderList = cs._occluderList;
|
|
|
|
_pixelSizeVector = cs._pixelSizeVector;
|
|
|
|
_smallFeatureCullingPixelSize = cs._smallFeatureCullingPixelSize;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void set(const CullingSet& cs,const Matrix& matrix, const Vec4& pixelSizeVector)
|
|
|
|
{
|
|
|
|
_mask = cs._mask;
|
|
|
|
_occluderList = cs._occluderList;
|
|
|
|
_pixelSizeVector = pixelSizeVector;
|
|
|
|
_smallFeatureCullingPixelSize = cs._smallFeatureCullingPixelSize;
|
|
|
|
|
2003-12-04 05:45:32 +08:00
|
|
|
//_frustum = cs._frustum;
|
|
|
|
//_frustum.transformProvidingInverse(matrix);
|
|
|
|
|
|
|
|
_frustum.setAndTransformProvidingInverse(cs._frustum,matrix);
|
2003-09-26 05:54:33 +08:00
|
|
|
|
|
|
|
for(OccluderList::iterator itr=_occluderList.begin();
|
|
|
|
itr!=_occluderList.end();
|
|
|
|
++itr)
|
|
|
|
{
|
|
|
|
itr->transformProvidingInverse(matrix);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2002-06-10 19:21:21 +08:00
|
|
|
typedef std::vector<ShadowVolumeOccluder> OccluderList;
|
2002-06-04 01:49:28 +08:00
|
|
|
|
|
|
|
typedef unsigned int Mask;
|
|
|
|
|
|
|
|
enum MaskValues
|
|
|
|
{
|
2003-08-20 20:50:54 +08:00
|
|
|
NO_CULLING = 0x0,
|
|
|
|
VIEW_FRUSTUM_SIDES_CULLING = 0x1,
|
|
|
|
NEAR_PLANE_CULLING = 0x2,
|
|
|
|
FAR_PLANE_CULLING = 0x4,
|
|
|
|
VIEW_FRUSTUM_CULLING = VIEW_FRUSTUM_SIDES_CULLING|
|
|
|
|
NEAR_PLANE_CULLING|
|
|
|
|
FAR_PLANE_CULLING,
|
|
|
|
SMALL_FEATURE_CULLING = 0x8,
|
|
|
|
SHADOW_OCCLUSION_CULLING = 0x10,
|
|
|
|
DEFAULT_CULLING = VIEW_FRUSTUM_SIDES_CULLING|
|
|
|
|
SMALL_FEATURE_CULLING|
|
|
|
|
SHADOW_OCCLUSION_CULLING,
|
|
|
|
ENABLE_ALL_CULLING = VIEW_FRUSTUM_CULLING|
|
|
|
|
SMALL_FEATURE_CULLING|
|
|
|
|
SHADOW_OCCLUSION_CULLING
|
2002-06-04 01:49:28 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
void setCullingMask(Mask mask) { _mask = mask; }
|
|
|
|
|
2002-06-10 19:21:21 +08:00
|
|
|
void setFrustum(Polytope& cv) { _frustum = cv; }
|
2002-06-04 01:49:28 +08:00
|
|
|
|
2002-06-10 19:21:21 +08:00
|
|
|
Polytope& getFrustum() { return _frustum; }
|
2002-06-15 20:14:42 +08:00
|
|
|
const Polytope& getFrustum() const { return _frustum; }
|
2002-06-04 01:49:28 +08:00
|
|
|
|
2002-06-10 19:21:21 +08:00
|
|
|
void addOccluder(ShadowVolumeOccluder& cv) { _occluderList.push_back(cv); }
|
2002-06-04 01:49:28 +08:00
|
|
|
|
2002-06-04 23:21:24 +08:00
|
|
|
|
|
|
|
void setPixelSizeVector(const Vec4& v) { _pixelSizeVector = v; }
|
|
|
|
|
|
|
|
Vec4& getPixelSizeVector() { return _pixelSizeVector; }
|
|
|
|
const Vec4& getPixelSizeVector() const { return _pixelSizeVector; }
|
|
|
|
|
|
|
|
void setSmallFeatureCullingPixelSize(float value) { _smallFeatureCullingPixelSize=value; }
|
|
|
|
|
|
|
|
float& getSmallFeatureCullingPixelSize() { return _smallFeatureCullingPixelSize; }
|
|
|
|
|
|
|
|
float getSmallFeatureCullingPixelSize() const { return _smallFeatureCullingPixelSize; }
|
|
|
|
|
|
|
|
|
|
|
|
/** Compute the pixel of an object at position v, with specified radius.*/
|
|
|
|
float pixelSize(const Vec3& v,float radius) const { return radius/(v*_pixelSizeVector); }
|
|
|
|
|
2004-08-31 22:49:33 +08:00
|
|
|
/** Compute the pixel of a bounding sphere.*/
|
2002-06-04 23:21:24 +08:00
|
|
|
float pixelSize(const BoundingSphere& bs) const { return bs.radius()/(bs.center()*_pixelSizeVector); }
|
|
|
|
|
2002-06-09 03:58:05 +08:00
|
|
|
|
2002-06-12 17:22:30 +08:00
|
|
|
inline bool isCulled(const std::vector<Vec3>& vertices)
|
|
|
|
{
|
|
|
|
if (_mask&VIEW_FRUSTUM_CULLING)
|
|
|
|
{
|
|
|
|
// is it outside the view frustum...
|
|
|
|
if (!_frustum.contains(vertices)) return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_mask&SMALL_FEATURE_CULLING)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_mask&SHADOW_OCCLUSION_CULLING)
|
|
|
|
{
|
|
|
|
// is it in one of the shadow occluder volumes.
|
|
|
|
if (!_occluderList.empty())
|
|
|
|
{
|
|
|
|
for(OccluderList::iterator itr=_occluderList.begin();
|
|
|
|
itr!=_occluderList.end();
|
|
|
|
++itr)
|
|
|
|
{
|
|
|
|
if (itr->contains(vertices)) return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2002-06-04 01:49:28 +08:00
|
|
|
inline bool isCulled(const BoundingBox& bb)
|
|
|
|
{
|
|
|
|
if (_mask&VIEW_FRUSTUM_CULLING)
|
|
|
|
{
|
|
|
|
// is it outside the view frustum...
|
2002-06-10 19:21:21 +08:00
|
|
|
if (!_frustum.contains(bb)) return true;
|
2002-06-04 01:49:28 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (_mask&SMALL_FEATURE_CULLING)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_mask&SHADOW_OCCLUSION_CULLING)
|
|
|
|
{
|
|
|
|
// is it in one of the shadow occluder volumes.
|
|
|
|
if (!_occluderList.empty())
|
|
|
|
{
|
|
|
|
for(OccluderList::iterator itr=_occluderList.begin();
|
|
|
|
itr!=_occluderList.end();
|
|
|
|
++itr)
|
|
|
|
{
|
|
|
|
if (itr->contains(bb)) return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline bool isCulled(const BoundingSphere& bs)
|
|
|
|
{
|
|
|
|
if (_mask&VIEW_FRUSTUM_CULLING)
|
|
|
|
{
|
|
|
|
// is it outside the view frustum...
|
2002-06-10 19:21:21 +08:00
|
|
|
if (!_frustum.contains(bs)) return true;
|
2002-06-04 01:49:28 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (_mask&SMALL_FEATURE_CULLING)
|
|
|
|
{
|
2002-06-04 23:21:24 +08:00
|
|
|
if (((bs.center()*_pixelSizeVector)*_smallFeatureCullingPixelSize)>bs.radius()) return true;
|
2002-06-04 01:49:28 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (_mask&SHADOW_OCCLUSION_CULLING)
|
|
|
|
{
|
|
|
|
// is it in one of the shadow occluder volumes.
|
|
|
|
if (!_occluderList.empty())
|
|
|
|
{
|
|
|
|
for(OccluderList::iterator itr=_occluderList.begin();
|
|
|
|
itr!=_occluderList.end();
|
|
|
|
++itr)
|
|
|
|
{
|
|
|
|
if (itr->contains(bs)) return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void pushCurrentMask()
|
|
|
|
{
|
2002-06-10 19:21:21 +08:00
|
|
|
_frustum.pushCurrentMask();
|
2002-06-04 01:49:28 +08:00
|
|
|
|
|
|
|
if (!_occluderList.empty())
|
|
|
|
{
|
|
|
|
for(OccluderList::iterator itr=_occluderList.begin();
|
|
|
|
itr!=_occluderList.end();
|
|
|
|
++itr)
|
|
|
|
{
|
|
|
|
itr->pushCurrentMask();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void popCurrentMask()
|
|
|
|
{
|
2002-06-10 19:21:21 +08:00
|
|
|
_frustum.popCurrentMask();
|
2002-06-04 01:49:28 +08:00
|
|
|
|
|
|
|
if (!_occluderList.empty())
|
|
|
|
{
|
|
|
|
for(OccluderList::iterator itr=_occluderList.begin();
|
|
|
|
itr!=_occluderList.end();
|
|
|
|
++itr)
|
|
|
|
{
|
|
|
|
itr->popCurrentMask();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-06-17 17:10:26 +08:00
|
|
|
void disableAndPushOccludersCurrentMask(NodePath& nodePath);
|
|
|
|
|
|
|
|
void popOccludersCurrentMask(NodePath& nodePath);
|
2002-06-04 01:49:28 +08:00
|
|
|
|
2003-09-26 05:54:33 +08:00
|
|
|
virtual ~CullingSet();
|
|
|
|
|
2002-06-04 01:49:28 +08:00
|
|
|
protected:
|
|
|
|
|
2003-01-10 17:25:42 +08:00
|
|
|
|
2002-06-04 23:21:24 +08:00
|
|
|
Mask _mask;
|
2002-06-10 19:21:21 +08:00
|
|
|
Polytope _frustum;
|
2002-06-04 23:21:24 +08:00
|
|
|
OccluderList _occluderList;
|
|
|
|
Vec4 _pixelSizeVector;
|
|
|
|
float _smallFeatureCullingPixelSize;
|
2002-06-04 01:49:28 +08:00
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
} // end of namespace
|
|
|
|
|
|
|
|
#endif
|