From 29805391bdda8e2f99a792c9fc29dd0eea0e05a1 Mon Sep 17 00:00:00 2001 From: Christophe Riccio Date: Sat, 22 Dec 2012 00:32:07 +0100 Subject: [PATCH] Added tests for slerp function --- glm/gtc/epsilon.hpp | 20 ++++++------ glm/gtc/epsilon.inl | 30 ------------------ glm/gtc/quaternion.inl | 4 +-- readme.txt | 1 + test/gtc/gtc_quaternion.cpp | 62 +++++++++++++++++++++++++++++++++++++ 5 files changed, 75 insertions(+), 42 deletions(-) diff --git a/glm/gtc/epsilon.hpp b/glm/gtc/epsilon.hpp index aa0d318c..eab8336d 100644 --- a/glm/gtc/epsilon.hpp +++ b/glm/gtc/epsilon.hpp @@ -57,11 +57,11 @@ namespace glm /// Returns the component-wise compare of |x - y| < epsilon. /// @see gtc_epsilon template - typename genType::boolType epsilonEqual( - genType const & x, - genType const & y, - typename genType::value_type const & epsilon); - + typename genType::boolType epsilonEqual( + genType const & x, + genType const & y, + typename genType::value_type const & epsilon); + /// Returns the component-wise compare of |x - y| < epsilon. /// @see gtc_epsilon template @@ -73,11 +73,11 @@ namespace glm /// Returns the component-wise compare of |x - y| < epsilon. /// @see gtc_epsilon template - typename genType::boolType epsilonNotEqual( - genType const & x, - genType const & y, - typename genType::value_type const & epsilon); - + typename genType::boolType epsilonNotEqual( + genType const & x, + genType const & y, + typename genType::value_type const & epsilon); + /// Returns the component-wise compare of |x - y| >= epsilon. /// @see gtc_epsilon template diff --git a/glm/gtc/epsilon.inl b/glm/gtc/epsilon.inl index 754d1413..6faef711 100644 --- a/glm/gtc/epsilon.inl +++ b/glm/gtc/epsilon.inl @@ -269,21 +269,6 @@ namespace glm abs(x.w - y.w) < epsilon); } - template - GLM_FUNC_QUALIFIER detail::tvec4 epsilonEqual - ( - detail::tquat const & x, - detail::tquat const & y, - detail::tquat const & epsilon - ) - { - return detail::tvec4( - abs(x.x - y.x) < epsilon.x, - abs(x.y - y.y) < epsilon.y, - abs(x.z - y.z) < epsilon.z, - abs(x.w - y.w) < epsilon.w); - } - template GLM_FUNC_QUALIFIER detail::tvec4 epsilonNotEqual ( @@ -298,19 +283,4 @@ namespace glm abs(x.z - y.z) >= epsilon, abs(x.w - y.w) >= epsilon); } - - template - GLM_FUNC_QUALIFIER detail::tvec4 epsilonNotEqual - ( - detail::tquat const & x, - detail::tquat const & y, - detail::tquat const & epsilon - ) - { - return detail::tvec4( - abs(x.x - y.x) >= epsilon.x, - abs(x.y - y.y) >= epsilon.y, - abs(x.z - y.z) >= epsilon.z, - abs(x.w - y.w) >= epsilon.w); - } }//namespace glm diff --git a/glm/gtc/quaternion.inl b/glm/gtc/quaternion.inl index 36385aae..1906a16f 100644 --- a/glm/gtc/quaternion.inl +++ b/glm/gtc/quaternion.inl @@ -468,7 +468,7 @@ namespace detail { // Essential Mathematics, page 467 T angle = acos(cosTheta); - return (sin((T(1) - a) * angle) * x + sin(a * angle) * z) / sin(angle); + return (sin((T(1) - a) * angle) * x + sin(a * angle) * y) / sin(angle); } } @@ -658,7 +658,7 @@ namespace detail typename detail::tquat::value_type fourYSquaredMinus1 = m[1][1] - m[0][0] - m[2][2]; typename detail::tquat::value_type fourZSquaredMinus1 = m[2][2] - m[0][0] - m[1][1]; typename detail::tquat::value_type fourWSquaredMinus1 = m[0][0] + m[1][1] + m[2][2]; - + int biggestIndex = 0; typename detail::tquat::value_type fourBiggestSquaredMinus1 = fourWSquaredMinus1; if(fourXSquaredMinus1 > fourBiggestSquaredMinus1) diff --git a/readme.txt b/readme.txt index 38817e59..38d35523 100644 --- a/readme.txt +++ b/readme.txt @@ -47,6 +47,7 @@ GLM 0.9.4.1: 2012-12-21 - Fixed quat slerp using mix function when cosTheta close to 1 - Improved fvec4SIMD and fmat4x4SIMD implementations - Fixed assert messages +- Added slerp and lerp quaternion functions and tests ================================================================================ GLM 0.9.4.0: 2012-11-18 diff --git a/test/gtc/gtc_quaternion.cpp b/test/gtc/gtc_quaternion.cpp index 1de749b9..66f39635 100644 --- a/test/gtc/gtc_quaternion.cpp +++ b/test/gtc/gtc_quaternion.cpp @@ -144,6 +144,67 @@ int test_quat_euler() return Error; } +int test_quat_slerp() +{ + int Error(0); + + float const Epsilon = 0.0001f;//glm::epsilon(); + + float sqrt2 = sqrt(2.0f)/2.0f; + glm::quat id; + glm::quat Y90rot(sqrt2, 0.0f, sqrt2, 0.0f); + glm::quat Y180rot(0.0f, 0.0f, 1.0f, 0.0f); + + // Testing a == 0 + // Must be id + glm::quat id2 = glm::slerp(id, Y90rot, 0.0f); + Error += glm::all(glm::epsilonEqual(id, id2, Epsilon)) ? 0 : 1; + + // Testing a == 1 + // Must be 90° rotation on Y : 0 0.7 0 0.7 + glm::quat Y90rot2 = glm::slerp(id, Y90rot, 1.0f); + Error += glm::all(glm::epsilonEqual(Y90rot, Y90rot2, Epsilon)) ? 0 : 1; + + // Testing standard, easy case + // Must be 45° rotation on Y : 0 0.38 0 0.92 + glm::quat Y45rot1 = glm::slerp(id, Y90rot, 0.5f); + + // Testing reverse case + // Must be 45° rotation on Y : 0 0.38 0 0.92 + glm::quat Ym45rot2 = glm::slerp(Y90rot, id, 0.5f); + + // Testing against full circle around the sphere instead of shortest path + // Must be 45° rotation on Y + // certainly not a 135° rotation + glm::quat Y45rot3 = glm::slerp(id , -Y90rot, 0.5f); + float Y45angle3 = glm::angle(Y45rot3); + Error += glm::epsilonEqual(Y45angle3, 45.f, Epsilon) ? 0 : 1; + Error += glm::all(glm::epsilonEqual(Ym45rot2, Y45rot3, Epsilon)) ? 0 : 1; + + // Same, but inverted + // Must also be 45° rotation on Y : 0 0.38 0 0.92 + // -0 -0.38 -0 -0.92 is ok too + glm::quat Y45rot4 = glm::slerp(-Y90rot, id, 0.5f); + Error += glm::all(glm::epsilonEqual(Ym45rot2, -Y45rot4, Epsilon)) ? 0 : 1; + + // Testing q1 = q2 + // Must be 90° rotation on Y : 0 0.7 0 0.7 + glm::quat Y90rot3 = glm::slerp(Y90rot, Y90rot, 0.5f); + Error += glm::all(glm::epsilonEqual(Y90rot, Y90rot3, Epsilon)) ? 0 : 1; + + // Testing 180° rotation + // Must be 90° rotation on almost any axis that is on the XZ plane + glm::quat XZ90rot = glm::slerp(id, -Y90rot, 0.5f); + float XZ90angle = glm::angle(XZ90rot); // Must be PI/4 = 0.78; + Error += glm::epsilonEqual(XZ90angle, 45.f, Epsilon) ? 0 : 1; + + // Testing almost equal quaternions (this test should pass through the linear interpolation) + // Must be 0 0.00X 0 0.99999 + glm::quat almostid = glm::slerp(id, glm::angleAxis(0.1f, 0.0f, 1.0f, 0.0f), 0.5f); + + return Error; +} + int test_quat_type() { glm::quat A; @@ -163,6 +224,7 @@ int main() Error += test_quat_mix(); Error += test_quat_normalize(); Error += test_quat_euler(); + Error += test_quat_slerp(); return Error; }