2022-04-07 18:46:57 +02:00

400 lines
12 KiB
C++

/*
This file is part of Nori, a simple educational ray tracer
Copyright (c) 2015 by Wenzel Jakob
Nori is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License Version 3
as published by the Free Software Foundation.
Nori 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
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#pragma once
#include <nori/ray.h>
NORI_NAMESPACE_BEGIN
/**
* \brief Generic n-dimensional bounding box data structure
*
* Maintains a minimum and maximum position along each dimension and provides
* various convenience functions for querying and modifying them.
*
* This class is parameterized by the underlying point data structure,
* which permits the use of different scalar types and dimensionalities, e.g.
* \code
* TBoundingBox<Vector3i> integerBBox(Point3i(0, 1, 3), Point3i(4, 5, 6));
* TBoundingBox<Vector2d> doubleBBox(Point2d(0.0, 1.0), Point2d(4.0, 5.0));
* \endcode
*
* \tparam T The underlying point data type (e.g. \c Point2d)
* \ingroup libcore
*/
template <typename _PointType> struct TBoundingBox {
enum {
Dimension = _PointType::Dimension
};
typedef _PointType PointType;
typedef typename PointType::Scalar Scalar;
typedef typename PointType::VectorType VectorType;
/**
* \brief Create a new invalid bounding box
*
* Initializes the components of the minimum
* and maximum position to \f$\infty\f$ and \f$-\infty\f$,
* respectively.
*/
TBoundingBox() {
reset();
}
/// Create a collapsed bounding box from a single point
TBoundingBox(const PointType &p)
: min(p), max(p) { }
/// Create a bounding box from two positions
TBoundingBox(const PointType &min, const PointType &max)
: min(min), max(max) {
}
/// Test for equality against another bounding box
bool operator==(const TBoundingBox &bbox) const {
return min == bbox.min && max == bbox.max;
}
/// Test for inequality against another bounding box
bool operator!=(const TBoundingBox &bbox) const {
return min != bbox.min || max != bbox.max;
}
/// Calculate the n-dimensional volume of the bounding box
Scalar getVolume() const {
return (max - min).prod();
}
/// Calculate the n-1 dimensional volume of the boundary
float getSurfaceArea() const {
VectorType d = max - min;
float result = 0.0f;
for (int i=0; i<Dimension; ++i) {
float term = 1.0f;
for (int j=0; j<Dimension; ++j) {
if (i == j)
continue;
term *= d[j];
}
result += term;
}
return 2.0f * result;
}
/// Return the center point
PointType getCenter() const {
return (max + min) * (Scalar) 0.5f;
}
/**
* \brief Check whether a point lies \a on or \a inside the bounding box
*
* \param p The point to be tested
*
* \param strict Set this parameter to \c true if the bounding
* box boundary should be excluded in the test
*/
bool contains(const PointType &p, bool strict = false) const {
if (strict) {
return (p.array() > min.array()).all()
&& (p.array() < max.array()).all();
} else {
return (p.array() >= min.array()).all()
&& (p.array() <= max.array()).all();
}
}
/**
* \brief Check whether a specified bounding box lies \a on or \a within
* the current bounding box
*
* Note that by definition, an 'invalid' bounding box (where min=\f$\infty\f$
* and max=\f$-\infty\f$) does not cover any space. Hence, this method will always
* return \a true when given such an argument.
*
* \param strict Set this parameter to \c true if the bounding
* box boundary should be excluded in the test
*/
bool contains(const TBoundingBox &bbox, bool strict = false) const {
if (strict) {
return (bbox.min.array() > min.array()).all()
&& (bbox.max.array() < max.array()).all();
} else {
return (bbox.min.array() >= min.array()).all()
&& (bbox.max.array() <= max.array()).all();
}
}
/**
* \brief Check two axis-aligned bounding boxes for possible overlap.
*
* \param strict Set this parameter to \c true if the bounding
* box boundary should be excluded in the test
*
* \return \c true If overlap was detected.
*/
bool overlaps(const TBoundingBox &bbox, bool strict = false) const {
if (strict) {
return (bbox.min.array() < max.array()).all()
&& (bbox.max.array() > min.array()).all();
} else {
return (bbox.min.array() <= max.array()).all()
&& (bbox.max.array() >= min.array()).all();
}
}
/**
* \brief Calculate the smallest squared distance between
* the axis-aligned bounding box and the point \c p.
*/
Scalar squaredDistanceTo(const PointType &p) const {
Scalar result = 0;
for (int i=0; i<Dimension; ++i) {
Scalar value = 0;
if (p[i] < min[i])
value = min[i] - p[i];
else if (p[i] > max[i])
value = p[i] - max[i];
result += value*value;
}
return result;
}
/**
* \brief Calculate the smallest distance between
* the axis-aligned bounding box and the point \c p.
*/
Scalar distanceTo(const PointType &p) const {
return std::sqrt(squaredDistanceTo(p));
}
/**
* \brief Calculate the smallest square distance between
* the axis-aligned bounding box and \c bbox.
*/
Scalar squaredDistanceTo(const TBoundingBox &bbox) const {
Scalar result = 0;
for (int i=0; i<Dimension; ++i) {
Scalar value = 0;
if (bbox.max[i] < min[i])
value = min[i] - bbox.max[i];
else if (bbox.min[i] > max[i])
value = bbox.min[i] - max[i];
result += value*value;
}
return result;
}
/**
* \brief Calculate the smallest distance between
* the axis-aligned bounding box and \c bbox.
*/
Scalar distanceTo(const TBoundingBox &bbox) const {
return std::sqrt(squaredDistanceTo(bbox));
}
/**
* \brief Check whether this is a valid bounding box
*
* A bounding box \c bbox is valid when
* \code
* bbox.min[dim] <= bbox.max[dim]
* \endcode
* holds along each dimension \c dim.
*/
bool isValid() const {
return (max.array() >= min.array()).all();
}
/// Check whether this bounding box has collapsed to a single point
bool isPoint() const {
return (max.array() == min.array()).all();
}
/// Check whether this bounding box has any associated volume
bool hasVolume() const {
return (max.array() > min.array()).all();
}
/// Return the dimension index with the largest associated side length
int getMajorAxis() const {
VectorType d = max - min;
int largest = 0;
for (int i=1; i<Dimension; ++i)
if (d[i] > d[largest])
largest = i;
return largest;
}
/// Return the dimension index with the shortest associated side length
int getMinorAxis() const {
VectorType d = max - min;
int shortest = 0;
for (int i=1; i<Dimension; ++i)
if (d[i] < d[shortest])
shortest = i;
return shortest;
}
/**
* \brief Calculate the bounding box extents
* \return max-min
*/
VectorType getExtents() const {
return max - min;
}
/// Clip to another bounding box
void clip(const TBoundingBox &bbox) {
min = min.cwiseMax(bbox.min);
max = max.cwiseMin(bbox.max);
}
/**
* \brief Mark the bounding box as invalid.
*
* This operation sets the components of the minimum
* and maximum position to \f$\infty\f$ and \f$-\infty\f$,
* respectively.
*/
void reset() {
min.setConstant( std::numeric_limits<Scalar>::infinity());
max.setConstant(-std::numeric_limits<Scalar>::infinity());
}
/// Expand the bounding box to contain another point
void expandBy(const PointType &p) {
min = min.cwiseMin(p);
max = max.cwiseMax(p);
}
/// Expand the bounding box to contain another bounding box
void expandBy(const TBoundingBox &bbox) {
min = min.cwiseMin(bbox.min);
max = max.cwiseMax(bbox.max);
}
/// Merge two bounding boxes
static TBoundingBox merge(const TBoundingBox &bbox1, const TBoundingBox &bbox2) {
return TBoundingBox(
bbox1.min.cwiseMin(bbox2.min),
bbox1.max.cwiseMax(bbox2.max)
);
}
/// Return the index of the largest axis
int getLargestAxis() const {
VectorType extents = max-min;
if (extents[0] >= extents[1] && extents[0] >= extents[2])
return 0;
else if (extents[1] >= extents[0] && extents[1] >= extents[2])
return 1;
else
return 2;
}
/// Return the position of a bounding box corner
PointType getCorner(int index) const {
PointType result;
for (int i=0; i<Dimension; ++i)
result[i] = (index & (1 << i)) ? max[i] : min[i];
return result;
}
/// Return a string representation of the bounding box
std::string toString() const {
if (!isValid())
return "BoundingBox[invalid]";
else
return tfm::format("BoundingBox[min=%s, max=%s]", min.toString(), max.toString());
}
/// Check if a ray intersects a bounding box
bool rayIntersect(const Ray3f &ray) const {
float nearT = -std::numeric_limits<float>::infinity();
float farT = std::numeric_limits<float>::infinity();
for (int i=0; i<3; i++) {
float origin = ray.o[i];
float minVal = min[i], maxVal = max[i];
if (ray.d[i] == 0) {
if (origin < minVal || origin > maxVal)
return false;
} else {
float t1 = (minVal - origin) * ray.dRcp[i];
float t2 = (maxVal - origin) * ray.dRcp[i];
if (t1 > t2)
std::swap(t1, t2);
nearT = std::max(t1, nearT);
farT = std::min(t2, farT);
if (!(nearT <= farT))
return false;
}
}
return ray.mint <= farT && nearT <= ray.maxt;
}
/// Return the overlapping region of the bounding box and an unbounded ray
bool rayIntersect(const Ray3f &ray, float &nearT, float &farT) const {
nearT = -std::numeric_limits<float>::infinity();
farT = std::numeric_limits<float>::infinity();
for (int i=0; i<3; i++) {
float origin = ray.o[i];
float minVal = min[i], maxVal = max[i];
if (ray.d[i] == 0) {
if (origin < minVal || origin > maxVal)
return false;
} else {
float t1 = (minVal - origin) * ray.dRcp[i];
float t2 = (maxVal - origin) * ray.dRcp[i];
if (t1 > t2)
std::swap(t1, t2);
nearT = std::max(t1, nearT);
farT = std::min(t2, farT);
if (!(nearT <= farT))
return false;
}
}
return true;
}
PointType min; ///< Component-wise minimum
PointType max; ///< Component-wise maximum
};
NORI_NAMESPACE_END