Greasy Fork is available in English.

WME URComments-Enhanced

Handle WME update requests more quickly and efficiently.

اعتبارا من 2018-12-18 23:35:41 UTC. شاهد أحدث إصدار.

// ==UserScript==
// @name        WME URComments-Enhanced
// @namespace   daniel@dbsooner.com
// @version     2018.12.18.04
// @description Handle WME update requests more quickly and efficiently.
// @grant       none
// @include     /^https:\/\/(www|beta)\.waze\.com\/(?!user\/)(.{2,6}\/)?editor\/?.*$/
// @require     https://greasyfork.org/scripts/24851-wazewrap/code/WazeWrap.js
// @require     https://apis.google.com/js/api.js
// @author      dBsooner
// @license     MIT/BSD/X11
// @icon        
// @contributionURL https://github.com/WazeDev/Thank-The-Authors
// ==/UserScript==

/*
 * Original concept and code for URComments (URC) was written by rickzabel and licensed under MIT/BSD/X11.
 * This script is a ground-up rewrite of URC. Special thanks is definitely given to rickzabel for his hard
 * work and dedication to the original script. You can reach him at rickzabel@gmail.com.
 *
 */

/*
 * Portions of this script were inspired by URO+ written by Twister-UK. Credit is given to him and his team
 * for development of that script. Where code was directly copied, function name or variable name was retained
 * and a comment was placed before the code. The code that was inspired by was a complete rewrite using URO+
 * as a reference for the logic. URO+ is located at: https://greasyfork.org/en/scripts/1952-uroverview-plus-uro
 *
 */

/* global GM_info */
/* global W */
/* global I18n */
/* global $ */
/* global WazeWrap */
/* global OL */
/* global gapi */

