xref: /webtrees/resources/js/webtrees.js (revision e99ac601cee7ced57ac8545ceba47ece1b15d883)
1/**
2 * webtrees: online genealogy
3 * Copyright (C) 2022 webtrees development team
4 * This program is free software: you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License as published by
6 * the Free Software Foundation, either version 3 of the License, or
7 * (at your option) any later version.
8 * This program is distributed in the hope that it will be useful,
9 * but WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 * GNU General Public License for more details.
12 * You should have received a copy of the GNU General Public License
13 * along with this program. If not, see <http://www.gnu.org/licenses/>.
14 */
15
16'use strict';
17
18(function (webtrees) {
19  const lang = document.documentElement.lang;
20
21  // Identify the script used by some text.
22  const scriptRegexes = {
23    Han: /[\u3400-\u9FCC]/,
24    Grek: /[\u0370-\u03FF]/,
25    Cyrl: /[\u0400-\u04FF]/,
26    Hebr: /[\u0590-\u05FF]/,
27    Arab: /[\u0600-\u06FF]/
28  };
29
30  /**
31   * Tidy the whitespace in a string.
32   * @param {string} str
33   * @returns {string}
34   */
35  function trim (str) {
36    return str.replace(/\s+/g, ' ').trim();
37  }
38
39  /**
40   * Look for non-latin characters in a string.
41   * @param {string} str
42   * @returns {string}
43   */
44  webtrees.detectScript = function (str) {
45    for (const script in scriptRegexes) {
46      if (str.match(scriptRegexes[script])) {
47        return script;
48      }
49    }
50
51    return 'Latn';
52  };
53
54  /**
55   * In some languages, the SURN uses a male/default form, but NAME uses a gender-inflected form.
56   * @param {string} surname
57   * @param {string} sex
58   * @returns {string}
59   */
60  function inflectSurname (surname, sex) {
61    if (lang === 'pl' && sex === 'F') {
62      return surname
63        .replace(/ski$/, 'ska')
64        .replace(/cki$/, 'cka')
65        .replace(/dzki$/, 'dzka')
66        .replace(/żki$/, 'żka');
67    }
68
69    return surname;
70  }
71
72  /**
73   * Build a NAME from a NPFX, GIVN, SPFX, SURN and NSFX parts.
74   * Assumes the language of the document is the same as the language of the name.
75   * @param {string} npfx
76   * @param {string} givn
77   * @param {string} spfx
78   * @param {string} surn
79   * @param {string} nsfx
80   * @param {string} sex
81   * @returns {string}
82   */
83  webtrees.buildNameFromParts = function (npfx, givn, spfx, surn, nsfx, sex) {
84    const usesCJK = webtrees.detectScript(npfx + givn + spfx + givn + surn + nsfx) === 'Han';
85    const separator = usesCJK ? '' : ' ';
86    const surnameFirst = usesCJK || ['hu', 'jp', 'ko', 'vi', 'zh-Hans', 'zh-Hant'].indexOf(lang) !== -1;
87    const patronym = ['is'].indexOf(lang) !== -1;
88    const slash = patronym ? '' : '/';
89
90    // GIVN and SURN may be a comma-separated lists.
91    npfx = trim(npfx);
92    givn = trim(givn.replace(/,/g, separator));
93    spfx = trim(spfx);
94    surn = inflectSurname(trim(surn.replace(/,/g, separator)), sex);
95    nsfx = trim(nsfx);
96
97    const surname_separator = spfx.endsWith('\'') || spfx.endsWith('‘') ? '' : ' ';
98
99    const surname = trim(spfx + surname_separator + surn);
100
101    const name = surnameFirst ? slash + surname + slash + separator + givn : givn + separator + slash + surname + slash;
102
103    return trim(npfx + separator + name + separator + nsfx);
104  };
105
106  // Insert text at the current cursor position in a text field.
107  webtrees.pasteAtCursor = function (element, text) {
108    if (element !== null) {
109      const caret_pos = element.selectionStart + text.length;
110      const textBefore = element.value.substring(0, element.selectionStart);
111      const textAfter = element.value.substring(element.selectionEnd);
112      element.value = textBefore + text + textAfter;
113      element.setSelectionRange(caret_pos, caret_pos);
114      element.focus();
115    }
116  };
117
118  /**
119   * @param {Element} datefield
120   * @param {string} dmy
121   */
122  webtrees.reformatDate = function (datefield, dmy) {
123    const months = ['JAN', 'FEB', 'MAR', 'APR', 'MAY', 'JUN', 'JUL', 'AUG', 'SEP', 'OCT', 'NOV', 'DEC'];
124    const hijri_months = ['MUHAR', 'SAFAR', 'RABIA', 'RABIT', 'JUMAA', 'JUMAT', 'RAJAB', 'SHAAB', 'RAMAD', 'SHAWW', 'DHUAQ', 'DHUAH'];
125    const hebrew_months = ['TSH', 'CSH', 'KSL', 'TVT', 'SHV', 'ADR', 'ADS', 'NSN', 'IYR', 'SVN', 'TMZ', 'AAV', 'ELL'];
126    const french_months = ['VEND', 'BRUM', 'FRIM', 'NIVO', 'PLUV', 'VENT', 'GERM', 'FLOR', 'PRAI', 'MESS', 'THER', 'FRUC', 'COMP'];
127    const jalali_months = ['FARVA', 'ORDIB', 'KHORD', 'TIR', 'MORDA', 'SHAHR', 'MEHR', 'ABAN', 'AZAR', 'DEY', 'BAHMA', 'ESFAN'];
128
129    let datestr = datefield.value;
130    // if a date has a date phrase marked by () this has to be excluded from altering
131    let datearr = datestr.split('(');
132    let datephrase = '';
133    if (datearr.length > 1) {
134      datestr = datearr[0];
135      datephrase = datearr[1];
136    }
137
138    // Gedcom dates are upper case
139    datestr = datestr.toUpperCase();
140    // Gedcom dates have no leading/trailing/repeated whitespace
141    datestr = datestr.replace(/\s+/g, ' ');
142    datestr = datestr.replace(/(^\s)|(\s$)/, '');
143    // Gedcom dates have spaces between letters and digits, e.g. "01JAN2000" => "01 JAN 2000"
144    datestr = datestr.replace(/(\d)([A-Z])/g, '$1 $2');
145    datestr = datestr.replace(/([A-Z])(\d)/g, '$1 $2');
146
147    // Shortcut for quarter format, "Q1 1900" => "BET JAN 1900 AND MAR 1900".
148    if (datestr.match(/^Q ([1-4]) (\d\d\d\d)$/)) {
149      datestr = 'BET ' + months[RegExp.$1 * 3 - 3] + ' ' + RegExp.$2 + ' AND ' + months[RegExp.$1 * 3 - 1] + ' ' + RegExp.$2;
150    }
151
152    // Shortcut for @#Dxxxxx@ 01 01 1400, etc.
153    if (datestr.match(/^(@#DHIJRI@|HIJRI)( \d?\d )(\d?\d)( \d?\d?\d?\d)$/)) {
154      datestr = '@#DHIJRI@' + RegExp.$2 + hijri_months[parseInt(RegExp.$3, 10) - 1] + RegExp.$4;
155    }
156    if (datestr.match(/^(@#DJALALI@|JALALI)( \d?\d )(\d?\d)( \d?\d?\d?\d)$/)) {
157      datestr = '@#DJALALI@' + RegExp.$2 + jalali_months[parseInt(RegExp.$3, 10) - 1] + RegExp.$4;
158    }
159    if (datestr.match(/^(@#DHEBREW@|HEBREW)( \d?\d )(\d?\d)( \d?\d?\d?\d)$/)) {
160      datestr = '@#DHEBREW@' + RegExp.$2 + hebrew_months[parseInt(RegExp.$3, 10) - 1] + RegExp.$4;
161    }
162    if (datestr.match(/^(@#DFRENCH R@|FRENCH)( \d?\d )(\d?\d)( \d?\d?\d?\d)$/)) {
163      datestr = '@#DFRENCH R@' + RegExp.$2 + french_months[parseInt(RegExp.$3, 10) - 1] + RegExp.$4;
164    }
165
166    // All digit dates
167    datestr = datestr.replace(/(\d\d)(\d\d)(\d\d)(\d\d)/g, function () {
168      if (RegExp.$1 > '12' && RegExp.$3 <= '12' && RegExp.$4 <= '31') {
169        return RegExp.$4 + ' ' + months[RegExp.$3 - 1] + ' ' + RegExp.$1 + RegExp.$2;
170      }
171      if (RegExp.$1 <= '31' && RegExp.$2 <= '12' && RegExp.$3 > '12') {
172        return RegExp.$1 + ' ' + months[RegExp.$2 - 1] + ' ' + RegExp.$3 + RegExp.$4;
173      }
174      return RegExp.$1 + RegExp.$2 + RegExp.$3 + RegExp.$4;
175    });
176
177    // e.g. 17.11.1860, 2 4 1987, 3/4/2005, 1999-12-31. Use locale settings since DMY order is ambiguous.
178    datestr = datestr.replace(/(\d+)([ ./-])(\d+)(\2)(\d+)/g, function () {
179      let f1 = parseInt(RegExp.$1, 10);
180      let f2 = parseInt(RegExp.$3, 10);
181      let f3 = parseInt(RegExp.$5, 10);
182      let yyyy = new Date().getFullYear();
183      let yy = yyyy % 100;
184      let cc = yyyy - yy;
185      if ((dmy === 'DMY' || f1 > 13 && f3 > 31) && f1 <= 31 && f2 <= 12) {
186        return f1 + ' ' + months[f2 - 1] + ' ' + (f3 >= 100 ? f3 : (f3 <= yy ? f3 + cc : f3 + cc - 100));
187      }
188      if ((dmy === 'MDY' || f2 > 13 && f3 > 31) && f1 <= 12 && f2 <= 31) {
189        return f2 + ' ' + months[f1 - 1] + ' ' + (f3 >= 100 ? f3 : (f3 <= yy ? f3 + cc : f3 + cc - 100));
190      }
191      if ((dmy === 'YMD' || f1 > 31) && f2 <= 12 && f3 <= 31) {
192        return f3 + ' ' + months[f2 - 1] + ' ' + (f1 >= 100 ? f1 : (f1 <= yy ? f1 + cc : f1 + cc - 100));
193      }
194      return RegExp.$1 + RegExp.$2 + RegExp.$3 + RegExp.$4 + RegExp.$5;
195    });
196
197    datestr = datestr
198      // Shortcuts for date ranges
199      .replace(/^[>]([\w ]+)$/, 'AFT $1')
200      .replace(/^[<]([\w ]+)$/, 'BEF $1')
201      .replace(/^([\w ]+)[-]$/, 'FROM $1')
202      .replace(/^[-]([\w ]+)$/, 'TO $1')
203      .replace(/^[~]([\w ]+)$/, 'ABT $1')
204      .replace(/^[*]([\w ]+)$/, 'EST $1')
205      .replace(/^[#]([\w ]+)$/, 'CAL $1')
206      .replace(/^([\w ]+) ?- ?([\w ]+)$/, 'BET $1 AND $2')
207      .replace(/^([\w ]+) ?~ ?([\w ]+)$/, 'FROM $1 TO $2')
208      // Convert full months to short months
209      .replace(/JANUARY/g, 'JAN')
210      .replace(/FEBRUARY/g, 'FEB')
211      .replace(/MARCH/g, 'MAR')
212      .replace(/APRIL/g, 'APR')
213      .replace(/JUNE/g, 'JUN')
214      .replace(/JULY/g, 'JUL')
215      .replace(/AUGUST/g, 'AUG')
216      .replace(/SEPTEMBER/g, 'SEP')
217      .replace(/OCTOBER/, 'OCT')
218      .replace(/NOVEMBER/g, 'NOV')
219      .replace(/DECEMBER/g, 'DEC')
220      // Americans enter dates as SEP 20, 1999
221      .replace(/(JAN|FEB|MAR|APR|MAY|JUN|JUL|AUG|SEP|OCT|NOV|DEC)\.? (\d\d?)[, ]+(\d\d\d\d)/g, '$2 $1 $3')
222      // Apply leading zero to day numbers
223      .replace(/(^| )(\d [A-Z]{3,5} \d{4})/g, '$10$2');
224
225    if (datephrase) {
226      datestr = datestr + ' (' + datephrase;
227    }
228
229    // Only update it if is has been corrected - otherwise input focus
230    // moves to the end of the field unnecessarily
231    if (datefield.value !== datestr) {
232      datefield.value = datestr;
233    }
234  };
235
236  let monthLabels = [];
237  monthLabels[1] = 'January';
238  monthLabels[2] = 'February';
239  monthLabels[3] = 'March';
240  monthLabels[4] = 'April';
241  monthLabels[5] = 'May';
242  monthLabels[6] = 'June';
243  monthLabels[7] = 'July';
244  monthLabels[8] = 'August';
245  monthLabels[9] = 'September';
246  monthLabels[10] = 'October';
247  monthLabels[11] = 'November';
248  monthLabels[12] = 'December';
249
250  let monthShort = [];
251  monthShort[1] = 'JAN';
252  monthShort[2] = 'FEB';
253  monthShort[3] = 'MAR';
254  monthShort[4] = 'APR';
255  monthShort[5] = 'MAY';
256  monthShort[6] = 'JUN';
257  monthShort[7] = 'JUL';
258  monthShort[8] = 'AUG';
259  monthShort[9] = 'SEP';
260  monthShort[10] = 'OCT';
261  monthShort[11] = 'NOV';
262  monthShort[12] = 'DEC';
263
264  let daysOfWeek = [];
265  daysOfWeek[0] = 'S';
266  daysOfWeek[1] = 'M';
267  daysOfWeek[2] = 'T';
268  daysOfWeek[3] = 'W';
269  daysOfWeek[4] = 'T';
270  daysOfWeek[5] = 'F';
271  daysOfWeek[6] = 'S';
272
273  let weekStart = 0;
274
275  /**
276   * @param {string} jan
277   * @param {string} feb
278   * @param {string} mar
279   * @param {string} apr
280   * @param {string} may
281   * @param {string} jun
282   * @param {string} jul
283   * @param {string} aug
284   * @param {string} sep
285   * @param {string} oct
286   * @param {string} nov
287   * @param {string} dec
288   * @param {string} sun
289   * @param {string} mon
290   * @param {string} tue
291   * @param {string} wed
292   * @param {string} thu
293   * @param {string} fri
294   * @param {string} sat
295   * @param {number} day
296   */
297  webtrees.calLocalize = function (jan, feb, mar, apr, may, jun, jul, aug, sep, oct, nov, dec, sun, mon, tue, wed, thu, fri, sat, day) {
298    monthLabels[1] = jan;
299    monthLabels[2] = feb;
300    monthLabels[3] = mar;
301    monthLabels[4] = apr;
302    monthLabels[5] = may;
303    monthLabels[6] = jun;
304    monthLabels[7] = jul;
305    monthLabels[8] = aug;
306    monthLabels[9] = sep;
307    monthLabels[10] = oct;
308    monthLabels[11] = nov;
309    monthLabels[12] = dec;
310    daysOfWeek[0] = sun;
311    daysOfWeek[1] = mon;
312    daysOfWeek[2] = tue;
313    daysOfWeek[3] = wed;
314    daysOfWeek[4] = thu;
315    daysOfWeek[5] = fri;
316    daysOfWeek[6] = sat;
317
318    if (day >= 0 && day < 7) {
319      weekStart = day;
320    }
321  };
322
323  /**
324   * @param {string} dateDivId
325   * @param {string} dateFieldId
326   * @returns {boolean}
327   */
328  webtrees.calendarWidget = function (dateDivId, dateFieldId) {
329    let dateDiv = document.getElementById(dateDivId);
330    let dateField = document.getElementById(dateFieldId);
331
332    if (dateDiv.style.visibility === 'visible') {
333      dateDiv.style.visibility = 'hidden';
334      return false;
335    }
336    if (dateDiv.style.visibility === 'show') {
337      dateDiv.style.visibility = 'hide';
338      return false;
339    }
340
341    /* Javascript calendar functions only work with precise gregorian dates "D M Y" or "Y" */
342    let greg_regex = /(?:(\d*) ?(JAN|FEB|MAR|APR|MAY|JUN|JUL|AUG|SEP|OCT|NOV|DEC) )?(\d+)/i;
343    let date;
344    if (greg_regex.exec(dateField.value)) {
345      let day   = RegExp.$1 || '1';
346      let month = RegExp.$2 || 'JAN'
347      let year  = RegExp.$3;
348      date = new Date(day + ' ' + month + ' ' + year);
349    } else {
350      date = new Date();
351    }
352
353    dateDiv.innerHTML = calGenerateSelectorContent(dateFieldId, dateDivId, date);
354    if (dateDiv.style.visibility === 'hidden') {
355      dateDiv.style.visibility = 'visible';
356      return false;
357    }
358    if (dateDiv.style.visibility === 'hide') {
359      dateDiv.style.visibility = 'show';
360      return false;
361    }
362
363    return false;
364  };
365
366  /**
367   * @param {string} dateFieldId
368   * @param {string} dateDivId
369   * @param {Date} date
370   * @returns {string}
371   */
372  function calGenerateSelectorContent (dateFieldId, dateDivId, date) {
373    let i, j;
374    let content = '<table border="1"><tr>';
375    content += '<td><select class="form-control" id="' + dateFieldId + '_daySelect" onchange="return webtrees.calUpdateCalendar(\'' + dateFieldId + '\', \'' + dateDivId + '\');">';
376    for (i = 1; i < 32; i++) {
377      content += '<option value="' + i + '"';
378      if (date.getDate() === i) {
379        content += ' selected="selected"';
380      }
381      content += '>' + i + '</option>';
382    }
383    content += '</select></td>';
384    content += '<td><select class="form-control" id="' + dateFieldId + '_monSelect" onchange="return webtrees.calUpdateCalendar(\'' + dateFieldId + '\', \'' + dateDivId + '\');">';
385    for (i = 1; i < 13; i++) {
386      content += '<option value="' + i + '"';
387      if (date.getMonth() + 1 === i) {
388        content += ' selected="selected"';
389      }
390      content += '>' + monthLabels[i] + '</option>';
391    }
392    content += '</select></td>';
393    content += '<td><input class="form-control" type="text" id="' + dateFieldId + '_yearInput" size="5" value="' + date.getFullYear() + '" onchange="return webtrees.calUpdateCalendar(\'' + dateFieldId + '\', \'' + dateDivId + '\');" /></td></tr>';
394    content += '<tr><td colspan="3">';
395    content += '<table width="100%">';
396    content += '<tr>';
397    j = weekStart;
398    for (i = 0; i < 7; i++) {
399      content += '<td ';
400      content += 'class="descriptionbox"';
401      content += '>';
402      content += daysOfWeek[j];
403      content += '</td>';
404      j++;
405      if (j > 6) {
406        j = 0;
407      }
408    }
409    content += '</tr>';
410
411    let tdate = new Date(date.getFullYear(), date.getMonth(), 1);
412    let day = tdate.getDay();
413    day = day - weekStart;
414    let daymilli = 1000 * 60 * 60 * 24;
415    tdate = tdate.getTime() - (day * daymilli) + (daymilli / 2);
416    tdate = new Date(tdate);
417
418    for (j = 0; j < 6; j++) {
419      content += '<tr>';
420      for (i = 0; i < 7; i++) {
421        content += '<td ';
422        if (tdate.getMonth() === date.getMonth()) {
423          if (tdate.getDate() === date.getDate()) {
424            content += 'class="descriptionbox"';
425          } else {
426            content += 'class="optionbox"';
427          }
428        } else {
429          content += 'style="background-color:#EAEAEA; border: solid #AAAAAA 1px;"';
430        }
431        content += '><a href="#" onclick="return webtrees.calDateClicked(\'' + dateFieldId + '\', \'' + dateDivId + '\', ' + tdate.getFullYear() + ', ' + tdate.getMonth() + ', ' + tdate.getDate() + ');">';
432        content += tdate.getDate();
433        content += '</a></td>';
434        let datemilli = tdate.getTime() + daymilli;
435        tdate = new Date(datemilli);
436      }
437      content += '</tr>';
438    }
439    content += '</table>';
440    content += '</td></tr>';
441    content += '</table>';
442
443    return content;
444  }
445
446  /**
447   * @param {string} dateFieldId
448   * @param {number} year
449   * @param {number} month
450   * @param {number} day
451   * @returns {boolean}
452   */
453  function calSetDateField (dateFieldId, year, month, day) {
454    let dateField = document.getElementById(dateFieldId);
455    dateField.value = (day < 10 ? '0' : '') + day + ' ' + monthShort[month + 1] + ' ' + year;
456    return false;
457  }
458
459  /**
460   * @param {string} dateFieldId
461   * @param {string} dateDivId
462   * @returns {boolean}
463   */
464  webtrees.calUpdateCalendar = function (dateFieldId, dateDivId) {
465    let dateSel = document.getElementById(dateFieldId + '_daySelect');
466    if (!dateSel) {
467      return false;
468    }
469    let monthSel = document.getElementById(dateFieldId + '_monSelect');
470    if (!monthSel) {
471      return false;
472    }
473    let yearInput = document.getElementById(dateFieldId + '_yearInput');
474    if (!yearInput) {
475      return false;
476    }
477
478    let month = parseInt(monthSel.options[monthSel.selectedIndex].value, 10);
479    month = month - 1;
480
481    let date = new Date(yearInput.value, month, dateSel.options[dateSel.selectedIndex].value);
482    calSetDateField(dateFieldId, date.getFullYear(), date.getMonth(), date.getDate());
483
484    let dateDiv = document.getElementById(dateDivId);
485    if (!dateDiv) {
486      alert('no dateDiv ' + dateDivId);
487      return false;
488    }
489    dateDiv.innerHTML = calGenerateSelectorContent(dateFieldId, dateDivId, date);
490
491    return false;
492  };
493
494  /**
495   * @param {string} dateFieldId
496   * @param {string} dateDivId
497   * @param {number} year
498   * @param {number} month
499   * @param {number} day
500   * @returns {boolean}
501   */
502  webtrees.calDateClicked = function (dateFieldId, dateDivId, year, month, day) {
503    calSetDateField(dateFieldId, year, month, day);
504    webtrees.calendarWidget(dateDivId, dateFieldId);
505    return false;
506  };
507
508  /**
509   * Persistent checkbox options to hide/show extra data.
510   * @param {HTMLInputElement} element
511   */
512  webtrees.persistentToggle = function (element) {
513    if (element instanceof HTMLInputElement && element.type === 'checkbox') {
514      const key = 'state-of-' + element.dataset.wtPersist;
515      const state = localStorage.getItem(key);
516
517      // Previously selected? Select again now.
518      if (state === 'true') {
519        element.click();
520      }
521
522      // Remember state for the next page load.
523      element.addEventListener('change', function () {
524        localStorage.setItem(key, element.checked.toString());
525      });
526    }
527  };
528
529  /**
530   * @param {Element} field
531   * @param {string} pos
532   * @param {string} neg
533   */
534  function reformatLatLong (field, pos, neg) {
535    // valid LATI or LONG according to Gedcom standard
536    // pos (+) : N or E
537    // neg (-) : S or W
538    let txt = field.value.toUpperCase();
539    txt = txt.replace(/(^\s*)|(\s*$)/g, ''); // trim
540    txt = txt.replace(/ /g, ':'); // N12 34 ==> N12.34
541    txt = txt.replace(/\+/g, ''); // +17.1234 ==> 17.1234
542    txt = txt.replace(/-/g, neg); // -0.5698 ==> W0.5698
543    txt = txt.replace(/,/g, '.'); // 0,5698 ==> 0.5698
544    // 0°34'11 ==> 0:34:11
545    txt = txt.replace(/\u00b0/g, ':'); // °
546    txt = txt.replace(/\u0027/g, ':'); // '
547    // 0:34:11.2W ==> W0.5698
548    txt = txt.replace(/^([0-9]+):([0-9]+):([0-9.]+)(.*)/g, function ($0, $1, $2, $3, $4) {
549      let n = parseFloat($1);
550      n += ($2 / 60);
551      n += ($3 / 3600);
552      n = Math.round(n * 1E4) / 1E4;
553      return $4 + n;
554    });
555    // 0:34W ==> W0.5667
556    txt = txt.replace(/^([0-9]+):([0-9]+)(.*)/g, function ($0, $1, $2, $3) {
557      let n = parseFloat($1);
558      n += ($2 / 60);
559      n = Math.round(n * 1E4) / 1E4;
560      return $3 + n;
561    });
562    // 0.5698W ==> W0.5698
563    txt = txt.replace(/(.*)(NSEW])$/g, '$2$1');
564    // 17.1234 ==> N17.1234
565    if (txt && txt.charAt(0) !== neg && txt.charAt(0) !== pos) {
566      txt = pos + txt;
567    }
568    field.value = txt;
569  }
570
571  /**
572   * @param {Element} field
573   */
574  webtrees.reformatLatitude = function (field) {
575    return reformatLatLong(field, 'N', 'S');
576  };
577
578  /**
579   * @param {Element} field
580   */
581  webtrees.reformatLongitude = function (field) {
582    return reformatLatLong(field, 'E', 'W');
583  };
584
585  /**
586   * Initialize autocomplete elements.
587   * @param {string} selector
588   */
589  webtrees.autocomplete = function (selector) {
590    // Use typeahead/bloodhound for autocomplete
591    $(selector).each(function () {
592      const that = this;
593      $(this).typeahead(null, {
594        display: 'value',
595        limit: 10,
596        minLength: 2,
597        source: new Bloodhound({
598          datumTokenizer: Bloodhound.tokenizers.obj.whitespace('value'),
599          queryTokenizer: Bloodhound.tokenizers.whitespace,
600          remote: {
601            url: this.dataset.wtAutocompleteUrl,
602            replace: function (url, uriEncodedQuery) {
603              const symbol = (url.indexOf("?") > 0) ? '&' : '?';
604              if (that.dataset.wtAutocompleteExtra === 'SOUR') {
605                let row_group = that.closest('.form-group').previousElementSibling;
606                while (row_group.querySelector('select') === null) {
607                  row_group = row_group.previousElementSibling;
608                }
609                const element = row_group.querySelector('select');
610                const extra   = element.options[element.selectedIndex].value.replace(/@/g, '');
611                return url + symbol + "query=" + uriEncodedQuery + '&extra=' + encodeURIComponent(extra);
612              }
613              return url + symbol + "query=" + uriEncodedQuery
614            }
615          }
616        })
617      });
618    });
619  };
620
621  /**
622   * Create a LeafletJS map from a list of providers/layers.
623   * @param {string} id
624   * @param {object} config
625   * @param {function} resetCallback
626   * @returns Map
627   */
628  webtrees.buildLeafletJsMap = function (id, config, resetCallback) {
629    const zoomControl = new L.control.zoom({
630      zoomInTitle: config.i18n.zoomIn,
631      zoomoutTitle: config.i18n.zoomOut,
632    });
633
634    const resetControl = L.Control.extend({
635      options: {
636        position: 'topleft',
637      },
638      onAdd: function (map) {
639        let container = L.DomUtil.create('div', 'leaflet-bar leaflet-control leaflet-control-custom');
640        container.onclick = resetCallback;
641        let reset = config.i18n.reset;
642        let anchor = L.DomUtil.create('a', 'leaflet-control-reset', container);
643        anchor.setAttribute('aria-label', reset);
644        anchor.href = '#';
645        anchor.title = reset;
646        anchor.role = 'button';
647        L.DomEvent.addListener(anchor, 'click', L.DomEvent.preventDefault);
648        let image = L.DomUtil.create('i', 'fas fa-redo', anchor);
649        image.alt = reset;
650
651        return container;
652      },
653    });
654
655    let defaultLayer = null;
656
657    for (let [, provider] of Object.entries(config.mapProviders)) {
658      for (let [, child] of Object.entries(provider.children)) {
659        if ('bingMapsKey' in child) {
660          child.layer = L.tileLayer.bing(child);
661        } else {
662          child.layer = L.tileLayer(child.url, child);
663        }
664        if (provider.default && child.default) {
665          defaultLayer = child.layer;
666        }
667      }
668    }
669
670    if (defaultLayer === null) {
671      console.log('No default map layer defined - using the first one.');
672      let defaultLayer = config.mapProviders[0].children[0].layer;
673    }
674
675
676    // Create the map with all controls and layers
677    return L.map(id, {
678      zoomControl: false,
679    })
680      .addControl(zoomControl)
681      .addControl(new resetControl())
682      .addLayer(defaultLayer)
683      .addControl(L.control.layers.tree(config.mapProviders, null, {
684        closedSymbol: config.icons.expand,
685        openedSymbol: config.icons.collapse,
686      }));
687
688  };
689
690  /**
691   * Initialize a tom-select input
692   * @param {Element} element
693   * @returns {TomSelect}
694   */
695  webtrees.initializeTomSelect = function (element) {
696    if (element.tomselect) {
697      return element.tomselect;
698    }
699
700    let options = {};
701
702    if (element.dataset.url) {
703      let plugins = ['dropdown_input', 'virtual_scroll'];
704
705      if (element.multiple) {
706        plugins.push('remove_button');
707      } else if (!element.required) {
708        plugins.push('clear_button');
709      }
710
711      options = {
712        plugins: plugins,
713        maxOptions: false,
714        render: {
715          item: (data, escape) => '<div>' + data.text + '</div>',
716          option: (data, escape) => '<div>' + data.text + '</div>',
717        },
718        firstUrl: query => element.dataset.url + '&query=' + encodeURIComponent(query),
719        load: function (query, callback) {
720          fetch(this.getUrl(query))
721            .then(response => response.json())
722            .then(json => {
723              if (json.nextUrl !== null) {
724                this.setNextUrl(query, json.nextUrl + '&query=' + encodeURIComponent(query));
725              }
726              callback(json.data);
727            })
728            .catch(callback);
729        },
730      };
731    }
732
733    return new TomSelect(element, options);
734  }
735
736  /**
737   * Reset a tom-select input to have a single selected option
738   * @param {TomSelect} tomSelect
739   * @param {string} value
740   * @param {string} text
741   */
742  webtrees.resetTomSelect = function (tomSelect, value, text) {
743    tomSelect.clear(true);
744    tomSelect.clearOptions();
745    tomSelect.addOption({ value: value, text: text });
746    tomSelect.refreshOptions();
747    tomSelect.addItem(value, true);
748    tomSelect.refreshItems();
749  };
750
751  /**
752   * Toggle the visibility/status of INDI/FAM/SOUR/REPO/OBJE selectors
753   *
754   * @param {Element} select
755   * @param {Element} container
756   */
757  webtrees.initializeIFSRO = function(select, container) {
758    select.addEventListener('change', function () {
759      // Show only the selected selector.
760      console.log(select.value);
761      container.querySelectorAll('.select-record').forEach(element => element.classList.add('d-none'));
762      container.querySelectorAll('.select-' + select.value).forEach(element => element.classList.remove('d-none'));
763      // Enable only the selected selector (so that disabled ones do not get submitted).
764      container.querySelectorAll('.select-record select').forEach(element => {
765        element.disabled = true;
766        element.tomselect.disable();
767      });
768      container.querySelectorAll('.select-' + select.value + ' select').forEach(element => {
769        element.disabled = false;
770        element.tomselect.enable();
771      });
772    });
773  }
774}(window.webtrees = window.webtrees || {}));
775
776// Send the CSRF token on all AJAX requests
777$.ajaxSetup({
778  headers: {
779    'X-CSRF-TOKEN': $('meta[name=csrf]').attr('content')
780  }
781});
782
783/**
784 * Initialisation
785 */
786$(function () {
787  // Page elements that load automatically via AJAX.
788  // This prevents bad robots from crawling resource-intensive pages.
789  $('[data-wt-ajax-url]').each(function () {
790    $(this).load(this.dataset.wtAjaxUrl);
791  });
792
793  // Autocomplete
794  webtrees.autocomplete('input[data-wt-autocomplete-url]');
795
796  document.querySelectorAll('.tom-select').forEach(element => webtrees.initializeTomSelect(element));
797
798  // If we clear the select (using the "X" button), we need an empty value
799  // (rather than no value at all) for (non-multiple) selects with name="array[]"
800  document.querySelectorAll('select.tom-select:not([multiple])')
801    .forEach(function (element) {
802      element.addEventListener('clear', function () {
803        webtrees.resetTomSelect(element.tomselect, '', '');
804      });
805    });
806
807  // Datatables - locale aware sorting
808  $.fn.dataTableExt.oSort['text-asc'] = function (x, y) {
809    return x.localeCompare(y, document.documentElement.lang, { sensitivity: 'base' });
810  };
811  $.fn.dataTableExt.oSort['text-desc'] = function (x, y) {
812    return y.localeCompare(x, document.documentElement.lang, { sensitivity: 'base' });
813  };
814
815  // DataTables - start hidden to prevent FOUC.
816  $('table.datatables').each(function () {
817    $(this).DataTable();
818    $(this).removeClass('d-none');
819  });
820
821  // Save button/checkbox state between pages
822  document.querySelectorAll('[data-wt-persist]')
823    .forEach((element) => webtrees.persistentToggle(element));
824
825  // Activate the on-screen keyboard
826  let osk_focus_element;
827  $('.wt-osk-trigger').click(function () {
828    // When a user clicks the icon, set focus to the corresponding input
829    osk_focus_element = document.getElementById(this.dataset.wtId);
830    osk_focus_element.focus();
831    $('.wt-osk').show();
832  });
833  $('.wt-osk-script-button').change(function () {
834    $('.wt-osk-script').prop('hidden', true);
835    $('.wt-osk-script-' + this.dataset.wtOskScript).prop('hidden', false);
836  });
837  $('.wt-osk-shift-button').click(function () {
838    document.querySelector('.wt-osk-keys').classList.toggle('shifted');
839  });
840  $('.wt-osk-keys').on('click', '.wt-osk-key', function () {
841    let key = $(this).contents().get(0).nodeValue;
842    let shift_state = $('.wt-osk-shift-button').hasClass('active');
843    let shift_key = $('sup', this)[0];
844    if (shift_state && shift_key !== undefined) {
845      key = shift_key.innerText;
846    }
847    webtrees.pasteAtCursor(osk_focus_element, key);
848    if ($('.wt-osk-pin-button').hasClass('active') === false) {
849      $('.wt-osk').hide();
850    }
851    osk_focus_element.dispatchEvent(new Event('input'));
852  });
853
854  $('.wt-osk-close').on('click', function () {
855    $('.wt-osk').hide();
856  });
857
858  // Hide/Show password fields
859  $('input[type=password]').each(function () {
860    $(this).hideShowPassword('infer', true, {
861      states: {
862        shown: {
863          toggle: {
864            content: this.dataset.wtHidePasswordText,
865            attr: {
866              title: this.dataset.wtHidePasswordTitle,
867              'aria-label': this.dataset.wtHidePasswordTitle,
868            }
869          }
870        },
871        hidden: {
872          toggle: {
873            content: this.dataset.wtShowPasswordText,
874            attr: {
875              title: this.dataset.wtShowPasswordTitle,
876              'aria-label': this.dataset.wtShowPasswordTitle,
877            }
878          }
879        }
880      }
881    });
882  });
883});
884
885// Prevent form re-submission via accidental double-click.
886document.addEventListener('submit', function (event) {
887  const form = event.target;
888
889  if (form.reportValidity()) {
890    form.addEventListener('submit', (event) => {
891      if (form.classList.contains('form-is-submitting')) {
892        event.preventDefault();
893      }
894
895      form.classList.add('form-is-submitting');
896    });
897  }
898});
899
900// Convert data-wt-confirm and data-wt-post-url/data-wt-reload-url attributes into useful behavior.
901document.addEventListener('click', (event) => {
902  const target = event.target.closest('a,button');
903
904  if (target === null) {
905    return;
906  }
907
908  if ('wtConfirm' in target.dataset && !confirm(target.dataset.wtConfirm)) {
909    event.preventDefault();
910    return;
911  }
912
913  if ('wtPostUrl' in target.dataset) {
914    const token = document.querySelector('meta[name=csrf]').content;
915
916    fetch(target.dataset.wtPostUrl, {
917      method: 'POST',
918      headers: {
919        'X-CSRF-TOKEN': token,
920        'X-Requested-with': 'XMLHttpRequest',
921      },
922    }).then(() => {
923      if ('wtReloadUrl' in target.dataset) {
924        // Go somewhere else. e.g. the home page after logout.
925        document.location = target.dataset.wtReloadUrl;
926      } else {
927        // Reload the current page. e.g. change language.
928        document.location.reload();
929      }
930    }).catch((error) => {
931      alert(error);
932    });
933  }
934});
935