forked from MarScaper/ephemeris
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathEphemeris.hpp
341 lines (271 loc) · 14.9 KB
/
Ephemeris.hpp
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
/*
* Ephemeris.hpp
*/
/*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#include <string.h>
// To speed up upload, you can disable planets calculations if not needed.
// VSOP87 and ELP2000 will not be loaded and solarSystemObjectAtDateAndTime()
// will simply return an empty object.
#define DISABLE_PLANETS 0
#ifndef Ephemeris_h
#define Ephemeris_h
#include "Calendar.hpp"
#if !DISABLE_PLANETS
#include "VSOP87.hpp"
#include "ELP2000.hpp"
#endif
/*! This structure describes equatorial coordinates. */
struct EquatorialCoordinates
{
/*! Floating value for Right Ascension. */
FLOAT ra;
/*! Floating value for Declination */
FLOAT dec;
};
/*! This structure describes horizontal coordinates. */
struct HorizontalCoordinates
{
/*! Floating value for altitude. */
FLOAT alt;
/*! Floating value for azimuth */
FLOAT azi;
};
/*! This structure describes Heliocentric ecliptic coordinates. */
struct HeliocentricCoordinates
{
/*! Floating value for ecliptic longitude. */
FLOAT lon;
/*! Floating value for ecliptic latitude.*/
FLOAT lat;
/*! Floating value for radius vector (distance from Sun). */
FLOAT radius;
};
/*! This structure describes geocentric coordinates. */
struct GeocentricCoordinates
{
/*! Floating value for longitude. */
FLOAT lon;
/*! Floating value for latitude.*/
FLOAT lat;
};
/*! This structure describes rectangular coordinates. */
struct RectangularCoordinates
{
FLOAT x;
FLOAT y;
FLOAT z;
};
/*! This structure describes available solar system objects for computation of ephemerides. */
enum SolarSystemObjectIndex
{
Sun = 0,
Mercury = 1,
Venus = 2,
Earth = 3,
Mars = 4,
Jupiter = 5,
Saturn = 6,
Uranus = 7,
Neptune = 8,
EarthsMoon = 9
};
enum RiseAndSetState
{
LocationOnEarthUnitialized,
RiseAndSetUdefined,
RiseAndSetOk,
ObjectAlwaysInSky,
ObjectNeverInSky
};
/*! This structure describes a planet for a specific date and time. */
struct SolarSystemObject
{
/*! Equatorial coordinates (RA/Dec). */
EquatorialCoordinates equaCoordinates;
/*! Horizontal coordinates (Alt/Az). */
HorizontalCoordinates horiCoordinates;
/*! Apparent diameter from earth in arc minutes. */
FLOAT diameter;
/*! Distance from earth in astronomical unit. */
FLOAT distance;
/*! Rise/Set state. */
RiseAndSetState riseAndSetState;
/*! Rise in floating hours. */
FLOAT rise;
/*! Set in floating hours. */
FLOAT set;
};
/*! This structure describes planetary orbit. */
struct PlanetayOrbit
{
/*! Mean longitude. */
FLOAT L;
/*! Semimajor axis. */
FLOAT a;
/*! Eccentricity. */
FLOAT e;
/*! Inclination. */
FLOAT i;
/*! Longitude ascending node. */
FLOAT omega;
/*! Perihelion. */
FLOAT pi;
/*! Mean anomaly. */
FLOAT M;
/*! Perihelion argument. */
FLOAT w;
};
/*!
* This class is used for astronomical calculations. The code is based on the book "Astronomical Algorithms" by Jean Meeus.
*/
class Ephemeris
{
public:
/*! Flip longitude coordinate. Default: West is negative and East is positive. */
static void flipLongitude(bool flip);
/*! Set location on earth (used for horizontal coordinates conversion). */
static void setLocationOnEarth(FLOAT floatingLatitude, FLOAT floatingLongitude);
/*! Set location on earth (used for horizontal coordinates conversion). */
static void setLocationOnEarth(FLOAT latDegrees, FLOAT latMinutes, FLOAT latSeconds,
FLOAT lonDegrees, FLOAT lonMinutes, FLOAT lonSeconds);
/*! Set altitude in meters for location on earth (improve precision for rise and set). */
static void setAltitude(int altitude);
/*! Convert floating hours to integer hours, minutes, seconds. */
static void floatingHoursToHoursMinutesSeconds(FLOAT floatingHours, int *hours, int *minutes, FLOAT *seconds);
/*! Convert integer hours, minutes, seconds to floating hours. */
static FLOAT hoursMinutesSecondsToFloatingHours(int hours, int minutes, FLOAT seconds);
/*! Convert floating degrees to integer degrees, minutes, seconds. */
static void floatingDegreesToDegreesMinutesSeconds(FLOAT floatingDegrees, int *degrees, int *minutes, FLOAT *seconds);
/*! Convert integer degrees, minutes, seconds to floating degrees. */
static FLOAT degreesMinutesSecondsToFloatingDegrees(int degrees, int minutes, FLOAT seconds);
/*! Convert floating hours by applying UTC offset. */
static FLOAT floatingHoursWithUTCOffset(float floatingHours, int UTCOffset);
/*! Convert equatorial coordinates for a specified equinox to apparent equatorial coordinates (JNow)
* for a specified date and time. Conversion applies, drift per year, precession of the equinoxes, nutation and aberration.
* eqDriftPerYear.ra must be expressed in s/year.
* eqDriftPerYear.dec must be expressed in "/year. */
static EquatorialCoordinates equatorialEquinoxToEquatorialJNowAtDateAndTime(EquatorialCoordinates eqEquinoxCoordinates,
int equinox,
EquatorialCoordinates eqDriftPerYear,
unsigned int day, unsigned int month, unsigned int year,
unsigned int hours, unsigned int minutes, unsigned int seconds);
/*! Convert equatorial coordinates for a specified equinox to apparent equatorial coordinates (JNow)
* for a specified date and time. Conversion applies precession of the equinoxes, nutation and aberration. */
static EquatorialCoordinates equatorialEquinoxToEquatorialJNowAtDateAndTime(EquatorialCoordinates eqEquinoxCoordinates,
int equinox,
unsigned int day, unsigned int month, unsigned int year,
unsigned int hours, unsigned int minutes, unsigned int seconds);
/*! Convert equatorial coordinates to horizontal coordinates. Location on Earth must be initialized first. */
static HorizontalCoordinates equatorialToHorizontalCoordinatesAtDateAndTime(EquatorialCoordinates eqCoordinates,
unsigned int day, unsigned int month, unsigned int year,
unsigned int hours, unsigned int minutes, unsigned int seconds);
/*! Convert horizontal coordinates to equatorial coordinates. Location on Earth must be initialized first. */
static EquatorialCoordinates horizontalToEquatorialCoordinatesAtDateAndTime(HorizontalCoordinates hCoordinates,
unsigned int day, unsigned int month, unsigned int year,
unsigned int hours, unsigned int minutes, unsigned int seconds);
/*! Compute solar system object for a specific date, time and location on earth (if location has been initialized first). */
static SolarSystemObject solarSystemObjectAtDateAndTime(SolarSystemObjectIndex planet,
unsigned int day, unsigned int month, unsigned int year,
unsigned int hours, unsigned int minutes, unsigned int seconds);
/*! Compute rise and set for the equatorial coordinates we want. */
static RiseAndSetState riseAndSetForEquatorialCoordinatesAtDateAndTime(EquatorialCoordinates coord,
FLOAT *rise, FLOAT *set,
unsigned int day, unsigned int month, unsigned int year,
unsigned int hours, unsigned int minutes, unsigned int seconds);
private:
/*! Compute apparent sideral time (in floating hours) for a given date and time.
* Reference: Chapter 7, page 35: Temps sidéral à Greenwich. */
static FLOAT apparentSideralTime(unsigned int day, unsigned int month, unsigned int year,
unsigned int hours, unsigned int minutes, unsigned int seconds);
/*! Compute mean sideral time for Greenwich.
* Reference: Chapter 7, page 35: Temps sidéral à Greenwich. */
static FLOAT meanGreenwichSiderealTimeAtDateAndTime(unsigned int day, unsigned int month, unsigned int year,
unsigned int hours, unsigned int minutes, unsigned int seconds);
/*! Compute mean sideral time for Greenwich.
* Reference: Chapter 7, page 35: Temps sidéral à Greenwich. */
static FLOAT meanGreenwichSiderealTimeAtJD(JulianDay jd);
/*! Compute heliocentric coordinates.
* Reference: Chapter 22, page 83: Position des planètes. */
static HeliocentricCoordinates heliocentricCoordinatesForPlanetAndT(SolarSystemObjectIndex planet, FLOAT T);
/*! Compute Kepler equation.
* Reference: Chapter 20, page 73: Equation de Kepler. */
static FLOAT kepler(FLOAT M, FLOAT e);
/*! Convert equatorial coordinates to horizontal coordinates.
* Reference: Chapter 8, page 37: Transformation de coordonnées. */
static HorizontalCoordinates equatorialToHorizontal(FLOAT H, FLOAT delta, FLOAT phi);
/*! Convert horizontal coordinates to equatorial coordinates.
* Reference: Chapter 8, page 37: Transformation de coordonnées. */
static EquatorialCoordinates horizontalToEquatorial(FLOAT azimuth, FLOAT altitude, FLOAT latitude);
/*! Convert ecliptic coordinates to equatorial coordinates.
* Reference: Chapter 8, page 37: Transformation de coordonnées. */
static EquatorialCoordinates EclipticToEquatorial(FLOAT lambda, FLOAT beta, FLOAT epsilon);
/*! Convert heliocentric coordinates to rectangular coordinates.
* Reference: Chapter 23, page 87: Mouvement elliptique. */
static RectangularCoordinates HeliocentricToRectangular(HeliocentricCoordinates hc, HeliocentricCoordinates hc0);
/*! Compute the true obliquity (angle in floating degrees) of the ecliptic,
* delta obliquity and delta nutation for T.
* Reference: Chapter 13, page 53: Nutation et obliquité de l'écliptique. */
static FLOAT obliquityAndNutationForT(FLOAT T, FLOAT *deltaObliquity, FLOAT *deltaNutation);
/*! Compute planet informations for T.
* Reference: Chapter 21, page 77: Eléments des orbites planétaires. */
#if !DISABLE_PLANETS
static PlanetayOrbit planetayOrbitForPlanetAndT(SolarSystemObjectIndex planet, FLOAT T);
#endif
/*! Compute Moon coordinates in the sky (R.A.,Dec) for a specific date and time.
* Reference: Chapter 28, page 109: Position de la Lune.
* Chapter 8, page 37: Transformation de coordonnées. */
#if !DISABLE_PLANETS
static EquatorialCoordinates equatorialCoordinatesForEarthsMoonAtJD(JulianDay jd, FLOAT *distance);
#endif
/*! Compute Sun coordinates in the sky (R.A.,Dec) for a specific date and time.
* Reference: Chapter 16, page 63: Les coordonnées du soleil. */
#if !DISABLE_PLANETS
static EquatorialCoordinates equatorialCoordinatesForSunAtJD(JulianDay jd, FLOAT *distance);
#endif
/*! Compute planet equatorial coordinates (and geocentric if needed) for a a specific Julian day.
* Reference: Chapter 23, page 87: Mouvement elliptique.
* Chapter 8, page 37: Transformation de coordonnées. */
#if !DISABLE_PLANETS
static EquatorialCoordinates equatorialCoordinatesForPlanetAtJD(SolarSystemObjectIndex planet, JulianDay jd, FLOAT *distance);
#endif
#if !DISABLE_PLANETS
/*! Compute VSOP87 (Planets) coefficients for T.
* Reference: Chapter 22, page 83: Position des planètes. */
static FLOAT sumVSOP87Coefs(const VSOP87Coefficient *valuePlanetCoefficients, int coefCount, FLOAT T);
#endif
#if !DISABLE_PLANETS
/*! Compute ELP2000 (Earth's Moon) coefficients for T.
* Reference: Chapter 28, page 109: Position de la Lune. */
static FLOAT sumELP2000Coefs(const FLOAT *moonCoefficients, const ELP2000Coefficient *moonAngleCoefficients, int coefCount,
FLOAT E, FLOAT D, FLOAT M, FLOAT Mp, FLOAT F, bool squareMultiplicator);
#endif
/*! Compute rise and set for specified equatorial coordinates, T0 (Mean sideral time at midnight), paralax, apparent diameter, and altitude.
* Reference: https://www.imcce.fr/langues/en/grandpublic/systeme/promenade-en/pages3/367.html */
static RiseAndSetState riseAndSetForEquatorialCoordinatesAndT0(EquatorialCoordinates coord, FLOAT T0, FLOAT *rise, FLOAT *set,
FLOAT paralax, FLOAT apparentDiameter);
/*! Convert equatorial coordinates for a specified equinox to apparent equatorial coordinates (JNow) for a specified T.
* Conversion applies, drift per year, precession of the equinoxes, nutation and aberration.
* eqDriftPerYear.ra must be expressed in s/year.
* eqDriftPerYear.dec must be expressed in "/year.
* Reference: Chapter 12, page 49: Precession.
* Chapter 14, page 57: Position apparente d'une étoile. */
static EquatorialCoordinates equatorialEquinoxToEquatorialJNowAtDateForT(EquatorialCoordinates eqEquinoxCoordinates,
int equinox,
EquatorialCoordinates eqDriftPerYear,
FLOAT T,
unsigned int year);
};
#endif