CQuaternion::lookAt imp

This commit is contained in:
Phillip Stephens 2016-09-16 15:20:12 -07:00
parent d03f1157dd
commit 0a7c192dd5
2 changed files with 48 additions and 1 deletions

View File

@ -15,6 +15,17 @@
namespace zeus namespace zeus
{ {
static inline float normalize_angle(float angle)
{
if (angle > M_PIF)
angle = -((2.f * angle) - M_PIF);
else if (angle < -M_PIF)
angle = 2.f * angle + M_PIF;
return angle;
}
class alignas(16) CQuaternion class alignas(16) CQuaternion
{ {
#if __atdna__ && ZE_ATHENA_TYPES #if __atdna__ && ZE_ATHENA_TYPES
@ -181,6 +192,40 @@ public:
return {q.x, q.y, q.z}; return {q.x, q.y, q.z};
} }
static inline CQuaternion lookAt(const CUnitVector3f& target, const CUnitVector3f& up, const CRelAngle& c)
{
CQuaternion q = skNoRotation;
zeus::CVector3f upCpy = up;
zeus::CVector3f targetCpy = target;
upCpy.z = 0.f;
targetCpy.z = 0.f;
zeus::CVector3f tmp;
if (upCpy.magnitude() > 0.0009f && upCpy.magnitude() > 0.0009f)
{
targetCpy.normalize();
upCpy.normalize();
CRelAngle angleBetween = normalize_angle(std::atan2(targetCpy.x, targetCpy.y) - std::atan2(upCpy.x, upCpy.y));
CRelAngle realAngle = zeus::clamp<CRelAngle>(-c, angleBetween, c);
CQuaternion tmpQ;
tmpQ.rotateZ(realAngle);
q = tmpQ;
CQuaternion q2 = (q * CQuaternion{0.f, targetCpy}) * -tmpQ;
tmp.x = q2.x;
tmp.y = q2.y;
tmp.z = q2.z;
}
else if (upCpy.magnitude() > 0.0009f)
tmp = targetCpy.normalized();
else if (upCpy.magnitude() > 0.0009f)
tmp = upCpy.normalized();
else
return skNoRotation;
CRelAngle realAngle = zeus::clamp<CRelAngle>(-c, normalize_angle(std::acos(up.z) - std::acos(target.z)), c);
return CQuaternion::fromAxisAngle(tmp.cross(kUpVec), realAngle) * q;
}
CVector3f transform(const CVector3f& v) const { return rotate(*this, v); } CVector3f transform(const CVector3f& v) const { return rotate(*this, v); }
CQuaternion log() const; CQuaternion log() const;

View File

@ -267,7 +267,9 @@ public:
} }
inline CVector3f cross(const CVector3f& rhs) const inline CVector3f cross(const CVector3f& rhs) const
{ {
return CVector3f(y * rhs.z - z * rhs.y, z * rhs.x - x * rhs.z, x * rhs.y - y * rhs.x); return CVector3f(y * rhs.z - z * rhs.y,
z * rhs.x - x * rhs.z,
x * rhs.y - y * rhs.x);
} }
inline float dot(const CVector3f& rhs) const inline float dot(const CVector3f& rhs) const