42660d13aa
- Added iOS project - Added mutli touch support (thanks to Redshift Software). git-svn-id: http://svn.code.sf.net/p/irrlicht/code/branches/ogl-es@4184 dfc29bdd-3216-0410-991c-e03cc46cb475
251 lines
8.8 KiB
C++
251 lines
8.8 KiB
C++
// Copyright (C) 2002-2011 Nikolaus Gebhardt
|
|
// This file is part of the "Irrlicht Engine".
|
|
// For conditions of distribution and use, see copyright notice in irrlicht.h
|
|
|
|
#ifndef __IRR_TRIANGLE_3D_H_INCLUDED__
|
|
#define __IRR_TRIANGLE_3D_H_INCLUDED__
|
|
|
|
#include "vector3d.h"
|
|
#include "line3d.h"
|
|
#include "plane3d.h"
|
|
#include "aabbox3d.h"
|
|
|
|
namespace irr
|
|
{
|
|
namespace core
|
|
{
|
|
|
|
//! 3d triangle template class for doing collision detection and other things.
|
|
template <class T>
|
|
class triangle3d
|
|
{
|
|
public:
|
|
|
|
//! Constructor for an all 0 triangle
|
|
triangle3d() {}
|
|
//! Constructor for triangle with given three vertices
|
|
triangle3d(vector3d<T> v1, vector3d<T> v2, vector3d<T> v3) : pointA(v1), pointB(v2), pointC(v3) {}
|
|
|
|
//! Equality operator
|
|
bool operator==(const triangle3d<T>& other) const
|
|
{
|
|
return other.pointA==pointA && other.pointB==pointB && other.pointC==pointC;
|
|
}
|
|
|
|
//! Inequality operator
|
|
bool operator!=(const triangle3d<T>& other) const
|
|
{
|
|
return !(*this==other);
|
|
}
|
|
|
|
//! Determines if the triangle is totally inside a bounding box.
|
|
/** \param box Box to check.
|
|
\return True if triangle is within the box, otherwise false. */
|
|
bool isTotalInsideBox(const aabbox3d<T>& box) const
|
|
{
|
|
return (box.isPointInside(pointA) &&
|
|
box.isPointInside(pointB) &&
|
|
box.isPointInside(pointC));
|
|
}
|
|
|
|
//! Determines if the triangle is totally outside a bounding box.
|
|
/** \param box Box to check.
|
|
\return True if triangle is outside the box, otherwise false. */
|
|
bool isTotalOutsideBox(const aabbox3d<T>& box) const
|
|
{
|
|
return ((pointA.X > box.MaxEdge.X && pointB.X > box.MaxEdge.X && pointC.X > box.MaxEdge.X) ||
|
|
|
|
(pointA.Y > box.MaxEdge.Y && pointB.Y > box.MaxEdge.Y && pointC.Y > box.MaxEdge.Y) ||
|
|
(pointA.Z > box.MaxEdge.Z && pointB.Z > box.MaxEdge.Z && pointC.Z > box.MaxEdge.Z) ||
|
|
(pointA.X < box.MinEdge.X && pointB.X < box.MinEdge.X && pointC.X < box.MinEdge.X) ||
|
|
(pointA.Y < box.MinEdge.Y && pointB.Y < box.MinEdge.Y && pointC.Y < box.MinEdge.Y) ||
|
|
(pointA.Z < box.MinEdge.Z && pointB.Z < box.MinEdge.Z && pointC.Z < box.MinEdge.Z));
|
|
}
|
|
|
|
//! Get the closest point on a triangle to a point on the same plane.
|
|
/** \param p Point which must be on the same plane as the triangle.
|
|
\return The closest point of the triangle */
|
|
core::vector3d<T> closestPointOnTriangle(const core::vector3d<T>& p) const
|
|
{
|
|
const core::vector3d<T> rab = line3d<T>(pointA, pointB).getClosestPoint(p);
|
|
const core::vector3d<T> rbc = line3d<T>(pointB, pointC).getClosestPoint(p);
|
|
const core::vector3d<T> rca = line3d<T>(pointC, pointA).getClosestPoint(p);
|
|
|
|
const T d1 = rab.getDistanceFrom(p);
|
|
const T d2 = rbc.getDistanceFrom(p);
|
|
const T d3 = rca.getDistanceFrom(p);
|
|
|
|
if (d1 < d2)
|
|
return d1 < d3 ? rab : rca;
|
|
|
|
return d2 < d3 ? rbc : rca;
|
|
}
|
|
|
|
//! Check if a point is inside the triangle (border-points count also as inside)
|
|
/** NOTE: When working with T='int' you should prefer isPointInsideFast, as
|
|
isPointInside will run into number-overflows already with coordinates in the 3-digit-range.
|
|
\param p Point to test. Assumes that this point is already
|
|
on the plane of the triangle.
|
|
\return True if the point is inside the triangle, otherwise false. */
|
|
bool isPointInside(const vector3d<T>& p) const
|
|
{
|
|
return (isOnSameSide(p, pointA, pointB, pointC) &&
|
|
isOnSameSide(p, pointB, pointA, pointC) &&
|
|
isOnSameSide(p, pointC, pointA, pointB));
|
|
}
|
|
|
|
//! Check if a point is inside the triangle (border-points count also as inside)
|
|
/** This method uses a barycentric coordinate system.
|
|
It is faster than isPointInside but is more susceptible to floating point rounding
|
|
errors. This will especially be noticable when the FPU is in single precision mode
|
|
(which is for example set on default by Direct3D).
|
|
\param p Point to test. Assumes that this point is already
|
|
on the plane of the triangle.
|
|
\return True if point is inside the triangle, otherwise false. */
|
|
bool isPointInsideFast(const vector3d<T>& p) const
|
|
{
|
|
const vector3d<T> a = pointC - pointA;
|
|
const vector3d<T> b = pointB - pointA;
|
|
const vector3d<T> c = p - pointA;
|
|
|
|
const f64 dotAA = a.dotProduct( a);
|
|
const f64 dotAB = a.dotProduct( b);
|
|
const f64 dotAC = a.dotProduct( c);
|
|
const f64 dotBB = b.dotProduct( b);
|
|
const f64 dotBC = b.dotProduct( c);
|
|
|
|
// get coordinates in barycentric coordinate system
|
|
const f64 invDenom = 1/(dotAA * dotBB - dotAB * dotAB);
|
|
const f64 u = (dotBB * dotAC - dotAB * dotBC) * invDenom;
|
|
const f64 v = (dotAA * dotBC - dotAB * dotAC ) * invDenom;
|
|
|
|
// We count border-points as inside to keep downward compatibility.
|
|
// That's why we use >= and <= instead of > and < as more commonly seen on the web.
|
|
return (u >= 0) && (v >= 0) && (u + v <= 1);
|
|
|
|
}
|
|
|
|
|
|
//! Get an intersection with a 3d line.
|
|
/** \param line Line to intersect with.
|
|
\param outIntersection Place to store the intersection point, if there is one.
|
|
\return True if there was an intersection, false if not. */
|
|
bool getIntersectionWithLimitedLine(const line3d<T>& line,
|
|
vector3d<T>& outIntersection) const
|
|
{
|
|
return getIntersectionWithLine(line.start,
|
|
line.getVector(), outIntersection) &&
|
|
outIntersection.isBetweenPoints(line.start, line.end);
|
|
}
|
|
|
|
|
|
//! Get an intersection with a 3d line.
|
|
/** Please note that also points are returned as intersection which
|
|
are on the line, but not between the start and end point of the line.
|
|
If you want the returned point be between start and end
|
|
use getIntersectionWithLimitedLine().
|
|
\param linePoint Point of the line to intersect with.
|
|
\param lineVect Vector of the line to intersect with.
|
|
\param outIntersection Place to store the intersection point, if there is one.
|
|
\return True if there was an intersection, false if there was not. */
|
|
bool getIntersectionWithLine(const vector3d<T>& linePoint,
|
|
const vector3d<T>& lineVect, vector3d<T>& outIntersection) const
|
|
{
|
|
if (getIntersectionOfPlaneWithLine(linePoint, lineVect, outIntersection))
|
|
return isPointInside(outIntersection);
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
//! Calculates the intersection between a 3d line and the plane the triangle is on.
|
|
/** \param lineVect Vector of the line to intersect with.
|
|
\param linePoint Point of the line to intersect with.
|
|
\param outIntersection Place to store the intersection point, if there is one.
|
|
\return True if there was an intersection, else false. */
|
|
bool getIntersectionOfPlaneWithLine(const vector3d<T>& linePoint,
|
|
const vector3d<T>& lineVect, vector3d<T>& outIntersection) const
|
|
{
|
|
const vector3d<T> normal = getNormal().normalize();
|
|
T t2;
|
|
|
|
if ( core::iszero ( t2 = normal.dotProduct(lineVect) ) )
|
|
return false;
|
|
|
|
T d = pointA.dotProduct(normal);
|
|
T t = -(normal.dotProduct(linePoint) - d) / t2;
|
|
outIntersection = linePoint + (lineVect * t);
|
|
return true;
|
|
}
|
|
|
|
|
|
//! Get the normal of the triangle.
|
|
/** Please note: The normal is not always normalized. */
|
|
vector3d<T> getNormal() const
|
|
{
|
|
return (pointB - pointA).crossProduct(pointC - pointA);
|
|
}
|
|
|
|
//! Test if the triangle would be front or backfacing from any point.
|
|
/** Thus, this method assumes a camera position from which the
|
|
triangle is definitely visible when looking at the given direction.
|
|
Do not use this method with points as it will give wrong results!
|
|
\param lookDirection Look direction.
|
|
\return True if the plane is front facing and false if it is backfacing. */
|
|
bool isFrontFacing(const vector3d<T>& lookDirection) const
|
|
{
|
|
const vector3d<T> n = getNormal().normalize();
|
|
const f32 d = (f32)n.dotProduct(lookDirection);
|
|
return F32_LOWER_EQUAL_0(d);
|
|
}
|
|
|
|
//! Get the plane of this triangle.
|
|
plane3d<T> getPlane() const
|
|
{
|
|
return plane3d<T>(pointA, pointB, pointC);
|
|
}
|
|
|
|
//! Get the area of the triangle
|
|
T getArea() const
|
|
{
|
|
return (pointB - pointA).crossProduct(pointC - pointA).getLength() * 0.5f;
|
|
|
|
}
|
|
|
|
//! sets the triangle's points
|
|
void set(const core::vector3d<T>& a, const core::vector3d<T>& b, const core::vector3d<T>& c)
|
|
{
|
|
pointA = a;
|
|
pointB = b;
|
|
pointC = c;
|
|
}
|
|
|
|
//! the three points of the triangle
|
|
vector3d<T> pointA;
|
|
vector3d<T> pointB;
|
|
vector3d<T> pointC;
|
|
|
|
private:
|
|
bool isOnSameSide(const vector3d<T>& p1, const vector3d<T>& p2,
|
|
const vector3d<T>& a, const vector3d<T>& b) const
|
|
{
|
|
vector3d<T> bminusa = b - a;
|
|
vector3d<T> cp1 = bminusa.crossProduct(p1 - a);
|
|
vector3d<T> cp2 = bminusa.crossProduct(p2 - a);
|
|
return (cp1.dotProduct(cp2) >= 0.0f);
|
|
}
|
|
};
|
|
|
|
|
|
//! Typedef for a f32 3d triangle.
|
|
typedef triangle3d<f32> triangle3df;
|
|
|
|
//! Typedef for an integer 3d triangle.
|
|
typedef triangle3d<s32> triangle3di;
|
|
|
|
} // end namespace core
|
|
} // end namespace irr
|
|
|
|
#endif
|
|
|