forked from astropy/astropy-api
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathcoordinates_example.py
422 lines (322 loc) · 14.3 KB
/
coordinates_example.py
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
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
#!/usr/bin/python
# -*- coding: utf-8 -*-
from __future__ import print_function
from astropy.coordinates import Angle
from astropy.units import Units as u
'''
The emphasis of this package is that common tasks should
be performed in as simple and intuitive manner as possible.
Complex or unusual tasks are still possible, but should not
interfere with the simplicity of common tasks.
One design principle is that objects should silently accept parameters
of any data type that can unambiguously be determined. Where data types
are accepted, they should be specified as constants, not as strings.
If an unknown unit is specified as a string, the code certainly
cannot handle it.
A note on units: Units (as with everything else in Python) are objects.
Any instance of a "unit" keyword will only accept "Unit" objects, not
bare strings. All commonly used units will be predefined in astropy.units
(which is still under development). The exact syntax might change,
but the ideas will not. Units may be specified as part of a string in
an initializer (subscribing to the "accept and parse values that are
unambiguous to a human reader" philosophy), but will not be accepted as
a bare string anywhere else.
On arrays: The interface outlined here only accepts scalar angles (and hence
coordinates) - we will likely later expand it to allow arrays to be stored in
these objects, but only after the scalar interfaces is complete and working.
'''
# ============
# Base Objects
# ============
'''
The "angle" is a fundamental object. The internal representation
is stored in radians, but this is transparent to the user. Units
*must* be specified rather than a default value be assumed. This is
as much for self-documenting code as anything else.
Angle objects simply represent a single angular coordinate. More specific
angular coordinates (e.g. RA, Dec) are subclasses of Angle.
'''
# Creating Angles
# ---------------
angle = Angle(54.12412, unit=u.degree)
angle = Angle("54.12412", unit=u.degree)
angle = Angle("54:07:26.832", unit=u.degree)
angle = Angle("54.12412 deg")
angle = Angle("54.12412 degrees")
angle = Angle("54.12412°") # because we like Unicode
angle = Angle((54,7,26.832), unit=unit=u.degree)
# (deg,min,sec) *tuples* are acceptable, but lists/arrays are *not* because of the need to eventually support arrays of coordinates
angle = Angle(3.60827466667, unit=u.hour)
angle = Angle("3:36:29.7888000120", unit=u.hour)
angle = Angle((3,36,29.7888000120), unit=u.hour) #as above, *must* be a tuple
angle = Angle(0.944644098745, unit=u.radian)
angle = Angle(54.12412)
#raises an exception because this is ambiguous
# Angle operations
'''
Angles can be added and subtracted. Multiplication and division by
a scalar is also permitted. A negative operator is also valid.
All of these operate in a single dimension. Attempting to
multiply or divide two Angle objects will raise an exception.
'''
a1 = Angle(3.60827466667, unit=u.hour)
a2 = Angle("54:07:26.832", unit=u.degree)
a3 = a1 + a2 # creates new Angle object
a4 = a1 - a2
a5 = -a1
a6 = a1 / a2 # raises an exception
a6 = a1 * a2 # raises an exception
a7 = Angle(a1) # makes a *copy* of the object, but identical content as a1
# Bounds
# ------
'''
By default the Angle object can accept any value, but will return
values in [-360,360] (retaining the sign that was specified).
One can also set artificial bounds for custom applications and range checking.
As Angle objects are intended to be immutable (the angle value anyway), this provides a bound check
upon creation. The units given in the "bounds" keyword must match the units
specified in the scalar.
'''
a1 = Angle(13343, unit=u.degree)
print(a1.degrees)
# 23
a2 = Angle(-50, unit=u.degree)
print(a2.degrees)
# -50
a3 = Angle(-361, unit=u.degree)
print (a3.degrees)
# -1
# custom bounds
a4 = Angle(66, unit=u.degree, bounds=(-45,45))
# RangeError
a5 = Angle(390, unit=u.degree, bounds=(-75,75))
print(a5.degrees)
# 30, no RangeError because while 390>75, 30 is within the bounds
a6 = Angle(390, unit=u.degree, bounds=(-720, 720))
print(a6.degrees)
# 390
a7 = Angle(1020, unit=u.degree, bounds=None)
print(a7.degrees)
# 1020
# bounds and operations
a8 = a5 + a6
# ValueError - the bounds don't match
a9 = a5 + a5
print(a9.bounds)
# (-75, 75) - if the bounds match, there is no error and the bound is kept
print(a9.degrees)
# 60
#To get the default bounds back, you need to create a new object with the equivalent angle
a9 = Angle(a5.degrees + a5.degrees, unit=u.degree)
a10 = Angle(a5.degrees + a6.degrees, unit=u.degree, bounds=[-180,180])
print(a10.degrees)
# 60 - if they don't match and you want to combine, just re-assign the bounds yourself
a11 = a7 - a7
print(a11.degrees)
# 0 - bounds of None can also be operated on without complaint
# Converting units
# ----------------
angle = Angle("54.12412", unit=u.degree)
print("Angle in hours: {0}.".format(angle.hours))
# Angle in hours: 3.60827466667.
print("Angle in radians: {0}.".format(angle.radians))
# Angle in radians: 0.944644098745.
print("Angle in degrees: {0}.".format(angle.degrees))
# Angle in degrees: 54.12412.
print("Angle in HMS: {0}".format(angle.hms)) # returns a tuple, e.g. (12, 21, 2.343)
# Angle in HMS: (3, 36, 29.78879999999947)
print("Angle in DMS: {0}".format(angle.dms)) # returns a tuple, e.g. (12, 21, 2.343)
# Angle in DMS: (54, 7, 26.831999999992036)
# String formatting
# -----------------
'''
The string method of Angle has this signature:
def string(self, unit=DEGREE, decimal=False, sep=" ", precision=5, pad=False):
The "decimal" parameter defaults to False since if you need to print the
Angle as a decimal, there's no need to use the "to_string" method (see above).
'''
print("Angle as HMS: {0}".format(angle.to_string(unit=u.hour)))
# Angle as HMS: 3 36 29.78880
print("Angle as HMS: {0}".format(angle.to_string(unit=u.hour, sep=":")))
# Angle as HMS: 3:36:29.78880
print("Angle as HMS: {0}".format(angle.to_string(unit=u.hour, sep=":", precision=2)))
# Angle as HMS: 3:36:29.79
# Note that you can provide one, two, or three separators passed as a tuple or list
#
print("Angle as HMS: {0}".format(angle.string(unit=u.hour, sep=("h","m","s"), precision=4)))
# Angle as HMS: 3h36m29.7888s
print("Angle as HMS: {0}".format(angle.string(unit=u.hour, sep=["-","|"], precision=4)))
# Angle as HMS: 3-36|29.7888
print("Angle as HMS: {0}".format(angle.string(unit=u.hour, sep="-", precision=4)))
# Angle as HMS: 3-36-29.7888
# The "pad" parameter will add leading zeros to numbers less than 10.
#
print("Angle as HMS: {0}".format(angle.string(unit=u.hour, precision=4, pad=True)))
# Angle as HMS: 03 36 29.7888
# RA/Dec Objects
# --------------
from astropy.coordinates import RA, Dec
'''
RA and Dec are objects that are subclassed from Angle. As with Angle, RA and Dec can
parse any unambiguous format (tuples, formatted strings, etc.).
The intention is not to create an Angle subclass for every possible coordinate object
(e.g. galactic l, galactic b). However, equatorial RA/dec are so prevalent in astronomy
that it's worth creating ones for these units. They will be noted as "special" in the
docs and use of the just the Angle class is to be used for other coordinate systems.
'''
ra = RA("4:08:15.162342") # error - hours or degrees?
ra = RA("26:34:65.345634") # unambiguous
# Units can be specified
ra = RA("4:08:15.162342", unit=u.hour)
# Where RA values are commonly found in hours or degrees, declination is nearly always
# specified in degrees, so this is the default.
dec = Dec("-41:08:15.162342")
dec = Dec("-41:08:15.162342", unit=u.degree) # same as above
# The Dec object will have bounds hard-coded at [-90,90] degrees.
# Coordinates
# -----------
'''
A coordinate marks a position on the sky. This is an object that contains two Angle
objects. There are a wide array of coordinate systems that should be implemented, and
it will be easy to subclass to create custom user-made coordinates with conversions to
standard coordinates.
'''
from astropy.coordinates import ICRSCoordinate, GalacticCoordinate, HorizontalCoordinate, Coordinate
# A coordinate in the ICRS standard frame (~= J2000)
c = ICRSCoordinate(ra, dec) #ra and dec are RA and Dec objects, or Angle objects
c = ICRSCoordinate("54.12412 deg", "-41:08:15.162342") #both strings are unambiguous
dec = c.dec
# dec is a Dec object
print(dec.degrees)
# -41.137545095
# It would be convenient to accept both (e.g. ra, dec) coordinates as a single
# string in the initializer. This can lead to ambiguities particularly when both
# are different units. The solution is to accept an array for the "unit" keyword
# that one would expect to sent to Angle:
c = ICRSCoordinate('4 23 43.43 +23 45 12.324', unit=[u.hour])
# The first element in 'units' refers to the first coordinate.
# DEGREE is assumed for the second coordinate
c = ICRSCoordinate('4 23 43.43 +23 45 12.324', unit=[u.hour, u.degree])
# Both can be specified and should be when there is ambiguity.
# Other types of coordinate systems have their own classes
c = GalacticCoordinates(l, b) #this only accepts Angles, *not* RA and Dec objects
c.l
# this is an Angle object, *not* RA or Dec
#some coordinates require an epoch - the argument will be an astropy.time.Time object
#and the syntax for that is still being ironed out
c = HorizontalCoordinates(alt, az, epoch=timeobj)
# Coordinate Factory
# ------------------
'''
To simplify usage, syntax will be provided to figure out the type of coordinates the user
wants without requiring them to know exactly which frame they want. The coordinate
system is determined from the keywords used when the object is initialized or can
be explicitly specified using "a1" and "a2" as angle parameters.
Question for the community: Should this be a *class* that overrides __new__ to provide
a new object, or a generator *function*. Erik prefers a function because he thinks it's
less magical to create an object from a function. Demitri prefers an abstract class
(Coordinate) that will return a new object that is subclassed from Coordinate, as
it's a more object-oriented solution and Demitri doesn't like functions in Python.
'''
#Note: `Coordinate` as used below would be `coordinate` if a function were used
c = Coordinate(ra="12:43:53", dec=-23, angle1_unit=u.hour, angle2_unit=u.degree)
# The ra and dec keywords imply equatorial coordinates, which will default to ICRS
# hence this returns an ICRSCoordinate
c = Coordinate(l=158.558650, b=-43.350066, angle1_unit=u.degree, angle2_unit=u.degree)
# l and b are for galactic coordinates, so this returns a GalacticCoordinate object
# Any acceptable input for RA() is accepted in Coordinate, etc.
c = Coordinate(ra="24:08:15.162342", dec=-41.432345, angle1_unit=u.hour, angle2_unit=u.degree)
# Mismatched keywords produce an error
c = Coordinate(ra="24:08:15.162342", b=-43.350066, angle1_unit=u.hour, angle2_unit=u.degree) # error
# Angle objects also accepted, and thus do not require units
ra = RA("4:08:15.162342", unit=u.hour)
dec = Dec("-41:08:15.162342")
c = Coordinate(ra=ra, dec=dec)
# Coordinate Conversions
# ----------------------
'''
Coordinate conversion occurs on-demand internally
'''
# using the c from above, which is RA,Dec = 4:08:15, -41:08:15.2
print(c.galactic)
# GalacticCoordinate(245.28098,-47.554501)
print(c.galactic.l, c.galactic.b) #the `galactic` result will be cached to speed this up
# Angle(158.558650) Angle(-43.350066)
# can also explicitly specify a coordinate class to convert to
gal = c.convert_to(GalacticCoordinate)
# can still convert back to equatorial using the shorthand
print(gal.equatorial.ra.to_string(unit=u.hour, sep=":", precision=2))
# 4:08:15.16
horiz = c.convert_to(HorizontalCoordinate)
# ConvertError - there's no way to convert to alt/az without a specified location
# users can specify their own coordinates and conversions
class CustomCoordinate(BaseCoordinate):
coordsysname = 'my_coord'
... specify conversions somewhere in the class ...
# allows both ways of converting
mycoord1 = c.my_coord
mycoord2 = c.convert_to(CustomCoordinate)
# Separations
# -----------
'''
Angular separations between two points on a sphere are supported via the
`separation` method.
'''
c1 = Coordinate(ra=0, dec=0, unit=u.degree)
c2 = Coordinate(ra=0, dec=1, unit=u.degree)
sep = c2.separation(c1)
#returns an AngularSeparation object (a subclass of Angle)
print(sep.degrees)
# 1.0
print(sep.arcmin)
# 60.0
c1 + c2
c1 - c2
# TypeError - these operations have ambiguous interpretations for points on a sphere
c3 = Coordinate(l=0, b=0, unit=u.degree) #Galactic Coordinates
# if there is a defined conversion between the relevant coordinate systems, it will
# be automatically performed to get the right angular separation
print c3.separation(c1).degrees
# 62.8716627659 - distance from the north galactic pole to celestial pole
c4 = CustomCoordinate(0, 0, unit=u.degree)
c4.separation(c1)
# raises an error if no conversion from the custom to equatorial
# coordinates is available
# Distances
# ---------
'''
Distances can also be specified, and allow for a full 3D definition of the coordinate.
'''
from astropy.coordinates import Distance
d = Distance(12, u.PARSECS)
# need to provide a unit
#standard units are pre-defined
print(distance.light_years)
# 39.12
print(distance.km)
# 3.7e14
print(distance.z) # redshift, assuming "current" cosmology
# (very small for 12 pc)
print(distance(<cosmology>).z) # custom cosmology possible
#Coordinate objects can be assigned a distance object, giving them a full 3D position:
c.distance = Distance(12, u.PARSECS)
# Coordinate objects can be initialized with a distance using special syntax
c1 = Coordinate(l=158.558650, b=-43.350066, unit=u.degree, distance=12 * u.KPC)
# Coordinate objects can be instantiated with cartesian coordinates
# Internally they will immediately be converted to two angles + a distance
c2 = ICRSCoordinates(x=2, y=4, z=8, unit=u.PARSEC)
c1.separation3d(c2, cosmology=<cosmology>) #if cosmology isn't given, use current
# returns a *3d* distance between the c1 and c2 coordinates with units
# Cartesian Coordinates
# ----------------------
'''
All spherical coordinate systems with distances can be converted to
cartesian coordinates.
'''
(x, y, z) = (c.x, c.y, c.z)
#this only computes the CartesianPoint *once*, and then caches it
c.cartesian
# returns CartesianPoint object, raise exception if no distance was specified
# CartesianPoint objects can be added and subtracted, which are vector/elementwise
# they can also be given as arguments to a coordinate system
csum = ICRSCoordinates(c1.cartesian + c2.cartesian)