no 'dangling' functions

@staticmethod is what we need here
This commit is contained in:
Martin Diehl 2020-02-21 10:45:14 +01:00
parent 1e1cb3f151
commit c84a6e90c9
1 changed files with 427 additions and 427 deletions

View File

@ -4,6 +4,13 @@ from . import Lambert
P = -1 P = -1
def isone(a):
return np.isclose(a,1.0,atol=1.0e-7,rtol=0.0)
def iszero(a):
return np.isclose(a,0.0,atol=1.0e-12,rtol=0.0)
#################################################################################################### ####################################################################################################
class Rotation: class Rotation:
u""" u"""
@ -183,7 +190,7 @@ class Rotation:
return angles in degrees. return angles in degrees.
""" """
eu = qu2eu(self.quaternion) eu = Rotation.qu2eu(self.quaternion)
if degrees: eu = np.degrees(eu) if degrees: eu = np.degrees(eu)
return eu return eu
@ -201,13 +208,13 @@ class Rotation:
return tuple of axis and angle. return tuple of axis and angle.
""" """
ax = qu2ax(self.quaternion) ax = Rotation.qu2ax(self.quaternion)
if degrees: ax[3] = np.degrees(ax[3]) if degrees: ax[3] = np.degrees(ax[3])
return (ax[:3],np.degrees(ax[3])) if pair else ax return (ax[:3],np.degrees(ax[3])) if pair else ax
def asMatrix(self): def asMatrix(self):
"""Rotation matrix.""" """Rotation matrix."""
return qu2om(self.quaternion) return Rotation.qu2om(self.quaternion)
def asRodrigues(self, def asRodrigues(self,
vector = False): vector = False):
@ -221,16 +228,16 @@ class Rotation:
return as actual Rodrigues--Frank vector, i.e. rotation axis scaled by tan(ω/2). return as actual Rodrigues--Frank vector, i.e. rotation axis scaled by tan(ω/2).
""" """
ro = qu2ro(self.quaternion) ro = Rotation.qu2ro(self.quaternion)
return ro[:3]*ro[3] if vector else ro return ro[:3]*ro[3] if vector else ro
def asHomochoric(self): def asHomochoric(self):
"""Homochoric vector: (h_1, h_2, h_3).""" """Homochoric vector: (h_1, h_2, h_3)."""
return qu2ho(self.quaternion) return Rotation.qu2ho(self.quaternion)
def asCubochoric(self): def asCubochoric(self):
"""Cubochoric vector: (c_1, c_2, c_3).""" """Cubochoric vector: (c_1, c_2, c_3)."""
return qu2cu(self.quaternion) return Rotation.qu2cu(self.quaternion)
def asM(self): def asM(self):
""" """
@ -276,7 +283,7 @@ class Rotation:
if np.any(eu < 0.0) or np.any(eu > 2.0*np.pi) or eu[1] > np.pi: if np.any(eu < 0.0) or np.any(eu > 2.0*np.pi) or eu[1] > np.pi:
raise ValueError('Euler angles outside of [0..2π],[0..π],[0..2π].\n{} {} {}.'.format(*eu)) raise ValueError('Euler angles outside of [0..2π],[0..π],[0..2π].\n{} {} {}.'.format(*eu))
return Rotation(eu2qu(eu)) return Rotation(Rotation.eu2qu(eu))
@staticmethod @staticmethod
def fromAxisAngle(angleAxis, def fromAxisAngle(angleAxis,
@ -294,7 +301,7 @@ class Rotation:
if not np.isclose(np.linalg.norm(ax[0:3]), 1.0): if not np.isclose(np.linalg.norm(ax[0:3]), 1.0):
raise ValueError('Axis angle rotation axis is not of unit length.\n{} {} {}'.format(*ax[0:3])) raise ValueError('Axis angle rotation axis is not of unit length.\n{} {} {}'.format(*ax[0:3]))
return Rotation(ax2qu(ax)) return Rotation(Rotation.ax2qu(ax))
@staticmethod @staticmethod
def fromBasis(basis, def fromBasis(basis,
@ -316,7 +323,7 @@ class Rotation:
or not np.isclose(np.dot(om[2],om[0]), 0.0): or not np.isclose(np.dot(om[2],om[0]), 0.0):
raise ValueError('matrix is not orthogonal.\n{}'.format(om)) raise ValueError('matrix is not orthogonal.\n{}'.format(om))
return Rotation(om2qu(om)) return Rotation(Rotation.om2qu(om))
@staticmethod @staticmethod
def fromMatrix(om, def fromMatrix(om,
@ -338,7 +345,7 @@ class Rotation:
if ro[3] < 0.0: if ro[3] < 0.0:
raise ValueError('Rodriques rotation angle not positive.\n'.format(ro[3])) raise ValueError('Rodriques rotation angle not positive.\n'.format(ro[3]))
return Rotation(ro2qu(ro)) return Rotation(Rotation.ro2qu(ro))
@staticmethod @staticmethod
def fromHomochoric(homochoric, def fromHomochoric(homochoric,
@ -348,7 +355,7 @@ class Rotation:
else np.array(homochoric,dtype=float) else np.array(homochoric,dtype=float)
if P > 0: ho *= -1 # convert from P=1 to P=-1 if P > 0: ho *= -1 # convert from P=1 to P=-1
return Rotation(ho2qu(ho)) return Rotation(Rotation.ho2qu(ho))
@staticmethod @staticmethod
def fromCubochoric(cubochoric, def fromCubochoric(cubochoric,
@ -356,10 +363,10 @@ class Rotation:
cu = cubochoric if isinstance(cubochoric, np.ndarray) and cubochoric.dtype == np.dtype(float) \ cu = cubochoric if isinstance(cubochoric, np.ndarray) and cubochoric.dtype == np.dtype(float) \
else np.array(cubochoric,dtype=float) else np.array(cubochoric,dtype=float)
ho = cu2ho(cu) ho = Rotation.cu2ho(cu)
if P > 0: ho *= -1 # convert from P=1 to P=-1 if P > 0: ho *= -1 # convert from P=1 to P=-1
return Rotation(ho2qu(ho)) return Rotation(Rotation.ho2qu(ho))
@staticmethod @staticmethod
@ -437,15 +444,8 @@ class Rotation:
# OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE # OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE
# USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. # USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
#################################################################################################### ####################################################################################################
def isone(a):
return np.isclose(a,1.0,atol=1.0e-7,rtol=0.0)
def iszero(a):
return np.isclose(a,0.0,atol=1.0e-12,rtol=0.0)
#---------- Quaternion ---------- #---------- Quaternion ----------
@staticmethod
def qu2om(qu): def qu2om(qu):
"""Quaternion to rotation matrix.""" """Quaternion to rotation matrix."""
qq = qu[0]**2-(qu[1]**2 + qu[2]**2 + qu[3]**2) qq = qu[0]**2-(qu[1]**2 + qu[2]**2 + qu[3]**2)
@ -459,7 +459,7 @@ def qu2om(qu):
om[2,0] = 2.0*(qu[3]*qu[1]-qu[0]*qu[2]) om[2,0] = 2.0*(qu[3]*qu[1]-qu[0]*qu[2])
return om if P > 0.0 else om.T return om if P > 0.0 else om.T
@staticmethod
def qu2eu(qu): def qu2eu(qu):
"""Quaternion to Bunge-Euler angles.""" """Quaternion to Bunge-Euler angles."""
q03 = qu[0]**2+qu[3]**2 q03 = qu[0]**2+qu[3]**2
@ -478,7 +478,7 @@ def qu2eu(qu):
eu = np.where(eu<0, (eu+2.0*np.pi)%np.array([2.0*np.pi,np.pi,2.0*np.pi]),eu) eu = np.where(eu<0, (eu+2.0*np.pi)%np.array([2.0*np.pi,np.pi,2.0*np.pi]),eu)
return eu return eu
@staticmethod
def qu2ax(qu): def qu2ax(qu):
""" """
Quaternion to axis angle pair. Quaternion to axis angle pair.
@ -496,7 +496,7 @@ def qu2ax(qu):
return np.array(ax) return np.array(ax)
@staticmethod
def qu2ro(qu): def qu2ro(qu):
"""Quaternion to Rodriques-Frank vector.""" """Quaternion to Rodriques-Frank vector."""
if iszero(qu[0]): if iszero(qu[0]):
@ -508,7 +508,7 @@ def qu2ro(qu):
return np.array(ro) return np.array(ro)
@staticmethod
def qu2ho(qu): def qu2ho(qu):
"""Quaternion to homochoric vector.""" """Quaternion to homochoric vector."""
omega = 2.0 * np.arccos(np.clip(qu[0],-1.0,1.0)) # avoid numerical difficulties omega = 2.0 * np.arccos(np.clip(qu[0],-1.0,1.0)) # avoid numerical difficulties
@ -522,23 +522,23 @@ def qu2ho(qu):
return ho return ho
@staticmethod
def qu2cu(qu): def qu2cu(qu):
"""Quaternion to cubochoric vector.""" """Quaternion to cubochoric vector."""
return ho2cu(qu2ho(qu)) return Rotation.ho2cu(Rotation.qu2ho(qu))
#---------- Rotation matrix ---------- #---------- Rotation matrix ----------
@staticmethod
def om2qu(om): def om2qu(om):
""" """
Rotation matrix to quaternion. Rotation matrix to quaternion.
The original formulation (direct conversion) had (numerical?) issues The original formulation (direct conversion) had (numerical?) issues
""" """
return eu2qu(om2eu(om)) return Rotation.eu2qu(Rotation.om2eu(om))
@staticmethod
def om2eu(om): def om2eu(om):
"""Rotation matrix to Bunge-Euler angles.""" """Rotation matrix to Bunge-Euler angles."""
if abs(om[2,2]) < 1.0: if abs(om[2,2]) < 1.0:
@ -553,7 +553,7 @@ def om2eu(om):
eu = np.where(eu<0, (eu+2.0*np.pi)%np.array([2.0*np.pi,np.pi,2.0*np.pi]),eu) eu = np.where(eu<0, (eu+2.0*np.pi)%np.array([2.0*np.pi,np.pi,2.0*np.pi]),eu)
return eu return eu
@staticmethod
def om2ax(om): def om2ax(om):
"""Rotation matrix to axis angle pair.""" """Rotation matrix to axis angle pair."""
ax=np.empty(4) ax=np.empty(4)
@ -574,24 +574,24 @@ def om2ax(om):
return np.array(ax) return np.array(ax)
@staticmethod
def om2ro(om): def om2ro(om):
"""Rotation matrix to Rodriques-Frank vector.""" """Rotation matrix to Rodriques-Frank vector."""
return eu2ro(om2eu(om)) return Rotation.eu2ro(Rotation.om2eu(om))
@staticmethod
def om2ho(om): def om2ho(om):
"""Rotation matrix to homochoric vector.""" """Rotation matrix to homochoric vector."""
return ax2ho(om2ax(om)) return Rotation.ax2ho(Rotation.om2ax(om))
@staticmethod
def om2cu(om): def om2cu(om):
"""Rotation matrix to cubochoric vector.""" """Rotation matrix to cubochoric vector."""
return ho2cu(om2ho(om)) return Rotation.ho2cu(Rotation.om2ho(om))
#---------- Bunge-Euler angles ---------- #---------- Bunge-Euler angles ----------
@staticmethod
def eu2qu(eu): def eu2qu(eu):
"""Bunge-Euler angles to quaternion.""" """Bunge-Euler angles to quaternion."""
ee = 0.5*eu ee = 0.5*eu
@ -604,7 +604,7 @@ def eu2qu(eu):
if qu[0] < 0.0: qu*=-1 if qu[0] < 0.0: qu*=-1
return qu return qu
@staticmethod
def eu2om(eu): def eu2om(eu):
"""Bunge-Euler angles to rotation matrix.""" """Bunge-Euler angles to rotation matrix."""
c = np.cos(eu) c = np.cos(eu)
@ -617,7 +617,7 @@ def eu2om(eu):
om[np.where(iszero(om))] = 0.0 om[np.where(iszero(om))] = 0.0
return om return om
@staticmethod
def eu2ax(eu): def eu2ax(eu):
"""Bunge-Euler angles to axis angle pair.""" """Bunge-Euler angles to axis angle pair."""
t = np.tan(eu[1]*0.5) t = np.tan(eu[1]*0.5)
@ -636,7 +636,7 @@ def eu2ax(eu):
return ax return ax
@staticmethod
def eu2ro(eu): def eu2ro(eu):
"""Bunge-Euler angles to Rodriques-Frank vector.""" """Bunge-Euler angles to Rodriques-Frank vector."""
ro = eu2ax(eu) # convert to axis angle pair representation ro = eu2ax(eu) # convert to axis angle pair representation
@ -649,19 +649,19 @@ def eu2ro(eu):
return ro return ro
@staticmethod
def eu2ho(eu): def eu2ho(eu):
"""Bunge-Euler angles to homochoric vector.""" """Bunge-Euler angles to homochoric vector."""
return ax2ho(eu2ax(eu)) return Rotation.ax2ho(Rotation.eu2ax(eu))
@staticmethod
def eu2cu(eu): def eu2cu(eu):
"""Bunge-Euler angles to cubochoric vector.""" """Bunge-Euler angles to cubochoric vector."""
return ho2cu(eu2ho(eu)) return Rotation.ho2cu(Rotation.eu2ho(eu))
#---------- Axis angle pair ---------- #---------- Axis angle pair ----------
@staticmethod
def ax2qu(ax): def ax2qu(ax):
"""Axis angle pair to quaternion.""" """Axis angle pair to quaternion."""
if iszero(ax[3]): if iszero(ax[3]):
@ -673,7 +673,7 @@ def ax2qu(ax):
return qu return qu
@staticmethod
def ax2om(ax): def ax2om(ax):
"""Axis angle pair to rotation matrix.""" """Axis angle pair to rotation matrix."""
c = np.cos(ax[3]) c = np.cos(ax[3])
@ -688,12 +688,12 @@ def ax2om(ax):
return om if P < 0.0 else om.T return om if P < 0.0 else om.T
@staticmethod
def ax2eu(ax): def ax2eu(ax):
"""Rotation matrix to Bunge Euler angles.""" """Rotation matrix to Bunge Euler angles."""
return om2eu(ax2om(ax)) return Rotation.om2eu(Rotation.ax2om(ax))
@staticmethod
def ax2ro(ax): def ax2ro(ax):
"""Axis angle pair to Rodriques-Frank vector.""" """Axis angle pair to Rodriques-Frank vector."""
if iszero(ax[3]): if iszero(ax[3]):
@ -706,36 +706,36 @@ def ax2ro(ax):
return np.array(ro) return np.array(ro)
@staticmethod
def ax2ho(ax): def ax2ho(ax):
"""Axis angle pair to homochoric vector.""" """Axis angle pair to homochoric vector."""
f = (0.75 * ( ax[3] - np.sin(ax[3]) ))**(1.0/3.0) f = (0.75 * ( ax[3] - np.sin(ax[3]) ))**(1.0/3.0)
ho = ax[0:3] * f ho = ax[0:3] * f
return ho return ho
@staticmethod
def ax2cu(ax): def ax2cu(ax):
"""Axis angle pair to cubochoric vector.""" """Axis angle pair to cubochoric vector."""
return ho2cu(ax2ho(ax)) return Rotation.ho2cu(Rotation.ax2ho(ax))
#---------- Rodrigues-Frank vector ---------- #---------- Rodrigues-Frank vector ----------
@staticmethod
def ro2qu(ro): def ro2qu(ro):
"""Rodriques-Frank vector to quaternion.""" """Rodriques-Frank vector to quaternion."""
return ax2qu(ro2ax(ro)) return Rotation.ax2qu(Rotation.ro2ax(ro))
@staticmethod
def ro2om(ro): def ro2om(ro):
"""Rodgrigues-Frank vector to rotation matrix.""" """Rodgrigues-Frank vector to rotation matrix."""
return ax2om(ro2ax(ro)) return Rotation.ax2om(Rotation.ro2ax(ro))
@staticmethod
def ro2eu(ro): def ro2eu(ro):
"""Rodriques-Frank vector to Bunge-Euler angles.""" """Rodriques-Frank vector to Bunge-Euler angles."""
return om2eu(ro2om(ro)) return Rotation.om2eu(Rotation.ro2om(ro))
@staticmethod
def ro2ax(ro): def ro2ax(ro):
"""Rodriques-Frank vector to axis angle pair.""" """Rodriques-Frank vector to axis angle pair."""
ta = ro[3] ta = ro[3]
@ -751,7 +751,7 @@ def ro2ax(ro):
return np.array(ax) return np.array(ax)
@staticmethod
def ro2ho(ro): def ro2ho(ro):
"""Rodriques-Frank vector to homochoric vector.""" """Rodriques-Frank vector to homochoric vector."""
if iszero(np.sum(ro[0:3]**2.0)): if iszero(np.sum(ro[0:3]**2.0)):
@ -762,29 +762,29 @@ def ro2ho(ro):
return np.array(ho) return np.array(ho)
@staticmethod
def ro2cu(ro): def ro2cu(ro):
"""Rodriques-Frank vector to cubochoric vector.""" """Rodriques-Frank vector to cubochoric vector."""
return ho2cu(ro2ho(ro)) return ho2cu(ro2ho(ro))
#---------- Homochoric vector---------- #---------- Homochoric vector----------
@staticmethod
def ho2qu(ho): def ho2qu(ho):
"""Homochoric vector to quaternion.""" """Homochoric vector to quaternion."""
return ax2qu(ho2ax(ho)) return Rotation.ax2qu(Rotation.ho2ax(ho))
@staticmethod
def ho2om(ho): def ho2om(ho):
"""Homochoric vector to rotation matrix.""" """Homochoric vector to rotation matrix."""
return ax2om(ho2ax(ho)) return Rotation.ax2om(Rotation.ho2ax(ho))
@staticmethod
def ho2eu(ho): def ho2eu(ho):
"""Homochoric vector to Bunge-Euler angles.""" """Homochoric vector to Bunge-Euler angles."""
return ax2eu(ho2ax(ho)) return Rotation.ax2eu(Rotation.ho2ax(ho))
@staticmethod
def ho2ax(ho): def ho2ax(ho):
"""Homochoric vector to axis angle pair.""" """Homochoric vector to axis angle pair."""
tfit = np.array([+1.0000000000018852, -0.5000000002194847, tfit = np.array([+1.0000000000018852, -0.5000000002194847,
@ -810,44 +810,44 @@ def ho2ax(ho):
ax = np.append(ho/np.sqrt(hmag_squared),2.0*np.arccos(np.clip(s,-1.0,1.0))) ax = np.append(ho/np.sqrt(hmag_squared),2.0*np.arccos(np.clip(s,-1.0,1.0)))
return ax return ax
@staticmethod
def ho2ro(ho): def ho2ro(ho):
"""Axis angle pair to Rodriques-Frank vector.""" """Axis angle pair to Rodriques-Frank vector."""
return ax2ro(ho2ax(ho)) return Rotation.ax2ro(Rotation.ho2ax(ho))
@staticmethod
def ho2cu(ho): def ho2cu(ho):
"""Homochoric vector to cubochoric vector.""" """Homochoric vector to cubochoric vector."""
return Lambert.BallToCube(ho) return Lambert.BallToCube(ho)
#---------- Cubochoric ---------- #---------- Cubochoric ----------
@staticmethod
def cu2qu(cu): def cu2qu(cu):
"""Cubochoric vector to quaternion.""" """Cubochoric vector to quaternion."""
return ho2qu(cu2ho(cu)) return Rotation.ho2qu(Rotation.cu2ho(cu))
@staticmethod
def cu2om(cu): def cu2om(cu):
"""Cubochoric vector to rotation matrix.""" """Cubochoric vector to rotation matrix."""
return ho2om(cu2ho(cu)) return Rotation.ho2om(Rotation.cu2ho(cu))
@staticmethod
def cu2eu(cu): def cu2eu(cu):
"""Cubochoric vector to Bunge-Euler angles.""" """Cubochoric vector to Bunge-Euler angles."""
return ho2eu(cu2ho(cu)) return Rotation.ho2eu(Rotation.cu2ho(cu))
@staticmethod
def cu2ax(cu): def cu2ax(cu):
"""Cubochoric vector to axis angle pair.""" """Cubochoric vector to axis angle pair."""
return ho2ax(cu2ho(cu)) return Rotation.ho2ax(Rotation.cu2ho(cu))
@staticmethod
def cu2ro(cu): def cu2ro(cu):
"""Cubochoric vector to Rodriques-Frank vector.""" """Cubochoric vector to Rodriques-Frank vector."""
return ho2ro(cu2ho(cu)) return Rotation.ho2ro(Rotation.cu2ho(cu))
@staticmethod
def cu2ho(cu): def cu2ho(cu):
"""Cubochoric vector to homochoric vector.""" """Cubochoric vector to homochoric vector."""
return Lambert.CubeToBall(cu) return Lambert.CubeToBall(cu)