5930 lines
172 KiB

#pragma once
#include <cstdio>
#include <cassert>
#include <cmath>
#include <stdlib.h>
#include <vector>
#include <map>
#include <limits>
namespace CELL
{
#define PI 3.14159265358979323
#define TWO_PI 6.28318530717958647
#define HALF_PI 1.57079632679489661
#define DEG2RAD(theta) (0.01745329251994329 * (theta))
#define RAD2DEG 57.2957795130823208
#define LOG2 0.69314718055994529
#define WGS_84_RADIUS_EQUATOR 6378137.0
#define WGS_84_RADIUS_POLAR 6356752.3142
#define MIN(a,b) ((a) < (b) ? (a) : (b))
#define MAX(a,b) ((a) > (b) ? (a) : (b))
#ifndef FLT_MAX
#define FLT_MAX 3.402823466e+38F
#endif
#ifndef FLT_MIN
#define FLT_MIN 1.175494351e-38F
#endif
#define MAKE_INT(a, b) ((int)(((short)(((int)(a)) & 0xffff)) | ((int)((short)(((int)(b)) & 0xffff))) << 16))
typedef unsigned char byte;
typedef long long int64;
typedef unsigned short ushort;
typedef unsigned int uint;
typedef unsigned long ulong;
template<class T> inline T tmin(T a,T b)
{
return a < b ? a:b;
}
template<class T> inline T tmax(T a,T b)
{
return a > b ? a:b;
}
union LargeInt
{
struct __LARGE_INT
{
unsigned int LowPart;
unsigned int HighPart;
}_largeInt;
int64 int64Data;
} ;
inline float unitRandom ()
{
return float(rand()) / float( RAND_MAX );
}
//-----------------------------------------------------------------------
inline float rangeRandom (float fLow, float fHigh)
{
return (fHigh-fLow)*unitRandom() + fLow;
}
/**
* 产生64位数字
*/
inline int64 makeInt64(unsigned low,unsigned hi)
{
LargeInt intdata;
intdata._largeInt.HighPart = low;
intdata._largeInt.LowPart = hi;
return intdata.int64Data;
}
template <typename T>
struct tvec2
{
typedef T value_type;
typedef std::size_t size_type;
typedef tvec2<T> type;
value_type x;
value_type y;
value_type & operator[](size_type i)
{
assert(i < this->length());
return (&x)[i];
}
value_type const & operator[]( size_type i ) const
{
assert(i < this->length());
return (&x)[i];
}
tvec2() :
x(value_type(0)),
y(value_type(0))
{}
tvec2(tvec2<T> const & v) :
x(v.x),
y(v.y)
{}
tvec2(value_type const & s) :
x(s),
y(s)
{}
tvec2(value_type const & s1, value_type const & s2) :
x(s1),
y(s2)
{}
template <typename U>
tvec2(U const & x) :
x(value_type(x)),
y(value_type(x))
{}
template <typename U, typename V>
tvec2(U const & a, V b) :
x(value_type(a)),
y(value_type(b))
{}
template <typename U>
tvec2(tvec2<U> const & v) :
x(value_type(v.x)),
y(value_type(v.y))
{}
tvec2<T> & operator= (tvec2<T> const & v)
{
this->x = v.x;
this->y = v.y;
return *this;
}
template <typename U>
tvec2<T> & operator= (tvec2<U> const & v)
{
this->x = T(v.x);
this->y = T(v.y);
return *this;
}
template <typename U>
tvec2<T> & operator+=(U const & s)
{
this->x += T(s);
this->y += T(s);
return *this;
}
template <typename U>
tvec2<T> & operator+=(tvec2<U> const & v)
{
this->x += T(v.x);
this->y += T(v.y);
return *this;
}
template <typename U>
tvec2<T> & operator-=(U const & s)
{
this->x -= T(s);
this->y -= T(s);
return *this;
}
template <typename U>
tvec2<T> & operator-=(tvec2<U> const & v)
{
this->x -= T(v.x);
this->y -= T(v.y);
return *this;
}
template <typename U>
tvec2<T> & operator*=(U s)
{
this->x *= T(s);
this->y *= T(s);
return *this;
}
template <typename U>
tvec2<T> & operator*=(tvec2<U> const & v)
{
this->x *= T(v.x);
this->y *= T(v.y);
return *this;
}
template <typename U>
tvec2<T> & operator/=(U s)
{
this->x /= T(s);
this->y /= T(s);
return *this;
}
template <typename U>
tvec2<T> & operator/=(tvec2<U> const & v)
{
this->x /= T(v.x);
this->y /= T(v.y);
return *this;
}
tvec2<T> & operator++()
{
++ this->x;
++ this->y;
return *this;
}
tvec2<T> & operator--()
{
--this->x;
--this->y;
return *this;
}
void makeCeil( tvec2<T> cmp )
{
if( cmp.x > x ) x = cmp.x;
if( cmp.y > y ) y = cmp.y;
}
void makeFloor( tvec2<T> cmp )
{
if( cmp.x < x ) x = cmp.x;
if( cmp.y < y ) y = cmp.y;
}
};
template <typename T>
tvec2<T> rotate(tvec2<T> const & v, T angle)
{
tvec2<T> res;
T const c(cos(DEG2RAD(angle)));
T const s(sin(DEG2RAD(angle)));
res.x = v.x * c - v.y * s;
res.y = v.x * s + v.y * c;
return res;
}
template <typename T>
bool operator==(tvec2<T> const & v1, tvec2<T> const & v2)
{
return (v1.x == v2.x) && (v1.y == v2.y);
}
template <typename T>
bool operator!=(tvec2<T> const & v1, tvec2<T> const & v2)
{
return (v1.x != v2.x) || (v1.y != v2.y);
}
template <typename T>
tvec2<T> operator+ (tvec2<T> const & v, T const & s)
{
return tvec2<T>(
v.x + T(s),
v.y + T(s));
}
template <typename T>
tvec2<T> operator+ (T const & s, tvec2<T> const & v)
{
return tvec2<T>(
T(s) + v.x,
T(s) + v.y);
}
template <typename T>
tvec2<T> operator+ (tvec2<T> const & v1, tvec2<T> const & v2)
{
return tvec2<T>(
v1.x + T(v2.x),
v1.y + T(v2.y));
}
template <typename T>
tvec2<T> operator-(tvec2<T> const & v, T const & s)
{
return tvec2<T>(
v.x - T(s),
v.y - T(s));
}
template <typename T>
tvec2<T> operator- (T const & s, tvec2<T> const & v)
{
return tvec2<T>(
T(s) - v.x,
T(s) - v.y);
}
template <typename T>
tvec2<T> operator- (tvec2<T> const & v1, tvec2<T> const & v2)
{
return tvec2<T>(
v1.x - T(v2.x),
v1.y - T(v2.y));
}
template <typename T>
tvec2<T> operator* (tvec2<T> const & v, T const & s)
{
return tvec2<T>(
v.x * T(s),
v.y * T(s));
}
template <typename T>
tvec2<T> operator* (T const & s, tvec2<T> const & v)
{
return tvec2<T>(
T(s) * v.x,
T(s) * v.y);
}
template <typename T>
tvec2<T> operator*(tvec2<T> const & v1, tvec2<T> const & v2)
{
return tvec2<T>(
v1.x * T(v2.x),
v1.y * T(v2.y));
}
template <typename T>
tvec2<T> operator/(tvec2<T> const & v, T const & s)
{
return tvec2<T>(
v.x / T(s),
v.y / T(s));
}
template <typename T>
tvec2<T> operator/(T const & s, tvec2<T> const & v)
{
return tvec2<T>(
T(s) / v.x,
T(s) / v.y);
}
template <typename T>
tvec2<T> operator/ (tvec2<T> const & v1, tvec2<T> const & v2 )
{
return tvec2<T>(
v1.x / T(v2.x),
v1.y / T(v2.y)
);
}
template <typename T>
tvec2<T> operator- (tvec2<T> const & v)
{
return tvec2<T> (
-v.x,
-v.y
);
}
template <typename T>
tvec2<T> operator++ (tvec2<T> const & v, int)
{
return tvec2<T>(
v.x + T(1),
v.y + T(1)
);
}
template <typename T>
tvec2<T> operator-- (tvec2<T> const & v, int)
{
return tvec2<T>(
v.x - T(1),
v.y - T(1)
);
}
template <typename T>
struct tvec3
{
typedef T value_type;
typedef std::size_t size_type;
typedef tvec3<T> type;
value_type x;
value_type y;
value_type z;
size_type length() const
{
return 3;
}
value_type & operator[](size_type i)
{
assert(i < this->length());
return (&x)[i];
}
value_type const & operator[](size_type i) const
{
assert(i < this->length());
return (&x)[i];
}
inline tvec3() :
x(value_type(0)),
y(value_type(0)),
z(value_type(0))
{}
inline tvec3(tvec3<T> const & v) :
x(v.x),
y(v.y),
z(v.z)
{}
inline tvec3(value_type s) :
x(s),
y(s),
z(s)
{}
inline tvec3(value_type s0, value_type s1, value_type s2) :
x(s0),
y(s1),
z(s2)
{}
template <typename U>
tvec3(U s) :
x(value_type(s)),
y(value_type(s)),
z(value_type(s))
{}
template <typename A, typename B, typename C>
tvec3(A x, B y, C z) :
x(value_type(x)),
y(value_type(y)),
z(value_type(z))
{}
template <typename A, typename B>
tvec3(tvec2<A> const& v, B s) :
x(value_type(v.x)),
y(value_type(v.y)),
z(value_type(s))
{}
template <typename A, typename B>
tvec3(A s,tvec2<B> const& v) :
x(value_type(s)),
y(value_type(v.x)),
z(value_type(v.y))
{}
template <typename U>
tvec3(tvec3<U> const & v) :
x(value_type(v.x)),
y(value_type(v.y)),
z(value_type(v.z))
{}
tvec3<T>& operator= (tvec3<T> const & v)
{
this->x = v.x;
this->y = v.y;
this->z = v.z;
return *this;
}
template <typename U>
tvec3<T>& operator= (tvec3<U> const & v)
{
this->x = T(v.x);
this->y = T(v.y);
this->z = T(v.z);
return *this;
}
template <typename U>
tvec3<T> & operator+=(U const & s)
{
this->x += T(s);
this->y += T(s);
this->z += T(s);
return *this;
}
template <typename U>
tvec3<T> & operator+=(tvec3<U> const & v)
{
this->x += T(v.x);
this->y += T(v.y);
this->z += T(v.z);
return *this;
}
template <typename U>
tvec3<T> & operator-=(U const & s)
{
this->x -= T(s);
this->y -= T(s);
this->z -= T(s);
return *this;
}
template <typename U>
tvec3<T> & operator-=(tvec3<U> const & v)
{
this->x -= T(v.x);
this->y -= T(v.y);
this->z -= T(v.z);
return *this;
}
template <typename U>
tvec3<T> & operator*=(U const & s)
{
this->x *= T(s);
this->y *= T(s);
this->z *= T(s);
return *this;
}
template <typename U>
tvec3<T> & operator*=(tvec3<U> const & v)
{
this->x *= T(v.x);
this->y *= T(v.y);
this->z *= T(v.z);
return *this;
}
template <typename U>
tvec3<T> & operator/=(U const & s)
{
this->x /= T(s);
this->y /= T(s);
this->z /= T(s);
return *this;
}
template <typename U>
tvec3<T> & operator/=(tvec3<U> const & v)
{
this->x /= T(v.x);
this->y /= T(v.y);
this->z /= T(v.z);
return *this;
}
tvec3<T> & operator++()
{
++this->x;
++this->y;
++this->z;
return *this;
}
tvec3<T> & operator--()
{
--this->x;
--this->y;
--this->z;
return *this;
}
void makeFloor( const tvec3<T>& cmp )
{
if( cmp.x < x ) x = cmp.x;
if( cmp.y < y ) y = cmp.y;
if( cmp.z < z ) z = cmp.z;
}
void makeCeil( const tvec3<T>& cmp )
{
if( cmp.x > x ) x = cmp.x;
if( cmp.y > y ) y = cmp.y;
if( cmp.z > z ) z = cmp.z;
}
T lengthf() const
{
return (T)sqrtf( x * x + y * y + z * z );
}
};
template<typename T>
bool operator >(const tvec3<T>& left ,const tvec3<T>& right)
{
return left.x > right.x && left.y > right.y && left.z > right.z;
}
template<typename T>
bool operator <(const tvec3<T>& left ,const tvec3<T>& right)
{
return left.x < right.x && left.y < right.y && left.z < right.z;
}
template <typename T>
tvec3<T> rotateX(const tvec3<T>& v, T angle)
{
tvec3<T> res(v);
T c = cos(T(DEG2RAD(angle)));
T s = sin(T(DEG2RAD(angle)));
res.y = v.y * c - v.z * s;
res.z = v.y * s + v.z * c;
return res;
}
template <typename T>
tvec3<T> rotateY(tvec3<T> const & v, T angle)
{
tvec3<T> res = v;
T c = cos(T(DEG2RAD(angle)));
T s = sin(T(DEG2RAD(angle)));
res.x = v.x * c + v.z * s;
res.z = -v.x * s + v.z * c;
return res;
}
template <typename T>
tvec3<T> rotateZ(tvec3<T> const & v, T angle)
{
tvec3<T> res = v;
T c = cos(DEG2RAD(angle));
T s = sin(DEG2RAD(angle));
res.x = v.x * c - v.y * s;
res.y = v.x * s + v.y * c;
return res;
}
/**
* 两个向量的夹角
* 定义两个向量 A,B
* A·B = |A|*|B|*cos(@)
* cos(@) = A·B/|A|*|B|
* @ = acos(@)
*/
template <typename T>
T angleBetweenVector(const tvec3<T>& a, const tvec3<T>& b)
{
#define Mag(V) (sqrtf(V.x*V.x + V.y*V.y + V.z*V.z))
T dotProduct = dot(a, b);
T vectorsMagnitude = Mag(a) * Mag(b) ;
T angle = acos( dotProduct / vectorsMagnitude );
T result = angle * T(RAD2DEG);
if(_isnan(result))
{
return T(0);
}
else
{
return result;
}
}
template<typename T>
inline bool _isnan(T t)
{
return t == t;
}
template <typename T>
T angleBetweenVector(const tvec2<T>& a, const tvec2<T>& b)
{
#define Mag2D(V) (sqrtf(V.x*V.x + V.y*V.y))
T dotProduct = dot(a, b);
T vectorsMagnitude = Mag2D(a) * Mag2D(b) ;
T angle = acos( dotProduct / vectorsMagnitude );
T result = angle * T(RAD2DEG);
if(_isnan(result))
{
return T(0);
}
else
{
return result;
}
}
template <typename T>
static T clamp(T val, T minval, T maxval)
{
assert (minval < maxval && "Invalid clamp range");
return MAX(MIN(val, maxval), minval);
}
template <typename T>
inline T acosEx (T val)
{
if ( T(-1.0f) < val )
{
if ( val < 1.0f )
return T(acos(val));
else
return T(0);
}
else
{
return T(PI);
}
}
template<typename T>
inline T angleBetween(const tvec3<T>& a, const tvec3<T>& b)
{
T lenProduct = a.lengthf() * b.lengthf();
// Divide by zero check
if(lenProduct < 1e-6f)
lenProduct = 1e-6f;
float f = dot(a,b) / lenProduct;
f = clamp(f, T(-1.0), T(1.0));
return acosEx(f);
}
/**
* 点在多边形里
* 如果点在多边形中,则,点与边的夹角之和 == 360
*/
template<typename T>
bool insidePolyon( const tvec3<T>& point, const tvec3<T> polygon[], size_t count)
{
tvec3<T> vA, vB;
T angle = T(0.0);
for (size_t i = 0; i < count; ++i)
{
vA = polygon[i] - point;
vB = polygon[(i + 1) % count] - point;
angle += angleBetweenVector(vA, vB);
}
if( abs(angle - 360 ) >= 0.5f)
{
return true;
}
return false;
}
template<typename T>
bool insidePolyon( const tvec2<T>& point, const tvec2<T> polygon[], size_t count)
{
T angle = T(0.0);
tvec2<T> vA, vB;
for (size_t i = 0; i < count; ++i)
{
vA = polygon[i] - point;
vB = polygon[(i + 1) % count] - point;
tvec3<T> a(vA.x,vA.y,0);
tvec3<T> b(vB.x,vB.y,0);
angle += angleBetweenVector<T>(a, b);
}
if( abs(angle - 360 ) >= 0.5f)
{
return true;
}
return false;
}
template<typename T>
bool pointinTriangle(tvec3<T> A, tvec3<T> B, tvec3<T> C, tvec3<T> P)
{
tvec3<T> v0 = C - A ;
tvec3<T> v1 = B - A ;
tvec3<T> v2 = P - A ;
float dot00 = dot(v0,v0) ;
float dot01 = dot(v0,v1) ;
float dot02 = dot(v0,v2) ;
float dot11 = dot(v1,v1) ;
float dot12 = dot(v1,v2) ;
float inverDeno = 1 / (dot00 * dot11 - dot01 * dot01) ;
float u = (dot11 * dot02 - dot01 * dot12) * inverDeno ;
if (u < 0 || u > 1) // if u out of range, return directly
{
return false ;
}
float v = (dot00 * dot12 - dot01 * dot02) * inverDeno ;
if (v < 0 || v > 1) // if v out of range, return directly
{
return false ;
}
return u + v <= 1 ;
}
template<typename T>
bool pointinTriangle(tvec2<T> A, tvec2<T> B, tvec2<T> C, tvec2<T> P)
{
return pointinTriangle(
tvec3<T>(A.x,A.y,0),
tvec3<T>(B.x,B.y,0),
tvec3<T>(C.x,C.y,0),
tvec3<T>(P.x,P.y,0));
}
/**
* 射线与三角形相交
*/
template<typename T>
bool intersectTriangle(
const tvec3<T>& orig,
const tvec3<T>& dir,
tvec3<T>& v0,
tvec3<T>& v1,
tvec3<T>& v2,
T* t,
T* u,
T* v
)
{
// Find vectors for two edges sharing vert0
tvec3<T> edge1 = v1 - v0;
tvec3<T> edge2 = v2 - v0;
// Begin calculating determinant - also used to calculate U parameter
tvec3<T> pvec;
pvec = cross(dir, edge2 );
// If determinant is near zero, ray lies in plane of triangle
T det = dot( edge1,pvec );
tvec3<T> tvec;
if( det > 0 )
{
tvec = orig - v0;
}
else
{
tvec = v0 - orig;
det = -det;
}
if( det < 0.0001f )
return false;
// Calculate U parameter and test bounds
*u = dot( tvec, pvec );
if( *u < 0.0f || *u > det )
return false;
// Prepare to test V parameter
tvec3<T> qvec;
qvec = cross(tvec, edge1 );
// Calculate V parameter and test bounds
*v = dot( dir, qvec );
if( *v < T(0.0f) || *u + *v > det )
return false;
*t = dot( edge2,qvec );
T fInvDet = T(1.0) / det;
*t *= fInvDet;
*u *= fInvDet;
*v *= fInvDet;
return true;
}
/**
* 计算三角形面积
*/
template<typename T> T calcTriangleArea(const tvec3<T>& pt1,const tvec3<T>& pt2,const tvec3<T>& pt3)
{
tvec3<T> e1 = pt2 - pt1;
tvec3<T> e2 = pt3 - pt1;
tvec3<T> e3 = cross(e1,e2);
return length(e3) * T(0.5);
}
template <typename T>
bool operator==(tvec3<T> const & v1, tvec3<T> const & v2)
{
return (v1.x == v2.x) && (v1.y == v2.y) && (v1.z == v2.z);
}
template <typename T>
bool operator!=(tvec3<T> const & v1, tvec3<T> const & v2)
{
return (v1.x != v2.x) || (v1.y != v2.y) || (v1.z != v2.z);
}
template <typename T>
tvec3<T> operator+(tvec3<T> const & v, T const & s)
{
return tvec3<T>(
v.x + T(s),
v.y + T(s),
v.z + T(s));
}
template <typename T>
tvec3<T> operator+ ( T const & s, tvec3<T> const & v)
{
return tvec3<T>(
T(s) + v.x,
T(s) + v.y,
T(s) + v.z);
}
template <typename T>
tvec3<T> operator+ (tvec3<T> const & v1, tvec3<T> const & v2)
{
return tvec3<T>(
v1.x + T(v2.x),
v1.y + T(v2.y),
v1.z + T(v2.z));
}
template <typename T>
tvec3<T> operator- (tvec3<T> const & v, T const & s)
{
return tvec3<T>(
v.x - T(s),
v.y - T(s),
v.z - T(s));
}
template <typename T>
tvec3<T> operator- (T const & s, tvec3<T> const & v)
{
return tvec3<T>(
T(s) - v.x,
T(s) - v.y,
T(s) - v.z);
}
template <typename T>
tvec3<T> operator- (tvec3<T> const & v1, tvec3<T> const & v2)
{
return tvec3<T>(
v1.x - T(v2.x),
v1.y - T(v2.y),
v1.z - T(v2.z));
}
template <typename T>
tvec3<T> operator*(tvec3<T> const & v, T const & s)
{
return tvec3<T>(
v.x * T(s),
v.y * T(s),
v.z * T(s));
}
template <typename T>
tvec3<T> operator* (T const & s, tvec3<T> const & v)
{
return tvec3<T>(
T(s) * v.x,
T(s) * v.y,
T(s) * v.z);
}
template <typename T>
tvec3<T> operator* (tvec3<T> const & v1, tvec3<T> const & v2)
{
return tvec3<T>(
v1.x * T(v2.x),
v1.y * T(v2.y),
v1.z * T(v2.z));
}
template <typename T>
tvec3<T> operator/ (tvec3<T> const & v, T const & s)
{
return tvec3<T>(
v.x / T(s),
v.y / T(s),
v.z / T(s));
}
template <typename T>
tvec3<T> operator/ (T const & s, tvec3<T> const & v)
{
return tvec3<T>(
T(s) / v.x,
T(s) / v.y,
T(s) / v.z);
}
template <typename T>
tvec3<T> operator/ (tvec3<T> const & v1, tvec3<T> const & v2)
{
return tvec3<T>(
v1.x / T(v2.x),
v1.y / T(v2.y),
v1.z / T(v2.z));
}
template <typename T>
tvec3<T> operator- (tvec3<T> const & v)
{
return tvec3<T>(
-v.x,
-v.y,
-v.z);
}
template <typename T>
tvec3<T> operator++ (tvec3<T> const & v, int)
{
return tvec3<T>(
v.x + T(1),
v.y + T(1),
v.z + T(1));
}
template <typename T>
tvec3<T> operator-- (tvec3<T> const & v, int)
{
return tvec3<T>(
v.x - T(1),
v.y - T(1),
v.z - T(1));
}
template <typename T>
struct tvec4
{
typedef T value_type;
typedef std::size_t size_type;
typedef tvec4<T> type;
value_type x, y, z, w;
size_type length() const
{
return 4;
}
value_type & operator[](size_type i)
{
assert(i < this->length());
return (&x)[i];
}
value_type const & operator[](size_type i) const
{
assert(i < this->length());
return (&x)[i];
}
tvec4() :
x(value_type(0)),
y(value_type(0)),
z(value_type(0)),
w(value_type(0))
{}
tvec4(tvec3<T> const& v, T s) :
x(v.x),
y(v.y),
z(v.z),
w(s)
{}
tvec4(T s) :
x(s),
y(s),
z(s),
w(s)
{}
tvec4(tvec4<T> const & v) :
x(v.x),
y(v.y),
z(v.z),
w(v.w)
{}
template <typename A, typename B>
tvec4(tvec3<A> const & v, B s):
x(value_type(v.x)),
y(value_type(v.y)),
z(value_type(v.z)),
w(value_type(s))
{}
template <typename A, typename B>
tvec4(A s,tvec3<B> const & v):
x(value_type(s)),
y(value_type(v.x)),
z(value_type(v.y)),
w(value_type(v.z))
{}
template<typename U>
tvec4(tvec4<U> const & v) :
x(value_type(v.x)),
y(value_type(v.y)),
z(value_type(v.z)),
w(value_type(v.w))
{}
tvec4
(
value_type s1,
value_type s2,
value_type s3,
value_type s4
) :
x(s1),
y(s2),
z(s3),
w(s4)
{}
tvec4<T> & operator=(tvec4<T> const & v)
{
this->x = v.x;
this->y = v.y;
this->z = v.z;
this->w = v.w;
return *this;
}
template <typename U>
tvec4<T> & operator= (tvec4<U> const & v)
{
this->x = T(v.x);
this->y = T(v.y);
this->z = T(v.z);
this->w = T(v.w);
return *this;
}
template <typename U>
tvec4<T> & operator+=(U const & s)
{
this->x += T(s);
this->y += T(s);
this->z += T(s);
this->w += T(s);
return *this;
}
template <typename U>
tvec4<T> & operator+=(tvec4<U> const & v)
{
this->x += T(v.x);
this->y += T(v.y);
this->z += T(v.z);
this->w += T(v.w);
return *this;
}
template <typename U>
tvec4<T> & operator-=(U const & s)
{
this->x -= T(s);
this->y -= T(s);
this->z -= T(s);
this->w -= T(s);
return *this;
}
template <typename U>
tvec4<T> & operator-=(tvec4<U> const & v)
{
this->x -= T(v.x);
this->y -= T(v.y);
this->z -= T(v.z);
this->w -= T(v.w);
return *this;
}
template <typename U>
tvec4<T> & operator*= (U const & s)
{
this->x *= T(s);
this->y *= T(s);
this->z *= T(s);
this->w *= T(s);
return *this;
}
template <typename U>
tvec4<T> & operator*=( tvec4<U> const & v)
{
this->x *= T(v.x);
this->y *= T(v.y);
this->z *= T(v.z);
this->w *= T(v.w);
return *this;
}
template <typename U>
tvec4<T> & operator/=(U const & s)
{
this->x /= T(s);
this->y /= T(s);
this->z /= T(s);
this->w /= T(s);
return *this;
}
template <typename U>
tvec4<T> & operator/=(tvec4<U> const & v)
{
this->x /= T(v.x);
this->y /= T(v.y);
this->z /= T(v.z);
this->w /= T(v.w);
return *this;
}
tvec4<T> & operator++()
{
++this->x;
++this->y;
++this->z;
++this->w;
return *this;
}
tvec4<T> & operator--()
{
--this->x;
--this->y;
--this->z;
--this->w;
return *this;
}
};
template <typename T>
tvec4<T> rotateX(const tvec4<T>& v, T angle)
{
tvec4<T> res(v);
T c = cos(DEG2RAD(angle));
T s = sin(DEG2RAD(angle));
res.y = v.y * c - v.z * s;
res.z = v.y * s + v.z * c;
return res;
}
template <typename T>
tvec4<T> rotateY(tvec4<T> const & v, T angle)
{
tvec4<T> res = v;
T c = cos(DEG2RAD(angle));
T s = sin(DEG2RAD(angle));
res.x = v.x * c + v.z * s;
res.z = -v.x * s + v.z * c;
return res;
}
template <typename T>
tvec4<T> rotateZ(tvec4<T> const & v, T angle)
{
tvec4<T> res = v;
T c = cos(DEG2RAD(angle));
T s = sin(DEG2RAD(angle));
res.x = v.x * c - v.y * s;
res.y = v.x * s + v.y * c;
return res;
}
template <typename T>
tvec4<T> operator+ (tvec4<T> const & v, T const & s)
{
return tvec4<T>(
v.x + s,
v.y + s,
v.z + s,
v.w + s);
}
template <typename T>
tvec4<T> operator+ (T const & s, tvec4<T> const & v)
{
return tvec4<T>(
s + v.x,
s + v.y,
s + v.z,
s + v.w);
}
template <typename T>
tvec4<T> operator+ (tvec4<T> const & v1, tvec4<T> const & v2)
{
return tvec4<T>(
v1.x + v2.x,
v1.y + v2.y,
v1.z + v2.z,
v1.w + v2.w);
}
template <typename T>
tvec4<T> operator- (tvec4<T> const & v, T const & s)
{
return tvec4<T>(
v.x - s,
v.y - s,
v.z - s,
v.w - s);
}
template <typename T>
tvec4<T> operator- (T const & s, tvec4<T> const & v)
{
return tvec4<T>(
s - v.x,
s - v.y,
s - v.z,
s - v.w);
}
template <typename T>
tvec4<T> operator-
(
tvec4<T> const & v1,
tvec4<T> const & v2
)
{
return tvec4<T>(
v1.x - v2.x,
v1.y - v2.y,
v1.z - v2.z,
v1.w - v2.w);
}
template <typename T>
tvec4<T> operator* (tvec4<T> const & v, T const & s)
{
return tvec4<T>(
v.x * s,
v.y * s,
v.z * s,
v.w * s);
}
template <typename T>
tvec4<T> operator* (T const & s, tvec4<T> const & v)
{
return tvec4<T>(
s * v.x,
s * v.y,
s * v.z,
s * v.w);
}
template <typename T>
tvec4<T> operator*(tvec4<T> const & v1, tvec4<T> const & v2)
{
return tvec4<T>(
v1.x * v2.x,
v1.y * v2.y,
v1.z * v2.z,
v1.w * v2.w);
}
template <typename T>
tvec4<T> operator/ (tvec4<T> const & v, T const & s)
{
return tvec4<T>(
v.x / s,
v.y / s,
v.z / s,
v.w / s);
}
template <typename T>
tvec4<T> operator/ (T const & s, tvec4<T> const & v)
{
return tvec4<T>(
s / v.x,
s / v.y,
s / v.z,
s / v.w);
}
template <typename T>
tvec4<T> operator/ ( tvec4<T> const & v1, tvec4<T> const & v2)
{
return tvec4<T>(
v1.x / v2.x,
v1.y / v2.y,
v1.z / v2.z,
v1.w / v2.w);
}
template <typename T>
tvec4<T> operator- ( tvec4<T> const & v)
{
return tvec4<T>(
-v.x,
-v.y,
-v.z,
-v.w);
}
template <typename T>
bool operator==
(
tvec4<T> const & v1,
tvec4<T> const & v2
)
{
return (v1.x == v2.x) && (v1.y == v2.y) && (v1.z == v2.z) && (v1.w == v2.w);
}
template <typename T>
bool operator!=(tvec4<T> const & v1, tvec4<T> const & v2)
{
return (v1.x != v2.x) || (v1.y != v2.y) || (v1.z != v2.z) || (v1.w != v2.w);
}
template<typename T>
class trect
{
public:
trect(T left = 0,T top = 0,T right = 0,T bottom = 0)
{
_left = left;
_top = top;
_right = right;
_bottom = bottom;
}
void fromCenter(T x,T y,T size)
{
_left = x - size * T(0.5f);
_top = y - size * T(0.5f);
_right = x + size * T(0.5f);
_bottom = y + size * T(0.5f);
}
void fromCenter(T x,T y,T sizeX,T sizeY)
{
_left = x - sizeX * T(0.5f);
_top = y - sizeY * T(0.5f);
_right = x + sizeX * T(0.5f);
_bottom = y + sizeY * T(0.5f);
}
bool ptInRect(T x,T y)
{
return x >= _left && x <= _right && y >= _top && y <= _bottom;
}
tvec2<T>center() const
{
return tvec2<T>((_left + _right) * T(0.5f),(_bottom + _top) * T(0.5f));
}
tvec2<T>halSize() const
{
return tvec2<T>((_right - _left) * T(0.5f),(_bottom - _top) * T(0.5f));
}
public:
T _left;
T _top;
T _right;
T _bottom;
};
template <typename T>
struct tmat2x2
{
typedef T value_type;
typedef std::size_t size_type;
typedef tvec2<T> col_type;
typedef tvec2<T> row_type;
typedef tmat2x2<T> type;
typedef tmat2x2<T> transpose_type;
public:
tmat2x2<T> _inverse() const
{
value_type Determinant = this->value[0][0] * this->value[1][1] - this->value[1][0] * this->value[0][1];
tmat2x2<T> Inverse(
+ this->value[1][1] / Determinant,
- this->value[0][1] / Determinant,
- this->value[1][0] / Determinant,
+ this->value[0][0] / Determinant);
return Inverse;
}
private:
col_type value[2];
public:
size_type length() const
{
return 2;
}
static size_type col_size()
{
return 2;
}
static size_type row_size()
{
return 2;
}
col_type &operator[](size_type i)
{
assert(i < this->length());
return this->value[i];
}
col_type const &operator[](size_type i) const
{
assert(i < this->length());
return this->value[i];
}
tmat2x2()
{
this->value[0] = col_type(1, 0);
this->value[1] = col_type(0, 1);
}
tmat2x2(tmat2x2<T> const & m)
{
this->value[0] = m.value[0];
this->value[1] = m.value[1];
}
tmat2x2(value_type s)
{
value_type const Zero(0);
this->value[0] = col_type(s, Zero);
this->value[1] = col_type(Zero, s);
}
tmat2x2(value_type x0, value_type y0, value_type x1, value_type y1)
{
this->value[0] = col_type(x0, y0);
this->value[1] = col_type(x1, y1);
}
tmat2x2(col_type const & v0, col_type const & v1)
{
this->value[0] = v0;
this->value[1] = v1;
}
template <typename U>
tmat2x2(U s)
{
value_type const Zero(0);
this->value[0] = tvec2<T>(value_type(s), Zero);
this->value[1] = tvec2<T>(Zero, value_type(s));
}
template <typename X1, typename Y1, typename X2, typename Y2>
tmat2x2(X1 x1, Y1 y1, X2 x2, Y2 y2)
{
this->value[0] = col_type(value_type(x1), value_type(y1));
this->value[1] = col_type(value_type(x2), value_type(y2));
}
template <typename V1, typename V2>
tmat2x2
(
tvec2<V1> const & v1,
tvec2<V2> const & v2
)
{
this->value[0] = col_type(v1);
this->value[1] = col_type(v2);
}
template <typename U>
tmat2x2(tmat2x2<U> const & m)
{
this->value[0] = col_type(m[0]);
this->value[1] = col_type(m[1]);
}
tmat2x2<T>& operator=(tmat2x2<T> const & m)
{
this->value[0] = m[0];
this->value[1] = m[1];
return *this;
}
template <typename U>
tmat2x2<T>& operator=
(
tmat2x2<U> const & m
)
{
this->value[0] = m[0];
this->value[1] = m[1];
return *this;
}
template <typename U>
tmat2x2<T>& operator+=(U const & s)
{
this->value[0] += s;
this->value[1] += s;
return *this;
}
template <typename U>
tmat2x2<T>& operator+=
(
tmat2x2<U> const & m
)
{
this->value[0] += m[0];
this->value[1] += m[1];
return *this;
}
template <typename U>
tmat2x2<T>& operator-=(U const & s)
{
this->value[0] -= s;
this->value[1] -= s;
return *this;
}
template <typename U>
tmat2x2<T>& operator-=(tmat2x2<U> const & m)
{
this->value[0] -= m[0];
this->value[1] -= m[1];
return *this;
}
template <typename U>
tmat2x2<T>& operator*= ( U const & s)
{
this->value[0] *= s;
this->value[1] *= s;
return *this;
}
template <typename U>
tmat2x2<T>& operator*= (tmat2x2<U> const & m)
{
return (*this = *this * m);
}
template <typename U>
tmat2x2<T>& operator/= (U const & s)
{
this->value[0] /= s;
this->value[1] /= s;
return *this;
}
template <typename U>
tmat2x2<T>& operator/= (tmat2x2<U> const & m)
{
return (*this = *this / m);
}
tmat2x2<T>& operator++ ()
{
++this->value[0];
++this->value[1];
return *this;
}
tmat2x2<T>& operator-- ()
{
--this->value[0];
--this->value[1];
return *this;
};
};
template <typename T>
tmat2x2<T> rotate(T angle)
{
T c = cos(DEG2RAD(angle));
T s = sin(DEG2RAD(angle));
return tmat2x2<T>(c,s,-s,c);
}
template <typename T>
tmat2x2<T> operator+ (tmat2x2<T> const & m, T const & s)
{
return tmat2x2<T>(m[0] + s,m[1] + s);
}
template <typename T>
tmat2x2<T> operator+ (T const & s, tmat2x2<T> const & m)
{
return tmat2x2<T>(m[0] + s,m[1] + s);
}
template <typename T>
tmat2x2<T> operator+ (tmat2x2<T> const & m1, tmat2x2<T> const & m2)
{
return tmat2x2<T>(m1[0] + m2[0],m1[1] + m2[1]);
}
template <typename T>
tmat2x2<T> operator- (tmat2x2<T> const & m, T const & s)
{
return tmat2x2<T>(m[0] - s,m[1] - s);
}
template <typename T>
tmat2x2<T> operator- (T const & s, tmat2x2<T> const & m)
{
return tmat2x2<T>(s - m[0],s - m[1]);
}
template <typename T>
tmat2x2<T> operator- (tmat2x2<T> const & m1, tmat2x2<T> const & m2)
{
return tmat2x2<T>(m1[0] - m2[0],m1[1] - m2[1]);
}
template <typename T>
tmat2x2<T> operator* (tmat2x2<T> const & m, T const & s)
{
return tmat2x2<T>(m[0] * s,m[1] * s);
}
template <typename T>
tmat2x2<T> operator* ( T const & s, tmat2x2<T> const & m)
{
return tmat2x2<T>(m[0] * s,m[1] * s);
}
template <typename T>
tvec2<T> operator*(tmat2x2<T> const & m, tvec2<T> const & v)
{
return tvec2<T>(
m[0][0] * v.x + m[1][0] * v.y,
m[0][1] * v.x + m[1][1] * v.y);
}
template <typename T>
tvec2<T> operator*(tvec2<T> const & v, tmat2x2<T> const & m)
{
return tvec2<T>(
v.x * m[0][0] + v.y * m[0][1],
v.x * m[1][0] + v.y * m[1][1]);
}
template <typename T>
tmat2x2<T> operator*(tmat2x2<T> const & m1,tmat2x2<T> const & m2)
{
return tmat2x2<T>(
m1[0][0] * m2[0][0] + m1[1][0] * m2[0][1],
m1[0][1] * m2[0][0] + m1[1][1] * m2[0][1],
m1[0][0] * m2[1][0] + m1[1][0] * m2[1][1],
m1[0][1] * m2[1][0] + m1[1][1] * m2[1][1]);
}
template <typename T>
struct tmat3x3
{
typedef T value_type;
typedef std::size_t size_type;
typedef tvec3<T> col_type;
typedef tvec3<T> row_type;
typedef tmat3x3<T> type;
typedef tmat3x3<T> transpose_type;
private:
// Data
col_type value[3];
public:
size_type length() const
{
return 3;
}
size_type col_size()
{
return 3;
}
size_type row_size()
{
return 3;
}
tmat3x3()
{
value_type const Zero(0);
value_type const One(1);
this->value[0] = col_type(One, Zero, Zero);
this->value[1] = col_type(Zero, One, Zero);
this->value[2] = col_type(Zero, Zero, One);
}
tmat3x3
(
tmat3x3<T> const & m
)
{
this->value[0] = m.value[0];
this->value[1] = m.value[1];
this->value[2] = m.value[2];
}
tmat3x3(value_type const & s)
{
value_type const Zero(0);
this->value[0] = col_type(s, Zero, Zero);
this->value[1] = col_type(Zero, s, Zero);
this->value[2] = col_type(Zero, Zero, s);
}
tmat3x3
(
value_type const & x0, value_type const & y0, value_type const & z0,
value_type const & x1, value_type const & y1, value_type const & z1,
value_type const & x2, value_type const & y2, value_type const & z2
)
{
this->value[0] = col_type(x0, y0, z0);
this->value[1] = col_type(x1, y1, z1);
this->value[2] = col_type(x2, y2, z2);
}
tmat3x3
(
col_type const & v0,
col_type const & v1,
col_type const & v2
)
{
this->value[0] = v0;
this->value[1] = v1;
this->value[2] = v2;
}
template <typename U>
tmat3x3(U const & s)
{
value_type const Zero(0);
this->value[0] = tvec3<T>(value_type(s), Zero, Zero);
this->value[1] = tvec3<T>(Zero, value_type(s), Zero);
this->value[2] = tvec3<T>(Zero, Zero, value_type(s));
}
template <
typename X1, typename Y1, typename Z1,
typename X2, typename Y2, typename Z2,
typename X3, typename Y3, typename Z3>
tmat3x3
(
X1 const & x1, Y1 const & y1, Z1 const & z1,
X2 const & x2, Y2 const & y2, Z2 const & z2,
X3 const & x3, Y3 const & y3, Z3 const & z3
)
{
this->value[0] = col_type(value_type(x1), value_type(y1), value_type(z1));
this->value[1] = col_type(value_type(x2), value_type(y2), value_type(z2));
this->value[2] = col_type(value_type(x3), value_type(y3), value_type(z3));
}
template <typename V1, typename V2, typename V3>
tmat3x3
(
tvec3<V1> const & v1,
tvec3<V2> const & v2,
tvec3<V3> const & v3
)
{
this->value[0] = col_type(v1);
this->value[1] = col_type(v2);
this->value[2] = col_type(v3);
}
template <typename U>
tmat3x3(tmat3x3<U> const & m)
{
this->value[0] = col_type(m[0]);
this->value[1] = col_type(m[1]);
this->value[2] = col_type(m[2]);
}
tmat3x3<T> _inverse() const
{
T S00 = value[0][0];
T S01 = value[0][1];
T S02 = value[0][2];
T S10 = value[1][0];
T S11 = value[1][1];
T S12 = value[1][2];
T S20 = value[2][0];
T S21 = value[2][1];
T S22 = value[2][2];
tmat3x3<T> Inverse(
S11 * S22 - S21 * S12,
S12 * S20 - S22 * S10,
S10 * S21 - S20 * S11,
S02 * S21 - S01 * S22,
S00 * S22 - S02 * S20,
S01 * S20 - S00 * S21,
S12 * S01 - S11 * S02,
S10 * S02 - S12 * S00,
S11 * S00 - S10 * S01);
T Determinant = S00 * (S11 * S22 - S21 * S12)
- S10 * (S01 * S22 - S21 * S02)
+ S20 * (S01 * S12 - S11 * S02);
Inverse /= Determinant;
return Inverse;
}
col_type & operator[](size_type i)
{
assert(i < this->length());
return this->value[i];
}
col_type const & operator[](size_type i) const
{
assert(i < this->length());
return this->value[i];
}
tmat3x3<T> & operator=(tmat3x3<T> const & m)
{
this->value[0] = m[0];
this->value[1] = m[1];
this->value[2] = m[2];
return *this;
}
template <typename U>
tmat3x3<T> & operator=(tmat3x3<U> const & m)
{
this->value[0] = m[0];
this->value[1] = m[1];
this->value[2] = m[2];
return *this;
}
template <typename U>
tmat3x3<T> & operator+= (U const & s)
{
this->value[0] += s;
this->value[1] += s;
this->value[2] += s;
return *this;
}
template <typename U>
tmat3x3<T> & operator+=(tmat3x3<U> const & m)
{
this->value[0] += m[0];
this->value[1] += m[1];
this->value[2] += m[2];
return *this;
}
template <typename U>
tmat3x3<T> & operator-= (U const & s)
{
this->value[0] -= s;
this->value[1] -= s;
this->value[2] -= s;
return *this;
}
template <typename U>
tmat3x3<T> & operator-= (tmat3x3<U> const & m)
{
this->value[0] -= m[0];
this->value[1] -= m[1];
this->value[2] -= m[2];
return *this;
}
template <typename U>
tmat3x3<T> & operator*= (U const & s)
{
this->value[0] *= s;
this->value[1] *= s;
this->value[2] *= s;
return *this;
}
template <typename U>
tmat3x3<T> & operator*= (tmat3x3<U> const & m)
{
return (*this = *this * m);
}
template <typename U>
tmat3x3<T> & operator/= (U const & s)
{
this->value[0] /= s;
this->value[1] /= s;
this->value[2] /= s;
return *this;
}
template <typename U>
tmat3x3<T> & operator/= (tmat3x3<U> const & m)
{
return (*this = *this / m);
}
tmat3x3<T> & operator++ ()
{
++this->value[0];
++this->value[1];
++this->value[2];
return *this;
}
tmat3x3<T> & operator-- ()
{
--this->value[0];
--this->value[1];
--this->value[2];
return *this;
}
};
template <typename T>
tmat3x3<T> operator+ (tmat3x3<T> const & m, T const & s)
{
return tmat3x3<T>(
m[0] + s,
m[1] + s,
m[2] + s);
}
template <typename T>
tmat3x3<T> operator+ (T const & s, tmat3x3<T> const & m)
{
return tmat3x3<T>(
m[0] + s,
m[1] + s,
m[2] + s);
}
template <typename T>
tmat3x3<T> operator+ (tmat3x3<T> const & m1, tmat3x3<T> const & m2)
{
return tmat3x3<T>(
m1[0] + m2[0],
m1[1] + m2[1],
m1[2] + m2[2]);
}
template <typename T>
tmat3x3<T> operator- (tmat3x3<T> const & m, T const & s)
{
return tmat3x3<T>(
m[0] - s,
m[1] - s,
m[2] - s);
}
template <typename T>
tmat3x3<T> operator- (T const & s, tmat3x3<T> const & m)
{
return tmat3x3<T>(
s - m[0],
s - m[1],
s - m[2]);
}
template <typename T>
tmat3x3<T> operator- (tmat3x3<T> const & m1, tmat3x3<T> const & m2)
{
return tmat3x3<T>(
m1[0] - m2[0],
m1[1] - m2[1],
m1[2] - m2[2]);
}
template <typename T>
tmat3x3<T> operator* (tmat3x3<T> const & m, T const & s)
{
return tmat3x3<T>(
m[0] * s,
m[1] * s,
m[2] * s);
}
template <typename T>
tmat3x3<T> operator* (T const & s, tmat3x3<T> const & m)
{
return tmat3x3<T>(
m[0] * s,
m[1] * s,
m[2] * s);
}
template <typename T>
tvec3<T> operator* (tmat3x3<T> const & m, tvec3<T> const & v)
{
return tvec3<T>(
m[0][0] * v.x + m[1][0] * v.y + m[2][0] * v.z,
m[0][1] * v.x + m[1][1] * v.y + m[2][1] * v.z,
m[0][2] * v.x + m[1][2] * v.y + m[2][2] * v.z);
}
template <typename T>
tvec3<T> operator* (tvec3<T> const & v, tmat3x3<T> const & m)
{
return tvec3<T>(
m[0][0] * v.x + m[0][1] * v.y + m[0][2] * v.z,
m[1][0] * v.x + m[1][1] * v.y + m[1][2] * v.z,
m[2][0] * v.x + m[2][1] * v.y + m[2][2] * v.z);
}
template <typename T>
tmat3x3<T> operator* (tmat3x3<T> const & m1, tmat3x3<T> const & m2)
{
T const srcA00 = m1[0][0];
T const srcA01 = m1[0][1];
T const srcA02 = m1[0][2];
T const srcA10 = m1[1][0];
T const srcA11 = m1[1][1];
T const srcA12 = m1[1][2];
T const srcA20 = m1[2][0];
T const srcA21 = m1[2][1];
T const srcA22 = m1[2][2];
T const srcB00 = m2[0][0];
T const srcB01 = m2[0][1];
T const srcB02 = m2[0][2];
T const srcB10 = m2[1][0];
T const srcB11 = m2[1][1];
T const srcB12 = m2[1][2];
T const srcB20 = m2[2][0];
T const srcB21 = m2[2][1];
T const srcB22 = m2[2][2];
tmat3x3<T> res;
res[0][0] = srcA00 * srcB00 + srcA10 * srcB01 + srcA20 * srcB02;
res[0][1] = srcA01 * srcB00 + srcA11 * srcB01 + srcA21 * srcB02;
res[0][2] = srcA02 * srcB00 + srcA12 * srcB01 + srcA22 * srcB02;
res[1][0] = srcA00 * srcB10 + srcA10 * srcB11 + srcA20 * srcB12;
res[1][1] = srcA01 * srcB10 + srcA11 * srcB11 + srcA21 * srcB12;
res[1][2] = srcA02 * srcB10 + srcA12 * srcB11 + srcA22 * srcB12;
res[2][0] = srcA00 * srcB20 + srcA10 * srcB21 + srcA20 * srcB22;
res[2][1] = srcA01 * srcB20 + srcA11 * srcB21 + srcA21 * srcB22;
res[2][2] = srcA02 * srcB20 + srcA12 * srcB21 + srcA22 * srcB22;
return res;
}
template <typename T>
tmat3x3<T> operator/ (tmat3x3<T> const & m, T const & s)
{
return tmat3x3<T>(
m[0] / s,
m[1] / s,
m[2] / s);
}
template <typename T>
tmat3x3<T> operator/ (T const & s, tmat3x3<T> const & m)
{
return tmat3x3<T>(
s / m[0],
s / m[1],
s / m[2]
);
}
template <typename T>
tvec3<T> operator/ (tmat3x3<T> const & m, tvec3<T> const & v)
{
return m._inverse() * v;
}
template <typename T>
tvec3<T> operator/ (tvec3<T> const & v, tmat3x3<T> const & m)
{
return v * m._inverse();
}
template <typename T>
tmat3x3<T> operator/ (tmat3x3<T> const & m1, tmat3x3<T> const & m2)
{
return m1 * m2._inverse();
}
template <typename T>
tmat3x3<T> const operator- (tmat3x3<T> const & m)
{
return tmat3x3<T>(
-m[0],
-m[1],
-m[2]);
}
template <typename T>
tmat3x3<T> const operator++ (tmat3x3<T> const & m, int)
{
return tmat3x3<T>(
m[0] + T(1),
m[1] + T(1),
m[2] + T(1));
}
template <typename T>
tmat3x3<T> const operator-- (tmat3x3<T> const & m, int)
{
return tmat3x3<T>(
m[0] - T(1),
m[1] - T(1),
m[2] - T(1));
}
template <typename T>
bool operator==(tmat3x3<T> const & m1, tmat3x3<T> const & m2)
{
return (m1[0] == m2[0]) && (m1[1] == m2[1]) && (m1[2] == m2[2]);
}
template <typename T>
bool operator!=(tmat3x3<T> const & m1, tmat3x3<T> const & m2)
{
return (m1[0] != m2[0]) || (m1[1] != m2[1]) || (m1[2] != m2[2]);
}
template <typename T>
struct tmat4x4
{
typedef T value_type;
typedef std::size_t size_type;
typedef tvec4<T> col_type;
typedef tvec4<T> row_type;
typedef tmat4x4<T> type;
typedef tmat4x4<T> transpose_type;
public:
tmat4x4<T> inverse() const
{
value_type subFactor00 = this->value[2][2] * this->value[3][3] - this->value[3][2] * this->value[2][3];
value_type subFactor01 = this->value[2][1] * this->value[3][3] - this->value[3][1] * this->value[2][3];
value_type subFactor02 = this->value[2][1] * this->value[3][2] - this->value[3][1] * this->value[2][2];
value_type subFactor03 = this->value[2][0] * this->value[3][3] - this->value[3][0] * this->value[2][3];
value_type subFactor04 = this->value[2][0] * this->value[3][2] - this->value[3][0] * this->value[2][2];
value_type subFactor05 = this->value[2][0] * this->value[3][1] - this->value[3][0] * this->value[2][1];
value_type subFactor06 = this->value[1][2] * this->value[3][3] - this->value[3][2] * this->value[1][3];
value_type subFactor07 = this->value[1][1] * this->value[3][3] - this->value[3][1] * this->value[1][3];
value_type subFactor08 = this->value[1][1] * this->value[3][2] - this->value[3][1] * this->value[1][2];
value_type subFactor09 = this->value[1][0] * this->value[3][3] - this->value[3][0] * this->value[1][3];
value_type subFactor10 = this->value[1][0] * this->value[3][2] - this->value[3][0] * this->value[1][2];
value_type subFactor11 = this->value[1][1] * this->value[3][3] - this->value[3][1] * this->value[1][3];
value_type SubFactor12 = this->value[1][0] * this->value[3][1] - this->value[3][0] * this->value[1][1];
value_type subFactor13 = this->value[1][2] * this->value[2][3] - this->value[2][2] * this->value[1][3];
value_type subFactor14 = this->value[1][1] * this->value[2][3] - this->value[2][1] * this->value[1][3];
value_type subFactor15 = this->value[1][1] * this->value[2][2] - this->value[2][1] * this->value[1][2];
value_type subFactor16 = this->value[1][0] * this->value[2][3] - this->value[2][0] * this->value[1][3];
value_type subFactor17 = this->value[1][0] * this->value[2][2] - this->value[2][0] * this->value[1][2];
value_type subFactor18 = this->value[1][0] * this->value[2][1] - this->value[2][0] * this->value[1][1];
tmat4x4<T> res(
+ this->value[1][1] * subFactor00 - this->value[1][2] * subFactor01 + this->value[1][3] * subFactor02,
- this->value[1][0] * subFactor00 + this->value[1][2] * subFactor03 - this->value[1][3] * subFactor04,
+ this->value[1][0] * subFactor01 - this->value[1][1] * subFactor03 + this->value[1][3] * subFactor05,
- this->value[1][0] * subFactor02 + this->value[1][1] * subFactor04 - this->value[1][2] * subFactor05,
- this->value[0][1] * subFactor00 + this->value[0][2] * subFactor01 - this->value[0][3] * subFactor02,
+ this->value[0][0] * subFactor00 - this->value[0][2] * subFactor03 + this->value[0][3] * subFactor04,
- this->value[0][0] * subFactor01 + this->value[0][1] * subFactor03 - this->value[0][3] * subFactor05,
+ this->value[0][0] * subFactor02 - this->value[0][1] * subFactor04 + this->value[0][2] * subFactor05,
+ this->value[0][1] * subFactor06 - this->value[0][2] * subFactor07 + this->value[0][3] * subFactor08,
- this->value[0][0] * subFactor06 + this->value[0][2] * subFactor09 - this->value[0][3] * subFactor10,
+ this->value[0][0] * subFactor11 - this->value[0][1] * subFactor09 + this->value[0][3] * SubFactor12,
- this->value[0][0] * subFactor08 + this->value[0][1] * subFactor10 - this->value[0][2] * SubFactor12,
- this->value[0][1] * subFactor13 + this->value[0][2] * subFactor14 - this->value[0][3] * subFactor15,
+ this->value[0][0] * subFactor13 - this->value[0][2] * subFactor16 + this->value[0][3] * subFactor17,
- this->value[0][0] * subFactor14 + this->value[0][1] * subFactor16 - this->value[0][3] * subFactor18,
+ this->value[0][0] * subFactor15 - this->value[0][1] * subFactor17 + this->value[0][2] * subFactor18);
value_type determinant =
+ this->value[0][0] * res[0][0]
+ this->value[0][1] * res[1][0]
+ this->value[0][2] * res[2][0]
+ this->value[0][3] * res[3][0];
res /= determinant;
return res;
}
private:
col_type value[4];
public:
size_type length() const
{
return 4;
}
size_type col_size()
{
return 4;
}
size_type row_size()
{
return 4;
}
void identify()
{
this->value[0] = col_type(1, 0, 0, 0);
this->value[1] = col_type(0, 1, 0, 0);
this->value[2] = col_type(0, 0, 1, 0);
this->value[3] = col_type(0, 0, 0, 1);
}
col_type & operator[](size_type i)
{
assert(i < this->length());
return this->value[i];
}
col_type const & operator[](size_type i) const
{
assert(i < this->length());
return this->value[i];
}
tmat4x4(tmat4x4<T> const & m)
{
this->value[0] = m.value[0];
this->value[1] = m.value[1];
this->value[2] = m.value[2];
this->value[3] = m.value[3];
}
tmat4x4(tmat3x3<T> const & m)
{
this->value[0] = col_type(m[0], value_type(0));
this->value[1] = col_type(m[1], value_type(0));
this->value[2] = col_type(m[2], value_type(0));
this->value[3] = col_type(value_type(0), value_type(0), value_type(0), value_type(1));
}
tmat4x4()
{
}
tmat4x4(value_type s)
{
value_type const Zero(0);
this->value[0] = col_type(s, Zero, Zero, Zero);
this->value[1] = col_type(Zero, s, Zero, Zero);
this->value[2] = col_type(Zero, Zero, s, Zero);
this->value[3] = col_type(Zero, Zero, Zero, s);
}
tmat4x4
(
value_type const & x0, value_type const & y0, value_type const & z0, value_type const & w0,
value_type const & x1, value_type const & y1, value_type const & z1, value_type const & w1,
value_type const & x2, value_type const & y2, value_type const & z2, value_type const & w2,
value_type const & x3, value_type const & y3, value_type const & z3, value_type const & w3
)
{
this->value[0] = col_type(x0, y0, z0, w0);
this->value[1] = col_type(x1, y1, z1, w1);
this->value[2] = col_type(x2, y2, z2, w2);
this->value[3] = col_type(x3, y3, z3, w3);
}
tmat4x4
(
col_type const & v0,
col_type const & v1,
col_type const & v2,
col_type const & v3
)
{
this->value[0] = v0;
this->value[1] = v1;
this->value[2] = v2;
this->value[3] = v3;
}
template <typename U>
tmat4x4(tmat4x4<U> const & m)
{
this->value[0] = col_type(m[0]);
this->value[1] = col_type(m[1]);
this->value[2] = col_type(m[2]);
this->value[3] = col_type(m[3]);
}
template <typename U>
tmat4x4(U const & s)
{
value_type const Zero(0);
this->value[0] = tvec4<T>(value_type(s), Zero, Zero, Zero);
this->value[1] = tvec4<T>(Zero, value_type(s), Zero, Zero);
this->value[2] = tvec4<T>(Zero, Zero, value_type(s), Zero);
this->value[3] = tvec4<T>(Zero, Zero, Zero, value_type(s));
}
template <
typename X1, typename Y1, typename Z1, typename W1,
typename X2, typename Y2, typename Z2, typename W2,
typename X3, typename Y3, typename Z3, typename W3,
typename X4, typename Y4, typename Z4, typename W4>
tmat4x4
(
X1 const & x1, Y1 const & y1, Z1 const & z1, W1 const & w1,
X2 const & x2, Y2 const & y2, Z2 const & z2, W2 const & w2,
X3 const & x3, Y3 const & y3, Z3 const & z3, W3 const & w3,
X4 const & x4, Y4 const & y4, Z4 const & z4, W4 const & w4
)
{
this->value[0] = col_type(value_type(x1), value_type(y1), value_type(z1), value_type(w1));
this->value[1] = col_type(value_type(x2), value_type(y2), value_type(z2), value_type(w2));
this->value[2] = col_type(value_type(x3), value_type(y3), value_type(z3), value_type(w3));
this->value[3] = col_type(value_type(x4), value_type(y4), value_type(z4), value_type(w4));
}
template <typename V1, typename V2, typename V3, typename V4>
tmat4x4
(
tvec4<V1> const & v1,
tvec4<V2> const & v2,
tvec4<V3> const & v3,
tvec4<V4> const & v4
)
{
this->value[0] = col_type(v1);
this->value[1] = col_type(v2);
this->value[2] = col_type(v3);
this->value[3] = col_type(v4);
}
T const * data() const
{
return &this->value[0][0];
}
tmat4x4<T>& operator= (tmat4x4<T> const & m)
{
this->value[0] = m[0];
this->value[1] = m[1];
this->value[2] = m[2];
this->value[3] = m[3];
return *this;
}
template <typename U>
tmat4x4<T>& operator= (tmat4x4<U> const & m)
{
this->value[0] = m[0];
this->value[1] = m[1];
this->value[2] = m[2];
this->value[3] = m[3];
return *this;
}
template <typename U>
tmat4x4<T>& operator+= (U const & s)
{
this->value[0] += s;
this->value[1] += s;
this->value[2] += s;
this->value[3] += s;
return *this;
}
template <typename U>
tmat4x4<T>& operator+= (tmat4x4<U> const & m)
{
this->value[0] += m[0];
this->value[1] += m[1];
this->value[2] += m[2];
this->value[3] += m[3];
return *this;
}
template <typename U>
tmat4x4<T> & operator-= (U const & s)
{
this->value[0] -= s;
this->value[1] -= s;
this->value[2] -= s;
this->value[3] -= s;
return *this;
}
template <typename U>
tmat4x4<T> & operator-= (tmat4x4<U> const & m)
{
this->value[0] -= m[0];
this->value[1] -= m[1];
this->value[2] -= m[2];
this->value[3] -= m[3];
return *this;
}
template <typename U>
tmat4x4<T> & operator*= (U const & s)
{
this->value[0] *= s;
this->value[1] *= s;
this->value[2] *= s;
this->value[3] *= s;
return *this;
}
template <typename U>
tmat4x4<T> & operator*= (tmat4x4<U> const & m)
{
return (*this = *this * m);
}
template <typename U>
tmat4x4<T> & operator/= (U const & s)
{
this->value[0] /= s;
this->value[1] /= s;
this->value[2] /= s;
this->value[3] /= s;
return *this;
}
template <typename U>
tmat4x4<T> & operator/= (tmat4x4<U> const & m)
{
return (*this = *this / m);
}
tmat4x4<T> & operator++ ()
{
++this->value[0];
++this->value[1];
++this->value[2];
++this->value[3];
return *this;
}
tmat4x4<T> & operator-- ()
{
--this->value[0];
--this->value[1];
--this->value[2];
--this->value[3];
return *this;
}
tmat4x4<T>& translate( value_type x,value_type y,value_type z)
{
this->value[0] = col_type(1, 0, 0, 0);
this->value[1] = col_type(0, 1, 0, 0);
this->value[2] = col_type(0, 0, 1, 0);
this->value[3] = col_type(x, y, z, 1);
return *this;
}
template<typename U>
tmat4x4<T>& translate( U x,U y,U z)
{
this->value[0] = col_type(1, 0, 0, 0);
this->value[1] = col_type(0, 1, 0, 0);
this->value[2] = col_type(0, 0, 1, 0);
this->value[3] = col_type(T(x), T(y), T(z), 1);
return *this;
}
tmat4x4<T>& translate(tvec3<T> const& pos)
{
this->value[0] = col_type(1, 0, 0, 0);
this->value[1] = col_type(0, 1, 0, 0);
this->value[2] = col_type(0, 0, 1, 0);
this->value[3] = col_type(pos.x,pos.y, pos.z, 1);
return *this;
}
template<typename U>
tmat4x4<T>& translate(tvec3<U> const& pos)
{
this->value[0] = col_type(1, 0, 0, 0);
this->value[1] = col_type(0, 1, 0, 0);
this->value[2] = col_type(0, 0, 1, 0);
this->value[3] = col_type(T(pos.x),T(pos.y), T(pos.z), 1);
return *this;
}
tmat4x4<T>& rotate(value_type angle,tvec3<T> const & v )
{
T a = DEG2RAD(angle);
T c = cos(a);
T s = sin(a);
tvec3<T> axis = normalize(v);
tvec3<T> temp = (T(1) - c) * axis;
tmat4x4<T> res;
this->value[0][0] = c + temp[0] * axis[0];
this->value[0][1] = 0 + temp[0] * axis[1] + s * axis[2];
this->value[0][2] = 0 + temp[0] * axis[2] - s * axis[1];
this->value[0][3] = 0;
this->value[1][0] = 0 + temp[1] * axis[0] - s * axis[2];
this->value[1][1] = c + temp[1] * axis[1];
this->value[1][2] = 0 + temp[1] * axis[2] + s * axis[0];
this->value[1][3] = 0;
this->value[2][0] = 0 + temp[2] * axis[0] + s * axis[1];
this->value[2][1] = 0 + temp[2] * axis[1] - s * axis[0];
this->value[2][2] = c + temp[2] * axis[2];
this->value[2][3] = 0;
this->value[3][0] = 0;
this->value[3][1] = 0;
this->value[3][2] = 0;
this->value[3][3] = 1;
return *this;
}
tmat4x4<T>& rotateX(value_type angle)
{
T a = DEG2RAD(angle);
T c = cos(a);
T s = sin(a);
this->value[0] = col_type(1, 0, 0, 0);
this->value[1] = col_type(0, c, s, 0);
this->value[2] = col_type(0,-s, c, 0);
this->value[3] = col_type(0, 0, 0, 1);
return *this;
}
template<typename U>
tmat4x4<T>& rotateX(U angle)
{
T a = DEG2RAD(angle);
T c = cos(a);
T s = sin(a);
this->value[0] = col_type(1, 0, 0, 0);
this->value[1] = col_type(0, c, s, 0);
this->value[2] = col_type(0,-s, c, 0);
this->value[3] = col_type(0, 0, 0, 1);
return *this;
}
tmat4x4<T>& rotateY(value_type angle)
{
T a = DEG2RAD(angle);
T c = cos(a);
T s = sin(a);
this->value[0] = col_type(c, 0,-s, 0);
this->value[1] = col_type(0, 1, 0, 0);
this->value[2] = col_type(s, 0, c, 0);
this->value[3] = col_type(0, 0, 0, 1);
return *this;
}
template<typename U>
tmat4x4<T>& rotateY(U angle)
{
T a = DEG2RAD(angle);
T c = cos(a);
T s = sin(a);
this->value[0] = col_type(c, 0,-s, 0);
this->value[1] = col_type(0, 1, 0, 0);
this->value[2] = col_type(s, 0, c, 0);
this->value[3] = col_type(0, 0, 0, 1);
return *this;
}
tmat4x4<T>& rotateZ(value_type angle)
{
T a = T(DEG2RAD(angle));
T c = cos(a);
T s = sin(a);
this->value[0] = col_type( c, s, 0, 0);
this->value[1] = col_type(-s, c, 0, 0);
this->value[2] = col_type( 0, 0, 1, 0);
this->value[3] = col_type( 0, 0, 0, 1);
return *this;
}
template<typename U>
tmat4x4<T>& rotateZ(U angle)
{
T a = DEG2RAD(angle);
T c = cos(a);
T s = sin(a);
this->value[0] = col_type( c, s, 0, 0);
this->value[1] = col_type(-s, c, 0, 0);
this->value[2] = col_type( 0, 0, 1, 0);
this->value[3] = col_type( 0, 0, 0, 1);
return *this;
}
tmat4x4<T> rotateXY(T angleX, T angleY)
{
T cosX = cos(DEG2RAD(angleX));
T sinX = sin(DEG2RAD(angleX));
T cosY = cos(DEG2RAD(angleY));
T sinY = sin(DEG2RAD(angleY));
this->value[0] = col_type( cosY, -sinX * sinY, cosX * sinY,0);
this->value[1] = col_type( 0, cosX, sinX, 0);
this->value[2] = col_type( -sinY , -sinX * cosY, cosX * cosY,0);
this->value[3] = col_type( 0, 0, 0, 1);
return *this;
}
tmat4x4<T> rotateYX(T angleX, T angleY)
{
T cosX = cos(DEG2RAD(angleX));
T sinX = sin(DEG2RAD(angleX));
T cosY = cos(DEG2RAD(angleY));
T sinY = sin(DEG2RAD(angleY));
this->value[0] = col_type( cosY, 0, sinY, 0);
this->value[1] = col_type( -sinX * sinY,cosX, sinX * cosY,0);
this->value[2] = col_type( -cosX * sinY,-sinX, cosX * cosY,0);
this->value[3] = col_type( 0, 0, 0, 1);
return *this;
}
tmat4x4<T> rotateYXZ( T yaw, T pitch, T roll)
{
T tmp_ch = cos(DEG2RAD(yaw));
T tmp_sh = sin(DEG2RAD(yaw));
T tmp_cp = cos(DEG2RAD(pitch));
T tmp_sp = sin(DEG2RAD(pitch));
T tmp_cb = cos(DEG2RAD(roll));
T tmp_sb = sin(DEG2RAD(roll));
tmat4x4<T> Result;
this->value[0][0] = tmp_ch * tmp_cb + tmp_sh * tmp_sp * tmp_sb;
this->value[0][1] = tmp_sb * tmp_cp;
this->value[0][2] = -tmp_sh * tmp_cb + tmp_ch * tmp_sp * tmp_sb;
this->value[0][3] = T(0);
this->value[1][0] = -tmp_ch * tmp_sb + tmp_sh * tmp_sp * tmp_cb;
this->value[1][1] = tmp_cb * tmp_cp;
this->value[1][2] = tmp_sb * tmp_sh + tmp_ch * tmp_sp * tmp_cb;
this->value[1][3] = T(0);
this->value[2][0] = tmp_sh * tmp_cp;
this->value[2][1] = -tmp_sp;
this->value[2][2] = tmp_ch * tmp_cp;
this->value[2][3] = T(0);
this->value[3][0] = T(0);
this->value[3][1] = T(0);
this->value[3][2] = T(0);
this->value[3][3] = T(1);
return *this;
}
tmat4x4<T> yawPitchRoll( T yaw, T pitch, T roll)
{
T tmp_ch = cos(DEG2RAD(yaw));
T tmp_sh = sin(DEG2RAD(yaw));
T tmp_cp = cos(DEG2RAD(pitch));
T tmp_sp = sin(DEG2RAD(pitch));
T tmp_cb = cos(DEG2RAD(roll));
T tmp_sb = sin(DEG2RAD(roll));
this->value[0][0] = tmp_ch * tmp_cb + tmp_sh * tmp_sp * tmp_sb;
this->value[0][1] = tmp_sb * tmp_cp;
this->value[0][2] = -tmp_sh * tmp_cb + tmp_ch * tmp_sp * tmp_sb;
this->value[0][3] = T(0);
this->value[1][0] = -tmp_ch * tmp_sb + tmp_sh * tmp_sp * tmp_cb;
this->value[1][1] = tmp_cb * tmp_cp;
this->value[1][2] = tmp_sb * tmp_sh + tmp_ch * tmp_sp * tmp_cb;
this->value[1][3] = T(0);
this->value[2][0] = tmp_sh * tmp_cp;
this->value[2][1] = -tmp_sp;
this->value[2][2] = tmp_ch * tmp_cp;
this->value[2][3] = T(0);
this->value[3][0] = T(0);
this->value[3][1] = T(0);
this->value[3][2] = T(0);
this->value[3][3] = T(1);
return *this;
}
tmat4x4<T>& scale(tvec3<T> const& s)
{
this->value[0] = col_type(s[0], 0, 0, 0);
this->value[1] = col_type(0, s[1], 0, 0);
this->value[2] = col_type(0, 0, s[2], 0);
this->value[3] = col_type(0, 0, 0, 1);
return *this;
}
tmat4x4<T>& scale(value_type x,value_type y,value_type z)
{
this->value[0] = col_type(x, 0, 0, 0);
this->value[1] = col_type(0, y, 0, 0);
this->value[2] = col_type(0, 0, z, 0);
this->value[3] = col_type(0, 0, 0, 1);
return *this;
}
template<typename U>
tmat4x4<T>& scale(U x,U y,U z)
{
this->value[0] = col_type(value_type(x), 0, 0, 0);
this->value[1] = col_type(0, value_type(y), 0, 0);
this->value[2] = col_type(0, 0, value_type(z), 0);
this->value[3] = col_type(0, 0, 0, 1);
return *this;
}
template<typename U,typename V,typename W>
tmat4x4<T>& scale(U x,V y,W z)
{
this->value[0] = col_type(value_type(x), 0, 0, 0);
this->value[1] = col_type(0, value_type(y), 0, 0);
this->value[2] = col_type(0, 0, value_type(z), 0);
this->value[3] = col_type(0, 0, 0, 1);
return *this;
}
tmat4x4<T> transpose( ) const
{
return tmat4x4<T>(
this->value[0][0], this->value[1][0], this->value[2][0], this->value[3][0],
this->value[0][1], this->value[1][1], this->value[2][1], this->value[3][1],
this->value[0][2], this->value[1][2], this->value[2][2], this->value[3][2],
this->value[0][3], this->value[1][3], this->value[2][3], this->value[3][3]
);
}
tmat4x4<T> extractMatrixRotation() const
{
return tmat4x4<T>(
this->value[0][0], this->value[0][1], this->value[0][2], 0.0,
this->value[1][0], this->value[1][1], this->value[1][2], 0.0,
this->value[2][0], this->value[2][1], this->value[2][2], 0.0,
0.0, 0.0, 0.0, 1.0
);
}
};
template <typename T>
tmat4x4<T> rotateX(T angleX)
{
T cosX = cos(DEG2RAD(angleX));
T sinX = sin(DEG2RAD(angleX));
return tmat4x4<T>(
T(1), T(0), T(0), T(0),
T(0), cosX, sinX, T(0),
T(0),-sinX, cosX, T(0),
T(0), T(0), T(0), T(1));
}
template <typename T>
tmat4x4<T> rotateY(T angleY)
{
T cosY = cos(DEG2RAD(angleY));
T sinY = sin(DEG2RAD(angleY));
return tmat4x4<T>(
cosY, T(0), sinY, T(0),
T(0), T(1), T(0), T(0),
-sinY, T(0), cosY, T(0),
T(0), T(0), T(0), T(1));
}
template <typename T>
tmat4x4<T> rotateZ(T angleZ)
{
T cosZ = (T)cos(DEG2RAD(angleZ));
T sinZ = (T)sin(DEG2RAD(angleZ));
return tmat4x4<T>(
cosZ, sinZ, T(0), T(0),
-sinZ, cosZ, T(0), T(0),
T(0), T(0), T(1), T(0),
T(0), T(0), T(0), T(1));
}
template <typename T>
tmat4x4<T> rotateXY(T angleX, T angleY)
{
T cosX = cos(DEG2RAD(angleX));
T sinX = sin(DEG2RAD(angleX));
T cosY = cos(DEG2RAD(angleY));
T sinY = sin(DEG2RAD(angleY));
return tmat4x4<T>(
cosY, -sinX * sinY, cosX * sinY, T(0),
T(0), cosX, sinX, T(0),
-sinY, -sinX * cosY, cosX * cosY, T(0),
T(0), T(0), T(0), T(1));
}
template <typename T>
tmat4x4<T> rotateYX(T angleY, T angleX)
{
T cosX = cos(DEG2RAD(angleX));
T sinX = sin(DEG2RAD(angleX));
T cosY = cos(DEG2RAD(angleY));
T sinY = sin(DEG2RAD(angleY));
return tmat4x4<T>(
cosY, T(0), sinY, T(0),
-sinX * sinY, cosX, sinX * cosY, T(0),
-cosX * sinY, -sinX, cosX * cosY, T(0),
T(0), T(0), T(0), T(1));
}
template <typename T>
tmat4x4<T> rotateXZ(T angleX, T angleZ)
{
return rotateX(angleX) * rotateZ(angleZ);
}
template <typename T>
tmat4x4<T> rotateZX(T angleX, T angleZ)
{
return rotateZ(angleZ) * rotateX(angleX);
}
template <typename T>
tmat4x4<T> rotateYXZ(T yaw, T pitch, T roll)
{
T tmp_ch = cos(DEG2RAD(yaw));
T tmp_sh = sin(DEG2RAD(yaw));
T tmp_cp = cos(DEG2RAD(pitch));
T tmp_sp = sin(DEG2RAD(pitch));
T tmp_cb = cos(DEG2RAD(roll));
T tmp_sb = sin(DEG2RAD(roll));
tmat4x4<T> res;
res[0][0] = tmp_ch * tmp_cb + tmp_sh * tmp_sp * tmp_sb;
res[0][1] = tmp_sb * tmp_cp;
res[0][2] = -tmp_sh * tmp_cb + tmp_ch * tmp_sp * tmp_sb;
res[0][3] = T(0);
res[1][0] = -tmp_ch * tmp_sb + tmp_sh * tmp_sp * tmp_cb;
res[1][1] = tmp_cb * tmp_cp;
res[1][2] = tmp_sb * tmp_sh + tmp_ch * tmp_sp * tmp_cb;
res[1][3] = T(0);
res[2][0] = tmp_sh * tmp_cp;
res[2][1] = -tmp_sp;
res[2][2] = tmp_ch * tmp_cp;
res[2][3] = T(0);
res[3][0] = T(0);
res[3][1] = T(0);
res[3][2] = T(0);
res[3][3] = T(1);
return res;
}
template <typename T>
tmat4x4<T> yawPitchRoll(T yaw, T pitch, T roll)
{
T tmp_ch = cos(DEG2RAD(yaw));
T tmp_sh = sin(DEG2RAD(yaw));
T tmp_cp = cos(DEG2RAD(pitch));
T tmp_sp = sin(DEG2RAD(pitch));
T tmp_cb = cos(DEG2RAD(roll));
T tmp_sb = sin(DEG2RAD(roll));
tmat4x4<T> res;
res[0][0] = tmp_ch * tmp_cb + tmp_sh * tmp_sp * tmp_sb;
res[0][1] = tmp_sb * tmp_cp;
res[0][2] = -tmp_sh * tmp_cb + tmp_ch * tmp_sp * tmp_sb;
res[0][3] = T(0);
res[1][0] = -tmp_ch * tmp_sb + tmp_sh * tmp_sp * tmp_cb;
res[1][1] = tmp_cb * tmp_cp;
res[1][2] = tmp_sb * tmp_sh + tmp_ch * tmp_sp * tmp_cb;
res[1][3] = T(0);
res[2][0] = tmp_sh * tmp_cp;
res[2][1] = -tmp_sp;
res[2][2] = tmp_ch * tmp_cp;
res[2][3] = T(0);
res[3][0] = T(0);
res[3][1] = T(0);
res[3][2] = T(0);
res[3][3] = T(1);
return res;
}
template <typename T>
void axisAngle
(
tmat4x4<T> const & mat,
tvec3<T> & axis,
T & angle
)
{
T epsilon = (T)0.01;
T epsilon2 = (T)0.1;
if ((fabs(mat[1][0] - mat[0][1]) < epsilon) &&
(fabs(mat[2][0] - mat[0][2]) < epsilon) &&
(fabs(mat[2][1] - mat[1][2]) < epsilon))
{
if ((fabs(mat[1][0] + mat[0][1]) < epsilon2) &&
(fabs(mat[2][0] + mat[0][2]) < epsilon2) &&
(fabs(mat[2][1] + mat[1][2]) < epsilon2) &&
(fabs(mat[0][0] + mat[1][1] + mat[2][2] - (T)3.0) < epsilon2))
{
angle = (T)0.0;
axis.x = (T)1.0;
axis.y = (T)0.0;
axis.z = (T)0.0;
return;
}
angle = T(3.1415926535897932384626433832795);
T xx = (mat[0][0] + (T)1.0) / (T)2.0;
T yy = (mat[1][1] + (T)1.0) / (T)2.0;
T zz = (mat[2][2] + (T)1.0) / (T)2.0;
T xy = (mat[1][0] + mat[0][1]) / (T)4.0;
T xz = (mat[2][0] + mat[0][2]) / (T)4.0;
T yz = (mat[2][1] + mat[1][2]) / (T)4.0;
if ((xx > yy) && (xx > zz))
{
if (xx < epsilon)
{
axis.x = (T)0.0;
axis.y = (T)0.7071;
axis.z = (T)0.7071;
}
else
{
axis.x = sqrt(xx);
axis.y = xy / axis.x;
axis.z = xz / axis.x;
}
}
else if (yy > zz)
{
if (yy < epsilon)
{
axis.x = (T)0.7071;
axis.y = (T)0.0;
axis.z = (T)0.7071;
}
else
{
axis.y = sqrt(yy);
axis.x = xy / axis.y;
axis.z = yz / axis.y;
}
}
else
{
if (zz < epsilon)
{
axis.x = (T)0.7071;
axis.y = (T)0.7071;
axis.z = (T)0.0;
}
else
{
axis.z = sqrt(zz);
axis.x = xz / axis.z;
axis.y = yz / axis.z;
}
}
return;
}
T s = sqrt((mat[2][1] - mat[1][2]) * (mat[2][1] - mat[1][2]) + (mat[2][0] - mat[0][2]) * (mat[2][0] - mat[0][2]) + (mat[1][0] - mat[0][1]) * (mat[1][0] - mat[0][1]));
if (abs(s) < T(0.001))
s = (T)1.0;
angle = acos((mat[0][0] + mat[1][1] + mat[2][2] - (T)1.0) / (T)2.0);
axis.x = (mat[1][2] - mat[2][1]) / s;
axis.y = (mat[2][0] - mat[0][2]) / s;
axis.z = (mat[0][1] - mat[1][0]) / s;
}
template <typename T>
tmat4x4<T> axisAngleMatrix(tvec3<T> const & axis,T const angle)
{
T c = cos(angle);
T s = sin(angle);
T t = T(1) - c;
tvec3<T> n = normalize(axis);
return tmat4x4<T>(
t * n.x * n.x + c, t * n.x * n.y + n.z * s, t * n.x * n.z - n.y * s, T(0),
t * n.x * n.y - n.z * s, t * n.y * n.y + c, t * n.y * n.z + n.x * s, T(0),
t * n.x * n.z + n.y * s, t * n.y * n.z - n.x * s, t * n.z * n.z + c, T(0),
T(0), T(0), T(0), T(1)
);
}
template <typename T>
tmat4x4<T> interpolate
(
tmat4x4<T> const & m1,
tmat4x4<T> const & m2,
T const delta
)
{
tmat4x4<T> m1rot = m1.extractMatrixRotation();
tmat4x4<T> dltRotation = m2 * m1rot.transpose();
tvec3<T> dltAxis;
T dltAngle;
axisAngle(dltRotation, dltAxis, dltAngle);
tmat4x4<T> out = axisAngleMatrix(dltAxis, dltAngle * delta) * m1rot;
out[3][0] = m1[3][0] + delta * (m2[3][0] - m1[3][0]);
out[3][1] = m1[3][1] + delta * (m2[3][1] - m1[3][1]);
out[3][2] = m1[3][2] + delta * (m2[3][2] - m1[3][2]);
return out;
}
template <typename T>
tvec3<T> operator * (tvec3<T> const& v, tmat4x4<T> const& mat)
{
return tvec3<T>
(
v.x*mat[0][0] + v.y*mat[1][0] + v.z*mat[2][0] + 1*mat[3][0],
v.x*mat[0][1] + v.y*mat[1][1] + v.z*mat[2][1] + 1*mat[3][1],
v.x*mat[0][2] + v.y*mat[1][2] + v.z*mat[2][2] + 1*mat[3][2]
);
}
template <typename T>
tmat4x4<T> operator+ (tmat4x4<T> const & m, typename tmat4x4<T>::value_type s)
{
return tmat4x4<T>(
m[0] + s,
m[1] + s,
m[2] + s,
m[3] + s);
}
template <typename T>
tmat4x4<T> operator+ (typename tmat4x4<T>::value_type s, tmat4x4<T> const & m)
{
return tmat4x4<T>(
m[0] + s,
m[1] + s,
m[2] + s,
m[3] + s);
}
template <typename T>
tmat4x4<T> operator+ (tmat4x4<T> const & m1, tmat4x4<T> const & m2)
{
return tmat4x4<T>(
m1[0] + m2[0],
m1[1] + m2[1],
m1[2] + m2[2],
m1[3] + m2[3]);
}
template <typename T>
tmat4x4<T> operator- (tmat4x4<T> const & m, typename tmat4x4<T>::value_type s)
{
return tmat4x4<T>(
m[0] - s,
m[1] - s,
m[2] - s,
m[3] - s);
}
template <typename T>
tmat4x4<T> operator- (typename tmat4x4<T>::value_type s, tmat4x4<T> const & m)
{
return tmat4x4<T>(
s - m[0],
s - m[1],
s - m[2],
s - m[3]);
}
template <typename T>
tmat4x4<T> operator- (tmat4x4<T> const & m1, tmat4x4<T> const & m2)
{
return tmat4x4<T>(
m1[0] - m2[0],
m1[1] - m2[1],
m1[2] - m2[2],
m1[3] - m2[3]);
}
template <typename T>
tmat4x4<T> operator* (tmat4x4<T> const & m, typename tmat4x4<T>::value_type s)
{
return tmat4x4<T>(
m[0] * s,
m[1] * s,
m[2] * s,
m[3] * s);
}
template <typename T>
tmat4x4<T> operator* (typename tmat4x4<T>::value_type s, tmat4x4<T> const & m)
{
return tmat4x4<T>(
m[0] * s,
m[1] * s,
m[2] * s,
m[3] * s);
}
template <typename T>
typename tmat4x4<T>::col_type operator* (tmat4x4<T> const & m, typename tmat4x4<T>::row_type const & v)
{
return typename tmat4x4<T>::col_type(
m[0][0] * v.x + m[1][0] * v.y + m[2][0] * v.z + m[3][0] * v.w,
m[0][1] * v.x + m[1][1] * v.y + m[2][1] * v.z + m[3][1] * v.w,
m[0][2] * v.x + m[1][2] * v.y + m[2][2] * v.z + m[3][2] * v.w,
m[0][3] * v.x + m[1][3] * v.y + m[2][3] * v.z + m[3][3] * v.w);
}
template <typename T>
typename tmat4x4<T>::row_type operator* (typename tmat4x4<T>::col_type const & v, tmat4x4<T> const & m)
{
return typename tmat4x4<T>::row_type(
m[0][0] * v.x + m[0][1] * v.y + m[0][2] * v.z + m[0][3] * v.w,
m[1][0] * v.x + m[1][1] * v.y + m[1][2] * v.z + m[1][3] * v.w,
m[2][0] * v.x + m[2][1] * v.y + m[2][2] * v.z + m[2][3] * v.w,
m[3][0] * v.x + m[3][1] * v.y + m[3][2] * v.z + m[3][3] * v.w);
}
template <typename T>
tmat4x4<T> operator* (tmat4x4<T> const & m1, tmat4x4<T> const & m2)
{
typename tmat4x4<T>::col_type const srcA0 = m1[0];
typename tmat4x4<T>::col_type const srcA1 = m1[1];
typename tmat4x4<T>::col_type const srcA2 = m1[2];
typename tmat4x4<T>::col_type const srcA3 = m1[3];
typename tmat4x4<T>::col_type const srcB0 = m2[0];
typename tmat4x4<T>::col_type const srcB1 = m2[1];
typename tmat4x4<T>::col_type const srcB2 = m2[2];
typename tmat4x4<T>::col_type const srcB3 = m2[3];
tmat4x4<T> res;
res[0] = srcA0 * srcB0[0] + srcA1 * srcB0[1] + srcA2 * srcB0[2] + srcA3 * srcB0[3];
res[1] = srcA0 * srcB1[0] + srcA1 * srcB1[1] + srcA2 * srcB1[2] + srcA3 * srcB1[3];
res[2] = srcA0 * srcB2[0] + srcA1 * srcB2[1] + srcA2 * srcB2[2] + srcA3 * srcB2[3];
res[3] = srcA0 * srcB3[0] + srcA1 * srcB3[1] + srcA2 * srcB3[2] + srcA3 * srcB3[3];
return res;
}
template <typename T>
tmat4x4<T> operator/ (tmat4x4<T> const & m, typename tmat4x4<T>::value_type s)
{
return tmat4x4<T>(
m[0] / s,
m[1] / s,
m[2] / s,
m[3] / s);
}
template <typename T>
tmat4x4<T> operator/ (typename tmat4x4<T>::value_type s, tmat4x4<T> const & m)
{
return tmat4x4<T>(
s / m[0],
s / m[1],
s / m[2],
s / m[3]);
}
template <typename T>
typename tmat4x4<T>::col_type operator/ (tmat4x4<T> const & m, typename tmat4x4<T>::row_type const & v)
{
return m.inverse() * v;
}
template <typename T>
typename tmat4x4<T>::row_type operator/ (typename tmat4x4<T>::col_type const & v, tmat4x4<T> const & m)
{
return v * m.inverse();
}
template <typename T>
tmat4x4<T> operator/ (tmat4x4<T> const & m1, tmat4x4<T> const & m2)
{
return m1 * m2.inverse();
}
template <typename T>
tmat4x4<T> const operator- (tmat4x4<T> const & m)
{
return tmat4x4<T>(
-m[0],
-m[1],
-m[2],
-m[3]);
}
template <typename T>
tmat4x4<T> const operator++ (tmat4x4<T> const & m, int)
{
return tmat4x4<T>(
m[0] + T(1),
m[1] + T(1),
m[2] + T(1),
m[3] + T(1));
}
template <typename T>
tmat4x4<T> const operator-- (tmat4x4<T> const & m, int)
{
return tmat4x4<T>(
m[0] - T(1),
m[1] - T(1),
m[2] - T(1),
m[3] - T(1));
}
template <typename T>
bool operator==(tmat4x4<T> const & m1, tmat4x4<T> const & m2)
{
return (m1[0] == m2[0]) && (m1[1] == m2[1]) && (m1[2] == m2[2]) && (m1[3] == m2[3]);
}
template <typename T>
bool operator!=(tmat4x4<T> const & m1, tmat4x4<T> const & m2)
{
return (m1[0] != m2[0]) || (m1[1] != m2[1]) || (m1[2] != m2[2]) || (m1[3] != m2[3]);
}
//////////////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////////////
//! 射线类
//////////////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////////////////////////////////////////////
template <typename T>
typename tvec2<T>::value_type length(tvec2<T> const & v)
{
typename tvec2<T>::value_type sqr = v.x * v.x + v.y * v.y;
return sqrt(sqr);
}
template <typename T>
typename tvec3<T>::value_type length(tvec3<T> const & v)
{
typename tvec3<T>::value_type sqr = v.x * v.x + v.y * v.y + v.z * v.z;
return sqrt(sqr);
}
template <typename T>
typename tvec4<T>::value_type length(tvec4<T> const & v)
{
typename tvec4<T>::value_type sqr = v.x * v.x + v.y * v.y + v.z * v.z + v.w * v.w;
return sqrt(sqr);
}
template <typename T>
typename tvec2<T>::value_type distance(tvec2<T> const & p0,tvec2<T> const & p1)
{
return length(p1 - p0);
}
template <typename T>
typename tvec3<T>::value_type distance(tvec3<T> const & p0,tvec3<T> const & p1)
{
return length(p1 - p0);
}
template <typename T>
typename tvec4<T>::value_type distance(tvec4<T> const & p0,tvec4<T> const & p1)
{
return length(p1 - p0);
}
template <typename T>
typename tvec2<T>::value_type dot(tvec2<T> const & x, tvec2<T> const & y)
{
return x.x * y.x + x.y * y.y;
}
template <typename T>
typename tvec3<T>::value_type dot(tvec3<T> const & x, tvec3<T> const & y)
{
return x.x * y.x + x.y * y.y + x.z * y.z;
}
template <typename T>
typename tvec4<T>::value_type dot(tvec4<T> const & x, tvec4<T> const & y)
{
return x.x * y.x + x.y * y.y + x.z * y.z + x.w * y.w;
}
template <typename T>
tvec3<T> cross(tvec3<T> const & x, tvec3<T> const & y)
{
return tvec3<T>
(
x.y * y.z - y.y * x.z,
x.z * y.x - y.z * x.x,
x.x * y.y - y.x * x.y
);
}
template <typename T>
T inversesqrt(T x)
{
return T(1) / sqrt(x);
}
template <typename T>
tvec2<T> normalize(tvec2<T> const & x)
{
typename tvec2<T>::value_type sqr = x.x * x.x + x.y * x.y;
return x * inversesqrt(sqr);
}
template <typename T>
tvec3<T> normalize(tvec3<T> const & x)
{
typename tvec3<T>::value_type sqr = x.x * x.x + x.y * x.y + x.z * x.z;
return x * inversesqrt(sqr);
}
template <typename T>
tvec4<T> normalize(tvec4<T> const & x)
{
typename tvec4<T>::value_type sqr = x.x * x.x + x.y * x.y + x.z * x.z + x.w * x.w;
return x * inversesqrt(sqr);
}
//////////////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////////////
template <typename T>
struct tquat
{
typedef T value_type;
typedef std::size_t size_type;
public:
value_type x;
value_type y;
value_type z;
value_type w;
size_type length() const
{
return 4;
}
tquat():
x(0),
y(0),
z(0),
w(1)
{}
explicit tquat(value_type s, tvec3<T> const & v):
x(v.x),
y(v.y),
z(v.z),
w(s)
{
}
explicit tquat(tvec3<T> const & v,value_type s):
x(v.x),
y(v.y),
z(v.z),
w(s)
{
}
explicit tquat(value_type w, value_type x, value_type y, value_type z):
x(x),
y(y),
z(z),
w(w)
{}
explicit tquat(tvec3<T> const & eulerAngle)
{
tvec3<T> c = cos(eulerAngle * value_type(0.5));
tvec3<T> s = sin(eulerAngle * value_type(0.5));
this->w = c.x * c.y * c.z + s.x * s.y * s.z;
this->x = s.x * c.y * c.z - c.x * s.y * s.z;
this->y = c.x * s.y * c.z + s.x * c.y * s.z;
this->z = c.x * c.y * s.z - s.x * s.y * c.z;
}
explicit tquat(tmat3x3<T> const & m)
{
*this = quat_cast(m);
}
explicit tquat(tmat4x4<T> const & m)
{
*this = quat_cast(m);
}
value_type & operator[](int i)
{
return (&x)[i];
}
value_type const & operator[](int i) const
{
return (&x)[i];
}
tquat<T> & operator*=(value_type s)
{
this->w *= s;
this->x *= s;
this->y *= s;
this->z *= s;
return *this;
}
tquat<T> & operator = (const tquat<T>& right)
{
this->w = right.w;
this->x = right.x;
this->y = right.y;
this->z = right.z;
return *this;
}
tquat<T> & operator/=(value_type s)
{
this->w /= s;
this->x /= s;
this->y /= s;
this->z /= s;
return *this;
}
};
template< typename T>
tmat4x4<T> makeTransform( tvec3<T> const & position, tvec3<T> const& scale, const tquat<T>& orientation)
{
tmat3x3<T> rot3x3 = mat3_cast(orientation);
return tmat4x4<T>
(
scale.x * rot3x3[0][0], scale.x * rot3x3[0][1], scale.x * rot3x3[0][2], 0,
scale.y * rot3x3[1][0], scale.y * rot3x3[1][1], scale.y * rot3x3[1][2], 0,
scale.z * rot3x3[2][0], scale.z * rot3x3[2][1], scale.z * rot3x3[2][2], 0,
position.x, position.y, position.z, 1
);
}
template <typename T>
T dot(tquat<T> const & q1, tquat<T> const & q2)
{
return q1.x * q2.x + q1.y * q2.y + q1.z * q2.z + q1.w * q2.w;
}
template <typename T>
tquat<T> cross(tquat<T> const & q1, tquat<T> const & q2)
{
return tquat<T>(
q1.w * q2.w - q1.x * q2.x - q1.y * q2.y - q1.z * q2.z,
q1.w * q2.x + q1.x * q2.w + q1.y * q2.z - q1.z * q2.y,
q1.w * q2.y + q1.y * q2.w + q1.z * q2.x - q1.x * q2.z,
q1.w * q2.z + q1.z * q2.w + q1.x * q2.y - q1.y * q2.x);
}
template <typename T>
T length(tquat<T> const & q)
{
return sqrt(dot(q, q));
}
template <typename genType>
genType mix(genType x, genType y, genType a)
{
return x + a * (y - x);
}
template <typename T>
T epsilon()
{
return std::numeric_limits<T>::epsilon();
}
template <typename T>
tquat<T> conjugate(tquat<T> const & q)
{
return tquat<T>(q.w, -q.x, -q.y, -q.z);
}
template <typename T>
tquat<T> inverse(tquat<T> const & q)
{
return conjugate(q) / dot(q, q);
}
template <typename T>
bool operator==(tquat<T> const & q1, tquat<T> const & q2)
{
return (q1.x == q2.x) && (q1.y == q2.y) && (q1.z == q2.z) && (q1.w == q2.w);
}
template <typename T>
bool operator!=(tquat<T> const & q1, tquat<T> const & q2)
{
return (q1.x != q2.x) || (q1.y != q2.y) || (q1.z != q2.z) || (q1.w != q2.w);
}
template <typename T>
tquat<T> operator- (tquat<T> const & q)
{
return tquat<T>(-q.w, -q.x, -q.y, -q.z);
}
template <typename T>
tquat<T> operator+ ( tquat<T> const & q, tquat<T> const & p)
{
return tquat<T>(
q.w + p.w,
q.x + p.x,
q.y + p.y,
q.z + p.z
);
}
template <typename T>
tquat<T> operator* ( tquat<T> const & q, tquat<T> const & p)
{
return tquat<T>(
q.w * p.w - q.x * p.x - q.y * p.y - q.z * p.z,
q.w * p.x + q.x * p.w + q.y * p.z - q.z * p.y,
q.w * p.y + q.y * p.w + q.z * p.x - q.x * p.z,
q.w * p.z + q.z * p.w + q.x * p.y - q.y * p.x
);
}
template <typename T>
tvec3<T> operator* (tquat<T> const & q, tvec3<T> const & v)
{
typename tquat<T>::value_type two(2);
tvec3<T> uv;
tvec3<T> uuv;
tvec3<T> quatVector(q.x, q.y, q.z);
uv = cross(quatVector, v);
uuv = cross(quatVector, uv);
uv *= two * q.w;
uuv *= two;
return v + uv + uuv;
}
template <typename T>
tvec3<T> operator* (tvec3<T> const & v,tquat<T> const & q)
{
return inverse(q) * v;
}
template <typename T>
tquat<T> operator* (tquat<T> const & q, typename tquat<T>::value_type s)
{
return tquat<T>(q.w * s, q.x * s, q.y * s, q.z * s);
}
template <typename T>
tquat<T> operator* (typename tquat<T>::value_type s,tquat<T> const & q)
{
return q * s;
}
template <typename T>
tquat<T> operator/ (tquat<T> const & q, typename tquat<T>::value_type s)
{
return tquat<T>(q.w / s, q.x / s, q.y / s, q.z / s);
}
template <typename T>
tquat<T> mix(tquat<T> const & x, tquat<T> const & y, T const & a)
{
T cosTheta = dot(x, y);
if(cosTheta > T(1) - epsilon<T>())
{
return tquat<T>(
mix(x.w, y.w, a),
mix(x.x, y.x, a),
mix(x.y, y.y, a),
mix(x.z, y.z, a)
);
}
else
{
// Essential Mathematics, page 467
T angle = acos(cosTheta);
return (sin((T(1) - a) * angle) * x + sin(a * angle) * y) / sin(angle);
}
}
template <typename T>
tquat<T> lerp(tquat<T> const & x, tquat<T> const & y, T a)
{
assert(a >= T(0));
assert(a <= T(1));
return x * (T(1) - a) + (y * a);
}
template <typename T>
tquat<T> slerp(tquat<T> const & x, tquat<T> const & y, T a)
{
tquat<T> z = y;
T cosTheta = dot(x, y);
if (cosTheta < T(0))
{
z = -y;
cosTheta = -cosTheta;
}
if(cosTheta > T(1) - epsilon<T>())
{
return tquat<T>
(
mix(x.w, z.w, a),
mix(x.x, z.x, a),
mix(x.y, z.y, a),
mix(x.z, z.z, a)
);
}
else
{
// Essential Mathematics, page 467
T angle = acos(cosTheta);
return (sin((T(1) - a) * angle) * x + sin(a * angle) * z) / sin(angle);
}
}
template <typename T>
tquat<T> rotate
(
typename tquat<T>::value_type angle,
tvec3<T> const & axis
)
{
tvec3<T> Tmp = axis;
typename tquat<T>::value_type len = length(Tmp);
if(abs(len - T(1)) > T(0.001f))
{
T oneOverLen = T(1) / len;
Tmp.x *= oneOverLen;
Tmp.y *= oneOverLen;
Tmp.z *= oneOverLen;
}
typename tquat<T>::value_type const AngleRad = (T)DEG2RAD(angle);
typename tquat<T>::value_type const Sin = (T)sin(AngleRad * T(0.5));
return tquat<T>((T)cos(AngleRad * T(0.5)), Tmp.x * Sin, Tmp.y * Sin, Tmp.z * Sin);
}
template <typename valType>
valType roll(tquat<valType> const & q)
{
return atan2(valType(2) * (q.x * q.y + q.w * q.z), q.w * q.w + q.x * q.x - q.y * q.y - q.z * q.z) * valType(RAD2DEG);
}
template <typename valType>
valType pitch(tquat<valType> const & q)
{
return ::atan2(valType(2) * (q.y * q.z + q.w * q.x), q.w * q.w - q.x * q.x - q.y * q.y + q.z * q.z)* valType(RAD2DEG);
}
template <typename valType>
valType yaw(tquat<valType> const & q)
{
return ::asin(valType(-2) * (q.x * q.z - q.w * q.y)) * valType(RAD2DEG);
}
template <typename T>
tvec3<T> eulerAngles(tquat<T> const & x)
{
return tvec3<T>(pitch(x), yaw(x), roll(x));
}
template <typename T>
tmat3x3<T> mat3_cast(const tquat<T>& q)
{
return tmat3x3<T>
(
1 - 2 * q.y * q.y - 2 * q.z * q.z, 2 * q.x * q.y + 2 * q.w * q.z, 2 * q.x * q.z - 2 * q.w * q.y,
2 * q.x * q.y - 2 * q.w * q.z, 1 - 2 * q.x * q.x - 2 * q.z * q.z, 2 * q.y * q.z + 2 * q.w * q.x,
2 * q.x * q.z + 2 * q.w * q.y, 2 * q.y * q.z - 2 * q.w * q.x, 1 - 2 * q.x * q.x - 2 * q.y * q.y
);
}
template <typename T>
tmat4x4<T> mat4_cast(tquat<T> const & q)
{
return tmat4x4<T>(mat3_cast(q));
}
template <typename T>
tquat<T> quat_cast(tmat3x3<T> const & m)
{
typename tquat<T>::value_type fourXSquaredMinus1 = m[0][0] - m[1][1] - m[2][2];
typename tquat<T>::value_type fourYSquaredMinus1 = m[1][1] - m[0][0] - m[2][2];
typename tquat<T>::value_type fourZSquaredMinus1 = m[2][2] - m[0][0] - m[1][1];
typename tquat<T>::value_type fourWSquaredMinus1 = m[0][0] + m[1][1] + m[2][2];
int biggestIndex = 0;
typename tquat<T>::value_type fourBiggestSquaredMinus1 = fourWSquaredMinus1;
if(fourXSquaredMinus1 > fourBiggestSquaredMinus1)
{
fourBiggestSquaredMinus1 = fourXSquaredMinus1;
biggestIndex = 1;
}
if(fourYSquaredMinus1 > fourBiggestSquaredMinus1)
{
fourBiggestSquaredMinus1 = fourYSquaredMinus1;
biggestIndex = 2;
}
if(fourZSquaredMinus1 > fourBiggestSquaredMinus1)
{
fourBiggestSquaredMinus1 = fourZSquaredMinus1;
biggestIndex = 3;
}
typename tquat<T>::value_type biggestVal = sqrt(fourBiggestSquaredMinus1 + T(1)) * T(0.5);
typename tquat<T>::value_type mult = T(0.25) / biggestVal;
tquat<T> res;
switch(biggestIndex)
{
case 0:
res.w = biggestVal;
res.x = (m[1][2] - m[2][1]) * mult;
res.y = (m[2][0] - m[0][2]) * mult;
res.z = (m[0][1] - m[1][0]) * mult;
break;
case 1:
res.w = (m[1][2] - m[2][1]) * mult;
res.x = biggestVal;
res.y = (m[0][1] + m[1][0]) * mult;
res.z = (m[2][0] + m[0][2]) * mult;
break;
case 2:
res.w = (m[2][0] - m[0][2]) * mult;
res.x = (m[0][1] + m[1][0]) * mult;
res.y = biggestVal;
res.z = (m[1][2] + m[2][1]) * mult;
break;
case 3:
res.w = (m[0][1] - m[1][0]) * mult;
res.x = (m[2][0] + m[0][2]) * mult;
res.y = (m[1][2] + m[2][1]) * mult;
res.z = biggestVal;
break;
default:
assert(false);
break;
}
return res;
}
template <typename T>
tquat<T> quat_cast(tmat4x4<T> const & m4)
{
return quat_cast(tmat3x3<T>(m4[0][0],m4[0][1],m4[0][2],
m4[1][0],m4[1][1],m4[1][2],
m4[2][0],m4[2][1],m4[2][2]));
}
template <typename T>
T angle(tquat<T> const & x)
{
return acos(x.w) * T(2) * T(RAD2DEG);
}
template <typename T>
tvec3<T> axis(tquat<T> const & x)
{
T tmp1 = T(1) - x.w * x.w;
if(tmp1 <= T(0))
{
return tvec3<T>(0, 0, 1);
}
T tmp2 = T(1) / sqrt(tmp1);
return tvec3<T>(x.x * tmp2, x.y * tmp2, x.z * tmp2);
}
template <typename valType>
tquat<valType> angleAxis(valType angle, tvec3<valType> const & axis)
{
tquat<valType> result;
valType a = (valType)(valType(DEG2RAD(angle)));
valType s = sin(a * valType(0.5));
result.w = cos(a * valType(0.5));
result.x = axis.x * s;
result.y = axis.y * s;
result.z = axis.z * s;
return result;
}
////////////////////////////////////////////////////////////////////////////////////////////////////////
////////////////////////////////////////////////////////////////////////////////////////////////////////
////////////////////////////////////////////////////////////////////////////////////////////////////////
template <typename T>
tmat4x4<T> translate(tmat4x4<T> const & m,tvec3<T> const & v)
{
tmat4x4<T> res(m);
res[3] = m[0] * v[0] + m[1] * v[1] + m[2] * v[2] + m[3];
return res;
}
template <typename T>
tmat4x4<T> rotate
(
tmat4x4<T> const & m,
T const & angle,
tvec3<T> const & v
)
{
T a = DEG2RAD(angle);
T c = cos(a);
T s = sin(a);
tvec3<T> axis = normalize(v);
tvec3<T> temp = (T(1) - c) * axis;
tmat4x4<T> res;
res[0][0] = c + temp[0] * axis[0];
res[0][1] = 0 + temp[0] * axis[1] + s * axis[2];
res[0][2] = 0 + temp[0] * axis[2] - s * axis[1];
res[1][0] = 0 + temp[1] * axis[0] - s * axis[2];
res[1][1] = c + temp[1] * axis[1];
res[1][2] = 0 + temp[1] * axis[2] + s * axis[0];
res[2][0] = 0 + temp[2] * axis[0] + s * axis[1];
res[2][1] = 0 + temp[2] * axis[1] - s * axis[0];
res[2][2] = c + temp[2] * axis[2];
tmat4x4<T> rot;
rot[0] = m[0] * res[0][0] + m[1] * res[0][1] + m[2] * res[0][2];
rot[1] = m[0] * res[1][0] + m[1] * res[1][1] + m[2] * res[1][2];
rot[2] = m[0] * res[2][0] + m[1] * res[2][1] + m[2] * res[2][2];
rot[3] = m[3];
return rot;
}
template <typename T>
tmat4x4<T> scale(tmat4x4<T> const & m,tvec3<T> const & v)
{
tmat4x4<T> res;
res[0] = m[0] * v[0];
res[1] = m[1] * v[1];
res[2] = m[2] * v[2];
res[3] = m[3];
return res;
}
template <typename T>
tmat4x4<T> rotate_slow
(
tmat4x4<T> const & m,
T const & angle,
tvec3<T> const & v
)
{
T const a = DEG2RAD(angle);
T c = cos(a);
T s = sin(a);
tmat4x4<T> res;
tvec3<T> axis = normalize(v);
res[0][0] = c + (1 - c) * axis.x * axis.x;
res[0][1] = (1 - c) * axis.x * axis.y + s * axis.z;
res[0][2] = (1 - c) * axis.x * axis.z - s * axis.y;
res[0][3] = 0;
res[1][0] = (1 - c) * axis.y * axis.x - s * axis.z;
res[1][1] = c + (1 - c) * axis.y * axis.y;
res[1][2] = (1 - c) * axis.y * axis.z + s * axis.x;
res[1][3] = 0;
res[2][0] = (1 - c) * axis.z * axis.x + s * axis.y;
res[2][1] = (1 - c) * axis.z * axis.y - s * axis.x;
res[2][2] = c + (1 - c) * axis.z * axis.z;
res[2][3] = 0;
res[3] = tvec4<T>(0, 0, 0, 1);
return m * res;
}
template <typename T>
tmat4x4<T> scale_slow(tmat4x4<T> const & m,tvec3<T> const & v)
{
tmat4x4<T> res(T(1));
res[0][0] = v.x;
res[1][1] = v.y;
res[2][2] = v.z;
return m * res;
}
template <typename valType>
tmat4x4<valType> ortho
(
valType left,
valType right,
valType bottom,
valType top,
valType zNear,
valType zFar
)
{
tmat4x4<valType> res(1);
res[0][0] = valType(2) / (right - left);
res[1][1] = valType(2) / (top - bottom);
res[2][2] = - valType(2) / (zFar - zNear);
res[3][0] = - (right + left) / (right - left);
res[3][1] = - (top + bottom) / (top - bottom);
res[3][2] = - (zFar + zNear) / (zFar - zNear);
return res;
}
template <typename valType>
tmat4x4<valType> frustum
(
valType left,
valType right,
valType bottom,
valType top,
valType nearVal,
valType farVal
)
{
tmat4x4<valType> res(0);
res[0][0] = (valType(2) * nearVal) / (right - left);
res[1][1] = (valType(2) * nearVal) / (top - bottom);
res[2][0] = (right + left) / (right - left);
res[2][1] = (top + bottom) / (top - bottom);
res[2][2] = -(farVal + nearVal) / (farVal - nearVal);
res[2][3] = valType(-1);
res[3][2] = -(valType(2) * farVal * nearVal) / (farVal - nearVal);
return res;
}
template <typename valType>
tmat4x4<valType> perspective(valType fovy, valType aspect, valType zNear, valType zFar)
{
valType range = tan(fovy * valType(DEG2RAD(0.5))) * zNear;
valType left = -range * aspect;
valType right = range * aspect;
valType bottom = -range;
valType top = range;
tmat4x4<valType> res(valType(0));
res[0][0] = (valType(2) * zNear) / (right - left);
res[1][1] = (valType(2) * zNear) / (top - bottom);
res[2][2] = - (zFar + zNear) / (zFar - zNear);
res[2][3] = - valType(1);
res[3][2] = - (valType(2) * zFar * zNear) / (zFar - zNear);
return res;
}
template <typename T>
tvec3<T> project
(
tvec3<T> const & obj,
tmat4x4<T> const & model,
tmat4x4<T> const & proj,
tvec4<T> const & viewport
)
{
tvec4<T> tmp = tvec4<T>(obj.x, obj.y,obj.z,T(1));
tmp = model * tmp;
tmp = proj * tmp;
tmp /= tmp.w;
tmp = tmp * T(0.5) + T(0.5);
tmp[0] = tmp[0] * T(viewport[2]) + T(viewport[0]);
tmp[1] = tmp[1] * T(viewport[3]) + T(viewport[1]);
return tvec3<T>(tmp.x,tmp.y,tmp.z);
}
template <typename T>
tvec3<T> unProject
(
tvec3<T> const & win,
tmat4x4<T> const & model,
tmat4x4<T> const & proj,
tvec4<T> const & viewport
)
{
tmat4x4<T> inverses = (proj * model).inverse();
tvec4<T> tmp = tvec4<T>(win.x,win.y,win.z, T(1));
tmp.x = (tmp.x - T(viewport[0])) / T(viewport[2]);
tmp.y = (tmp.y - T(viewport[1])) / T(viewport[3]);
tmp = tmp * T(2) - T(1);
tvec4<T> obj = inverses * tmp;
obj /= obj.w;
return tvec3<T>(obj.x,obj.y,obj.z);
}
template <typename T>
tmat4x4<T> pickMatrix
(
tvec2<T> const & center,
tvec2<T> const & delta,
tvec4<T> const & viewport
)
{
assert(delta.x > T(0) && delta.y > T(0));
tmat4x4<T> res(1.0f);
if(!(delta.x > T(0) && delta.y > T(0)))
{
return res;
}
tvec3<T> Temp
(
(T(viewport[2]) - T(2) * (center.x - T(viewport[0]))) / delta.x,
(T(viewport[3]) - T(2) * (center.y - T(viewport[1]))) / delta.y,
T(0)
);
res = translate(res, Temp);
return scale(res, tvec3<T>(T(viewport[2]) / delta.x, T(viewport[3]) / delta.y, T(1)));
}
template <typename T>
tmat4x4<T> lookAt
(
tvec3<T> const & eye,
tvec3<T> const & center,
tvec3<T> const & up
)
{
tvec3<T> f = normalize(center - eye);
tvec3<T> u = normalize(up);
tvec3<T> s = normalize(cross(f, u));
u = cross(s, f);
tmat4x4<T> res(1);
res[0][0] = s.x;
res[1][0] = s.y;
res[2][0] = s.z;
res[0][1] = u.x;
res[1][1] = u.y;
res[2][1] = u.z;
res[0][2] = -f.x;
res[1][2] = -f.y;
res[2][2] = -f.z;
res[3][0] = -dot(s, eye);
res[3][1] = -dot(u, eye);
res[3][2] = dot(f, eye);
return res;
}
template<typename T>
class AxisAlignedBox2D
{
public:
enum Extent
{
EXTENT_NULL,
EXTENT_FINITE,
EXTENT_INFINITE
};
public:
tvec2<T> _minimum;
tvec2<T> _maximum;
Extent _extent;
public:
/*
1-----2
/| /|
/ | / |
5-----4 |
| 0--|--3
| / | /
|/ |/
6-----7
*/
typedef enum
{
FAR_LEFT_BOTTOM = 0,
FAR_LEFT_TOP = 1,
FAR_RIGHT_TOP = 2,
FAR_RIGHT_BOTTOM = 3,
NEAR_RIGHT_BOTTOM = 7,
NEAR_LEFT_BOTTOM = 6,
NEAR_LEFT_TOP = 5,
NEAR_RIGHT_TOP = 4
} CornerEnum;
AxisAlignedBox2D()
{
_minimum = tvec2<T>( T(-0.5), T(-0.5));
_maximum = tvec2<T>( T(0.5), T(0.5));
_extent = EXTENT_NULL;
}
AxisAlignedBox2D(const AxisAlignedBox2D & rkBox)
{
setExtents( rkBox._minimum, rkBox._maximum );
_extent = rkBox._extent;
}
AxisAlignedBox2D( const tvec2<T>& min, const tvec2<T>& max )
{
setExtents( min, max );
}
AxisAlignedBox2D(
T mx, T my,
T Mx, T My
)
{
setExtents( tvec2<T>(mx, my), tvec2<T>(Mx, My));
}
AxisAlignedBox2D<T>& operator=(const AxisAlignedBox2D<T>& right)
{
setExtents(right._minimum, right._maximum);
return *this;
}
~AxisAlignedBox2D()
{
}
/**
* Gets the minimum corner of the box.
*/
const tvec2<T>& getMinimum(void) const
{
return _minimum;
}
/**
* Gets a modifiable version of the minimum
* corner of the box.
*/
tvec2<T>& getMinimum(void)
{
return _minimum;
}
void setMinimum( const tvec2<T>& vec )
{
_minimum = vec;
}
void setMinimum( T x,T y )
{
_minimum = tvec2<T>(x,y);
}
/**
* Gets the maximum corner of the box.
*/
const tvec2<T>& getMaximum(void) const
{
return _maximum;
}
/**
* Gets a modifiable version of the maximum
* corner of the box.
*/
tvec2<T>& getMaximum(void)
{
return _maximum;
}
/**
* Sets the maximum corner of the box.
*/
void setMaximum( const tvec2<T>& vec )
{
_maximum = vec;
}
void setMaximum( T x, T y )
{
_maximum.x = x;
_maximum.y = y;
}
/**
* Sets both minimum and maximum extents at once.
*/
void setExtents( const tvec2<T>& min, const tvec2<T>& max )
{
_minimum = min;
_maximum = max;
_extent = EXTENT_FINITE;
}
void setExtents(
T mx, T my,
T Mx, T My
)
{
_minimum.x = mx;
_minimum.y = my;
_maximum.x = Mx;
_maximum.y = My;
_extent = EXTENT_FINITE;
}
inline bool intersects(const AxisAlignedBox2D& b2) const
{
if (_maximum.x < b2._minimum.x)
return false;
if (_maximum.y < b2._minimum.y)
return false;
if (_minimum.x > b2._maximum.x)
return false;
if (_minimum.y > b2._maximum.y)
return false;
return true;
}
inline AxisAlignedBox2D<T> intersection(const AxisAlignedBox2D<T>& b2) const
{
tvec2<T> intMin = _minimum;
tvec2<T> intMax = _maximum;
intMin.makeCeil(b2.getMinimum());
intMax.makeFloor(b2.getMaximum());
if (intMin.x < intMax.x &&
intMin.y < intMax.y)
{
return AxisAlignedBox2D<T>(intMin, intMax);
}
return AxisAlignedBox2D<T>();
}
inline void setNull()
{
_extent = EXTENT_NULL;
}
inline bool isNull(void) const
{
return (_extent == EXTENT_NULL);
}
bool isFinite(void) const
{
return (_extent == EXTENT_FINITE);
}
inline void setInfinite()
{
_extent = EXTENT_INFINITE;
}
inline bool isInfinite(void) const
{
return (_extent == EXTENT_INFINITE);
}
inline bool intersects(const tvec2<T>& v) const
{
return( v.x >= _minimum.x && v.x <= _maximum.x &&
v.y >= _minimum.y && v.y <= _maximum.y );
}
inline tvec2<T> getCenter(void) const
{
return tvec2<T>(
(_maximum.x + _minimum.x) * T(0.5f),
(_maximum.y + _minimum.y) * T(0.5f));
}
/**
* Gets the size of the box
*/
inline tvec2<T> getSize(void) const
{
return _maximum - _minimum;
}
inline tvec2<T> getHalfSize(void) const
{
return (_maximum - _minimum) * T(0.5);
}
inline bool contains(const tvec2<T>& v) const
{
return _minimum.x <= v.x && v.x <= _maximum.x &&
_minimum.y <= v.y && v.y <= _maximum.y;
}
inline bool contains(const AxisAlignedBox2D& other) const
{
return this->_minimum.x <= other._minimum.x &&
this->_minimum.y <= other._minimum.y &&
other._maximum.x <= this->_maximum.x &&
other._maximum.y <= this->_maximum.y;
}
inline bool operator== (const AxisAlignedBox2D& right) const
{
return this->_minimum == right._minimum &&
this->_maximum == right._maximum;
}
inline bool operator!= (const AxisAlignedBox2D& right) const
{
return !(*this == right);
}
inline void merge(tvec2<T> point)
{
if (_minimum.x > point.x)
{
_minimum.x = point.x;
}
if (_minimum.y > point.y)
{
_minimum.y = point.y;
}
if (_maximum.x < point.x)
{
_maximum.x = point.x;
}
if (_maximum.y < point.y)
{
_maximum.y = point.y;
}
}
inline void merge(AxisAlignedBox2D<T> other)
{
_maximum.makeCeil(other._maximum);
_minimum.makeFloor(other._minimum);
}
};
template<typename T>
class AxisAlignedBox
{
public:
enum Extent
{
EXTENT_NULL,
EXTENT_FINITE,
EXTENT_INFINITE
};
public:
tvec3<T> _minimum;
tvec3<T> _maximum;
Extent _extent;
public:
/*
1-----2
/| /|
/ | / |
5-----4 |
| 0--|--3
| / | /
|/ |/
6-----7
*/
typedef enum
{
FAR_LEFT_BOTTOM = 0,
FAR_LEFT_TOP = 1,
FAR_RIGHT_TOP = 2,
FAR_RIGHT_BOTTOM = 3,
NEAR_RIGHT_BOTTOM = 7,
NEAR_LEFT_BOTTOM = 6,
NEAR_LEFT_TOP = 5,
NEAR_RIGHT_TOP = 4
} CornerEnum;
AxisAlignedBox()
{
_minimum = tvec3<T>( T(-0.5), T(-0.5), T(-0.5) );
_maximum = tvec3<T>( T(0.5), T(0.5), T(0.5) );
_extent = EXTENT_NULL;
}
AxisAlignedBox(const AxisAlignedBox & rkBox)
{
setExtents( rkBox._minimum, rkBox._maximum );
_extent = rkBox._extent;
}
AxisAlignedBox( const tvec3<T>& min, const tvec3<T>& max )
{
setExtents( min, max );
}
AxisAlignedBox(
T mx, T my, T mz,
T Mx, T My, T Mz
)
{
setExtents( mx, my, mz, Mx, My, Mz );
}
AxisAlignedBox<T>& operator=(const AxisAlignedBox<T>& right)
{
setExtents(right._minimum, right._maximum);
return *this;
}
~AxisAlignedBox()
{
}
/**
* Gets the minimum corner of the box.
*/
const tvec3<T>& getMinimum(void) const
{
return _minimum;
}
/**
* Gets a modifiable version of the minimum
* corner of the box.
*/
tvec3<T>& getMinimum(void)
{
return _minimum;
}
void setMinimum(const tvec3<T>& mins)
{
_minimum = mins;
}
void setMinimum(T x,T y, T z)
{
_minimum = tvec3<T>(x,y,z);
}
/**
* Gets the maximum corner of the box.
*/
const tvec3<T>& getMaximum(void) const
{
return _maximum;
}
/**
* Gets a modifiable version of the maximum
* corner of the box.
*/
tvec3<T>& getMaximum(void)
{
return _maximum;
}
/**
* Sets the maximum corner of the box.
*/
void setMaximum( const tvec3<T>& vec )
{
_maximum = vec;
}
void setMaximum( T x, T y, T z )
{
_maximum.x = x;
_maximum.y = y;
_maximum.z = z;
}
/**
* Changes one of the components of the maximum corner of the box
* used to resize only one dimension of the box
*/
void setMaximumX( T x )
{
_maximum.x = x;
}
void setMaximumY( T y )
{
_maximum.y = y;
}
void setMaximumZ( T z )
{
_maximum.z = z;
}
/**
* Sets both minimum and maximum extents at once.
*/
void setExtents( const tvec3<T>& min, const tvec3<T>& max )
{
_minimum = min;
_maximum = max;
_extent = EXTENT_FINITE;
}
void setExtents(
T mx, T my, T mz,
T Mx, T My, T Mz )
{
_minimum.x = mx;
_minimum.y = my;
_minimum.z = mz;
_maximum.x = Mx;
_maximum.y = My;
_maximum.z = Mz;
_extent = EXTENT_FINITE;
}
/** Returns a pointer to an array of 8 corner points, useful for
collision vs. non-aligned objects.
@remarks
If the order of these corners is important, they are as
follows: The 4 points of the minimum Z face (note that
because Ogre uses right-handed coordinates, the minimum Z is
at the 'back' of the box) starting with the minimum point of
all, then anticlockwise around this face (if you are looking
onto the face from outside the box). Then the 4 points of the
maximum Z face, starting with maximum point of all, then
anticlockwise around this face (looking onto the face from
outside the box). Like this:
<pre>
1-----2
/| /|
/ | / |
5-----4 |
| 0--|--3
| / | /
|/ |/
6-----7
</pre>
@remarks as this implementation uses a static member, make sure to use your own copy !
*/
void getAllCorners(tvec3<T> mpCorners[8] ) const
{
mpCorners[0] = _minimum;
mpCorners[1].x = _minimum.x; mpCorners[1].y = _maximum.y; mpCorners[1].z = _minimum.z;
mpCorners[2].x = _maximum.x; mpCorners[2].y = _maximum.y; mpCorners[2].z = _minimum.z;
mpCorners[3].x = _maximum.x; mpCorners[3].y = _minimum.y; mpCorners[3].z = _minimum.z;
mpCorners[4] = _maximum;
mpCorners[5].x = _minimum.x; mpCorners[5].y = _maximum.y; mpCorners[5].z = _maximum.z;
mpCorners[6].x = _minimum.x; mpCorners[6].y = _minimum.y; mpCorners[6].z = _maximum.z;
mpCorners[7].x = _maximum.x; mpCorners[7].y = _minimum.y; mpCorners[7].z = _maximum.z;
}
/**
* gets the position of one of the corners
*/
tvec3<T> getCorner(CornerEnum cornerToGet) const
{
switch(cornerToGet)
{
case FAR_LEFT_BOTTOM:
return _minimum;
case FAR_LEFT_TOP:
return tvec3<T>(_minimum.x, _maximum.y, _minimum.z);
case FAR_RIGHT_TOP:
return tvec3<T>(_maximum.x, _maximum.y, _minimum.z);
case FAR_RIGHT_BOTTOM:
return tvec3<T>(_maximum.x, _minimum.y, _minimum.z);
case NEAR_RIGHT_BOTTOM:
return tvec3<T>(_maximum.x, _minimum.y, _maximum.z);
case NEAR_LEFT_BOTTOM:
return tvec3<T>(_minimum.x, _minimum.y, _maximum.z);
case NEAR_LEFT_TOP:
return tvec3<T>(_minimum.x, _maximum.y, _maximum.z);
case NEAR_RIGHT_TOP:
return _maximum;
default:
return tvec3<T>();
}
}
/**
* Merges the passed in box into the current box. The result is the
* box which encompasses both.
*/
void merge( const AxisAlignedBox<T>& right )
{
if ((right._extent == EXTENT_NULL) || (_extent == EXTENT_INFINITE))
{
return;
}
else if (right._extent == EXTENT_INFINITE)
{
_extent = EXTENT_INFINITE;
}
else if (_extent == EXTENT_NULL)
{
setExtents(right._minimum, right._maximum);
}
else
{
//! merge
tvec3<T> min = _minimum;
tvec3<T> max = _maximum;
max.makeCeil(right._maximum);
min.makeFloor(right._minimum);
setExtents(min, max);
}
}
/**
* Extends the box to encompass the specified point (if needed).
*/
void merge( const tvec3<T>& point )
{
switch (_extent)
{
case EXTENT_NULL: // if null, use this point
setExtents(point, point);
return;
case EXTENT_FINITE:
_maximum.makeCeil(point);
_minimum.makeFloor(point);
return;
case EXTENT_INFINITE:
return;
}
}
void transform( const tmat4x4<T>& matrix )
{
tvec3<T> oldMin;
tvec3<T> oldMax;
tvec3<T> currentCorner;
oldMin = _minimum;
oldMax = _maximum;
// We sequentially compute the corners in the following order :
// 0, 6, 5, 1, 2, 4 ,7 , 3
// This sequence allows us to only change one member at a time to get at all corners.
// For each one, we transform it using the matrix
// Which gives the resulting point and merge the resulting point.
currentCorner = oldMin;
tvec3<T> vVert = currentCorner * matrix;
setExtents(vVert,vVert);
// First corner
// min min min
currentCorner = oldMin;
merge( currentCorner * matrix );
// min,min,max
currentCorner.z = oldMax.z;
merge( currentCorner * matrix );
// min max max
currentCorner.y = oldMax.y;
merge( currentCorner * matrix );
// min max min
currentCorner.z = oldMin.z;
merge( currentCorner * matrix );
// max max min
currentCorner.x = oldMax.x;
merge( currentCorner * matrix );
// max max max
currentCorner.z = oldMax.z;
merge( currentCorner * matrix );
// max min max
currentCorner.y = oldMin.y;
merge( currentCorner * matrix);
// max min min
currentCorner.z = oldMin.z;
merge( currentCorner * matrix);
}
/**
* Returns whether or not this box intersects another.
*/
bool intersects(const AxisAlignedBox& b2) const
{
if (_maximum.x < b2._minimum.x)
return false;
if (_maximum.y < b2._minimum.y)
return false;
if (_maximum.z < b2._minimum.z)
return false;
if (_minimum.x > b2._maximum.x)
return false;
if (_minimum.y > b2._maximum.y)
return false;
if (_minimum.z > b2._maximum.z)
return false;
return true;
}
/**
* Returns whether or not this box intersects another.
*/
bool intersectsNoZ(const AxisAlignedBox& b2) const
{
if (_maximum.x < b2._minimum.x)
return false;
if (_maximum.y < b2._minimum.y)
return false;
if (_minimum.x > b2._maximum.x)
return false;
if (_minimum.y > b2._maximum.y)
return false;
return true;
}
AxisAlignedBox<T> intersection(const AxisAlignedBox<T>& b2) const
{
tvec3<T> intMin = _minimum;
tvec3<T> intMax = _maximum;
intMin.makeCeil(b2.getMinimum());
intMax.makeFloor(b2.getMaximum());
if (intMin.x < intMax.x &&
intMin.y < intMax.y &&
intMin.z < intMax.z)
{
return AxisAlignedBox<T>(intMin, intMax);
}
return AxisAlignedBox<T>();
}
void setNull()
{
_extent = EXTENT_NULL;
}
bool isNull(void) const
{
return (_extent == EXTENT_NULL);
}
bool isFinite(void) const
{
return (_extent == EXTENT_FINITE);
}
void setInfinite()
{
_extent = EXTENT_INFINITE;
}
bool isInfinite(void) const
{
return (_extent == EXTENT_INFINITE);
}
void scale(const tvec3<T>& s)
{
tvec3<T> min = _minimum * s;
tvec3<T> max = _maximum * s;
setExtents(min, max);
}
bool intersects(const tvec3<T>& v) const
{
return( v.x >= _minimum.x && v.x <= _maximum.x &&
v.y >= _minimum.y && v.y <= _maximum.y &&
v.z >= _minimum.z && v.z <= _maximum.z);
}
bool intersects(const tvec2<T>& v) const
{
return( v.x >= _minimum.x && v.x <= _maximum.x &&
v.y >= _minimum.y && v.y <= _maximum.y );
}
tvec3<T> getCenter(void) const
{
return tvec3<T>(
(_maximum.x + _minimum.x) * T(0.5f),
(_maximum.y + _minimum.y) * T(0.5f),
(_maximum.z + _minimum.z) * T(0.5f)
);
}
/**
* Gets the size of the box
*/
tvec3<T> getSize(void) const
{
return _maximum - _minimum;
}
tvec3<T> getHalfSize(void) const
{
return (_maximum - _minimum) * T(0.5);
}
bool contains(const tvec3<T>& v) const
{
return _minimum.x <= v.x && v.x <= _maximum.x &&
_minimum.y <= v.y && v.y <= _maximum.y &&
_minimum.z <= v.z && v.z <= _maximum.z;
}
bool contains(const AxisAlignedBox& other) const
{
return this->_minimum.x <= other._minimum.x &&
this->_minimum.y <= other._minimum.y &&
this->_minimum.z <= other._minimum.z &&
other._maximum.x <= this->_maximum.x &&
other._maximum.y <= this->_maximum.y &&
other._maximum.z <= this->_maximum.z;
}
bool operator== (const AxisAlignedBox& right) const
{
return this->_minimum == right._minimum &&
this->_maximum == right._maximum;
}
bool operator!= (const AxisAlignedBox& right) const
{
return !(*this == right);
}
};
template<typename T>
class tspline
{
public:
tspline()
{
mCoeffs[0][0] = 2;
mCoeffs[0][1] = -2;
mCoeffs[0][2] = 1;
mCoeffs[0][3] = 1;
mCoeffs[1][0] = -3;
mCoeffs[1][1] = 3;
mCoeffs[1][2] = -2;
mCoeffs[1][3] = -1;
mCoeffs[2][0] = 0;
mCoeffs[2][1] = 0;
mCoeffs[2][2] = 1;
mCoeffs[2][3] = 0;
mCoeffs[3][0] = 1;
mCoeffs[3][1] = 0;
mCoeffs[3][2] = 0;
mCoeffs[3][3] = 0;
mCoeffs = mCoeffs.transpose();
mAutoCalc = true;
}
~tspline(){};
void addPoint(const tvec3<T>& p)
{
mPoints.push_back(p);
if (mAutoCalc)
{
recalcTangents();
}
}
const tvec3<T>& getPoint(size_t index) const
{
assert (index < mPoints.size() && "Point index is out of bounds!!");
return mPoints[index];
}
tvec3<T>& getPoint(size_t index)
{
assert (index < mPoints.size() && "Point index is out of bounds!!");
return mPoints[index];
}
/**
* 获取点的数量
*/
size_t getNumPoints(void) const
{
return mPoints.size();
}
/**
* 清除所有的点数据
*/
void clear(void)
{
mPoints.clear();
mTangents.clear();
}
/**
* 更新点数据
*/
void updatePoint(size_t index, const tvec3<T>& value)
{
assert (index < mPoints.size() && "Point index is out of bounds!!");
mPoints[index] = value;
if (mAutoCalc)
{
recalcTangents();
}
}
/**
* 差值获取点数据
*/
tvec3<T> interpolate(T time) const
{
T fSeg = time * (mPoints.size() - 1);
unsigned segIdx = (unsigned)fSeg;
// Apportion t
time = fSeg - segIdx;
return interpolate(segIdx, time);
}
/**
* 根据索引差值
*/
tvec3<T> interpolate(size_t fromIndex, T t) const
{
// Bounds check
assert (fromIndex < mPoints.size() && "fromIndex out of bounds");
if ((fromIndex + 1) == mPoints.size())
{
// Duff request, cannot blend to nothing
// Just return source
return mPoints[fromIndex];
}
// Fast special cases
if (t == 0.0f)
{
return mPoints[fromIndex];
}
else if(t == 1.0f)
{
return mPoints[fromIndex + 1];
}
// float interpolation
// Form a vector of powers of t
T t2, t3;
t2 = t * t;
t3 = t2 * t;
tvec4<T> powers(t3, t2, t, 1);
const tvec3<T>& point1 = mPoints[fromIndex];
const tvec3<T>& point2 = mPoints[fromIndex+1];
const tvec3<T>& tan1 = mTangents[fromIndex];
const tvec3<T>& tan2 = mTangents[fromIndex+1];
tmat4x4<T> pt;
pt[0][0] = point1.x;
pt[0][1] = point1.y;
pt[0][2] = point1.z;
pt[0][3] = 1.0f;
pt[1][0] = point2.x;
pt[1][1] = point2.y;
pt[1][2] = point2.z;
pt[1][3] = 1.0f;
pt[2][0] = tan1.x;
pt[2][1] = tan1.y;
pt[2][2] = tan1.z;
pt[2][3] = 1.0f;
pt[3][0] = tan2.x;
pt[3][1] = tan2.y;
pt[3][2] = tan2.z;
pt[3][3] = 1.0f;
pt = pt.transpose();
tvec4<T> ret = powers * mCoeffs * pt;
return tvec3<T>(ret.x, ret.y, ret.z);
}
/**
* 自动计算标记
*/
void setAutoCalculate(bool autoCalc)
{
mAutoCalc = autoCalc;
}
/**
* 计算切线
*/
void recalcTangents(void)
{
size_t i, numPoints;
bool isClosed;
numPoints = mPoints.size();
if (numPoints < 2)
{
return;
}
if (mPoints[0] == mPoints[numPoints-1])
{
isClosed = true;
}
else
{
isClosed = false;
}
mTangents.resize(numPoints);
for(i = 0; i < numPoints; ++i)
{
if (i ==0)
{
// Special case start
if (isClosed)
{
// Use numPoints-2 since numPoints-1 is the last point and == [0]
mTangents[i] = 0.5f * (mPoints[1] - mPoints[numPoints-2]);
}
else
{
mTangents[i] = 0.5f * (mPoints[1] - mPoints[0]);
}
}
else if (i == numPoints-1)
{
if (isClosed)
{
mTangents[i] = mTangents[0];
}
else
{
mTangents[i] = 0.5f * (mPoints[i] - mPoints[i-1]);
}
}
else
{
mTangents[i] = 0.5f * (mPoints[i+1] - mPoints[i-1]);
}
}
}
public:
bool mAutoCalc;
std::vector< tvec3<T> > mPoints;
std::vector< tvec3<T> > mTangents;
tmat4x4<T> mCoeffs;
};
template < typename T >
class tellipsoidModel
{
public:
tellipsoidModel(T radiusEquator = T(WGS_84_RADIUS_EQUATOR),T radiusPolar = T(WGS_84_RADIUS_POLAR))
{
_radiusEquator = radiusEquator;
_radiusPolar = radiusPolar;
T flattening = (_radiusEquator-_radiusPolar)/_radiusEquator;
_eccentricitySquared= T(2)*flattening - flattening*flattening;
}
~tellipsoidModel(void)
{
}
void convertLatLongHeightToXYZ(
T latitude,
T longitude,
T height,
T& X,
T& Y,
T& Z
) const
{
// for details on maths see http://www.colorado.edu/geography/gcraft/notes/datum/gif/llhxyz.gif
T sin_latitude = sin(latitude);
T cos_latitude = cos(latitude);
T N = _radiusEquator / sqrt( 1.0 - _eccentricitySquared*sin_latitude*sin_latitude);
X = (N+height) * cos_latitude*cos(longitude);
Y = (N+height) * cos_latitude*sin(longitude);
Z = (N*(1-_eccentricitySquared)+height)*sin_latitude;
}
void convertXYZToLatLongHeight(
T X,
T Y,
T Z,
T& latitude,
T& longitude,
T& height
) const
{
// http://www.colorado.edu/geography/gcraft/notes/datum/gif/xyzllh.gif
T p = (T)sqrt(X*X + Y*Y);
T theta = (T)atan2(Z*_radiusEquator , (p*_radiusPolar));
T eDashSquared = (_radiusEquator*_radiusEquator - _radiusPolar*_radiusPolar) / (_radiusPolar*_radiusPolar);
T sin_theta = (T)sin(theta);
T cos_theta = (T)cos(theta);
latitude = (T)atan( (Z + eDashSquared*_radiusPolar*sin_theta*sin_theta*sin_theta) /
(p - _eccentricitySquared*_radiusEquator*cos_theta*cos_theta*cos_theta) );
longitude = (T)atan2(Y,X);
T sin_latitude = (T)sin(latitude);
T N = _radiusEquator / (T)sqrt( 1.0 - _eccentricitySquared*sin_latitude*sin_latitude);
height = p/(T)cos(latitude) - N;
}
protected:
T _radiusEquator;
T _radiusPolar;
T _eccentricitySquared;
};
class Rgba4Byte
{
public:
Rgba4Byte(
unsigned char r = 255,
unsigned char g = 255,
unsigned char b = 255,
unsigned char a = 255
)
{
_r = r;
_g = g;
_b = b;
_a = a;
}
friend bool operator == (const Rgba4Byte& left,const Rgba4Byte& right)
{
return left._r==right._r &&
left._g==right._g &&
left._b==right._b &&
left._a==right._a;
}
friend bool operator != (const Rgba4Byte& left,const Rgba4Byte& right)
{
return left._r !=right._r ||
left._g!=right._g ||
left._b!=right._b ||
left._a!=right._a;
}
operator unsigned()
{
unsigned color;
char* pColor = (char*)&color;
pColor[0] = _r;
pColor[1] = _g;
pColor[2] = _b;
pColor[3] = _a;
return color;
}
operator int()
{
int color;
char* pColor = (char*)&color;
pColor[0] = _r;
pColor[1] = _g;
pColor[2] = _b;
pColor[3] = _a;
return color;
}
operator ulong()
{
return toUint();
}
uint toUint()
{
return (_b) | (_g << 8) | (_r << 16) | (_a << 24);
}
public:
unsigned char _b;
unsigned char _g;
unsigned char _r;
unsigned char _a;
};
typedef Rgba4Byte Rgba;
inline Rgba4Byte colorLerp(const Rgba4Byte& c1, const Rgba4Byte& c2, float s)
{
Rgba4Byte color;
color._r = (unsigned char)(c1._r + s * (c2._r - c1._r));
color._g = (unsigned char)(c1._g + s * (c2._g - c1._g));
color._b = (unsigned char)(c1._b + s * (c2._b - c1._b));
color._a = (unsigned char)(c1._a + s * (c2._a - c1._a));
return color;
}
template <typename T>
class tAxisAlignedBox2
{
public:
enum Extent
{
EXTENT_NULL,
EXTENT_FINITE,
EXTENT_INFINITE
};
public:
tvec2<T> _vMin;
tvec2<T> _vMax;
Extent mExtent;
public:
tvec2<T> center() const
{
return (_vMin + _vMax) * T(0.5);
}
tvec2<T> size() const
{
return _vMax - _vMin;
}
tvec2<T> halfSize() const
{
return (_vMax - _vMin) * T(0.5);
}
bool intersects(tvec2<T> v) const
{
return( v.x >= _vMin.x && v.x <= _vMax.x &&
v.y >= _vMin.y && v.y <= _vMax.y );
}
void merge(tvec2<T> point)
{
if (_vMin.x > point.x)
{
_vMin.x = point.x;
}
if (_vMin.y > point.y)
{
_vMin.y = point.y;
}
if (_vMax.x < point.x)
{
_vMax.x = point.x;
}
if (_vMax.y < point.y)
{
_vMax.y = point.y;
}
}
void merge(tAxisAlignedBox2<T> other)
{
_vMax.makeCeil(other._vMax);
_vMin.makeFloor(other._vMin);
}
bool contains(tvec2<T> point) const
{
return _vMin.x <= point.x && point.x <= _vMax.x &&
_vMin.y <= point.y && point.y <= _vMax.y ;
}
bool contains(tAxisAlignedBox2<T> other) const
{
return this->_vMin.x <= other._vMin.x &&
this->_vMin.y <= other._vMin.y &&
other._vMax.x <= this->_vMax.x &&
other._vMax.y <= this->_vMax.y ;
}
};
template<typename T>
class tray
{
typedef T value_type;
typedef tray<T> type;
protected:
tvec3<T> _origin;
tvec3<T> _direction;
public:
tray():
_origin(value_type(0),value_type(0),value_type(0)),
_direction(value_type(0),value_type(0),value_type(1))
{}
tray(const tvec3<T>& origin, const tvec3<T>& direction):
_origin(origin),
_direction(direction)
{}
/**
* 设置射线的起点
*/
void setOrigin(const tvec3<T>& origin)
{
_origin = origin;
}
/**
* 返回射线的起点
*/
const tvec3<T>& getOrigin(void) const
{
return _origin;
}
/**
* 设置射线的方向
*/
void setDirection(const tvec3<T>& dir)
{
_direction = dir;
}
/**
* 返回射线的方向
*/
const tvec3<T>& getDirection(void) const
{
return _direction;
}
/**
* Gets the position of a point t units along the ray.
*/
tvec3<T> getPoint(T time) const
{
return tvec3<T>(_origin + (_direction * time));
}
/**
* 测试射线box相交
* 如果相交,返回值中的first == true.否则false
* second为射线到点的距离
* 调用getPoint方法,则返回交点
*/
std::pair<bool, T> intersects(const AxisAlignedBox<T>& box) const
{
T lowt = 0.0f;
T t;
bool hit = false;
tvec3<T> hitpoint;
tvec3<T> min = box.getMinimum();
tvec3<T> max = box.getMaximum();
/**
* 点在包围盒里面
*/
if ( _origin > min && _origin < max )
{
return std::pair<bool, T>(true, 0.0f);
}
// Check each face in turn, only check closest 3
// Min x
if (_origin.x <= min.x && _direction.x > 0)
{
t = (min.x - _origin.x) / _direction.x;
if (t >= 0)
{
// Substitute t back into ray and check bounds and dist
hitpoint = _origin + _direction * t;
if (hitpoint.y >= min.y &&
hitpoint.y <= max.y &&
hitpoint.z >= min.z &&
hitpoint.z <= max.z &&
(!hit || t < lowt))
{
hit = true;
lowt = t;
}
}
}
// Max x
if (_origin.x >= max.x && _direction.x < 0)
{
t = (max.x - _origin.x) / _direction.x;
if (t >= 0)
{
// Substitute t back into ray and check bounds and dist
hitpoint = _origin + _direction * t;
if (hitpoint.y >= min.y &&
hitpoint.y <= max.y &&
hitpoint.z >= min.z &&
hitpoint.z <= max.z &&
(!hit || t < lowt))
{
hit = true;
lowt = t;
}
}
}
// Min y
if (_origin.y <= min.y && _direction.y > 0)
{
t = (min.y - _origin.y) / _direction.y;
if (t >= 0)
{
// Substitute t back into ray and check bounds and dist
hitpoint = _origin + _direction * t;
if (hitpoint.x >= min.x &&
hitpoint.x <= max.x &&
hitpoint.z >= min.z &&
hitpoint.z <= max.z &&
(!hit || t < lowt))
{
hit = true;
lowt = t;
}
}
}
// Max y
if (_origin.y >= max.y && _direction.y < 0)
{
t = (max.y - _origin.y) / _direction.y;
if (t >= 0)
{
// Substitute t back into ray and check bounds and dist
hitpoint = _origin + _direction * t;
if (hitpoint.x >= min.x &&
hitpoint.x <= max.x &&
hitpoint.z >= min.z &&
hitpoint.z <= max.z &&
(!hit || t < lowt))
{
hit = true;
lowt = t;
}
}
}
// Min z
if (_origin.z <= min.z && _direction.z > 0)
{
t = (min.z - _origin.z) / _direction.z;
if (t >= 0)
{
// Substitute t back into ray and check bounds and dist
hitpoint = _origin + _direction * t;
if (hitpoint.x >= min.x &&
hitpoint.x <= max.x &&
hitpoint.y >= min.y &&
hitpoint.y <= max.y &&
(!hit || t < lowt))
{
hit = true;
lowt = t;
}
}
}
// Max z
if (_origin.z >= max.z && _direction.z < 0)
{
t = (max.z - _origin.z) / _direction.z;
if (t >= 0)
{
// Substitute t back into ray and check bounds and dist
hitpoint = _origin + _direction * t;
if (hitpoint.x >= min.x &&
hitpoint.x <= max.x &&
hitpoint.y >= min.y &&
hitpoint.y <= max.y &&
(!hit || t < lowt))
{
hit = true;
lowt = t;
}
}
}
return std::pair<bool, T>(hit, lowt);
}
};
template<class T>
class Plane
{
public:
tvec3<T> _normal;
T _distance;
public:
Plane ()
{
_normal = tvec3<T>(0,0,0);
_distance = 0.0f;
}
Plane (const Plane& right)
{
_normal = right._normal;
_distance = right._distance;
}
/** Construct a plane through a normal, and a distance to move the plane along the normal.*/
Plane (const tvec3<T>& rkNormal, T fConstant)
{
_normal = rkNormal;
_distance = -fConstant;
}
/** Construct a plane using the 4 constants directly **/
Plane (T x, T y, T z, T o)
{
_normal = tvec3<T>(x, y, z);
T invLen = 1.0f / (_normal).length();
_normal *= invLen;
_distance = o * invLen;
}
Plane (const tvec3<T>& rkNormal, const tvec3<T>& rkPoint)
{
redefine(rkNormal, rkPoint);
}
Plane (const tvec3<T>& rkPoint0, const tvec3<T>& rkPoint1,const tvec3<T>& rkPoint2)
{
redefine(rkPoint0, rkPoint1, rkPoint2);
}
/**
* 到点的距离
*/
float distance(const tvec3<T> &pos) const
{
return dot(_normal,pos) + _distance;
}
/** The "positive side" of the plane is the half space to which the
plane normal points. The "negative side" is the other half
space. The flag "no side" indicates the plane itself.
*/
enum Side
{
NO_SIDE,
POSITIVE_SIDE,
NEGATIVE_SIDE,
BOTH_SIDE
};
Side getSide (const tvec3<T>& rkPoint) const
{
float fDistance = getDistance(rkPoint);
if ( fDistance < 0.0 )
return Plane::NEGATIVE_SIDE;
if ( fDistance > 0.0 )
return Plane::POSITIVE_SIDE;
return Plane::NO_SIDE;
}
Side getSide (const tvec3<T>& centre, const tvec3<T>& halfSize) const
{
// Calculate the distance between box centre and the plane
float dist = getDistance(centre);
// Calculate the maximise allows absolute distance for
// the distance between box centre and plane
float maxAbsDist = _normal.absDot(halfSize);
if (dist < -maxAbsDist)
return Plane::NEGATIVE_SIDE;
if (dist > +maxAbsDist)
return Plane::POSITIVE_SIDE;
return Plane::BOTH_SIDE;
}
float getDistance (const tvec3<T>& rkPoint) const
{
return _normal.dot(rkPoint) + _distance;
}
void redefine(const tvec3<T>& rkPoint0, const tvec3<T>& rkPoint1,
const tvec3<T>& rkPoint2)
{
tvec3<T> kEdge1 = rkPoint1 - rkPoint0;
tvec3<T> kEdge2 = rkPoint2 - rkPoint0;
_normal = cross(kEdge1,kEdge2);
_normal.normalise();
_distance = -dot(_normal,rkPoint0);
}
/** Redefine this plane based on a normal and a point. */
void redefine(const tvec3<T>& rkNormal, const tvec3<T>& rkPoint)
{
_normal = rkNormal;
_distance = -dot(rkNormal,rkPoint);
}
// tvec3<T> projectVector(const tvec3<T>& p) const
// {
// matrix3 xform;
// xform[0][0] = 1.0f - _normal.x * _normal.x;
// xform[0][1] = -_normal.x * _normal.y;
// xform[0][2] = -_normal.x * _normal.z;
// xform[1][0] = -_normal.y * _normal.x;
// xform[1][1] = 1.0f - _normal.y * _normal.y;
// xform[1][2] = -_normal.y * _normal.z;
// xform[2][0] = -_normal.z * _normal.x;
// xform[2][1] = -_normal.z * _normal.y;
// xform[2][2] = 1.0f - _normal.z * _normal.z;
// return xform * p;
// }
/** Normalises the plane.
@remarks
This method normalises the plane's normal and the length scale of d
is as well.
@note
This function will not crash for zero-sized vectors, but there
will be no changes made to their components.
@returns The previous length of the plane's normal.
*/
float normalise(void)
{
float fLength = _normal.length();
// Will also work for zero-sized vectors, but will change nothing
if (fLength > 1e-08f)
{
float fInvLength = 1.0f / fLength;
_normal *= fInvLength;
_distance *= fInvLength;
}
return fLength;
}
/// Comparison operator
bool operator==(const Plane& right) const
{
return (right._distance == _distance && right._normal == _normal);
}
bool operator!=(const Plane& right) const
{
return (right._distance != _distance && right._normal != _normal);
}
};
template<class T>
class tfrustum
{
public:
enum
{
FRUSTUM_LEFT = 0,
FRUSTUM_RIGHT = 1,
FRUSTUM_TOP = 2,
FRUSTUM_BOTTOM = 3,
FRUSTUM_FAR = 4,
FRUSTUM_NEAR = 5,
};
public:
/**
* project * modleview
*/
void loadFrustum(const tmat4x4<T> &mvp)
{
const T* dataPtr = mvp.data();
_planes[FRUSTUM_LEFT ] = Plane<T>(dataPtr[12] - dataPtr[0], dataPtr[13] - dataPtr[1], dataPtr[14] - dataPtr[2], dataPtr[15] - dataPtr[3]);
_planes[FRUSTUM_RIGHT ] = Plane<T>(dataPtr[12] + dataPtr[0], dataPtr[13] + dataPtr[1], dataPtr[14] + dataPtr[2], dataPtr[15] + dataPtr[3]);
_planes[FRUSTUM_TOP ] = Plane<T>(dataPtr[12] - dataPtr[4], dataPtr[13] - dataPtr[5], dataPtr[14] - dataPtr[6], dataPtr[15] - dataPtr[7]);
_planes[FRUSTUM_BOTTOM] = Plane<T>(dataPtr[12] + dataPtr[4], dataPtr[13] + dataPtr[5], dataPtr[14] + dataPtr[6], dataPtr[15] + dataPtr[7]);
_planes[FRUSTUM_FAR ] = Plane<T>(dataPtr[12] - dataPtr[8], dataPtr[13] - dataPtr[9], dataPtr[14] - dataPtr[10], dataPtr[15] - dataPtr[11]);
_planes[FRUSTUM_NEAR ] = Plane<T>(dataPtr[12] + dataPtr[8], dataPtr[13] + dataPtr[9], dataPtr[14] + dataPtr[10], dataPtr[15] + dataPtr[11]);
}
bool pointInFrustum(const tvec3<T> &pos) const
{
for (int i = 0; i < 6; i++)
{
if (_planes[i].distance(pos) <= 0)
return false;
}
return true;
}
bool sphereInFrustum(const tvec3<T> &pos, const float radius) const
{
for (int i = 0; i < 6; i++)
{
if (_planes[i].distance(pos) <= -radius)
return false;
}
return true;
}
bool cubeInFrustum(T minX,T maxX,T minY,T maxY,T minZ,T maxZ) const
{
for (int i = 0; i < 6; i++)
{
if (_planes[i].distance(tvec3<T>(minX, minY, minZ)) > 0) continue;
if (_planes[i].distance(tvec3<T>(minX, minY, maxZ)) > 0) continue;
if (_planes[i].distance(tvec3<T>(minX, maxY, minZ)) > 0) continue;
if (_planes[i].distance(tvec3<T>(minX, maxY, maxZ)) > 0) continue;
if (_planes[i].distance(tvec3<T>(maxX, minY, minZ)) > 0) continue;
if (_planes[i].distance(tvec3<T>(maxX, minY, maxZ)) > 0) continue;
if (_planes[i].distance(tvec3<T>(maxX, maxY, minZ)) > 0) continue;
if (_planes[i].distance(tvec3<T>(maxX, maxY, maxZ)) > 0) continue;
return false;
}
return true;
}
const Plane<T> &getPlane(const int plane) const
{
return _planes[plane];
}
protected:
Plane<T> _planes[6];
};
typedef float real;
typedef tvec2<int> int2;
typedef tvec2<float> float2;
typedef tvec2<double> double2;
typedef tvec2<real> real2;
typedef tvec3<int> int3;
typedef tvec3<unsigned> uint3;
typedef tvec3<float> float3;
typedef tvec3<double> double3;
typedef tvec3<real> real3;
typedef tvec4<int> int4;
typedef tvec4<float> float4;
typedef tvec4<double> double4;
typedef tvec4<real> real4;
typedef trect<real> rect4;
typedef trect<int> rect4i;
typedef AxisAlignedBox<float> aabb3d;
typedef AxisAlignedBox<real> aabbr;
typedef AxisAlignedBox2D<float> AABB2D;
typedef AxisAlignedBox2D<real> aabb2dr;
typedef AxisAlignedBox2D<int> aabb2di;
typedef tmat2x2<float> matrix2;
typedef tmat3x3<float> matrix3;
typedef tmat4x4<float> matrix4;
typedef tmat4x4<real> matrix4r;
typedef tquat<float> quaternion;
typedef tquat<real> quatr;
typedef tray<float> Ray;
typedef tfrustum<float> Frustum;
typedef tellipsoidModel<float> ellipsoid;
}