001/*
002 * This file is part of McIDAS-V
003 *
004 * Copyright 2007-2023
005 * Space Science and Engineering Center (SSEC)
006 * University of Wisconsin - Madison
007 * 1225 W. Dayton Street, Madison, WI 53706, USA
008 * https://www.ssec.wisc.edu/mcidas
009 * 
010 * All Rights Reserved
011 * 
012 * McIDAS-V is built on Unidata's IDV and SSEC's VisAD libraries, and
013 * some McIDAS-V source code is based on IDV and VisAD source code.  
014 * 
015 * McIDAS-V is free software; you can redistribute it and/or modify
016 * it under the terms of the GNU Lesser Public License as published by
017 * the Free Software Foundation; either version 3 of the License, or
018 * (at your option) any later version.
019 * 
020 * McIDAS-V is distributed in the hope that it will be useful,
021 * but WITHOUT ANY WARRANTY; without even the implied warranty of
022 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
023 * GNU Lesser Public License for more details.
024 * 
025 * You should have received a copy of the GNU Lesser Public License
026 * along with this program.  If not, see http://www.gnu.org/licenses.
027 */
028package edu.wisc.ssec.mcidasv.util;
029
030import java.util.Date;
031import java.util.Locale;
032import java.util.TimeZone;
033import java.text.DateFormat;
034import java.text.ParseException;
035import java.text.SimpleDateFormat;
036import java.io.PrintStream;
037
038
039/**
040 * Approximations de quelques calculs astronomiques relatifs aux calendriers terrestres.
041 * Les différents cycles astronomiques (notamment le jour, le mois et l'année) ne sont pas
042 * constants. Par exemple, la longueur de l'année tropicale (le nombre moyen de jours entre
043 * deux équinoxes vernales) était d'environ 365,242196 jours en 1900 et devrait être d'environ
044 * 365,242184 jours en 2100, soit un changement d'environ 1 seconde. Cette classe permet de
045 * calculer la longueur d'une année ou d'un mois à une date spécifiée. Toutefois, il est
046 * important de noter que les intervalles de temps calculés par les méthodes de cette classe
047 * sont des <strong>moyennes</strong>. Pour une année en particulier, l'intervalle de temps
048 * d'un équinoxe vernale au prochain peut s'écarter de cette moyenne de plusieurs minutes.
049 * <p>
050 * Les calculs de la longueur de l'année tropicale sont basés sur les travaux de Laskar (1986).
051 * Les calculs de la longueur des mois synodiques sont basés sur les travaux de Chapront-Touze et
052 * Chapront (1988).On peut lire plus de détails au sujet des calendrier terrestre au site
053 * <a href="http://webexhibits.org/calendars/year-astronomy.html">http://webexhibits.org/calendars/year-astronomy.html</a>
054 * ainsi que
055 * <a href="http://www.treasure-troves.com/astro/TropicalYear.html">http://www.treasure-troves.com/astro/TropicalYear.html</a>.
056 *
057 * @author Martin Desruisseaux (IRD)
058 * @version 3.00
059 *
060 * @since 2.0
061 * @module
062 */
063public final class Calendar {
064    /**
065     * Nombre de millisecondes dans une journée. Cette constante est
066     * utilisée pour convertir des intervalles de temps du Java en
067     * nombre de jours.
068     */
069    private static final double MILLIS_IN_DAY = 1000*60*60*24;
070
071    /**
072     * Jour julien correspondant à l'époch du Java (1er janvier 1970 à minuit).
073     * Cette constante est utilisée pour convertir des dates du Java en jour
074     * julien.
075     *
076     * La valeur {@link #julianDay}   du 1er janvier 2000 00:00 GMT est 2451544.5 jours.
077     * La valeur {@link Date#getTime} du 1er janvier 2000 00:00 GMT est 10957 jours.
078     */
079    private static final double JULIAN_DAY_1970 = 2451544.5-10957;
080
081    /**
082     * Do not allow instantiation of this class.
083     */
084    private Calendar() {
085    }
086
087    /**
088     * Returns the julian day of the given date. This is the number of days elapsed since
089     * January 1st, 4713 before J.C. at noon GMT. This is named after <cite>Julius Scaliger</cite>,
090     * not to be confused to the number of days elapsed since the beginning of the year (named after
091     * <cite>Julius Caesar</cite>).
092     *
093     * @param  time The time for which to evaluate the julian day.
094     * @return Number of days elapsed since January 1st, 4713 before J.C. at noon GMT.
095     */
096    public static double julianDay(final Date time) {
097        return julianDay(time.getTime());
098    }
099
100    /**
101     * Computes the {@linkplain #julianDay(Date) julian day}.
102     *
103     * @param time The date in milliseconds elapsed since January 1st, 1970.
104     */
105    static double julianDay(final long time) {
106        return (time/MILLIS_IN_DAY) + JULIAN_DAY_1970;
107    }
108
109    /**
110     * Retourne le nombre de siècles écoulés depuis le 1 janvier 2000 à midi.
111     * Cette information est utilisée dans les formules de Laskar (1986) pour
112     * calculer la longueur d'une année tropicale, ainsi que par Chapront-Touze
113     * et Chapront (1988) pour la longueur d'un mois synodique.
114     */
115    static double julianCentury(final Date time) {
116        return ((time.getTime() / MILLIS_IN_DAY) + (JULIAN_DAY_1970 - 2451545.0)) / 36525;
117    }
118
119    /**
120     * Retourne la longueur de l'année tropicale. L'année tropicale est définie comme l'intervalle
121     * moyen entre deux équinoxes vernales (autour du 21 mars dans l'hémisphère nord). Il correspond
122     * au cycle des saisons. Cet intervalle de temps est une <strong>moyenne</strong>. Un cycle réel
123     * peut s'écarter de plusieurs minutes de cette moyenne. Notez aussi qu'une année tropicale
124     * n'est pas identique à une année sidérale, qui est le temps requis par la Terre pour compléter
125     * un orbite autour du Soleil. En l'an 2000, l'année tropicale avait une longueur d'environ
126     * 365,2422 jours tandis que l'année sidérale avait une longueur de 365,2564 jours.
127     *
128     * @param  time A date in the year for which to compute the length.
129     * @return The tropical length of the given year.
130     */
131    public static double tropicalYearLength(final Date time) {
132        final double T = julianCentury(time);
133        return 365.2421896698 + T*(-0.00000615359 + T*(-7.29E-10 + T*(2.64E-10)));
134    }
135
136    /**
137     * Retourne la longueur du mois synodique. Le mois synodique est l'intervalle de temps moyen
138     * entre deux conjonctions de la lune et du soleil. Il correspond au cycle des phases de la
139     * lune. Cet intervalle de temps est une <strong>moyenne</strong>. Un cycle réel peut s'écarter
140     * de plusieurs heures de cette moyenne.
141     *
142     * @param  time A date in the month for which to compute the length.
143     * @return The synodic length of the given month.
144     */
145    public static double synodicMonthLength(final Date time) {
146        final double T=julianCentury(time);
147        return 29.5305888531 + T*(0.00000021621 + T*(-3.64E-10));
148    }
149
150    /**
151     * Returns a date format to be used by the command line tools.
152     */
153    static DateFormat getDateFormat() {
154        final DateFormat format = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss", Locale.CANADA);
155        format.setTimeZone(TimeZone.getTimeZone("UTC"));
156        return format;
157    }
158
159    /**
160     * Prints the length of a tropical year and a synodic month for the given date.
161     * Cette application peut être lancée avec la syntaxe suivante:
162     * This application can be launch from the command line as below:
163     *
164     * {@preformat shell
165     *     java org.geotoolkit.nature.Calendar [date]
166     * }
167     *
168     * where <var>date</var> is an optional argument specifying the date and time in the
169     * {@code "yyyy-MM-dd HH:mm:ss"} format, UTC time zone. If this argument is omitted,
170     * then the current time is used.
171     *
172     * @param args The command line argument.
173     * @throws ParseException If the date is not properly formatted.
174     */
175    public static void main(final String[] args) throws ParseException {
176        final DateFormat format = getDateFormat();
177        final Date time = (args.length != 0) ? format.parse(args[0]) : new Date();
178        final PrintStream out = System.out;
179        out.print("Date (UTC)   : "); out.println(format.format(time));
180        out.print("Tropical year: "); out.println(tropicalYearLength(time));
181        out.print("Synodic month: "); out.println(synodicMonthLength(time));
182    }
183}