2012-03-12 15:22:03 +08:00
|
|
|
/*
|
|
|
|
Copyright (c) 2008, Luke Benstead.
|
|
|
|
All rights reserved.
|
|
|
|
|
|
|
|
Redistribution and use in source and binary forms, with or without modification,
|
|
|
|
are permitted provided that the following conditions are met:
|
|
|
|
|
|
|
|
* Redistributions of source code must retain the above copyright notice,
|
|
|
|
this list of conditions and the following disclaimer.
|
|
|
|
* Redistributions in binary form must reproduce the above copyright notice,
|
|
|
|
this list of conditions and the following disclaimer in the documentation
|
|
|
|
and/or other materials provided with the distribution.
|
|
|
|
|
|
|
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
|
|
|
ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
|
|
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
|
|
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
|
|
|
|
ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
|
|
(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
|
|
LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
|
|
|
|
ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
|
|
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
#include <assert.h>
|
|
|
|
#include <memory.h>
|
|
|
|
|
|
|
|
#include "kazmath/utility.h"
|
|
|
|
#include "kazmath/mat3.h"
|
|
|
|
#include "kazmath/vec3.h"
|
|
|
|
#include "kazmath/quaternion.h"
|
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
#ifndef NULL
|
|
|
|
#define NULL ((void *)0)
|
2012-03-12 15:22:03 +08:00
|
|
|
#endif
|
|
|
|
|
|
|
|
///< Returns pOut, sets pOut to the conjugate of pIn
|
|
|
|
kmQuaternion* const kmQuaternionConjugate(kmQuaternion* pOut, const kmQuaternion* pIn)
|
|
|
|
{
|
2012-04-19 14:35:52 +08:00
|
|
|
pOut->x = -pIn->x;
|
|
|
|
pOut->y = -pIn->y;
|
|
|
|
pOut->z = -pIn->z;
|
|
|
|
pOut->w = pIn->w;
|
2012-03-12 15:22:03 +08:00
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
return pOut;
|
2012-03-12 15:22:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
///< Returns the dot product of the 2 quaternions
|
|
|
|
const kmScalar kmQuaternionDot(const kmQuaternion* q1, const kmQuaternion* q2)
|
|
|
|
{
|
2012-04-19 14:35:52 +08:00
|
|
|
// A dot B = B dot A = AtBt + AxBx + AyBy + AzBz
|
2012-03-12 15:22:03 +08:00
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
return (q1->w * q2->w +
|
|
|
|
q1->x * q2->x +
|
|
|
|
q1->y * q2->y +
|
|
|
|
q1->z * q2->z);
|
2012-03-12 15:22:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
///< Returns the exponential of the quaternion
|
|
|
|
kmQuaternion* kmQuaternionExp(kmQuaternion* pOut, const kmQuaternion* pIn)
|
|
|
|
{
|
2012-04-19 14:35:52 +08:00
|
|
|
assert(0);
|
2012-03-12 15:22:03 +08:00
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
return pOut;
|
2012-03-12 15:22:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
///< Makes the passed quaternion an identity quaternion
|
|
|
|
kmQuaternion* kmQuaternionIdentity(kmQuaternion* pOut)
|
|
|
|
{
|
2012-04-19 14:35:52 +08:00
|
|
|
pOut->x = 0.0;
|
|
|
|
pOut->y = 0.0;
|
|
|
|
pOut->z = 0.0;
|
|
|
|
pOut->w = 1.0;
|
2012-03-12 15:22:03 +08:00
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
return pOut;
|
2012-03-12 15:22:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
///< Returns the inverse of the passed Quaternion
|
|
|
|
kmQuaternion* kmQuaternionInverse(kmQuaternion* pOut,
|
2012-04-19 14:35:52 +08:00
|
|
|
const kmQuaternion* pIn)
|
2012-03-12 15:22:03 +08:00
|
|
|
{
|
2012-04-19 14:35:52 +08:00
|
|
|
kmScalar l = kmQuaternionLength(pIn);
|
2012-03-12 15:22:03 +08:00
|
|
|
kmQuaternion tmp;
|
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
if (fabs(l) > kmEpsilon)
|
|
|
|
{
|
|
|
|
pOut->x = 0.0;
|
|
|
|
pOut->y = 0.0;
|
|
|
|
pOut->z = 0.0;
|
|
|
|
pOut->w = 0.0;
|
2012-03-12 15:22:03 +08:00
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
return pOut;
|
|
|
|
}
|
2012-03-12 15:22:03 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
///Get the conjugute and divide by the length
|
|
|
|
kmQuaternionScale(pOut,
|
|
|
|
kmQuaternionConjugate(&tmp, pIn), 1.0f / l);
|
2012-03-12 15:22:03 +08:00
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
return pOut;
|
2012-03-12 15:22:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
///< Returns true if the quaternion is an identity quaternion
|
|
|
|
int kmQuaternionIsIdentity(const kmQuaternion* pIn)
|
|
|
|
{
|
2012-04-19 14:35:52 +08:00
|
|
|
return (pIn->x == 0.0 && pIn->y == 0.0 && pIn->z == 0.0 &&
|
|
|
|
pIn->w == 1.0);
|
2012-03-12 15:22:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
///< Returns the length of the quaternion
|
|
|
|
kmScalar kmQuaternionLength(const kmQuaternion* pIn)
|
|
|
|
{
|
2012-04-19 14:35:52 +08:00
|
|
|
return sqrtf(kmQuaternionLengthSq(pIn));
|
2012-03-12 15:22:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
///< Returns the length of the quaternion squared (prevents a sqrt)
|
|
|
|
kmScalar kmQuaternionLengthSq(const kmQuaternion* pIn)
|
|
|
|
{
|
2012-04-19 14:35:52 +08:00
|
|
|
return pIn->x * pIn->x + pIn->y * pIn->y +
|
|
|
|
pIn->z * pIn->z + pIn->w * pIn->w;
|
2012-03-12 15:22:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
///< Returns the natural logarithm
|
|
|
|
kmQuaternion* kmQuaternionLn(kmQuaternion* pOut,
|
2012-04-19 14:35:52 +08:00
|
|
|
const kmQuaternion* pIn)
|
2012-03-12 15:22:03 +08:00
|
|
|
{
|
2012-04-19 14:35:52 +08:00
|
|
|
/*
|
|
|
|
A unit quaternion, is defined by:
|
|
|
|
Q == (cos(theta), sin(theta) * v) where |v| = 1
|
|
|
|
The natural logarithm of Q is, ln(Q) = (0, theta * v)
|
|
|
|
*/
|
2012-03-12 15:22:03 +08:00
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
assert(0);
|
2012-03-12 15:22:03 +08:00
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
return pOut;
|
2012-03-12 15:22:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
///< Multiplies 2 quaternions together
|
|
|
|
extern
|
|
|
|
kmQuaternion* kmQuaternionMultiply(kmQuaternion* pOut,
|
2012-04-19 14:35:52 +08:00
|
|
|
const kmQuaternion* q1,
|
|
|
|
const kmQuaternion* q2)
|
2012-03-12 15:22:03 +08:00
|
|
|
{
|
2012-04-19 14:35:52 +08:00
|
|
|
pOut->w = q1->w * q2->w - q1->x * q2->x - q1->y * q2->y - q1->z * q2->z;
|
|
|
|
pOut->x = q1->w * q2->x + q1->x * q2->w + q1->y * q2->z - q1->z * q2->y;
|
|
|
|
pOut->y = q1->w * q2->y + q1->y * q2->w + q1->z * q2->x - q1->x * q2->z;
|
|
|
|
pOut->z = q1->w * q2->z + q1->z * q2->w + q1->x * q2->y - q1->y * q2->x;
|
2012-03-12 15:22:03 +08:00
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
return pOut;
|
2012-03-12 15:22:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
///< Normalizes a quaternion
|
|
|
|
kmQuaternion* kmQuaternionNormalize(kmQuaternion* pOut,
|
2012-04-19 14:35:52 +08:00
|
|
|
const kmQuaternion* pIn)
|
2012-03-12 15:22:03 +08:00
|
|
|
{
|
2012-04-19 14:35:52 +08:00
|
|
|
kmScalar length = kmQuaternionLength(pIn);
|
|
|
|
assert(fabs(length) > kmEpsilon);
|
|
|
|
kmQuaternionScale(pOut, pIn, 1.0f / length);
|
2012-03-12 15:22:03 +08:00
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
return pOut;
|
2012-03-12 15:22:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
///< Rotates a quaternion around an axis
|
|
|
|
kmQuaternion* kmQuaternionRotationAxis(kmQuaternion* pOut,
|
2012-04-19 14:35:52 +08:00
|
|
|
const kmVec3* pV,
|
|
|
|
kmScalar angle)
|
2012-03-12 15:22:03 +08:00
|
|
|
{
|
2012-04-19 14:35:52 +08:00
|
|
|
kmScalar rad = angle * 0.5f;
|
|
|
|
kmScalar scale = sinf(rad);
|
2012-03-12 15:22:03 +08:00
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
pOut->w = cosf(rad);
|
|
|
|
pOut->x = pV->x * scale;
|
|
|
|
pOut->y = pV->y * scale;
|
|
|
|
pOut->z = pV->z * scale;
|
2012-03-12 15:22:03 +08:00
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
return pOut;
|
2012-03-12 15:22:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
///< Creates a quaternion from a rotation matrix
|
|
|
|
kmQuaternion* kmQuaternionRotationMatrix(kmQuaternion* pOut,
|
2012-04-19 14:35:52 +08:00
|
|
|
const kmMat3* pIn)
|
2012-03-12 15:22:03 +08:00
|
|
|
{
|
|
|
|
/*
|
|
|
|
Note: The OpenGL matrices are transposed from the description below
|
|
|
|
taken from the Matrix and Quaternion FAQ
|
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
if ( mat[0] > mat[5] && mat[0] > mat[10] ) { // Column 0:
|
2012-03-12 15:22:03 +08:00
|
|
|
S = sqrt( 1.0 + mat[0] - mat[5] - mat[10] ) * 2;
|
|
|
|
X = 0.25 * S;
|
|
|
|
Y = (mat[4] + mat[1] ) / S;
|
|
|
|
Z = (mat[2] + mat[8] ) / S;
|
|
|
|
W = (mat[9] - mat[6] ) / S;
|
2012-04-19 14:35:52 +08:00
|
|
|
} else if ( mat[5] > mat[10] ) { // Column 1:
|
2012-03-12 15:22:03 +08:00
|
|
|
S = sqrt( 1.0 + mat[5] - mat[0] - mat[10] ) * 2;
|
|
|
|
X = (mat[4] + mat[1] ) / S;
|
|
|
|
Y = 0.25 * S;
|
|
|
|
Z = (mat[9] + mat[6] ) / S;
|
|
|
|
W = (mat[2] - mat[8] ) / S;
|
2012-04-19 14:35:52 +08:00
|
|
|
} else { // Column 2:
|
2012-03-12 15:22:03 +08:00
|
|
|
S = sqrt( 1.0 + mat[10] - mat[0] - mat[5] ) * 2;
|
|
|
|
X = (mat[2] + mat[8] ) / S;
|
|
|
|
Y = (mat[9] + mat[6] ) / S;
|
|
|
|
Z = 0.25 * S;
|
|
|
|
W = (mat[4] - mat[1] ) / S;
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
float x, y, z, w;
|
|
|
|
float *pMatrix = NULL;
|
|
|
|
float m4x4[16] = {0};
|
|
|
|
float scale = 0.0f;
|
|
|
|
float diagonal = 0.0f;
|
|
|
|
|
|
|
|
if(!pIn) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* 0 3 6
|
|
|
|
1 4 7
|
|
|
|
2 5 8
|
|
|
|
|
|
|
|
0 1 2 3
|
|
|
|
4 5 6 7
|
|
|
|
8 9 10 11
|
|
|
|
12 13 14 15*/
|
|
|
|
|
|
|
|
m4x4[0] = pIn->mat[0];
|
|
|
|
m4x4[1] = pIn->mat[3];
|
|
|
|
m4x4[2] = pIn->mat[6];
|
|
|
|
m4x4[4] = pIn->mat[1];
|
|
|
|
m4x4[5] = pIn->mat[4];
|
|
|
|
m4x4[6] = pIn->mat[7];
|
|
|
|
m4x4[8] = pIn->mat[2];
|
|
|
|
m4x4[9] = pIn->mat[5];
|
|
|
|
m4x4[10] = pIn->mat[8];
|
|
|
|
m4x4[15] = 1;
|
|
|
|
pMatrix = &m4x4[0];
|
|
|
|
|
|
|
|
diagonal = pMatrix[0] + pMatrix[5] + pMatrix[10] + 1;
|
|
|
|
|
|
|
|
if(diagonal > kmEpsilon) {
|
|
|
|
// Calculate the scale of the diagonal
|
|
|
|
scale = (float)sqrt(diagonal ) * 2;
|
|
|
|
|
|
|
|
// Calculate the x, y, x and w of the quaternion through the respective equation
|
|
|
|
x = ( pMatrix[9] - pMatrix[6] ) / scale;
|
|
|
|
y = ( pMatrix[2] - pMatrix[8] ) / scale;
|
|
|
|
z = ( pMatrix[4] - pMatrix[1] ) / scale;
|
|
|
|
w = 0.25f * scale;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// If the first element of the diagonal is the greatest value
|
|
|
|
if ( pMatrix[0] > pMatrix[5] && pMatrix[0] > pMatrix[10] )
|
|
|
|
{
|
|
|
|
// Find the scale according to the first element, and double that value
|
|
|
|
scale = (float)sqrt( 1.0f + pMatrix[0] - pMatrix[5] - pMatrix[10] ) * 2.0f;
|
|
|
|
|
|
|
|
// Calculate the x, y, x and w of the quaternion through the respective equation
|
|
|
|
x = 0.25f * scale;
|
|
|
|
y = (pMatrix[4] + pMatrix[1] ) / scale;
|
|
|
|
z = (pMatrix[2] + pMatrix[8] ) / scale;
|
|
|
|
w = (pMatrix[9] - pMatrix[6] ) / scale;
|
|
|
|
}
|
|
|
|
// Else if the second element of the diagonal is the greatest value
|
|
|
|
else if (pMatrix[5] > pMatrix[10])
|
|
|
|
{
|
|
|
|
// Find the scale according to the second element, and double that value
|
|
|
|
scale = (float)sqrt( 1.0f + pMatrix[5] - pMatrix[0] - pMatrix[10] ) * 2.0f;
|
|
|
|
|
|
|
|
// Calculate the x, y, x and w of the quaternion through the respective equation
|
|
|
|
x = (pMatrix[4] + pMatrix[1] ) / scale;
|
|
|
|
y = 0.25f * scale;
|
|
|
|
z = (pMatrix[9] + pMatrix[6] ) / scale;
|
|
|
|
w = (pMatrix[2] - pMatrix[8] ) / scale;
|
|
|
|
}
|
|
|
|
// Else the third element of the diagonal is the greatest value
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// Find the scale according to the third element, and double that value
|
|
|
|
scale = (float)sqrt( 1.0f + pMatrix[10] - pMatrix[0] - pMatrix[5] ) * 2.0f;
|
|
|
|
|
|
|
|
// Calculate the x, y, x and w of the quaternion through the respective equation
|
|
|
|
x = (pMatrix[2] + pMatrix[8] ) / scale;
|
|
|
|
y = (pMatrix[9] + pMatrix[6] ) / scale;
|
|
|
|
z = 0.25f * scale;
|
|
|
|
w = (pMatrix[4] - pMatrix[1] ) / scale;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pOut->x = x;
|
|
|
|
pOut->y = y;
|
|
|
|
pOut->z = z;
|
|
|
|
pOut->w = w;
|
|
|
|
|
|
|
|
return pOut;
|
2012-03-12 15:22:03 +08:00
|
|
|
|
|
|
|
#if 0
|
2012-04-19 14:35:52 +08:00
|
|
|
kmScalar T = pIn->mat[0] + pIn->mat[5] + pIn->mat[10];
|
|
|
|
|
|
|
|
if (T > kmEpsilon) {
|
|
|
|
//If the trace is greater than zero we always use this calculation:
|
|
|
|
/* S = sqrt(T) * 2;
|
|
|
|
X = ( mat[9] - mat[6] ) / S;
|
|
|
|
Y = ( mat[2] - mat[8] ) / S;
|
|
|
|
Z = ( mat[4] - mat[1] ) / S;
|
|
|
|
W = 0.25 * S;*/
|
|
|
|
|
|
|
|
/* kmScalar s = sqrtf(T) * 2;
|
|
|
|
pOut->x = (pIn->mat[9] - pIn->mat[6]) / s;
|
|
|
|
pOut->y = (pIn->mat[8] - pIn->mat[2]) / s;
|
|
|
|
pOut->z = (pIn->mat[1] - pIn->mat[4]) / s;
|
|
|
|
pOut->w = 0.25f * s;
|
|
|
|
|
|
|
|
kmQuaternionNormalize(pOut, pOut);
|
|
|
|
return pOut;
|
|
|
|
}
|
|
|
|
|
|
|
|
//Otherwise the calculation depends on which major diagonal element has the greatest value.
|
|
|
|
|
|
|
|
if (pIn->mat[0] > pIn->mat[5] && pIn->mat[0] > pIn->mat[10]) {
|
|
|
|
kmScalar s = sqrtf(1 + pIn->mat[0] - pIn->mat[5] - pIn->mat[10]) * 2;
|
|
|
|
pOut->x = 0.25f * s;
|
|
|
|
pOut->y = (pIn->mat[1] + pIn->mat[4]) / s;
|
|
|
|
pOut->z = (pIn->mat[8] + pIn->mat[2]) / s;
|
|
|
|
pOut->w = (pIn->mat[9] - pIn->mat[6]) / s;
|
|
|
|
}
|
|
|
|
else if (pIn->mat[5] > pIn->mat[10]) {
|
|
|
|
kmScalar s = sqrtf(1 + pIn->mat[5] - pIn->mat[0] - pIn->mat[10]) * 2;
|
|
|
|
pOut->x = (pIn->mat[1] + pIn->mat[4]) / s;
|
|
|
|
pOut->y = 0.25f * s;
|
|
|
|
pOut->z = (pIn->mat[9] + pIn->mat[6]) / s;
|
|
|
|
pOut->w = (pIn->mat[8] - pIn->mat[2]) / s;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
kmScalar s = sqrt(1.0f + pIn->mat[10] - pIn->mat[0] - pIn->mat[5]) * 2.0f;
|
2012-03-12 15:22:03 +08:00
|
|
|
pOut->x = (pIn->mat[8] + pIn->mat[2] ) / s;
|
|
|
|
pOut->y = (pIn->mat[6] + pIn->mat[9] ) / s;
|
|
|
|
pOut->z = 0.25f * s;
|
|
|
|
pOut->w = (pIn->mat[1] - pIn->mat[4] ) / s;
|
2012-04-19 14:35:52 +08:00
|
|
|
}
|
2012-03-12 15:22:03 +08:00
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
kmQuaternionNormalize(pOut, pOut);
|
|
|
|
return pOut;*/
|
2012-03-12 15:22:03 +08:00
|
|
|
#endif // 0
|
|
|
|
}
|
|
|
|
|
|
|
|
///< Create a quaternion from yaw, pitch and roll
|
|
|
|
kmQuaternion* kmQuaternionRotationYawPitchRoll(kmQuaternion* pOut,
|
2012-04-19 14:35:52 +08:00
|
|
|
kmScalar yaw,
|
|
|
|
kmScalar pitch,
|
|
|
|
kmScalar roll)
|
2012-03-12 15:22:03 +08:00
|
|
|
{
|
2012-04-19 14:35:52 +08:00
|
|
|
kmScalar ex, ey, ez; // temp half euler angles
|
|
|
|
kmScalar cr, cp, cy, sr, sp, sy, cpcy, spsy; // temp vars in roll,pitch yaw
|
2012-03-12 15:22:03 +08:00
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
ex = kmDegreesToRadians(pitch) / 2.0f; // convert to rads and half them
|
|
|
|
ey = kmDegreesToRadians(yaw) / 2.0f;
|
|
|
|
ez = kmDegreesToRadians(roll) / 2.0f;
|
2012-03-12 15:22:03 +08:00
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
cr = cosf(ex);
|
|
|
|
cp = cosf(ey);
|
|
|
|
cy = cosf(ez);
|
2012-03-12 15:22:03 +08:00
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
sr = sinf(ex);
|
|
|
|
sp = sinf(ey);
|
|
|
|
sy = sinf(ez);
|
2012-03-12 15:22:03 +08:00
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
cpcy = cp * cy;
|
|
|
|
spsy = sp * sy;
|
2012-03-12 15:22:03 +08:00
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
pOut->w = cr * cpcy + sr * spsy;
|
2012-03-12 15:22:03 +08:00
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
pOut->x = sr * cpcy - cr * spsy;
|
|
|
|
pOut->y = cr * sp * cy + sr * cp * sy;
|
|
|
|
pOut->z = cr * cp * sy - sr * sp * cy;
|
2012-03-12 15:22:03 +08:00
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
kmQuaternionNormalize(pOut, pOut);
|
2012-03-12 15:22:03 +08:00
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
return pOut;
|
2012-03-12 15:22:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
///< Interpolate between 2 quaternions
|
|
|
|
kmQuaternion* kmQuaternionSlerp(kmQuaternion* pOut,
|
2012-04-19 14:35:52 +08:00
|
|
|
const kmQuaternion* q1,
|
|
|
|
const kmQuaternion* q2,
|
|
|
|
kmScalar t)
|
2012-03-12 15:22:03 +08:00
|
|
|
{
|
|
|
|
|
|
|
|
/*float CosTheta = Q0.DotProd(Q1);
|
|
|
|
float Theta = acosf(CosTheta);
|
|
|
|
float SinTheta = sqrtf(1.0f-CosTheta*CosTheta);
|
|
|
|
|
|
|
|
float Sin_T_Theta = sinf(T*Theta)/SinTheta;
|
|
|
|
float Sin_OneMinusT_Theta = sinf((1.0f-T)*Theta)/SinTheta;
|
|
|
|
|
|
|
|
Quaternion Result = Q0*Sin_OneMinusT_Theta;
|
|
|
|
Result += (Q1*Sin_T_Theta);
|
|
|
|
|
|
|
|
return Result;*/
|
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
if (q1->x == q2->x &&
|
|
|
|
q1->y == q2->y &&
|
|
|
|
q1->z == q2->z &&
|
|
|
|
q1->w == q2->w) {
|
|
|
|
|
|
|
|
pOut->x = q1->x;
|
|
|
|
pOut->y = q1->y;
|
|
|
|
pOut->z = q1->z;
|
|
|
|
pOut->w = q1->w;
|
|
|
|
|
|
|
|
return pOut;
|
|
|
|
}
|
|
|
|
{
|
|
|
|
kmScalar ct = kmQuaternionDot(q1, q2);
|
|
|
|
kmScalar theta = acosf(ct);
|
|
|
|
kmScalar st = sqrtf(1.0 - kmSQR(ct));
|
|
|
|
|
|
|
|
kmScalar stt = sinf(t * theta) / st;
|
|
|
|
kmScalar somt = sinf((1.0 - t) * theta) / st;
|
|
|
|
|
|
|
|
kmQuaternion temp, temp2;
|
|
|
|
kmQuaternionScale(&temp, q1, somt);
|
|
|
|
kmQuaternionScale(&temp2, q2, stt);
|
|
|
|
kmQuaternionAdd(pOut, &temp, &temp2);
|
|
|
|
}
|
|
|
|
return pOut;
|
2012-03-12 15:22:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
///< Get the axis and angle of rotation from a quaternion
|
|
|
|
void kmQuaternionToAxisAngle(const kmQuaternion* pIn,
|
2012-04-19 14:35:52 +08:00
|
|
|
kmVec3* pAxis,
|
|
|
|
kmScalar* pAngle)
|
2012-03-12 15:22:03 +08:00
|
|
|
{
|
2012-04-19 14:35:52 +08:00
|
|
|
kmScalar tempAngle; // temp angle
|
|
|
|
kmScalar scale; // temp vars
|
|
|
|
|
|
|
|
tempAngle = acosf(pIn->w);
|
|
|
|
scale = sqrtf(kmSQR(pIn->x) + kmSQR(pIn->y) + kmSQR(pIn->z));
|
|
|
|
|
|
|
|
if (((scale > -kmEpsilon) && scale < kmEpsilon)
|
|
|
|
|| (scale < 2*kmPI + kmEpsilon && scale > 2*kmPI - kmEpsilon)) // angle is 0 or 360 so just simply set axis to 0,0,1 with angle 0
|
|
|
|
{
|
|
|
|
*pAngle = 0.0f;
|
|
|
|
|
|
|
|
pAxis->x = 0.0f;
|
|
|
|
pAxis->y = 0.0f;
|
|
|
|
pAxis->z = 1.0f;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
*pAngle = tempAngle * 2.0f; // angle in radians
|
|
|
|
|
|
|
|
pAxis->x = pIn->x / scale;
|
|
|
|
pAxis->y = pIn->y / scale;
|
|
|
|
pAxis->z = pIn->z / scale;
|
|
|
|
kmVec3Normalize(pAxis, pAxis);
|
|
|
|
}
|
2012-03-12 15:22:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
kmQuaternion* kmQuaternionScale(kmQuaternion* pOut,
|
2012-04-19 14:35:52 +08:00
|
|
|
const kmQuaternion* pIn,
|
|
|
|
kmScalar s)
|
2012-03-12 15:22:03 +08:00
|
|
|
{
|
2012-04-19 14:35:52 +08:00
|
|
|
pOut->x = pIn->x * s;
|
|
|
|
pOut->y = pIn->y * s;
|
|
|
|
pOut->z = pIn->z * s;
|
|
|
|
pOut->w = pIn->w * s;
|
2012-03-12 15:22:03 +08:00
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
return pOut;
|
2012-03-12 15:22:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
kmQuaternion* kmQuaternionAssign(kmQuaternion* pOut, const kmQuaternion* pIn)
|
|
|
|
{
|
2012-04-19 14:35:52 +08:00
|
|
|
memcpy(pOut, pIn, sizeof(float) * 4);
|
2012-03-12 15:22:03 +08:00
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
return pOut;
|
2012-03-12 15:22:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
kmQuaternion* kmQuaternionAdd(kmQuaternion* pOut, const kmQuaternion* pQ1, const kmQuaternion* pQ2)
|
|
|
|
{
|
2012-04-19 14:35:52 +08:00
|
|
|
pOut->x = pQ1->x + pQ2->x;
|
|
|
|
pOut->y = pQ1->y + pQ2->y;
|
|
|
|
pOut->z = pQ1->z + pQ2->z;
|
|
|
|
pOut->w = pQ1->w + pQ2->w;
|
2012-03-12 15:22:03 +08:00
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
return pOut;
|
2012-03-12 15:22:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/** Adapted from the OGRE engine!
|
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
Gets the shortest arc quaternion to rotate this vector to the destination
|
|
|
|
vector.
|
2012-03-12 15:22:03 +08:00
|
|
|
@remarks
|
2012-04-19 14:35:52 +08:00
|
|
|
If you call this with a dest vector that is close to the inverse
|
|
|
|
of this vector, we will rotate 180 degrees around the 'fallbackAxis'
|
|
|
|
(if specified, or a generated axis if not) since in this case
|
|
|
|
ANY axis of rotation is valid.
|
2012-03-12 15:22:03 +08:00
|
|
|
*/
|
|
|
|
|
|
|
|
kmQuaternion* kmQuaternionRotationBetweenVec3(kmQuaternion* pOut, const kmVec3* vec1, const kmVec3* vec2, const kmVec3* fallback) {
|
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
kmVec3 v1, v2;
|
2012-03-12 15:22:03 +08:00
|
|
|
kmScalar a;
|
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
kmVec3Assign(&v1, vec1);
|
|
|
|
kmVec3Assign(&v2, vec2);
|
2012-03-12 15:22:03 +08:00
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
kmVec3Normalize(&v1, &v1);
|
|
|
|
kmVec3Normalize(&v2, &v2);
|
2012-03-12 15:22:03 +08:00
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
a = kmVec3Dot(&v1, &v2);
|
2012-03-12 15:22:03 +08:00
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
if (a >= 1.0) {
|
|
|
|
kmQuaternionIdentity(pOut);
|
|
|
|
return pOut;
|
|
|
|
}
|
2012-03-12 15:22:03 +08:00
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
if (a < (1e-6f - 1.0f)) {
|
|
|
|
if (fabs(kmVec3LengthSq(fallback)) < kmEpsilon) {
|
|
|
|
kmQuaternionRotationAxis(pOut, fallback, kmPI);
|
|
|
|
} else {
|
|
|
|
kmVec3 axis;
|
|
|
|
kmVec3 X;
|
|
|
|
X.x = 1.0;
|
|
|
|
X.y = 0.0;
|
|
|
|
X.z = 0.0;
|
2012-03-12 15:22:03 +08:00
|
|
|
|
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
kmVec3Cross(&axis, &X, vec1);
|
2012-03-12 15:22:03 +08:00
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
//If axis is zero
|
|
|
|
if (fabs(kmVec3LengthSq(&axis)) < kmEpsilon) {
|
|
|
|
kmVec3 Y;
|
|
|
|
Y.x = 0.0;
|
|
|
|
Y.y = 1.0;
|
|
|
|
Y.z = 0.0;
|
2012-03-12 15:22:03 +08:00
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
kmVec3Cross(&axis, &Y, vec1);
|
|
|
|
}
|
2012-03-12 15:22:03 +08:00
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
kmVec3Normalize(&axis, &axis);
|
2012-03-12 15:22:03 +08:00
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
kmQuaternionRotationAxis(pOut, &axis, kmPI);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
kmScalar s = sqrtf((1+a) * 2);
|
|
|
|
kmScalar invs = 1 / s;
|
2012-03-12 15:22:03 +08:00
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
kmVec3 c;
|
|
|
|
kmVec3Cross(&c, &v1, &v2);
|
2012-03-12 15:22:03 +08:00
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
pOut->x = c.x * invs;
|
|
|
|
pOut->y = c.y * invs;
|
2012-03-12 15:22:03 +08:00
|
|
|
pOut->z = c.z * invs;
|
|
|
|
pOut->w = s * 0.5f;
|
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
kmQuaternionNormalize(pOut, pOut);
|
|
|
|
}
|
2012-03-12 15:22:03 +08:00
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
return pOut;
|
2012-03-12 15:22:03 +08:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
kmVec3* kmQuaternionMultiplyVec3(kmVec3* pOut, const kmQuaternion* q, const kmVec3* v) {
|
2012-04-19 14:35:52 +08:00
|
|
|
kmVec3 uv, uuv, qvec;
|
2012-03-12 15:22:03 +08:00
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
qvec.x = q->x;
|
|
|
|
qvec.y = q->y;
|
|
|
|
qvec.z = q->z;
|
2012-03-12 15:22:03 +08:00
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
kmVec3Cross(&uv, &qvec, v);
|
|
|
|
kmVec3Cross(&uuv, &qvec, &uv);
|
2012-03-12 15:22:03 +08:00
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
kmVec3Scale(&uv, &uv, (2.0f * q->w));
|
|
|
|
kmVec3Scale(&uuv, &uuv, 2.0f);
|
2012-03-12 15:22:03 +08:00
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
kmVec3Add(pOut, v, &uv);
|
|
|
|
kmVec3Add(pOut, pOut, &uuv);
|
2012-03-12 15:22:03 +08:00
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
return pOut;
|
2012-03-12 15:22:03 +08:00
|
|
|
}
|
|
|
|
|