xref: /webtrees/app/Date.php (revision e73150af6665c6617220df2d362a9ac84e040860)
1<?php
2
3/**
4 * webtrees: online genealogy
5 * Copyright (C) 2021 webtrees development team
6 * This program is free software: you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation, either version 3 of the License, or
9 * (at your option) any later version.
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
14 * You should have received a copy of the GNU General Public License
15 * along with this program. If not, see <https://www.gnu.org/licenses/>.
16 */
17
18declare(strict_types=1);
19
20namespace Fisharebest\Webtrees;
21
22use DomainException;
23use Fisharebest\ExtCalendar\GregorianCalendar;
24use Fisharebest\Webtrees\Date\AbstractCalendarDate;
25use Fisharebest\Webtrees\Date\FrenchDate;
26use Fisharebest\Webtrees\Date\GregorianDate;
27use Fisharebest\Webtrees\Date\HijriDate;
28use Fisharebest\Webtrees\Date\JalaliDate;
29use Fisharebest\Webtrees\Date\JewishDate;
30use Fisharebest\Webtrees\Date\JulianDate;
31use Fisharebest\Webtrees\Date\RomanDate;
32
33/**
34 * A representation of GEDCOM dates and date ranges.
35 *
36 * Since different calendars start their days at different times, (civil
37 * midnight, solar midnight, sunset, sunrise, etc.), we convert on the basis of
38 * midday.
39 *
40 * We assume that years start on the first day of the first month. Where
41 * this is not the case (e.g. England prior to 1752), we need to use modified
42 * years or the OS/NS notation "4 FEB 1750/51".
43 */
44class Date
45{
46    // Optional qualifier, such as BEF, FROM, ABT
47    public string $qual1 = '';
48
49    // The first (or only) date
50    private AbstractCalendarDate $date1;
51
52    // Optional qualifier, such as TO, AND
53    public string $qual2 = '';
54
55    // Optional second date
56    private ?AbstractCalendarDate $date2 = null;
57
58    // Optional text, as included with an INTerpreted date
59    private string $text = '';
60
61    /**
62     * Create a date, from GEDCOM data.
63     *
64     * @param string $date A date in GEDCOM format
65     */
66    public function __construct(string $date)
67    {
68        $calendar_date_factory = Registry::calendarDateFactory();
69
70        // Extract any explanatory text
71        if (preg_match('/^(.*) ?[(](.*)[)]/', $date, $match)) {
72            $date       = $match[1];
73            $this->text = $match[2];
74        }
75        if (preg_match('/^(FROM|BET) (.+) (AND|TO) (.+)/', $date, $match)) {
76            $this->qual1 = $match[1];
77            $this->date1 = $calendar_date_factory->make($match[2]);
78            $this->qual2 = $match[3];
79            $this->date2 = $calendar_date_factory->make($match[4]);
80        } elseif (preg_match('/^(TO|FROM|BEF|AFT|CAL|EST|INT|ABT) (.+)/', $date, $match)) {
81            $this->qual1 = $match[1];
82            $this->date1 = $calendar_date_factory->make($match[2]);
83        } else {
84            $this->date1 = $calendar_date_factory->make($date);
85        }
86    }
87
88    /**
89     * When we copy a date object, we need to create copies of
90     * its child objects.
91     */
92    public function __clone()
93    {
94        $this->date1 = clone $this->date1;
95        if ($this->date2 !== null) {
96            $this->date2 = clone $this->date2;
97        }
98    }
99
100    /**
101     * Convert a date to the preferred format and calendar(s) display.
102     *
103     * @param Tree|null   $tree              Wrap the date in a link to the calendar page for the tree
104     * @param string|null $date_format       Override the default date format
105     * @param bool        $convert_calendars Convert the date into other calendars (requires a tree)
106     *
107     * @return string
108     */
109    public function display(Tree $tree = null, string $date_format = null, bool $convert_calendars = false): string
110    {
111        if ($tree instanceof Tree) {
112            $CALENDAR_FORMAT = $tree->getPreference('CALENDAR_FORMAT');
113        } else {
114            $CALENDAR_FORMAT = 'none';
115        }
116
117        $date_format = $date_format ?? I18N::dateFormat();
118
119        if ($convert_calendars) {
120            $calendar_format = explode('_and_', $CALENDAR_FORMAT);
121        } else {
122            $calendar_format = [];
123        }
124
125        // Two dates with text before, between and after
126        $q1 = $this->qual1;
127        $d1 = $this->date1->format($date_format, $this->qual1);
128        $q2 = $this->qual2;
129        if ($this->date2 === null) {
130            $d2 = '';
131        } else {
132            $d2 = $this->date2->format($date_format, $this->qual2);
133        }
134        // Con vert to other calendars, if requested
135        $conv1 = '';
136        $conv2 = '';
137        foreach ($calendar_format as $cal_fmt) {
138            if ($cal_fmt !== 'none') {
139                $d1conv = $this->date1->convertToCalendar($cal_fmt);
140                if ($d1conv->inValidRange()) {
141                    $d1tmp = $d1conv->format($date_format, $this->qual1);
142                } else {
143                    $d1tmp = '';
144                }
145                if ($this->date2 === null) {
146                    $d2conv = null;
147                    $d2tmp  = '';
148                } else {
149                    $d2conv = $this->date2->convertToCalendar($cal_fmt);
150                    if ($d2conv->inValidRange()) {
151                        $d2tmp = $d2conv->format($date_format, $this->qual2);
152                    } else {
153                        $d2tmp = '';
154                    }
155                }
156                // If the date is different from the unconverted date, add it to the date string.
157                if ($d1 != $d1tmp && $d1tmp !== '') {
158                    if ($tree instanceof Tree) {
159                        if ($CALENDAR_FORMAT !== 'none') {
160                            $conv1 .= ' <span dir="' . I18N::direction() . '">(<a href="' . e($d1conv->calendarUrl($date_format, $tree)) . '" rel="nofollow">' . $d1tmp . '</a>)</span>';
161                        } else {
162                            $conv1 .= ' <span dir="' . I18N::direction() . '"><br><a href="' . e($d1conv->calendarUrl($date_format, $tree)) . '" rel="nofollow">' . $d1tmp . '</a></span>';
163                        }
164                    } else {
165                        $conv1 .= ' <span dir="' . I18N::direction() . '">(' . $d1tmp . ')</span>';
166                    }
167                }
168                if ($this->date2 !== null && $d2 != $d2tmp && $d1tmp != '') {
169                    if ($tree instanceof Tree) {
170                        $conv2 .= ' <span dir="' . I18N::direction() . '">(<a href="' . e($d2conv->calendarUrl($date_format, $tree)) . '" rel="nofollow">' . $d2tmp . '</a>)</span>';
171                    } else {
172                        $conv2 .= ' <span dir="' . I18N::direction() . '">(' . $d2tmp . ')</span>';
173                    }
174                }
175            }
176        }
177
178        // Add URLs, if requested
179        if ($tree instanceof Tree) {
180            $d1 = '<a href="' . e($this->date1->calendarUrl($date_format, $tree)) . '" rel="nofollow">' . $d1 . '</a>';
181            if ($this->date2 instanceof AbstractCalendarDate) {
182                $d2 = '<a href="' . e($this->date2->calendarUrl($date_format, $tree)) . '" rel="nofollow">' . $d2 . '</a>';
183            }
184        }
185
186        // Localise the date
187        switch ($q1 . $q2) {
188            case '':
189                $tmp = $d1 . $conv1;
190                break;
191            case 'ABT':
192                /* I18N: Gedcom ABT dates */
193                $tmp = I18N::translate('about %s', $d1 . $conv1);
194                break;
195            case 'CAL':
196                /* I18N: Gedcom CAL dates */
197                $tmp = I18N::translate('calculated %s', $d1 . $conv1);
198                break;
199            case 'EST':
200                /* I18N: Gedcom EST dates */
201                $tmp = I18N::translate('estimated %s', $d1 . $conv1);
202                break;
203            case 'INT':
204                /* I18N: Gedcom INT dates */
205                $tmp = I18N::translate('interpreted %s (%s)', $d1 . $conv1, e($this->text));
206                break;
207            case 'BEF':
208                /* I18N: Gedcom BEF dates */
209                $tmp = I18N::translate('before %s', $d1 . $conv1);
210                break;
211            case 'AFT':
212                /* I18N: Gedcom AFT dates */
213                $tmp = I18N::translate('after %s', $d1 . $conv1);
214                break;
215            case 'FROM':
216                /* I18N: Gedcom FROM dates */
217                $tmp = I18N::translate('from %s', $d1 . $conv1);
218                break;
219            case 'TO':
220                /* I18N: Gedcom TO dates */
221                $tmp = I18N::translate('to %s', $d1 . $conv1);
222                break;
223            case 'BETAND':
224                /* I18N: Gedcom BET-AND dates */
225                $tmp = I18N::translate('between %s and %s', $d1 . $conv1, $d2 . $conv2);
226                break;
227            case 'FROMTO':
228                /* I18N: Gedcom FROM-TO dates */
229                $tmp = I18N::translate('from %s to %s', $d1 . $conv1, $d2 . $conv2);
230                break;
231            default:
232                $tmp = I18N::translate('Invalid date');
233                break;
234        }
235
236        if (strip_tags($tmp) === '') {
237            return '';
238        }
239
240        return '<span class="date">' . $tmp . '</span>';
241    }
242
243    /**
244     * Get the earliest calendar date from this GEDCOM date.
245     *
246     * In the date “FROM 1900 TO 1910”, this would be 1900.
247     *
248     * @return AbstractCalendarDate
249     */
250    public function minimumDate(): AbstractCalendarDate
251    {
252        return $this->date1;
253    }
254
255    /**
256     * Get the latest calendar date from this GEDCOM date.
257     *
258     * In the date “FROM 1900 TO 1910”, this would be 1910.
259     *
260     * @return AbstractCalendarDate
261     */
262    public function maximumDate(): AbstractCalendarDate
263    {
264        return $this->date2 ?? $this->date1;
265    }
266
267    /**
268     * Get the earliest Julian day number from this GEDCOM date.
269     *
270     * @return int
271     */
272    public function minimumJulianDay(): int
273    {
274        return $this->minimumDate()->minimumJulianDay();
275    }
276
277    /**
278     * Get the latest Julian day number from this GEDCOM date.
279     *
280     * @return int
281     */
282    public function maximumJulianDay(): int
283    {
284        return $this->maximumDate()->maximumJulianDay();
285    }
286
287    /**
288     * Get the middle Julian day number from the GEDCOM date.
289     *
290     * For a month-only date, this would be somewhere around the 16th day.
291     * For a year-only date, this would be somewhere around 1st July.
292     *
293     * @return int
294     */
295    public function julianDay(): int
296    {
297        return intdiv($this->minimumJulianDay() + $this->maximumJulianDay(), 2);
298    }
299
300    /**
301     * Offset this date by N years, and round to the whole year.
302     *
303     * This is typically used to create an estimated death date,
304     * which is before a certain number of years after the birth date.
305     *
306     * @param int    $years     a number of years, positive or negative
307     * @param string $qualifier typically “BEF” or “AFT”
308     *
309     * @return Date
310     */
311    public function addYears(int $years, string $qualifier = ''): Date
312    {
313        $tmp               = clone $this;
314        $tmp->date1->year  += $years;
315        $tmp->date1->month = 0;
316        $tmp->date1->day   = 0;
317        $tmp->date1->setJdFromYmd();
318        $tmp->qual1 = $qualifier;
319        $tmp->qual2 = '';
320        $tmp->date2 = null;
321
322        return $tmp;
323    }
324
325    /**
326     * Compare two dates, so they can be sorted.
327     *
328     * return -1 if $a<$b
329     * return +1 if $b>$a
330     * return  0 if dates same/overlap
331     * BEF/AFT sort as the day before/after
332     *
333     * @param Date $a
334     * @param Date $b
335     *
336     * @return int
337     */
338    public static function compare(Date $a, Date $b): int
339    {
340        // Get min/max JD for each date.
341        switch ($a->qual1) {
342            case 'BEF':
343                $amin = $a->minimumJulianDay() - 1;
344                $amax = $amin;
345                break;
346            case 'AFT':
347                $amax = $a->maximumJulianDay() + 1;
348                $amin = $amax;
349                break;
350            default:
351                $amin = $a->minimumJulianDay();
352                $amax = $a->maximumJulianDay();
353                break;
354        }
355        switch ($b->qual1) {
356            case 'BEF':
357                $bmin = $b->minimumJulianDay() - 1;
358                $bmax = $bmin;
359                break;
360            case 'AFT':
361                $bmax = $b->maximumJulianDay() + 1;
362                $bmin = $bmax;
363                break;
364            default:
365                $bmin = $b->minimumJulianDay();
366                $bmax = $b->maximumJulianDay();
367                break;
368        }
369        if ($amax < $bmin) {
370            return -1;
371        }
372
373        if ($amin > $bmax && $bmax > 0) {
374            return 1;
375        }
376
377        if ($amin < $bmin && $amax <= $bmax) {
378            return -1;
379        }
380
381        if ($amin > $bmin && $amax >= $bmax && $bmax > 0) {
382            return 1;
383        }
384
385        return 0;
386    }
387
388    /**
389     * Check whether a gedcom date contains usable calendar date(s).
390     *
391     * An incomplete date such as "12 AUG" would be invalid, as
392     * we cannot sort it.
393     *
394     * @return bool
395     */
396    public function isOK(): bool
397    {
398        return $this->minimumJulianDay() && $this->maximumJulianDay();
399    }
400
401    /**
402     * Calculate the gregorian year for a date. This should NOT be used internally
403     * within WT - we should keep the code "calendar neutral" to allow support for
404     * jewish/arabic users. This is only for interfacing with external entities,
405     * such as the ancestry.com search interface or the dated fact icons.
406     *
407     * @return int
408     */
409    public function gregorianYear(): int
410    {
411        if ($this->isOK()) {
412            $gregorian_calendar = new GregorianCalendar();
413            [$year] = $gregorian_calendar->jdToYmd($this->julianDay());
414
415            return $year;
416        }
417
418        return 0;
419    }
420}
421