nuclear@10: /* nuclear@10: libvmath - a vector math library nuclear@10: Copyright (C) 2004-2015 John Tsiombikas nuclear@10: nuclear@10: This program is free software: you can redistribute it and/or modify nuclear@10: it under the terms of the GNU Lesser General Public License as published nuclear@10: by the Free Software Foundation, either version 3 of the License, or nuclear@10: (at your option) any later version. nuclear@10: nuclear@10: This program is distributed in the hope that it will be useful, nuclear@10: but WITHOUT ANY WARRANTY; without even the implied warranty of nuclear@10: MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the nuclear@10: GNU Lesser General Public License for more details. nuclear@10: nuclear@10: You should have received a copy of the GNU Lesser General Public License nuclear@10: along with this program. If not, see . nuclear@10: */ nuclear@10: #include "quat.h" nuclear@10: #include "vmath.h" nuclear@10: nuclear@10: Quaternion::Quaternion() nuclear@10: { nuclear@10: s = 1.0; nuclear@10: v.x = v.y = v.z = 0.0; nuclear@10: } nuclear@10: nuclear@10: Quaternion::Quaternion(scalar_t s, const Vector3 &v) nuclear@10: { nuclear@10: this->s = s; nuclear@10: this->v = v; nuclear@10: } nuclear@10: nuclear@10: Quaternion::Quaternion(scalar_t s, scalar_t x, scalar_t y, scalar_t z) nuclear@10: { nuclear@10: v.x = x; nuclear@10: v.y = y; nuclear@10: v.z = z; nuclear@10: this->s = s; nuclear@10: } nuclear@10: nuclear@10: Quaternion::Quaternion(const Vector3 &axis, scalar_t angle) nuclear@10: { nuclear@10: set_rotation(axis, angle); nuclear@10: } nuclear@10: nuclear@10: Quaternion::Quaternion(const quat_t &quat) nuclear@10: { nuclear@10: v.x = quat.x; nuclear@10: v.y = quat.y; nuclear@10: v.z = quat.z; nuclear@10: s = quat.w; nuclear@10: } nuclear@10: nuclear@10: Quaternion Quaternion::operator +(const Quaternion &quat) const nuclear@10: { nuclear@10: return Quaternion(s + quat.s, v + quat.v); nuclear@10: } nuclear@10: nuclear@10: Quaternion Quaternion::operator -(const Quaternion &quat) const nuclear@10: { nuclear@10: return Quaternion(s - quat.s, v - quat.v); nuclear@10: } nuclear@10: nuclear@10: Quaternion Quaternion::operator -() const nuclear@10: { nuclear@10: return Quaternion(-s, -v); nuclear@10: } nuclear@10: nuclear@10: /** Quaternion Multiplication: nuclear@10: * Q1*Q2 = [s1*s2 - v1.v2, s1*v2 + s2*v1 + v1(x)v2] nuclear@10: */ nuclear@10: Quaternion Quaternion::operator *(const Quaternion &quat) const nuclear@10: { nuclear@10: Quaternion newq; nuclear@10: newq.s = s * quat.s - dot_product(v, quat.v); nuclear@10: newq.v = quat.v * s + v * quat.s + cross_product(v, quat.v); nuclear@10: return newq; nuclear@10: } nuclear@10: nuclear@10: void Quaternion::operator +=(const Quaternion &quat) nuclear@10: { nuclear@10: *this = Quaternion(s + quat.s, v + quat.v); nuclear@10: } nuclear@10: nuclear@10: void Quaternion::operator -=(const Quaternion &quat) nuclear@10: { nuclear@10: *this = Quaternion(s - quat.s, v - quat.v); nuclear@10: } nuclear@10: nuclear@10: void Quaternion::operator *=(const Quaternion &quat) nuclear@10: { nuclear@10: *this = *this * quat; nuclear@10: } nuclear@10: nuclear@10: void Quaternion::reset_identity() nuclear@10: { nuclear@10: s = 1.0; nuclear@10: v.x = v.y = v.z = 0.0; nuclear@10: } nuclear@10: nuclear@10: Quaternion Quaternion::conjugate() const nuclear@10: { nuclear@10: return Quaternion(s, -v); nuclear@10: } nuclear@10: nuclear@10: scalar_t Quaternion::length() const nuclear@10: { nuclear@10: return (scalar_t)sqrt(v.x*v.x + v.y*v.y + v.z*v.z + s*s); nuclear@10: } nuclear@10: nuclear@10: /** Q * ~Q = ||Q||^2 */ nuclear@10: scalar_t Quaternion::length_sq() const nuclear@10: { nuclear@10: return v.x*v.x + v.y*v.y + v.z*v.z + s*s; nuclear@10: } nuclear@10: nuclear@10: void Quaternion::normalize() nuclear@10: { nuclear@10: scalar_t len = (scalar_t)sqrt(v.x*v.x + v.y*v.y + v.z*v.z + s*s); nuclear@10: v.x /= len; nuclear@10: v.y /= len; nuclear@10: v.z /= len; nuclear@10: s /= len; nuclear@10: } nuclear@10: nuclear@10: Quaternion Quaternion::normalized() const nuclear@10: { nuclear@10: Quaternion nq = *this; nuclear@10: scalar_t len = (scalar_t)sqrt(v.x*v.x + v.y*v.y + v.z*v.z + s*s); nuclear@10: nq.v.x /= len; nuclear@10: nq.v.y /= len; nuclear@10: nq.v.z /= len; nuclear@10: nq.s /= len; nuclear@10: return nq; nuclear@10: } nuclear@10: nuclear@10: /** Quaternion Inversion: Q^-1 = ~Q / ||Q||^2 */ nuclear@10: Quaternion Quaternion::inverse() const nuclear@10: { nuclear@10: Quaternion inv = conjugate(); nuclear@10: scalar_t lensq = length_sq(); nuclear@10: inv.v /= lensq; nuclear@10: inv.s /= lensq; nuclear@10: nuclear@10: return inv; nuclear@10: } nuclear@10: nuclear@10: nuclear@10: void Quaternion::set_rotation(const Vector3 &axis, scalar_t angle) nuclear@10: { nuclear@10: scalar_t half_angle = angle / 2.0; nuclear@10: s = cos(half_angle); nuclear@10: v = axis * sin(half_angle); nuclear@10: } nuclear@10: nuclear@10: void Quaternion::rotate(const Vector3 &axis, scalar_t angle) nuclear@10: { nuclear@10: Quaternion q; nuclear@10: scalar_t half_angle = angle / 2.0; nuclear@10: q.s = cos(half_angle); nuclear@10: q.v = axis * sin(half_angle); nuclear@10: nuclear@10: *this *= q; nuclear@10: } nuclear@10: nuclear@10: void Quaternion::rotate(const Quaternion &q) nuclear@10: { nuclear@10: *this = q * *this * q.conjugate(); nuclear@10: } nuclear@10: nuclear@10: Matrix3x3 Quaternion::get_rotation_matrix() const nuclear@10: { nuclear@10: return Matrix3x3( nuclear@10: 1.0 - 2.0 * v.y*v.y - 2.0 * v.z*v.z, 2.0 * v.x * v.y - 2.0 * s * v.z, 2.0 * v.z * v.x + 2.0 * s * v.y, nuclear@10: 2.0 * v.x * v.y + 2.0 * s * v.z, 1.0 - 2.0 * v.x*v.x - 2.0 * v.z*v.z, 2.0 * v.y * v.z - 2.0 * s * v.x, nuclear@10: 2.0 * v.z * v.x - 2.0 * s * v.y, 2.0 * v.y * v.z + 2.0 * s * v.x, 1.0 - 2.0 * v.x*v.x - 2.0 * v.y*v.y); nuclear@10: } nuclear@10: nuclear@10: nuclear@10: /** Spherical linear interpolation (slerp) */ nuclear@10: Quaternion slerp(const Quaternion &quat1, const Quaternion &q2, scalar_t t) nuclear@10: { nuclear@10: Quaternion q1 = quat1; nuclear@10: scalar_t dot = q1.s * q2.s + q1.v.x * q2.v.x + q1.v.y * q2.v.y + q1.v.z * q2.v.z; nuclear@10: nuclear@10: if(dot < 0.0) { nuclear@10: /* make sure we interpolate across the shortest arc */ nuclear@10: q1 = -quat1; nuclear@10: dot = -dot; nuclear@10: } nuclear@10: nuclear@10: /* clamp dot to [-1, 1] in order to avoid domain errors in acos due to nuclear@10: * floating point imprecisions nuclear@10: */ nuclear@10: if(dot < -1.0) dot = -1.0; nuclear@10: if(dot > 1.0) dot = 1.0; nuclear@10: nuclear@10: scalar_t angle = acos(dot); nuclear@10: scalar_t a, b; nuclear@10: nuclear@10: scalar_t sin_angle = sin(angle); nuclear@10: if(fabs(sin_angle) < SMALL_NUMBER) { nuclear@10: /* for very small angles or completely opposite orientations nuclear@10: * use linear interpolation to avoid div/zero (in the first case it makes sense, nuclear@10: * the second case is pretty much undefined anyway I guess ... nuclear@10: */ nuclear@10: a = 1.0f - t; nuclear@10: b = t; nuclear@10: } else { nuclear@10: a = sin((1.0f - t) * angle) / sin_angle; nuclear@10: b = sin(t * angle) / sin_angle; nuclear@10: } nuclear@10: nuclear@10: scalar_t x = q1.v.x * a + q2.v.x * b; nuclear@10: scalar_t y = q1.v.y * a + q2.v.y * b; nuclear@10: scalar_t z = q1.v.z * a + q2.v.z * b; nuclear@10: scalar_t s = q1.s * a + q2.s * b; nuclear@10: nuclear@10: return Quaternion(s, Vector3(x, y, z)); nuclear@10: } nuclear@10: nuclear@10: nuclear@10: /* nuclear@10: std::ostream &operator <<(std::ostream &out, const Quaternion &q) nuclear@10: { nuclear@10: out << "(" << q.s << ", " << q.v << ")"; nuclear@10: return out; nuclear@10: } nuclear@10: */