171239cb6SGreg Roach/** 271239cb6SGreg Roach * webtrees: online genealogy 3242a7862SGreg Roach * Copyright (C) 2019 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 1671239cb6SGreg Roach'use strict'; 1771239cb6SGreg Roach 18*59e18f0cSGreg Roachlet webtrees = function () { 19*59e18f0cSGreg Roach const lang = document.documentElement.lang; 20*59e18f0cSGreg Roach 21*59e18f0cSGreg Roach /** 22*59e18f0cSGreg Roach * Tidy the whitespace in a string. 23*59e18f0cSGreg Roach */ 24*59e18f0cSGreg Roach function trim(str) { 25*59e18f0cSGreg Roach return str.replace(/\s+/g, " ").trim(); 26*59e18f0cSGreg Roach 27*59e18f0cSGreg Roach } 28*59e18f0cSGreg Roach 29*59e18f0cSGreg Roach /** 30*59e18f0cSGreg Roach * Look for non-latin characters in a string. 31*59e18f0cSGreg Roach */ 32*59e18f0cSGreg Roach function detectScript(str) { 33*59e18f0cSGreg Roach if (str.match(/[\u3400-\u9FCC]/)) { 34*59e18f0cSGreg Roach return "cjk"; 35*59e18f0cSGreg Roach } else if (str.match(/[\u0370-\u03FF]/)) { 36*59e18f0cSGreg Roach return "greek"; 37*59e18f0cSGreg Roach } else if (str.match(/[\u0400-\u04FF]/)) { 38*59e18f0cSGreg Roach return "cyrillic"; 39*59e18f0cSGreg Roach } else if (str.match(/[\u0590-\u05FF]/)) { 40*59e18f0cSGreg Roach return "hebrew"; 41*59e18f0cSGreg Roach } else if (str.match(/[\u0600-\u06FF]/)) { 42*59e18f0cSGreg Roach return "arabic"; 43*59e18f0cSGreg Roach } 44*59e18f0cSGreg Roach 45*59e18f0cSGreg Roach return "latin"; 46*59e18f0cSGreg Roach } 47*59e18f0cSGreg Roach 48*59e18f0cSGreg Roach /** 49*59e18f0cSGreg Roach * In some languages, the SURN uses a male/default form, but NAME uses a gender-inflected form. 50*59e18f0cSGreg Roach */ 51*59e18f0cSGreg Roach function inflectSurname(surname, sex) { 52*59e18f0cSGreg Roach if (lang === "pl" && sex === "F") { 53*59e18f0cSGreg Roach return surname 54*59e18f0cSGreg Roach .replace(/ski$/, "ska") 55*59e18f0cSGreg Roach .replace(/cki$/, "cka") 56*59e18f0cSGreg Roach .replace(/dzki$/, "dzka") 57*59e18f0cSGreg Roach .replace(/żki$/, "żka"); 58*59e18f0cSGreg Roach } 59*59e18f0cSGreg Roach 60*59e18f0cSGreg Roach return surname; 61*59e18f0cSGreg Roach } 62*59e18f0cSGreg Roach 63*59e18f0cSGreg Roach /** 64*59e18f0cSGreg Roach * Build a NAME from a NPFX, GIVN, SPFX, SURN and NSFX parts. 65*59e18f0cSGreg Roach * 66*59e18f0cSGreg Roach * Assumes the language of the document is the same as the language of the name. 67*59e18f0cSGreg Roach */ 68*59e18f0cSGreg Roach function buildNameFromParts(npfx, givn, spfx, surn, nsfx, sex) { 69*59e18f0cSGreg Roach const usesCJK = detectScript(npfx + givn + spfx + givn + surn + nsfx) === "cjk"; 70*59e18f0cSGreg Roach const separator = usesCJK ? "" : " "; 71*59e18f0cSGreg Roach const surnameFirst = usesCJK || ['hu', 'jp', 'ko', 'vi', 'zh-Hans', 'zh-Hant'].indexOf(lang) !== -1; 72*59e18f0cSGreg Roach const patronym = ['is'].indexOf(lang) !== -1; 73*59e18f0cSGreg Roach const slash = patronym ? "" : "/"; 74*59e18f0cSGreg Roach 75*59e18f0cSGreg Roach // GIVN and SURN may be a comma-separated lists. 76*59e18f0cSGreg Roach npfx = trim(npfx); 77*59e18f0cSGreg Roach givn = trim(givn.replace(",", separator)); 78*59e18f0cSGreg Roach spfx = trim(spfx); 79*59e18f0cSGreg Roach surn = inflectSurname(trim(surn.replace(",", separator)), sex); 80*59e18f0cSGreg Roach nsfx = trim(nsfx); 81*59e18f0cSGreg Roach 82*59e18f0cSGreg Roach const surname = trim(spfx + separator + surn); 83*59e18f0cSGreg Roach 84*59e18f0cSGreg Roach const name = surnameFirst ? slash + surname + slash + separator + givn : givn + separator + slash + surname + slash; 85*59e18f0cSGreg Roach 86*59e18f0cSGreg Roach return trim(npfx + separator + name + separator + nsfx); 87*59e18f0cSGreg Roach } 88*59e18f0cSGreg Roach 89*59e18f0cSGreg Roach // Public methods 90*59e18f0cSGreg Roach return { 91*59e18f0cSGreg Roach buildNameFromParts: buildNameFromParts, 92*59e18f0cSGreg Roach detectScript: detectScript, 93*59e18f0cSGreg Roach }; 94*59e18f0cSGreg Roach}(); 95*59e18f0cSGreg Roach 9671239cb6SGreg Roachfunction expand_layer(sid) 9771239cb6SGreg Roach{ 9871239cb6SGreg Roach $('#' + sid + '_img').toggleClass('icon-plus icon-minus'); 9971239cb6SGreg Roach $('#' + sid).slideToggle('fast'); 10071239cb6SGreg Roach $('#' + sid + '-alt').toggle(); // hide something when we show the layer - and vice-versa 10171239cb6SGreg Roach return false; 10271239cb6SGreg Roach} 10371239cb6SGreg Roach 10471239cb6SGreg Roach// Accept the changes to a record - and reload the page 10571239cb6SGreg Roachfunction accept_changes(xref, ged) 10671239cb6SGreg Roach{ 10771239cb6SGreg Roach $.post( 10871239cb6SGreg Roach 'index.php', 10971239cb6SGreg Roach { 11071239cb6SGreg Roach route: 'accept-changes', 11171239cb6SGreg Roach xref: xref, 11271239cb6SGreg Roach ged: ged, 11371239cb6SGreg Roach }, 11471239cb6SGreg Roach function () { 115070932ceSGreg Roach document.location.reload(); 11671239cb6SGreg Roach } 11771239cb6SGreg Roach ); 11871239cb6SGreg Roach return false; 11971239cb6SGreg Roach} 12071239cb6SGreg Roach 12171239cb6SGreg Roach// Reject the changes to a record - and reload the page 12271239cb6SGreg Roachfunction reject_changes(xref, ged) 12371239cb6SGreg Roach{ 12471239cb6SGreg Roach $.post( 12571239cb6SGreg Roach 'index.php', 12671239cb6SGreg Roach { 12771239cb6SGreg Roach route: 'reject-changes', 12871239cb6SGreg Roach xref: xref, 12971239cb6SGreg Roach ged: ged, 13071239cb6SGreg Roach }, 13171239cb6SGreg Roach function () { 132070932ceSGreg Roach document.location.reload(); 13371239cb6SGreg Roach } 13471239cb6SGreg Roach ); 13571239cb6SGreg Roach return false; 13671239cb6SGreg Roach} 13771239cb6SGreg Roach 13871239cb6SGreg Roach// Delete a record - and reload the page 13971239cb6SGreg Roachfunction delete_record(xref, gedcom) 14071239cb6SGreg Roach{ 14171239cb6SGreg Roach $.post( 14271239cb6SGreg Roach 'index.php', 14371239cb6SGreg Roach { 14471239cb6SGreg Roach route: 'delete-record', 14571239cb6SGreg Roach xref: xref, 14671239cb6SGreg Roach ged: gedcom, 14771239cb6SGreg Roach }, 14871239cb6SGreg Roach function () { 149070932ceSGreg Roach document.location.reload(); 15071239cb6SGreg Roach } 15171239cb6SGreg Roach ); 15271239cb6SGreg Roach 15371239cb6SGreg Roach return false; 15471239cb6SGreg Roach} 15571239cb6SGreg Roach 15671239cb6SGreg Roach// Delete a fact - and reload the page 15771239cb6SGreg Roachfunction delete_fact(message, ged, xref, fact_id) 15871239cb6SGreg Roach{ 15971239cb6SGreg Roach if (confirm(message)) { 16071239cb6SGreg Roach $.post( 16171239cb6SGreg Roach 'index.php', 16271239cb6SGreg Roach { 16371239cb6SGreg Roach route: 'delete-fact', 16471239cb6SGreg Roach xref: xref, 16571239cb6SGreg Roach fact_id: fact_id, 16671239cb6SGreg Roach ged: ged 16771239cb6SGreg Roach }, 16871239cb6SGreg Roach function () { 169070932ceSGreg Roach document.location.reload(); 17071239cb6SGreg Roach } 17171239cb6SGreg Roach ); 17271239cb6SGreg Roach } 17371239cb6SGreg Roach return false; 17471239cb6SGreg Roach} 17571239cb6SGreg Roach 17671239cb6SGreg Roach// Copy a fact to the clipboard 17771239cb6SGreg Roachfunction copy_fact(ged, xref, fact_id) 17871239cb6SGreg Roach{ 17971239cb6SGreg Roach $.post( 18071239cb6SGreg Roach 'index.php', 18171239cb6SGreg Roach { 18271239cb6SGreg Roach route: 'copy-fact', 18371239cb6SGreg Roach xref: xref, 18471239cb6SGreg Roach fact_id: fact_id, 18571239cb6SGreg Roach ged: ged, 18671239cb6SGreg Roach }, 18771239cb6SGreg Roach function () { 188070932ceSGreg Roach document.location.reload(); 18971239cb6SGreg Roach } 19071239cb6SGreg Roach ); 19171239cb6SGreg Roach return false; 19271239cb6SGreg Roach} 19371239cb6SGreg Roach 19471239cb6SGreg Roach// Paste a fact from the clipboard 19571239cb6SGreg Roachfunction paste_fact(ged, xref, element) 19671239cb6SGreg Roach{ 19771239cb6SGreg Roach $.post( 19871239cb6SGreg Roach 'index.php', 19971239cb6SGreg Roach { 20071239cb6SGreg Roach route: 'paste-fact', 20171239cb6SGreg Roach xref: xref, 20271239cb6SGreg Roach fact_id: $(element).val(), // element is the <select> containing the option 20371239cb6SGreg Roach ged: ged, 20471239cb6SGreg Roach }, 20571239cb6SGreg Roach function () { 206070932ceSGreg Roach document.location.reload(); 20771239cb6SGreg Roach } 20871239cb6SGreg Roach ); 20971239cb6SGreg Roach return false; 21071239cb6SGreg Roach} 21171239cb6SGreg Roach 21271239cb6SGreg Roach// Delete a user - and reload the page 21371239cb6SGreg Roachfunction delete_user(message, user_id) 21471239cb6SGreg Roach{ 21571239cb6SGreg Roach if (confirm(message)) { 21671239cb6SGreg Roach $.post( 21771239cb6SGreg Roach 'index.php', 21871239cb6SGreg Roach { 21971239cb6SGreg Roach route: 'delete-user', 22071239cb6SGreg Roach user_id: user_id, 22171239cb6SGreg Roach }, 22271239cb6SGreg Roach function () { 223070932ceSGreg Roach document.location.reload(); 22471239cb6SGreg Roach } 22571239cb6SGreg Roach ); 22671239cb6SGreg Roach } 22771239cb6SGreg Roach return false; 22871239cb6SGreg Roach} 22971239cb6SGreg Roach 23071239cb6SGreg Roach// Masquerade as another user - and reload the page. 23171239cb6SGreg Roachfunction masquerade(user_id) 23271239cb6SGreg Roach{ 23371239cb6SGreg Roach $.post( 23471239cb6SGreg Roach 'index.php', 23571239cb6SGreg Roach { 23671239cb6SGreg Roach route: 'masquerade', 23771239cb6SGreg Roach user_id: user_id, 23871239cb6SGreg Roach }, 23971239cb6SGreg Roach function () { 240070932ceSGreg Roach document.location.reload(); 24171239cb6SGreg Roach } 24271239cb6SGreg Roach ); 24371239cb6SGreg Roach return false; 24471239cb6SGreg Roach} 24571239cb6SGreg Roach 24671239cb6SGreg Roachvar pastefield; 24771239cb6SGreg Roachfunction addmedia_links(field, iid, iname) 24871239cb6SGreg Roach{ 24971239cb6SGreg Roach pastefield = field; 25071239cb6SGreg Roach insertRowToTable(iid, iname); 25171239cb6SGreg Roach return false; 25271239cb6SGreg Roach} 25371239cb6SGreg Roach 25471239cb6SGreg Roachfunction valid_date(datefield, dmy) 25571239cb6SGreg Roach{ 25671239cb6SGreg Roach var months = ['JAN', 'FEB', 'MAR', 'APR', 'MAY', 'JUN', 'JUL', 'AUG', 'SEP', 'OCT', 'NOV', 'DEC']; 25771239cb6SGreg Roach var hijri_months = ['MUHAR', 'SAFAR', 'RABIA', 'RABIT', 'JUMAA', 'JUMAT', 'RAJAB', 'SHAAB', 'RAMAD', 'SHAWW', 'DHUAQ', 'DHUAH']; 25871239cb6SGreg Roach var hebrew_months = ['TSH', 'CSH', 'KSL', 'TVT', 'SHV', 'ADR', 'ADS', 'NSN', 'IYR', 'SVN', 'TMZ', 'AAV', 'ELL']; 25971239cb6SGreg Roach var french_months = ['VEND', 'BRUM', 'FRIM', 'NIVO', 'PLUV', 'VENT', 'GERM', 'FLOR', 'PRAI', 'MESS', 'THER', 'FRUC', 'COMP']; 26071239cb6SGreg Roach var jalali_months = ['FARVA', 'ORDIB', 'KHORD', 'TIR', 'MORDA', 'SHAHR', 'MEHR', 'ABAN', 'AZAR', 'DEY', 'BAHMA', 'ESFAN']; 26171239cb6SGreg Roach 26271239cb6SGreg Roach var datestr = datefield.value; 26371239cb6SGreg Roach // if a date has a date phrase marked by () this has to be excluded from altering 26471239cb6SGreg Roach var datearr = datestr.split('('); 26571239cb6SGreg Roach var datephrase = ''; 26671239cb6SGreg Roach if (datearr.length > 1) { 26771239cb6SGreg Roach datestr = datearr[0]; 26871239cb6SGreg Roach datephrase = datearr[1]; 26971239cb6SGreg Roach } 27071239cb6SGreg Roach 27171239cb6SGreg Roach // Gedcom dates are upper case 27271239cb6SGreg Roach datestr = datestr.toUpperCase(); 27371239cb6SGreg Roach // Gedcom dates have no leading/trailing/repeated whitespace 27471239cb6SGreg Roach datestr = datestr.replace(/\s+/, ' '); 27571239cb6SGreg Roach datestr = datestr.replace(/(^\s)|(\s$)/, ''); 27671239cb6SGreg Roach // Gedcom dates have spaces between letters and digits, e.g. "01JAN2000" => "01 JAN 2000" 27771239cb6SGreg Roach datestr = datestr.replace(/(\d)([A-Z])/, '$1 $2'); 27871239cb6SGreg Roach datestr = datestr.replace(/([A-Z])(\d)/, '$1 $2'); 27971239cb6SGreg Roach 28071239cb6SGreg Roach // Shortcut for quarter format, "Q1 1900" => "BET JAN 1900 AND MAR 1900". See [ 1509083 ] 28171239cb6SGreg Roach if (datestr.match(/^Q ([1-4]) (\d\d\d\d)$/)) { 28271239cb6SGreg Roach datestr = 'BET ' + months[RegExp.$1 * 3 - 3] + ' ' + RegExp.$2 + ' AND ' + months[RegExp.$1 * 3 - 1] + ' ' + RegExp.$2; 28371239cb6SGreg Roach } 28471239cb6SGreg Roach 28571239cb6SGreg Roach // Shortcut for @#Dxxxxx@ 01 01 1400, etc. 28671239cb6SGreg Roach if (datestr.match(/^(@#DHIJRI@|HIJRI)( \d?\d )(\d?\d)( \d?\d?\d?\d)$/)) { 28771239cb6SGreg Roach datestr = '@#DHIJRI@' + RegExp.$2 + hijri_months[parseInt(RegExp.$3, 10) - 1] + RegExp.$4; 28871239cb6SGreg Roach } 28971239cb6SGreg Roach if (datestr.match(/^(@#DJALALI@|JALALI)( \d?\d )(\d?\d)( \d?\d?\d?\d)$/)) { 29071239cb6SGreg Roach datestr = '@#DJALALI@' + RegExp.$2 + jalali_months[parseInt(RegExp.$3, 10) - 1] + RegExp.$4; 29171239cb6SGreg Roach } 29271239cb6SGreg Roach if (datestr.match(/^(@#DHEBREW@|HEBREW)( \d?\d )(\d?\d)( \d?\d?\d?\d)$/)) { 29371239cb6SGreg Roach datestr = '@#DHEBREW@' + RegExp.$2 + hebrew_months[parseInt(RegExp.$3, 10) - 1] + RegExp.$4; 29471239cb6SGreg Roach } 29571239cb6SGreg Roach if (datestr.match(/^(@#DFRENCH R@|FRENCH)( \d?\d )(\d?\d)( \d?\d?\d?\d)$/)) { 29671239cb6SGreg Roach datestr = '@#DFRENCH R@' + RegExp.$2 + french_months[parseInt(RegExp.$3, 10) - 1] + RegExp.$4; 29771239cb6SGreg Roach } 29871239cb6SGreg Roach 29971239cb6SGreg Roach // e.g. 17.11.1860, 03/04/2005 or 1999-12-31. Use locale settings where DMY order is ambiguous. 30071239cb6SGreg Roach var qsearch = /^([^\d]*)(\d+)[^\d](\d+)[^\d](\d+)$/i; 30171239cb6SGreg Roach if (qsearch.exec(datestr)) { 30271239cb6SGreg Roach var f0 = RegExp.$1; 30371239cb6SGreg Roach var f1 = parseInt(RegExp.$2, 10); 30471239cb6SGreg Roach var f2 = parseInt(RegExp.$3, 10); 30571239cb6SGreg Roach var f3 = parseInt(RegExp.$4, 10); 30671239cb6SGreg Roach var yyyy = new Date().getFullYear(); 30771239cb6SGreg Roach var yy = yyyy % 100; 30871239cb6SGreg Roach var cc = yyyy - yy; 30971239cb6SGreg Roach if (dmy === 'DMY' && f1 <= 31 && f2 <= 12 || f1 > 13 && f1 <= 31 && f2 <= 12 && f3 > 31) { 31071239cb6SGreg Roach datestr = f0 + f1 + ' ' + months[f2 - 1] + ' ' + (f3 >= 100 ? f3 : (f3 <= yy ? f3 + cc : f3 + cc - 100)); 31171239cb6SGreg Roach } else { 31271239cb6SGreg Roach if (dmy === 'MDY' && f1 <= 12 && f2 <= 31 || f2 > 13 && f2 <= 31 && f1 <= 12 && f3 > 31) { 31371239cb6SGreg Roach datestr = f0 + f2 + ' ' + months[f1 - 1] + ' ' + (f3 >= 100 ? f3 : (f3 <= yy ? f3 + cc : f3 + cc - 100)); 31471239cb6SGreg Roach } else { 31571239cb6SGreg Roach if (dmy === 'YMD' && f2 <= 12 && f3 <= 31 || f3 > 13 && f3 <= 31 && f2 <= 12 && f1 > 31) { 31671239cb6SGreg Roach datestr = f0 + f3 + ' ' + months[f2 - 1] + ' ' + (f1 >= 100 ? f1 : (f1 <= yy ? f1 + cc : f1 + cc - 100)); 31771239cb6SGreg Roach } 31871239cb6SGreg Roach } 31971239cb6SGreg Roach } 32071239cb6SGreg Roach } 32171239cb6SGreg Roach 32271239cb6SGreg Roach // Shortcuts for date ranges 32371239cb6SGreg Roach datestr = datestr.replace(/^[>]([\w ]+)$/, 'AFT $1'); 32471239cb6SGreg Roach datestr = datestr.replace(/^[<]([\w ]+)$/, 'BEF $1'); 32571239cb6SGreg Roach datestr = datestr.replace(/^([\w ]+)[-]$/, 'FROM $1'); 32671239cb6SGreg Roach datestr = datestr.replace(/^[-]([\w ]+)$/, 'TO $1'); 32771239cb6SGreg Roach datestr = datestr.replace(/^[~]([\w ]+)$/, 'ABT $1'); 32871239cb6SGreg Roach datestr = datestr.replace(/^[*]([\w ]+)$/, 'EST $1'); 32971239cb6SGreg Roach datestr = datestr.replace(/^[#]([\w ]+)$/, 'CAL $1'); 33071239cb6SGreg Roach datestr = datestr.replace(/^([\w ]+) ?- ?([\w ]+)$/, 'BET $1 AND $2'); 33171239cb6SGreg Roach datestr = datestr.replace(/^([\w ]+) ?~ ?([\w ]+)$/, 'FROM $1 TO $2'); 33271239cb6SGreg Roach 33371239cb6SGreg Roach // Convert full months to short months 33471239cb6SGreg Roach datestr = datestr.replace(/(JANUARY)/, 'JAN'); 33571239cb6SGreg Roach datestr = datestr.replace(/(FEBRUARY)/, 'FEB'); 33671239cb6SGreg Roach datestr = datestr.replace(/(MARCH)/, 'MAR'); 33771239cb6SGreg Roach datestr = datestr.replace(/(APRIL)/, 'APR'); 33871239cb6SGreg Roach datestr = datestr.replace(/(MAY)/, 'MAY'); 33971239cb6SGreg Roach datestr = datestr.replace(/(JUNE)/, 'JUN'); 34071239cb6SGreg Roach datestr = datestr.replace(/(JULY)/, 'JUL'); 34171239cb6SGreg Roach datestr = datestr.replace(/(AUGUST)/, 'AUG'); 34271239cb6SGreg Roach datestr = datestr.replace(/(SEPTEMBER)/, 'SEP'); 34371239cb6SGreg Roach datestr = datestr.replace(/(OCTOBER)/, 'OCT'); 34471239cb6SGreg Roach datestr = datestr.replace(/(NOVEMBER)/, 'NOV'); 34571239cb6SGreg Roach datestr = datestr.replace(/(DECEMBER)/, 'DEC'); 34671239cb6SGreg Roach 34771239cb6SGreg Roach // Americans frequently enter dates as SEP 20, 1999 34871239cb6SGreg Roach // No need to internationalise this, as this is an english-language issue 34971239cb6SGreg Roach datestr = datestr.replace(/(JAN|FEB|MAR|APR|MAY|JUN|JUL|AUG|SEP|OCT|NOV|DEC)\.? (\d\d?)[, ]+(\d\d\d\d)/, '$2 $1 $3'); 35071239cb6SGreg Roach 35171239cb6SGreg Roach // Apply leading zero to day numbers 35271239cb6SGreg Roach datestr = datestr.replace(/(^| )(\d [A-Z]{3,5} \d{4})/, '$10$2'); 35371239cb6SGreg Roach 35471239cb6SGreg Roach if (datephrase) { 35571239cb6SGreg Roach datestr = datestr + ' (' + datephrase; 35671239cb6SGreg Roach } 35771239cb6SGreg Roach // Only update it if is has been corrected - otherwise input focus 35871239cb6SGreg Roach // moves to the end of the field unnecessarily 35971239cb6SGreg Roach if (datefield.value !== datestr) { 36071239cb6SGreg Roach datefield.value = datestr; 36171239cb6SGreg Roach } 36271239cb6SGreg Roach} 36371239cb6SGreg Roach 36471239cb6SGreg Roachvar menutimeouts = []; 36571239cb6SGreg Roach 36671239cb6SGreg Roachfunction show_submenu(elementid, parentid) 36771239cb6SGreg Roach{ 36871239cb6SGreg Roach var pagewidth = document.body.scrollWidth + document.documentElement.scrollLeft; 36971239cb6SGreg Roach var element = document.getElementById(elementid); 37071239cb6SGreg Roach 37171239cb6SGreg Roach if (element && element.style) { 37271239cb6SGreg Roach if (document.all) { 37371239cb6SGreg Roach pagewidth = document.body.offsetWidth; 37471239cb6SGreg Roach } else { 37571239cb6SGreg Roach pagewidth = document.body.scrollWidth + document.documentElement.scrollLeft - 55; 37671239cb6SGreg Roach if (document.documentElement.dir === 'rtl') { 37771239cb6SGreg Roach boxright = element.offsetLeft + element.offsetWidth + 10; 37871239cb6SGreg Roach } 37971239cb6SGreg Roach } 38071239cb6SGreg Roach 38171239cb6SGreg Roach // -- make sure the submenu is the size of the largest child 38271239cb6SGreg Roach var maxwidth = 0; 38371239cb6SGreg Roach var count = element.childNodes.length; 38471239cb6SGreg Roach for (var i = 0; i < count; i++) { 38571239cb6SGreg Roach var child = element.childNodes[i]; 38671239cb6SGreg Roach if (child.offsetWidth > maxwidth + 5) { 38771239cb6SGreg Roach maxwidth = child.offsetWidth; 38871239cb6SGreg Roach } 38971239cb6SGreg Roach } 39071239cb6SGreg Roach if (element.offsetWidth < maxwidth) { 39171239cb6SGreg Roach element.style.width = maxwidth + 'px'; 39271239cb6SGreg Roach } 39371239cb6SGreg Roach var pelement, boxright; 39471239cb6SGreg Roach pelement = document.getElementById(parentid); 39571239cb6SGreg Roach if (pelement) { 39671239cb6SGreg Roach element.style.left = pelement.style.left; 39771239cb6SGreg Roach boxright = element.offsetLeft + element.offsetWidth + 10; 39871239cb6SGreg Roach if (boxright > pagewidth) { 39971239cb6SGreg Roach var menuleft = pagewidth - element.offsetWidth; 40071239cb6SGreg Roach element.style.left = menuleft + 'px'; 40171239cb6SGreg Roach } 40271239cb6SGreg Roach } 40371239cb6SGreg Roach 40471239cb6SGreg Roach if (element.offsetLeft < 0) { 40571239cb6SGreg Roach element.style.left = '0px'; 40671239cb6SGreg Roach } 40771239cb6SGreg Roach 40871239cb6SGreg Roach // -- put scrollbars on really long menus 40971239cb6SGreg Roach if (element.offsetHeight > 500) { 41071239cb6SGreg Roach element.style.height = '400px'; 41171239cb6SGreg Roach element.style.overflow = 'auto'; 41271239cb6SGreg Roach } 41371239cb6SGreg Roach 41471239cb6SGreg Roach element.style.visibility = 'visible'; 41571239cb6SGreg Roach } 41671239cb6SGreg Roach clearTimeout(menutimeouts[elementid]); 41771239cb6SGreg Roach menutimeouts[elementid] = null; 41871239cb6SGreg Roach} 41971239cb6SGreg Roach 42071239cb6SGreg Roachfunction hide_submenu(elementid) 42171239cb6SGreg Roach{ 42271239cb6SGreg Roach if (typeof menutimeouts[elementid] !== 'number') { 42371239cb6SGreg Roach return; 42471239cb6SGreg Roach } 42571239cb6SGreg Roach var element = document.getElementById(elementid); 42671239cb6SGreg Roach if (element && element.style) { 42771239cb6SGreg Roach element.style.visibility = 'hidden'; 42871239cb6SGreg Roach } 42971239cb6SGreg Roach clearTimeout(menutimeouts[elementid]); 43071239cb6SGreg Roach menutimeouts[elementid] = null; 43171239cb6SGreg Roach} 43271239cb6SGreg Roach 43371239cb6SGreg Roachfunction timeout_submenu(elementid) 43471239cb6SGreg Roach{ 43571239cb6SGreg Roach if (typeof menutimeouts[elementid] !== 'number') { 43671239cb6SGreg Roach menutimeouts[elementid] = setTimeout("hide_submenu('" + elementid + "')", 100); 43771239cb6SGreg Roach } 43871239cb6SGreg Roach} 43971239cb6SGreg Roach 44071239cb6SGreg Roachvar monthLabels = []; 44171239cb6SGreg RoachmonthLabels[1] = 'January'; 44271239cb6SGreg RoachmonthLabels[2] = 'February'; 44371239cb6SGreg RoachmonthLabels[3] = 'March'; 44471239cb6SGreg RoachmonthLabels[4] = 'April'; 44571239cb6SGreg RoachmonthLabels[5] = 'May'; 44671239cb6SGreg RoachmonthLabels[6] = 'June'; 44771239cb6SGreg RoachmonthLabels[7] = 'July'; 44871239cb6SGreg RoachmonthLabels[8] = 'August'; 44971239cb6SGreg RoachmonthLabels[9] = 'September'; 45071239cb6SGreg RoachmonthLabels[10] = 'October'; 45171239cb6SGreg RoachmonthLabels[11] = 'November'; 45271239cb6SGreg RoachmonthLabels[12] = 'December'; 45371239cb6SGreg Roach 45471239cb6SGreg Roachvar monthShort = []; 45571239cb6SGreg RoachmonthShort[1] = 'JAN'; 45671239cb6SGreg RoachmonthShort[2] = 'FEB'; 45771239cb6SGreg RoachmonthShort[3] = 'MAR'; 45871239cb6SGreg RoachmonthShort[4] = 'APR'; 45971239cb6SGreg RoachmonthShort[5] = 'MAY'; 46071239cb6SGreg RoachmonthShort[6] = 'JUN'; 46171239cb6SGreg RoachmonthShort[7] = 'JUL'; 46271239cb6SGreg RoachmonthShort[8] = 'AUG'; 46371239cb6SGreg RoachmonthShort[9] = 'SEP'; 46471239cb6SGreg RoachmonthShort[10] = 'OCT'; 46571239cb6SGreg RoachmonthShort[11] = 'NOV'; 46671239cb6SGreg RoachmonthShort[12] = 'DEC'; 46771239cb6SGreg Roach 46871239cb6SGreg Roachvar daysOfWeek = []; 46971239cb6SGreg RoachdaysOfWeek[0] = 'S'; 47071239cb6SGreg RoachdaysOfWeek[1] = 'M'; 47171239cb6SGreg RoachdaysOfWeek[2] = 'T'; 47271239cb6SGreg RoachdaysOfWeek[3] = 'W'; 47371239cb6SGreg RoachdaysOfWeek[4] = 'T'; 47471239cb6SGreg RoachdaysOfWeek[5] = 'F'; 47571239cb6SGreg RoachdaysOfWeek[6] = 'S'; 47671239cb6SGreg Roach 47771239cb6SGreg Roachvar weekStart = 0; 47871239cb6SGreg Roach 47971239cb6SGreg Roachfunction cal_setMonthNames(jan, feb, mar, apr, may, jun, jul, aug, sep, oct, nov, dec) 48071239cb6SGreg Roach{ 48171239cb6SGreg Roach monthLabels[1] = jan; 48271239cb6SGreg Roach monthLabels[2] = feb; 48371239cb6SGreg Roach monthLabels[3] = mar; 48471239cb6SGreg Roach monthLabels[4] = apr; 48571239cb6SGreg Roach monthLabels[5] = may; 48671239cb6SGreg Roach monthLabels[6] = jun; 48771239cb6SGreg Roach monthLabels[7] = jul; 48871239cb6SGreg Roach monthLabels[8] = aug; 48971239cb6SGreg Roach monthLabels[9] = sep; 49071239cb6SGreg Roach monthLabels[10] = oct; 49171239cb6SGreg Roach monthLabels[11] = nov; 49271239cb6SGreg Roach monthLabels[12] = dec; 49371239cb6SGreg Roach} 49471239cb6SGreg Roach 49571239cb6SGreg Roachfunction cal_setDayHeaders(sun, mon, tue, wed, thu, fri, sat) 49671239cb6SGreg Roach{ 49771239cb6SGreg Roach daysOfWeek[0] = sun; 49871239cb6SGreg Roach daysOfWeek[1] = mon; 49971239cb6SGreg Roach daysOfWeek[2] = tue; 50071239cb6SGreg Roach daysOfWeek[3] = wed; 50171239cb6SGreg Roach daysOfWeek[4] = thu; 50271239cb6SGreg Roach daysOfWeek[5] = fri; 50371239cb6SGreg Roach daysOfWeek[6] = sat; 50471239cb6SGreg Roach} 50571239cb6SGreg Roach 50671239cb6SGreg Roachfunction cal_setWeekStart(day) 50771239cb6SGreg Roach{ 50871239cb6SGreg Roach if (day >= 0 && day < 7) { 50971239cb6SGreg Roach weekStart = day; 51071239cb6SGreg Roach } 51171239cb6SGreg Roach} 51271239cb6SGreg Roach 51371239cb6SGreg Roachfunction calendarWidget(dateDivId, dateFieldId) 51471239cb6SGreg Roach{ 51571239cb6SGreg Roach var dateDiv = document.getElementById(dateDivId); 51671239cb6SGreg Roach var dateField = document.getElementById(dateFieldId); 51771239cb6SGreg Roach 51871239cb6SGreg Roach if (dateDiv.style.visibility === 'visible') { 51971239cb6SGreg Roach dateDiv.style.visibility = 'hidden'; 52071239cb6SGreg Roach return false; 52171239cb6SGreg Roach } 52271239cb6SGreg Roach if (dateDiv.style.visibility === 'show') { 52371239cb6SGreg Roach dateDiv.style.visibility = 'hide'; 52471239cb6SGreg Roach return false; 52571239cb6SGreg Roach } 52671239cb6SGreg Roach 52771239cb6SGreg Roach /* Javascript calendar functions only work with precise gregorian dates "D M Y" or "Y" */ 52871239cb6SGreg Roach var greg_regex = /((\d+ (JAN|FEB|MAR|APR|MAY|JUN|JUL|AUG|SEP|OCT|NOV|DEC) )?\d+)/i; 52971239cb6SGreg Roach var date; 53071239cb6SGreg Roach if (greg_regex.exec(dateField.value)) { 53171239cb6SGreg Roach date = new Date(RegExp.$1); 53271239cb6SGreg Roach } else { 53371239cb6SGreg Roach date = new Date(); 53471239cb6SGreg Roach } 53571239cb6SGreg Roach 53671239cb6SGreg Roach dateDiv.innerHTML = cal_generateSelectorContent(dateFieldId, dateDivId, date); 53771239cb6SGreg Roach if (dateDiv.style.visibility === 'hidden') { 53871239cb6SGreg Roach dateDiv.style.visibility = 'visible'; 53971239cb6SGreg Roach return false; 54071239cb6SGreg Roach } 54171239cb6SGreg Roach if (dateDiv.style.visibility === 'hide') { 54271239cb6SGreg Roach dateDiv.style.visibility = 'show'; 54371239cb6SGreg Roach return false; 54471239cb6SGreg Roach } 54571239cb6SGreg Roach 54671239cb6SGreg Roach return false; 54771239cb6SGreg Roach} 54871239cb6SGreg Roach 54971239cb6SGreg Roachfunction cal_generateSelectorContent(dateFieldId, dateDivId, date) 55071239cb6SGreg Roach{ 55171239cb6SGreg Roach var i, j; 55271239cb6SGreg Roach var content = '<table border="1"><tr>'; 55371239cb6SGreg Roach content += '<td><select class="form-control" id="' + dateFieldId + '_daySelect" onchange="return cal_updateCalendar(\'' + dateFieldId + '\', \'' + dateDivId + '\');">'; 55471239cb6SGreg Roach for (i = 1; i < 32; i++) { 55571239cb6SGreg Roach content += '<option value="' + i + '"'; 55671239cb6SGreg Roach if (date.getDate() === i) { 55771239cb6SGreg Roach content += ' selected="selected"'; 55871239cb6SGreg Roach } 55971239cb6SGreg Roach content += '>' + i + '</option>'; 56071239cb6SGreg Roach } 56171239cb6SGreg Roach content += '</select></td>'; 56271239cb6SGreg Roach content += '<td><select class="form-control" id="' + dateFieldId + '_monSelect" onchange="return cal_updateCalendar(\'' + dateFieldId + '\', \'' + dateDivId + '\');">'; 56371239cb6SGreg Roach for (i = 1; i < 13; i++) { 56471239cb6SGreg Roach content += '<option value="' + i + '"'; 56571239cb6SGreg Roach if (date.getMonth() + 1 === i) { 56671239cb6SGreg Roach content += ' selected="selected"'; 56771239cb6SGreg Roach } 56871239cb6SGreg Roach content += '>' + monthLabels[i] + '</option>'; 56971239cb6SGreg Roach } 57071239cb6SGreg Roach content += '</select></td>'; 57171239cb6SGreg 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>'; 57271239cb6SGreg Roach content += '<tr><td colspan="3">'; 57371239cb6SGreg Roach content += '<table width="100%">'; 57471239cb6SGreg Roach content += '<tr>'; 57571239cb6SGreg Roach j = weekStart; 57671239cb6SGreg Roach for (i = 0; i < 7; i++) { 57771239cb6SGreg Roach content += '<td '; 57871239cb6SGreg Roach content += 'class="descriptionbox"'; 57971239cb6SGreg Roach content += '>'; 58071239cb6SGreg Roach content += daysOfWeek[j]; 58171239cb6SGreg Roach content += '</td>'; 58271239cb6SGreg Roach j++; 58371239cb6SGreg Roach if (j > 6) { 58471239cb6SGreg Roach j = 0; 58571239cb6SGreg Roach } 58671239cb6SGreg Roach } 58771239cb6SGreg Roach content += '</tr>'; 58871239cb6SGreg Roach 58971239cb6SGreg Roach var tdate = new Date(date.getFullYear(), date.getMonth(), 1); 59071239cb6SGreg Roach var day = tdate.getDay(); 59171239cb6SGreg Roach day = day - weekStart; 59271239cb6SGreg Roach var daymilli = 1000 * 60 * 60 * 24; 59371239cb6SGreg Roach tdate = tdate.getTime() - (day * daymilli) + (daymilli / 2); 59471239cb6SGreg Roach tdate = new Date(tdate); 59571239cb6SGreg Roach 59671239cb6SGreg Roach for (j = 0; j < 6; j++) { 59771239cb6SGreg Roach content += '<tr>'; 59871239cb6SGreg Roach for (i = 0; i < 7; i++) { 59971239cb6SGreg Roach content += '<td '; 60071239cb6SGreg Roach if (tdate.getMonth() === date.getMonth()) { 60171239cb6SGreg Roach if (tdate.getDate() === date.getDate()) { 60271239cb6SGreg Roach content += 'class="descriptionbox"'; 60371239cb6SGreg Roach } else { 60471239cb6SGreg Roach content += 'class="optionbox"'; 60571239cb6SGreg Roach } 60671239cb6SGreg Roach } else { 60771239cb6SGreg Roach content += 'style="background-color:#EAEAEA; border: solid #AAAAAA 1px;"'; 60871239cb6SGreg Roach } 60971239cb6SGreg Roach content += '><a href="#" onclick="return cal_dateClicked(\'' + dateFieldId + '\', \'' + dateDivId + '\', ' + tdate.getFullYear() + ', ' + tdate.getMonth() + ', ' + tdate.getDate() + ');">'; 61071239cb6SGreg Roach content += tdate.getDate(); 61171239cb6SGreg Roach content += '</a></td>'; 61271239cb6SGreg Roach var datemilli = tdate.getTime() + daymilli; 61371239cb6SGreg Roach tdate = new Date(datemilli); 61471239cb6SGreg Roach } 61571239cb6SGreg Roach content += '</tr>'; 61671239cb6SGreg Roach } 61771239cb6SGreg Roach content += '</table>'; 61871239cb6SGreg Roach content += '</td></tr>'; 61971239cb6SGreg Roach content += '</table>'; 62071239cb6SGreg Roach 62171239cb6SGreg Roach return content; 62271239cb6SGreg Roach} 62371239cb6SGreg Roach 62471239cb6SGreg Roachfunction cal_setDateField(dateFieldId, year, month, day) 62571239cb6SGreg Roach{ 62671239cb6SGreg Roach var dateField = document.getElementById(dateFieldId); 62771239cb6SGreg Roach if (!dateField) { 62871239cb6SGreg Roach return false; 62971239cb6SGreg Roach } 63071239cb6SGreg Roach if (day < 10) { 63171239cb6SGreg Roach day = '0' + day; 63271239cb6SGreg Roach } 63371239cb6SGreg Roach dateField.value = day + ' ' + monthShort[month + 1] + ' ' + year; 63471239cb6SGreg Roach return false; 63571239cb6SGreg Roach} 63671239cb6SGreg Roach 63771239cb6SGreg Roachfunction cal_updateCalendar(dateFieldId, dateDivId) 63871239cb6SGreg Roach{ 63971239cb6SGreg Roach var dateSel = document.getElementById(dateFieldId + '_daySelect'); 64071239cb6SGreg Roach if (!dateSel) { 64171239cb6SGreg Roach return false; 64271239cb6SGreg Roach } 64371239cb6SGreg Roach var monthSel = document.getElementById(dateFieldId + '_monSelect'); 64471239cb6SGreg Roach if (!monthSel) { 64571239cb6SGreg Roach return false; 64671239cb6SGreg Roach } 64771239cb6SGreg Roach var yearInput = document.getElementById(dateFieldId + '_yearInput'); 64871239cb6SGreg Roach if (!yearInput) { 64971239cb6SGreg Roach return false; 65071239cb6SGreg Roach } 65171239cb6SGreg Roach 65271239cb6SGreg Roach var month = parseInt(monthSel.options[monthSel.selectedIndex].value, 10); 65371239cb6SGreg Roach month = month - 1; 65471239cb6SGreg Roach 65571239cb6SGreg Roach var date = new Date(yearInput.value, month, dateSel.options[dateSel.selectedIndex].value); 65671239cb6SGreg Roach cal_setDateField(dateFieldId, date.getFullYear(), date.getMonth(), date.getDate()); 65771239cb6SGreg Roach 65871239cb6SGreg Roach var dateDiv = document.getElementById(dateDivId); 65971239cb6SGreg Roach if (!dateDiv) { 66071239cb6SGreg Roach alert('no dateDiv ' + dateDivId); 66171239cb6SGreg Roach return false; 66271239cb6SGreg Roach } 66371239cb6SGreg Roach dateDiv.innerHTML = cal_generateSelectorContent(dateFieldId, dateDivId, date); 66471239cb6SGreg Roach 66571239cb6SGreg Roach return false; 66671239cb6SGreg Roach} 66771239cb6SGreg Roach 66871239cb6SGreg Roachfunction cal_dateClicked(dateFieldId, dateDivId, year, month, day) 66971239cb6SGreg Roach{ 67071239cb6SGreg Roach cal_setDateField(dateFieldId, year, month, day); 67171239cb6SGreg Roach calendarWidget(dateDivId, dateFieldId); 67271239cb6SGreg Roach return false; 67371239cb6SGreg Roach} 67471239cb6SGreg Roach 67571239cb6SGreg Roachfunction openerpasteid(id) 67671239cb6SGreg Roach{ 67771239cb6SGreg Roach if (window.opener.paste_id) { 67871239cb6SGreg Roach window.opener.paste_id(id); 67971239cb6SGreg Roach } 68071239cb6SGreg Roach window.close(); 68171239cb6SGreg Roach} 68271239cb6SGreg Roach 68371239cb6SGreg Roachfunction paste_id(value) 68471239cb6SGreg Roach{ 68571239cb6SGreg Roach pastefield.value = value; 68671239cb6SGreg Roach} 68771239cb6SGreg Roach 68871239cb6SGreg Roachfunction pastename(name) 68971239cb6SGreg Roach{ 69071239cb6SGreg Roach if (nameElement) { 69171239cb6SGreg Roach nameElement.innerHTML = name; 69271239cb6SGreg Roach } 69371239cb6SGreg Roach if (remElement) { 69471239cb6SGreg Roach remElement.style.display = 'block'; 69571239cb6SGreg Roach } 69671239cb6SGreg Roach} 69771239cb6SGreg Roach 69871239cb6SGreg Roachfunction paste_char(value) 69971239cb6SGreg Roach{ 70071239cb6SGreg Roach if (document.selection) { 70171239cb6SGreg Roach // IE 70271239cb6SGreg Roach pastefield.focus(); 70371239cb6SGreg Roach document.selection.createRange().text = value; 70471239cb6SGreg Roach } else if (pastefield.selectionStart || pastefield.selectionStart === 0) { 70571239cb6SGreg Roach // Mozilla/Chrome/Safari 70671239cb6SGreg Roach pastefield.value = 70771239cb6SGreg Roach pastefield.value.substring(0, pastefield.selectionStart) + 70871239cb6SGreg Roach value + 70971239cb6SGreg Roach pastefield.value.substring(pastefield.selectionEnd, pastefield.value.length); 71071239cb6SGreg Roach pastefield.selectionStart = pastefield.selectionEnd = pastefield.selectionStart + value.length; 71171239cb6SGreg Roach } else { 71271239cb6SGreg Roach // Fallback? - just append 71371239cb6SGreg Roach pastefield.value += value; 71471239cb6SGreg Roach } 71571239cb6SGreg Roach 71671239cb6SGreg Roach if (pastefield.id === 'NPFX' || pastefield.id === 'GIVN' || pastefield.id === 'SPFX' || pastefield.id === 'SURN' || pastefield.id === 'NSFX') { 71771239cb6SGreg Roach updatewholename(); 71871239cb6SGreg Roach } 71971239cb6SGreg Roach} 72071239cb6SGreg Roach 72171239cb6SGreg Roach/** 72271239cb6SGreg Roach * Persistant checkbox options to hide/show extra data. 72371239cb6SGreg Roach 72464490ee2SGreg Roach * @param element_id 72571239cb6SGreg Roach */ 72664490ee2SGreg Roachfunction persistent_toggle(element_id) 72771239cb6SGreg Roach{ 72864490ee2SGreg Roach let element = document.getElementById(element_id); 72964490ee2SGreg Roach let key = 'state-of-' + element_id; 73064490ee2SGreg Roach let state = localStorage.getItem(key); 73171239cb6SGreg Roach 73264490ee2SGreg Roach // Previously selected? 73364490ee2SGreg Roach if (state === 'true') { 73464490ee2SGreg Roach $(element).click(); 73571239cb6SGreg Roach } 73671239cb6SGreg Roach 73764490ee2SGreg Roach // Remember state for the next page load. 73864490ee2SGreg Roach $(element).on('change', function() { localStorage.setItem(key, element.checked); }); 73971239cb6SGreg Roach} 74071239cb6SGreg Roach 74171239cb6SGreg Roachfunction valid_lati_long(field, pos, neg) 74271239cb6SGreg Roach{ 74371239cb6SGreg Roach // valid LATI or LONG according to Gedcom standard 74471239cb6SGreg Roach // pos (+) : N or E 74571239cb6SGreg Roach // neg (-) : S or W 74671239cb6SGreg Roach var txt = field.value.toUpperCase(); 74771239cb6SGreg Roach txt = txt.replace(/(^\s*)|(\s*$)/g, ''); // trim 74871239cb6SGreg Roach txt = txt.replace(/ /g, ':'); // N12 34 ==> N12.34 74971239cb6SGreg Roach txt = txt.replace(/\+/g, ''); // +17.1234 ==> 17.1234 75071239cb6SGreg Roach txt = txt.replace(/-/g, neg); // -0.5698 ==> W0.5698 75171239cb6SGreg Roach txt = txt.replace(/,/g, '.'); // 0,5698 ==> 0.5698 75271239cb6SGreg Roach // 0°34'11 ==> 0:34:11 75371239cb6SGreg Roach txt = txt.replace(/\u00b0/g, ':'); // ° 75471239cb6SGreg Roach txt = txt.replace(/\u0027/g, ':'); // ' 75571239cb6SGreg Roach // 0:34:11.2W ==> W0.5698 75671239cb6SGreg Roach txt = txt.replace(/^([0-9]+):([0-9]+):([0-9.]+)(.*)/g, function ($0, $1, $2, $3, $4) { 75771239cb6SGreg Roach var n = parseFloat($1); 75871239cb6SGreg Roach n += ($2 / 60); 75971239cb6SGreg Roach n += ($3 / 3600); 76071239cb6SGreg Roach n = Math.round(n * 1E4) / 1E4; 76171239cb6SGreg Roach return $4 + n; 76271239cb6SGreg Roach }); 76371239cb6SGreg Roach // 0:34W ==> W0.5667 76471239cb6SGreg Roach txt = txt.replace(/^([0-9]+):([0-9]+)(.*)/g, function ($0, $1, $2, $3) { 76571239cb6SGreg Roach var n = parseFloat($1); 76671239cb6SGreg Roach n += ($2 / 60); 76771239cb6SGreg Roach n = Math.round(n * 1E4) / 1E4; 76871239cb6SGreg Roach return $3 + n; 76971239cb6SGreg Roach }); 77071239cb6SGreg Roach // 0.5698W ==> W0.5698 77171239cb6SGreg Roach txt = txt.replace(/(.*)([N|S|E|W]+)$/g, '$2$1'); 77271239cb6SGreg Roach // 17.1234 ==> N17.1234 77371239cb6SGreg Roach if (txt && txt.charAt(0) !== neg && txt.charAt(0) !== pos) { 77471239cb6SGreg Roach txt = pos + txt; 77571239cb6SGreg Roach } 77671239cb6SGreg Roach field.value = txt; 77771239cb6SGreg Roach} 77871239cb6SGreg Roach 77971239cb6SGreg Roach// This is the default way for webtrees to show image galleries. 78071239cb6SGreg Roach// Custom themes may use a different viewer. 78171239cb6SGreg Roachfunction activate_colorbox(config) 78271239cb6SGreg Roach{ 78371239cb6SGreg Roach $.extend($.colorbox.settings, { 78471239cb6SGreg Roach // Don't scroll window with document 78571239cb6SGreg Roach fixed: true, 78671239cb6SGreg Roach current: '', 78771239cb6SGreg Roach previous: '\uf048', 78871239cb6SGreg Roach next: '\uf051', 78971239cb6SGreg Roach slideshowStart: '\uf04b', 79071239cb6SGreg Roach slideshowStop: '\uf04c', 79171239cb6SGreg Roach close: '\uf00d' 79271239cb6SGreg Roach }); 79371239cb6SGreg Roach if (config) { 79471239cb6SGreg Roach $.extend($.colorbox.settings, config); 79571239cb6SGreg Roach } 79671239cb6SGreg Roach 79771239cb6SGreg Roach // Trigger an event when we click on an (any) image 79871239cb6SGreg Roach $('body').on('click', 'a.gallery', function () { 79971239cb6SGreg Roach // Enable colorbox for images 80071239cb6SGreg Roach $('a[type^=image].gallery').colorbox({ 80171239cb6SGreg Roach photo: true, 80271239cb6SGreg Roach maxWidth: '95%', 80371239cb6SGreg Roach maxHeight: '95%', 80471239cb6SGreg Roach rel: 'gallery', // Turn all images on the page into a slideshow 80571239cb6SGreg Roach slideshow: true, 80671239cb6SGreg Roach slideshowAuto: false, 80771239cb6SGreg Roach // Add wheelzoom to the displayed image 80871239cb6SGreg Roach onComplete: function () { 80971239cb6SGreg Roach // Disable click on image triggering next image 81071239cb6SGreg Roach // https://github.com/jackmoore/colorbox/issues/668 81171239cb6SGreg Roach $('.cboxPhoto').unbind('click'); 81271239cb6SGreg Roach 81371239cb6SGreg Roach wheelzoom(document.querySelectorAll('.cboxPhoto')); 81471239cb6SGreg Roach } 81571239cb6SGreg Roach }); 81671239cb6SGreg Roach 81771239cb6SGreg Roach // Enable colorbox for audio using <audio></audio>, where supported 81871239cb6SGreg Roach // $('html.video a[type^=video].gallery').colorbox({ 81971239cb6SGreg Roach // rel: 'nofollow' // Slideshows are just for images 82071239cb6SGreg Roach // }); 82171239cb6SGreg Roach 82271239cb6SGreg Roach // Enable colorbox for video using <video></video>, where supported 82371239cb6SGreg Roach // $('html.audio a[type^=audio].gallery').colorbox({ 82471239cb6SGreg Roach // rel: 'nofollow', // Slideshows are just for images 82571239cb6SGreg Roach // }); 82671239cb6SGreg Roach 82771239cb6SGreg Roach // Allow all other media types remain as download links 82871239cb6SGreg Roach }); 82971239cb6SGreg Roach} 83071239cb6SGreg Roach 83171239cb6SGreg Roach// Initialize autocomplete elements. 83271239cb6SGreg Roachfunction autocomplete(selector) 83371239cb6SGreg Roach{ 83471239cb6SGreg Roach // Use typeahead/bloodhound for autocomplete 83571239cb6SGreg Roach $(selector).each(function () { 836f4abaf12SGreg Roach let that = this; 83771239cb6SGreg Roach $(this).typeahead(null, { 83871239cb6SGreg Roach display: 'value', 83971239cb6SGreg Roach source: new Bloodhound({ 84071239cb6SGreg Roach datumTokenizer: Bloodhound.tokenizers.obj.whitespace('value'), 84171239cb6SGreg Roach queryTokenizer: Bloodhound.tokenizers.whitespace, 84271239cb6SGreg Roach remote: { 84371239cb6SGreg Roach url: this.dataset.autocompleteUrl, 844f4abaf12SGreg Roach replace: function(url, uriEncodedQuery) { 845f4abaf12SGreg Roach if (that.dataset.autocompleteExtra) { 846f4abaf12SGreg Roach let extra = $(document.querySelector(that.dataset.autocompleteExtra)).val(); 847f4abaf12SGreg Roach return url.replace("QUERY",uriEncodedQuery) + '&extra=' + encodeURIComponent(extra) 848f4abaf12SGreg Roach } 849f4abaf12SGreg Roach return url.replace("QUERY",uriEncodedQuery); 850f4abaf12SGreg Roach }, 851f4abaf12SGreg Roach wildcard: 'QUERY', 852f4abaf12SGreg Roach 85371239cb6SGreg Roach } 85471239cb6SGreg Roach }) 85571239cb6SGreg Roach }); 85671239cb6SGreg Roach }); 85771239cb6SGreg Roach} 85871239cb6SGreg Roach 85971239cb6SGreg Roach/** 86071239cb6SGreg Roach * Insert text at the current cursor position in an input field. 86171239cb6SGreg Roach * 86271239cb6SGreg Roach * @param e The input element. 86371239cb6SGreg Roach * @param t The text to insert. 86471239cb6SGreg Roach */ 86571239cb6SGreg Roachfunction insertTextAtCursor(e, t) 86671239cb6SGreg Roach{ 86771239cb6SGreg Roach var scrollTop = e.scrollTop; 86871239cb6SGreg Roach var selectionStart = e.selectionStart; 86971239cb6SGreg Roach var prefix = e.value.substring(0, selectionStart); 87071239cb6SGreg Roach var suffix = e.value.substring(e.selectionEnd, e.value.length); 87171239cb6SGreg Roach e.value = prefix + t + suffix; 87271239cb6SGreg Roach e.selectionStart = selectionStart + t.length; 87371239cb6SGreg Roach e.selectionEnd = e.selectionStart; 87471239cb6SGreg Roach e.focus(); 87571239cb6SGreg Roach e.scrollTop = scrollTop; 87671239cb6SGreg Roach} 87771239cb6SGreg Roach 87888de55fdSRico Sonntag 87988de55fdSRico Sonntag/** 88088de55fdSRico Sonntag * Draws a google pie chart. 88188de55fdSRico Sonntag * 88288de55fdSRico Sonntag * @param {String} elementId The element id of the HTML element the chart is rendered too 88388de55fdSRico Sonntag * @param {Array} data The chart data array 88488de55fdSRico Sonntag * @param {Array} colors The chart color array 88588de55fdSRico Sonntag * @param {String} title The chart title 88688de55fdSRico Sonntag * @param {String} labeledValueText The type of how to display the slice text 88788de55fdSRico Sonntag */ 88888de55fdSRico Sonntagfunction drawPieChart(elementId, data, colors, title, labeledValueText) 88988de55fdSRico Sonntag{ 89088de55fdSRico Sonntag var data = google.visualization.arrayToDataTable(data); 89188de55fdSRico Sonntag var options = { 89288de55fdSRico Sonntag title: title, 89388de55fdSRico Sonntag height: '100%', 89488de55fdSRico Sonntag width: '100%', 89588de55fdSRico Sonntag pieStartAngle: 0, 89688de55fdSRico Sonntag pieSliceText: 'none', 89788de55fdSRico Sonntag pieSliceTextStyle: { 89888de55fdSRico Sonntag color: '#777' 89988de55fdSRico Sonntag }, 90088de55fdSRico Sonntag pieHole: 0.4, // Donut 90188de55fdSRico Sonntag //is3D: true, // 3D (not together with pieHole) 90288de55fdSRico Sonntag legend: { 90388de55fdSRico Sonntag alignment: 'center', 90488de55fdSRico Sonntag // Flickers on mouseover :( 90588de55fdSRico Sonntag labeledValueText: labeledValueText || 'value', 90688de55fdSRico Sonntag position: 'labeled' 90788de55fdSRico Sonntag }, 90888de55fdSRico Sonntag chartArea: { 90988de55fdSRico Sonntag left: 0, 91088de55fdSRico Sonntag top: '5%', 91188de55fdSRico Sonntag height: '90%', 91288de55fdSRico Sonntag width: '100%' 91388de55fdSRico Sonntag }, 91488de55fdSRico Sonntag tooltip: { 91588de55fdSRico Sonntag trigger: 'none', 91688de55fdSRico Sonntag text: 'both' 91788de55fdSRico Sonntag }, 91888de55fdSRico Sonntag backgroundColor: 'transparent', 91988de55fdSRico Sonntag colors: colors 92088de55fdSRico Sonntag }; 92188de55fdSRico Sonntag 92288de55fdSRico Sonntag var chart = new google.visualization.PieChart(document.getElementById(elementId)); 92388de55fdSRico Sonntag 92488de55fdSRico Sonntag chart.draw(data, options); 92588de55fdSRico Sonntag} 92688de55fdSRico Sonntag 92788de55fdSRico Sonntag/** 92888de55fdSRico Sonntag * Draws a google column chart. 92988de55fdSRico Sonntag * 93088de55fdSRico Sonntag * @param {String} elementId The element id of the HTML element the chart is rendered too 93188de55fdSRico Sonntag * @param {Array} data The chart data array 93288de55fdSRico Sonntag * @param {Object} options The chart specific options to overwrite the default ones 93388de55fdSRico Sonntag */ 93488de55fdSRico Sonntagfunction drawColumnChart(elementId, data, options) 93588de55fdSRico Sonntag{ 93688de55fdSRico Sonntag var defaults = { 93788de55fdSRico Sonntag title: '', 93888de55fdSRico Sonntag subtitle: '', 939a81e5019SRico Sonntag titleTextStyle: { 940a81e5019SRico Sonntag color: '#757575', 941a81e5019SRico Sonntag fontName: 'Roboto', 942a81e5019SRico Sonntag fontSize: '16px', 943a81e5019SRico Sonntag bold: false, 944a81e5019SRico Sonntag italic: false 945a81e5019SRico Sonntag }, 94688de55fdSRico Sonntag height: '100%', 94788de55fdSRico Sonntag width: '100%', 94888de55fdSRico Sonntag vAxis: { 94988de55fdSRico Sonntag title: '' 95088de55fdSRico Sonntag }, 95188de55fdSRico Sonntag hAxis: { 95288de55fdSRico Sonntag title: '' 95388de55fdSRico Sonntag }, 95488de55fdSRico Sonntag legend: { 95588de55fdSRico Sonntag position: 'none' 95688de55fdSRico Sonntag }, 95788de55fdSRico Sonntag backgroundColor: 'transparent' 95888de55fdSRico Sonntag }; 95988de55fdSRico Sonntag 96088de55fdSRico Sonntag options = Object.assign(defaults, options); 96188de55fdSRico Sonntag 96288de55fdSRico Sonntag var chart = new google.visualization.ColumnChart(document.getElementById(elementId)); 96388de55fdSRico Sonntag var data = google.visualization.arrayToDataTable(data); 96488de55fdSRico Sonntag 96588de55fdSRico Sonntag chart.draw(data, options); 96688de55fdSRico Sonntag} 96788de55fdSRico Sonntag 96888de55fdSRico Sonntag/** 96988de55fdSRico Sonntag * Draws a google combo chart. 97088de55fdSRico Sonntag * 97188de55fdSRico Sonntag * @param {String} elementId The element id of the HTML element the chart is rendered too 97288de55fdSRico Sonntag * @param {Array} data The chart data array 97388de55fdSRico Sonntag * @param {Object} options The chart specific options to overwrite the default ones 97488de55fdSRico Sonntag */ 97588de55fdSRico Sonntagfunction drawComboChart(elementId, data, options) 97688de55fdSRico Sonntag{ 97788de55fdSRico Sonntag var defaults = { 97888de55fdSRico Sonntag title: '', 97988de55fdSRico Sonntag subtitle: '', 98088de55fdSRico Sonntag titleTextStyle: { 98188de55fdSRico Sonntag color: '#757575', 98288de55fdSRico Sonntag fontName: 'Roboto', 98388de55fdSRico Sonntag fontSize: '16px', 98488de55fdSRico Sonntag bold: false, 98588de55fdSRico Sonntag italic: false 98688de55fdSRico Sonntag }, 98788de55fdSRico Sonntag height: '100%', 98888de55fdSRico Sonntag width: '100%', 98988de55fdSRico Sonntag vAxis: { 99088de55fdSRico Sonntag title: '' 99188de55fdSRico Sonntag }, 99288de55fdSRico Sonntag hAxis: { 99388de55fdSRico Sonntag title: '' 99488de55fdSRico Sonntag }, 99588de55fdSRico Sonntag legend: { 99688de55fdSRico Sonntag position: 'none' 99788de55fdSRico Sonntag }, 99888de55fdSRico Sonntag seriesType: 'bars', 99988de55fdSRico Sonntag series: { 100088de55fdSRico Sonntag 2: { 100188de55fdSRico Sonntag type: 'line' 100288de55fdSRico Sonntag } 100388de55fdSRico Sonntag }, 100488de55fdSRico Sonntag colors: [], 100588de55fdSRico Sonntag backgroundColor: 'transparent' 100688de55fdSRico Sonntag }; 100788de55fdSRico Sonntag 100888de55fdSRico Sonntag options = Object.assign(defaults, options); 100988de55fdSRico Sonntag 101088de55fdSRico Sonntag var chart = new google.visualization.ComboChart(document.getElementById(elementId)); 101188de55fdSRico Sonntag var data = google.visualization.arrayToDataTable(data); 101288de55fdSRico Sonntag 101388de55fdSRico Sonntag chart.draw(data, options); 101488de55fdSRico Sonntag} 101588de55fdSRico Sonntag 1016a81e5019SRico Sonntag/** 1017a81e5019SRico Sonntag * Draws a google geo chart. 1018a81e5019SRico Sonntag * 1019a81e5019SRico Sonntag * @param {String} elementId The element id of the HTML element the chart is rendered too 1020a81e5019SRico Sonntag * @param {Array} data The chart data array 1021a81e5019SRico Sonntag * @param {Object} options The chart specific options to overwrite the default ones 1022a81e5019SRico Sonntag */ 1023a81e5019SRico Sonntagfunction drawGeoChart(elementId, data, options) 1024a81e5019SRico Sonntag{ 1025a81e5019SRico Sonntag var defaults = { 1026a81e5019SRico Sonntag title: '', 1027a81e5019SRico Sonntag subtitle: '', 1028a81e5019SRico Sonntag height: '100%', 1029a81e5019SRico Sonntag width: '100%' 1030a81e5019SRico Sonntag }; 1031a81e5019SRico Sonntag 1032a81e5019SRico Sonntag options = Object.assign(defaults, options); 1033a81e5019SRico Sonntag 1034a81e5019SRico Sonntag var chart = new google.visualization.GeoChart(document.getElementById(elementId)); 1035a81e5019SRico Sonntag var data = google.visualization.arrayToDataTable(data); 1036a81e5019SRico Sonntag 1037a81e5019SRico Sonntag chart.draw(data, options); 1038a81e5019SRico Sonntag} 103988de55fdSRico Sonntag 104071239cb6SGreg Roach// Send the CSRF token on all AJAX requests 104171239cb6SGreg Roach$.ajaxSetup({ 104271239cb6SGreg Roach headers: { 104371239cb6SGreg Roach 'X-CSRF-TOKEN': $('meta[name=csrf]').attr('content') 104471239cb6SGreg Roach } 104571239cb6SGreg Roach}); 104671239cb6SGreg Roach 104771239cb6SGreg Roach// Initialisation 104871239cb6SGreg Roach$(function () { 104971239cb6SGreg Roach // Page elements that load automaticaly via AJAX. 105071239cb6SGreg Roach // This prevents bad robots from crawling resource-intensive pages. 105171239cb6SGreg Roach $("[data-ajax-url]").each(function () { 105271239cb6SGreg Roach $(this).load($(this).data('ajaxUrl')); 105371239cb6SGreg Roach }); 105471239cb6SGreg Roach 105571239cb6SGreg Roach // Select2 - format entries in the select list 105671239cb6SGreg Roach function templateOptionForSelect2(data) 105771239cb6SGreg Roach { 105871239cb6SGreg Roach if (data.loading) { 105971239cb6SGreg Roach // If we're waiting for the server, this will be a "waiting..." message 106071239cb6SGreg Roach return data.text; 106171239cb6SGreg Roach } else { 106271239cb6SGreg Roach // The response from the server is already in HTML, so no need to format it here. 106371239cb6SGreg Roach return data.text; 106471239cb6SGreg Roach } 106571239cb6SGreg Roach } 106671239cb6SGreg Roach 106771239cb6SGreg Roach // Autocomplete 106871239cb6SGreg Roach autocomplete('input[data-autocomplete-url]'); 106971239cb6SGreg Roach 107071239cb6SGreg Roach // Select2 - activate autocomplete fields 107171239cb6SGreg Roach $('select.select2').select2({ 107271239cb6SGreg Roach // Do not escape. 107371239cb6SGreg Roach escapeMarkup: function (x) { 107471239cb6SGreg Roach return x } 107571239cb6SGreg Roach // Same formatting for both selections and rsult 107671239cb6SGreg Roach //templateResult: templateOptionForSelect2, 107771239cb6SGreg Roach //templateSelection: templateOptionForSelect2 107871239cb6SGreg Roach }) 107971239cb6SGreg Roach // If we clear the select (using the "X" button), we need an empty 108071239cb6SGreg Roach // value (rather than no value at all) for inputs with name="array[]" 108171239cb6SGreg Roach .on('select2:unselect', function (evt) { 108271239cb6SGreg Roach $(evt.delegateTarget).append('<option value="" selected="selected"></option>'); 108371239cb6SGreg Roach }) 108471239cb6SGreg Roach 108571239cb6SGreg Roach // Datatables - locale aware sorting 108671239cb6SGreg Roach $.fn.dataTableExt.oSort['text-asc'] = function (x, y) { 108771239cb6SGreg Roach return x.localeCompare(y, document.documentElement.lang, {'sensitivity': 'base'}); 108871239cb6SGreg Roach }; 108971239cb6SGreg Roach $.fn.dataTableExt.oSort['text-desc'] = function (x, y) { 109071239cb6SGreg Roach return y.localeCompare(x, document.documentElement.lang, {'sensitivity': 'base'}); 109171239cb6SGreg Roach }; 109271239cb6SGreg Roach 109371239cb6SGreg Roach // DataTables - start hidden to prevent FOUC. 109471239cb6SGreg Roach $('table.datatables').each(function () { 109571239cb6SGreg Roach $(this).DataTable(); $(this).removeClass('d-none'); }); 109671239cb6SGreg Roach 109771239cb6SGreg Roach // Create a new record while editing an existing one. 109871239cb6SGreg Roach // Paste the XREF and description into the Select2 element. 109971239cb6SGreg Roach $('.wt-modal-create-record').on('show.bs.modal', function (event) { 110071239cb6SGreg Roach // Find the element ID that needs to be updated with the new value. 110171239cb6SGreg Roach $('form', $(this)).data('element-id', $(event.relatedTarget).data('element-id')); 110271239cb6SGreg Roach $('form .form-group input:first', $(this)).focus(); 110371239cb6SGreg Roach }); 110471239cb6SGreg Roach 110571239cb6SGreg Roach // Submit the modal form using AJAX, and paste the returned record ID/NAME into the parent form. 110671239cb6SGreg Roach $('.wt-modal-create-record form').on('submit', function (event) { 110771239cb6SGreg Roach event.preventDefault(); 110871239cb6SGreg Roach var elementId = $(this).data('element-id'); 110971239cb6SGreg Roach $.ajax({ 111071239cb6SGreg Roach url: 'index.php', 111171239cb6SGreg Roach type: 'POST', 111271239cb6SGreg Roach data: new FormData(this), 111371239cb6SGreg Roach async: false, 111471239cb6SGreg Roach cache: false, 111571239cb6SGreg Roach contentType: false, 111671239cb6SGreg Roach processData: false, 111771239cb6SGreg Roach success: function (data) { 111871239cb6SGreg Roach $('#' + elementId).select2().empty().append(new Option(data.text, data.id)).val(data.id).trigger('change'); 111971239cb6SGreg Roach }, 112071239cb6SGreg Roach failure: function (data) { 112171239cb6SGreg Roach alert(data.error_message); 112271239cb6SGreg Roach } 112371239cb6SGreg Roach }); 112471239cb6SGreg Roach // Clear the form 112571239cb6SGreg Roach this.reset(); 112671239cb6SGreg Roach // Close the modal 112771239cb6SGreg Roach $(this).closest('.wt-modal-create-record').modal('hide'); 112871239cb6SGreg Roach }); 112971239cb6SGreg Roach 113071239cb6SGreg Roach // Activate the langauge selection menu. 113171239cb6SGreg Roach $('.menu-language').on('click', '[data-language]', function () { 113271239cb6SGreg Roach $.post('index.php', { 113371239cb6SGreg Roach route: 'language', 113471239cb6SGreg Roach language: $(this).data('language') 113571239cb6SGreg Roach }, function () { 1136070932ceSGreg Roach document.location.reload(); 113771239cb6SGreg Roach }); 113871239cb6SGreg Roach 113971239cb6SGreg Roach return false; 114071239cb6SGreg Roach }); 114171239cb6SGreg Roach 114271239cb6SGreg Roach // Activate the theme selection menu. 114371239cb6SGreg Roach $('.menu-theme').on('click', '[data-theme]', function () { 114471239cb6SGreg Roach $.post('index.php', { 114571239cb6SGreg Roach route: 'theme', 114671239cb6SGreg Roach theme: $(this).data('theme') 114771239cb6SGreg Roach }, function () { 1148070932ceSGreg Roach document.location.reload(); 114971239cb6SGreg Roach }); 115071239cb6SGreg Roach 115171239cb6SGreg Roach return false; 115271239cb6SGreg Roach }); 115371239cb6SGreg Roach 115471239cb6SGreg Roach // Activate the on-screen keyboard 115571239cb6SGreg Roach var osk_focus_element; 115671239cb6SGreg Roach $('.wt-osk-trigger').click(function () { 115771239cb6SGreg Roach // When a user clicks the icon, set focus to the corresponding input 115871239cb6SGreg Roach osk_focus_element = document.getElementById($(this).data('id')); 115971239cb6SGreg Roach osk_focus_element.focus(); 116071239cb6SGreg Roach $('.wt-osk').show(); 116171239cb6SGreg Roach 116271239cb6SGreg Roach }); 116371239cb6SGreg Roach 116471239cb6SGreg Roach $('.wt-osk-script-button').change(function () { 116571239cb6SGreg Roach $('.wt-osk-script').prop('hidden', true); 116671239cb6SGreg Roach $('.wt-osk-script-' + $(this).data('script')).prop('hidden', false); 116771239cb6SGreg Roach }); 116871239cb6SGreg Roach $('.wt-osk-shift-button').click(function () { 116971239cb6SGreg Roach document.querySelector('.wt-osk-keys').classList.toggle('shifted'); 117071239cb6SGreg Roach }); 117171239cb6SGreg Roach $('.wt-osk-keys').on('click', '.wt-osk-key', function () { 117271239cb6SGreg Roach var key = $(this).contents().get(0).nodeValue; 117371239cb6SGreg Roach var shift_state = $('.wt-osk-shift-button').hasClass('active'); 117471239cb6SGreg Roach var shift_key = $('sup', this)[0]; 117571239cb6SGreg Roach if (shift_state && shift_key !== undefined) { 117671239cb6SGreg Roach key = shift_key.innerText; 117771239cb6SGreg Roach } 117871239cb6SGreg Roach if (osk_focus_element !== null) { 117971239cb6SGreg Roach var cursorPos = osk_focus_element.selectionStart; 118071239cb6SGreg Roach var v = osk_focus_element.value; 118171239cb6SGreg Roach var textBefore = v.substring(0, cursorPos); 118271239cb6SGreg Roach var textAfter = v.substring(cursorPos, v.length); 118371239cb6SGreg Roach osk_focus_element.value = textBefore + key + textAfter; 118471239cb6SGreg Roach if ($('.wt-osk-pin-button').hasClass('active') === false) { 118571239cb6SGreg Roach $('.wt-osk').hide(); 118671239cb6SGreg Roach } 118771239cb6SGreg Roach } 118871239cb6SGreg Roach }); 118971239cb6SGreg Roach 119071239cb6SGreg Roach $('.wt-osk-close').on('click', function () { 119171239cb6SGreg Roach $('.wt-osk').hide(); 119271239cb6SGreg Roach }); 119371239cb6SGreg Roach}); 1194