2021-11-01 03:07:54 +05:30
|
|
|
|
import copy
|
2022-11-23 02:56:15 +05:30
|
|
|
|
from typing import Optional, Union, Sequence, Tuple, Literal, List, TypeVar
|
2021-11-01 03:07:54 +05:30
|
|
|
|
|
2020-02-21 03:46:35 +05:30
|
|
|
|
import numpy as np
|
|
|
|
|
|
2022-02-13 05:54:02 +05:30
|
|
|
|
from ._typehints import FloatSequence, IntSequence, NumpyRngSeed
|
2020-11-16 03:44:46 +05:30
|
|
|
|
from . import tensor
|
2020-09-28 16:44:23 +05:30
|
|
|
|
from . import util
|
|
|
|
|
from . import grid_filters
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-04-12 04:49:11 +05:30
|
|
|
|
_P = -1
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-04-22 16:53:33 +05:30
|
|
|
|
# parameters for conversion from/to cubochoric
|
|
|
|
|
_sc = np.pi**(1./6.)/6.**(1./6.)
|
|
|
|
|
_beta = np.pi**(5./6.)/6.**(1./6.)/2.
|
|
|
|
|
_R1 = (3.*np.pi/4.)**(1./3.)
|
|
|
|
|
|
2022-02-02 16:44:00 +05:30
|
|
|
|
MyType = TypeVar('MyType', bound='Rotation')
|
|
|
|
|
|
2020-02-21 03:46:35 +05:30
|
|
|
|
class Rotation:
|
|
|
|
|
u"""
|
2020-11-10 01:50:56 +05:30
|
|
|
|
Rotation with functionality for conversion between different representations.
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-04-12 19:08:38 +05:30
|
|
|
|
The following conventions apply:
|
|
|
|
|
|
2020-11-10 01:50:56 +05:30
|
|
|
|
- Coordinate frames are right-handed.
|
|
|
|
|
- A rotation angle ω is taken to be positive for a counterclockwise rotation
|
2020-04-12 19:08:38 +05:30
|
|
|
|
when viewing from the end point of the rotation axis towards the origin.
|
2022-11-29 10:35:58 +05:30
|
|
|
|
- Rotations will be interpreted in the passive sense, i.e. as rotation of
|
|
|
|
|
the coordinate frame.
|
2020-04-12 19:08:38 +05:30
|
|
|
|
- P = -1 (as default).
|
|
|
|
|
|
|
|
|
|
Examples
|
|
|
|
|
--------
|
2021-03-28 15:05:40 +05:30
|
|
|
|
Rotate vector 'a' (defined in coordinate system 'A') to
|
|
|
|
|
coordinates 'b' expressed in system 'B':
|
2020-04-12 19:08:38 +05:30
|
|
|
|
|
2021-03-28 15:05:40 +05:30
|
|
|
|
>>> import numpy as np
|
2023-02-21 20:57:06 +05:30
|
|
|
|
>>> import damask
|
2021-03-28 15:05:40 +05:30
|
|
|
|
>>> Q = damask.Rotation.from_random()
|
|
|
|
|
>>> a = np.random.rand(3)
|
2021-04-26 22:39:11 +05:30
|
|
|
|
>>> b = Q @ a
|
2021-03-28 15:05:40 +05:30
|
|
|
|
>>> np.allclose(np.dot(Q.as_matrix(),a),b)
|
|
|
|
|
True
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2021-01-13 05:26:40 +05:30
|
|
|
|
Compound rotations R1 (first) and R2 (second):
|
|
|
|
|
|
2021-03-28 15:05:40 +05:30
|
|
|
|
>>> import numpy as np
|
2023-02-21 20:57:06 +05:30
|
|
|
|
>>> import damask
|
2021-03-28 15:05:40 +05:30
|
|
|
|
>>> R1 = damask.Rotation.from_random()
|
|
|
|
|
>>> R2 = damask.Rotation.from_random()
|
|
|
|
|
>>> R = R2 * R1
|
|
|
|
|
>>> np.allclose(R.as_matrix(), np.dot(R2.as_matrix(),R1.as_matrix()))
|
|
|
|
|
True
|
2021-01-13 05:26:40 +05:30
|
|
|
|
|
2020-02-21 03:46:35 +05:30
|
|
|
|
References
|
|
|
|
|
----------
|
|
|
|
|
D. Rowenhorst et al., Modelling and Simulation in Materials Science and Engineering 23:083501, 2015
|
|
|
|
|
https://doi.org/10.1088/0965-0393/23/8/083501
|
|
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
__slots__ = ['quaternion']
|
|
|
|
|
|
2022-01-26 19:39:09 +05:30
|
|
|
|
def __init__(self,
|
2022-11-18 05:09:32 +05:30
|
|
|
|
rotation: Union[FloatSequence, 'Rotation'] = np.array([1.,0.,0.,0.])):
|
2020-02-21 03:46:35 +05:30
|
|
|
|
"""
|
2021-03-27 14:40:35 +05:30
|
|
|
|
New rotation.
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
|
----------
|
2023-02-21 20:57:06 +05:30
|
|
|
|
rotation : list, numpy.ndarray, or Rotation, optional
|
2020-06-20 15:50:43 +05:30
|
|
|
|
Unit quaternion in positive real hemisphere.
|
|
|
|
|
Use .from_quaternion to perform a sanity check.
|
2020-11-10 01:50:56 +05:30
|
|
|
|
Defaults to no rotation.
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
|
|
|
|
"""
|
2021-12-14 21:35:00 +05:30
|
|
|
|
self.quaternion: np.ndarray
|
2020-11-10 01:50:56 +05:30
|
|
|
|
if isinstance(rotation,Rotation):
|
|
|
|
|
self.quaternion = rotation.quaternion.copy()
|
|
|
|
|
elif np.array(rotation).shape[-1] == 4:
|
2022-11-18 05:09:32 +05:30
|
|
|
|
self.quaternion = np.array(rotation,dtype=float)
|
2020-11-10 01:50:56 +05:30
|
|
|
|
else:
|
2022-02-03 20:41:09 +05:30
|
|
|
|
raise TypeError('"rotation" is neither a Rotation nor a quaternion')
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-05-02 19:20:46 +05:30
|
|
|
|
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def __repr__(self) -> str:
|
2022-07-08 21:36:41 +05:30
|
|
|
|
"""
|
|
|
|
|
Return repr(self).
|
|
|
|
|
|
2023-02-21 20:57:06 +05:30
|
|
|
|
Give short, human-readable summary.
|
2022-07-08 21:36:41 +05:30
|
|
|
|
|
|
|
|
|
"""
|
2021-03-27 20:24:29 +05:30
|
|
|
|
return f'Quaternion{" " if self.quaternion.shape == (4,) else "s of shape "+str(self.quaternion.shape[:-1])+chr(10)}'\
|
2021-01-13 14:20:58 +05:30
|
|
|
|
+ str(self.quaternion)
|
2020-05-02 19:20:46 +05:30
|
|
|
|
|
|
|
|
|
|
2022-02-02 16:44:00 +05:30
|
|
|
|
def __copy__(self: MyType,
|
2022-11-23 02:56:15 +05:30
|
|
|
|
rotation: Union[None, FloatSequence, 'Rotation'] = None) -> MyType:
|
2022-07-08 21:36:41 +05:30
|
|
|
|
"""
|
|
|
|
|
Return deepcopy(self).
|
|
|
|
|
|
|
|
|
|
Create deep copy.
|
|
|
|
|
|
|
|
|
|
"""
|
2021-11-01 03:07:54 +05:30
|
|
|
|
dup = copy.deepcopy(self)
|
|
|
|
|
if rotation is not None:
|
|
|
|
|
dup.quaternion = Rotation(rotation).quaternion
|
|
|
|
|
return dup
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
|
|
|
|
copy = __copy__
|
|
|
|
|
|
|
|
|
|
|
2022-01-26 19:39:09 +05:30
|
|
|
|
def __getitem__(self,
|
2023-10-04 20:00:25 +05:30
|
|
|
|
item: Union[Tuple[Union[None, int, slice]], int, bool, np.bool_, np.ndarray]):
|
2022-07-08 21:36:41 +05:30
|
|
|
|
"""
|
2022-08-09 18:59:22 +05:30
|
|
|
|
Return self[item].
|
2022-07-08 21:36:41 +05:30
|
|
|
|
|
|
|
|
|
Return slice according to item.
|
|
|
|
|
|
|
|
|
|
"""
|
2022-02-11 22:40:29 +05:30
|
|
|
|
return self.copy() if self.shape == () else \
|
2022-02-13 05:54:02 +05:30
|
|
|
|
self.copy(self.quaternion[item+(slice(None),)] if isinstance(item,tuple) else self.quaternion[item])
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
|
|
|
|
|
2022-02-11 22:40:29 +05:30
|
|
|
|
def __eq__(self,
|
|
|
|
|
other: object) -> bool:
|
2020-11-10 01:50:56 +05:30
|
|
|
|
"""
|
2022-07-08 21:36:41 +05:30
|
|
|
|
Return self==other.
|
|
|
|
|
|
2022-07-08 21:37:07 +05:30
|
|
|
|
Test equality of other.
|
2020-11-10 01:50:56 +05:30
|
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
|
----------
|
|
|
|
|
other : Rotation
|
|
|
|
|
Rotation to check for equality.
|
|
|
|
|
|
|
|
|
|
"""
|
2022-02-11 22:40:29 +05:30
|
|
|
|
return NotImplemented if not isinstance(other, Rotation) else \
|
2022-11-18 05:09:32 +05:30
|
|
|
|
np.logical_or(np.all(self.quaternion == other.quaternion,axis=-1),
|
|
|
|
|
np.all(self.quaternion == -1.*other.quaternion,axis=-1))
|
2021-04-05 21:54:03 +05:30
|
|
|
|
|
2021-01-04 02:19:01 +05:30
|
|
|
|
|
2022-02-04 21:27:25 +05:30
|
|
|
|
def __ne__(self,
|
|
|
|
|
other: object) -> bool:
|
2021-01-04 02:19:01 +05:30
|
|
|
|
"""
|
2022-07-08 21:36:41 +05:30
|
|
|
|
Return self!=other.
|
|
|
|
|
|
2022-07-08 21:37:07 +05:30
|
|
|
|
Test inequality of other.
|
2021-01-04 02:19:01 +05:30
|
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
|
----------
|
|
|
|
|
other : Rotation
|
2021-09-24 18:24:31 +05:30
|
|
|
|
Rotation to check for inequality.
|
2021-01-04 02:19:01 +05:30
|
|
|
|
|
|
|
|
|
"""
|
2022-02-04 21:27:25 +05:30
|
|
|
|
return np.logical_not(self==other) if isinstance(other, Rotation) else NotImplemented
|
2020-11-10 01:50:56 +05:30
|
|
|
|
|
2022-02-16 02:38:12 +05:30
|
|
|
|
def isclose(self: MyType,
|
|
|
|
|
other: MyType,
|
2022-11-18 05:09:32 +05:30
|
|
|
|
rtol: float = 1.e-5,
|
|
|
|
|
atol: float = 1.e-8,
|
2022-01-26 19:39:09 +05:30
|
|
|
|
equal_nan: bool = True) -> bool:
|
2021-04-05 21:54:03 +05:30
|
|
|
|
"""
|
|
|
|
|
Report where values are approximately equal to corresponding ones of other Rotation.
|
|
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
|
----------
|
|
|
|
|
other : Rotation
|
|
|
|
|
Rotation to compare against.
|
|
|
|
|
rtol : float, optional
|
|
|
|
|
Relative tolerance of equality.
|
|
|
|
|
atol : float, optional
|
|
|
|
|
Absolute tolerance of equality.
|
|
|
|
|
equal_nan : bool, optional
|
|
|
|
|
Consider matching NaN values as equal. Defaults to True.
|
|
|
|
|
|
|
|
|
|
Returns
|
|
|
|
|
-------
|
2022-02-13 05:54:02 +05:30
|
|
|
|
mask : numpy.ndarray of bool, shape (self.shape)
|
2021-04-05 21:54:03 +05:30
|
|
|
|
Mask indicating where corresponding rotations are close.
|
|
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
s = self.quaternion
|
|
|
|
|
o = other.quaternion
|
2022-11-18 05:09:32 +05:30
|
|
|
|
return np.logical_or(np.all(np.isclose(s, o,rtol,atol,equal_nan),axis=-1),
|
|
|
|
|
np.all(np.isclose(s,-1.*o,rtol,atol,equal_nan),axis=-1))
|
2021-04-05 21:54:03 +05:30
|
|
|
|
|
|
|
|
|
|
2022-02-16 02:38:12 +05:30
|
|
|
|
def allclose(self: MyType,
|
|
|
|
|
other: MyType,
|
2022-11-18 05:09:32 +05:30
|
|
|
|
rtol: float = 1.e-5,
|
|
|
|
|
atol: float = 1.e-8,
|
2022-01-26 19:39:09 +05:30
|
|
|
|
equal_nan: bool = True) -> Union[np.bool_, bool]:
|
2021-04-05 21:54:03 +05:30
|
|
|
|
"""
|
|
|
|
|
Test whether all values are approximately equal to corresponding ones of other Rotation.
|
|
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
|
----------
|
|
|
|
|
other : Rotation
|
|
|
|
|
Rotation to compare against.
|
|
|
|
|
rtol : float, optional
|
|
|
|
|
Relative tolerance of equality.
|
|
|
|
|
atol : float, optional
|
|
|
|
|
Absolute tolerance of equality.
|
|
|
|
|
equal_nan : bool, optional
|
|
|
|
|
Consider matching NaN values as equal. Defaults to True.
|
|
|
|
|
|
|
|
|
|
Returns
|
|
|
|
|
-------
|
|
|
|
|
answer : bool
|
|
|
|
|
Whether all values are close between both rotations.
|
|
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
return np.all(self.isclose(other,rtol,atol,equal_nan))
|
|
|
|
|
|
|
|
|
|
|
2021-02-22 23:22:06 +05:30
|
|
|
|
def __array__(self):
|
|
|
|
|
"""Initializer for numpy."""
|
|
|
|
|
return self.quaternion
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@property
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def size(self) -> int:
|
2021-02-22 23:22:06 +05:30
|
|
|
|
return self.quaternion[...,0].size
|
|
|
|
|
|
2020-11-10 01:50:56 +05:30
|
|
|
|
@property
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def shape(self) -> Tuple[int, ...]:
|
2021-02-22 23:22:06 +05:30
|
|
|
|
return self.quaternion[...,0].shape
|
2020-06-30 15:46:47 +05:30
|
|
|
|
|
|
|
|
|
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def __len__(self) -> int:
|
2022-07-08 21:36:41 +05:30
|
|
|
|
"""
|
|
|
|
|
Return len(self).
|
|
|
|
|
|
|
|
|
|
Length of leading/leftmost dimension of array.
|
|
|
|
|
|
|
|
|
|
"""
|
2020-06-30 15:46:47 +05:30
|
|
|
|
return 0 if self.shape == () else self.shape[0]
|
|
|
|
|
|
|
|
|
|
|
2022-02-02 16:44:00 +05:30
|
|
|
|
def __invert__(self: MyType) -> MyType:
|
2022-07-08 21:36:41 +05:30
|
|
|
|
"""
|
|
|
|
|
Return ~self.
|
|
|
|
|
|
|
|
|
|
Inverse rotation (backward rotation).
|
|
|
|
|
|
|
|
|
|
"""
|
2020-11-10 01:50:56 +05:30
|
|
|
|
dup = self.copy()
|
2022-11-18 05:09:32 +05:30
|
|
|
|
dup.quaternion[...,1:] *= -1.
|
2020-11-10 01:50:56 +05:30
|
|
|
|
return dup
|
|
|
|
|
|
|
|
|
|
|
2022-02-02 16:44:00 +05:30
|
|
|
|
def __pow__(self: MyType,
|
2022-02-11 22:40:29 +05:30
|
|
|
|
exp: Union[float, int]) -> MyType:
|
2020-11-10 01:50:56 +05:30
|
|
|
|
"""
|
2022-07-08 21:36:41 +05:30
|
|
|
|
Return self**exp.
|
|
|
|
|
|
2021-01-13 14:20:58 +05:30
|
|
|
|
Perform the rotation 'exp' times.
|
2020-11-10 01:50:56 +05:30
|
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
|
----------
|
2022-02-13 05:54:02 +05:30
|
|
|
|
exp : float
|
2021-01-13 14:20:58 +05:30
|
|
|
|
Exponent.
|
2020-11-10 01:50:56 +05:30
|
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
phi = np.arccos(self.quaternion[...,0:1])
|
|
|
|
|
p = self.quaternion[...,1:]/np.linalg.norm(self.quaternion[...,1:],axis=-1,keepdims=True)
|
2022-02-13 05:54:02 +05:30
|
|
|
|
return self.copy(Rotation(np.block([np.cos(exp*phi),np.sin(exp*phi)*p]))._standardize())
|
2020-11-10 01:50:56 +05:30
|
|
|
|
|
2022-02-02 16:44:00 +05:30
|
|
|
|
def __ipow__(self: MyType,
|
2022-02-11 22:40:29 +05:30
|
|
|
|
exp: Union[float, int]) -> MyType:
|
2021-01-03 20:24:41 +05:30
|
|
|
|
"""
|
2022-07-08 21:36:41 +05:30
|
|
|
|
Return self**=exp.
|
|
|
|
|
|
2021-01-13 14:20:58 +05:30
|
|
|
|
Perform the rotation 'exp' times (in-place).
|
2021-01-03 20:24:41 +05:30
|
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
|
----------
|
2022-02-13 05:54:02 +05:30
|
|
|
|
exp : float
|
2021-01-13 14:20:58 +05:30
|
|
|
|
Exponent.
|
2021-01-03 20:24:41 +05:30
|
|
|
|
|
|
|
|
|
"""
|
2021-01-13 14:20:58 +05:30
|
|
|
|
return self**exp
|
2021-01-03 20:24:41 +05:30
|
|
|
|
|
2020-11-10 01:50:56 +05:30
|
|
|
|
|
2022-02-04 21:27:25 +05:30
|
|
|
|
def __mul__(self: MyType,
|
|
|
|
|
other: MyType) -> MyType:
|
2021-01-03 20:24:41 +05:30
|
|
|
|
"""
|
2022-07-08 21:36:41 +05:30
|
|
|
|
Return self*other.
|
|
|
|
|
|
2021-03-27 14:40:35 +05:30
|
|
|
|
Compose with other.
|
2021-01-03 20:24:41 +05:30
|
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
|
----------
|
2022-02-11 22:40:29 +05:30
|
|
|
|
other : Rotation, shape (self.shape)
|
2021-01-13 05:26:40 +05:30
|
|
|
|
Rotation for composition.
|
2023-10-04 19:04:23 +05:30
|
|
|
|
Compatible innermost dimensions will blend.
|
2021-01-13 05:26:40 +05:30
|
|
|
|
|
|
|
|
|
Returns
|
|
|
|
|
-------
|
|
|
|
|
composition : Rotation
|
|
|
|
|
Compound rotation self*other, i.e. first other then self rotation.
|
2021-01-03 20:24:41 +05:30
|
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
if isinstance(other,Rotation):
|
2023-10-04 19:04:23 +05:30
|
|
|
|
blend = util.shapeblender( self.shape,other.shape)
|
|
|
|
|
s_m = util.shapeshifter( self.shape,blend,mode='right')
|
|
|
|
|
s_o = util.shapeshifter(other.shape,blend,mode='left')
|
|
|
|
|
|
|
|
|
|
q_m = self.broadcast_to(s_m).quaternion[...,0:1]
|
|
|
|
|
p_m = self.broadcast_to(s_m).quaternion[...,1:]
|
|
|
|
|
q_o = other.broadcast_to(s_o).quaternion[...,0:1]
|
|
|
|
|
p_o = other.broadcast_to(s_o).quaternion[...,1:]
|
|
|
|
|
|
2023-08-18 02:40:26 +05:30
|
|
|
|
qmo = q_m*q_o
|
|
|
|
|
q = (qmo - np.einsum('...i,...i',p_m,p_o).reshape(qmo.shape))
|
2021-01-13 05:26:40 +05:30
|
|
|
|
p = q_m*p_o + q_o*p_m + _P * np.cross(p_m,p_o)
|
2022-02-13 05:54:02 +05:30
|
|
|
|
return self.copy(Rotation(np.block([q,p]))._standardize())
|
2021-01-03 20:24:41 +05:30
|
|
|
|
else:
|
2022-02-22 21:12:05 +05:30
|
|
|
|
raise TypeError('use "R@b", i.e. matmul, to apply rotation "R" to object "b"')
|
2021-01-03 20:24:41 +05:30
|
|
|
|
|
2022-02-04 14:27:42 +05:30
|
|
|
|
def __imul__(self: MyType,
|
|
|
|
|
other: MyType) -> MyType:
|
2021-01-03 20:24:41 +05:30
|
|
|
|
"""
|
2022-07-08 21:36:41 +05:30
|
|
|
|
Return self*=other.
|
|
|
|
|
|
2021-03-27 14:40:35 +05:30
|
|
|
|
Compose with other (in-place).
|
2021-01-03 20:24:41 +05:30
|
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
|
----------
|
2022-02-11 22:40:29 +05:30
|
|
|
|
other : Rotation, shape (self.shape)
|
2021-01-13 05:26:40 +05:30
|
|
|
|
Rotation for composition.
|
2023-10-04 19:04:23 +05:30
|
|
|
|
Compatible innermost dimensions will blend.
|
2021-01-03 20:24:41 +05:30
|
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
return self*other
|
|
|
|
|
|
|
|
|
|
|
2022-02-04 14:27:42 +05:30
|
|
|
|
def __truediv__(self: MyType,
|
|
|
|
|
other: MyType) -> MyType:
|
2021-01-03 20:24:41 +05:30
|
|
|
|
"""
|
2022-07-08 21:36:41 +05:30
|
|
|
|
Return self/other.
|
|
|
|
|
|
2021-03-27 14:40:35 +05:30
|
|
|
|
Compose with inverse of other.
|
2021-01-03 20:24:41 +05:30
|
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
|
----------
|
2022-02-11 22:40:29 +05:30
|
|
|
|
other : damask.Rotation, shape (self.shape)
|
2021-09-24 18:24:31 +05:30
|
|
|
|
Rotation to invert for composition.
|
2023-10-04 19:04:23 +05:30
|
|
|
|
Compatible innermost dimensions will blend.
|
2021-01-03 20:24:41 +05:30
|
|
|
|
|
2021-01-13 05:26:40 +05:30
|
|
|
|
Returns
|
|
|
|
|
-------
|
|
|
|
|
composition : Rotation
|
|
|
|
|
Compound rotation self*(~other), i.e. first inverse of other then self rotation.
|
|
|
|
|
|
2021-01-03 20:24:41 +05:30
|
|
|
|
"""
|
|
|
|
|
if isinstance(other,Rotation):
|
2021-01-13 05:26:40 +05:30
|
|
|
|
return self*~other
|
2021-01-03 20:24:41 +05:30
|
|
|
|
else:
|
2022-02-22 21:12:05 +05:30
|
|
|
|
raise TypeError('use "R@b", i.e. matmul, to apply rotation "R" to object "b"')
|
2021-01-03 20:24:41 +05:30
|
|
|
|
|
2022-02-04 14:27:42 +05:30
|
|
|
|
def __itruediv__(self: MyType,
|
|
|
|
|
other: MyType) -> MyType:
|
2021-01-03 20:24:41 +05:30
|
|
|
|
"""
|
2022-07-08 21:36:41 +05:30
|
|
|
|
Return self/=other.
|
|
|
|
|
|
2021-03-27 14:40:35 +05:30
|
|
|
|
Compose with inverse of other (in-place).
|
2021-01-03 20:24:41 +05:30
|
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
|
----------
|
2022-02-11 22:40:29 +05:30
|
|
|
|
other : Rotation, shape (self.shape)
|
2021-09-24 18:24:31 +05:30
|
|
|
|
Rotation to invert for composition.
|
2021-01-03 20:24:41 +05:30
|
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
return self/other
|
|
|
|
|
|
|
|
|
|
|
2022-02-04 21:27:25 +05:30
|
|
|
|
def __matmul__(self,
|
|
|
|
|
other: np.ndarray) -> np.ndarray:
|
2020-02-21 03:46:35 +05:30
|
|
|
|
"""
|
2022-07-08 21:36:41 +05:30
|
|
|
|
Return self@other.
|
|
|
|
|
|
2022-08-09 18:59:22 +05:30
|
|
|
|
Rotate vector, second-order tensor, or fourth-order tensor.
|
2023-09-20 03:20:28 +05:30
|
|
|
|
`other` is interpreted as an array of tensor quantities with the highest-possible order
|
2023-09-22 02:57:38 +05:30
|
|
|
|
considering the shape of `self`. Compatible innermost dimensions will blend.
|
2023-09-20 03:20:28 +05:30
|
|
|
|
For instance, shapes of (2,) and (3,3) for `self` and `other` prompt interpretation of
|
|
|
|
|
`other` as a second-rank tensor and result in (2,) rotated tensors, whereas
|
|
|
|
|
shapes of (2,1) and (3,3) for `self` and `other` result in (2,3) rotated vectors.
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
|
----------
|
2022-02-11 22:40:29 +05:30
|
|
|
|
other : numpy.ndarray, shape (...,3), (...,3,3), or (...,3,3,3,3)
|
2021-01-13 05:26:40 +05:30
|
|
|
|
Vector or tensor on which to apply the rotation.
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-07-01 02:47:50 +05:30
|
|
|
|
Returns
|
|
|
|
|
-------
|
2022-02-11 22:40:29 +05:30
|
|
|
|
rotated : numpy.ndarray, shape (...,3), (...,3,3), or (...,3,3,3,3)
|
2021-01-13 05:26:40 +05:30
|
|
|
|
Rotated vector or tensor, i.e. transformed to frame defined by rotation.
|
2020-05-20 22:40:16 +05:30
|
|
|
|
|
2023-09-22 02:57:38 +05:30
|
|
|
|
Examples
|
|
|
|
|
--------
|
|
|
|
|
All below examples rely on imported modules:
|
|
|
|
|
>>> import numpy as np
|
|
|
|
|
>>> import damask
|
|
|
|
|
|
|
|
|
|
Application of twelve (random) rotations to a set of five vectors.
|
|
|
|
|
|
|
|
|
|
>>> r = damask.Rotation.from_random(shape=(12))
|
|
|
|
|
>>> o = np.ones((5,3))
|
|
|
|
|
>>> (r@o).shape # (12) @ (5, 3)
|
|
|
|
|
(12,5, 3)
|
|
|
|
|
|
|
|
|
|
Application of a (random) rotation to all twelve second-rank tensors.
|
|
|
|
|
|
|
|
|
|
>>> r = damask.Rotation.from_random()
|
|
|
|
|
>>> o = np.ones((12,3,3))
|
|
|
|
|
>>> (r@o).shape # (1) @ (12, 3,3)
|
|
|
|
|
(12,3,3)
|
|
|
|
|
|
|
|
|
|
Application of twelve (random) rotations to the corresponding twelve second-rank tensors.
|
|
|
|
|
|
|
|
|
|
>>> r = damask.Rotation.from_random(shape=(12))
|
|
|
|
|
>>> o = np.ones((12,3,3))
|
|
|
|
|
>>> (r@o).shape # (12) @ (3,3)
|
|
|
|
|
(12,3,3)
|
|
|
|
|
|
|
|
|
|
Application of each of three (random) rotations to all three vectors.
|
|
|
|
|
|
|
|
|
|
>>> r = damask.Rotation.from_random(shape=(3))
|
|
|
|
|
>>> o = np.ones((3,3))
|
|
|
|
|
>>> (r[...,np.newaxis]@o[np.newaxis,...]).shape # (3,1) @ (1,3, 3)
|
|
|
|
|
(3,3,3)
|
|
|
|
|
|
|
|
|
|
Application of twelve (random) rotations to all twelve second-rank tensors.
|
|
|
|
|
|
|
|
|
|
>>> r = damask.Rotation.from_random(shape=(12))
|
|
|
|
|
>>> o = np.ones((12,3,3))
|
|
|
|
|
>>> (r@o[np.newaxis,...]).shape # (12) @ (1,12, 3,3)
|
|
|
|
|
(12,3,3,3)
|
|
|
|
|
|
2020-05-02 18:46:26 +05:30
|
|
|
|
"""
|
2022-01-26 19:39:09 +05:30
|
|
|
|
if isinstance(other, np.ndarray):
|
2023-10-04 19:04:23 +05:30
|
|
|
|
obs = util.shapeblender(self.shape,other.shape)[len(self.shape):]
|
2023-09-20 03:20:28 +05:30
|
|
|
|
for l in [4,2,1]:
|
|
|
|
|
if obs[-l:] == l*(3,):
|
|
|
|
|
bs = util.shapeblender(self.shape,other.shape[:-l],False)
|
|
|
|
|
self_ = self.broadcast_to(bs) if self.shape != bs else self
|
|
|
|
|
if l==1:
|
|
|
|
|
q_m = self_.quaternion[...,0]
|
|
|
|
|
p_m = self_.quaternion[...,1:]
|
|
|
|
|
A = q_m**2 - np.einsum('...i,...i',p_m,p_m)
|
|
|
|
|
B = 2. * np.einsum('...i,...i',p_m,other)
|
|
|
|
|
C = 2. * _P * q_m
|
|
|
|
|
return np.block([(A * other[...,i]) +
|
|
|
|
|
(B * p_m[...,i]) +
|
|
|
|
|
(C * ( p_m[...,(i+1)%3]*other[...,(i+2)%3]
|
|
|
|
|
- p_m[...,(i+2)%3]*other[...,(i+1)%3]))
|
|
|
|
|
for i in [0,1,2]]).reshape(bs+(3,),order='F')
|
|
|
|
|
else:
|
|
|
|
|
return np.einsum({2: '...im,...jn,...mn',
|
|
|
|
|
4: '...im,...jn,...ko,...lp,...mnop'}[l],
|
|
|
|
|
*l*[self_.as_matrix()],
|
|
|
|
|
other)
|
|
|
|
|
raise ValueError('can only rotate vectors, second-order tensors, and fourth-order tensors')
|
2022-01-26 19:39:09 +05:30
|
|
|
|
elif isinstance(other, Rotation):
|
2023-09-20 03:20:28 +05:30
|
|
|
|
raise TypeError('use "R2*R1", i.e. multiplication, to compose rotations "R1" and "R2"')
|
2020-05-02 19:20:46 +05:30
|
|
|
|
else:
|
2022-02-22 21:12:05 +05:30
|
|
|
|
raise TypeError(f'cannot rotate "{type(other)}"')
|
2020-05-02 18:46:26 +05:30
|
|
|
|
|
2021-01-13 14:05:42 +05:30
|
|
|
|
apply = __matmul__
|
|
|
|
|
|
2020-06-20 15:50:43 +05:30
|
|
|
|
|
2022-02-02 16:44:00 +05:30
|
|
|
|
def _standardize(self: MyType) -> MyType:
|
2020-11-10 01:50:56 +05:30
|
|
|
|
"""Standardize quaternion (ensure positive real hemisphere)."""
|
2022-11-18 05:09:32 +05:30
|
|
|
|
self.quaternion[self.quaternion[...,0] < 0.] *= -1.
|
2020-06-20 15:50:43 +05:30
|
|
|
|
return self
|
|
|
|
|
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2022-02-02 16:44:00 +05:30
|
|
|
|
def append(self: MyType,
|
|
|
|
|
other: Union[MyType, List[MyType]]) -> MyType:
|
2021-01-04 11:53:14 +05:30
|
|
|
|
"""
|
2021-03-27 14:40:35 +05:30
|
|
|
|
Extend array along first dimension with other array(s).
|
2021-01-04 11:53:14 +05:30
|
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
|
----------
|
2022-01-28 18:10:37 +05:30
|
|
|
|
other : (list of) damask.Rotation
|
2021-01-04 11:53:14 +05:30
|
|
|
|
|
|
|
|
|
"""
|
2022-02-13 05:54:02 +05:30
|
|
|
|
return self.copy(np.vstack(tuple(map(lambda x:x.quaternion,
|
|
|
|
|
[self]+other if isinstance(other,list) else [self,other]))))
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
|
|
|
|
|
2022-02-02 16:44:00 +05:30
|
|
|
|
def flatten(self: MyType,
|
|
|
|
|
order: Literal['C','F','A'] = 'C') -> MyType:
|
2021-04-23 22:45:11 +05:30
|
|
|
|
"""
|
|
|
|
|
Flatten array.
|
|
|
|
|
|
2022-02-11 22:40:29 +05:30
|
|
|
|
Parameters
|
|
|
|
|
----------
|
|
|
|
|
order : {'C', 'F', 'A'}, optional
|
|
|
|
|
'C' flattens in row-major (C-style) order.
|
|
|
|
|
'F' flattens in column-major (Fortran-style) order.
|
|
|
|
|
'A' flattens in column-major order if object is Fortran contiguous in memory,
|
|
|
|
|
row-major order otherwise.
|
|
|
|
|
Defaults to 'C'.
|
|
|
|
|
|
2021-04-23 22:45:11 +05:30
|
|
|
|
Returns
|
|
|
|
|
-------
|
2021-04-25 11:17:00 +05:30
|
|
|
|
flattened : damask.Rotation
|
2021-04-23 22:45:11 +05:30
|
|
|
|
Rotation flattened to single dimension.
|
|
|
|
|
|
|
|
|
|
"""
|
2022-02-13 05:54:02 +05:30
|
|
|
|
return self.copy(self.quaternion.reshape((-1,4),order=order))
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-11-10 01:50:56 +05:30
|
|
|
|
|
2022-02-02 16:44:00 +05:30
|
|
|
|
def reshape(self: MyType,
|
2022-03-19 15:25:03 +05:30
|
|
|
|
shape: Union[int, IntSequence],
|
2022-02-02 16:44:00 +05:30
|
|
|
|
order: Literal['C','F','A'] = 'C') -> MyType:
|
2021-04-23 22:45:11 +05:30
|
|
|
|
"""
|
|
|
|
|
Reshape array.
|
|
|
|
|
|
2022-02-11 22:40:29 +05:30
|
|
|
|
Parameters
|
|
|
|
|
----------
|
2022-11-14 19:39:45 +05:30
|
|
|
|
shape : (sequence of) int
|
2022-03-19 15:25:03 +05:30
|
|
|
|
New shape, number of elements needs to match the original shape.
|
2022-02-11 22:40:29 +05:30
|
|
|
|
If an integer is supplied, then the result will be a 1-D array of that length.
|
|
|
|
|
order : {'C', 'F', 'A'}, optional
|
|
|
|
|
'C' flattens in row-major (C-style) order.
|
|
|
|
|
'F' flattens in column-major (Fortran-style) order.
|
|
|
|
|
'A' flattens in column-major order if object is Fortran contiguous in memory,
|
|
|
|
|
row-major order otherwise.
|
|
|
|
|
Defaults to 'C'.
|
|
|
|
|
|
2021-04-23 22:45:11 +05:30
|
|
|
|
Returns
|
|
|
|
|
-------
|
2021-04-25 11:17:00 +05:30
|
|
|
|
reshaped : damask.Rotation
|
2021-04-23 22:45:11 +05:30
|
|
|
|
Rotation of given shape.
|
|
|
|
|
|
|
|
|
|
"""
|
2020-11-10 01:50:56 +05:30
|
|
|
|
if isinstance(shape,(int,np.integer)): shape = (shape,)
|
2022-02-13 05:54:02 +05:30
|
|
|
|
return self.copy(self.quaternion.reshape(tuple(shape)+(4,),order=order))
|
2020-11-10 01:50:56 +05:30
|
|
|
|
|
|
|
|
|
|
2022-02-02 16:44:00 +05:30
|
|
|
|
def broadcast_to(self: MyType,
|
2022-03-19 15:25:03 +05:30
|
|
|
|
shape: Union[int, IntSequence],
|
2022-02-02 16:44:00 +05:30
|
|
|
|
mode: Literal['left', 'right'] = 'right') -> MyType:
|
2020-02-21 03:46:35 +05:30
|
|
|
|
"""
|
2021-03-27 14:40:35 +05:30
|
|
|
|
Broadcast array.
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
|
----------
|
2022-11-14 19:39:45 +05:30
|
|
|
|
shape : (sequence of) int
|
2022-03-19 15:25:03 +05:30
|
|
|
|
Shape of broadcasted array, needs to be compatible with the original shape.
|
2020-11-10 01:50:56 +05:30
|
|
|
|
mode : str, optional
|
|
|
|
|
Where to preferentially locate missing dimensions.
|
|
|
|
|
Either 'left' or 'right' (default).
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2021-04-23 22:45:11 +05:30
|
|
|
|
Returns
|
|
|
|
|
-------
|
2021-04-25 11:17:00 +05:30
|
|
|
|
broadcasted : damask.Rotation
|
2021-04-23 22:45:11 +05:30
|
|
|
|
Rotation broadcasted to given shape.
|
|
|
|
|
|
2020-02-21 03:46:35 +05:30
|
|
|
|
"""
|
2022-03-19 15:25:03 +05:30
|
|
|
|
shape_ = (shape,) if isinstance(shape,(int,np.integer)) else tuple(shape)
|
|
|
|
|
return self.copy(np.broadcast_to(self.quaternion.reshape(util.shapeshifter(self.shape,shape_,mode)+(4,)),
|
2022-11-18 05:09:32 +05:30
|
|
|
|
shape_+(4,)))
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
|
|
|
|
|
2022-02-13 05:54:02 +05:30
|
|
|
|
def average(self: MyType,
|
2022-11-23 02:56:15 +05:30
|
|
|
|
weights: Optional[FloatSequence] = None) -> MyType:
|
2020-11-10 01:50:56 +05:30
|
|
|
|
"""
|
2021-03-27 14:40:35 +05:30
|
|
|
|
Average along last array dimension.
|
2020-11-10 01:50:56 +05:30
|
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
|
----------
|
2022-02-13 05:54:02 +05:30
|
|
|
|
weights : numpy.ndarray, shape (self.shape), optional
|
2020-11-10 01:50:56 +05:30
|
|
|
|
Relative weight of each rotation.
|
|
|
|
|
|
|
|
|
|
Returns
|
|
|
|
|
-------
|
2021-04-23 22:45:11 +05:30
|
|
|
|
average : damask.Rotation
|
2020-11-10 01:50:56 +05:30
|
|
|
|
Weighted average of original Rotation field.
|
|
|
|
|
|
|
|
|
|
References
|
|
|
|
|
----------
|
2021-03-18 20:06:40 +05:30
|
|
|
|
F. Landis Markley et al., Journal of Guidance, Control, and Dynamics 30(4):1193-1197, 2007
|
|
|
|
|
https://doi.org/10.2514/1.28949
|
2020-11-10 01:50:56 +05:30
|
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
def _M(quat):
|
|
|
|
|
"""Intermediate representation supporting quaternion averaging."""
|
|
|
|
|
return np.einsum('...i,...j',quat,quat)
|
|
|
|
|
|
2022-02-04 21:27:25 +05:30
|
|
|
|
weights_ = np.ones(self.shape,dtype=float) if weights is None else np.array(weights,float)
|
2020-11-10 01:50:56 +05:30
|
|
|
|
|
2022-11-18 05:09:32 +05:30
|
|
|
|
eig, vec = np.linalg.eig(np.sum(_M(self.quaternion) * weights_[...,np.newaxis,np.newaxis],axis=-3)
|
2022-02-04 21:27:25 +05:30
|
|
|
|
/np.sum( weights_[...,np.newaxis,np.newaxis],axis=-3))
|
2020-11-10 01:50:56 +05:30
|
|
|
|
|
2022-02-13 05:54:02 +05:30
|
|
|
|
return self.copy(Rotation.from_quaternion(np.real(
|
|
|
|
|
np.squeeze(
|
|
|
|
|
np.take_along_axis(vec,
|
|
|
|
|
eig.argmax(axis=-1)[...,np.newaxis,np.newaxis],
|
|
|
|
|
axis=-1),
|
|
|
|
|
axis=-1)),
|
|
|
|
|
accept_homomorph = True))
|
2020-05-02 19:20:46 +05:30
|
|
|
|
|
|
|
|
|
|
2022-02-04 14:27:42 +05:30
|
|
|
|
def misorientation(self: MyType,
|
|
|
|
|
other: MyType) -> MyType:
|
2020-02-21 03:46:35 +05:30
|
|
|
|
"""
|
2021-03-27 14:40:35 +05:30
|
|
|
|
Calculate misorientation to other Rotation.
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
|
----------
|
2021-04-23 22:45:11 +05:30
|
|
|
|
other : damask.Rotation
|
2020-11-10 01:50:56 +05:30
|
|
|
|
Rotation to which the misorientation is computed.
|
2024-02-10 06:22:32 +05:30
|
|
|
|
Compatible innermost dimensions will blend.
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2021-04-23 22:45:11 +05:30
|
|
|
|
Returns
|
|
|
|
|
-------
|
2021-04-25 11:17:00 +05:30
|
|
|
|
g : damask.Rotation
|
2021-04-23 22:45:11 +05:30
|
|
|
|
Misorientation.
|
|
|
|
|
|
2020-02-21 03:46:35 +05:30
|
|
|
|
"""
|
2024-02-10 06:22:32 +05:30
|
|
|
|
return ~(self*~other)
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
|
|
|
|
|
|
|
|
|
################################################################################################
|
|
|
|
|
# convert to different orientation representations (numpy arrays)
|
|
|
|
|
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def as_quaternion(self) -> np.ndarray:
|
2020-02-21 03:46:35 +05:30
|
|
|
|
"""
|
2020-06-20 15:50:43 +05:30
|
|
|
|
Represent as unit quaternion.
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-06-20 15:50:43 +05:30
|
|
|
|
Returns
|
|
|
|
|
-------
|
2022-02-11 22:40:29 +05:30
|
|
|
|
q : numpy.ndarray, shape (...,4)
|
2021-09-08 22:40:49 +05:30
|
|
|
|
Unit quaternion (q_0, q_1, q_2, q_3) in positive real hemisphere, i.e. ǀqǀ = 1, q_0 ≥ 0.
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
|
|
|
|
"""
|
2020-06-20 15:50:43 +05:30
|
|
|
|
return self.quaternion.copy()
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-11-18 03:26:22 +05:30
|
|
|
|
def as_Euler_angles(self,
|
2021-12-14 21:35:00 +05:30
|
|
|
|
degrees: bool = False) -> np.ndarray:
|
2020-02-21 03:46:35 +05:30
|
|
|
|
"""
|
2021-09-09 19:35:09 +05:30
|
|
|
|
Represent as Bunge Euler angles.
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
|
----------
|
|
|
|
|
degrees : bool, optional
|
2022-04-27 20:55:39 +05:30
|
|
|
|
Return angles in degrees. Defaults to False.
|
2020-06-20 15:50:43 +05:30
|
|
|
|
|
|
|
|
|
Returns
|
|
|
|
|
-------
|
2022-02-11 22:40:29 +05:30
|
|
|
|
phi : numpy.ndarray, shape (...,3)
|
2021-09-09 19:35:09 +05:30
|
|
|
|
Bunge Euler angles (φ_1 ∈ [0,2π], ϕ ∈ [0,π], φ_2 ∈ [0,2π])
|
2021-09-08 22:40:49 +05:30
|
|
|
|
or (φ_1 ∈ [0,360], ϕ ∈ [0,180], φ_2 ∈ [0,360]) if degrees == True.
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2021-09-09 19:35:09 +05:30
|
|
|
|
Notes
|
|
|
|
|
-----
|
|
|
|
|
Bunge Euler angles correspond to a rotation axis sequence of z–x'–z''.
|
|
|
|
|
|
2021-07-16 13:51:06 +05:30
|
|
|
|
Examples
|
|
|
|
|
--------
|
2021-09-09 19:35:09 +05:30
|
|
|
|
Cube orientation as Bunge Euler angles.
|
2021-07-16 13:51:06 +05:30
|
|
|
|
|
|
|
|
|
>>> import damask
|
2022-11-08 01:26:38 +05:30
|
|
|
|
>>> damask.Rotation([1,0,0,0]).as_Euler_angles()
|
2021-07-16 13:51:06 +05:30
|
|
|
|
array([0., 0., 0.])
|
|
|
|
|
|
2020-02-21 03:46:35 +05:30
|
|
|
|
"""
|
2020-05-20 14:41:07 +05:30
|
|
|
|
eu = Rotation._qu2eu(self.quaternion)
|
2022-02-11 22:40:29 +05:30
|
|
|
|
return np.degrees(eu) if degrees else eu
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-04-21 15:57:50 +05:30
|
|
|
|
def as_axis_angle(self,
|
2021-12-14 21:35:00 +05:30
|
|
|
|
degrees: bool = False,
|
2022-01-14 19:25:08 +05:30
|
|
|
|
pair: bool = False) -> Union[Tuple[np.ndarray, np.ndarray], np.ndarray]:
|
2020-02-21 03:46:35 +05:30
|
|
|
|
"""
|
2021-09-08 22:40:49 +05:30
|
|
|
|
Represent as axis–angle pair.
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
|
----------
|
|
|
|
|
degrees : bool, optional
|
2020-06-20 15:50:43 +05:30
|
|
|
|
Return rotation angle in degrees. Defaults to False.
|
2020-02-21 03:46:35 +05:30
|
|
|
|
pair : bool, optional
|
2020-06-20 15:50:43 +05:30
|
|
|
|
Return tuple of axis and angle. Defaults to False.
|
|
|
|
|
|
|
|
|
|
Returns
|
|
|
|
|
-------
|
2022-12-11 20:39:54 +05:30
|
|
|
|
n_omega : numpy.ndarray, shape (...,4) or tuple ((...,3), (...)) if pair == True
|
2021-09-09 19:35:09 +05:30
|
|
|
|
Axis and angle [n_1, n_2, n_3, ω] with ǀnǀ = 1 and ω ∈ [0,π]
|
2021-09-08 22:40:49 +05:30
|
|
|
|
or ω ∈ [0,180] if degrees == True.
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2021-07-16 13:51:06 +05:30
|
|
|
|
Examples
|
|
|
|
|
--------
|
2021-09-08 22:40:49 +05:30
|
|
|
|
Cube orientation as axis–angle pair.
|
2021-07-16 13:51:06 +05:30
|
|
|
|
|
|
|
|
|
>>> import damask
|
2022-11-08 01:26:38 +05:30
|
|
|
|
>>> damask.Rotation([1,0,0,0]).as_axis_angle(pair=True)
|
2021-09-08 22:40:49 +05:30
|
|
|
|
(array([0., 0., 1.]), array(0.))
|
2021-07-16 13:51:06 +05:30
|
|
|
|
|
2020-02-21 03:46:35 +05:30
|
|
|
|
"""
|
2021-12-14 21:35:00 +05:30
|
|
|
|
ax: np.ndarray = Rotation._qu2ax(self.quaternion)
|
2020-04-21 15:57:50 +05:30
|
|
|
|
if degrees: ax[...,3] = np.degrees(ax[...,3])
|
|
|
|
|
return (ax[...,:3],ax[...,3]) if pair else ax
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def as_matrix(self) -> np.ndarray:
|
2020-06-20 15:50:43 +05:30
|
|
|
|
"""
|
|
|
|
|
Represent as rotation matrix.
|
|
|
|
|
|
|
|
|
|
Returns
|
|
|
|
|
-------
|
2022-02-11 22:40:29 +05:30
|
|
|
|
R : numpy.ndarray, shape (...,3,3)
|
2021-09-08 22:40:49 +05:30
|
|
|
|
Rotation matrix R with det(R) = 1, R.T ∙ R = I.
|
2020-06-20 15:50:43 +05:30
|
|
|
|
|
2021-07-16 13:51:06 +05:30
|
|
|
|
Examples
|
|
|
|
|
--------
|
|
|
|
|
Cube orientation as rotation matrix.
|
|
|
|
|
|
|
|
|
|
>>> import damask
|
2022-11-08 01:26:38 +05:30
|
|
|
|
>>> damask.Rotation([1,0,0,0]).as_matrix()
|
2021-07-16 13:51:06 +05:30
|
|
|
|
array([[1., 0., 0.],
|
|
|
|
|
[0., 1., 0.],
|
|
|
|
|
[0., 0., 1.]])
|
2021-07-16 14:04:56 +05:30
|
|
|
|
|
2020-06-20 15:50:43 +05:30
|
|
|
|
"""
|
2020-05-20 14:41:07 +05:30
|
|
|
|
return Rotation._qu2om(self.quaternion)
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-11-18 03:26:22 +05:30
|
|
|
|
def as_Rodrigues_vector(self,
|
2021-12-14 21:35:00 +05:30
|
|
|
|
compact: bool = False) -> np.ndarray:
|
2020-02-21 03:46:35 +05:30
|
|
|
|
"""
|
2021-09-09 00:09:06 +05:30
|
|
|
|
Represent as Rodrigues–Frank vector with separate axis and angle argument.
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
|
----------
|
2021-03-02 20:56:18 +05:30
|
|
|
|
compact : bool, optional
|
2021-09-09 00:09:06 +05:30
|
|
|
|
Return three-component Rodrigues–Frank vector,
|
2021-03-02 20:56:18 +05:30
|
|
|
|
i.e. axis and angle argument are not separated.
|
2020-06-20 15:50:43 +05:30
|
|
|
|
|
|
|
|
|
Returns
|
|
|
|
|
-------
|
2022-02-11 22:40:29 +05:30
|
|
|
|
rho : numpy.ndarray, shape (...,4) or (...,3) if compact == True
|
2021-09-08 22:40:49 +05:30
|
|
|
|
Rodrigues–Frank vector [n_1, n_2, n_3, tan(ω/2)] with ǀnǀ = 1 and ω ∈ [0,π]
|
2021-09-24 01:16:53 +05:30
|
|
|
|
or [n_1, n_2, n_3] with ǀnǀ = tan(ω/2) and ω ∈ [0,π] if compact == True.
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2021-07-16 13:51:06 +05:30
|
|
|
|
Examples
|
|
|
|
|
--------
|
2021-09-08 22:40:49 +05:30
|
|
|
|
Cube orientation as three-component Rodrigues–Frank vector.
|
2021-07-16 13:51:06 +05:30
|
|
|
|
|
|
|
|
|
>>> import damask
|
2022-11-08 01:26:38 +05:30
|
|
|
|
>>> damask.Rotation([1,0,0,0]).as_Rodrigues_vector(compact=True)
|
2021-07-16 13:51:06 +05:30
|
|
|
|
array([ 0., 0., 0.])
|
|
|
|
|
|
2020-02-21 03:46:35 +05:30
|
|
|
|
"""
|
2020-05-20 14:41:07 +05:30
|
|
|
|
ro = Rotation._qu2ro(self.quaternion)
|
2020-11-19 19:01:14 +05:30
|
|
|
|
if compact:
|
2020-07-01 19:56:56 +05:30
|
|
|
|
with np.errstate(invalid='ignore'):
|
|
|
|
|
return ro[...,:3]*ro[...,3:4]
|
|
|
|
|
else:
|
|
|
|
|
return ro
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def as_homochoric(self) -> np.ndarray:
|
2020-06-20 15:50:43 +05:30
|
|
|
|
"""
|
|
|
|
|
Represent as homochoric vector.
|
|
|
|
|
|
|
|
|
|
Returns
|
|
|
|
|
-------
|
2022-02-11 22:40:29 +05:30
|
|
|
|
h : numpy.ndarray, shape (...,3)
|
2021-09-08 22:40:49 +05:30
|
|
|
|
Homochoric vector (h_1, h_2, h_3) with ǀhǀ < (3/4*π)^(1/3).
|
2020-06-20 15:50:43 +05:30
|
|
|
|
|
2021-07-16 13:51:06 +05:30
|
|
|
|
Examples
|
|
|
|
|
--------
|
|
|
|
|
Cube orientation as homochoric vector.
|
|
|
|
|
|
|
|
|
|
>>> import damask
|
2022-11-08 01:26:38 +05:30
|
|
|
|
>>> damask.Rotation([1,0,0,0]).as_homochoric()
|
2021-07-16 13:51:06 +05:30
|
|
|
|
array([0., 0., 0.])
|
|
|
|
|
|
2020-06-20 15:50:43 +05:30
|
|
|
|
"""
|
2020-05-20 14:41:07 +05:30
|
|
|
|
return Rotation._qu2ho(self.quaternion)
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def as_cubochoric(self) -> np.ndarray:
|
2020-06-20 15:50:43 +05:30
|
|
|
|
"""
|
2020-06-20 21:43:34 +05:30
|
|
|
|
Represent as cubochoric vector.
|
2020-06-20 15:50:43 +05:30
|
|
|
|
|
|
|
|
|
Returns
|
|
|
|
|
-------
|
2022-02-11 22:40:29 +05:30
|
|
|
|
x : numpy.ndarray, shape (...,3)
|
2022-11-19 13:40:00 +05:30
|
|
|
|
Cubochoric vector (x_1, x_2, x_3) with max(x_i) < 1/2*π^(2/3).
|
2020-06-20 15:50:43 +05:30
|
|
|
|
|
2021-07-16 13:51:06 +05:30
|
|
|
|
Examples
|
|
|
|
|
--------
|
|
|
|
|
Cube orientation as cubochoric vector.
|
|
|
|
|
|
|
|
|
|
>>> import damask
|
2022-11-08 01:26:38 +05:30
|
|
|
|
>>> damask.Rotation([1,0,0,0]).as_cubochoric()
|
2021-07-16 13:51:06 +05:30
|
|
|
|
array([0., 0., 0.])
|
|
|
|
|
|
2020-06-20 15:50:43 +05:30
|
|
|
|
"""
|
2020-05-20 14:41:07 +05:30
|
|
|
|
return Rotation._qu2cu(self.quaternion)
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
|
|
|
|
################################################################################################
|
2020-04-21 15:57:50 +05:30
|
|
|
|
# Static constructors. The input data needs to follow the conventions, options allow to
|
|
|
|
|
# relax the conventions.
|
2020-02-21 03:46:35 +05:30
|
|
|
|
@staticmethod
|
2022-01-14 19:07:48 +05:30
|
|
|
|
def from_quaternion(q: Union[Sequence[FloatSequence], np.ndarray],
|
2021-12-14 21:35:00 +05:30
|
|
|
|
accept_homomorph: bool = False,
|
2022-11-16 02:41:29 +05:30
|
|
|
|
normalize: bool = False,
|
2022-02-02 15:41:59 +05:30
|
|
|
|
P: Literal[1, -1] = -1) -> 'Rotation':
|
2020-06-20 15:50:43 +05:30
|
|
|
|
"""
|
|
|
|
|
Initialize from quaternion.
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-06-20 15:50:43 +05:30
|
|
|
|
Parameters
|
|
|
|
|
----------
|
2022-02-11 22:40:29 +05:30
|
|
|
|
q : numpy.ndarray, shape (...,4)
|
2022-05-11 01:15:19 +05:30
|
|
|
|
Unit quaternion (q_0, q_1, q_2, q_3) in positive real hemisphere, i.e. ǀqǀ = 1 and q_0 ≥ 0.
|
2022-01-13 03:43:38 +05:30
|
|
|
|
accept_homomorph : bool, optional
|
2020-06-20 15:50:43 +05:30
|
|
|
|
Allow homomorphic variants, i.e. q_0 < 0 (negative real hemisphere).
|
|
|
|
|
Defaults to False.
|
2022-11-16 02:41:29 +05:30
|
|
|
|
normalize: bool, optional
|
|
|
|
|
Allow ǀqǀ ≠ 1. Defaults to False.
|
2020-08-09 00:26:17 +05:30
|
|
|
|
P : int ∈ {-1,1}, optional
|
2021-09-08 22:40:49 +05:30
|
|
|
|
Sign convention. Defaults to -1.
|
2020-06-20 15:50:43 +05:30
|
|
|
|
|
2022-11-19 13:40:00 +05:30
|
|
|
|
Returns
|
|
|
|
|
-------
|
|
|
|
|
new : damask.Rotation
|
|
|
|
|
|
2020-06-20 15:50:43 +05:30
|
|
|
|
"""
|
2022-02-13 05:54:02 +05:30
|
|
|
|
qu = np.array(q,dtype=float)
|
2022-11-18 05:09:32 +05:30
|
|
|
|
if qu.shape[:-2:-1] != (4,): raise ValueError('invalid shape')
|
|
|
|
|
if abs(P) != 1: raise ValueError('P ∉ {-1,1}')
|
2020-04-16 02:30:00 +05:30
|
|
|
|
|
2020-11-10 01:50:56 +05:30
|
|
|
|
qu[...,1:4] *= -P
|
2022-11-16 02:41:29 +05:30
|
|
|
|
|
2020-05-19 12:29:03 +05:30
|
|
|
|
if accept_homomorph:
|
2022-11-18 05:09:32 +05:30
|
|
|
|
qu[qu[...,0]<0.] *= -1.
|
|
|
|
|
elif np.any(qu[...,0] < 0.):
|
2022-11-16 02:41:29 +05:30
|
|
|
|
raise ValueError('quaternion with negative first (real) component')
|
|
|
|
|
if normalize:
|
|
|
|
|
qu /= np.linalg.norm(qu,axis=-1,keepdims=True)
|
2022-11-18 05:09:32 +05:30
|
|
|
|
elif not np.allclose(np.linalg.norm(qu,axis=-1),1.,rtol=1.e-8):
|
2022-02-22 21:12:05 +05:30
|
|
|
|
raise ValueError('quaternion is not of unit length')
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
|
|
|
|
return Rotation(qu)
|
|
|
|
|
|
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def from_Euler_angles(phi: np.ndarray,
|
2022-02-02 15:41:59 +05:30
|
|
|
|
degrees: bool = False) -> 'Rotation':
|
2020-06-20 15:50:43 +05:30
|
|
|
|
"""
|
2021-09-09 19:35:09 +05:30
|
|
|
|
Initialize from Bunge Euler angles.
|
2020-04-16 02:30:00 +05:30
|
|
|
|
|
2020-06-20 15:50:43 +05:30
|
|
|
|
Parameters
|
|
|
|
|
----------
|
2022-02-11 22:40:29 +05:30
|
|
|
|
phi : numpy.ndarray, shape (...,3)
|
2021-09-09 19:35:09 +05:30
|
|
|
|
Euler angles (φ_1 ∈ [0,2π], ϕ ∈ [0,π], φ_2 ∈ [0,2π])
|
2021-09-09 00:15:12 +05:30
|
|
|
|
or (φ_1 ∈ [0,360], ϕ ∈ [0,180], φ_2 ∈ [0,360]) if degrees == True.
|
2022-01-13 03:43:38 +05:30
|
|
|
|
degrees : bool, optional
|
2021-09-09 19:35:09 +05:30
|
|
|
|
Euler angles are given in degrees. Defaults to False.
|
|
|
|
|
|
2022-11-19 13:40:00 +05:30
|
|
|
|
Returns
|
|
|
|
|
-------
|
|
|
|
|
new : damask.Rotation
|
|
|
|
|
|
2021-09-09 19:35:09 +05:30
|
|
|
|
Notes
|
|
|
|
|
-----
|
|
|
|
|
Bunge Euler angles correspond to a rotation axis sequence of z–x'–z''.
|
2020-06-20 15:50:43 +05:30
|
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
eu = np.array(phi,dtype=float)
|
2022-11-18 05:09:32 +05:30
|
|
|
|
if eu.shape[:-2:-1] != (3,): raise ValueError('invalid shape')
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
|
|
|
|
eu = np.radians(eu) if degrees else eu
|
2022-11-18 05:09:32 +05:30
|
|
|
|
if np.any(eu < 0.) or np.any(eu > np.pi*np.array([2.,1.,2.])):
|
2022-02-22 20:16:12 +05:30
|
|
|
|
raise ValueError('Euler angles outside of [0..2π],[0..π],[0..2π]')
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-05-20 14:41:07 +05:30
|
|
|
|
return Rotation(Rotation._eu2qu(eu))
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
|
|
|
|
@staticmethod
|
2022-12-11 20:39:54 +05:30
|
|
|
|
def from_axis_angle(n_omega: np.ndarray,
|
2022-05-11 01:15:19 +05:30
|
|
|
|
degrees: bool = False,
|
2021-12-14 21:35:00 +05:30
|
|
|
|
normalize: bool = False,
|
2022-02-02 15:41:59 +05:30
|
|
|
|
P: Literal[1, -1] = -1) -> 'Rotation':
|
2020-06-20 15:50:43 +05:30
|
|
|
|
"""
|
2022-05-11 01:15:19 +05:30
|
|
|
|
Initialize from axis–angle pair.
|
2020-04-16 02:30:00 +05:30
|
|
|
|
|
2020-06-20 15:50:43 +05:30
|
|
|
|
Parameters
|
|
|
|
|
----------
|
2022-12-11 20:39:54 +05:30
|
|
|
|
n_omega : numpy.ndarray, shape (...,4)
|
2021-09-08 22:40:49 +05:30
|
|
|
|
Axis and angle (n_1, n_2, n_3, ω) with ǀnǀ = 1 and ω ∈ [0,π]
|
|
|
|
|
or ω ∈ [0,180] if degrees == True.
|
2022-01-13 03:43:38 +05:30
|
|
|
|
degrees : bool, optional
|
2020-06-20 15:50:43 +05:30
|
|
|
|
Angle ω is given in degrees. Defaults to False.
|
2022-01-13 03:43:38 +05:30
|
|
|
|
normalize: bool, optional
|
2020-09-23 19:51:20 +05:30
|
|
|
|
Allow ǀnǀ ≠ 1. Defaults to False.
|
2020-08-09 00:26:17 +05:30
|
|
|
|
P : int ∈ {-1,1}, optional
|
2021-09-08 22:40:49 +05:30
|
|
|
|
Sign convention. Defaults to -1.
|
2020-06-20 15:50:43 +05:30
|
|
|
|
|
2022-11-19 13:40:00 +05:30
|
|
|
|
Returns
|
|
|
|
|
-------
|
|
|
|
|
new : damask.Rotation
|
|
|
|
|
|
2020-06-20 15:50:43 +05:30
|
|
|
|
"""
|
2022-12-11 20:39:54 +05:30
|
|
|
|
ax = np.array(n_omega,dtype=float)
|
2022-11-18 05:09:32 +05:30
|
|
|
|
if ax.shape[:-2:-1] != (4,): raise ValueError('invalid shape')
|
|
|
|
|
if abs(P) != 1: raise ValueError('P ∉ {-1,1}')
|
2020-04-16 02:30:00 +05:30
|
|
|
|
|
2020-11-10 01:50:56 +05:30
|
|
|
|
ax[...,0:3] *= -P
|
2022-11-18 05:09:32 +05:30
|
|
|
|
if degrees: ax[..., 3] = np.radians(ax[...,3])
|
|
|
|
|
if np.any(ax[...,3] < 0.) or np.any(ax[...,3] > np.pi):
|
2022-02-22 21:12:05 +05:30
|
|
|
|
raise ValueError('axis–angle rotation angle outside of [0..π]')
|
2022-11-18 05:09:32 +05:30
|
|
|
|
|
|
|
|
|
if normalize:
|
|
|
|
|
ax[...,0:3] /= np.linalg.norm(ax[...,0:3],axis=-1,keepdims=True)
|
|
|
|
|
elif not np.allclose(np.linalg.norm(ax[...,0:3],axis=-1),1.):
|
2022-02-22 21:12:05 +05:30
|
|
|
|
raise ValueError('axis–angle rotation axis is not of unit length')
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-05-20 14:41:07 +05:30
|
|
|
|
return Rotation(Rotation._ax2qu(ax))
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def from_basis(basis: np.ndarray,
|
|
|
|
|
orthonormal: bool = True,
|
2022-02-02 15:41:59 +05:30
|
|
|
|
reciprocal: bool = False) -> 'Rotation':
|
2020-06-20 15:50:43 +05:30
|
|
|
|
"""
|
2022-05-11 01:15:19 +05:30
|
|
|
|
Initialize from basis vector triplet.
|
2020-04-16 02:30:00 +05:30
|
|
|
|
|
2020-06-20 15:50:43 +05:30
|
|
|
|
Parameters
|
|
|
|
|
----------
|
2022-02-11 22:40:29 +05:30
|
|
|
|
basis : numpy.ndarray, shape (...,3,3)
|
2022-05-11 01:15:19 +05:30
|
|
|
|
Three three-dimensional basis vectors.
|
2022-01-13 03:43:38 +05:30
|
|
|
|
orthonormal : bool, optional
|
2020-06-20 23:27:49 +05:30
|
|
|
|
Basis is strictly orthonormal, i.e. is free of stretch components. Defaults to True.
|
2022-01-13 03:43:38 +05:30
|
|
|
|
reciprocal : bool, optional
|
2020-06-20 23:27:49 +05:30
|
|
|
|
Basis vectors are given in reciprocal (instead of real) space. Defaults to False.
|
2020-06-20 15:50:43 +05:30
|
|
|
|
|
2022-11-24 14:45:23 +05:30
|
|
|
|
Returns
|
|
|
|
|
-------
|
|
|
|
|
new : damask.Rotation
|
|
|
|
|
|
2020-06-20 15:50:43 +05:30
|
|
|
|
"""
|
2020-04-16 02:30:00 +05:30
|
|
|
|
om = np.array(basis,dtype=float)
|
2022-11-18 05:09:32 +05:30
|
|
|
|
if om.shape[-2:] != (3,3): raise ValueError('invalid shape')
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
|
|
|
|
if reciprocal:
|
2020-11-16 03:44:46 +05:30
|
|
|
|
om = np.linalg.inv(tensor.transpose(om)/np.pi) # transform reciprocal basis set
|
2020-02-21 03:46:35 +05:30
|
|
|
|
orthonormal = False # contains stretch
|
2022-11-18 05:09:32 +05:30
|
|
|
|
|
2020-02-21 03:46:35 +05:30
|
|
|
|
if not orthonormal:
|
|
|
|
|
(U,S,Vh) = np.linalg.svd(om) # singular value decomposition
|
2020-11-16 11:42:37 +05:30
|
|
|
|
om = np.einsum('...ij,...jl',U,Vh)
|
2022-11-18 05:09:32 +05:30
|
|
|
|
elif not np.allclose(np.einsum('...i,...i',om[...,0],om[...,1]),0.) \
|
|
|
|
|
or not np.allclose(np.einsum('...i,...i',om[...,1],om[...,2]),0.) \
|
|
|
|
|
or not np.allclose(np.einsum('...i,...i',om[...,2],om[...,0]),0.):
|
2022-02-22 21:12:05 +05:30
|
|
|
|
raise ValueError('orientation matrix is not orthogonal')
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2022-11-18 05:09:32 +05:30
|
|
|
|
if not np.allclose(np.linalg.det(om),1.):
|
|
|
|
|
raise ValueError('orientation matrix has determinant ≠ 1')
|
|
|
|
|
|
2020-05-20 14:41:07 +05:30
|
|
|
|
return Rotation(Rotation._om2qu(om))
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
|
|
|
|
@staticmethod
|
2023-01-17 05:23:49 +05:30
|
|
|
|
def from_matrix(R: np.ndarray,
|
|
|
|
|
normalize: bool = False) -> 'Rotation':
|
2020-06-20 15:50:43 +05:30
|
|
|
|
"""
|
|
|
|
|
Initialize from rotation matrix.
|
|
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
|
----------
|
2022-02-11 22:40:29 +05:30
|
|
|
|
R : numpy.ndarray, shape (...,3,3)
|
2022-05-11 01:15:19 +05:30
|
|
|
|
Rotation matrix with det(R) = 1 and R.T ∙ R = I.
|
2023-01-17 05:23:49 +05:30
|
|
|
|
normalize : bool, optional
|
|
|
|
|
Rescales rotation matrix to unit determinant. Defaults to False.
|
2020-06-20 15:50:43 +05:30
|
|
|
|
|
2022-11-19 13:40:00 +05:30
|
|
|
|
Returns
|
|
|
|
|
-------
|
|
|
|
|
new : damask.Rotation
|
|
|
|
|
|
2020-06-20 15:50:43 +05:30
|
|
|
|
"""
|
2023-01-17 05:23:49 +05:30
|
|
|
|
return Rotation.from_basis(np.array(R,dtype=float) * (np.linalg.det(R)**(-1./3.))[...,np.newaxis,np.newaxis]
|
|
|
|
|
if normalize else
|
|
|
|
|
R)
|
2020-06-20 15:50:43 +05:30
|
|
|
|
|
2020-11-10 01:50:56 +05:30
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def from_parallel(a: np.ndarray,
|
2022-02-02 15:41:59 +05:30
|
|
|
|
b: np.ndarray ) -> 'Rotation':
|
2020-11-10 01:50:56 +05:30
|
|
|
|
"""
|
2022-05-11 01:15:19 +05:30
|
|
|
|
Initialize from pairs of two orthogonal basis vectors.
|
2020-11-10 01:50:56 +05:30
|
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
|
----------
|
2022-02-11 22:40:29 +05:30
|
|
|
|
a : numpy.ndarray, shape (...,2,3)
|
2022-05-11 01:15:19 +05:30
|
|
|
|
Two three-dimensional vectors of first orthogonal basis.
|
2022-02-11 22:40:29 +05:30
|
|
|
|
b : numpy.ndarray, shape (...,2,3)
|
2022-05-11 01:15:19 +05:30
|
|
|
|
Corresponding three-dimensional vectors of second basis.
|
2020-11-10 01:50:56 +05:30
|
|
|
|
|
2022-11-19 13:40:00 +05:30
|
|
|
|
Returns
|
|
|
|
|
-------
|
|
|
|
|
new : damask.Rotation
|
|
|
|
|
|
2020-11-10 01:50:56 +05:30
|
|
|
|
"""
|
2022-11-18 05:09:32 +05:30
|
|
|
|
a_ = np.array(a,dtype=float)
|
|
|
|
|
b_ = np.array(b,dtype=float)
|
2020-11-10 01:50:56 +05:30
|
|
|
|
if a_.shape[-2:] != (2,3) or b_.shape[-2:] != (2,3) or a_.shape != b_.shape:
|
2022-02-22 21:12:05 +05:30
|
|
|
|
raise ValueError('invalid shape')
|
2020-11-10 01:50:56 +05:30
|
|
|
|
am = np.stack([ a_[...,0,:],
|
|
|
|
|
a_[...,1,:],
|
|
|
|
|
np.cross(a_[...,0,:],a_[...,1,:]) ],axis=-2)
|
|
|
|
|
bm = np.stack([ b_[...,0,:],
|
|
|
|
|
b_[...,1,:],
|
|
|
|
|
np.cross(b_[...,0,:],b_[...,1,:]) ],axis=-2)
|
|
|
|
|
|
|
|
|
|
return Rotation.from_basis(np.swapaxes(am/np.linalg.norm(am,axis=-1,keepdims=True),-1,-2))\
|
|
|
|
|
.misorientation(Rotation.from_basis(np.swapaxes(bm/np.linalg.norm(bm,axis=-1,keepdims=True),-1,-2)))
|
|
|
|
|
|
|
|
|
|
|
2020-06-20 15:50:43 +05:30
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def from_Rodrigues_vector(rho: np.ndarray,
|
|
|
|
|
normalize: bool = False,
|
2022-02-02 15:41:59 +05:30
|
|
|
|
P: Literal[1, -1] = -1) -> 'Rotation':
|
2020-06-20 15:50:43 +05:30
|
|
|
|
"""
|
2022-05-11 01:15:19 +05:30
|
|
|
|
Initialize from Rodrigues–Frank vector (with angle separated from axis).
|
2020-06-20 15:50:43 +05:30
|
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
|
----------
|
2022-02-11 22:40:29 +05:30
|
|
|
|
rho : numpy.ndarray, shape (...,4)
|
2021-09-08 22:40:49 +05:30
|
|
|
|
Rodrigues–Frank vector (n_1, n_2, n_3, tan(ω/2)) with ǀnǀ = 1 and ω ∈ [0,π].
|
2022-01-13 03:43:38 +05:30
|
|
|
|
normalize : bool, optional
|
2020-09-23 19:51:20 +05:30
|
|
|
|
Allow ǀnǀ ≠ 1. Defaults to False.
|
2020-08-09 00:26:17 +05:30
|
|
|
|
P : int ∈ {-1,1}, optional
|
2021-09-08 22:40:49 +05:30
|
|
|
|
Sign convention. Defaults to -1.
|
2020-04-16 02:30:00 +05:30
|
|
|
|
|
2022-11-19 13:40:00 +05:30
|
|
|
|
Returns
|
|
|
|
|
-------
|
|
|
|
|
new : damask.Rotation
|
|
|
|
|
|
2020-06-20 15:50:43 +05:30
|
|
|
|
"""
|
|
|
|
|
ro = np.array(rho,dtype=float)
|
2022-11-18 05:09:32 +05:30
|
|
|
|
if ro.shape[:-2:-1] != (4,): raise ValueError('invalid shape')
|
|
|
|
|
if abs(P) != 1: raise ValueError('P ∉ {-1,1}')
|
2020-04-16 02:30:00 +05:30
|
|
|
|
|
2020-11-10 01:50:56 +05:30
|
|
|
|
ro[...,0:3] *= -P
|
2022-11-18 05:09:32 +05:30
|
|
|
|
if np.any(ro[...,3] < 0.): raise ValueError('Rodrigues vector rotation angle is negative')
|
|
|
|
|
|
|
|
|
|
if normalize:
|
|
|
|
|
ro[...,0:3] /= np.linalg.norm(ro[...,0:3],axis=-1,keepdims=True)
|
|
|
|
|
elif not np.allclose(np.linalg.norm(ro[...,0:3],axis=-1),1.):
|
2022-02-22 20:16:12 +05:30
|
|
|
|
raise ValueError('Rodrigues vector rotation axis is not of unit length')
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-05-20 14:41:07 +05:30
|
|
|
|
return Rotation(Rotation._ro2qu(ro))
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def from_homochoric(h: np.ndarray,
|
2022-02-02 15:41:59 +05:30
|
|
|
|
P: Literal[1, -1] = -1) -> 'Rotation':
|
2020-06-20 15:50:43 +05:30
|
|
|
|
"""
|
|
|
|
|
Initialize from homochoric vector.
|
2020-04-16 02:30:00 +05:30
|
|
|
|
|
2020-06-20 15:50:43 +05:30
|
|
|
|
Parameters
|
|
|
|
|
----------
|
2022-02-11 22:40:29 +05:30
|
|
|
|
h : numpy.ndarray, shape (...,3)
|
2021-09-08 22:40:49 +05:30
|
|
|
|
Homochoric vector (h_1, h_2, h_3) with ǀhǀ < (3/4*π)^(1/3).
|
2020-08-09 00:26:17 +05:30
|
|
|
|
P : int ∈ {-1,1}, optional
|
2021-09-08 22:40:49 +05:30
|
|
|
|
Sign convention. Defaults to -1.
|
2020-06-20 15:50:43 +05:30
|
|
|
|
|
2022-11-19 13:40:00 +05:30
|
|
|
|
Returns
|
|
|
|
|
-------
|
|
|
|
|
new : damask.Rotation
|
|
|
|
|
|
2020-06-20 15:50:43 +05:30
|
|
|
|
"""
|
|
|
|
|
ho = np.array(h,dtype=float)
|
2022-11-18 05:09:32 +05:30
|
|
|
|
if ho.shape[:-2:-1] != (3,): raise ValueError('invalid shape')
|
|
|
|
|
if abs(P) != 1: raise ValueError('P ∉ {-1,1}')
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-11-10 01:50:56 +05:30
|
|
|
|
ho *= -P
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2022-11-18 05:09:32 +05:30
|
|
|
|
if np.any(np.linalg.norm(ho,axis=-1) > _R1+1.e-9):
|
2022-02-22 21:12:05 +05:30
|
|
|
|
raise ValueError('homochoric coordinate outside of the sphere')
|
2020-04-12 04:59:11 +05:30
|
|
|
|
|
2020-05-20 14:41:07 +05:30
|
|
|
|
return Rotation(Rotation._ho2qu(ho))
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def from_cubochoric(x: np.ndarray,
|
2022-02-02 15:41:59 +05:30
|
|
|
|
P: Literal[1, -1] = -1) -> 'Rotation':
|
2020-06-20 15:50:43 +05:30
|
|
|
|
"""
|
|
|
|
|
Initialize from cubochoric vector.
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-06-20 15:50:43 +05:30
|
|
|
|
Parameters
|
|
|
|
|
----------
|
2022-02-11 22:40:29 +05:30
|
|
|
|
x : numpy.ndarray, shape (...,3)
|
2021-09-08 22:40:49 +05:30
|
|
|
|
Cubochoric vector (x_1, x_2, x_3) with max(x_i) < 1/2*π^(2/3).
|
2020-08-09 00:26:17 +05:30
|
|
|
|
P : int ∈ {-1,1}, optional
|
2021-09-08 22:40:49 +05:30
|
|
|
|
Sign convention. Defaults to -1.
|
2020-06-20 15:50:43 +05:30
|
|
|
|
|
2022-11-19 13:40:00 +05:30
|
|
|
|
Returns
|
|
|
|
|
-------
|
|
|
|
|
new : damask.Rotation
|
|
|
|
|
|
2020-06-20 15:50:43 +05:30
|
|
|
|
"""
|
2021-02-12 16:45:08 +05:30
|
|
|
|
cu = np.array(x,dtype=float)
|
2022-11-18 05:09:32 +05:30
|
|
|
|
if cu.shape[:-2:-1] != (3,): raise ValueError('invalid shape')
|
|
|
|
|
if abs(P) != 1: raise ValueError('P ∉ {-1,1}')
|
|
|
|
|
if np.abs(np.max(cu)) > np.pi**(2./3.) * 0.5+1.e-9:
|
2022-02-22 21:12:05 +05:30
|
|
|
|
raise ValueError('cubochoric coordinate outside of the cube')
|
2020-04-12 04:59:11 +05:30
|
|
|
|
|
2020-11-10 01:50:56 +05:30
|
|
|
|
ho = -P * Rotation._cu2ho(cu)
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-05-20 14:41:07 +05:30
|
|
|
|
return Rotation(Rotation._ho2qu(ho))
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@staticmethod
|
2022-11-23 02:56:15 +05:30
|
|
|
|
def from_random(shape: Union[None, int, IntSequence] = None,
|
|
|
|
|
rng_seed: Optional[NumpyRngSeed] = None) -> 'Rotation':
|
2020-09-15 11:42:02 +05:30
|
|
|
|
"""
|
2022-03-19 15:25:03 +05:30
|
|
|
|
Initialize with samples from a uniform distribution.
|
2020-09-15 11:42:02 +05:30
|
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
|
----------
|
2022-11-14 19:39:45 +05:30
|
|
|
|
shape : (sequence of) int, optional
|
2022-03-19 15:25:03 +05:30
|
|
|
|
Shape of the returned array. Defaults to None, which gives a scalar.
|
2020-11-15 17:36:26 +05:30
|
|
|
|
rng_seed : {None, int, array_like[ints], SeedSequence, BitGenerator, Generator}, optional
|
2021-09-08 22:40:49 +05:30
|
|
|
|
A seed to initialize the BitGenerator.
|
|
|
|
|
Defaults to None, i.e. unpredictable entropy will be pulled from the OS.
|
2020-09-15 11:42:02 +05:30
|
|
|
|
|
2022-11-19 13:40:00 +05:30
|
|
|
|
Returns
|
|
|
|
|
-------
|
|
|
|
|
new : damask.Rotation
|
|
|
|
|
|
2020-09-15 11:42:02 +05:30
|
|
|
|
"""
|
2022-02-13 05:54:02 +05:30
|
|
|
|
rng = np.random.default_rng(rng_seed)
|
2022-05-23 11:31:17 +05:30
|
|
|
|
r = rng.random(3 if shape is None else tuple(shape)+(3,) if hasattr(shape, '__iter__') else (shape,3)) # type: ignore
|
2020-04-21 15:29:42 +05:30
|
|
|
|
|
2020-04-16 02:30:00 +05:30
|
|
|
|
A = np.sqrt(r[...,2])
|
2022-11-18 05:09:32 +05:30
|
|
|
|
B = np.sqrt(1.-r[...,2])
|
|
|
|
|
q = np.stack([np.cos(2.*np.pi*r[...,0])*A,
|
|
|
|
|
np.sin(2.*np.pi*r[...,1])*B,
|
|
|
|
|
np.cos(2.*np.pi*r[...,1])*B,
|
|
|
|
|
np.sin(2.*np.pi*r[...,0])*A],axis=-1)
|
2020-04-21 15:29:42 +05:30
|
|
|
|
|
2020-11-10 01:50:56 +05:30
|
|
|
|
return Rotation(q if shape is None else q.reshape(r.shape[:-1]+(4,)))._standardize()
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-09-15 15:26:13 +05:30
|
|
|
|
|
2020-09-28 16:44:23 +05:30
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def from_ODF(weights: np.ndarray,
|
|
|
|
|
phi: np.ndarray,
|
2022-11-23 02:56:15 +05:30
|
|
|
|
shape: Union[None, int, IntSequence] = None,
|
2022-04-28 19:05:50 +05:30
|
|
|
|
degrees: bool = False,
|
2021-12-14 21:35:00 +05:30
|
|
|
|
fractions: bool = True,
|
2022-11-23 02:56:15 +05:30
|
|
|
|
rng_seed: Optional[NumpyRngSeed] = None) -> 'Rotation':
|
2020-09-28 16:44:23 +05:30
|
|
|
|
"""
|
2022-03-19 15:25:03 +05:30
|
|
|
|
Initialize with samples from a binned orientation distribution function (ODF).
|
2020-09-28 16:44:23 +05:30
|
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
|
----------
|
2022-02-11 22:40:29 +05:30
|
|
|
|
weights : numpy.ndarray, shape (n)
|
2021-09-08 22:40:49 +05:30
|
|
|
|
Texture intensity values (probability density or volume fraction) at Euler space grid points.
|
2022-02-11 22:40:29 +05:30
|
|
|
|
phi : numpy.ndarray, shape (n,3)
|
2020-09-29 20:45:10 +05:30
|
|
|
|
Grid coordinates in Euler space at which weights are defined.
|
2022-11-14 19:39:45 +05:30
|
|
|
|
shape : (sequence of) int, optional
|
2022-03-19 15:25:03 +05:30
|
|
|
|
Shape of the returned array. Defaults to None, which gives a scalar.
|
2022-01-13 03:43:38 +05:30
|
|
|
|
degrees : bool, optional
|
2021-09-08 22:40:49 +05:30
|
|
|
|
Euler space grid coordinates are in degrees. Defaults to True.
|
2022-01-13 03:43:38 +05:30
|
|
|
|
fractions : bool, optional
|
2021-09-08 22:40:49 +05:30
|
|
|
|
ODF values correspond to volume fractions, not probability densities.
|
2020-09-28 16:44:23 +05:30
|
|
|
|
Defaults to True.
|
2020-11-15 17:36:26 +05:30
|
|
|
|
rng_seed: {None, int, array_like[ints], SeedSequence, BitGenerator, Generator}, optional
|
2021-09-08 22:40:49 +05:30
|
|
|
|
A seed to initialize the BitGenerator.
|
|
|
|
|
Defaults to None, i.e. unpredictable entropy will be pulled from the OS.
|
2020-09-28 16:44:23 +05:30
|
|
|
|
|
2022-11-19 13:40:00 +05:30
|
|
|
|
Returns
|
|
|
|
|
-------
|
|
|
|
|
new : damask.Rotation
|
|
|
|
|
|
2020-09-29 22:25:00 +05:30
|
|
|
|
Notes
|
|
|
|
|
-----
|
|
|
|
|
Due to the distortion of Euler space in the vicinity of ϕ = 0, probability densities, p, defined on
|
|
|
|
|
grid points with ϕ = 0 will never result in reconstructed orientations as their dV/V = p dγ = p × 0.
|
2022-02-11 22:40:29 +05:30
|
|
|
|
Hence, it is recommended to transform any such dataset to a cell-centered version, which avoids grid points at ϕ = 0.
|
2020-09-29 22:25:00 +05:30
|
|
|
|
|
|
|
|
|
References
|
|
|
|
|
----------
|
2021-03-18 20:06:40 +05:30
|
|
|
|
P. Eisenlohr and F. Roters, Computational Materials Science 42(4):670-678, 2008
|
2020-09-29 22:25:00 +05:30
|
|
|
|
https://doi.org/10.1016/j.commatsci.2007.09.015
|
|
|
|
|
|
2020-09-28 16:44:23 +05:30
|
|
|
|
"""
|
|
|
|
|
def _dg(eu,deg):
|
|
|
|
|
"""Return infinitesimal Euler space volume of bin(s)."""
|
2020-11-18 18:16:48 +05:30
|
|
|
|
phi_sorted = eu[np.lexsort((eu[:,0],eu[:,1],eu[:,2]))]
|
2020-12-07 22:19:37 +05:30
|
|
|
|
steps,size,_ = grid_filters.cellsSizeOrigin_coordinates0_point(phi_sorted)
|
2020-09-28 16:44:23 +05:30
|
|
|
|
delta = np.radians(size/steps) if deg else size/steps
|
2022-11-18 05:09:32 +05:30
|
|
|
|
return delta[0]*2.*np.sin(delta[1]/2.)*delta[2] / 8. / np.pi**2 * np.sin(np.radians(eu[:,1]) if deg else eu[:,1])
|
2020-09-28 16:44:23 +05:30
|
|
|
|
|
2022-11-18 05:09:32 +05:30
|
|
|
|
dg = 1. if fractions else _dg(phi,degrees)
|
|
|
|
|
dV_V = dg * np.maximum(0.,weights.squeeze())
|
2020-09-28 16:44:23 +05:30
|
|
|
|
|
2022-11-22 20:09:51 +05:30
|
|
|
|
N = 1 if shape is None else np.prod(shape).astype(int)
|
2022-03-18 13:36:48 +05:30
|
|
|
|
return Rotation.from_Euler_angles(phi[util.hybrid_IA(dV_V,N,rng_seed)],degrees).reshape(() if shape is None else shape)
|
2020-09-28 16:44:23 +05:30
|
|
|
|
|
|
|
|
|
|
2020-09-15 15:26:13 +05:30
|
|
|
|
@staticmethod
|
2022-02-02 15:41:59 +05:30
|
|
|
|
def from_spherical_component(center: 'Rotation',
|
2021-12-14 21:35:00 +05:30
|
|
|
|
sigma: float,
|
2022-11-23 02:56:15 +05:30
|
|
|
|
shape: Union[None, int, IntSequence] = None,
|
2022-04-28 19:05:50 +05:30
|
|
|
|
degrees: bool = False,
|
2022-11-23 02:56:15 +05:30
|
|
|
|
rng_seed: Optional[NumpyRngSeed] = None) -> 'Rotation':
|
2020-09-15 15:26:13 +05:30
|
|
|
|
"""
|
2022-03-19 15:25:03 +05:30
|
|
|
|
Initialize with samples from a Gaussian distribution around a given center.
|
2020-09-15 15:26:13 +05:30
|
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
|
----------
|
2022-03-19 15:25:03 +05:30
|
|
|
|
center : Rotation or Orientation
|
|
|
|
|
Central rotation.
|
2020-09-16 04:10:05 +05:30
|
|
|
|
sigma : float
|
|
|
|
|
Standard deviation of (Gaussian) misorientation distribution.
|
2022-11-14 19:39:45 +05:30
|
|
|
|
shape : (sequence of) int, optional
|
2022-03-19 15:25:03 +05:30
|
|
|
|
Shape of the returned array. Defaults to None, which gives a scalar.
|
2022-01-13 03:43:38 +05:30
|
|
|
|
degrees : bool, optional
|
2022-11-07 22:43:10 +05:30
|
|
|
|
sigma is given in degrees. Defaults to False.
|
2020-11-15 17:36:26 +05:30
|
|
|
|
rng_seed : {None, int, array_like[ints], SeedSequence, BitGenerator, Generator}, optional
|
2021-09-08 22:40:49 +05:30
|
|
|
|
A seed to initialize the BitGenerator.
|
|
|
|
|
Defaults to None, i.e. unpredictable entropy will be pulled from the OS.
|
2020-09-15 15:26:13 +05:30
|
|
|
|
|
2022-11-07 22:43:10 +05:30
|
|
|
|
Examples
|
|
|
|
|
--------
|
|
|
|
|
Create a brass texture consisting of
|
|
|
|
|
200 orientations:
|
|
|
|
|
|
|
|
|
|
>>> import damask
|
2022-11-18 05:09:32 +05:30
|
|
|
|
>>> center = damask.Rotation.from_Euler_angles([35.,45.,0.],degrees=True)
|
|
|
|
|
>>> brass = damask.Rotation.from_spherical_component(center=center,sigma=3.,shape=200,degrees=True)
|
2022-11-07 22:43:10 +05:30
|
|
|
|
|
|
|
|
|
Create a Goss texture consisting of
|
|
|
|
|
100 orientations:
|
|
|
|
|
|
|
|
|
|
>>> import damask
|
2022-11-18 05:09:32 +05:30
|
|
|
|
>>> center = damask.Rotation.from_Euler_angles([0.,45.,0.],degrees=True)
|
|
|
|
|
>>> goss = damask.Rotation.from_spherical_component(center=center,sigma=3.,shape=100,degrees=True)
|
2022-11-07 22:43:10 +05:30
|
|
|
|
|
2020-09-15 15:26:13 +05:30
|
|
|
|
"""
|
2020-11-15 17:36:26 +05:30
|
|
|
|
rng = np.random.default_rng(rng_seed)
|
2020-09-16 04:10:05 +05:30
|
|
|
|
sigma = np.radians(sigma) if degrees else sigma
|
2022-03-15 01:54:05 +05:30
|
|
|
|
N = 1 if shape is None else np.prod(shape)
|
2022-11-18 05:09:32 +05:30
|
|
|
|
u,Theta = (rng.random((N,2)) * 2. * np.array([1.,np.pi]) - np.array([1.,0.])).T
|
2020-09-16 04:42:30 +05:30
|
|
|
|
omega = abs(rng.normal(scale=sigma,size=N))
|
2022-11-18 05:09:32 +05:30
|
|
|
|
p = np.column_stack([np.sqrt(1.-u**2)*np.cos(Theta),
|
|
|
|
|
np.sqrt(1.-u**2)*np.sin(Theta),
|
2020-09-16 04:10:05 +05:30
|
|
|
|
u, omega])
|
2020-09-16 10:03:17 +05:30
|
|
|
|
|
2022-03-18 13:36:48 +05:30
|
|
|
|
return Rotation.from_axis_angle(p).reshape(() if shape is None else shape) * center
|
2020-09-15 15:26:13 +05:30
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@staticmethod
|
2022-04-27 18:29:55 +05:30
|
|
|
|
def from_fiber_component(crystal: IntSequence,
|
|
|
|
|
sample: IntSequence,
|
2022-11-18 05:09:32 +05:30
|
|
|
|
sigma: float = 0.,
|
2022-11-23 02:56:15 +05:30
|
|
|
|
shape: Union[None, int, IntSequence] = None,
|
2022-04-28 19:05:50 +05:30
|
|
|
|
degrees: bool = False,
|
2022-11-23 02:56:15 +05:30
|
|
|
|
rng_seed: Optional[NumpyRngSeed] = None) -> 'Rotation':
|
2020-09-15 15:26:13 +05:30
|
|
|
|
"""
|
2022-03-19 15:25:03 +05:30
|
|
|
|
Initialize with samples from a Gaussian distribution around a given direction.
|
2020-09-15 15:26:13 +05:30
|
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
|
----------
|
2022-04-27 18:29:55 +05:30
|
|
|
|
crystal : numpy.ndarray, shape (2)
|
2022-04-28 05:03:33 +05:30
|
|
|
|
Polar coordinates (polar angle θ from [0 0 1], azimuthal angle φ from [1 0 0])
|
|
|
|
|
of fiber direction in crystal frame.
|
2022-04-27 18:29:55 +05:30
|
|
|
|
sample : numpy.ndarray, shape (2)
|
2022-04-28 05:03:33 +05:30
|
|
|
|
Polar coordinates (polar angle θ from z, azimuthal angle φ from x)
|
|
|
|
|
of fiber direction in sample frame.
|
2020-09-16 03:44:15 +05:30
|
|
|
|
sigma : float, optional
|
|
|
|
|
Standard deviation of (Gaussian) misorientation distribution.
|
2020-09-16 02:04:19 +05:30
|
|
|
|
Defaults to 0.
|
2022-11-14 19:39:45 +05:30
|
|
|
|
shape : (sequence of) int, optional
|
2022-03-19 15:25:03 +05:30
|
|
|
|
Shape of the returned array. Defaults to None, which gives a scalar.
|
2022-01-13 03:43:38 +05:30
|
|
|
|
degrees : bool, optional
|
2022-11-07 22:43:10 +05:30
|
|
|
|
sigma and polar coordinates are given in degrees. Defaults to False.
|
2020-11-15 17:36:26 +05:30
|
|
|
|
rng_seed : {None, int, array_like[ints], SeedSequence, BitGenerator, Generator}, optional
|
2021-09-08 22:40:49 +05:30
|
|
|
|
A seed to initialize the BitGenerator.
|
|
|
|
|
Defaults to None, i.e. unpredictable entropy will be pulled from the OS.
|
2020-09-15 15:26:13 +05:30
|
|
|
|
|
2022-11-19 13:40:00 +05:30
|
|
|
|
Returns
|
|
|
|
|
-------
|
|
|
|
|
new : damask.Rotation
|
|
|
|
|
|
2022-04-28 05:03:33 +05:30
|
|
|
|
Notes
|
|
|
|
|
-----
|
2022-05-09 02:52:03 +05:30
|
|
|
|
The crystal direction for (θ=0,φ=0) is [0 0 1],
|
|
|
|
|
the sample direction for (θ=0,φ=0) is z.
|
2022-04-28 05:03:33 +05:30
|
|
|
|
|
2022-04-28 19:05:50 +05:30
|
|
|
|
Polar coordinates follow the ISO 80000-2:2019 convention
|
|
|
|
|
typically used in physics.
|
|
|
|
|
See https://en.wikipedia.org/wiki/Spherical_coordinate_system.
|
|
|
|
|
|
|
|
|
|
Ranges 0≤θ≤π and 0≤φ≤2π give a unique set of coordinates.
|
2022-04-28 05:03:33 +05:30
|
|
|
|
|
|
|
|
|
Examples
|
|
|
|
|
--------
|
2022-05-09 02:52:03 +05:30
|
|
|
|
Create an ideal α-fiber texture (<1 1 0> ǀǀ RD=x) consisting of
|
|
|
|
|
200 orientations:
|
|
|
|
|
|
|
|
|
|
>>> import damask
|
|
|
|
|
>>> import numpy as np
|
|
|
|
|
>>> alpha = damask.Rotation.from_fiber_component([np.pi/4.,0.],[np.pi/2.,0.],shape=200)
|
|
|
|
|
|
|
|
|
|
Create an ideal γ-fiber texture (<1 1 1> ǀǀ ND=z) consisting of
|
|
|
|
|
100 orientations:
|
|
|
|
|
|
|
|
|
|
>>> import damask
|
2022-11-18 05:09:32 +05:30
|
|
|
|
>>> gamma = damask.Rotation.from_fiber_component([54.7,45.],[0.,0.],shape=100,degrees=True)
|
2022-04-28 05:03:33 +05:30
|
|
|
|
|
2020-09-15 15:26:13 +05:30
|
|
|
|
"""
|
2020-11-15 17:36:26 +05:30
|
|
|
|
rng = np.random.default_rng(rng_seed)
|
2022-04-28 05:03:33 +05:30
|
|
|
|
sigma_,alpha,beta = (np.radians(coordinate) for coordinate in (sigma,crystal,sample)) if degrees else \
|
|
|
|
|
map(np.array, (sigma,crystal,sample))
|
2020-09-16 02:04:19 +05:30
|
|
|
|
|
2022-04-28 19:26:24 +05:30
|
|
|
|
d_cr = np.array([np.sin(alpha[0])*np.cos(alpha[1]), np.sin(alpha[0])*np.sin(alpha[1]), np.cos(alpha[0])])
|
|
|
|
|
d_lab = np.array([np.sin( beta[0])*np.cos( beta[1]), np.sin( beta[0])*np.sin( beta[1]), np.cos( beta[0])])
|
2020-09-16 02:04:19 +05:30
|
|
|
|
ax_align = np.append(np.cross(d_lab,d_cr), np.arccos(np.dot(d_lab,d_cr)))
|
2022-11-18 05:09:32 +05:30
|
|
|
|
if np.isclose(ax_align[3],0.): ax_align[:3] = np.array([1.,0.,0.])
|
|
|
|
|
R_align = Rotation.from_axis_angle(ax_align if ax_align[3] > 0. else -ax_align,normalize=True) # rotate fiber axis from sample to crystal frame
|
2020-09-16 02:04:19 +05:30
|
|
|
|
|
2022-11-22 20:09:51 +05:30
|
|
|
|
N = 1 if shape is None else np.prod(shape).astype(int)
|
2022-11-18 05:09:32 +05:30
|
|
|
|
u,Theta = (rng.random((N,2)) * 2. * np.array([1.,np.pi]) - np.array([1.,0.])).T
|
2020-09-16 04:42:30 +05:30
|
|
|
|
omega = abs(rng.normal(scale=sigma_,size=N))
|
2022-11-22 20:09:51 +05:30
|
|
|
|
p = np.column_stack([np.sqrt(1.-u**2)*np.cos(Theta),
|
|
|
|
|
np.sqrt(1.-u**2)*np.sin(Theta),
|
2020-09-16 04:10:05 +05:30
|
|
|
|
u, omega])
|
2020-09-16 10:03:17 +05:30
|
|
|
|
p[:,:3] = np.einsum('ij,...j',np.eye(3)-np.outer(d_lab,d_lab),p[:,:3]) # remove component along fiber axis
|
|
|
|
|
f = np.column_stack((np.broadcast_to(d_lab,(N,3)),rng.random(N)*np.pi))
|
2022-11-18 05:09:32 +05:30
|
|
|
|
f[::2,:3] *= -1. # flip half the rotation axes to negative sense
|
2020-09-16 10:03:17 +05:30
|
|
|
|
|
2022-03-15 01:54:05 +05:30
|
|
|
|
return (R_align.broadcast_to(N)
|
|
|
|
|
* Rotation.from_axis_angle(p,normalize=True)
|
2022-03-18 13:36:48 +05:30
|
|
|
|
* Rotation.from_axis_angle(f)).reshape(() if shape is None else shape)
|
2020-09-15 15:26:13 +05:30
|
|
|
|
|
|
|
|
|
|
2020-02-21 03:46:35 +05:30
|
|
|
|
####################################################################################################
|
|
|
|
|
# Code below available according to the following conditions on https://github.com/MarDiehl/3Drotations
|
|
|
|
|
####################################################################################################
|
2020-06-21 13:34:45 +05:30
|
|
|
|
# Copyright (c) 2017-2020, Martin Diehl/Max-Planck-Institut für Eisenforschung GmbH
|
2020-02-21 03:46:35 +05:30
|
|
|
|
# Copyright (c) 2013-2014, Marc De Graef/Carnegie Mellon University
|
|
|
|
|
# All rights reserved.
|
|
|
|
|
#
|
|
|
|
|
# Redistribution and use in source and binary forms, with or without modification, are
|
|
|
|
|
# permitted provided that the following conditions are met:
|
|
|
|
|
#
|
|
|
|
|
# - Redistributions of source code must retain the above copyright notice, this list
|
|
|
|
|
# of conditions and the following disclaimer.
|
|
|
|
|
# - Redistributions in binary form must reproduce the above copyright notice, this
|
|
|
|
|
# list of conditions and the following disclaimer in the documentation and/or
|
|
|
|
|
# other materials provided with the distribution.
|
|
|
|
|
# - Neither the names of Marc De Graef, Carnegie Mellon University nor the names
|
|
|
|
|
# of its contributors may be used to endorse or promote products derived from
|
|
|
|
|
# this software without specific prior written permission.
|
|
|
|
|
#
|
|
|
|
|
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
|
|
|
# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
|
|
|
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
|
|
|
# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
|
|
|
|
|
# LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
|
|
|
# DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
|
|
|
|
# SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
|
|
|
|
# CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
|
|
|
|
# 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.
|
|
|
|
|
####################################################################################################
|
2020-02-21 15:15:14 +05:30
|
|
|
|
#---------- Quaternion ----------
|
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _qu2om(qu: np.ndarray) -> np.ndarray:
|
2020-05-17 01:26:30 +05:30
|
|
|
|
qq = qu[...,0:1]**2-(qu[...,1:2]**2 + qu[...,2:3]**2 + qu[...,3:4]**2)
|
2022-11-18 05:09:32 +05:30
|
|
|
|
om = np.block([qq + 2.*qu[...,1:2]**2,
|
|
|
|
|
2.*(qu[...,2:3]*qu[...,1:2]-_P*qu[...,0:1]*qu[...,3:4]),
|
|
|
|
|
2.*(qu[...,3:4]*qu[...,1:2]+_P*qu[...,0:1]*qu[...,2:3]),
|
|
|
|
|
2.*(qu[...,1:2]*qu[...,2:3]+_P*qu[...,0:1]*qu[...,3:4]),
|
|
|
|
|
qq + 2.*qu[...,2:3]**2,
|
|
|
|
|
2.*(qu[...,3:4]*qu[...,2:3]-_P*qu[...,0:1]*qu[...,1:2]),
|
|
|
|
|
2.*(qu[...,1:2]*qu[...,3:4]-_P*qu[...,0:1]*qu[...,2:3]),
|
|
|
|
|
2.*(qu[...,2:3]*qu[...,3:4]+_P*qu[...,0:1]*qu[...,1:2]),
|
|
|
|
|
qq + 2.*qu[...,3:4]**2,
|
2020-05-17 01:26:30 +05:30
|
|
|
|
]).reshape(qu.shape[:-1]+(3,3))
|
2020-05-18 19:24:58 +05:30
|
|
|
|
return om
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-02-21 15:15:14 +05:30
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _qu2eu(qu: np.ndarray) -> np.ndarray:
|
2023-09-01 00:57:22 +05:30
|
|
|
|
"""
|
|
|
|
|
Quaternion to Bunge Euler angles.
|
|
|
|
|
|
|
|
|
|
References
|
|
|
|
|
----------
|
2023-09-01 16:35:28 +05:30
|
|
|
|
E. Bernardes and S. Viollet, PLoS ONE 17(11):e0276302, 2022
|
2023-09-01 00:57:22 +05:30
|
|
|
|
https://doi.org/10.1371/journal.pone.0276302
|
|
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
a = qu[...,0:1]
|
|
|
|
|
b = -_P*qu[...,3:4]
|
|
|
|
|
c = -_P*qu[...,1:2]
|
|
|
|
|
d = -_P*qu[...,2:3]
|
|
|
|
|
|
|
|
|
|
eu = np.block([
|
|
|
|
|
np.arctan2(b,a),
|
|
|
|
|
np.arccos(2*(a**2+b**2)/(a**2+b**2+c**2+d**2)-1),
|
|
|
|
|
np.arctan2(-d,c),
|
|
|
|
|
])
|
|
|
|
|
|
|
|
|
|
eu_sum = eu[...,0] + eu[...,2]
|
|
|
|
|
eu_diff = eu[...,0] - eu[...,2]
|
|
|
|
|
|
|
|
|
|
is_zero = np.isclose(eu[...,1],0.0)
|
|
|
|
|
is_pi = np.isclose(eu[...,1],np.pi)
|
|
|
|
|
is_ok = ~np.logical_or(is_zero,is_pi)
|
|
|
|
|
|
|
|
|
|
eu[...,0][is_zero] = 2*eu[...,0][is_zero]
|
|
|
|
|
eu[...,0][is_pi] = -2*eu[...,2][is_pi]
|
|
|
|
|
eu[...,2][~is_ok] = 0.0
|
|
|
|
|
eu[...,0][is_ok] = eu_diff[is_ok]
|
|
|
|
|
eu[...,2][is_ok] = eu_sum [is_ok]
|
|
|
|
|
|
|
|
|
|
eu[np.logical_or(np.abs(eu) < 1.e-6,
|
|
|
|
|
np.abs(eu-2*np.pi) < 1.e-6)] = 0.
|
2022-11-18 05:09:32 +05:30
|
|
|
|
return np.where(eu < 0., eu%(np.pi*np.array([2.,1.,2.])),eu)
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-02-21 15:15:14 +05:30
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _qu2ax(qu: np.ndarray) -> np.ndarray:
|
2020-02-21 15:23:44 +05:30
|
|
|
|
"""
|
2021-09-08 22:40:49 +05:30
|
|
|
|
Quaternion to axis–angle pair.
|
2020-02-21 15:15:14 +05:30
|
|
|
|
|
2021-09-08 22:40:49 +05:30
|
|
|
|
Modified version of the original formulation, should be numerically more stable.
|
2020-02-21 15:23:44 +05:30
|
|
|
|
"""
|
2020-05-17 01:26:30 +05:30
|
|
|
|
with np.errstate(invalid='ignore',divide='ignore'):
|
|
|
|
|
s = np.sign(qu[...,0:1])/np.sqrt(qu[...,1:2]**2+qu[...,2:3]**2+qu[...,3:4]**2)
|
2022-11-18 05:09:32 +05:30
|
|
|
|
omega = 2. * np.arccos(np.clip(qu[...,0:1],-1.,1.))
|
|
|
|
|
ax = np.where(np.broadcast_to(qu[...,0:1] < 1.e-8,qu.shape),
|
2020-06-19 15:55:46 +05:30
|
|
|
|
np.block([qu[...,1:4],np.broadcast_to(np.pi,qu[...,0:1].shape)]),
|
2020-05-17 01:26:30 +05:30
|
|
|
|
np.block([qu[...,1:4]*s,omega]))
|
2022-11-18 05:09:32 +05:30
|
|
|
|
ax[np.isclose(qu[...,0],1.,rtol=0.)] = np.array([0.,0.,1.,0.])
|
2020-04-09 02:41:48 +05:30
|
|
|
|
return ax
|
|
|
|
|
|
2020-02-21 15:15:14 +05:30
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _qu2ro(qu: np.ndarray) -> np.ndarray:
|
2021-09-08 22:40:49 +05:30
|
|
|
|
"""Quaternion to Rodrigues–Frank vector."""
|
2020-05-17 01:26:30 +05:30
|
|
|
|
with np.errstate(invalid='ignore',divide='ignore'):
|
|
|
|
|
s = np.linalg.norm(qu[...,1:4],axis=-1,keepdims=True)
|
2022-11-18 05:09:32 +05:30
|
|
|
|
ro = np.where(np.broadcast_to(np.abs(qu[...,0:1]) < 1.e-12,qu.shape),
|
2020-06-19 15:55:46 +05:30
|
|
|
|
np.block([qu[...,1:2], qu[...,2:3], qu[...,3:4], np.broadcast_to(np.inf,qu[...,0:1].shape)]),
|
2020-05-17 01:26:30 +05:30
|
|
|
|
np.block([qu[...,1:2]/s,qu[...,2:3]/s,qu[...,3:4]/s,
|
2022-11-18 05:09:32 +05:30
|
|
|
|
np.tan(np.arccos(np.clip(qu[...,0:1],-1.,1.)))
|
2020-05-17 01:26:30 +05:30
|
|
|
|
])
|
|
|
|
|
)
|
2022-11-18 05:09:32 +05:30
|
|
|
|
ro[np.abs(s).squeeze(-1) < 1.e-12] = np.array([0.,0.,_P,0.])
|
2020-04-09 02:41:48 +05:30
|
|
|
|
return ro
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-02-21 15:15:14 +05:30
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _qu2ho(qu: np.ndarray) -> np.ndarray:
|
2020-02-21 15:23:44 +05:30
|
|
|
|
"""Quaternion to homochoric vector."""
|
2020-05-17 01:26:30 +05:30
|
|
|
|
with np.errstate(invalid='ignore'):
|
2022-11-18 05:09:32 +05:30
|
|
|
|
omega = 2. * np.arccos(np.clip(qu[...,0:1],-1.,1.))
|
|
|
|
|
ho = np.where(np.abs(omega) < 1.e-12,
|
2020-05-17 01:26:30 +05:30
|
|
|
|
np.zeros(3),
|
2022-11-18 05:09:32 +05:30
|
|
|
|
qu[...,1:4]/np.linalg.norm(qu[...,1:4],axis=-1,keepdims=True)
|
2020-05-17 01:26:30 +05:30
|
|
|
|
* (0.75*(omega - np.sin(omega)))**(1./3.))
|
2020-02-21 15:23:44 +05:30
|
|
|
|
return ho
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-02-21 15:15:14 +05:30
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _qu2cu(qu: np.ndarray) -> np.ndarray:
|
2020-02-21 15:23:44 +05:30
|
|
|
|
"""Quaternion to cubochoric vector."""
|
2020-05-20 14:41:07 +05:30
|
|
|
|
return Rotation._ho2cu(Rotation._qu2ho(qu))
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
|
|
|
|
|
2020-02-21 15:15:14 +05:30
|
|
|
|
#---------- Rotation matrix ----------
|
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _om2qu(om: np.ndarray) -> np.ndarray:
|
2020-02-21 15:23:44 +05:30
|
|
|
|
"""
|
|
|
|
|
Rotation matrix to quaternion.
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-05-18 22:11:48 +05:30
|
|
|
|
This formulation is from www.euclideanspace.com/maths/geometry/rotations/conversions/matrixToQuaternion.
|
|
|
|
|
The original formulation had issues.
|
2020-02-21 15:23:44 +05:30
|
|
|
|
"""
|
2022-11-18 05:09:32 +05:30
|
|
|
|
trace = om[...,0,0:1] + om[...,1,1:2] + om[...,2,2:3]
|
2020-05-19 11:05:58 +05:30
|
|
|
|
|
|
|
|
|
with np.errstate(invalid='ignore',divide='ignore'):
|
2022-11-18 05:09:32 +05:30
|
|
|
|
s = np.array([
|
|
|
|
|
0.5 / np.sqrt( 1. + trace),
|
|
|
|
|
2. * np.sqrt( 1. + om[...,0,0:1] - om[...,1,1:2] - om[...,2,2:3]),
|
|
|
|
|
2. * np.sqrt( 1. + om[...,1,1:2] - om[...,2,2:3] - om[...,0,0:1]),
|
|
|
|
|
2. * np.sqrt( 1. + om[...,2,2:3] - om[...,0,0:1] - om[...,1,1:2] )
|
|
|
|
|
])
|
|
|
|
|
qu = np.where(trace>0,
|
|
|
|
|
np.block([0.25 / s[0],
|
|
|
|
|
(om[...,2,1:2] - om[...,1,2:3] ) * s[0],
|
|
|
|
|
(om[...,0,2:3] - om[...,2,0:1] ) * s[0],
|
|
|
|
|
(om[...,1,0:1] - om[...,0,1:2] ) * s[0]]),
|
|
|
|
|
np.where(om[...,0,0:1] > np.maximum(om[...,1,1:2],om[...,2,2:3]),
|
|
|
|
|
np.block([(om[...,2,1:2] - om[...,1,2:3]) / s[1],
|
|
|
|
|
0.25 * s[1],
|
|
|
|
|
(om[...,0,1:2] + om[...,1,0:1]) / s[1],
|
|
|
|
|
(om[...,0,2:3] + om[...,2,0:1]) / s[1]]),
|
|
|
|
|
np.where(om[...,1,1:2] > om[...,2,2:3],
|
|
|
|
|
np.block([(om[...,0,2:3] - om[...,2,0:1]) / s[2],
|
|
|
|
|
(om[...,0,1:2] + om[...,1,0:1]) / s[2],
|
|
|
|
|
0.25 * s[2],
|
|
|
|
|
(om[...,1,2:3] + om[...,2,1:2]) / s[2]]),
|
|
|
|
|
np.block([(om[...,1,0:1] - om[...,0,1:2]) / s[3],
|
|
|
|
|
(om[...,0,2:3] + om[...,2,0:1]) / s[3],
|
|
|
|
|
(om[...,1,2:3] + om[...,2,1:2]) / s[3],
|
|
|
|
|
0.25 * s[3]]),
|
|
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
)*np.array([1.,_P,_P,_P])
|
|
|
|
|
qu[qu[...,0] < 0.] *= -1.
|
2020-05-19 11:05:58 +05:30
|
|
|
|
return qu
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-02-21 15:15:14 +05:30
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _om2eu(om: np.ndarray) -> np.ndarray:
|
2021-09-09 19:35:09 +05:30
|
|
|
|
"""Rotation matrix to Bunge Euler angles."""
|
2020-05-17 01:26:30 +05:30
|
|
|
|
with np.errstate(invalid='ignore',divide='ignore'):
|
2022-11-18 05:09:32 +05:30
|
|
|
|
zeta = 1./np.sqrt(1.-om[...,2,2:3]**2)
|
|
|
|
|
eu = np.where(np.isclose(np.abs(om[...,2,2:3]),1.,0.),
|
2020-05-17 01:26:30 +05:30
|
|
|
|
np.block([np.arctan2(om[...,0,1:2],om[...,0,0:1]),
|
2022-11-22 20:09:51 +05:30
|
|
|
|
np.pi*0.5*(1.-om[...,2,2:3]),
|
2020-05-17 01:26:30 +05:30
|
|
|
|
np.zeros(om.shape[:-2]+(1,)),
|
|
|
|
|
]),
|
|
|
|
|
np.block([np.arctan2(om[...,2,0:1]*zeta,-om[...,2,1:2]*zeta),
|
2020-05-18 19:24:58 +05:30
|
|
|
|
np.arccos( om[...,2,2:3]),
|
2020-05-17 01:26:30 +05:30
|
|
|
|
np.arctan2(om[...,0,2:3]*zeta,+om[...,1,2:3]*zeta)
|
|
|
|
|
])
|
|
|
|
|
)
|
2022-11-18 05:09:32 +05:30
|
|
|
|
eu[np.abs(eu) < 1.e-8] = 0.0
|
|
|
|
|
return np.where(eu < 0., eu%(np.pi*np.array([2.,1.,2.])),eu)
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-02-21 15:15:14 +05:30
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _om2ax(om: np.ndarray) -> np.ndarray:
|
2021-09-08 22:40:49 +05:30
|
|
|
|
"""Rotation matrix to axis–angle pair."""
|
2020-05-17 01:26:30 +05:30
|
|
|
|
diag_delta = -_P*np.block([om[...,1,2:3]-om[...,2,1:2],
|
2020-05-17 11:31:34 +05:30
|
|
|
|
om[...,2,0:1]-om[...,0,2:3],
|
|
|
|
|
om[...,0,1:2]-om[...,1,0:1]
|
2020-05-17 01:26:30 +05:30
|
|
|
|
])
|
2022-11-18 05:09:32 +05:30
|
|
|
|
t = 0.5*(om.trace(axis2=-2,axis1=-1) -1.).reshape(om.shape[:-2]+(1,))
|
2020-05-17 01:26:30 +05:30
|
|
|
|
w,vr = np.linalg.eig(om)
|
|
|
|
|
# mask duplicated real eigenvalues
|
2022-11-18 05:09:32 +05:30
|
|
|
|
w[np.isclose(w[...,0],1.+0.j),1:] = 0.
|
|
|
|
|
w[np.isclose(w[...,1],1.+0.j),2:] = 0.
|
2020-05-17 01:26:30 +05:30
|
|
|
|
vr = np.swapaxes(vr,-1,-2)
|
2022-11-18 05:09:32 +05:30
|
|
|
|
ax = np.where(np.abs(diag_delta)<1.e-13,
|
|
|
|
|
np.real(vr[np.isclose(w,1.+0.j)]).reshape(om.shape[:-2]+(3,)),
|
|
|
|
|
np.abs(np.real(vr[np.isclose(w,1.+0.j)]).reshape(om.shape[:-2]+(3,)))
|
2020-05-17 01:26:30 +05:30
|
|
|
|
*np.sign(diag_delta))
|
2022-11-18 05:09:32 +05:30
|
|
|
|
ax = np.block([ax,np.arccos(np.clip(t,-1.,1.))])
|
|
|
|
|
ax[np.abs(ax[...,3]) < 1.e-8] = np.array([0.,0.,1.,0.])
|
2020-04-09 17:50:43 +05:30
|
|
|
|
return ax
|
|
|
|
|
|
2020-02-21 15:15:14 +05:30
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _om2ro(om: np.ndarray) -> np.ndarray:
|
2021-09-08 22:40:49 +05:30
|
|
|
|
"""Rotation matrix to Rodrigues–Frank vector."""
|
2020-05-20 14:41:07 +05:30
|
|
|
|
return Rotation._eu2ro(Rotation._om2eu(om))
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-02-21 15:15:14 +05:30
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _om2ho(om: np.ndarray) -> np.ndarray:
|
2020-02-21 15:23:44 +05:30
|
|
|
|
"""Rotation matrix to homochoric vector."""
|
2020-05-20 14:41:07 +05:30
|
|
|
|
return Rotation._ax2ho(Rotation._om2ax(om))
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-02-21 15:15:14 +05:30
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _om2cu(om: np.ndarray) -> np.ndarray:
|
2020-02-21 15:23:44 +05:30
|
|
|
|
"""Rotation matrix to cubochoric vector."""
|
2020-05-20 14:41:07 +05:30
|
|
|
|
return Rotation._ho2cu(Rotation._om2ho(om))
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
|
|
|
|
|
2021-09-09 19:35:09 +05:30
|
|
|
|
#---------- Bunge Euler angles ----------
|
2020-02-21 15:15:14 +05:30
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _eu2qu(eu: np.ndarray) -> np.ndarray:
|
2021-09-09 19:35:09 +05:30
|
|
|
|
"""Bunge Euler angles to quaternion."""
|
2020-05-17 01:26:30 +05:30
|
|
|
|
ee = 0.5*eu
|
|
|
|
|
cPhi = np.cos(ee[...,1:2])
|
|
|
|
|
sPhi = np.sin(ee[...,1:2])
|
2020-05-18 23:46:50 +05:30
|
|
|
|
qu = np.block([ cPhi*np.cos(ee[...,0:1]+ee[...,2:3]),
|
|
|
|
|
-_P*sPhi*np.cos(ee[...,0:1]-ee[...,2:3]),
|
|
|
|
|
-_P*sPhi*np.sin(ee[...,0:1]-ee[...,2:3]),
|
|
|
|
|
-_P*cPhi*np.sin(ee[...,0:1]+ee[...,2:3])])
|
2022-11-18 05:09:32 +05:30
|
|
|
|
qu[qu[...,0] < 0.] *= -1.
|
2020-02-21 15:23:44 +05:30
|
|
|
|
return qu
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-02-21 15:15:14 +05:30
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _eu2om(eu: np.ndarray) -> np.ndarray:
|
2021-09-09 19:35:09 +05:30
|
|
|
|
"""Bunge Euler angles to rotation matrix."""
|
2020-05-17 01:26:30 +05:30
|
|
|
|
c = np.cos(eu)
|
|
|
|
|
s = np.sin(eu)
|
|
|
|
|
om = np.block([+c[...,0:1]*c[...,2:3]-s[...,0:1]*s[...,2:3]*c[...,1:2],
|
|
|
|
|
+s[...,0:1]*c[...,2:3]+c[...,0:1]*s[...,2:3]*c[...,1:2],
|
|
|
|
|
+s[...,2:3]*s[...,1:2],
|
|
|
|
|
-c[...,0:1]*s[...,2:3]-s[...,0:1]*c[...,2:3]*c[...,1:2],
|
|
|
|
|
-s[...,0:1]*s[...,2:3]+c[...,0:1]*c[...,2:3]*c[...,1:2],
|
|
|
|
|
+c[...,2:3]*s[...,1:2],
|
|
|
|
|
+s[...,0:1]*s[...,1:2],
|
|
|
|
|
-c[...,0:1]*s[...,1:2],
|
|
|
|
|
+c[...,1:2]
|
|
|
|
|
]).reshape(eu.shape[:-1]+(3,3))
|
2022-11-18 05:09:32 +05:30
|
|
|
|
om[np.abs(om) < 1.e-12] = 0.
|
2020-02-21 15:23:44 +05:30
|
|
|
|
return om
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-02-21 15:15:14 +05:30
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _eu2ax(eu: np.ndarray) -> np.ndarray:
|
2021-09-09 19:35:09 +05:30
|
|
|
|
"""Bunge Euler angles to axis–angle pair."""
|
2020-05-17 01:26:30 +05:30
|
|
|
|
t = np.tan(eu[...,1:2]*0.5)
|
|
|
|
|
sigma = 0.5*(eu[...,0:1]+eu[...,2:3])
|
|
|
|
|
delta = 0.5*(eu[...,0:1]-eu[...,2:3])
|
|
|
|
|
tau = np.linalg.norm(np.block([t,np.sin(sigma)]),axis=-1,keepdims=True)
|
2022-11-18 05:09:32 +05:30
|
|
|
|
alpha = np.where(np.abs(np.cos(sigma))<1.e-12,np.pi,2.*np.arctan(tau/np.cos(sigma)))
|
2020-05-17 01:26:30 +05:30
|
|
|
|
with np.errstate(invalid='ignore',divide='ignore'):
|
2022-11-18 05:09:32 +05:30
|
|
|
|
ax = np.where(np.broadcast_to(np.abs(alpha)<1.e-12,eu.shape[:-1]+(4,)),
|
|
|
|
|
[0.,0.,1.,0.],
|
2020-05-17 01:26:30 +05:30
|
|
|
|
np.block([-_P/tau*t*np.cos(delta),
|
|
|
|
|
-_P/tau*t*np.sin(delta),
|
|
|
|
|
-_P/tau* np.sin(sigma),
|
|
|
|
|
alpha
|
|
|
|
|
]))
|
2022-11-18 05:09:32 +05:30
|
|
|
|
ax[(alpha<0.).squeeze()] *= -1.
|
2020-02-21 15:23:44 +05:30
|
|
|
|
return ax
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-02-21 15:15:14 +05:30
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _eu2ro(eu: np.ndarray) -> np.ndarray:
|
2021-09-09 19:35:09 +05:30
|
|
|
|
"""Bunge Euler angles to Rodrigues–Frank vector."""
|
2020-05-20 14:41:07 +05:30
|
|
|
|
ax = Rotation._eu2ax(eu)
|
2020-05-17 01:26:30 +05:30
|
|
|
|
ro = np.block([ax[...,:3],np.tan(ax[...,3:4]*.5)])
|
2022-11-18 05:09:32 +05:30
|
|
|
|
ro[ax[...,3] >= np.pi,3] = np.inf
|
|
|
|
|
ro[np.abs(ax[...,3])<1.e-16] = np.array([0.,0.,_P,0.])
|
2020-02-21 15:23:44 +05:30
|
|
|
|
return ro
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-02-21 15:15:14 +05:30
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _eu2ho(eu: np.ndarray) -> np.ndarray:
|
2021-09-09 19:35:09 +05:30
|
|
|
|
"""Bunge Euler angles to homochoric vector."""
|
2020-05-20 14:41:07 +05:30
|
|
|
|
return Rotation._ax2ho(Rotation._eu2ax(eu))
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-02-21 15:15:14 +05:30
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _eu2cu(eu: np.ndarray) -> np.ndarray:
|
2021-09-09 19:35:09 +05:30
|
|
|
|
"""Bunge Euler angles to cubochoric vector."""
|
2020-05-20 14:41:07 +05:30
|
|
|
|
return Rotation._ho2cu(Rotation._eu2ho(eu))
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
|
|
|
|
|
2020-02-21 15:15:14 +05:30
|
|
|
|
#---------- Axis angle pair ----------
|
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _ax2qu(ax: np.ndarray) -> np.ndarray:
|
2021-09-08 22:40:49 +05:30
|
|
|
|
"""Axis–angle pair to quaternion."""
|
2020-05-17 01:26:30 +05:30
|
|
|
|
c = np.cos(ax[...,3:4]*.5)
|
|
|
|
|
s = np.sin(ax[...,3:4]*.5)
|
2022-11-18 05:09:32 +05:30
|
|
|
|
qu = np.where(np.abs(ax[...,3:4]) < 1.e-6,[1.,0.,0.,0.],np.block([c,ax[...,:3]*s]))
|
2020-04-21 15:29:42 +05:30
|
|
|
|
return qu
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-02-21 15:15:14 +05:30
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _ax2om(ax: np.ndarray) -> np.ndarray:
|
2021-09-08 22:40:49 +05:30
|
|
|
|
"""Axis-angle pair to rotation matrix."""
|
2020-05-17 01:26:30 +05:30
|
|
|
|
c = np.cos(ax[...,3:4])
|
|
|
|
|
s = np.sin(ax[...,3:4])
|
2022-11-18 05:09:32 +05:30
|
|
|
|
omc = 1.-c
|
2020-05-17 01:26:30 +05:30
|
|
|
|
om = np.block([c+omc*ax[...,0:1]**2,
|
2022-11-18 05:09:32 +05:30
|
|
|
|
omc*ax[...,0:1]*ax[...,1:2] + s*ax[...,2:3],
|
|
|
|
|
omc*ax[...,0:1]*ax[...,2:3] - s*ax[...,1:2],
|
|
|
|
|
omc*ax[...,0:1]*ax[...,1:2] - s*ax[...,2:3],
|
2020-05-17 01:26:30 +05:30
|
|
|
|
c+omc*ax[...,1:2]**2,
|
2022-11-18 05:09:32 +05:30
|
|
|
|
omc*ax[...,1:2]*ax[...,2:3] + s*ax[...,0:1],
|
|
|
|
|
omc*ax[...,0:1]*ax[...,2:3] + s*ax[...,1:2],
|
|
|
|
|
omc*ax[...,1:2]*ax[...,2:3] - s*ax[...,0:1],
|
2020-05-17 01:26:30 +05:30
|
|
|
|
c+omc*ax[...,2:3]**2]).reshape(ax.shape[:-1]+(3,3))
|
2022-11-18 05:09:32 +05:30
|
|
|
|
return om if _P < 0. else np.swapaxes(om,-1,-2)
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-02-21 15:15:14 +05:30
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _ax2eu(ax: np.ndarray) -> np.ndarray:
|
2021-09-09 19:35:09 +05:30
|
|
|
|
"""Rotation matrix to Bunge Euler angles."""
|
2020-05-20 14:41:07 +05:30
|
|
|
|
return Rotation._om2eu(Rotation._ax2om(ax))
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-02-21 15:15:14 +05:30
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _ax2ro(ax: np.ndarray) -> np.ndarray:
|
2021-09-08 22:40:49 +05:30
|
|
|
|
"""Axis–angle pair to Rodrigues–Frank vector."""
|
2020-05-17 01:26:30 +05:30
|
|
|
|
ro = np.block([ax[...,:3],
|
|
|
|
|
np.where(np.isclose(ax[...,3:4],np.pi,atol=1.e-15,rtol=.0),
|
|
|
|
|
np.inf,
|
|
|
|
|
np.tan(ax[...,3:4]*0.5))
|
|
|
|
|
])
|
2022-11-18 05:09:32 +05:30
|
|
|
|
ro[np.abs(ax[...,3]) < 1.e-6] = np.array([.0,.0,_P,.0])
|
2020-04-21 15:29:42 +05:30
|
|
|
|
return ro
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-02-21 15:15:14 +05:30
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _ax2ho(ax: np.ndarray) -> np.ndarray:
|
2021-09-08 22:40:49 +05:30
|
|
|
|
"""Axis–angle pair to homochoric vector."""
|
2022-11-18 05:09:32 +05:30
|
|
|
|
f = (0.75 * ( ax[...,3:4] - np.sin(ax[...,3:4]) ))**(1./3.)
|
|
|
|
|
return ax[...,:3] * f
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-02-21 15:15:14 +05:30
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _ax2cu(ax: np.ndarray) -> np.ndarray:
|
2021-09-08 22:40:49 +05:30
|
|
|
|
"""Axis–angle pair to cubochoric vector."""
|
2020-05-20 14:41:07 +05:30
|
|
|
|
return Rotation._ho2cu(Rotation._ax2ho(ax))
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
|
|
|
|
|
2020-02-21 15:15:14 +05:30
|
|
|
|
#---------- Rodrigues-Frank vector ----------
|
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _ro2qu(ro: np.ndarray) -> np.ndarray:
|
2021-09-08 22:40:49 +05:30
|
|
|
|
"""Rodrigues–Frank vector to quaternion."""
|
2020-05-20 14:41:07 +05:30
|
|
|
|
return Rotation._ax2qu(Rotation._ro2ax(ro))
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-02-21 15:15:14 +05:30
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _ro2om(ro: np.ndarray) -> np.ndarray:
|
2021-09-08 22:40:49 +05:30
|
|
|
|
"""Rodgrigues–Frank vector to rotation matrix."""
|
2020-05-20 14:41:07 +05:30
|
|
|
|
return Rotation._ax2om(Rotation._ro2ax(ro))
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-02-21 15:15:14 +05:30
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _ro2eu(ro: np.ndarray) -> np.ndarray:
|
2021-09-09 19:35:09 +05:30
|
|
|
|
"""Rodrigues–Frank vector to Bunge Euler angles."""
|
2020-05-20 14:41:07 +05:30
|
|
|
|
return Rotation._om2eu(Rotation._ro2om(ro))
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-02-21 15:15:14 +05:30
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _ro2ax(ro: np.ndarray) -> np.ndarray:
|
2021-09-08 22:40:49 +05:30
|
|
|
|
"""Rodrigues–Frank vector to axis–angle pair."""
|
2020-05-17 01:26:30 +05:30
|
|
|
|
with np.errstate(invalid='ignore',divide='ignore'):
|
|
|
|
|
ax = np.where(np.isfinite(ro[...,3:4]),
|
|
|
|
|
np.block([ro[...,0:3]*np.linalg.norm(ro[...,0:3],axis=-1,keepdims=True),2.*np.arctan(ro[...,3:4])]),
|
|
|
|
|
np.block([ro[...,0:3],np.broadcast_to(np.pi,ro[...,3:4].shape)]))
|
2022-11-18 05:09:32 +05:30
|
|
|
|
ax[np.abs(ro[...,3]) < 1.e-8] = np.array([0.,0.,1.,0.])
|
2020-04-11 17:27:05 +05:30
|
|
|
|
return ax
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-02-21 15:15:14 +05:30
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _ro2ho(ro: np.ndarray) -> np.ndarray:
|
2021-09-08 22:40:49 +05:30
|
|
|
|
"""Rodrigues–Frank vector to homochoric vector."""
|
2022-11-18 05:09:32 +05:30
|
|
|
|
f = np.where(np.isfinite(ro[...,3:4]),2.*np.arctan(ro[...,3:4]) -np.sin(2.*np.arctan(ro[...,3:4])),np.pi)
|
|
|
|
|
return np.where(np.broadcast_to(np.sum(ro[...,0:3]**2,axis=-1,keepdims=True) < 1.e-8,ro[...,0:3].shape),
|
|
|
|
|
np.zeros(3), ro[...,0:3]* (0.75*f)**(1./3.))
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-02-21 15:15:14 +05:30
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _ro2cu(ro: np.ndarray) -> np.ndarray:
|
2021-09-08 22:40:49 +05:30
|
|
|
|
"""Rodrigues–Frank vector to cubochoric vector."""
|
2020-05-20 14:41:07 +05:30
|
|
|
|
return Rotation._ho2cu(Rotation._ro2ho(ro))
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
|
|
|
|
|
2020-02-21 15:15:14 +05:30
|
|
|
|
#---------- Homochoric vector----------
|
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _ho2qu(ho: np.ndarray) -> np.ndarray:
|
2020-02-21 15:23:44 +05:30
|
|
|
|
"""Homochoric vector to quaternion."""
|
2020-05-20 14:41:07 +05:30
|
|
|
|
return Rotation._ax2qu(Rotation._ho2ax(ho))
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-02-21 15:15:14 +05:30
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _ho2om(ho: np.ndarray) -> np.ndarray:
|
2020-02-21 15:23:44 +05:30
|
|
|
|
"""Homochoric vector to rotation matrix."""
|
2020-05-20 14:41:07 +05:30
|
|
|
|
return Rotation._ax2om(Rotation._ho2ax(ho))
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-02-21 15:15:14 +05:30
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _ho2eu(ho: np.ndarray) -> np.ndarray:
|
2021-09-09 19:35:09 +05:30
|
|
|
|
"""Homochoric vector to Bunge Euler angles."""
|
2020-05-20 14:41:07 +05:30
|
|
|
|
return Rotation._ax2eu(Rotation._ho2ax(ho))
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-02-21 15:15:14 +05:30
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _ho2ax(ho: np.ndarray) -> np.ndarray:
|
2021-09-08 22:40:49 +05:30
|
|
|
|
"""Homochoric vector to axis–angle pair."""
|
2022-06-10 02:46:02 +05:30
|
|
|
|
tfit = np.array([+0.9999999999999968, -0.49999999999986866, -0.025000000000632055,
|
|
|
|
|
-0.003928571496460683, -0.0008164666077062752, -0.00019411896443261646,
|
|
|
|
|
-0.00004985822229871769, -0.000014164962366386031, -1.9000248160936107e-6,
|
|
|
|
|
-5.72184549898506e-6, +7.772149920658778e-6, -0.00001053483452909705,
|
|
|
|
|
+9.528014229335313e-6, -5.660288876265125e-6, +1.2844901692764126e-6,
|
|
|
|
|
+1.1255185726258763e-6, -1.3834391419956455e-6, +7.513691751164847e-7,
|
|
|
|
|
-2.401996891720091e-7, +4.386887017466388e-8, -3.5917775353564864e-9])
|
2022-11-18 05:09:32 +05:30
|
|
|
|
hmag_squared = np.sum(ho**2,axis=-1,keepdims=True)
|
2022-02-17 22:46:53 +05:30
|
|
|
|
s = np.sum(tfit*hmag_squared**np.arange(len(tfit)),axis=-1,keepdims=True)
|
2020-05-17 01:26:30 +05:30
|
|
|
|
with np.errstate(invalid='ignore'):
|
2022-11-18 05:09:32 +05:30
|
|
|
|
return np.where(np.broadcast_to(np.abs(hmag_squared)<1.e-8,ho.shape[:-1]+(4,)),
|
|
|
|
|
[0.,0.,1.,0.],
|
|
|
|
|
np.block([ho/np.sqrt(hmag_squared),2.*np.arccos(np.clip(s,-1.,1.))]))
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-02-21 15:15:14 +05:30
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _ho2ro(ho: np.ndarray) -> np.ndarray:
|
2021-09-08 22:40:49 +05:30
|
|
|
|
"""Axis–angle pair to Rodrigues–Frank vector."""
|
2020-05-20 14:41:07 +05:30
|
|
|
|
return Rotation._ax2ro(Rotation._ho2ax(ho))
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-02-21 15:15:14 +05:30
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _ho2cu(ho: np.ndarray) -> np.ndarray:
|
2020-04-22 16:53:33 +05:30
|
|
|
|
"""
|
|
|
|
|
Homochoric vector to cubochoric vector.
|
|
|
|
|
|
|
|
|
|
References
|
|
|
|
|
----------
|
|
|
|
|
D. Roşca et al., Modelling and Simulation in Materials Science and Engineering 22:075013, 2014
|
|
|
|
|
https://doi.org/10.1088/0965-0393/22/7/075013
|
|
|
|
|
|
|
|
|
|
"""
|
2020-05-17 01:26:30 +05:30
|
|
|
|
rs = np.linalg.norm(ho,axis=-1,keepdims=True)
|
|
|
|
|
|
|
|
|
|
xyz3 = np.take_along_axis(ho,Rotation._get_pyramid_order(ho,'forward'),-1)
|
|
|
|
|
|
|
|
|
|
with np.errstate(invalid='ignore',divide='ignore'):
|
|
|
|
|
# inverse M_3
|
2022-11-18 05:09:32 +05:30
|
|
|
|
xyz2 = xyz3[...,0:2] * np.sqrt( 2.*rs/(rs+np.abs(xyz3[...,2:3])) )
|
2020-05-17 01:26:30 +05:30
|
|
|
|
qxy = np.sum(xyz2**2,axis=-1,keepdims=True)
|
|
|
|
|
|
|
|
|
|
q2 = qxy + np.max(np.abs(xyz2),axis=-1,keepdims=True)**2
|
|
|
|
|
sq2 = np.sqrt(q2)
|
2022-11-18 05:09:32 +05:30
|
|
|
|
q = (_beta/np.sqrt(2.)/_R1) * np.sqrt(q2*qxy/(q2-np.max(np.abs(xyz2),axis=-1,keepdims=True)*sq2))
|
2020-05-17 01:26:30 +05:30
|
|
|
|
tt = np.clip((np.min(np.abs(xyz2),axis=-1,keepdims=True)**2\
|
2022-11-18 05:09:32 +05:30
|
|
|
|
+np.max(np.abs(xyz2),axis=-1,keepdims=True)*sq2)/np.sqrt(2.)/qxy,-1.,1.)
|
2020-05-17 01:26:30 +05:30
|
|
|
|
T_inv = np.where(np.abs(xyz2[...,1:2]) <= np.abs(xyz2[...,0:1]),
|
2022-11-18 05:09:32 +05:30
|
|
|
|
np.block([np.ones_like(tt),np.arccos(tt)/np.pi*12.]),
|
|
|
|
|
np.block([np.arccos(tt)/np.pi*12.,np.ones_like(tt)]))*q
|
|
|
|
|
T_inv[xyz2<0.] *= -1.
|
|
|
|
|
T_inv[np.broadcast_to(np.isclose(qxy,0.,rtol=0.,atol=1.e-12),T_inv.shape)] = 0.
|
|
|
|
|
cu = np.block([T_inv, np.where(xyz3[...,2:3]<0.,-np.ones_like(xyz3[...,2:3]),np.ones_like(xyz3[...,2:3])) \
|
|
|
|
|
* rs/np.sqrt(6./np.pi),
|
2020-05-17 01:26:30 +05:30
|
|
|
|
])/ _sc
|
|
|
|
|
|
2022-11-18 05:09:32 +05:30
|
|
|
|
cu[np.isclose(np.sum(np.abs(ho),axis=-1),0.,rtol=0.,atol=1.e-16)] = 0.
|
|
|
|
|
return np.take_along_axis(cu,Rotation._get_pyramid_order(ho,'backward'),-1)
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-02-21 15:15:14 +05:30
|
|
|
|
#---------- Cubochoric ----------
|
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _cu2qu(cu: np.ndarray) -> np.ndarray:
|
2020-02-21 15:23:44 +05:30
|
|
|
|
"""Cubochoric vector to quaternion."""
|
2020-05-20 14:41:07 +05:30
|
|
|
|
return Rotation._ho2qu(Rotation._cu2ho(cu))
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-02-21 15:15:14 +05:30
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _cu2om(cu: np.ndarray) -> np.ndarray:
|
2020-02-21 15:23:44 +05:30
|
|
|
|
"""Cubochoric vector to rotation matrix."""
|
2020-05-20 14:41:07 +05:30
|
|
|
|
return Rotation._ho2om(Rotation._cu2ho(cu))
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-02-21 15:15:14 +05:30
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _cu2eu(cu: np.ndarray) -> np.ndarray:
|
2021-09-09 19:35:09 +05:30
|
|
|
|
"""Cubochoric vector to Bunge Euler angles."""
|
2020-05-20 14:41:07 +05:30
|
|
|
|
return Rotation._ho2eu(Rotation._cu2ho(cu))
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-02-21 15:15:14 +05:30
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _cu2ax(cu: np.ndarray) -> np.ndarray:
|
2021-09-08 22:40:49 +05:30
|
|
|
|
"""Cubochoric vector to axis–angle pair."""
|
2020-05-20 14:41:07 +05:30
|
|
|
|
return Rotation._ho2ax(Rotation._cu2ho(cu))
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-02-21 15:15:14 +05:30
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _cu2ro(cu: np.ndarray) -> np.ndarray:
|
2021-09-08 22:40:49 +05:30
|
|
|
|
"""Cubochoric vector to Rodrigues–Frank vector."""
|
2020-05-20 14:41:07 +05:30
|
|
|
|
return Rotation._ho2ro(Rotation._cu2ho(cu))
|
2020-02-21 03:46:35 +05:30
|
|
|
|
|
2020-02-21 15:15:14 +05:30
|
|
|
|
@staticmethod
|
2021-12-14 21:35:00 +05:30
|
|
|
|
def _cu2ho(cu: np.ndarray) -> np.ndarray:
|
2020-04-22 16:53:33 +05:30
|
|
|
|
"""
|
|
|
|
|
Cubochoric vector to homochoric vector.
|
|
|
|
|
|
|
|
|
|
References
|
|
|
|
|
----------
|
|
|
|
|
D. Roşca et al., Modelling and Simulation in Materials Science and Engineering 22:075013, 2014
|
|
|
|
|
https://doi.org/10.1088/0965-0393/22/7/075013
|
|
|
|
|
|
|
|
|
|
"""
|
2020-05-17 01:26:30 +05:30
|
|
|
|
with np.errstate(invalid='ignore',divide='ignore'):
|
2022-06-10 02:08:13 +05:30
|
|
|
|
# get pyramid and scale by grid parameter ratio
|
2020-05-17 01:26:30 +05:30
|
|
|
|
XYZ = np.take_along_axis(cu,Rotation._get_pyramid_order(cu,'forward'),-1) * _sc
|
|
|
|
|
order = np.abs(XYZ[...,1:2]) <= np.abs(XYZ[...,0:1])
|
2022-11-18 05:09:32 +05:30
|
|
|
|
q = np.pi/12. * np.where(order,XYZ[...,1:2],XYZ[...,0:1]) \
|
2020-05-17 01:26:30 +05:30
|
|
|
|
/ np.where(order,XYZ[...,0:1],XYZ[...,1:2])
|
|
|
|
|
c = np.cos(q)
|
|
|
|
|
s = np.sin(q)
|
2022-11-18 05:09:32 +05:30
|
|
|
|
q = _R1*2.**0.25/_beta/ np.sqrt(np.sqrt(2.)-c) \
|
2020-05-17 01:26:30 +05:30
|
|
|
|
* np.where(order,XYZ[...,0:1],XYZ[...,1:2])
|
|
|
|
|
|
2022-11-18 05:09:32 +05:30
|
|
|
|
T = np.block([(np.sqrt(2.)*c - 1.), np.sqrt(2.) * s]) * q
|
2020-05-17 01:26:30 +05:30
|
|
|
|
|
|
|
|
|
# transform to sphere grid (inverse Lambert)
|
|
|
|
|
c = np.sum(T**2,axis=-1,keepdims=True)
|
2022-11-18 05:09:32 +05:30
|
|
|
|
s = c * np.pi/24. /XYZ[...,2:3]**2
|
|
|
|
|
c = c * np.sqrt(np.pi/24.)/XYZ[...,2:3]
|
|
|
|
|
q = np.sqrt( 1. - s)
|
2020-05-17 01:26:30 +05:30
|
|
|
|
|
2022-11-18 05:09:32 +05:30
|
|
|
|
ho = np.where(np.isclose(np.sum(np.abs(XYZ[...,0:2]),axis=-1,keepdims=True),0.,rtol=0.,atol=1.e-16),
|
|
|
|
|
np.block([np.zeros_like(XYZ[...,0:2]),np.sqrt(6./np.pi)*XYZ[...,2:3]]),
|
2020-05-17 01:26:30 +05:30
|
|
|
|
np.block([np.where(order,T[...,0:1],T[...,1:2])*q,
|
|
|
|
|
np.where(order,T[...,1:2],T[...,0:1])*q,
|
2022-11-18 05:09:32 +05:30
|
|
|
|
np.sqrt(6./np.pi) * XYZ[...,2:3] - c])
|
2020-05-17 01:26:30 +05:30
|
|
|
|
)
|
|
|
|
|
|
2022-11-18 05:09:32 +05:30
|
|
|
|
ho[np.isclose(np.sum(np.abs(cu),axis=-1),0.,rtol=0.,atol=1.e-16)] = 0.
|
|
|
|
|
return np.take_along_axis(ho,Rotation._get_pyramid_order(cu,'backward'),-1)
|
2020-05-03 22:21:30 +05:30
|
|
|
|
|
2020-04-22 16:53:33 +05:30
|
|
|
|
|
2020-04-29 21:22:09 +05:30
|
|
|
|
@staticmethod
|
2022-01-26 19:39:09 +05:30
|
|
|
|
def _get_pyramid_order(xyz: np.ndarray,
|
|
|
|
|
direction: Literal['forward', 'backward']) -> np.ndarray:
|
2020-04-29 21:22:09 +05:30
|
|
|
|
"""
|
|
|
|
|
Get order of the coordinates.
|
2020-04-22 16:53:33 +05:30
|
|
|
|
|
2020-04-29 21:22:09 +05:30
|
|
|
|
Depending on the pyramid in which the point is located, the order need to be adjusted.
|
2020-04-22 16:53:33 +05:30
|
|
|
|
|
2020-04-29 21:22:09 +05:30
|
|
|
|
Parameters
|
|
|
|
|
----------
|
|
|
|
|
xyz : numpy.ndarray
|
2021-09-08 22:40:49 +05:30
|
|
|
|
Coordinates of a point on a uniform refinable grid on a ball or
|
2020-04-29 21:22:09 +05:30
|
|
|
|
in a uniform refinable cubical grid.
|
2020-04-22 16:53:33 +05:30
|
|
|
|
|
2020-04-29 21:22:09 +05:30
|
|
|
|
References
|
|
|
|
|
----------
|
|
|
|
|
D. Roşca et al., Modelling and Simulation in Materials Science and Engineering 22:075013, 2014
|
|
|
|
|
https://doi.org/10.1088/0965-0393/22/7/075013
|
2020-04-22 16:53:33 +05:30
|
|
|
|
|
2020-04-29 21:22:09 +05:30
|
|
|
|
"""
|
2020-05-17 01:26:30 +05:30
|
|
|
|
order = {'forward': np.array([[0,1,2],[1,2,0],[2,0,1]]),
|
2020-05-03 22:21:30 +05:30
|
|
|
|
'backward':np.array([[0,1,2],[2,0,1],[1,2,0]])}
|
2020-05-17 01:26:30 +05:30
|
|
|
|
|
|
|
|
|
p = np.where(np.maximum(np.abs(xyz[...,0]),np.abs(xyz[...,1])) <= np.abs(xyz[...,2]),0,
|
|
|
|
|
np.where(np.maximum(np.abs(xyz[...,1]),np.abs(xyz[...,2])) <= np.abs(xyz[...,0]),1,2))
|
2020-04-29 21:22:09 +05:30
|
|
|
|
|
|
|
|
|
return order[direction][p]
|