File: example_gallery_plot_galactocentric_frame.rst

package info (click to toggle)
astropy 7.2.0-1
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 34,156 kB
  • sloc: python: 240,839; ansic: 55,852; lex: 8,621; sh: 3,318; xml: 2,399; makefile: 149
file content (174 lines) | stat: -rw-r--r-- 8,386 bytes parent folder | download
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
.. _sphx_glr_generated_examples_coordinates_plot_galactocentric-frame.py:

Transforming positions and velocities to and from a Galactocentric frame
========================================================================

..
  EXAMPLE START
  Transforming positions and velocities to and from a Galactocentric frames

This example shows a few examples of how to use and customize the
`~astropy.coordinates.Galactocentric` frame to transform Heliocentric sky
positions, distance, proper motions, and radial velocities to a Galactocentric,
Cartesian frame, and the same in reverse.

The main configurable parameters of the `~astropy.coordinates.Galactocentric`
frame control the position and velocity of the solar system barycenter within
the Galaxy. These are specified by setting the ICRS coordinates of the
Galactic center, the distance to the Galactic center (the sun-galactic center
line is always assumed to be the x-axis of the Galactocentric frame), and the
Cartesian 3-velocity of the sun in the Galactocentric frame. We will first
demonstrate how to customize these values, then show how to set the solar motion
instead by inputting the proper motion of Sgr A*.

Note that, for brevity, we may refer to the solar system barycenter as just "the
sun" in the examples below.

Let's first define a barycentric coordinate and velocity in the ICRS frame.
We will use the data for the star HD 39881 from the
`Simbad <https://simbad.unistra.fr/simbad/>`_ database:


>>> import astropy.coordinates as coord
>>> from astropy import units as u
>>> c1 = coord.SkyCoord(
...     ra=89.014303 * u.degree,
...     dec=13.924912 * u.degree,
...     distance=(37.59 * u.mas).to(u.pc, u.parallax()),
...     pm_ra_cosdec=372.72 * (u.mas / u.yr),
...     pm_dec=-483.69 * (u.mas / u.yr),
...     radial_velocity=0.37 * (u.km / u.s),
...     frame="icrs",
... )

This is a high proper-motion star; suppose we'd like to transform its position
and velocity to a Galactocentric frame to see if it has a large 3D velocity
as well. To use the Astropy default solar position and motion parameters, we
can do the following:

>>> gc1 = c1.transform_to(coord.Galactocentric)

From here, we can access the components of the resulting
`~astropy.coordinates.Galactocentric` instance to see the 3D Cartesian
velocity components:

>>> print(gc1.v_x, gc1.v_y, gc1.v_z)  # doctest: +FLOAT_CMP
30.254684717897074 km / s 171.29916086104885 km / s 18.19390627095307 km / s

The default parameters for the `~astropy.coordinates.Galactocentric` frame
are detailed in the linked documentation, but we can modify the most commonly
changed values using the keywords ``galcen_distance``, ``galcen_v_sun``, and
``z_sun`` which set the sun-Galactic center distance, the 3D velocity vector
of the sun, and the height of the sun above the Galactic midplane,
respectively. The velocity of the sun can be specified as an
`~astropy.units.Quantity` object with velocity units and is interpreted as a
Cartesian velocity, as in the example below. Note that, as with the positions,
the Galactocentric frame is a right-handed system (i.e., the Sun is at negative
x values) so ``v_x`` is opposite of the Galactocentric radial velocity:

>>> v_sun = [11.1, 244, 7.25] * (u.km / u.s)  # [vx, vy, vz]
>>> gc_frame = coord.Galactocentric(
...     galcen_distance=8 * u.kpc, galcen_v_sun=v_sun, z_sun=0 * u.pc
... )

We can then transform to this frame instead, with our custom parameters:

>>> gc2 = c1.transform_to(gc_frame)
>>> print(gc2.v_x, gc2.v_y, gc2.v_z)  # doctest: +FLOAT_CMP
28.427958360720748 km / s 169.69916086104888 km / s 17.70831652451455 km / s

It is sometimes useful to specify the solar motion using the
`proper motion of Sgr A* <https://arxiv.org/abs/astro-ph/0408107>`_
instead of Cartesian velocity components. With an assumed distance, we can convert
proper motion components to Cartesian velocity components using `astropy.units`:

>>> galcen_distance = 8 * u.kpc
>>> pm_gal_sgrA = [-6.379, -0.202] * (u.mas / u.yr)  # from Reid & Brunthaler 2004
>>> vy, vz = -(galcen_distance * pm_gal_sgrA).to(u.km / u.s, u.dimensionless_angles())

We still have to assume a line-of-sight velocity for the Galactic center,
which we will again take to be 11 km/s:

