mirror of
https://github.com/ptitSeb/Serious-Engine
synced 2024-11-29 21:25:54 +01:00
287 lines
9.5 KiB
C++
287 lines
9.5 KiB
C++
/* Copyright (c) 2002-2012 Croteam Ltd.
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of version 2 of the GNU General Public License as published by
|
|
the Free Software Foundation
|
|
|
|
|
|
This program 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, write to the Free Software Foundation, Inc.,
|
|
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */
|
|
|
|
#ifndef SE_INCL_OBBOX_H
|
|
#define SE_INCL_OBBOX_H
|
|
#ifdef PRAGMA_ONCE
|
|
#pragma once
|
|
#endif
|
|
|
|
#include <Engine/Math/Vector.h>
|
|
#include <Engine/Math/Matrix.h>
|
|
#include <Engine/Math/Functions.h>
|
|
#include <Engine/Math/Plane.h>
|
|
#include <Engine/Math/obbox.h>
|
|
|
|
/*
|
|
* Template for oriented bounding box of arbitrary type in 3D
|
|
*/
|
|
template<class Type>
|
|
class OBBox {
|
|
// implementation:
|
|
public:
|
|
Vector<Type, 3> box_vO; // center of the box
|
|
Vector<Type, 3> box_avAxis[3]; // axis direction vectors
|
|
Type box_atSize[3]; // size on each of the axis (in both directions)
|
|
|
|
/* Clear to normalized empty bounding box. */
|
|
inline void SetToNormalizedEmpty(void);
|
|
// interface:
|
|
public:
|
|
/* Default constructor. */
|
|
inline OBBox(void);
|
|
/* Constructor from components. */
|
|
inline OBBox(const Vector<Type, 3> &vO,
|
|
const Vector<Type, 3> &vAxis0, const Vector<Type, 3> &vAxis1, const Vector<Type, 3> &vAxis2,
|
|
Type tSize0, Type tSize1, Type tSize2);
|
|
/* Constructor from axis aligned box and placement. */
|
|
inline OBBox(const AABBox<Type, 3> &aabbox,
|
|
const Vector<Type, 3> &vPos, const Matrix<Type, 3, 3> &mRot);
|
|
/* Constructor from axis aligned box without placement. */
|
|
inline OBBox(const AABBox<Type, 3> &aabbox);
|
|
|
|
// classify box with respect to a plane
|
|
inline Type TestAgainstPlane(const Plane<Type, 3> &pl) const;
|
|
// check if two boxes intersect/touch
|
|
inline BOOL HasContactWith(const OBBox<Type> &boxB) const;
|
|
|
|
/* Check if empty. */
|
|
inline BOOL IsEmpty(void) const;
|
|
};
|
|
|
|
/*
|
|
* Clear to normalized empty bounding box.
|
|
*/
|
|
template<class Type>
|
|
inline void OBBox<Type>::SetToNormalizedEmpty(void) {
|
|
for ( int i=0; i<3; i++ ) {
|
|
box_atSize[i] = LowerLimit(Type(0));
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Constructor for empty bounding box.
|
|
*/
|
|
template<class Type>
|
|
inline OBBox<Type>::OBBox<Type>() {
|
|
SetToNormalizedEmpty();
|
|
}
|
|
|
|
/* Constructor from axis aligned box and placement. */
|
|
template<class Type>
|
|
inline OBBox<Type>::OBBox(const AABBox<Type, 3> &aabbox,
|
|
const Vector<Type, 3> &vPos, const Matrix<Type, 3, 3> &mRot)
|
|
{
|
|
// translate and rotate the center
|
|
box_vO = aabbox.Center()*mRot+vPos;
|
|
// extracted orientation from the rotation matrix
|
|
box_avAxis[0](1) = mRot(1,1); box_avAxis[0](2) = mRot(2,1); box_avAxis[0](3) = mRot(3,1);
|
|
box_avAxis[1](1) = mRot(1,2); box_avAxis[1](2) = mRot(2,2); box_avAxis[1](3) = mRot(3,2);
|
|
box_avAxis[2](1) = mRot(1,3); box_avAxis[2](2) = mRot(2,3); box_avAxis[2](3) = mRot(3,3);
|
|
|
|
// get sizes from obbox sizes
|
|
box_atSize[0] = aabbox.Size()(1)*0.5f;
|
|
box_atSize[1] = aabbox.Size()(2)*0.5f;
|
|
box_atSize[2] = aabbox.Size()(3)*0.5f;
|
|
}
|
|
/* Constructor from axis aligned box without placement. */
|
|
template<class Type>
|
|
inline OBBox<Type>::OBBox(const AABBox<Type, 3> &aabbox)
|
|
{
|
|
box_vO = aabbox.Center();
|
|
box_avAxis[0] = Vector<Type, 3>(1,0,0);
|
|
box_avAxis[1] = Vector<Type, 3>(0,1,0);
|
|
box_avAxis[2] = Vector<Type, 3>(0,0,1);
|
|
box_atSize[0] = aabbox.Size()(1)*0.5f;
|
|
box_atSize[1] = aabbox.Size()(2)*0.5f;
|
|
box_atSize[2] = aabbox.Size()(3)*0.5f;
|
|
}
|
|
|
|
/* Constructor from components. */
|
|
template<class Type>
|
|
inline OBBox<Type>::OBBox(const Vector<Type, 3> &vO,
|
|
const Vector<Type, 3> &vAxis0, const Vector<Type, 3> &vAxis1, const Vector<Type, 3> &vAxis2,
|
|
Type tSize0, Type tSize1, Type tSize2) {
|
|
box_vO = vO;
|
|
box_avAxis[0] = vAxis0; box_avAxis[1] = vAxis1; box_avAxis[2] = vAxis2;
|
|
box_atSize[0] = tSize0; box_atSize[1] = tSize1; box_atSize[2] = tSize2;
|
|
};
|
|
|
|
/*
|
|
* Check if empty.
|
|
*/
|
|
template<class Type>
|
|
inline BOOL OBBox<Type>::IsEmpty(void) const {
|
|
// if any dimension is empty, it is empty
|
|
for ( int i=0; i<3; i++ ) {
|
|
if (box_atSize[i] < Type(0)) {
|
|
return TRUE;
|
|
}
|
|
}
|
|
// otherwise, it is not empty
|
|
return FALSE;
|
|
}
|
|
|
|
// classify a box with respect to a plane
|
|
template<class Type>
|
|
inline Type OBBox<Type>::TestAgainstPlane(const Plane<Type, 3> &pl) const
|
|
{
|
|
// project each axis to the plane normal
|
|
Type tNX = ((const Vector<Type,3> &)pl)%box_avAxis[0];
|
|
Type tNY = ((const Vector<Type,3> &)pl)%box_avAxis[1];
|
|
Type tNZ = ((const Vector<Type,3> &)pl)%box_avAxis[2];
|
|
// calculate overall size of the box along the plane normal
|
|
Type tSize = Abs(tNX*box_atSize[0]) + Abs(tNY*box_atSize[1]) + Abs(tNZ*box_atSize[2]);
|
|
// get distance of the center from the plane
|
|
Type tCenterD = pl.PointDistance(box_vO);
|
|
|
|
// if the center is further front than box's size
|
|
if (tCenterD>tSize) {
|
|
// completely in front `
|
|
return Type(1);
|
|
// if the center is further back than box's size
|
|
} else if (tCenterD<-tSize) {
|
|
// completely back
|
|
return Type(-1);
|
|
// otherwise, it touches the plane
|
|
} else {
|
|
return Type(0);
|
|
}
|
|
}
|
|
|
|
// check if two boxes intersect/touch
|
|
// using the separating axes theorem
|
|
template<class Type>
|
|
inline BOOL OBBox<Type>::HasContactWith(const OBBox<Type> &boxB) const
|
|
{
|
|
const OBBox<Type> &boxA = *this;
|
|
|
|
// find offset in abs space
|
|
Vector<Type, 3> vOffAbs = boxB.box_vO - boxA.box_vO;
|
|
// rotate offset to A space
|
|
Type vOffA[3] = {
|
|
vOffAbs%boxA.box_avAxis[0],
|
|
vOffAbs%boxA.box_avAxis[1],
|
|
vOffAbs%boxA.box_avAxis[2]};
|
|
|
|
// calculate rotation matrix from B to A
|
|
Type mR[3][3];
|
|
{for(INDEX i=0; i<3; i++) {
|
|
{for(INDEX j=0; j<3; j++) {
|
|
mR[i][j] = boxA.box_avAxis[i]%boxB.box_avAxis[j];
|
|
}}
|
|
}}
|
|
|
|
Type tRa, tRb, tT;
|
|
|
|
// check each axis of A
|
|
{for(INDEX i=0; i<3; i++ ) {
|
|
tRa = boxA.box_atSize[i];
|
|
tRb = boxB.box_atSize[0]*Abs(mR[i][0]) + boxB.box_atSize[1]*Abs(mR[i][1]) + boxB.box_atSize[2]*Abs(mR[i][2]);
|
|
tT = Abs( vOffA[i] );
|
|
if (tT>tRa+tRb) return FALSE;
|
|
}}
|
|
|
|
// check each axis of B
|
|
{for(INDEX i=0; i<3; i++ ) {
|
|
tRa = boxA.box_atSize[0]*Abs(mR[0][i]) + boxA.box_atSize[1]*Abs(mR[1][i]) + boxA.box_atSize[2]*Abs(mR[2][i]);
|
|
tRb = boxB.box_atSize[i];
|
|
tT = Abs( vOffA[0]*mR[0][i] + vOffA[1]*mR[1][i] + vOffA[2]*mR[2][i] );
|
|
if (tT>tRa+tRb) return FALSE;
|
|
}}
|
|
|
|
// check A0 x B0
|
|
tRa = boxA.box_atSize[1]*Abs(mR[2][0]) + boxA.box_atSize[2]*Abs(mR[1][0]);
|
|
tRb = boxB.box_atSize[1]*Abs(mR[0][2]) + boxB.box_atSize[2]*Abs(mR[0][1]);
|
|
tT = Abs( vOffA[2]*mR[1][0] - vOffA[1]*mR[2][0] );
|
|
if(tT>tRa+tRb) return FALSE;
|
|
|
|
// check A0 x B1
|
|
tRa = boxA.box_atSize[1]*Abs(mR[2][1]) + boxA.box_atSize[2]*Abs(mR[1][1]);
|
|
tRb = boxB.box_atSize[0]*Abs(mR[0][2]) + boxB.box_atSize[2]*Abs(mR[0][0]);
|
|
tT = Abs( vOffA[2]*mR[1][1] - vOffA[1]*mR[2][1] );
|
|
if(tT>tRa+tRb) return FALSE;
|
|
|
|
// check A0 x B2
|
|
tRa = boxA.box_atSize[1]*Abs(mR[2][2]) + boxA.box_atSize[2]*Abs(mR[1][2]);
|
|
tRb = boxB.box_atSize[0]*Abs(mR[0][1]) + boxB.box_atSize[1]*Abs(mR[0][0]);
|
|
tT = Abs( vOffA[2]*mR[1][2] - vOffA[1]*mR[2][2] );
|
|
if(tT>tRa+tRb) return FALSE;
|
|
|
|
// check A1 x B0
|
|
tRa = boxA.box_atSize[0]*Abs(mR[2][0]) + boxA.box_atSize[2]*Abs(mR[0][0]);
|
|
tRb = boxB.box_atSize[1]*Abs(mR[1][2]) + boxB.box_atSize[2]*Abs(mR[1][1]);
|
|
tT = Abs( vOffA[0]*mR[2][0] - vOffA[2]*mR[0][0] );
|
|
if(tT>tRa+tRb) return FALSE;
|
|
|
|
// check A1 x B1
|
|
tRa = boxA.box_atSize[0]*Abs(mR[2][1]) + boxA.box_atSize[2]*Abs(mR[0][1]);
|
|
tRb = boxB.box_atSize[0]*Abs(mR[1][2]) + boxB.box_atSize[2]*Abs(mR[1][0]);
|
|
tT = Abs( vOffA[0]*mR[2][1] - vOffA[2]*mR[0][1] );
|
|
if(tT>tRa+tRb) return FALSE;
|
|
|
|
// check A1 x B2
|
|
tRa = boxA.box_atSize[0]*Abs(mR[2][2]) + boxA.box_atSize[2]*Abs(mR[0][2]);
|
|
tRb = boxB.box_atSize[0]*Abs(mR[1][1]) + boxB.box_atSize[1]*Abs(mR[1][0]);
|
|
tT = Abs( vOffA[0]*mR[2][2] - vOffA[2]*mR[0][2] );
|
|
if(tT>tRa+tRb) return FALSE;
|
|
|
|
// check A2 x B0
|
|
tRa = boxA.box_atSize[0]*Abs(mR[1][0]) + boxA.box_atSize[1]*Abs(mR[0][0]);
|
|
tRb = boxB.box_atSize[1]*Abs(mR[2][2]) + boxB.box_atSize[2]*Abs(mR[2][1]);
|
|
tT = Abs( vOffA[1]*mR[0][0] - vOffA[0]*mR[1][0] );
|
|
if(tT>tRa+tRb) return FALSE;
|
|
|
|
// check A2 x B1
|
|
tRa = boxA.box_atSize[0]*Abs(mR[1][1]) + boxA.box_atSize[1]*Abs(mR[0][1]);
|
|
tRb = boxB.box_atSize[0] *Abs(mR[2][2]) + boxB.box_atSize[2]*Abs(mR[2][0]);
|
|
tT = Abs( vOffA[1]*mR[0][1] - vOffA[0]*mR[1][1] );
|
|
if(tT>tRa+tRb) return FALSE;
|
|
|
|
// check A2 x B2
|
|
tRa = boxA.box_atSize[0]*Abs(mR[1][2]) + boxA.box_atSize[1]*Abs(mR[0][2]);
|
|
tRb = boxB.box_atSize[0]*Abs(mR[2][1]) + boxB.box_atSize[1]*Abs(mR[2][0]);
|
|
tT = Abs( vOffA[1]*mR[0][2] - vOffA[0]*mR[1][2] );
|
|
if(tT>tRa+tRb) return FALSE;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
// helper functions for converting between FLOAT and DOUBLE obboxes
|
|
inline DOUBLEobbox3D FLOATtoDOUBLE(const FLOATobbox3D &boxf) {
|
|
return DOUBLEobbox3D(
|
|
FLOATtoDOUBLE(boxf.box_vO),
|
|
FLOATtoDOUBLE(boxf.box_avAxis[0]),
|
|
FLOATtoDOUBLE(boxf.box_avAxis[1]),
|
|
FLOATtoDOUBLE(boxf.box_avAxis[2]),
|
|
FLOATtoDOUBLE(boxf.box_atSize[0]),
|
|
FLOATtoDOUBLE(boxf.box_atSize[1]),
|
|
FLOATtoDOUBLE(boxf.box_atSize[2]));
|
|
}
|
|
inline FLOATobbox3D DOUBLEtoFLOAT(const DOUBLEobbox3D &boxd) {
|
|
return FLOATobbox3D(
|
|
DOUBLEtoFLOAT(boxd.box_vO),
|
|
DOUBLEtoFLOAT(boxd.box_avAxis[0]),
|
|
DOUBLEtoFLOAT(boxd.box_avAxis[1]),
|
|
DOUBLEtoFLOAT(boxd.box_avAxis[2]),
|
|
DOUBLEtoFLOAT(boxd.box_atSize[0]),
|
|
DOUBLEtoFLOAT(boxd.box_atSize[1]),
|
|
DOUBLEtoFLOAT(boxd.box_atSize[2]));
|
|
}
|
|
|
|
|
|
#endif /* include-once check. */
|
|
|