2017-08-30 00:00:16 +00:00
|
|
|
///////////////////////////////////////////
|
2019-03-23 04:13:18 +00:00
|
|
|
// LibFile: quaternions.scad
|
|
|
|
// Support for Quaternions.
|
|
|
|
// To use, add the following line to the beginning of your file:
|
|
|
|
// ```
|
2019-04-19 07:25:10 +00:00
|
|
|
// include <BOSL2/std.scad>
|
2019-03-23 04:13:18 +00:00
|
|
|
// ```
|
2017-08-30 00:00:16 +00:00
|
|
|
///////////////////////////////////////////
|
|
|
|
|
2019-02-02 07:33:07 +00:00
|
|
|
|
2019-03-23 04:13:18 +00:00
|
|
|
// Section: Quaternions
|
|
|
|
// Quaternions are fast methods of storing and calculating arbitrary rotations.
|
|
|
|
// Quaternions contain information on both axis of rotation, and rotation angle.
|
|
|
|
// You can chain multiple rotation together by multiplying quaternions together.
|
2020-02-16 01:13:33 +00:00
|
|
|
// They don't suffer from the gimbal-lock issues that `[X,Y,Z]` rotation angles do.
|
2019-03-23 04:13:18 +00:00
|
|
|
// Quaternions are stored internally as a 4-value vector:
|
2020-02-16 01:13:33 +00:00
|
|
|
// `[X,Y,Z,W]`, where the quaternion formula is `W+Xi+Yj+Zk`
|
2019-03-23 04:13:18 +00:00
|
|
|
|
|
|
|
|
|
|
|
// Internal
|
2017-08-30 00:00:16 +00:00
|
|
|
function _Quat(a,s,w) = [a[0]*s, a[1]*s, a[2]*s, w];
|
2019-03-23 04:13:18 +00:00
|
|
|
|
2020-07-17 19:28:28 +00:00
|
|
|
function _Qvec(q) = [q.x,q.y,q.z];
|
|
|
|
|
|
|
|
function _Qreal(q) = q[3];
|
|
|
|
|
|
|
|
function _Qset(v,r) = concat( v, r );
|
|
|
|
|
|
|
|
// normalizes without checking
|
2020-08-26 13:07:12 +00:00
|
|
|
function _Qunit(q) = q/norm(q);
|
2020-07-17 19:28:28 +00:00
|
|
|
|
|
|
|
|
|
|
|
// Function: Q_is_quat()
|
|
|
|
// Usage:
|
|
|
|
// if(Q_is_quat(q)) a=0;
|
|
|
|
// Description: Return true if q is a valid non-zero quaternion.
|
|
|
|
// Arguments:
|
|
|
|
// q = object to check.
|
2020-08-26 13:07:12 +00:00
|
|
|
function Q_is_quat(q) = is_vector(q,4) ;//&& ! approx(norm(q),0) ;
|
2020-07-17 19:28:28 +00:00
|
|
|
|
2019-03-23 04:13:18 +00:00
|
|
|
|
|
|
|
// Function: Quat()
|
|
|
|
// Usage:
|
|
|
|
// Quat(ax, ang);
|
2020-07-17 19:28:28 +00:00
|
|
|
// Description: Create a normalized Quaternion from axis and angle of rotation.
|
2019-03-23 04:13:18 +00:00
|
|
|
// Arguments:
|
|
|
|
// ax = Vector of axis of rotation.
|
|
|
|
// ang = Number of degrees to rotate around the axis counter-clockwise, when facing the origin.
|
2020-07-17 19:28:28 +00:00
|
|
|
function Quat(ax=[0,0,1], ang=0) =
|
2020-07-19 21:09:33 +00:00
|
|
|
assert( is_vector(ax,3) && is_finite(ang), "Invalid input")
|
|
|
|
let( n = norm(ax) )
|
|
|
|
approx(n,0)
|
|
|
|
? _Quat([0,0,0], sin(ang/2), cos(ang/2))
|
|
|
|
: _Quat(ax/n, sin(ang/2), cos(ang/2));
|
2019-02-06 11:32:27 +00:00
|
|
|
|
2019-03-23 04:13:18 +00:00
|
|
|
|
|
|
|
// Function: QuatX()
|
|
|
|
// Usage:
|
|
|
|
// QuatX(a);
|
2020-07-17 19:28:28 +00:00
|
|
|
// Description: Create a normalized Quaternion for rotating around the X axis [1,0,0].
|
2019-03-23 04:13:18 +00:00
|
|
|
// Arguments:
|
|
|
|
// a = Number of degrees to rotate around the axis counter-clockwise, when facing the origin.
|
2020-07-17 19:28:28 +00:00
|
|
|
function QuatX(a=0) =
|
2020-07-19 21:09:33 +00:00
|
|
|
assert( is_finite(a), "Invalid angle" )
|
|
|
|
Quat([1,0,0],a);
|
2019-03-23 04:13:18 +00:00
|
|
|
|
|
|
|
|
|
|
|
// Function: QuatY()
|
|
|
|
// Usage:
|
|
|
|
// QuatY(a);
|
2020-07-17 19:28:28 +00:00
|
|
|
// Description: Create a normalized Quaternion for rotating around the Y axis [0,1,0].
|
2019-03-23 04:13:18 +00:00
|
|
|
// Arguments:
|
|
|
|
// a = Number of degrees to rotate around the axis counter-clockwise, when facing the origin.
|
2020-07-17 19:28:28 +00:00
|
|
|
function QuatY(a=0) =
|
2020-07-19 21:09:33 +00:00
|
|
|
assert( is_finite(a), "Invalid angle" )
|
|
|
|
Quat([0,1,0],a);
|
2020-07-17 19:28:28 +00:00
|
|
|
|
2019-03-23 04:13:18 +00:00
|
|
|
|
|
|
|
// Function: QuatZ()
|
|
|
|
// Usage:
|
|
|
|
// QuatZ(a);
|
2020-07-17 19:28:28 +00:00
|
|
|
// Description: Create a normalized Quaternion for rotating around the Z axis [0,0,1].
|
2019-03-23 04:13:18 +00:00
|
|
|
// Arguments:
|
|
|
|
// a = Number of degrees to rotate around the axis counter-clockwise, when facing the origin.
|
2020-07-17 19:28:28 +00:00
|
|
|
function QuatZ(a=0) =
|
2020-07-19 21:09:33 +00:00
|
|
|
assert( is_finite(a), "Invalid angle" )
|
|
|
|
Quat([0,0,1],a);
|
2019-02-06 11:32:27 +00:00
|
|
|
|
2019-03-23 04:13:18 +00:00
|
|
|
|
|
|
|
// Function: QuatXYZ()
|
|
|
|
// Usage:
|
|
|
|
// QuatXYZ([X,Y,Z])
|
|
|
|
// Description:
|
2020-07-17 19:28:28 +00:00
|
|
|
// Creates a normalized quaternion from standard [X,Y,Z] rotation angles in degrees.
|
2019-03-23 04:13:18 +00:00
|
|
|
// Arguments:
|
|
|
|
// a = The triplet of rotation angles, [X,Y,Z]
|
|
|
|
function QuatXYZ(a=[0,0,0]) =
|
2020-07-19 21:09:33 +00:00
|
|
|
assert( is_vector(a,3), "Invalid angles")
|
2020-05-30 02:04:34 +00:00
|
|
|
let(
|
2020-07-19 21:09:33 +00:00
|
|
|
qx = QuatX(a[0]),
|
|
|
|
qy = QuatY(a[1]),
|
|
|
|
qz = QuatZ(a[2])
|
2020-05-30 02:04:34 +00:00
|
|
|
)
|
2020-08-26 13:07:12 +00:00
|
|
|
_Qmul(qz, _Qmul(qy, qx));
|
2017-08-30 00:00:16 +00:00
|
|
|
|
2019-03-23 04:13:18 +00:00
|
|
|
|
2020-07-17 19:28:28 +00:00
|
|
|
// Function: Q_From_to()
|
|
|
|
// Usage:
|
|
|
|
// q = Q_From_to(v1, v2);
|
|
|
|
// Description:
|
|
|
|
// Returns the normalized quaternion that rotates the non zero 3D vector v1
|
|
|
|
// to the non zero 3D vector v2.
|
|
|
|
function Q_From_to(v1, v2) =
|
2020-07-19 21:09:33 +00:00
|
|
|
assert( is_vector(v1,3) && is_vector(v2,3)
|
|
|
|
&& ! approx(norm(v1),0) && ! approx(norm(v2),0)
|
|
|
|
, "Invalid vector(s)")
|
|
|
|
let( ax = cross(v1,v2),
|
|
|
|
n = norm(ax) )
|
|
|
|
approx(n, 0)
|
|
|
|
? v1*v2>0 ? Q_Ident() : Quat([ v1.y, -v1.x, 0], 180)
|
|
|
|
: Quat(ax, atan2( n , v1*v2 ));
|
2020-07-17 19:28:28 +00:00
|
|
|
|
|
|
|
|
2019-03-23 04:13:18 +00:00
|
|
|
// Function: Q_Ident()
|
|
|
|
// Description: Returns the "Identity" zero-rotation Quaternion.
|
2017-08-30 00:00:16 +00:00
|
|
|
function Q_Ident() = [0, 0, 0, 1];
|
|
|
|
|
|
|
|
|
2019-03-23 04:13:18 +00:00
|
|
|
// Function: Q_Add_S()
|
|
|
|
// Usage:
|
|
|
|
// Q_Add_S(q, s)
|
2020-07-17 19:28:28 +00:00
|
|
|
// Description:
|
|
|
|
// Adds a scalar value `s` to the W part of a quaternion `q`.
|
|
|
|
// The returned quaternion is usually not normalized.
|
|
|
|
function Q_Add_S(q, s) =
|
2020-07-19 21:09:33 +00:00
|
|
|
assert( is_finite(s), "Invalid scalar" )
|
|
|
|
q+[0,0,0,s];
|
2019-03-23 04:13:18 +00:00
|
|
|
|
|
|
|
|
|
|
|
// Function: Q_Sub_S()
|
|
|
|
// Usage:
|
|
|
|
// Q_Sub_S(q, s)
|
2020-07-17 19:28:28 +00:00
|
|
|
// Description:
|
|
|
|
// Subtracts a scalar value `s` from the W part of a quaternion `q`.
|
|
|
|
// The returned quaternion is usually not normalized.
|
|
|
|
function Q_Sub_S(q, s) =
|
2020-07-19 21:09:33 +00:00
|
|
|
assert( is_finite(s), "Invalid scalar" )
|
|
|
|
q-[0,0,0,s];
|
2019-03-23 04:13:18 +00:00
|
|
|
|
|
|
|
|
|
|
|
// Function: Q_Mul_S()
|
|
|
|
// Usage:
|
|
|
|
// Q_Mul_S(q, s)
|
2020-07-17 19:28:28 +00:00
|
|
|
// Description:
|
|
|
|
// Multiplies each part of a quaternion `q` by a scalar value `s`.
|
|
|
|
// The returned quaternion is usually not normalized.
|
|
|
|
function Q_Mul_S(q, s) =
|
2020-07-19 21:09:33 +00:00
|
|
|
assert( is_finite(s), "Invalid scalar" )
|
|
|
|
q*s;
|
2019-03-23 04:13:18 +00:00
|
|
|
|
|
|
|
|
|
|
|
// Function: Q_Div_S()
|
|
|
|
// Usage:
|
|
|
|
// Q_Div_S(q, s)
|
2020-07-17 19:28:28 +00:00
|
|
|
// Description:
|
|
|
|
// Divides each part of a quaternion `q` by a scalar value `s`.
|
|
|
|
// The returned quaternion is usually not normalized.
|
|
|
|
function Q_Div_S(q, s) =
|
2020-07-19 21:09:33 +00:00
|
|
|
assert( is_finite(s) && ! approx(s,0) , "Invalid scalar" )
|
|
|
|
q/s;
|
2019-03-23 04:13:18 +00:00
|
|
|
|
|
|
|
|
|
|
|
// Function: Q_Add()
|
|
|
|
// Usage:
|
|
|
|
// Q_Add(a, b)
|
2020-07-17 19:28:28 +00:00
|
|
|
// Description:
|
|
|
|
// Adds each part of two quaternions together.
|
|
|
|
// The returned quaternion is usually not normalized.
|
|
|
|
function Q_Add(a, b) =
|
2020-07-19 21:09:33 +00:00
|
|
|
assert( Q_is_quat(a) && Q_is_quat(a), "Invalid quaternion(s)")
|
|
|
|
assert( ! approx(norm(a+b),0), "Quaternions cannot be opposed" )
|
|
|
|
a+b;
|
2019-03-23 04:13:18 +00:00
|
|
|
|
|
|
|
|
|
|
|
// Function: Q_Sub()
|
|
|
|
// Usage:
|
|
|
|
// Q_Sub(a, b)
|
2020-07-17 19:28:28 +00:00
|
|
|
// Description:
|
|
|
|
// Subtracts each part of quaternion `b` from quaternion `a`.
|
|
|
|
// The returned quaternion is usually not normalized.
|
|
|
|
function Q_Sub(a, b) =
|
2020-07-19 21:09:33 +00:00
|
|
|
assert( Q_is_quat(a) && Q_is_quat(a), "Invalid quaternion(s)")
|
|
|
|
assert( ! approx(a,b), "Quaternions cannot be equal" )
|
|
|
|
a-b;
|
2019-03-23 04:13:18 +00:00
|
|
|
|
|
|
|
|
|
|
|
// Function: Q_Mul()
|
|
|
|
// Usage:
|
|
|
|
// Q_Mul(a, b)
|
2020-07-17 19:28:28 +00:00
|
|
|
// Description:
|
|
|
|
// Multiplies quaternion `a` by quaternion `b`.
|
|
|
|
// The returned quaternion is normalized if both `a` and `b` are normalized
|
|
|
|
function Q_Mul(a, b) =
|
2020-07-19 21:09:33 +00:00
|
|
|
assert( Q_is_quat(a) && Q_is_quat(b), "Invalid quaternion(s)")
|
2020-08-26 13:07:12 +00:00
|
|
|
_Qmul(a,b);
|
|
|
|
|
|
|
|
function _Qmul(a,b) =
|
2020-07-19 21:09:33 +00:00
|
|
|
[
|
|
|
|
a[3]*b.x + a.x*b[3] + a.y*b.z - a.z*b.y,
|
|
|
|
a[3]*b.y - a.x*b.z + a.y*b[3] + a.z*b.x,
|
|
|
|
a[3]*b.z + a.x*b.y - a.y*b.x + a.z*b[3],
|
2020-08-26 13:07:12 +00:00
|
|
|
a[3]*b[3] - a.x*b.x - a.y*b.y - a.z*b.z
|
2020-07-19 21:09:33 +00:00
|
|
|
];
|
2020-08-26 13:07:12 +00:00
|
|
|
// [ [a[3], -a.z, a.y, a.x],
|
|
|
|
// [ a.z, a[3], -a.x, a.y],
|
|
|
|
// [-a.y, a.x, a[3], a.z],
|
|
|
|
// [-a.x, -a.y, -a.z, a[3]] ]*[b.x,b.y,b.z,b[3]]
|
|
|
|
|
2019-03-23 04:13:18 +00:00
|
|
|
|
2020-04-15 02:16:24 +00:00
|
|
|
// Function: Q_Cumulative()
|
|
|
|
// Usage:
|
2020-08-26 13:07:12 +00:00
|
|
|
// Q_Cumulative(ql);
|
2020-04-15 02:16:24 +00:00
|
|
|
// Description:
|
|
|
|
// Given a list of Quaternions, cumulatively multiplies them, returning a list
|
|
|
|
// of each cumulative Quaternion product. It starts with the first quaternion
|
|
|
|
// given in the list, and applies successive quaternion rotations in list order.
|
2020-07-17 19:28:28 +00:00
|
|
|
// The quaternion in the returned list are normalized if each quaternion in v
|
|
|
|
// is normalized.
|
2020-08-26 13:07:12 +00:00
|
|
|
function Q_Cumulative(ql) =
|
|
|
|
assert( is_matrix(ql,undef,4) && len(ql)>0, "Invalid list of quaternions." )
|
|
|
|
[for( i = 0, q = ql[0];
|
|
|
|
i<=len(ql);
|
|
|
|
i = i+1, q = (i==len(ql))? 0: _Qmul(q,ql[i]) )
|
|
|
|
q ];
|
2020-04-15 02:16:24 +00:00
|
|
|
|
|
|
|
|
2019-03-23 04:13:18 +00:00
|
|
|
// Function: Q_Dot()
|
|
|
|
// Usage:
|
|
|
|
// Q_Dot(a, b)
|
|
|
|
// Description: Calculates the dot product between quaternions `a` and `b`.
|
2020-07-17 19:28:28 +00:00
|
|
|
function Q_Dot(a, b) =
|
2020-07-19 21:09:33 +00:00
|
|
|
assert( Q_is_quat(a) && Q_is_quat(b), "Invalid quaternion(s)" )
|
|
|
|
a*b;
|
2017-08-30 00:00:16 +00:00
|
|
|
|
2019-03-23 04:13:18 +00:00
|
|
|
// Function: Q_Neg()
|
|
|
|
// Usage:
|
|
|
|
// Q_Neg(q)
|
|
|
|
// Description: Returns the negative of quaternion `q`.
|
2020-07-17 19:28:28 +00:00
|
|
|
function Q_Neg(q) =
|
2020-07-19 21:09:33 +00:00
|
|
|
assert( Q_is_quat(q), "Invalid quaternion" )
|
|
|
|
-q;
|
2019-03-23 04:13:18 +00:00
|
|
|
|
|
|
|
|
|
|
|
// Function: Q_Conj()
|
|
|
|
// Usage:
|
|
|
|
// Q_Conj(q)
|
|
|
|
// Description: Returns the conjugate of quaternion `q`.
|
2020-07-17 19:28:28 +00:00
|
|
|
function Q_Conj(q) =
|
2020-08-26 13:07:12 +00:00
|
|
|
assert( Q_is_quat(q), str("Invalid quaternion",q) )
|
2020-07-19 21:09:33 +00:00
|
|
|
[-q.x, -q.y, -q.z, q[3]];
|
2020-07-17 19:28:28 +00:00
|
|
|
|
|
|
|
|
|
|
|
// Function: Q_Inverse()
|
|
|
|
// Usage:
|
|
|
|
// qc = Q_Inverse(q)
|
|
|
|
// Description: Returns the multiplication inverse of quaternion `q` that is normalized only if `q` is normalized.
|
|
|
|
function Q_Inverse(q) =
|
2020-07-19 21:09:33 +00:00
|
|
|
assert( Q_is_quat(q), "Invalid quaternion" )
|
2020-08-26 13:07:12 +00:00
|
|
|
// let(q = q/norm(q) )
|
2020-07-19 21:09:33 +00:00
|
|
|
[-q.x, -q.y, -q.z, q[3]];
|
2019-03-23 04:13:18 +00:00
|
|
|
|
|
|
|
|
|
|
|
// Function: Q_Norm()
|
|
|
|
// Usage:
|
|
|
|
// Q_Norm(q)
|
2020-07-17 19:28:28 +00:00
|
|
|
// Description:
|
|
|
|
// Returns the `norm()` "length" of quaternion `q`.
|
|
|
|
// Normalized quaternions have unitary norm.
|
|
|
|
function Q_Norm(q) =
|
2020-07-19 21:09:33 +00:00
|
|
|
assert( Q_is_quat(q), "Invalid quaternion" )
|
|
|
|
norm(q);
|
2019-03-23 04:13:18 +00:00
|
|
|
|
|
|
|
|
|
|
|
// Function: Q_Normalize()
|
|
|
|
// Usage:
|
|
|
|
// Q_Normalize(q)
|
|
|
|
// Description: Normalizes quaternion `q`, so that norm([W,X,Y,Z]) == 1.
|
2020-07-17 19:28:28 +00:00
|
|
|
function Q_Normalize(q) =
|
2020-07-19 21:09:33 +00:00
|
|
|
assert( Q_is_quat(q) , "Invalid quaternion" )
|
2020-08-26 13:07:12 +00:00
|
|
|
approx(_Qvec(q), [0,0,0])
|
|
|
|
? Q_Ident()
|
|
|
|
: q/norm(q);
|
2019-03-23 04:13:18 +00:00
|
|
|
|
|
|
|
|
|
|
|
// Function: Q_Dist()
|
|
|
|
// Usage:
|
|
|
|
// Q_Dist(q1, q2)
|
|
|
|
// Description: Returns the "distance" between two quaternions.
|
2020-07-17 19:28:28 +00:00
|
|
|
function Q_Dist(q1, q2) =
|
2020-07-19 21:09:33 +00:00
|
|
|
assert( Q_is_quat(q1) && Q_is_quat(q2), "Invalid quaternion(s)" )
|
|
|
|
norm(q2-q1);
|
2019-03-23 04:13:18 +00:00
|
|
|
|
|
|
|
|
|
|
|
// Function: Q_Slerp()
|
|
|
|
// Usage:
|
|
|
|
// Q_Slerp(q1, q2, u);
|
|
|
|
// Description:
|
|
|
|
// Returns a quaternion that is a spherical interpolation between two quaternions.
|
|
|
|
// Arguments:
|
|
|
|
// q1 = The first quaternion. (u=0)
|
|
|
|
// q2 = The second quaternion. (u=1)
|
|
|
|
// u = The proportional value, from 0 to 1, of what part of the interpolation to return.
|
2020-02-16 00:49:02 +00:00
|
|
|
// Example(3D): Giving `u` as a Scalar
|
|
|
|
// a = QuatY(-135);
|
|
|
|
// b = QuatXYZ([0,-30,30]);
|
|
|
|
// for (u=[0:0.1:1])
|
|
|
|
// Qrot(Q_Slerp(a, b, u))
|
|
|
|
// right(80) cube([10,10,1]);
|
|
|
|
// #sphere(r=80);
|
|
|
|
// Example(3D): Giving `u` as a Range
|
|
|
|
// a = QuatZ(-135);
|
|
|
|
// b = QuatXYZ([90,0,-45]);
|
|
|
|
// for (q = Q_Slerp(a, b, [0:0.1:1]))
|
|
|
|
// Qrot(q) right(80) cube([10,10,1]);
|
|
|
|
// #sphere(r=80);
|
2020-07-17 19:28:28 +00:00
|
|
|
function Q_Slerp(q1, q2, u, _dot) =
|
2020-08-26 13:07:12 +00:00
|
|
|
assert(is_finite(u) || is_range(u) || is_vector(u), "Invalid interpolation coefficient(s)")
|
|
|
|
assert(Q_is_quat(q1) && Q_is_quat(q2), "Invalid quaternion(s)" )
|
|
|
|
let(
|
|
|
|
_dot = q1*q2,
|
|
|
|
q1 = q1/norm(q1),
|
|
|
|
q2 = _dot<0 ? -q2/norm(q2) : q2/norm(q2),
|
|
|
|
dot = abs(_dot),
|
|
|
|
q3 = dot>0.9995? q2: _Qunit(q2 - dot*q1)
|
|
|
|
)
|
|
|
|
! is_num(u)
|
|
|
|
? [for (uu=u) _Qslerp(q1, q3, uu, dot)]
|
|
|
|
: _Qslerp(q1, q3, u, dot);
|
|
|
|
|
|
|
|
function _Qslerp(q1, q2, u, dot) =
|
|
|
|
dot>0.9995
|
|
|
|
? _Qunit(q1 + u*(q2-q1))
|
|
|
|
: let( theta = u*acos(dot) )
|
|
|
|
_Qunit(q1*cos(theta) + q2*sin(theta));
|
|
|
|
|
|
|
|
|
|
|
|
// Function: Q_to_matrix3()
|
|
|
|
// Usage:
|
|
|
|
// Q_to_matrix3(q);
|
2019-03-23 04:13:18 +00:00
|
|
|
// Description:
|
|
|
|
// Returns the 3x3 rotation matrix for the given normalized quaternion q.
|
2020-08-26 13:07:12 +00:00
|
|
|
function Q_to_matrix3(q) =
|
2020-07-19 21:09:33 +00:00
|
|
|
let( q = Q_Normalize(q) )
|
|
|
|
[
|
|
|
|
[1-2*q[1]*q[1]-2*q[2]*q[2], 2*q[0]*q[1]-2*q[2]*q[3], 2*q[0]*q[2]+2*q[1]*q[3]],
|
|
|
|
[ 2*q[0]*q[1]+2*q[2]*q[3], 1-2*q[0]*q[0]-2*q[2]*q[2], 2*q[1]*q[2]-2*q[0]*q[3]],
|
|
|
|
[ 2*q[0]*q[2]-2*q[1]*q[3], 2*q[1]*q[2]+2*q[0]*q[3], 1-2*q[0]*q[0]-2*q[1]*q[1]]
|
|
|
|
];
|
2017-08-30 00:00:16 +00:00
|
|
|
|
|
|
|
|
2020-08-26 13:07:12 +00:00
|
|
|
// Function: Q_to_matrix4()
|
2019-03-23 04:13:18 +00:00
|
|
|
// Usage:
|
2020-08-26 13:07:12 +00:00
|
|
|
// Q_to_matrix4(q);
|
2019-03-23 04:13:18 +00:00
|
|
|
// Description:
|
|
|
|
// Returns the 4x4 rotation matrix for the given normalized quaternion q.
|
2020-08-26 13:07:12 +00:00
|
|
|
function Q_to_matrix4(q) =
|
2020-07-19 21:09:33 +00:00
|
|
|
let( q = Q_Normalize(q) )
|
|
|
|
[
|
|
|
|
[1-2*q[1]*q[1]-2*q[2]*q[2], 2*q[0]*q[1]-2*q[2]*q[3], 2*q[0]*q[2]+2*q[1]*q[3], 0],
|
|
|
|
[ 2*q[0]*q[1]+2*q[2]*q[3], 1-2*q[0]*q[0]-2*q[2]*q[2], 2*q[1]*q[2]-2*q[0]*q[3], 0],
|
|
|
|
[ 2*q[0]*q[2]-2*q[1]*q[3], 2*q[1]*q[2]+2*q[0]*q[3], 1-2*q[0]*q[0]-2*q[1]*q[1], 0],
|
|
|
|
[ 0, 0, 0, 1]
|
|
|
|
];
|
2017-08-30 00:00:16 +00:00
|
|
|
|
|
|
|
|
2020-08-26 13:07:12 +00:00
|
|
|
// Function: Q_from_matrix()
|
|
|
|
// Usage:
|
|
|
|
// Q_from_matrix(M)
|
|
|
|
// Description:
|
|
|
|
// Returns a normalized quaternion corresponding to the rotation matrix M.
|
|
|
|
// M may be a 3x3 rotation matrix or a homogeneous 4x4 rotation matrix.
|
|
|
|
// The last row and last column of M are ignored for 4x4 matrices.
|
|
|
|
// It doesn't check whether M is in fact a rotation matrix.
|
|
|
|
// If M is not a rotation, the returned quaternion is unpredictable.
|
|
|
|
//
|
|
|
|
// based on https://en.wikipedia.org/wiki/Rotation_matrix
|
|
|
|
//
|
|
|
|
function Q_from_matrix(M) =
|
|
|
|
assert( is_matrix(M) && (len(M)==3 || len(M)==4) && (len(M[0])==3 || len(M[0])==4),
|
|
|
|
"The matrix should be 3x3 or 4x4")
|
|
|
|
let( tr = M[0][0]+M[1][1]+M[2][2] ) // M trace
|
|
|
|
tr>0
|
|
|
|
? let( r = sqrt(1+tr), s = -1/r/2 )
|
|
|
|
_Qunit( _Qset([ M[1][2]-M[2][1], M[2][0]-M[0][2], M[0][1]-M[1][0] ]*s, r/2 ) )
|
|
|
|
: let(
|
|
|
|
i = max_index([ M[0][0], M[1][1], M[2][2] ]),
|
|
|
|
r = sqrt(1 + 2*M[i][i] -M[0][0] -M[1][1] -M[2][2] ),
|
|
|
|
s = 1/r/2
|
|
|
|
)
|
|
|
|
i==0 ? _Qunit( _Qset( [ r/2, s*(M[1][0]+M[0][1]), s*(M[0][2]+M[2][0]) ], s*(M[2][1]-M[1][2])) ):
|
|
|
|
i==1 ? _Qunit( _Qset( [ s*(M[1][0]+M[0][1]), r/2, s*(M[2][1]+M[1][2]) ], s*(M[0][2]-M[2][0])) )
|
|
|
|
: _Qunit( _Qset( [ s*(M[2][0]+M[0][2]), s*(M[1][2]+M[2][1]), r/2 ], s*(M[1][0]-M[0][1])) ) ;
|
|
|
|
|
|
|
|
|
2019-03-23 04:13:18 +00:00
|
|
|
// Function: Q_Axis()
|
|
|
|
// Usage:
|
|
|
|
// Q_Axis(q)
|
|
|
|
// Description:
|
|
|
|
// Returns the axis of rotation of a normalized quaternion `q`.
|
2020-07-17 19:28:28 +00:00
|
|
|
// The input doesn't need to be normalized.
|
|
|
|
function Q_Axis(q) =
|
2020-07-19 21:09:33 +00:00
|
|
|
assert( Q_is_quat(q) , "Invalid quaternion" )
|
|
|
|
let( d = norm(_Qvec(q)) )
|
|
|
|
approx(d,0)? [0,0,1] : _Qvec(q)/d;
|
2019-02-06 11:32:27 +00:00
|
|
|
|
2019-03-23 04:13:18 +00:00
|
|
|
// Function: Q_Angle()
|
|
|
|
// Usage:
|
2020-07-17 19:28:28 +00:00
|
|
|
// a = Q_Angle(q)
|
|
|
|
// a12 = Q_Angle(q1,q2);
|
2019-03-23 04:13:18 +00:00
|
|
|
// Description:
|
2020-07-17 19:28:28 +00:00
|
|
|
// If only q1 is given, returns the angle of rotation (in degrees) of that quaternion.
|
|
|
|
// If both q1 and q2 are given, returns the angle (in degrees) between them.
|
|
|
|
// The input quaternions don't need to be normalized.
|
|
|
|
function Q_Angle(q1,q2) =
|
2020-07-19 21:09:33 +00:00
|
|
|
assert(Q_is_quat(q1) && (is_undef(q2) || Q_is_quat(q2)), "Invalid quaternion(s)" )
|
|
|
|
let( n1 = is_undef(q2)? norm(_Qvec(q1)): norm(q1) )
|
|
|
|
is_undef(q2)
|
|
|
|
? 2 * atan2(n1,_Qreal(q1))
|
|
|
|
: let( q1 = q1/norm(q1),
|
|
|
|
q2 = q2/norm(q2) )
|
|
|
|
4 * atan2(norm(q1 - q2), norm(q1 + q2));
|
2019-02-06 11:32:27 +00:00
|
|
|
|
2019-11-12 03:09:46 +00:00
|
|
|
// Function&Module: Qrot()
|
|
|
|
// Usage: As Module
|
2019-03-23 04:13:18 +00:00
|
|
|
// Qrot(q) ...
|
2019-11-12 03:09:46 +00:00
|
|
|
// Usage: As Function
|
|
|
|
// pts = Qrot(q,p);
|
2019-03-23 04:13:18 +00:00
|
|
|
// Description:
|
2019-11-12 03:09:46 +00:00
|
|
|
// When called as a module, rotates all children by the rotation stored in quaternion `q`.
|
|
|
|
// When called as a function with a `p` argument, rotates the point or list of points in `p` by the rotation stored in quaternion `q`.
|
|
|
|
// When called as a function without a `p` argument, returns the affine3d rotation matrix for the rotation stored in quaternion `q`.
|
2019-03-23 04:13:18 +00:00
|
|
|
// Example(FlatSpin):
|
2020-02-16 00:49:02 +00:00
|
|
|
// module shape() translate([80,0,0]) cube([10,10,1]);
|
|
|
|
// q = QuatXYZ([90,-15,-45]);
|
|
|
|
// Qrot(q) shape();
|
|
|
|
// #shape();
|
2019-11-12 03:09:46 +00:00
|
|
|
// Example(NORENDER):
|
|
|
|
// q = QuatXYZ([45,35,10]);
|
|
|
|
// mat4x4 = Qrot(q);
|
|
|
|
// Example(NORENDER):
|
|
|
|
// q = QuatXYZ([45,35,10]);
|
|
|
|
// pt = Qrot(q, p=[4,5,6]);
|
|
|
|
// Example(NORENDER):
|
|
|
|
// q = QuatXYZ([45,35,10]);
|
|
|
|
// pts = Qrot(q, p=[[2,3,4], [4,5,6], [9,2,3]]);
|
2017-08-30 00:00:16 +00:00
|
|
|
module Qrot(q) {
|
2020-08-26 13:07:12 +00:00
|
|
|
multmatrix(Q_to_matrix4(q)) {
|
2020-05-30 02:04:34 +00:00
|
|
|
children();
|
|
|
|
}
|
2017-08-30 00:00:16 +00:00
|
|
|
}
|
|
|
|
|
2019-11-12 03:09:46 +00:00
|
|
|
function Qrot(q,p) =
|
2020-08-26 13:07:12 +00:00
|
|
|
is_undef(p)? Q_to_matrix4(q) :
|
2020-07-19 21:09:33 +00:00
|
|
|
is_vector(p)? Qrot(q,[p])[0] :
|
2020-08-26 13:07:12 +00:00
|
|
|
apply(Q_to_matrix4(q), p);
|
2019-11-12 03:09:46 +00:00
|
|
|
|
2017-08-30 00:00:16 +00:00
|
|
|
|
2020-02-16 00:49:02 +00:00
|
|
|
// Module: Qrot_copies()
|
|
|
|
// Usage:
|
|
|
|
// Qrot_copies(quats) ...
|
|
|
|
// Description:
|
|
|
|
// For each quaternion given in the list `quats`, rotates to that orientation and creates a copy
|
|
|
|
// of all children. This is equivalent to `for (q=quats) Qrot(q) ...`.
|
|
|
|
// Arguments:
|
|
|
|
// quats = A list containing all quaternions to rotate to and create copies of all children for.
|
|
|
|
// Example:
|
|
|
|
// a = QuatZ(-135);
|
|
|
|
// b = QuatXYZ([0,-30,30]);
|
|
|
|
// Qrot_copies(Q_Slerp(a, b, [0:0.1:1]))
|
|
|
|
// right(80) cube([10,10,1]);
|
|
|
|
// #sphere(r=80);
|
|
|
|
module Qrot_copies(quats) for (q=quats) Qrot(q) children();
|
|
|
|
|
|
|
|
|
2020-07-17 19:28:28 +00:00
|
|
|
// Function&Module: Q_Rotation_path(q1, n, [q2])
|
2020-07-19 21:09:33 +00:00
|
|
|
// Usage: As a function
|
2020-07-17 19:28:28 +00:00
|
|
|
// path = Q_Rotation_path(q1, n, q2);
|
|
|
|
// path = Q_Rotation_path(q1, n);
|
2020-07-19 21:09:33 +00:00
|
|
|
// Usage: As a module
|
2020-07-17 19:28:28 +00:00
|
|
|
// Q_Rotation_path(q1, n, q2) ...
|
|
|
|
// Description:
|
|
|
|
// If q2 is undef and it is called as a function, the path, with length n+1 (n>=1), will be the
|
|
|
|
// cumulative multiplications of the matrix rotation of q1 by itself.
|
|
|
|
// If q2 is defined and it is called as a function, returns a rotation matrix path of length n+1 (n>=1)
|
|
|
|
// that interpolates two given rotation quaternions. The first matrix of the sequence is the
|
|
|
|
// matrix rotation of q1 and the last one, the matrix rotation of q2. The intermediary matrix
|
|
|
|
// rotations are an uniform interpolation of the path extreme matrices.
|
|
|
|
// When called as a module, applies to its children() each rotation of the sequence computed
|
|
|
|
// by the function.
|
|
|
|
// The input quaternions don't need to be normalized.
|
|
|
|
// Arguments:
|
|
|
|
// q1 = The quaternion of the first rotation.
|
|
|
|
// q2 = The quaternion of the last rotation.
|
|
|
|
// n = An integer defining the path length ( path length = n+1).
|
|
|
|
// Example(3D): as a function
|
|
|
|
// a = QuatY(-135);
|
|
|
|
// b = QuatXYZ([0,-30,30]);
|
|
|
|
// for (M=Q_Rotation_path(a, 10, b))
|
|
|
|
// multmatrix(M)
|
|
|
|
// right(80) cube([10,10,1]);
|
|
|
|
// #sphere(r=80);
|
|
|
|
// Example(3D): as a module
|
|
|
|
// a = QuatY(-135);
|
|
|
|
// b = QuatXYZ([0,-30,30]);
|
|
|
|
// Q_Rotation_path(a, 10, b)
|
|
|
|
// right(80) cube([10,10,1]);
|
|
|
|
// #sphere(r=80);
|
|
|
|
// Example(3D): as a function
|
|
|
|
// a = QuatY(5);
|
|
|
|
// for (M=Q_Rotation_path(a, 10))
|
|
|
|
// multmatrix(M)
|
|
|
|
// right(80) cube([10,10,1]);
|
|
|
|
// #sphere(r=80);
|
|
|
|
// Example(3D): as a module
|
|
|
|
// a = QuatY(5);
|
|
|
|
// Q_Rotation_path(a, 10)
|
|
|
|
// right(80) cube([10,10,1]);
|
|
|
|
// #sphere(r=80);
|
|
|
|
function Q_Rotation_path(q1, n=1, q2) =
|
2020-07-19 21:09:33 +00:00
|
|
|
assert( Q_is_quat(q1) && (is_undef(q2) || Q_is_quat(q2) ), "Invalid quaternion(s)" )
|
|
|
|
assert( is_finite(n) && n>=1 && n==floor(n), "Invalid integer" )
|
|
|
|
assert( is_undef(q2) || ! approx(norm(q1+q2),0), "Quaternions cannot be opposed" )
|
|
|
|
is_undef(q2)
|
2020-08-26 13:07:12 +00:00
|
|
|
? [for( i=0, dR=Q_to_matrix4(q1), R=dR; i<=n; i=i+1, R=dR*R ) R]
|
2020-07-19 21:09:33 +00:00
|
|
|
: let( q2 = Q_Normalize( q1*q2<0 ? -q2: q2 ),
|
2020-08-26 13:07:12 +00:00
|
|
|
dq = Q_pow( _Qmul( q2, Q_Inverse(q1) ), 1/n ),
|
|
|
|
dR = Q_to_matrix4(dq) )
|
|
|
|
[for( i=0, R=Q_to_matrix4(q1); i<=n; i=i+1, R=dR*R ) R];
|
2020-07-17 19:28:28 +00:00
|
|
|
|
|
|
|
module Q_Rotation_path(q1, n=1, q2) {
|
2020-07-19 21:09:33 +00:00
|
|
|
for(Mi=Q_Rotation_path(q1, n, q2))
|
|
|
|
multmatrix(Mi)
|
|
|
|
children();
|
2020-07-17 19:28:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Function: Q_Nlerp()
|
|
|
|
// Usage:
|
|
|
|
// q = Q_Nlerp(q1, q2, u);
|
|
|
|
// Description:
|
|
|
|
// Returns a quaternion that is a normalized linear interpolation between two quaternions
|
|
|
|
// when u is a number.
|
|
|
|
// If u is a list of numbers, computes the interpolations for each value in the
|
|
|
|
// list and returns the interpolated quaternions in a list.
|
|
|
|
// The input quaternions don't need to be normalized.
|
|
|
|
// Arguments:
|
|
|
|
// q1 = The first quaternion. (u=0)
|
|
|
|
// q2 = The second quaternion. (u=1)
|
|
|
|
// u = A value (or a list of values), between 0 and 1, of the proportion(s) of each quaternion in the interpolation.
|
|
|
|
// Example(3D): Giving `u` as a Scalar
|
|
|
|
// a = QuatY(-135);
|
|
|
|
// b = QuatXYZ([0,-30,30]);
|
|
|
|
// for (u=[0:0.1:1])
|
|
|
|
// Qrot(Q_Nlerp(a, b, u))
|
|
|
|
// right(80) cube([10,10,1]);
|
|
|
|
// #sphere(r=80);
|
|
|
|
// Example(3D): Giving `u` as a Range
|
|
|
|
// a = QuatZ(-135);
|
|
|
|
// b = QuatXYZ([90,0,-45]);
|
|
|
|
// for (q = Q_Nlerp(a, b, [0:0.1:1]))
|
|
|
|
// Qrot(q) right(80) cube([10,10,1]);
|
|
|
|
// #sphere(r=80);
|
|
|
|
function Q_Nlerp(q1,q2,u) =
|
2020-07-19 21:09:33 +00:00
|
|
|
assert(is_finite(u) || is_range(u) || is_vector(u) ,
|
|
|
|
"Invalid interpolation coefficient(s)" )
|
|
|
|
assert(Q_is_quat(q1) && Q_is_quat(q2), "Invalid quaternion(s)" )
|
|
|
|
assert( ! approx(norm(q1+q2),0), "Quaternions cannot be opposed" )
|
|
|
|
let( q1 = Q_Normalize(q1),
|
|
|
|
q2 = Q_Normalize(q2) )
|
|
|
|
is_num(u)
|
2020-08-26 13:07:12 +00:00
|
|
|
? _Qunit((1-u)*q1 + u*q2 )
|
|
|
|
: [for (ui=u) _Qunit((1-ui)*q1 + ui*q2 ) ];
|
2020-07-17 19:28:28 +00:00
|
|
|
|
|
|
|
|
|
|
|
// Function: Q_Squad()
|
|
|
|
// Usage:
|
|
|
|
// qn = Q_Squad(q1,q2,q3,q4,u);
|
|
|
|
// Description:
|
|
|
|
// Returns a quaternion that is a cubic spherical interpolation of the quaternions
|
|
|
|
// q1 and q4 taking the other two quaternions, q2 and q3, as parameter of a cubic
|
|
|
|
// on the sphere similar to the control points of a Bezier curve.
|
|
|
|
// If u is a number, usually between 0 and 1, returns the quaternion that results
|
|
|
|
// from the interpolation.
|
|
|
|
// If u is a list of numbers, computes the interpolations for each value in the
|
|
|
|
// list and returns the interpolated quaternions in a list.
|
|
|
|
// The input quaternions don't need to be normalized.
|
|
|
|
// Arguments:
|
|
|
|
// q1 = The start quaternion. (u=0)
|
|
|
|
// q1 = The first intermediate quaternion.
|
|
|
|
// q2 = The second intermediate quaternion.
|
|
|
|
// q4 = The end quaternion. (u=1)
|
|
|
|
// u = A value (or a list of values), of the proportion(s) of each quaternion in the cubic interpolation.
|
|
|
|
// Example(3D): Giving `u` as a Scalar
|
|
|
|
// a = QuatY(-135);
|
|
|
|
// b = QuatXYZ([-50,-50,120]);
|
|
|
|
// c = QuatXYZ([-50,-40,30]);
|
|
|
|
// d = QuatY(-45);
|
|
|
|
// color("red"){
|
|
|
|
// Qrot(b) right(80) cube([10,10,1]);
|
|
|
|
// Qrot(c) right(80) cube([10,10,1]);
|
|
|
|
// }
|
|
|
|
// for (u=[0:0.05:1])
|
|
|
|
// Qrot(Q_Squad(a, b, c, d, u))
|
|
|
|
// right(80) cube([10,10,1]);
|
|
|
|
// #sphere(r=80);
|
|
|
|
// Example(3D): Giving `u` as a Range
|
|
|
|
// a = QuatY(-135);
|
|
|
|
// b = QuatXYZ([-50,-50,120]);
|
|
|
|
// c = QuatXYZ([-50,-40,30]);
|
|
|
|
// d = QuatY(-45);
|
|
|
|
// for (q = Q_Squad(a, b, c, d, [0:0.05:1]))
|
|
|
|
// Qrot(q) right(80) cube([10,10,1]);
|
|
|
|
// #sphere(r=80);
|
|
|
|
function Q_Squad(q1,q2,q3,q4,u) =
|
2020-07-19 21:09:33 +00:00
|
|
|
assert(is_finite(u) || is_range(u) || is_vector(u) ,
|
|
|
|
"Invalid interpolation coefficient(s)" )
|
|
|
|
is_num(u)
|
|
|
|
? Q_Slerp( Q_Slerp(q1,q4,u), Q_Slerp(q2,q3,u), 2*u*(1-u))
|
|
|
|
: [for(ui=u) Q_Slerp( Q_Slerp(q1,q4,ui), Q_Slerp(q2,q3,ui), 2*ui*(1-ui) ) ];
|
2020-07-17 19:28:28 +00:00
|
|
|
|
|
|
|
|
2020-08-26 13:07:12 +00:00
|
|
|
function Q_Scubic(q1,q2,q3,q4,u) =
|
|
|
|
assert(is_finite(u) || is_range(u) || is_vector(u) ,
|
|
|
|
"Invalid interpolation coefficient(s)" )
|
|
|
|
is_num(u)
|
|
|
|
? let( q12 = Q_Slerp(q1,q2,u),
|
|
|
|
q23 = Q_Slerp(q2,q3,u),
|
|
|
|
q34 = Q_Slerp(q3,q4,u) )
|
|
|
|
Q_Slerp(Q_Slerp(q12,q23,u),Q_Slerp(q23,q34,u),u)
|
|
|
|
: [for(ui=u) Q_Scubic( q1,q2,q3,q4,ui) ];
|
|
|
|
|
|
|
|
|
2020-07-17 19:28:28 +00:00
|
|
|
// Function: Q_exp()
|
|
|
|
// Usage:
|
|
|
|
// q2 = Q_exp(q);
|
|
|
|
// Description:
|
|
|
|
// Returns the quaternion that is the exponential of the quaternion q in base e
|
|
|
|
// The returned quaternion is usually not normalized.
|
|
|
|
function Q_exp(q) =
|
2020-07-19 21:09:33 +00:00
|
|
|
assert( is_vector(q,4), "Input is not a valid quaternion")
|
|
|
|
let( nv = norm(_Qvec(q)) ) // q may be equal to zero here!
|
|
|
|
exp(_Qreal(q))*Quat(_Qvec(q),2*nv);
|
2020-07-17 19:28:28 +00:00
|
|
|
|
|
|
|
|
|
|
|
// Function: Q_ln()
|
|
|
|
// Usage:
|
|
|
|
// q2 = Q_ln(q);
|
|
|
|
// Description:
|
|
|
|
// Returns the quaternion that is the natural logarithm of the quaternion q.
|
|
|
|
// The returned quaternion is usually not normalized and may be zero.
|
|
|
|
function Q_ln(q) =
|
2020-07-19 21:09:33 +00:00
|
|
|
assert(Q_is_quat(q), "Input is not a valid quaternion")
|
|
|
|
let( nq = norm(q),
|
|
|
|
nv = norm(_Qvec(q)) )
|
|
|
|
approx(nv,0) ? _Qset([0,0,0] , ln(nq) ) :
|
|
|
|
_Qset(_Qvec(q)*atan2(nv,_Qreal(q))/nv, ln(nq));
|
2020-07-17 19:28:28 +00:00
|
|
|
|
|
|
|
|
|
|
|
// Function: Q_pow()
|
|
|
|
// Usage:
|
|
|
|
// q2 = Q_pow(q, r);
|
|
|
|
// Description:
|
|
|
|
// Returns the quaternion that is the power of the quaternion q to the real exponent r.
|
|
|
|
// The returned quaternion is normalized if `q` is normalized.
|
|
|
|
function Q_pow(q,r=1) =
|
2020-07-19 21:09:33 +00:00
|
|
|
assert( Q_is_quat(q) && is_finite(r),
|
|
|
|
"Invalid inputs")
|
|
|
|
let( theta = 2*atan2(norm(_Qvec(q)),_Qreal(q)) )
|
|
|
|
Quat(_Qvec(q), r*theta); // Q_exp(r*Q_ln(q));
|
2020-07-17 19:28:28 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
2020-05-30 02:04:34 +00:00
|
|
|
// vim: expandtab tabstop=4 shiftwidth=4 softtabstop=4 nowrap
|