#region --- License ---
/*
Copyright (c) 2006 - 2008 The Open Toolkit library.
Permission is hereby granted, free of charge, to any person obtaining a copy of
this software and associated documentation files (the "Software"), to deal in
the Software without restriction, including without limitation the rights to
use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies
of the Software, and to permit persons to whom the Software is furnished to do
so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
SOFTWARE.
*/
#endregion
using System;
using System.Runtime.InteropServices;
using System.ComponentModel;
using System.Xml.Serialization;
namespace OpenTK
{
///
/// Represents a double-precision Quaternion.
///
[Serializable]
[StructLayout(LayoutKind.Sequential)]
public struct Quaterniond : IEquatable
{
#region Fields
Vector3d xyz;
double w;
#endregion
#region Constructors
///
/// Construct a new Quaterniond from vector and w components
///
/// The vector part
/// The w part
public Quaterniond(Vector3d v, double w)
{
this.xyz = v;
this.w = w;
}
///
/// Construct a new Quaterniond
///
/// The x component
/// The y component
/// The z component
/// The w component
public Quaterniond(double x, double y, double z, double w)
: this(new Vector3d(x, y, z), w)
{ }
///
/// Construct a new Quaterniond from given Euler angles
///
/// The pitch (attitude), rotation around X axis
/// The yaw (heading), rotation around Y axis
/// The roll (bank), rotation around Z axis
public Quaterniond(double pitch, double yaw, double roll)
{
yaw *= 0.5;
pitch *= 0.5;
roll *= 0.5;
double c1 = Math.Cos(yaw);
double c2 = Math.Cos(pitch);
double c3 = Math.Cos(roll);
double s1 = Math.Sin(yaw);
double s2 = Math.Sin(pitch);
double s3 = Math.Sin(roll);
this.w = c1 * c2 * c3 - s1 * s2 * s3;
this.xyz.X = s1 * s2 * c3 + c1 * c2 * s3;
this.xyz.Y = s1 * c2 * c3 + c1 * s2 * s3;
this.xyz.Z = c1 * s2 * c3 - s1 * c2 * s3;
}
///
/// Construct a new Quaterniond from given Euler angles
///
/// The euler angles as a Vector3d
public Quaterniond(Vector3d eulerAngles)
:this(eulerAngles.X, eulerAngles.Y, eulerAngles.Z)
{ }
#endregion
#region Public Members
#region Properties
#pragma warning disable 3005 // Identifier differing only in case is not CLS-compliant, compiler bug in Mono 3.4.0
///
/// Gets or sets an OpenTK.Vector3d with the X, Y and Z components of this instance.
///
[Obsolete("Use Xyz property instead.")]
[CLSCompliant(false)]
[EditorBrowsable(EditorBrowsableState.Never)]
[XmlIgnore]
public Vector3d XYZ { get { return Xyz; } set { Xyz = value; } }
///
/// Gets or sets an OpenTK.Vector3d with the X, Y and Z components of this instance.
///
public Vector3d Xyz { get { return xyz; } set { xyz = value; } }
#pragma warning restore 3005
///
/// Gets or sets the X component of this instance.
///
[XmlIgnore]
public double X { get { return xyz.X; } set { xyz.X = value; } }
///
/// Gets or sets the Y component of this instance.
///
[XmlIgnore]
public double Y { get { return xyz.Y; } set { xyz.Y = value; } }
///
/// Gets or sets the Z component of this instance.
///
[XmlIgnore]
public double Z { get { return xyz.Z; } set { xyz.Z = value; } }
///
/// Gets or sets the W component of this instance.
///
public double W { get { return w; } set { w = value; } }
#endregion
#region Instance
#region ToAxisAngle
///
/// Convert the current quaternion to axis angle representation
///
/// The resultant axis
/// The resultant angle
public void ToAxisAngle(out Vector3d axis, out double angle)
{
Vector4d result = ToAxisAngle();
axis = result.Xyz;
angle = result.W;
}
///
/// Convert this instance to an axis-angle representation.
///
/// A Vector4 that is the axis-angle representation of this quaternion.
public Vector4d ToAxisAngle()
{
Quaterniond q = this;
if (Math.Abs(q.W) > 1.0f)
q.Normalize();
Vector4d result = new Vector4d();
result.W = 2.0f * (float)System.Math.Acos(q.W); // angle
float den = (float)System.Math.Sqrt(1.0 - q.W * q.W);
if (den > 0.0001f)
{
result.Xyz = q.Xyz / den;
}
else
{
// This occurs when the angle is zero.
// Not a problem: just set an arbitrary normalized axis.
result.Xyz = Vector3d.UnitX;
}
return result;
}
#endregion
#region public double Length
///
/// Gets the length (magnitude) of the Quaterniond.
///
///
public double Length
{
get
{
return (double)System.Math.Sqrt(W * W + Xyz.LengthSquared);
}
}
#endregion
#region public double LengthSquared
///
/// Gets the square of the Quaterniond length (magnitude).
///
public double LengthSquared
{
get
{
return W * W + Xyz.LengthSquared;
}
}
#endregion
///
/// Returns a copy of the Quaterniond scaled to unit length.
///
public Quaterniond Normalized()
{
Quaterniond q = this;
q.Normalize();
return q;
}
///
/// Reverses the rotation angle of this Quaterniond.
///
public void Invert()
{
W = -W;
}
///
/// Returns a copy of this Quaterniond with its rotation angle reversed.
///
public Quaterniond Inverted()
{
var q = this;
q.Invert();
return q;
}
#region public void Normalize()
///
/// Scales the Quaterniond to unit length.
///
public void Normalize()
{
double scale = 1.0f / this.Length;
Xyz *= scale;
W *= scale;
}
#endregion
#region public void Conjugate()
///
/// Inverts the Vector3d component of this Quaterniond.
///
public void Conjugate()
{
Xyz = -Xyz;
}
#endregion
#endregion
#region Static
#region Fields
///
/// Defines the identity quaternion.
///
public readonly static Quaterniond Identity = new Quaterniond(0, 0, 0, 1);
#endregion
#region Add
///
/// Add two quaternions
///
/// The first operand
/// The second operand
/// The result of the addition
public static Quaterniond Add(Quaterniond left, Quaterniond right)
{
return new Quaterniond(
left.Xyz + right.Xyz,
left.W + right.W);
}
///
/// Add two quaternions
///
/// The first operand
/// The second operand
/// The result of the addition
public static void Add(ref Quaterniond left, ref Quaterniond right, out Quaterniond result)
{
result = new Quaterniond(
left.Xyz + right.Xyz,
left.W + right.W);
}
#endregion
#region Sub
///
/// Subtracts two instances.
///
/// The left instance.
/// The right instance.
/// The result of the operation.
public static Quaterniond Sub(Quaterniond left, Quaterniond right)
{
return new Quaterniond(
left.Xyz - right.Xyz,
left.W - right.W);
}
///
/// Subtracts two instances.
///
/// The left instance.
/// The right instance.
/// The result of the operation.
public static void Sub(ref Quaterniond left, ref Quaterniond right, out Quaterniond result)
{
result = new Quaterniond(
left.Xyz - right.Xyz,
left.W - right.W);
}
#endregion
#region Mult
///
/// Multiplies two instances.
///
/// The first instance.
/// The second instance.
/// A new instance containing the result of the calculation.
[Obsolete("Use Multiply instead.")]
public static Quaterniond Mult(Quaterniond left, Quaterniond right)
{
return new Quaterniond(
right.W * left.Xyz + left.W * right.Xyz + Vector3d.Cross(left.Xyz, right.Xyz),
left.W * right.W - Vector3d.Dot(left.Xyz, right.Xyz));
}
///
/// Multiplies two instances.
///
/// The first instance.
/// The second instance.
/// A new instance containing the result of the calculation.
[Obsolete("Use Multiply instead.")]
public static void Mult(ref Quaterniond left, ref Quaterniond right, out Quaterniond result)
{
result = new Quaterniond(
right.W * left.Xyz + left.W * right.Xyz + Vector3d.Cross(left.Xyz, right.Xyz),
left.W * right.W - Vector3d.Dot(left.Xyz, right.Xyz));
}
///
/// Multiplies two instances.
///
/// The first instance.
/// The second instance.
/// A new instance containing the result of the calculation.
public static Quaterniond Multiply(Quaterniond left, Quaterniond right)
{
Quaterniond result;
Multiply(ref left, ref right, out result);
return result;
}
///
/// Multiplies two instances.
///
/// The first instance.
/// The second instance.
/// A new instance containing the result of the calculation.
public static void Multiply(ref Quaterniond left, ref Quaterniond right, out Quaterniond result)
{
result = new Quaterniond(
right.W * left.Xyz + left.W * right.Xyz + Vector3d.Cross(left.Xyz, right.Xyz),
left.W * right.W - Vector3d.Dot(left.Xyz, right.Xyz));
}
///
/// Multiplies an instance by a scalar.
///
/// The instance.
/// The scalar.
/// A new instance containing the result of the calculation.
public static void Multiply(ref Quaterniond quaternion, double scale, out Quaterniond result)
{
result = new Quaterniond(quaternion.X * scale, quaternion.Y * scale, quaternion.Z * scale, quaternion.W * scale);
}
///
/// Multiplies an instance by a scalar.
///
/// The instance.
/// The scalar.
/// A new instance containing the result of the calculation.
public static Quaterniond Multiply(Quaterniond quaternion, double scale)
{
return new Quaterniond(quaternion.X * scale, quaternion.Y * scale, quaternion.Z * scale, quaternion.W * scale);
}
#endregion
#region Conjugate
///
/// Get the conjugate of the given Quaterniond
///
/// The Quaterniond
/// The conjugate of the given Quaterniond
public static Quaterniond Conjugate(Quaterniond q)
{
return new Quaterniond(-q.Xyz, q.W);
}
///
/// Get the conjugate of the given Quaterniond
///
/// The Quaterniond
/// The conjugate of the given Quaterniond
public static void Conjugate(ref Quaterniond q, out Quaterniond result)
{
result = new Quaterniond(-q.Xyz, q.W);
}
#endregion
#region Invert
///
/// Get the inverse of the given Quaterniond
///
/// The Quaterniond to invert
/// The inverse of the given Quaterniond
public static Quaterniond Invert(Quaterniond q)
{
Quaterniond result;
Invert(ref q, out result);
return result;
}
///
/// Get the inverse of the given Quaterniond
///
/// The Quaterniond to invert
/// The inverse of the given Quaterniond
public static void Invert(ref Quaterniond q, out Quaterniond result)
{
double lengthSq = q.LengthSquared;
if (lengthSq != 0.0)
{
double i = 1.0f / lengthSq;
result = new Quaterniond(q.Xyz * -i, q.W * i);
}
else
{
result = q;
}
}
#endregion
#region Normalize
///
/// Scale the given Quaterniond to unit length
///
/// The Quaterniond to normalize
/// The normalized Quaterniond
public static Quaterniond Normalize(Quaterniond q)
{
Quaterniond result;
Normalize(ref q, out result);
return result;
}
///
/// Scale the given Quaterniond to unit length
///
/// The Quaterniond to normalize
/// The normalized Quaterniond
public static void Normalize(ref Quaterniond q, out Quaterniond result)
{
double scale = 1.0f / q.Length;
result = new Quaterniond(q.Xyz * scale, q.W * scale);
}
#endregion
#region FromAxisAngle
///
/// Build a Quaterniond from the given axis and angle
///
/// The axis to rotate about
/// The rotation angle in radians
///
public static Quaterniond FromAxisAngle(Vector3d axis, double angle)
{
if (axis.LengthSquared == 0.0f)
return Identity;
Quaterniond result = Identity;
angle *= 0.5f;
axis.Normalize();
result.Xyz = axis * (double)System.Math.Sin(angle);
result.W = (double)System.Math.Cos(angle);
return Normalize(result);
}
#endregion
#region FromEulerAngles
///
/// Builds a Quaterniond from the given euler angles
///
/// The pitch (attitude), rotation around X axis
/// The yaw (heading), rotation around Y axis
/// The roll (bank), rotation around Z axis
///
public static Quaterniond FromEulerAngles(double pitch, double yaw, double roll)
{
return new Quaterniond(pitch, yaw, roll);
}
///
/// Builds a Quaterniond from the given euler angles
///
/// The euler angles as a vector
/// The equivalent Quaterniond
public static Quaterniond FromEulerAngles(Vector3d eulerAngles)
{
return new Quaterniond(eulerAngles);
}
///
/// Builds a Quaterniond from the given euler angles
///
/// The euler angles a vector
/// The equivalent Quaterniond
public static void FromEulerAngles(ref Vector3d eulerAngles, out Quaterniond result)
{
double c1 = Math.Cos(eulerAngles.Y * 0.5);
double c2 = Math.Cos(eulerAngles.X * 0.5);
double c3 = Math.Cos(eulerAngles.Z * 0.5);
double s1 = Math.Sin(eulerAngles.Y * 0.5);
double s2 = Math.Sin(eulerAngles.X * 0.5);
double s3 = Math.Sin(eulerAngles.Z * 0.5);
result.w = c1 * c2 * c3 - s1 * s2 * s3;
result.xyz.X = s1 * s2 * c3 + c1 * c2 * s3;
result.xyz.Y = s1 * c2 * c3 + c1 * s2 * s3;
result.xyz.Z = c1 * s2 * c3 - s1 * c2 * s3;
}
#endregion
#region FromMatrix
///
/// Builds a quaternion from the given rotation matrix
///
/// A rotation matrix
/// The equivalent quaternion
public static Quaterniond FromMatrix(Matrix3d matrix)
{
Quaterniond result;
FromMatrix(ref matrix, out result);
return result;
}
///
/// Builds a quaternion from the given rotation matrix
///
/// A rotation matrix
/// The equivalent quaternion
public static void FromMatrix(ref Matrix3d matrix, out Quaterniond result)
{
double trace = matrix.Trace;
if (trace > 0)
{
double s = Math.Sqrt(trace + 1) * 2;
double invS = 1.0 / s;
result.w = s * 0.25;
result.xyz.X = (matrix.Row2.Y - matrix.Row1.Z) * invS;
result.xyz.Y = (matrix.Row0.Z - matrix.Row2.X) * invS;
result.xyz.Z = (matrix.Row1.X - matrix.Row0.Y) * invS;
}
else
{
double m00 = matrix.Row0.X, m11 = matrix.Row1.Y, m22 = matrix.Row2.Z;
if (m00 > m11 && m00 > m22)
{
double s = Math.Sqrt(1 + m00 - m11 - m22) * 2;
double invS = 1.0 / s;
result.w = (matrix.Row2.Y - matrix.Row1.Z) * invS;
result.xyz.X = s * 0.25;
result.xyz.Y = (matrix.Row0.Y + matrix.Row1.X) * invS;
result.xyz.Z = (matrix.Row0.Z + matrix.Row2.X) * invS;
}
else if (m11 > m22)
{
double s = Math.Sqrt(1 + m11 - m00 - m22) * 2;
double invS = 1.0 / s;
result.w = (matrix.Row0.Z - matrix.Row2.X) * invS;
result.xyz.X = (matrix.Row0.Y + matrix.Row1.X) * invS;
result.xyz.Y = s * 0.25;
result.xyz.Z = (matrix.Row1.Z + matrix.Row2.Y) * invS;
}
else
{
double s = Math.Sqrt(1 + m22 - m00 - m11) * 2;
double invS = 1.0 / s;
result.w = (matrix.Row1.X - matrix.Row0.Y) * invS;
result.xyz.X = (matrix.Row0.Z + matrix.Row2.X) * invS;
result.xyz.Y = (matrix.Row1.Z + matrix.Row2.Y) * invS;
result.xyz.Z = s * 0.25;
}
}
}
#endregion
#region Slerp
///
/// Do Spherical linear interpolation between two quaternions
///
/// The first Quaterniond
/// The second Quaterniond
/// The blend factor
/// A smooth blend between the given quaternions
public static Quaterniond Slerp(Quaterniond q1, Quaterniond q2, double blend)
{
// if either input is zero, return the other.
if (q1.LengthSquared == 0.0f)
{
if (q2.LengthSquared == 0.0f)
{
return Identity;
}
return q2;
}
else if (q2.LengthSquared == 0.0f)
{
return q1;
}
double cosHalfAngle = q1.W * q2.W + Vector3d.Dot(q1.Xyz, q2.Xyz);
if (cosHalfAngle >= 1.0f || cosHalfAngle <= -1.0f)
{
// angle = 0.0f, so just return one input.
return q1;
}
else if (cosHalfAngle < 0.0f)
{
q2.Xyz = -q2.Xyz;
q2.W = -q2.W;
cosHalfAngle = -cosHalfAngle;
}
double blendA;
double blendB;
if (cosHalfAngle < 0.99f)
{
// do proper slerp for big angles
double halfAngle = (double)System.Math.Acos(cosHalfAngle);
double sinHalfAngle = (double)System.Math.Sin(halfAngle);
double oneOverSinHalfAngle = 1.0f / sinHalfAngle;
blendA = (double)System.Math.Sin(halfAngle * (1.0f - blend)) * oneOverSinHalfAngle;
blendB = (double)System.Math.Sin(halfAngle * blend) * oneOverSinHalfAngle;
}
else
{
// do lerp if angle is really small.
blendA = 1.0f - blend;
blendB = blend;
}
Quaterniond result = new Quaterniond(blendA * q1.Xyz + blendB * q2.Xyz, blendA * q1.W + blendB * q2.W);
if (result.LengthSquared > 0.0f)
return Normalize(result);
else
return Identity;
}
#endregion
#endregion
#region Operators
///
/// Adds two instances.
///
/// The first instance.
/// The second instance.
/// The result of the calculation.
public static Quaterniond operator +(Quaterniond left, Quaterniond right)
{
left.Xyz += right.Xyz;
left.W += right.W;
return left;
}
///
/// Subtracts two instances.
///
/// The first instance.
/// The second instance.
/// The result of the calculation.
public static Quaterniond operator -(Quaterniond left, Quaterniond right)
{
left.Xyz -= right.Xyz;
left.W -= right.W;
return left;
}
///
/// Multiplies two instances.
///
/// The first instance.
/// The second instance.
/// The result of the calculation.
public static Quaterniond operator *(Quaterniond left, Quaterniond right)
{
Multiply(ref left, ref right, out left);
return left;
}
///
/// Multiplies an instance by a scalar.
///
/// The instance.
/// The scalar.
/// A new instance containing the result of the calculation.
public static Quaterniond operator *(Quaterniond quaternion, double scale)
{
Multiply(ref quaternion, scale, out quaternion);
return quaternion;
}
///
/// Multiplies an instance by a scalar.
///
/// The instance.
/// The scalar.
/// A new instance containing the result of the calculation.
public static Quaterniond operator *(double scale, Quaterniond quaternion)
{
return new Quaterniond(quaternion.X * scale, quaternion.Y * scale, quaternion.Z * scale, quaternion.W * scale);
}
///
/// Compares two instances for equality.
///
/// The first instance.
/// The second instance.
/// True, if left equals right; false otherwise.
public static bool operator ==(Quaterniond left, Quaterniond right)
{
return left.Equals(right);
}
///
/// Compares two instances for inequality.
///
/// The first instance.
/// The second instance.
/// True, if left does not equal right; false otherwise.
public static bool operator !=(Quaterniond left, Quaterniond right)
{
return !left.Equals(right);
}
#endregion
#region Overrides
#region public override string ToString()
///
/// Returns a System.String that represents the current Quaterniond.
///
///
public override string ToString()
{
return String.Format("V: {0}, W: {1}", Xyz, W);
}
#endregion
#region public override bool Equals (object o)
///
/// Compares this object instance to another object for equality.
///
/// The other object to be used in the comparison.
/// True if both objects are Quaternions of equal value. Otherwise it returns false.
public override bool Equals(object other)
{
if (other is Quaterniond == false) return false;
return this == (Quaterniond)other;
}
#endregion
#region public override int GetHashCode ()
///
/// Provides the hash code for this object.
///
/// A hash code formed from the bitwise XOR of this objects members.
public override int GetHashCode()
{
unchecked
{
return (this.xyz.GetHashCode() * 397) ^ this.w.GetHashCode();
}
}
#endregion
#endregion
#endregion
#if false
#region Fields
/// The W component of the Quaterniond.
public double W;
/// The X component of the Quaterniond.
public double X;
/// The Y component of the Quaterniond.
public double Y;
/// The Z component of the Quaterniond.
public double Z;
#endregion
#region Constructors
/// Constructs left Quaterniond that is left copy of the given Quaterniond.
/// The Quaterniond to copy.
public Quaterniond(ref Quaterniond Quaterniond) : this(Quaterniond.W, Quaterniond.X, Quaterniond.Y, Quaterniond.Z) { }
/// Constructs left Quaterniond from the given components.
/// The W component for the Quaterniond.
/// A Vector representing the X, Y, and Z componets for the quaterion.
public Quaterniond(double w, ref Vector3d vector3d) : this(w, vector3d.X, vector3d.Y, vector3d.Z) { }
/// Constructs left Quaterniond from the given axis and angle.
/// The axis for the Quaterniond.
/// The angle for the quaternione.
public Quaterniond(ref Vector3d axis, double angle)
{
double halfAngle = Functions.DTOR * angle / 2;
this.W = System.Math.Cos(halfAngle);
double sin = System.Math.Sin(halfAngle);
Vector3d axisNormalized;
Vector3d.Normalize(ref axis, out axisNormalized);
this.X = axisNormalized.X * sin;
this.Y = axisNormalized.Y * sin;
this.Z = axisNormalized.Z * sin;
}
/// Constructs left Quaterniond from the given components.
/// The W component for the Quaterniond.
/// The X component for the Quaterniond.
/// The Y component for the Quaterniond.
/// The Z component for the Quaterniond.
public Quaterniond(double w, double x, double y, double z)
{
this.W = w;
this.X = x;
this.Y = y;
this.Z = z;
}
/// Constructs left Quaterniond from the given array of double-precision floating-point numbers.
/// The array of doubles for the components of the Quaterniond.
public Quaterniond(double[] doubleArray)
{
if (doubleArray == null || doubleArray.GetLength(0) < 4) throw new MissingFieldException();
this.W = doubleArray[0];
this.X = doubleArray[1];
this.Y = doubleArray[2];
this.Z = doubleArray[3];
}
/// Constructs left Quaterniond from the given matrix. Only contains rotation information.
/// The matrix for the components of the Quaterniond.
public Quaterniond(ref Matrix4d matrix)
{
double scale = System.Math.Pow(matrix.Determinant, 1.0d/3.0d);
W = System.Math.Sqrt(System.Math.Max(0, scale + matrix[0, 0] + matrix[1, 1] + matrix[2, 2])) / 2;
X = System.Math.Sqrt(System.Math.Max(0, scale + matrix[0, 0] - matrix[1, 1] - matrix[2, 2])) / 2;
Y = System.Math.Sqrt(System.Math.Max(0, scale - matrix[0, 0] + matrix[1, 1] - matrix[2, 2])) / 2;
Z = System.Math.Sqrt(System.Math.Max(0, scale - matrix[0, 0] - matrix[1, 1] + matrix[2, 2])) / 2;
if( matrix[2,1] - matrix[1,2] < 0 ) X = -X;
if( matrix[0,2] - matrix[2,0] < 0 ) Y = -Y;
if( matrix[1,0] - matrix[0,1] < 0 ) Z = -Z;
}
public Quaterniond(ref Matrix3d matrix)
{
double scale = System.Math.Pow(matrix.Determinant, 1.0d / 3.0d);
W = System.Math.Sqrt(System.Math.Max(0, scale + matrix[0, 0] + matrix[1, 1] + matrix[2, 2])) / 2;
X = System.Math.Sqrt(System.Math.Max(0, scale + matrix[0, 0] - matrix[1, 1] - matrix[2, 2])) / 2;
Y = System.Math.Sqrt(System.Math.Max(0, scale - matrix[0, 0] + matrix[1, 1] - matrix[2, 2])) / 2;
Z = System.Math.Sqrt(System.Math.Max(0, scale - matrix[0, 0] - matrix[1, 1] + matrix[2, 2])) / 2;
if (matrix[2, 1] - matrix[1, 2] < 0) X = -X;
if (matrix[0, 2] - matrix[2, 0] < 0) Y = -Y;
if (matrix[1, 0] - matrix[0, 1] < 0) Z = -Z;
}
#endregion
#region Arithmetic Operators
public void Add(ref Quaterniond Quaterniond)
{
W = W + Quaterniond.W;
X = X + Quaterniond.X;
Y = Y + Quaterniond.Y;
Z = Z + Quaterniond.Z;
}
public void Add(ref Quaterniond Quaterniond, out Quaterniond result)
{
result.W = W + Quaterniond.W;
result.X = X + Quaterniond.X;
result.Y = Y + Quaterniond.Y;
result.Z = Z + Quaterniond.Z;
}
public static void Add(ref Quaterniond left, ref Quaterniond right, out Quaterniond result)
{
result.W = left.W + right.W;
result.X = left.X + right.X;
result.Y = left.Y + right.Y;
result.Z = left.Z + right.Z;
}
public void Subtract(ref Quaterniond Quaterniond)
{
W = W - Quaterniond.W;
X = X - Quaterniond.X;
Y = Y - Quaterniond.Y;
Z = Z - Quaterniond.Z;
}
public void Subtract(ref Quaterniond Quaterniond, out Quaterniond result)
{
result.W = W - Quaterniond.W;
result.X = X - Quaterniond.X;
result.Y = Y - Quaterniond.Y;
result.Z = Z - Quaterniond.Z;
}
public static void Subtract(ref Quaterniond left, ref Quaterniond right, out Quaterniond result)
{
result.W = left.W - right.W;
result.X = left.X - right.X;
result.Y = left.Y - right.Y;
result.Z = left.Z - right.Z;
}
public void Multiply(ref Quaterniond Quaterniond)
{
double w = W * Quaterniond.W - X * Quaterniond.X - Y * Quaterniond.Y - Z * Quaterniond.Z;
double x = W * Quaterniond.X + X * Quaterniond.W + Y * Quaterniond.Z - Z * Quaterniond.Y;
double y = W * Quaterniond.Y + Y * Quaterniond.W + Z * Quaterniond.X - X * Quaterniond.Z;
Z = W * Quaterniond.Z + Z * Quaterniond.W + X * Quaterniond.Y - Y * Quaterniond.X;
W = w;
X = x;
Y = y;
}
public void Multiply(ref Quaterniond Quaterniond, out Quaterniond result)
{
result.W = W * Quaterniond.W - X * Quaterniond.X - Y * Quaterniond.Y - Z * Quaterniond.Z;
result.X = W * Quaterniond.X + X * Quaterniond.W + Y * Quaterniond.Z - Z * Quaterniond.Y;
result.Y = W * Quaterniond.Y + Y * Quaterniond.W + Z * Quaterniond.X - X * Quaterniond.Z;
result.Z = W * Quaterniond.Z + Z * Quaterniond.W + X * Quaterniond.Y - Y * Quaterniond.X;
}
public static void Multiply(ref Quaterniond left, ref Quaterniond right, out Quaterniond result)
{
result.W = left.W * right.W - left.X * right.X - left.Y * right.Y - left.Z * right.Z;
result.X = left.W * right.X + left.X * right.W + left.Y * right.Z - left.Z * right.Y;
result.Y = left.W * right.Y + left.Y * right.W + left.Z * right.X - left.X * right.Z;
result.Z = left.W * right.Z + left.Z * right.W + left.X * right.Y - left.Y * right.X;
}
public void Multiply(double scalar)
{
W = W * scalar;
X = X * scalar;
Y = Y * scalar;
Z = Z * scalar;
}
public void Multiply(double scalar, out Quaterniond result)
{
result.W = W * scalar;
result.X = X * scalar;
result.Y = Y * scalar;
result.Z = Z * scalar;
}
public static void Multiply(ref Quaterniond Quaterniond, double scalar, out Quaterniond result)
{
result.W = Quaterniond.W * scalar;
result.X = Quaterniond.X * scalar;
result.Y = Quaterniond.Y * scalar;
result.Z = Quaterniond.Z * scalar;
}
public void Divide(double scalar)
{
if (scalar == 0) throw new DivideByZeroException();
W = W / scalar;
X = X / scalar;
Y = Y / scalar;
Z = Z / scalar;
}
public void Divide(double scalar, out Quaterniond result)
{
if (scalar == 0) throw new DivideByZeroException();
result.W = W / scalar;
result.X = X / scalar;
result.Y = Y / scalar;
result.Z = Z / scalar;
}
public static void Divide(ref Quaterniond Quaterniond, double scalar, out Quaterniond result)
{
if (scalar == 0) throw new DivideByZeroException();
result.W = Quaterniond.W / scalar;
result.X = Quaterniond.X / scalar;
result.Y = Quaterniond.Y / scalar;
result.Z = Quaterniond.Z / scalar;
}
#endregion
#region Functions
public double Modulus
{
get
{
return System.Math.Sqrt(W * W + X * X + Y * Y + Z * Z);
}
}
public double ModulusSquared
{
get
{
return W * W + X * X + Y * Y + Z * Z;
}
}
public static double DotProduct(Quaterniond left, Quaterniond right)
{
return left.W * right.W + left.X * right.X + left.Y * right.Y + left.Z * right.Z;
}
public void Normalize()
{
double modulus = System.Math.Sqrt(W * W + X * X + Y * Y + Z * Z);
if (modulus == 0) throw new DivideByZeroException();
W = W / modulus;
X = X / modulus;
Y = Y / modulus;
Z = Z / modulus;
}
public void Normalize( out Quaterniond result )
{
double modulus = System.Math.Sqrt(W * W + X * X + Y * Y + Z * Z);
if (modulus == 0) throw new DivideByZeroException();
result.W = W / modulus;
result.X = X / modulus;
result.Y = Y / modulus;
result.Z = Z / modulus;
}
public static void Normalize(ref Quaterniond Quaterniond, out Quaterniond result)
{
double modulus = System.Math.Sqrt(Quaterniond.W * Quaterniond.W + Quaterniond.X * Quaterniond.X + Quaterniond.Y * Quaterniond.Y + Quaterniond.Z * Quaterniond.Z);
if (modulus == 0) throw new DivideByZeroException();
result.W = Quaterniond.W / modulus;
result.X = Quaterniond.X / modulus;
result.Y = Quaterniond.Y / modulus;
result.Z = Quaterniond.Z / modulus;
}
public void Conjugate()
{
X = -X;
Y = -Y;
Z = -Z;
}
public void Conjugate( out Quaterniond result )
{
result.W = W;
result.X = -X;
result.Y = -Y;
result.Z = -Z;
}
public static void Conjugate(ref Quaterniond Quaterniond, out Quaterniond result)
{
result.W = Quaterniond.W;
result.X = -Quaterniond.X;
result.Y = -Quaterniond.Y;
result.Z = -Quaterniond.Z;
}
public void Inverse()
{
double modulusSquared = W * W + X * X + Y * Y + Z * Z;
if (modulusSquared <= 0) throw new InvalidOperationException();
double inverseModulusSquared = 1.0 / modulusSquared;
W = W * inverseModulusSquared;
X = X * -inverseModulusSquared;
Y = Y * -inverseModulusSquared;
Z = Z * -inverseModulusSquared;
}
public void Inverse( out Quaterniond result )
{
double modulusSquared = W * W + X * X + Y * Y + Z * Z;
if (modulusSquared <= 0) throw new InvalidOperationException();
double inverseModulusSquared = 1.0 / modulusSquared;
result.W = W * inverseModulusSquared;
result.X = X * -inverseModulusSquared;
result.Y = Y * -inverseModulusSquared;
result.Z = Z * -inverseModulusSquared;
}
public static void Inverse(ref Quaterniond Quaterniond, out Quaterniond result)
{
double modulusSquared = Quaterniond.W * Quaterniond.W + Quaterniond.X * Quaterniond.X + Quaterniond.Y * Quaterniond.Y + Quaterniond.Z * Quaterniond.Z;
if (modulusSquared <= 0) throw new InvalidOperationException();
double inverseModulusSquared = 1.0 / modulusSquared;
result.W = Quaterniond.W * inverseModulusSquared;
result.X = Quaterniond.X * -inverseModulusSquared;
result.Y = Quaterniond.Y * -inverseModulusSquared;
result.Z = Quaterniond.Z * -inverseModulusSquared;
}
public void Log()
{
if (System.Math.Abs(W) < 1.0)
{
double angle = System.Math.Acos(W);
double sin = System.Math.Sin(angle);
if (System.Math.Abs(sin) >= 0)
{
double coefficient = angle / sin;
X = X * coefficient;
Y = Y * coefficient;
Z = Z * coefficient;
}
}
else
{
X = 0;
Y = 0;
Z = 0;
}
W = 0;
}
public void Log( out Quaterniond result )
{
if (System.Math.Abs(W) < 1.0)
{
double angle = System.Math.Acos(W);
double sin = System.Math.Sin(angle);
if (System.Math.Abs(sin) >= 0)
{
double coefficient = angle / sin;
result.X = X * coefficient;
result.Y = Y * coefficient;
result.Z = Z * coefficient;
}
else
{
result.X = X;
result.Y = Y;
result.Z = Z;
}
}
else
{
result.X = 0;
result.Y = 0;
result.Z = 0;
}
result.W = 0;
}
public static void Log(ref Quaterniond Quaterniond, out Quaterniond result)
{
if (System.Math.Abs(Quaterniond.W) < 1.0)
{
double angle = System.Math.Acos(Quaterniond.W);
double sin = System.Math.Sin(angle);
if (System.Math.Abs(sin) >= 0)
{
double coefficient = angle / sin;
result.X = Quaterniond.X * coefficient;
result.Y = Quaterniond.Y * coefficient;
result.Z = Quaterniond.Z * coefficient;
}
else
{
result.X = Quaterniond.X;
result.Y = Quaterniond.Y;
result.Z = Quaterniond.Z;
}
}
else
{
result.X = 0;
result.Y = 0;
result.Z = 0;
}
result.W = 0;
}
public void Exp()
{
double angle = System.Math.Sqrt(X * X + Y * Y + Z * Z);
double sin = System.Math.Sin(angle);
if (System.Math.Abs(sin) > 0)
{
double coefficient = angle / sin;
W = 0;
X = X * coefficient;
Y = Y * coefficient;
Z = Z * coefficient;
}
else
{
W = 0;
}
}
public void Exp(out Quaterniond result)
{
double angle = System.Math.Sqrt(X * X + Y * Y + Z * Z);
double sin = System.Math.Sin(angle);
if (System.Math.Abs(sin) > 0)
{
double coefficient = angle / sin;
result.W = 0;
result.X = X * coefficient;
result.Y = Y * coefficient;
result.Z = Z * coefficient;
}
else
{
result.W = 0;
result.X = X;
result.Y = Y;
result.Z = Z;
}
}
public static void Exp(ref Quaterniond Quaterniond, out Quaterniond result)
{
double angle = System.Math.Sqrt(Quaterniond.X * Quaterniond.X + Quaterniond.Y * Quaterniond.Y + Quaterniond.Z * Quaterniond.Z);
double sin = System.Math.Sin(angle);
if (System.Math.Abs(sin) > 0)
{
double coefficient = angle / sin;
result.W = 0;
result.X = Quaterniond.X * coefficient;
result.Y = Quaterniond.Y * coefficient;
result.Z = Quaterniond.Z * coefficient;
}
else
{
result.W = 0;
result.X = Quaterniond.X;
result.Y = Quaterniond.Y;
result.Z = Quaterniond.Z;
}
}
/// Returns left matrix for this Quaterniond.
public void Matrix4d(out Matrix4d result)
{
// TODO Expand
result = new Matrix4d(ref this);
}
public void GetAxisAndAngle(out Vector3d axis, out double angle)
{
Quaterniond Quaterniond;
Normalize(out Quaterniond);
double cos = Quaterniond.W;
angle = System.Math.Acos(cos) * 2 * Functions.RTOD;
double sin = System.Math.Sqrt( 1.0d - cos * cos );
if ( System.Math.Abs( sin ) < 0.0001 ) sin = 1;
axis = new Vector3d(X / sin, Y / sin, Z / sin);
}
public static void Slerp(ref Quaterniond start, ref Quaterniond end, double blend, out Quaterniond result)
{
if (start.W == 0 && start.X == 0 && start.Y == 0 && start.Z == 0)
{
if (end.W == 0 && end.X == 0 && end.Y == 0 && end.Z == 0)
{
result.W = 1;
result.X = 0;
result.Y = 0;
result.Z = 0;
}
else
{
result = end;
}
}
else if (end.W == 0 && end.X == 0 && end.Y == 0 && end.Z == 0)
{
result = start;
}
Vector3d startVector = new Vector3d(start.X, start.Y, start.Z);
Vector3d endVector = new Vector3d(end.X, end.Y, end.Z);
double cosHalfAngle = start.W * end.W + Vector3d.Dot(startVector, endVector);
if (cosHalfAngle >= 1.0f || cosHalfAngle <= -1.0f)
{
// angle = 0.0f, so just return one input.
result = start;
}
else if (cosHalfAngle < 0.0f)
{
end.W = -end.W;
end.X = -end.X;
end.Y = -end.Y;
end.Z = -end.Z;
cosHalfAngle = -cosHalfAngle;
}
double blendA;
double blendB;
if (cosHalfAngle < 0.99f)
{
// do proper slerp for big angles
double halfAngle = (double)System.Math.Acos(cosHalfAngle);
double sinHalfAngle = (double)System.Math.Sin(halfAngle);
double oneOverSinHalfAngle = 1.0f / sinHalfAngle;
blendA = (double)System.Math.Sin(halfAngle * (1.0f - blend)) * oneOverSinHalfAngle;
blendB = (double)System.Math.Sin(halfAngle * blend) * oneOverSinHalfAngle;
}
else
{
// do lerp if angle is really small.
blendA = 1.0f - blend;
blendB = blend;
}
result.W = blendA * start.W + blendB * end.W;
result.X = blendA * start.X + blendB * end.X;
result.Y = blendA * start.Y + blendB * end.Y;
result.Z = blendA * start.Z + blendB * end.Z;
if (result.W != 0 || result.X != 0 || result.Y != 0 || result.Z != 0)
{
result.Normalize();
}
else
{
result.W = 1;
result.X = 0;
result.Y = 0;
result.Z = 0;
}
}
#endregion
#region HashCode
/// Returns the hash code for this instance.
/// A 32-bit signed integer that is the hash code for this instance.
public override int GetHashCode()
{
base.GetHashCode();
return W.GetHashCode() ^ X.GetHashCode() ^ Y.GetHashCode() ^ Z.GetHashCode();
}
#endregion
#region String and Parse
/// Returns the fully qualified type name of this instance.
/// A System.String containing left fully qualified type name.
public override string ToString()
{
return string.Format("({0}, {1}, {2}, {3})", W, X, Y, Z);
}
/// Parses left string, converting it to left Quaterniond.
/// The string to parse.
/// The Quaterniond represented by the string.
public static void Parse(string str, out Quaterniond result)
{
Match match = new Regex(@"\((?.*),(?.*),(?.*),(?.*)\)", RegexOptions.None).Match(str);
if (!match.Success) throw new Exception("Parse failed!");
result.W = double.Parse(match.Result("${w}"));
result.X = double.Parse(match.Result("${x}"));
result.Y = double.Parse(match.Result("${y}"));
result.Z = double.Parse(match.Result("${z}"));
}
#endregion
#region Constants
/// A quaterion with all zero components.
public static readonly Quaterniond Zero = new Quaterniond(0, 0, 0, 0);
/// A quaterion representing an identity.
public static readonly Quaterniond Identity = new Quaterniond(1, 0, 0, 0);
/// A quaterion representing the W axis.
public static readonly Quaterniond WAxis = new Quaterniond(1, 0, 0, 0);
/// A quaterion representing the X axis.
public static readonly Quaterniond XAxis = new Quaterniond(0, 1, 0, 0);
/// A quaterion representing the Y axis.
public static readonly Quaterniond YAxis = new Quaterniond(0, 0, 1, 0);
/// A quaterion representing the Z axis.
public static readonly Quaterniond ZAxis = new Quaterniond(0, 0, 0, 1);
#endregion
#endif
#region IEquatable Members
///
/// Compares this Quaterniond instance to another Quaterniond for equality.
///
/// The other Quaterniond to be used in the comparison.
/// True if both instances are equal; false otherwise.
public bool Equals(Quaterniond other)
{
return Xyz == other.Xyz && W == other.W;
}
#endregion
}
}