>>> vx = 11.1 * (u.km / u.s)
>>> v_sun2 = u.Quantity([vx, vy, vz])  # List of Quantity -> a single Quantity
>>> gc_frame2 = coord.Galactocentric(
...     galcen_distance=galcen_distance, galcen_v_sun=v_sun2, z_sun=0 * u.pc
... )
>>> gc3 = c1.transform_to(gc_frame2)
>>> print(gc3.v_x, gc3.v_y, gc3.v_z)  # doctest: +FLOAT_CMP
28.427958360720748 km / s 167.61484955608267 km / s 18.118916793584443 km / s

The transformations also work in the opposite direction. This can be useful
for transforming simulated or theoretical data to observable quantities. As
an example, we will generate 4 theoretical circular orbits at different
Galactocentric radii with the same circular velocity, and transform them to
Heliocentric coordinates:

.. plot::
   :include-source:

    >>> import matplotlib.pyplot as plt
    >>> import numpy as np
    >>> import astropy.coordinates as coord
    >>> from astropy import units as u
    >>> ring_distances = np.arange(10, 26, 5) * u.kpc
    >>> circ_velocity = 220 * (u.km / u.s)
    >>> phi_grid = np.linspace(90, 270, 512) * u.degree  # grid of azimuths
    >>> ring_rep = coord.CylindricalRepresentation(
    ...     rho=ring_distances[:, np.newaxis],
    ...     phi=phi_grid[np.newaxis],
    ...     z=np.zeros_like(ring_distances)[:, np.newaxis],
    ... )
    >>> angular_velocity = (-circ_velocity / ring_distances).to(
    ...     u.mas / u.yr, u.dimensionless_angles()
    ... )
    >>> ring_dif = coord.CylindricalDifferential(
    ...     d_rho=np.zeros(phi_grid.shape)[np.newaxis] * (u.km / u.s),
    ...     d_phi=angular_velocity[:, np.newaxis],
    ...     d_z=np.zeros(phi_grid.shape)[np.newaxis] * (u.km / u.s),
    ... )
    >>> ring_rep = ring_rep.with_differentials(ring_dif)
    >>> gc_rings = coord.SkyCoord(ring_rep, frame=coord.Galactocentric)

    First, let's visualize the geometry in Galactocentric coordinates. Here are
    the positions and velocities of the rings; note that in the velocity plot,
    the velocities of the 4 rings are identical and thus overlaid under the same
    curve:

    >>> fig, axes = plt.subplots(1, 2, figsize=(12, 6))
    >>> axes[0].plot(gc_rings.x.T, gc_rings.y.T, marker="None", linewidth=3)  # doctest: +IGNORE_OUTPUT
    >>> axes[0].text(-8.0, 0, r"$\odot$", fontsize=20)  # doctest: +IGNORE_OUTPUT
    >>> axes[0].set_xlim(-30, 30)  # doctest: +IGNORE_OUTPUT
    >>> axes[0].set_ylim(-30, 30)  # doctest: +IGNORE_OUTPUT
    >>> axes[0].set_xlabel("$x$ [kpc]")  # doctest: +IGNORE_OUTPUT
    >>> axes[0].set_ylabel("$y$ [kpc]")  # doctest: +IGNORE_OUTPUT
    >>> axes[0].set_title("Positions")  # doctest: +IGNORE_OUTPUT
    >>> axes[1].plot(gc_rings.v_x.T, gc_rings.v_y.T, marker="None", linewidth=3)  # doctest: +IGNORE_OUTPUT
    >>> axes[1].set_xlim(-250, 250)  # doctest: +IGNORE_OUTPUT
    >>> axes[1].set_ylim(-250, 250)  # doctest: +IGNORE_OUTPUT
    >>> axes[1].set_xlabel(f"$v_x$ [{(u.km / u.s).to_string('latex_inline')}]")  # doctest: +IGNORE_OUTPUT
    >>> axes[1].set_ylabel(f"$v_y$ [{(u.km / u.s).to_string('latex_inline')}]")  # doctest: +IGNORE_OUTPUT
    >>> axes[1].set_title("Velocities")  # doctest: +IGNORE_OUTPUT
    >>> fig.tight_layout()

    Now we can transform to Galactic coordinates and visualize the rings in
    observable coordinates:

    >>> gal_rings = gc_rings.transform_to(coord.Galactic)
    >>> fig, ax = plt.subplots(1, 1, figsize=(8, 6))
    >>> for i in range(len(ring_distances)):
    ...     ax.plot(
    ...         gal_rings[i].l.degree,
    ...         gal_rings[i].pm_l_cosb.value,
    ...         label=str(ring_distances[i]),
    ...         marker="None",
    ...         linewidth=3,
    ...     )  # doctest: +IGNORE_OUTPUT
    >>> ax.set_xlim(360, 0)  # doctest: +IGNORE_OUTPUT
    >>> ax.set_xlabel("$l$ [deg]")  # doctest: +IGNORE_OUTPUT
    >>> ax.set_ylabel(rf'$\mu_l \, \cos b$ [{(u.mas/u.yr).to_string("latex_inline")}]')  # doctest: +IGNORE_OUTPUT
    >>> ax.legend()  # doctest: +IGNORE_OUTPUT
    >>> plt.draw()

..
  EXAMPLE END