1
0
mirror of https://git.dev.opencascade.org/repos/occt.git synced 2025-04-05 18:16:23 +03:00

0031186: Foundation Classes - add additional useful methods to BVH_Box.

Add Transform() and Transformed() methods by Mat4 for BVH_Box (only for 3D dimension).
Add PointBoxProjection() method to project point to box and RayBoxIntersection() method to compute hit time of ray-box intersection to BVH_Tools.
Add BVH_Ray class to define ray object.
This commit is contained in:
osa 2019-11-22 16:02:02 +03:00 committed by bugmaster
parent e513a6c538
commit 2b5a58a33c
4 changed files with 232 additions and 1 deletions

View File

@ -23,11 +23,84 @@
#include <limits>
//! Base class for BVH_Box (CRTP idiom is used).
//! @tparam T Numeric data type
//! @tparam N Vector dimension
//! @tparam TheDerivedBox Template of derived class that defined axis aligned bounding box.
template <class T, int N, template <class /*T*/, int /*N*/> class TheDerivedBox>
class BVH_BaseBox {};
// forward declaration
template <class T, int N> class BVH_Box;
//! Partial template specialization for BVH_Box when N = 3.
template <class T>
class BVH_BaseBox<T, 3, BVH_Box>
{
public:
//! Transforms this box with given transformation.
void Transform (const NCollection_Mat4<T>& theTransform)
{
if (theTransform.IsIdentity())
{
return;
}
BVH_Box<T, 3> *aThis = static_cast<BVH_Box<T, 3>*>(this);
if (!aThis->IsValid())
{
return;
}
BVH_Box<T, 3> aBox = Transformed (theTransform);
aThis->CornerMin() = aBox.CornerMin();
aThis->CornerMax() = aBox.CornerMax();
}
//! Returns a box which is the result of applying the
//! given transformation to this box.
BVH_Box<T, 3> Transformed (const NCollection_Mat4<T>& theTransform) const
{
BVH_Box<T, 3> aResultBox;
if (theTransform.IsIdentity())
{
return aResultBox;
}
const BVH_Box<T, 3> *aThis = static_cast<const BVH_Box<T, 3>*>(this);
if (!aThis->IsValid())
{
return aResultBox;
}
for (size_t aX = 0; aX <= 1; ++aX)
{
for (size_t aY = 0; aY <= 1; ++aY)
{
for (size_t aZ = 0; aZ <= 1; ++aZ)
{
typename BVH::VectorType<T, 4>::Type aPnt =
theTransform * typename BVH::VectorType<T, 4>::Type (aX ? aThis->CornerMax().x() : aThis->CornerMin().x(),
aY ? aThis->CornerMax().y() : aThis->CornerMin().y(),
aZ ? aThis->CornerMax().z() : aThis->CornerMin().z(),
static_cast<T> (1.0));
aResultBox.Add (aPnt.xyz());
}
}
}
return aResultBox;
}
};
//! Defines axis aligned bounding box (AABB) based on BVH vectors.
//! \tparam T Numeric data type
//! \tparam N Vector dimension
template<class T, int N>
class BVH_Box
class BVH_Box : public BVH_BaseBox<T, N, BVH_Box>
{
public:

39
src/BVH/BVH_Ray.hxx Normal file
View File

@ -0,0 +1,39 @@
// Created by: Olga Suryaninova
// Created on: 2019-11-25
// Copyright (c) 2019 OPEN CASCADE SAS
//
// This file is part of Open CASCADE Technology software library.
//
// This library is free software; you can redistribute it and/or modify it under
// the terms of the GNU Lesser General Public License version 2.1 as published
// by the Free Software Foundation, with special exception defined in the file
// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT
// distribution for complete text of the license and disclaimer of any warranty.
//
// Alternatively, this file may be used under the terms of Open CASCADE
// commercial license or contractual agreement.
#ifndef _BVH_Ray_Header
#define _BVH_Ray_Header
//! Describes a ray based on BVH vectors.
template<class T, int N>
class BVH_Ray
{
public:
typedef typename BVH::VectorType<T, N>::Type BVH_VecNt;
public:
BVH_VecNt Origin;
BVH_VecNt Direct;
public:
BVH_Ray (const BVH_VecNt& theOrigin,
const BVH_VecNt& theDirect) : Origin (theOrigin),
Direct (theDirect) { }
};
#endif // _BVH_Ray_Header

View File

@ -17,6 +17,7 @@
#define _BVH_Tools_Header
#include <BVH_Box.hxx>
#include <BVH_Ray.hxx>
#include <BVH_Types.hxx>
//! Defines a set of static methods operating with points and bounding boxes.
@ -35,6 +36,10 @@ public: //! @name Box-Box Square distance
static T BoxBoxSquareDistance (const BVH_Box<T, N>& theBox1,
const BVH_Box<T, N>& theBox2)
{
if (!theBox1.IsValid() || !theBox2.IsValid())
{
return static_cast<T>(0);
}
return BoxBoxSquareDistance (theBox1.CornerMin(), theBox1.CornerMax(),
theBox2.CornerMin(), theBox2.CornerMax());
}
@ -60,6 +65,10 @@ public: //! @name Point-Box Square distance
static T PointBoxSquareDistance (const BVH_VecNt& thePoint,
const BVH_Box<T, N>& theBox)
{
if (!theBox.IsValid())
{
return static_cast<T>(0);
}
return PointBoxSquareDistance (thePoint,
theBox.CornerMin(),
theBox.CornerMax());
@ -79,6 +88,29 @@ public: //! @name Point-Box Square distance
return aDist;
}
public: //! @name Point-Box projection
//! Computes projection of point on bounding box
static BVH_VecNt PointBoxProjection (const BVH_VecNt& thePoint,
const BVH_Box<T, N>& theBox)
{
if (!theBox.IsValid())
{
return thePoint;
}
return PointBoxProjection (thePoint,
theBox.CornerMin(),
theBox.CornerMax());
}
//! Computes projection of point on bounding box
static BVH_VecNt PointBoxProjection (const BVH_VecNt& thePoint,
const BVH_VecNt& theCMin,
const BVH_VecNt& theCMax)
{
return thePoint.cwiseMax (theCMin).cwiseMin (theCMax);
}
public: //! @name Point-Triangle Square distance
//! Computes square distance between point and triangle
@ -160,6 +192,92 @@ public: //! @name Point-Triangle Square distance
return (aDirect.Dot(aDirect));
}
public: //! @name Ray-Box Intersection
//! Computes hit time of ray-box intersection
static Standard_Boolean RayBoxIntersection (const BVH_Ray<T, N>& theRay,
const BVH_Box<T, N>& theBox,
T& theTimeEnter,
T& theTimeLeave)
{
if (!theBox.IsValid())
{
return Standard_False;
}
return RayBoxIntersection (theRay, theBox.CornerMin(), theBox.CornerMax(), theTimeEnter, theTimeLeave);
}
//! Computes hit time of ray-box intersection
static Standard_Boolean RayBoxIntersection (const BVH_Ray<T, N>& theRay,
const BVH_VecNt& theBoxCMin,
const BVH_VecNt& theBoxCMax,
T& theTimeEnter,
T& theTimeLeave)
{
return RayBoxIntersection (theRay.Origin, theRay.Direct,
theBoxCMin, theBoxCMax, theTimeEnter, theTimeLeave);
}
//! Computes hit time of ray-box intersection
static Standard_Boolean RayBoxIntersection (const BVH_VecNt& theRayOrigin,
const BVH_VecNt& theRayDirection,
const BVH_Box<T, N>& theBox,
T& theTimeEnter,
T& theTimeLeave)
{
if (!theBox.IsValid())
{
return Standard_False;
}
return RayBoxIntersection (theRayOrigin, theRayDirection,
theBox.CornerMin(), theBox.CornerMax(),
theTimeEnter, theTimeLeave);
}
//! Computes hit time of ray-box intersection
static Standard_Boolean RayBoxIntersection (const BVH_VecNt& theRayOrigin,
const BVH_VecNt& theRayDirection,
const BVH_VecNt& theBoxCMin,
const BVH_VecNt& theBoxCMax,
T& theTimeEnter,
T& theTimeLeave)
{
BVH_VecNt aNodeMin, aNodeMax;
for (int i = 0; i < N; ++i)
{
if (theRayDirection[i] == 0)
{
aNodeMin[i] = (theBoxCMin[i] - theRayOrigin[i]) < 0 ?
(std::numeric_limits<T>::min)() : (std::numeric_limits<T>::max)();
aNodeMax[i] = (theBoxCMax[i] - theRayOrigin[i]) < 0 ?
(std::numeric_limits<T>::min)() : (std::numeric_limits<T>::max)();
}
else
{
aNodeMin[i] = (theBoxCMin[i] - theRayOrigin[i]) / theRayDirection[i];
aNodeMax[i] = (theBoxCMax[i] - theRayOrigin[i]) / theRayDirection[i];
}
}
BVH_VecNt aTimeMin, aTimeMax;
for (int i = 0; i < N; ++i)
{
aTimeMin[i] = Min (aNodeMin[i], aNodeMax[i]);
aTimeMax[i] = Max (aNodeMin[i], aNodeMax[i]);
}
T aTimeEnter = Max (aTimeMin[0], Max (aTimeMin[1], aTimeMin[2]));
T aTimeLeave = Min (aTimeMax[0], Min (aTimeMax[1], aTimeMax[2]));
Standard_Boolean hasIntersection = aTimeEnter < aTimeLeave && aTimeLeave > 0;
if (hasIntersection)
{
theTimeEnter = aTimeEnter;
theTimeLeave = aTimeLeave;
}
return hasIntersection;
}
};
#endif

View File

@ -22,6 +22,7 @@ BVH_PrimitiveSet3d.hxx
BVH_Properties.cxx
BVH_Properties.hxx
BVH_QueueBuilder.hxx
BVH_Ray.hxx
BVH_Set.hxx
BVH_Sorter.hxx
BVH_QuickSorter.hxx