1
0
mirror of https://bitbucket.org/librepilot/librepilot.git synced 2025-01-09 20:46:07 +01:00
LibrePilot/ground/openpilotgcs/src/libs/glc_lib/glc_boundingbox.cpp

213 lines
5.5 KiB
C++
Raw Normal View History

/****************************************************************************
This file is part of the GLC-lib library.
Copyright (C) 2005-2008 Laurent Ribon (laumaya@users.sourceforge.net)
Version 2.0.0, packaged on July 2010.
http://glc-lib.sourceforge.net
GLC-lib is free software; you can redistribute it and/or modify
it under the terms of the GNU Lesser General Public License as published by
the Free Software Foundation; either version 3 of the License, or
(at your option) any later version.
GLC-lib 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 Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public License
along with GLC-lib; if not, write to the Free Software
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
*****************************************************************************/
//! \file glc_boundingbox.cpp implementation of the GLC_BoundingBox class.
#include "glc_boundingbox.h"
#include "maths/glc_matrix4x4.h"
quint32 GLC_BoundingBox::m_ChunkId= 0xA707;
//////////////////////////////////////////////////////////////////////
// Constructor Destructor
//////////////////////////////////////////////////////////////////////
GLC_BoundingBox::GLC_BoundingBox()
: m_Lower(0, 0, 0)
, m_Upper(0, 0, 0)
, m_IsEmpty(true)
{
}
GLC_BoundingBox::GLC_BoundingBox(const GLC_BoundingBox& boundingBox)
: m_Lower(boundingBox.m_Lower)
, m_Upper(boundingBox.m_Upper)
, m_IsEmpty(boundingBox.m_IsEmpty)
{
}
GLC_BoundingBox::GLC_BoundingBox(const GLC_Point3d& lower, const GLC_Point3d& upper)
: m_Lower(lower)
, m_Upper(upper)
, m_IsEmpty(false)
{
}
//////////////////////////////////////////////////////////////////////
// Get Functions
//////////////////////////////////////////////////////////////////////
quint32 GLC_BoundingBox::chunckID()
{
return m_ChunkId;
}
bool GLC_BoundingBox::intersect(const GLC_Point3d& point) const
{
if (!m_IsEmpty)
{
bool result= (point.x() < m_Upper.x()) && (point.y() < m_Upper.y())
&& (point.z() < m_Upper.z()) && (point.x() > m_Lower.x())
&& (point.y() > m_Lower.y()) && (point.z() > m_Lower.z());
return result;
}
else
{
return false;
}
}
bool GLC_BoundingBox::intersectBoundingSphere(const GLC_Point3d& point) const
{
const double distance= (center() - point).length();
return distance < boundingSphereRadius();
}
bool GLC_BoundingBox::intersectBoundingSphere(const GLC_BoundingBox& boundingSphere) const
{
const double distance= (center() - boundingSphere.center()).length();
return distance < (boundingSphereRadius() + boundingSphere.boundingSphereRadius());
}
//////////////////////////////////////////////////////////////////////
// Set Functions
//////////////////////////////////////////////////////////////////////
GLC_BoundingBox& GLC_BoundingBox::combine(const GLC_Point3d& point)
{
if (m_IsEmpty)
{
m_Lower= point;
m_Upper= point;
m_IsEmpty= false;
}
else
{
double lowerX= qMin(point.x(), m_Lower.x());
double lowerY= qMin(point.y(), m_Lower.y());
double lowerZ= qMin(point.z(), m_Lower.z());
m_Lower.setVect(lowerX, lowerY, lowerZ);
double upperX= qMax(point.x(), m_Upper.x());
double upperY= qMax(point.y(), m_Upper.y());
double upperZ= qMax(point.z(), m_Upper.z());
m_Upper.setVect(upperX, upperY, upperZ);
}
return *this;
}
GLC_BoundingBox& GLC_BoundingBox::combine(const GLC_Point3df& pointf)
{
GLC_Point3d point(pointf);
if (m_IsEmpty)
{
m_Lower= point;
m_Upper= point;
m_IsEmpty= false;
}
else
{
double lowerX= qMin(point.x(), m_Lower.x());
double lowerY= qMin(point.y(), m_Lower.y());
double lowerZ= qMin(point.z(), m_Lower.z());
m_Lower.setVect(lowerX, lowerY, lowerZ);
double upperX= qMax(point.x(), m_Upper.x());
double upperY= qMax(point.y(), m_Upper.y());
double upperZ= qMax(point.z(), m_Upper.z());
m_Upper.setVect(upperX, upperY, upperZ);
}
return *this;
}
GLC_BoundingBox& GLC_BoundingBox::combine(const GLC_BoundingBox& box)
{
if (m_IsEmpty && !box.m_IsEmpty)
{
m_Lower= box.m_Lower;
m_Upper= box.m_Upper;
m_IsEmpty= box.m_IsEmpty;
}
else if (! box.m_IsEmpty)
{
double lowerX= qMin(box.m_Lower.x(), m_Lower.x());
double lowerY= qMin(box.m_Lower.y(), m_Lower.y());
double lowerZ= qMin(box.m_Lower.z(), m_Lower.z());
m_Lower.setVect(lowerX, lowerY, lowerZ);
double upperX= qMax(box.m_Upper.x(), m_Upper.x());
double upperY= qMax(box.m_Upper.y(), m_Upper.y());
double upperZ= qMax(box.m_Upper.z(), m_Upper.z());
m_Upper.setVect(upperX, upperY, upperZ);
}
return *this;
}
GLC_BoundingBox& GLC_BoundingBox::transform(const GLC_Matrix4x4& matrix)
{
if (!m_IsEmpty)
{
GLC_Point3d newLower= matrix * m_Lower;
GLC_Point3d newUpper= matrix * m_Upper;
GLC_BoundingBox boundingBox;
boundingBox.combine(newLower);
boundingBox.combine(newUpper);
m_Lower= boundingBox.m_Lower;
m_Upper= boundingBox.m_Upper;
}
return *this;
}
QDataStream &operator<<(QDataStream &stream, const GLC_BoundingBox &bBox)
{
quint32 chunckId= GLC_BoundingBox::m_ChunkId;
stream << chunckId;
stream << bBox.m_IsEmpty;
stream << bBox.lowerCorner();
stream << bBox.upperCorner();
return stream;
}
QDataStream &operator>>(QDataStream &stream, GLC_BoundingBox &bBox)
{
quint32 chunckId;
stream >> chunckId;
Q_ASSERT(chunckId == GLC_BoundingBox::m_ChunkId);
stream >> bBox.m_IsEmpty;
stream >> bBox.m_Lower;
stream >> bBox.m_Upper;
return stream;
}