Specifying rotations or origin shifts in Potential
classes#
Most of the gravitational potential classes implemented in gala
support
shifting the origin of the potential relative to the coordinate system, and
specifying a rotation of the potential relative to the coordinate system.
By default, the origin is assumed to be at (0,0,0) or (0,0), and there is no
rotation assumed.
For the examples below the following imports have already been executed:
>>> import astropy.units as u
>>> import numpy as np
>>> import gala.potential as gp
>>> from gala.units import galactic, solarsystem
Origin shifts#
For potential classes that support these options, origin shifts are specified by
passing in a Quantity
to set the origin of the potential in the
given coordinate system. For example, if we are working with two
KeplerPotential
objects, and we want them to be offset from
one another such that one potential is at (1, 0, 0)
AU and the other is at
(-2, 0, 0)
AU, we would define the two objects as:
>>> p1 = gp.KeplerPotential(m=1*u.Msun, origin=[1, 0, 0]*u.au,
... units=solarsystem)
>>> p2 = gp.KeplerPotential(m=0.5*u.Msun, origin=[-2, 0, 0]*u.au,
... units=solarsystem)
To see that these are shifted from the coordinate system origin, let’s combine
these two objects into a CCompositePotential
and
visualize the potential:
>>> pot = gp.CCompositePotential(p1=p1, p2=p2)
>>> fig, ax = plt.subplots(1, 1, figsize=(5, 5))
>>> grid = np.linspace(-5, 5, 100)
>>> p.plot_contours(grid=(grid, grid, 0.), ax=ax)
>>> ax.set_xlabel("$x$ [kpc]")
>>> ax.set_ylabel("$y$ [kpc]")
(Source code
, png
, pdf
)
Rotations#
Rotations can be specified either by passing in a
scipy.spatial.transform.Rotation
instance, or by passing in a 2D numpy
array
specifying a rotation matrix. For example, let’s see what happens if we rotate a
bar potential using these two possible inputs. First, we’ll define a rotation
matrix specifying a 30 degree rotation around the z axis (i.e.
counter-clockwise) using astropy.coordinates.matrix_utilities.rotation_matrix
.
Next, we’ll define a rotation using a scipy
Rotation
object:
>>> from astropy.coordinates.matrix_utilities import rotation_matrix
>>> from scipy.spatial.transform import Rotation
>>> R_arr = rotation_matrix(30*u.deg, 'z')
>>> R_scipy = Rotation.from_euler('z', 30, degrees=True)
Warning
Note that astropy and scipy have different rotation conventions, so even though both of the above look like identical 30 degree rotations around the z axis, they result in different (i.e. transposed or inverse) rotation matrices:
>>> R_arr
array([[ 0.8660254, 0.5 , 0. ],
[-0.5 , 0.8660254, 0. ],
[ 0. , 0. , 1. ]])
>>> R_scipy.as_matrix()
array([[ 0.8660254, -0.5 , 0. ],
[ 0.5 , 0.8660254, 0. ],
[ 0. , 0. , 1. ]])
Let’s see what happens to the bar potential when we specify these rotations:
>>> bar1 = gp.LongMuraliBarPotential(m=1e10, a=3.5, b=0.5, c=0.5,
... units=galactic)
>>> bar2 = gp.LongMuraliBarPotential(m=1e10, a=3.5, b=0.5, c=0.5,
... units=galactic, R=R_arr)
>>> bar3 = gp.LongMuraliBarPotential(m=1e10, a=3.5, b=0.5, c=0.5,
... units=galactic, R=R_scipy)
(Source code
, png
, pdf
)