From 4d64dcda3cf5c0ccde53674337ecdb57b2e6d5b3 Mon Sep 17 00:00:00 2001 From: Jonathan Youett <youett@mi.fu-berlin.de> Date: Tue, 18 Oct 2011 12:19:15 +0000 Subject: [PATCH] Add the cayley and inverse cayley mapping. By now these formulas use the matrix representation of a skew matrix/rotation which is not very pretty. I will replace it by a quaternion counterpart when I figured out how to do it properly [[Imported from SVN: r7943]] --- dune/gfe/rotation.hh | 60 ++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 60 insertions(+) diff --git a/dune/gfe/rotation.hh b/dune/gfe/rotation.hh index 2b82d924..c1b6748b 100644 --- a/dune/gfe/rotation.hh +++ b/dune/gfe/rotation.hh @@ -450,6 +450,66 @@ public: return APseudoInv; } + /** \brief The cayley mapping from \f$ \mathfrak{so}(3) \f$ to \f$ SO(3) \f$. */ + static Rotation<3,T> cayley(const SkewMatrix<3,T>& s) { + Rotation<3,T> q; + + Dune::FieldVector<T,3> vAxial = v.axial(); + T norm = 0.25*vAxial.two_norm2() + 1; + + Dune::FieldMatrix<T,3,3> mat = s.toMatrix(); + mat *= 0.5; + Dune::FieldMatrix<T,3,3> skewSquare = mat.rightmultiply(mat); + mat += skewSquare; + mat *= 2/norm; + + for (int i=0;i<3;i++) + mat[i][i] += 1; + + q.set(mat); + return q; + } + + /** \brief The inverse of the cayley mapping. */ + static SkewMatrix<3,T> cayleyInv(const Rotation<3,T> q) { + + Dune::FieldMatrix<T,3,3> mat; + + // compute the trace of the rotation matrix + double trace = -(*this)[0]*(*this)[0] -(*this)[1]*(*this)[1] -(*this)[2]*(*this)[2]+3*(*this)[3]*(*this)[3]; + + if ( (trace+1)>1e-6 && (trace+1)<-1e-6) { // if this term doesn't vanish we can use a direct formula + + (*this).matrix(mat); + Dune::FieldMatrix<T,3,3> matT; + (*this).inverse().matrix(matT); + mat -= matT; + mat *= 1/(1+trace); + } + else { // use the formula that involves the computation of an inverse + Dune::FieldMatrix<T,3,3> inv; + (*this).matrix(inv); + Dune::FieldMatrix<T,3,3> notInv = inv; + + for (int i=0;i<3;i++) { + inv[i][i] +=1; + notInv[i][i] -=1; + } + inv.invert(); + mat = notInv.leftmultiply(inv); + mat *= 2; + } + + // result is a skew symmetric matrix + SkewMatrix<3,T> res; + res.axial()[0] = mat[2][1]; + res.axial()[1] = mat[0][2]; + res.axial()[2] = mat[1][0]; + + return res; + + } + static T distance(const Rotation<3,T>& a, const Rotation<3,T>& b) { Quaternion<T> diff = a; -- GitLab