mirror of
https://git.dev.opencascade.org/repos/occt.git
synced 2025-04-04 18:06:22 +03:00
0031511: Point Cloud Rendering, Volume Rendering - reuse Graphic3d_CullingTool
Graphic3d_CullingTool::IsCulled() has been extended with theIsInside argument for full inclusion test. Graphic3d_Layer::UpdateCulling() now avoids frustum culling tests for BVH children for parent nodes completely included into frustum. Graphic3d_CullingTool::SetViewVolume() has been extended by optional model-world matrix.
This commit is contained in:
parent
89fcfe1551
commit
9ad4ff93a0
@ -1475,7 +1475,8 @@ Standard_EXPORT void NCollection_Lerp<Handle(Graphic3d_Camera)>::Interpolate (co
|
|||||||
//function : FrustumPoints
|
//function : FrustumPoints
|
||||||
//purpose :
|
//purpose :
|
||||||
//=======================================================================
|
//=======================================================================
|
||||||
void Graphic3d_Camera::FrustumPoints (NCollection_Array1<Graphic3d_Vec3d>& thePoints) const
|
void Graphic3d_Camera::FrustumPoints (NCollection_Array1<Graphic3d_Vec3d>& thePoints,
|
||||||
|
const Graphic3d_Mat4d& theModelWorld) const
|
||||||
{
|
{
|
||||||
if (thePoints.Length() != FrustumVerticesNB)
|
if (thePoints.Length() != FrustumVerticesNB)
|
||||||
{
|
{
|
||||||
@ -1483,7 +1484,7 @@ void Graphic3d_Camera::FrustumPoints (NCollection_Array1<Graphic3d_Vec3d>& thePo
|
|||||||
}
|
}
|
||||||
|
|
||||||
const Graphic3d_Mat4d& aProjectionMat = ProjectionMatrix();
|
const Graphic3d_Mat4d& aProjectionMat = ProjectionMatrix();
|
||||||
const Graphic3d_Mat4d& aWorldViewMat = OrientationMatrix();
|
const Graphic3d_Mat4d aWorldViewMat = OrientationMatrix() * theModelWorld;
|
||||||
|
|
||||||
Standard_Real nLeft = 0.0, nRight = 0.0, nTop = 0.0, nBottom = 0.0;
|
Standard_Real nLeft = 0.0, nRight = 0.0, nTop = 0.0, nBottom = 0.0;
|
||||||
Standard_Real fLeft = 0.0, fRight = 0.0, fTop = 0.0, fBottom = 0.0;
|
Standard_Real fLeft = 0.0, fRight = 0.0, fTop = 0.0, fBottom = 0.0;
|
||||||
|
@ -670,7 +670,8 @@ public:
|
|||||||
//! Fill array of current view frustum corners.
|
//! Fill array of current view frustum corners.
|
||||||
//! The size of this array is equal to FrustumVerticesNB.
|
//! The size of this array is equal to FrustumVerticesNB.
|
||||||
//! The order of vertices is as defined in FrustumVert_* enumeration.
|
//! The order of vertices is as defined in FrustumVert_* enumeration.
|
||||||
Standard_EXPORT void FrustumPoints (NCollection_Array1<Graphic3d_Vec3d>& thePoints) const;
|
Standard_EXPORT void FrustumPoints (NCollection_Array1<Graphic3d_Vec3d>& thePoints,
|
||||||
|
const Graphic3d_Mat4d& theModelWorld = Graphic3d_Mat4d()) const;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
|
||||||
|
@ -34,12 +34,17 @@ Graphic3d_CullingTool::Graphic3d_CullingTool()
|
|||||||
|
|
||||||
// =======================================================================
|
// =======================================================================
|
||||||
// function : SetViewVolume
|
// function : SetViewVolume
|
||||||
// purpose : Retrieves view volume's planes equations and its vertices from projection and world-view matrices.
|
// purpose :
|
||||||
// =======================================================================
|
// =======================================================================
|
||||||
void Graphic3d_CullingTool::SetViewVolume (const Handle(Graphic3d_Camera)& theCamera)
|
void Graphic3d_CullingTool::SetViewVolume (const Handle(Graphic3d_Camera)& theCamera,
|
||||||
|
const Graphic3d_Mat4d& theModelWorld)
|
||||||
|
{
|
||||||
|
const bool hasModelTrsf = !theModelWorld.IsIdentity();
|
||||||
|
if (!myWorldViewProjState.IsChanged (theCamera->WorldViewProjState())
|
||||||
|
&& !hasModelTrsf)
|
||||||
{
|
{
|
||||||
if (!myWorldViewProjState.IsChanged (theCamera->WorldViewProjState()))
|
|
||||||
return;
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
myIsProjectionParallel = theCamera->IsOrthographic();
|
myIsProjectionParallel = theCamera->IsOrthographic();
|
||||||
const gp_Dir aCamDir = theCamera->Direction();
|
const gp_Dir aCamDir = theCamera->Direction();
|
||||||
@ -50,12 +55,19 @@ void Graphic3d_CullingTool::SetViewVolume (const Handle(Graphic3d_Camera)& theCa
|
|||||||
myWorldViewProjState = theCamera->WorldViewProjState();
|
myWorldViewProjState = theCamera->WorldViewProjState();
|
||||||
myCamEye.SetValues (theCamera->Eye().X(), theCamera->Eye().Y(), theCamera->Eye().Z());
|
myCamEye.SetValues (theCamera->Eye().X(), theCamera->Eye().Y(), theCamera->Eye().Z());
|
||||||
myCamDir.SetValues (aCamDir.X(), aCamDir.Y(), aCamDir.Z());
|
myCamDir.SetValues (aCamDir.X(), aCamDir.Y(), aCamDir.Z());
|
||||||
|
if (hasModelTrsf)
|
||||||
|
{
|
||||||
|
Graphic3d_Mat4d aModelInv;
|
||||||
|
theModelWorld.Inverted (aModelInv);
|
||||||
|
myCamEye = (aModelInv * Graphic3d_Vec4d (myCamEye, 1.0)).xyz();
|
||||||
|
myCamDir = (aModelInv * Graphic3d_Vec4d (myCamDir, 0.0)).xyz();
|
||||||
|
}
|
||||||
myCamScale = theCamera->IsOrthographic()
|
myCamScale = theCamera->IsOrthographic()
|
||||||
? theCamera->Scale()
|
? theCamera->Scale()
|
||||||
: 2.0 * Tan (theCamera->FOVy() * M_PI / 360.0); // same as theCamera->Scale()/theCamera->Distance()
|
: 2.0 * Tan (theCamera->FOVy() * M_PI / 360.0); // same as theCamera->Scale()/theCamera->Distance()
|
||||||
|
|
||||||
// Compute frustum points
|
// Compute frustum points
|
||||||
theCamera->FrustumPoints (myClipVerts);
|
theCamera->FrustumPoints (myClipVerts, theModelWorld);
|
||||||
|
|
||||||
// Compute frustum planes
|
// Compute frustum planes
|
||||||
// Vertices go in order:
|
// Vertices go in order:
|
||||||
|
@ -59,7 +59,10 @@ public:
|
|||||||
Standard_EXPORT Graphic3d_CullingTool();
|
Standard_EXPORT Graphic3d_CullingTool();
|
||||||
|
|
||||||
//! Retrieves view volume's planes equations and its vertices from projection and world-view matrices.
|
//! Retrieves view volume's planes equations and its vertices from projection and world-view matrices.
|
||||||
Standard_EXPORT void SetViewVolume (const Handle(Graphic3d_Camera)& theCamera);
|
//! @param theCamera [in] camera definition
|
||||||
|
//! @param theModelWorld [in] optional object transformation for computing frustum in object local coordinate system
|
||||||
|
Standard_EXPORT void SetViewVolume (const Handle(Graphic3d_Camera)& theCamera,
|
||||||
|
const Graphic3d_Mat4d& theModelWorld = Graphic3d_Mat4d());
|
||||||
|
|
||||||
Standard_EXPORT void SetViewportSize (Standard_Integer theViewportWidth,
|
Standard_EXPORT void SetViewportSize (Standard_Integer theViewportWidth,
|
||||||
Standard_Integer theViewportHeight,
|
Standard_Integer theViewportHeight,
|
||||||
@ -79,16 +82,19 @@ public:
|
|||||||
|
|
||||||
//! Checks whether given AABB should be entirely culled or not.
|
//! Checks whether given AABB should be entirely culled or not.
|
||||||
//! @param theCtx [in] culling properties
|
//! @param theCtx [in] culling properties
|
||||||
//! @param theMinPt [in] maximum point of AABB
|
//! @param theMinPnt [in] maximum point of AABB
|
||||||
//! @param theMaxPt [in] minimum point of AABB
|
//! @param theMaxPnt [in] minimum point of AABB
|
||||||
//! @return Standard_True, if AABB is in viewing area, Standard_False otherwise
|
//! @param theIsInside [out] flag indicating if AABB is fully inside; initial value should be set to TRUE
|
||||||
|
//! @return TRUE if AABB is completely outside of view frustum or culled by size/distance;
|
||||||
|
//! FALSE in case of partial or complete overlap (use theIsInside to distinguish)
|
||||||
bool IsCulled (const CullingContext& theCtx,
|
bool IsCulled (const CullingContext& theCtx,
|
||||||
const Graphic3d_Vec3d& theMinPt,
|
const Graphic3d_Vec3d& theMinPnt,
|
||||||
const Graphic3d_Vec3d& theMaxPt) const
|
const Graphic3d_Vec3d& theMaxPnt,
|
||||||
|
Standard_Boolean* theIsInside = NULL) const
|
||||||
{
|
{
|
||||||
return isFullOut (theMinPt, theMaxPt)
|
return IsOutFrustum(theMinPnt, theMaxPnt, theIsInside)
|
||||||
|| isTooDistant(theCtx, theMinPt, theMaxPt)
|
|| IsTooDistant(theCtx, theMinPnt, theMaxPnt, theIsInside)
|
||||||
|| isTooSmall (theCtx, theMinPt, theMaxPt);
|
|| IsTooSmall (theCtx, theMinPnt, theMaxPnt);
|
||||||
}
|
}
|
||||||
|
|
||||||
//! Return the camera definition.
|
//! Return the camera definition.
|
||||||
@ -122,7 +128,13 @@ public:
|
|||||||
return myWorldViewProjState;
|
return myWorldViewProjState;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected:
|
//! Returns camera eye position.
|
||||||
|
const Graphic3d_Vec3d& CameraEye() const { return myCamEye; }
|
||||||
|
|
||||||
|
//! Returns camera direction.
|
||||||
|
const Graphic3d_Vec3d& CameraDirection() const { return myCamDir; }
|
||||||
|
|
||||||
|
public:
|
||||||
|
|
||||||
//! Calculates signed distance from plane to point.
|
//! Calculates signed distance from plane to point.
|
||||||
//! @param theNormal [in] the plane's normal.
|
//! @param theNormal [in] the plane's normal.
|
||||||
@ -131,76 +143,89 @@ protected:
|
|||||||
const Graphic3d_Vec4d& thePnt);
|
const Graphic3d_Vec4d& thePnt);
|
||||||
|
|
||||||
//! Detects if AABB overlaps view volume using separating axis theorem (SAT).
|
//! Detects if AABB overlaps view volume using separating axis theorem (SAT).
|
||||||
//! @param theMinPt [in] maximum point of AABB.
|
//! @param theMinPnt [in] maximum point of AABB
|
||||||
//! @param theMaxPt [in] minimum point of AABB.
|
//! @param theMaxPnt [in] minimum point of AABB
|
||||||
//! @return FALSE, if AABB is in viewing area, TRUE otherwise.
|
//! @param theIsInside [out] flag indicating if AABB is fully inside; initial value should be set to TRUE
|
||||||
bool isFullOut (const Graphic3d_Vec3d& theMinPt,
|
//! @return TRUE if AABB is completely outside of view frustum;
|
||||||
const Graphic3d_Vec3d& theMaxPt) const
|
//! FALSE in case of partial or complete overlap (use theIsInside to distinguish)
|
||||||
|
//! @sa SelectMgr_Frustum::hasOverlap()
|
||||||
|
bool IsOutFrustum (const Graphic3d_Vec3d& theMinPnt,
|
||||||
|
const Graphic3d_Vec3d& theMaxPnt,
|
||||||
|
Standard_Boolean* theIsInside = NULL) const
|
||||||
{
|
{
|
||||||
// E1
|
// E1
|
||||||
// |_ E0
|
// |_ E0
|
||||||
// /
|
// /
|
||||||
// E2
|
// E2
|
||||||
|
if (theMinPnt[0] > myMaxOrthoProjectionPts[0] // E0 test (x axis)
|
||||||
// E0 test (x axis)
|
|| theMaxPnt[0] < myMinOrthoProjectionPts[0]
|
||||||
if (theMinPt.x() > myMaxOrthoProjectionPts[0]
|
|| theMinPnt[1] > myMaxOrthoProjectionPts[1] // E1 test (y axis)
|
||||||
|| theMaxPt.x() < myMinOrthoProjectionPts[0])
|
|| theMaxPnt[1] < myMinOrthoProjectionPts[1]
|
||||||
|
|| theMinPnt[2] > myMaxOrthoProjectionPts[2] // E2 test (z axis)
|
||||||
|
|| theMaxPnt[2] < myMinOrthoProjectionPts[2])
|
||||||
{
|
{
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
if (theIsInside != NULL
|
||||||
// E1 test (y axis)
|
&& *theIsInside)
|
||||||
if (theMinPt.y() > myMaxOrthoProjectionPts[1]
|
|
||||||
|| theMaxPt.y() < myMinOrthoProjectionPts[1])
|
|
||||||
{
|
{
|
||||||
return true;
|
*theIsInside = theMinPnt[0] >= myMinOrthoProjectionPts[0] // E0 test (x axis)
|
||||||
}
|
&& theMaxPnt[0] <= myMaxOrthoProjectionPts[0]
|
||||||
|
&& theMinPnt[1] >= myMinOrthoProjectionPts[1] // E1 test (y axis)
|
||||||
// E2 test (z axis)
|
&& theMaxPnt[1] <= myMaxOrthoProjectionPts[1]
|
||||||
if (theMinPt.z() > myMaxOrthoProjectionPts[2]
|
&& theMinPnt[1] >= myMinOrthoProjectionPts[2] // E2 test (z axis)
|
||||||
|| theMaxPt.z() < myMinOrthoProjectionPts[2])
|
&& theMaxPnt[1] <= myMaxOrthoProjectionPts[2];
|
||||||
{
|
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const Standard_Integer anIncFactor = myIsProjectionParallel ? 2 : 1;
|
const Standard_Integer anIncFactor = myIsProjectionParallel ? 2 : 1;
|
||||||
for (Standard_Integer aPlaneIter = 0; aPlaneIter < PlanesNB - 1; aPlaneIter += anIncFactor)
|
for (Standard_Integer aPlaneIter = 0; aPlaneIter < PlanesNB - 1; aPlaneIter += anIncFactor)
|
||||||
{
|
{
|
||||||
// frustum normals
|
// frustum normals
|
||||||
const Graphic3d_Vec3d anAxis = myClipPlanes[aPlaneIter].Normal;
|
const Graphic3d_Vec3d& anAxis = myClipPlanes[aPlaneIter].Normal;
|
||||||
|
const Graphic3d_Vec3d aPVertex (anAxis.x() > 0.0 ? theMaxPnt.x() : theMinPnt.x(),
|
||||||
const Graphic3d_Vec3d aPVertex (anAxis.x() > 0.0 ? theMaxPt.x() : theMinPt.x(),
|
anAxis.y() > 0.0 ? theMaxPnt.y() : theMinPnt.y(),
|
||||||
anAxis.y() > 0.0 ? theMaxPt.y() : theMinPt.y(),
|
anAxis.z() > 0.0 ? theMaxPnt.z() : theMinPnt.z());
|
||||||
anAxis.z() > 0.0 ? theMaxPt.z() : theMinPt.z());
|
const Standard_Real aPnt0 = aPVertex.Dot (anAxis);
|
||||||
Standard_Real aPnt0 = aPVertex.Dot (anAxis);
|
if (theIsInside == NULL
|
||||||
|
&& aPnt0 >= myMinClipProjectionPts[aPlaneIter]
|
||||||
if (aPnt0 >= myMinClipProjectionPts[aPlaneIter]
|
|
||||||
&& aPnt0 <= myMaxClipProjectionPts[aPlaneIter])
|
&& aPnt0 <= myMaxClipProjectionPts[aPlaneIter])
|
||||||
{
|
{
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
const Graphic3d_Vec3d aNVertex (anAxis.x() > 0.0 ? theMinPt.x() : theMaxPt.x(),
|
const Graphic3d_Vec3d aNVertex (anAxis.x() > 0.0 ? theMinPnt.x() : theMaxPnt.x(),
|
||||||
anAxis.y() > 0.0 ? theMinPt.y() : theMaxPt.y(),
|
anAxis.y() > 0.0 ? theMinPnt.y() : theMaxPnt.y(),
|
||||||
anAxis.z() > 0.0 ? theMinPt.z() : theMaxPt.z());
|
anAxis.z() > 0.0 ? theMinPnt.z() : theMaxPnt.z());
|
||||||
Standard_Real aPnt1 = aNVertex.Dot (anAxis);
|
const Standard_Real aPnt1 = aNVertex.Dot (anAxis);
|
||||||
|
|
||||||
const Standard_Real aMin = aPnt0 < aPnt1 ? aPnt0 : aPnt1;
|
const Standard_Real aBoxProjMin = aPnt0 < aPnt1 ? aPnt0 : aPnt1;
|
||||||
const Standard_Real aMax = aPnt0 > aPnt1 ? aPnt0 : aPnt1;
|
const Standard_Real aBoxProjMax = aPnt0 > aPnt1 ? aPnt0 : aPnt1;
|
||||||
|
if (aBoxProjMin > myMaxClipProjectionPts[aPlaneIter]
|
||||||
if (aMin > myMaxClipProjectionPts[aPlaneIter]
|
|| aBoxProjMax < myMinClipProjectionPts[aPlaneIter])
|
||||||
|| aMax < myMinClipProjectionPts[aPlaneIter])
|
|
||||||
{
|
{
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (theIsInside != NULL
|
||||||
|
&& *theIsInside)
|
||||||
|
{
|
||||||
|
*theIsInside = aBoxProjMin >= myMinClipProjectionPts[aPlaneIter]
|
||||||
|
&& aBoxProjMax <= myMaxClipProjectionPts[aPlaneIter];
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
//! Returns TRUE if given AABB should be discarded by distance culling criterion.
|
//! Returns TRUE if given AABB should be discarded by distance culling criterion.
|
||||||
bool isTooDistant (const CullingContext& theCtx,
|
//! @param theMinPnt [in] maximum point of AABB
|
||||||
const Graphic3d_Vec3d& theMinPt,
|
//! @param theMaxPnt [in] minimum point of AABB
|
||||||
const Graphic3d_Vec3d& theMaxPt) const
|
//! @param theIsInside [out] flag indicating if AABB is fully inside; initial value should be set to TRUE
|
||||||
|
//! @return TRUE if AABB is completely behind culling distance;
|
||||||
|
//! FALSE in case of partial or complete overlap (use theIsInside to distinguish)
|
||||||
|
bool IsTooDistant (const CullingContext& theCtx,
|
||||||
|
const Graphic3d_Vec3d& theMinPnt,
|
||||||
|
const Graphic3d_Vec3d& theMaxPnt,
|
||||||
|
Standard_Boolean* theIsInside = NULL) const
|
||||||
{
|
{
|
||||||
if (theCtx.DistCull <= 0.0)
|
if (theCtx.DistCull <= 0.0)
|
||||||
{
|
{
|
||||||
@ -208,22 +233,34 @@ protected:
|
|||||||
}
|
}
|
||||||
|
|
||||||
// check distance to the bounding sphere as fast approximation
|
// check distance to the bounding sphere as fast approximation
|
||||||
const Graphic3d_Vec3d aSphereCenter = (theMinPt + theMaxPt) * 0.5;
|
const Graphic3d_Vec3d aSphereCenter = (theMinPnt + theMaxPnt) * 0.5;
|
||||||
const Standard_Real aSphereRadius = (theMaxPt - theMinPt).maxComp() * 0.5;
|
const Standard_Real aSphereRadius = (theMaxPnt - theMinPnt).maxComp() * 0.5;
|
||||||
return (aSphereCenter - myCamEye).Modulus() - aSphereRadius > theCtx.DistCull;
|
const Standard_Real aDistToCenter = (aSphereCenter - myCamEye).Modulus();
|
||||||
|
if ((aDistToCenter - aSphereRadius) > theCtx.DistCull)
|
||||||
|
{
|
||||||
|
// clip if closest point is behind culling distance
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
if (theIsInside != NULL
|
||||||
|
&& *theIsInside)
|
||||||
|
{
|
||||||
|
// check if farthest point is before culling distance
|
||||||
|
*theIsInside = (aDistToCenter + aSphereRadius) <= theCtx.DistCull;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
//! Returns TRUE if given AABB should be discarded by size culling criterion.
|
//! Returns TRUE if given AABB should be discarded by size culling criterion.
|
||||||
bool isTooSmall (const CullingContext& theCtx,
|
bool IsTooSmall (const CullingContext& theCtx,
|
||||||
const Graphic3d_Vec3d& theMinPt,
|
const Graphic3d_Vec3d& theMinPnt,
|
||||||
const Graphic3d_Vec3d& theMaxPt) const
|
const Graphic3d_Vec3d& theMaxPnt) const
|
||||||
{
|
{
|
||||||
if (theCtx.SizeCull2 <= 0.0)
|
if (theCtx.SizeCull2 <= 0.0)
|
||||||
{
|
{
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
const Standard_Real aBoxDiag2 = (theMaxPt - theMinPt).SquareModulus();
|
const Standard_Real aBoxDiag2 = (theMaxPnt - theMinPnt).SquareModulus();
|
||||||
if (myIsProjectionParallel)
|
if (myIsProjectionParallel)
|
||||||
{
|
{
|
||||||
return aBoxDiag2 < theCtx.SizeCull2;
|
return aBoxDiag2 < theCtx.SizeCull2;
|
||||||
@ -231,7 +268,7 @@ protected:
|
|||||||
|
|
||||||
// note that distances behind the Eye (aBndDist < 0) are not scaled correctly here,
|
// note that distances behind the Eye (aBndDist < 0) are not scaled correctly here,
|
||||||
// but majority of such objects should be culled by frustum
|
// but majority of such objects should be culled by frustum
|
||||||
const Graphic3d_Vec3d aBndCenter = (theMinPt + theMaxPt) * 0.5;
|
const Graphic3d_Vec3d aBndCenter = (theMinPnt + theMaxPnt) * 0.5;
|
||||||
const Standard_Real aBndDist = (aBndCenter - myCamEye).Dot (myCamDir);
|
const Standard_Real aBndDist = (aBndCenter - myCamEye).Dot (myCamDir);
|
||||||
return aBoxDiag2 < theCtx.SizeCull2 * aBndDist * aBndDist;
|
return aBoxDiag2 < theCtx.SizeCull2 * aBndDist * aBndDist;
|
||||||
}
|
}
|
||||||
|
@ -452,6 +452,19 @@ void Graphic3d_Layer::updateBVH() const
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
namespace
|
||||||
|
{
|
||||||
|
//! This structure describes the node in BVH
|
||||||
|
struct NodeInStack
|
||||||
|
{
|
||||||
|
NodeInStack (Standard_Integer theId = 0,
|
||||||
|
Standard_Boolean theIsFullInside = false) : Id (theId), IsFullInside (theIsFullInside) {}
|
||||||
|
|
||||||
|
Standard_Integer Id; //!< node identifier
|
||||||
|
Standard_Boolean IsFullInside; //!< if the node is completely inside
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
// =======================================================================
|
// =======================================================================
|
||||||
// function : UpdateCulling
|
// function : UpdateCulling
|
||||||
// purpose :
|
// purpose :
|
||||||
@ -517,33 +530,54 @@ void Graphic3d_Layer::UpdateCulling (Standard_Integer theViewId,
|
|||||||
aBVHTree = myBVHPrimitives.BVH();
|
aBVHTree = myBVHPrimitives.BVH();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (theSelector.IsCulled (aCullCtx, aBVHTree->MinPoint (0), aBVHTree->MaxPoint (0)))
|
const bool toCheckFullInside = true;
|
||||||
|
NodeInStack aNode (0, toCheckFullInside); // a root node
|
||||||
|
if (theSelector.IsCulled (aCullCtx, aBVHTree->MinPoint (0), aBVHTree->MaxPoint (0), toCheckFullInside ? &aNode.IsFullInside : NULL))
|
||||||
{
|
{
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
Standard_Integer aStack[BVH_Constants_MaxTreeDepth];
|
NodeInStack aStack[BVH_Constants_MaxTreeDepth];
|
||||||
Standard_Integer aHead = -1;
|
Standard_Integer aHead = -1;
|
||||||
Standard_Integer aNode = 0; // a root node
|
|
||||||
for (;;)
|
for (;;)
|
||||||
{
|
{
|
||||||
if (!aBVHTree->IsOuter (aNode))
|
if (!aBVHTree->IsOuter (aNode.Id))
|
||||||
{
|
{
|
||||||
const Standard_Integer aLeftChildIdx = aBVHTree->Child<0> (aNode);
|
NodeInStack aLeft (aBVHTree->Child<0> (aNode.Id), toCheckFullInside);
|
||||||
const Standard_Integer aRightChildIdx = aBVHTree->Child<1> (aNode);
|
NodeInStack aRight(aBVHTree->Child<1> (aNode.Id), toCheckFullInside);
|
||||||
const Standard_Boolean isLeftChildIn = !theSelector.IsCulled (aCullCtx, aBVHTree->MinPoint (aLeftChildIdx), aBVHTree->MaxPoint (aLeftChildIdx));
|
bool isLeftChildIn = true, isRightChildIn = true;
|
||||||
const Standard_Boolean isRightChildIn = !theSelector.IsCulled (aCullCtx, aBVHTree->MinPoint (aRightChildIdx), aBVHTree->MaxPoint (aRightChildIdx));
|
if (aNode.IsFullInside)
|
||||||
|
{
|
||||||
|
// small size should be always checked
|
||||||
|
isLeftChildIn = !theSelector.IsTooSmall (aCullCtx, aBVHTree->MinPoint (aLeft.Id), aBVHTree->MaxPoint (aLeft.Id));
|
||||||
|
isRightChildIn = !theSelector.IsTooSmall (aCullCtx, aBVHTree->MinPoint (aRight.Id), aBVHTree->MaxPoint (aRight.Id));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
isLeftChildIn = !theSelector.IsCulled (aCullCtx, aBVHTree->MinPoint (aLeft.Id), aBVHTree->MaxPoint (aLeft.Id), toCheckFullInside ? &aLeft.IsFullInside : NULL);
|
||||||
|
if (!isLeftChildIn)
|
||||||
|
{
|
||||||
|
aLeft.IsFullInside = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
isRightChildIn = !theSelector.IsCulled (aCullCtx, aBVHTree->MinPoint (aRight.Id), aBVHTree->MaxPoint (aRight.Id), toCheckFullInside ? &aRight.IsFullInside : NULL);
|
||||||
|
if (!isRightChildIn)
|
||||||
|
{
|
||||||
|
aRight.IsFullInside = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (isLeftChildIn
|
if (isLeftChildIn
|
||||||
&& isRightChildIn)
|
&& isRightChildIn)
|
||||||
{
|
{
|
||||||
aNode = myBVHIsLeftChildQueuedFirst ? aLeftChildIdx : aRightChildIdx;
|
aNode = myBVHIsLeftChildQueuedFirst ? aLeft : aRight;
|
||||||
aStack[++aHead] = myBVHIsLeftChildQueuedFirst ? aRightChildIdx : aLeftChildIdx;
|
aStack[++aHead] = myBVHIsLeftChildQueuedFirst ? aRight : aLeft;
|
||||||
myBVHIsLeftChildQueuedFirst = !myBVHIsLeftChildQueuedFirst;
|
myBVHIsLeftChildQueuedFirst = !myBVHIsLeftChildQueuedFirst;
|
||||||
}
|
}
|
||||||
else if (isLeftChildIn
|
else if (isLeftChildIn
|
||||||
|| isRightChildIn)
|
|| isRightChildIn)
|
||||||
{
|
{
|
||||||
aNode = isLeftChildIn ? aLeftChildIdx : aRightChildIdx;
|
aNode = isLeftChildIn ? aLeft : aRight;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@ -557,8 +591,8 @@ void Graphic3d_Layer::UpdateCulling (Standard_Integer theViewId,
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
const Standard_Integer aStartIdx = aBVHTree->BegPrimitive (aNode);
|
const Standard_Integer aStartIdx = aBVHTree->BegPrimitive (aNode.Id);
|
||||||
const Standard_Integer anEndIdx = aBVHTree->EndPrimitive (aNode);
|
const Standard_Integer anEndIdx = aBVHTree->EndPrimitive (aNode.Id);
|
||||||
for (Standard_Integer anIdx = aStartIdx; anIdx <= anEndIdx; ++anIdx)
|
for (Standard_Integer anIdx = aStartIdx; anIdx <= anEndIdx; ++anIdx)
|
||||||
{
|
{
|
||||||
const Graphic3d_CStructure* aStruct = isTrsfPers
|
const Graphic3d_CStructure* aStruct = isTrsfPers
|
||||||
|
Loading…
x
Reference in New Issue
Block a user