https://mooseframework.inl.gov
EulerAngles.C
Go to the documentation of this file.
1 //* This file is part of the MOOSE framework
2 //* https://mooseframework.inl.gov
3 //*
4 //* All rights reserved, see COPYRIGHT for full restrictions
5 //* https://github.com/idaholab/moose/blob/master/COPYRIGHT
6 //*
7 //* Licensed under LGPL 2.1, please see LICENSE for details
8 //* https://www.gnu.org/licenses/lgpl-2.1.html
9 
10 #include "libmesh/ignore_warnings.h"
11 #include "EulerAngles.h"
12 #include "MooseRandom.h"
13 
15 {
16  phi1 = 0.0;
17  Phi = 0.0;
18  phi2 = 0.0;
19 }
20 
21 EulerAngles::EulerAngles(const Eigen::Quaternion<Real> & q)
22 {
23  phi1 = std::atan2((q.x() * q.z() + q.w() * q.y()), -(-q.w() * q.x() + q.y() * q.z())) *
24  (180.0 / libMesh::pi);
25 
26  // avoid NAN value from sqrt(val) while computing Phi
27  auto val = 1.0 - std::pow(q.w() * q.w() - q.x() * q.x() - q.y() * q.y() + q.z() * q.z(), 2.0);
28  if (val < 0.0 && !MooseUtils::absoluteFuzzyEqual(val, 0.0))
29  mooseError("Euler angle conversion is not successful due to invalid quaternion value.");
30 
31  Phi = std::atan2(std::sqrt(std::abs(val)),
32  q.w() * q.w() - q.x() * q.x() - q.y() * q.y() + q.z() * q.z()) *
33  (180.0 / libMesh::pi);
34  phi2 = std::atan2((q.x() * q.z() - q.w() * q.y()), (q.w() * q.x() + q.y() * q.z())) *
35  (180.0 / libMesh::pi);
36 
37  // Following checks and updates are done only to comply with bunge euler angle definitions, 0.0
38  // <= phi1/phi2 <= 360.0
39  if (phi1 < 0.0)
40  phi1 += 360.0;
41  if (phi2 < 0.0)
42  phi2 += 360.0;
43  if (Phi < 0.0)
44  mooseError("Euler angle out of range.");
45 }
46 
47 EulerAngles::EulerAngles(const Real & v0, const Real & v1, const Real & v2)
48 {
49  phi1 = v0;
50  Phi = v1;
51  phi2 = v2;
52 }
53 
54 Eigen::Quaternion<Real>
56 {
57  Eigen::Quaternion<Real> q;
58 
59  Real cPhi1PlusPhi2, cphi, cPhi1MinusPhi2;
60  Real sPhi1PlusPhi2, sphi, sPhi1MinusPhi2;
61 
67  cPhi1PlusPhi2 = std::cos((phi1 * libMesh::pi / 180.0 + phi2 * libMesh::pi / 180.0) / 2.0);
68  cphi = std::cos(Phi * libMesh::pi / 360.0);
69  cPhi1MinusPhi2 = std::cos((phi1 * libMesh::pi / 180.0 - phi2 * libMesh::pi / 180.0) / 2.0);
70 
71  sPhi1PlusPhi2 = std::sin((phi1 * libMesh::pi / 180.0 + phi2 * libMesh::pi / 180.0) / 2.0);
72  sphi = std::sin(Phi * libMesh::pi / 360.0);
73  sPhi1MinusPhi2 = std::sin((phi1 * libMesh::pi / 180.0 - phi2 * libMesh::pi / 180.0) / 2.0);
74 
75  q.w() = cphi * cPhi1PlusPhi2;
76  q.x() = sphi * cPhi1MinusPhi2;
77  q.y() = sphi * sPhi1MinusPhi2;
78  q.z() = cphi * sPhi1PlusPhi2;
79 
80  return q;
81 }
82 
83 void
85 {
86  phi1 = MooseRandom::rand() * 360.0;
87  Phi = std::acos(1.0 - 2.0 * MooseRandom::rand()) / libMesh::pi * 180.0;
88  phi2 = MooseRandom::rand() * 360;
89 }
90 
91 void
93 {
94  phi1 = random.rand(0) * 360.0;
95  Phi = std::acos(1.0 - 2.0 * random.rand(0)) / libMesh::pi * 180.0;
96  phi2 = random.rand(0) * 360;
97 }
98 #include "libmesh/restore_warnings.h"
void mooseError(Args &&... args)
void random()
Definition: EulerAngles.C:84
DIE A HORRIBLE DEATH HERE typedef LIBMESH_DEFAULT_SCALAR_TYPE Real
static Real rand()
MooseUnits pow(const MooseUnits &, int)
const Real pi
Eigen::Quaternion< Real > toQuaternion()
Definition: EulerAngles.C:55