xref: /webtrees/resources/js/webtrees.js (revision 59e18f0c2efdecfc20593b681d464c0c9736d2f9)
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