WME URComments-Enhanced

URComments-Enhanced (URC-E) allows Waze editors to handle WME update requests more quickly and efficiently. Also adds many UR filtering options, ability to change the markers, plus much, much, more!

19.04.2023 itibariyledir. En son verisyonu görün.

// ==UserScript==
// @name        WME URComments-Enhanced
// @namespace   https://greasyfork.org/users/166843
// @version     2023.04.19.01
// eslint-disable-next-line max-len
// @description URComments-Enhanced (URC-E) allows Waze editors to handle WME update requests more quickly and efficiently. Also adds many UR filtering options, ability to change the markers, plus much, much, more!
// @grant       GM_xmlhttpRequest
// @match       *://*.waze.com/*editor*
// @exclude     *://*.waze.com/user/editor*
// @require     https://greasyfork.org/scripts/24851-wazewrap/code/WazeWrap.js
// @author      dBsooner
// @license     MIT/BSD/X11
// @connect     greasyfork.org
// @icon        
// @contributionURL https://github.com/WazeDev/Thank-The-Authors
// ==/UserScript==

/* global unsafeWindow, $, GM_info, GM_xmlhttpRequest, I18n, OpenLayers, W, WazeWrap */

/*
 * 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
 *
 */
(function () {
    'use strict';

    let _settings = {},
        _selUr = {
            doubleClick: false,
            handling: false,
            newStatus: undefined,
            urId: -1,
            urOpen: false
        },
        _restrictionsEnforce = {},
        _restrictionsEnforcedTitle,
        _commentList = [],
        _commentListLoaded = false,
        _customReplaceVars = [],
        _markerStackArray = [],
        _currentCommentList = null,
        _filtersAppliedOnZoom = false,
        _initialUrLayerScan = false,
        _markerCountOnInit = -1,
        _mousedOverMarkerId = null,
        _mouseIsDown = false,
        _needTranslation = false,
        _unstackedMasterId = null,
        _restoreZoom = null,
        _restoreDrawerTab,
        _restoreTab,
        _restoreTabPosition,
        _wmeUserId = null,
        _initUrIdInUrlObserver,
        _lastVersionChecked = '0',
        _needUrId = false,
        _mapUpdateRequests = {};

    // eslint-disable-next-line no-nested-ternary
    const _SCRIPT_SHORT_NAME = `URC-E${(/beta/.test(GM_info.script.name) ? ' β' : /\(DEV\)/i.test(GM_info.script.name) ? ' Ω' : '')}`,
        _SCRIPT_LONG_NAME = GM_info.script.name,
        _IS_ALPHA_VERSION = /[Ω]/.test(_SCRIPT_SHORT_NAME),
        _IS_BETA_VERSION = /[β]/.test(_SCRIPT_SHORT_NAME),
        _SCRIPT_AUTHOR = GM_info.script.author,
        _PROD_URL = 'https://greasyfork.org/scripts/375430-wme-urcomments-enhanced/code/WME-URComments-Enhanced.user.js',
        _PROD_META_URL = 'https://greasyfork.org/scripts/375430-wme-urcomments-enhanced/code/WME-URComments-Enhanced.meta.js',
        _FORUM_URL = 'https://www.waze.com/forum/viewtopic.php?f=819&t=275608',
        _SETTINGS_STORE_NAME = 'WME_URC-E',
        _BETA_URL = 'YUhSMGNITTZMeTluY21WaGMzbG1iM0pyTG05eVp5OXpZM0pwY0hSekx6TTNOelEyTkMxM2JXVXRkWEpqYjIxdFpXNTBjeTFsYm1oaGJtTmxaQzFpWlhSaEwyTnZaR1V2VjAxRkxWVlNRMjl0YldWdWRITXRSVzVvWVc1alpXUXVkWE5sY2k1cWN3PT0=',
        _BETA_META_URL = 'YUhSMGNITTZMeTluY21WaGMzbG1iM0pyTG05eVp5OXpZM0pwY0hSekx6TTNOelEyTkMxM2JXVXRkWEpqYjIxdFpXNTBjeTFsYm1oaGJtTmxaQzFpWlhSaEwyTnZaR1V2VjAxRkxWVlNRMjl0YldWdWRITXRSVzVvWVc1alpXUXViV1YwWVM1cWN3PT0=',
        _ALERT_UPDATE = true,
        _SCRIPT_VERSION = GM_info.script.version.toString(),
        _SCRIPT_VERSION_CHANGES = ['<b>CHANGE:</b> Steps for custom Google sheet creation updated.',
            '<b>CHANGE:</b> WME production now includes function from WME beta.'
        ],
        _DEBUG = /[βΩ]/.test(_SCRIPT_SHORT_NAME),
        _LOAD_BEGIN_TIME = performance.now(),
        _STATIC_ONLY_USERS = ['itzwolf'],
        _URCE_API_KEY = 'AIzaSyA2xOeUfopDqhB8r8esEa2A-G0X64UMr1c',
        _URCE_SPREADSHEET_ID = '1aVKBOwjYmO88x96fIHtIQgAwMaCV_NfklvPqf0J0pzQ',
        _autoSwitch = {},
        _commentLists = [],
        _currentArea = { country: undefined, state: undefined },
        _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
        },
        _overflowUrsUrls = [],
        _restrictions = {},
        _spinners = {
            buildCommentList: false,
            changeCommentList: false,
            checkMarkerStacking: false,
            checkRestrictions: false,
            handleAfterCommentMutation: false,
            handleClickedShortcut: false,
            handleUpdateRequestContainer: false,
            handleUrLayer: false,
            handleUrOverflow: false,
            init: false,
            markerMouseDown: false,
            postUrComment: false
        },
        _timeouts = {
            autoCloseUrPanel: undefined,
            autoScrollComments: undefined,
            checkForStaticListArray: undefined,
            checkRestrictions: {},
            getMapUrsAsync: {},
            getOverflowUrsFromUrl: {},
            initUrIdInUrl: undefined,
            isDomElementReady: {},
            isPanelReady: {},
            onWmeReady: undefined,
            popup: undefined,
            popupDelay: undefined,
            saveSettingsToStorage: undefined
        },
        _saveButtonObserver = new MutationObserver((mutations) => {
            if ((W.model.actionManager._redoStack.length === 0)
                // 2023.04.05.01: Production save button observer mutations
                && (mutations.some((mutation) => (mutation.attributeName === 'class')
                        && mutation.target.classList.contains('waze-icon-save')
                        && (mutation.oldValue.indexOf('ItemDisabled') === -1)
                        && mutation.target.classList.contains('ItemDisabled'))
                // 2023.04.05.01: Beta save button observer mutations
                    || mutations.some((mutation) => ((mutation.attributeName === 'disabled')
                        && (mutation.oldValue === 'false')
                        && (mutation.target.attributes.disabled.value === 'true')))
                )
            )
                handleAfterSave();
        }),
        _urDataStateObserver = new MutationObserver((mutations) => {
            if (_selUr.handling) {
                const dataStateMutations = mutations.filter((mutation) => mutation.attributeName === 'data-state');
                if (dataStateMutations.length > 0) {
                    const newDataState = dataStateMutations[0].target.attributes['data-state'].nodeValue.replace('-', '');
                    if ((newDataState === 'open') || (newDataState === 'solved') || (newDataState === 'notidentified'))
                        _selUr.newStatus = newDataState;
                    else
                        logWarning(`INVALID DATA STATE CHANGE: ${dataStateMutations[0].target.attributes['data-state'].nodeValue}`);
                }
            }
        }),
        _urCommentsObserver = new MutationObserver((mutations) => {
            if (_selUr.handling && (mutations[0].addedNodes.length > 0))
                handleAfterCommentMutation(mutations[0].addedNodes[0]);
        }),
        _urPanelContainerObserver = new MutationObserver((mutations) => {
            if (_selUr.handling
                && (mutations.filter(
                    (mutation) => (mutation.removedNodes.length > 0) && $(mutation.target).is('#panel-container')
                ).filter(
                    (removedChild) => removedChild.removedNodes[0].classList.contains('show') && removedChild.removedNodes[0].classList.contains('mapUpdateRequest')
                ).length > 0)
            )
                handleAfterCloseUpdateContainer();
            else if (mutations.filter(
                (mutation) => (mutation.type === 'attributes')
            ).filter(
                (mutation) => (mutation.oldValue?.match('mapUpdateRequest panel') && mutation.target.classList.contains('show'))
            ).length > 0)
                handleUpdateRequestContainer();
        }),
        _urMarkerObserver = new MutationObserver((mutations) => {
            mutations.filter((mutation) => (mutation.addedNodes.length > 0)).forEach((newMarker) => {
                const $addedNode = $(newMarker.addedNodes[0]);
                if (!$addedNode.data('urce_hasListeners'))
                    $addedNode.on({ mouseover: markerMouseOver, mouseout: markerMouseOut, click: markerClick }).data('urce_hasListeners', true);
            });
            if (_needUrId) {
                const urId = mutations.filter(
                    (mutation) => ((mutation.attributeName === 'class') && !/marker-selected/.test(mutation.oldValue) && /marker-selected/.test(mutation.target.className))
                )[0]?.target.attributes['data-id'].value;
                if (urId > 0) {
                    _needUrId = false;
                    _selUr.urId = parseInt(urId);
                    logDebug(`Selected UR from marker mutation: ${_selUr.urId}`);
                    maskBoxes(undefined, true, 'needUrId', true);
                    handleUpdateRequestContainer();
                }
            }
        }),
        _urceSidepanelContentObserver = new MutationObserver((mutations) => {
            if (mutations.some((mutation) => mutation.target.classList.contains('active')))
                checkSidebarHeight();
        }),
        _userscriptsApiDocsLinkIntersectionObserver = new IntersectionObserver((entries) => {
            if (entries.some((entry) => entry.isIntersecting)) {
                checkSidebarHeight();
                _userscriptsApiDocsLinkIntersectionObserver.disconnect();
                _userscriptsApiDocsLinkIntersectionObserver.isObserving = false;
            }
        }, { root: document.documentElement }),
        _userInfoTabContentObserver = new MutationObserver((mutations) => {
            mutations.forEach((mutation) => {
                mutation.addedNodes.forEach((node) => {
                    if (/^userscripts-api-docs-link-container/.test(node.className) && !_userscriptsApiDocsLinkIntersectionObserver.isObserving) {
                        _userscriptsApiDocsLinkIntersectionObserver.observing = true;
                        _userscriptsApiDocsLinkIntersectionObserver.observe(node);
                    }
                });
            });
        });

    function log(message, data = '') { console.log(`${_SCRIPT_SHORT_NAME}:`, message, data); }
    function logError(message, data = '') { console.error(`${_SCRIPT_SHORT_NAME}:`, new Error(message), data); }
    function logWarning(message, data = '') { console.warn(`${_SCRIPT_SHORT_NAME}:`, message, data); }
    function logDebug(message, data = '') {
        if (_DEBUG)
            log(message, data);
    }

    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]);
            return a[property].localeCompare(b[property]);
        };
    }

    function dec(s = '') {
        return atob(atob(s));
    }

    function getRandomId() {
        return Math.random().toString(36).slice(2);
    }

    async function loadSettingsFromStorage(restoreSettings, proceedWithRestore) {
        if (!restoreSettings)
            logDebug('Loading settings from storage.');
        const invalidRestoreSettings = [],
            retainedSettings = [],
            defaultSettings = {
                lastSaved: 0,
                lastVersion: undefined,
                wmeUserId: undefined,
                expandMoreInfo: false,
                expandShortcuts: true,
                // Comment List
                commentList: 0,
                commentListStyle: 'default',
                commentListCollapses: {},
                customSsId: '',
                customTagline: '',
                tagEmail: '',
                autoSwitchCommentList: false,
                enableAppendMode: false,
                // Per Comment List Settings
                perCommentListSettings: {},
                // URC-E Master Settings
                autoCenterOnUr: false,
                autoClickOpenSolvedNi: false,
                autoCloseUrPanel: (_settings.autoCloseCommentWindow),
                autoSaveAfterSolvedOrNiComment: false,
                autoSendReminders: false,
                autoSendRemindersExceptTagged: false,
                autoSetNewUrComment: false,
                autoSetNewUrCommentSlur: false,
                autoSetNewUrCommentWithDescription: false,
                autoSetReminderUrComment: false,
                autoSwitchToUrCommentsTab: false,
                autoZoomInOnNewUr: false,
                autoZoomOutAfterClosePanel: false,
                autoZoomOutAfterComment: false,
                disableDoneNextButtons: false,
                replaceNextWithDoneButton: false,
                doubleClickLinkNiComments: false,
                doubleClickLinkOpenComments: false,
                doubleClickLinkSolvedComments: false,
                hideZoomOutLinks: false,
                unfollowUrAfterSend: false,
                enableUrOverflowHandling: false,
                enableAutoRefresh: false,
                autoScrollComments: false,
                reverseCommentSort: false,
                reminderDays: 0,
                closeDays: 7,
                // UR Marker Settings
                enableUrPillCounts: false,
                disableUrMarkerPopup: false,
                urMarkerPopupDelay: 2,
                urMarkerPopupTimeout: 3,
                doNotShowTagNameOnPill: false,
                replaceTagNameWithEditorName: false,
                unstackMarkers: false,
                unstackDisableAboveZoom: 15,
                unstackSensitivity: 15,
                customMarkersRoadworks: false,
                customMarkersConstruction: false,
                customMarkersClosures: false,
                customMarkersEvents: false,
                customMarkersNotes: false,
                customMarkersWslm: false,
                customMarkersBog: false,
                customMarkersDifficult: false,
                customMarkersNativeSl: false,
                customMarkersCustom: false,
                customMarkersCustomText: '',
                // UR Filtering Settings
                enableUrceUrFiltering: false,
                invertFilters: false,
                hideOutsideEditableArea: false,
                doNotFilterTaggedUrs: false,
                doNotHideSelectedUr: false,
                disableFilteringAboveZoom: false,
                disableFilteringAboveZoomLevel: 12,
                disableFilteringBelowZoom: false,
                disableFilteringBelowZoomLevel: 22,
                // -- Lifecycle
                hideWaiting: false,
                hideUrsCloseNeeded: false,
                hideUrsReminderNeeded: false,
                // -- Hide by status
                hideByStatusOpen: false,
                hideByStatusClosed: false,
                hideByStatusNotIdentified: false,
                hideByStatusSolved: false,
                hideByStatusClosedBy: false,
                hideByStatusClosedByUsers: '',
                // -- Hide by type
                hideByTypeBlockedRoad: false,
                hideByTypeGeneralError: false,
                hideByTypeIncorrectAddress: false,
                hideByTypeIncorrectJunction: false,
                hideByTypeIncorrectRoute: false,
                hideByTypeIncorrectStreetPrefixOrSuffix: false,
                hideByTypeIncorrectTurn: false,
                hideByTypeMissingBridgeOverpass: false,
                hideByTypeMissingExit: false,
                hideByTypeMissingLandmark: false,
                hideByTypeMissingOrInvalidSpeedLimit: false,
                hideByTypeMissingRoad: false,
                hideByTypeMissingRoundabout: false,
                hideByTypeMissingStreetName: false,
                hideByTypeTurnNotAllowed: false,
                hideByTypeUndefined: 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: '',
                hideByAgeOfSubmissionMoreThan: false,
                hideByAgeOfSubmissionMoreThanDaysOld: '',
                // -- Hide by Descriptions / Comments / Following
                hideFollowing: false,
                hideNotFollowing: false,
                hideWithDescription: false,
                hideWithoutDescription: false,
                hideWithCommentsFromMe: false,
                hideWithoutCommentsFromMe: false,
                hideFirstCommentByMe: false,
                hideFirstCommentNotByMe: false,
                hideLastCommentByMe: false,
                hideLastCommentNotByMe: false,
                hideLastCommentByReporter: false,
                hideLastCommentNotByReporter: false,
                hideByCommentCountLessThan: false,
                hideByCommentCountLessThanNumber: '',
                hideByCommentCountMoreThan: false,
                hideByCommentCountMoreThanNumber: '',
                hideByAgeOfFirstCommentLessThan: false,
                hideByAgeOfFirstCommentLessThanDaysOld: '',
                hideByAgeOfFirstCommentMoreThan: false,
                hideByAgeOfFirstCommentMoreThanDaysOld: '',
                hideByAgeOfLastCommentLessThan: false,
                hideByAgeOfLastCommentLessThanDaysOld: '',
                hideByAgeOfLastCommentMoreThan: false,
                hideByAgeOfLastCommentMoreThanDaysOld: '',
                hideByKeywordIncluding: false,
                hideByKeywordIncludingKeyword: '',
                hideByKeywordNotIncluding: false,
                hideByKeywordNotIncludingKeyword: '',
                hideByKeywordCaseInsensitive: false,
                hideWithCommentBy: false,
                hideWithCommentByUsers: '',
                hideWithoutCommentBy: false,
                hideWithoutCommentByUsers: ''
            };
        if (restoreSettings && (restoreSettings !== 'resetSettings') && !proceedWithRestore) {
            Object.keys(restoreSettings).forEach((prop) => {
                if (!defaultSettings.hasOwnProperty(prop)) {
                    invalidRestoreSettings.push(prop);
                    delete (restoreSettings[prop]);
                }
                else if ((restoreSettings[prop] === 'true') || (restoreSettings[prop] === true)) {
                    restoreSettings[prop] = true;
                }
                else if ((restoreSettings[prop] === 'false') || (restoreSettings[prop] === false)) {
                    restoreSettings[prop] = false;
                }
                else if ((typeof restoreSettings[prop] !== 'object')
                    && !isNaN(restoreSettings[prop])
                    && (restoreSettings[prop].length > 0)
                    && (restoreSettings[prop] !== parseInt(restoreSettings[prop]))
                ) {
                    restoreSettings[prop] = parseInt(restoreSettings[prop]);
                }
            });
            Object.keys(_settings).forEach((prop) => {
                if (!restoreSettings.hasOwnProperty(prop)) {
                    restoreSettings[prop] = _settings[prop];
                    retainedSettings.push(prop);
                }
            });
            let outputText = `<b>${I18n.t('urce.prompts.RestoreSettingsNumOfSettings')}:</b> ${Object.keys(restoreSettings).length - retainedSettings.length}`
                + `<br><b>${I18n.t('urce.prompts.RestoreSettingsRetainedSettings')}:</b> `;
            if (retainedSettings.length > 0)
                outputText += `${retainedSettings.join(', ')} <i>(<b>${I18n.t('urce.common.Total')}:</b> ${retainedSettings.length})</i>`;
            else
                outputText += `<i>${I18n.t('urce.common.None')}</i>`;
            outputText += `<br><b>${I18n.t('urce.prompts.RestoreSettingsInvalidSettings')}:</b> `;
            if (invalidRestoreSettings.length > 0)
                outputText += `${invalidRestoreSettings.join(', ')} <i>(<b>${I18n.t('urce.common.Total')}:</b> ${invalidRestoreSettings.length})</i>`;
            else
                outputText += `<i>${I18n.t('urce.common.None')}</i>`;
            outputText += `<br><br><b>${I18n.t('urce.prompts.RestoreSettingsConfirmation')}</b>`;
            WazeWrap.Alerts.confirm(
                _SCRIPT_SHORT_NAME,
                formatText(outputText, true, false, -1),
                () => { loadSettingsFromStorage(restoreSettings, true); },
                () => { },
                I18n.t('urce.common.Yes'),
                I18n.t('urce.common.No')
            );
            return Promise.resolve();
        }
        const loadedSettings = (restoreSettings === 'resetSettings') ? {} : restoreSettings || $.parseJSON(localStorage.getItem(_SETTINGS_STORE_NAME));
        _settings = $.extend(true, {}, defaultSettings, loadedSettings);

        const serverSettings = await WazeWrap.Remote.RetrieveSettings(_SETTINGS_STORE_NAME);
        if (!restoreSettings && (serverSettings?.lastSaved > _settings.lastSaved)) {
            _settings = $.extend(true, _settings, serverSettings);
            _timeouts.saveSettingsToStorage = window.setTimeout(saveSettingsToStorage, 5000);
        }

        if (_settings.wmeUserId !== _wmeUserId)
            _settings.wmeUserId = _wmeUserId;
            // Remove old settings
        ['autoCloseCommentWindow', 'hideClosedUrs', 'showOthersUrsPastReminderClose', 'onlyShowMyUrs', 'hideTaggedUrs', 'hideUrsWoComments', 'hideUrsWoCommentsOrDescriptions',
            'hideUrsWoCommentsWithDescriptions', 'hideUrsWithUserReplies', 'disableAboveZoomLevel', 'hideByAgeOfLastCommentLessThanDaysAgo',
            'hideByAgeOfLastCommentMoreThanDaysAgo', 'hideByAgeOfFirstCommentMoreThanDaysAgo', 'hideByTypeWazeAutomatic', 'sortCommentsOldestFirst'].forEach((oldSetting) => {
            if (_settings.hasOwnProperty(oldSetting))
                delete (_settings[oldSetting]);
        });
        // Fix bad settings
        ['reminderDays', 'closeDays', 'hideByAgeOfLastCommentMoreThanDaysOld', 'hideByAgeOfLastCommentLessThanDaysOld', 'hideByAgeOfFirstCommentMoreThanDaysOld',
            'hideByAgeOfFirstCommentLessThanDaysOld', 'hideByCommentCountMoreThanNumber', 'hideByCommentCountLessThanNumber', 'hideByAgeOfSubmissionMoreThanDaysOld',
            'hideByAgeOfSubmissionLessThanDaysOld'].forEach((setting) => {
            if ((_settings[setting] === undefined) || (_settings[setting] === null) || ((_settings[setting].length === 0) && (_settings[setting] !== '')))
                _settings[setting] = '';
        });
        ['disableFilteringAboveZoomLevel', 'disableFilteringBelowZoomLevel', 'unstackDisableAboveZoom'].forEach((setting) => {
            if (_settings[setting] < 11)
                _settings[setting] += 12;
        });
        _timeouts.saveSettingsToStorage = window.setTimeout(saveSettingsToStorage, 5000);
        if (proceedWithRestore) {
            await initGui(false);
            await changeCommentList(_settings.commentList, false, true);
            handleUrLayer('settingsToggle', undefined, getMapUrsObjArr());
            saveSettingsToStorage();
            WazeWrap.Alerts.success(_SCRIPT_SHORT_NAME, ((restoreSettings === 'resetSettings') ? `${I18n.t('urce.prompts.ResetSettingsComplete')}.` : `${I18n.t('urce.prompts.RestoreSettingsComplete')}.`));
        }
        return Promise.resolve();
    }

    function showScriptInfoAlert() {
        if (_ALERT_UPDATE && (_SCRIPT_VERSION !== _settings.lastVersion)) {
            let releaseNotes = '';
            releaseNotes += '<p>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>';
            }
            WazeWrap.Interface.ShowScriptUpdate(_SCRIPT_SHORT_NAME, _SCRIPT_VERSION, releaseNotes, (_IS_BETA_VERSION ? dec(_BETA_URL) : _PROD_URL).replace(/code\/.*\.js/, ''), _FORUM_URL);
        }
    }

    async function saveSettingsToStorage() {
        checkTimeout({ timeout: 'saveSettingsToStorage' });
        if (localStorage) {
            _settings.commentListCollapses = _settings.commentListCollapses || {};
            _settings.commentListCollapses[_settings.commentList] = await getCollapsedGroups();
            _settings.lastVersion = _SCRIPT_VERSION;
            _settings.lastSaved = Date.now();
            localStorage.setItem(_SETTINGS_STORE_NAME, JSON.stringify(_settings));
            WazeWrap.Remote.SaveSettings(_SETTINGS_STORE_NAME, _settings);
            logDebug('Settings saved.');
        }
    }

    function checkTimeout(obj) {
        if (obj.toIndex) {
            if (_timeouts[obj.timeout]?.[obj.toIndex]) {
                window.clearTimeout(_timeouts[obj.timeout][obj.toIndex]);
                delete (_timeouts[obj.timeout][obj.toIndex]);
            }
        }
        else {
            if (_timeouts[obj.timeout])
                window.clearTimeout(_timeouts[obj.timeout]);
            _timeouts[obj.timeout] = undefined;
        }
    }

    async function checkSidebarHeight() {
        const urceNode = document.querySelector('#sidepanel-urc-e .URCE-divTabs'),
            userscriptsApiDocsLinkNode = document.querySelector('#user-info .userscripts-api-docs-link-container'),
            wzMapOlFooter = document.querySelector('#waze-map-container .wz-map-ol-footer'),
            { top } = urceNode.getBoundingClientRect(),
            { scrollY, pageYOffset } = window,
            { scrollTop } = document.body;
        if (!userscriptsApiDocsLinkNode || !wzMapOlFooter)
            return;
        let offset = (top + (scrollY || pageYOffset || scrollTop)) || 0;
        offset += userscriptsApiDocsLinkNode.offsetHeight;
        offset += wzMapOlFooter.offsetHeight;
        if (offset !== parseInt(getComputedStyle(urceNode).getPropertyValue('--height-offset'))) {
            logDebug(`Changing --height-offset to: ${offset}`);
            urceNode.style.setProperty('--height-offset', `${offset}px`);
        }
    }

    function disableWme(message, remove) {
        if (remove && ($('#urce-disableWme').length > 0)) {
            $('#urce-disableWme').remove();
        }
        else if (!remove && ($('#urce-disableWme').length < 1)) {
            $('body').append('<div id="urce-disableWme" class="URCE-disableWme-main">'
                + `     <div id="urce-disableWme-text" class="URCE-disableWme-text">${message}</div>`
                + '</div>');
        }
    }

    function dismissAlertBoxInPanel(event, idx) {
        if (((event?.data?.idx > -1) && ($(`#urceAlertPanelBox-${event?.data?.idx}`).length > 0))
            || ((idx > 0) && ($(`#urceAlertPanelBox-${idx}`).length > 0))
        )
            $(`#urceAlertPanelBox-${((idx > -1) ? idx : event.data.idx)}`).remove();
    }

    function alertBoxInPanel(message, panelBoxTitle, panelBoxDismiss, index) {
        if ($(`#urceAlertPanelBox-${index}`).length > 0)
            $(`#urceAlertPanelBox-${index}`).remove();
        const htmlOut = `<div id="urceAlertPanelBox-${index}" class="URCE-divWarningBox" title="${(panelBoxTitle || '')}">`
            + `${(panelBoxDismiss ? `<div id="urceAlertPanelBox-${index}-dismiss" class="URCE-divDismiss"><i class="fa fa-close" "aria-hidden"="true"></i></div>` : '')}`
            + `${message}</div>`;
        $('#panel-urce-comments').prepend(htmlOut);
        if (panelBoxDismiss)
            $(`#urceAlertPanelBox-${index}-dismiss`).off().on('click', { idx: index }, dismissAlertBoxInPanel);
    }

    function isChecked(obj) {
        return $(obj).is(':checked');
    }

    function isPanelReady(waitForAttr = '', retryInterval = 10, maxTries = 200) {
        return new Promise((resolve) => {
            (function retry(waitForAttrStr, tries, toIndex, retryInt, maxNumTries) {
                checkTimeout({ timeout: 'isPanelReady', toIndex });
                if (tries > maxNumTries)
                    resolve({ error: true });
                else if (!W.map.panelRegion.currentView?.model?.attributes?.[waitForAttrStr])
                    resolve({ error: false });
                else
                    _timeouts.isPanelReady[toIndex] = window.setTimeout(retry, retryInt, waitForAttrStr, ++tries, toIndex, retryInt, maxNumTries);
            }(waitForAttr, 1, getRandomId(), retryInterval, maxTries));
        });
    }

    function getDomElement(element, shadowHost, retryInterval = 10, maxTries = 200) {
        return new Promise((resolve) => {
            (function retry(elementStr, shadowHostStr, tries, toIndex, retryInt, maxNumTries) {
                checkTimeout({ timeout: 'isDomElementReady', toIndex });
                if (tries > maxNumTries) {
                    logError(`Timed out waiting for DOM element to appear: ${elementStr}`);
                    resolve(undefined);
                }
                else {
                    let $returnElem;
                    if (shadowHostStr) {
                        const $shadowHostElem = $(shadowHostStr);
                        if ($shadowHostElem?.length > 0)
                            $returnElem = $(elementStr, $shadowHostElem[0].shadowRoot);
                    }
                    else {
                        $returnElem = $(elementStr);
                    }
                    if ($returnElem?.length > 0)
                        resolve($returnElem);
                    else
                        _timeouts.isDomElementReady[toIndex] = window.setTimeout(retry, retryInt, elementStr, shadowHostStr, ++tries, toIndex, retryInt, maxNumTries);
                }
            }(element, shadowHost, 1, getRandomId(), retryInterval, maxTries));
        });
    }

    function handleReadyError(reopenUrPanel = false, spinnerStop = false, spinnerName = '', errorDisplay = false, errorText = '') {
        if (errorText.length > 0) {
            logError(errorText);
            if (errorDisplay)
                WazeWrap.Alerts.error(_SCRIPT_SHORT_NAME, errorText);
        }
        if (spinnerStop)
            doSpinner(spinnerName, false);
        if (reopenUrPanel && (parseInt($('.update-requests .marker-selected').attr('data-id')) > 0))
            openUrPanel(0, reopenUrPanel);
    }

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

    function doSpinner(spinnerName = '', spin = true) {
        const $btn = $('#urceUrMarkerProcessingSpinner');
        if ($btn.length === 0) {
            Object.keys(_spinners).forEach((a) => { _spinners[a] = false; });
        }
        else if (!spin) {
            _spinners[spinnerName] = false;
            if (!Object.values(_spinners).some((a) => a === true))
                $btn.removeClass('fa-spin').css({ color: 'lightgray' }).attr('title', I18n.t('urce.mouseOver.URMarkerProcessingInactive'));
        }
        else {
            _spinners[spinnerName] = true;
            if (!$btn.hasClass('fa-spin'))
                $btn.css({ color: 'black' }).attr('title', I18n.t('urce.mouseOver.URMarkerProcessingActive')).addClass('fa-spin');
        }
    }

    function checkRestrictions(event) {
        return new Promise((resolve) => {
            (function retry(tries, toIndex, evt) {
                checkTimeout({ timeout: 'checkRestrictions', toIndex });
                // 2023.04.05.01: W.model.getTopCountry() and W.model.getTopState() return null when zoom level < 12.
                if (W.map.getZoom() < 12) {
                    resolve();
                    return;
                }
                doSpinner('checkRestrictions', true);
                const displayWarning = (content, remove) => {
                    if (remove) {
                        $('#restrictionsEnforcedWarning, #restrictionsEnforcedWarning-Settings').hide();
                    }
                    else if (((evt?.[0]?.type === 'init') || (evt?.[0]?.type === 'modeChange'))
                        || ($('#restrictionsEnforcedWarning').length === 0)
                    ) {
                        _restrictionsEnforcedTitle = content;
                    }
                    else {
                        $('#restrictionsEnforcedWarning, #restrictionsEnforcedWarning-Settings').children()[0].setAttribute('title', content);
                        $('#restrictionsEnforcedWarning, #restrictionsEnforcedWarning-Settings').show();
                    }
                };
                let moved = false,
                    state,
                    country;
                if ((tries === 1) && (evt?.[0]?.type !== 'init') && (evt?.[0]?.type !== 'modeChange')) {
                    _timeouts.checkRestrictions[toIndex] = window.setTimeout(retry, 500, ++tries, toIndex, evt);
                }
                else if (tries < 301) {
                    if ((evt?.[0]?.type === 'state') && (evt?.[0]?.name === W.model.getTopState().name)) {
                        country = (evt[0].countryID !== 0) ? W.model.countries.getObjectById(evt[0].countryID).abbr : W.model.getTopCountry().abbr;
                        state = evt[0].name;
                    }
                    else if ((evt?.[0]?.type === 'country') && (evt?.[0]?.abbr === W.model.getTopCountry().abbr)) {
                        country = evt[0].abbr;
                        state = false;
                    }
                    if (!country && !W.model.getTopCountry()) {
                        logDebug(`Waiting on Waze model for countries to populate. Try ${tries} of 300.`);
                        _timeouts.checkRestrictions[toIndex] = window.setTimeout(retry, 100, ++tries, toIndex, evt);
                    }
                    else {
                        checkTimeout({ timeout: 'checkRestrictions', toIndex });
                        country = country || W.model.getTopCountry().abbr;
                        state = state || (W.model.getTopState() ? W.model.getTopState().name : false);
                        if (state !== _currentArea.state) {
                            _currentArea.state = state;
                            moved = true;
                        }
                        if (country !== _currentArea.country) {
                            _currentArea.country = country;
                            moved = true;
                        }
                        if (moved) {
                            logDebug((((evt?.[0]?.type === 'init') || (evt?.[0]?.type === 'modeChange')) ? 'Setting up restrictions.' : 'Checking restrictions.'));
                            _restrictionsEnforce = {};
                            let restrictionsAlertBannerTitle = `${I18n.t('urce.prompts.RestrictionsEnforced')}\n\n${I18n.t('urce.prompts.RestrictionsEnforcedTitle')}:\n`;
                            if (_restrictions[country]) {
                                if (_restrictions[country][state]) {
                                    restrictionsAlertBannerTitle += `\n${W.model.countries.getByAttributes({ abbr: country })[0].name} - ${state}:`;
                                    Object.keys(_restrictions[country][state]).forEach((restriction) => {
                                        _restrictionsEnforce[restriction] = _restrictions[country][state][restriction];
                                        restrictionsAlertBannerTitle += `\n${I18n.t(`urce.prefs.${restriction.charAt(0).toUpperCase()}${restriction.slice(1)}`)}: `;
                                        if (_restrictionsEnforce[restriction] === true)
                                            restrictionsAlertBannerTitle += I18n.t('urce.common.Enabled');
                                        else if (_restrictionsEnforce[restriction] === false)
                                            restrictionsAlertBannerTitle += I18n.t('urce.common.Disabled');
                                        else
                                            restrictionsAlertBannerTitle += I18n.t('common.time.days', { days: _restrictionsEnforce[restriction] });
                                    });
                                }
                                else if (Object.keys(_restrictions[country]?.ALL || {}).length > 0) {
                                    restrictionsAlertBannerTitle += `\n${W.model.countries.getByAttributes({ abbr: country })[0].name} - ${I18n.t('urce.common.All')}:`;
                                    Object.keys(_restrictions[country].ALL).forEach((restriction) => {
                                        _restrictionsEnforce[restriction] = _restrictions[country].ALL[restriction];
                                        restrictionsAlertBannerTitle += `\n${I18n.t(`urce.prefs.${restriction.charAt(0).toUpperCase()}${restriction.slice(1)}`)}: `;
                                        if (_restrictionsEnforce[restriction] === true)
                                            restrictionsAlertBannerTitle += I18n.t('urce.common.Enabled');
                                        else if (_restrictionsEnforce[restriction] === false)
                                            restrictionsAlertBannerTitle += I18n.t('urce.common.Disabled');
                                        else
                                            restrictionsAlertBannerTitle += I18n.t('common.time.days', { days: _restrictionsEnforce[restriction] });
                                    });
                                }
                                if (Object.values(_restrictionsEnforce).length > 0)
                                    displayWarning(restrictionsAlertBannerTitle, false);
                                else
                                    displayWarning(false, true);
                            }
                            else {
                                displayWarning(false, true);
                            }
                            resolve();
                        }
                        else {
                            resolve();
                        }
                    }
                }
                else {
                    resolve(logError('Unable to check for restrictions'));
                }
                doSpinner('checkRestrictions', false);
            }(1, getRandomId(), event));
        });
    }

    function getMapUrsObjArr(urIds = []) {
        const mUrsObjArr = [],
            objsArray = (urIds.length > 0) ? W.model.mapUpdateRequests.getByIds(urIds) : W.model.mapUpdateRequests.getObjectArray();
        objsArray.forEach((urObj) => {
            if (mUrsObjArr.indexOf(urObj.attributes.id) === -1)
                mUrsObjArr.push(urObj);
        });
        return mUrsObjArr;
    }

    function mUrsAdded(objectsArr) {
        if (objectsArr?.length === 0)
            return;
        const zoomLevel = W.map.getZoom();
        let filter = true;
        if ((_settings.disableFilteringAboveZoom && (zoomLevel < _settings.disableFilteringAboveZoomLevel))
                || (_settings.disableFilteringBelowZoom && (zoomLevel > _settings.disableFilteringBelowZoomLevel))
        )
            filter = false;
        handleUrLayer('mUrsAdded', filter, objectsArr.sort((a, b) => a.attributes.id - b.attributes.id));
    }

    function mUrsRemoved(objectsArr = []) {
        if (W.model.mapUpdateRequests.getObjectArray().length === 0)
            _mapUpdateRequests = {};
        else
            objectsArr.forEach((mUrObj) => delete (_mapUpdateRequests[mUrObj.attributes.id]));
    }

    async function getUpdateRequestSessions(urIds = []) {
        let data = {};
        try {
            data = await W.controller.descartesClient.getUpdateRequestSessionsByIds(urIds);
            if (data?.updateRequestSessions?.objects.length > 0)
                // 2023.04.05.01: No need to merge the data to the W.map.mapUpdateRequests repo. Let WME control that repo.
                // W.model.mergeResponse(data);
                data = Object.fromEntries(data.updateRequestSessions.objects.map((o) => [o.id, o]));
            else
                data = {};
        }
        catch (error) {
            logWarning(error);
        }
        return Promise.resolve(data);
    }

    async function handleAfterCommentMutation(domElem) {
        logDebug(`Handling new comment mutation for urId: ${_selUr.urId}`);
        doSpinner('handleAfterCommentMutation', true);
        if (_settings.unfollowUrAfterSend)
            unfollowUrAfterSend();
        if (_settings.autoZoomOutAfterComment)
            autoZoomOut();
        if (_settings.autoCloseUrPanel || _selUr.doubleClick) {
            await autoCloseUrPanel();
        }
        else {
            await updateUrceData(getMapUrsObjArr([_selUr.urId]));
            if ($(domElem).has('#urceDaysAgo').length === 0) {
                const shadowDOMstyle = document.createElement('style');
                shadowDOMstyle.innerHTML = '.key-with-image-wrapper, .key-wrapper { width: 100% } '
                    + '.wz-list-item, .wz-list-item.with-subtitle { --wz-list-item-vertical-padding: 0px !important; margin: 2px 0px !important; }';
                $(domElem).find('span.date').css('float', 'right');
                domElem.shadowRoot.appendChild(shadowDOMstyle.cloneNode(true));
                $($(domElem).children()[0]).append(''
                    + '<div class="date urce">'
                    + `     <div>(${parseDaysAgo(uroDateToDays(W.model.updateRequestSessions.objects[_selUr.urId].comments[(_mapUpdateRequests[_selUr.urId].urceData.commentCount - 1)].createdOn))})</div>`
                    + '</div>');
            }
            if (_settings.reverseCommentSort) {
                const $commentList = $('#panel-container .mapUpdateRequest.panel.show .top-section .body .conversation.section .conversation-view .comment-list'),
                    numComments = $commentList.children().length;
                domElem.remove();
                $commentList.prepend(domElem);
                autoScrollComments(numComments);
            }
            if (_settings.autoSaveAfterSolvedOrNiComment && ((_selUr.newStatus === 'solved') || (_selUr.newStatus === 'notidentified')))
                $('.toolbar-button.waze-icon-save').trigger('click');
            else
                handleUrLayer('sendComment', undefined, getMapUrsObjArr([_selUr.urId]));
        }
        doSpinner('handleAfterCommentMutation', false);
    }

    async function handleAfterCloseUpdateContainer() {
        _urPanelContainerObserver.disconnect();
        _urPanelContainerObserver.observe(document.getElementById('panel-container'), {
            childList: true, attributes: true, attributeOldValue: true, characterData: false, characterDataOldValue: false, subtree: true
        });
        /**
         * 2023.03.21
         * Commented out because the marker is still selected when the panel is first closed.
         * We need to go ahead and process the rest as it's important for the rest of our script. This will have a side effect of the zoom features being
         * a little wonky.
         * TODO: Find another way to know if we clicked a new UR marker with the current one still open OR if we truly closed the UR panel.
         *
        if (parseInt($('.update-requests .marker-selected').data('id')) > 0)
            return;
         */
        const { urId, newStatus } = _selUr;
        _selUr = {
            doubleClick: false,
            handling: false,
            newStatus: undefined,
            urId: -1,
            urOpen: false
        };
        if (_urDataStateObserver.isObserving) {
            _urDataStateObserver.disconnect();
            _urDataStateObserver.isObserving = false;
        }
        if (_urCommentsObserver.isObserving) {
            _urCommentsObserver.disconnect();
            _urCommentsObserver.isObserving = false;
        }
        if (_settings.autoZoomOutAfterClosePanel)
            autoZoomOut();
        if (_settings.autoSaveAfterSolvedOrNiComment && ((newStatus === 'solved') || (newStatus === 'notidentified'))) {
            $('.toolbar-button.waze-icon-save').trigger('click');
        }
        else {
            if (_settings.autoSwitchToUrCommentsTab)
                autoSwitchToPrevTab();
            handleUrLayer('close', undefined, getMapUrsObjArr([urId]));
        }
    }

    function handleAfterSave() {
        if (_settings.autoZoomOutAfterComment)
            autoZoomOut();
        if (_settings.autoSwitchToUrCommentsTab)
            autoSwitchToPrevTab();
        handleUrLayer('save', undefined, getMapUrsObjArr());
    }

    async function handleUpdateRequestContainer() {
        if (!_commentListLoaded)
            return;
        if ((parseInt($('.update-requests .marker-selected').attr('data-id')) > 0)
            && (!(_selUr.urId > 0) || (_selUr.urId !== parseInt($('.update-requests .marker-selected').attr('data-id'))))
        ) {
            _selUr.urId = parseInt($('.update-requests .marker-selected').attr('data-id'));
            logDebug(`Selected UR from handleURContainer: ${_selUr.urId}`);
        }
        else if (!(_selUr.urId > 0)) {
            _needUrId = true;
            maskBoxes(`${I18n.t('urce.prompts.WaitingToGetUrId')}<br><br>${I18n.t('urce.common.PleaseWait')}`, false, 'needUrId', true);
            return;
        }
        if (_settings.replaceNextWithDoneButton && W.map.panelRegion.currentView.options.showNext) {
            openUrPanel(_selUr.urId, true);
            return;
        }
        _selUr.handling = true;
        _urPanelContainerObserver.disconnect();
        _urPanelContainerObserver.observe(document.getElementById('panel-container'), {
            childList: true, attributes: false, attributeOldValue: false, characterData: false, characterDataOldValue: false, subtree: false
        });
        doSpinner('handleUpdateRequestContainer', true);
        _restoreZoom = W.map.getZoom();
        if (_timeouts.popup)
            hidePopup();
        logDebug(`Handling update request container for urId: ${_selUr.urId}`);
        if (_settings.autoSwitchCommentList) {
            const topCountry = W.model.getTopCountry(),
                topState = W.model.getTopState();
            if ((_autoSwitch[topCountry.abbr]?.ALL > -1) || (_autoSwitch[topCountry.abbr]?.[topState.name] > -1)) {
                let commentList;
                if (_autoSwitch[topCountry.abbr][topState.name] > -1)
                    commentList = _autoSwitch[topCountry.abbr][topState.name];
                else if (_autoSwitch[topCountry.abbr].ALL)
                    commentList = _autoSwitch[topState.abbr].ALL;
                else
                    commentList = -1;
                if ((commentList > -1) && (commentList !== _currentCommentList))
                    await changeCommentList(commentList, (commentList !== _settings.commentList), false);
            }
            else if (_currentCommentList !== _settings.commentList) {
                await changeCommentList(_settings.commentList, true, false);
            }
        }
        await updateUrceData(getMapUrsObjArr([_selUr.urId]));
        let $domElement = await getDomElement('#panel-container .top-section .header .main-title');
        if (!$domElement) {
            handleReadyError(true, true, 'handleUpdateRequestContainer', false, 'isDomElementReady: .main-title');
            return;
        }
        if ($domElement.html().indexOf(_selUr.urId) === -1)
            $domElement.append(` (${_selUr.urId}) `);
        $domElement = await getDomElement('#panel-container .top-section .header .reported');
        if (!$domElement) {
            handleReadyError(true, true, 'handleUpdateRequestContainer', false, '');
            return;
        }
        if (!$domElement.text().endsWith(')'))
            $domElement[0].textContent += ` (${parseDaysAgo(_mapUpdateRequests[_selUr.urId].urceData.driveDaysOld)})`;
        $domElement = (W.model.mapUpdateRequests.objects[_selUr.urId].attributes.description) ? await getDomElement('#panel-container .top-section .body .problem-data .description .content') : undefined;
        if ($domElement?.children().length === 0) {
            const $content = await getDomElement('#panel-container .top-section .body .problem-data .description .content'),
                newDiv = `<div class="URCE-divDesc">${$content.text()}</div>`;
            if ($content) {
                $content.text('');
                $content.append(newDiv);
            }
        }
        if (_mapUpdateRequests[_selUr.urId].urceData.commentCount > 0) {
            const shadowDOMstyle = document.createElement('style');
            shadowDOMstyle.innerHTML = '.key-with-image-wrapper, .key-wrapper { width: 100% } '
                    + '.wz-list-item, .wz-list-item.with-subtitle { --wz-list-item-vertical-padding: 0px !important; margin: 2px 0px !important; }';
            for (let idx = 0; idx < _mapUpdateRequests[_selUr.urId].urceData.commentCount; idx++) {
                // eslint-disable-next-line no-await-in-loop
                $domElement = await getDomElement('#panel-container .mapUpdateRequest .top-section .body .conversation .comment .comment-title');
                if (!$domElement) {
                    handleReadyError(false, false, '', false, 'isConversationLoaded: loadingConversation');
                }
                else {
                    const $currComment = $($domElement[idx]);
                    if ($currComment.has('#urceDaysAgo').length === 0) {
                        $currComment.children().filter('span.date').css('float', 'right');
                        $currComment.parent()[0].shadowRoot.appendChild(shadowDOMstyle.cloneNode(true));
                        $currComment.append(''
                                + '<div class="date urce">'
                                + `     <div>(${parseDaysAgo(uroDateToDays(W.model.updateRequestSessions.objects[_selUr.urId].comments[idx].createdOn))})</div>`
                                + '</div>');
                    }
                }
            }
            if (_settings.reverseCommentSort && (_mapUpdateRequests[_selUr.urId].urceData.commentCount > 1)) {
                const $commentList = await getDomElement(
                        '#panel-container .mapUpdateRequest.panel.show .top-section .body .conversation.section .conversation-view .comment-list'
                    ),
                    sortedComments = $commentList.children('wz-list-item').get().reverse();
                if ($commentList)
                    $commentList.append(sortedComments);
            }
        }
        _selUr.urOpen = W.model.mapUpdateRequests.objects[_selUr.urId].attributes.open;
        if (_settings.autoSwitchToUrCommentsTab)
            autoSwitchToUrceTab();
        if (_settings.disableDoneNextButtons) {
            $domElement = await getDomElement('#panel-container .mapUpdateRequest .actions .content .navigation');
            if ($domElement)
                $domElement.css({ display: 'none' });
        }
        $domElement = await getDomElement('#panel-container .mapUpdateRequest .top-section .header .title .focus');
        if ($domElement)
            $domElement.off('click', recenterOnUr).on('click', { urId: _selUr.urId }, recenterOnUr);
        $domElement = await getDomElement('textarea[id=id]', '#panel-container .mapUpdateRequest .top-section .body .conversation .new-comment-text');
        if (!$domElement) {
            handleReadyError(true, true, 'handleUpdateRequestContainer', false, '');
            return;
        }
        $domElement
            .css('background-color', (_settings.enableAppendMode ? 'peachpuff' : ''))
            .off('keyup', checkValue).on('keyup', checkValue);
        if ($('#urceShortcuts').length === 0) {
            $('#panel-container .mapUpdateRequest .top-section .body .conversation .new-comment-form').prepend(''
                + '<div id="urceShortcuts" style="text-align:left;padding-bottom:8px;font-size:12px;width:100%">'
                + `<div id="urceShortcutsExpand" style="padding-bottom:4px;font-size:13px;cursor:pointer;border-bottom:1px solid darkgray;">${I18n.t('urce.urPanel.Shortcuts')}`
                + `<div style="display:inline;float:right;padding-right:8px;font-size:11px;"><i class="fa fa-fw ${(_settings.expandShortcuts ? 'fa-chevron-up' : 'fa-chevron-down')} URCE-chevron"></i></div></div>`
                + `<div id="urceShortcutsExpandDiv" style="${(_settings.expandShortcuts ? '' : 'display:none;')}border-bottom:1px solid darkgray;padding: 5px 0 5px 0;"><div><i id="urceShortcuts-selSegs" class="fa fa-road" "aria-hidden"="true" style="cursor:pointer;padding-left:4px;" title="${I18n.t('urce.urPanel.InsertSelSegsTitle')}"></i>`
                + '<span class="fa-stack" style="width:1.8em;height:1.8em;line-height:1.8em;">'
                + `<i id="urceShortcuts-selSegsWithCity" class="fa fa-road" "aria-hidden"="true" style="cursor:pointer;padding-left:4px;" title="${I18n.t('urce.urPanel.InsertSelSegsWithCityTitle')}"></i>`
                + `<i id="urceShortcuts-selSegsWithCity" class="fa fa-plus fa-stack-2x" "area-hidden"="true" style="font-size:xx-small;text-align:right;cursor:pointer;" title="${I18n.t('urce.urPanel.InsertSelSegsWithCityTitle')}"></i>`
                + '</span><div style="display:inline-block;padding-left:8px;padding-right:8px;">||</div>'
                + `<i id="urceShortcuts-placeName" class="fa fa-home" "aria-hidden"="true" style="cursor:pointer;" title="${I18n.t('urce.urPanel.InsertPlaceNameTitle')}"></i>`
                + `<i id="urceShortcuts-placeAddress" class="fa fa-map-marker" "aria-hidden"="true" style="cursor:pointer;padding-left:4px;" title="${I18n.t('urce.urPanel.InsertPlaceAddressTitle')}"></i>`
                + '<div style="display:inline-block;padding-left:8px;padding-right:8px;">||</div>'
                + `<i id="urceShortcuts-description" class="fa fa-paragraph" "aria-hidden"="true" style="cursor:pointer;" title="${I18n.t('urce.urPanel.InsertDescriptionTitle')}"></i>`
                + `<i id="urceShortcuts-wazeUsername" class="fa fa-user-o" "aria-hidden"="true" style="cursor:pointer;padding-left:4px;" title="${I18n.t('urce.urPanel.InsertWazeUsernameTitle')}"></i>`
                + `<i id="urceShortcuts-urType" class="fa fa-info-circle" "aria-hidden"="true" style="cursor:pointer;padding-left:4px;" title="${I18n.t('urce.urPanel.InsertUrTypeTitle')}"></i>`
                + `<i id="urceShortcuts-customTagline" class="fa fa-tag" "aria-hidden"="true" style="cursor:pointer;padding-left:4px;" title="${I18n.t('urce.urPanel.InsertCustomTaglineTitle')}"></i>`
                + `<i id="urceShortcuts-urPermalink" class="fa fa-link" "aria-hidden"="true" style="cursor:pointer;padding-left:4px;" title="${I18n.t('urce.urPanel.InsertUrPermalinkTitle')}"></i>`
                + '</div>'
                + `<div><i id="urceShortcuts-driveTime" class="fa fa-clock-o" "aria-hidden"="true" style="cursor:pointer;padding-left:4px;" title="${I18n.t('urce.urPanel.InsertTimeTitle')}"></i>`
                + `<i id="urceShortcuts-driveDayOfWeek" class="fa fa-sun-o" "aria-hidden"="true" style="cursor:pointer;padding-left:4px;" title="${I18n.t('urce.urPanel.InsertDayOfWeekTitle')}"></i>`
                + `<i id="urceShortcuts-driveDate" class="fa fa-calendar-o" "aria-hidden"="true" style="cursor:pointer;padding-left:4px;" title="${I18n.t('urce.urPanel.InsertDateTitle')}"></i>`
                + '<div style="display:inline-block;padding-left:8px;padding-right:8px;">||</div>'
                + `<i id="urceShortcuts-driveTimeCasual" class="fa fa-clock-o" "aria-hidden"="true" style="cursor:pointer" title="${I18n.t('urce.urPanel.InsertTimeCasualTitle')}"></i>`
                + `<i id="urceShortcuts-driveDateCasual" class="fa fa-calendar" "aria-hidden"="true" style="cursor:pointer;padding-left:4px;" title="${I18n.t('urce.urPanel.InsertDateCasualTitle')}"></i>`
                + `<i id="urceShortcuts-driveDateTimeCasualMode" class="fa fa-calendar-plus-o" "aria-hidden"="true" style="cursor:pointer;padding-left:4px;" title="${I18n.t('urce.urPanel.InsertDateTimeCasualModeTitle')}"></i>`
                + `<div style="display:inline-block;padding-left:8px;">- ${I18n.t('urce.urPanel.DriveDate')}</div></div>`
                + `<div><i id="urceShortcuts-currentTime" class="fa fa-clock-o" "aria-hidden"="true" style="cursor:pointer;padding-left:4px;" title="${I18n.t('urce.urPanel.InsertCurrentTimeTitle')}"></i>`
                + `<i id="urceShortcuts-currentDayOfWeek" class="fa fa-sun-o" "aria-hidden"="true" style="cursor:pointer;padding-left:4px;" title="${I18n.t('urce.urPanel.InsertCurrentDayOfWeekTitle')}"></i>`
                + `<i id="urceShortcuts-currentDate" class="fa fa-calendar-o" "aria-hidden"="true" style="cursor:pointer;padding-left:4px;" title="${I18n.t('urce.urPanel.InsertCurrentDateTitle')}"></i>`
                + '<div style="display:inline-block;padding-left:8px;padding-right:8px;">||</div>'
                + `<i id="urceShortcuts-currentTimeCasual" class="fa fa-clock-o" "aria-hidden"="true" style="cursor:pointer;" title="${I18n.t('urce.urPanel.InsertCurrentTimeCasualTitle')}"></i>`
                + `<i id="urceShortcuts-currentDateCasual" class="fa fa-calendar" "aria-hidden"="true" style="cursor:pointer;padding-left:4px;" title="${I18n.t('urce.urPanel.InsertCurrentDateCasualTitle')}"></i>`
                + `<div style="display:inline-block;padding-left:24px;">- ${I18n.t('urce.urPanel.CurrentDate')}</div></div></div></div>`);
            $('i[id|="urceShortcuts"]').off().on('click', function () {
                handleClickedShortcut(this.id.substr(14));
            });
            $('#urceShortcutsExpand').off().on('click', function () {
                if ($($(this).find('i')[0]).hasClass('fa-chevron-up'))
                    _settings.expandShortcuts = false;
                else
                    _settings.expandShortcuts = true;
                $($(this).find('i')[0]).toggleClass('fa-chevron-down fa-chevron-up');
                if (_settings.expandShortcuts)
                    $('#urceShortcutsExpandDiv').show();
                else
                    $('#urceShortcutsExpandDiv').hide();
                saveSettingsToStorage();
            });
        }
        $domElement = await getDomElement('#panel-container .mapUpdateRequest .top-section .body .conversation .new-comment-form .send-button');
        if ($domElement)
            $domElement.on('click', () => { logDebug('Clicked send to submit the comment.'); });
        const shadowDOMstyle = document.createElement('style');
        shadowDOMstyle.innerHTML = '.wz-button.md { height: 30px !important; min-width: 60px !important; max-widt: 140px !important; padding: 0px 10px !important; overflow: hidden; }';
        $domElement = await getDomElement('#panel-container .mapUpdateRequest .top-section .body .conversation .new-comment-form .send-button');
        if ($domElement)
            $domElement[0].shadowRoot.appendChild(shadowDOMstyle.cloneNode(true));
        else
            logWarning('Could not appendChild CSS to send-button shadow root due to timeout waiting for DOM element.');

        shadowDOMstyle.innerHTML = '.wz-textarea textarea { font-size: 13px !important; line-height: 14px; padding: 6px !important; } '
            + '.wz-textarea .length-text { font-size: 12px !important; padding: 2px 6px 0px 0px !important; }';
        $domElement = await getDomElement('#panel-container .mapUpdateRequest .top-section .body .conversation .new-comment-form .new-comment-text');
        if ($domElement)
            $domElement[0].shadowRoot.appendChild(shadowDOMstyle.cloneNode(true));
        else
            logWarning('Could not appendChild CSS to new-comment-text shadow root due to timeout waiting for DOM element.');
        shadowDOMstyle.innerHTML = '.wz-checkbox { font-size: 12px !important; } .wz-checkbox .border { height: 15px !important; width: 15px !important; }';
        $domElement = await getDomElement('#panel-container .mapUpdateRequest .top-section .body .conversation .new-comment-form wz-checkbox[name=follow]');
        if ($domElement)
            $domElement[0].shadowRoot.appendChild(shadowDOMstyle.cloneNode(true));
        else
            logWarning('Could not appendChild CSS to new-comment-follow shadow root due to timeout waiting for DOM element.');
        if (!_urDataStateObserver.isObserving) {
            _urDataStateObserver.observe($('#panel-container .mapUpdateRequest.panel.show .problem-edit')[0], {
                childList: false, attributes: true, attributeOldValue: true, characterData: false, characterDataOldValue: false, subtree: false
            });
            _urDataStateObserver.isObserving = true;
        }
        if (!_urCommentsObserver.isObserving) {
            _urCommentsObserver.observe($('#panel-container .mapUpdateRequest.panel.show .top-section .body .conversation.section .conversation-view .comment-list')[0], {
                childList: true, attributes: false, attributeOldValue: false, characterData: false, characterDataOldValue: false, subtree: false
            });
            _urCommentsObserver.isObserving = true;
        }
        if (!(await isPanelReady('loadingMoreInfo', 10, 200)).error) {
            $domElement = await getDomElement('#panel-container .mapUpdateRequest .top-section .body .more-info');
            if (_settings.expandMoreInfo && W.map.panelRegion.currentView.model.attributes.moreInfoAvailable && $domElement.hasClass('collapsed')) {
                $domElement = await getDomElement('#panel-container .mapUpdateRequest .top-section .body .more-info .title');
                $domElement.click();
            }
            else if (!_settings.expandMoreInfo && !$domElement.hasClass('collapsed')) {
                $domElement = await getDomElement('#panel-container .mapUpdateRequest .top-section .body .more-info .title');
                $domElement.click();
            }
        }
        $domElement = await getDomElement('#panel-container .mapUpdateRequest .top-section .body .more-info .title');
        $domElement.on('click', function () {
            if (this.parentElement.classList.contains('collapsed'))
                _settings.expandMoreInfo = true;
            else
                _settings.expandMoreInfo = false;
            saveSettingsToStorage();
        });
        $domElement = await getDomElement('#panel-container .mapUpdateRequest .top-section .body .conversation');
        if ($domElement?.hasClass('collapsed')) {
            $domElement = await getDomElement('#panel-container .mapUpdateRequest .top-section .body .conversation .title');
            $domElement.click();
        }
        $domElement = await getDomElement('#panel-container .mapUpdateRequest .top-section');
        if ($domElement)
            $domElement.scrollTop($domElement[0].scrollHeight);
        autoScrollComments(_mapUpdateRequests[_selUr.urId].urceData.commentCount, 10, 1);
        if (_mapUpdateRequests[_selUr.urId].urceData.commentCount === 0) {
            if (_settings.autoZoomInOnNewUr)
                autoZoomIn();
            const { commentNum } = Object.values(_defaultComments).find((defaultComment) => defaultComment.urNum === W.model.mapUpdateRequests.objects[_selUr.urId].attributes.type);
            if (_selUr.urOpen && commentNum) {
                if (
                    ((_settings.perCommentListSettings[_currentCommentList].autoSetNewUrComment || (_restrictionsEnforce.autoSetNewUrComment === true))
                        && (_restrictionsEnforce.autoSetNewUrComment !== false)
                        && !W.model.mapUpdateRequests.objects[_selUr.urId].attributes.description)
                    || ((_settings.perCommentListSettings[_currentCommentList].autoSetNewUrCommentWithDescription || (_restrictionsEnforce.autoSetNewUrCommentWithDescription === true))
                        && (_restrictionsEnforce.autoSetNewUrCommentWithDescription !== false)
                        && W.model.mapUpdateRequests.objects[_selUr.urId].attributes.description
                        && (W.model.mapUpdateRequests.objects[_selUr.urId].attributes.type !== 23))
                    || ((_settings.perCommentListSettings[_currentCommentList].autoSetNewUrCommentSlur || (_restrictionsEnforce.autoSetNewUrCommentSlur === true))
                        && (_restrictionsEnforce.autoSetNewUrCommentSlur !== false)
                        && W.model.mapUpdateRequests.objects[_selUr.urId].attributes.type === 23)
                ) {
                    if (_settings.autoClickOpenSolvedNi)
                        autoClickOpenSolvedNi(commentNum);
                    postUrComment(_commentList[commentNum].comment, false);
                }
            }
        }
        else if (_mapUpdateRequests[_selUr.urId].urceData.commentCount === 1) {
            if (_selUr.urOpen
                && (_settings.perCommentListSettings[_currentCommentList].autoSetReminderUrComment || (_restrictionsEnforce.autoSetReminderUrComment === true))
                && (_restrictionsEnforce.autoSetReminderUrComment !== false)
                && _defaultComments.dr.commentNum
                && (_mapUpdateRequests[_selUr.urId].urceData.commentCount > 0)
                && (_settings.perCommentListSettings[_currentCommentList].reminderDays !== 0)
                && (_restrictionsEnforce.reminderDays !== 0)
                && ((_mapUpdateRequests[_selUr.urId].urceData.lastCommentDaysOld > (_settings.perCommentListSettings[_currentCommentList].reminderDays - 1))
                    || (_mapUpdateRequests[_selUr.urId].urceData.lastCommentDaysOld > (_restrictionsEnforce.reminderDays - 1)))
                && (_mapUpdateRequests[_selUr.urId].urceData.lastCommentBy > 0)
            ) {
                if (_settings.autoClickOpenSolvedNi)
                    autoClickOpenSolvedNi(_defaultComments.dr.commentNum);
                postUrComment(_commentList[_defaultComments.dr.commentNum].comment, false);
            }
        }
        if (_settings.autoCenterOnUr)
            recenterOnUr({ data: { urId: _selUr.urId } });
        doSpinner('handleUpdateRequestContainer', false);
    }

    function checkValue() {
        const varsFound = this.value.match(/\B\$\S*\$\B/gm) || this.value.match(/(\$SELSEGS|\$USERNAME|\$URD)/gm);
        if (varsFound) {
            let title;
            if (this.value.indexOf('$SELSEGS') > -1)
                title = I18n.t('urce.prompts.SelSegsFound');
            else if (this.value.indexOf('$PLACE_ADDRESS$') > -1)
                title = I18n.t('urce.prompts.PlaceAddressFound');
            else if (this.value.indexOf('$PLACE_NAME$') > -1)
                title = I18n.t('urce.prompts.PlaceNameFound');
            else
                title = I18n.t('urce.prompts.VarFound').replace('$VARSFOUND$', varsFound.join(', '));
            $('#panel-container .mapUpdateRequest .top-section .body .conversation .new-comment-form .send-button').prop('disabled', true).attr('title', title);
        }
        else if ($('#panel-container .mapUpdateRequest .top-section .body .conversation .new-comment-form .send-button').prop('disabled') === true) {
            $('#panel-container .mapUpdateRequest .top-section .body .conversation .new-comment-form .send-button').prop('disabled', false).attr('title', '');
        }
    }

    async function handleClickedComment(commentNum, doubleClick) {
        _selUr.doubleClick = doubleClick;
        const $domElement = await getDomElement('textarea[id=id]', '#panel-container .mapUpdateRequest .top-section .body .conversation .new-comment-text');
        if (!$domElement) {
            logWarning('No comment box found after clicking a comment from the list.');
            WazeWrap.Alerts.info(_SCRIPT_SHORT_NAME, I18n.t('urce.prompts.NoCommentBox'));
            return;
        }
        if (doubleClick)
            $domElement.off('blur', autoClickSendButton).on('blur', autoClickSendButton);
        if (_settings.autoClickOpenSolvedNi && _selUr.urOpen)
            autoClickOpenSolvedNi(commentNum);
        postUrComment(_commentList[commentNum].comment, doubleClick);
    }

    function autoSwitchToUrceTab() {
        _restoreDrawerTab = $('#drawer').find('[selected="true"]').children('.w-icon')[0] || 'none-selected';
        if (!$(_restoreDrawerTab).hasClass('w-icon-script')) {
            document.querySelector('.w-icon-script').dispatchEvent(new MouseEvent('click', {
                view: (typeof unsafeWindow !== 'undefined' ? unsafeWindow : window), bubbles: true, cancelable: true, button: 0
            }));
        }
        _restoreTab = _restoreTab || $('#user-tabs .nav-tabs .active > a').first()[0];
        _restoreTabPosition = _restoreTabPosition || $($('#user-info .tab-content')[0]).scrollTop();
        document.querySelector('img#urceIcon').closest('a[href^="#userscript"').dispatchEvent(new MouseEvent('click', {
            view: (typeof unsafeWindow !== 'undefined' ? unsafeWindow : window), bubbles: true, cancelable: true, button: 0
        }));
        document.querySelector('a[href="#panel-urce-comments"]').dispatchEvent(new MouseEvent('click', {
            view: (typeof unsafeWindow !== 'undefined' ? unsafeWindow : window), bubbles: true, cancelable: true, button: 0
        }));
        $($('#user-info .tab-content')[0]).scrollTop(0);
    }

    function autoSwitchToPrevTab() {
        if (!$(W.map.getOLMap().div).hasClass('problem-selected')) {
            if ($(_restoreDrawerTab).hasClass('w-icon-script')) {
                if ($(_restoreTab).length > 0) {
                    _restoreTab.dispatchEvent(new MouseEvent('click', {
                        view: (typeof unsafeWindow !== 'undefined' ? unsafeWindow : window), bubbles: true, cancelable: true, button: 0
                    }));
                    $($('#user-info .tab-content')[0]).scrollTop(_restoreTabPosition);
                }
            }
            else if (_restoreDrawerTab && _restoreDrawerTab !== 'none-selected') {
                _restoreDrawerTab.dispatchEvent(new MouseEvent('click', {
                    view: (typeof unsafeWindow !== 'undefined' ? unsafeWindow : window), bubbles: true, cancelable: true, button: 0
                }));
            }
            else if (_restoreDrawerTab === 'none-selected') {
                $('#drawer').find('.w-icon-script')[0].dispatchEvent(new MouseEvent('click', {
                    view: (typeof unsafeWindow !== 'undefined' ? unsafeWindow : window), bubbles: true, cancelable: true, button: 0
                }));
            }
            _restoreDrawerTab = undefined;
            _restoreTab = undefined;
            _restoreTabPosition = undefined;
        }
    }

    async function unfollowUrAfterSend() {
        if (!W.model.updateRequestSessions.objects[_selUr.urId]) {
            const data = await W.controller.descartesClient.getUpdateRequestSessionsByIds([_selUr.urId]);
            if (data.updateRequestSessions.objects.length > 0)
                W.model.mergeResponse(data);
            else
                return;
        }
        W.model.updateRequestSessions.objects[_selUr.urId].setFollowing('false');
    }

    async function autoCloseUrPanel() {
        const areCommentsLoaded = await new Promise((resolve) => {
            (function retry(tries, retryInt, maxNumTries) {
                checkTimeout({ timeout: 'autoCloseUrPanel' });
                if (tries > maxNumTries)
                    resolve({ error: true });
                else if (!$('#panel-container .mapUpdateRequest .top-section .body .problem-data .more-info').hasClass('loading'))
                    resolve({ error: false });
                else
                    _timeouts.autoCloseUrPanel = window.setTimeout(retry, ++tries, retryInt, maxNumTries);
            }(1, 10, 200));
        });
        if (areCommentsLoaded.error) {
            handleReadyError(false, false, '', false, '');
            return Promise.resolve();
        }
        W.map.panelRegion.currentView.destroy();
        return Promise.resolve();
    }

    async function autoClickSendButton() {
        let $domElement = await getDomElement('#panel-container .mapUpdateRequest .top-section .body .conversation .new-comment-form .send-button');
        if ($domElement)
            $domElement.trigger('click');
        $domElement = await getDomElement('textarea[id=id]', '#panel-container .mapUpdateRequest .top-section .body .conversation .new-comment-text');
        if ($domElement)
            $domElement.off('blur', autoClickSendButton);
    }

    async function autoClickOpenSolvedNi(commentNum) {
        const confirmHold = window.confirm;
        window.confirm = function () {
            // Dummy confirm to prevent WME from being able to send confirmations during auto clicking
            return true;
        };
        let $domElement = await getDomElement('#panel-container .mapUpdateRequest .top-section .body');
        if ($domElement)
            $domElement.scrollTop($domElement[0].scrollHeight);
        if ((_commentList[commentNum].urstatus === 'notidentified') && (_selUr.newStatus !== 'notidentified')) {
            $domElement = await getDomElement('#panel-container .mapUpdateRequest .actions .content .controls-container input[value="not-identified"]');
            if ($domElement)
                $domElement.trigger('click');
        }
        else if ((_commentList[commentNum].urstatus === 'solved') && (_selUr.newStatus !== 'solved')) {
            $domElement = await getDomElement('#panel-container .mapUpdateRequest .actions .content .controls-container input[value="solved"]');
            if ($domElement)
                $domElement.trigger('click');
        }
        else if ((_commentList[commentNum].urstatus === 'open') && ((_selUr.newStatus === 'solved') || (_selUr.newStatus === 'notidentified'))) {
            $domElement = await getDomElement('#panel-container .mapUpdateRequest .actions .content .controls-container input[value="open"]');
            if ($domElement)
                $domElement.trigger('click');
        }
        window.confirm = confirmHold;
    }

    function autoZoomIn() {
        if (W.map.getZoom() < 17)
            W.map.getOLMap().moveTo(W.map.updateRequestLayer.featureMarkers[_selUr.urId].marker.lonlat, 17);
    }

    function autoZoomOut() {
        if (_restoreZoom && !$(W.map.getOLMap().div).hasClass('problem-selected')) {
            if (_restoreZoom !== W.map.olMap.getZoom()) {
                W.map.getOLMap().zoomTo(_restoreZoom);
                _restoreZoom = null;
            }
        }
    }

    function convertTimeOfDayToCasual(hour) {
        let casualText;
        if ((hour > 20) || (hour < 4))
            casualText = I18n.t('urce.time.Night');
        else if ((hour > 3) && (hour < 12))
            casualText = I18n.t('urce.time.Morning');
        else if ((hour > 11) && (hour < 18))
            casualText = I18n.t('urce.time.Afternoon');
        else if ((hour > 17) && (hour < 21))
            casualText = I18n.t('urce.time.Evening');
        else
            casualText = '';
        return casualText;
    }

    function formatText(text = '', replaceVars = false, shortcutClicked = false, urId = -1) {
        if (!(urId > 0) && _selUr?.urId)
            ({ urId } = _selUr);
        if (replaceVars && shortcutClicked && (text.indexOf('$SELSEGS') > -1)) {
            const selFeatures = W.selectionManager.getSelectedDataModelObjects();
            let output = '';
            if ((selFeatures.length > 0) && (selFeatures.length < 3)) {
                let street1Name = '',
                    street2Name = '',
                    cityName = '',
                    firstCityId;
                for (let idx = 0; idx < selFeatures.length; idx++) {
                    if (selFeatures[idx].type === 'segment') {
                        if (selFeatures.length > 1) {
                            const streetObj = W.model.streets.getObjectById(selFeatures[idx].attributes.primaryStreetID);
                            if (idx === 0) {
                                street1Name = (streetObj.name?.length > 0) ? streetObj.name : I18n.t('urce.tools.UnknownRoadName');
                                firstCityId = streetObj.cityID;
                            }
                            else {
                                street2Name = (streetObj.name?.length > 0) ? streetObj.name : I18n.t('urce.tools.UnknownRoadName');
                                if ((firstCityId !== 999940) && (text.indexOf('$SELSEGS_WITH_CITY$') > -1)) {
                                    if ((firstCityId === streetObj.cityID) || (streetObj.cityID === 999940)) {
                                        const cityObj = W.model.cities.getObjectById(firstCityId);
                                        cityName = (cityObj.attributes.name !== '') ? cityObj.attributes.name : '';
                                    }
                                }
                                else if ((firstCityId === 999940) && (streetObj.cityID !== 999940) && (text.indexOf('$SELSEGS_WITH_CITY$') > -1)) {
                                    const cityObj = W.model.cities.getObjectById(streetObj.cityID);
                                    cityName = (cityObj.attributes.name !== '') ? cityObj.attributes.name : '';
                                }
                            }
                        }
                        else {
                            const streetObj = W.model.streets.getObjectById(selFeatures[idx].attributes.primaryStreetID);
                            street1Name = (streetObj.name?.length > 0) ? streetObj.name : '';
                            const cityObj = W.model.cities.getObjectById(streetObj.cityID);
                            if ((cityObj.attributes?.name !== '') && (street1Name !== '') && (text.indexOf('$SELSEGS_WITH_CITY$') > -1))
                                cityName = cityObj.attributes.name;
                        }
                    }
                }
                if ((street1Name !== '') || (street2Name !== '')) {
                    if (street2Name !== '') {
                        if (cityName !== '')
                            output = I18n.t('urce.tools.IntersectionOfWithCity');
                        else
                            output = I18n.t('urce.tools.IntersectionOf');
                    }
                    else if (cityName !== '') {
                        output = I18n.t('urce.tools.SegmentWithCity');
                    }
                    else {
                        output = '$SEG1NAME$';
                    }
                    output = output.replace('$SEGCITY$', cityName).replace('$SEG1NAME$', street1Name).replace('$SEG2NAME$', street2Name);
                    text = text.replace(/\$SELSEGS(\$|_WITH_CITY\$)?/gm, output);
                }
                else {
                    WazeWrap.Alerts.error(_SCRIPT_SHORT_NAME, I18n.t('urce.prompts.SelSegsInsertError'));
                }
            }
            else {
                WazeWrap.Alerts.error(_SCRIPT_SHORT_NAME, I18n.t('urce.prompts.SelSegsInsertError'));
            }
        }
        if (replaceVars && (urId > -1)) {
            if (/(\$(CURRENTDATE_DAY_OF_WEEK|CURRENTDATE_DATE|CURRENTDATE_DATE_CASUAL|CURRENTDATE_TIME|CURRENTDATE_TIME_CASUAL)\$)/gm.test(text)) {
                if (text.indexOf('$CURRENTDATE_DAY_OF_WEEK$') > -1)
                    text = text.replace('$CURRENTDATE_DAY_OF_WEEK$', new Date().toLocaleDateString(I18n.currentLocale(), { weekday: 'long' }));
                if (text.indexOf('$CURRENTDATE_DATE$') > -1)
                    text = text.replace('$CURRENTDATE_DATE$', new Date().toLocaleDateString(I18n.currentLocale(), { month: '2-digit', day: '2-digit', year: 'numeric' }));
                if (text.indexOf('$CURRENTDATE_DATE_CASUAL$') > -1)
                    text = text.replace('$CURRENTDATE_DATE_CASUAL$', new Date().toLocaleDateString(I18n.currentLocale(), { month: 'long', day: '2-digit' }));
                if (text.indexOf('$CURRENTDATE_TIME$') > -1)
                    text = text.replace('$CURRENTDATE_TIME$', new Date().toLocaleDateString(I18n.currentLocale(), { hour: '2-digit', minute: '2-digit', timeZoneName: 'short' }));
                if (text.indexOf('$CURRENTDATE_TIME_CASUAL$') > -1)
                    text = text.replace('$CURRENTDATE_TIME_CASUAL$', convertTimeOfDayToCasual(new Date().getHours()));
            }
            if (/(\$(DRIVEDATE_DAY_OF_WEEK|DRIVEDATE_DATE|DRIVEDATE_DATE_CASUAL|DRIVEDATE_DAYS_AGO|DRIVEDATE_TIME|DRIVEDATE_TIME_CASUAL|DRIVEDATE_TIME_CASUALMODE)\$)/gm.test(text)) {
                if (W.model.mapUpdateRequests.objects[urId]) {
                    if (text.indexOf('$DRIVEDATE_DAY_OF_WEEK$') > -1) {
                        if (W.model.mapUpdateRequests.objects[urId]?.attributes.driveDate > -1) {
                            text = text.replace(
                                '$DRIVEDATE_DAY_OF_WEEK$',
                                new Date(W.model.mapUpdateRequests.objects[urId].attributes.driveDate).toLocaleDateString(I18n.currentLocale(), { weekday: 'long' })
                            );
                        }
                        else {
                            text = text.replace('$DRIVEDATE_DAY_OF_WEEK$', '');
                        }
                    }
                    if (text.indexOf('$DRIVEDATE_DATE$') > -1) {
                        if (W.model.mapUpdateRequests.objects[urId]?.attributes.driveDate > -1) {
                            text = text.replace(
                                '$DRIVEDATE_DATE$',
                                new Date(W.model.mapUpdateRequests.objects[urId].attributes.driveDate).toLocaleDateString(
                                    I18n.currentLocale(),
                                    { month: '2-digit', day: '2-digit', year: 'numeric' }
                                )
                            );
                        }
                        else {
                            text = text.replace('$DRIVEDATE_DATE$', '');
                        }
                    }
                    if (text.indexOf('$DRIVEDATE_DATE_CASUAL$') > -1) {
                        if (W.model.mapUpdateRequests.objects[urId]?.attributes.driveDate > -1) {
                            text = text.replace(
                                '$DRIVEDATE_DATE_CASUAL$',
                                new Date(W.model.mapUpdateRequests.objects[urId].attributes.driveDate).toLocaleDateString(I18n.currentLocale(), { month: 'long', day: '2-digit' })
                            );
                        }
                        else {
                            text = text.replace('$DRIVEDATE_DATE_CASUAL$', '');
                        }
                    }
                    if (text.indexOf('$DRIVEDATE_DAYS_AGO$') > -1) {
                        if (_mapUpdateRequests[urId]?.urceData?.driveDaysOld > -1)
                            text = text.replace('$DRIVEDATE_DAYS_AGO$', parseDaysAgo(_mapUpdateRequests[urId].urceData.driveDaysOld));
                        else
                            text = text.replace('$DRIVEDATE_DAYS_AGO$', '');
                    }
                    if (text.indexOf('$DRIVEDATE_TIME$') > -1) {
                        if (W.model.mapUpdateRequests.objects[urId]?.attributes.driveDate > -1) {
                            text = text.replace(
                                '$DRIVEDATE_TIME$',
                                new Date(W.model.mapUpdateRequests.objects[urId].attributes.driveDate).toLocaleTimeString(
                                    I18n.currentLocale(),
                                    { hour: '2-digit', minute: '2-digit', timeZoneName: 'short' }
                                )
                            );
                        }
                        else {
                            text = text.replace('$DRIVEDATE_TIME$', '');
                        }
                    }
                    if (text.indexOf('$DRIVEDATE_TIME_CASUAL$') > -1) {
                        if (W.model.mapUpdateRequests.objects[urId]?.attributes.driveDate > -1)
                            text = text.replace('$DRIVEDATE_TIME_CASUAL$', convertTimeOfDayToCasual(new Date(W.model.mapUpdateRequests.objects[urId].attributes.driveDate).getHours()));
                        else
                            text = text.replace('$DRIVEDATE_TIME_CASUAL$', '');
                    }
                    if (text.indexOf('$DRIVEDATE_TIME_CASUALMODE$') > -1) {
                        if (_mapUpdateRequests[urId]?.urceData) {
                            const driveDaysAgo = _mapUpdateRequests[urId].urceData.driveDaysOld,
                                driveHour = new Date(W.model.mapUpdateRequests.objects[urId].attributes.driveDate).getHours();
                            let dayOfWeek = new Date(W.model.mapUpdateRequests.objects[urId].attributes.driveDate).getDay(),
                                casualText;
                            if ((driveDaysAgo < 21) && (driveHour > -1) && (driveHour < 4))
                                dayOfWeek = (dayOfWeek > 0) ? dayOfWeek - 1 : 6;
                            if (driveDaysAgo === 0) {
                                if ((driveHour > 20) && (driveHour < 25))
                                    casualText = I18n.t('urce.time.Tonight');
                                else if ((driveHour > -1) && (driveHour < 4))
                                    casualText = I18n.t('urce.time.LastNight');
                                else
                                    casualText = I18n.t('urce.time.ThisCasualTOD').replace('$THIS_CASUAL_TOD$', convertTimeOfDayToCasual(driveHour));
                            }
                            else if (driveDaysAgo === 1) {
                                if ((driveHour > 20) && (driveHour < 25))
                                    casualText = I18n.t('urce.time.LastNight');
                                else if ((driveHour > -1) && (driveHour < 4))
                                    casualText = I18n.t('urce.time.ThisWeekTOD').replace('$CASUAL_TOD$', convertTimeOfDayToCasual(driveHour));
                                else
                                    casualText = I18n.t('urce.time.YesterdayCasualTOD').replace('$YESTERDAY_CASUAL_TOD$', convertTimeOfDayToCasual(driveHour));
                                casualText = casualText.replace('$DAY_NAME$', I18n.translations[I18n.currentLocale()].date.day_names[dayOfWeek]);
                            }
                            else if ((driveDaysAgo > 1) && (driveDaysAgo < 21)) {
                                if ((driveDaysAgo > 1) && (driveDaysAgo < 7)) {
                                    if ((driveDaysAgo === 6) && (driveHour > -1) && (driveHour < 4))
                                        casualText = I18n.t('urce.time.LastWeekTOD').replace('$CASUAL_TOD$', convertTimeOfDayToCasual(driveHour));
                                    else
                                        casualText = I18n.t('urce.time.ThisWeekTOD').replace('$CASUAL_TOD$', convertTimeOfDayToCasual(driveHour));
                                }
                                else if ((driveDaysAgo > 6) && (driveDaysAgo < 14)) {
                                    casualText = I18n.t('urce.time.LastWeekTOD').replace('$CASUAL_TOD$', convertTimeOfDayToCasual(driveHour));
                                }
                                else if ((driveDaysAgo > 13) && (driveDaysAgo < 21)) {
                                    casualText = I18n.t('urce.time.TwoWeeksAgo');
                                }
                                casualText = casualText.replace('$DAY_NAME$', I18n.translations[I18n.currentLocale()].date.day_names[dayOfWeek]);
                            }
                            else if ((driveDaysAgo > 20) && (driveDaysAgo < 28)) {
                                casualText = I18n.t('urce.time.ThreeWeeksAgo');
                            }
                            else if ((driveDaysAgo > 27) && (driveDaysAgo < 61)) {
                                casualText = I18n.t('urce.time.AFewWeeksAgo');
                            }
                            else if ((driveDaysAgo > 60) && (driveDaysAgo < 121)) {
                                casualText = I18n.t('urce.time.ACoupleMonthsAgo');
                            }
                            else if (driveDaysAgo > 120) {
                                casualText = I18n.t('urce.time.AWhileBack');
                            }
                            else {
                                casualText = '';
                            }
                            text = text.replace('$DRIVEDATE_TIME_CASUALMODE$', casualText);
                        }
                        else {
                            text = text.replace('$DRIVEDATE_TIME_CASUALMODE$', '');
                        }
                    }
                }
            }
            if (text.indexOf('$URD') > -1) {
                if (W.model.mapUpdateRequests.objects[urId]?.attributes.description)
                    text = text.replace(/("?\$URD\$?"?)+/gmi, `"${W.model.mapUpdateRequests.objects[urId].attributes.description}"`).replace(/\n+/gmi, '');
                else
                    text = text.replace(/("?\$URD\$?"?)+/gmi, '');
            }
            if (text.indexOf('$CUSTOMTAGLINE$') > -1) {
                if (_settings.perCommentListSettings[_currentCommentList].customTagline.length > 0)
                    text = text.replace('$CUSTOMTAGLINE$', formatText(_settings.perCommentListSettings[_currentCommentList].customTagline, true, shortcutClicked, urId));
                else
                    text = text.replace('$CUSTOMTAGLINE$', '');
            }
            if (text.indexOf('$URTYPE$') > -1)
                text = text.replace('$URTYPE$', W.model.mapUpdateRequests.objects[urId].attributes.typeText);
            if (text.indexOf('$PERMALINK$') > -1) {
                if (W.model.mapUpdateRequests.objects[urId]) {
                    const lonLat = WazeWrap.Geometry.ConvertTo4326(W.model.mapUpdateRequests.objects[urId].attributes.geometry.x, W.model.mapUpdateRequests.objects[urId].attributes.geometry.y),
                        urlParams = new URLSearchParams(window.location.search);
                    const urPermalink = `https://${document.location.hostname.replace(/beta/i, 'www')}${document.location.pathname}?${(urlParams.get('env') ? `env=${urlParams.get('env')}&` : '')}lon=${lonLat.lon.toFixed(5)}&lat=${lonLat.lat.toFixed(5)}&s=20489175039&zoomLevel=17&mapUpdateRequest=${urId}`;
                    text = text.replace('$PERMALINK$', urPermalink);
                }
                else {
                    text = text.replace('$PERMALINK$', '');
                }
            }
        }
        if (replaceVars && (text.indexOf('$CLOSED_NOR_EMAIL_TAG$') > -1)) {
            if ((_settings.perCommentListSettings[_currentCommentList].tagEmail.length > 0) && (W.loginManager.user.userName.length > 0))
                text = text.replace('$CLOSED_NOR_EMAIL_TAG$', `Since this report is closed, please send further correspondence to ${_settings.perCommentListSettings[_currentCommentList].tagEmail} and include ${W.loginManager.user.userName} in the subject line.`);
            else
                text = text.replace('$CLOSED_NOR_EMAIL_TAG$', '');
        }
        if (replaceVars && (text.indexOf('$TAG_EMAIL$') > -1)) {
            if (_settings.perCommentListSettings[_currentCommentList].tagEmail.length > 0)
                text = text.replace('$TAG_EMAIL$', _settings.perCommentListSettings[_currentCommentList].tagEmail);
            else
                text = text.replace('$TAG_EMAIL$', '');
        }
        if (replaceVars && (text.indexOf('$USERNAME') > -1)) {
            if (W.loginManager.user.userName)
                text = text.replace(/(\$USERNAME\$?)+/gmi, W.loginManager.user.userName);
            else
                text = text.replace(/(\$USERNAME\$?)+/gmi, '');
        }
        if (replaceVars && (text.indexOf('$PLACE_NAME$') > -1)) {
            const placeObj = W.selectionManager.getSelectedDataModelObjects()[0];
            if (placeObj?.type === 'venue') {
                if (placeObj.attributes.residential === true)
                    text = text.replace('$PLACE_NAME$', I18n.t('objects.venue.fields.residential'));
                else
                    text = text.replace('$PLACE_NAME$', ((placeObj.attributes.name.length > 0) ? placeObj.attributes.name : I18n.t('urce.tools.UnknownVenueName')));
            }
            else {
                WazeWrap.Alerts.error(_SCRIPT_SHORT_NAME, I18n.t('urce.prompts.PlaceNameInsertError'));
            }
        }
        if (replaceVars && (text.indexOf('$PLACE_ADDRESS$') > -1)) {
            const placeObj = W.selectionManager.getSelectedDataModelObjects()[0];
            if ((placeObj?.type === 'venue') && ((placeObj?.attributes.houseNumber.length > 0) || (placeObj?.attributes.streetID.length > 0))) {
                let placeAddress = '';
                if (placeObj.attributes.houseNumber.length > 0)
                    placeAddress += `${placeObj.attributes.houseNumber} `;
                if (placeObj.attributes.streetID > 0) {
                    const streetObj = W.model.streets.getObjectById(placeObj.attributes.streetID);
                    if (streetObj?.name !== '')
                        placeAddress += streetObj.name;
                    if ((streetObj.cityID > 0) && (streetObj.cityID !== 999940)) {
                        const cityObj = W.model.cities.getObjectById(streetObj.cityID);
                        placeAddress += `, ${cityObj.attributes.name}`;
                        if (cityObj.attributes.stateID > 0) {
                            const stateObj = W.model.states.getObjectById(cityObj.attributes.stateID);
                            if (stateObj.name !== '')
                                placeAddress += `, ${stateObj.name}`;
                        }
                    }
                }
                if (placeAddress !== '')
                    text = text.replace('$PLACE_ADDRESS$', placeAddress);
                else
                    WazeWrap.Alerts.error(_SCRIPT_SHORT_NAME, I18n.t('urce.prompts.PlaceAddressInsertError'));
            }
            else {
                WazeWrap.Alerts.error(_SCRIPT_SHORT_NAME, I18n.t('urce.prompts.PlaceAddressInsertError'));
            }
        }
        if (replaceVars && (_customReplaceVars?.length > 0)) {
            _customReplaceVars.forEach((customReplaceVar) => {
                if (text.indexOf(customReplaceVar.customVar) > -1)
                    text = text.replace(customReplaceVar.customVar, formatText(customReplaceVar.replaceText, true, shortcutClicked, urId));
            });
        }
        return text.replace(/\\[r|n]+/gm, '\n');
    }

    async function handleClickedShortcut(shortcut) {
        doSpinner('handleClickedShortcut', true);
        let $domElement = await getDomElement('textarea[id=id]', '#panel-container .mapUpdateRequest .top-section .body .conversation .new-comment-text');
        if (!$domElement) {
            handleReadyError(false, true, 'handleClickedShortcut', true, 'UR panel comment box is missing.');
            return;
        }
        const cursorPos = $domElement[0].selectionStart,
            currVal = $domElement.val();
        let newVal = currVal.slice(0, cursorPos),
            useCurrVal = false,
            replaceText;
        if (shortcut === 'selSegs') {
            if (/\$SELSEGS(\$|_WITH_CITY\$)?/.test(currVal)) {
                useCurrVal = true;
                replaceText = currVal;
            }
            else {
                replaceText = '$SELSEGS$';
            }
        }
        else if (shortcut === 'selSegsWithCity') {
            const checkVal = currVal.match(/(\$SELSEGS(\$|(?!_))|\$SELSEGS_WITH_CITY\$?)/);
            if (checkVal?.length > 0) {
                useCurrVal = true;
                replaceText = currVal.replace(checkVal[0], '$SELSEGS_WITH_CITY$');
            }
            else {
                replaceText = '$SELSEGS_WITH_CITY$';
            }
        }
        else if (shortcut === 'driveTime') {
            replaceText = '$DRIVEDATE_TIME$';
        }
        else if (shortcut === 'driveDayOfWeek') {
            replaceText = '$DRIVEDATE_DAY_OF_WEEK$';
        }
        else if (shortcut === 'driveDate') {
            replaceText = '$DRIVEDATE_DATE$';
        }
        else if (shortcut === 'description') {
            replaceText = '$URD$';
        }
        else if (shortcut === 'wazeUsername') {
            replaceText = '$USERNAME$';
        }
        else if (shortcut === 'driveTimeCasual') {
            replaceText = '$DRIVEDATE_TIME_CASUAL$';
        }
        else if (shortcut === 'driveDateCasual') {
            replaceText = '$DRIVEDATE_DATE_CASUAL$';
        }
        else if (shortcut === 'driveDateTimeCasualMode') {
            replaceText = '$DRIVEDATE_TIME_CASUALMODE$';
        }
        else if (shortcut === 'currentDayOfWeek') {
            replaceText = '$CURRENTDATE_DAY_OF_WEEK$';
        }
        else if (shortcut === 'currentDate') {
            replaceText = '$CURRENTDATE_DATE$';
        }
        else if (shortcut === 'currentDateCasual') {
            replaceText = '$CURRENTDATE_DATE_CASUAL$';
        }
        else if (shortcut === 'currentTime') {
            replaceText = '$CURRENTDATE_TIME$';
        }
        else if (shortcut === 'currentTimeCasual') {
            replaceText = '$CURRENTDATE_TIME_CASUAL$';
        }
        else if (shortcut === 'urType') {
            replaceText = '$URTYPE$';
        }
        else if (shortcut === 'customTagline') {
            replaceText = '$CUSTOMTAGLINE$';
        }
        else if (shortcut === 'placeName') {
            if (/\$PLACE_NAME\$/.test(currVal)) {
                useCurrVal = true;
                replaceText = currVal;
            }
            else {
                replaceText = '$PLACE_NAME$';
            }
        }
        else if (shortcut === 'placeAddress') {
            if (/\$PLACE_ADDRESS\$/.test(currVal)) {
                useCurrVal = true;
                replaceText = currVal;
            }
            else {
                replaceText = '$PLACE_ADDRESS$';
            }
        }
        else if (shortcut === 'urPermalink') {
            replaceText = '$PERMALINK$';
        }
        else {
            doSpinner('handleClickedShortcut', false);
            return;
        }
        let outputText = formatText(replaceText, true, true, -1);
        if ((((shortcut === 'selSegs') || (shortcut === 'selSegsWithCity')) && /\$SELSEGS\$?/gm.test(outputText))
            || ((shortcut === 'placeName') && (outputText.indexOf('$PLACE_NAME$') > -1))
            || ((shortcut === 'placeAddress') && (outputText.indexOf('$PLACE_ADDRESS$') > -1))
        ) {
            doSpinner('handleClickedShortcut', false);
            return;
        }
        if (outputText && !useCurrVal) {
            if ((newVal.length > 0) && !/\s/.test(newVal.slice(-1)))
                outputText = ` ${outputText}`;
            if (currVal.slice(cursorPos).length > 0) {
                if (!/[\t\f\v ]/.test(currVal.substr(cursorPos, 1)))
                    outputText = `${outputText} `;
            }
            newVal = `${newVal}${outputText}${currVal.slice(cursorPos)}`;
            if (newVal.length > 2000) {
                WazeWrap.Alerts.error(_SCRIPT_SHORT_NAME, I18n.t('urce.prompts.CommentTooLong'));
                doSpinner('handleClickedShortcut', false);
                return;
            }
            $domElement = await getDomElement('textarea[id=id]', '#panel-container .mapUpdateRequest .top-section .body .conversation .new-comment-text');
            if (!$domElement)
                logWarning('Timed out waiting for DOM elements before setting value of comment box after clicking a shortcut with selectRange.');
            else
                $domElement.val(newVal).selectRange((cursorPos + outputText.length));
            $domElement = await getDomElement('textarea[id=id]', '#panel-container .mapUpdateRequest .top-section .body .conversation .new-comment-text');
            if (!$domElement)
                logWarning('Timed out waiting for DOM elements before dispatching input event after clicking a shortcut with selectRange.');
            else
                $domElement[0].dispatchEvent(new Event('input'));
        }
        else if (useCurrVal) {
            $domElement = await getDomElement('textarea[id=id]', '#panel-container .mapUpdateRequest .top-section .body .conversation .new-comment-text');
            if (!$domElement)
                logWarning('Timed out waiting for DOM elements before setting value of comment box after clicking a shortcut without selectRange.');
            else
                $domElement.val(outputText).selectRange((cursorPos + outputText.length));
            $domElement = await getDomElement('textarea[id=id]', '#panel-container .mapUpdateRequest .top-section .body .conversation .new-comment-text');
            if (!$domElement)
                logWarning('Timed out waiting for DOM elements before dispatching input event after clicking a shortcut without selectRange.');
            else
                $domElement[0].dispatchEvent(new Event('input'));
        }
        if ((outputText && !useCurrVal) || useCurrVal) {
            $domElement = await getDomElement('textarea[id=id]', '#panel-container .mapUpdateRequest .top-section .body .conversation .new-comment-text');
            if (!$domElement)
                logWarning('Timed out waiting for DOM elements before triggering keyup and focus events after clicking a shortcut.');
            else
                $domElement.keyup().focus();
        }
        doSpinner('handleClickedShortcut', false);
    }

    async function autoPostReminderComment(urId) {
        if (W.map.getZoom() < 10)
            return Promise.resolve({ error: true, message: 'zoomIn' });
        const comment = formatText(_commentList[_defaultComments.dr.commentNum].comment, true, false, urId);
        try {
            if (/\B\$\S*\$\B/gm.test(comment) || /(\$SELSEGS|\$USERNAME|\$URD)/gm.test(comment))
                throw new Error(`Did not auto-post reminder comment for urId ${urId} because a variable was not replaced.`);
            if (!W.model.updateRequestSessions.objects[urId]) {
                const data = await W.controller.descartesClient.getUpdateRequestSessionsByIds([urId]);
                if (data.updateRequestSessions.objects.length > 0)
                    W.model.mergeResponse(data);
                else
                    throw new Error(`Failed to merge updateRequestSession for urId ${urId}`);
            }
            W.model.updateRequestSessions.objects[urId].addComment(comment);
            return Promise.resolve({ error: false });
        }
        catch (error) {
            return Promise.resolve({ error: true, message: error });
        }
    }

    async function postUrComment(comment, doubleClick) {
        doSpinner('postUrComment', true);
        let commentOutput,
            cursorPos,
            newCursorPos,
            postNls = 0;
        let $domElement = await getDomElement('textarea[id=id]', '#panel-container .mapUpdateRequest .top-section .body .conversation .new-comment-text');
        if (!$domElement) {
            logError('UR panel comment box is missing at beginning of postUrComment function.');
            handleReadyError(false, true, 'postUrComment', true, 'UR panel comment box is missing.');
            return;
        }
        if (_settings.enableAppendMode && ($domElement.val() !== '') && !doubleClick) {
            cursorPos = $domElement[0].selectionStart;
            newCursorPos = cursorPos;
            const currVal = $domElement.val();
            let newVal = currVal.slice(0, cursorPos);
            if ((newVal.length > 0) && /[\n\r]/.test(newVal.slice(-1))) {
                if (!/[\n\r]/.test(newVal.slice(-2, -1))) {
                    newVal += '\n';
                    newCursorPos++;
                }
            }
            else {
                newVal += '\n\n';
                newCursorPos += 2;
            }
            newVal += formatText(comment, true, false, -1);
            if (currVal.slice(cursorPos).length > 0) {
                if (/[\n\r]/.test(currVal.substr(cursorPos, 1))) {
                    if (!/[\n\r]/.test(currVal.substr(cursorPos + 1, 1))) {
                        newVal += '\n';
                        postNls++;
                    }
                }
                else {
                    newVal += '\n\n';
                    postNls += 2;
                }
                newVal += currVal.slice(cursorPos);
            }
            commentOutput = newVal;
        }
        else {
            commentOutput = formatText(comment, true, false, -1);
        }
        if (commentOutput.length > 2000) {
            WazeWrap.Alerts.error(_SCRIPT_SHORT_NAME, I18n.t('urce.prompts.CommentTooLong'));
            logError(I18n.t('urce.prompts.CommentTooLong'));
        }
        else {
            if (cursorPos) {
                $domElement = await getDomElement('textarea[id=id]', '#panel-container .mapUpdateRequest .top-section .body .conversation .new-comment-text');
                if (!$domElement) {
                    logError('Timed out waiting on text box to set value with selectRange.');
                    handleReadyError(false, true, 'postUrComment', true, 'UR panel comment box is missing.');
                    return;
                }
                $domElement.val(commentOutput).selectRange((newCursorPos + comment.replace(/\\[r|n]+/gm, ' ').length + postNls));
                $domElement = await getDomElement('textarea[id=id]', '#panel-container .mapUpdateRequest .top-section .body .conversation .new-comment-text');
                if (!$domElement) {
                    logError('Timed out waiting on text box shadow root to dispatch event with selectRange.');
                    handleReadyError(false, true, 'postUrComment', true, 'UR panel comment box is missing.');
                    return;
                }
                $domElement[0].dispatchEvent(new Event('input'));
            }
            else {
                $domElement = await getDomElement('textarea[id=id]', '#panel-container .mapUpdateRequest .top-section .body .conversation .new-comment-text');
                if (!$domElement) {
                    logError('Timed out waiting on text box to set value without selectRange.');
                    handleReadyError(false, true, 'postUrComment', true, 'UR panel comment box is missing.');
                    return;
                }
                $domElement.val(commentOutput);
                $domElement = await getDomElement('textarea[id=id]', '#panel-container .mapUpdateRequest .top-section .body .conversation .new-comment-text');
                if (!$domElement) {
                    logError('Timed out waiting on text box shadow root to dispatch event without selectRange.');
                    handleReadyError(false, true, 'postUrComment', true, 'UR panel comment box is missing.');
                    return;
                }
                $domElement[0].dispatchEvent(new Event('input'));
            }
            if (doubleClick && !/\B\$\S*\$\B/gm.test(commentOutput) && !/(\$SELSEGS|\$USERNAME|\$URD)/gm.test(commentOutput)) {
                $domElement = await getDomElement('textarea[id=id]', '#panel-container .mapUpdateRequest .top-section .body .conversation .new-comment-text');
                if ($domElement)
                    $domElement.blur();
            }
            else {
                $domElement = await getDomElement('textarea[id=id]', '#panel-container .mapUpdateRequest .top-section .body .conversation .new-comment-text');
                if ($domElement)
                    $domElement.keyup().focus();
            }
        }
        doSpinner('postUrComment', false);
    }

    function getUsernameAndRank(userId) {
        let username,
            rank;
        if (W.model.users.objects[userId]) {
            username = W.model.users.objects[userId].userName || userId;
            rank = W.model.users.objects[userId].rank + 1;
        }
        else {
            username = userId;
            rank = '?';
        }
        return { username, rank };
    }

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

    function parseDaysAgo(days) {
        if (days === 0)
            return I18n.t('date.today');
        if (days === 1)
            return I18n.t('date.yesterday');
        return I18n.t('common.time.ago', { time: I18n.t('common.time.days', { days }) });
    }

    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;
    }

    class StackListObj {
        constructor(urId, x, y) {
            this.urId = parseInt(urId);
            this.x = parseInt(x);
            this.y = parseInt(y);
        }
    }

    function restackMarkers() {
        if (_markerStackArray.length === 0)
            return;
        let filter = true;
        if ((_settings.disableFilteringAboveZoom && (W.map.getZoom() < _settings.disableFilteringAboveZoomLevel))
            || (_settings.disableFilteringBelowZoom && (W.map.getZoom() > _settings.disableFilteringBelowZoomLevel))
        )
            filter = false;
        const markerMapCollection = { ...W.map.updateRequestLayer.featureMarkers };
        const markerModelCollection = { ...W.model.mapUpdateRequests.objects };
        if (markerMapCollection) {
            Object.keys(markerMapCollection).forEach((marker) => {
                if (markerMapCollection.hasOwnProperty(marker)) {
                    const markerMapObj = markerMapCollection[marker];
                    const markerModelObj = markerModelCollection[marker];
                    if (markerModelObj.attributes.geometry.urceRealX) {
                        markerModelObj.attributes.geometry.x = markerModelObj.attributes.geometry.urceRealX;
                        markerModelObj.attributes.geometry.y = markerModelObj.attributes.geometry.urceRealY;
                        delete (markerModelObj.attributes.geometry.urceRealX);
                        delete (markerModelObj.attributes.geometry.urceRealY);
                    }
                    if (!(filter
                            && _settings.enableUrceUrFiltering
                            && _mapUpdateRequests[marker]?.urceData?.hideUr
                            && (!((_selUr.urId === markerModelObj.attributes.id) && _settings.doNotHideSelectedUr))
                            && (!((markerModelObj.attributes.urceData.tagType !== -1) && _settings.doNotFilterTaggedUrs)))
                    ) {
                        if (markerMapObj.marker.icon.imageDiv.style.display === 'none')
                            $(markerMapObj.marker.icon.imageDiv).show();
                    }
                }
            });
            for (let idx = 0; idx < _markerStackArray.length; idx++) {
                if (markerMapCollection[_markerStackArray[idx].urId]) {
                    markerMapCollection[_markerStackArray[idx].urId].marker.icon.imageDiv.style.left = `${_markerStackArray[idx].x}px`;
                    markerMapCollection[_markerStackArray[idx].urId].marker.icon.imageDiv.style.top = `${_markerStackArray[idx].y}px`;
                }
            }
            _markerStackArray = [];
            _unstackedMasterId = null;
        }
    }

    function checkMarkerStacking(urId, unstackedX, unstackedY) {
        urId = parseInt(urId);
        if (!_settings.unstackMarkers || (isIdAlreadyUnstacked(urId) === true))
            return;
        doSpinner('checkMarkerStacking', true);
        const stackList = [],
            markerMapCollection = { ...W.map.updateRequestLayer.featureMarkers },
            markerModelCollection = { ...W.model.mapUpdateRequests.objects };
        let offset = 1000000000;
        stackList.push(urId);
        if (markerMapCollection) {
            Object.keys(markerMapCollection).forEach((marker) => {
                if (markerMapCollection.hasOwnProperty(marker)) {
                    if (!markerModelCollection[marker].attributes.geometry.urceRealX) {
                        markerModelCollection[marker].attributes.geometry.urceRealX = (markerModelCollection[marker].attributes.geometry.realX)
                            ? markerModelCollection[marker].attributes.geometry.realX
                            : markerModelCollection[marker].attributes.geometry.x;
                        markerModelCollection[marker].attributes.geometry.x = (markerModelCollection[marker].attributes.geometry.realX)
                            ? (markerModelCollection[marker].attributes.geometry.realX + offset)
                            : (markerModelCollection[marker].attributes.geometry.x + offset);
                        markerModelCollection[marker].attributes.geometry.urceRealY = (markerModelCollection[marker].attributes.geometry.realY)
                            ? markerModelCollection[marker].attributes.geometry.realY
                            : markerModelCollection[marker].attributes.geometry.y;
                        markerModelCollection[marker].attributes.geometry.y = (markerModelCollection[marker].attributes.geometry.realY)
                            ? (markerModelCollection[marker].attributes.geometry.realY + offset)
                            : (markerModelCollection[marker].attributes.geometry.y + offset);
                        offset += 1000;
                    }
                    if (!(markerMapCollection[marker].marker.icon.imageDiv.classList.contains('recently-closed') && (W.map.updateRequestLayer.showHidden === false))
                        && ((markerMapCollection[marker].marker.icon.imageDiv.style.display !== 'none') || (markerMapCollection[marker].marker.icon.imageDiv.style.visibility !== 'hidden'))
                    ) {
                        if (parseInt(marker) !== urId) {
                            const xDiff = unstackedX - parsePxString(markerMapCollection[marker].marker.icon.imageDiv.style.left),
                                yDiff = unstackedY - parsePxString(markerMapCollection[marker].marker.icon.imageDiv.style.top),
                                distSquared = ((xDiff * xDiff) + (yDiff * yDiff));
                            if (distSquared < (_settings.unstackSensitivity * _settings.unstackSensitivity))
                                stackList.push(parseInt(marker));
                        }
                    }
                }
            });
        }
        if (stackList.length > 0) {
            if (stackList.length === 1)
                logDebug('Single marker highlighted. Adjusting geometry properties to prevent recentering.');
            else if (W.map.getZoom() < _settings.unstackDisableAboveZoom)
                logDebug(`Zoom level is ${W.map.getZoom()} which is less than setting for disable above zoom of ${_settings.unstackDisableAboveZoom}. Adjusting geometry properties to prevent recentering.`);
            else
                logDebug(`${stackList.length} 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++) {
                    const thisUrId = stackList[idx],
                        x = parsePxString(markerMapCollection[thisUrId].marker.icon.imageDiv.style.left),
                        y = parsePxString(markerMapCollection[thisUrId].marker.icon.imageDiv.style.top);
                    _markerStackArray.push(new StackListObj(thisUrId, x, y));
                    if (!((W.map.getZoom() < _settings.unstackDisableAboveZoom) || (stackList.length === 1))) {
                        markerMapCollection[thisUrId].marker.icon.imageDiv.style.left = `${unstackedX}px`;
                        markerMapCollection[thisUrId].marker.icon.imageDiv.style.top = `${unstackedY}px`;
                        unstackedX += 10;
                        unstackedY -= 30;
                    }
                }
                if (!((W.map.getZoom() < _settings.unstackDisableAboveZoom) || (stackList.length === 1))) {
                    Object.keys(markerMapCollection).forEach((marker) => {
                        if (markerMapCollection.hasOwnProperty(marker)) {
                            if (!isIdAlreadyUnstacked(parseInt(marker)))
                                $(markerMapCollection[marker].marker.icon.imageDiv).hide();
                        }
                    });
                }
            }
        }
        else {
            restackMarkers();
        }
        doSpinner('checkMarkerStacking', false);
    }

    async function markerMouseOver() {
        if (_mouseIsDown)
            return;
        const popupDelayTime = (Date.now() + (_settings.urMarkerPopupDelay * 100));
        let popupX,
            popupY;
        if (this.className.indexOf('user-generated') > -1) {
            const markerId = parseInt(this.attributes['data-id'].value);
            if ((markerId > 0) && ((_mousedOverMarkerId !== markerId) || ($('#urceDiv').css('visibility') === 'hidden'))) {
                _mousedOverMarkerId = markerId;
                const targetTab = `_urceTab_${Math.round(Math.random() * 1000000)}`,
                    popupXOffset = parsePxString($('#sidebar').css('width')) + parsePxString($('#drawer').css('width')),
                    unstackedX = parsePxString(W.map.updateRequestLayer.featureMarkers[markerId].marker.icon.imageDiv.style.left),
                    unstackedY = parsePxString(W.map.updateRequestLayer.featureMarkers[markerId].marker.icon.imageDiv.style.top);
                checkMarkerStacking(markerId, unstackedX, unstackedY);
                if (!_settings.disableUrMarkerPopup) {
                    doSpinner('markerMouseDown', true);
                    if (!_mapUpdateRequests[markerId]?.urceData)
                        await updateUrceData(getMapUrsObjArr([markerId]));
                    popupX = unstackedX - parsePxString(W.map.segmentLayer.div.style.left) + popupXOffset + 10;
                    popupY = unstackedY - parsePxString(W.map.segmentLayer.div.style.top) + 10;
                    let popupContent = `<b>${I18n.t('problems.panel.titles.map_update_request')} (${markerId}): `
                        + `${I18n.t(`update_requests.types.${W.model.mapUpdateRequests.objects[markerId].attributes.type}`)}</b><br>`;
                    if (!W.model.mapUpdateRequests.objects[markerId].attributes.description)
                        popupContent += `<i>${I18n.t('urce.mouseOver.NoDescription')}</i>`;
                    else
                        popupContent += clickify(W.model.mapUpdateRequests.objects[markerId].attributes.description.replace(/((?:.{80}).*?)\s/gm, '$1\n'), '');
                    if (_mapUpdateRequests[markerId].urceData.driveDaysOld > -1) {
                        popupContent += `<br><i>${I18n.t('mte.edit.submitted')} ${parseDaysAgo(_mapUpdateRequests[markerId].urceData.driveDaysOld)} `;
                        if (W.model.mapUpdateRequests.objects[markerId].attributes.driveDate > -1) {
                            popupContent += `(${new Date(W.model.mapUpdateRequests.objects[markerId].attributes.driveDate).toLocaleDateString('en-us')}`
                                + ` ${new Date(W.model.mapUpdateRequests.objects[markerId].attributes.driveDate).toLocaleTimeString('en-us')}) `;
                        }
                        if (W.model.mapUpdateRequests.objects[markerId]?.attributes.guestUserName) {
                            popupContent += I18n.t('urce.mouseOver.ViaLivemap');
                            if (W.model.mapUpdateRequests.objects[markerId].attributes.guestUserName !== '')
                                popupContent += ` by ${W.model.mapUpdateRequests.objects[markerId].attributes.guestUserName.replace(/<\/?[^>]+(>|$)/g, '')}`;
                        }
                        popupContent += '</i>';
                    }
                    if ((W.model.mapUpdateRequests.objects[markerId].attributes.resolvedOn) && (_mapUpdateRequests[markerId].urceData.resolveDaysAgo > -1)) {
                        popupContent += `<br><i>${I18n.t('urce.urStatus.Closed')} ${parseDaysAgo(_mapUpdateRequests[markerId].urceData.resolveDaysAgo)} `
                            + `(${new Date(W.model.mapUpdateRequests.objects[markerId].attributes.resolvedOn).toLocaleDateString('en-us')}`
                            + ` ${new Date(W.model.mapUpdateRequests.objects[markerId].attributes.resolvedOn).toLocaleTimeString('en-us')})</i>`
                            + `<br><i>${I18n.t('urce.mouseOver.MarkedAs')} `;
                        if (W.model.mapUpdateRequests.objects[markerId].attributes.resolution === 0)
                            popupContent += I18n.t('venues.update_requests.panel.solved');
                        else if (W.model.mapUpdateRequests.objects[markerId].attributes.resolution === 1)
                            popupContent += I18n.t('urce.urStatus.NotIdentified');
                        else
                            popupContent += I18n.t('segment.direction.0');
                        if (W.model.mapUpdateRequests.objects[markerId].attributes.resolvedBy) {
                            popupContent += ` ${I18n.t('element_history.changed_by')} `
                                + `<a href="${W.Config.user_profile.url}${getUsernameAndRank(W.model.mapUpdateRequests.objects[markerId].attributes.resolvedBy).username}">`
                                + `${getUsernameAndRank(W.model.mapUpdateRequests.objects[markerId].attributes.resolvedBy).username}</a>`
                                + ` (${getUsernameAndRank(W.model.mapUpdateRequests.objects[markerId].attributes.resolvedBy).rank})`;
                        }
                        popupContent += '</i>';
                    }
                    popupContent += `<br><br>${_mapUpdateRequests[markerId].urceData.commentCount} ${I18n.t('urce.tabs.Comments')}`;
                    if (!_mapUpdateRequests[markerId].urceData.commentsByMe && (_mapUpdateRequests[markerId].urceData.commentCount > 0))
                        popupContent += ` (${I18n.t('urce.mouseOver.NoneByMe')})`;
                    if ((_mapUpdateRequests[markerId].urceData.commentCount > 0) && (_mapUpdateRequests[markerId].urceData.lastCommentDaysOld > -1))
                        popupContent += `, ${I18n.t('element_history.actions.default.UPDATE')} ${parseDaysAgo(_mapUpdateRequests[markerId].urceData.lastCommentDaysOld)}`;
                    if (_mapUpdateRequests[markerId].urceData.commentCount > 0) {
                        popupContent += `<br>${I18n.t('urce.mouseOver.FirstComment')}: ${parseDaysAgo(_mapUpdateRequests[markerId].urceData.firstCommentDaysOld)}`
                            + ` ${I18n.t('element_history.changed_by')} `;
                        if (_mapUpdateRequests[markerId].urceData.firstCommentBy === -1) {
                            popupContent += I18n.t('conversation.reporter');
                        }
                        else {
                            popupContent += `<a href="${W.Config.user_profile.url}${getUsernameAndRank(_mapUpdateRequests[markerId].urceData.firstCommentBy).username}">`
                                + `${getUsernameAndRank(_mapUpdateRequests[markerId].urceData.firstCommentBy).username}</a>`
                                + ` (${getUsernameAndRank(_mapUpdateRequests[markerId].urceData.firstCommentBy).rank})`;
                        }
                        if (_mapUpdateRequests[markerId].urceData.commentCount > 1) {
                            popupContent += `<br>${I18n.t('urce.mouseOver.LastComment')}: ${parseDaysAgo(_mapUpdateRequests[markerId].urceData.lastCommentDaysOld)}`
                                + ` ${I18n.t('element_history.changed_by')} `;
                            if (_mapUpdateRequests[markerId].urceData.lastCommentBy === -1) {
                                popupContent += I18n.t('conversation.reporter');
                            }
                            else {
                                popupContent += `<a href="${W.Config.user_profile.url}${getUsernameAndRank(_mapUpdateRequests[markerId].urceData.lastCommentBy).username}">`
                                    + `${getUsernameAndRank(_mapUpdateRequests[markerId].urceData.lastCommentBy).username}</a>`
                                    + ` (${getUsernameAndRank(_mapUpdateRequests[markerId].urceData.lastCommentBy).rank})`;
                            }
                        }
                        popupContent += `<br>${I18n.t('urce.mouseOver.ReporterHasCommented')}: <i>${(_mapUpdateRequests[markerId].urceData.reporterHasCommented) ? I18n.t('urce.common.Yes') : I18n.t('urce.common.No')}</i>`;
                    }
                    let x,
                        y;
                    if (W.model.mapUpdateRequests.objects[markerId].attributes.geometry.urceRealX)
                        x = W.model.mapUpdateRequests.objects[markerId].attributes.geometry.urceRealX;
                    else if (W.model.mapUpdateRequests.objects[markerId].attributes.geometry.realX)
                        x = W.model.mapUpdateRequests.objects[markerId].attributes.geometry.realX;
                    else
                        ({ x } = W.model.mapUpdateRequests.objects[markerId].attributes.geometry);
                    if (W.model.mapUpdateRequests.objects[markerId].attributes.geometry.urceRealY)
                        y = W.model.mapUpdateRequests.objects[markerId].attributes.geometry.urceRealY;
                    else if (W.model.mapUpdateRequests.objects[markerId].attributes.geometry.realY)
                        y = W.model.mapUpdateRequests.objects[markerId].attributes.geometry.realY;
                    else
                        ({ y } = W.model.mapUpdateRequests.objects[markerId].attributes.geometry);
                    const urPos = WazeWrap.Geometry.ConvertTo4326(x, y);
                    let urLink = $(document)[0].location.href;
                    urLink = `${urLink.substr(0, urLink.indexOf('?zoom'))}?zoomLevel=17&lat=${urPos.lat}&lon=${urPos.lon}&mapUpdateRequest=${markerId}`;
                    popupContent += `<hr><ul><li><a href="${urLink}" id="_urceOpenInNewTab" target="${targetTab}">${I18n.t('urce.mouseOver.OpenInNewTab')}</a> - `
                        + `<a href="#" id="_urceRecenterSession" data-id="${markerId}">${I18n.t('urce.mouseOver.CenterInCurrentTab')}</a>`;
                    let lmLink;
                    if ($('#livemap-link').length > 0)
                        lmLink = $('#livempa-link').href;
                    else if ($('.livemap-link').length > 0)
                        lmLink = $('.livemap-link')[0].href;
                    if (lmLink) {
                        lmLink = `${((lmLink.indexOf('?') > -1) ? lmLink.substr(0, lmLink.indexOf('?')) : lmLink)}?zoomLevel=17&lat=${urPos.lat}&lon=${urPos.lon}&layers=BTTTT`;
                        popupContent += `<li><a href="${lmLink}" target="${targetTab}_lmTab">${I18n.t('urce.mouseOver.OpenInNewLivemapTab')}</a>`;
                    }
                    const popupDelay = (Date.now() > popupDelayTime) ? -1 : (popupDelayTime - Date.now());
                    if (popupDelay < 0) {
                        handlePopup({
                            popupContent, popupX, popupY, urId: markerId
                        });
                    }
                    else {
                        checkTimeout({ timeout: 'popupDelay' });
                        _timeouts.popupDelay = window.setTimeout(handlePopup, popupDelay, {
                            popupContent, popupX, popupY, urId: markerId
                        });
                    }
                    doSpinner('markerMouseDown', false);
                }
            }
        }
    }

    function markerMouseOut(event) {
        const newUrId = (parseInt($(event?.relatedTarget).attr('data-id')) > -1) ? parseInt($(event.relatedTarget).attr('data-id')) : false;
        if (((newUrId > 0) && isIdAlreadyUnstacked(newUrId))
            || ((event?.relatedTarget?.id === 'urceDiv') || (event?.relatedTarget?.id?.indexOf('urceCounts') > -1) || (event?.relatedTarget?.parentNode?.id?.indexOf('urce') > -1))
        )
            return;
        if (!newUrId)
            _mousedOverMarkerId = null;
        hidePopup();
        restackMarkers();
    }

    function markerClick(event) {
        if (!_selUr.handling && _commentListLoaded && (!(_selUr.urId > 0) || (_selUr.urId !== parseInt(event.target.attributes['data-id'].value)))) {
            _selUr = {
                doubleClick: false,
                handling: false,
                newStatus: undefined,
                urId: parseInt(event.target.attributes['data-id'].value),
                urOpen: false
            };
            logDebug(`Clicked UR marker for urId: ${_selUr.urId}`);
        }
    }

    function handlePopup(popupObj) {
        if (_mousedOverMarkerId !== popupObj.urId)
            return;
        $('#urceDiv .urceDivContent').html(popupObj.popupContent);
        $('#urceDiv')
            .css({ height: 'auto', width: 'auto' })
            .off()
            .on('mouseleave', hidePopup)
            .on('mouseenter', () => {
                checkTimeout({ timeout: 'popup' });
                checkTimeout({ timeout: 'popupDelay' });
            })
            .on('dblclick', { doubleClick: true }, hidePopup);
        $('#_urceOpenInNewTab').off().on('mouseup', saveSettingsToStorage);
        $('#_urceRecenterSession').off().on('click', { urId: popupObj.urId }, recenterOnUr);
        let rw = parseInt($('#urceDiv')[0].clientWidth);
        if (rw > (window.innerWidth * 0.45)) {
            rw = (window.innerWidth * 0.45);
            $('#urceDiv').css({ width: `${rw}px` });
        }
        const rh = parseInt($('#urceDiv')[0].clientHeight);
        if ((popupObj.popupX + rw) > window.innerWidth)
            popupObj.popupX -= (rw + 20);
        if ((popupObj.popupY + rh) > window.innerHeight)
            popupObj.popupY -= (((popupObj.popupY + rh) - window.innerHeight) + 30);
        popupObj.popupX = (popupObj.popupX < 0) ? 0 : popupObj.popupX;
        popupObj.popupY = (popupObj.popupY < 0) ? 0 : popupObj.popupY;
        $('#urceDiv').css({ top: `${popupObj.popupY}px`, left: `${popupObj.popupX}px`, visibility: 'visible' });
        checkTimeout({ timeout: 'popup' });
        checkTimeout({ timeout: 'popupDelay' });
        if (_settings.urMarkerPopupTimeout > 0)
            _timeouts.popup = window.setTimeout(hidePopup, (_settings.urMarkerPopupTimeout * 1000));
    }

    function hidePopup(event) {
        const newUrId = (parseInt($(event?.relatedTarget).attr('data-id')) > -1) ? parseInt($(event.relatedTarget).attr('data-id')) : -1;
        checkTimeout({ timeout: 'popup' });
        if ($('#urceDiv').css('visibility') !== 'hidden')
            $('#urceDiv').css({ visibility: 'hidden' });
        $('#urceDiv').off();
        if (((newUrId > 0) && isIdAlreadyUnstacked(newUrId))
            || ((event?.relatedTarget?.id === 'urceDiv') || (event?.relatedTarget?.id?.indexOf('urceCounts') > -1) || (event?.relatedTarget?.parentNode?.id?.indexOf('urce') > -1))
        ) {
            if (!event?.data?.doubleClick)
                return;
        }
        if (!newUrId && (event?.type === 'mouseleave') && ((event?.target?.id === 'urceDiv') || (event?.target?.offsetParent?.id === 'urceDiv')))
            _mousedOverMarkerId = null;
        if (!_mousedOverMarkerId)
            restackMarkers();
    }

    async function openUrPanel(urId = -1, closeUrPanel = false) {
        if (!(urId > 0) && _selUr?.urId)
            ({ urId } = _selUr);
        if (!(urId > 0))
            return;
        if (closeUrPanel)
            await autoCloseUrPanel();
        const t = (_settings.replaceNextWithDoneButton)
            ? { showNext: false, nextButtonString: I18n.t('problems.panel.done') }
            : { showNext: true, nextButtonString: I18n.t('problems.panel.next') };
        if (urId !== _selUr.urId) {
            _selUr = {
                doubleClick: false,
                handling: false,
                newStatus: undefined,
                urId,
                urOpen: false
            };
        }
        W.reqres.request('problems:browse', $.extend(t, { problem: W.model.mapUpdateRequests.objects[urId] }));
    }

    function recenterOnUr(event) {
        const urId = event?.data?.urId || -1;
        if (urId < 0)
            return;
        if (this?.id === '_urceRecenterSession')
            openUrPanel(urId, true);
        hidePopup();
        W.map.setCenter(W.map.updateRequestLayer.featureMarkers[urId].marker.lonlat);
    }

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

    function removeCustomMarker(urId) {
        if ($(`#urceCustomMarker_${urId}`).length > 0) {
            $(`#urceCustomMarker_${urId}`).remove();
            return 'removed';
        }
        return false;
    }

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

    function renderCustomMarker(urId, urOpen, customType, $node) {
        // URO+ Alt Markers, courtesy of URO+. Thank you!
        const 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
            [
                '',
                '',
                '',
                ''
            ]
        ];
        let status = 'updated';
        if ($(`#urceCustomMarker_${urId}`).length === 0) {
            $($node).append(`<span id="urceCustomMarker_${urId}" style="position:absolute;pointer-events:none;top:-3px;left:-2px;"></span>`);
            status = 'added';
        }
        const variant = (!urOpen) ? 2 : 0;
        $(`#urceCustomMarker_${urId}`).empty().append(`<img src="${uroAltMarkers[getCustomMarkerIdx(customType)][variant]}">`);
        return status;
    }

    function getCustomMarkerIdx(customType) {
        if (customType === 0) // ROADWORKS
            return 1;
        if (customType === 1) // CONSTRUCTION
            return 1;
        if (customType === 2) // CLOSURE
            return 0;
        if (customType === 3) // EVENT
            return 4;
        if (customType === 4) // NOTE
            return 3;
        if (customType === 5) // WSLM
            return 5;
        if (customType === 6) // BOG
            return 11;
        if (customType === 7) // DIFFICULT
            return 12;
        if (customType === 98) // Native speed limit URs
            return 5;
        if (customType === 99) // custom text
            return 2;
        return -1;
    }

    function convertTagToCustomType(tag) {
        if (tag === 'ROADWORKS')
            return 0;
        if (tag === 'CONSTRUCTION')
            return 1;
        if (tag === 'CLOSURE')
            return 2;
        if (tag === 'EVENT')
            return 3;
        if (tag === 'NOTE')
            return 4;
        if (tag === 'WSLM')
            return 5;
        if ((tag === 'BOG') || (tag === 'BOTG'))
            return 6;
        if (tag === 'DIFFICULT')
            return 7;
        return -1;
    }

    function updateUrMapMarkers(mUrsObjArr, filter) {
        const closeDays = _restrictionsEnforce.closeDays || _settings.perCommentListSettings[_currentCommentList].closeDays || 7,
            markerChanges = {
                markers: {
                    hidden: [],
                    unhidden: [],
                    missing: []
                },
                pills: {
                    added: [],
                    updated: [],
                    removed: []
                },
                customMarkers: {
                    added: [],
                    updated: [],
                    removed: []
                }
            };
        mUrsObjArr.forEach((mUrObj) => {
            const $marker = $(`.map-problem.user-generated[data-id="${mUrObj.attributes.id}"]`),
                mUrObjUrceData = _mapUpdateRequests[mUrObj.attributes.id];
            if (($marker.length > 0) && mUrObjUrceData?.urceData) {
                if (!$marker.data('urce_hasListeners'))
                    $marker.on({ mouseover: markerMouseOver, mouseout: markerMouseOut, click: markerClick }).data('urce_hasListeners', true);
                if (filter
                    && _settings.enableUrceUrFiltering
                    && (mUrObjUrceData.urceData.hideUr || (_settings.hideOutsideEditableArea && !mUrObj.canEdit()))
                    && (!((_selUr.urId === mUrObj.attributes.id) && _settings.doNotHideSelectedUr))
                    && (!((mUrObjUrceData.urceData.tagType !== -1) && _settings.doNotFilterTaggedUrs))
                ) {
                    markerChanges.markers.hidden.push(mUrObj.attributes.id);
                    $marker.hide();
                }
                else {
                    if ($marker.css('display') === 'none') {
                        markerChanges.markers.unhidden.push(mUrObj.attributes.id);
                        $marker.show();
                    }
                    if (_settings.enableUrPillCounts || customMarkersEnabledCheck()) {
                        if (_settings.enableUrPillCounts) {
                            let tagContent = '',
                                tagOffset,
                                urCountBackground;
                            if (mUrObjUrceData.urceData.tagType !== -1)
                                urCountBackground = '#CCCCCC';
                            else if ((mUrObjUrceData.urceData.commentCount > 0) && (mUrObjUrceData.urceData.lastCommentDaysOld > (closeDays - 1)))
                                urCountBackground = '#FF8B8B';
                            else if (!mUrObjUrceData.urceData.commentsByMe && (mUrObjUrceData.urceData.lastCommentBy < 0) && (mUrObjUrceData.urceData.lastCommentDaysOld < closeDays))
                                urCountBackground = '#FFCC99';
                            else if (!mUrObjUrceData.urceData.commentsByMe && (mUrObjUrceData.urceData.lastCommentBy > -1) && (mUrObjUrceData.urceData.lastCommentDaysOld < closeDays))
                                urCountBackground = '#FFFF99';
                            else if (mUrObjUrceData.urceData.commentsByMe && (mUrObjUrceData.urceData.lastCommentBy < 0))
                                urCountBackground = '#79B5C7';
                            else if (_wmeUserId === mUrObjUrceData.urceData.lastCommentBy)
                                urCountBackground = '#FFFFFF';
                            else
                                urCountBackground = '#84FFFF';
                            if ((mUrObjUrceData.urceData.tagType !== -1) && _settings.doNotShowTagNameOnPill) {
                                if (mUrObjUrceData.urceData.commentCount > 0)
                                    tagContent += `${mUrObjUrceData.urceData.commentCount}c `;
                                tagContent += `${((mUrObjUrceData.urceData.lastCommentDaysOld !== -1) ? mUrObjUrceData.urceData.lastCommentDaysOld : mUrObjUrceData.urceData.driveDaysOld)}d`;
                                tagOffset = (tagContent.length < 3) ? 0 : Math.round(tagContent.length * 2.28);
                            }
                            else if (mUrObjUrceData.urceData.tagType !== -1) {
                                tagContent += mUrObjUrceData.urceData.tagType;
                                if (mUrObjUrceData.urceData.commentCount > 0)
                                    tagContent += ` ${mUrObjUrceData.urceData.commentCount}c`;
                                tagOffset = (tagContent.length < 10) ? Math.round(tagContent.length * 2.86) : Math.round(tagContent.length * 3.33);
                            }
                            else {
                                if (mUrObjUrceData.urceData.commentCount > 0)
                                    tagContent += `${mUrObjUrceData.urceData.commentCount}c `;
                                tagContent += `${((mUrObjUrceData.urceData.lastCommentDaysOld !== -1) ? mUrObjUrceData.urceData.lastCommentDaysOld : mUrObjUrceData.urceData.driveDaysOld)}d`;
                                tagOffset = (tagContent.length < 3) ? 0 : Math.round(tagContent.length * 2.28);
                            }
                            tagOffset = `-${tagOffset}px`;
                            if ($(`#urceCounters-${mUrObj.attributes.id}`).length > 0) {
                                markerChanges.pills.updated.push(mUrObj.attributes.id);
                                $(`#urceCounters-${mUrObj.attributes.id}`).remove();
                            }
                            else {
                                markerChanges.pills.added.push(mUrObj.attributes.id);
                            }
                            const $divContainer = $(`<div id="urceCounters-${mUrObj.attributes.id}" data-id="${mUrObj.attributes.id}" style="clear:both; margin-bottom:10px;"></div>`),
                                $divInner = $(`<div id="urceCounters-${mUrObj.attributes.id}-text" data-id="${mUrObj.attributes.id}" class="urceCountersPill urceCounts" style="background-color:${urCountBackground}; right:${tagOffset}">${tagContent}</div>`);
                            $divInner.data('model', $marker.data('model'));
                            $marker.append($divContainer.append($divInner));
                        }
                        else if ($(`#urceCounters-${mUrObj.attributes.id}`).length > 0) {
                            markerChanges.pills.removed.push(mUrObj.attributes.id);
                            $(`#urceCounters-${mUrObj.attributes.id}`).remove();
                        }
                        let status;
                        if (customMarkersEnabledCheck()) {
                            if (mUrObjUrceData.urceData.customType > -1)
                                status = addCustomMarker(mUrObj.attributes.id, mUrObj.attributes.open, mUrObjUrceData.urceData.customType, $marker);
                            else
                                status = removeCustomMarker(mUrObj.attributes.id);
                        }
                        else {
                            status = removeCustomMarker(mUrObj.attributes.id);
                        }
                        if (status)
                            markerChanges.customMarkers[status].push(mUrObj.attributes.id);
                    }
                    else {
                        if ($(`#urceCounters-${mUrObj.attributes.id}`).length > 0) {
                            markerChanges.pills.removed.push(mUrObj.attributes.id);
                            $(`#urceCounters-${mUrObj.attributes.id}`).remove();
                        }
                        removeCustomMarker(mUrObj.attributes.id);
                    }
                }
            }
        });
        if (markerChanges.markers.hidden.length > 0)
            logDebug(`Hid UR markers for UR(s): ${markerChanges.markers.hidden.join(', ')} (Total: ${markerChanges.markers.hidden.length})`);
        if (markerChanges.markers.unhidden.length > 0)
            logDebug(`Unhid UR markers for UR(s): ${markerChanges.markers.unhidden.join(', ')} (Total: ${markerChanges.markers.unhidden.length})`);
        if (markerChanges.markers.missing.length > 0)
            logDebug(`Missing UR markers for UR(s): ${markerChanges.markers.missing.join(', ')} (Total: ${markerChanges.markers.missing.length})`);
        if (markerChanges.pills.added.length > 0)
            logDebug(`Added marker pills for UR(s): ${markerChanges.pills.added.join(', ')} (Total: ${markerChanges.pills.added.length})`);
        if (markerChanges.pills.updated.length > 0)
            logDebug(`Updated marker pills for UR(s): ${markerChanges.pills.updated.join(', ')} (Total: ${markerChanges.pills.updated.length})`);
        if (markerChanges.pills.removed.length > 0)
            logDebug(`Updated marker pills for UR(s): ${markerChanges.pills.removed.join(', ')} (Total: ${markerChanges.pills.removed.length})`);
        if (markerChanges.customMarkers.added.length > 0)
            logDebug(`Added custom markers for UR(s): ${markerChanges.customMarkers.added.join(', ')} (Total: ${markerChanges.customMarkers.added.length})`);
        if (markerChanges.customMarkers.updated.length > 0)
            logDebug(`Updated custom markers for UR(s): ${markerChanges.customMarkers.updated.join(', ')} (Total: ${markerChanges.customMarkers.updated.length})`);
        if (markerChanges.customMarkers.removed.length > 0)
            logDebug(`Removed custom markers for UR(s): ${markerChanges.customMarkers.removed.join(', ')} (Total: ${markerChanges.customMarkers.removed.length})`);
    }

    async function updateUrceData(mUrsObjArr) {
        const updateMarkersArr = [];
        let reopenPanel = false;
        if (!mUrsObjArr)
            return Promise.resolve();
        const eg = W.map.getExtent().toGeometry(),
            processMUrObjs = [...mUrsObjArr],
            reminderDays = _restrictionsEnforce.reminderDays || _settings.perCommentListSettings[_currentCommentList].reminderDays || 0,
            closeDays = _restrictionsEnforce.closeDays || _settings.perCommentListSettings[_currentCommentList].closeDays || 7,
            tagRegex = /^.*?\[(ROADWORKS|CONSTRUCTION|CLOSURE|EVENT|NOTE|WSLM|BOG|BOTG|DIFFICULT)\].*$/gim,
            tagUsernameRegex = new RegExp(` ${W.loginManager.user.userName} `, 'gi'),
            getUserIdsToCheck = (usersToCheck) => {
                if (usersToCheck.length > 0) {
                    // eslint-disable-next-line array-callback-return, consistent-return
                    const userIdsArr = usersToCheck.split(',').map((username) => {
                        if (W.model.users.getByAttributes({ userName: username.trim() }).length > 0)
                            return W.model.users.getByAttributes({ userName: username.trim() })[0].id;
                    });
                    return userIdsArr;
                }
                return [];
            },
            checkCommentedUsers = (userIdsToCheck, userIds) => {
                let retVal = false;
                userIdsToCheck.forEach((userId) => {
                    if (userIds.indexOf(parseInt(userId)) > -1)
                        retVal = true;
                });
                return retVal;
            };
        while (processMUrObjs.length > 0) {
            const autoSentRemindersFor = [],
                // 2023.04.05.01: Lowered to 400 due to request header size limits experienced by ojlaw
                chunk = processMUrObjs.splice(0, 400),
                urIds = chunk.map((a) => a.attributes.id);
            let includingKeyword,
                keywordIncludingRegex,
                notIncludingKeyword,
                keywordNotIncludingRegex;
            logDebug(`Updating urceData for urIds: ${urIds.join(', ')} (Total Count: ${urIds.length})`);
            if (_settings.hideByKeywordIncludingKeyword.length > 0) {
                includingKeyword = _settings.hideByKeywordIncludingKeyword.trim(); // Make regex compat ... .replace(/[-[\]{}()*+?.,\\^$|#\s]/g, '\\$&');
                if (_settings.hideByKeywordCaseInsensitive)
                    keywordIncludingRegex = new RegExp(includingKeyword, 'gim');
                else
                    keywordIncludingRegex = new RegExp(includingKeyword, 'gm');
            }
            if (_settings.hideByKeywordNotIncludingKeyword.length > 0) {
                notIncludingKeyword = _settings.hideByKeywordNotIncludingKeyword.trim(); // Make regex compat ... .replace(/[-[\]{}()*+?.,\\^$|#\s]/g, '\\$&');
                if (_settings.hideByKeywordCaseInsensitive)
                    keywordNotIncludingRegex = new RegExp(notIncludingKeyword, 'gim');
                else
                    keywordNotIncludingRegex = new RegExp(notIncludingKeyword, 'gm');
            }
            // eslint-disable-next-line no-await-in-loop
            const urSessions = await getUpdateRequestSessions(urIds);
            for (let idx = 0; idx < chunk.length; idx++) {
                const urSessionsObj = urSessions[chunk[idx].attributes.id];
                if (urSessionsObj) {
                    let autoSendReminder = false;
                    let urceData = {
                        commentCount: urSessionsObj.comments.length,
                        commentsByMe: false,
                        commentUserIds: [],
                        containsSquareBrackets: false,
                        customType: -1,
                        driveDaysOld: (chunk[idx].attributes.driveDate) ? uroDateToDays(chunk[idx].attributes.driveDate) : -1,
                        firstCommentBy: -2,
                        firstCommentDaysOld: -1,
                        fullText: '',
                        hideByStatusClosedBy: false,
                        hideUr: false,
                        hideWithCommentBy: false,
                        hideWithoutCommentBy: false,
                        inMapExtent: eg.intersects(chunk[idx].attributes.geometry),
                        keywordIncluding: false,
                        keywordNotIncluding: false,
                        lastCommentBy: -2,
                        lastCommentDaysOld: -1,
                        needsClosed: false,
                        needsReminder: false,
                        reporterHasCommented: false,
                        resolveDaysAgo: (chunk[idx].attributes.resolvedOn) ? uroDateToDays(chunk[idx].attributes.resolvedOn) : undefined,
                        tagType: -1,
                        waiting: false
                    };
                    if (urceData.commentCount > 0) {
                        urceData.firstCommentDaysOld = uroDateToDays(urSessionsObj.comments[0].createdOn);
                        urceData.firstCommentBy = parseInt(urSessionsObj.comments[0].userID);
                        urceData.lastCommentDaysOld = uroDateToDays(urSessionsObj.comments[(urceData.commentCount - 1)].createdOn);
                        urceData.lastCommentBy = parseInt(urSessionsObj.comments[(urceData.commentCount - 1)].userID);
                        urceData.fullText += (chunk[idx].attributes.description) ? `${chunk[idx].attributes.description} ` : '';
                        for (let commentIdx = 0; commentIdx < urceData.commentCount; commentIdx++) {
                            urceData.fullText += `${urSessionsObj.comments[commentIdx].text} `;
                            urceData.commentUserIds.push(urSessionsObj.comments[commentIdx].userID);
                        }
                        if (/\[\s*\S+[\s\S]*\]/m.test(urceData.fullText))
                            urceData.containsSquareBrackets = true;
                        if (urceData.commentUserIds.indexOf(_wmeUserId) > -1)
                            urceData.commentsByMe = true;
                        if (urceData.commentUserIds.indexOf(-1) > -1)
                            urceData.reporterHasCommented = true;
                        if (chunk[idx].attributes.open && urceData.commentCount === 1) {
                            if ((reminderDays !== 0) && (_restrictionsEnforce.reminderDays !== 0) && (urceData.lastCommentDaysOld > (reminderDays - 1))) {
                                if ((_settings.perCommentListSettings[_currentCommentList].autoSendReminders || (_restrictionsEnforce.autoSendReminders === true))
                                    && (_restrictionsEnforce.autoSendReminders !== false)
                                    && _defaultComments.dr.commentNum
                                    && (urceData.lastCommentBy > 1)
                                    && (_wmeUserId === urceData.lastCommentBy)
                                    && urceData.inMapExtent
                                    && !urceData.containsSquareBrackets
                                    && ((W.loginManager.user.rank > 2) || ((parseInt(W.loginManager.user.rank) === 2) && W.loginManager.user.isAreaManager))
                                    && !_mapUpdateRequests[chunk[idx].attributes.id]?.urceData?.autoSentReminder
                                )
                                    autoSendReminder = true;
                                else
                                    urceData.needsReminder = true;
                            }
                            else if (((reminderDays === 0) || (_restrictionsEnforce.reminderDays === 0)) && (urceData.lastCommentDaysOld > (closeDays - 1))) {
                                urceData.needsClosed = true;
                            }
                            else {
                                urceData.waiting = true;
                            }
                        }
                        if (chunk[idx].attributes.open && (urceData.commentCount > 1)) {
                            if (urceData.lastCommentBy > 1) {
                                if ((closeDays > 0) && (urceData.lastCommentDaysOld > (closeDays - 1))) {
                                    if (_wmeUserId === urceData.lastCommentBy)
                                        urceData.needsClosed = true;
                                    else if (urceData.lastCommentDaysOld < (reminderDays + closeDays))
                                        urceData.waiting = true;
                                    else if (urceData.lastCommentDaysOld > (reminderDays + closeDays - 1))
                                        urceData.needsClosed = true;
                                }
                                else {
                                    urceData.waiting = true;
                                }
                            }
                        }
                    }
                    else {
                        urceData.fullText += (chunk[idx].attributes.description) ? chunk[idx].attributes.description : '';
                    }
                    if (!chunk[idx].attributes.open && _settings.hideByStatusClosedBy && (_settings.hideByStatusClosedByUsers.length > 0)) {
                        const userIdsToCheck = getUserIdsToCheck(_settings.hideByStatusClosedByUsers);
                        if ((userIdsToCheck.length > 0) && (userIdsToCheck.indexOf(chunk[idx].attributes.resolvedBy) > -1))
                            urceData.hideByStatusClosedBy = true;
                    }
                    if (_settings.hideWithCommentBy && (_settings.hideWithCommentByUsers.length > 0)) {
                        const userIdsToCheck = getUserIdsToCheck(_settings.hideWithCommentByUsers);
                        if ((userIdsToCheck.length > 0) && (urceData.commentUserIds.length > 0)) {
                            if (checkCommentedUsers(userIdsToCheck, urceData.commentUserIds) === true)
                                urceData.hideWithCommentBy = true;
                        }
                    }
                    if (_settings.hideWithoutCommentBy && (_settings.hideWithoutCommentByUsers.length > 0)) {
                        const userIdsToCheck = getUserIdsToCheck(_settings.hideWithoutCommentByUsers);
                        if ((userIdsToCheck.length > 0) && (urceData.commentUserIds.length > 0)) {
                            if (checkCommentedUsers(userIdsToCheck, urceData.commentUserIds) !== true)
                                urceData.hideWithoutCommentBy = true;
                        }
                        else if (urceData.commentUserIds.length === 0) {
                            urceData.hideWithoutCommentBy = true;
                        }
                    }
                    // eslint-disable-next-line no-control-regex
                    urceData.fullText = urceData.fullText.replace(/[\r\n\x0B\x0C\u0085\u2028\u2029]+/g, ' ');
                    urceData.tagType = tagRegex.test(urceData.fullText) ? urceData.fullText.replace(tagRegex, '$1') : -1;
                    if (urceData.tagType !== -1) {
                        urceData.customType = convertTagToCustomType(urceData.tagType);
                    }
                    else if (chunk[idx].attributes.type === 23) {
                        urceData.customType = 98;
                    }
                    else if (_settings.customMarkersCustom && (_settings.customMarkersCustomText.length > 0) && new RegExp(`\\[${_settings.customMarkersCustomText.replace(/[[\]]+/gim, '')}\\]`).test(urceData.fullText)) {
                        urceData.tagType = 99;
                        urceData.customType = 99;
                    }
                    else {
                        urceData.customType = -1;
                    }
                    if (keywordIncludingRegex && keywordIncludingRegex.test(urceData.fullText))
                        urceData.keywordIncluding = true;
                    if (keywordNotIncludingRegex && !keywordNotIncludingRegex.test(urceData.fullText))
                        urceData.keywordNotIncluding = true;
                    if ((urceData.tagType === -1) && (chunk[idx].attributes.type === 23))
                        urceData.customType = 98;
                    else if (urceData.tagType === -1)
                        urceData.customType = -1;
                    if ((urceData.tagType === -1) || _settings.replaceTagNameWithEditorName) {
                        if (_settings.replaceTagNameWithEditorName && tagUsernameRegex.test(urceData.fullText))
                            urceData.tagType = W.loginManager.user.userName;
                        else if (!urceData.tagType)
                            urceData.tagType = -1;
                    }
                    if (autoSendReminder) {
                        if (((urceData.customType > -1) && !_settings.perCommentListSettings[_currentCommentList].autoSendRemindersExceptTagged)
                            || (urceData.customType === -1)
                        ) {
                            try {
                                // eslint-disable-next-line no-await-in-loop
                                const autoPostReminderCommentResult = await autoPostReminderComment(chunk[idx].attributes.id);
                                if (autoPostReminderCommentResult.error) {
                                    if (autoPostReminderCommentResult.message === 'zoomIn')
                                        logDebug(`Did not auto post reminder comment due to zoom being less than 10 (Zoom: ${W.map.getZoom()}) for urId ${chunk[idx].attributes.id}.`);
                                    else
                                        logError(autoPostReminderCommentResult.message);
                                }
                                else {
                                    autoSentRemindersFor.push(chunk[idx].attributes.id);
                                    if (_settings.unfollowAfterSend)
                                        unfollowUrAfterSend(chunk[idx].attributes.id);
                                    urceData = $.extend({}, urceData, {
                                        autoSentReminder: true,
                                        commentCount: urceData.commentCount + 1,
                                        commentsByMe: true,
                                        lastCommentDaysOld: 0,
                                        lastCommentBy: _wmeUserId,
                                        needsReminder: false,
                                        waiting: true
                                    });
                                }
                                if (parseInt($('.update-requests .marker-selected').attr('data-id')) === _selUr.urId)
                                    reopenPanel = true;
                                updateMarkersArr.push(chunk[idx]);
                            }
                            catch (error) {
                                logWarning(error);
                                urceData.needsReminder = true;
                            }
                        }
                        else {
                            urceData.needsReminder = true;
                        }
                    }
                    if ((_settings.hideWaiting && urceData.waiting)
                        || (_settings.hideUrsCloseNeeded && urceData.needsClosed)
                        || (_settings.hideUrsReminderNeeded && urceData.needsReminder)
                        || (_settings.hideByStatusOpen && chunk[idx].attributes.open)
                        || (_settings.hideByStatusClosed && !chunk[idx].attributes.open)
                        || (_settings.hideByStatusNotIdentified && (chunk[idx].attributes.resolution === 1))
                        || (_settings.hideByStatusSolved && (chunk[idx].attributes.resolution === 0))
                        || (_settings.hideByStatusClosedBy && urceData.hideByStatusClosedBy)
                        // Types
                        || (_settings.hideByTypeBlockedRoad && (chunk[idx].attributes.type === 19))
                        || (_settings.hideByTypeGeneralError && (chunk[idx].attributes.type === 10))
                        || (_settings.hideByTypeIncorrectAddress && (chunk[idx].attributes.type === 7))
                        || (_settings.hideByTypeIncorrectJunction && (chunk[idx].attributes.type === 12))
                        || (_settings.hideByTypeIncorrectRoute && (chunk[idx].attributes.type === 8))
                        || (_settings.hideByTypeIncorrectStreetPrefixOrSuffix && (chunk[idx].attributes.type === 22))
                        || (_settings.hideByTypeIncorrectTurn && (chunk[idx].attributes.type === 6))
                        || (_settings.hideByTypeMissingBridgeOverpass && (chunk[idx].attributes.type === 13))
                        || (_settings.hideByTypeMissingExit && (chunk[idx].attributes.type === 15))
                        || (_settings.hideByTypeMissingLandmark && (chunk[idx].attributes.type === 18))
                        || (_settings.hideByTypeMissingOrInvalidSpeedLimit && (chunk[idx].attributes.type === 23))
                        || (_settings.hideByTypeMissingRoad && (chunk[idx].attributes.type === 16))
                        || (_settings.hideByTypeMissingRoundabout && (chunk[idx].attributes.type === 9))
                        || (_settings.hideByTypeMissingStreetName && (chunk[idx].attributes.type === 21))
                        || (_settings.hideByTypeTurnNotAllowed && (chunk[idx].attributes.type === 11))
                        || (_settings.hideByTypeUndefined
                            && (!chunk[idx].attributes.type
                                || (chunk[idx].attributes.type > 23)
                                || (chunk[idx].attributes.type < 6)
                                || (chunk[idx].attributes.type === 17)
                                || (chunk[idx].attributes.type === 20)))
                        || (_settings.hideByTypeWrongDrivingDirection && (chunk[idx].attributes.type === 14))
                        // Tags
                        || (_settings.hideByTaggedBog && (urceData.customType === 6))
                        || (_settings.hideByTaggedClosure && (urceData.customType === 2))
                        || (_settings.hideByTaggedConstruction && (urceData.customType === 1))
                        || (_settings.hideByTaggedDifficult && (urceData.customType === 7))
                        || (_settings.hideByTaggedEvent && (urceData.customType === 3))
                        || (_settings.hideByTaggedNote && (urceData.customType === 4))
                        || (_settings.hideByTaggedRoadworks && (urceData.customType === 0))
                        || (_settings.hideByTaggedWslm && (urceData.customType === 5))
                        // Age of submission
                        || (_settings.hideByAgeOfSubmissionLessThan && (urceData.driveDaysOld < _settings.hideByAgeOfSubmissionLessThanDaysOld))
                        || (_settings.hideByAgeOfSubmissionMoreThan && (urceData.driveDaysOld > _settings.hideByAgeOfSubmissionMoreThanDaysOld))
                        // Following, description, comments
                        || (_settings.hideFollowing && urSessionsObj.isFollowing)
                        || (_settings.hideNotFollowing && !urSessionsObj.isFollowing)
                        || (_settings.hideDescription
                            && (chunk[idx]?.attributes.description.length > 0) && (chunk[idx]?.attributes.description !== ''))
                        || (_settings.hideWithoutDescription
                            && (!chunk[idx].attributes.description || (chunk[idx].attributes.description.length === 0) || (chunk[idx].attributes.description === '')))
                        || (_settings.hideWithCommentsFromMe && (urceData.commentsByMe))
                        || (_settings.hideWithoutCommentsFromMe && (!urceData.commentsByMe))
                        || (_settings.hideLastCommentByMe && (urceData.lastCommentBy === _wmeUserId))
                        || (_settings.hideLastCommentNotByMe && (urceData.lastCommentBy !== _wmeUserId))
                        || (_settings.hideLastCommentByReporter && (urceData.lastCommentBy === -1))
                        || (_settings.hideLastCommentNotByReporter && (urceData.lastCommentBy > 0))
                        || (_settings.hideByCommentCountLessThan && (urceData.commentCount < _settings.hideByCommentCountLessThanNumber))
                        || (_settings.hideByCommentCountMoreThan && (urceData.commentCount > _settings.hideByCommentCountMoreThanNumber))
                        || (_settings.hideByAgeOfFirstCommentLessThan && (urceData.commentCount > 0) && (urceData.firstCommentDaysOld < _settings.hideByAgeOfFirstCommentLessThanDaysOld))
                        || (_settings.hideByAgeOfFirstCommentMoreThan && (urceData.commentCount > 0) && (urceData.firstCommentDaysOld > _settings.hideByAgeOfFirstCommentMoreThanDaysOld))
                        || (_settings.hideByAgeOfLastCommentLessThan && (urceData.commentCount > 0) && (urceData.lastCommentDaysOld < _settings.hideByAgeOfLastCommentLessThanDaysOld))
                        || (_settings.hideByAgeOfLastCommentMoreThan && (urceData.commentCount > 0) && (urceData.lastCommentDaysOld > _settings.hideByAgeOfLastCommentMoreThanDaysOld))
                        || (_settings.hideByKeywordIncluding && urceData.keywordIncluding)
                        || (_settings.hideByKeywordNotIncluding && urceData.keywordNotIncluding)
                        || (_settings.hideWithCommentBy && urceData.hideWithCommentBy)
                        || (_settings.hideWithoutCommentBy && urceData.hideWithoutCommentBy)
                    )
                        urceData.hideUr = true;
                    if (_settings.invertFilters) {
                        if (urceData.hideUr)
                            urceData.hideUr = false;
                        else
                            urceData.hideUr = true;
                    }
                    _mapUpdateRequests[chunk[idx].attributes.id] = { urceData };
                }
                else {
                    logDebug(`Could not load urSessionsObj for urID: ${chunk[idx].attributes.id}`);
                }
            }
            if (autoSentRemindersFor.length > 0) {
                logDebug(`Automatically sent reminder comments to urId(s): ${autoSentRemindersFor.join(', ')} (${autoSentRemindersFor.length})`);
                WazeWrap.Alerts.info(_SCRIPT_SHORT_NAME, `${I18n.t('urce.prompts.ReminderMessageAuto')}: ${autoSentRemindersFor.join(', ')} (${autoSentRemindersFor.length})`);
            }
        }
        if (updateMarkersArr.length > 0) {
            const zoomLevel = W.map.getZoom(),
                filter = !(((_settings.disableFilteringAboveZoom && (zoomLevel < _settings.disableFilteringAboveZoomLevel))
                            || (_settings.disableFilteringBelowZoom && (zoomLevel > _settings.disableFilteringBelowZoomLevel))));
            updateUrMapMarkers(updateMarkersArr, filter);
        }
        if (reopenPanel)
            openUrPanel(0, true);
        return Promise.resolve();
    }

    async function handleUrLayer(phase, filter, mUrsObjArr) {
        if (!mUrsObjArr)
            return Promise.resolve();
        doSpinner('handleUrLayer', true);
        const zoomLevel = W.map.getZoom();
        filter = (filter !== undefined) ? filter : !(((_settings.disableFilteringAboveZoom && (zoomLevel < _settings.disableFilteringAboveZoomLevel))
                                                    || (_settings.disableFilteringBelowZoom && (zoomLevel > _settings.disableFilteringBelowZoomLevel))));
        if (phase === 'init')
            logDebug('Checking for UR markers already present before URC-E completed initialization.');
        else if (phase === 'init_end')
            logDebug('Updating UR markers that appeared after initialization completed.');
        else if (phase === 'close')
            logDebug('Updating UR markers after closing UR panel.');
        else if (phase === 'settingsToggle')
            logDebug('Updating UR markers after a setting toggle.');
        else if (phase === 'sendComment')
            logDebug('Updating UR markers after sending a comment.');
        else if (phase === 'overflow')
            logDebug('Updating UR markers from being added through UR overflow handling.');
        else
            logDebug(`Updating UR markers that appeared after a ${phase} event.`);
        mUrsObjArr.sort((a, b) => a.attributes.id - b.attributes.id);
        if (phase === 'init')
            _markerCountOnInit = mUrsObjArr.length;
        if ((phase === 'overflow') && _initialUrLayerScan) {
            if (_markerCountOnInit > -1)
                _markerCountOnInit += mUrsObjArr.length;
            else
                _markerCountOnInit = mUrsObjArr.length;
        }
        await updateUrceData(mUrsObjArr);
        updateUrMapMarkers(mUrsObjArr, filter);
        // 2023.04.05.01: Used to check for conditions and either run handleUrOverflow or panel alert box or remove it. But now using W.app.on('change:loadingFeedMapData') event listener.
        if (phase !== 'overflow')
            _filtersAppliedOnZoom = filter;
        doSpinner('handleUrLayer', false);
        return Promise.resolve();
    }

    function getOverflowUrsFromUrl(urlStr) {
        logDebug(`Getting URs from: ${urlStr}`);
        return new Promise((resolve) => {
            (async function retry(url, tries, toIndex) {
                checkTimeout({ timeout: 'getOverflowUrsFromUrl', toIndex });
                const errorObj = { error: false, url };
                let data;
                try {
                    data = await $.getJSON(url);
                    data.url = url;
                }
                catch (error) {
                    if (error.status === 429)
                        data = { error: { status: 429 } };
                    else if (!errorObj.error)
                        errorObj.error = (typeof error === 'object') ? error : { error };
                }
                if (tries > 10) {
                    resolve({ error: { reason: 'Too many retries.' }, url });
                }
                else if (errorObj.error) {
                    resolve(errorObj);
                }
                else if (!data || (data?.error?.status === 429)) {
                    log('Rate limited by Waze server. Retrying overflow request.');
                    _timeouts.getOverflowUrsFromUrl[toIndex] = window.setTimeout(retry, 100, url, ++tries, toIndex);
                }
                else {
                    checkTimeout({ timeout: 'getOverflowUrsFromUrl', toIndex });
                    resolve(data);
                }
            }(urlStr, 1, getRandomId()));
        });
    }

    function handleUrOverflow(evt) {
        if (evt?.changed?.loadingFeedMapData || evt?.changed?.attributes?.loadingFeedMapData)
            return;
        if (W.map.getZoom() < 10) {
            logDebug('UR overflow handling does not work with zoom levels < 10.');
            return;
        }
        if (W.model.mapUpdateRequests.getObjectArray().length < 500) {
            dismissAlertBoxInPanel(undefined, 9999);
            return;
        }
        if (!_settings.enableUrOverflowHandling && (W.model.mapUpdateRequests.getObjectArray().length > 499)) {
            alertBoxInPanel(I18n.t('urce.prompts.UrOverflowErrorWithoutOverflowEnabled'), undefined, true, 9999);
            return;
        }
        doSpinner('handleUrOverflow', true);
        const baseUrl = `https://${document.location.host}${W.Config.paths.features}?language=en&mapUpdateRequestFilter=`
                + `${(($('#layer-switcher-item_closed_update_requests').is(':checked')) ? '3' : '1')}%2C0&bbox=`,
            vpBounds = W.map.getExtent().transform(W.map.getProjectionObject(), new OpenLayers.Projection('EPSG:4326')),
            vpBoundsFrom = { lon: vpBounds.left, lat: vpBounds.bottom },
            vpBoundsTo = { lon: vpBounds.right, lat: vpBounds.top },
            vpCenter = W.map.getCenter().transform(W.map.getProjectionObject(), new OpenLayers.Projection('EPSG:4326')),
            processData = (data) => {
                if (data.error) {
                    logWarning(data.error);
                }
                else if (data.mapUpdateRequests?.objects?.length > 499) {
                    logDebug('More than 499 objects returned in overflow request, queueing sub quadrants for further checking.');
                    const bbox = data.url.split('bbox=')[1].split(','),
                        bboxFrom = WazeWrap.Geometry.ConvertTo900913(bbox[0], bbox[1]),
                        bboxTo = WazeWrap.Geometry.ConvertTo900913(bbox[2], bbox[3]),
                        subQuadCenter = WazeWrap.Geometry.ConvertTo4326((bboxFrom.lon - ((bboxFrom.lon - bboxTo.lon) / 2)), (bboxTo.lat - ((bboxTo.lat - bboxFrom.lat) / 2)));
                    [`${baseUrl}${subQuadCenter.lon.toFixed(6)},${subQuadCenter.lat.toFixed(6)},${bbox[2]},${bbox[3]}`,
                        `${baseUrl}${bbox[0]},${subQuadCenter.lat.toFixed(6)},${subQuadCenter.lon.toFixed(6)},${bbox[3]}`,
                        `${baseUrl}${bbox[0]},${bbox[1]},${subQuadCenter.lon.toFixed(6)},${subQuadCenter.lat.toFixed(6)}`,
                        `${baseUrl}${subQuadCenter.lon.toFixed(6)},${bbox[1]},${bbox[2]},${subQuadCenter.lat.toFixed(6)}`
                    ].forEach((url) => {
                        _overflowUrsUrls.push(url);
                        getOverflowUrsFromUrl(url).then(processData);
                    });
                }
                else if (data.mapUpdateRequests?.objects?.length > 0) {
                    const overflowUrsToPut = [];
                    data.mapUpdateRequests.objects.forEach((mapUr) => {
                        if (!W.model.mapUpdateRequests.objects[mapUr.id]) {
                            const NewUr = require('Waze/Feature/Vector/UpdateRequest'),
                                toPutUr = new NewUr(mapUr),
                                toPutPoint = new OpenLayers.Geometry.Point(mapUr.geometry.coordinates[0], mapUr.geometry.coordinates[1])
                                    .transform(new OpenLayers.Projection('EPSG:4326'), W.map.getProjectionObject());
                            toPutUr.geometry = toPutPoint;
                            const toPutReqBounds = new OpenLayers.Geometry.Polygon(),
                                toPutBounds = new OpenLayers.Bounds(toPutPoint.x, toPutPoint.y, toPutPoint.x, toPutPoint.y);
                            toPutReqBounds.bounds = toPutBounds;
                            toPutUr.requestBounds = toPutReqBounds;
                            overflowUrsToPut.push(toPutUr);
                        }
                    });
                    if (overflowUrsToPut.length > 0) {
                        logDebug(`${overflowUrsToPut.length} URs added from overflow.`);
                        W.model.mapUpdateRequests.put(overflowUrsToPut);
                    }
                    else {
                        logDebug('All URs submitted for overflow processing already exist on map.');
                    }
                }
                _overflowUrsUrls.splice(_overflowUrsUrls.indexOf(data.url), 1);
                if (_overflowUrsUrls.length === 0)
                    doSpinner('handleUrOverflow', false);
            };
        [`${baseUrl}${vpCenter.lon.toFixed(6)},${vpCenter.lat.toFixed(6)},${vpBoundsTo.lon.toFixed(6)},${vpBoundsTo.lat.toFixed(6)}`,
            `${baseUrl}${vpBoundsFrom.lon.toFixed(6)},${vpCenter.lat.toFixed(6)},${vpCenter.lon.toFixed(6)},${vpBoundsTo.lat.toFixed(6)}`,
            `${baseUrl}${vpBoundsFrom.lon.toFixed(6)},${vpBoundsFrom.lat.toFixed(6)},${vpCenter.lon.toFixed(6)},${vpCenter.lat.toFixed(6)}`,
            `${baseUrl}${vpCenter.lon.toFixed(6)},${vpBoundsFrom.lat.toFixed(6)},${vpBoundsTo.lon.toFixed(6)},${vpCenter.lat.toFixed(6)}`
        ].forEach((url) => {
            _overflowUrsUrls.push(url);
            getOverflowUrsFromUrl(url).then(processData);
        });
    }

    function mouseDown() {
        _mouseIsDown = true;
    }

    function mouseUp() {
        _mouseIsDown = false;
    }

    /**
     * 2024.04.05.01: With the removal of the handleUrOverflow call from this function, it and the event listener are no longer needed.
    function invokeMoveEnd(/* evt *-/) {
        /**
         * Enable Auto Refresh: Disabled 2023.03.29
         *      Due to W.controller.reloadData() causing issues with new Issue Tracker.
         *      Specifically if user had more than 500 URs loaded in tracker, it would reset them back to 500 due to reloadData
         *      wiping and reloading the model data. For now, this setting is disabled and the functionality has been replaced
         *      with the already working handleOverflow function (if the user has it enabled).
        const zoomLevel = evt.object.zoom || W.map.getZoom();
        if (_settings.enableAutoRefresh
            && (zoomLevel > 14)
            && (W.model.mapUpdateRequests.getObjectArray().length > 499)
            && (W.model.actionManager.getActionsNum() === 0)
        )
            W.controller.reloadData();
        else if (!_settings.enableUrOverflowHandling && (W.model.mapUpdateRequests.getObjectArray().length > 499))
            alertBoxInPanel(I18n.t('urce.prompts.UrOverflowErrorWithoutOverflowEnabled'), undefined, true, 9999);
        else
            dismissAlertBoxInPanel(undefined, 9999);
         *-/
        // 2023.04.05.01: Used to check for conditions and either run handleUrOverflow or panel alert box or remove it. But now using W.app.on('change:loadingFeedMapData') event listener.
    }
     */

    function invokeZoomEnd(evt) {
        const zoomLevel = evt?.object?.zoom || W.map.getZoom();
        /**
         * Enable Auto Refresh: Disabled 2023.03.29
         *      Due to W.controller.reloadData() causing issues with new Issue Tracker.
         *      Specifically if user had more than 500 URs loaded in tracker, it would reset them back to 500 due to reloadData
         *      wiping and reloading the model data. For now, this setting is disabled and the functionality has been replaced
         *      with the already working handleOverflow function (if the user has it enabled).
         *
        if (_settings.enableAutoRefresh && (zoomLevel > 14) && (W.model.mapUpdateRequests.getObjectArray().length > 499) && (W.model.actionManager.getActionsNum() === 0)) {
            W.controller.reloadData();
            return;
        }
         */
        let filter;
        if (_settings.disableFilteringAboveZoom || _settings.disableFilteringBelowZoom) {
            if (!_filtersAppliedOnZoom && _settings.disableFilteringAboveZoom && (zoomLevel > (_settings.disableFilteringAboveZoomLevel - 1)))
                filter = true;
            if (!_filtersAppliedOnZoom && _settings.disableFilteringBelowZoom && (zoomLevel < (_settings.disableFilteringBelowZoomLevel + 1)))
                filter = true;
            if (_filtersAppliedOnZoom && _settings.disableFilteringAboveZoom && (zoomLevel < _settings.disableFilteringAboveZoomLevel))
                filter = false;
            if (_filtersAppliedOnZoom && _settings.disableFilteringBelowZoom && (zoomLevel > _settings.disableFilteringBelowZoomLevel))
                filter = false;
        }
        // 2023.04.05.01: Used to check for conditions and either run handleUrOverflow or panel alert box or remove it. But now using W.app.on('change:loadingFeedMapData') event listener.
        if (filter !== undefined)
            handleUrLayer('zoomEnd', filter, getMapUrsObjArr());
    }

    async function invokeModeChange(event) {
        if (event?.changed?.mode === 1)
            await initBackgroundTasks('disable', 'invokeModeChange');
        else if (event?.changed?.mode === 0)
            await initBackgroundTasks('enable', 'invokeModeChange');
        if ((event?.changed?.mode === 0) || (event?.changed?.isImperial === true) || (event?.changed?.isImperial === false)) {
            await changeCommentList(_currentCommentList, false, true);
            await checkRestrictions([{ type: 'modeChange' }]);
            handleUrLayer('modeChange', undefined, getMapUrsObjArr());
        }
    }

    async function maskBoxes(message, unmask, phase, maskUrPanel) {
        const zIndex = (phase === 'init') ? 19999 : 10000;
        if (unmask) {
            $(`#urceTabLightbox-${phase}`).remove();
            $(`#urPanelLightbox-${phase}`).remove();
            if ($('[id^="urceTabLightbox-"').length === 0)
                $('#sidepanel-urc-e').css('position', '');
        }
        else if (!unmask) {
            if ($(`#urceTabLightbox-${phase}`).length === 0) {
                $('#sidepanel-urc-e').css('position', 'relative');
                const $urceTabDisabled = $('<div>', { id: `urceTabLightbox-${phase}`, class: 'urceMaskLightbox', style: `z-index:${zIndex};` });
                $urceTabDisabled.html(`<div class="urceMaskLightbox text">${message}</div>`);
                $('#sidepanel-urc-e').prepend($urceTabDisabled);
            }
            if (maskUrPanel && ($(`#urPanelLightbox-${phase}`).length === 0)) {
                const $domElement = await getDomElement('#panel-container .mapUpdateRequest.panel.show');
                $($domElement.children()[0]).css('position', 'relative');
                const $urPanelDisabled = $('<div>', { id: `urPanelLightbox-${phase}`, class: 'urceMaskLightbox', style: `z-index:${zIndex};` });
                $urPanelDisabled.html(`<div class="urceMaskLightbox text">${message}</div>`);
                if (!$domElement)
                    handleReadyError(false, false, '', false, 'Timed out trying to add mask to UR panel.');
                else
                    $($domElement.children()[0]).prepend($urPanelDisabled);
            }
        }
    }

    function autoScrollComments(commentCount = 0, retryInterval = 10, maxTries = 200) {
        (async function retry(commentCountInt, tries, retryInt, maxNumTries) {
            checkTimeout({ timeout: 'autoScrollComments' });
            if (tries > maxNumTries) {
                logWarning(`Timed out trying to scroll to the bottom. commentCount: ${commentCountInt}, tries: ${tries}, retryInt: ${retryInt}, maxTries: ${maxTries}`);
                return;
            }
            const $commentList = await getDomElement('#panel-container .mapUpdateRequest.panel.show .top-section .body .conversation.section .conversation-view .comment-list');
            if ($commentList && (commentCountInt > 0)
                && (commentCountInt === $commentList.children().length)
                && (!_settings.autoScrollComments
                    || (($commentList[0].scrollHeight - $commentList.scrollTop() - $commentList.outerHeight()) > 1)
                )
            )
                $commentList.scrollTop((_settings.autoScrollComments ? $commentList[0].scrollHeight : 0));
            else if ((commentCountInt !== 0) && W.map.panelRegion.currentView.model.attributes.loadingConversation)
                _timeouts.autoScrollComments = window.setTimeout(retry, retryInt, commentCountInt, ++tries, retryInt, maxNumTries);
        }(commentCount, 1, retryInterval, maxTries));
    }

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

    function changeCommentList(commentListIdx, autoSwitch, refresh) {
        return new Promise((resolve) => {
            doSpinner('changeCommentList', true);
            refresh = (refresh === true);
            commentListIdx = (isNaN(commentListIdx)) ? _settings.commentList : commentListIdx;
            if (refresh || (!autoSwitch && ((commentListIdx !== _settings.commentList) || (commentListIdx !== _currentCommentList))) || (autoSwitch && (commentListIdx !== _currentCommentList))) {
                let debugMsg;
                if (autoSwitch)
                    debugMsg = 'Automatically switching comment lists';
                else if (refresh)
                    debugMsg = 'Refreshing comment list';
                else
                    debugMsg = 'Switching comment lists';
                if (parseInt(_currentCommentList) > -1)
                    debugMsg += ` from ${getCommentListInfo(_currentCommentList).name}`;
                logDebug(`${debugMsg} to ${getCommentListInfo(commentListIdx).name}.`);
                if (!autoSwitch && !refresh)
                    _settings.commentList = commentListIdx;
                buildCommentList(commentListIdx, 'changeList', autoSwitch).catch((error) => {
                    error.maskUrPanel = (_selUr.urId > 0);
                    error.phase = 'changeList';
                    error.staticList = (getCommentListInfo(commentListIdx).type === 'static');
                    handleError(error);
                })
                    .finally(() => {
                        if (!autoSwitch && !refresh)
                            saveSettingsToStorage();
                        doSpinner('changeCommentList', false);
                        resolve();
                    });
            }
            else {
                doSpinner('changeCommentList', false);
                resolve();
            }
        });
    }

    function getCommentListInfo(commentListIdx) {
        commentListIdx = (isNaN(commentListIdx)) ? _settings.commentList : commentListIdx;
        return _commentLists.find((cList) => cList.idx === commentListIdx);
    }

    function createCommentListCSV(section) {
        section = section || 'comments';
        const date = new Date(),
            dateTs = `${date.getFullYear()}${(date.getMonth() < 9) ? `0${(date.getMonth() + 1)}` : (date.getMonth() + 1)
            }${date.getDate() < 9 ? `0${(date.getDate() + 1)}` : (date.getDate() + 1)
            }${date.getHours() < 10 ? `0${date.getHours()}` : date.getHours()
            }${date.getMinutes() < 10 ? `0${date.getMinutes()}` : date.getMinutes()
            }${date.getSeconds() < 10 ? `0${date.getSeconds()}` : date.getSeconds()
            }`;
        if (section === 'defaults') {
            let idx,
                defaultsArr = {};
            Object.keys(_defaultComments).forEach((key) => {
                if (key === 'dr')
                    idx = 0;
                else if (key === 'dc')
                    idx = 1;
                else
                    idx = _defaultComments[key].urNum;
                defaultsArr[idx] = (_defaultComments[key].commentNum !== null) ? [_commentList[_defaultComments[key].commentNum].title] : [''];
            });
            defaultsArr = Object.values(defaultsArr);
            const a = document.createElement('a');
            a.href = URL.createObjectURL(new Blob([defaultsArr.join('\n')], { type: 'text/csv' }));
            a.download = `default_responses_${dateTs}.csv`;
            a.click();
            $(a).remove();
        }
        else if (section === 'comments') {
            const commentsArr = _commentList.map((entry) => {
                let urstatus;
                if (entry.urstatus === 'open')
                    urstatus = 'Open';
                else if (entry.urstatus === 'solved')
                    urstatus = 'Solved';
                else if (entry.urstatus === 'notidentified')
                    urstatus = 'NotIdentified';
                else if (entry.urstatus !== '')
                    urstatus = entry.urstatus.toUpperCase();
                else
                    urstatus = '';
                return [`"${entry.title.replace(/"/gmi, '\\"')}"|"${entry.comment.replace(/"/gmi, '\\"')}"|${urstatus}`];
            });
            const a = document.createElement('a');
            a.href = URL.createObjectURL(new Blob([commentsArr.join('\n')], { type: 'text/csv' }));
            a.download = `comment_list_${dateTs}.csv`;
            a.click();
            $(a).remove();
        }
    }

    function createStaticToGoogleSheet(convert) {
        if (convert && (getCommentListInfo(_currentCommentList).type !== 'static')) {
            WazeWrap.Alerts.error(_SCRIPT_SHORT_NAME, I18n.t('urce.prompts.ConversionLoadAddonFirst'));
            return;
        }
        const createSteps = Object.keys(I18n.translations['en-US'].urce.tools).filter((k) => /^CreateStep[0-9]+/.test(k)),
            finalSteps = Object.keys(I18n.translations['en-US'].urce.tools).filter((k) => /^FinalStep[0-9]+/.test(k));
        let spreadsheetStep,
            downloadDefaultsStep,
            downloadCommentsStep,
            htmlOut = `<div class="URCE-disableWme-text-header">${(convert ? I18n.t('urce.tools.ConvertCreateConvertProcess') : I18n.t('urce.tools.ConvertCreateCreateProcess'))}</div>`
                + `<div class="URCE-disableWme-text-body"><div style="font-size:14px;font-weight:600;">${I18n.t('urce.tools.ConvertCreateSteps')}:</div><ol>`;
        for (let i = 0; i < createSteps.length; i++) {
            if (/\$SPREADSHEET_STEP\$/.test(I18n.t(`urce.tools.${createSteps[i]}`)))
                spreadsheetStep = i + 1;
            htmlOut += `<li><p>${
                I18n.t(`urce.tools.${createSteps[i]}`).replace('$TEMPLATE_LINK$', `<a id="_openTemplate" href="https://bit.ly/urc-e_cc-template" target="_blank">${I18n.t('urce.common.Link')}</a>`).replace('$SPREADSHEET_STEP$', '')
            }</p></li>`;
        }
        if (convert) {
            const convertSteps = Object.keys(I18n.translations['en-US'].urce.tools).filter((k) => /^ConvertStep[0-9]+/.test(k));
            for (let i = 0; i < convertSteps.length; i++) {
                if (/\$DOWNLOAD_DEFAULTS_LINK\$/.test(I18n.t(`urce.tools.${convertSteps[i]}`)))
                    downloadDefaultsStep = createSteps.length + i + 1;
                if (/\$DOWNLOAD_COMMENTS_LINK\$/.test(I18n.t(`urce.tools.${convertSteps[i]}`)))
                    downloadCommentsStep = createSteps.length + i + 1;
                htmlOut += `<li><p>${
                    I18n.t(`urce.tools.${convertSteps[i]}`).replace('$DOWNLOAD_DEFAULTS_LINK$', `<a id="_downloadDefaults">${I18n.t('urce.common.Link')}</a>`)
                        .replace('$DOWNLOAD_COMMENTS_LINK$', `<a id="_downloadComments">${I18n.t('urce.common.Link')}</a>`).replace('$DOWNLOAD_DEFAULTS_STEP$', downloadDefaultsStep).replace('$DOWNLOAD_COMMENTS_STEP$', downloadCommentsStep)
                }</p></li>`;
            }
        }
        for (let i = 0; i < finalSteps.length; i++) {
            const stepText = I18n.t(`urce.tools.${finalSteps[i]}`).replace('$SPREADSHEET_STEP$', spreadsheetStep);
            htmlOut += `<li><p>${stepText}</p></li>`;
        }
        htmlOut += `</ol></div><div class="URCE-disableWme-text-footer"><button id="_dismissSteps" urceprefs="tools" class="urceToolsButton">${I18n.t('urce.common.Finish')}</button></div>`;
        disableWme(htmlOut, false);
        $('#_dismissSteps').off().on('click', () => disableWme(undefined, true));
        $('#_downloadDefaults').off().on('click', () => createCommentListCSV('defaults'));
        $('#_downloadComments').off().on('click', () => createCommentListCSV('comments'));
    }

    function checkForStaticListArray(oldVarNameStr) {
        return new Promise((resolve, reject) => {
            (function retry(oldVarName, tries) {
                checkTimeout({ timeout: 'checkForStaticListArray' });
                if (tries > 100)
                    reject(new Error(I18n.t('urce.prompts.TimedOutWaitingStatic')));
                else if (!(typeof unsafeWindow !== 'undefined' ? unsafeWindow : window)[`Urcomments${oldVarName}Array2`])
                    _timeouts.checkForStaticListArray = window.setTimeout(retry, 100, oldVarName, ++tries);
                else
                    resolve();
            }(oldVarNameStr, 1));
        });
    }

    function convertCommentListStatic(commentListIdx) {
        return new Promise((resolve, reject) => {
            commentListIdx = (isNaN(commentListIdx)) ? _settings.commentList : commentListIdx;
            const { oldVarName } = getCommentListInfo(commentListIdx);
            checkForStaticListArray(oldVarName).then(() => {
                try {
                    const oldUrcArr = (typeof unsafeWindow !== 'undefined' ? unsafeWindow : window)[`Urcomments${oldVarName}Array2`],
                        defaultReminderIdx = parseInt((typeof unsafeWindow !== 'undefined' ? unsafeWindow : window)[`Urcomments${oldVarName}ReminderPosistion`]),
                        closedNiIdx = parseInt((typeof unsafeWindow !== 'undefined' ? unsafeWindow : window)[`Urcomments${oldVarName}CloseNotIdentifiedPosistion`]),
                        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 (!/<br>/gi.test(oldUrcArr[0]) && (oldUrcArr[2] !== '')) {
                        data[3] = ['||GROUP TITLE||||||||||||||||||'];
                        entryIdx = 4;
                    }
                    else {
                        entryIdx = 3;
                    }
                    for (let oldUrcArrIdx = 0; oldUrcArrIdx < oldUrcArr.length; oldUrcArrIdx += 3) {
                        if (/<br>/gi.test(oldUrcArr[oldUrcArrIdx]) || (oldUrcArr[oldUrcArrIdx + 2] === '')) {
                            oldUrcArr[oldUrcArrIdx + 2] = 'GROUP TITLE';
                            oldUrcArr[oldUrcArrIdx] = $('<div>').html(oldUrcArr[oldUrcArrIdx]).text();
                        }
                        let temp = `${oldUrcArr[oldUrcArrIdx]}|${oldUrcArr[oldUrcArrIdx + 1]}|`
                        + `${((oldUrcArr[oldUrcArrIdx + 2] !== '') ? oldUrcArr[oldUrcArrIdx + 2].toLowerCase() : '')}`
                        + `${((oldUrcArrIdx === defaultReminderIdx) ? '|default_is_true' : '|')}`
                        + `${((oldUrcArrIdx === closedNiIdx) ? '|default_is_true' : '|')}`;
                        for (let i = 6; i < 24; i++) {
                            if ((i !== 17) && (i !== 20))
                                temp += (((typeof unsafeWindow !== 'undefined' ? unsafeWindow : window)[`Urcomments${oldVarName}def_names`][i]) && ((typeof unsafeWindow !== 'undefined' ? unsafeWindow : window)[`Urcomments${oldVarName}def_names`][i].toLowerCase() === oldUrcArr[oldUrcArrIdx].toLowerCase())) ? '|default_is_true' : '|';
                        }
                        data[entryIdx] = [temp];
                        entryIdx++;
                    }
                    resolve(data);
                }
                catch (error) {
                    reject(error);
                }
            })
                .catch((error) => {
                    reject(error);
                });
        });
    }

    function processCommentList(data) {
        return new Promise((resolve, reject) => {
            logDebug('Processing comment list data.');
            if (data) {
                let ssFieldNames;
                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'],
                    outputItems = [],
                    rowObj = {},
                    findObjIndex = (array, fldName, value) => array.map((a) => a[fldName]).indexOf(value),
                    checkFieldNames = (fldName) => ssFieldNames.indexOf(fldName) > -1,
                    createRowObj = (entry, i) => {
                        i = i || 0;
                        if (ssFieldNames[i].trim().toLowerCase() === 'comment')
                            rowObj[ssFieldNames[i].trim().toLowerCase()] = entry;
                        else if (ssFieldNames[i].trim().toLowerCase() === 'title')
                            rowObj[ssFieldNames[i].trim().toLowerCase()] = entry.trim();
                        else
                            rowObj[ssFieldNames[i].trim().toLowerCase()] = entry.trim().toLowerCase();
                        i++;
                    };
                let groupDivId,
                    commentId = 0,
                    blankGroup = 0;
                for (let entryIdx = 0; entryIdx < data.length; entryIdx++) {
                    if (entryIdx === 0) {
                        if (data[entryIdx][0] !== 'URCE')
                            return reject(new Error('Incorrect format in spreadsheet data received.'));
                    }
                    else if (entryIdx === 1) {
                        if (_SCRIPT_VERSION < data[entryIdx][0])
                            return reject(new Error(`updateRequired|${data[entryIdx][0]}`));
                    }
                    else if (entryIdx === 2) {
                        ssFieldNames = data[entryIdx][0].split('|').map((fldName) => fldName.trim());
                        if (ssFieldNames.length !== EXPECTED_FIELD_NAMES.length)
                            return reject(new Error(`Expected ${EXPECTED_FIELD_NAMES.length} columns in comment definition data. Spreadsheet returned ${ssFieldNames.length}.`));
                        if (!EXPECTED_FIELD_NAMES.every((fldName) => checkFieldNames(fldName)))
                            return reject(new Error(`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(', ')}`));
                    }
                    else if (data[entryIdx][0]) {
                        data[entryIdx][0].split('|').forEach((entry, i) => createRowObj(entry, i));
                        if (rowObj.title !== 'URCE_REMOVED_SO_SKIP') {
                            if (rowObj.title === 'URCE_ERROR')
                                return reject(new Error(`There is an unknown error in the spreadsheet output. Please contact the list owner: ${getCommentListInfo(_settings.commentList).listOwner}`));
                            if (rowObj.urstatus === 'custom var') {
                                _customReplaceVars.push({ customVar: `$${rowObj.title}$`, replaceText: rowObj.comment });
                            }
                            else if ((entryIdx === 3) && (rowObj.urstatus !== 'group title')) {
                                return reject(new Error(`Row 25 on the spreadsheet must be set to "GROUP TITLE" for the URSTATUS column. Please contact the list owner: ${getCommentListInfo(_settings.commentList).listOwner}`));
                            }
                            else if (rowObj.urstatus === 'group title') { // Group title row. Nothing to set in the arrays, but build html.
                                groupDivId = 'urceComments-for-';
                                if (rowObj.title !== '') {
                                    groupDivId += rowObj.title.replace(/[^\w]+/gi, '').toLowerCase();
                                    if (rowObj.title === rowObj.title.toUpperCase()) {
                                        if (rowObj.title.length > 30) {
                                            rowObj.titleMouseOver = rowObj.title;
                                            rowObj.title = `${rowObj.title.substring(0, 30)}...`;
                                        }
                                    }
                                    else if (rowObj.title.length > 35) {
                                        rowObj.titleMouseOver = rowObj.title;
                                        rowObj.title = `${rowObj.title.substring(0, 35)}...`;
                                    }
                                }
                                else {
                                    groupDivId += `blankGroup${(++blankGroup)}`;
                                }
                                const collapsed = (
                                        _settings.commentListCollapses.hasOwnProperty(_settings.commentList)
                                        && _settings.commentListCollapses[_settings.commentList].hasOwnProperty(`${groupDivId}_body_urce`)
                                        && (_settings.commentListCollapses[_settings.commentList][`${groupDivId}_body_urce`] === true)
                                    )
                                        ? 'collapse'
                                        : '',
                                    chevron = (collapsed === 'collapse') ? 'fa-chevron-right' : 'fa-chevron-down';
                                outputItems.push({
                                    chevron,
                                    collapsed,
                                    groupDivId,
                                    items: [],
                                    name: rowObj.title,
                                    title: rowObj.titleMouseOver
                                });
                                _commentList[commentId] = { title: rowObj.title, comment: '', urstatus: 'GROUP TITLE' };
                                commentId++;
                            }
                            else { // SHOULD be a normal comments row, push values to arrays and build html.
                                if ((rowObj.urstatus !== 'solved') && (rowObj.urstatus !== 'notidentified') && (rowObj.urstatus !== 'open') && (rowObj.urstatus !== 'blank line'))
                                    return reject(new Error(`Your current selected list does not have a status set for ${rowObj.title}. Please contact list owner: ${getCommentListInfo(_settings.commentList).listOwner}`));
                                _commentList[commentId] = { title: rowObj.title, comment: rowObj.comment, urstatus: rowObj.urstatus };
                                if (Object.values(rowObj).indexOf('default_is_true') > -1) {
                                    const drIdx = ssFieldNames.indexOf('DR'),
                                        splitRowDefaultCommentsBoolean = Object.values(rowObj).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 divDoubleClickStyle = 'display:initial;';
                                if (rowObj.urstatus === 'solved') {
                                    linkClass = 'URCE-solvedLink';
                                    divDoubleClickId = 'URCE-divDoubleClickSolved';
                                    if (!_settings.doubleClickLinkSolvedComments)
                                        divDoubleClickStyle = 'display:none;';
                                }
                                else if (rowObj.urstatus === 'notidentified') {
                                    linkClass = 'URCE-niLink';
                                    divDoubleClickId = 'URCE-divDoubleClickNi';
                                    if (!_settings.doubleClickLinkNiComments)
                                        divDoubleClickStyle = 'display:none;';
                                }
                                else {
                                    linkClass = (rowObj.urstatus === 'blank line') ? 'URCE-blankLine' : 'URCE-openLink';
                                    divDoubleClickId = (rowObj.title !== '') ? 'URCE-divDoubleClickOpen' : 'URCE-divDoubleClickOpen-Hidden';
                                    if (!_settings.doubleClickLinkOpenComments || (rowObj.urstatus === 'blank line'))
                                        divDoubleClickStyle = 'display:none;';
                                }
                                const idx = findObjIndex(outputItems, 'groupDivId', groupDivId);
                                outputItems[idx].items.push({
                                    linkClass,
                                    commentId,
                                    title: formatText(rowObj.comment, false, false, -1),
                                    name: rowObj.title,
                                    divDoubleClickId,
                                    divDoubleClickStyle,
                                    divDoubleClickTitle: `${I18n.t('urce.common.DoubleClickTitle')}:\n${formatText(rowObj.comment, false, false, -1)}`
                                });
                                commentId++;
                            }
                        }
                    }
                }
                if (outputItems.length > 0) {
                    const urStyle = (_settings.commentListStyle === 'urStyle') ? ' urStyle' : '',
                        DOUBLE_CLICK_ICON = '';
                    let htmlOut = '';
                    outputItems.forEach((item) => {
                        htmlOut += `<fieldset id="${item.groupDivId}" class="URCE-field ${urStyle}">`
                        + `<legend id="${item.groupDivId}_legend" class="URCE-legend ${urStyle}"><i class="fa fa-fw ${item.chevron} URCE-chevron"></i>`
                        + `<span class="URCE-span"${(item.title ? ` title="${item.title}` : '')}">${item.name}</span></legend>`
                        + `<div id="${item.groupDivId}_body_urce" class="${item.collapsed} URCE-group_body ${urStyle}">`;
                        if (item.items.length > 0) {
                            for (let idx = 0; idx < item.items.length; idx++) {
                                htmlOut += `<div class="URCE-divComment hover expand ${item.items[idx].linkClass}" style="position:relative;">`
                                    + '         <div style="width:225px;display:inline-flex;">'
                                    + `             <a id="urce-cid-${item.items[idx].commentId}" class="URCE-Comments ${item.items[idx].linkClass} URCE-Comments" title="${item.items[idx].title.replace(/"/g, '\'')}">${item.items[idx].name}</a>`
                                    + '         </div>'
                                    + `         <div id="${item.items[idx].divDoubleClickId}" class="URCE-divDoubleClick" style="${item.items[idx].divDoubleClickStyle}" title="${item.items[idx].divDoubleClickTitle.replace(/"/g, '\'')}">`
                                    + `             <img id="urce-img-cid-${item.items[idx].commentId}" src="${DOUBLE_CLICK_ICON}" class="URCE-doubleClickIcon"></img>`
                                    + '         </div>'
                                    + '</div>';
                            }
                        }
                        htmlOut += '</div></fieldset>';
                    });
                    $('#_commentList').append(htmlOut);
                    $('legend[id$="_legend"]').off().on('click', function () {
                        $($(this).children()[0]).toggleClass('fa-chevron-down fa-chevron-right');
                        $($(this).siblings()[0]).toggleClass('collapse');
                        saveSettingsToStorage();
                    });
                    $('a[id^="urce-cid-"]').off().on('click', function () {
                        handleClickedComment(parseInt(this.id.replace('urce-cid-', '')), false);
                    });
                    $('img[id^="urce-img-cid-"]').off().on('dblclick', function () {
                        handleClickedComment(parseInt(this.id.replace('urce-img-cid-', '')), true);
                    });
                }
                else {
                    return reject(new Error(`There is an error in the output. Please contact the list owner: ${getCommentListInfo(_settings.commentList).listOwner}`));
                }
            }
            else {
                return reject(new Error('No data passed to the JSON processing function.'));
            }
            return resolve();
        });
    }

    function commentListAsync(commentListIdx) {
        return new Promise((resolve, reject) => {
            commentListIdx = (isNaN(commentListIdx)) ? _settings.commentList : commentListIdx;
            const commentListInfo = getCommentListInfo(commentListIdx);
            logDebug(`Beginning comment list async for comment list: ${commentListInfo.name}`);
            $.getJSON(`https://sheets.googleapis.com/v4/spreadsheets/${((commentListIdx === 1001) ? _settings.customSsId : _URCE_SPREADSHEET_ID)}/values/${commentListInfo.gSheetRange}?key=${_URCE_API_KEY}`, (data) => {
                if (data.values.length > 0)
                    resolve(data.values);
                else
                    reject(new Error('No comments found in the spreadsheet sheet.'));
            })
                .fail((response) => {
                    reject(new Error(`Spreadsheet call failed. Code: ${response.status} - Text: ${response.statusText}`));
                });
        });
    }

    function buildCommentList(commentListIdx, phase, autoSwitch) {
        return new Promise((resolve, reject) => {
            doSpinner('buildCommentList', true);
            commentListIdx = commentListIdx || _settings.commentList;
            const commentListInfo = getCommentListInfo(commentListIdx);
            Object.values(_defaultComments).forEach((a) => { a.commentNum = null; });
            logDebug(`Building comment list for: ${commentListInfo.name}`);
            if (phase !== 'init')
                maskBoxes(`${I18n.t('urce.prompts.SwitchingCommentLists')}<br><br>${I18n.t('urce.common.PleaseWait')}.`, false, phase, (_selUr.urId > 0));
            let htmlOut = `<div class="URCE-commentListName">${I18n.t('urce.common.CommentList')}: `
                + `<select id="_selcurrentCommentList" title="${I18n.t('urce.common.CurrentCommentListTitle')}">`;
            _commentLists.forEach((cList) => {
                if (cList.status !== 'disabled')
                    htmlOut += `<option value="${cList.idx}"${((cList.idx === commentListIdx) ? ' selected' : '')}>${(!autoSwitch ? cList.name : `${cList.name} (${I18n.t('urce.common.AutoSwitched')})`)}`;
            });
            htmlOut += '</select>'
                + `<div id="restrictionsEnforcedWarning" style="float:right;padding-top:8px;color:red;font-size:18px;${(_restrictionsEnforcedTitle ? '' : 'display:none;')}">`
                + `     <i class="fa fa-exclamation-triangle" aria-hidden="true"${(_restrictionsEnforcedTitle ? ` title="${_restrictionsEnforcedTitle}"` : '')}></i>`
                + '</div></div>'
                + '<div class="URCE-commentListName URCE-controls URCE-divCC">'
                + `     <input type="checkbox" id="_cbenableAppendMode" class="urceSettingsCheckbox2" title="${I18n.t('urce.prefs.EnableAppendModeTitle')}"${(_settings.enableAppendMode ? ' checked' : '')}>`
                + `     <label for="_cbenableAppendMode" title="${I18n.t('urce.prefs.EnableAppendModeTitle')}" class="URCE-label">${I18n.t('urce.prefs.EnableAppendMode')}</label><br>`
                + '</div>'
                + `<div id="URCE-expandCollapseAllComments" class="URCE-expandCollapseAll${((_settings.commentListStyle === 'urStyle') ? ' urStyle' : '')}">`
                + `     <div id="URCE-expandAllComments" class="URCE-expandCollapseAllItem">${I18n.t('urce.common.ExpandAll')}</div>`
                + '     <div style="display:inline;"> : </div>'
                + `     <div id="URCE-collapseAllComments" class="URCE-expandCollapseAllItem">${I18n.t('urce.common.CollapseAll')}</div>`
                + '</div>';
            $('#_commentList').empty().append(htmlOut);
            _restrictionsEnforcedTitle = undefined;
            $('#_selcurrentCommentList').off().on('change', function () {
                if ((parseInt(this.value) === 1001) && (!_settings.customSsId || (_settings.customSsId.length < 1))) {
                    $(this).val(_currentCommentList);
                    WazeWrap.Alerts.error(_SCRIPT_SHORT_NAME, I18n.t('urce.prompts.SetCustomSsIdFirst'));
                }
                else {
                    changeCommentList(parseInt(this.value), false, true);
                }
            });
            $('#_cbenableAppendMode').off().on('change', function () {
                _settings[$(this)[0].id.substr(3)] = isChecked(this);
                if (W.map.panelRegion.currentView?.adapter?.problem) {
                    $('textarea[id=id]', $('#panel-container .mapUpdateRequest .top-section .body .conversation .new-comment-text')[0].shadowRoot)
                        .css('background-color', (isChecked(this) ? 'peachpuff' : ''));
                }
                saveSettingsToStorage();
            });
            $('#URCE-expandAllComments, #URCE-collapseAllComments').off().on('click', function () {
                const $legends = $('legend[id^="urceComments-for"');
                for (let idx = 0; idx < $legends.length; idx++) {
                    if (this.id === 'URCE-expandAllComments') {
                        if ($legends[idx].nextElementSibling.className.indexOf('collapse') > -1)
                            $($legends[idx]).click();
                    }
                    else if (this.id === 'URCE-collapseAllComments') {
                        if ($legends[idx].nextElementSibling.className.indexOf('collapse') === -1)
                            $($legends[idx]).click();
                    }
                }
            });
            _commentList = [];
            _customReplaceVars = [];
            _currentCommentList = commentListIdx;
            processPerCommentListSettings(commentListIdx);
            if (commentListInfo.type === 'static') {
                convertCommentListStatic(commentListIdx).then((data) => {
                    processCommentList(data).then(() => {
                        _commentListLoaded = true;
                        doSpinner('buildCommentList', false);
                        if (phase !== 'init')
                            maskBoxes(undefined, true, phase, (_selUr.urId > 0));
                        resolve();
                    })
                        .catch((error) => {
                            doSpinner('buildCommentList', false);
                            error.phase = phase;
                            error.maskUrPanel = (_selUr.urId > 0);
                            error.staticList = (commentListInfo.type === 'static');
                            error.commentList = commentListIdx;
                            reject(error);
                        });
                }).catch((error) => {
                    doSpinner('buildCommentList', false);
                    error.phase = phase;
                    error.maskUrPanel = (_selUr.urId > 0);
                    error.staticList = (commentListInfo.type === 'static');
                    error.commentList = commentListIdx;
                    reject(error);
                });
            }
            else {
                commentListAsync(commentListIdx).then((data) => {
                    processCommentList(data).then(() => {
                        _commentListLoaded = true;
                        doSpinner('buildCommentList', false);
                        if (phase !== 'init')
                            maskBoxes(undefined, true, phase, (_selUr.urId > 0));
                        resolve();
                    })
                        .catch((error) => {
                            doSpinner('buildCommentList', false);
                            error.phase = phase;
                            error.maskUrPanel = (_selUr.urId > 0);
                            error.staticList = (commentListInfo.type === 'static');
                            error.commentList = commentListIdx;
                            reject(error);
                        });
                }).catch((error) => {
                    doSpinner('buildCommentList', false);
                    error.phase = phase;
                    error.maskUrPanel = (_selUr.urId > 0);
                    error.staticList = (commentListInfo.type === 'static');
                    error.commentList = commentListIdx;
                    reject(error);
                });
            }
        });
    }

    function processPerCommentListSettings(commentListIdx) {
        const defaultPerCommentListSettings = {
            autoSendReminders: _settings.autoSendReminders,
            autoSendReminders_useDefault: true,
            autoSendRemindersExceptTagged: _settings.autoSendRemindersExceptTagged,
            autoSendRemindersExceptTagged_useDefault: true,
            autoSetNewUrComment: _settings.autoSetNewUrComment,
            autoSetNewUrComment_useDefault: true,
            autoSetNewUrCommentSlur: _settings.autoSetNewUrCommentSlur,
            autoSetNewUrCommentSlur_useDefault: true,
            autoSetNewUrCommentWithDescription: _settings.autoSetNewUrCommentWithDescription,
            autoSetNewUrCommentWithDescription_useDefault: true,
            autoSetReminderUrComment: _settings.autoSetReminderUrComment,
            autoSetReminderUrComment_useDefault: true,
            customTagline: _settings.customTagline,
            customTagline_useDefault: true,
            tagEmail: _settings.tagEmail,
            tagEmail_useDefault: true,
            reminderDays: _settings.reminderDays,
            reminderDays_useDefault: true,
            closeDays: _settings.closeDays,
            closeDays_useDefault: true
        };
        let perCommentListSettingsChanged = false;
        if (!_settings.perCommentListSettings[commentListIdx]) {
            _settings.perCommentListSettings[commentListIdx] = defaultPerCommentListSettings;
            perCommentListSettingsChanged = true;
        }
        else {
            Object.keys(defaultPerCommentListSettings).forEach((prop) => {
                if (!_settings.perCommentListSettings[commentListIdx].hasOwnProperty(prop)) {
                    _settings.perCommentListSettings[commentListIdx][prop] = defaultPerCommentListSettings[prop];
                    perCommentListSettingsChanged = true;
                }
            });
        }
        if (perCommentListSettingsChanged)
            saveSettingsToStorage();
        const perCListSettings = _settings.perCommentListSettings[commentListIdx],
            htmlOut = ''
                + `<div>${I18n.t('urce.prefs.SettingsFor')}: ${getCommentListInfo(commentListIdx).name}</div>`
                + '<div class="URCE-controls">'
                + '   <div>'
                + '      <div style="width:calc(100% - 18px); display:inline-block">'
                + `         <input type="checkbox" id="_cbperCommentList_autoSendReminders" urceprefs="perCommentList" class="urceSettingsCheckbox${((perCListSettings.autoSendReminders_useDefault) ? ' urceDisabled' : '')}" title="${I18n.t('urce.prefs.AutoSendRemindersTitle')}" ${((perCListSettings.autoSendReminders) ? 'checked="true"' : '')} ${((perCListSettings.autoSendReminders_useDefault) ? 'disabled="true"' : '')}>`
                + `         <label for="_cbperCommentList_autoSendReminders" title="${I18n.t('urce.prefs.AutoSendRemindersTitle')}" urceprefs="perCommentList" class="URCE-label${((perCListSettings.autoSendReminders_useDefault) ? ' urceDisabled' : '')}">${I18n.t('urce.prefs.AutoSendReminders')}</label>`
                + `         <div class="URCE-divWarning URCE-divWarningPre">(<div class="URCE-divWarning URCE-divWarningTitle" title="${I18n.t('urce.prefs.AutoSendRemindersWarningTitle')}">${I18n.t('urce.prefs.AutoSendRemindersWarning')}</div><div class="URCE-divWarning">)</div></div>`
                + '      </div>'
                + `      <input type="checkbox" style="float:right;" id="_cbperCommentList_autoSendReminders_useDefault" urceprefs="perCommentList" class="urceSettingsCheckbox" title="${I18n.t('urce.prefs.UseDefault')}" ${((perCListSettings.autoSendReminders_useDefault) ? 'checked="true"' : '')}>`
                + '   </div>'
                + '   <div>'
                + '      <div style="width:calc(100% - 18px); display:inline-block; padding-left:15px; font-style:italic;">'
                + `         <input type="checkbox" id="_cbperCommentList_autoSendRemindersExceptTagged" urceprefs="perCommentList" class="urceSettingsCheckbox${((perCListSettings.autoSendRemindersExceptTagged_useDefault) ? ' urceDisabled' : '')}" title="${I18n.t('urce.prefs.AutoSendRemindersExceptTaggedTitle')}" ${((perCListSettings.autoSendRemindersExceptTagged) ? 'checked="true"' : '')} ${((perCListSettings.autoSendRemindersExceptTagged_useDefault) ? 'disabled="true"' : '')}>`
                + `         <label for="_cbperCommentList_autoSendRemindersExceptTagged" title="${I18n.t('urce.prefs.AutoSendRemindersExceptTaggedTitle')}" urceprefs="perCommentList" class="URCE-label${((perCListSettings.autoSendRemindersExceptTagged_useDefault) ? ' urceDisabled' : '')}">${I18n.t('urce.prefs.AutoSendRemindersExceptTagged')}</label>`
                + '      </div>'
                + `      <input type="checkbox" style="float:right;" id="_cbperCommentList_autoSendRemindersExceptTagged_useDefault" urceprefs="perCommentList" class="urceSettingsCheckbox" title="${I18n.t('urce.prefs.UseDefault')}" ${((perCListSettings.autoSendRemindersExceptTagged_useDefault) ? 'checked="true"' : '')}>`
                + '   </div>'
                + '   <div>'
                + '      <div style="width:calc(100% - 18px); display:inline-block">'
                + `         <input type="checkbox" id="_cbperCommentList_autoSetNewUrComment" urceprefs="perCommentList" class="urceSettingsCheckbox${((perCListSettings.autoSetNewUrComment_useDefault) ? ' urceDisabled' : '')}" title="${I18n.t('urce.prefs.AutoSetNewUrCommentTitle')}" ${((perCListSettings.autoSetNewUrComment) ? 'checked="true"' : '')} ${((perCListSettings.autoSetNewUrComment_useDefault) ? 'disabled="true"' : '')}>`
                + `         <label for="_cbperCommentList_autoSetNewUrComment" title="${I18n.t('urce.prefs.AutoSetNewUrCommentTitle')}" urceprefs="perCommentList" class="URCE-label${((perCListSettings.autoSetNewUrComment_useDefault) ? ' urceDisabled' : '')}">${I18n.t('urce.prefs.AutoSetNewUrComment')}</label>`
                + '      </div>'
                + `      <input type="checkbox" style="float:right;" id="_cbperCommentList_autoSetNewUrComment_useDefault" urceprefs="perCommentList" class="urceSettingsCheckbox" title="${I18n.t('urce.prefs.UseDefault')}" ${((perCListSettings.autoSetNewUrComment_useDefault) ? 'checked="true"' : '')}>`
                + '   </div>'
                + '   <div>'
                + '      <div style="width:calc(100% - 18px); display:inline-block">'
                + `         <input type="checkbox" id="_cbperCommentList_autoSetNewUrCommentSlur" urceprefs="perCommentList" class="urceSettingsCheckbox${((perCListSettings.autoSetNewUrCommentSlur_useDefault) ? ' urceDisabled' : '')}" title="${I18n.t('urce.prefs.AutoSetNewUrCommentSlurTitle')}" ${((perCListSettings.autoSetNewUrCommentSlur) ? 'checked="true"' : '')} ${((perCListSettings.autoSetNewUrCommentSlur_useDefault) ? 'disabled="true"' : '')}>`
                + `         <label for="_cbperCommentList_autoSetNewUrCommentSlur" title="${I18n.t('urce.prefs.AutoSetNewUrCommentSlurTitle')}" urceprefs="perCommentList" class="URCE-label${((perCListSettings.autoSetNewUrCommentSlur_useDefault) ? ' urceDisabled' : '')}">${I18n.t('urce.prefs.AutoSetNewUrCommentSlur')}</label>`
                + '      </div>'
                + `      <input type="checkbox" style="float:right;" id="_cbperCommentList_autoSetNewUrCommentSlur_useDefault" urceprefs="perCommentList" class="urceSettingsCheckbox" title="${I18n.t('urce.prefs.UseDefault')}" ${((perCListSettings.autoSetNewUrCommentSlur_useDefault) ? 'checked="true"' : '')}>`
                + '   </div>'
                + '   <div>'
                + '      <div style="width:calc(100% - 18px); display:inline-block">'
                + `         <input type="checkbox" id="_cbperCommentList_autoSetNewUrCommentWithDescription" urceprefs="perCommentList" class="urceSettingsCheckbox${((perCListSettings.autoSetNewUrCommentWithDescription_useDefault) ? ' urceDisabled' : '')}" title="${I18n.t('urce.prefs.AutoSetNewUrCommentWithDescriptionTitle')}" ${((perCListSettings.autoSetNewUrCommentWithDescription) ? 'checked="true"' : '')} ${((perCListSettings.autoSetNewUrCommentWithDescription_useDefault) ? 'disabled="true"' : '')}>`
                + `         <label for="_cbperCommentList_autoSetNewUrCommentWithDescription" title="${I18n.t('urce.prefs.AutoSetNewUrCommentWithDescriptionTitle')}" urceprefs="perCommentList" class="URCE-label${((perCListSettings.autoSetNewUrCommentWithDescription_useDefault) ? ' urceDisabled' : '')}">${I18n.t('urce.prefs.AutoSetNewUrCommentWithDescription')}</label>`
                + '      </div>'
                + `      <input type="checkbox" style="float:right;" id="_cbperCommentList_autoSetNewUrCommentWithDescription_useDefault" urceprefs="perCommentList" class="urceSettingsCheckbox" title="${I18n.t('urce.prefs.UseDefault')}" ${((perCListSettings.autoSetNewUrCommentWithDescription_useDefault) ? 'checked="true"' : '')}>`
                + '   </div>'
                + '   <div>'
                + '      <div style="width:calc(100% - 18px); display:inline-block">'
                + `         <input type="checkbox" id="_cbperCommentList_autoSetReminderUrComment" urceprefs="perCommentList" class="urceSettingsCheckbox${((perCListSettings.autoSetReminderUrComment_useDefault) ? ' urceDisabled' : '')}" title="${I18n.t('urce.prefs.AutoSetReminderUrCommentTitle')}" ${((perCListSettings.autoSetReminderUrComment) ? 'checked="true"' : '')} ${((perCListSettings.autoSetReminderUrComment_useDefault) ? 'disabled="true"' : '')}>`
                + `         <label for="_cbperCommentList_autoSetReminderUrComment" title="${I18n.t('urce.prefs.AutoSetReminderUrCommentTitle')}" urceprefs="perCommentList" class="URCE-label${((perCListSettings.autoSetReminderUrComment_useDefault) ? ' urceDisabled' : '')}">${I18n.t('urce.prefs.AutoSetReminderUrComment')}</label>`
                + '      </div>'
                + `      <input type="checkbox" style="float:right;" id="_cbperCommentList_autoSetReminderUrComment_useDefault" urceprefs="perCommentList" class="urceSettingsCheckbox" title="${I18n.t('urce.prefs.UseDefault')}" ${((perCListSettings.autoSetReminderUrComment_useDefault) ? 'checked="true"' : '')}>`
                + '   </div>'
                + '</div>'
                + '<div class="URCE-controls URCE-textFirst">'
                + '   <div>'
                + '      <div style="width:calc(100% - 18px); display:inline-block">'
                + `         <div style="display:inline;" title="${I18n.t('urce.prefs.TagEmailTitle')}" class="URCE-label${((perCListSettings.tagEmail_useDefault) ? ' urceDisabled' : '')}">${I18n.t('urce.prefs.TagEmail')}:</div>`
                + `         <input type="text" id="_textperCommentList_tagEmail" class="URCE-textInput urceSettingsTextBox${((perCListSettings.tagEmail_useDefault) ? ' urceDisabled' : '')}" urceprefs="perCommentList" value="${perCListSettings.tagEmail}" title="${I18n.t('urce.prefs.TagEmailTitle')}" ${((perCListSettings.tagEmail_useDefault) ? 'disabled="true"' : '')}>`
                + '      </div>'
                + `      <input type="checkbox" style="float:right;" id="_cbperCommentList_tagEmail_useDefault" urceprefs="perCommentList" class="urceSettingsCheckbox" title="${I18n.t('urce.prefs.UseDefault')}" ${((perCListSettings.tagEmail_useDefault) ? 'checked="true"' : '')}>`
                + '   </div>'
                + '   <div>'
                + '      <div style="width:calc(100% - 18px); display:inline-block">'
                + `         <div style="display:inline;" title="${I18n.t('urce.prefs.ReminderDaysTitle')}" class="URCE-label${((perCListSettings.reminderDays_useDefault) ? ' urceDisabled' : '')}" urceprefs="perCommentList">${I18n.t('urce.prefs.ReminderDays')}:</div>`
                + `         <input type="number" id="_numperCommentList_reminderDays" class="URCE-daysInput urceSettingsNumberBox${((perCListSettings.reminderDays_useDefault) ? ' urceDisabled' : '')}" urceprefs="perCommentList" min="0" max="9999" step="1" value="${perCListSettings.reminderDays}" title="${I18n.t('urce.prefs.ReminderDaysTitle')}" ${((perCListSettings.reminderDays_useDefault) ? 'disabled="true"' : '')}>`
                + `         <div style="display:inline;" class="URCE-label${((perCListSettings.reminderDays_useDefault) ? ' urceDisabled' : '')}" urceprefs="perCommentList">${I18n.t('common.time.days', { days: 0 }).replace('0 ', '')}</div>`
                + '      </div>'
                + `      <input type="checkbox" style="float:right;" id="_cbperCommentList_reminderDays_useDefault" urceprefs="perCommentList" class="urceSettingsCheckbox" title="${I18n.t('urce.prefs.UseDefault')}" ${((perCListSettings.reminderDays_useDefault) ? 'checked="true"' : '')}>`
                + '   </div>'
                + '   <div>'
                + '      <div style="width:calc(100% - 18px); display:inline-block">'
                + `         <div style="display:inline;" title="${I18n.t('urce.prefs.CloseDaysTitle')}" class="URCE-label${((perCListSettings.closeDays_useDefault) ? ' urceDisabled' : '')}" urceprefs="perCommentList">${I18n.t('urce.prefs.CloseDays')}:</div>`
                + `         <input type="number" id="_numperCommentList_closeDays" class="URCE-daysInput urceSettingsNumberBox${((perCListSettings.closeDays_useDefault) ? ' urceDisabled' : '')}" urceprefs="perCommentList" min="1" max="9999" step="1" value="${perCListSettings.closeDays}" title="${I18n.t('urce.prefs.CloseDaysTitle')}" ${((perCListSettings.closeDays_useDefault) ? 'disabled="true"' : '')}>`
                + `         <div style="display:inline;" class="URCE-label${((perCListSettings.closeDays_useDefault) ? ' urceDisabled' : '')}" urceprefs="perCommentList">${I18n.t('common.time.days', { days: 0 }).replace('0 ', '')}</div>`
                + '      </div>'
                + `      <input type="checkbox" style="float:right;" id="_cbperCommentList_closeDays_useDefault" urceprefs="perCommentList" class="urceSettingsCheckbox" title="${I18n.t('urce.prefs.UseDefault')}" ${((perCListSettings.closeDays_useDefault) ? 'checked="true"' : '')}>`
                + '   </div>'
                + '   <div>'
                + '      <div style="width:calc(100% - 18px); display:inline-block">'
                + `         <div style="display:inline;" title="${I18n.t('urce.prefs.CustomTaglineTitle')}" class="URCE-label${((perCListSettings.customTagline_useDefault) ? ' urceDisabled' : '')}">${I18n.t('urce.prefs.CustomTagline')}:</div>`
                + `         <textarea id="_textperCommentList_customTagline" class="URCE-textInput urceSettingsTextBox${((perCListSettings.customTagline_useDefault) ? ' urceDisabled' : '')}" urceprefs="perCommentList" title="${I18n.t('urce.prefs.CustomTaglineTitle')}" style="resize:none;width:230px;height:50px" ${((perCListSettings.customTagline_useDefault) ? 'disabled="true"' : '')}>${perCListSettings.customTagline}</textarea>`
                + '      </div>'
                + `      <input type="checkbox" style="float:right;" id="_cbperCommentList_customTagline_useDefault" urceprefs="perCommentList" class="urceSettingsCheckbox" title="${I18n.t('urce.prefs.UseDefault')}" ${((perCListSettings.customTagline_useDefault) ? 'checked="true"' : '')}>`
                + '   </div>'
                + '</div>';
        $('#URCE-divPerCommentListSettings').html(htmlOut);
        $('input[urceprefs="perCommentList"], textarea[urceprefs="perCommentList"]').off().on('change', function () {
            const settingName = $(this)[0].id.replace(/(_.+perCommentList_)/gmi, '');
            if (this.type === 'checkbox') {
                _settings.perCommentListSettings[_currentCommentList][settingName] = isChecked(this);
                saveSettingsToStorage();
            }
            if (this.type === 'text') {
                const newVal = this.value.trim();
                if ((newVal !== this.value) || (_settings.perCommentListSettings[_currentCommentList][settingName] !== newVal)) {
                    if (newVal !== this.value)
                        this.value = newVal;
                    _settings.perCommentListSettings[_currentCommentList][settingName] = newVal;
                    saveSettingsToStorage();
                    if (settingName === 'tagEmail')
                        changeCommentList(_currentCommentList, false, true);
                }
            }
            if (this.type === 'number') {
                const val = Math.min(9999, Math.max(0, Math.abs((parseInt(this.value) || 0))));
                if ((val !== this.value) || (_settings.perCommentListSettings[_currentCommentList][settingName] !== val)) {
                    if (val !== parseInt(this.value))
                        this.value = val;
                    _settings.perCommentListSettings[_currentCommentList][settingName] = val;
                    saveSettingsToStorage();
                    handleUrLayer('settingsToggle', undefined, getMapUrsObjArr());
                }
            }
            if (settingName.indexOf('_useDefault') > -1) {
                if (!isChecked(this)) {
                    $(this).siblings().removeClass('urceDisabled').removeAttr('disabled');
                    $(this).siblings().children().removeClass('urceDisabled').removeAttr('disabled');
                }
                else {
                    $(this).siblings().addClass('urceDisabled').attr('disabled', true);
                    $(this).siblings().children().addClass('urceDisabled').attr('disabled', true);
                }
                const parentSettingName = settingName.replace('_useDefault', '');
                if (_settings.perCommentListSettings[_currentCommentList][parentSettingName] !== _settings[parentSettingName]) {
                    _settings.perCommentListSettings[_currentCommentList][parentSettingName] = _settings[parentSettingName];
                    if ($(`input[id$="${parentSettingName}"][urceprefs!="perCommentList"]`)[0].type === 'checkbox')
                        $(`#_cbperCommentList_${parentSettingName}`).prop('checked', _settings[parentSettingName]);
                    else if ($(`input[id$="${parentSettingName}"][urceprefs!="perCommentList"]`)[0].type === 'number')
                        $(`#_numperCommentList_${parentSettingName}`).val(_settings[parentSettingName]);
                    else if ($(`input[id$="${parentSettingName}"][urceprefs!="perCommentList"]`)[0].type === 'text')
                        $(`#_textperCommentList_${parentSettingName}`).val(_settings[parentSettingName]);
                    else if ($(`textarea[id$="${parentSettingName}"][urceprefs!="perCommentList"]`).length > 0)
                        $(`#_textperCommentList_${parentSettingName}`).val(_settings[parentSettingName]);
                }
                saveSettingsToStorage();
            }
        });
    }

    function handleError(err) {
        let errorText,
            htmlOut = '<div class="URCE-divLoading">';
        if (err.message?.indexOf('|') > -1) {
            if (err.message.split('|')[0] === 'updateRequired')
                errorText = `${I18n.t('urce.prompts.UpdateRequired')}: ${err.message.split('|')[1]}`;
            htmlOut += errorText;
        }
        else {
            errorText = err.message;
            htmlOut += (err.commentList === 1001) ? I18n.t('urce.prompts.CustomGSheetLoadError') : I18n.t('urce.common.ErrorGeneric');
        }
        logError(errorText);
        _commentListLoaded = false;
        if (err.phase === 'changeList') {
            if (err.staticList) {
                htmlOut += `\n\n${I18n.t('urce.common.Type')}: ${I18n.t('urce.common.Static')}`;
            }
            else if (err.commentList) {
                const commentListInfo = getCommentListInfo(err.commentList);
                htmlOut += `\n\n${I18n.t('urce.common.CommentList')}: ${commentListInfo.name}`
                    + `\n${I18n.t('urce.common.ListOwner')}: ${commentListInfo.listOwner}`;
            }
        }
        htmlOut += '</div>';
        $('#_commentList').empty().append(htmlOut);
        WazeWrap.Alerts.error(_SCRIPT_SHORT_NAME, htmlOut);
        maskBoxes(undefined, true, err.phase, err.maskUrPanel);
    }

    async function initBackgroundTasks(status, phase) {
        logDebug(`${((status === 'enable') ? 'Initializing' : 'Uninitializing')} background tasks.`);
        if (status === 'enable') {
            logDebug('Setting event listeners for UR markers.');
            if (phase === 'init') {
                _initialUrLayerScan = true;
                await handleUrLayer('init', undefined, getMapUrsObjArr());
                _initialUrLayerScan = false;
            }
            if (!_saveButtonObserver.isObserving
                || !_urPanelContainerObserver.isObserving
                || !_urMarkerObserver.isObserving
                || !_urceSidepanelContentObserver.isObserving
                || !_userInfoTabContentObserver.isObserving
            ) {
                logDebug('Enabling MOs.');
                if (!_saveButtonObserver.isObserving) {
                    _saveButtonObserver.observe(document.querySelector('#toolbar .js-save-popover-target'), {
                        childList: false, attributes: true, attributeOldValue: true, characterData: false, characterDataOldValue: false, subtree: false
                    });
                    _saveButtonObserver.isObserving = true;
                }
                if (!_urPanelContainerObserver.isObserving) {
                    _urPanelContainerObserver.observe(document.getElementById('panel-container'), {
                        childList: true, attributes: true, attributeOldValue: true, characterData: false, characterDataOldValue: false, subtree: true
                    });
                    _urPanelContainerObserver.isObserving = true;
                }
                if (!_urMarkerObserver.isObserving) {
                    _urMarkerObserver.observe(W.map.updateRequestLayer.div, {
                        childList: true, attributes: true, attributeOldValue: true, characterData: false, characterDataOldValue: false, subtree: true
                    });
                    _urMarkerObserver.isObserving = true;
                }
                if (!_urceSidepanelContentObserver.isObserving) {
                    _urceSidepanelContentObserver.observe($('#sidepanel-urc-e').parent()[0], {
                        childList: false, attributes: true, attributeOldValue: true, characterData: false, characterOldValue: false, subtree: false
                    });
                    _urceSidepanelContentObserver.isObserving = true;
                }
                if (!_userInfoTabContentObserver.isObserving) {
                    _userInfoTabContentObserver.observe(document.querySelector('#user-info .tab-content'), {
                        childList: true, attributes: false, attributeOldValue: false, subtree: false
                    });
                    _userInfoTabContentObserver.isObserving = true;
                }
            }
            logDebug('Registering event hooks.');
            W.map.events.registerPriority('mousedown', null, mouseDown);
            W.map.events.register('zoomend', undefined, invokeZoomEnd);
            W.map.events.register('mouseup', undefined, mouseUp);
            W.prefs.on('change:isImperial', invokeModeChange);
            W.model.mapUpdateRequests.on('objectsadded', mUrsAdded);
            W.model.mapUpdateRequests.on('objectsremoved', mUrsRemoved);
            W.app.on('change:loadingFeedMapData', handleUrOverflow);
            W.model.states.on('objectsadded', checkRestrictions);
            W.model.countries.on('objectsadded', checkRestrictions);
        }
        else if (status === 'disable') {
            if (_saveButtonObserver.isObserving || _urPanelContainerObserver.isObserving) {
                logDebug('Disabling MOs.');
                if (_saveButtonObserver.isObserving) {
                    _saveButtonObserver.disconnect();
                    _saveButtonObserver.isObserving = false;
                }
                if (_urPanelContainerObserver.isObserving) {
                    _urPanelContainerObserver.disconnect();
                    _urPanelContainerObserver.isObserving = false;
                }
            }
            logDebug('Disabling event listeners for UR markers.');
            const markerMapCollection = { ...W.map.updateRequestLayer.featureMarkers };
            if (markerMapCollection) {
                Object.keys(markerMapCollection).forEach((marker) => {
                    if (markerMapCollection.hasOwnProperty(marker))
                        markerMapCollection[marker].marker.icon.$div.off({ mouseover: markerMouseOver, mouseout: markerMouseOut, click: markerClick }).data('urce_hasListeners', false);
                });
            }
            logDebug('Unregistering map.events event hook.');
            W.map.events.unregister('mousedown', undefined, mouseDown);
            W.map.events.unregister('zoomend', undefined, invokeZoomEnd);
            W.map.events.unregister('mouseup', undefined, mouseUp);
            W.model.mapUpdateRequests.off('objectsadded', mUrsAdded);
            W.model.mapUpdateRequests.off('objectsremoved', mUrsRemoved);
            W.app.off('change:loadingFeedMapData', handleUrOverflow);
            W.model.states.off('objectsadded', checkRestrictions);
            W.model.countries.off('objectsadded', checkRestrictions);
        }
        return Promise.resolve();
    }

    function injectCss() {
        logDebug('Injecting CSS.');
        $('<style = type="text/css">'
            // Comments tab
            + '#sidepanel-urc-e #panel-urce-comments .URCE-Comments { text-decoration:none; cursor:pointer; color: #000000; font-size:11px; }'
            + '#sidepanel-urc-e #panel-urce-comments .URCE-commentListName { padding-top:5px; font-size:10px; }'
            + '#sidepanel-urc-e #panel-urce-comments .URCE-divLoading { text-align:left; color:red; font-size:12px; }'
            + '#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-divComment { padding:0px 4px 0px 4px; }'
            + '#sidepanel-urc-e #panel-urce-comments .URCE-divComment:before, #panel-urce-comments .URCE-divComment.hover:after { '
            + '     content:""; position:absolute; bottom:-2px; width:0px; height:2px; transition:all 0.2s ease-in-out; transition-duration:0.5s; opacity:0;'
            + '}'
            + '#sidepanel-urc-e #panel-urce-comments .URCE-divComment.hover.expand.URCE-openLink:before { left:calc(50%); background-color:#000000; margin-right:5px; }'
            + '#sidepanel-urc-e #panel-urce-comments .URCE-divComment.hover.expand.URCE-openLink:after { right:calc(50%); background-color:#000000; }'
            + '#sidepanel-urc-e #panel-urce-comments .URCE-divComment.hover.expand.URCE-solvedLink:before { left:calc(50%); background-color:#008F00; margin-right:5px; }'
            + '#sidepanel-urc-e #panel-urce-comments .URCE-divComment.hover.expand.URCE-solvedLink:after { right:calc(50%); background-color:#008F00;}'
            + '#sidepanel-urc-e #panel-urce-comments .URCE-divComment.hover.expand.URCE-niLink:before { left:calc(50%); background-color:#E68A00; margin-right:5px; }'
            + '#sidepanel-urc-e #panel-urce-comments .URCE-divComment.hover.expand.URCE-niLink:after { right:calc(50%); background-color:#E68A00; }'
            + '#sidepanel-urc-e #panel-urce-comments .URCE-divComment.hover:hover { cursor:pointer; }'
            + '#sidepanel-urc-e #panel-urce-comments .URCE-divComment.hover.URCE-blankLine:hover { cursor:default; }'
            + '#sidepanel-urc-e #panel-urce-comments .URCE-divComment:hover:after { width:100%; opacity:1; }'
            + '#sidepanel-urc-e #panel-urce-comments .URCE-divComment:hover:before { width:100%; opacity:1; margin-right:5px; }'
            + '#sidepanel-urc-e #panel-urce-comments .URCE-divComment.hover:hover.expand:after { width:50%; }'
            + '#sidepanel-urc-e #panel-urce-comments .URCE-divComment.hover:hover.expand:before { width:50%; margin-right:5px; }'
            + '#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-top:4px; height:14px; 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 { line-height: 15px; }'
            + '#sidepanel-urc-e #panel-urce-comments .URCE-group_body.urStyle { padding-left:23px !important; }'
            + '#sidepanel-urc-e #panel-urce-comments .URCE-controls input[type="checkbox"] { margin:2px; vertical-align:middle; cursor:pointer; }'
            + '#sidepanel-urc-e #panel-urce-comments .URCE-controls label { font-weight:normal; cursor:pointer; display:inline-block; position:relative; padding-left:16px; }'
            // Settings tab
            + '#sidepanel-urc-e #panel-urce-settings .URCE-divDefaultSettings {'
            + '     background-color:lightgray; border-top:1px solid gray; border-bottom:1px solid gray; margin-top:8px; text-align:center; font-size:10px; font-weight:600;'
            + '     text-transform:uppercase;'
            + '}'
            + '#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-textInput { width:175px; height:20px; }'
            + '#sidepanel-urc-e #panel-urce-settings .URCE-textInput2 { width:75px; height:20px; }'
            + '#sidepanel-urc-e #panel-urce-settings .URCE-span { font-size:12px; text-transform:uppercase; cursor:pointer; }'
            + '#sidepanel-urc-e #panel-urce-settings .URCE-controls { padding:5px 0 5px 0; font-size:10px;}'
            + '#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 16px !important; }'
            + '#sidepanel-urc-e #panel-urce-settings .URCE-controls .URCE-textFirst.urceDisabled, .URCE-controls.URCE-textFirst.urceDisabled, div.URCE-label.urceDisabled { color:#808080; }'
            + '#sidepanel-urc-e #panel-urce-settings .URCE-controls .URCE-divDaysInline { display:inline; padding-left:3px; }'
            + '#sidepanel-urc-e #panel-urce-settings .URCE-controls .URCE-divDaysInline.urceDisabled { display:inline; padding-left:2px; cursor:default; color:#808080; }'
            + '#sidepanel-urc-e #panel-urce-settings .URCE-controls input[type="checkbox"] { margin:2px; vertical-align:middle; cursor:pointer; }'
            + '#sidepanel-urc-e #panel-urce-settings .URCE-controls input[type="checkbox"][disabled] { margin:2px; vertical-align:middle; cursor:default; }'
            + '#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; display:inline-block; position:relative; }'
            + '#sidepanel-urc-e #panel-urce-settings .URCE-controls label.urceDisabled { font-weight:normal; cursor:default; color:#808080;  display:inline-block; position:relative; }'
            + '#sidepanel-urc-e #panel-urce-settings .URCE-spreadsheetLink { font-size:11px; text-align:right; }'
            // Tools tab
            + '#sidepanel-urc-e #panel-urce-tools .URCE-divCC { text-align:center; }'
            + '#sidepanel-urc-e #panel-urce-tools .URCE-span { font-size:12px; text-transform:uppercase; cursor:pointer; }'
            + '#sidepanel-urc-e #panel-urce-tools .urceToolsButtonFile {'
            + '     font-size:11px; background-color:lightgray; border:1px solid gray; cursor:default; height:22px; margin-top:6px; border-radius:4px;'
            + '}'
            + '#sidepanel-urc-e #panel-urce-tools .URCE-divRestoreFileError { font-weight:600; margin:6px; font-size:11px; text-align:left; }'
            + '#sidepanel-urc-e #panel-urce-tools #urceGSheetCreateConvert { margin-top:5px; }'
            // disable WME
            + '#urce-disableWme.URCE-disableWme-main { position:absolute; top:0; left:0; width:100%; height:100%; background:rgba(0,0,0,0.75); z-index:2000; }'
            + '#urce-disableWme #urce-disableWme-text.URCE-disableWme-text {'
            + '     position:absolute; padding:8px; top:50%; left:50%; transform:translate(-50%,-50%); border:2px solid black; border-radius:5px;box-shadow:5px 5px 10px black;'
            + '     background-color:#2F96B4; color:white; z-index:20001;'
            + '}'
            + '#urce-disableWme #urce-disableWme-text .URCE-disableWme-text-header { border-bottom:1px solid;padding: 0 5px 5px 5px;font-size: 18px;text-align:center;font-weight: 600; }'
            + '#urce-disableWme #urce-disableWme-text .URCE-disableWme-text-body { border-bottom: 1px solid;padding: 5px 0 0 0; }'
            + '#urce-disableWme #urce-disableWme-text .URCE-disableWme-text-footer { padding-top:5px; text-align:center; }'
            + '#urce-disableWme #urce-disableWme-text li { font-weight:600; font-size:13px; }'
            + '#urce-disableWme #urce-disableWme-text p { font-weight:normal; margin:0; font-size:13px; }'
            + '#urce-disableWme #urce-disableWme-text a { cursor:pointer; }'
            // Common
            + '.urceToolsButton {'
            + '     font-size:11px; margin-left:10px; background-color:lightgray; border:none !important; cursor:default; height:22px; border-radius:4px; border:1px solid gray;'
            + '}'
            + '.urceToolsButton.active, .urceToolsButtonFile:hover, .urceToolsButton:hover { background-color:gray !important; }'
            + '#sidepanel-urc-e .URCE-divDismiss {'
            + '     display:inline-block; float:right; width:16px; height:16px; margin-top:-12px; border-radius:50%; border:1px solid black; background-color:white; text-align:center; cursor:pointer;'
            + '}'
            + '#sidepanel-urc-e .URCE-divWarningBox { background-color:indianred; border:1px solid silver; margin:6px 0 6px 0; font-size:12px; border-radius:4px; padding:5px; font-weight:600; }'
            + '#sidepanel-urc-e .URCE-expandCollapseAll { font-size:10px; margin-bottom:-10px; text-align:right; }'
            + '#sidepanel-urc-e .URCE-expandCollapseAll.urStyle { margin-bottom:unset !important; }'
            + '#sidepanel-urc-e .URCE-expandCollapseAllItem { display:inline; cursor:pointer; }'
            + '#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:13px; font-weight:600; }'
            + '#sidepanel-urc-e .URCE-spanTitle { font-size:14px; font-weight:600; }'
            + '#sidepanel-urc-e .URCE-spanVersion { font-size:11px; margin-left:11px; color:#aaa; }'
            + '#sidepanel-urc-e .URCE-divTabs { padding-right:5px; height:calc(100vh - var(--height-offset)); }'
            + '#sidepanel-urc-e .URCE-navTabs { padding:0 0 6px; }'
            + '#sidepanel-urc-e .URCE-navTabs li { flex-grow:1 !important; }' // Compatibility with FUME "Compress/enhance side panel contents" setting
            + '#panel-urce-comments { padding: 0px !important; width:100% !important; }'
            + '#panel-urce-settings { padding: 0px !important; width:100% !important; }'
            + '#panel-urce-tools { padding: 0px !important; width:100% !important; }'
            // Main Tabs
            + '.URCE-tabIcon { margin-bottom:3px; width:18px; }'
            + '.URCE-urFilteringToggleBtn { margin-left:4px; cursor:pointer; font-size:13px; }'
            + '.URCE-spinner { margin-left:2px; color:lightgray; }'
            // urceDiv
            + '#urceDiv { position:absolute; visibility:hidden; top:0; left:0; z-index:15000; background-color:aliceBlue; border-width:3px; border-style:solid; border-radius:10px;'
            + '     box-shadow:5px 5px 10px silver;'
            + '}'
            + '#urceDiv hr { border-top:1px solid #000000; }'
            + '#urceDiv .urceDivCloseButton { float:right; cursor:pointer; padding:2px 6px; margin-top:-15px; margin-right:-15px; background-color:aliceblue; border-width:3px; border-style:solid;'
            + '     border-radius:10px; box-shadow: 5px 5px 10px silver; font-weight:600;'
            + '}'
            + '#urceDiv .urceDivContent { padding:5px 15px 0px 5px; }'
            + '#urceDiv .urceDivDisablePopups { float:right; cursor:pointer; margin-top:-12px; margin-right:10px; font-size:10px; text-decoration:underline; }'
            // UR panel Manipulation
            + '#panel-container .mapUpdateRequest.panel { width:380px; max-height:87vh; }'
            + '#panel-container .mapUpdateRequest.panel>* { max-height:87vh; }'
            + '#panel-container .mapUpdateRequest.panel .problem-edit .conversation-view .comment-list { padding: 0px 6px; margin-bottom: 6px; max-height: 26vh; }'
            + '#panel-container .mapUpdateRequest.panel .problem-edit .conversation-view .new-comment-form .new-comment-text { margin-bottom: 0px; }'
            + '#panel-container .mapUpdateRequest.panel .problem-edit .conversation-view .comment .comment-title .date.urce { display: flex; justify-content: flex-end; margin-top: -4px; }'
            + '#panel-container .mapUpdateRequest.panel .problem-edit .header { padding-top: 5px; padding-bottom: 5px; font-size: 12px; line-height: 14px; padding-right: 0px; }'
            + '#panel-container .mapUpdateRequest.panel .problem-edit .header .main-title { font-size: 14px; line-height: 14px; }'
            + '#panel-container .mapUpdateRequest.panel .problem-edit .header .dot { top: 6px; }'
            + '#panel-container .mapUpdateRequest.panel .problem-edit .section .content { padding: 5px 12px; font-size: 12px; line-height: 14px; }'
            + '#panel-container .mapUpdateRequest.panel .problem-edit .section .content .URCE-divDesc { max-height: 82px; overflow-y: auto; }'
            + '#panel-container .mapUpdateRequest.panel .problem-edit .section .title { padding: 0 6px 0 6px; font-size: 13px; line-height: 13px; }'
            + '#panel-container .mapUpdateRequest.panel .problem-edit .actions .controls-container { margin-top: -2px; margin-bottom: -2px; text-align: center; }'
            + '#panel-container .mapUpdateRequest.panel .problem-edit .more-info .more-info-checkbox label { font-size: 12px; line-height: 14px; }'
            + '#panel-container .mapUpdateRequest.panel .problem-edit .actions .controls-container label[for|="state"] { height: 22px; width: unset; min-width: 162px; line-height: 26px; margin: 2px; }'
            + '#panel-container .mapUpdateRequest.panel .problem-edit[data-state="open"] .actions .controls-container label[for="state-solved"]  { display: inline-block; }'
            + '#panel-container .mapUpdateRequest.panel .problem-edit[data-state="open"] .actions .controls-container label[for|="state-not-identified"] { display: inline-block; }'
            + '#panel-container .mapUpdateRequest.panel .problem-edit[data-state="solved"] .actions .controls-container label[for|="state-open"], '
            + '     #panel-container .mapUpdateRequest.panel .problem-edit[data-state="not-identified"] .actions .controls-container label[for|="state-open"] { display: inline-block !important; }'
            + '#panel-container .mapUpdateRequest.panel .problem-edit[data-state="not-identified"] .actions .controls-container label[for|="state-not-identified"], '
            + '     #panel-container .mapUpdateRequest.panel .problem-edit[data-state="not-identified"] .actions .controls-container label[for|="state-solved"] { display: none !important; }'
            + '#panel-container .mapUpdateRequest.panel .problem-edit .actions .navigation .waze-plain-btn { height: 30px; line-height: 18px; font-size: 13px; }'
            + '#panel-container .mapUpdateRequest.panel .problem-edit .actions .no-permissions-alert { margin-bottom: 8px; margin-top: 2px; padding: 4px; }'
            // Map content
            + '.urceCountersPill { color:black; position:absolute; top:30px; display:block; width:auto; white-space:nowrap; padding-left:5px; padding-right:5px; border:1px solid; border-radius:25px; }'
            // URCE Lightbox
            + '.urceMaskLightbox { position:absolute; width:100%; height:100%; background:rgba(0,0,0,.75); color:white; display:flex; align-items:center; justify-content:center; }'
            + '.urceMaskLightbox .text { font-weight:800; }'
            + '</style>').appendTo('head');
    }

    function initCommentsTab() {
        logDebug('Initializing Comments tab.');
        $('#panel-urce-comments').empty().append(''
            + `<div id="_divZoomOutLinks" class="URCE-divCCLinks" style="${(_settings.hideZoomOutLinks ? 'display:none;' : '')}">`
            + `     <div id="urceIcon" class="URCE-divIcon"><img src="${GM_info.script.icon}" class="URCE-icon"></div>`
            + `     <a id="zoomOutLink1" class="URCE-Comments" zoomTo="12" title="${I18n.t('urce.commentsTab.ZoomOutLink1Title')}">${I18n.t('urce.commentsTab.ZoomOutLink1')}</a><br>`
            + `     <a id="zoomOutLink2" class="URCE-Comments" zoomTo="14" title="${I18n.t('urce.commentsTab.ZoomOutLink2Title')}">${I18n.t('urce.commentsTab.ZoomOutLink2')}</a><br>`
            + `     <a id="zoomOutLink3" class="URCE-Comments" zoomTo="15" title="${I18n.t('urce.commentsTab.ZoomOutLink3Title')}">${I18n.t('urce.commentsTab.ZoomOutLink3')}</a><br>`
            + '</div>'
            + '<div id="_commentList" class="URCE-divCC"></div>');
        $('a[id^="zoomOutLink"]').off().on('click', function () {
            if ($('#panel-container .mapUpdateRequest .top-section .close-panel').length > 0)
                autoCloseUrPanel();
            W.map.getOLMap().zoomTo(parseInt(this.getAttribute('zoomTo')));
        });
        if (_needTranslation)
            alertBoxInPanel(`URC-E does not currently have a translation for your WME Language Setting (<i>${I18n.currentLocale()}</i>). Translations are setup on a Google Sheet, so they are simple to do.<br><br>If you would like to provide a translation for your WME Language Setting (<i>${I18n.currentLocale()}</i>), please contact ${_SCRIPT_AUTHOR} via forum PM or Discord, or click reply on the forum thread:<br><a href="https://www.waze.com/forum/viewtopic.php?f=819&t=275608#p1920278" target="_blank">https://www.waze.com/forum/viewtopic.php?f=819&t=275608#p1920278</a>`, undefined, true, 9998);
    }

    function initToolsTab() {
        logDebug('Initializing Tools tab.');
        const urStyle = (_settings.commentListStyle === 'urStyle') ? ' urStyle' : '';
        $('#panel-urce-tools').empty().append(''
            + `<div id="expandCollapseAllTools" class="URCE-expandCollapseAll${urStyle}">`
            + `     <div id="URCE-expandAllTools" class="URCE-expandCollapseAllItem">${I18n.t('urce.common.ExpandAll')}</div>`
            + '     <div style="display:inline;"> : </div>'
            + `     <div id="URCE-collapseAllTools" class="URCE-expandCollapseAllItem">${I18n.t('urce.common.CollapseAll')}</div>`
            + '</div>'
            + `<fieldset id="urce-tools-fieldset-settings" class="URCE-field${urStyle}">`
            + `     <legend id="urce-tools-legend-settings" class="URCE-legend${urStyle}"><i class="fa fa-fw fa-chevron-down URCE-chevron"></i><span class="URCE-span">${I18n.t('urce.tabs.Settings')}</span></legend>`
            + '     <div class="URCE-controls URCE-divCC">'
            + `         <button id="_butbackupSettings" urceprefs="tools" class="urceToolsButton" title="${I18n.t('urce.tools.BackupSettingsTitle')}">${I18n.t('urce.common.Backup')}</button>`
            + `         <button id="_butrestoreSettings" urceprefs="tools" class="urceToolsButton" title="${I18n.t('urce.tools.RestoreSettingsTitle')}">${I18n.t('urce.common.Restore')}</button>`
            + `         <button id="_butresetSettings" urceprefs="tools" class="urceToolsButton" title="${I18n.t('urce.tools.ResetSettingsTitle')}">${I18n.t('urce.common.Reset')}</button>`
            + '         <div id="urceRestoreSettingsFile" style="display:none;">'
            + `             <input type="file" id="_filerestoreSettings" urceprefs="tools" class="urceToolsButtonFile" title="${I18n.t('urce.tools.RestoreSettingsSelectFileTitle')}">`
            + `             <div id="urceRestoreSettingsFileError" class="URCE-divRestoreFileError error" style="display:none;">* ${I18n.t('urce.tools.RestoreSettingsFileError')}</div>`
            + '         </div>'
            + '     </div>'
            + '</fieldset>'
            + `<fieldset id="urce-tools-fieldset-customGoogleSpreadsheet" class="URCE-field${urStyle}">`
            + `     <legend id="urce-tools-legend-customGoogleSpreadsheet" class="URCE-legend${urStyle}"><i class="fa fa-fw fa-chevron-down URCE-chevron"></i><span class="URCE-span">${I18n.t('urce.tools.CustomGoogleSpreadsheet')}</span></legend>`
            + '     <div id="urceGSheetCreateConvert" urceprefs="tools">'
            + `         <button id="_butconvertCurrentCustom" urceprefs="tools" class="urceToolsButton" title="${I18n.t('urce.tools.ConvertCurrentCustomTitle')}">${I18n.t('urce.tools.ConvertCurrentCustom')}</button>`
            + `         <button id="_butcreateNewCustom" urceprefs="tools" class="urceToolsButton" title="${I18n.t('urce.tools.CreateNewCustomTitle')}">${I18n.t('urce.tools.CreateNewCustom')}</button>`
            + '     </div>'
            + '</fieldset>');
        $('#_butbackupSettings').off().on('click', () => {
            saveSettingsToStorage();
            const a = document.createElement('a');
            a.href = URL.createObjectURL(new Blob([JSON.stringify({ URCE: _settings })], { type: 'application/json' }));
            a.download = `urce-settings-v${_SCRIPT_VERSION}.json`;
            a.click();
            $(a).remove();
        });
        $('#_butrestoreSettings').off().on('click', function () {
            if (this.className.indexOf('active') === -1) {
                $(this).addClass('active');
                $('#urceRestoreSettingsFile').show();
            }
            else {
                $(this).removeClass('active');
                $('#_filerestoreSettings').val('');
                $('#urceRestoreSettingsFile').hide().removeClass('error');
                $('#urceRestoreSettingsFileError').empty();
            }
        });
        $('#_butresetSettings').off().on('click', () => {
            WazeWrap.Alerts.confirm(
                _SCRIPT_SHORT_NAME,
                I18n.t('urce.prompts.ResetSettingsConfirmation'),
                () => { loadSettingsFromStorage('resetSettings', true); },
                () => { },
                I18n.t('urce.common.Yes'),
                I18n.t('urce.common.No')
            );
        });
        $('#_filerestoreSettings').off().on('change', function () {
            const file = this.files[0];
            if (((file.type === 'application/json') || (file.type === 'text/json')) && (file.size < 102400)) {
                const reader = new FileReader();
                reader.onload = function () {
                    let restoreSettings;
                    try {
                        restoreSettings = $.parseJSON(this.result);
                    }
                    catch (error) {
                        logError('Unable to parse the input file.', error);
                        return $('#_filerestoreSettings').val('');
                    }
                    if (!restoreSettings.hasOwnProperty('URCE')) {
                        logWarning('Invalid URCE settings JSON file.');
                        $('#urceRestoreSettingsFileError').show();
                        $('#urceRestoreSettingsFile').addClass('error');
                    }
                    else {
                        loadSettingsFromStorage(restoreSettings.URCE, false);
                    }
                    return true;
                };
                reader.readAsText(file);
            }
            else {
                logWarning('Invalid URCE settings JSON file.');
                $('#urceRestoreSettingsFileError').show();
                $('#urceRestoreSettingsFile').addClass('error');
            }
        });
        $('#URCE-expandAllTools, #URCE-collapseAllTools').off().on('click', function () {
            const $legends = $('legend[id^="urce-tools-legend"');
            for (let idx = 0; idx < $legends.length; idx++) {
                if (this.id === 'URCE-expandAllTools') {
                    if ($legends[idx].nextElementSibling.className.indexOf('collapse') > -1)
                        $($legends[idx]).click();
                }
                else if (this.id === 'URCE-collapseAllTools') {
                    if ($legends[idx].nextElementSibling.className.indexOf('collapse') === -1)
                        $($legends[idx]).click();
                }
            }
        });
        $('legend[id^="urce-tools-legend-"]').off().on('click', function () {
            $($(this).children()[0]).toggleClass('fa-chevron-down fa-chevron-right');
            $($(this).siblings()[0]).toggleClass('collapse');
        });
        $('#_butconvertCurrentCustom').off().on('click', () => createStaticToGoogleSheet(true));
        $('#_butcreateNewCustom').off().on('click', () => createStaticToGoogleSheet(false));
    }

    function initSettingsTab() {
        logDebug('Initializing Settings tab.');
        const urStyle = (_settings.commentListStyle === 'urStyle') ? ' urStyle' : '',
            buildStandardCbSetting = (setting, urceprefs) => {
                let translationName,
                    translationTitle;
                if (setting === 'disableFilteringBelowZoom') {
                    translationName = I18n.t('urce.prefs.DisableFilteringBelowZoomLevel');
                    translationTitle = I18n.t('urce.prefs.DisableFilteringBelowZoomLevelTitle');
                }
                else if (setting === 'disableFilteringAboveZoom') {
                    translationName = I18n.t('urce.prefs.DisableFilteringAboveZoomLevel');
                    translationTitle = I18n.t('urce.prefs.DisableFilteringAboveZoomLevelTitle');
                }
                else {
                    if (setting === 'hideByStatusOpen')
                        translationName = I18n.t('venues.update_requests.panel.action.open');
                    else if (setting === 'hideByStatusClosed')
                        translationName = I18n.t('urce.urStatus.Closed');
                    else if (setting === 'hideByStatusNotIdentified')
                        translationName = I18n.t('urce.urStatus.NotIdentified');
                    else if (setting === 'hideByStatusSolved')
                        translationName = I18n.t('venues.update_requests.panel.solved');
                    else if (setting === 'hideByStatusClosedBy')
                        translationName = `${I18n.t('urce.common.ClosedBy')}: `;
                    else if (setting === 'hideByTypeIncorrectTurn')
                        translationName = I18n.t('update_requests.types.6');
                    else if (setting === 'hideByTypeIncorrectAddress')
                        translationName = I18n.t('update_requests.types.7');
                    else if (setting === 'hideByTypeIncorrectRoute')
                        translationName = I18n.t('update_requests.types.8');
                    else if (setting === 'hideByTypeMissingRoundabout')
                        translationName = I18n.t('update_requests.types.9');
                    else if (setting === 'hideByTypeGeneralError')
                        translationName = I18n.t('update_requests.types.10');
                    else if (setting === 'hideByTypeTurnNotAllowed')
                        translationName = I18n.t('update_requests.types.11');
                    else if (setting === 'hideByTypeIncorrectJunction')
                        translationName = I18n.t('update_requests.types.12');
                    else if (setting === 'hideByTypeMissingBridgeOverpass')
                        translationName = I18n.t('update_requests.types.13');
                    else if (setting === 'hideByTypeWrongDrivingDirection')
                        translationName = I18n.t('update_requests.types.14');
                    else if (setting === 'hideByTypeMissingExit')
                        translationName = I18n.t('update_requests.types.15');
                    else if (setting === 'hideByTypeMissingRoad')
                        translationName = I18n.t('update_requests.types.16');
                    else if (setting === 'hideByTypeMissingLandmark')
                        translationName = I18n.t('update_requests.types.18');
                    else if (setting === 'hideByTypeBlockedRoad')
                        translationName = I18n.t('update_requests.types.19');
                    else if (setting === 'hideByTypeMissingStreetName')
                        translationName = I18n.t('update_requests.types.21');
                    else if (setting === 'hideByTypeIncorrectStreetPrefixOrSuffix')
                        translationName = I18n.t('update_requests.types.22');
                    else if (setting === 'hideByTypeMissingOrInvalidSpeedLimit')
                        translationName = I18n.t('update_requests.types.23');
                    else if (setting === 'hideByTypeUndefined')
                        translationName = I18n.t('urce.urTypes.Undefined');
                    else
                        translationName = I18n.t(`urce.prefs.${setting.charAt(0).toUpperCase()}${setting.slice(1)}`);
                    translationTitle = I18n.t(`urce.prefs.${setting.charAt(0).toUpperCase()}${setting.slice(1)}Title`);
                }
                let rVal = `<input type="checkbox" id="_cb${setting}" urceprefs="${urceprefs}" class="urceSettingsCheckbox" title="${translationTitle}"${((_settings[setting] === true) ? ' checked' : '')}>`
                    + `<label for="_cb${setting}" urceprefs="${urceprefs}" title="${translationTitle}" class="URCE-label">${translationName}</label>`;
                if (setting === 'autoSendReminders') {
                    rVal += `<div class="URCE-divWarning URCE-divWarningPre">(<div class="URCE-divWarning URCE-divWarningTitle" title="${I18n.t('urce.prefs.AutoSendRemindersWarningTitle')}">${I18n.t('urce.prefs.AutoSendRemindersWarning')}</div><div class="URCE-divWarning">)</div></div>`
                    + `<div style="padding-left:15px; font-style:italic;"><input type="checkbox" id="_cbautoSendRemindersExceptTagged" urceprefs="${urceprefs}" class="urceSettingsCheckbox" title="${I18n.t('urce.prefs.AutoSendRemindersExceptTaggedTitle')}"${((_settings.autoSendRemindersExceptTagged === true) ? ' checked' : '')}>`
                    + `<label for="_cbautoSendRemindersExceptTagged" urceprefs="${urceprefs}" title="${I18n.t('urce.prefs.AutoSendRemindersExceptTaggedTitle')}" class="URCE-label">${I18n.t('urce.prefs.AutoSendRemindersExceptTagged')}</label></div>`;
                }
                if (setting === 'disableFilteringAboveZoom')
                    rVal += `<div class="URCE-divDaysInline"><input type="number" id="_numdisableFilteringAboveZoomLevel" class="URCE-daysInput urceSettingsNumberBox" urceprefs="filtering" min="12" max="22" step="1" value="${_settings.disableFilteringAboveZoomLevel}" title=${translationTitle}"></div>`;
                if (setting === 'disableFilteringBelowZoom')
                    rVal += `<div class="URCE-divDaysInline"><input type="number" id="_numdisableFilteringBelowZoomLevel" class="URCE-daysInput urceSettingsNumberBox" urceprefs="filtering" min="12" max="22" step="1" value="${_settings.disableFilteringBelowZoomLevel}" title=${translationTitle}"></div>`;
                if (setting === 'hideByStatusClosedBy')
                    rVal += `<div class="URCE-divDaysInline"><input type="text" id="_texthideByStatusClosedByUsers" class="urceSettingsTextBox" style="width:150px;height:20px" urceprefs="filtering" value="${_settings.hideByStatusClosedByUsers}" title="${translationTitle}"></div>`;
                if (setting !== 'autoSendReminders')
                    rVal += '<br>';
                return rVal;
            },
            // eslint-disable-next-line arrow-body-style
            buildTagCbSetting = (setting, urceprefs, translationName, translationTitle, lineBreak) => {
                return `<input type="checkbox" id="_cb${setting}" urceprefs="${urceprefs}" class="urceSettingsCheckbox" title="${translationTitle}"${((_settings[setting] === true) ? ' checked' : '')}>`
                    + `<label for="_cb${setting}" urceprefs="${urceprefs}" title="${translationTitle}" class="URCE-label">${translationName}</label>${(lineBreak ? '<br>' : '')}`;
            },
            buildTextFirstTextSetting = (setting, urceprefs) => {
                const translationName = I18n.t(`urce.prefs.${setting.charAt(0).toUpperCase()}${setting.slice(1)}`),
                    translationTitle = I18n.t(`urce.prefs.${setting.charAt(0).toUpperCase()}${setting.slice(1)}Title`);
                return `<div title="${translationTitle}" class="URCE-label" urceprefs="${urceprefs}">${translationName}: <input type="text" id="_text${setting}" class="URCE-textInput urceSettingsTextBox" urceprefs="${urceprefs}" value="${_settings[setting]}" title="${translationTitle}"></div>`;
            },
            buildTextFirstNumSetting = (setting, urceprefs, min, max, step, postText) => {
                const translationName = I18n.t(`urce.prefs.${setting.charAt(0).toUpperCase()}${setting.slice(1)}`),
                    translationTitle = formatText(I18n.t(`urce.prefs.${setting.charAt(0).toUpperCase()}${setting.slice(1)}Title`), false, false, -1);
                let rVal = `<div title="${translationTitle}" class="URCE-label" urceprefs="${urceprefs}">${translationName} <input type="number" id="_num${setting}" class="URCE-daysInput urceSettingsNumberBox" urceprefs="${urceprefs}" value="${_settings[setting]}" title="${translationTitle}" min="${min}" max="${max}" step="${step}">`;
                if (postText)
                    rVal += `<div class="URCE-divDaysInline" urceprefs="${urceprefs}">${postText}</div>`;
                rVal += '</div>';
                return rVal;
            },
            buildTextFirstDoubleCbSetting = (setting1, setting2, textBefore, urceprefs) => {
                let translationName1,
                    translationName2;
                if (setting1 === 'hideFollowing') {
                    translationName1 = I18n.t('urce.common.Following');
                    translationName2 = I18n.t('urce.common.NotFollowing');
                }
                else if ((setting1 === 'hideWithDescription') || (setting1 === 'hideWithCommentsFromMe')) {
                    translationName1 = I18n.t('urce.common.With');
                    translationName2 = I18n.t('urce.common.Without');
                }
                else if ((setting1 === 'hideFirstCommentByMe') || (setting1 === 'hideLastCommentByMe') || (setting1 === 'hideLastCommentByReporter')) {
                    translationName1 = I18n.t('urce.common.Yes');
                    translationName2 = I18n.t('urce.common.No');
                }
                const translationTitle1 = I18n.t(`urce.prefs.${setting1.charAt(0).toUpperCase()}${setting1.slice(1)}Title`),
                    translationTitle2 = I18n.t(`urce.prefs.${setting2.charAt(0).toUpperCase()}${setting2.slice(1)}Title`);
                return `<div style="display:inline-flex;">${textBefore}: `
                    + '     <div style="display:inline;">'
                    + `         <input type="checkbox" id="_cb${setting1}" urceprefs="${urceprefs}" class="urceSettingsCheckbox" title="${translationTitle1}"${((_settings[setting1] === true) ? ' checked' : '')}>`
                    + `         <label for="_cb${setting1}" urceprefs="${urceprefs}" class="URCE-label" title="${translationTitle1}">${translationName1}</label>`
                    + `         <input type="checkbox" id="_cb${setting2}" urceprefs="${urceprefs}" class="urceSettingsCheckbox" title="${translationTitle2}"${((_settings[setting2] === true) ? ' checked' : '')}>`
                    + `         <label for="_cb${setting2}" urceprefs="${urceprefs}" class="URCE-label" title="${translationTitle2}">${translationName2}</label>`
                    + '     </div>'
                    + '</div>';
            },
            buildCbandNumSetting = (setting, numSetting, urceprefs, min, max, step, postText) => {
                let translationName;
                if ((setting === 'hideByAgeOfSubmissionLessThan') || (setting === 'hideByCommentCountLessThan'))
                    translationName = I18n.t('urce.common.LessThan');
                else if ((setting === 'hideByAgeOfSubmissionMoreThan') || (setting === 'hideByCommentCountMoreThan'))
                    translationName = I18n.t('urce.common.MoreThan');
                else
                    translationName = I18n.t(`urce.prefs.${setting.charAt(0).toUpperCase()}${setting.slice(1)}`);
                const translationTitle = I18n.t(`urce.prefs.${setting.charAt(0).toUpperCase()}${setting.slice(1)}Title`);
                let rVal = '<div>'
                    + `<input type="checkbox" id="_cb${setting}" urceprefs="${urceprefs}" class="urceSettingsCheckbox" title="${translationTitle}"${((_settings[setting] === true) ? ' checked' : '')}>`
                    + `<label for="_cb${setting}" urceprefs="${urceprefs}" class="URCE-label" title="${translationTitle}">${translationName}: </label>`
                    + `<div class="URCE-divDaysInline" urceprefs="${urceprefs}"><input type="number" id="_num${numSetting}" class="URCE-daysInput urceSettingsNumberBox" urceprefs="${urceprefs}" value="${_settings[numSetting]}" title="${translationTitle}" min="${min}" max="${max}" step="${step}"></div>`;
                if (postText)
                    rVal += `<div class="URCE-divDaysInline" urceprefs="${urceprefs}">${postText}</div>`;
                rVal += '</div>';
                return rVal;
            },
            buildCbandTextSetting = (setting, textSetting, urceprefs, postText) => {
                const translationName = I18n.t(`urce.prefs.${setting.charAt(0).toUpperCase()}${setting.slice(1)}`),
                    translationTitle = I18n.t(`urce.prefs.${setting.charAt(0).toUpperCase()}${setting.slice(1)}Title`);
                let rVal = '<div style="display:inline-flex;">'
                    + `<input type="checkbox" id="_cb${setting}" urceprefs="${urceprefs}" class="urceSettingsCheckbox" title="${translationTitle}"${((_settings[setting] === true) ? ' checked' : '')}>`
                    + `<label for="_cb${setting}" urceprefs="${urceprefs}" class="URCE-label" title="${translationTitle}">${translationName}: </label>`
                    + `<div class="URCE-divDaysInline" urceprefs="${urceprefs}"><input type="text" id="_text${textSetting}" class="URCE-textInput2 urceSettingsTextBox" urceprefs="${urceprefs}" value="${_settings[textSetting]}" title="${translationTitle}"></div>`;
                if (postText)
                    rVal += `<div class="URCE-divDaysInline" urceprefs="${urceprefs}">${postText}</div>`;
                rVal += '</div>';
                return rVal;
            };
        let htmlOut = ''
            + `<div id="expandCollapseAllSettings" class="URCE-expandCollapseAll${urStyle}">`
            + `     <div id="URCE-expandAllSettings" class="URCE-expandCollapseAllItem">${I18n.t('urce.common.ExpandAll')}</div>`
            + '     <div style="display:inline;"> : </div>'
            + `     <div id="URCE-collapseAllSettings" class="URCE-expandCollapseAllItem">${I18n.t('urce.common.CollapseAll')}</div>`
            + '</div>'
            + `<fieldset id="urce-prefs-fieldset-commentList" class="URCE-field${urStyle}">`
            + `     <legend id="urce-prefs-legend-commentList" class="URCE-legend${urStyle}"><i class="fa fa-fw fa-chevron-down URCE-chevron"></i><span class="URCE-span">${I18n.t('urce.common.CommentList')}</span></legend>`
            + '     <div class="URCE-controls URCE-divCC">'
            + `         <div>${I18n.t('urce.prefs.DefaultList')}: <select id="_selCommentList" title="${I18n.t('urce.prefs.DefaultListTitle')}" urceprefs="commentList">`;
        _commentLists.forEach((cList) => {
            if (cList.status !== 'disabled')
                htmlOut += `<option value="${cList.idx}"${((cList.idx === _settings.commentList) ? ' selected="true"' : '')}>${cList.name}</option>`;
        });
        htmlOut += ''
            + '             </select>'
            + `             <div id="restrictionsEnforcedWarning-Settings" style="float:right;padding-right:4px;color:red;font-size:18px;${(_restrictionsEnforcedTitle ? '' : 'display:none;')}">`
            + `                 <i class="fa fa-exclamation-triangle" aria-hidden="true"${(_restrictionsEnforcedTitle ? ` title="${_restrictionsEnforcedTitle}"` : '')}></i>`
            + '         </div></div>'
            + `         <div>${I18n.t('urce.prefs.CustomSsId')}: <input type="text" id="_textcustomSsId" class="URCE-textInput urceSettingsTextBox" urceprefs="commentList" value="${_settings.customSsId}" title="${I18n.t('urce.prefs.CustomSsIdTitle')}" style="width:100px;margin-left:5px;"></div>`
            + `         <div>${I18n.t('urce.common.Style')}: <select id="_selCommentListStyle" title="${I18n.t('urce.prefs.CommentListStyleTitle')}" urceprefs="commentList">`
            + `             <option value="default"${((_settings.commentListStyle === 'default') ? ' selected="true"' : '')}>${I18n.t('urce.prefs.StyleDefault')}</option>`
            + `             <option value="urStyle"${((_settings.commentListStyle === 'urStyle') ? ' selected="true"' : '')}>${I18n.t('urce.prefs.StyleUrStyle')}</option>`
            + '         </select></div>'
            + `         <input type="checkbox" id="_cbautoSwitchCommentList" urceprefs="commentList" class="urceSettingsCheckbox" title="${I18n.t('urce.prefs.AutoSwitchCommentListTitle')}"${(_settings.autoSwitchCommentList ? ' checked' : '')}">`
            + `         <label for="_cbautoSwitchCommentList" urceprefs="commentList" title="${I18n.t('urce.prefs.AutoSwitchCommentListTitle')}" class="URCE-label">${I18n.t('urce.prefs.AutoSwitchCommentList')}</label>`
            + '         <div id="urceSpreadsheetLinks">'
            + `             <div class="URCE-spreadsheetLink" id="urceSpreadsheetLinkDiv"><a class="URCE-Controls URCE-spreadsheetLink" id="urce-spreadsheet-link" title="${I18n.t('urce.prefs.SpreadsheetLinkTitle')}" href="http://bit.ly/urc-e_ss">URC-E Master Spreadsheet</a></div>`;
        if (_settings.customSsId?.length > 0)
            htmlOut += `    <div class="URCE-spreadsheetLink" id="urceCustomSpreadsheetLinkDiv"><a class="URCE-Controls URCE-spreadsheetLink" id="urce-customSpreadsheet-link" title="${I18n.t('urce.prefs.CustomSpreadsheetLinkTitle')}" href="https://docs.google.com/spreadsheets/d/${_settings.customSsId}/edit">${I18n.t('urce.prefs.CustomSpreadsheetLink')}</a></div>`;
        htmlOut += ''
            + '</div></div></fieldset>'
            // Per Comment List Settings Settings
            + `<fieldset id="urce-prefs-fieldset-perCommentListSettings" class="URCE-field${urStyle}">`
            + `     <legend id="urce-prefs-legend-perCommentListSettings" class="URCE-legend${urStyle}"><i class="fa fa-fw fa-chevron-down URCE-chevron"></i><span class="URCE-span">${I18n.t('urce.prefs.PerCommentListSettings')}</span></legend>`
            + '<div id="URCE-divPerCommentListSettings"></div></fieldset>'
            // URC-E Master Settings
            + `<fieldset id="urce-prefs-fieldset-urc-e-prefs" class="URCE-field${urStyle}">`
            + `     <legend id="urce-prefs-legend-urc-e-prefs" class="URCE-legend${urStyle}"><i class="fa fa-fw fa-chevron-down URCE-chevron"></i><span class="URCE-span">${I18n.t('urce.prefs.UrcePrefs')}</span></legend>`
            + '     <div class="URCE-controls URCE-divCC">';
        ['autoCenterOnUr', 'autoClickOpenSolvedNi', 'autoCloseUrPanel', 'autoSaveAfterSolvedOrNiComment', 'autoSendReminders', 'autoSetNewUrComment', 'autoSetNewUrCommentSlur',
            'autoSetNewUrCommentWithDescription', 'autoSetReminderUrComment', 'autoSwitchToUrCommentsTab', 'autoZoomInOnNewUr', 'autoZoomOutAfterClosePanel', 'autoZoomOutAfterComment',
            'disableDoneNextButtons', 'replaceNextWithDoneButton', 'doubleClickLinkNiComments', 'doubleClickLinkOpenComments', 'doubleClickLinkSolvedComments', 'hideZoomOutLinks',
            'unfollowUrAfterSend', 'enableUrOverflowHandling', 'enableAutoRefresh', 'expandMoreInfo', 'expandShortcuts', 'autoScrollComments',
            'reverseCommentSort'].forEach((setting) => { htmlOut += buildStandardCbSetting(setting, 'urce'); });
        htmlOut += '<div class="URCE-controls URCE-textFirst">';
        htmlOut += buildTextFirstTextSetting('tagEmail', 'commentList');
        htmlOut += buildTextFirstNumSetting('reminderDays', 'urce', '0', '9999', '1', I18n.t('common.time.days', { days: 0 }).replace('0 ', ''));
        htmlOut += buildTextFirstNumSetting('closeDays', 'urce', '1', '9999', '1', I18n.t('common.time.days', { days: 0 }).replace('0 ', ''));
        htmlOut += `<div title="${I18n.t('urce.prefs.CustomTaglineTitle')}" class="URCE-label" urceprefs="commentList">${I18n.t('urce.prefs.CustomTagline')}: `
            + `<textarea id="_textcustomTagline" class="URCE-textInput urceSettingsTextBox" style="width:230px;height:50px;resize:none;" urceprefs="commentList" title="${I18n.t('urce.prefs.CustomTaglineTitle')}">${_settings.customTagline}</textarea>`
            + '</div></div></div></fieldset>'
            // UR Marker Settings
            + `<fieldset id="urce-prefs-fieldset-ur-marker-prefs" class="URCE-field${urStyle}">`
            + `     <legend id="urce-prefs-legend-ur-marker-prefs" class="URCE-legend${urStyle}"><i class="fa fa-fw fa-chevron-down URCE-chevron"></i><span class="URCE-span">${I18n.t('urce.prefs.UrMarkerPrefs')}</span></legend>`
            + '     <div class="URCE-controls URCE-divCC">';
        htmlOut += buildStandardCbSetting('enableUrPillCounts', 'markerMaster');
        htmlOut += buildStandardCbSetting('disableUrMarkerPopup', 'marker-nodisable');
        htmlOut += '    <div class="URCE-textFirst" urceprefs="marker-nodisable">';
        htmlOut += buildTextFirstNumSetting('urMarkerPopupDelay', 'marker-nodisable', '1', '99', '1', ' * 100ms');
        htmlOut += buildTextFirstNumSetting('urMarkerPopupTimeout', 'marker-nodisable', '1', '99', '1', I18n.t('datetime.distance_in_words.x_seconds.other', { count: 30 }).replace('30', ''));
        htmlOut += '    </div>';
        ['doNotShowTagNameOnPill', 'replaceTagNameWithEditorName'].forEach((setting) => { htmlOut += buildStandardCbSetting(setting, 'marker'); });
        htmlOut += buildStandardCbSetting('unstackMarkers', 'marker-nodisable');
        htmlOut += '    <div class="URCE-textFirst" urceprefs="marker-nodisable-stack">';
        htmlOut += buildTextFirstNumSetting('unstackSensitivity', 'marker-nodisable-unstack', '1', '99', '1', undefined);
        htmlOut += buildTextFirstNumSetting('unstackDisableAboveZoom', 'marker-nodisable-unstack', '12', '22', '1', undefined);
        htmlOut += '    </div>'
            // -- Custom markers
            + '<div>'
            + `     <div class="URCE-subHeading" style="font-weight:600;">${I18n.t('urce.prefs.UseCustomMarkersFor')}<br></div>`;
        ['Bog', 'Closures', 'Construction', 'Difficult', 'Events', 'Notes', 'Roadworks', 'Wslm', 'NativeSl', 'Custom'].forEach((setting) => {
            let translationName,
                translationTitle,
                lineBreak = true;
            if ((setting === 'Closures') || (setting === 'Events') || (setting === 'Notes')) {
                translationName = I18n.t(`urce.tags.${setting.slice(0, -1)}`);
                translationTitle = I18n.t(`urce.prefs.${setting.slice(0, -1)}Title`);
            }
            else if (setting === 'NativeSl') {
                translationName = I18n.t('urce.prefs.NativeSpeedLimits');
                translationTitle = I18n.t('urce.prefs.NativeSpeedLimitsTitle');
            }
            else if (setting === 'Custom') {
                translationName = `${I18n.t('urce.common.Custom')}: `;
                translationTitle = I18n.t('urce.prefs.CustomTitle');
                lineBreak = false;
            }
            else {
                translationName = I18n.t(`urce.tags.${setting}`);
                translationTitle = I18n.t(`urce.prefs.${setting}Title`);
            }
            htmlOut += buildTagCbSetting(`customMarkers${setting}`, 'marker-nodisable', translationName, translationTitle, lineBreak);
        });
        htmlOut += `<div class="URCE-divDaysInline"><input type="text" id="_textcustomMarkersCustomText" class="urceSettingsTextBox" style="width:100px;height:20px;" value="${_settings.customMarkersCustomText}" title="${I18n.t('urce.prefs.CustomTitle')}"></div>`
            + '</div></div></fieldset>'
            // UR Filtering Settings
            + `<fieldset id="urce-prefs-fieldset-ur-filtering-prefs" class="URCE-field${urStyle}">`
            + `     <legend id="urce-prefs-legend-ur-filtering-prefs" class="URCE-legend${urStyle}"><i class="fa fa-fw fa-chevron-down URCE-chevron"></i><span class="URCE-span">${I18n.t('urce.prefs.UrFilteringPrefs')}</span></legend>`
            + '     <div class="URCE-controls URCE-divCC">';
        htmlOut += buildStandardCbSetting('enableUrceUrFiltering', 'filteringMaster');
        ['invertFilters', 'hideOutsideEditableArea', 'doNotFilterTaggedUrs', 'doNotHideSelectedUr', 'disableFilteringAboveZoom', 'disableFilteringBelowZoom'
        ].forEach((setting) => { htmlOut += buildStandardCbSetting(setting, 'filtering'); });
        // -- Lifecycle
        htmlOut += `     <div class="URCE-subHeading" style="font-weight:600;">${I18n.t(`urce.prefs.LifeCycleStatus${((_settings.invertFilters) ? 'Inverted' : '')}`)}:<br></div>`;
        ['hideWaiting', 'hideUrsCloseNeeded', 'hideUrsReminderNeeded'].forEach((setting) => { htmlOut += buildStandardCbSetting(setting, 'filtering'); });
        // -- Hide by status
        htmlOut += `     <div class="URCE-subHeading" style="font-weight:600;">${I18n.t(`urce.prefs.HideByStatus${((_settings.invertFilters) ? 'Inverted' : '')}`)}:<br></div>`;
        ['hideByStatusOpen', 'hideByStatusClosed', 'hideByStatusNotIdentified', 'hideByStatusSolved',
            'hideByStatusClosedBy'].forEach((setting) => { htmlOut += buildStandardCbSetting(setting, 'filtering'); });
        // -- Hide by type
        htmlOut += `     <div class="URCE-subHeading" style="font-weight:600;">${I18n.t(`urce.prefs.HideByType${((_settings.invertFilters) ? 'Inverted' : '')}`)}:<br></div>`;
        ['hideByTypeBlockedRoad', 'hideByTypeGeneralError', 'hideByTypeIncorrectAddress', 'hideByTypeIncorrectJunction', 'hideByTypeIncorrectRoute', 'hideByTypeIncorrectStreetPrefixOrSuffix',
            'hideByTypeIncorrectTurn', 'hideByTypeMissingBridgeOverpass', 'hideByTypeMissingExit', 'hideByTypeMissingLandmark', 'hideByTypeMissingOrInvalidSpeedLimit', 'hideByTypeMissingRoad',
            'hideByTypeMissingRoundabout', 'hideByTypeMissingStreetName', 'hideByTypeTurnNotAllowed', 'hideByTypeUndefined', 'hideByTypeWrongDrivingDirection'
        ].forEach((setting) => { htmlOut += buildStandardCbSetting(setting, 'filtering'); });
        // -- Hide by tagged
        htmlOut += `     <div class="URCE-subHeading" style="font-weight:600;">${I18n.t(`urce.prefs.HideByTagged${((_settings.invertFilters) ? 'Inverted' : '')}`)}:<br></div>`;
        ['Bog', 'Closure', 'Construction', 'Difficult', 'Event', 'Note', 'Roadworks', 'Wslm'].forEach((setting) => { htmlOut += buildTagCbSetting(`hideByTagged${setting}`, 'filtering', I18n.t(`urce.tags.${setting}`), I18n.t(`urce.prefs.HideByTagged${setting}Title`), true); });
        // -- Hide by age of submission
        htmlOut += `     <div class="URCE-subHeading" style="font-weight:600;">${I18n.t(`urce.prefs.HideByAgeOfSubmission${((_settings.invertFilters) ? 'Inverted' : '')}`)}:<br></div>`;
        htmlOut += buildCbandNumSetting('hideByAgeOfSubmissionLessThan', 'hideByAgeOfSubmissionLessThanDaysOld', 'filtering', '0', '9999', '1', I18n.t('common.time.days', { days: 0 }).replace('0 ', ''));
        htmlOut += buildCbandNumSetting('hideByAgeOfSubmissionMoreThan', 'hideByAgeOfSubmissionMoreThanDaysOld', 'filtering', '0', '9999', '1', I18n.t('common.time.days', { days: 0 }).replace('0 ', ''));
        // -- Hide by description, comments, following
        htmlOut += `     <div class="URCE-subHeading" style="font-weight:600;">${I18n.t(`urce.prefs.DescriptionCommentsFollowing${((_settings.invertFilters) ? 'Inverted' : '')}`)}:<br></div>`
            + '             <div class="URCE-textFirst" urceprefs="filtering">';
        htmlOut += buildTextFirstDoubleCbSetting('hideFollowing', 'hideNotFollowing', I18n.t('urce.common.Following'), 'filtering');
        htmlOut += buildTextFirstDoubleCbSetting('hideWithDescription', 'hideWithoutDescription', I18n.t('objects.venue.fields.description'), 'filtering');
        htmlOut += buildTextFirstDoubleCbSetting('hideWithCommentsFromMe', 'hideWithoutCommentsFromMe', I18n.t('urce.prefs.HideCommentsFromMe'), 'filtering');
        htmlOut += buildTextFirstDoubleCbSetting('hideFirstCommentByMe', 'hideFirstCommentNotByMe', I18n.t('urce.prefs.HideFirstCommentByMe'), 'filtering');
        htmlOut += buildTextFirstDoubleCbSetting('hideLastCommentByMe', 'hideLastCommentNotByMe', I18n.t('urce.prefs.HideLastCommentByMe'), 'filtering');
        htmlOut += buildTextFirstDoubleCbSetting('hideLastCommentByReporter', 'hideLastCommentNotByReporter', I18n.t('urce.prefs.HideLastCommentByReporter'), 'filtering');
        htmlOut += '        </div>';
        htmlOut += buildCbandNumSetting('hideByCommentCountLessThan', 'hideByCommentCountLessThanNumber', 'filtering', '0', '9999', '1', I18n.t('urce.tabs.Comments'));
        htmlOut += buildCbandNumSetting('hideByCommentCountMoreThan', 'hideByCommentCountMoreThanNumber', 'filtering', '0', '9999', '1', I18n.t('urce.tabs.Comments'));
        htmlOut += buildCbandNumSetting('hideByAgeOfFirstCommentLessThan', 'hideByAgeOfFirstCommentLessThanDaysOld', 'filtering', '0', '9999', '1', I18n.t('common.time.days', { days: 0 })
            .replace('0 ', ''));
        htmlOut += buildCbandNumSetting('hideByAgeOfFirstCommentMoreThan', 'hideByAgeOfFirstCommentMoreThanDaysOld', 'filtering', '0', '9999', '1', I18n.t('common.time.days', { days: 0 })
            .replace('0 ', ''));
        htmlOut += buildCbandNumSetting('hideByAgeOfLastCommentLessThan', 'hideByAgeOfLastCommentLessThanDaysOld', 'filtering', '0', '9999', '1', I18n.t('common.time.days', { days: 0 })
            .replace('0 ', ''));
        htmlOut += buildCbandNumSetting('hideByAgeOfLastCommentMoreThan', 'hideByAgeOfLastCommentMoreThanDaysOld', 'filtering', '0', '9999', '1', I18n.t('common.time.days', { days: 0 })
            .replace('0 ', ''));
        htmlOut += buildCbandTextSetting('hideByKeywordIncluding', 'hideByKeywordIncludingKeyword', 'filtering', undefined);
        htmlOut += buildCbandTextSetting('hideByKeywordNotIncluding', 'hideByKeywordNotIncludingKeyword', 'filtering', undefined);
        htmlOut += '        <div style="padding-left:15px;font-style:italic;">';
        htmlOut += buildStandardCbSetting('hideByKeywordCaseInsensitive', 'filtering');
        htmlOut += '        </div>';
        htmlOut += buildCbandTextSetting('hideWithCommentBy', 'hideWithCommentByUsers', 'filtering', undefined);
        htmlOut += buildCbandTextSetting('hideWithoutCommentBy', 'hideWithoutCommentByUsers', 'filtering', undefined);
        htmlOut += '</div></fieldset>';
        $('#panel-urce-settings').empty().append(htmlOut);
        if (!isChecked('#_cbenableUrPillCounts'))
            $('[urceprefs=marker]').prop('disabled', true).addClass('urceDisabled');
        else
            $('[urceprefs=marker]').prop('disabled', false).removeClass('urceDisabled');
        if (!isChecked('#_cbenableUrceUrFiltering'))
            $('[urceprefs=filtering]').prop('disabled', true).addClass('urceDisabled');
        else
            $('[urceprefs=filtering]').prop('disabled', false).removeClass('urceDisabled');
        if (!isChecked('#_cbunstackMarkers'))
            $('[urceprefs$="-unstack"]').prop('disabled', true).addClass('urceDisabled');
        else
            $('[urceprefs$="-unstack"]').prop('disabled', false).removeClass('urceDisabled');
        $('#URCE-expandAllSettings, #URCE-collapseAllSettings').off().on('click', function () {
            const $legends = $('legend[id^="urce-prefs-legend"');
            for (let idx = 0; idx < $legends.length; idx++) {
                if (this.id === 'URCE-expandAllSettings') {
                    if ($legends[idx].nextElementSibling.className.indexOf('collapse') > -1)
                        $($legends[idx]).click();
                }
                else if (this.id === 'URCE-collapseAllSettings') {
                    if ($legends[idx].nextElementSibling.className.indexOf('collapse') === -1)
                        $($legends[idx]).click();
                }
            }
        });
        $('legend[id^="urce-prefs-legend-"]').off().on('click', function () {
            $($(this).children()[0]).toggleClass('fa-chevron-down fa-chevron-right');
            $($(this).siblings()[0]).toggleClass('collapse');
        });
        $('#_selCommentList').off().on('change', function () {
            if ((parseInt(this.value) === 1001) && (!_settings.customSsId || (_settings.customSsId.length < 1))) {
                $(this).val(_currentCommentList);
                WazeWrap.Alerts.error(_SCRIPT_SHORT_NAME, I18n.t('urce.prompts.SetCustomSsIdFirst'));
            }
            else {
                changeCommentList(parseInt(this.value), false, false);
            }
        });
        $('#_selCommentListStyle').off().on('change', function () {
            changeCommentListStyle(this.value);
        });
        $('#urce-spreadsheet-link, #urce-customSpreadsheet-link').off().on('click', function (e) {
            e.preventDefault();
            e.stopPropagation();
            window.open(this.href, '_blank');
        });
        $('.urceSettingsCheckbox').off().on('change', function () {
            let otherSettingName;
            const settingName = $(this)[0].id.substr(3),
                urcePrefs = $(this).attr('urceprefs');
            if (settingName === 'hideFollowing')
                otherSettingName = 'hideNotFollowing';
            if (settingName === 'hideNotFollowing')
                otherSettingName = 'hideFollowing';
            if (settingName === 'hideWithDescription')
                otherSettingName = 'hideWithoutDescription';
            if (settingName === 'hideWithoutDescription')
                otherSettingName = 'hideWithDescription';
            if (settingName === 'hideWithCommentsFromMe')
                otherSettingName = 'hideWithoutCommentsFromMe';
            if (settingName === 'hideWithoutCommentsFromMe')
                otherSettingName = 'hideWithCommentsFromMe';
            if (settingName === 'hideFirstCommentByMe')
                otherSettingName = 'hideFirstCommentNotByMe';
            if (settingName === 'hideFirstCommentNotByMe')
                otherSettingName = 'hideFirstCommentByMe';
            if (settingName === 'hideLastCommentByMe')
                otherSettingName = 'hideLastCommentNotByMe';
            if (settingName === 'hideLastCommentNotByMe')
                otherSettingName = 'hideLastCommentByMe';
            if (settingName === 'hideLastCommentByReporter')
                otherSettingName = 'hideLastCommentNotByReporter';
            if (settingName === 'hideLastCommentNotByReporter')
                otherSettingName = 'hideLastCommentByReporter';
            if (settingName === 'replaceNextWithDoneButton')
                otherSettingName = 'disableDoneNextButtons';
            if (settingName === 'disableDoneNextButtons')
                otherSettingName = 'replaceNextWithDoneButton';
            if (otherSettingName) {
                if (isChecked(this) && isChecked(`#_cb${otherSettingName}`)) {
                    _settings[otherSettingName] = false;
                    $(`#_cb${otherSettingName}`).prop('checked', false);
                }
            }
            if (settingName === 'disableDoneNextButtons') {
                if (isChecked(this))
                    $('#panel-container .content .navigation').css({ display: 'none' });
                else
                    $('#panel-container .content .navigation').css({ display: 'block' });
            }
            if (settingName === 'hideZoomOutLinks') {
                if (isChecked(this))
                    $('div#_divZoomOutLinks').hide();
                else
                    $('div#_divZoomOutLinks').show();
            }
            if (settingName === 'unstackMarkers') {
                if (isChecked(this))
                    $('[urceprefs$="-unstack"]').prop('disabled', false).removeClass('urceDisabled');
                else
                    $('[urceprefs$="-unstack"]').prop('disabled', true).addClass('urceDisabled');
            }
            if (settingName === 'autoClickOpenSolvedNi') {
                if (isChecked('#_cbdoubleClickLinkNiComments')) {
                    _settings.doubleClickLinkNiComments = false;
                    $('#_cbdoubleClickLinkNiComments').prop('checked', false);
                    $('div#URCE-divDoubleClickNi').hide();
                }
                if (isChecked('#_cbdoubleClickLinkOpenComments')) {
                    _settings.doubleClickLinkOpenComments = false;
                    $('#_cbdoubleClickLinkOpenComments').prop('checked', false);
                    $('div#URCE-divDoubleClickOpen').hide();
                }
                if (isChecked('#_cbdoubleClickLinkSolvedComments')) {
                    _settings.doubleClickLinkSolvedComments = false;
                    $('#_cbdoubleClickLinkSolvedComments').prop('checked', false);
                    $('div#URCE-divDoubleClickSolved').hide();
                }
            }
            if (settingName === 'doubleClickLinkOpenComments') {
                if (!isChecked(this)) {
                    $('div#URCE-divDoubleClickOpen').hide();
                }
                else {
                    if (!isChecked('#_cbautoClickOpenSolvedNi')) {
                        _settings.autoClickOpenSolvedNi = true;
                        $('#_cbautoClickOpenSolvedNi').prop('checked', true);
                    }
                    $('div#URCE-divDoubleClickOpen').show();
                }
            }
            if (settingName === 'doubleClickLinkNiComments') {
                if (!isChecked(this)) {
                    $('div#URCE-divDoubleClickNi').hide();
                }
                else {
                    if (!isChecked('#_cbautoClickOpenSolvedNi')) {
                        _settings.autoClickOpenSolvedNi = true;
                        $('#_cbautoClickOpenSolvedNi').prop('checked', true);
                    }
                    $('div#URCE-divDoubleClickNi').show();
                }
            }
            if (settingName === 'doubleClickLinkSolvedComments') {
                if (!isChecked(this)) {
                    $('div#URCE-divDoubleClickSolved').hide();
                }
                else {
                    if (!isChecked('#_cbautoClickOpenSolvedNi')) {
                        _settings.autoClickOpenSolvedNi = true;
                        $('#_cbautoClickOpenSolvedNi').prop('checked', true);
                    }
                    $('div#URCE-divDoubleClickSolved').show();
                }
            }
            if (urcePrefs === 'markerMaster') {
                if (!isChecked(this))
                    $('[urceprefs=marker]').prop('disabled', true).addClass('urceDisabled');
                else
                    $('[urceprefs=marker]').prop('disabled', false).removeClass('urceDisabled');
            }
            if (urcePrefs === 'filteringMaster') {
                if (!isChecked(this)) {
                    $('[urceprefs=filtering]').prop('disabled', true).addClass('urceDisabled');
                    $('#urceUrFilteringToggleBtn').css('color', '#ccc');
                }
                else {
                    $('[urceprefs=filtering]').prop('disabled', false).removeClass('urceDisabled');
                    $('#urceUrFilteringToggleBtn').css('color', '#00bd00');
                }
            }
            _settings[settingName] = isChecked(this);
            if (isChecked($(`input[id="_cbperCommentList_${settingName}_useDefault"]`)) && (isChecked($(`input[id="_cbperCommentList_${settingName}"]`)) !== isChecked(this)))
                $(`input[id="_cbperCommentList_${settingName}"]`).prop('checked', isChecked(this));
            Object.values(_settings.perCommentListSettings).forEach((arr) => {
                if (arr[`${settingName}_useDefault`]) {
                    if (arr[settingName] !== isChecked(this))
                        arr[settingName] = isChecked(this);
                }
            });
            saveSettingsToStorage();
            if (settingName === 'invertFilters')
                initSettingsTab();
            if ((((urcePrefs.indexOf('marker') > -1) || (urcePrefs.indexOf('filtering') > -1)) && (settingName.indexOf('unstack') === -1))
                || (settingName === 'enableUrOverflowHandling')
            )
                handleUrLayer('settingsToggle', undefined, getMapUrsObjArr());
        });
        $('.urceSettingsNumberBox').off().on('change', function () {
            const settingName = $(this)[0].id.substr(4),
                maxVal = (['disableFilteringAboveZoomLevel', 'disableFilteringBelowZoomLevel', 'unstackDisableAboveZoom'].indexOf(settingName) > -1) ? 22 : 9999,
                minVal = (['disableFilteringAboveZoomLevel', 'disableFilteringBelowZoomLevel', 'unstackDisableAboveZoom'].indexOf(settingName) > -1) ? 12 : 0,
                val = Math.min(maxVal, Math.max(minVal, parseInt(Math.abs(parseInt(this.value) || minVal))));
            if ((val !== parseInt(this.value)) || (_settings[settingName] !== val)) {
                if (val !== parseInt(this.value))
                    this.value = val;
                _settings[settingName] = val;
                if (isChecked($(`input[id="_cbperCommentList_${settingName}_useDefault"]`)) && ($(`input[id="_numperCommentList_${settingName}"]`) !== val))
                    $(`input[id="_numperCommentList_${settingName}"]`).val(val);
                Object.values(_settings.perCommentListSettings).forEach((arr) => {
                    if (arr[`${settingName}_useDefault`]) {
                        if (arr[settingName] !== val)
                            arr[settingName] = val;
                    }
                });
                saveSettingsToStorage();
                if (settingName.indexOf('unstack') === -1)
                    handleUrLayer('settingsToggle', undefined, getMapUrsObjArr());
            }
        });
        $('.urceSettingsTextBox').off().on('change', function () {
            const settingName = $(this)[0].id.substr(5),
                val = this.value.trim();
            if ((val !== this.value) || (_settings[settingName] !== val)) {
                if (val !== this.value)
                    this.value = val;
                _settings[settingName] = val;
                if (isChecked($(`input[id="_cbperCommentList_${settingName}_useDefault"]`)) && ($(`input[id="_numperCommentList_${settingName}"]`) !== val))
                    $(`input[id="_textperCommentList_${settingName}"]`).val(val);
                Object.values(_settings.perCommentListSettings).forEach((arr) => {
                    if (arr[`${settingName}_useDefault`]) {
                        if (arr[settingName] !== val)
                            arr[settingName] = val;
                    }
                });
                saveSettingsToStorage();
                if (settingName === 'customSsId') {
                    if (val && (val.length > 0)) {
                        const $customSsIdLink = $('<a>', {
                            class: 'URCE-Controls URCE-spreadsheetLink', id: 'urce-customSpreadsheet-link', title: I18n.t('urce.prefs.CustomSpreadsheetLinkTitle'), href: `https://docs.google.com/spreadsheets/d/${val}/edit`
                        }).text(I18n.t('urce.prefs.CustomSpreadsheetLink')).on('click', function (e) {
                            e.preventDefault();
                            e.stopPropagation();
                            window.open(this.href, '_blank');
                        });
                        if ($('#urceCustomSpreadsheetLinkDiv').length === 0) {
                            $('#urceSpreadsheetLinks').append(
                                $('<div>', { class: 'URCE-spreadsheetLink', id: 'urceCustomSpreadsheetLinkDiv' }).append($customSsIdLink)
                            );
                        }
                        else {
                            $('#urceCustomSpreadsheetLinkDiv').empty().append($customSsIdLink);
                        }
                    }
                    else if ($('#urceCustomSpreadsheetLinkDiv').length > 0) {
                        $('#urceCustomSpreadsheetLinkDiv').remove();
                    }
                }
                if ((settingName !== 'tagEmail') && (settingName !== 'customSsId') && (settingName !== 'customTagline'))
                    handleUrLayer('settingsToggle', undefined, getMapUrsObjArr());
                else if ((settingName === 'tagEmail') || (settingName === 'customTagline') || ((settingName === 'customSsId') && (_currentCommentList === 1001)))
                    changeCommentList(_settings.commentList, false, true);
            }
        });
        /**
         * Disable settings for compatibility
         * Enable Auto Refresh: Disabled 2023.03.29
         *      Due to W.controller.reloadData() causing issues with new Issue Tracker.
         *      Specifically if user had more than 500 URs loaded in tracker, it would reset them back to 500 due to reloadData
         *      wiping and reloading the model data. For now, this setting is disabled and the functionality has been replaced
         *      with the already working handleOverflow function (if the user has it enabled).
         */
        $('#_cbenableAutoRefresh, label[for="_cbenableAutoRefresh"')
            .attr('disabled', true)
            .addClass('urceDisabled')
            .attr('title', `${I18n.t('urce.prefs.DisabledUnusedSettingTitle')}\r\n\r\n${I18n.t('urce.prefs.EnableAutoRefreshTitle')}`);
    }

    async function initGui(firstCall = true) {
        logDebug('Initializing GUI.');
        injectCss();
        if ($('#urceDiv').length === 0) {
            $('body').append('<div id="urceDiv">'
                + '<span class="urceDivCloseButton">X</span>'
                + '<div class="urceDivContent"></div>'
                + `<span class="urceDivDisablePopups">${I18n.t('urce.prefs.DisableUrMarkerPopup')}</span>`
                + '</div>');
        }
        $('#urceDiv .urceDivCloseButton').off().on('click', { doubleClick: false }, hidePopup);
        $('#urceDiv .urceDivDisablePopups').off().on('click', undefined, () => {
            hidePopup(true);
            $('#_cbdisableUrMarkerPopup').click();
        });
        if (firstCall) {
            const { tabLabel, tabPane } = W.userscripts.registerSidebarTab('URC-E');
            tabLabel.innerHTML = `<img id="urceIcon" class="URCE-tabIcon" src="${GM_info.script.icon}">`
                + `<span id="urceUrMarkerProcessingSpinner" class="fa fa-spinner URCE-spinner" title="${I18n.t('urce.mouseOver.URMarkerProcessingInactive')}"></span>`
                + `<span id="urceUrFilteringToggleBtn" class="fa fa-filter URCE-urFilteringToggleBtn" style="color:${(_settings.enableUrceUrFiltering ? '#00bd00' : '#ccc')};" title="${I18n.t('urce.mouseOver.ToggleUrceURFiltering')}">`;
            tabLabel.title = 'URC-E';
            tabPane.innerHTML = `<span class="URCE-spanTitle">${_SCRIPT_SHORT_NAME}</span><span class="URCE-spanVersion">${_SCRIPT_VERSION}</span>`
                + '<div class="URCE-navTabs"><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>`
                + `     <li><a data-toggle="tab" href="#panel-urce-tools" aria-expanded="true">${I18n.t('urce.tabs.Tools')}</a></li>`
                + '</ul></div>'
                + '<div class="tab-content URCE-divTabs" style="--height-offset:0px;">'
                + '     <div class="tab-pane active" id="panel-urce-comments"></div>'
                + '     <div class="tab-pane" id="panel-urce-settings"></div>'
                + '     <div class="tab-pane" id="panel-urce-tools"></div>'
                + '</div></span>';
            Object.assign(tabPane.parentElement.style, { width: 'auto', padding: '0 15px' });
            tabPane.id = 'sidepanel-urc-e';
            await W.userscripts.waitForElementConnected(tabPane);
            $('span#urceUrFilteringToggleBtn').on('click', (evt) => {
                evt.stopPropagation();
                $('#_cbenableUrceUrFiltering').click();
            });
            showScriptInfoAlert();
        }
        initSettingsTab();
        initCommentsTab();
        initToolsTab();
        return Promise.resolve();
    }

    function initCommentLists() {
        return new Promise((resolve, reject) => {
            logDebug('Initializing available comment lists.');
            let errorText;
            $.getJSON(`https://sheets.googleapis.com/v4/spreadsheets/${_URCE_SPREADSHEET_ID}/values/CommentLists!A3:G?key=${_URCE_API_KEY}`, (data) => {
                if (data?.values.length > 0) {
                    let ssFieldNames;
                    const EXPECTED_FIELD_NAMES = ['idx', 'name', 'status', 'type', 'oldVarName', 'Prefix', 'listOwner'],
                        checkFieldNames = (fldName) => ssFieldNames.indexOf(fldName) > -1;
                    for (let entryIdx = 0; entryIdx < data.values.length; entryIdx++) {
                        if (entryIdx === 0) {
                            if (_SCRIPT_VERSION < data.values[entryIdx][0]) {
                                errorText = `updateRequired|${data.values[entryIdx][0]}`;
                                break;
                            }
                        }
                        else if (entryIdx === 1) {
                            ssFieldNames = data.values[entryIdx].map((fldName) => fldName.trim());
                            if (ssFieldNames.length !== EXPECTED_FIELD_NAMES.length)
                                errorText = `Expected ${EXPECTED_FIELD_NAMES.length} columns in comment lists data. Spreadsheet returned ${ssFieldNames.length}.`;
                            else if (!EXPECTED_FIELD_NAMES.every((fldName) => checkFieldNames(fldName)))
                                errorText = `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(', ')}`;
                            if (errorText)
                                break;
                        }
                        else {
                            const output = Object.create({});
                            for (let valIdx = 0; valIdx < data.values[entryIdx].length; valIdx++) {
                                if (ssFieldNames[valIdx] === 'idx')
                                    output[ssFieldNames[valIdx]] = parseInt(data.values[entryIdx][valIdx]);
                                else if (ssFieldNames[valIdx] === 'Prefix')
                                    output.gSheetRange = `${data.values[entryIdx][valIdx]}_Output_(do_not_edit)!A1:A`;
                                else
                                    output[ssFieldNames[valIdx]] = data.values[entryIdx][valIdx];
                            }
                            _commentLists.push(output);
                        }
                    }
                    if (!errorText)
                        _commentLists.sort(dynamicSort('name'));
                }
                else {
                    errorText = errorText || 'No lists available.';
                }
            })
                .fail((response) => {
                    errorText = `Spreadsheet call failed. Code: ${response.status} - Text: ${response.responseText}`;
                })
                .always(() => {
                    if (errorText)
                        logWarning(errorText);
                    if (errorText && (_STATIC_ONLY_USERS.indexOf(W.loginManager.user.userName) > -1)) {
                        _commentLists.push({
                            idx: 1, name: 'Custom', status: 'enabled', type: 'static', oldVarName: 'Custom', listOwner: 'Custom', gSheetRange: ''
                        });
                        _commentLists.push({
                            idx: 3, name: 'USA - SER', status: 'enabled', type: 'static', oldVarName: 'USA_Southeast', listOwner: 'itzwolf', gSheetRange: ''
                        });
                        resolve();
                    }
                    else if (errorText) {
                        reject(new Error(errorText));
                    }
                    else {
                        resolve();
                    }
                });
        });
    }

    function initAutoSwitchArrays() {
        return new Promise((resolve, reject) => {
            logDebug('Initializing auto switch setup.');
            let errorText;
            $.getJSON(`https://sheets.googleapis.com/v4/spreadsheets/${_URCE_SPREADSHEET_ID}/values/CommentLists_AutoSwitch!A3:ZZ?majorDimension=COLUMNS&key=${_URCE_API_KEY}`, (data) => {
                if (data?.values.length > 0) {
                    for (let entryIdx = 0; entryIdx < data.values.length; entryIdx++) {
                        if ((entryIdx === 0) && (_SCRIPT_VERSION < data.values[entryIdx][0])) {
                            errorText = `updateRequired|${data.values[entryIdx][0]}`;
                            break;
                        }
                        if (data.values[entryIdx].length > 3) {
                            const commentListNum = parseInt(data.values[entryIdx][2]);
                            for (let idx = 3; idx < data.values[entryIdx].length; idx++) {
                                const values = data.values[entryIdx][idx].split('.');
                                if ((values[0] === 'state') && !_autoSwitch[values[1]])
                                    _autoSwitch[values[1]] = {};
                                if (values[0] === 'state')
                                    _autoSwitch[values[1]][values[2]] = commentListNum;
                                if ((values[0] === 'country') && !_autoSwitch[values[1]])
                                    _autoSwitch[values[1]] = { ALL: commentListNum };
                            }
                        }
                    }
                }
                else {
                    errorText = errorText || 'No autoswitch data available';
                }
            })
                .fail((response) => {
                    errorText = `Spreadsheet call failed. Code: ${response.status} - Text: ${response.responseText}`;
                })
                .always(() => {
                    if (errorText)
                        logWarning(errorText);
                    if (!errorText || (errorText && (_STATIC_ONLY_USERS.indexOf(W.loginManager.user.userName) > -1)))
                        resolve();
                    else
                        reject(new Error(errorText));
                });
        });
    }

    function initRestrictions() {
        return new Promise((resolve, reject) => {
            logDebug('Initializing restrictions.');
            let errorText;
            $.getJSON(`https://sheets.googleapis.com/v4/spreadsheets/${_URCE_SPREADSHEET_ID}/values/Restrictions!A3:ZZ?majorDimension=COLUMNS&key=${_URCE_API_KEY}`, (data) => {
                if (data?.values.length > 0) {
                    for (let entryIdx = 0; entryIdx < data.values.length; entryIdx++) {
                        if ((entryIdx === 0) && (_SCRIPT_VERSION < data.values[entryIdx][0])) {
                            errorText = `updateRequired|${data.values[entryIdx][0]}`;
                            break;
                        }
                        if (data.values[entryIdx].length > 1) {
                            const restriction = data.values[entryIdx][1];
                            for (let idx = 2; idx < data.values[entryIdx].length; idx++) {
                                const values = data.values[entryIdx][idx].split('.');
                                if (!_restrictions[values[1]])
                                    _restrictions[values[1]] = {};
                                if (values[0] === 'state') {
                                    if (!_restrictions[values[1]][values[2]])
                                        _restrictions[values[1]][values[2]] = {};
                                    _restrictions[values[1]][values[2]][restriction] = ((restriction !== 'reminderDays') && (restriction !== 'closeDays'))
                                        ? (values[3] === 'true')
                                        : parseInt(values[3]);
                                }
                                if (values[0] === 'country') {
                                    if (!_restrictions[values[1]][values[2]])
                                        _restrictions[values[1]][values[2]] = {};
                                    _restrictions[values[1]][values[2]][restriction] = ((restriction !== 'reminderDays') && (restriction !== 'closeDays'))
                                        ? (values[3] === 'true')
                                        : parseInt(values[3]);
                                }
                            }
                        }
                    }
                }
                else {
                    errorText = errorText || 'No restrictions data available';
                }
            })
                .fail((response) => {
                    errorText = `Spreadsheet call failed. Code: ${response.status} - Text: ${response.responseText}`;
                })
                .always(() => {
                    if (errorText)
                        logWarning(errorText);
                    if (!errorText || (errorText && (_STATIC_ONLY_USERS.indexOf(W.loginManager.user.userName) > -1)))
                        resolve();
                    else
                        reject(new Error(errorText));
                });
        });
    }

    async function initFinish(urId, urPanelMissing) {
        checkTimeout({ timeout: 'initUrIdInUrl' });
        if (_initUrIdInUrlObserver?.isObserving) {
            _initUrIdInUrlObserver.isObserving = false;
            _initUrIdInUrlObserver.disconnect();
        }
        await initBackgroundTasks('enable', 'initFinish');
        if (W.model.mapUpdateRequests.getObjectArray().length > _markerCountOnInit)
            await handleUrLayer('init_end', undefined, getMapUrsObjArr());
        _markerCountOnInit = -1;
        maskBoxes(undefined, true, 'init', (urId > 0));
        if (urPanelMissing && W.model.mapUpdateRequests.objects[urId]) {
            logDebug(`UR ${urId} marker in URL. UR panel did not appear after 15 seconds, attempting to activate marker.`);
            openUrPanel(urId, false);
        }
        else if (urPanelMissing) {
            log(`UR ${urId} marker in URL. UR panel did not appear after 15 seconds. Marker not found.`);
        }
        else {
            logDebug(`UR ${urId} marker in URL. Re-opening.`);
            openUrPanel(urId, true);
        }
    }

    function initCheckForUrPanel(urId, tries) {
        checkTimeout({ timeout: 'initUrIdInUrl' });
        if (tries < 150) {
            if ($('#panel-container').children().length === 0) {
                if ($(`.map-problem.user-generated[data-id="${urId}"]`).length > 0)
                    openUrPanel(urId, true);
                else
                    _timeouts.initUrIdInUrl = window.setTimeout(initCheckForUrPanel, 100, urId, ++tries);
            }
        }
        else {
            logWarning(`UR ${urId} found in URL, but UR panel failed to open after 15 seconds.`);
            initFinish(urId, true);
        }
    }

    function checkUrceVersion() {
        if (_IS_ALPHA_VERSION)
            return;
        try {
            const metaUrl = _IS_BETA_VERSION ? dec(_BETA_META_URL) : _PROD_META_URL;
            GM_xmlhttpRequest({
                url: metaUrl,
                onload(res) {
                    const latestVersion = res.responseText.match(/@version\s+(.*)/)[1];
                    if ((latestVersion > _SCRIPT_VERSION) && (latestVersion > (_lastVersionChecked || '0'))) {
                        _lastVersionChecked = latestVersion;
                        WazeWrap.Alerts.info(
                            _SCRIPT_LONG_NAME,
                            `<a href="${(_IS_BETA_VERSION ? dec(_BETA_URL) : _PROD_URL)}" target = "_blank">Version ${latestVersion}</a> is available.<br>Update now to get the latest features and fixes.`,
                            true,
                            false
                        );
                    }
                },
                onerror(res) {
                    // Silently fail with an error message in the console.
                    logError('Upgrade version check:', res);
                }
            });
        }
        catch (err) {
            // Silently fail with an error message in the console.
            logError('Upgrade version check:', err);
        }
    }

    async function onWazeWrapReady() {
        log('Initializing.');
        checkUrceVersion();
        setInterval(checkUrceVersion, 60 * 60 * 1000);
        _wmeUserId = W.loginManager.user.id;
        await loadSettingsFromStorage(false, false);
        const urIdInUrl = parseInt(window.location.search.split('mapUpdateRequest=')[1]);
        Promise.all([loadTranslations(), initCommentLists(), initAutoSwitchArrays(), initRestrictions()]).catch((error) => {
            error.staticList = false;
            error.phase = 'init';
            error.maskUrPanel = (urIdInUrl > 0);
            error.commentList = false;
            handleError(error);
        }).then(() => {
            initGui().then(() => {
                doSpinner('init', true);
                maskBoxes(`${I18n.t('urce.prompts.WaitingOnInit')}<br><br>${I18n.t('urce.common.PleaseWait')}.`, false, 'init', (urIdInUrl > 0));
                checkRestrictions([{ type: 'init' }]).then(() => {
                    buildCommentList(undefined, 'init', false).catch((error) => {
                        handleError(error);
                    })
                        .finally(() => {
                            window.addEventListener('beforeunload', () => { saveSettingsToStorage(); }, false);
                            log(`Fully initialized in ${Math.round(performance.now() - _LOAD_BEGIN_TIME)} ms.`);
                            if (!urIdInUrl || !(urIdInUrl > 0)) {
                                initBackgroundTasks('enable', 'init').then(() => {
                                    if (W.model.mapUpdateRequests.getObjectArray().length > _markerCountOnInit)
                                        handleUrLayer('init_end', undefined, getMapUrsObjArr());
                                    _markerCountOnInit = -1;
                                    maskBoxes(undefined, true, 'init', (urIdInUrl > 0));
                                });
                            }
                            else if ($('#panel-container').children().length === 0) {
                                logDebug(`urId ${urIdInUrl} found in URL, but the UR panel has not shown up yet. Waiting up to 15 seconds.`);
                                _initUrIdInUrlObserver = new MutationObserver((mutations) => {
                                    mutations.forEach((mutation) => {
                                        if ((mutation.type === 'attributes')
                                            && $(mutation.target.parentNode).is('#panel-container')
                                            && (mutation.target.className.indexOf('show') > -1)
                                            && (mutation.oldValue.indexOf('show') === -1)
                                        )
                                            initFinish(urIdInUrl, false);
                                    });
                                });
                                _initUrIdInUrlObserver.observe(document.getElementById('panel-container'), {
                                    childList: true, attributes: true, attributeOldValue: true, characterData: true, characterDataOldValue: true, subtree: true
                                });
                                _initUrIdInUrlObserver.isObserving = true;
                                _timeouts.initUrIdInUrl = window.setTimeout(initCheckForUrPanel, 100, urIdInUrl, 1);
                            }
                            else {
                                initFinish(urIdInUrl, false);
                            }
                            unsafeWindow._mapUpdateRequests = _mapUpdateRequests;
                            doSpinner('init', false);
                        });
                });
            });
        });
    }

    function onWmeReady(tries = 1) {
        if (typeof tries === 'object')
            tries = 1;
        checkTimeout({ timeout: 'onWmeReady' });
        if (WazeWrap?.Ready) {
            logDebug('WazeWrap is ready. Proceeding with initialization.');
            onWazeWrapReady();
        }
        else if (tries < 1000) {
            logDebug(`WazeWrap is not in Ready state. Retrying ${tries} of 1000.`);
            _timeouts.onWmeReady = window.setTimeout(onWmeReady, 200, ++tries);
        }
        else {
            logError('onWmeReady timed out waiting for WazeWrap Ready state.');
        }
    }

    function onWmeInitialized() {
        if (W.userscripts?.state?.isReady) {
            logDebug('W is ready and already in "wme-ready" state. Proceeding with initialization.');
            onWmeReady(1);
        }
        else {
            logDebug('W is ready, but state is not "wme-ready". Adding event listener.');
            document.addEventListener('wme-ready', onWmeReady, { once: true });
        }
    }

    function bootstrap() {
        if (!W) {
            logDebug('W is not available. Adding event listener.');
            document.addEventListener('wme-initialized', onWmeInitialized, { once: true });
        }
        else {
            onWmeInitialized();
        }
    }

    bootstrap();

    function loadTranslations() {
        return new Promise((resolve, reject) => {
            logDebug('Loading translations.');
            let errorText;
            /* FIX FOR WME MISSING DAY NAMES 2021.07.13 - snhroc */
            if (!I18n.translations[I18n.locale].date.day_names) {
                I18n.translations[I18n.locale].date.day_names = [];
                Object.entries(I18n.translations[I18n.locale].date).forEach(([k, v]) => {
                    if (k.indexOf('day_names_') === 0)
                        I18n.translations[I18n.locale].date.day_names.push(v);
                });
            }
            /* END FIX */
            $.getJSON(`https://sheets.googleapis.com/v4/spreadsheets/${_URCE_SPREADSHEET_ID}/values/Script_Translations!A3:AA?key=${_URCE_API_KEY}`, (data) => {
                const translationLocales = [];
                let translations = {};
                if (data?.values.length > 0) {
                    for (let entryIdx = 0; entryIdx < data.values.length; entryIdx++) {
                        if (entryIdx === 0) {
                            if (_SCRIPT_VERSION < data.values[entryIdx][0]) {
                                errorText = `updateRequired|${data.values[entryIdx][0]}`;
                                break;
                            }
                        }
                        else if (entryIdx === 1) {
                            for (let idx = 1; idx < data.values[entryIdx].length; idx++) {
                                translationLocales.push(data.values[entryIdx][idx].trim());
                                translations[data.values[entryIdx][idx].trim()] = {};
                            }
                        }
                        else {
                            let translationDefinition = [];
                            for (let valIdx = 0; valIdx < data.values[entryIdx].length; valIdx++) {
                                if (valIdx === 0) {
                                    translationDefinition = data.values[entryIdx][valIdx].split('.');
                                }
                                else {
                                    const translationLocale = translationLocales[(valIdx - 1)],
                                        translationDef0 = translationDefinition[0],
                                        translationDef1 = translationDefinition[1];
                                    if (typeof translations[translationLocale][translationDef0] === 'undefined')
                                        translations[translationLocale][translationDef0] = {};
                                    translations[translationLocale][translationDef0][translationDef1] = data.values[entryIdx][valIdx]
                                        .replace('$SCRIPT_AUTHOR$', _SCRIPT_AUTHOR).replace(/\\[r|n]+/gm, '\n');
                                }
                            }
                        }
                    }
                }
                else {
                    errorText = errorText || 'No translations available.';
                }
                translations = (!errorText && (data?.values.length > 0)) ? translations : {
                    en: {
                        commentsTab: {
                            ZoomOutLink1: 'Zoom out 12 & close UR',
                            ZoomOutLink1Title: 'Zooms all the way out and closes the UR panel.',
                            ZoomOutLink2: 'Zoom out 14 & close UR',
                            ZoomOutLink2Title: 'Zooms out to level 14 and closes the UR panel.',
                            ZoomOutLink3: 'Zoom out 15 & close UR',
                            ZoomOutLink3Title: 'Zooms out to level 15 and closes the UR panel.'
                        },
                        common: {
                            All: 'All',
                            AutoSwitched: 'auto switched',
                            Backup: 'Backup',
                            ClosedBy: 'Closed by',
                            CollapseAll: 'Collapse all',
                            CommentList: 'Comment List',
                            CurrentCommentListTitle: 'You can change the currently loaded comment list using this drop down.\nChanging this drop down will not be saved as a setting and '
                                        + 'will not change your default list (located on the settings tab).\nThis is only to allow you to quickly switch between lists.',
                            Custom: 'Custom',
                            Description: 'Description',
                            Disabled: 'Disabled',
                            DoubleClickTitle: 'Double click here to send this comment',
                            Enabled: 'Enabled',
                            ErrorGeneric: 'An error has occurred within URC-E. Please contact a WazeDev team member.',
                            ErrorHeader: 'URC-E Error',
                            ExpandAll: 'Expand all',
                            Failed: 'Failed',
                            Finish: 'Finish',
                            Following: 'Following',
                            LessThan: 'Less than',
                            Link: 'Link',
                            List: 'List',
                            ListOwner: 'List owner',
                            Loading: 'Loading',
                            MoreThan: 'More than',
                            NeedsChecked: 'Needs checked',
                            No: 'No',
                            None: 'None',
                            NotApplicable: 'Not applicable',
                            NotFollowing: 'Not following',
                            PleaseWait: 'Please wait',
                            Reset: 'Reset',
                            Restore: 'Restore',
                            Static: 'Static',
                            Style: 'Style',
                            Success: 'Success',
                            Total: 'Total',
                            Type: 'Type',
                            With: 'With',
                            Without: 'Without',
                            Yes: 'Yes'
                        },
                        mouseOver: {
                            CenterInCurrentTab: 'Center in current tab',
                            FirstComment: 'First comment',
                            LastComment: 'Last comment',
                            MarkedAs: 'Marked as',
                            NoDescription: 'No description',
                            NoneByMe: 'none by me',
                            OpenInNewLivemapTab: 'Open in new Livemap tab',
                            OpenInNewTab: 'Open in new tab',
                            ReporterHasCommented: 'Reporter has commented',
                            ToggleUrceURFiltering: 'Toggle URC-E UR filtering.',
                            URMarkerProcessingActive: 'URC-E is currently processing UR markers... Please wait.',
                            URMarkerProcessingInactive: 'URC-E is not currently processing any UR markers.',
                            ViaLivemap: 'via Livemap'
                        },
                        prefs: {
                            DisabledUnusedSettingTitle: 'This setting is currently disabled and unused due to WME compatibility issues.',
                            DefaultList: 'Default list',
                            DefaultListTitle: '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 a WazeDev team member.',
                            CustomSsId: 'Custom Google Spreadsheet ID',
                            CustomSsIdTitle: 'Enter the Google Spreadsheet ID for the Custom comment list you would like to load when you select \'Custom G Sheet\' comment list.',
                            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.',
                            SpreadsheetLink: 'URC-E Master Spreadsheet',
                            SpreadsheetLinkTitle: 'Click here to view the URC-E master spreadsheet.',
                            CustomSpreadsheetLink: 'Custom Google Spreadsheet',
                            CustomSpreadsheetLinkTitle: 'Click here to view your custom google spreadsheet.',
                            StyleDefault: 'Default',
                            StyleUrStyle: 'UR Style',
                            TagEmail: 'Tag email',
                            TagEmailTitle: 'Some comment lists have specific comments that use a replacement tag.\nThe replacement tag is used to specify an email address to send correspondence '
                                        + 'to.\nIf you are setup to use one of these email addresses, please specify it here. If not, leave it blank.',
                            CustomTagline: 'Custom tagline',
                            CustomTaglineTitle: 'Some comments use a custom tagline variable.\nSpecify the text, if any, you would like to have put in place of this custom tagline variable in the comment.',
                            AutoSwitchCommentList: 'Automatically switch comment lists',
                            AutoSwitchCommentListTitle: 'Automatically switch to the comment list designated for the area the UR is in, if there is a list associated with the area.\nOpening a UR '
                                        + 'in an area that does not have a list associated will use the \'Comment List\' you have selected above.',
                            PerCommentListSettings: 'Per comment list settings',
                            SettingsFor: 'Settings for',
                            UseDefault: 'Use \'URC-E Master Settings\' setting',
                            UrcePrefs: 'URC-E Master Settings',
                            AutoCenterOnUr: 'Auto center on UR',
                            AutoCenterOnUrTitle: 'Auto center the map to the selected UR at the current map zoom level.',
                            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, automatically '
                                        + 'select Open, Solved or Not Identified.',
                            AutoCloseUrPanel: 'Auto close UR panel',
                            AutoCloseUrPanelTitle: 'Automatically close the UR panel after you click send on a comment.',
                            AutoSaveAfterSolvedOrNiComment: 'Auto save after solved or NI',
                            AutoSaveAfterSolvedOrNiCommentTitle: 'This will automatically click the save button after you close the UR panel in with you set the UR status to Solved or Not Identified.',
                            AutoSendReminders: 'Auto send reminders',
                            AutoSendRemindersTitle: 'Automatically send the reminder comment to the URs in the map window (as you pan around) you were the last to comment on and it has reached the '
                                        + 'days specified in \'Reminder Days\'. Restricted to editor rank 4+.',
                            AutoSendRemindersWarning: 'WARNING',
                            AutoSendRemindersWarningTitle: 'AUTOMATICALLY SEND REMINDERS at the reminder days setting.\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\'.',
                            AutoSendRemindersExceptTagged: 'Except tagged',
                            AutoSendRemindersExceptTaggedTitle: 'Enabling this setting will prevent the \'Auto send reminders\' setting (if enabled as well) from automatically sending a reminder '
                                        + 'comment to a tagged UR.',
                            AutoSetNewUrComment: 'Auto set new UR comment (w/o description)',
                            AutoSetNewUrCommentTitle: 'Automatically set the default UR comment for the UR type on new (do not already have comments) URs that do not have a description.',
                            AutoSetNewUrCommentSlur: 'Auto set new UR comment (SLURs)',
                            AutoSetNewUrCommentSlurTitle: 'Automatically set the default UR comment for new (do not already have comments) SLURs.',
                            AutoSetNewUrCommentWithDescription: 'Auto set new UR comment (with description)',
                            AutoSetNewUrCommentWithDescriptionTitle: 'Automatically set the default UR comment for the UR type on new (do not already have comments) URs that have a description.',
                            AutoSetReminderUrComment: 'Auto set reminder UR comment',
                            AutoSetReminderUrCommentTitle: 'Automatically 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: 'Automatically switch to the URComments-Enhanced tab when opening a UR. When the UR panel is closed you will be switched back to your previous tab.',
                            AutoZoomInOnNewUr: 'Auto zoom in on new UR',
                            AutoZoomInOnNewUrTitle: 'Automatically zoom in when opening new (no comments) URs.',
                            AutoZoomOutAfterClosePanel: 'Auto zoom out after close panel',
                            AutoZoomOutAfterClosePanelTitle: 'Automatically zoom the map back to the previous zoom after closing the UR panel.',
                            AutoZoomOutAfterComment: 'Auto zoom out after comment',
                            AutoZoomOutAfterCommentTitle: 'Automatically zoom the map back to the previous zoom after clicking send on a UR comment.',
                            DisableDoneNextButtons: 'Disable done / next buttons',
                            DisableDoneNextButtonsTitle: 'Disable the done / next buttons at the bottom of the UR panel.',
                            ReplaceNextWithDoneButton: 'Replace Next UR button with Done button',
                            ReplaceNextWithDoneButtonTitle: 'Replace the Next Update Request button with a Done button.',
                            DoubleClickLinkNiComments: 'Double click link - NI comments',
                            DoubleClickLinkNiCommentsTitle: 'Add an image (extra link) to the \'not identified\' comments. When double clicked it will automatically set and send the UR comment of the '
                                            + 'one you double clicked, and then launch all of the other options that are enabled.',
                            DoubleClickLinkOpenComments: 'Double click link - Open comments',
                            DoubleClickLinkOpenCommentsTitle: 'Add an image (extra link) to the \'open\' comments. When double clicked it will automatically set and send the UR comment of the one you '
                                            + 'double clicked, and then launch all of the other options that are enabled.',
                            DoubleClickLinkSolvedComments: 'Double click link - Solved comments',
                            DoubleClickLinkSolvedCommentsTitle: 'Add an image (extra link) to the \'solved\' comments. When double clicked it will automatically set and send the UR comment of the one '
                                            + 'you double clicked, and then 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.',
                            EnableUrOverflowHandling: 'Enable UR overflow handling',
                            EnableUrOverflowHandlingTitle: 'If this setting is enabled and there are more than 499 URs on the screen, URC-E will attempt to gather more URs and add them to the map, '
                                            + 'if they do not already exist.\nWME does not display more than 500 URs on a single screen on its own.',
                            EnableAutoRefresh: 'Enable auto refresh on zoom / pan',
                            EnableAutoRefreshTitle: 'Reloads the map data when zooming or panning to show URs that may have been missed due to WME\'s 500 UR limit.  Will only reload if the zoom '
                                            + 'level is between 15 and 22, there are not pending edits, and there are more than 499 URs loaded.',
                            ExpandMoreInfo: 'Auto expand more info section',
                            ExpandMoreInfoTitle: 'Automatically expand the more info section of the UR Panel.',
                            ExpandShortcuts: 'Auto expand shortcuts section',
                            ExpandShortcutsTitle: 'Automatically expand the shortcuts section of the UR Panel.',
                            AutoScrollComments: 'Auto scroll comments in UR panel',
                            AutoScrollCommentsTitle: 'Automatically scroll the comments in the UR panel to the bottom if enabled, or top if disabled.',
                            ReverseCommentSort: 'Sort Comments in reverse order',
                            ReverseCommentSortTitle: 'Sort comments in the UR Panel in the reverse order.',
                            UrMarkerPrefs: 'UR Marker Settings',
                            EnableUrPillCounts: 'Enable UR pill counts',
                            EnableUrPillCountsTitle: 'Enable or disable the pill with UR counts on the map marker.',
                            DisableUrMarkerPopup: 'Disable UR marker popups',
                            DisableUrMarkerPopupTitle: 'Do not show the UR popup tooltip when you mouse over a UR marker.',
                            UrMarkerPopupDelay: 'UR marker popup delay',
                            UrMarkerPopupDelayTitle: 'The number of milliseconds (* 100) to delay before the UR marker tooltip will be displayed.',
                            UrMarkerPopupTimeout: 'UR marker popup timeout',
                            UrMarkerPopupTimeoutTitle: 'Specify the number of seconds to leave the UR marker tooltip displayed, while hovering over the marker.\nLeaving the marker, unless to the '
                                            + 'tooltip itself, will cause the tooltip to close.\nEntering the tooltip will cancel the timer and leaving the tooltip will close the tooltip.\nDouble '
                                            + 'click to quickly close.',
                            DoNotShowTagNameOnPill: 'Don\'t show tag name on pill',
                            DoNotShowTagNameOnPillTitle: 'Do not show the tag name on the pill where there is a tag. Example: [NOTE]',
                            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. Similar to how URO+ unstacks markers.',
                            UnstackSensitivity: 'Unstack sensitivity',
                            UnstackSensitivityTitle: 'Specify the sensitivity for which markers are considered stacked.\nDefault: 15',
                            UnstackDisableAboveZoom: 'Unstack disable when zoom level <',
                            UnstackDisableAboveZoomTitle: 'When you zoom out wider than the specified zoom level, marker unstacking will be disabled.\nDefault: 15',
                            UseCustomMarkersFor: 'Use Custom Markers for',
                            BogTitle: 'Replace default UR marker with custom marker for the URs with \'[BOG]\' (boots on ground) / \'[BOTG]\' (boots on the ground) in the description or comments.',
                            ClosureTitle: 'Replace default UR marker with custom marker for the URs with \'[CLOSURE]\' in the description or comments.',
                            ConstructionTitle: 'Replace default UR marker with custom marker for the URs with \'[CONSTRUCTION]\' in the description or comments.',
                            DifficultTitle: 'Replace default UR marker with custom marker for the URs with \'[DIFFICULT]\' in the description or comments.',
                            EventTitle: 'Replace default UR marker with custom marker for the URs with \'[EVENT]\' in the description or comments.',
                            NoteTitle: 'Replace default UR marker with custom marker for the URs with \'[NOTE]\' in the description or comments.',
                            RoadworksTitle: 'Replace default UR marker with custom marker for the URs with \'[ROADWORKS]\' in the description or comments. Used in the UK.',
                            WslmTitle: 'Waze Speed Limit Marker',
                            NativeSpeedLimits: 'Native speed limits',
                            NativeSpeedLimitsTitle: 'Replace default UR marker with custom marker for the URs with \'speed limit\' type.',
                            CustomTitle: 'Replace default UR marker with custom marker for the URs with the text in the box to the right in the description or comments.',
                            UrFilteringPrefs: 'UR Filtering Settings',
                            EnableUrceUrFiltering: 'Enable URC-E UR filtering',
                            EnableUrceUrFilteringTitle: 'Enable or disable URComments-Enhanced built-in UR filtering.',
                            InvertFilters: 'Invert filters',
                            InvertFiltersTitle: 'This will invert the filters you select / do not select.\nExample: If a filter were to match a selected / enabled setting, it would normally '
                                        + 'be hidden.\nBut, if you enable \'invert filters\', the marker would be shown and all others that do not match will be hidden.',
                            HideOutsideEditableArea: 'Hide outside editable area',
                            HideOutsideEditableAreaTitle: 'Hide URs outside your editable area.',
                            DoNotFilterTaggedUrs: 'Do not filter tagged URs',
                            DoNotFilterTaggedUrsTitle: 'Do not filter URs that are tagged with a [] tag. Example: [NOTE]',
                            DoNotHideSelectedUr: 'Do not hide selected UR',
                            DoNotHideSelectedUrTitle: 'Do not hide a UR if it is currently being selected.',
                            DisableFilteringAboveZoomLevel: 'Disable filtering when zoom level <',
                            DisableFilteringAboveZoomLevelTitle: 'Disable UR filtering when zoomed out wider than the specified zoom level. Set to \'12\' to enable all filtering.',
                            DisableFilteringBelowZoomLevel: 'Disable filtering when zoom level >',
                            DisableFilteringBelowZoomLevelTitle: 'Disable UR filtering when zoomed in tighter than the specified zoom level. Set to \'22\' to enable all filtering.',
                            LifeCycleStatus: 'Hide by lifecycle status',
                            LifeCycleStatusInverted: 'Show by lifecycle status',
                            HideWaiting: 'Waiting',
                            HideWaitingTitle: 'Hide/show URs that do not currently need work.',
                            HideUrsCloseNeeded: 'Close needed',
                            HideUrsCloseNeededTitle: 'Hide/show URs that need closing.',
                            HideUrsReminderNeeded: 'Reminders needed',
                            HideUrsReminderNeededTitle: 'Hide/show URs where reminders are needed.',
                            HideByStatus: 'Hide by status',
                            HideByStatusInverted: 'Show by status',
                            HideByStatusOpenTitle: 'Hide/show all open URs.',
                            HideByStatusClosedTitle: 'Hide/show all closed (solved and not identified) URs.',
                            HideByStatusNotIdentifiedTitle: 'Hide/show all closed as not identified URs.',
                            HideByStatusSolvedTitle: 'Hide/show all closed as solved URs.',
                            HideByStatusClosedByTitle: 'Hide/show closed URs that were closed by any of the specified users.\nInput username(s) in text box. Separate multiple entries with a comma.',
                            HideByType: 'Hide by type',
                            HideByTypeInverted: 'Show by type',
                            HideByTypeBlockedRoadTitle: 'Hide/show all blocked road URs.',
                            HideByTypeGeneralErrorTitle: 'Hide/show all general error URs.',
                            HideByTypeIncorrectAddressTitle: 'Hide/show all incorrect address URs.',
                            HideByTypeIncorrectJunctionTitle: 'Hide/show all incorrect junction URs.',
                            HideByTypeIncorrectRouteTitle: 'Hide/show all incorrect route URs.',
                            HideByTypeIncorrectStreetPrefixOrSuffixTitle: 'Hide/show all incorrect street prefix or suffix URs.',
                            HideByTypeIncorrectTurnTitle: 'Hide/show all incorrect turn URs.',
                            HideByTypeMissingBridgeOverpassTitle: 'Hide/show all missing bridge overpass URs.',
                            HideByTypeMissingExitTitle: 'Hide/show all missing exit URs.',
                            HideByTypeMissingLandmarkTitle: 'Hide/show all missing landmark URs.',
                            HideByTypeMissingOrInvalidSpeedLimitTitle: 'Hide/show all missing or invalid speed limit URs.',
                            HideByTypeMissingRoadTitle: 'Hide/show all missing road URs.',
                            HideByTypeMissingRoundaboutTitle: 'Hide/show all missing roundabout URs.',
                            HideByTypeMissingStreetNameTitle: 'Hide/show all missing street name URs.',
                            HideByTypeTurnNotAllowedTitle: 'Hide/show all turn not allowed URs.',
                            HideByTypeUndefinedTitle: 'Hide/show all undefined URs.',
                            HideByTypeWrongDrivingDirectionTitle: 'Hide/show all wrong driving direction URs.',
                            HideByTagged: 'Hide by tagged',
                            HideByTaggedInverted: 'Show by tagged',
                            HideByTaggedBogTitle: 'Hide/show all URs with [BOG] (boots on ground) / [BOTG] (boots on the ground) in description or comments.',
                            HideByTaggedClosureTitle: 'Hide/show all URs with [CLOSURE] in description or comments.',
                            HideByTaggedConstructionTitle: 'Hide/show all URs with [CONSTRUCTION] in description or comments.',
                            HideByTaggedDifficultTitle: 'Hide/show all URs with [DIFFICULT] in description or comments.',
                            HideByTaggedEventTitle: 'Hide/show all URs with [EVENT] in description or comments.',
                            HideByTaggedNoteTitle: 'Hide/show all URs with [NOTE] in description or comments.',
                            HideByTaggedRoadworksTitle: 'Hide/show all URs with [ROADWORKS] in description or comments. Used in the UK.',
                            HideByTaggedWslmTitle: 'Hide/show all URs with [WSLM] in description or comments.',
                            HideByAgeOfSubmission: 'Hide by age of submission',
                            HideByAgeOfSubmissionInverted: 'Show by age of submission',
                            HideByAgeOfSubmissionLessThanTitle: 'Hide/show URs that were originally created less than specified number of days ago.',
                            HideByAgeOfSubmissionMoreThanTitle: 'Hide/show URs that were originally created more than specified number of days ago.',
                            DescriptionCommentsFollowing: 'Hide by description, comment, following',
                            DescriptionCommentsFollowingInverted: 'Show by description, comment, following',
                            HideFollowingTitle: 'Hide/show URs you are following.',
                            HideNotFollowingTitle: 'Hide/show URs you are not following.',
                            HideWithDescriptionTitle: 'Hide/show URs that have a description.',
                            HideWithoutDescriptionTitle: 'Hide/show URs that do not have a description.',
                            HideCommentsFromMe: 'Comments from me',
                            HideWithCommentsFromMeTitle: 'Hide/show URs you have commented on.',
                            HideWithoutCommentsFromMeTitle: 'Hide/show URs you have not commented on.',
                            HideFirstCommentByMe: 'First comment by me',
                            HideFirstCommentByMeTitle: 'Hide/show URs where you were the first person to comment.',
                            HideFirstCommentNotByMeTitle: 'Hide/show URs where someone else was the first person to comment.',
                            HideLastCommentByMe: 'Last comment by me',
                            HideLastCommentByMeTitle: 'Hide/show URs where you are the last person to comment.',
                            HideLastCommentNotByMeTitle: 'Hide/show URs where someone else is the last person to comment.',
                            HideLastCommentByReporter: 'Last comment by reporter',
                            HideLastCommentByReporterTitle: 'Hide/show URs where the reporter is the last person to comment.',
                            HideLastCommentNotByReporterTitle: 'Hide/show URs where the reporter is not the last person to comment.',
                            HideByCommentCountLessThanTitle: 'Hide/show URs that contain less comments than the number specified.',
                            HideByCommentCountMoreThanTitle: 'Hide/show URs that contain more comments than the number specified.',
                            HideByAgeOfFirstCommentLessThan: 'First comment less than',
                            HideByAgeOfFirstCommentLessThanTitle: 'Hide/show URs where the first comment is less than the days specified ago.',
                            HideByAgeOfFirstCommentMoreThan: 'First comment more than',
                            HideByAgeOfFirstCommentMoreThanTitle: 'Hide/show URs where the first comment is more than the days specified ago.',
                            HideByAgeOfLastCommentLessThan: 'Last comment less than',
                            HideByAgeOfLastCommentLessThanTitle: 'Hide/show URs where the last comment is less than the days specified ago.',
                            HideByAgeOfLastCommentMoreThan: 'Last comment more than',
                            HideByAgeOfLastCommentMoreThanTitle: 'Hide/show URs where the last comment is more than the days specified ago.',
                            HideByKeywordIncluding: 'Including keyword',
                            HideByKeywordIncludingTitle: 'Hide/show URs that include the custom word / text specified. Regex compatible.',
                            HideByKeywordNotIncluding: 'Not including keyword',
                            HideByKeywordNotIncludingTitle: 'Hide/show URs that do not include the custom word / text specified. Regex compatible.',
                            HideByKeywordCaseInsensitive: 'Case-insensitive keyword matches',
                            HideByKeywordCaseInsensitiveTitle: 'If enabled, searching for the above including or not including keywords will be done using case insensitive searching.',
                            HideWithCommentBy: 'With comment by',
                            HideWithCommentByTitle: 'Hide/show URs that have been commented on by the specified user(s).\nUse a comma (,) to separate usernames.',
                            HideWithoutCommentBy: 'Without comment by',
                            HideWithoutCommentByTitle: 'Hide/show URs that have not been commented on by the specified user(s).\nUse a comma (,) to separate usernames.',
                            ReminderDays: 'Reminder days',
                            ReminderDaysTitle: 'Number of days to use when calculating UR filtering and when setting and/or sending the reminder comment.\nThis is the number of days since the first '
                                        + 'comment.\nSet to 0 if you do not use reminders.',
                            CloseDays: 'Close days',
                            CloseDaysTitle: 'Number of days to use when calculating UR filtering.\nThis is the number of days since the last comment.\nExample: If you close 4 days after the last '
                                        + 'comment, set to 4.\nAnything less than this time will be considered \'waiting\' as long as there is at least one comment already.',
                            EnableAppendMode: 'Enable append comment mode',
                            EnableAppendModeTitle: 'Enabling append comment mode will allow you to append a comment to the existing text in the new-comment box.\nThe comment is appended with a blank '
                                        + 'line between the existing text and the new text.\nThe status of the UR is set to the status of the new comment you clicked to append.\nIf the comment would end '
                                        + 'up being longer than 2000 characters, append mode will give a warning and not alter the text in the comment box, but the status would have been changed.'
                        },
                        tabs: {
                            Comments: 'Comments',
                            Settings: 'Settings',
                            Tools: 'Tools'
                        },
                        tags: {
                            Bog: '[BOG] / [BOTG]',
                            Closure: '[CLOSURE]',
                            Construction: '[CONSTRUCTION]',
                            Difficult: '[DIFFICULT]',
                            Event: '[EVENT]',
                            Note: '[NOTE]',
                            Roadworks: '[ROADWORKS]',
                            Wslm: '[WSLM]'
                        },
                        tools: {
                            BackupSettingsTitle: 'Download a backup copy of your URC-E settings in JSON format.\nThis backup can be used to restore your settings to another computer, or in the event '
                                        + 'you lose your settings.\n\nNote: Please do not modify the JSON file in any way. The format is crucial to proper restoral of settings.',
                            CreateStep1: 'Open the template spreadsheet: $TEMPLATE_LINK$',
                            CreateStep2: 'Click <i>File</i> then <i>Make a copy</i>.',
                            CreateStep3: 'Give your file a name and specify the folder you want to save it in.',
                            CreateStep4: 'Click <i>Make a copy</i>.',
                            CreateStep5: 'Your sheet will open in a new tab. Look at the URL to get the <i>Spreadsheet ID</i>.<br>It is between <i>https://docs.google.com/spreadsheets/d/</i> and <i>/edit#...</i>.\n<br><b>Example:</b> <i>1JVtw4xwjKmPX_H1Xo1uwyYwxguM-oSi0LotD4lEwmK4</i> $SPREADSHEET_STEP$',
                            CreateStep6: 'Click <i>Share</i> at the top right.',
                            CreateStep7: 'Under <i>General access</i>, click the down arrow next to <i>Restricted</i>.',
                            CreateStep8: 'Select <i>Anyone with the link</i>.',
                            CreateStep9: 'Ensure <i>Viewer</i> is listed as to right and <i>Anyone on the internet with the link can view</i> below.',
                            CreateStep10: 'Click <i>Done</i>.',
                            ConvertStep1: 'Download your converted default UR type responses: $DOWNLOAD_DEFAULTS_LINK$',
                            ConvertStep2: 'Download your converted comment list: $DOWNLOAD_COMMENTS_LINK$',
                            ConvertStep3: 'In your custom Google Sheet, select the <b>CustomComments</b> tab, then click cell <b>B5</b>.',
                            ConvertStep4: 'Click <i>File</i> then <i>Import</i>.',
                            ConvertStep5: 'Click <i>Upload</i> at the top.',
                            ConvertStep6: 'Either click <i>Browse</i> and select your <i>default UR type responses</i> file that was downloaded in step $DOWNLOAD_DEFAULTS_STEP$ '
                                        + '(<b>default_responses_DATETIME.csv</b>),<br>or drag and drop your <i>default UR type responses</i> file to the box.',
                            ConvertStep7: 'For <i>Import location</i> select <b>Replace data at selected cell</b>.',
                            ConvertStep8: 'For <i>Separator type</i> select <i>Custom</i> and put <b>|</b> in the box.',
                            ConvertStep9: 'Click <i>Import data</i>.',
                            ConvertStep10: 'Click cell <b>A25</b>.',
                            ConvertStep11: 'Click <i>File</i> then <i>Import</i>.',
                            ConvertStep12: 'Click <i>Upload</i> at the top.',
                            ConvertStep13: 'Either click <i>Browse</i> and select your <i>comment list</i> file that was downloaded in step $DOWNLOAD_COMMENTS_STEP$ '
                                        + '(<b>comment_list_DATETIME.csv</b>),<br>or drag and drop your <i>comment list</i> file to the box.',
                            ConvertStep14: 'For <i>Import location</i> select <b>Replace data at selected cell</b>.',
                            ConvertStep15: 'For <i>Separator type</i> select <i>Custom</i> and put <b>|</b> in the box.',
                            ConvertStep16: 'Click <i>Import data</i>.',
                            FinalStep1: 'Put your <b>Spreadsheet ID</b> (step $SPREADSHEET_STEP$) in the Custom <i>Google Spreadsheet ID</i> settings box in URC-E settings.',
                            FinalStep2: 'You can now use the <i>Custom G Sheet</i> comment list.',
                            ConvertCreateConvertProcess: 'Convert Process',
                            ConvertCreateCreateProcess: 'Create Process',
                            ConvertCreateSteps: 'Steps',
                            ConvertCurrentCustom: 'Convert current custom',
                            ConvertCurrentCustomTitle: 'Convert your current custom comment list addon to URC-E style Google Sheet for easy maintenance, sharing, etc.',
                            CreateNewCustom: 'Create new custom',
                            CreateNewCustomTitle: 'Create your own URC-E custom comment list Google sheet for easy maintenance, sharing, etc.',
                            CustomGoogleSpreadsheet: 'Custom Google Spreadsheet',
                            IntersectionOf: 'the intersection of $SEG1NAME$ and $SEG2NAME$',
                            IntersectionOfWithCity: 'the intersection of $SEG1NAME$ and $SEG2NAME$ in $SEGCITY$',
                            ResetSettingsTitle: 'Reset all URC-E settings back to their default values.\n\nNote: Almost all settings default to disabled.',
                            RestoreSettingsFileError: 'Invalid URC-E settings JSON file.',
                            RestoreSettingsSelectFileTitle: 'Select the JSON file created by the URC-E \'Backup\' settings button.',
                            RestoreSettingsTitle: 'Restore a backup copy of your URC-E settings from the JSON backup file created with the backup settings button.\n\nNote: Please do not modify the '
                                        + 'JSON file in any way. The format is crucial to proper restoral of settings.',
                            SegmentWithCity: '$SEG1NAME$ in $SEGCITY$',
                            UnknownRoadName: 'unnamed road',
                            UnknownVenueName: 'unnamed venue'
                        },
                        urPanel: {
                            CurrentDate: 'Current date',
                            DriveDate: 'Drive date',
                            InsertCurrentDateCasualTitle: 'Shortcut - Current date (casual): Click this icon to insert the current date into the new comment box at the cursor position (full month '
                                        + 'name, 2-digit day in locale format).',
                            InsertCurrentDateTitle: 'Shortcut - Current date: Click this icon to insert the current date into the new comment box at the cursor position (2-digit month, 2-digit day, '
                                        + '4-digit year in locale format).',
                            InsertCurrentDayOfWeekTitle: 'Shortcut - Current day of week: Click this icon to insert the current day of the week into the new comment box at the cursor position (full '
                                        + 'day of week name in locale language).',
                            InsertCurrentTimeCasualTitle: 'Shortcut - Current time of day (casual): Click this icon to insert the current time of day into the new comment box at the cursor position '
                                        + '(in locale language).\n\n04:00am-11:59am: morning\n12:00pm-05:59pm: afternoon\n06:00pm-08:59pm: evening\n09:00pm-03:59am: night',
                            InsertCurrentTimeTitle: 'Shortcut - Current time of day: Click this icon to insert the current time of day into the new comment box at the cursor position (2-digit hour, '
                                        + '2-digit minute in locale format).',
                            InsertCustomTaglineTitle: 'Shortcut - Custom tagline: Click this icon to insert your custom tagline into the new comment box at the cursor position.',
                            InsertDateTimeCasualModeTitle: 'Shortcut - Drive day and time (fully casual): Click this icon to insert the drive date into the new comment box at the cursor position.\n\n'
                                        + '0 days: this morning, this afternoon, this evening, tonight\n'
                                        + '1 days: yesterday morning, yesterday afternoon, yesterrday evening, last night\n'
                                        + '2-6 days: Day_of_Week morning/afternoon/evening/night\n'
                                        + '7-13 days: last Day_of_Week morning/afternoon/evening/night\n'
                                        + '14-20 days: Day_of_Week before last\n'
                                        + '21-27 days: three weeks ago\n'
                                        + '28-60 days: a few weeks ago\n'
                                        + '61-120 days: a couple months back\n'
                                        + '121+ days: a while ago',
                            InsertDateCasualTitle: 'Shortcut - Drive date (casual): Click this icon to insert the drive date into the new comment box at the cursor position (full month name, 2-digit '
                                        + 'day in locale format).',
                            InsertDateTitle: 'Shortcut - Drive date: Click this icon to insert the drive date into the new comment box at the cursor position (2-digit month, 2-digit day, 4-digit '
                                        + 'year in locale format).',
                            InsertDayOfWeekTitle: 'Shortcut - Drive day of week: Click this icon to insert the drive date day of the week into the new comment box at the cursor position (full day of '
                                        + 'week name in locale language).',
                            InsertDescriptionTitle: 'Shortcut - Description: Click this icon to insert the UR description into the new comment box at the cursor position.',
                            InsertPlaceAddressTitle: 'Shortcut - Place address: Click this icon to either replace \'$PLACE_ADDRESS$\' with the address of the currently selected place, or it will '
                                        + 'insert the address of the currently selected place into the comment box at the current cursor position.',
                            InsertPlaceNameTitle: 'Shortcut - Place name: Click this icon to either replace \'$PLACE_NAME$\' with the name of the currently selected place, or it will insert the name '
                                        + 'of the currently selected place into the comment box at the current cursor position.',
                            InsertSelSegsTitle: 'Shortcut - Segment name(s): Click this icon to either replace \'$SELSEGS$\' (or \'$SELSEGS\') with the name of the currently selected segment(s), or\nit '
                                        + 'will insert the name of the currently selected segment(s) into the comment box at the cursor position.',
                            InsertSelSegsWithCityTitle: 'Shortcut - Segment name(s) with city: Click this icon to either replace \'$SELSEGS_WITH_CITY$\' (or \'$SELSEGS$\' or \'$SELSEGS\') with the '
                                        + 'name and city of the currently selected segment(s), or \nit will insert the name and city of the currently selected segment(s) into the comment box at the '
                                        + 'cursor position.',
                            InsertTimeCasualTitle: 'Shortcut - Drive time of day (casual): Click this icon to insert the drive date time of day into the new comment box at the cursor position (in '
                                        + 'locale language).\n\n04:00am-11:59am: morning\n12:00pm-05:59pm: afternoon\n06:00pm-08:59pm: evening\n09:00pm-03:59am: night',
                            InsertTimeTitle: 'Shortcut - Drive time of day: Click this icon to insert the drive date time of day into the new comment box at the cursor position (2-digit hour, '
                                        + '2-digit minute in locale format).',
                            InsertUrPermalinkTitle: 'Shortcut - Insert permalink to this UR. URL will include your locale, your \'env\', the latitude and longitude of this UR, zoom level of 17, '
                                        + 'mapUpdateRequest of this UR ID number, and s=20489175039 (which ensures the UR layer is turned on).',
                            InsertUrTypeTitle: 'Shortcut - UR Type: Click this icon to insert the UR type into the new comment box at the cursor position.',
                            InsertWazeUsernameTitle: 'Shortcut - Waze username: Click this icon to insert your Waze username into the new comment box at the cursor position.',
                            Shortcuts: 'Shortcuts'
                        },
                        urStatus: {
                            Closed: 'Closed',
                            NotIdentified: 'Not identified'
                        },
                        urTypes: {
                            Undefined: 'Undefined'
                        },
                        prompts: {
                            CommentInsertTimedOut: 'URC-E timed out waiting for the comment text box to become available.',
                            CommentTooLong: 'Appending another comment to the current text will cause the comment to be longer than 2000 characters, which is too long. URC-E did not append the '
                                        + 'selected comment and left the new-comment box with the same text it had.',
                            ConversionLoadAddonFirst: 'In order to convert a custom comments addon script to the new URC-E style Google Sheet, you must first ensure your custom comments addon is '
                                        + 'enabled in TamperMonkey and selected in URC-E\'s Comment List settings box.',
                            CustomGSheetLoadError: 'An error has occurred loading your URC-E custom comment Google sheet.<br><br>Please make sure you have done the following:\n<ul><li>Set the correct <b><i>Custom '
                                        + 'Google Spreadsheet ID</i></b> (should be the ID of the copy created in the previous step) set on the settings tab.\n<li>Ensured your spreadsheet is shared '
                                        + 'to everyone with a link can view.\n</ul>If all these have been done and you are still having issues, please contact a WazeDev team member.',
                            NoCommentBox: 'URC-E: Unable to find the comment box! In order for this script to work, you need to have a UR open.',
                            PlaceAddressFound: 'The selected comment contains \'$PLACE_ADDRESS$\'.\n\nIn order to replace this text with the place address, please\nselect a place and click the place '
                                        + 'address shortcut button in the UR panel.',
                            PlaceAddressInsertError: 'In order to use the <i class="fa fa-map-marker" aria-hidden="true"></i> button in the UR Panel, you must first select a place.',
                            PlaceNameFound: 'The selected comment contains \'$PLACE_NAME$\'.\n\nIn order to replace this text with the place name, please\nselect a place and click the place name '
                                        + 'shortcut button in the UR panel.',
                            PlaceNameInsertError: 'In order to use the <i class="fa fa-home" aria-hidden="true"></i> button in the UR Panel, you must first select a place.',
                            ReminderMessageAuto: 'Automatically sent reminder message to UR(s)',
                            ResetSettings: 'Reset Settings',
                            ResetSettingsComplete: 'Settings reset complete',
                            ResetSettingsConfirmation: 'Are you sure you want to reset URC-E settings back to their default values?',
                            RestoreSettings: 'Restore Settings',
                            RestoreSettingsComplete: 'Settings restore complete',
                            RestoreSettingsConfirmation: 'Would you like to proceed with restoring your URC-E settings?',
                            RestoreSettingsInvalidSettings: 'Invalid settings removed from JSON file',
                            RestoreSettingsNumOfSettings: 'Number of settings to restore',
                            RestoreSettingsRetainedSettings: 'Retained current settings',
                            RestrictionsEnforced: 'Restrictions enforced!',
                            RestrictionsEnforcedTitle: 'The following restrictions have been enforced',
                            SelSegsFound: 'The selected comment contains \'$SELSEGS$\'.\n\nIn order to replace this text with the road name(s), please\nselect one or two segments and click the road '
                                        + 'button in the\nUR panel.',
                            SelSegsInsertError: 'In order to use the <i class="fa fa-road" aria-hidden="true"></i> button in the UR Panel, you must first select one or two segments.',
                            SetCustomSsIdFirst: 'Before you can select <b><i>Custom G Sheet</i></b> as your comment list, you must first set the <b><i>Custom Google Spreadsheet ID</i></b> setting on '
                                        + 'the URC-E settings tab.',
                            SwitchingCommentLists: 'Switching comment lists',
                            TimedOutWaitingStatic: 'Timed out waiting for the static list to become available. Is it enabled?',
                            UpdateRequired: 'You are using an older version of URC-E, which has caused an error. Please update to at least version',
                            UrOverflowErrorWithoutOverflowEnabled: 'WME will not load more than 500 URs per screen. Some URs may be missing. You can try to enable overflow handling, or zoom in and refresh.',
                            VarFound: 'The selected comment contains variables: $VARSFOUND$.\n\nUntil these are replaced, the send button is disabled.',
                            WaitingOnInit: 'Waiting for URC-E to fully initialize',
                            WaitingToGetUrId: 'Waiting to get urId'
                        },
                        time: {
                            ACoupleMonthsAgo: 'a couple months ago',
                            AFewWeeksAgo: 'a few weeks ago',
                            Afternoon: 'afternoon',
                            AWhileBack: 'a while ago',
                            Evening: 'evening',
                            Morning: 'morning',
                            LastNight: 'last night',
                            LastWeekTOD: 'last $DAY_NAME$ $CASUAL_TOD$',
                            Night: 'night',
                            ThisCasualTOD: 'this $THIS_CASUAL_TOD$',
                            ThisWeekTOD: '$DAY_NAME$ $CASUAL_TOD$',
                            Tonight: 'tonight',
                            TwoWeeksAgo: '$DAY_NAME$ before last',
                            ThreeWeeksAgo: 'three weeks ago',
                            YesterdayCasualTOD: 'yesterday $YESTERDAY_CASUAL_TOD$'
                        }
                    }
                };
                translations['en-US'] = { ...translations.en };
                const locale = I18n.currentLocale();
                if (translations[locale]) {
                    Object.keys(translations[locale]).forEach((obj1) => {
                        if (typeof translations[locale][obj1] === 'object') {
                            Object.keys(translations[locale][obj1]).forEach((obj2) => {
                                if (translations[locale][obj1][obj2].length === 0)
                                    delete (translations[locale][obj1][obj2]);
                            });
                        }
                        else if ((typeof translations[locale][obj1] === 'string') && (translations[locale][obj1].length === 0)) {
                            delete (translations[locale][obj1]);
                        }
                    });
                }
                I18n.translations[locale].urce = $.extend(true, {}, translations.en, translations[locale]);
                if (Object.keys(translations).indexOf(locale) === -1)
                    _needTranslation = true;
                if (errorText)
                    logWarning(errorText);
            })
                .fail((response) => {
                    errorText = `Spreadsheet call failed. Code: ${response.status} - Text: ${response.responseText}`;
                })
                .always(() => {
                    if (errorText && (_STATIC_ONLY_USERS.indexOf(W.loginManager.user.userName) === -1))
                        reject(new Error(errorText));
                    else
                        resolve();
                });
        });
    }

    // jQuery function to set cursor position in input
    $.fn.selectRange = function (start, end) {
        if (end === undefined)
            end = start;
        return this.each(function () {
            if ('selectionStart' in this) {
                this.selectionStart = start;
                this.selectionEnd = end;
            }
            else if (this.setSelectionRange) {
                this.setSelectionRange(start, end);
            }
            else if (this.createTextRange) {
                const range = this.createTextRange();
                range.collapse(true);
                range.moveEnd('character', end);
                range.moveStart('character', start);
                range.select();
            }
        });
    };

    // Modified version of uroClickify() -- Thank you!!
    function clickify(desc, suffix) {
        const terminators = [' ', ',', ')', ']', '\r', '\n'];
        if ((desc === null) || (desc === undefined) || (desc === ''))
            return '';
        desc = desc.replace(/<\/?[^>]+(>|$)/g, '');
        if (desc !== 'null') {
            if (desc.indexOf('http') > -1) {
                const links = desc.split('http');
                desc = '';
                for (let i = 0; i < links.length; i++) {
                    if (links[i][2] === '/') {
                        links[i] = `http${links[i]}`;
                        let linkEndPos = links[i].length + 1;
                        for (let j = 0; j < terminators.length; j++) {
                            if (links[i].indexOf(terminators[j]) !== -1)
                                linkEndPos = Math.min(linkEndPos, links[i].indexOf(terminators[j]));
                        }
                        let descPostLink = '';
                        if (linkEndPos < links[i].length) {
                            descPostLink = links[i].slice(linkEndPos);
                            links[i] = links[i].slice(0, linkEndPos);
                        }
                        desc += `<a target="_wazeUR" href="${links[i]}">${links[i]}</a>${descPostLink}`;
                    }
                    else {
                        desc += links[i];
                    }
                }
            }
            desc = desc.replace(/\n/g, '<br>');
            return desc + suffix;
        }
        return '';
    }

    // Date to Days, courtesy of URO+. Thank you!
    function uroDateToDays(dateToConvert) {
        const dateNow = new Date(),
            elapsedSinceEpoch = dateNow.getTime(),
            elapsedSinceEvent = elapsedSinceEpoch - dateToConvert;
        dateNow.setHours(0);
        dateNow.setMinutes(0);
        dateNow.setSeconds(0);
        dateNow.setMilliseconds(0);
        const elapsedSinceMidnight = elapsedSinceEpoch - dateNow.getTime();
        dateNow.setHours(24);
        const pendingUntilMidnight = elapsedSinceEpoch - dateNow.getTime();
        if ((elapsedSinceEvent < elapsedSinceMidnight) && (elapsedSinceEvent > pendingUntilMidnight)) {
            // event occurred today...
            return 0;
        }
        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));
        }
        // event occurred at some point prior to midnight this morning, so return a minimum value of 1...
        return (1 + Math.floor((elapsedSinceEvent - elapsedSinceMidnight) / 86400000));
    }
}
)();