1
0
cuberite-2a/src/Vector3.h

446 lines
8.7 KiB
C
Raw Normal View History

2014-03-11 10:01:17 -04:00
#pragma once
#include <list>
#include <vector>
2014-03-11 10:01:17 -04:00
template <typename T>
// tolua_begin
class Vector3
{
TOLUA_TEMPLATE_BIND((T, int, float, double))
public:
T x, y, z;
2014-03-11 12:32:33 -04:00
inline Vector3(void) : x(0), y(0), z(0) {}
2014-03-11 10:01:17 -04:00
inline Vector3(T a_x, T a_y, T a_z) : x(a_x), y(a_y), z(a_z) {}
2014-03-11 15:58:50 -04:00
// Hardcoded copy constructors (tolua++ does not support function templates .. yet)
Vector3(const Vector3<float> & a_Rhs) : x(static_cast<T>(a_Rhs.x)), y(static_cast<T>(a_Rhs.y)), z(static_cast<T>(a_Rhs.z)) {}
Vector3(const Vector3<double> & a_Rhs) : x(static_cast<T>(a_Rhs.x)), y(static_cast<T>(a_Rhs.y)), z(static_cast<T>(a_Rhs.z)) {}
Vector3(const Vector3<int> & a_Rhs) : x(static_cast<T>(a_Rhs.x)), y(static_cast<T>(a_Rhs.y)), z(static_cast<T>(a_Rhs.z)) {}
2014-03-11 15:58:50 -04:00
2014-03-11 10:01:17 -04:00
// tolua_end
template <typename _T>
Vector3(const Vector3<_T> & a_Rhs) : x(a_Rhs.x), y(a_Rhs.y), z(a_Rhs.z) {}
template <typename _T>
Vector3(const Vector3<_T> * a_Rhs) : x(a_Rhs->x), y(a_Rhs->y), z(a_Rhs->z) {}
// tolua_begin
inline void Set(T a_x, T a_y, T a_z)
{
x = a_x;
y = a_y;
z = a_z;
}
inline void Normalize(void)
{
double Len = 1.0 / Length();
2014-03-11 10:01:17 -04:00
x = static_cast<T>(x * Len);
y = static_cast<T>(y * Len);
z = static_cast<T>(z * Len);
2014-03-11 10:01:17 -04:00
}
inline Vector3<T> NormalizeCopy(void) const
{
double Len = 1.0 / Length();
2014-03-11 10:01:17 -04:00
return Vector3<T>(
static_cast<T>(x * Len),
static_cast<T>(y * Len),
static_cast<T>(z * Len)
2014-03-11 10:01:17 -04:00
);
}
// tolua_end
/** Sets the given vector to the normalized version of this vector.
Removed from LuaAPI, because Lua doesn't need distinguishing from the other overload. */
2014-03-11 10:01:17 -04:00
inline void NormalizeCopy(Vector3<T> & a_Rhs) const
{
double Len = 1.0 / Length();
2014-03-11 10:01:17 -04:00
a_Rhs.Set(
static_cast<T>(x * Len),
static_cast<T>(y * Len),
static_cast<T>(z * Len)
2014-03-11 10:01:17 -04:00
);
}
// tolua_begin
inline bool HasNonZeroLength(void) const
{
2015-05-08 18:32:02 -04:00
#ifdef __clang__
#pragma clang diagnostic push
#pragma clang diagnostic ignored "-Wfloat-equal"
#endif
return ((x != 0) || (y != 0) || (z != 0));
2015-05-08 18:32:02 -04:00
#ifdef __clang__
#pragma clang diagnostic pop
#endif
}
inline double Length(void) const
2014-03-11 10:01:17 -04:00
{
return sqrt(static_cast<double>(x * x + y * y + z * z));
2014-03-11 10:01:17 -04:00
}
inline double SqrLength(void) const
2014-03-11 10:01:17 -04:00
{
return x * x + y * y + z * z;
}
inline T Dot(const Vector3<T> & a_Rhs) const
{
return x * a_Rhs.x + y * a_Rhs.y + z * a_Rhs.z;
}
/** Updates each coord to its absolute value */
inline void Abs()
{
x = (x < 0) ? -x : x;
y = (y < 0) ? -y : y;
z = (z < 0) ? -z : z;
}
/** Clamps each coord into the specified range. */
inline void Clamp(T a_Min, T a_Max)
{
x = ::Clamp(x, a_Min, a_Max);
y = ::Clamp(y, a_Min, a_Max);
z = ::Clamp(z, a_Min, a_Max);
}
2016-02-05 16:45:45 -05:00
2014-03-11 10:01:17 -04:00
inline Vector3<T> Cross(const Vector3<T> & a_Rhs) const
{
return Vector3<T>(
y * a_Rhs.z - z * a_Rhs.y,
z * a_Rhs.x - x * a_Rhs.z,
x * a_Rhs.y - y * a_Rhs.x
);
}
inline bool Equals(const Vector3<T> & a_Rhs) const
{
// Perform a strict comparison of the contents - we want to know whether this object is exactly equal
// To perform EPS-based comparison, use the EqualsEps() function
2015-05-08 18:32:02 -04:00
#ifdef __clang__
#pragma clang diagnostic push
#pragma clang diagnostic ignored "-Wfloat-equal"
#endif
return !((x != a_Rhs.x) || (y != a_Rhs.y) || (z != a_Rhs.z));
2015-05-08 18:32:02 -04:00
#ifdef __clang__
#pragma clang diagnostic pop
#endif
}
2016-02-05 16:45:45 -05:00
inline bool EqualsEps(const Vector3<T> & a_Rhs, T a_Eps) const
2014-03-11 10:01:17 -04:00
{
return (Abs(x - a_Rhs.x) < a_Eps) && (Abs(y - a_Rhs.y) < a_Eps) && (Abs(z - a_Rhs.z) < a_Eps);
2014-03-11 10:01:17 -04:00
}
inline void Move(T a_X, T a_Y, T a_Z)
{
x += a_X;
y += a_Y;
z += a_Z;
}
2014-03-25 16:33:23 -04:00
inline void Move(const Vector3<T> & a_Diff)
{
x += a_Diff.x;
y += a_Diff.y;
z += a_Diff.z;
}
/** Returns a new Vector3i with coords set to std::floor() of this vector's coords. */
2014-07-04 12:42:40 -04:00
inline Vector3<int> Floor(void) const
2014-06-30 14:21:21 -04:00
{
2014-07-04 12:49:44 -04:00
return Vector3<int>(
FloorC(x),
FloorC(y),
FloorC(z)
);
2014-06-30 14:21:21 -04:00
}
2014-03-11 10:01:17 -04:00
// tolua_end
inline bool operator != (const Vector3<T> & a_Rhs) const
{
return !Equals(a_Rhs);
}
inline bool operator == (const Vector3<T> & a_Rhs) const
{
return Equals(a_Rhs);
}
inline bool operator > (const Vector3<T> & a_Rhs) const
{
return (SqrLength() > a_Rhs.SqrLength());
}
inline bool operator < (const Vector3<T> & a_Rhs) const
{
return (SqrLength() < a_Rhs.SqrLength());
}
2014-03-11 10:01:17 -04:00
inline void operator += (const Vector3<T> & a_Rhs)
{
x += a_Rhs.x;
y += a_Rhs.y;
z += a_Rhs.z;
}
inline void operator -= (const Vector3<T> & a_Rhs)
{
x -= a_Rhs.x;
y -= a_Rhs.y;
z -= a_Rhs.z;
}
inline void operator *= (const Vector3<T> & a_Rhs)
{
x *= a_Rhs.x;
y *= a_Rhs.y;
z *= a_Rhs.z;
}
inline void operator *= (T a_v)
{
x *= a_v;
y *= a_v;
z *= a_v;
}
inline Vector3<T> & operator = (const Vector3<T> & a_Rhs)
{
x = a_Rhs.x;
y = a_Rhs.y;
z = a_Rhs.z;
return *this;
}
2014-03-11 10:01:17 -04:00
// tolua_begin
2016-02-05 16:45:45 -05:00
2014-03-11 10:01:17 -04:00
inline Vector3<T> operator + (const Vector3<T>& a_Rhs) const
{
return Vector3<T>(
x + a_Rhs.x,
y + a_Rhs.y,
z + a_Rhs.z
);
}
inline Vector3<T> operator - (const Vector3<T>& a_Rhs) const
{
return Vector3<T>(
x - a_Rhs.x,
y - a_Rhs.y,
z - a_Rhs.z
);
}
inline Vector3<T> operator - (void) const
{
return Vector3<T>(-x, -y, -z);
}
2014-03-11 10:01:17 -04:00
inline Vector3<T> operator * (const Vector3<T>& a_Rhs) const
{
return Vector3<T>(
x * a_Rhs.x,
y * a_Rhs.y,
z * a_Rhs.z
);
}
inline Vector3<T> operator / (const Vector3<T> & a_Rhs)
{
return Vector3<T>(
x / a_Rhs.x,
y / a_Rhs.y,
z / a_Rhs.z
);
}
2014-03-11 10:01:17 -04:00
inline Vector3<T> operator * (T a_v) const
{
return Vector3<T>(
x * a_v,
y * a_v,
z * a_v
);
}
inline Vector3<T> operator / (T a_v) const
{
return Vector3<T>(
x / a_v,
y / a_v,
z / a_v
);
}
2014-03-11 12:10:15 -04:00
/** Returns the coefficient for the (a_OtherEnd - this) line to reach the specified Z coord.
The result satisfies the following equation:
(*this + Result * (a_OtherEnd - *this)).z = a_Z
If the line is too close to being parallel, this function returns NO_INTERSECTION
*/
2014-03-11 10:01:17 -04:00
inline double LineCoeffToXYPlane(const Vector3<T> & a_OtherEnd, T a_Z) const
{
if (Abs(z - a_OtherEnd.z) < EPS)
2014-03-11 10:01:17 -04:00
{
return NO_INTERSECTION;
}
return (a_Z - z) / (a_OtherEnd.z - z);
}
2014-03-11 12:10:15 -04:00
/** Returns the coefficient for the (a_OtherEnd - this) line to reach the specified Y coord.
The result satisfies the following equation:
(*this + Result * (a_OtherEnd - *this)).y = a_Y
If the line is too close to being parallel, this function returns NO_INTERSECTION
*/
2014-03-11 10:01:17 -04:00
inline double LineCoeffToXZPlane(const Vector3<T> & a_OtherEnd, T a_Y) const
{
if (Abs(y - a_OtherEnd.y) < EPS)
2014-03-11 10:01:17 -04:00
{
return NO_INTERSECTION;
}
return (a_Y - y) / (a_OtherEnd.y - y);
}
2014-03-11 12:10:15 -04:00
/** Returns the coefficient for the (a_OtherEnd - this) line to reach the specified X coord.
The result satisfies the following equation:
(*this + Result * (a_OtherEnd - *this)).x = a_X
If the line is too close to being parallel, this function returns NO_INTERSECTION
*/
2014-03-11 10:01:17 -04:00
inline double LineCoeffToYZPlane(const Vector3<T> & a_OtherEnd, T a_X) const
{
if (Abs(x - a_OtherEnd.x) < EPS)
2014-03-11 10:01:17 -04:00
{
return NO_INTERSECTION;
}
return (a_X - x) / (a_OtherEnd.x - x);
2014-06-29 17:41:31 -04:00
}
/** Rotates the vector 90 degrees clockwise around the vertical axis.
Note that this is specific to minecraft's axis ordering, which is X+ left, Z+ down. */
inline void TurnCW(void)
{
std::swap(x, z);
x = -x;
}
/** Rotates the vector 90 degrees counterclockwise around the vertical axis.
Note that this is specific to minecraft's axis ordering, which is X+ left, Z+ down. */
inline void TurnCCW(void)
{
std::swap(x, z);
z = -z;
}
2014-03-11 10:01:17 -04:00
/** The max difference between two coords for which the coords are assumed equal. */
static const double EPS;
/** Return value of LineCoeffToPlane() if the line is parallel to the plane. */
static const double NO_INTERSECTION;
2016-02-05 16:45:45 -05:00
protected:
2014-03-11 12:10:15 -04:00
/** Returns the absolute value of the given argument.
Templatized because the standard library differentiates between abs() and fabs(). */
static T Abs(T a_Value)
{
return (a_Value < 0) ? -a_Value : a_Value;
}
2014-06-29 17:41:31 -04:00
2014-03-11 10:01:17 -04:00
};
// tolua_end
2014-03-11 12:10:15 -04:00
2015-04-29 12:24:14 -04:00
2014-07-10 06:46:09 -04:00
template <> inline Vector3<int> Vector3<int>::Floor(void) const
{
return *this;
}
2015-06-07 06:52:14 -04:00
template <typename What>
class VectorHasher
{
public:
/** Provides a hash of a vector's contents */
size_t operator()(const Vector3<What> & a_Vector) const
{
2016-04-20 17:53:38 -04:00
// Guaranteed to have non repeating hashes for any 128x128x128 area
size_t Hash = static_cast<size_t>(a_Vector.y);
Hash <<= 16;
2015-06-07 06:52:14 -04:00
Hash ^= static_cast<size_t>(a_Vector.x);
2016-04-20 17:53:38 -04:00
Hash ^= static_cast<size_t>(a_Vector.z) << 8;
2015-06-07 06:52:14 -04:00
return Hash;
}
};
2014-03-11 10:01:17 -04:00
template <typename T>
const double Vector3<T>::EPS = 0.000001;
template <typename T>
const double Vector3<T>::NO_INTERSECTION = 1e70;
// tolua_begin
typedef Vector3<double> Vector3d;
typedef Vector3<float> Vector3f;
typedef Vector3<int> Vector3i;
// tolua_end
typedef std::vector<Vector3i> cVector3iArray;