(function() {
    'use strict';

    const SCRIPT_AUTHOR = GM_info.script.author;
    const SETTINGS_STORE_NAME = "WME_URC-E";
    const ALERT_UPDATE = true;
    const SCRIPT_VERSION = GM_info.script.version;
    const SCRIPT_VERSION_CHANGES = [ 'Initial release of URComments-Enhanced.','Google API v4', 'Added Dutch list' ];
    const DOUBLE_CLICK_ICON = '';
    const DEBUG = true;
    const LOAD_BEGIN_TIME = performance.now();

    const URCE_CLIENT_ID ='309982510721-uefelja68jujfalgeqjagit4dff7egv5.apps.googleusercontent.com';
    const URCE_API_KEY = 'AIzaSyA2xOeUfopDqhB8r8esEa2A-G0X64UMr1c';
    const URCE_DISCOVERY_DOCS = [ 'https://sheets.googleapis.com/$discovery/rest?version=v4' ];
    const URCE_SCOPES = 'https://www.googleapis.com/auth/spreadsheets.readonly';
    const URCE_SPREADSHEET_ID = '1aVKBOwjYmO88x96fIHtIQgAwMaCV_NfklvPqf0J0pzQ';

    let _selUr = {};
    let _settings = {};
    let _commentList = [];
    let _alertBoxArray = [];
    let _markerStackArray = [];
    let _customMarkerArray = [];
    let _defaultComments = {
        'dr': { 'commentNum':null, 'urNum':98 }, // Default reminder
        'dc': { 'commentNum':null, 'urNum':99 }, // Default closed / not identified
        'it': { 'commentNum':null, 'urNum':6 }, // Incorrect turn
        'ia': { 'commentNum':null, 'urNum':7 }, // Incorrect address
        'ir': { 'commentNum':null, 'urNum':8 }, // Incorrect route
        'mra': { 'commentNum':null, 'urNum':9 }, // Missing roundabout
        'ge': { 'commentNum':null, 'urNum':10 }, // General error
        'tna': { 'commentNum':null, 'urNum':11 }, // Turn not allowed
        'ij': { 'commentNum':null, 'urNum':12 }, // Incorrect junction
        'mbo': { 'commentNum':null, 'urNum':13 }, // Missing bridge overpass
        'wdd': { 'commentNum':null, 'urNum':14 }, // Wrong driving direction
        'me': { 'commentNum':null, 'urNum':15 }, // Missing exit
        'mr': { 'commentNum':null, 'urNum':16 }, // Missing road
        'ml': { 'commentNum':null, 'urNum':18 }, // Missing landmark
        'br': { 'commentNum':null, 'urNum':19 }, // Blocked road
        'msn': { 'commentNum':null, 'urNum':21 }, // Missing street name
        'isps': { 'commentNum':null, 'urNum':22 }, // Incorrect street prefix or suffix
        'sl': { 'commentNum':null, 'urNum':23 } // Speed Limit
    };
    let _urceInitialized = false;
    let _commentListLoaded = false;
    let _alertBoxInUse = false;
    let _unstackedMasterId = null;
    let _mousedOverMarkerId = null;
    let _mousedOverMarkerType = null;
    let _restoreZoom, _$restoreTab, _restoreTabPosition, _wmeUserId;
    let _commentLists = [{idx:0, name:'CommentTeam', status:'enabled', oldVarName:'CommentTeam', listOwner:'CommentTeam', region:'ALL', gSheetRange:'CommentTeam_Output_(do_not_edit)!A1:A' },
                         {idx:1, name:'Custom', status:'enabled', oldVarName:'Custom', listOwner:'Custom', gSheetUrl:'', type:'static' },
                         {idx:2, name:'USA - SCR', status:'enabled', oldVarName:'USA_SouthCentral', listOwner:'SCR CommentTeam', region:'USA_SCR', gSheetRange:'USA_SCR_Output_(do_not_edit)!A1:A' },
                         {idx:3, name:'USA - SER', status:'enabled', oldVarName:'USA_Southeast', listOwner:'itzwolf', region:'USA_SER', gSheetRange:'USA_SER_Output_(do_not_edit)!A1:A' },
                         {idx:4, name:'User: PesachZ', status:'enabled', oldVarName:'PesachZ', listOwner:'PesachZ', region:'ALL', gSheetRange:'PesachZ_Output_(do_not_edit)!A1:A' },
                         {idx:5, name:'Dutch', status:'enabled', oldVarName:'Dutch', listOwner:'WimVandierendonck', region:'ROW_BE', gSheetRange:'Dutch_Output_(do_not_edit)!A1:A' }
                          ].sort(dynamicSort('name'));

    function log(message) { console.log('URC-E:', message); }
    function logError(message) { console.error('URC-E:', message); }
    function logDebug(message) { if (DEBUG) console.log('URC-E:', message); }
    function logWarning(message) { console.warn('URC-E:', message); }

    function dynamicSort(property) {
        let sortOrder = 1;
        if (property[0] === "-") {
            sortOrder = -1;
            property = property.substr(1);
        }
        return function (a,b) {
            if (sortOrder == -1) {
                return b[property].localeCompare(a[property]);
            } else {
                return a[property].localeCompare(b[property]);
            }
        }
    }

    function loadSettingsFromStorage() {
        logDebug('Loading settings from storage.');
        let convertUrcSettings = false;
        let loadedSettings = $.parseJSON(localStorage.getItem(SETTINGS_STORE_NAME));
        let defaultSettings = {
            //Comment List
            commentList: 0,
            commentListStyle: 'default',
            commentListCollapses: {},
            // URC-E Preferences
            autoCenterOnUr: false,
            autoClickOpenSolvedNi: false,
            autoCloseCommentWindow: false,
            autoSaveAfterSolvedOrNiComment: false,
            autoSendReminders: false,
            autoSetNewUrComment: false,
            autoSetReminderUrComment:false,
            autoSwitchToUrCommentsTab: false,
            autoZoomInOnNewUr: false,
            autoZoomOutAfterComment: false,
            disableDoneNextButtons: false,
            doubleClickLinkNiComments: false,
            doubleClickLinkOpenComments: false,
            doubleClickLinkSolvedComments: false,
            hideZoomOutLinks: false,
            unfollowUrAfterSend: false,
            // UR Marker Prefs
            enableUrPillCounts: false,
            doNotShowTagNameOnPill: false,
            replaceTagNameWithEditorName: false,
            unstackMarkers: false,
            customMarkersRoadworks: false,
            customMarkersConstruction: false,
            customMarkersClosures: false,
            customMarkersEvents: false,
            customMarkersNotes: false,
            customMarkersWslm: false,
            customMarkersBog: false,
            customMarkersDifficult: false,
            customMarkersNativeSl: false,
            // UR Filtering Prefs
            enableUrceUrFiltering: false,
            hideOutsideEditableArea: false,
            doNotHideSelectedUr: false,
            // -- Lifecycle
            hideWaiting: false,
            hideUrsCloseNeeded: false,
            hideUrsReminderNeeded: false,
            // -- Hide by status
            hideByStatusOpen: false,
            hideByStatusClosed: false,
            hideByStatusNotIdentified: false,
            hideByStatusSolved: false,
            // -- Hide by type
            hideByTypeBlockedRoad: false,
            hideByTypeGeneralError: false,
            hideByTypeIncorrectAddress: false,
            hideByTypeIncorrectJunction: false,
            hideByTypeIncorrectRoute: false,
            hideByTypeIncorrectTurn: false,
            hideByTypeMissingBridgeOverpass: false,
            hideByTypeMissingExit: false,
            hideByTypeMissingLandmark: false,
            hideByTypeMissingOrInvalidSpeedLimit: false,
            hideByTypeMissingRoad: false,
            hideByTypeMissingRoundabout: false,
            hideByTypeTurnNotAllowed: false,
            hideByTypeUndefined: false,
            hideByTypeWazeAutomatic: false,
            hideByTypeWrongDrivingDirection: false,
            // -- Hide by tag
            hideByTaggedBog: false,
            hideByTaggedClosure: false,
            hideByTaggedConstruction: false,
            hideByTaggedDifficult: false,
            hideByTaggedEvent: false,
            hideByTaggedNote: false,
            hideByTaggedRoadworks: false,
            hideByTaggedWslm: false,
            // -- Hide by age of submission
            hideByAgeOfSubmissionLessThan: false,
            hideByAgeOfSubmissionLessThanDaysOld: null,
            hideByAgeOfSubmissionMoreThan: false,
            hideByAgeOfSubmissionMoreThanDaysOld: null,
            // -- Hide by Descriptions / Comments / Following
            hideFollowing: false,
            hideNotFollowing: false,
            hideWithDescription: false,
            hideWithoutDeescription: false,
            hideWithCommentsFromMe: false,
            hideWithoutCommentsFromMe: false,
            hideFirstCommentByMe: false,
            hideFirstCommentNotByMe: false,
            hideLastCommentByMe: false,
            hideLastCommentNotByMe: false,
            hideLastCommentByReporter: false,
            hideLastCommentNotByReporter: false,
            hideByCommentCountLessThan: false,
            hideByCommentCountLessThanNumber: null,
            hideByCommentCountMoreThan: false,
            hideByCommentCountMoreThanNumber: null,
            hideByAgeOfFirstCommentLessThan: false,
            hideByAgeOfFirstCommentLessThanDaysOld: null,
            hideByAgeOfFirstCommentMoreThan: false,
            hideByAgeOfFirstCommentMoreThanDaysOld: null,
            hideByAgeOfLastCommentLessThan: false,
            hideByAgeOfLastCommentLessThanDaysOld: null,
            hideByAgeOfLastCommentMoreThan: false,
            hideByAgeOfLastCommentMoreThanDaysOld: null,
            // Common Prefs
            reminderDays: 0,
            closeDays: 7,
            wmeUserId: null,
            lastVersion: null
        };
         _settings = loadedSettings ? loadedSettings : defaultSettings;
        for (let prop in defaultSettings) {
            if (!_settings.hasOwnProperty(prop)) {
                _settings[prop] = defaultSettings[prop];
            }
        }
        if (_settings.reminderDays > (_settings.closeDays - 1)) {
            _settings.reminderDays = (_settings.closeDays - 1) < 0 ? 0 : (_settings.closeDays - 1);
        }
        _settings.reminderDays = Math.min(13,Math.max(0,parseInt(_settings.reminderDays)));
        if (_settings.closeDays < (_settings.reminderDays + 1)) {
            _settings.closeDays = (_settings.reminderDays + 1) > 14 ? 14 : (_settings.reminderDays + 1);
        }
        _settings.closeDays = Math.min(14,Math.max(1,parseInt(_settings.closeDays)));
        if (_settings.wmeUserId !== _wmeUserId) _settings.wmeUserId = _wmeUserId;
        ['hideClosedUrs', 'showOthersUrsPastReminderClose', 'onlyShowMyUrs', 'hideTaggedUrs', 'hideUrsWoComments', 'hideUrsWoCommentsOrDescriptions', 'hideUrsWoCommentsWithDescriptions', 'hideUrsWithUserReplies' ].forEach(oldSetting => {
            if (_settings.hasOwnProperty(oldSetting)) delete(_settings[oldSetting]);
        });
    }

    async function saveSettingsToStorage() {
        if (localStorage) {
            if (_settings.commentListCollapses === undefined) _settings.commentListCollapses = {};
            _settings.commentListCollapses[_settings.commentList] = await getCollapsedGroups();
            _settings.lastVersion = SCRIPT_VERSION;
            localStorage.setItem(SETTINGS_STORE_NAME, JSON.stringify(_settings));
            logDebug('Settings saved.');
        }
    }

    function closeAlertBox() {
        $('#urceAlertBoxHeader').empty();
        $('#urceAlertBoxContent').empty();
        $('#urceAlertTickBtnCaption').text('');
        $('#urceAlertCrossBtnCaption').text('');
        $('#urceAlertBox').css('visibility', 'hidden');
        $('#urceAlertCrossBtn').css('visibility', 'hidden');
        _alertBoxInUse = false;
        if (_alertBoxArray.length > 0) buildAlertBoxFromArray();
    }

    function buildAlertBoxFromArray() {
        _alertBoxInUse = true;
        let alertBoxTickAction = null;
        let alertBoxCrossAction = null;
        $('#urceAlertBoxHeader').append(
            $('<span>').append(
                $('<i>', {class:'fa ' + _alertBoxArray[0].headerIcon})
            ).append(' ' + _alertBoxArray[0].title)
        );
        $('#urceAlertBoxContent').html(_alertBoxArray[0].content);
        $('#urceAlertTickBtnCaption').text(_alertBoxArray[0].tickText);
        if (typeof _alertBoxArray[0].tickAction === 'function') alertBoxTickAction = _alertBoxArray[0].tickAction;
        if (_alertBoxArray[0].hasCross) {
            $('#urceAlertCrossBtnCaption').text(_alertBoxArray[0].crossText);
            $('#urceAlertCrossBtn').css('visibility', 'visible');
            if(typeof _alertBoxArray[0].crossAction === "function") alertBoxCrossAction = _alertBoxArray[0].crossAction;
        } else {
            $('#urceAlertCrossBtn').css('visibility', 'hidden');
        }
        $('#urceAlertTickBtn').off('click');
        $('#urceAlertTickBtn').on('click', function() {
            if (alertBoxTickAction !== null) alertBoxTickAction();
            closeAlertBox();
        });
        $('#urceAlertCrossBtn').off('click');
        $('#urceAlertCrossBtn').on('click', function() {
            if (alertBoxCrossAction !== null) alertBoxCrossAction();
            closeAlertBox();
        });
        $('#urceAlertBox').css('visibility', 'visible');
        _alertBoxArray.shift();
    }

    function showAlertBox(headerIcon, title, content, hasCross, tickText, crossText, tickAction, crossAction) {
        _alertBoxArray.push( { headerIcon: headerIcon, title: title, content: content, hasCross: hasCross, tickText: tickText, crossText: crossText, tickAction: tickAction, crossAction: crossAction });
        if (!_alertBoxInUse) buildAlertBoxFromArray();
    }

    function showScriptInfoAlert() {
        if (ALERT_UPDATE && SCRIPT_VERSION !== _settings.lastVersion) {
            let releaseNotes = '';
            releaseNotes += '<p>' + GM_info.script.name + '<br>v' + SCRIPT_VERSION + '<br><br>What\'s New:</p>';
            if (SCRIPT_VERSION_CHANGES.length > 0) {
                releaseNotes += '<ul>';
                for (let idx=0; idx < SCRIPT_VERSION_CHANGES.length; idx++) {
                    releaseNotes += '<li>' + SCRIPT_VERSION_CHANGES[idx];
                }
                releaseNotes += '</ul>';
            } else {
                releaseNotes += '<ul><li>Nothing major.</ul>';
            }
            showAlertBox('fa-info-circle', 'URC-E Release Notes', releaseNotes, false, "OK", "", null, null);
        }
    }

    function convertOldVarName(oldVarName) {
        let filterArr = _commentLists.filter(obj => obj.oldVarName === oldVarName);
        return filterArr.length > 0 ? filterArr[0].idx : 0;
    }

    function isChecked(checkboxId) {
        return $('#' + checkboxId).is(':checked');
    }

    function changeSetting(settingId, settingVal) {
        _settings[settingId] = settingVal;
        saveSettingsToStorage();
    }

    function changeCommentListStyle(settingVal) {
        if (_settings.commentListStyle !== settingVal) {
            if (settingVal === 'default') {
                $('fieldset[id^="urceComments-for-"], legend[id^="urceComments-for-"], div[id^="urceComments-for-"]').removeClass('urStyle');
                $('fieldset[id^="urce-prefs-fieldset"], legend[id^="urce-prefs-legend"]').removeClass('urStyle');
            } else if (settingVal == 'urStyle') {
                $('fieldset[id^="urceComments-for-"], legend[id^="urceComments-for-"], div[id^="urceComments-for-"]').addClass('urStyle');
                $('fieldset[id^="urce-prefs-fieldset"], legend[id^="urce-prefs-legend"]').addClass('urStyle');
            }
            _settings.commentListStyle = settingVal;
        }
        saveSettingsToStorage();
    }

    function getCollapsedGroups() {
        return new Promise((resolve) => {
            let $getDivs = $('div[id$="_body"]');
            let rObj = {};
            for (let idx = 0; idx < $getDivs.length; idx++) {
                let div = $getDivs[idx];
                if (div.id.indexOf('urceComments-for-') > -1) {
                    rObj[div.id] = $(div).hasClass('collapse');
                }
            }
            resolve(rObj);
        });
    }

    function getUrSessionsAsync(urIds, tries) {
        return new Promise((resolve,reject) => {
            (async function retry(urIds, tries) {
                tries = tries || 1;
                let urSessionsObj;
                try {
                    urSessionsObj = await W.model.updateRequestSessions.getAsync(urIds);
                } catch(error) {
                    let msg = 'Error retreiving urSessions async for urIds: ' + urIds.join(', ') + ' on try ' + tries + '.';
                    if (tries < 50) msg += ' Retrying.';
                    logDebug(msg);
                }
                if (tries > 49 && !urSessionsObj) {
                    reject('50 tries at getting urSessions async have elapsed. Stopping loop.');
                } else if (!urSessionsObj) {
                    setTimeout(retry, 100, urIds, ++tries);
                } else {
                    resolve(urSessionsObj);
                }
            })(urIds, null);
        });
    }

    function getMapUrsAsync(urIds, tries) {
        return new Promise((resolve,reject) => {
            (async function retry(urIds, tries) {
                tries = tries || 1;
                let mapUrsObj;
                try {
                    mapUrsObj = await W.model.mapUpdateRequests.getByIds(urIds);
                } catch(error) {
                    let msg = 'Error retrives mapUpdateRequests async for urIds: ' + urIds.join(', ') + ' on try ' + tries + '.';
                    if (tries < 50) msg += ' Retrying.';
                    logDebug(msg);
                }
                if (tries > 49 && !mapUrsObj) {
                    reject('50 tries at getting mapUpdateRequests async have elapsed. Stopping loop.');
                } else if (!mapUrsObj) {
                    setTimeout(retry, 100, urIds, ++tries);
                } else {
                    resolve(mapUrsObj);
                }
            })(urIds, null);
        });
    }

    function handleAfterCommentMutation(urId) {
        logDebug('Handling new comment mutation for urId: ' + urId);
        let newStatus = _selUr.newStatus;
        let doubleClick = _selUr.doubleClick;
        if (_settings.unfollowUrAfterSend) unfollowUrAfterSend(urId);
        if ((_settings.autoCloseCommentWindow && !newStatus) || doubleClick) {
            autoCloseUrPanel();
        } else {
            if (_settings.autoSaveAfterSolvedOrNiComment && (newStatus === 'solved' || newStatus === 'notidentified')) {
                clickSaveButton();
            } else {
                try {
                    handleUrLayer('sendComment')
                } catch(error) {
                    logWarning(error);
                }
            }
        }
    }

    function handleAfterCloseUpdateContainer(urId) {
        let newStatus = _selUr.newStatus;
        if (_settings.autoSaveAfterSolvedOrNiComment && (newStatus === 'solved' || newStatus === 'notidentified')) {
            clickSaveButton();
        } else {
            if (_settings.autoZoomOutAfterComemnt) autoZoomOut();
            if (_settings.autoSwitchToUrCommentsTab) autoSwitchToPrevTab();
            try {
                handleUrLayer('sendComment')
            } catch(error) {
                logWarning(error);
            }
        }
        _selUr = {};
    }

    function handleAfterSave() {
        if (_settings.autoZoomOutAfterComemnt) autoZoomOut();
        if (_settings.autoSwitchToUrCommentsTab) autoSwitchToPrevTab();
        try {
            handleUrLayer('save');
        } catch(error) {
            logWarning(error);
        }
    }

    async function handleUpdateRequestContainer(urId) {
        let commentNum, urSessionObj, mapUrsObj;
        _selUr.handling = true;
        logDebug('Handling update request container mutation. urId: ' + urId);
        try {
            urSessionObj = await getUrSessionsAsync([urId]);
        } catch(error) {
            logError(error);
            return;
        }
        try {
            mapUrsObj = await getMapUrsAsync([urId]);
        } catch(error) {
            logDebug(error);
            return;
        }
        let urData = urSessionObj[0];
        let mUrObj = mapUrsObj[0];
        _selUr.urOpen = mUrObj.attributes.open;
        if (_settings.autoSwitchToUrCommentsTab) autoSwitchToUrceTab();
        if ($('#panel-container .problem-edit .conversation').hasClass('collapsed')) {
            logDebug('Expanding conversation list.');
           $('#panel-container .problem-edit .conversation').removeClass('collapsed');
        }
        if (_settings.disableDoneNextButtons) {
            logDebug('Removing the done / next buttons.');
            $('#panel-container .content .navigation').css({'display':'none'});
        }
        logDebug('Setting event hook for center on UR crosshairs in UR panel title bar.');
        $('#panel-container > div > div > div.top-section > div.header > div.title > div > a.focus').off('click', handleUrPanelCrosshairsClick);
        $('#panel-container > div > div > div.top-section > div.header > div.title > div > a.focus').on('click', {mUrObj:mUrObj}, handleUrPanelCrosshairsClick);
        logDebug('Waiting 250ms before scrolling to bottom of the conversation list to give it time to load.');
        await setTimeout(scrollToBottom, 250);
        if (urData.comments.length === 0) {
            commentNum = Object.values(_defaultComments).find(defaultComment => { return defaultComment.urNum === mUrObj.attributes.type }).commentNum;
            if (_settings.autoSetNewUrComment) {
                if (_settings.autoZoomInOnNewUr) autoZoomIn(urId);
                if (_selUr.urOpen) {
                    if (_settings.autoClickOpenSolvedNi) autoClickOpenSolvedNi(commentNum);
                    try {
                        await postUrComment(_commentList[commentNum].comment);
                    } catch(error) {
                        logError(error);
                        showAlertBox('fa-exclamation-circle', I18n.t('urce.common.ErrorHeader'), I18n.t('urce.prompts.CommentInsertTimedOut'), false, 'OK', '', null, null);
                    }
                }
            }
        } else if (urData.comments.length === 1) {
            commentNum = _defaultComments.dr.commentNum;
            if (_settings.autoCenterOnUr) autoCenterOnUr(urId);
            let lastCommentBy = urData.comments[(urData.comments.length - 1)].userID;
            let commentDaysOld = urData.comments[(urData.comments.length - 1)].createdOn === null ? -1 : uroDateToDays(urData.comments[(urData.comments.length - 1)].createdOn);
            if (_settings.autoSetReminderUrComment && urData.comments.length > 0 && _settings.reminderDays !== 0 &&commentDaysOld > (_settings.reminderDays - 1) && lastCommentBy > 1) {
                if (_selUr.urOpen) {
                    if (_settings.autoZoomInOnNewUr) autoZoomIn(urId);
                    if (_settings.autoClickOpenSolvedNi) autoClickOpenSolvedNi(commentNum);
                    try {
                        await postUrComment(_commentList[commentNum].comment);
                    } catch(error) {
                        logError(error);
                        showAlertBox('fa-exclamation-circle', I18n.t('urce.common.ErrorHeader'), I18n.t('urce.prompts.CommentInsertTimedOut'), false, 'OK', '', null, null);
                    }
                }
            }
        } else {
            if (_settings.autoCenterOnUr) autoCenterOnUr(urId);
        }
    }

    async function handleClickedComment(commentNum, doubleClick) {
        let urId = _selUr.urId || getUrId('clickedComment');
        logDebug('Handling clicked comment. commentNum: ' + commentNum + ' | doubleClick: ' + doubleClick);
        _selUr.doubleClick = doubleClick;
        if (!$('.new-comment-text')[0]) {
            logWarning('No comment box found after clicking a comment from the list.');
            showAlertBanner(I18n.t('urce.prompts.NoCommentBox'), 5000);
            return;
        }
        if (!urId) {
            logError('No urId was found.');
            return;
        }
        if (doubleClick) {
            $('.new-comment-text').off('blur', autoClickSendButton);
            $('.new-comment-text').on('blur', autoClickSendButton);
        }
        if (_settings.autoClickOpenSolvedNi && _selUr.urOpen) autoClickOpenSolvedNi(commentNum);
        try {
            await postUrComment(_commentList[commentNum].comment);
        } catch(error) {
            logError(error);
            showAlertBox('fa-exclamation-circle', I18n.t('urce.common.ErrorHeader'), I18n.t('urce.prompts.CommentInsertTimedOut'), false, 'OK', '', null, null);
            return;
        }
    }

    function autoSwitchToUrceTab() {
        logDebug('Switching to URC-E > Comments tab.');
        _$restoreTab = $('#user-tabs > ul > li.active > a');
        _restoreTabPosition = $($('#user-info .tab-content')[0]).scrollTop();
        $('a[href="#sidepanel-urc-e"]').trigger('click');
        $('a[href="#panel-urce-comments"]').trigger('click');
        $($('#user-info .tab-content')[0]).scrollTop(0);
    }

    function autoSwitchToPrevTab() {
        if ($(_$restoreTab)) {
            logDebug('Switching to previous tab.');
            $(_$restoreTab).click();
            $($('#user-info .tab-content')[0]).scrollTop(_restoreTabPosition);
            _$restoreTab = null;
            _restoreTabPosition = null;
        }
    }

    function scrollToBottom() {
        logDebug('Scrolling to the bottom of comment list.');
        $('.top-section').scrollTop($('.top-section')[0].scrollHeight);
    }

    function handleUrPanelCrosshairsClick(event) {
        logDebug('Handling UR Panel crosshairs click event.');
        let mUrObj = event.data.mUrObj;
            let x = getXY(null, mUrObj).x;
            let y = getXY(null, mUrObj).y;
        W.map.setCenter([x,y], 5);
    }

    function unfollowUrAfterSend(urId) {
        logDebug('Unfollowing UR: ' + urId);
        W.model.updateRequestSessions.objects[urId].setFollowing('false');
    }

    function autoCloseUrPanel() {
        logDebug('Clicking close on UR panel.');
        $("#panel-container > div > div > div.top-section > a").trigger('click');
    }

    function clickSaveButton() {
        logDebug('Clicking the save button.');
        $('.toolbar-button.waze-icon-save').trigger('click');
    }

    function autoClickSendButton() {
        logDebug('doubleClick is true. Clicking send.');
        $('.new-comment-form .send-button').trigger('click');
        $('.new-comment-text').off('blur', autoClickSendButton);
    }

    function autoClickOpenSolvedNi(commentNum) {
        logDebug('Running auto click open, solved or not identified routine.');
        logDebug('Masking confirm function.');
        let confirmHold = window.confirm;
        window.confirm = function(msg) {
            // Dummy confirm to prevent WME from being able to send confirmations during auto clicking
            return true;
        }
        $('.problem-edit .body').scrollTop($('.problem-edit .body')[0].scrollHeight);
        if (_commentList[commentNum].urstatus === 'notidentified' && _selUr.newStatus !== 'notidentified') {
            logDebug('Clicking Not Identified');
            $('input[value="not-identified"]').trigger('click');
        } else if (_commentList[commentNum].urstatus === 'solved' && _selUr.newStatusn !== 'solved') {
            logDebug('Clicking Solved.');
            $('input[value="solved"]').trigger('click');
        } else if (_commentList[commentNum].urstatus === 'open' && (_selUr.newStatus === 'solved' || _selUr.newStatus === 'notidentified')) {
            logDebug('Clicking Open.');
            $('input[value="open"]').trigger('click');
        }
        logDebug('Unmasking confirm function.');
        window.confirm = confirmHold;
    }

    function autoZoomIn(urId) {
        logDebug('Checking zoom level and zooming in on UR if zoom level is less than 4.');
        let zoom = 4;
        _restoreZoom = getZoomLevel();
        if (_restoreZoom < zoom) {
            logDebug('Zooming to 4 from ' + _restoreZoom + '.');
            let x = getXY(urId, null).x;
            let y = getXY(urId, null).y;
            W.map.setCenter([x,y], 5);
        }
    }

    function autoCenterOnUr(urId) {
        logDebug('Checking zoom level and centering on UR if zoom level is less than 3.');
        let _restoreZoom = getZoomLevel();
        if (_restoreZoom < 3) {
            logDebug('Centering on UR because zoom level is ' + _restoreZoom + '.');
            let x = getXY(urId, null).x;
            let y = getXY(urId, null).y;
            W.map.setCenter([x,y], _restoreZoom);
        }
    }

    function autoZoomOut() {
        _restoreZoom = _restoreZoom || 4;
        logDebug('Zooming out to ' + _restoreZoom + '.');
        W.map.setCenter(W.map.getCenter(), _restoreZoom);
        _restoreZoom = null;
    }

    function getZoomLevel() {
        logDebug('Getting zoom level: ' + W.map.mapState.mapLocation.zoom);
        return W.map.mapState.mapLocation.zoom;
    }

    function formatText(text) {
        if (text.indexOf('$URD') > 0) {
            if ($('#update-request-panel .solution p').length > 0) {
                text = text.replace('$URD', $('#update-request.panel .solution p').text()).replace(/\n+/gmi, '');
            } else if ($('.description .content').length > 0) {
                text = text.replace('$URD', $('.description .content').text()).replace(/\n+/gmi, '').replace('$USERNAME', W.model.loginManager.user.userName);
            } else {
                text = text.replace(' "$URD"', '');
            }
        }
        if (text.indexOf('$SELSEGS') > 0) {
            let selFeatures = W.selectionManager.getSelectedFeatures();
            let streetName;
            if (selFeatures.length > 0 && selFeatures.length < 3) {
                for (let idx = 0; idx < selFeatures.length; idx++) {
                    if (selFeatures[idx].model.CLASS_NAME === 'W.Feature.Vector.Segment') {
                        if (selFeatures.length === 1) {
                            streetName = W.model.streets.objects[selFeatures[idx].model.attributes.primaryStreetID].name;
                        } else {
                            if (idx === 0) {
                                streetName = 'the intersection of ' + W.model.streets.objects[selFeatures[idx].model.attributes.primaryStreetID].name + ' and ';
                            } else {
                                streetName += W.model.streets.objects[selFeatures[idx].model.attributes.primaryStreetID].name;
                            }
                        }
                    }
                }
                if (streetName && streetName.length > 0) {
                    text = text.replace('$SELSEGS', streetName);
                } else {
                    text = text.replace('$SELSEGS', '');
                }
            }
        }
        return text.replace(/\\[r|n]+/gmi, '\n');
    }

    function postUrComment(comment) {
        return new Promise((resolve, reject) => {
            (function retry(comment, tries) {
                logDebug('Attemping to insert comment into comment box. Tries: ' + tries);
                if (tries > 100) {
                    reject('Timed out waiting for the comment text box to become available.');
                } else if (!$('.new-comment-text')[0]) {
                    setTimeout(retry, 100, comment, ++tries);
                } else {
                    $('.new-comment-text').val(formatText(comment)).change().keyup();
                    $('.new-comment-text').blur();
                    resolve();
                }
            })(comment, 1);
        });
    }

    function showAlertBanner(message, delay) {
        let dateNow = new Date().getTime();
        let width = message.length * 10;
        $('#map').append('<div id="urceMessage" style="width:100%; font-size:15px; font-weight:bold; margin-left:auto; position:absolute; top:0px; left:10px; z-index:1000;"></div>');
        $('#urceMessage').append('<div id="urceMapNote' + dateNow + '" style="width:' + width + 'px; font-size: 15px; font-weight:bold; margin-left:auto; margin-right:auto; background-color:orange;"><center><b>' + message + '</b></center></div>');
        $('#urceMapNote' + dateNow).show().delay(delay).queue(function() {
            $('#urceMessage').remove();
            $(this).remove();
        });
    }

    function getXY(urId, mUrObj) {
        let x,y;
        if (!urId && mUrObj) {
            if (mUrObj.attributes.geometry.urceRealX !== undefined) x = mUrObj.attributes.geometry.urceRealX;
            else if (mUrObj.attributes.geometry.realX !== undefined) x = mUrObj.attributes.geometry.realX;
            else x = mUrObj.attributes.geometry.x;
            if (mUrObj.attributes.geometry.urceRealY !== undefined) y = mUrObj.attributes.geometry.urceRealY;
            else if (mUrObj.attributes.geometry.realY !== undefined) y = mUrObj.attributes.geometry.realY;
            else y = mUrObj.attributes.geometry.y;
        } else {
            if (W.model.mapUpdateRequests.objects[urId].attributes.geometry.urceRealX !== undefined) x = W.model.mapUpdateRequests.objects[urId].attributes.geometry.urceRealX;
            else if (W.model.mapUpdateRequests.objects[urId].attributes.geometry.realX !== undefined) x = W.model.mapUpdateRequests.objects[urId].attributes.geometry.realX;
            else x = W.model.mapUpdateRequests.objects[urId].attributes.geometry.x;
            if (W.model.mapUpdateRequests.objects[urId].attributes.geometry.urceRealY !== undefined) y = W.model.mapUpdateRequests.objects[urId].attributes.geometry.urceRealY;
            else if (W.model.mapUpdateRequests.objects[urId].attributes.geometry.realY !== undefined) y = W.model.mapUpdateRequests.objects[urId].attributes.geometry.realY;
            else y = W.model.mapUpdateRequests.objects[urId].attributes.geometry.y;
        }
        return {x:x,y:y};
    }

    function parsePxString(pxString) {
        return parseInt(pxString.split('px')[0]);
    }

    function isIdAlreadyUnstacked(urId) {
        if (_markerStackArray.length === 0) return false;
        for (let idx = 0; idx < _markerStackArray.length; idx++) {
            if (_markerStackArray[idx].urId == urId) return true;
        }
        return false;
    }

    function stackListObj(urId, x, y) {
        this.urId = urId;
        this.x = parseInt(x);
        this.y = parseInt(y);
    }

    function restackMarkers() {
        if (_markerStackArray.length === 0) return;
        let markerCollection = W.map.updateRequestLayer.markers;
        if (markerCollection !== null) {
            logDebug('Restacking markers.');
            for (let marker in markerCollection) {
                if (markerCollection.hasOwnProperty(marker)) {
                    let testMarkerObj = markerCollection[marker];
                    if (testMarkerObj.model.attributes.geometry.urceRealX != null) {
                        testMarkerObj.model.attributes.geometry.x = testMarkerObj.model.attributes.geometry.urceRealX;
                        testMarkerObj.model.attributes.geometry.y = testMarkerObj.model.attributes.geometry.urceRealY;
                        delete(testMarkerObj.model.attributes.geometry.urceRealX);
                        delete(testMarkerObj.model.attributes.geometry.urceRealY);
                    }
                }
            }
            for (let idx=0; idx < _markerStackArray.length; idx++) {
                let origX = _markerStackArray[idx].x + 'px';
                let origY = _markerStackArray[idx].y + 'px';
                let urId = _markerStackArray[idx].urId
                if (markerCollection[urId] != null) {
                    markerCollection[urId].icon.imageDiv.style.left = origX;
                    markerCollection[urId].icon.imageDiv.style.top = origY;
                }
            }
            _markerStackArray = [];
            _unstackedMasterId = null;
        }
    }

    function checkMarkerStacking(urId, unstackedX, unstackedY) {
        if (isIdAlreadyUnstacked(urId) === true) return;
        logDebug('Checking for marker stack, urId: ' + urId);
        let stackList = [];
        let unstackZoomLevel = 3;
        let unstackSensitivity = 15;
        let threshSquared = unstackSensitivity * unstackSensitivity;
        let markerCollection = W.map.updateRequestLayer.markers;
        let marker;
        let offset = 1000000000;
        let inhibitUnstacking = (W.map.getZoom() < unstackZoomLevel);
        stackList.push(urId);
        if (markerCollection !== null) {
            for (marker in markerCollection) {
                if (markerCollection.hasOwnProperty(marker)) {
                    let testMarkerObj = markerCollection[marker];
                    let includeInStack = (testMarkerObj.icon.imageDiv.style.visibility != 'hidden');
                    let suppressClosed = (testMarkerObj.icon.imageDiv.classList.contains('recently-closed') && (W.map.updateRequestLayer.showHidden === false));
                    if (testMarkerObj.model.attributes.geometry.urceRealX === undefined) {
                        testMarkerObj.model.attributes.geometry.urceRealX = (testMarkerObj.model.attributes.geometry.realX) ? testMarkerObj.model.attributes.geometry.realX : testMarkerObj.model.attributes.geometry.x;
                        testMarkerObj.model.attributes.geometry.x = (testMarkerObj.model.attributes.geometry.realX) ? (testMarkerObj.model.attributes.geometry.realX + offset) : (testMarkerObj.model.attributes.geometry.x + offset);
                        testMarkerObj.model.attributes.geometry.urceRealY = (testMarkerObj.model.attributes.geometry.realY) ? testMarkerObj.model.attributes.geometry.realY : testMarkerObj.model.attributes.geometry.y;
                        testMarkerObj.model.attributes.geometry.y = (testMarkerObj.model.attributes.geometry.realY) ? (testMarkerObj.model.attributes.geometry.realY + offset) : (testMarkerObj.model.attributes.geometry.y + offset);
                        offset += 1000;
                    }
                    if (includeInStack && !suppressClosed) {
                        if (testMarkerObj.id != urId) {
                            let xDiff = unstackedX - parsePxString(markerCollection[testMarkerObj.id].icon.imageDiv.style.left);
                            let yDiff = unstackedY - parsePxString(markerCollection[testMarkerObj.id].icon.imageDiv.style.top);
                            let distSquared = ((xDiff * xDiff) + (yDiff * yDiff));
                            if (distSquared < threshSquared) stackList.push(testMarkerObj.id);
                        }
                    }
                }
            }
        }
        inhibitUnstacking = (inhibitUnstacking || (stackList.length == 1));
        if (stackList.length > 0) {
            if (inhibitUnstacking) logDebug('Single marker highlighted. Adjusting geometry properties to prevent recentering.');
            else logDebug('Markers are stacked!');
            if (_unstackedMasterId != urId) {
                logDebug('Unstacked ID mismatch, relocating markers.');
                restackMarkers();
                _unstackedMasterId = urId;
                _markerStackArray = [];

                _markerStackArray.push(new stackListObj(urId, unstackedX, unstackedY));

                for (let idx = 0; idx < stackList.length; idx++) {
                    let thisUrId = stackList[idx];
                    let x = parsePxString(markerCollection[thisUrId].icon.imageDiv.style.left);
                    let y = parsePxString(markerCollection[thisUrId].icon.imageDiv.style.top);
                    _markerStackArray.push(new stackListObj(thisUrId, x, y));
                    if (!inhibitUnstacking) {
                        markerCollection[thisUrId].icon.imageDiv.style.left = unstackedX + 'px';
                        markerCollection[thisUrId].icon.imageDiv.style.top = unstackedY + 'px';
                        unstackedX += 10;
                        unstackedY -= 30;
                    }
                }
            }
        } else {
            restackMarkers();
        }
    }

    function highlightedItemsCheck(event) {
        if (_settings.unstackMarkers) {
            if (_mousedOverMarkerType == 'ur') {
                let unstackedX = parsePxString(W.map.updateRequestLayer.markers[_mousedOverMarkerId].icon.imageDiv.style.left);
                let unstackedY = parsePxString(W.map.updateRequestLayer.markers[_mousedOverMarkerId].icon.imageDiv.style.top);
                checkMarkerStacking(_mousedOverMarkerId, unstackedX, unstackedY);
            } else {
                restackMarkers();
            }
        }
    }

    function markerMouseOver(event) {
        let markerType = getMarkerType(this);
        if (markerType !== null) {
            let markerId = this.attributes['data-id'].value;
            logDebug('Hover over ' + markerType + ' ID ' + markerId);
            _mousedOverMarkerId = markerId;
            _mousedOverMarkerType = markerType;
        }
    }

    function markerMouseOut(event) {
        let markerType = getMarkerType(this);
        if (markerType !== null) {
            let markerId = this.attributes['data-id'].value;
            logDebug('Hover off ' + markerType + ' ID ' + markerId);
            _mousedOverMarkerId = null;
            _mousedOverMarkerType = null;
        }
    }

    function getMarkerType(marker) {
        if (marker.className.indexOf('user-generated') > -1) return 'ur';
        else return null;
    }

    function addCustomMarker(urId, urOpen, customType, $node) {
        let useCustomMarker = false;
        if (customType === 0) useCustomMarker = _settings.customMarkersRoadworks;
        else if (customType === 1) useCustomMarker = _settings.customMarkersConstruction;
        else if (customType === 2) useCustomMarker = _settings.customMarkersClosures;
        else if (customType === 3) useCustomMarker = _settings.customMarkersEvents;
        else if (customType === 4) useCustomMarker = _settings.customMarkersNotes;
        else if (customType === 5) useCustomMarker = _settings.customMarkersWslm;
        else if (customType === 6) useCustomMarker = _settings.customMarkersBog;
        else if (customType === 7) useCustomMarker = _settings.customMarkersDifficult;
        else if (customType === 98) useCustomMarker = _settings.customMarkersNativeSl;
        if (useCustomMarker) renderCustomMarker(urId, urOpen, customType, $node);
        else removeCustomMarker(urId);
    }

    function removeCustomMarker(urId) {
        if ($(`#urceCustomMarker_${urId}`).length > 0) {
            logDebug('Removing custom marker for UR: ' + urId);
            $(`#urceCustomMarker_${urId}`).remove();
        }
    }

    function customMarkersEnabledCheck() {
        if (_settings.customMarkersRoadworks || _settings.customMarkersConstruction || _settings.customMarkersClosures || _settings.customMarkersEvents ||
            _settings.customMarkersNotes || _settings.customMarkersWslm || _settings.customMarkersBog || _settings.customMarkersDifficult || _settings.customMarkersNativeSl)
            return true;
        else return false;
    }

    function renderCustomMarker(urId, urOpen, customType, $node) {
        if ($(`#urceCustomMarker_${urId}`).length === 0) {
            logDebug('Adding custom marker for UR: ' + urId);
            $($node).append(
                $('<span>', {id:`urceCustomMarker_${urId}`, style:'position:absolute;pointer-events:none;top:-3px;left:-2px;'})
            );
        } else {
            logDebug('Updating custom marker for UR: ' + urId);
        }
        let customMarker = getCustomMarkerIdx(customType);
        let customVariant = (!urOpen) ? 2 : 0;
        $(`#urceCustomMarker_${urId}`).empty().append(
            $('<img>', {src:uroAltMarkers[customMarker][customVariant]})
        );
    }

    function getCustomMarkerIdx(customType) {
        if (customType === 0) return 1;      // ROADWORKS
        if (customType === 1) return 1;      // CONSTRUCTION
        if (customType === 2) return 0;      // CLOSURE
        if (customType === 3) return 4;      // EVENT
        if (customType === 4) return 3;      // NOTE
        if (customType === 5) return 5;      // WSLM
        if (customType === 6) return 11;     // BOG
        if (customType === 7) return 12;     // DIFFICULT

        if (customType === 98) return 5;     // Native speed limit URs
        if (customType === 99) return 2;     // custom text

        return -1;
    }

    function converTagToCustomType(tag) {
        switch(tag) {
            case 'ROADWORKS':
                return 0;
            case 'CONSTRUCTION':
                return 1;
            case 'CLOSURE':
                return 2;
            case 'EVENT':
                return 3;
            case 'NOTE':
                return 4;
            case 'WSLM':
                return 5;
            case 'BOG':
                return 6;
            case 'DIFFICULT':
                return 7;
            default:
                return -1;
        }
    }

    function updateUrMapMarkers(urIds, urSessionsObj, mapUrsObj) {
        let tagRegex = /.*\[(ROADWORKS|CONSTRUCTION|CLOSURE|EVENT|NOTE|WSLM|BOG|DIFFICULT)\].*/gi;
        const customMarkersEnabled = customMarkersEnabledCheck();
        for (let idx = 0; idx < urIds.length; idx++) {
            let urId = urIds[idx];
            let $node = $(`[data-id="${urId}"]`);
            if (_settings.enableUrPillCounts || customMarkersEnabled) {
                let urData = urSessionsObj[idx];
                let mUrObj = mapUrsObj[idx];
                let urDesc = mUrObj.attributes.description;
                let urOpen = mUrObj.attributes.open;
                let urCommentCount = urData.comments.length;
                let wmeUsername = W.model.loginManager.user.userName;
                let commentDaysOld, lastCommentBy, fullText, tagType, urCountBackground, tagContent, tagOffset, customType;
                let commentUserIds = [];
                if (urCommentCount > 0) {
                    commentDaysOld = uroDateToDays(urData.comments[(urCommentCount-1)].createdOn);
                    lastCommentBy = urData.comments[(urCommentCount-1)].userID;
                    fullText = urDesc ? urDesc + ' ' : '';
                    for (let idx = 0; idx < urCommentCount; idx++) {
                        fullText += urData.comments[idx].text + ' ';
                        commentUserIds.push(urData.comments[idx].userID);
                    }
                } else {
                    fullText = urDesc ? urDesc : '';
                    commentDaysOld = uroDateToDays(mUrObj.attributes.driveDate);
                }
                tagType = (fullText.search(tagRegex) > -1) ? fullText.replace(tagRegex, '$1') : null;
                if (tagType) {
                    customType = converTagToCustomType(tagType);
                } else {
                    if (mUrObj.attributes.type === 23) customType = 98;
                    else customType = -1;
                }
                if (!tagType || _settings.replaceTagNameWithEditorName) {
                    let regex = new RegExp(' ' + wmeUsername + ' ', 'gi');
                    tagType = fullText.search(regex) > -1 ? wmeUsername : tagType;
                }
                let curEditorHasCommented = commentUserIds.indexOf(_wmeUserId) > -1 ? true : false;
                if (_settings.enableUrPillCounts) {
                    urCountBackground = '#FFFF99';
                    if (_wmeUserId === lastCommentBy) urCountBackground = '#FFFFFF';
                    if (curEditorHasCommented && lastCommentBy < 1) urCountBackground = '#79B5C7';
                    if (_wmeUserId !== lastCommentBy && lastCommentBy < 1 && commentDaysOld < _settings.closeDays) urCountBackground = '#FFCC99';
                    if (_wmeUserId !== lastCommentBy && lastCommentBy > 1 && commentDaysOld > (_settings.closeDays - 1)) urCountBackground = '#FF8B8B';
                    if (tagType) urCountBackground = '#CCCCCC';
                    if (commentDaysOld === null || commentDaysOld === '' || commentDaysOld === undefined) commentDaysOld = (urCommentCount === '0') ? '0' : '?';
                    if (tagType && _settings.doNotShowTagNameOnPill) {
                        tagContent = urCommentCount + 'c ' + commentDaysOld + 'd';
                        tagOffset = Math.round(tagContent.length * 2.28);
                    } else if (tagType) {
                        tagContent = tagType + ' ' + urCommentCount + 'c';
                        tagOffset = (tagContent.length < 10) ? Math.round(tagContent.length * 2.86) : Math.round(tagContent.length * 3.33);
                    } else {
                        tagContent = urCommentCount + 'c ' + commentDaysOld + 'd';
                        tagOffset = Math.round(tagContent.length * 2.28);
                    }
                    tagOffset = '-' + tagOffset + 'px';
                    if (urCountBackground === '#CCCCCC') {
                        $($node).css({'z-index':'999'});
                    } else if (urCountBackground === '#FFFFFF'|| urCountBackground === '#79B5C7') {
                        $($node).css({'z-index':'998'});
                    } else if (urCountBackground === '#FF8B8B') {
                        $($node).css({'z-index':'997'});
                    }
                    if ($(`#urceCounters-${urId}`).length > 0) {
                        logDebug('Updating marker counters on UR marker for UR: ' + urId);
                        $(`#urceCounters-${urId}`).remove();
                    } else {
                        logDebug('Adding marker counters on UR marker for UR: ' + urId);
                    }
                    $($node).append(
                        $('<div>', {id:`urceCounters-${urId}`}).css('clear', 'both').css('margin-bottom', '10px').append(
                            $('<div>').html(tagContent).css({'color':'black', 'background-color':urCountBackground, 'position':'absolute', 'top':'30px', 'right':tagOffset, 'display':'block', 'width':'auto', 'white-space':'nowrap', 'padding-left':'5px', 'padding-right':'5px', 'border':'1px solid', 'border-radius':'25px'}).addClass('urceCounts')
                        )
                    );
                } else {
                    if ($(`#urceCounters-${urId}`).length > 0) {
                        logDebug('Removing marker counters on UR marker for UR: ' + urId);
                        $(`#urceCounters-${urId}`).remove();
                    }
                }
                if (customMarkersEnabled) {
                    if (customType > -1) addCustomMarker(urId, urOpen, customType, $node);
                    else removeCustomMarker(urId);
                } else {
                    removeCustomMarker(urId);
                }
            } else {
                if ($(`#urceCounters-${urId}`).length > 0) {
                    logDebug('Removing marker counters on UR marker for UR: ' + urId);
                    $(`#urceCounters-${urId}`).remove();
                }
                removeCustomMarker(urId);
            }
        }
    }

    function filterUrMapMarkers(urIds, urSessionsObj, mapUrsObj) {
        let tagRegex = /.*\[(ROADWORKS|CONSTRUCTION|CLOSURE|EVENT|NOTE|WSLM|BOG|DIFFICULT)\].*/gi;
        let doNotHideSelectedUr = _settings.doNotHideSelectedUr;
        for (let idx = 0; idx < urIds.length; idx++) {
            let urId = urIds[idx];
            let $node = $(`[data-id="${urId}"]`);
            if (_settings.enableUrceUrFiltering || _settings.autoSendReminders) {
                let preventHiding = ((urId === _selUr.urId) && doNotHideSelectedUr);
                let urData = urSessionsObj[idx];
                let mUrObj = mapUrsObj[idx];
                let urDesc = mUrObj.attributes.description;
                let urOpen = mUrObj.attributes.open;
                let urType = mUrObj.attributes.type;
                let urDriveDate = (mUrObj.attributes.driveDate) ? mUrObj.attributes.driveDate : -1;
                let urResolvedOn = (mUrObj.attributes.resolvedOn) ? mUrObj.attributes.resolvedOn : -1;
                let urReminderSent = mUrObj.attributes.reminderSent;
                let urCommentCount = urData.comments.length;
                let wmeUsername = W.model.loginManager.user.userName;
                let commentDaysOld, lastCommentBy, fullText, tagType, customType;
                let commentUserIds = [];
                let hideUr = false;
                let needsReminder = false;
                let needsClosed = false;
                let urWaiting = false;
                if (urCommentCount > 0) {
                    commentDaysOld = uroDateToDays(urData.comments[(urCommentCount-1)].createdOn);
                    lastCommentBy = urData.comments[(urCommentCount-1)].userID;
                    fullText = urDesc ? urDesc + ' ' : '';
                    for (let idx = 0; idx < urCommentCount; idx++) {
                        fullText += urData.comments[idx].text + ' ';
                        commentUserIds.push(urData.comments[idx].userID);
                    }
                    if (urCommentCount === 1) {
                        if (lastCommentBy > 1) {
                            if (_settings.reminderDays !== 0 && commentDaysOld > (_settings.reminderDays - 1)) {
                                if (_wmeUserId === lastCommentBy && !urReminderSent && _settings.autoSendReminders) {
                                    showAlertBanner(I18n.t('urce.prompts.ReminderMessageAuto') + ' ' + urId, 3000);
                                    W.model.updateRequestSessions.objects[urId].addComment(_defaultComments.dr.commentNum);
                                    W.model.mapUpdateRequests.objects[urId].attributes.reminderSent = 'true';
                                    if (_settings.unfollowUrAfterSend) unfollowUrAfterSend(urId);
                                    urWaiting = true;
                                } else {
                                    needsReminder = true;
                                }
                            } else {
                                urWaiting = true;
                            }
                        }
                    }
                    if (urCommentCount > 1) {
                        if (lastCommentBy > 1) {
                            if (commentDaysOld > (_settings.closeDays - 1)) {
                                if (_wmeUserId === lastCommentBy) {
                                    needsClosed = true;
                                } else {
                                    if (commentDaysOld < (_settings.reminderDays + _settings.closeDays)) urWaiting = true;
                                }
                            } else {
                                urWaiting = true;
                            }
                        }
                    }
                } else {
                    fullText = urDesc ? urDesc : '';
                    commentDaysOld = uroDateToDays(urDriveDate);
                }
                tagType = (fullText.search(tagRegex) > -1) ? fullText.replace(tagRegex, '$1') : null;
                if (tagType) {
                    customType = converTagToCustomType(tagType);
                } else {
                    if (urType === 23) customType = 98;
                    else customType = -1;
                }
                if (!tagType) {
                    let regex = new RegExp(' ' + wmeUsername + ' ', 'gi');
                    tagType = fullText.search(regex) > -1 ? wmeUsername : null;
                }
                if ((!preventHiding && _settings.enableUrceUrFiltering) &&
                    ((_settings.hideOutsideEditableArea && !mUrObj.canEdit()) ||
                     (_settings.hideWaiting && urWaiting) ||
                     (_settings.needsClosed && needsClosed) ||
                     (_settings.hideUrsReminderNeeded && needsReminder) ||
                     (_settings.hideByStatusOpen && urOpen) ||
                     (_settings.hideByStatusClosed && !urOpen) ||
                     (_settings.hideByStatusNotIdentified && mUrObj.attributes.resolution === 1) ||
                     (_settings.hideByStatusSolved && mUrObj.attributes.resolution === 0) ||
                     // Types
                     (_settings.hideByTypeBlockedRoad && urType === 19) ||
                     (_settings.hideByTypeGeneralError && urType === 10) ||
                     (_settings.hideByTypeIncorrectAddress && urType === 7) ||
                     (_settings.hideByTypeIncorrectJunction && urType === 12) ||
                     (_settings.hideByTypeIncorrectRoute && urType === 8) ||
                     (_settings.hideByTypeIncorrectTurn && urType === 6) ||
                     (_settings.hideByTypeMissingBridgeOverpass && urType === 13) ||
                     (_settings.hideByTypeMissingExit && urType === 15) ||
                     (_settings.hideByTypeMissingLandmark && urType === 18) ||
                     (_settings.hideByTypeMissingOrInvalidSpeedLimit && urType === 23) ||
                     (_settings.hideByTypeMissingRoad && urType === 16) ||
                     (_settings.hideByTypeMissingRoundabout && urType === 9) ||
                     (_settings.hideByTypeTurnNotAllowed && urType === 11) ||
                     (_settings.hideByTypeUndefined && (!urType || (urType > 19 && urType !== 23) || urType < 6 || urType === 17)) ||
                     (_settings.hideByTypeWazeAutomatic && urDesc.indexOf('Waze Automatic:') > -1) ||
                     (_settings.hideByTypeWrongDrivingDirection && urType === 14) ||
                     //Tags
                     (_settings.hideByTaggedBog && customType === 6) ||
                     (_settings.hideByTaggedClosure && customType === 2) ||
                     (_settings.hideByTaggedConstruction && customType === 1) ||
                     (_settings.hideByTaggedDifficult && customType === 7) ||
                     (_settings.hideByTaggedEvent && customType === 3) ||
                     (_settings.hideByTaggedNote && customType === 4) ||
                     (_settings.hideByTaggedRoadworks && customType === 0) ||
                     (_settings.hideByTaggedWslm && customType === 5) ||
                     // Age of submission
                     (_settings.hideByAgeOfSubmissionLessThan && uroDateToDays(urDriveDate) < _settings.hideByAgeOfSubmissionLessThanDaysOld) ||
                     (_settings.hideByAgeOfSubmissionMoreThan && uroDateToDays(urDriveDate) > _settings.hideByAgeOfSubmissionMoreThanDaysOld) ||
                     // Following, description, comments
                     (_settings.hideFollowing && urData.isFollowing) ||
                     (_settings.hideNotFollowing && !urData.isFollowing) ||
                     (_settings.hideDescription && urDesc && urDesc.length > 0 && urDesc !== '') ||
                     (_settings.hideWithoutDescription && (!urDesc || urDesc.length === 0 || urDesc === '')) ||
                     (_settings.hideWithCommentsFromMe && commentUserIds.indexOf(_wmeUserId) > -1) ||
                     (_settings.hideWithoutCommentsFromMe && commentUserIds.indexOf(_wmeUserId) === -1) ||
                     (_settings.hideLastCommentByMe && lastCommentBy === _wmeUserId) ||
                     (_settings.hideLastCommentNotByMe && lastCommentBy !== _wmeUserId) ||
                     (_settings.hideLastCommentByReporter && lastCommentBy === -1) ||
                     (_settings.hideLastCommentNotByReporter && lastCommentBy > 0) ||
                     (_settings.hideByCommentCountLessThan && urCommentCount < _settings.hideByCommentCountLessThanNumber) ||
                     (_settings.hideByCommentCountMoreThan && urCommentCount > _settings.hideByCommentCountMoreThanNumber) ||
                     (_settings.hideByAgeOfFirstCommentLessThan && urData.comments.length > 0 && uroDateToDays(urData.comments[0].createdOn) < _settings.hideByAgeOfFirstCommentLessThanDaysOld) ||
                     (_settings.hideByAgeOfFirstCommentMoreThan && urData.comments.length > 0 && uroDateToDays(urdata.comments[0].createdOn) > _settings.hideByAgeOfFirstCommentMoreThanDaysOld) ||
                     (_settings.hideByAgeOfLastCommentLessThan && commentDaysOld < _settings.hideByAgeOfLastCommentLessThanDaysOld) ||
                     (_settings.hideByAgeOfLastCommentMoreThan && commentDaysOld > _settings.hideByAgeOfLastCommentMoreThanDaysOld)
                    )
                   ) hideUr = true;
                if (hideUr && _settings.enableUrceUrFiltering) {
                    logDebug('Hiding UR marker for UR: ' + urId);
                    $($node).hide();
                } else {
                    logDebug('Unhiding UR marker for UR: ' + urId);
                    $($node).show();
                }
            } else {
                logDebug('Unhiding UR marker for UR: ' + urId);
                $($node).show();
            }
            logDebug('Setting event listeners on ' + urId);
            $($node).off('click', handleUrMarkerClick);
            $($node).on('click', handleUrMarkerClick);
            $($node).off('mouseover', markerMouseOver);
            $($node).on('mouseover', markerMouseOver);
            $($node).off('mouseout', markerMouseOut);
            $($node).on('mouseout', markerMouseOut);
        }
    }

    async function handleUrMapMarkers(urIds) {
        let urSessionsObj, mapUrsObj;
        urIds = urIds.sort();
        try {
            urSessionsObj = await getUrSessionsAsync(urIds);
            urSessionsObj.sort(function (a, b) {
                return a.id - b.id;
            });
        } catch(error) {
            logDebug(error);
            return;
        }
        try {
            mapUrsObj = await getMapUrsAsync(urIds);
        } catch(error) {
            logDebug(error);
            return;
        }
        filterUrMapMarkers(urIds, urSessionsObj, mapUrsObj);
        updateUrMapMarkers(urIds, urSessionsObj, mapUrsObj);
    }

    function handleUrLayer(phase) {
        return new Promise(async (resolve,reject) => {
            switch(phase) {
                case 'init':
                    logDebug('Checking for UR markers already present before URC-E completed initialization.');
                    break;
                case 'save':
                    logDebug('Updating UR markers after save.');
                    break;
                case 'close':
                    logDebug('Updating UR markers after closing UR panel.');
                    break;
                case 'settingsToggle':
                    logDebug('Updating UR markers after a setting toggle.');
                    break;
                case 'sendComment':
                    logDebug('Updating UR markers after sending a comment.');
                    break;
                default:
                    reject('No phase available in request.');
                    return;
            }
            let urList = [];
            let urMapMarkerIds = [];
            for (let urId in W.map.updateRequestLayer.markers) {
                if (urMapMarkerIds.indexOf(urId) === -1) urMapMarkerIds.push(urId);
            }
            if (urMapMarkerIds.length > 0) {
                try {
                    await handleUrMapMarkers(urMapMarkerIds);
                } catch(error) {
                    reject(error);
                    return;
                }
            }
            resolve();
        });
    }

    function handleUrMarkerClick() {
        if ($(this).hasClass('user-generated') || $(this).hasClass('has-comments')) {
            if (!(_selUr.urId > 0)) {
                _selUr.urId = $(this).attr('data-id');
                logDebug('Clicked UR: ' + _selUr.urId);
            }
        }
    }

    function getUrId(phase) {
        if (!(_selUr.urId > 0)) {
            if (phase === 'urPanelMutation') logWarning('Had to get the urId from the back yard for the UR panel mutation. Please let dBsooner know.');
            else if (phase === 'clickedComment') logWarning('Had to get the urId from the back yard for the clicked comment phase. Please let dBsooner know.');
            else logWarning('Had to get the urId from the back yard because it was wandering around. Please let dBsooner know.');
            _selUr.urId = $(".update-requests .selected").data("id")
        }
    }

    async function changeCommentList(commentListIdx) {
        commentListIdx = parseInt(commentListIdx || 0);
        if (commentListIdx != _settings.commentList) {
            logDebug('Switching comment list from ' + _commentLists[_settings.commentList].name + ' to ' + _commentLists[commentListIdx].name + '.');
            _settings.commentList = parseInt(commentListIdx);
            let buildCommentListResult = await buildCommentList(commentListIdx);
            if (buildCommentListResult.error) {
                handleBuildCommentListError(buildCommentListResult.error);
            }
            saveSettingsToStorage();
        }
    }

    function getCommentListInfo(commentListIdx) {
        commentListIdx = parseInt(commentListIdx || _settings.commentList);
        return _commentLists.find(cList => { return cList.idx === commentListIdx });
    }

    function checkForStaticListArray(oldVarName) {
        return new Promise((resolve, reject) => {
            (function checking(oldVarName, tries) {
                tries = tries || 1;
                if (tries > 100) {
                    reject('Timed out waiting for static list variable to be set.');
                } else if (!window['Urcomments' + oldVarName + 'Array2']) {
                    setTimeout(checking, 100, oldVarName, ++tries);
                } else {
                    resolve();
                }
            })(oldVarName, null);
        });
    }

    function convertCommentListStatic(commentListIdx) {
        return new Promise(async (resolve, reject) => {
            commentListIdx = parseInt(commentListIdx || _settings.commentList);
            let oldVarName = getCommentListInfo(commentListIdx).oldVarName;
            try {
                await checkForStaticListArray(oldVarName);
            } catch (error) {
                reject(error);
                return;
            }
            let oldUrcArr = window['Urcomments' + oldVarName + 'Array2'];
            let defaultReminderIdx = window['Urcomments' + oldVarName + 'ReminderPosistion'];
            let closedNiIdx = window['Urcomments' + oldVarName + 'CloseNotIdentifiedPosistion'];
            let data = [];
            let entryIdx;
            logDebug('Converting static comment list to URC-E format for comment list: ' + oldVarName);
            data[0] = [ 'URCE' ];
            data[1] = [ '2018.11.28.01' ];
            data[2] = [ 'TITLE|COMMENT|URSTATUS|DR|DC|IT|IA|IR|MRA|GE|TNA|IJ|MBO|WDD|ME|MR|ML|BR|MSN|ISPS|SL' ];
            if (oldUrcArr[0].search(/<br>/gi) === -1) {
                data[3] = [ '||GROUP TITLE||||||||||||||||||' ];
                entryIdx = 4;
            } else {
                entryIdx = 3;
            }
            for (let oldUrcArrIdx = 0; oldUrcArrIdx < oldUrcArr.length; oldUrcArrIdx = oldUrcArrIdx + 3) {
                let temp;
                let title = oldUrcArr[oldUrcArrIdx];
                let comment = oldUrcArr[oldUrcArrIdx+1];
                let urstatus = oldUrcArr[oldUrcArrIdx+2] != '' ? oldUrcArr[oldUrcArrIdx+2].toLowerCase() : '';
                if (title.search(/<br>/gi) > -1) {
                    urstatus = 'GROUP TITLE';
                    title = $("<div>").html(title).text();
                }
                temp = title+'|'+comment+'|'+urstatus;
                temp += (oldUrcArrIdx == defaultReminderIdx) ? '|default_is_true' : '|';
                temp += (oldUrcArrIdx == closedNiIdx) ? '|default_is_true' : '|';
                for (let i=6; i<24; i++) {
                    if (i === 17 || i === 20) continue;
                    temp += (window['Urcomments' + oldVarName + 'def_names'][i] == title) ? '|default_is_true' : '|';
                }
                data[entryIdx] = [ temp ];
                entryIdx++;
            }
            resolve(data);
        });
    }

    function processCommentList(data) {
        return new Promise((resolve,reject) => {
            logDebug('Procesing comment list data.');
            if (data) {
                const EXPECTED_FIELD_NAMES = ['TITLE','COMMENT','URSTATUS','DR','DC','IT','IA','IR','MRA','GE','TNA','IJ','MBO','WDD','ME','MR','ML','BR','MSN','ISPS','SL'];
                let ssFieldNames, groupDivId;
                let checkFieldNames = fldName => ssFieldNames.indexOf(fldName) > -1;
                let commentId = 0
                let blankGroup = 0;
                let doubleClickLinkNiComments = _settings.doubleClickLinkNiComments;
                let doubleClickLinkOpenComments = _settings.doubleClickLinkOpenComments;
                let doubleClickLinkSolvedComments = _settings.doubleClickLinkSolvedComments;
                for (let entryIdx = 0; entryIdx < data.length; entryIdx++) {
                    let cellValue = data[entryIdx][0];
                    if (entryIdx === 0) {
                        if (cellValue !== 'URCE') {
                            reject('Incorrect format in spreadsheet data received.');
                            return;
                        }
                    } else if (entryIdx === 1) {
                        if (SCRIPT_VERSION < cellValue) {
                            reject('Script must be updated to at least version ' + cellValue + ' before comment definitions can be loaded.');
                            return;
                        }
                    } else if (entryIdx === 2) {
                        ssFieldNames = cellValue.split('|').map(fldName => fldName.trim());
                        if (ssFieldNames.length !== EXPECTED_FIELD_NAMES.length) {
                            reject('Expected ' + EXPECTED_FIELD_NAMES.length + ' columns in comment definition data. Spreadsheet returned ' + ssFieldNames.length + '.');
                            return;
                        } else if (!EXPECTED_FIELD_NAMES.every(fldName => checkFieldNames(fldName))) {
                            reject('Script expected to see the following column names in the comment definition spreadsheet:\n' + EXPECTED_FIELD_NAMES.join(', ') + '\nHowever, the spreadsheet returned these:\n' + ssFieldNames.join(', '));
                            return;
                        }
                    } else {
                        let splitRow = cellValue.split('|');
                        let rObj = {};
                        for (let i=0; i<splitRow.length; i++) {
                            let rObjKey = ssFieldNames[i].trim().toLowerCase();
                            rObj[rObjKey] = rObjKey === 'comment' ? splitRow[i] : rObjKey === 'title' ? splitRow[i].trim() : splitRow[i].trim().toLowerCase();
                        }
                        splitRow = rObj;
                        if (splitRow.title === 'URCE_REMOVED_SO_SKIP') {
                            // Nothing to do here. Move along. This is a comment that has been set to 'REMOVED' in the spreadsheet.
                            logDebug('SKIPPING a removed comment.');
                        } else if (splitRow.title === 'URCE_ERROR') {
                            // UH OH . This is bad. Something broke in the arrayformula on the spradsheet.
                            reject('There is an unknown error in the spreadsheet output. Please contact the list owner.');
                            return;
                        } else if (splitRow.urstatus === 'group title') {
                            // Group title row. Nothing to set in the arrays, but build html.
                            groupDivId = 'urceComments-for-';
                            if (splitRow.title != '') {
                                groupDivId += splitRow.title.replace(/[^\w]+/gi, '').toLowerCase();
                                if (splitRow.title === splitRow.title.toUpperCase()) {
                                    if (splitRow.title.length > 25) {
                                        splitRow.titleMouseOver = splitRow.title;
                                        splitRow.title = splitRow.title.substring(0, 25) + '...';
                                    }
                                } else if (splitRow.title.length > 30) {
                                    splitRow.titleMouseOver = splitRow.title;
                                    splitRow.title = splitRow.title.substring(0, 30) + '...';
                                }
                            } else {
                                groupDivId += 'blankGroup' + (++blankGroup);
                            }
                            let collapsed;
                            if (_settings.commentListCollapses.hasOwnProperty(_settings.commentList)) {
                                let cListCollapses = _settings.commentListCollapses[_settings.commentList];
                                collapsed = (cListCollapses.hasOwnProperty(groupDivId+'_body')) ? (cListCollapses[groupDivId+'_body'] === true) ? 'collapse' : '' : '';
                            } else {
                                collapsed = '';
                            }
                            let chevron = (collapsed === 'collapse') ? 'fa-chevron-right' : 'fa-chevron-down';
                            let urStyle = (_settings.commentListStyle === 'urStyle') ? ' urStyle' : '';
                            $('#_commentList').append(
                                $('<fieldset>', {id:groupDivId, class:`URCE-field ${urStyle}`}).append(
                                    $('<legend>', {id:groupDivId+'_legend', class:`URCE-legend ${urStyle}`}).append(
                                        $('<i>', {class:`fa fa-fw ${chevron} URCE-chevron`}),
                                        $('<span>', {class:'URCE-span', title:splitRow.titleMouseOver}).text(splitRow.title)
                                    ).click(function() {
                                        $($(this).children()[0]).toggleClass('fa fa-fw fa-chevron-down');
                                        $($(this).children()[0]).toggleClass('fa fa-fw fa-chevron-right');
                                        $($(this).siblings()[0]).toggleClass('collapse');
                                        saveSettingsToStorage();
                                    })
                                ).append(
                                    $('<div>', {id:groupDivId+'_body', class:`${collapsed} URCE-group_body ${urStyle}`})
                                )
                            )
                        } else {
                            // SHOULD be a normal comments row, push values to arrays and build html.
                            if (splitRow.urstatus !== 'solved' && splitRow.urstatus !== 'notidentified' && splitRow.urstatus !== 'open' && splitRow.urstatus !== 'blank line') {
                                return reject('Your current selected list does not have a status set for ' + splitRow.title + '. Please contact list owner.');
                            } else {
                                _commentList[commentId] = { 'title':splitRow.title, 'comment':splitRow.comment, 'urstatus':splitRow.urstatus };
                                if (Object.values(splitRow).indexOf('default_is_true') > -1) {
                                    let drIdx = ssFieldNames.indexOf('DR');
                                    let splitRowDefaultCommentsBoolean = Object.values(splitRow).slice(drIdx);
                                    for (let boolIdx = 0; boolIdx < splitRowDefaultCommentsBoolean.length; boolIdx++) {
                                        if (splitRowDefaultCommentsBoolean[boolIdx].toLowerCase() === 'default_is_true') {
                                            _defaultComments[ssFieldNames[(boolIdx+drIdx)].toLowerCase()].commentNum = commentId;
                                        }
                                    }
                                }
                                let linkClass;
                                let divDoubleClickId;
                                let divDoubleClickClass;
                                let divDoubleClickStyle = 'display:initial;';
                                if (splitRow.urstatus === 'solved') {
                                    linkClass = 'URCE-solvedLink';
                                    divDoubleClickId = 'URCE-divDoubleClickSolved';
                                    if (!doubleClickLinkSolvedComments) {
                                        divDoubleClickStyle = 'display:none;';
                                    }
                                } else if (splitRow.urstatus === 'notidentified') {
                                    linkClass = 'URCE-niLink';
                                    divDoubleClickId = 'URCE-divDoubleClickNi';
                                    if (!doubleClickLinkNiComments) {
                                        divDoubleClickStyle = 'display:none;';
                                    }
                                } else {
                                    linkClass = 'URCE-openLink';
                                    divDoubleClickId = splitRow.title != '' ? 'URCE-divDoubleClickOpen' : 'URCE-divDoubleClickOpen-Hidden';
                                    divDoubleClickClass = splitRow.title != '' ? 'URCE-divDoubleClick' : 'URCE-divDoubleClick-Hidden';
                                    if (!doubleClickLinkOpenComments || splitRow.urstatus === 'blank line') {
                                        divDoubleClickStyle = 'display:none;';
                                    }
                                }
                                $(`#${groupDivId}_body`).append(
                                    $('<div>').append(
                                        $('<a>', {class:'URCE-Comments', id:'urce-cid-'+commentId, title:splitRow.comment, class:linkClass + ' URCE-Comments'}).text(splitRow.title).click(function() {
                                            handleClickedComment(parseInt(this.id.replace(/urce-cid-/, '')), false);
                                        })
                                    ).append(
                                        $('<div>', {class:'URCE-divDoubleClick', id:divDoubleClickId, style:divDoubleClickStyle, title:I18n.t('urce.common.DoubleClickTitle') + '\n' + splitRow.comment}).append(
                                            $('<img>', {src:DOUBLE_CLICK_ICON, class:'URCE-doubleClickIcon', id:'urce-img-cid-'+commentId}).dblclick(function() {
                                                handleClickedComment(parseInt(this.id.replace(/urce-img-cid-/, '')), true);
                                            })
                                        )
                                    ).append(
                                        $('<br>')
                                    ),
                                )
                                commentId++;
                            }
                        }
                    }
                }
            } else {
                reject('No data passed to the JSON processing function.');
                return;
            }
            resolve();
        });
    }

    function commentListAsync(commentListIdx) {
        commentListIdx = parseInt(commentListIdx || _settings.commentList);
        logDebug('Beginning comment list async for comment list: ' + getCommentListInfo(commentListIdx).name);
        return new Promise((resolve, reject) => {
            gapi.load('client:auth2', () => {
                gapi.client.init({
                    apiKey: URCE_API_KEY,
                    clientId: URCE_CLIENT_ID,
                    discoveryDocs: URCE_DISCOVERY_DOCS,
                    scopes: URCE_SCOPES
                }).then(() => {
                    gapi.client.sheets.spreadsheets.values.get({
                        spreadsheetId: URCE_SPREADSHEET_ID,
                        range: getCommentListInfo(commentListIdx).gSheetRange
                    }).then((response) => {
                        let range = response.result;
                        if (range.values.length > 0) {
                            resolve(range.values);
                        } else {
                            reject('No comments found in spreadsheet sheet.');
                        }
                    }, (response) => {
                        reject(response.result.error.message);
                    });
                }, (error) => {
                    reject(JSON.stringify(error, null, 2));
                });
            });
        });
    }

    async function buildCommentList(commentListIdx) {
        commentListIdx = parseInt(commentListIdx || _settings.commentList);
        logDebug('Building comment list for: ' + getCommentListInfo(commentListIdx).name);
        _commentListLoaded = false;
        let data, result;
        $('#_commentList').empty();
        $('#_commentList').append(
            $('<div>', {class:'URCE-commentListName'}).text(I18n.t('urce.prefs.CommentList') + ': ' + getCommentListInfo(_settings.commentList).name)
        );
        _commentList = [];
        if (getCommentListInfo(commentListIdx).type === 'static') {
            try {
                data = await convertCommentListStatic(commentListIdx);
            } catch (error) {
                return {error:error};
            }
            try {
                await processCommentList(data);
            } catch (error) {
                return {error:error};
            }
        } else {
            try {
                data = await commentListAsync(commentListIdx);
            } catch (error) {
                return {error:error};
            }
            try {
                await processCommentList(data);
            } catch (error) {
                return {error:error};
            }
        }
        $('#_selCommentList').val(_settings.commentList);
        _commentListLoaded = true;
        return {error:false};
    }

    function handleBuildCommentListError(error) {
        logError(error);
        _commentListLoaded = false;
        $('#_commentList').empty();
        $('#_commentList').append(
            $('<div>', {class:'URCE-divLoading'}).text(I18n.t('urce.common.ErrorGeneric'))
        );
    }

    function initMutationObservers(status) {
        let saveButtonObserver = new MutationObserver(function(mutations) {
            mutations.forEach(function(mutation) {
                if ($(mutation.target).hasClass('waze-icon-save') && mutation.type === 'attributes' && mutation.attributeName === 'class' && mutation.target.classList.contains('ItemDisabled')) {
                    if (mutation.oldValue.toString().indexOf('ItemDisabled') === -1) {
                        handleAfterSave();
                    }
                }
            });
        });
        let urPanelContainerObserver = new MutationObserver(function(mutations) {
            let urId = _selUr.urId || getUrId('urPanelMutation');
            mutations.forEach(function(mutation) {
                if ($(mutation.target).is('#panel-container') && mutation.type === 'childList' && mutation.addedNodes.length > 0 && urId > 0) {
                    handleUpdateRequestContainer(urId);
                } else if ($(mutation.target).is('#panel-container') && mutation.type === 'childList' && mutation.removedNodes.length > 0 && urId > 0) {
                    handleAfterCloseUpdateContainer(urId);
                } else if ($(mutation.target).hasClass('comment-list') && mutation.type === 'childList' && mutation.addedNodes.length > 0 && urId > 0) {
                    handleAfterCommentMutation(urId);
                } else if (mutation.type === 'attributes' && mutation.attributeName === 'data-state') {
                    logDebug('Handling UR status change mutation.');
                    if (mutation.target.attributes['data-state'].nodeValue === 'open') {
                        _selUr.newStatus = 'open';
                    } else if (mutation.target.attributes['data-state'].nodeValue === 'solved') {
                        _selUr.newStatus = 'solved';
                    } else if (mutation.target.attributes['data-state'].nodeValue === 'not-identified') {
                        _selUr.newStatus = 'notidentified';
                    } else {
                        logWarning(mutation.target.attributes['data-state'].nodeValue);
                    }
                }
            });
        });
        let urMarkerObserver = new MutationObserver(function(mutations) {
            let urMapMarkerIds = [];
            let i = 0;
            mutations.forEach((mutation) => {
                if (mutation.type === 'childList') {
                    for (let idx = 0; idx < mutation.addedNodes.length; idx++) {
                        let addedNode = mutation.addedNodes[idx];
                        if (addedNode.classList && addedNode.classList.contains('map-marker') && (addedNode.classList.contains('user-generated') || addedNode.classList.contains('map-marker'))) {
                            let urId = addedNode.getAttribute('data-id');
                            if (urId > 0 && urMapMarkerIds.indexOf(urId) === -1) {
                                logDebug('Setting event listeners on ' + urId);
                                $(addedNode).off('click', handleUrMarkerClick);
                                $(addedNode).on('click', handleUrMarkerClick);
                                $(addedNode).off('mouseover', markerMouseOver);
                                $(addedNode).on('mouseover', markerMouseOver);
                                $(addedNode).off('mouseout', markerMouseOut);
                                $(addedNode).on('mouseout', markerMouseOut);
                                urMapMarkerIds.push(urId);
                            }
                        }
                    }
                } else if (mutation.type === 'attributes' && mutation.target.classList && (mutation.target.classList.contains('user-generated') || mutation.target.classList.contains('has-comments'))) {
                    if ((!mutation.oldValue || !mutation.oldValue.match(/\bselected\b/)) && mutation.target.classList.contains('selected')) {
                        if (mutation.target.attributes['data-id'].nodeValue > 0) {
                            if (!_selUr.handling) {
                                _selUr.urId = mutation.target.attributes['data-id'].nodeValue;
                                logDebug('Caught selected UR by backdoor. Firing the minions. urId: ' + _selUr.urId);
                                handleUpdateRequestContainer(_selUr.urId);
                            }
                        }
                    }
                }
            });
            if (urMapMarkerIds.length > 0) {
                handleUrMapMarkers(urMapMarkerIds);
            }
        });
        if (status === 'enable' && (!saveButtonObserver.isObserving || !urPanelContainerObserver.isObserving || !urMarkerObserver.isObserving)) {
            logDebug('Enabling MOs.');
            if (!saveButtonObserver.isObserving) {
                saveButtonObserver.observe(document.getElementById('toolbar'), { childList: true, attributes: true, attributeOldValue: true, characterData: true, characterDataOldValue: true, subtree: true });
                saveButtonObserver.isObserving = true;
            }
            if (!urPanelContainerObserver.isObserving) {
                urPanelContainerObserver.observe(document.getElementById('panel-container'), { childList: true, attributes: true, attributeOldValue: true, characterData: true, characterDataOldValue: true, subtree: true });
                urPanelContainerObserver.isObserving = true;
            }
            if (!urMarkerObserver.isObserving) {
                urMarkerObserver.observe(W.map.updateRequestLayer.div, { childList: true, attributes: true, attributeOldValue: true, characterData: true, characterDataOldValue: true, subtree: true });
                urMarkerObserver.isObserving = true;
            }
            logDebug('Registering map.events event hook.');
            W.map.events.register('mousemove', null, highlightedItemsCheck);
        } else if (status === 'disable' && (saveButtonObserver.isObserving || urPanelContainerObserver.isObserving || urMarkerObserver.isObserving)) {
            logDebug('Disabling MOs.');
            if (saveButtonObserver.isObserving) {
                saveButtonObserver.disconnect();
                saveButtonObserver.isObserving = false;
            }
            if (urPanelContainerObserver.isObserving) {
                urPanelContainerObserver.disconnect();
                urPanelContainerObserver.isObserving = false;
            }
            if (urMarkerObserver.isObserving) {
                urMarkerObserver.disconnect();
                urMarkerObserver.isObserving = false;
            }
            logDebug('Unregistering map.events event hook.');
            W.map.events.unregister('mousemove', null, highlightedItemsCheck);
        }
    }

    async function initBackgroundTasks() {
        logDebug('Initializing background tasks.');
        let parentLayerEnabled = $('#layer-switcher-group_issues').is(':checked');
        let mapIssuesEnabled = $('#layer-switcher-group_map_issues').is(':checked');
        let openUrsEnabled = $('#layer-switcher-item_update_requests').is(':checked');
        let closedUrsEnabled = $('#layer-switcher-item_closed_update_requests').is(':checked');
        logDebug('Setting event hooks on layer toggles.');
        $('#layer-switcher-group_issues').change(function() {
            if (!$(this).is(':checked')) {
                initMutationObservers('disable');
            } else {
                mapIssuesEnabled = $('#layer-switcher-group_map_issues').is(':checked');
                openUrsEnabled = $('#layer-switcher-item_update_requests').is(':checked');
                closedUrsEnabled = $('#layer-switcher-item_closed_update_requests').is(':checked');
                if ((mapIssuesEnabled) && (openUrsEnabled || closedUrsEnabled)) {
                    initMutationObservers('enable');
                } else {
                    initMutationObservers('disable');
                }
            }
        });
        $('#layer-switcher-group_map_issues').change(function() {
            if (!$(this).is(':checked')) {
                initMutationObservers('disable');
            } else {
                openUrsEnabled = $('#layer-switcher-item_update_requests').is(':checked');
                closedUrsEnabled = $('#layer-switcher-item_closed_update_requests').is(':checked');
                if (openUrsEnabled || closedUrsEnabled) {
                    initMutationObservers('enable');
                } else {
                    initMutationObservers('disable');
                }
            }
        });
        $('#layer-switcher-item_update_requests').change(function() {
            closedUrsEnabled = $('#layer-switcher-item_closed_update_requests').is(':checked');
            if (!$(this).is(':checked') && !closedUrsEnabled) {
                initMutationObservers('disable');
            } else {
                initMutationObservers('enable');
            }
        });
        $('#layer-switcher-item_closed_update_requests').change(function() {
            openUrsEnabled = $('#layer-switcher-item_update_requests').is(':checked');
            if (!$(this).is(':checked') && !openUrsEnabled) {
                initMutationObservers('disable');
            } else {
                initMutationObservers('enable');
            }
        });
        if (parentLayerEnabled && mapIssuesEnabled && (openUrsEnabled || closedUrsEnabled)) {
            try {
                await handleUrLayer('init');
            } catch(error) {
                logWarning(error);
                // Don't need to return here, go ahead and setup the MOs.
            }
            initMutationObservers('enable');
        }
    }

    function injectCss() {
        logDebug('Injecting CSS.');
        let css = [
            // Comments tab
            '#sidepanel-urc-e #panel-urce-comments .URCE-Comments { text-decoration:none; cursor:pointer; color: #000000; font-size:12px; }',
            '#sidepanel-urc-e #panel-urce-comments .URCE-commentListName { padding-left:12px; font-size:12px; }',
            '#sidepanel-urc-e #panel-urce-comments .URCE-divLoading { text-align:left; color:red; font-size:11px; }',
            '#sidepanel-urc-e #panel-urce-comments .URCE-divCCLinks { text-align:center; }',
            '#sidepanel-urc-e #panel-urce-comments .URCE-divIcon { height:0px; position:relative; top:-3px; left:-100px; }',
            '#sidepanel-urc-e #panel-urce-comments .URCE-icon { cursor:default; }',
            '#sidepanel-urc-e #panel-urce-comments .URCE-solvedLink { color:#008F00; }',
            '#sidepanel-urc-e #panel-urce-comments .URCE-niLink { color:#E68A00; }',
            '#sidepanel-urc-e #panel-urce-comments .URCE-openLink { color:#000000; }',
            '#sidepanel-urc-e #panel-urce-comments .URCE-doubleClickIcon { padding-bottom:4px; height:16px; float:right; }',
            '#sidepanel-urc-e #panel-urce-comments .URCE-divDoubleClick { display:inline; }',
            '#sidepanel-urc-e #panel-urce-comments .URCE-span { cursor:pointer; }',
            '#sidepanel-urc-e #panel-urce-comments .URCE-group_body.urStyle { padding-left:23px !important; }',
            // Settings tab
            '#sidepanel-urc-e #panel-urce-settings .URCE-divWarningPre { margin-left:3px; }',
            '#sidepanel-urc-e #panel-urce-settings .URCE-divWarning { display:inline; }',
            '#sidepanel-urc-e #panel-urce-settings .URCE-divWarningTitle { color:red; text-decoration:underline; }',
            '#sidepanel-urc-e #panel-urce-settings .URCE-daysInput { width:38px; height:20px; }',
            '#sidepanel-urc-e #panel-urce-settings .URCE-span { text-transform:uppercase; cursor:pointer; }',
            '#sidepanel-urc-e #panel-urce-settings .URCE-controls { padding:5px 0 5px 0; font-size:12px;}',
            '#sidepanel-urc-e #panel-urce-settings .URCE-controls .URCE-subHeading { font-weight:600; }',
            '#sidepanel-urc-e #panel-urce-settings .URCE-controls .URCE-textFirst, .URCE-controls.URCE-textFirst { padding:0 0 0 10px !important; line-height:12px }',
            '#sidepanel-urc-e #panel-urce-settings .URCE-controls .URCE-divDaysInline { display:inline; padding-left:4px; }',
            '#sidepanel-urc-e #panel-urce-settings .URCE-controls input[type="checkbox"] { margin:0 5px 0 10px; vertical-align:middle; cursor:pointer; }',
            '#sidepanel-urc-e #panel-urce-settings .URCE-controls select { height:22px; vertical-align:middle; }',
            '#sidepanel-urc-e #panel-urce-settings .URCE-controls label { font-weight:normal; cursor:pointer; }',
            // Common
            '#sidepanel-urc-e .URCE-chevron { cursor:pointer; font-size:12px; margin-right: 4px; }',
            '#sidepanel-urc-e .URCE-field { border:1px solid silver; padding:5px; border-radius:4px; -webkit-padding-before:0; }',
            '#sidepanel-urc-e .URCE-field.urStyle { border:unset !important; padding:unset !important; border-radius:unset !important; }',
            '#sidepanel-urc-e .URCE-legend { margin-bottom:0px; border-bottom-style:none; width:auto; }',
            '#sidepanel-urc-e .URCE-legend.urStyle { border-bottom-style:unset !important; margin-bottom:2px !important; width:100% !important; background-color:#F6F7F7 !important; line-height:20px !important; padding:0 2px 0 2px !important; border-top:1px solid #C0C0C0 !important; border-bottom:1px solid #C0C0C0 !important; }',
            '#sidepanel-urc-e .URCE-divCC { /* padding-top:2px !important; */ }',
            '#sidepanel-urc-e .URCE-label { white-space:pre-line; margin:0 0 0 0; }',
            '#sidepanel-urc-e .URCE-span { font-size:14px; font-weight:600; }',
            '#sidepanel-urc-e .URCE-spanTitle { font-size:14px; font-weight:600; }',
            '#sidepanel-urc-e .URCE-spanVersion { font-size:11px; margin-left:10px; color:#aaa; }',
            '#sidepanel-urc-e .URCE-divTabs { padding:8px; padding-top:2px; }',
            // Main Tabs
            '.URCE-tabIcon { padding-bottom:6px; width:18px; }',
            // Alert Box
            '#urceAlertBox { position:fixed; visibility:hidden; top:50%; left:50%; z-index:10000; background-color:aliceBlue; border-width:3px; border-style:solid; border-radius:10px; box-shadow:5px 5px 10px silver; padding:4px; -webkit-transform:translate(-50%, -50%); transform:translate(-50%, -50%); }',
            '.URCE-alertBox-header { padding:4px; background-color:LightBlue; font-weight:bold; font-size:14px; }',
            '.URCE-alertBox-content { padding:4px; background-color:White; overflow:auto; max-height:500px; }',
            '.URCE-alertBox-controls { text-align:center; padding:4px; }',
            '.URCE-alertBox-controls-span-urceAlertTickBtn, .URCE-alertBox-controls-span-urceAlertCrossBtn { cursor:pointer; font-size:14px; border:thin outset black; padding:2px 10px 2px 10px; }',
            '.URCE-alertBox-controls-span-urceAlertTickBtnCaption, .URCE-alertBox-controls-span-urceAlertCrossBtnCaption { font-weight:bold; }'
        ].join(' ');
        $('<style = type="text/css">' + css + '</style>').appendTo('head');
    }

    function initCommentsTab() {
        logDebug('Initializing Comments tab.');
        $('#panel-urce-comments').append(
            $('<div>', {id:'_divZoomOutLinks', class:'URCE-divCCLinks'}).append(
                $('<div>', {id:'urceIcon', class:'URCE-divIcon'}).append(
                    $('<img>', {src:GM_info.script.icon, class:'URCE-icon'})
                ),
                $('<a>', {id:'zoomOutLink1', class:'URCE-Comments', title:I18n.t('urce.commentsTab.ZoomOutLink1Title')}).text(I18n.t('urce.commentsTab.ZoomOutLink1')).on('click', function() {
                    if ($(".problem-edit .header .close-panel").length !== 0) $("#panel-container > div > div > div.top-section > a").trigger('click');
                    W.map.setCenter(W.map.getCenter(), 0);

                }).append('<br>'),
                $('<a>', {id:'zoomOutlink2', class:'URCE-Comments', title:I18n.t('urce.commentsTab.ZoomOutLink2Title')}).text(I18n.t('urce.commentsTab.ZoomOutLink2')).on('click', function() {
                    if ($(".problem-edit .header .close-panel").length !== 0) $("#panel-container > div > div > div.top-section > a").trigger('click');
                    W.map.setCenter(W.map.getCenter(), 2);

                }).append('<br>'),
                $('<a>', {id:'zoomOutlink3', class:'URCE-Comments', title:I18n.t('urce.commentsTab.ZoomOutLink3Title')}).text(I18n.t('urce.commentsTab.ZoomOutLink3')).on('click', function() {
                    if ($(".problem-edit .header .close-panel").length !== 0) $("#panel-container > div > div > div.top-section > a").trigger('click');
                    W.map.setCenter(W.map.getCenter(), 3);

                }).append('<br>')
            ).append(function() {
                if (_settings.hideZoomOutLinks) $(this).hide();
            }),
            $('<div>', {id:'_commentList', class:'URCE-divCC'}).append(
                $('<div>', {class:'URCE-divLoading'}).text(I18n.t('urce.common.Loading') + ' ' + getCommentListInfo().name + ' comment list. ' + I18n.t('urce.common.PleaseWait') + '...')
            )
        )
    }

    function initSettingsTab() {
        logDebug('Initializing Settings tab.');
        let urStyle = (_settings.commentListStyle === 'urStyle') ? ' urStyle' : '';
        $('#panel-urce-settings').append(
            // Comment List
            $('<fieldset>', {id:'urce-prefs-fieldset-commentList', class:`URCE-field${urStyle}`}).append(
                $('<legend>', {id:'urce-prefs-legend-commentList', class:`URCE-legend${urStyle}`}).append(
                    $('<i>', {class:'fa fa-fw fa-chevron-down URCE-chevron'}),
                    $('<span>', {class:'URCE-span'}).text(I18n.t('urce.prefs.CommentList'))
                ).click(function() {
                    $($(this).children()[0]).toggleClass('fa fa-fw fa-chevron-down');
                    $($(this).children()[0]).toggleClass('fa fa-fw fa-chevron-right');
                    $($(this).siblings()[0]).toggleClass('collapse');
                }),
                $('<div>', {class:'URCE-controls URCE-divCC'}).append(
                    // Comment list
                    $('<div>').text(I18n.t('urce.common.List') + ': ').append(function() {
                        let $selList = $('<select>', {id:'_selCommentList', title:I18n.t('urce.prefs.CommentListTitle'), urceprefs:'commentList'});
                        _commentLists.forEach(cList => {
                            if (cList.status === 'disabled') {
                                return;
                            } else if (cList.idx === _settings.commentList) {
                                $selList.append($('<option>', {value:cList.idx, selected:true}).text(cList.name));
                            } else {
                                $selList.append($('<option>', {value:cList.idx}).text(cList.name));
                            }
                        });
                        return $selList.val(_settings.commentList).change(function() {
                            changeCommentList($(this).val());
                        });
                    }),
                    // Comment list style
                    $('<div>').text(I18n.t('urce.common.Style') + ': ').append(function() {
                        let $selList = $('<select>', {id:'_selCommentListStyle', title:I18n.t('urce.prefs.CommentListStyleTitle'), urceprefs:'commentList'});
                        if (_settings.commentListStyle === 'default') {
                            $selList.append(
                                $('<option>', {value:'default', selected:true}).text(I18n.t('urce.prefs.StyleDefault'))
                            )
                        } else {
                            $selList.append(
                                $('<option>', {value:'default'}).text(I18n.t('urce.prefs.StyleDefault'))
                            )
                        }
                        if (_settings.commentListStyle === 'urStyle') {
                            $selList.append(
                                $('<option>', {value:'urStyle', selected:true}).text(I18n.t('urce.prefs.StyleUrStyle'))
                            )
                        } else {
                            $selList.append(
                                $('<option>', {value:'urStyle'}).text(I18n.t('urce.prefs.StyleUrStyle'))
                            )
                        }
                        $selList.change(function() {
                            changeCommentListStyle($(this).val());
                        })
                        return $selList;
                    })
                )
            ),
            // URC-E Preferences
            $('<fieldset>', {id:'urce-prefs-fieldset-urc-e-prefs', class:`URCE-field${urStyle}`}).append(
                $('<legend>', {id:'urce-prefs-legend-urc-e-prefs', class:`URCE-legend${urStyle}`}).append(
                    $('<i>', {class:'fa fa-fw fa-chevron-down URCE-chevron'}),
                    $('<span>', {class:'URCE-span'}).text(I18n.t('urce.prefs.UrcePrefs'))
                ).click(function() {
                    $($(this).children()[0]).toggleClass('fa fa-fw fa-chevron-down');
                    $($(this).children()[0]).toggleClass('fa fa-fw fa-chevron-right');
                    $($(this).siblings()[0]).toggleClass('collapse');
                }),
                $('<div>', {class:'URCE-controls URCE-divCC'}).append(
                    $('<input>', {type:'checkbox', id:'_cbAutoCenterOnUr', urceprefs:'urce'}).change(function() { changeSetting('autoCenterOnUr', $(this).is(':checked')); }).prop('checked', _settings.autoCenterOnUr),
                    $('<label>', {for:'_cbAutoCenterOnUr', title:I18n.t('urce.prefs.AutoCenterOnUrTitle'), class:'URCE-label'}).text(I18n.t('urce.prefs.AutoCenterOnUr')),
                    $('<br>'),
                    $('<input>', {type:'checkbox', id:'_cbAutoClickOpenSolvedNi', urceprefs:'urce'}).change(function() {
                        _settings.autoClickOpenSolvedNi = $(this).is(':checked');
                        if (!$(this).is(':checked')) {
                            if (isChecked('_cbAutoSaveAfterSolvedOrNiComment')) {
                                _settings.autoSaveAfterSolvedOrNiComment = false;
                                $('#_cbAutoSaveAfterSolvedOrNiComment').prop('checked', false);
                            }
                            if (isChecked('_cbDoubleClickLinkNiComments')) {
                                _settings.doubleClickLinkNiComments = false;
                                $('#_cbDoubleClickLinkNiComments').prop('checked', false);
                            }
                            if (isChecked('_cbDoubleClickLinkOpenComments')) {
                                _settings.doubleClickLinkOpenComments = false;
                                $('#_cbDoubleClickLinkOpenComments').prop('checked', false);
                            }
                            if (isChecked('_cbDoubleClickLinkSolvedComments')) {
                                _settings.doubleClickLinkSolvedComments = false;
                                $('#_cbDoubleClickLinkSolvedComments').prop('checked', false);
                            }
                        }
                        saveSettingsToStorage();
                    }).prop('checked', _settings.autoClickOpenSolvedNi),
                    $('<label>', {for:'_cbAutoClickOpenSolvedNi', title:I18n.t('urce.prefs.AutoClickOpenSolvedNiTitle'), class:'URCE-label'}).text(I18n.t('urce.prefs.AutoClickOpenSolvedNi')),
                    $('<br>'),
                    $('<input>', {type:'checkbox', id:'_cbAutoCloseCommentWindow', urceprefs:'urce'}).change(function() { changeSetting('autoCloseCommentWindow', $(this).is(':checked')); }).prop('checked', _settings.autoCloseCommentWindow),
                    $('<label>', {for:'_cbAutoCloseCommentWindow', title:I18n.t('urce.prefs.AutoCloseCommentWindowTitle'), class:'URCE-label'}).text(I18n.t('urce.prefs.AutoCloseCommentWindow')),
                    $('<br>'),
                    $('<input>', {type:'checkbox', id:'_cbAutoSaveAfterSolvedOrNiComment', urceprefs:'urce'}).change(function() {
                        _settings.autoSaveAfterSolvedOrNiComment = $(this).is(':checked');
                        if ($(this).is(':checked') && !isChecked('_cbAutoClickOpenSolvedNi')) {
                            _settings.autoClickOpenSolvedNi = true;
                            $('#_cbAutoClickOpenSolvedNi').prop('checked', true);
                        }
                        saveSettingsToStorage();
                    }).prop('checked', _settings.autoSaveAfterSolvedOrNiComment),
                    $('<label>', {for:'_cbAutoSaveAfterSolvedOrNiComment', title:I18n.t('urce.prefs.AutoSaveAfterSolvedOrNiCommentTitle'), class:'URCE-label'}).text(I18n.t('urce.prefs.AutoSaveAfterSolvedOrNiComment')),
                    $('<br>'),
                    $('<input>', {type:'checkbox', id:'_cbAutoSendReminders', urceprefs:'urce'}).change(function() { changeSetting('autoSendReminders', $(this).is(':checked')); }).prop('checked', _settings.autoSendReminders),
                    $('<label>', {for:'_cbAutoSendReminders', title:I18n.t('urce.prefs.AutoSendRemindersTitle'), class:'URCE-label'}).text(I18n.t('urce.prefs.AutoSendReminders')),
                    $('<div>', {class:'URCE-divWarning URCE-divWarningPre'}).text('(').append(
                        $('<div>', {class:'URCE-divWarning URCE-divWarningTitle', title:I18n.t('urce.prefs.AutoSendRemindersWarningTitle')}).text(I18n.t('urce.prefs.AutoSendRemindersWarning')),
                    ).append(
                        $('<div>', {class:'URCE-divWarning'}).text(')')
                    ),
                    $('<br>'),
                    $('<input>', {type:'checkbox', id:'_cbAutoSetNewUrComment', urceprefs:'urce'}).change(function() { changeSetting('autoSetNewUrComment', $(this).is(':checked')); }).prop('checked', _settings.autoSetNewUrComment),
                    $('<label>', {for:'_cbAutoSetNewUrComment', title:I18n.t('urce.prefs.AutoSetNewUrCommentTitle'), class:'URCE-label'}).text(I18n.t('urce.prefs.AutoSetNewUrComment')),
                    $('<br>'),
                    $('<input>', {type:'checkbox', id:'_cbAutoSetReminderUrComment', urceprefs:'urce'}).change(function() { changeSetting('autoSetReminderUrComment', $(this).is(':checked')); }).prop('checked', _settings.autoSetReminderUrComment),
                    $('<label>', {for:'_cbAutoSetReminderUrComment', title:I18n.t('urce.prefs.AutoSetReminderUrCommentTitle'), class:'URCE-label'}).text(I18n.t('urce.prefs.AutoSetReminderUrComment')),
                    $('<br>'),
                    $('<input>', {type:'checkbox', id:'_cbAutoSwitchToUrCommentsTab', urceprefs:'urce'}).change(function() { changeSetting('autoSwitchToUrCommentsTab', $(this).is(':checked')); }).prop('checked', _settings.autoSwitchToUrCommentsTab),
                    $('<label>', {for:'_cbAutoSwitchToUrCommentsTab', title:I18n.t('urce.prefs.AutoSwitchToUrCommentsTabTitle'), class:'URCE-label'}).text(I18n.t('urce.prefs.AutoSwitchToUrCommentsTab')),
                    $('<br>'),
                    $('<input>', {type:'checkbox', id:'_cbAutoZoomInOnNewUr', urceprefs:'urce'}).change(function() { changeSetting('autoZoomInOnNewUr', $(this).is(':checked')); }).prop('checked', _settings.autoZoomInOnNewUr),
                    $('<label>', {for:'_cbAutoZoomInOnNewUr', title:I18n.t('urce.prefs.AutoZoomInOnNewUrTitle'), class:'URCE-label'}).text(I18n.t('urce.prefs.AutoZoomInOnNewUr')),
                    $('<br>'),
                    $('<input>', {type:'checkbox', id:'_cbAutoZoomOutAfterComment', urceprefs:'urce'}).change(function() { changeSetting('autoZoomOutAfterComment', $(this).is(':checked')); }).prop('checked', _settings.autoZoomOutAfterComment),
                    $('<label>', {for:'_cbAutoZoomOutAfterComment', title:I18n.t('urce.prefs.AutoZoomOutAfterCommentTitle'), class:'URCE-label'}).text(I18n.t('urce.prefs.AutoZoomOutAfterComment')),
                    $('<br>'),
                    $('<input>', {type:'checkbox', id:'_cbDisableDoneNextButtons', urceprefs:'urce'}).change(function() {
                        changeSetting('disableDoneNextButtons', $(this).is(':checked'));
                        if ($(this).is(':checked')) {
                            $('#panel-container .content .navigation').css({'display':'none'});
                        } else {
                            $('#panel-container .content .navigation').css({'display':'block'});
                        }
                    }).prop('checked', _settings.disableDoneNextButtons),
                    $('<label>', {for:'_cbDisableDoneNextButtons', title:I18n.t('urce.prefs.DisableDoneNextButtonsTitle'), class:'URCE-label'}).text(I18n.t('urce.prefs.DisableDoneNextButtons')),
                    $('<br>'),
                    $('<input>', {type:'checkbox', id:'_cbDoubleClickLinkNiComments', urceprefs:'urce'}).change(function() {
                        _settings.doubleClickLinkNiComments = $(this).is(':checked');
                        if (!$(this).is(':checked')) {
                            $('div#URCE-divDoubleClickNi').hide();
                        } else {
                            if (!isChecked('_cbAutoClickOpenSolvedNi')) {
                                _settings.autoClickOpenSolvedNi = true;
                                $('#_cbAutoClickOpenSolvedNi').prop('checked', true);
                            }
                            $('div#URCE-divDoubleClickNi').show();
                        }
                        saveSettingsToStorage();
                    }).prop('checked', _settings.doubleClickLinkNiComments),
                    $('<label>', {for:'_cbDoubleClickLinkNiComments', title:I18n.t('urce.prefs.DoubleClickLinkNiCommentsTitle'), class:'URCE-label'}).text(I18n.t('urce.prefs.DoubleClickLinkNiComments')),
                    $('<br>'),
                    $('<input>', {type:'checkbox', id:'_cbDoubleClickLinkOpenComments', urceprefs:'urce'}).change(function() {
                        _settings.doubleClickLinkOpenComments = $(this).is(':checked');
                        if (!$(this).is(':checked')) {
                            $('div#URCE-divDoubleClickOpen').hide();
                        } else {
                            if (!isChecked('_cbAutoClickOpenSolvedNi')) {
                                _settings.autoClickOpenSolvedNi = true;
                                $('#_cbAutoClickOpenSolvedNi').prop('checked', true);
                            }
                            $('div#URCE-divDoubleClickOpen').show();
                        }
                        saveSettingsToStorage();
                    }).prop('checked', _settings.doubleClickLinkOpenComments),
                    $('<label>', {for:'_cbDoubleClickLinkOpenComments', title:I18n.t('urce.prefs.DoubleClickLinkOpenCommentsTitle'), class:'URCE-label'}).text(I18n.t('urce.prefs.DoubleClickLinkOpenComments')),
                    $('<br>'),
                    $('<input>', {type:'checkbox', id:'_cbDoubleClickLinkSolvedComments', urceprefs:'urce'}).change(function() {
                        _settings.doubleClickLinkSolvedComments = $(this).is(':checked');
                        if (!$(this).is(':checked')) {
                            $('div#URCE-divDoubleClickSolved').hide();
                        } else {
                            if (!isChecked('_cbAutoClickOpenSolvedNi')) {
                                _settings.autoClickOpenSolvedNi = true;
                                $('#_cbAutoClickOpenSolvedNi').prop('checked', true);
                            }
                            $('div#URCE-divDoubleClickSolved').show();
                        }
                        saveSettingsToStorage();
                    }).prop('checked', _settings.doubleClickLinkSolvedComments),
                    $('<label>', {for:'_cbDoubleClickLinkSolvedComments', title:I18n.t('urce.prefs.DoubleClickLinkSolvedCommentsTitle'), class:'URCE-label'}).text(I18n.t('urce.prefs.DoubleClickLinkSolvedComments')),
                    $('<br>'),
                    $('<input>', {type:'checkbox', id:'_cbHideZoomOutLinks', urceprefs:'urce'}).change(function() {
                        changeSetting('hideZoomOutLinks', $(this).is(':checked'));
                        if ($(this).is(':checked')) {
                            $('div#_divZoomOutLinks').hide();
                        } else {
                            $('div#_divZoomOutLinks').show();
                        }
                    }).prop('checked', _settings.hideZoomOutLinks),
                    $('<label>', {for:'_cbHideZoomOutLinks', title:I18n.t('urce.prefs.HideZoomOutLinksTitle'), class:'URCE-label'}).text(I18n.t('urce.prefs.HideZoomOutLinks')),
                    $('<br>'),
                    $('<input>', {type:'checkbox', id:'_cbUnfollowUrAfterSend', urceprefs:'urce'}).change(function() { changeSetting('unfollowUrAfterSend', $(this).is(':checked')); }).prop('checked', _settings.unfollowUrAfterSend),
                    $('<label>', {for:'_cbUnfollowUrAfterSend', title:I18n.t('urce.prefs.UnfollowUrAfterSendTitle'), class:'URCE-label'}).text(I18n.t('urce.prefs.UnfollowUrAfterSend'))
                )
            ),
            // UR Marker Preferences
            $('<fieldset>', {id:'urce-prefs-fieldset-ur-marker-prefs', class:`URCE-field${urStyle}`}).append(
                $('<legend>', {id:'urce-prefs-legend-ur-marker-prefs', class:`URCE-legend${urStyle}`}).append(
                    $('<i>', {class:'fa fa-fw fa-chevron-down URCE-chevron'}),
                    $('<span>', {class:'URCE-span'}).text(I18n.t('urce.prefs.UrMarkerPrefs'))
                ).click(function() {
                    $($(this).children()[0]).toggleClass('fa fa-fw fa-chevron-down');
                    $($(this).children()[0]).toggleClass('fa fa-fw fa-chevron-right');
                    $($(this).siblings()[0]).toggleClass('collapse');
                }),
                $('<div>', {class:'URCE-controls URCE-divCC'}).append(
                    $('<input>', {type:'checkbox', id:'_cbEnableUrPillCounts', urceprefs:'markerMaster'}).change(function() {
                        changeSetting('enableUrPillCounts', $(this).is(':checked'));
                        if (!$(this).is(':checked')) {
                            $('[urceprefs=marker]').prop('disabled', true);
                        } else {
                            $('[urceprefs=marker]').prop('disabled', false);
                        }
                        (async () => {
                            try {
                                await handleUrLayer('settingsToggle');
                            } catch(error) {
                                logWarning(error);
                            }
                        })();
                    }).prop('checked', _settings.enableUrPillCounts),
                    $('<label>', {for:'_cbEnableUrPillCounts', title:I18n.t('urce.prefs.EnableUrPillCountsTitle'), class:'URCE-label'}).text(I18n.t('urce.prefs.EnableUrPillCounts')),
                    $('<br>'),
                    $('<input>', {type:'checkbox', id:'_cbDoNotShowTagNameOnPill', urceprefs:'marker'}).change(function() {
                        changeSetting('doNotShowTagNameOnPill', $(this).is(':checked'));
                        (async () => {
                            try {
                                await handleUrLayer('settingsToggle');
                            } catch(error) {
                                logWarning(error);
                            }
                        })();
                    }).prop('checked', _settings.doNotShowTagNameOnPill),
                    $('<label>', {for:'_cbDoNotShowTagNameOnPill', title:I18n.t('urce.prefs.DoNotShowTagNameOnPillTitle'), class:'URCE-label'}).text(I18n.t('urce.prefs.DoNotShowTagNameOnPill')),
                    $('<br>'),
                    $('<input>', {type:'checkbox', id:'_cbReplaceTagNameWithEditorName', urceprefs:'marker'}).change(function() {
                        changeSetting('replaceTagNameWithEditorName', $(this).is(':checked'));
                        (async () => {
                            try {
                                await handleUrLayer('settingsToggle');
                            } catch(error) {
                                logWarning(error);
                            }
                        })();
                    }).prop('checked', _settings.replaceTagNameWithEditorName),
                    $('<label>', {for:'_cbReplaceTagNameWithEditorName', title:I18n.t('urce.prefs.ReplaceTagNameWithEditorNameTitle'), class:'URCE-label'}).text(I18n.t('urce.prefs.ReplaceTagNameWithEditorName')),
                    $('<br>'),
                    $('<input>', {type:'checkbox', id:'_cbUnstackMarkers', urceprefs:'marker-nodisable'}).change(function() {
                        changeSetting('unstackMarkers', $(this).is(':checked'));
                        (async () => {
                            try {
                                await handleUrLayer('settingsToggle');
                            } catch(error) {
                                logWarning(error);
                            }
                        })();
                    }).prop('checked', _settings.unstackMarkers),
                    $('<label>', {for:'_cbUnstackMarkers', title:I18n.t('urce.prefs.UnstackMarkersTitle'), class:'URCE-label'}).text(I18n.t('urce.prefs.UnstackMarkers')),
                    $('<br>'),
                    // -- Custom markers
                    $('<div>').append(
                        $('<div>', {class:'URCE-subHeading'}).text(I18n.t('urce.prefs.UseCustomMarkersFor') + ':').css({fontWeight:'600'}).append('<br>')
                    ).append(
                        $('<input>', {type:'checkbox', id:'_cbCustomMarkersBog', urceprefs:'marker-nodisable'}).change(function() {
                            changeSetting('customMarkersBog', $(this).is(':checked'));
                            (async () => {
                                try {
                                    await handleUrLayer('settingsToggle');
                                } catch(error) {
                                    logWarning(error);
                                }
                            })();
                        }).prop('checked', _settings.customMarkersBog),
                        $('<label>', {for:'_cbCustomMarkersBog', title:I18n.t('urce.tags.BogTitle'), class:'URCE-label'}).text(I18n.t('urce.tags.Bog')),
                        $('<br>'),
                        $('<input>', {type:'checkbox', id:'_cbCustomMarkersClosures', urceprefs:'marker-nodisable'}).change(function() {
                            changeSetting('customMarkersClosures', $(this).is(':checked'));
                            (async () => {
                                try {
                                    await handleUrLayer('settingsToggle');
                                } catch(error) {
                                    logWarning(error);
                                }
                            })();
                        }).prop('checked', _settings.customMarkersClosures),
                        $('<label>', {for:'_cbCustomMarkersClosures', title:I18n.t('urce.tags.ClosureTitle'), class:'URCE-label'}).text(I18n.t('urce.tags.Closure')),
                        $('<br>'),
                        $('<input>', {type:'checkbox', id:'_cbCustomMarkersConstruction', urceprefs:'marker-nodisable'}).change(function() {
                            changeSetting('customMarkersConstruction', $(this).is(':checked'));
                            (async () => {
                                try {
                                    await handleUrLayer('settingsToggle');
                                } catch(error) {
                                    logWarning(error);
                                }
                            })();
                        }).prop('checked', _settings.customMarkersConstruction),
                        $('<label>', {for:'_cbCustomMarkersConstruction', title:I18n.t('urce.tags.ConstructionTitle'), class:'URCE-label'}).text(I18n.t('urce.tags.Construction')),
                        $('<br>'),
                        $('<input>', {type:'checkbox', id:'_cbCustomMarkersDifficult', urceprefs:'marker-nodisable'}).change(function() {
                            changeSetting('customMarkersDifficult', $(this).is(':checked'));
                            (async () => {
                                try {
                                    await handleUrLayer('settingsToggle');
                                } catch(error) {
                                    logWarning(error);
                                }
                            })();
                        }).prop('checked', _settings.customMarkersDifficult),
                        $('<label>', {for:'_cbCustomMarkersDifficult', title:I18n.t('urce.tags.DifficultTitle'), class:'URCE-label'}).text(I18n.t('urce.tags.Difficult')),
                        $('<br>'),
                        $('<input>', {type:'checkbox', id:'_cbCustomMarkersEvents', urceprefs:'marker-nodisable'}).change(function() {
                            changeSetting('customMarkersEvents', $(this).is(':checked'));
                            (async () => {
                                try {
                                    await handleUrLayer('settingsToggle');
                                } catch(error) {
                                    logWarning(error);
                                }
                            })();
                        }).prop('checked', _settings.customMarkersEvents),
                        $('<label>', {for:'_cbCustomMarkersEvents', title:I18n.t('urce.tags.EventTitle'), class:'URCE-label'}).text(I18n.t('urce.tags.Event')),
                        $('<br>'),
                        $('<input>', {type:'checkbox', id:'_cbCustomMarkersNotes', urceprefs:'marker-nodisable'}).change(function() {
                            changeSetting('customMarkersNotes', $(this).is(':checked'));
                            (async () => {
                                try {
                                    await handleUrLayer('settingsToggle');
                                } catch(error) {
                                    logWarning(error);
                                }
                            })();
                        }).prop('checked', _settings.customMarkersNotes),
                        $('<label>', {for:'_cbCustomMarkersNotes', title:I18n.t('urce.tags.NoteTitle'), class:'URCE-label'}).text(I18n.t('urce.tags.Note')),
                        $('<br>'),
                        $('<input>', {type:'checkbox', id:'_cbCustomMarkersRoadworks', urceprefs:'marker-nodisable'}).change(function() {
                            changeSetting('customMarkersRoadworks', $(this).is(':checked'));
                            (async () => {
                                try {
                                    await handleUrLayer('settingsToggle');
                                } catch(error) {
                                    logWarning(error);
                                }
                            })();
                        }).prop('checked', _settings.customMarkersRoadworks),
                        $('<label>', {for:'_cbCustomMarkersRoadworks', title:I18n.t('urce.tags.RoadworksTitle'), class:'URCE-label'}).text(I18n.t('urce.tags.Roadworks')),
                        $('<br>'),
                        $('<input>', {type:'checkbox', id:'_cbCustomMarkersWslm', urceprefs:'marker-nodisable'}).change(function() {
                            changeSetting('customMarkersWslm', $(this).is(':checked'));
                            (async () => {
                                try {
                                    await handleUrLayer('settingsToggle');
                                } catch(error) {
                                    logWarning(error);
                                }
                            })();
                        }).prop('checked', _settings.customMarkersWslm),
                        $('<label>', {for:'_cbCustomMarkersWslm', title:I18n.t('urce.tags.WslmTitle'), class:'URCE-label'}).text(I18n.t('urce.tags.Wslm')),
                        $('<br>'),
                        $('<input>', {type:'checkbox', id:'_cbCustomMarkersNativeSl', urceprefs:'marker-nodisable'}).change(function() {
                            changeSetting('customMarkersNativeSl', $(this).is(':checked'));
                            (async () => {
                                try {
                                    await handleUrLayer('settingsToggle');
                                } catch(error) {
                                    logWarning(error);
                                }
                            })();
                        }).prop('checked', _settings.customMarkersNativeSl),
                        $('<label>', {for:'_cbCustomMarkersNativeSl', title:I18n.t('urce.prefs.NativeSpeedLimitsTitle'), class:'URCE-label'}).text(I18n.t('urce.prefs.NativeSpeedLimits'))
                    )
                )
            ),
            // UR Filtering Preferences
            $('<fieldset>', {id:'urce-prefs-fieldset-ur-filtering-prefs', class:`URCE-field${urStyle}`}).append(
                $('<legend>', {id:'urce-prefs-legend-ur-filtering-prefs', class:`URCE-legend${urStyle}`}).append(
                    $('<i>', {class:'fa fa-fw fa-chevron-down URCE-chevron'}),
                    $('<span>', {class:'URCE-span'}).text(I18n.t('urce.prefs.UrFilteringPrefs'))
                ).click(function() {
                    $($(this).children()[0]).toggleClass('fa fa-fw fa-chevron-down');
                    $($(this).children()[0]).toggleClass('fa fa-fw fa-chevron-right');
                    $($(this).siblings()[0]).toggleClass('collapse');
                }),
                $('<div>', {class:'URCE-controls URCE-divCC'}).append(
                    $('<input>', {type:'checkbox', id:'_cbEnableUrceUrFiltering', urceprefs:'filteringMaster'}).change(function() {
                        _settings.enableUrceUrFiltering = $(this).is(':checked');
                        if (!$(this).is(':checked')) {
                            $('[urceprefs=filtering]').prop('disabled', true);
                        } else {
                            $('[urceprefs=filtering]').prop('disabled', false);
                        }
                        saveSettingsToStorage();
                        (async () => {
                            try {
                                await handleUrLayer('settingsToggle');
                            } catch(error) {
                                logWarning(error);
                            }
                        })();
                    }).prop('checked', _settings.enableUrceUrFiltering),
                    $('<label>', {for:'_cbEnableUrceUrFiltering', title:I18n.t('urce.prefs.EnableUrceUrFilteringTitle'), class:'URCE-label'}).text(I18n.t('urce.prefs.EnableUrceUrFiltering')),
                    $('<br>'),
                    $('<input>', {type:'checkbox', id:'_cbHideOutsideEditableArea', urceprefs:'filtering'}).change(function() {
                        changeSetting('hideOutsideEditableArea', $(this).is(':checked'));
                        (async () => {
                            try {
                                await handleUrLayer('settingsToggle');
                            } catch(error) {
                                logWarning(error);
                            }
                        })();
                    }).prop('checked', _settings.hideOutsideEditableArea),
                    $('<label>', {for:'_cbHideOutsideEditableArea', title:I18n.t('urce.prefs.HideOutsideEditableAreaTitle'), class:'URCE-label'}).text(I18n.t('urce.prefs.HideOutsideEditableArea')),
                    $('<br>'),
                    $('<input>', {type:'checkbox', id:'_cbDoNotHideSelectedUr', urceprefs:'filtering'}).change(function() {
                        changeSetting('doNotHideSelectedUr', $(this).is(':checked'));
                        (async () => {
                            try {
                                await handleUrLayer('settingsToggle');
                            } catch(error) {
                                logWarning(error);
                            }
                        })();
                    }).prop('checked', _settings.doNotHideSelectedUr),
                    $('<label>', {for:'_cbDoNotHideSelectedUr', title:I18n.t('urce.prefs.DoNotHideSelectedUrTitle'), class:'URCE-label'}).text(I18n.t('urce.prefs.DoNotHideSelectedUr')),
                    $('<br>'),
                    // -- Lifecycle
                    $('<div>').append(
                        $('<div>', {class:'URCE-subHeading'}).text(I18n.t('urce.prefs.LifeCycleStatus') + ':').append('<br>')
                    ).append(
                        $('<input>', {type:'checkbox', id:'_cbHideWaiting', urceprefs:'filtering'}).change(function() {
                            changeSetting('hideWaiting', $(this).is(':checked'));
                            (async () => {
                                try {
                                    await handleUrLayer('settingsToggle');
                                } catch(error) {
                                    logWarning(error);
                                }
                            })();
                        }).prop('checked', _settings.hideWaiting),
                        $('<label>', {for:'_cbHideWaiting', title:I18n.t('urce.prefs.HideWaitingTitle'), class:'URCE-label'}).text(I18n.t('urce.prefs.HideWaiting')),
                        $('<br>'),
                        $('<input>', {type:'checkbox', id:'_cbHideUrsCloseNeeded', urceprefs:'filtering'}).change(function() {
                            changeSetting('hideUrsCloseNeeded', $(this).is(':checked'));
                            (async () => {
                                try {
                                    await handleUrLayer('settingsToggle');
                                } catch(error) {
                                    logWarning(error);
                                }
                            })();
                        }).prop('checked', _settings.hideUrsCloseNeeded),
                        $('<label>', {for:'_cbHideUrsCloseNeeded', title:I18n.t('urce.prefs.HideUrsCloseNeededTitle'), class:'URCE-label'}).text(I18n.t('urce.prefs.HideUrsCloseNeeded')),
                        $('<br>'),
                        $('<input>', {type:'checkbox', id:'_cbHideUrsReminderNeeded', urceprefs:'filtering'}).change(function() {
                            changeSetting('hideUrsReminderNeeded', $(this).is(':checked'));
                            (async () => {
                                try {
                                    await handleUrLayer('settingsToggle');
                                } catch(error) {
                                    logWarning(error);
                                }
                            })();
                        }).prop('checked', _settings.hideUrsReminderNeeded),
                        $('<label>', {for:'_cbHideUrsReminderNeeded', title:I18n.t('urce.prefs.HideUrsReminderNeededTitle'), class:'URCE-label'}).text(I18n.t('urce.prefs.HideUrsReminderNeeded')),
                        $('<br>')
                    ),
                    // -- Hide by status
                    $('<div>').append(
                        $('<div>', {class:'URCE-subHeading'}).text(I18n.t('urce.prefs.HideByStatus') + ':').append('<br>')
                    ).append(
                        $('<input>', {type:'checkbox', id:'_cbHideByStatusOpen', urceprefs:'filtering'}).change(function() {
                            changeSetting('hideByStatusOpen', $(this).is(':checked'));
                            (async () => {
                                try {
                                    await handleUrLayer('settingsToggle');
                                } catch(error) {
                                    logWarning(error);
                                }
                            })();
                        }).prop('checked', _settings.hideByStatusOpen),
                        $('<label>', {for:'_cbHideByStatusOpen', title:I18n.t('urce.prefs.HideByStatusOpenTitle'), class:'URCE-label'}).text(I18n.t('urce.urStatus.Open')),
                        $('<br>'),
                        $('<input>', {type:'checkbox', id:'_cbHideByStatusClosed', urceprefs:'filtering'}).change(function() {
                            changeSetting('hideByStatusClosed', $(this).is(':checked'));
                            (async () => {
                                try {
                                    await handleUrLayer('settingsToggle');
                                } catch(error) {
                                    logWarning(error);
                                }
                            })();
                        }).prop('checked', _settings.hideByStatusClosed),
                        $('<label>', {for:'_cbHideByStatusClosed', title:I18n.t('urce.prefs.HideByStatusClosedTitle'), class:'URCE-label'}).text(I18n.t('urce.urStatus.Closed')),
                        $('<br>'),
                        $('<input>', {type:'checkbox', id:'_cbHideByStatusNotIdentified', urceprefs:'filtering'}).change(function() {
                            changeSetting('hideByStatusNotIdentified', $(this).is(':checked'));
                            (async () => {
                                try {
                                    await handleUrLayer('settingsToggle');
                                } catch(error) {
                                    logWarning(error);
                                }
                            })();
                        }).prop('checked', _settings.hideByStatusNotIdentified),
                        $('<label>', {for:'_cbHideByStatusNotIdentified', title:I18n.t('urce.prefs.HideByStatusNotIdentifiedTitle'), class:'URCE-label'}).text(I18n.t('urce.urStatus.NotIdentified')),
                        $('<br>'),
                        $('<input>', {type:'checkbox', id:'_cbHideByStatusSolved', urceprefs:'filtering'}).change(function() {
                            changeSetting('hideByStatusSolved', $(this).is(':checked'));
                            (async () => {
                                try {
                                    await handleUrLayer('settingsToggle');
                                } catch(error) {
                                    logWarning(error);
                                }
                            })();
                        }).prop('checked', _settings.hideByStatusSolved),
                        $('<label>', {for:'_cbHideByStatusSolved', title:I18n.t('urce.prefs.HideByStatusSolvedTitle'), class:'URCE-label'}).text(I18n.t('urce.urStatus.Solved')),
                        $('<br>')
                    ),
                    // -- Hide by type
                    $('<div>').append(
                        $('<div>', {class:'URCE-subHeading'}).text(I18n.t('urce.prefs.HideByType') + ':').append('<br>')
                    ).append(
                        $('<input>', {type:'checkbox', id:'_cbHideByTypeBlockedRoad', urceprefs:'filtering'}).change(function() {
                            changeSetting('hideByTypeBlockedRoad', $(this).is(':checked'));
                            (async () => {
                                try {
                                    await handleUrLayer('settingsToggle');
                                } catch(error) {
                                    logWarning(error);
                                }
                            })();
                        }).prop('checked', _settings.hideByTypeBlockedRoad),
                        $('<label>', {for:'_cbHideByTypeBlockedRoad', title:I18n.t('urce.prefs.HideByTypeBlockedRoadTitle'), class:'URCE-label'}).text(I18n.t('urce.urTypes.BlockedRoad')),
                        $('<br>'),
                        $('<input>', {type:'checkbox', id:'_cbHideByTypeGeneralError', urceprefs:'filtering'}).change(function() {
                            changeSetting('hideByTypeGeneralError', $(this).is(':checked'));
                            (async () => {
                                try {
                                    await handleUrLayer('settingsToggle');
                                } catch(error) {
                                    logWarning(error);
                                }
                            })();
                        }).prop('checked', _settings.hideByTypeGeneralError),
                        $('<label>', {for:'_cbHideByTypeGeneralError', title:I18n.t('urce.prefs.HideByTypeGeneralErrorTitle'), class:'URCE-label'}).text(I18n.t('urce.urTypes.GeneralError')),
                        $('<br>'),
                        $('<input>', {type:'checkbox', id:'_cbHideByTypeIncorrectAddress', urceprefs:'filtering'}).change(function() {
                            changeSetting('hideByTypeIncorrectAddress', $(this).is(':checked'));
                            (async () => {
                                try {
                                    await handleUrLayer('settingsToggle');
                                } catch(error) {
                                    logWarning(error);
                                }
                            })();
                        }).prop('checked', _settings.hideByTypeIncorrectAddress),
                        $('<label>', {for:'_cbHideByTypeIncorrectAddress', title:I18n.t('urce.prefs.HideByTypeIncorrectAddressTitle'), class:'URCE-label'}).text(I18n.t('urce.urTypes.IncorrectAddress')),
                        $('<br>'),
                        $('<input>', {type:'checkbox', id:'_cbHideByTypeIncorrectJunction', urceprefs:'filtering'}).change(function() {
                            changeSetting('hideByTypeIncorrectJunction', $(this).is(':checked'));
                            (async () => {
                                try {
                                    await handleUrLayer('settingsToggle');
                                } catch(error) {
                                    logWarning(error);
                                }
                            })();
                        }).prop('checked', _settings.hideByTypeIncorrectJunction),
                        $('<label>', {for:'_cbHideByTypeIncorrectJunction', title:I18n.t('urce.prefs.HideByTypeIncorrectJunctionTitle'), class:'URCE-label'}).text(I18n.t('urce.urTypes.IncorrectJunction')),
                        $('<br>'),
                        $('<input>', {type:'checkbox', id:'_cbHideByTypeIncorrectRoute', urceprefs:'filtering'}).change(function() {
                            changeSetting('hideByTypeIncorrectRoute', $(this).is(':checked'));
                            (async () => {
                                try {
                                    await handleUrLayer('settingsToggle');
                                } catch(error) {
                                    logWarning(error);
                                }
                            })();
                        }).prop('checked', _settings.hideByTypeIncorrectRoute),
                        $('<label>', {for:'_cbHideByTypeIncorrectRoute', title:I18n.t('urce.prefs.HideByTypeIncorrectRouteTitle'), class:'URCE-label'}).text(I18n.t('urce.urTypes.IncorrectRoute')),
                        $('<br>'),
                        $('<input>', {type:'checkbox', id:'_cbHideByTypeIncorrectTurn', urceprefs:'filtering'}).change(function() {
                            changeSetting('hideByTypeIncorrectTurn', $(this).is(':checked'));
                            (async () => {
                                try {
                                    await handleUrLayer('settingsToggle');
                                } catch(error) {
                                    logWarning(error);
                                }
                            })();
                        }).prop('checked', _settings.hideByTypeIncorrectTurn),
                        $('<label>', {for:'_cbHideByTypeIncorrectTurn', title:I18n.t('urce.prefs.HideByTypeIncorrectTurnTitle'), class:'URCE-label'}).text(I18n.t('urce.urTypes.IncorrectTurn')),
                        $('<br>'),
                        $('<input>', {type:'checkbox', id:'_cbHideByTypeMissingBridgeOverpass', urceprefs:'filtering'}).change(function() {
                            changeSetting('hideByTypeMissingBridgeOverpass', $(this).is(':checked'));
                            (async () => {
                                try {
                                    await handleUrLayer('settingsToggle');
                                } catch(error) {
                                    logWarning(error);
                                }
                            })();
                        }).prop('checked', _settings.hideByTypeMissingBridgeOverpass),
                        $('<label>', {for:'_cbHideByTypeMissingBridgeOverpass', title:I18n.t('urce.prefs.HideByTypeMissingBridgeOverpassTitle'), class:'URCE-label'}).text(I18n.t('urce.urTypes.MissingBridgeOverpass')),
                        $('<br>'),
                        $('<input>', {type:'checkbox', id:'_cbHideByTypeMissingExit', urceprefs:'filtering'}).change(function() {
                            changeSetting('hideByTypeMissingExit', $(this).is(':checked'));
                            (async () => {
                                try {
                                    await handleUrLayer('settingsToggle');
                                } catch(error) {
                                    logWarning(error);
                                }
                            })();
                        }).prop('checked', _settings.hideByTypeMissingExit),
                        $('<label>', {for:'_cbHideByTypeMissingExit', title:I18n.t('urce.prefs.HideByTypeMissingExitTitle'), class:'URCE-label'}).text(I18n.t('urce.urTypes.MissingExit')),
                        $('<br>'),
                        $('<input>', {type:'checkbox', id:'_cbHideByTypeMissingLandmark', urceprefs:'filtering'}).change(function() {
                            changeSetting('hideByTypeMissingLandmark', $(this).is(':checked'));
                            (async () => {
                                try {
                                    await handleUrLayer('settingsToggle');
                                } catch(error) {
                                    logWarning(error);
                                }
                            })();
                        }).prop('checked', _settings.hideByTypeMissingLandmark),
                        $('<label>', {for:'_cbHideByTypeMissingLandmark', title:I18n.t('urce.prefs.HideByTypeMissingLandmarkTitle'), class:'URCE-label'}).text(I18n.t('urce.urTypes.MissingLandmark')),
                        $('<br>'),
                        $('<input>', {type:'checkbox', id:'_cbHideByTypeMissingOrInvalidSpeedLimit', urceprefs:'filtering'}).change(function() {
                            changeSetting('hideByTypeMissingOrInvalidSpeedLimit', $(this).is(':checked'));
                            (async () => {
                                try {
                                    await handleUrLayer('settingsToggle');
                                } catch(error) {
                                    logWarning(error);
                                }
                            })();
                        }).prop('checked', _settings.hideByTypeMissingOrInvalidSpeedLimit),
                        $('<label>', {for:'_cbHideByTypeMissingOrInvalidSpeedLimit', title:I18n.t('urce.prefs.HideByTypeMissingOrInvalidSpeedLimitTitle'), class:'URCE-label'}).text(I18n.t('urce.urTypes.MissingOrInvalidSpeedLimit')),
                        $('<br>'),
                        $('<input>', {type:'checkbox', id:'_cbHideByTypeMissingRoad', urceprefs:'filtering'}).change(function() {
                            changeSetting('hideByTypeMissingRoad', $(this).is(':checked'));
                            (async () => {
                                try {
                                    await handleUrLayer('settingsToggle');
                                } catch(error) {
                                    logWarning(error);
                                }
                            })();
                        }).prop('checked', _settings.hideByTypeMissingRoad),
                        $('<label>', {for:'_cbHideByTypeMissingRoad', title:I18n.t('urce.prefs.HideByTypeMissingRoadTitle'), class:'URCE-label'}).text(I18n.t('urce.urTypes.MissingRoad')),
                        $('<br>'),
                        $('<input>', {type:'checkbox', id:'_cbHideByTypeMissingRoundabout', urceprefs:'filtering'}).change(function() {
                            changeSetting('hideByTypeMissingRoundabout', $(this).is(':checked'));
                            (async () => {
                                try {
                                    await handleUrLayer('settingsToggle');
                                } catch(error) {
                                    logWarning(error);
                                }
                            })();
                        }).prop('checked', _settings.hideByTypeMissingRoundabout),
                        $('<label>', {for:'_cbHideByTypeMissingRoundabout', title:I18n.t('urce.prefs.HideByTypeMissingRoundaboutTitle'), class:'URCE-label'}).text(I18n.t('urce.urTypes.MissingRoundabout')),
                        $('<br>'),
                        $('<input>', {type:'checkbox', id:'_cbHideByTypeTurnNotAllowed', urceprefs:'filtering'}).change(function() {
                            changeSetting('hideByTypeTurnNotAllowed', $(this).is(':checked'));
                            (async () => {
                                try {
                                    await handleUrLayer('settingsToggle');
                                } catch(error) {
                                    logWarning(error);
                                }
                            })();
                        }).prop('checked', _settings.hideByTypeTurnNotAllowed),
                        $('<label>', {for:'_cbHideByTypeTurnNotAllowed', title:I18n.t('urce.prefs.HideByTypeTurnNotAllowedTitle'), class:'URCE-label'}).text(I18n.t('urce.urTypes.TurnNotAllowed')),
                        $('<br>'),
                        $('<input>', {type:'checkbox', id:'_cbHideByTypeUndefined', urceprefs:'filtering'}).change(function() {
                            changeSetting('hideByTypeUndefined', $(this).is(':checked'));
                            (async () => {
                                try {
                                    await handleUrLayer('settingsToggle');
                                } catch(error) {
                                    logWarning(error);
                                }
                            })();
                        }).prop('checked', _settings.hideByTypeUndefined),
                        $('<label>', {for:'_cbHideByTypeUndefined', title:I18n.t('urce.prefs.HideByTypeUndefinedTitle'), class:'URCE-label'}).text(I18n.t('urce.urTypes.Undefined')),
                        $('<br>'),
                        $('<input>', {type:'checkbox', id:'_cbHideByTypeWazeAutomatic', urceprefs:'filtering'}).change(function() {
                            changeSetting('hideByTypeWazeAutomatic', $(this).is(':checked'));
                            (async () => {
                                try {
                                    await handleUrLayer('settingsToggle');
                                } catch(error) {
                                    logWarning(error);
                                }
                            })();
                        }).prop('checked', _settings.hideByTypeWazeAutomatic),
                        $('<label>', {for:'_cbHideByTypeWazeAutomatic', title:I18n.t('urce.prefs.HideByTypeWazeAutomaticTitle'), class:'URCE-label'}).text(I18n.t('urce.urTypes.WazeAutomatic')),
                        $('<br>'),
                        $('<input>', {type:'checkbox', id:'_cbHideByTypeWrongDrivingDirection', urceprefs:'filtering'}).change(function() {
                            changeSetting('hideByTypeWrongDrivingDirection', $(this).is(':checked'));
                            (async () => {
                                try {
                                    await handleUrLayer('settingsToggle');
                                } catch(error) {
                                    logWarning(error);
                                }
                            })();
                        }).prop('checked', _settings.hideByTypeWrongDrivingDirection),
                        $('<label>', {for:'_cbHideByTypeWrongDrivingDirection', title:I18n.t('urce.prefs.HideByTypeWrongDrivingDirectionTitle'), class:'URCE-label'}).text(I18n.t('urce.urTypes.WrongDrivingDirection')),
                        $('<br>')
                    ),
                    // -- Hide by tagged
                    $('<div>').append(
                        $('<div>', {class:'URCE-subHeading'}).text(I18n.t('urce.prefs.HideByTagged') + ':').append('<br>')
                    ).append(
                        $('<input>', {type:'checkbox', id:'_cbHideByTaggedBog', urceprefs:'filtering'}).change(function() {
                            changeSetting('hideByTaggedBog', $(this).is(':checked'));
                            (async () => {
                                try {
                                    await handleUrLayer('settingsToggle');
                                } catch(error) {
                                    logWarning(error);
                                }
                            })();
                        }).prop('checked', _settings.hideByTaggedBog),
                        $('<label>', {for:'_cbHideByTaggedBog', title:I18n.t('urce.prefs.HideByTaggedBogTitle'), class:'URCE-label'}).text(I18n.t('urce.tags.Bog')),
                        $('<br>'),
                        $('<input>', {type:'checkbox', id:'_cbHideByTaggedClosure', urceprefs:'filtering'}).change(function() {
                            changeSetting('hideByTaggedClosure', $(this).is(':checked'));
                            (async () => {
                                try {
                                    await handleUrLayer('settingsToggle');
                                } catch(error) {
                                    logWarning(error);
                                }
                            })();
                        }).prop('checked', _settings.hideByTaggedClosure),
                        $('<label>', {for:'_cbHideByTaggedClosure', title:I18n.t('urce.prefs.HideByTaggedClosureTitle'), class:'URCE-label'}).text(I18n.t('urce.tags.Closure')),
                        $('<br>'),
                        $('<input>', {type:'checkbox', id:'_cbHideByTaggedConstruction', urceprefs:'filtering'}).change(function() {
                            changeSetting('hideByTaggedConstruction', $(this).is(':checked'));
                            (async () => {
                                try {
                                    await handleUrLayer('settingsToggle');
                                } catch(error) {
                                    logWarning(error);
                                }
                            })();
                        }).prop('checked', _settings.hideByTaggedConstruction),
                        $('<label>', {for:'_cbHideByTaggedConstruction', title:I18n.t('urce.prefs.HideByTaggedConstructionTitle'), class:'URCE-label'}).text(I18n.t('urce.tags.Construction')),
                        $('<br>'),
                        $('<input>', {type:'checkbox', id:'_cbHideByTaggedDifficult', urceprefs:'filtering'}).change(function() {
                            changeSetting('hideByTaggedDifficult', $(this).is(':checked'));
                            (async () => {
                                try {
                                    await handleUrLayer('settingsToggle');
                                } catch(error) {
                                    logWarning(error);
                                }
                            })();
                        }).prop('checked', _settings.hideByTaggedDifficult),
                        $('<label>', {for:'_cbHideByTaggedDifficult', title:I18n.t('urce.prefs.HideByTaggedDifficultTitle'), class:'URCE-label'}).text(I18n.t('urce.tags.Difficult')),
                        $('<br>'),
                        $('<input>', {type:'checkbox', id:'_cbHideByTaggedEvent', urceprefs:'filtering'}).change(function() {
                            changeSetting('hideByTaggedEvent', $(this).is(':checked'));
                            (async () => {
                                try {
                                    await handleUrLayer('settingsToggle');
                                } catch(error) {
                                    logWarning(error);
                                }
                            })();
                        }).prop('checked', _settings.hideByTaggedEvent),
                        $('<label>', {for:'_cbHideByTaggedEvent', title:I18n.t('urce.prefs.HideByTaggedEventTitle'), class:'URCE-label'}).text(I18n.t('urce.tags.Event')),
                        $('<br>'),
                        $('<input>', {type:'checkbox', id:'_cbHideByTaggedNote', urceprefs:'filtering'}).change(function() {
                            changeSetting('hideByTaggedNote', $(this).is(':checked'));
                            (async () => {
                                try {
                                    await handleUrLayer('settingsToggle');
                                } catch(error) {
                                    logWarning(error);
                                }
                            })();
                        }).prop('checked', _settings.hideByTaggedNote),
                        $('<label>', {for:'_cbHideByTaggedNote', title:I18n.t('urce.prefs.HideByTaggedNoteTitle'), class:'URCE-label'}).text(I18n.t('urce.tags.Note')),
                        $('<br>'),
                        $('<input>', {type:'checkbox', id:'_cbHideByTaggedRoadworks', urceprefs:'filtering'}).change(function() {
                            changeSetting('hideByTaggedRoadworks', $(this).is(':checked'));
                            (async () => {
                                try {
                                    await handleUrLayer('settingsToggle');
                                } catch(error) {
                                    logWarning(error);
                                }
                            })();
                        }).prop('checked', _settings.hideByTaggedRoadworks),
                        $('<label>', {for:'_cbHideByTaggedRoadworks', title:I18n.t('urce.prefs.HideByTaggedRoadworksTitle'), class:'URCE-label'}).text(I18n.t('urce.tags.Roadworks')),
                        $('<br>'),
                        $('<input>', {type:'checkbox', id:'_cbHideByTaggedWslm', urceprefs:'filtering'}).change(function() {
                            changeSetting('hideByTaggedWslm', $(this).is(':checked'));
                            (async () => {
                                try {
                                    await handleUrLayer('settingsToggle');
                                } catch(error) {
                                    logWarning(error);
                                }
                            })();
                        }).prop('checked', _settings.hideByTaggedWslm),
                        $('<label>', {for:'_cbHideByTaggedWslm', title:I18n.t('urce.prefs.HideByTaggedWslmTitle'), class:'URCE-label'}).text(I18n.t('urce.tags.Wslm')),
                        $('<br>')
                    ),
                    // -- Hide by age of submission
                    $('<div>').append(
                        $('<div>', {class:'URCE-subHeading'}).text(I18n.t('urce.prefs.HideByAgeOfSubmission') + ':').append('<br>')
                    ).append(
                        $('<div>').append(
                            $('<input>', {type:'checkbox', id:'_cbHideByAgeOfSubmissionLessThan', urceprefs:'filtering'}).change(function() {
                                changeSetting('hideByAgeOfSubmissionLessThan', $(this).is(':checked'));
                                (async () => {
                                    try {
                                        await handleUrLayer('settingsToggle');
                                    } catch(error) {
                                        logWarning(error);
                                    }
                                })();
                            }).prop('checked', _settings.hideByAgeOfSubmissionLessThan),
                            $('<label>', {for:'_cbHideByAgeOfSubmissionLessThan', class:'URCE-label'}).text(I18n.t('urce.common.LessThan')),
                            $('<div>', {class:'URCE-divDaysInline'}).append(
                                $('<input>', {type:'number', id:'_numHideByAgeOfSubmissionLessThanDaysOld', class:'URCE-daysInput', urceprefs:'filtering', min:'0', max:'9999', step:'1', value:_settings.hideByAgeOfSubmissionLessThanDaysOld}).on('change', function() {
                                    let val = Math.abs(parseInt(this.value, 10) || 0);
                                    val = Math.min(9999,Math.max(0,parseInt(val)));
                                    if (val !== this.value) {
                                        changeSetting('hideByAgeOfSubmissionLessThanDaysOld', val);
                                        this.value = val;
                                        (async () => {
                                            try {
                                                await handleUrLayer('settingsToggle');
                                            } catch(error) {
                                                logWarning(error);
                                            }
                                        })();
                                    }
                                }),
                                $('<div>', {class:'URCE-divDaysInline'}).append(I18n.t('urce.common.DaysOld'))
                            ),
                        ),
                        $('<div>').append(
                            $('<input>', {type:'checkbox', id:'_cbHideByAgeOfSubmissionMoreThan', urceprefs:'filtering'}).change(function() {
                                changeSetting('hideByAgeOfSubmissionMoreThan', $(this).is(':checked'));
                                (async () => {
                                    try {
                                        await handleUrLayer('settingsToggle');
                                    } catch(error) {
                                        logWarning(error);
                                    }
                                })();
                            }).prop('checked', _settings.hideByAgeOfSubmissionMoreThan),
                            $('<label>', {for:'_cbHideByAgeOfSubmissionMoreThan', class:'URCE-label'}).text(I18n.t('urce.common.MoreThan')),
                            $('<div>', {class:'URCE-divDaysInline'}).append(
                                $('<input>', {type:'number', id:'_numHideByAgeOfSubmissionMoreThanDaysOld', class:'URCE-daysInput', urceprefs:'filtering', min:'0', max:'9999', step:'1', value:_settings.hideByAgeOfSubmissionMoreThanDaysOld}).on('change', function() {
                                    let val = Math.abs(parseInt(this.value, 10) || 0);
                                    val = Math.min(9999,Math.max(0,parseInt(val)));
                                    if (val !== this.value) {
                                        changeSetting('hideByAgeOfSubmissionMoreThanDaysOld', val);
                                        this.value = val;
                                        (async () => {
                                            try {
                                                await handleUrLayer('settingsToggle');
                                            } catch(error) {
                                                logWarning(error);
                                            }
                                        })();
                                    }
                                }),
                                $('<div>', {class:'URCE-divDaysInline'}).append(I18n.t('urce.common.DaysOld'))
                            )
                        )
                    ),
                    // -- Hide by description, comments, following
                    $('<div>').append(
                        $('<div>', {class:'URCE-subHeading'}).text(I18n.t('urce.prefs.DescriptionCommentsFollowing') + ':').append('<br>')
                    ).append(
                        // -- -- Following / not following
                        $('<div>', {class:'URCE-textFirst'}).append(
                            $('<div>').text(I18n.t('urce.common.Following') + ': ').append(
                                $('<div>', {style:'display:inline;'}).append(
                                    $('<input>', {type:'checkbox', id:'_cbHideFollowing', urceprefs:'filtering'}).change(function() {
                                        _settings.hideFollowing = $(this).is(':checked');
                                        if ($(this).is(':checked')) {
                                            if (isChecked('_cbHideNotFollowing')) {
                                                _settings.hideNotFollowing = false;
                                                $('#_cbHideNotFollowing').prop('checked', false);
                                            }
                                        }
                                        saveSettingsToStorage();
                                        (async () => {
                                            try {
                                                await handleUrLayer('settingsToggle');
                                            } catch(error) {
                                                logWarning(error);
                                            }
                                        })();
                                    }).prop('checked', _settings.hideFollowing),
                                    $('<label>', {for:'_cbHideFollowing', title:I18n.t('urce.prefs.HideFollowingTitle')}).text(I18n.t('urce.common.Following')),
                                    $('<input>', {type:'checkbox', id:'_cbHideNotFollowing', urceprefs:'filtering'}).change(function() {
                                        _settings.hideNotFollowing = $(this).is(':checked');
                                        if ($(this).is(':checked')) {
                                            if (isChecked('_cbHideFollowing')) {
                                                _settings.hideFollowing = false;
                                                $('#_cbHideFollowing').prop('checked', false);
                                            }
                                        }
                                        saveSettingsToStorage();
                                        (async () => {
                                            try {
                                                await handleUrLayer('settingsToggle');
                                            } catch(error) {
                                                logWarning(error);
                                            }
                                        })();
                                    }).prop('checked', _settings.hideNotFollowing),
                                    $('<label>', {for:'_cbHideNotFollowing', title:I18n.t('urce.prefs.HideNotFollowingTitle')}).text(I18n.t('urce.common.NotFollowing')),
                                )
                            ),
                            // -- -- With / without description
                            $('<div>').text(I18n.t('urce.common.Description') + ': ').append(
                                $('<div>', {style:'display:inline;'}).append(
                                    $('<input>', {type:'checkbox', id:'_cbHideWithDescription', urceprefs:'filtering'}).change(function() {
                                        _settings.hideWithDescription = $(this).is(':checked');
                                        if ($(this).is(':checked')) {
                                            if (isChecked('_cbHideWithoutDescription')) {
                                                _settings.hideWithoutDescription = false;
                                                $('#_cbHideWithoutDescription').prop('checked', false);
                                            }
                                        }
                                        saveSettingsToStorage();
                                        (async () => {
                                            try {
                                                await handleUrLayer('settingsToggle');
                                            } catch(error) {
                                                logWarning(error);
                                            }
                                        })();
                                    }).prop('checked', _settings.hideWithDescription),
                                    $('<label>', {for:'_cbHideWithDescription', title:I18n.t('urce.prefs.HideWithDescriptionTitle')}).text(I18n.t('urce.common.With')),
                                    $('<input>', {type:'checkbox', id:'_cbHideWithoutDescription', urceprefs:'filtering'}).change(function() {
                                        _settings.hideWithoutDescription = $(this).is(':checked');
                                        if ($(this).is(':checked')) {
                                            if (isChecked('_cbHideWithDescription')) {
                                                _settings.hideWithDescription = false;
                                                $('#_cbHideWithDescription').prop('checked', false);
                                            }
                                        }
                                        saveSettingsToStorage();
                                        (async () => {
                                            try {
                                                await handleUrLayer('settingsToggle');
                                            } catch(error) {
                                                logWarning(error);
                                            }
                                        })();
                                    }).prop('checked', _settings.hideWithoutDescription),
                                    $('<label>', {for:'_cbHideWithoutDescription', title:I18n.t('urce.prefs.HideWithoutDescriptionTitle')}).text(I18n.t('urce.common.Without')),
                                )
                            ),
                            // -- -- With / without comments from me
                            $('<div>').text(I18n.t('urce.prefs.HideCommentsFromMe') + ': ').append(
                                $('<div>', {style:'display:inline;'}).append(
                                    $('<input>', {type:'checkbox', id:'_cbHideWithCommentsFromMe', urceprefs:'filtering'}).change(function() {
                                        _settings.hideWithCommentsFromMe = $(this).is(':checked');
                                        if ($(this).is(':checked')) {
                                            if (isChecked('_cbHideWithoutCommentsFromMe')) {
                                                _settings.hideWithoutCommentsFromMe = false;
                                                $('#_cbHideWithoutCommentsFromMe').prop('checked', false);
                                            }
                                        }
                                        saveSettingsToStorage();
                                        (async () => {
                                            try {
                                                await handleUrLayer('settingsToggle');
                                            } catch(error) {
                                                logWarning(error);
                                            }
                                        })();
                                    }).prop('checked', _settings.hideWithCommentsFromMe),
                                    $('<label>', {for:'_cbHideWithCommentsFromMe', title:I18n.t('urce.prefs.HideWithCommentsFromMeTitle')}).text(I18n.t('urce.common.With')),
                                    $('<input>', {type:'checkbox', id:'_cbHideWithoutCommentsFromMe', urceprefs:'filtering'}).change(function() {
                                        _settings.hideWithoutCommentsFromMe = $(this).is(':checked');
                                        if ($(this).is(':checked')) {
                                            if (isChecked('_cbHideWithCommentsFromMe')) {
                                                _settings.hideWithCommentsFromMe = false;
                                                $('#_cbHideWithCommentsFromMe').prop('checked', false);
                                            }
                                        }
                                        saveSettingsToStorage();
                                        (async () => {
                                            try {
                                                await handleUrLayer('settingsToggle');
                                            } catch(error) {
                                                logWarning(error);
                                            }
                                        })();
                                    }).prop('checked', _settings.hideWithoutCommentsFromMe),
                                    $('<label>', {for:'_cbHideWithoutCommentsFromMe', title:I18n.t('urce.prefs.HideWithoutCommentsFromMeTitle')}).text(I18n.t('urce.common.Without')),
                                )
                            ),
                            // -- -- First comment by me yes / no
                            $('<div>').text(I18n.t('urce.prefs.HideFirstCommentByMe') + ': ').append(
                                $('<div>', {style:'display:inline;'}).append(
                                    $('<input>', {type:'checkbox', id:'_cbHideFirstCommentByMe', urceprefs:'filtering'}).change(function() {
                                        _settings.hideFirstCommentByMe = $(this).is(':checked');
                                        if ($(this).is(':checked')) {
                                            if (isChecked('_cbHideFirstCommentNotByMe')) {
                                                _settings.hideFirstCommentNotByMe = false;
                                                $('#_cbHideFirstCommentNotByMe').prop('checked', false);
                                            }
                                        }
                                        saveSettingsToStorage();
                                        (async () => {
                                            try {
                                                await handleUrLayer('settingsToggle');
                                            } catch(error) {
                                                logWarning(error);
                                            }
                                        })();
                                    }).prop('checked', _settings.hideFirstCommentByMe),
                                    $('<label>', {for:'_cbHideFirstCommentByMe', title:I18n.t('urce.prefs.HideFirstCommentByMeTitle')}).text(I18n.t('urce.common.Yes')),
                                    $('<input>', {type:'checkbox', id:'_cbHideFirstCommentNotByMe', urceprefs:'filtering'}).change(function() {
                                        _settings.hideFirstCommentNotByMe = $(this).is(':checked');
                                        if ($(this).is(':checked')) {
                                            if (isChecked('_cbHideFirstCommentByMe')) {
                                                _settings.hideFirstCommentByMe = false;
                                                $('#_cbHideFirstCommentByMe').prop('checked', false);
                                            }
                                        }
                                        saveSettingsToStorage();
                                        (async () => {
                                            try {
                                                await handleUrLayer('settingsToggle');
                                            } catch(error) {
                                                logWarning(error);
                                            }
                                        })();
                                    }).prop('checked', _settings.hideFirstCommentNotByMe),
                                    $('<label>', {for:'_cbHideFirstCommentNotByMe', title:I18n.t('urce.prefs.HideFirstCommentNotByMeTitle')}).text(I18n.t('urce.common.No')),
                                )
                            ),
                            // -- -- Last comment by me yes / no
                            $('<div>').text(I18n.t('urce.prefs.HideLastCommentByMe') + ': ').append(
                                $('<div>', {style:'display:inline;'}).append(
                                    $('<input>', {type:'checkbox', id:'_cbHideLastCommentByMe', urceprefs:'filtering'}).change(function() {
                                        _settings.hideLastCommentByMe = $(this).is(':checked');
                                        if ($(this).is(':checked')) {
                                            if (isChecked('_cbHideLastCommentNotByMe')) {
                                                _settings.hideLastCommentNotByMe = false;
                                                $('#_cbHideLastCommentNotByMe').prop('checked', false);
                                            }
                                        }
                                        saveSettingsToStorage();
                                        (async () => {
                                            try {
                                                await handleUrLayer('settingsToggle');
                                            } catch(error) {
                                                logWarning(error);
                                            }
                                        })();
                                    }).prop('checked', _settings.hideLastCommentByMe),
                                    $('<label>', {for:'_cbHideLastCommentByMe', title:I18n.t('urce.prefs.HideLastCommentByMeTitle')}).text(I18n.t('urce.common.Yes')),
                                    $('<input>', {type:'checkbox', id:'_cbHideLastCommentNotByMe', urceprefs:'filtering'}).change(function() {
                                        _settings.hideLastCommentNotByMe = $(this).is(':checked');
                                        if ($(this).is(':checked')) {
                                            if (isChecked('_cbHideLastCommentByMe')) {
                                                _settings.hideLastCommentByMe = false;
                                                $('#_cbHideLastCommentByMe').prop('checked', false);
                                            }
                                        }
                                        saveSettingsToStorage();
                                        (async () => {
                                            try {
                                                await handleUrLayer('settingsToggle');
                                            } catch(error) {
                                                logWarning(error);
                                            }
                                        })();
                                    }).prop('checked', _settings.hideLastCommentNotByMe),
                                    $('<label>', {for:'_cbHideLastCommentNotByMe', title:I18n.t('urce.prefs.HideLastCommentNotByMeTitle')}).text(I18n.t('urce.common.No')),
                                )
                            ),
                            // -- -- Last comment by reporter yes / no
                            $('<div>').text(I18n.t('urce.prefs.HideLastCommentByReporter') + ': ').append(
                                $('<div>', {style:'display:inline;'}).append(
                                    $('<input>', {type:'checkbox', id:'_cbHideLastCommentByReporter', urceprefs:'filtering'}).change(function() {
                                        _settings.hideLastCommentByReporter = $(this).is(':checked');
                                        if ($(this).is(':checked')) {
                                            if (isChecked('_cbHideLastCommentNotByReporter')) {
                                                _settings.hideLastCommentNotByReporter = false;
                                                $('#_cbHideLastCommentNotByReporter').prop('checked', false);
                                            }
                                        }
                                        saveSettingsToStorage();
                                        (async () => {
                                            try {
                                                await handleUrLayer('settingsToggle');
                                            } catch(error) {
                                                logWarning(error);
                                            }
                                        })();
                                    }).prop('checked', _settings.hideLastCommentByReporter),
                                    $('<label>', {for:'_cbHideLastCommentByReporter', title:I18n.t('urce.prefs.HideLastCommentByReporterTitle')}).text(I18n.t('urce.common.Yes')),
                                    $('<input>', {type:'checkbox', id:'_cbHideLastCommentNotByReporter', urceprefs:'filtering'}).change(function() {
                                        _settings.hideLastCommentNotByReporter = $(this).is(':checked');
                                        if ($(this).is(':checked')) {
                                            if (isChecked('_cbHideLastCommentByReporter')) {
                                                _settings.hideLastCommentByReporter = false;
                                                $('#_cbHideLastCommentByReporter').prop('checked', false);
                                            }
                                        }
                                        saveSettingsToStorage();
                                        (async () => {
                                            try {
                                                await handleUrLayer('settingsToggle');
                                            } catch(error) {
                                                logWarning(error);
                                            }
                                        })();
                                    }).prop('checked', _settings.hideLastCommentNotByReporter),
                                    $('<label>', {for:'_cbHideLastCommentNotByReporter', title:I18n.t('urce.prefs.HideLastCommentNotByReporterTitle')}).text(I18n.t('urce.common.No')),
                                )
                            )
                        ),
                        // -- -- Less than / more than XX comments
                        $('<div>').append(
                            $('<input>', {type:'checkbox', id:'_cbHideByCommentCountLessThan', urceprefs:'filtering'}).change(function() {
                                changeSetting('hideByCommentCountLessThan', $(this).is(':checked'));
                                (async () => {
                                    try {
                                        await handleUrLayer('settingsToggle');
                                    } catch(error) {
                                        logWarning(error);
                                    }
                                })();
                            }).prop('checked', _settings.hideByCommentCountLessThan),
                            $('<label>', {for:'_cbHideByCommentCountLessThan', class:'URCE-label'}).text(I18n.t('urce.common.LessThan')),
                            $('<div>', {class:'URCE-divDaysInline'}).append(
                                $('<input>', {type:'number', id:'_numHideByCommentCountLessThanNumber', style:'width:36px; height:20px;', urceprefs:'filtering', min:'0', max:'9999', step:'1', value:_settings.hideByCommentCountLessThanNumber}).on('change', function() {
                                    let val = Math.abs(parseInt(this.value, 10) || 0);
                                    val = Math.min(9999,Math.max(0,parseInt(val)));
                                    if (val !== this.value) {
                                        changeSetting('hideByCommentCountLessThanNumber', val);
                                        this.value = val;
                                        (async () => {
                                            try {
                                                await handleUrLayer('settingsToggle');
                                            } catch(error) {
                                                logWarning(error);
                                            }
                                        })();
                                    }
                                }),
                                $('<div>', {class:'URCE-divDaysInline'}).append(I18n.t('urce.tabs.Comments').toLowerCase())
                            ),
                        ),
                        $('<div>').append(
                            $('<input>', {type:'checkbox', id:'_cbHideByCommentCountMoreThan', urceprefs:'filtering'}).change(function() {
                                changeSetting('hideByCommentCountMoreThan', $(this).is(':checked'));
                                (async () => {
                                    try {
                                        await handleUrLayer('settingsToggle');
                                    } catch(error) {
                                        logWarning(error);
                                    }
                                })();
                            }).prop('checked', _settings.hideByCommentCountMoreThan),
                            $('<label>', {for:'_cbHideByCommentCountMoreThan', class:'URCE-label'}).text(I18n.t('urce.common.MoreThan')),
                            $('<div>', {class:'URCE-divDaysInline'}).append(
                                $('<input>', {type:'number', id:'_numHideByCommentCountMoreThanNumber', style:'width:36px; height:20px;', urceprefs:'filtering', min:'0', max:'9999', step:'1', value:_settings.hideByCommentCountMoreThanNumber}).on('change', function() {
                                    let val = Math.abs(parseInt(this.value, 10) || 0);
                                    val = Math.min(9999,Math.max(0,parseInt(val)));
                                    if (val !== this.value) {
                                        changeSetting('hideByCommentCountMoreThanNumber', val);
                                        this.value = val;
                                        (async () => {
                                            try {
                                                await handleUrLayer('settingsToggle');
                                            } catch(error) {
                                                logWarning(error);
                                            }
                                        })();
                                    }
                                }),
                                $('<div>', {class:'URCE-divDaysInline'}).append(I18n.t('urce.tabs.Comments').toLowerCase())
                            ),
                        ),
                        // -- -- Age of first comment less than / more than XX days old
                        $('<div>').append(
                            $('<input>', {type:'checkbox', id:'_cbHideByAgeOfFirstCommentLessThan', urceprefs:'filtering'}).change(function() {
                                changeSetting('hideByAgeOfFirstCommentLessThan', $(this).is(':checked'));
                                (async () => {
                                    try {
                                        await handleUrLayer('settingsToggle');
                                    } catch(error) {
                                        logWarning(error);
                                    }
                                })();
                            }).prop('checked', _settings.hideByAgeOfFirstCommentLessThan),
                            $('<label>', {for:'_cbHideByAgeOfFirstCommentLessThan', class:'URCE-label'}).text(I18n.t('urce.prefs.HideByAgeOfFirstCommentLessThan')),
                            $('<div>', {class:'URCE-divDaysInline'}).append(
                                $('<input>', {type:'number', id:'_numHideByAgeOfFirstCommentLessThanDaysOld', style:'width:36px; height:20px;', urceprefs:'filtering', min:'0', max:'9999', step:'1', value:_settings.hideByAgeOfFirstCommentLessThanDaysOld}).on('change', function() {
                                    let val = Math.abs(parseInt(this.value, 10) || 0);
                                    val = Math.min(9999,Math.max(0,parseInt(val)));
                                    if (val !== this.value) {
                                        changeSetting('hideByAgeOfFirstCommentLessThanDaysOld', val);
                                        this.value = val;
                                        (async () => {
                                            try {
                                                await handleUrLayer('settingsToggle');
                                            } catch(error) {
                                                logWarning(error);
                                            }
                                        })();
                                    }
                                }),
                                $('<div>', {class:'URCE-divDaysInline'}).append(I18n.t('urce.common.DaysOld'))
                            ),
                        ),
                        $('<div>').append(
                            $('<input>', {type:'checkbox', id:'_cbHideByAgeOfFirstCommentMoreThan', urceprefs:'filtering'}).change(function() {
                                changeSetting('hideByAgeOfFirstCommentMoreThan', $(this).is(':checked'));
                                (async () => {
                                    try {
                                        await handleUrLayer('settingsToggle');
                                    } catch(error) {
                                        logWarning(error);
                                    }
                                })();
                            }).prop('checked', _settings.hideByAgeOfFirstCommentMoreThan),
                            $('<label>', {for:'_cbHideByAgeOfFirstCommentMoreThan', class:'URCE-label'}).text(I18n.t('urce.prefs.HideByAgeOfFirstCommentMoreThan')),
                            $('<div>', {class:'URCE-divDaysInline'}).append(
                                $('<input>', {type:'number', id:'_numHideByAgeOfFirstCommentMoreThanDaysAgo', style:'width:36px; height:20px;', urceprefs:'filtering', min:'0', max:'9999', step:'1', value:_settings.hideByAgeOfFirstCommentMoreThanDaysAgo}).on('change', function() {
                                    let val = Math.abs(parseInt(this.value, 10) || 0);
                                    val = Math.min(9999,Math.max(0,parseInt(val)));
                                    if (val !== this.value) {
                                        changeSetting('hideByAgeOfFirstCommentMoreThanDaysAgo', val);
                                        this.value = val;
                                        (async () => {
                                            try {
                                                await handleUrLayer('settingsToggle');
                                            } catch(error) {
                                                logWarning(error);
                                            }
                                        })();
                                    }
                                }),
                                $('<div>', {class:'URCE-divDaysInline'}).append(I18n.t('urce.common.DaysOld'))
                            ),
                        ),
                        // -- -- Age of last comment less than / more than XX days old
                        $('<div>').append(
                            $('<input>', {type:'checkbox', id:'_cbHideByAgeOfLastCommentLessThan', urceprefs:'filtering'}).change(function() {
                                changeSetting('hideByAgeOfLastCommentLessThan', $(this).is(':checked'));
                                (async () => {
                                    try {
                                        await handleUrLayer('settingsToggle');
                                    } catch(error) {
                                        logWarning(error);
                                    }
                                })();
                            }).prop('checked', _settings.hideByAgeOfLastCommentLessThan),
                            $('<label>', {for:'_cbHideByAgeOfLastCommentLessThan', class:'URCE-label'}).text(I18n.t('urce.prefs.HideByAgeOfLastCommentLessThan')),
                            $('<div>', {class:'URCE-divDaysInline'}).append(
                                $('<input>', {type:'number', id:'_numHideByAgeOfLastCommentLessThanDaysOld', class:'URCE-daysInput', urceprefs:'filtering', min:'0', max:'9999', step:'1', value:_settings.hideByAgeOfLastCommentLessThanDaysOld}).on('change', function() {
                                    let val = Math.abs(parseInt(this.value, 10) || 0);
                                    val = Math.min(9999,Math.max(0,parseInt(val)));
                                    if (val !== this.value) {
                                        changeSetting('hideByAgeOfLastCommentLessThanDaysOld', val);
                                        this.value = val;
                                        (async () => {
                                            try {
                                                await handleUrLayer('settingsToggle');
                                            } catch(error) {
                                                logWarning(error);
                                            }
                                        })();
                                    }
                                }),
                                $('<div>', {class:'URCE-divDaysInline'}).append(I18n.t('urce.common.DaysOld'))
                            ),
                        ),
                        $('<div>').append(
                            $('<input>', {type:'checkbox', id:'_cbHideByAgeOfLastCommentMoreThan', urceprefs:'filtering'}).change(function() {
                                changeSetting('hideByAgeOfLastCommentMoreThan', $(this).is(':checked'));
                                (async () => {
                                    try {
                                        await handleUrLayer('settingsToggle');
                                    } catch(error) {
                                        logWarning(error);
                                    }
                                })();
                            }).prop('checked', _settings.hideByAgeOfLastCommentMoreThan),
                            $('<label>', {for:'_cbHideByAgeOfLastCommentMoreThan', class:'URCE-label'}).text(I18n.t('urce.prefs.HideByAgeOfLastCommentMoreThan')),
                            $('<div>', {class:'URCE-divDaysInline'}).append(
                                $('<input>', {type:'number', id:'_numHideByAgeOfLastCommentMoreThanDaysAgo', style:'width:36px; height:20px;', urceprefs:'filtering', min:'0', max:'9999', step:'1', value:_settings.hideByAgeOfLastCommentMoreThanDaysAgo}).on('change', function() {
                                    let val = Math.abs(parseInt(this.value, 10) || 0);
                                    val = Math.min(9999,Math.max(0,parseInt(val)));
                                    if (val !== this.value) {
                                        changeSetting('hideByAgeOfLastCommentMoreThanDaysAgo', val);
                                        this.value = val;
                                        (async () => {
                                            try {
                                                await handleUrLayer('settingsToggle');
                                            } catch(error) {
                                                logWarning(error);
                                            }
                                        })();
                                    }
                                }),
                                $('<div>', {class:'URCE-divDaysInline'}).append(I18n.t('urce.common.DaysOld'))
                            )
                        )
                    )
                )
            ),
            // Common Preferences
            $('<fieldset>', {id:'urce-prefs-fieldset-common-prefs', class:`URCE-field${urStyle}`}).append(
                $('<legend>', {id:'urce-prefs-legend-common-prefs', class:`URCE-legend${urStyle}`}).append(
                    $('<i>', {class:'fa fa-fw fa-chevron-down URCE-chevron'}),
                    $('<span>', {class:'URCE-span'}).text(I18n.t('urce.prefs.CommonPrefs'))
                ).click(function() {
                    $($(this).children()[0]).toggleClass('fa fa-fw fa-chevron-down');
                    $($(this).children()[0]).toggleClass('fa fa-fw fa-chevron-right');
                    $($(this).siblings()[0]).toggleClass('collapse');
                }),
                $('<div>', {class:'URCE-controls URCE-textFirst'}).append(
                    $('<div>', {title:I18n.t('urce.prefs.ReminderDaysTitle'), class:'URCE-label', urceprefs:'common'}).append(I18n.t('urce.prefs.ReminderDays') + ': ').append(
                        $('<input>', {type:'number', id:'_numReminderDays', class:'URCE-daysInput', urceprefs:'common', min:'0', max:'13', step:'1', value:_settings.reminderDays, title:I18n.t('urce.prefs.ReminderDaysTitle')}).on('change', function() {
                            let numReminderDays = Math.abs(parseInt(this.value, 10) || 0);
                            if (numReminderDays >= _settings.closeDays) {
                                numReminderDays = (_settings.closeDays - 1) < 0 ? 0 : (_settings.closeDays - 1);
                            }
                            numReminderDays = Math.min(13,Math.max(0,parseInt(numReminderDays)));
                            if (numReminderDays !== this.value) {
                                changeSetting('reminderDays', numReminderDays);
                                this.value = numReminderDays;
                                (async () => {
                                    try {
                                        await handleUrLayer('settingsToggle');
                                    } catch(error) {
                                        logWarning(error);
                                    }
                                })();
                            }
                        })
                    ),
                    $('<div>', {title:I18n.t('urce.prefs.CloseDaysTitle'), class:'URCE-label', urceprefs:'common'}).append(I18n.t('urce.prefs.CloseDays') + ': ').append(
                        $('<input>', {type:'number', id:'_numCloseDays', class:'URCE-daysInput', urceprefs:'common', min:'1', max:'14', step:'1', value:_settings.closeDays, title:I18n.t('urce.prefs.CloseDaysTitle')}).on('change', function() {
                            let numCloseDays = Math.abs(parseInt(this.value, 10) || 1);
                            if (numCloseDays <= _settings.reminderDays) {
                                numCloseDays = (_settings.reminderDays + 1) > 14 ? 14 : (_settings.reminderDays + 1);
                            }
                            numCloseDays = Math.min(14,Math.max(1,parseInt(numCloseDays)));
                            if (numCloseDays !== this.value) {
                                changeSetting('closeDays', numCloseDays);
                                this.value = numCloseDays;
                                (async () => {
                                    try {
                                        await handleUrLayer('settingsToggle');
                                    } catch(error) {
                                        logWarning(error);
                                    }
                                })();
                            }
                        })
                    )
                )
            )
        );
        if (!isChecked('_cbEnableUrPillCounts')) {
            $('[urceprefs=marker]').prop('disabled', true);
        } else {
            $('[urceprefs=marker]').prop('disabled', false);
        }
        if (!isChecked('_cbEnableUrceUrFiltering')) {
            $('[urceprefs=filtering]').prop('disabled', true);
        } else {
            $('[urceprefs=filtering]').prop('disabled', false);
        }
    }

    function initTab() {
        logDebug('Firing initTab via callback.');
        initSettingsTab();
        initCommentsTab();
        $('a[href="#sidepanel-urc-e"]').prepend($('<img>', {class:'URCE-tabIcon', src:GM_info.script.icon}));
    }

    function initGui() {
        logDebug('Initializing GUI.');
        injectCss();
        $('body').append(
            $('<div>', {id:'urceAlertBox', class:'urceAlertBox'})
        );
        $('#urceAlertBox').append(
            $('<div>', {id:'urceAlertBoxHeader', class:'URCE-alertBox-header'})
        ).append(
            $('<div>', {id:'urceAlertBoxContent', class:'URCE-alertBox-content'})
        ).append(
            $('<div>', {id:'urceAlertBoxControls', class:'URCE-alertBox-controls'}).append(
                $('<span>', {id:'urceAlertTickBtn', class:'URCE-alertBox-controls-span-urceAlertTickBtn'}).append(
                    $('<i>', {class:'fa fa-check'})
                ).append(
                    $('<span>', {id:'urceAlertTickBtnCaption', class:'URCE-alertBox-controls-span-urceAlertTickBtnCaption'})
                )
            ).append(
                '&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;'
            ).append(
                $('<span>', {id:'urceAlertCrossBtn', class:'URCE-alertBox-controls-span-urceAlertCrossBtn'}).append(
                    $('<i>', {class:'fa fa-times'})
                ).append(
                    $('<span>', {id:'urceAlertCrossBtnCaption', class:'URCE-alertBox-controls-span-urceAlertCrossBtnCaption'})
                )
            )
        );
        let $content = $('<div>').append(
            $('<span>', {class:'URCE-spanTitle'}).text(I18n.t('urce.common.Title')),
            $('<span>', {class:'URCE-spanVersion'}).text(GM_info.script.version),
            '<ul class="nav nav-tabs">' +
            '<li class="active"><a data-toggle="tab" href="#panel-urce-comments" aria-expanded="true">' + I18n.t('urce.tabs.Comments') + '</a></li>' +
            '<li><a data-toggle="tab" href="#panel-urce-settings" aria-expanded="true">' + I18n.t('urce.tabs.Settings') + '</a></li>' +
            '</ul>',
            $('<div>', {class:'tab-content URCE-divTabs'}).append(
                $('<div>', {class:'tab-pane active', id:'panel-urce-comments'}),
                $('<div>', {class:'tab-pane', id:'panel-urce-settings'})
                )
            ).html();
        new WazeWrap.Interface.Tab('URC-E', $content, initTab, null);
        $('div#sidepanel-urc-e').width('290px');
        showScriptInfoAlert();
    }

    async function init() {
        log('Initializing.');
        _wmeUserId = W.loginManager.user.id;
        loadSettingsFromStorage();
        loadTranslations();
        initGui();
        window.addEventListener("beforeunload", function() {
            saveSettingsToStorage();
        }, false);
        log('Initialized.');
        setTimeout(saveSettingsToStorage, 5000);
        _urceInitialized = true;
        let buildCommentListResult = await buildCommentList();
        if (buildCommentListResult.error) {
            handleBuildCommentListError(buildCommentListResult.error);
        } else {
            initBackgroundTasks();
        }
        logDebug('Loaded in ' + Math.round(performance.now() - LOAD_BEGIN_TIME) + ' ms.');
    }

    function bootstrap(tries) {
        tries = tries || 1;
        if (W &&
            W.map &&
            W.model &&
            $ && WazeWrap.Ready) {
            log('Bootstrapping.');
            init();
        } else if (tries < 1000) {
            logDebug('Bootstrap failed. Retrying ' + tries + ' of 1000');
            setTimeout(function () { bootstrap(++tries); }, 200);
        } else {
            logError('Bootstrap timed out waiting for WME to become ready.');
        }
    }

    bootstrap();

    function loadTranslations() {
        logDebug('Loading translations.');
        setTranslations({
            en: {
                commentsTab: {
                    ZoomOutLink1: 'Zoom out 0 & close UR',
                    ZoomOutLink1Title: 'Zooms all the way out and closes the UR dialogue.',
                    ZoomOutLink2: 'Zoom out 2 & close UR',
                    ZoomOutLink2Title: 'Zooms out to level 2 and closes the UR dialogue.',
                    ZoomOutLink3: 'Zoom out 3 & close UR',
                    ZoomOutLink3Title: 'Zooms out to level 3 and closes the UR dialogue.'
                },
                common: {
                    Title: 'URComments-Enhanced',
                    DoubleClickTitle: 'Double click here to send this comment:',
                    Loading: 'Loading',
                    PleaseWait: 'Please wait',
                    ErrorHeader: 'URC-E Error',
                    ErrorGeneric: 'An error has occurred within URC-E. Please contact dBsooner via Discord or PM.',
                    Yes: 'Yes',
                    No: 'No',
                    All: 'All',
                    With: 'With',
                    Without: 'Without',
                    Following: 'Following',
                    NotFollowing: 'Not following',
                    LessThan: 'Less than',
                    MoreThan: 'More than',
                    DaysOld: 'days old',
                    List: 'List',
                    Style: 'Style',
                    Description: 'Description'
                },
                prefs: {
                    // Comment List
                    CommentList: 'Comment List',
                    CommentListTitle: 'Select the custom list you would like to use. CommentTeam is the default. If you would like your comment list built into this script or have suggestions on the CommentTeam list, please contact dBsooner on Discord or via PM.',
                    CommentListStyleTitle: 'Select the style you would like the URC-E panel to be displayed in. This only affects the look of the tab, no functionality is changed.',
                    StyleDefault: 'Default',
                    StyleUrStyle: 'UR Style',
                    // URC-E Preferences
                    UrcePrefs: 'URC-E Preferences',
                    AutoCenterOnUr: 'Auto center on UR',
                    AutoCenterOnUrTitle: 'Auto Center the map at the current map zoom to the selected UR when it has comments and the zoom is less than 3.',
                    AutoClickOpenSolvedNi: 'Auto click open, solved or not identified',
                    AutoClickOpenSolvedNiTitle: 'Suppress the message about recent pending questions to the reporter and then, depending on the choice set for that comment, clicks Open, Solved or Not Identified.',
                    AutoCloseCommentWindow: 'Auto close comment window',
                    AutoCloseCommentWindowTitle: 'This will automatically close the UR window for user requests that do not require saving after you click a UR comment in the list and then the send button.',
                    AutoSaveAfterSolvedOrNiComment: 'Auto save after solved or NI comment',
                    AutoSaveAfterSolvedOrNiCommentTitle: 'If \'Auto Click Open, Solved or Not Identified\' is also checked, this will click the save button after clicking a UR comment in the list and then the send button.',
                    AutoSendReminders: 'Auto send reminders',
                    AutoSendRemindersTitle: 'Auto send reminders to your URs on the screen.',
                    AutoSendRemindersWarning: 'WARNING',
                    AutoSendRemindersWarningTitle: 'This will AUTOMATICALLY send reminders at the reminder days setting (currently: ' + _settings.reminderDays + ' days).\nThis only happens when they are visible on your screen.\n\nNOTE: When using this feature you should not leave URs open unless you asked a question\nthat needs a response from the reporter, as this script will send reminders to all open URs\nafter \'Reminder days\'.',
                    AutoSetNewUrComment: 'Auto set new UR comment',
                    AutoSetNewUrCommentTitle: 'Auto set the default UR comment for the UR type on new URs that do not already have comments.',
                    AutoSetReminderUrComment: 'Auto set reminder UR comment',
                    AutoSetReminderUrCommentTitle: 'Auto set the UR reminder comment for URs that are older than the \'Reminder days\' setting and have only one comment.',
                    AutoSwitchToUrCommentsTab: 'Auto switch to the URC-E tab',
                    AutoSwitchToUrCommentsTabTitle: 'Auto switch to the URComments-Enhanced tab when opening a UR. When the UR window is closed you will be switched back to your previous tab.',
                    AutoZoomInOnNewUr: 'Auto zoom in on new UR',
                    AutoZoomInOnNewUrTitle: 'Auto zoom in when opening URs with no comments and when sending UR reminders.',
                    AutoZoomOutAfterComment: 'Auto zoom out after comment',
                    AutoZoomOutAfterCommentTitle: 'After clicking on a UR comment in the list and then clicking send on the UR, the map zoom will be set back to your previous zoom.',
                    DisableDoneNextButtons: 'Disable done / next buttons',
                    DisableDoneNextButtonsTitle: 'Disable the done / next buttons at the bottom of the new UR window.',
                    DoubleClickLinkNiComments: 'Double click link - NI comments',
                    DoubleClickLinkNiCommentsTitle: 'Add an extra link to the \'not identified\' comments. When double clicked it will automatically send the comment to the UR window, click send, and then will launch all of the other options that are enabled.',
                    DoubleClickLinkOpenComments: 'Double click link - Open comments',
                    DoubleClickLinkOpenCommentsTitle: 'Add an extra link to the \'open\' comments. When double clicked it will automatically send the comment to the UR window, click send, and then will launch all of the other options that are enabled.',
                    DoubleClickLinkSolvedComments: 'Double click link - Solved comments',
                    DoubleClickLinkSolvedCommentsTitle: 'Add an extra link to the \'solved\' comments. When double clicked it will automatically send the comment to the UR window, click send, and then will launch all of the other options that are enabled.',
                    HideZoomOutLinks: 'Hide zoom out links',
                    HideZoomOutLinksTitle: 'Hide the zoom out links on the comments tab.',
                    UnfollowUrAfterSend: 'Unfollow UR after send',
                    UnfollowUrAfterSendTitle: 'Unfollow the UR after sending a comment.',
                    // UR Marker Preferences
                    UrMarkerPrefs: 'UR Marker Preferences',
                    EnableUrPillCounts: 'Enable UR pill counts',
                    EnableUrPillCountsTitle: 'Enable or disable the pill with UR counts.',
                    DoNotShowTagNameOnPill: 'Don\'t show tag name on pill',
                    DoNotShowTagNameOnPillTitle: 'Do not show the tag name on the pill where there is a URO tag.',
                    ReplaceTagNameWithEditorName: 'Replace tag name with editor name',
                    ReplaceTagNameWithEditorNameTitle: 'When a UR has the logged in editors name in the description or any of the comments of the UR (not the name Waze automatically adds when commenting), replace the tag type with the editors name.',
                    UnstackMarkers: 'Unstack markers',
                    UnstackMarkersTitle: 'Attempt to unstack markers by offsetting them, URO+ style.',
                    UseCustomMarkersFor: 'Use Custom Markers for',
                    NativeSpeedLimits: 'Native speed limits',
                    NativeSpeedLimitsTitle: 'Replace default UR marker with custom marker for the URs with \'speed limit\' type.',
                    // UR Filtering Preferences
                    UrFilteringPrefs: 'UR Filtering Preferences',
                    EnableUrceUrFiltering: 'Enable URC-E UR filtering',
                    EnableUrceUrFilteringTitle: 'Enable or disable URComments-Enhanced built-in UR filtering.',
                    HideOutsideEditableArea: 'Hide outside editable area',
                    HideOutsideEditableAreaTitle: 'Hide URs outside your editable area.',
                    DoNotHideSelectedUr: 'Do not hide selected UR',
                    DoNotHideSelectedUrTitle: 'Do not hide a UR if it is actually being selected.',
                    // Hide by type
                    HideByType: 'Hide by type',
                    HideByTypeBlockedRoadTitle: 'Hide all blocked road URs.',
                    HideByTypeGeneralErrorTitle: 'Hide all general error URs.',
                    HideByTypeIncorrectAddressTitle: 'Hide all incorrect address URs.',
                    HideByTypeIncorrectJunctionTitle: 'Hide all incorrect junction URs.',
                    HideByTypeIncorrectRouteTitle: 'Hide all incorrect route URs.',
                    HideByTypeIncorrectTurnTitle: 'Hide all incorrect turn URs.',
                    HideByTypeMissingBridgeOverpassTitle: 'Hide all missing bridge overpass URs.',
                    HideByTypeMissingExitTitle: 'Hide all missing exit URs.',
                    HideByTypeMissingLandmarkTitle: 'Hide all missing landmark URs.',
                    HideByTypeMissingOrInvalidSpeedLimitTitle: 'Hide all missing or invalid speed limit URs',
                    HideByTypeMissingRoadTitle: 'Hide all missing road URs.',
                    HideByTypeMissingRoundaboutTitle: 'Hide all missing roundabout URs.',
                    HideByTypeTurnNotAllowedTitle: 'Hide all turn not allowed URs.',
                    HideByTypeUndefinedTitle: 'Hide all undefined URs.',
                    HideByTypeWazeAutomaticTitle: 'Hide all Waze automatic URs.',
                    HideByTypeWrongDrivingDirectionTitle: 'Hide all wrong driving direction URs.',
                    // Hide by tagged
                    HideByTagged: 'Hide by tag',
                    HideByTaggedBogTitle: 'Hide all URs with [BOG] in description or comments.',
                    HideByTaggedClosureTitle: 'Hide all URs with [CLOSURE] in description or comments.',
                    HideByTaggedConstructionTitle: 'Hide all URs with [CONSTRUCTION] in description or comments.',
                    HideByTaggedDifficultTitle: 'Hide all URs with [DIFFICULT] in description or comments.',
                    HideByTaggedEventTitle: 'Hide all URs with [EVENT] in description or comments.',
                    HideByTaggedNoteTitle: 'Hide all URs with [NOTE] in description or comments.',
                    HideByTaggedRoadworksTitle: 'Hide all URs with [ROADWORKS] in description or comments.',
                    HideByTaggedWslmTitle: 'Hide all URs with [WSLM] in description or comments.',
                    // Hide by status
                    HideByStatus: 'Hide by status',
                    HideByStatusOpenTitle: 'Hide all open URs.',
                    HideByStatusClosedTitle: 'Hide all closed (solved and not identified) URs.',
                    HideByStatusNotIdentifiedTitle: 'Hide all closed as not identified URs.',
                    HideByStatusSolvedTitle: 'Hide all closed as solved URs.',
                    // Hide by age of submission
                    HideByAgeOfSubmission: 'Hide by age of submission',
                    // Hide by description, comment, following
                    DescriptionCommentsFollowing: 'Hide by description, comment, following',
                    HideFollowingTitle: 'Hide URs you are following.',
                    HideNotFollowingTitle: 'Hide URs you are not following.',
                    HideWithDescriptionTitle: 'Hide URs that have a description.',
                    HideWithoutDescriptionTitle: 'Hide URs that do not have a description.',
                    HideCommentsFromMe: 'Comments from me',
                    HideWithCommentsFromMeTitle: 'Hide URs you have commented on.',
                    HideWithoutCommentsFromMeTitle: 'Hide URs you have not commented on.',
                    HideFirstCommentByMe: 'First comment by me',
                    HideFirstCommentByMeTitle: 'Hide URs where you were the first person to comment.',
                    HideFirstCommentNotByMeTitle: 'Hide URs where someone else was the first person to comment.',
                    HideLastCommentByMe: 'Last comment by me',
                    HideLastCommentByMeTitle: 'Hide URs where you are the last person to comment.',
                    HideLastCommentNotByMeTitle: 'Hide URs where someone else is the last person to comment.',
                    HideLastCommentByReporter: 'Last comment by reporter',
                    HideLastCommentByReporterTitle: 'Hide URs where the reporter is the last person to comment.',
                    HideLastCommentNotByReporterTitle: 'Hide URs where the reporter is not the last person to comment.',
                    HideByAgeOfFirstCommentLessThan: 'First comment less than',
                    HideByAgeOfFirstCommentMoreThan: 'First comment more than',
                    HideByAgeOfLastCommentLessThan: 'Last comment less than',
                    HideByAgeOfLastCommentMoreThan: 'Last comment more than',
                    // Lifecycle
                    LifeCycleStatus: 'Hide by lifecycle status',
                    HideWaiting: 'Waiting',
                    HideWaitingTitle: 'Only show URs that need work (hide URs in other parts of the life-cycle).',
                    HideUrsCloseNeeded: 'Close needed',
                    HideUrsCloseNeededTitle: 'Hide URs that need closing.',
                    HideUrsReminderNeeded: 'Reminders needed',
                    HideUrsReminderNeededTitle: 'Hide URs where reminders are needed.',
                    // Common Preferences
                    CommonPrefs: 'Common Preferences',
                    ReminderDays: 'Reminder days',
                    ReminderDaysTitle: 'Number of days to use when calculating UR filtering and when setting and/or sending the reminder comment. Must be between 0 and 13 and less than \'Close days\'. 0 is off (no reminder used).',
                    CloseDays: 'Close days',
                    CloseDaysTitle: 'Number of days to use when calculating UR filtering. Must be between 2 and 14 and greater than \'Reminder days\'.'
                },
                tabs: {
                    Comments: 'Comments',
                    Settings: 'Settings'
                },
                tags: {
                    Bog: '[BOG]',
                    BogTitle: 'Boots On Ground',
                    Closure: '[CLOSURE]',
                    ClosureTitle: 'Closures',
                    Construction: '[CONSTRUCTION]',
                    ConstructionTitle: 'Construction',
                    Difficult: '[DIFFICULT]',
                    DifficultTitle: 'Difficult',
                    Event: '[EVENT]',
                    EventTitle: 'Events',
                    Note: '[NOTE]',
                    NoteTitle: 'Notes',
                    Roadworks: '[ROADWORKS]',
                    RoadworksTitle: 'Roadworks (UK).',
                    Wslm: '[WSLM]',
                    WslmTitle: 'Waze Speed Limit Marker'
                },
                urStatus: {
                    Open: 'Open',
                    Closed: 'Closed',
                    Solved: 'Solved',
                    NotIdentified: 'Not identified'
                },
                urTypes: {
                    BlockedRoad: 'Blocked road',
                    GeneralError: 'General error',
                    IncorrectAddress: 'Incorrect address',
                    IncorrectJunction: 'Incorrect junction',
                    IncorrectRoute: 'Incorrect route',
                    IncorrectTurn: 'Incorrect turn',
                    MissingBridgeOverpass: 'Missing bridge overpass',
                    MissingExit: 'Missing exit',
                    MissingLandmark: 'Missing landmark',
                    MissingOrInvalidSpeedLimit: 'Missing/Invalid speed limit',
                    MissingRoad: 'Missing road',
                    MissingRoundabout: 'Missing roundabout',
                    TurnNotAllowed: 'Turn not allowed',
                    Undefined: 'Undefined',
                    WazeAutomatic: 'Waze automatic',
                    WrongDrivingDirection: 'Wrong driving direction'
                },
                prompts: {
                    NoCommentBox: 'URC-E: Unable to find the comment box! In order for this script to work, you need to have a UR open.',
                    CommentInsertTimedOut: 'URCE-E timed out waiting for the comment text box to become available.',
                    ReminderMessageAuto: 'URC-E: Automatically sending reminder message to UR:',
                    CustomListUsed: 'URC-E has loaded your "Custom" comment list. However, only the comments themselves have been loaded. The settings text and tooltips were not loaded. Further, this functionality is deprecated and may be discontinued at any time. An alternative solution may or may not be offered at that time.'
                }
            }
        });
    }

    function setTranslations(translations) {
        logDebug('Setting translations.');
        I18n.translations[I18n.currentLocale()].urce = translations.en;
        for (let i = 0; i < Object.keys(translations).length; i++) {
            let locale = Object.keys(translations)[i];
            if (I18n.currentLocale() == locale) {
                I18n.translations[locale].urce.prefs = translations[locale].prefs;
                return;
            }
        }
    }

    // Date to Days, courtesy of URO+. Thank you!
    function uroDateToDays(dateToConvert) {
        let dateNow = new Date();

        let elapsedSinceEpoch = dateNow.getTime();
        let elapsedSinceEvent = elapsedSinceEpoch - dateToConvert;

        dateNow.setHours(0);
        dateNow.setMinutes(0);
        dateNow.setSeconds(0);
        dateNow.setMilliseconds(0);
        let elapsedSinceMidnight = elapsedSinceEpoch - dateNow.getTime();
        dateNow.setHours(24);
        let pendingUntilMidnight = elapsedSinceEpoch - dateNow.getTime();

        if ((elapsedSinceEvent < elapsedSinceMidnight) && (elapsedSinceEvent > pendingUntilMidnight)) {
            // event occurred today...
            return 0;
        } else if (elapsedSinceEvent < 0) {
            // event occurrs at some point in the future after midnight today, so return a minimum value of -1...
            return -1 - Math.floor((pendingUntilMidnight - elapsedSinceEvent) / 86400000);
        } else {
            // event occurred at some point prior to midnight this morning, so return a minimum value of 1...
            return 1 + Math.floor((elapsedSinceEvent - elapsedSinceMidnight) / 86400000);
        }
    }

    // URO+ Alt Markers, courtesy of URO+. Thank you!
    let uroAltMarkers =
        [
            // each altMarker has 4 variants: 0 = normal open, 1 = selected open, 2 = normal closed, 3 = selected closed
            //  0: closure UR
            [
                "",
                "",
                "",
                ""
            ],
            //  1: roadworks UR
            [
                "",
                "",
                "",
                ""
            ],
            // 2: custom keyword UR
            [
                "",
                "",
                "",
                ""
            ],
            //  3: note UR
            [
                "",
                "",
                "",
                ""
            ],
            //  4: event UR
            [
                "",
                "",
                "",
                ""
            ],
            // 5: WMSL/SLUR UR
            [
                "",
                "",
                "",
                ""
            ],
            // 6: Elgin MP
            [
                "",
                "",
                "",
                ""
            ],
            // 7: TrafficCast MP
            [
                "",
                "",
                "",
                ""
            ],
            // 8: TrafficMaster MP
            [
                "",
                "",
                "",
                ""
            ],
            // 9: CalTrans
            [
                '',
                '',
                '',
                ''
            ],
            // 10: TfL
            [
                '',
                '',
                '',
                ''
            ],
            // 11: BOG
            [
                '',
                '',
                '',
                ''
            ],
            // 12: Difficult turn
            [
                '',
                '',
                '',
                ''
            ]
        ];
})();