261 lines
		
	
	
		
			7.0 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
		
		
			
		
	
	
			261 lines
		
	
	
		
			7.0 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
|   | /******************************************************************************/ | |||
|  | /* | |||
|  |   Project   - MudBun | |||
|  |   Publisher - Long Bunny Labs | |||
|  |               http://LongBunnyLabs.com | |||
|  |   Author    - Ming-Lun "Allen" Chou | |||
|  |               http://AllenChou.net | |||
|  | */ | |||
|  | /******************************************************************************/ | |||
|  | 
 | |||
|  | using UnityEngine; | |||
|  | 
 | |||
|  | namespace MudBun | |||
|  | { | |||
|  |   public class QuaternionUtil | |||
|  |   { | |||
|  |     // basic stuff | |||
|  |     // ------------------------------------------------------------------------ | |||
|  | 
 | |||
|  |     public static float Magnitude(Quaternion q) | |||
|  |     { | |||
|  |       return Mathf.Sqrt(q.x * q.x + q.y * q.y + q.z * q.z + q.w * q.w); | |||
|  |     } | |||
|  | 
 | |||
|  |     public static float MagnitudeSqr(Quaternion q) | |||
|  |     { | |||
|  |       return q.x * q.x + q.y * q.y + q.z * q.z + q.w * q.w; | |||
|  |     } | |||
|  | 
 | |||
|  |     public static Quaternion Normalize(Quaternion q) | |||
|  |     { | |||
|  |       float magInv = 1.0f / Magnitude(q); | |||
|  |       return new Quaternion(magInv * q.x, magInv * q.y, magInv * q.z, magInv * q.w); | |||
|  |     } | |||
|  | 
 | |||
|  |     public static Quaternion AngularVector(Vector3 v) | |||
|  |     { | |||
|  |       float len = v.magnitude; | |||
|  |       if (len < MathUtil.Epsilon) | |||
|  |         return Quaternion.identity; | |||
|  | 
 | |||
|  |       v /= len; | |||
|  | 
 | |||
|  |       float h = 0.5f * len; | |||
|  |       float s = Mathf.Sin(h); | |||
|  |       float c = Mathf.Cos(h); | |||
|  | 
 | |||
|  |       return new Quaternion(s * v.x, s * v.y, s * v.z, c); | |||
|  |     } | |||
|  | 
 | |||
|  |     // axis must be normalized | |||
|  |     public static Quaternion AxisAngle(Vector3 axis, float angle) | |||
|  |     { | |||
|  |       float h = 0.5f * angle; | |||
|  |       float s = Mathf.Sin(h); | |||
|  |       float c = Mathf.Cos(h); | |||
|  | 
 | |||
|  |       return new Quaternion(s * axis.x, s * axis.y, s * axis.z, c); | |||
|  |     } | |||
|  | 
 | |||
|  |     public static Vector3 GetAxis(Quaternion q) | |||
|  |     { | |||
|  |       Vector3 v = new Vector3(q.x, q.y, q.z); | |||
|  |       float len = v.magnitude; | |||
|  |       if (len < MathUtil.Epsilon) | |||
|  |         return Vector3.left; | |||
|  | 
 | |||
|  |       return v / len; | |||
|  |     } | |||
|  | 
 | |||
|  |     public static float GetAngle(Quaternion q) | |||
|  |     { | |||
|  |       return 2.0f * Mathf.Acos(Mathf.Clamp(q.w, -1.0f, 1.0f)); | |||
|  |     } | |||
|  | 
 | |||
|  |     public static Quaternion Pow(Quaternion q, float exp) | |||
|  |     { | |||
|  |       Vector3 axis = GetAxis(q); | |||
|  |       float angle = GetAngle(q); | |||
|  |       return AxisAngle(axis, angle * exp); | |||
|  |     } | |||
|  | 
 | |||
|  |     // v: derivative of q | |||
|  |     public static Quaternion Integrate(Quaternion q, Quaternion v, float dt) | |||
|  |     { | |||
|  |       return Pow(v, dt) * q; | |||
|  |     } | |||
|  | 
 | |||
|  |     // omega: angular velocity (direction is axis, magnitude is angle) | |||
|  |     // https://www.ashwinnarayan.com/post/how-to-integrate-quaternions/ | |||
|  |     // https://gafferongames.com/post/physics_in_3d/ | |||
|  |     public static Quaternion Integrate(Quaternion q, Vector3 omega, float dt) | |||
|  |     { | |||
|  |       omega *= 0.5f; | |||
|  |       Quaternion p = (new Quaternion(omega.x, omega.y, omega.z, 0.0f)) * q; | |||
|  |       return Normalize(new Quaternion(q.x + p.x * dt, q.y + p.y * dt, q.z + p.z * dt, q.w + p.w * dt)); | |||
|  |     } | |||
|  | 
 | |||
|  |     public static Vector4 ToVector4(Quaternion q) | |||
|  |     { | |||
|  |       return new Vector4(q.x, q.y, q.z, q.w); | |||
|  |     } | |||
|  | 
 | |||
|  |     public static Quaternion FromVector4(Vector4 v, bool normalize = true) | |||
|  |     { | |||
|  |       if (normalize) | |||
|  |       { | |||
|  |         float magSqr = v.sqrMagnitude; | |||
|  |         if (magSqr < MathUtil.Epsilon) | |||
|  |           return Quaternion.identity; | |||
|  | 
 | |||
|  |         v /= Mathf.Sqrt(magSqr); | |||
|  |       } | |||
|  | 
 | |||
|  |       return new Quaternion(v.x, v.y, v.z, v.w); | |||
|  |     } | |||
|  | 
 | |||
|  |     // ------------------------------------------------------------------------ | |||
|  |     // end: basic stuff | |||
|  | 
 | |||
|  | 
 | |||
|  |     // swing-twist decomposition & interpolation | |||
|  |     // ------------------------------------------------------------------------ | |||
|  | 
 | |||
|  |     public static void DecomposeSwingTwist | |||
|  |     ( | |||
|  |       Quaternion q,  | |||
|  |       Vector3 twistAxis,  | |||
|  |       out Quaternion swing,  | |||
|  |       out Quaternion twist | |||
|  |     ) | |||
|  |     { | |||
|  |       Vector3 r = new Vector3(q.x, q.y, q.z); // (rotaiton axis) * cos(angle / 2) | |||
|  | 
 | |||
|  |       // singularity: rotation by 180 degree | |||
|  |       if (r.sqrMagnitude < MathUtil.Epsilon) | |||
|  |       { | |||
|  |         Vector3 rotatedTwistAxis = q * twistAxis; | |||
|  |         Vector3 swingAxis = Vector3.Cross(twistAxis, rotatedTwistAxis); | |||
|  | 
 | |||
|  |         if (swingAxis.sqrMagnitude > MathUtil.Epsilon) | |||
|  |         { | |||
|  |           float swingAngle = Vector3.Angle(twistAxis, rotatedTwistAxis); | |||
|  |           swing = Quaternion.AngleAxis(swingAngle, swingAxis); | |||
|  |         } | |||
|  |         else | |||
|  |         { | |||
|  |           // more singularity: rotation axis parallel to twist axis | |||
|  |           swing = Quaternion.identity; // no swing | |||
|  |         } | |||
|  | 
 | |||
|  |         // always twist 180 degree on singularity | |||
|  |         twist = Quaternion.AngleAxis(180.0f, twistAxis); | |||
|  |         return; | |||
|  |       } | |||
|  | 
 | |||
|  |       // formula & proof:  | |||
|  |       // http://www.euclideanspace.com/maths/geometry/rotations/for/decomposition/ | |||
|  |       Vector3 p = Vector3.Project(r, twistAxis); | |||
|  |       twist = new Quaternion(p.x, p.y, p.z, q.w); | |||
|  |       twist = Normalize(twist); | |||
|  |       swing = q * Quaternion.Inverse(twist); | |||
|  |     } | |||
|  | 
 | |||
|  |     public enum SterpMode | |||
|  |     { | |||
|  |       // non-constant angular velocity, faster | |||
|  |       // use if interpolating across small angles or constant angular velocity is not important | |||
|  |       Nlerp, | |||
|  | 
 | |||
|  |       // constant angular velocity, slower | |||
|  |       // use if interpolating across large angles and constant angular velocity is important | |||
|  |       Slerp,  | |||
|  |     }; | |||
|  | 
 | |||
|  |     // same swing & twist parameters | |||
|  |     public static Quaternion Sterp | |||
|  |     ( | |||
|  |       Quaternion a,  | |||
|  |       Quaternion b,  | |||
|  |       Vector3 twistAxis,  | |||
|  |       float t,  | |||
|  |       SterpMode mode = SterpMode.Slerp | |||
|  |     ) | |||
|  |     { | |||
|  |       Quaternion swing; | |||
|  |       Quaternion twist; | |||
|  |       return Sterp(a, b, twistAxis, t, out swing, out twist, mode); | |||
|  |     } | |||
|  | 
 | |||
|  |     // same swing & twist parameters with individual interpolated swing & twist outputs | |||
|  |     public static Quaternion Sterp | |||
|  |     ( | |||
|  |       Quaternion a,  | |||
|  |       Quaternion b,  | |||
|  |       Vector3 twistAxis,  | |||
|  |       float t,  | |||
|  |       out Quaternion swing,  | |||
|  |       out Quaternion twist,  | |||
|  |       SterpMode mode = SterpMode.Slerp | |||
|  |     ) | |||
|  |     { | |||
|  |       return Sterp(a, b, twistAxis, t, t, out swing, out twist, mode); | |||
|  |     } | |||
|  | 
 | |||
|  |     // different swing & twist parameters | |||
|  |     public static Quaternion Sterp | |||
|  |     ( | |||
|  |       Quaternion a,  | |||
|  |       Quaternion b,  | |||
|  |       Vector3 twistAxis,  | |||
|  |       float tSwing,  | |||
|  |       float tTwist,  | |||
|  |       SterpMode mode = SterpMode.Slerp | |||
|  |     ) | |||
|  |     { | |||
|  |       Quaternion swing; | |||
|  |       Quaternion twist; | |||
|  |       return Sterp(a, b, twistAxis, tSwing, tTwist, out swing, out twist, mode); | |||
|  |     } | |||
|  | 
 | |||
|  |     // master sterp function | |||
|  |     public static Quaternion Sterp | |||
|  |     ( | |||
|  |       Quaternion a,  | |||
|  |       Quaternion b,  | |||
|  |       Vector3 twistAxis,  | |||
|  |       float tSwing,  | |||
|  |       float tTwist,  | |||
|  |       out Quaternion swing,  | |||
|  |       out Quaternion twist,  | |||
|  |       SterpMode mode | |||
|  |     ) | |||
|  |     { | |||
|  |       Quaternion q = b * Quaternion.Inverse(a); | |||
|  |       Quaternion swingFull; | |||
|  |       Quaternion twistFull; | |||
|  |       QuaternionUtil.DecomposeSwingTwist(q, twistAxis, out swingFull, out twistFull); | |||
|  | 
 | |||
|  |       switch (mode) | |||
|  |       { | |||
|  |         default: | |||
|  |         case SterpMode.Nlerp: | |||
|  |           swing = Quaternion.Lerp(Quaternion.identity, swingFull, tSwing); | |||
|  |           twist = Quaternion.Lerp(Quaternion.identity, twistFull, tTwist); | |||
|  |           break; | |||
|  |         case SterpMode.Slerp: | |||
|  |           swing = Quaternion.Slerp(Quaternion.identity, swingFull, tSwing); | |||
|  |           twist = Quaternion.Slerp(Quaternion.identity, twistFull, tTwist); | |||
|  |           break; | |||
|  |       } | |||
|  | 
 | |||
|  |       return twist * swing; | |||
|  |     } | |||
|  | 
 | |||
|  |     // ------------------------------------------------------------------------ | |||
|  |     // end: swing-twist decomposition & interpolation | |||
|  |   } | |||
|  | } | |||
|  | 
 |