198 lines
5.2 KiB
C++
198 lines
5.2 KiB
C++
/* -*- C++ -*- ------------------------------------------------------------
|
|
|
|
Copyright (c) 2007 Jesse Anders and Demian Nave http://cmldev.net/
|
|
|
|
The Configurable Math Library (CML) is distributed under the terms of the
|
|
Boost Software License, v1.0 (see cml/LICENSE for details).
|
|
|
|
*-----------------------------------------------------------------------*/
|
|
/** @file
|
|
* @brief Defines an operator for quaternion conjugation.
|
|
*/
|
|
|
|
#ifndef conjugate_h
|
|
#define conjugate_h
|
|
|
|
#include <cml/quaternion/quaternion_expr.h>
|
|
|
|
namespace cml {
|
|
namespace et {
|
|
|
|
/** An expression node for conjugating a quaternion. */
|
|
template<class ExprT>
|
|
class ConjugateOp
|
|
{
|
|
public:
|
|
|
|
typedef ConjugateOp<ExprT> expr_type;
|
|
|
|
/* Record ary-ness of the expression: */
|
|
typedef unary_expression expr_ary;
|
|
|
|
/* Copy the expression by value into higher-up expressions: */
|
|
typedef expr_type expr_const_reference;
|
|
|
|
typedef typename ExprT::value_type value_type;
|
|
typedef quaternion_result_tag result_tag;
|
|
typedef typename ExprT::size_tag size_tag;
|
|
|
|
/* Store the expression traits for the subexpression: */
|
|
typedef ExprTraits<ExprT> expr_traits;
|
|
|
|
/* Reference type for the subexpression: */
|
|
typedef typename expr_traits::const_reference expr_reference;
|
|
|
|
/* Get the result type (same as for subexpression): */
|
|
typedef typename expr_traits::result_type result_type;
|
|
|
|
/* For matching by assignability: */
|
|
typedef cml::et::not_assignable_tag assignable_tag;
|
|
|
|
/* Get the temporary type: */
|
|
typedef typename result_type::temporary_type temporary_type;
|
|
|
|
/* Get the vector type: */
|
|
typedef typename result_type::vector_type vector_type;
|
|
|
|
/* Get the imaginary part type: */
|
|
typedef typename vector_type::subvector_type imaginary_type;
|
|
|
|
/* Record the order type: */
|
|
typedef typename result_type::order_type order_type;
|
|
|
|
|
|
public:
|
|
|
|
/** Record result size as an enum. */
|
|
enum { array_size = ExprT::array_size };
|
|
|
|
/** Localize the ordering as an enum. */
|
|
enum {
|
|
W = order_type::W,
|
|
X = order_type::X,
|
|
Y = order_type::Y,
|
|
Z = order_type::Z
|
|
};
|
|
|
|
|
|
public:
|
|
|
|
/** Return the real part of the expression. */
|
|
value_type real() const {
|
|
return m_expr.real();
|
|
}
|
|
|
|
/** Return the vector part of the expression. */
|
|
imaginary_type imaginary() const {
|
|
return -m_expr.imaginary();
|
|
}
|
|
|
|
/** Return the Cayley norm of the expression. */
|
|
value_type norm() const {
|
|
return length_squared();
|
|
}
|
|
|
|
/** Return square of the quaternion length. */
|
|
value_type length_squared() const {
|
|
return dot(
|
|
QuaternionXpr<expr_type>(*this),
|
|
QuaternionXpr<expr_type>(*this));
|
|
}
|
|
|
|
/** Return the quaternion length. */
|
|
value_type length() const {
|
|
return std::sqrt(length_squared());
|
|
}
|
|
|
|
/** Return the result as a normalized quaternion. */
|
|
temporary_type normalize() const {
|
|
temporary_type q(QuaternionXpr<expr_type>(*this));
|
|
return q.normalize();
|
|
}
|
|
|
|
/** Compute conjugated result at index i.
|
|
*
|
|
* The conjugate of quaternion s + v is s - v.
|
|
*/
|
|
value_type operator[](size_t i) const {
|
|
return (i == W) ? m_expr[W] : - m_expr[i] ;
|
|
}
|
|
|
|
|
|
public:
|
|
|
|
/** Return size of this expression (same as argument's size). */
|
|
size_t size() const {
|
|
return m_expr.size();
|
|
}
|
|
|
|
/** Return reference to contained expression. */
|
|
expr_reference expression() const { return m_expr; }
|
|
|
|
|
|
public:
|
|
|
|
/** Construct from the subexpression. */
|
|
explicit ConjugateOp(expr_reference expr) : m_expr(expr) {}
|
|
|
|
/** Copy constructor. */
|
|
ConjugateOp(const expr_type& e) : m_expr(e.m_expr) {}
|
|
|
|
|
|
protected:
|
|
|
|
expr_reference m_expr;
|
|
|
|
|
|
private:
|
|
|
|
/* Cannot be assigned to: */
|
|
expr_type& operator=(const expr_type&);
|
|
};
|
|
|
|
/** Expression traits class for ConjugateOp<>. */
|
|
template<class ExprT>
|
|
struct ExprTraits< ConjugateOp<ExprT> >
|
|
{
|
|
typedef ConjugateOp<ExprT> expr_type;
|
|
typedef ExprT arg_type;
|
|
|
|
typedef typename expr_type::value_type value_type;
|
|
typedef typename expr_type::expr_const_reference const_reference;
|
|
typedef typename expr_type::result_tag result_tag;
|
|
typedef typename expr_type::size_tag size_tag;
|
|
typedef typename expr_type::result_type result_type;
|
|
typedef typename expr_type::assignable_tag assignable_tag;
|
|
typedef expr_node_tag node_tag;
|
|
|
|
value_type get(const expr_type& v, size_t i) const { return v[i]; }
|
|
size_t size(const expr_type& e) const { return e.size(); }
|
|
};
|
|
|
|
} // namespace et
|
|
|
|
/** Conjugation of a quaternion. */
|
|
template<typename E, class AT, class OT, class CT> inline
|
|
et::QuaternionXpr< et::ConjugateOp< quaternion<E,AT,OT,CT> > >
|
|
conjugate(const quaternion<E,AT,OT,CT>& arg)
|
|
{
|
|
typedef et::ConjugateOp< quaternion<E,AT,OT,CT> > ExprT;
|
|
return et::QuaternionXpr<ExprT>(ExprT(arg));
|
|
}
|
|
|
|
/** Conjugation of a QuaternionXpr. */
|
|
template<class XprT> inline
|
|
et::QuaternionXpr< et::ConjugateOp<XprT> >
|
|
conjugate(QUATXPR_ARG_TYPE arg)
|
|
{
|
|
typedef et::ConjugateOp<XprT> ExprT;
|
|
return et::QuaternionXpr<ExprT>(ExprT(arg.expression()));
|
|
}
|
|
|
|
} // namespace cml
|
|
|
|
#endif
|
|
|
|
// -------------------------------------------------------------------------
|
|
// vim:ft=cpp
|