171239cb6SGreg Roach/** 271239cb6SGreg Roach * webtrees: online genealogy 3063107dbSGreg Roach * Copyright (C) 2020 webtrees development team 471239cb6SGreg Roach * This program is free software: you can redistribute it and/or modify 571239cb6SGreg Roach * it under the terms of the GNU General Public License as published by 671239cb6SGreg Roach * the Free Software Foundation, either version 3 of the License, or 771239cb6SGreg Roach * (at your option) any later version. 871239cb6SGreg Roach * This program is distributed in the hope that it will be useful, 971239cb6SGreg Roach * but WITHOUT ANY WARRANTY; without even the implied warranty of 1071239cb6SGreg Roach * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 1171239cb6SGreg Roach * GNU General Public License for more details. 1271239cb6SGreg Roach * You should have received a copy of the GNU General Public License 1371239cb6SGreg Roach * along with this program. If not, see <http://www.gnu.org/licenses/>. 1471239cb6SGreg Roach */ 1571239cb6SGreg Roach 16efd89170SGreg Roach'use strict'; 1771239cb6SGreg Roach 180d2905f7SGreg Roach(function (webtrees) { 1959e18f0cSGreg Roach const lang = document.documentElement.lang; 2059e18f0cSGreg Roach 210d2905f7SGreg Roach // Identify the script used by some text. 220d2905f7SGreg Roach const scriptRegexes = { 230d2905f7SGreg Roach Han: /[\u3400-\u9FCC]/, 240d2905f7SGreg Roach Grek: /[\u0370-\u03FF]/, 250d2905f7SGreg Roach Cyrl: /[\u0400-\u04FF]/, 260d2905f7SGreg Roach Hebr: /[\u0590-\u05FF]/, 27efd89170SGreg Roach Arab: /[\u0600-\u06FF]/ 280d2905f7SGreg Roach }; 290d2905f7SGreg Roach 3059e18f0cSGreg Roach /** 3159e18f0cSGreg Roach * Tidy the whitespace in a string. 3206fe1eb5SGreg Roach * @param {string} str 3306fe1eb5SGreg Roach * @returns {string} 3459e18f0cSGreg Roach */ 3559e18f0cSGreg Roach function trim (str) { 36efd89170SGreg Roach return str.replace(/\s+/g, ' ').trim(); 3759e18f0cSGreg Roach } 3859e18f0cSGreg Roach 3959e18f0cSGreg Roach /** 4059e18f0cSGreg Roach * Look for non-latin characters in a string. 4106fe1eb5SGreg Roach * @param {string} str 4206fe1eb5SGreg Roach * @returns {string} 4359e18f0cSGreg Roach */ 440d2905f7SGreg Roach webtrees.detectScript = function (str) { 450d2905f7SGreg Roach for (const script in scriptRegexes) { 460d2905f7SGreg Roach if (str.match(scriptRegexes[script])) { 470d2905f7SGreg Roach return script; 480d2905f7SGreg Roach } 4959e18f0cSGreg Roach } 5059e18f0cSGreg Roach 51efd89170SGreg Roach return 'Latn'; 520d2905f7SGreg Roach }; 5359e18f0cSGreg Roach 5459e18f0cSGreg Roach /** 5559e18f0cSGreg Roach * In some languages, the SURN uses a male/default form, but NAME uses a gender-inflected form. 5606fe1eb5SGreg Roach * @param {string} surname 5706fe1eb5SGreg Roach * @param {string} sex 5806fe1eb5SGreg Roach * @returns {string} 5959e18f0cSGreg Roach */ 6059e18f0cSGreg Roach function inflectSurname (surname, sex) { 61efd89170SGreg Roach if (lang === 'pl' && sex === 'F') { 6259e18f0cSGreg Roach return surname 63efd89170SGreg Roach .replace(/ski$/, 'ska') 64efd89170SGreg Roach .replace(/cki$/, 'cka') 65efd89170SGreg Roach .replace(/dzki$/, 'dzka') 66efd89170SGreg Roach .replace(/żki$/, 'żka'); 6759e18f0cSGreg Roach } 6859e18f0cSGreg Roach 6959e18f0cSGreg Roach return surname; 7059e18f0cSGreg Roach } 7159e18f0cSGreg Roach 7259e18f0cSGreg Roach /** 7359e18f0cSGreg Roach * Build a NAME from a NPFX, GIVN, SPFX, SURN and NSFX parts. 7459e18f0cSGreg Roach * Assumes the language of the document is the same as the language of the name. 7506fe1eb5SGreg Roach * @param {string} npfx 7606fe1eb5SGreg Roach * @param {string} givn 7706fe1eb5SGreg Roach * @param {string} spfx 7806fe1eb5SGreg Roach * @param {string} surn 7906fe1eb5SGreg Roach * @param {string} nsfx 8006fe1eb5SGreg Roach * @param {string} sex 8106fe1eb5SGreg Roach * @returns {string} 8259e18f0cSGreg Roach */ 830d2905f7SGreg Roach webtrees.buildNameFromParts = function (npfx, givn, spfx, surn, nsfx, sex) { 84efd89170SGreg Roach const usesCJK = webtrees.detectScript(npfx + givn + spfx + givn + surn + nsfx) === 'Han'; 85efd89170SGreg Roach const separator = usesCJK ? '' : ' '; 8659e18f0cSGreg Roach const surnameFirst = usesCJK || ['hu', 'jp', 'ko', 'vi', 'zh-Hans', 'zh-Hant'].indexOf(lang) !== -1; 8759e18f0cSGreg Roach const patronym = ['is'].indexOf(lang) !== -1; 88efd89170SGreg Roach const slash = patronym ? '' : '/'; 8959e18f0cSGreg Roach 9059e18f0cSGreg Roach // GIVN and SURN may be a comma-separated lists. 9159e18f0cSGreg Roach npfx = trim(npfx); 92063107dbSGreg Roach givn = trim(givn.replace(/,/g, separator)); 9359e18f0cSGreg Roach spfx = trim(spfx); 94063107dbSGreg Roach surn = inflectSurname(trim(surn.replace(/,/g, separator)), sex); 9559e18f0cSGreg Roach nsfx = trim(nsfx); 9659e18f0cSGreg Roach 9759e18f0cSGreg Roach const surname = trim(spfx + separator + surn); 9859e18f0cSGreg Roach 9959e18f0cSGreg Roach const name = surnameFirst ? slash + surname + slash + separator + givn : givn + separator + slash + surname + slash; 10059e18f0cSGreg Roach 10159e18f0cSGreg Roach return trim(npfx + separator + name + separator + nsfx); 1020d2905f7SGreg Roach }; 1030d2905f7SGreg Roach 1040d2905f7SGreg Roach // Insert text at the current cursor position in a text field. 1050d2905f7SGreg Roach webtrees.pasteAtCursor = function (element, text) { 1060d2905f7SGreg Roach if (element !== null) { 1070d2905f7SGreg Roach const caret_pos = element.selectionStart + text.length; 1080d2905f7SGreg Roach const textBefore = element.value.substring(0, element.selectionStart); 1090d2905f7SGreg Roach const textAfter = element.value.substring(element.selectionEnd); 1100d2905f7SGreg Roach element.value = textBefore + text + textAfter; 1110d2905f7SGreg Roach element.setSelectionRange(caret_pos, caret_pos); 1120d2905f7SGreg Roach element.focus(); 11359e18f0cSGreg Roach } 114efd89170SGreg Roach }; 11571239cb6SGreg Roach 11606fe1eb5SGreg Roach /** 1177fb78f8aSGreg Roach * @param {Element} datefield 11806fe1eb5SGreg Roach * @param {string} dmy 11906fe1eb5SGreg Roach */ 120*a7a3d6dbSGreg Roach webtrees.reformatDate = function (datefield, dmy) { 121*a7a3d6dbSGreg Roach const months = ['JAN', 'FEB', 'MAR', 'APR', 'MAY', 'JUN', 'JUL', 'AUG', 'SEP', 'OCT', 'NOV', 'DEC']; 122*a7a3d6dbSGreg Roach const hijri_months = ['MUHAR', 'SAFAR', 'RABIA', 'RABIT', 'JUMAA', 'JUMAT', 'RAJAB', 'SHAAB', 'RAMAD', 'SHAWW', 'DHUAQ', 'DHUAH']; 123*a7a3d6dbSGreg Roach const hebrew_months = ['TSH', 'CSH', 'KSL', 'TVT', 'SHV', 'ADR', 'ADS', 'NSN', 'IYR', 'SVN', 'TMZ', 'AAV', 'ELL']; 124*a7a3d6dbSGreg Roach const french_months = ['VEND', 'BRUM', 'FRIM', 'NIVO', 'PLUV', 'VENT', 'GERM', 'FLOR', 'PRAI', 'MESS', 'THER', 'FRUC', 'COMP']; 125*a7a3d6dbSGreg Roach const jalali_months = ['FARVA', 'ORDIB', 'KHORD', 'TIR', 'MORDA', 'SHAHR', 'MEHR', 'ABAN', 'AZAR', 'DEY', 'BAHMA', 'ESFAN']; 12671239cb6SGreg Roach 127*a7a3d6dbSGreg Roach let datestr = datefield.value; 12871239cb6SGreg Roach // if a date has a date phrase marked by () this has to be excluded from altering 129*a7a3d6dbSGreg Roach let datearr = datestr.split('('); 130*a7a3d6dbSGreg Roach let datephrase = ''; 13171239cb6SGreg Roach if (datearr.length > 1) { 13271239cb6SGreg Roach datestr = datearr[0]; 13371239cb6SGreg Roach datephrase = datearr[1]; 13471239cb6SGreg Roach } 13571239cb6SGreg Roach 13671239cb6SGreg Roach // Gedcom dates are upper case 13771239cb6SGreg Roach datestr = datestr.toUpperCase(); 13871239cb6SGreg Roach // Gedcom dates have no leading/trailing/repeated whitespace 13980d699d6SGreg Roach datestr = datestr.replace(/\s+/g, ' '); 14071239cb6SGreg Roach datestr = datestr.replace(/(^\s)|(\s$)/, ''); 14171239cb6SGreg Roach // Gedcom dates have spaces between letters and digits, e.g. "01JAN2000" => "01 JAN 2000" 14280d699d6SGreg Roach datestr = datestr.replace(/(\d)([A-Z])/g, '$1 $2'); 14380d699d6SGreg Roach datestr = datestr.replace(/([A-Z])(\d)/g, '$1 $2'); 14471239cb6SGreg Roach 14571239cb6SGreg Roach // Shortcut for quarter format, "Q1 1900" => "BET JAN 1900 AND MAR 1900". See [ 1509083 ] 14671239cb6SGreg Roach if (datestr.match(/^Q ([1-4]) (\d\d\d\d)$/)) { 14771239cb6SGreg Roach datestr = 'BET ' + months[RegExp.$1 * 3 - 3] + ' ' + RegExp.$2 + ' AND ' + months[RegExp.$1 * 3 - 1] + ' ' + RegExp.$2; 14871239cb6SGreg Roach } 14971239cb6SGreg Roach 15071239cb6SGreg Roach // Shortcut for @#Dxxxxx@ 01 01 1400, etc. 15171239cb6SGreg Roach if (datestr.match(/^(@#DHIJRI@|HIJRI)( \d?\d )(\d?\d)( \d?\d?\d?\d)$/)) { 15271239cb6SGreg Roach datestr = '@#DHIJRI@' + RegExp.$2 + hijri_months[parseInt(RegExp.$3, 10) - 1] + RegExp.$4; 15371239cb6SGreg Roach } 15471239cb6SGreg Roach if (datestr.match(/^(@#DJALALI@|JALALI)( \d?\d )(\d?\d)( \d?\d?\d?\d)$/)) { 15571239cb6SGreg Roach datestr = '@#DJALALI@' + RegExp.$2 + jalali_months[parseInt(RegExp.$3, 10) - 1] + RegExp.$4; 15671239cb6SGreg Roach } 15771239cb6SGreg Roach if (datestr.match(/^(@#DHEBREW@|HEBREW)( \d?\d )(\d?\d)( \d?\d?\d?\d)$/)) { 15871239cb6SGreg Roach datestr = '@#DHEBREW@' + RegExp.$2 + hebrew_months[parseInt(RegExp.$3, 10) - 1] + RegExp.$4; 15971239cb6SGreg Roach } 16071239cb6SGreg Roach if (datestr.match(/^(@#DFRENCH R@|FRENCH)( \d?\d )(\d?\d)( \d?\d?\d?\d)$/)) { 16171239cb6SGreg Roach datestr = '@#DFRENCH R@' + RegExp.$2 + french_months[parseInt(RegExp.$3, 10) - 1] + RegExp.$4; 16271239cb6SGreg Roach } 16371239cb6SGreg Roach 1647fb78f8aSGreg Roach // All digit dates 1657fb78f8aSGreg Roach datestr = datestr.replaceAll(/(\d\d)(\d\d)(\d\d)(\d\d)/g, function () { 1667fb78f8aSGreg Roach if (RegExp.$1 > '12' && RegExp.$3 <= '12' && RegExp.$4 <= '31') { 1677fb78f8aSGreg Roach return RegExp.$4 + ' ' + months[RegExp.$3 - 1] + ' ' + RegExp.$1 + RegExp.$2; 1687fb78f8aSGreg Roach } 1697fb78f8aSGreg Roach if (RegExp.$1 <= '31' && RegExp.$2 <= '12' && RegExp.$3 > '12') { 1707fb78f8aSGreg Roach return RegExp.$1 + ' ' + months[RegExp.$2 - 1] + ' ' + RegExp.$3 + RegExp.$4; 1717fb78f8aSGreg Roach } 1727fb78f8aSGreg Roach return RegExp.$1 + RegExp.$2 + RegExp.$3 + RegExp.$4; 1737fb78f8aSGreg Roach }); 1747fb78f8aSGreg Roach 1757fb78f8aSGreg Roach // e.g. 17.11.1860, 3/4/2005 or 1999-12-31. Use locale settings since DMY order is ambiguous. 1767fb78f8aSGreg Roach datestr = datestr.replaceAll(/(\d+)([./-])(\d+)([./-])(\d+)/g, function () { 1777fb78f8aSGreg Roach var f1 = parseInt(RegExp.$1, 10); 17871239cb6SGreg Roach var f2 = parseInt(RegExp.$3, 10); 1797fb78f8aSGreg Roach var f3 = parseInt(RegExp.$5, 10); 18071239cb6SGreg Roach var yyyy = new Date().getFullYear(); 18171239cb6SGreg Roach var yy = yyyy % 100; 18271239cb6SGreg Roach var cc = yyyy - yy; 18371239cb6SGreg Roach if (dmy === 'DMY' && f1 <= 31 && f2 <= 12 || f1 > 13 && f1 <= 31 && f2 <= 12 && f3 > 31) { 1847fb78f8aSGreg Roach return f1 + ' ' + months[f2 - 1] + ' ' + (f3 >= 100 ? f3 : (f3 <= yy ? f3 + cc : f3 + cc - 100)); 1857fb78f8aSGreg Roach } 18671239cb6SGreg Roach if (dmy === 'MDY' && f1 <= 12 && f2 <= 31 || f2 > 13 && f2 <= 31 && f1 <= 12 && f3 > 31) { 1877fb78f8aSGreg Roach return f2 + ' ' + months[f1 - 1] + ' ' + (f3 >= 100 ? f3 : (f3 <= yy ? f3 + cc : f3 + cc - 100)); 1887fb78f8aSGreg Roach } 18971239cb6SGreg Roach if (dmy === 'YMD' && f2 <= 12 && f3 <= 31 || f3 > 13 && f3 <= 31 && f2 <= 12 && f1 > 31) { 1907fb78f8aSGreg Roach return f3 + ' ' + months[f2 - 1] + ' ' + (f1 >= 100 ? f1 : (f1 <= yy ? f1 + cc : f1 + cc - 100)); 19171239cb6SGreg Roach } 1927fb78f8aSGreg Roach return RegExp.$1 + RegExp.$2 + RegExp.$3 + RegExp.$4 + RegExp.$5; 1937fb78f8aSGreg Roach }); 19471239cb6SGreg Roach 195*a7a3d6dbSGreg Roach datestr = datestr 19671239cb6SGreg Roach // Shortcuts for date ranges 197*a7a3d6dbSGreg Roach .replace(/^[>]([\w ]+)$/, 'AFT $1') 198*a7a3d6dbSGreg Roach .replace(/^[<]([\w ]+)$/, 'BEF $1') 199*a7a3d6dbSGreg Roach .replace(/^([\w ]+)[-]$/, 'FROM $1') 200*a7a3d6dbSGreg Roach .replace(/^[-]([\w ]+)$/, 'TO $1') 201*a7a3d6dbSGreg Roach .replace(/^[~]([\w ]+)$/, 'ABT $1') 202*a7a3d6dbSGreg Roach .replace(/^[*]([\w ]+)$/, 'EST $1') 203*a7a3d6dbSGreg Roach .replace(/^[#]([\w ]+)$/, 'CAL $1') 204*a7a3d6dbSGreg Roach .replace(/^([\w ]+) ?- ?([\w ]+)$/, 'BET $1 AND $2') 205*a7a3d6dbSGreg Roach .replace(/^([\w ]+) ?~ ?([\w ]+)$/, 'FROM $1 TO $2') 20671239cb6SGreg Roach // Convert full months to short months 207*a7a3d6dbSGreg Roach .replaceAll('JANUARY', 'JAN') 208*a7a3d6dbSGreg Roach .replaceAll('FEBRUARY', 'FEB') 209*a7a3d6dbSGreg Roach .replaceAll('MARCH', 'MAR') 210*a7a3d6dbSGreg Roach .replaceAll('APRIL', 'APR') 211*a7a3d6dbSGreg Roach .replaceAll('JUNE', 'JUN') 212*a7a3d6dbSGreg Roach .replaceAll('JULY', 'JUL') 213*a7a3d6dbSGreg Roach .replaceAll('AUGUST', 'AUG') 214*a7a3d6dbSGreg Roach .replaceAll('SEPTEMBER', 'SEP') 215*a7a3d6dbSGreg Roach .replaceAll('OCTOBER', 'OCT') 216*a7a3d6dbSGreg Roach .replaceAll('NOVEMBER', 'NOV') 217*a7a3d6dbSGreg Roach .replaceAll('DECEMBER', 'DEC') 218*a7a3d6dbSGreg Roach // Americans enter dates as SEP 20, 1999 219*a7a3d6dbSGreg Roach .replaceAll(/(JAN|FEB|MAR|APR|MAY|JUN|JUL|AUG|SEP|OCT|NOV|DEC)\.? (\d\d?)[, ]+(\d\d\d\d)/, '$2 $1 $3') 22071239cb6SGreg Roach // Apply leading zero to day numbers 221*a7a3d6dbSGreg Roach .replaceAll(/(^| )(\d [A-Z]{3,5} \d{4})/, '$10$2'); 22271239cb6SGreg Roach 22371239cb6SGreg Roach if (datephrase) { 22471239cb6SGreg Roach datestr = datestr + ' (' + datephrase; 22571239cb6SGreg Roach } 226*a7a3d6dbSGreg Roach 22771239cb6SGreg Roach // Only update it if is has been corrected - otherwise input focus 22871239cb6SGreg Roach // moves to the end of the field unnecessarily 22971239cb6SGreg Roach if (datefield.value !== datestr) { 23071239cb6SGreg Roach datefield.value = datestr; 23171239cb6SGreg Roach } 23271239cb6SGreg Roach } 233*a7a3d6dbSGreg Roach}(window.webtrees = window.webtrees || {})); 23471239cb6SGreg Roach 23571239cb6SGreg Roachvar monthLabels = []; 23671239cb6SGreg RoachmonthLabels[1] = 'January'; 23771239cb6SGreg RoachmonthLabels[2] = 'February'; 23871239cb6SGreg RoachmonthLabels[3] = 'March'; 23971239cb6SGreg RoachmonthLabels[4] = 'April'; 24071239cb6SGreg RoachmonthLabels[5] = 'May'; 24171239cb6SGreg RoachmonthLabels[6] = 'June'; 24271239cb6SGreg RoachmonthLabels[7] = 'July'; 24371239cb6SGreg RoachmonthLabels[8] = 'August'; 24471239cb6SGreg RoachmonthLabels[9] = 'September'; 24571239cb6SGreg RoachmonthLabels[10] = 'October'; 24671239cb6SGreg RoachmonthLabels[11] = 'November'; 24771239cb6SGreg RoachmonthLabels[12] = 'December'; 24871239cb6SGreg Roach 24971239cb6SGreg Roachvar monthShort = []; 25071239cb6SGreg RoachmonthShort[1] = 'JAN'; 25171239cb6SGreg RoachmonthShort[2] = 'FEB'; 25271239cb6SGreg RoachmonthShort[3] = 'MAR'; 25371239cb6SGreg RoachmonthShort[4] = 'APR'; 25471239cb6SGreg RoachmonthShort[5] = 'MAY'; 25571239cb6SGreg RoachmonthShort[6] = 'JUN'; 25671239cb6SGreg RoachmonthShort[7] = 'JUL'; 25771239cb6SGreg RoachmonthShort[8] = 'AUG'; 25871239cb6SGreg RoachmonthShort[9] = 'SEP'; 25971239cb6SGreg RoachmonthShort[10] = 'OCT'; 26071239cb6SGreg RoachmonthShort[11] = 'NOV'; 26171239cb6SGreg RoachmonthShort[12] = 'DEC'; 26271239cb6SGreg Roach 26371239cb6SGreg Roachvar daysOfWeek = []; 26471239cb6SGreg RoachdaysOfWeek[0] = 'S'; 26571239cb6SGreg RoachdaysOfWeek[1] = 'M'; 26671239cb6SGreg RoachdaysOfWeek[2] = 'T'; 26771239cb6SGreg RoachdaysOfWeek[3] = 'W'; 26871239cb6SGreg RoachdaysOfWeek[4] = 'T'; 26971239cb6SGreg RoachdaysOfWeek[5] = 'F'; 27071239cb6SGreg RoachdaysOfWeek[6] = 'S'; 27171239cb6SGreg Roach 27271239cb6SGreg Roachvar weekStart = 0; 27371239cb6SGreg Roach 27406fe1eb5SGreg Roach/** 27506fe1eb5SGreg Roach * @param {string} jan 27606fe1eb5SGreg Roach * @param {string} feb 27706fe1eb5SGreg Roach * @param {string} mar 27806fe1eb5SGreg Roach * @param {string} apr 27906fe1eb5SGreg Roach * @param {string} may 28006fe1eb5SGreg Roach * @param {string} jun 28106fe1eb5SGreg Roach * @param {string} jul 28206fe1eb5SGreg Roach * @param {string} aug 28306fe1eb5SGreg Roach * @param {string} sep 28406fe1eb5SGreg Roach * @param {string} oct 28506fe1eb5SGreg Roach * @param {string} nov 28606fe1eb5SGreg Roach * @param {string} dec 28706fe1eb5SGreg Roach */ 288efd89170SGreg Roachfunction cal_setMonthNames (jan, feb, mar, apr, may, jun, jul, aug, sep, oct, nov, dec) { 28971239cb6SGreg Roach monthLabels[1] = jan; 29071239cb6SGreg Roach monthLabels[2] = feb; 29171239cb6SGreg Roach monthLabels[3] = mar; 29271239cb6SGreg Roach monthLabels[4] = apr; 29371239cb6SGreg Roach monthLabels[5] = may; 29471239cb6SGreg Roach monthLabels[6] = jun; 29571239cb6SGreg Roach monthLabels[7] = jul; 29671239cb6SGreg Roach monthLabels[8] = aug; 29771239cb6SGreg Roach monthLabels[9] = sep; 29871239cb6SGreg Roach monthLabels[10] = oct; 29971239cb6SGreg Roach monthLabels[11] = nov; 30071239cb6SGreg Roach monthLabels[12] = dec; 30171239cb6SGreg Roach} 30271239cb6SGreg Roach 30306fe1eb5SGreg Roach/** 30406fe1eb5SGreg Roach * @param {string} sun 30506fe1eb5SGreg Roach * @param {string} mon 30606fe1eb5SGreg Roach * @param {string} tue 30706fe1eb5SGreg Roach * @param {string} wed 30806fe1eb5SGreg Roach * @param {string} thu 30906fe1eb5SGreg Roach * @param {string} fri 31006fe1eb5SGreg Roach * @param {string} sat 31106fe1eb5SGreg Roach */ 312efd89170SGreg Roachfunction cal_setDayHeaders (sun, mon, tue, wed, thu, fri, sat) { 31371239cb6SGreg Roach daysOfWeek[0] = sun; 31471239cb6SGreg Roach daysOfWeek[1] = mon; 31571239cb6SGreg Roach daysOfWeek[2] = tue; 31671239cb6SGreg Roach daysOfWeek[3] = wed; 31771239cb6SGreg Roach daysOfWeek[4] = thu; 31871239cb6SGreg Roach daysOfWeek[5] = fri; 31971239cb6SGreg Roach daysOfWeek[6] = sat; 32071239cb6SGreg Roach} 32171239cb6SGreg Roach 32206fe1eb5SGreg Roach/** 32306fe1eb5SGreg Roach * @param {number} day 32406fe1eb5SGreg Roach */ 325efd89170SGreg Roachfunction cal_setWeekStart (day) { 32671239cb6SGreg Roach if (day >= 0 && day < 7) { 32771239cb6SGreg Roach weekStart = day; 32871239cb6SGreg Roach } 32971239cb6SGreg Roach} 33071239cb6SGreg Roach 33106fe1eb5SGreg Roach/** 33206fe1eb5SGreg Roach * @param {string} dateDivId 33306fe1eb5SGreg Roach * @param {string} dateFieldId 33406fe1eb5SGreg Roach * @returns {boolean} 33506fe1eb5SGreg Roach */ 336efd89170SGreg Roachfunction calendarWidget (dateDivId, dateFieldId) { 33771239cb6SGreg Roach var dateDiv = document.getElementById(dateDivId); 33871239cb6SGreg Roach var dateField = document.getElementById(dateFieldId); 33971239cb6SGreg Roach 34071239cb6SGreg Roach if (dateDiv.style.visibility === 'visible') { 34171239cb6SGreg Roach dateDiv.style.visibility = 'hidden'; 34271239cb6SGreg Roach return false; 34371239cb6SGreg Roach } 34471239cb6SGreg Roach if (dateDiv.style.visibility === 'show') { 34571239cb6SGreg Roach dateDiv.style.visibility = 'hide'; 34671239cb6SGreg Roach return false; 34771239cb6SGreg Roach } 34871239cb6SGreg Roach 34971239cb6SGreg Roach /* Javascript calendar functions only work with precise gregorian dates "D M Y" or "Y" */ 35071239cb6SGreg Roach var greg_regex = /((\d+ (JAN|FEB|MAR|APR|MAY|JUN|JUL|AUG|SEP|OCT|NOV|DEC) )?\d+)/i; 35171239cb6SGreg Roach var date; 35271239cb6SGreg Roach if (greg_regex.exec(dateField.value)) { 35371239cb6SGreg Roach date = new Date(RegExp.$1); 35471239cb6SGreg Roach } else { 35571239cb6SGreg Roach date = new Date(); 35671239cb6SGreg Roach } 35771239cb6SGreg Roach 35871239cb6SGreg Roach dateDiv.innerHTML = cal_generateSelectorContent(dateFieldId, dateDivId, date); 35971239cb6SGreg Roach if (dateDiv.style.visibility === 'hidden') { 36071239cb6SGreg Roach dateDiv.style.visibility = 'visible'; 36171239cb6SGreg Roach return false; 36271239cb6SGreg Roach } 36371239cb6SGreg Roach if (dateDiv.style.visibility === 'hide') { 36471239cb6SGreg Roach dateDiv.style.visibility = 'show'; 36571239cb6SGreg Roach return false; 36671239cb6SGreg Roach } 36771239cb6SGreg Roach 36871239cb6SGreg Roach return false; 36971239cb6SGreg Roach} 37071239cb6SGreg Roach 37106fe1eb5SGreg Roach/** 37206fe1eb5SGreg Roach * @param {string} dateFieldId 37306fe1eb5SGreg Roach * @param {string} dateDivId 37406fe1eb5SGreg Roach * @param {Date} date 37506fe1eb5SGreg Roach * @returns {string} 37606fe1eb5SGreg Roach */ 377efd89170SGreg Roachfunction cal_generateSelectorContent (dateFieldId, dateDivId, date) { 37871239cb6SGreg Roach var i, j; 37971239cb6SGreg Roach var content = '<table border="1"><tr>'; 38071239cb6SGreg Roach content += '<td><select class="form-control" id="' + dateFieldId + '_daySelect" onchange="return cal_updateCalendar(\'' + dateFieldId + '\', \'' + dateDivId + '\');">'; 38171239cb6SGreg Roach for (i = 1; i < 32; i++) { 38271239cb6SGreg Roach content += '<option value="' + i + '"'; 38371239cb6SGreg Roach if (date.getDate() === i) { 38471239cb6SGreg Roach content += ' selected="selected"'; 38571239cb6SGreg Roach } 38671239cb6SGreg Roach content += '>' + i + '</option>'; 38771239cb6SGreg Roach } 38871239cb6SGreg Roach content += '</select></td>'; 38971239cb6SGreg Roach content += '<td><select class="form-control" id="' + dateFieldId + '_monSelect" onchange="return cal_updateCalendar(\'' + dateFieldId + '\', \'' + dateDivId + '\');">'; 39071239cb6SGreg Roach for (i = 1; i < 13; i++) { 39171239cb6SGreg Roach content += '<option value="' + i + '"'; 39271239cb6SGreg Roach if (date.getMonth() + 1 === i) { 39371239cb6SGreg Roach content += ' selected="selected"'; 39471239cb6SGreg Roach } 39571239cb6SGreg Roach content += '>' + monthLabels[i] + '</option>'; 39671239cb6SGreg Roach } 39771239cb6SGreg Roach content += '</select></td>'; 39871239cb6SGreg Roach content += '<td><input class="form-control" type="text" id="' + dateFieldId + '_yearInput" size="5" value="' + date.getFullYear() + '" onchange="return cal_updateCalendar(\'' + dateFieldId + '\', \'' + dateDivId + '\');" /></td></tr>'; 39971239cb6SGreg Roach content += '<tr><td colspan="3">'; 40071239cb6SGreg Roach content += '<table width="100%">'; 40171239cb6SGreg Roach content += '<tr>'; 40271239cb6SGreg Roach j = weekStart; 40371239cb6SGreg Roach for (i = 0; i < 7; i++) { 40471239cb6SGreg Roach content += '<td '; 40571239cb6SGreg Roach content += 'class="descriptionbox"'; 40671239cb6SGreg Roach content += '>'; 40771239cb6SGreg Roach content += daysOfWeek[j]; 40871239cb6SGreg Roach content += '</td>'; 40971239cb6SGreg Roach j++; 41071239cb6SGreg Roach if (j > 6) { 41171239cb6SGreg Roach j = 0; 41271239cb6SGreg Roach } 41371239cb6SGreg Roach } 41471239cb6SGreg Roach content += '</tr>'; 41571239cb6SGreg Roach 41671239cb6SGreg Roach var tdate = new Date(date.getFullYear(), date.getMonth(), 1); 41771239cb6SGreg Roach var day = tdate.getDay(); 41871239cb6SGreg Roach day = day - weekStart; 41971239cb6SGreg Roach var daymilli = 1000 * 60 * 60 * 24; 42071239cb6SGreg Roach tdate = tdate.getTime() - (day * daymilli) + (daymilli / 2); 42171239cb6SGreg Roach tdate = new Date(tdate); 42271239cb6SGreg Roach 42371239cb6SGreg Roach for (j = 0; j < 6; j++) { 42471239cb6SGreg Roach content += '<tr>'; 42571239cb6SGreg Roach for (i = 0; i < 7; i++) { 42671239cb6SGreg Roach content += '<td '; 42771239cb6SGreg Roach if (tdate.getMonth() === date.getMonth()) { 42871239cb6SGreg Roach if (tdate.getDate() === date.getDate()) { 42971239cb6SGreg Roach content += 'class="descriptionbox"'; 43071239cb6SGreg Roach } else { 43171239cb6SGreg Roach content += 'class="optionbox"'; 43271239cb6SGreg Roach } 43371239cb6SGreg Roach } else { 43471239cb6SGreg Roach content += 'style="background-color:#EAEAEA; border: solid #AAAAAA 1px;"'; 43571239cb6SGreg Roach } 43671239cb6SGreg Roach content += '><a href="#" onclick="return cal_dateClicked(\'' + dateFieldId + '\', \'' + dateDivId + '\', ' + tdate.getFullYear() + ', ' + tdate.getMonth() + ', ' + tdate.getDate() + ');">'; 43771239cb6SGreg Roach content += tdate.getDate(); 43871239cb6SGreg Roach content += '</a></td>'; 43971239cb6SGreg Roach var datemilli = tdate.getTime() + daymilli; 44071239cb6SGreg Roach tdate = new Date(datemilli); 44171239cb6SGreg Roach } 44271239cb6SGreg Roach content += '</tr>'; 44371239cb6SGreg Roach } 44471239cb6SGreg Roach content += '</table>'; 44571239cb6SGreg Roach content += '</td></tr>'; 44671239cb6SGreg Roach content += '</table>'; 44771239cb6SGreg Roach 44871239cb6SGreg Roach return content; 44971239cb6SGreg Roach} 45071239cb6SGreg Roach 45106fe1eb5SGreg Roach/** 45206fe1eb5SGreg Roach * @param {string} dateFieldId 45306fe1eb5SGreg Roach * @param {number} year 45406fe1eb5SGreg Roach * @param {number} month 45506fe1eb5SGreg Roach * @param {number} day 45606fe1eb5SGreg Roach * @returns {boolean} 45706fe1eb5SGreg Roach */ 458efd89170SGreg Roachfunction cal_setDateField (dateFieldId, year, month, day) { 45971239cb6SGreg Roach var dateField = document.getElementById(dateFieldId); 46071239cb6SGreg Roach if (!dateField) { 46171239cb6SGreg Roach return false; 46271239cb6SGreg Roach } 46371239cb6SGreg Roach if (day < 10) { 46471239cb6SGreg Roach day = '0' + day; 46571239cb6SGreg Roach } 46671239cb6SGreg Roach dateField.value = day + ' ' + monthShort[month + 1] + ' ' + year; 46771239cb6SGreg Roach return false; 46871239cb6SGreg Roach} 46971239cb6SGreg Roach 47006fe1eb5SGreg Roach/** 47106fe1eb5SGreg Roach * @param {string} dateFieldId 47206fe1eb5SGreg Roach * @param {string} dateDivId 47306fe1eb5SGreg Roach * @returns {boolean} 47406fe1eb5SGreg Roach */ 475efd89170SGreg Roachfunction cal_updateCalendar (dateFieldId, dateDivId) { 47671239cb6SGreg Roach var dateSel = document.getElementById(dateFieldId + '_daySelect'); 47771239cb6SGreg Roach if (!dateSel) { 47871239cb6SGreg Roach return false; 47971239cb6SGreg Roach } 48071239cb6SGreg Roach var monthSel = document.getElementById(dateFieldId + '_monSelect'); 48171239cb6SGreg Roach if (!monthSel) { 48271239cb6SGreg Roach return false; 48371239cb6SGreg Roach } 48471239cb6SGreg Roach var yearInput = document.getElementById(dateFieldId + '_yearInput'); 48571239cb6SGreg Roach if (!yearInput) { 48671239cb6SGreg Roach return false; 48771239cb6SGreg Roach } 48871239cb6SGreg Roach 48971239cb6SGreg Roach var month = parseInt(monthSel.options[monthSel.selectedIndex].value, 10); 49071239cb6SGreg Roach month = month - 1; 49171239cb6SGreg Roach 49271239cb6SGreg Roach var date = new Date(yearInput.value, month, dateSel.options[dateSel.selectedIndex].value); 49371239cb6SGreg Roach cal_setDateField(dateFieldId, date.getFullYear(), date.getMonth(), date.getDate()); 49471239cb6SGreg Roach 49571239cb6SGreg Roach var dateDiv = document.getElementById(dateDivId); 49671239cb6SGreg Roach if (!dateDiv) { 49771239cb6SGreg Roach alert('no dateDiv ' + dateDivId); 49871239cb6SGreg Roach return false; 49971239cb6SGreg Roach } 50071239cb6SGreg Roach dateDiv.innerHTML = cal_generateSelectorContent(dateFieldId, dateDivId, date); 50171239cb6SGreg Roach 50271239cb6SGreg Roach return false; 50371239cb6SGreg Roach} 50471239cb6SGreg Roach 50506fe1eb5SGreg Roach/** 50606fe1eb5SGreg Roach * @param {string} dateFieldId 50706fe1eb5SGreg Roach * @param {string} dateDivId 50806fe1eb5SGreg Roach * @param {number} year 50906fe1eb5SGreg Roach * @param {number} month 51006fe1eb5SGreg Roach * @param {number} day 51106fe1eb5SGreg Roach * @returns {boolean} 51206fe1eb5SGreg Roach */ 513efd89170SGreg Roachfunction cal_dateClicked (dateFieldId, dateDivId, year, month, day) { 51471239cb6SGreg Roach cal_setDateField(dateFieldId, year, month, day); 51571239cb6SGreg Roach calendarWidget(dateDivId, dateFieldId); 51671239cb6SGreg Roach return false; 51771239cb6SGreg Roach} 51871239cb6SGreg Roach 51906fe1eb5SGreg Roach/** 52071239cb6SGreg Roach * Persistant checkbox options to hide/show extra data. 52164490ee2SGreg Roach * @param element_id 52271239cb6SGreg Roach */ 523efd89170SGreg Roachfunction persistent_toggle (element_id) { 524efd89170SGreg Roach const element = document.getElementById(element_id); 525efd89170SGreg Roach const key = 'state-of-' + element_id; 526efd89170SGreg Roach const state = localStorage.getItem(key); 52771239cb6SGreg Roach 52864490ee2SGreg Roach // Previously selected? 52964490ee2SGreg Roach if (state === 'true') { 53064490ee2SGreg Roach $(element).click(); 53171239cb6SGreg Roach } 53271239cb6SGreg Roach 53364490ee2SGreg Roach // Remember state for the next page load. 53464490ee2SGreg Roach $(element).on('change', function () { localStorage.setItem(key, element.checked); }); 53571239cb6SGreg Roach} 53671239cb6SGreg Roach 53706fe1eb5SGreg Roach/** 53806fe1eb5SGreg Roach * @param {string} field 53906fe1eb5SGreg Roach * @param {string} pos 54006fe1eb5SGreg Roach * @param {string} neg 54106fe1eb5SGreg Roach */ 542efd89170SGreg Roachfunction valid_lati_long (field, pos, neg) { 54371239cb6SGreg Roach // valid LATI or LONG according to Gedcom standard 54471239cb6SGreg Roach // pos (+) : N or E 54571239cb6SGreg Roach // neg (-) : S or W 54671239cb6SGreg Roach var txt = field.value.toUpperCase(); 54771239cb6SGreg Roach txt = txt.replace(/(^\s*)|(\s*$)/g, ''); // trim 54871239cb6SGreg Roach txt = txt.replace(/ /g, ':'); // N12 34 ==> N12.34 54971239cb6SGreg Roach txt = txt.replace(/\+/g, ''); // +17.1234 ==> 17.1234 55071239cb6SGreg Roach txt = txt.replace(/-/g, neg); // -0.5698 ==> W0.5698 55171239cb6SGreg Roach txt = txt.replace(/,/g, '.'); // 0,5698 ==> 0.5698 55271239cb6SGreg Roach // 0°34'11 ==> 0:34:11 55371239cb6SGreg Roach txt = txt.replace(/\u00b0/g, ':'); // ° 55471239cb6SGreg Roach txt = txt.replace(/\u0027/g, ':'); // ' 55571239cb6SGreg Roach // 0:34:11.2W ==> W0.5698 55671239cb6SGreg Roach txt = txt.replace(/^([0-9]+):([0-9]+):([0-9.]+)(.*)/g, function ($0, $1, $2, $3, $4) { 55771239cb6SGreg Roach var n = parseFloat($1); 55871239cb6SGreg Roach n += ($2 / 60); 55971239cb6SGreg Roach n += ($3 / 3600); 56071239cb6SGreg Roach n = Math.round(n * 1E4) / 1E4; 56171239cb6SGreg Roach return $4 + n; 56271239cb6SGreg Roach }); 56371239cb6SGreg Roach // 0:34W ==> W0.5667 56471239cb6SGreg Roach txt = txt.replace(/^([0-9]+):([0-9]+)(.*)/g, function ($0, $1, $2, $3) { 56571239cb6SGreg Roach var n = parseFloat($1); 56671239cb6SGreg Roach n += ($2 / 60); 56771239cb6SGreg Roach n = Math.round(n * 1E4) / 1E4; 56871239cb6SGreg Roach return $3 + n; 56971239cb6SGreg Roach }); 57071239cb6SGreg Roach // 0.5698W ==> W0.5698 57171239cb6SGreg Roach txt = txt.replace(/(.*)([N|S|E|W]+)$/g, '$2$1'); 57271239cb6SGreg Roach // 17.1234 ==> N17.1234 57371239cb6SGreg Roach if (txt && txt.charAt(0) !== neg && txt.charAt(0) !== pos) { 57471239cb6SGreg Roach txt = pos + txt; 57571239cb6SGreg Roach } 57671239cb6SGreg Roach field.value = txt; 57771239cb6SGreg Roach} 57871239cb6SGreg Roach 57906fe1eb5SGreg Roach/** 58006fe1eb5SGreg Roach * Initialize autocomplete elements. 58106fe1eb5SGreg Roach * @param {string} selector 58206fe1eb5SGreg Roach */ 583efd89170SGreg Roachfunction autocomplete (selector) { 58471239cb6SGreg Roach // Use typeahead/bloodhound for autocomplete 58571239cb6SGreg Roach $(selector).each(function () { 586efd89170SGreg Roach const that = this; 58771239cb6SGreg Roach $(this).typeahead(null, { 58871239cb6SGreg Roach display: 'value', 5895e6816beSGreg Roach limit: 0, 59071239cb6SGreg Roach source: new Bloodhound({ 59171239cb6SGreg Roach datumTokenizer: Bloodhound.tokenizers.obj.whitespace('value'), 59271239cb6SGreg Roach queryTokenizer: Bloodhound.tokenizers.whitespace, 59371239cb6SGreg Roach remote: { 59471239cb6SGreg Roach url: this.dataset.autocompleteUrl, 595f4abaf12SGreg Roach replace: function (url, uriEncodedQuery) { 596f4abaf12SGreg Roach if (that.dataset.autocompleteExtra) { 597efd89170SGreg Roach const extra = $(document.querySelector(that.dataset.autocompleteExtra)).val(); 598efd89170SGreg Roach return url.replace('QUERY', uriEncodedQuery) + '&extra=' + encodeURIComponent(extra); 599f4abaf12SGreg Roach } 600efd89170SGreg Roach return url.replace('QUERY', uriEncodedQuery); 601f4abaf12SGreg Roach }, 602efd89170SGreg Roach wildcard: 'QUERY' 603f4abaf12SGreg Roach 60471239cb6SGreg Roach } 60571239cb6SGreg Roach }) 60671239cb6SGreg Roach }); 60771239cb6SGreg Roach }); 60871239cb6SGreg Roach} 60971239cb6SGreg Roach 61071239cb6SGreg Roach/** 61171239cb6SGreg Roach * Insert text at the current cursor position in an input field. 61206fe1eb5SGreg Roach * @param {Element} e The input element. 61306fe1eb5SGreg Roach * @param {string} t The text to insert. 61471239cb6SGreg Roach */ 615efd89170SGreg Roachfunction insertTextAtCursor (e, t) { 61671239cb6SGreg Roach var scrollTop = e.scrollTop; 61771239cb6SGreg Roach var selectionStart = e.selectionStart; 61871239cb6SGreg Roach var prefix = e.value.substring(0, selectionStart); 61971239cb6SGreg Roach var suffix = e.value.substring(e.selectionEnd, e.value.length); 62071239cb6SGreg Roach e.value = prefix + t + suffix; 62171239cb6SGreg Roach e.selectionStart = selectionStart + t.length; 62271239cb6SGreg Roach e.selectionEnd = e.selectionStart; 62371239cb6SGreg Roach e.focus(); 62471239cb6SGreg Roach e.scrollTop = scrollTop; 62571239cb6SGreg Roach} 62671239cb6SGreg Roach 62771239cb6SGreg Roach// Send the CSRF token on all AJAX requests 62871239cb6SGreg Roach$.ajaxSetup({ 62971239cb6SGreg Roach headers: { 63071239cb6SGreg Roach 'X-CSRF-TOKEN': $('meta[name=csrf]').attr('content') 63171239cb6SGreg Roach } 63271239cb6SGreg Roach}); 63371239cb6SGreg Roach 63406fe1eb5SGreg Roach/** 63506fe1eb5SGreg Roach * Initialisation 63606fe1eb5SGreg Roach */ 63771239cb6SGreg Roach$(function () { 63871239cb6SGreg Roach // Page elements that load automaticaly via AJAX. 63971239cb6SGreg Roach // This prevents bad robots from crawling resource-intensive pages. 640efd89170SGreg Roach $('[data-ajax-url]').each(function () { 64171239cb6SGreg Roach $(this).load($(this).data('ajaxUrl')); 64271239cb6SGreg Roach }); 64371239cb6SGreg Roach 64406fe1eb5SGreg Roach /** 64506fe1eb5SGreg Roach * Select2 - format entries in the select list 64606fe1eb5SGreg Roach * @param {Object} data 64706fe1eb5SGreg Roach * @returns {string} 64806fe1eb5SGreg Roach */ 649efd89170SGreg Roach function templateOptionForSelect2 (data) { 65080d699d6SGreg Roach // This could be a "waiting..." message (data.loading is true) or a response from the server. 65180d699d6SGreg Roach // Both are already HTML, so no need to reformat it. 65271239cb6SGreg Roach return data.text; 65371239cb6SGreg Roach } 65471239cb6SGreg Roach 65571239cb6SGreg Roach // Autocomplete 65671239cb6SGreg Roach autocomplete('input[data-autocomplete-url]'); 65771239cb6SGreg Roach 65871239cb6SGreg Roach // Select2 - activate autocomplete fields 659bdbdb10cSGreg Roach const lang = document.documentElement.lang; 660bdbdb10cSGreg Roach const select2_languages = { 661bdbdb10cSGreg Roach 'zh-Hans': 'zh-CN', 662efd89170SGreg Roach 'zh-Hant': 'zh-TW' 663bdbdb10cSGreg Roach }; 664efd89170SGreg Roach $('select.select2').select2({ 665bdbdb10cSGreg Roach language: select2_languages[lang] || lang, 666aee7167dSGreg Roach // Needed for elements that are initially hidden. 667efd89170SGreg Roach width: '100%', 668896a5721SGreg Roach // Do not escape - we do it on the server. 66971239cb6SGreg Roach escapeMarkup: function (x) { 6708ec20abdSGreg Roach return x; 671efd89170SGreg Roach } 672896a5721SGreg Roach }); 673896a5721SGreg Roach 674896a5721SGreg Roach // If we clear the select (using the "X" button), we need an empty value 675896a5721SGreg Roach // (rather than no value at all) for (non-multiple) selects with name="array[]" 676efd89170SGreg Roach $('select.select2:not([multiple])') 677efd89170SGreg Roach .on('select2:unselect', function (evt) { 678efd89170SGreg Roach $(evt.delegateTarget).html('<option value="" selected></option>'); 679bdbdb10cSGreg Roach }); 68071239cb6SGreg Roach 68171239cb6SGreg Roach // Datatables - locale aware sorting 68271239cb6SGreg Roach $.fn.dataTableExt.oSort['text-asc'] = function (x, y) { 683efd89170SGreg Roach return x.localeCompare(y, document.documentElement.lang, { sensitivity: 'base' }); 68471239cb6SGreg Roach }; 68571239cb6SGreg Roach $.fn.dataTableExt.oSort['text-desc'] = function (x, y) { 686efd89170SGreg Roach return y.localeCompare(x, document.documentElement.lang, { sensitivity: 'base' }); 68771239cb6SGreg Roach }; 68871239cb6SGreg Roach 68971239cb6SGreg Roach // DataTables - start hidden to prevent FOUC. 69071239cb6SGreg Roach $('table.datatables').each(function () { 6914843b94fSGreg Roach $(this).DataTable(); 6924843b94fSGreg Roach $(this).removeClass('d-none'); 6934843b94fSGreg Roach }); 6944843b94fSGreg Roach 6954843b94fSGreg Roach // Save button state between pages 696efd89170SGreg Roach document.querySelectorAll('[data-toggle=button][data-persist]').forEach((element) => { 6974843b94fSGreg Roach // Previously selected? 698efd89170SGreg Roach if (localStorage.getItem('state-of-' + element.dataset.persist) === 'T') { 6994843b94fSGreg Roach element.click(); 7004843b94fSGreg Roach } 7014843b94fSGreg Roach // Save state on change 702efd89170SGreg Roach element.addEventListener('click', (event) => { 7034843b94fSGreg Roach // Event occurs *before* the state changes, so reverse T/F. 704efd89170SGreg Roach localStorage.setItem('state-of-' + event.target.dataset.persist, event.target.classList.contains('active') ? 'F' : 'T'); 7054843b94fSGreg Roach }); 7064843b94fSGreg Roach }); 70771239cb6SGreg Roach 70871239cb6SGreg Roach // Activate the on-screen keyboard 70971239cb6SGreg Roach var osk_focus_element; 71071239cb6SGreg Roach $('.wt-osk-trigger').click(function () { 71171239cb6SGreg Roach // When a user clicks the icon, set focus to the corresponding input 71271239cb6SGreg Roach osk_focus_element = document.getElementById($(this).data('id')); 71371239cb6SGreg Roach osk_focus_element.focus(); 71471239cb6SGreg Roach $('.wt-osk').show(); 71571239cb6SGreg Roach }); 71671239cb6SGreg Roach $('.wt-osk-script-button').change(function () { 71771239cb6SGreg Roach $('.wt-osk-script').prop('hidden', true); 71871239cb6SGreg Roach $('.wt-osk-script-' + $(this).data('script')).prop('hidden', false); 71971239cb6SGreg Roach }); 72071239cb6SGreg Roach $('.wt-osk-shift-button').click(function () { 72171239cb6SGreg Roach document.querySelector('.wt-osk-keys').classList.toggle('shifted'); 72271239cb6SGreg Roach }); 72371239cb6SGreg Roach $('.wt-osk-keys').on('click', '.wt-osk-key', function () { 72471239cb6SGreg Roach var key = $(this).contents().get(0).nodeValue; 72571239cb6SGreg Roach var shift_state = $('.wt-osk-shift-button').hasClass('active'); 72671239cb6SGreg Roach var shift_key = $('sup', this)[0]; 72771239cb6SGreg Roach if (shift_state && shift_key !== undefined) { 72871239cb6SGreg Roach key = shift_key.innerText; 72971239cb6SGreg Roach } 7300d2905f7SGreg Roach webtrees.pasteAtCursor(osk_focus_element, key); 73171239cb6SGreg Roach if ($('.wt-osk-pin-button').hasClass('active') === false) { 73271239cb6SGreg Roach $('.wt-osk').hide(); 73371239cb6SGreg Roach } 73471239cb6SGreg Roach }); 73571239cb6SGreg Roach 73671239cb6SGreg Roach $('.wt-osk-close').on('click', function () { 73771239cb6SGreg Roach $('.wt-osk').hide(); 73871239cb6SGreg Roach }); 73971239cb6SGreg Roach}); 7407adfb8e5SGreg Roach 741b3a775f6SGreg Roach// Convert data-confirm and data-post-url attributes into useful behavior. 742efd89170SGreg Roachdocument.addEventListener('click', (event) => { 743*a7a3d6dbSGreg Roach const target = event.target.closest('a,button'); 7447adfb8e5SGreg Roach 7457adfb8e5SGreg Roach if (target === null) { 7467adfb8e5SGreg Roach return; 7477adfb8e5SGreg Roach } 7487adfb8e5SGreg Roach 749efd89170SGreg Roach if ('confirm' in target.dataset && !confirm(target.dataset.confirm)) { 7507adfb8e5SGreg Roach event.preventDefault(); 7517adfb8e5SGreg Roach return; 7527adfb8e5SGreg Roach } 7537adfb8e5SGreg Roach 754efd89170SGreg Roach if ('postUrl' in target.dataset) { 755efd89170SGreg Roach const token = document.querySelector('meta[name=csrf]').content; 756ea101122SGreg Roach 757ea101122SGreg Roach fetch(target.dataset.postUrl, { 758ea101122SGreg Roach method: 'POST', 759ea101122SGreg Roach headers: { 760ea101122SGreg Roach 'X-CSRF-TOKEN': token, 761ea101122SGreg Roach 'X-Requested-with': 'XMLHttpRequest', 762ea101122SGreg Roach }, 763ea101122SGreg Roach }).then((response) => { 764ea101122SGreg Roach if ('reloadUrl' in target.dataset) { 765ea101122SGreg Roach // Go somewhere else. e.g. home page after logout. 766ea101122SGreg Roach document.location = target.dataset.reloadUrl; 767ea101122SGreg Roach } else { 768ea101122SGreg Roach // Reload the current page. e.g. change language. 7697adfb8e5SGreg Roach document.location.reload(); 7707adfb8e5SGreg Roach } 771cb8f307bSGreg Roach }).catch((error) => { 772ea101122SGreg Roach alert(error); 773ea101122SGreg Roach }); 7747adfb8e5SGreg Roach } 7757adfb8e5SGreg Roach}); 776