17 #include <vsg/maths/mat4.h>
20 # pragma GCC diagnostic push
21 # pragma GCC diagnostic ignored "-Wpedantic"
23 #if defined(__clang__)
24 # pragma clang diagnostic push
25 # pragma clang diagnostic ignored "-Wgnu-anonymous-struct"
26 # pragma clang diagnostic ignored "-Wnested-anon-types"
43 value_type x, y, z, w;
50 value{v.x, v.y, v.z, v.w} {}
51 constexpr
t_quat(value_type in_x, value_type in_y, value_type in_z, value_type in_w) :
52 value{in_x, in_y, in_z, in_w} {}
55 set(angle_radians, axis);
64 value{
static_cast<T
>(v.x),
static_cast<T
>(v.y),
static_cast<T
>(v.z),
static_cast<T
>(v.w)} {}
68 constexpr std::size_t size()
const {
return 4; }
70 value_type& operator[](std::size_t i) {
return value[i]; }
71 value_type operator[](std::size_t i)
const {
return value[i]; }
76 value[0] =
static_cast<value_type
>(rhs[0]);
77 value[1] =
static_cast<value_type
>(rhs[1]);
78 value[2] =
static_cast<value_type
>(rhs[2]);
79 value[3] =
static_cast<value_type
>(rhs[3]);
83 T* data() {
return value; }
84 const T* data()
const {
return value; }
86 void set(value_type in_x, value_type in_y, value_type in_z, value_type in_w)
96 const value_type epsilon = 1e-7;
97 value_type len = length(axis);
105 value_type inversenorm = 1.0 / len;
106 value_type coshalfangle = cos(0.5 * angle_radians);
107 value_type sinhalfangle = sin(0.5 * angle_radians);
109 x = axis.x * sinhalfangle * inversenorm;
110 y = axis.y * sinhalfangle * inversenorm;
111 z = axis.z * sinhalfangle * inversenorm;
117 const value_type epsilon = 1e-7;
119 value_type dot_pd = vsg::dot(from, to);
120 value_type div = std::sqrt(length2(from) * length2(to));
122 if (div - dot_pd < epsilon)
124 axis = orthogonal(from);
128 axis = cross(from, to);
131 value_type len = length(axis);
133 double angle_radians = acos(dot_pd / div);
135 value_type inversenorm = 1.0 / len;
136 value_type coshalfangle = cos(0.5 * angle_radians);
137 value_type sinhalfangle = sin(0.5 * angle_radians);
139 x = axis.x * sinhalfangle * inversenorm;
140 y = axis.y * sinhalfangle * inversenorm;
141 z = axis.z * sinhalfangle * inversenorm;
155 return lhs[0] == rhs[0] && lhs[1] == rhs[1] && lhs[2] == rhs[2] && lhs[3] == rhs[3];
159 constexpr
bool operator!=(
const t_quat<T>& lhs,
const t_quat<T>& rhs)
161 return lhs[0] != rhs[0] || lhs[1] != rhs[1] || lhs[2] != rhs[2] || lhs[3] != rhs[3];
165 constexpr
bool operator<(
const t_quat<T>& lhs,
const t_quat<T>& rhs)
167 if (lhs[0] < rhs[0])
return true;
168 if (lhs[0] > rhs[0])
return false;
169 if (lhs[1] < rhs[1])
return true;
170 if (lhs[1] > rhs[1])
return false;
171 if (lhs[2] < rhs[2])
return true;
172 if (lhs[2] > rhs[2])
return false;
173 return lhs[3] < rhs[3];
177 constexpr t_quat<T> operator-(
const t_quat<T>& lhs,
const t_quat<T>& rhs)
179 return t_quat<T>(lhs[0] - rhs[0], lhs[1] - rhs[1], lhs[2] - rhs[2], lhs[3] - rhs[3]);
183 constexpr t_quat<T> conjugate(
const t_quat<T>& v)
185 return t_quat<T>(-v[0], -v[1], -v[2], -v[3]);
189 constexpr t_quat<T> operator-(
const t_quat<T>& v)
195 constexpr t_quat<T> operator+(
const t_quat<T>& lhs,
const t_quat<T>& rhs)
197 return t_quat<T>(lhs[0] + rhs[0], lhs[1] + rhs[1], lhs[2] + rhs[2], lhs[3] + rhs[3]);
202 constexpr t_quat<T> operator*(
const t_quat<T>& lhs,
const t_quat<T>& rhs)
204 t_quat<T> q(rhs[3] * lhs[0] + rhs[0] * lhs[3] + rhs[1] * lhs[2] - rhs[2] * lhs[1],
205 rhs[3] * lhs[1] - rhs[0] * lhs[2] + rhs[1] * lhs[3] + rhs[2] * lhs[0],
206 rhs[3] * lhs[2] + rhs[0] * lhs[1] - rhs[1] * lhs[0] + rhs[2] * lhs[3],
207 rhs[3] * lhs[3] - rhs[0] * lhs[0] - rhs[1] * lhs[1] - rhs[2] * lhs[2]);
214 constexpr t_vec3<T> operator*(
const t_quat<T>& q,
const t_vec3<T>& v)
218 t_vec3<T> qvec(q[0], q[1], q[2]);
220 uuv = cross(qvec, uv);
228 constexpr t_quat<T> operator*(
const t_quat<T>& lhs, T rhs)
230 return t_quat<T>(lhs[0] * rhs, lhs[1] * rhs, lhs[2] * rhs, lhs[3] * rhs);
234 constexpr t_quat<T> operator/(
const t_quat<T>& lhs, T rhs)
236 T inv =
static_cast<T
>(1.0) / rhs;
237 return t_quat<T>(lhs[0] * inv, lhs[1] * inv, lhs[2] * inv, lhs[3] * inv);
241 constexpr T length(
const t_quat<T>& v)
243 return std::sqrt(v[0] * v[0] + v[1] * v[1] + v[2] * v[2] + v[3] * v[3]);
247 constexpr t_quat<T> normalize(
const t_quat<T>& v)
249 T inverse_len =
static_cast<T
>(1.0) / length(v);
250 return t_quat<T>(v[0] * inverse_len, v[1] * inverse_len, v[2] * inverse_len, v[3] * inverse_len);
254 constexpr T dot(
const t_quat<T>& lhs,
const t_quat<T>& rhs)
256 return lhs[0] * rhs[0] + lhs[1] * rhs[1] + lhs[2] * rhs[2] + lhs[3] * rhs[3];
260 constexpr t_quat<T> inverse(
const t_quat<T>& v)
262 t_quat<T> c = conjugate(v);
263 T inverse_len =
static_cast<T
>(1.0) / length(v);
264 return t_quat<T>(c[0] * inverse_len, c[1] * inverse_len, c[2] * inverse_len, c[3] * inverse_len);
268 constexpr t_quat<T> mix(
const t_quat<T>& from, t_quat<T> to, T r)
270 T epsilon = std::numeric_limits<T>::epsilon();
273 T cosomega = dot(from, to);
276 cosomega = -cosomega;
283 if ((one - cosomega) > epsilon)
285 T omega = acos(cosomega);
286 T sinomega = sin(omega);
287 T scale_from = sin((one - r) * omega) / sinomega;
288 T scale_to = sin(r * omega) / sinomega;
289 return (from * scale_from) + (to * scale_to);
294 return (from * (one - r)) + (to * r);
300 #if defined(__clang__)
301 # pragma clang diagnostic pop
303 #if defined(__GNUC__)
304 # pragma GCC diagnostic pop
t_quat template class that a represents quaternion
Definition: quat.h:35