WME Place Harmonizer

Harmonizes, formats, and locks a selected place

目前为 2023-04-21 提交的版本。查看 最新版本

// ==UserScript==
// @name        WME Place Harmonizer
// @namespace   WazeUSA
// @version     2023.04.21.005
// @description Harmonizes, formats, and locks a selected place
// @author      WMEPH Development Group
// @include     /^https:\/\/(www|beta)\.waze\.com\/(?!user\/)(.{2,6}\/)?editor\/?.*$/
// @require     https://gf.qytechs.cn/scripts/24851-wazewrap/code/WazeWrap.js
// @require     https://gf.qytechs.cn/scripts/37486-wme-utils-hoursparser/code/WME%20Utils%20-%20HoursParser.js
// @require     https://cdnjs.cloudflare.com/ajax/libs/lz-string/1.4.4/lz-string.min.js
// @license     GNU GPL v3
// @connect     gf.qytechs.cn
// @grant       GM_addStyle
// @grant       GM_xmlhttpRequest
// ==/UserScript==

/* global W */
/* global OpenLayers */
/* global _ */
/* global WazeWrap */
/* global LZString */
/* global HoursParser */
/* global I18n */
/* global google */

/* eslint-disable max-classes-per-file */

(function main() {
    'use strict';

    // Script update info

    // BE SURE TO SET THIS TO NULL OR AN EMPTY STRING WHEN RELEASING A NEW UPDATE.
    const _SCRIPT_UPDATE_MESSAGE = '';
    const _CSS = `
    #edit-panel .venue-feature-editor {
        overflow: initial;
    }
    #sidebar .wmeph-pane {
        width: auto;
        padding: 8px !important;
    }
    #WMEPH_banner .wmeph-btn { 
        background-color: #fbfbfb;
        box-shadow: 0 2px 0 #aaa;
        border: solid 1px #bbb;
        font-weight:normal;
        margin-bottom: 2px;
        margin-right:4px
    }
    .wmeph-btn, .wmephwl-btn {
        height: 19px;
        font-family: "Boing", sans-serif;
    }
    .btn.wmeph-btn {
        padding: 0px 3px;
    }
    .btn.wmephwl-btn {
        padding: 0px 1px 0px 2px;
        height: 18px;
        box-shadow: 0 2px 0 #b3b3b3;
    }
    
    #WMEPH_banner .banner-row {
        padding:2px 4px;
        cursor: default;
    }
    #WMEPH_banner .banner-row.red {
        color:#b51212;
        background-color:#f0dcdc;
    }
    #WMEPH_banner .banner-row.blue {
        color:#3232e6;
        background-color:#dcdcf0;
    }
    #WMEPH_banner .banner-row.yellow {
        color:#584a04;
        background-color:#f0f0c2;
    }
    #WMEPH_banner .banner-row.gray {
        color:#3a3a3a;
        background-color:#eeeeee;
    }
    #WMEPH_banner .banner-row.lightgray {
        color:#3a3a3a;
        background-color: #f5f5f5;
    }
    #WMEPH_banner .banner-row .dupe {
        padding-left:8px;
    }
    #WMEPH_banner {
        background-color:#fff;
        color:black; font-size:14px;
        padding-top:8px;
        padding-bottom:8px;
        margin-left:4px;
        margin-right:4px;
        line-height:18px;
        margin-top:2px;
        border: solid 1px #8d8c8c;
        border-radius: 6px;
        margin-bottom: 4px;
    }
    #WMEPH_banner input[type=text] {
        font-size: 13px !important;
        height:22px !important;
        font-family: "Open Sans", Alef, helvetica, sans-serif !important;
    }
    #WMEPH_banner div:last-child {
        padding-bottom: 3px !important;
    }
    #wmeph-run-panel {
        padding-bottom: 6px;
        padding-top: 3px;
        width: 290;
        color: black;
        font-size: 15px;
        margin-right: auto;
        margin-left: 4px;
    }
    #WMEPH_tools div {
        padding-bottom: 2px !important;
    }
    .wmeph-fat-btn {
        padding-left:8px;
        padding-right:8px;
        padding-top:4px;
        margin-right:3px;
        display:inline-block;
        font-weight:normal;
        height:24px;
        font-family: "Boing", sans-serif;
    }
    .ui-autocomplete {
        max-height: 300px;
        overflow-y: auto;
        overflow-x: hidden;
    } 
    .wmeph-hr {
        border-color: #ccc;
    }
    .wmeph-hr {
        border-color: #ccc;
    }
    
    @keyframes highlight {
        0% {
            background: #ffff99; 
        }
        100% {
            background: none;
        }
    }
    
    .highlight {
        animation: highlight 1.5s;
    }

    .google-logo {
        /*font-size: 16px*/
    }
    .google-logo.red{
        color: #ea4335
    }
    .google-logo.blue {
        color: #4285f4
    }
    .google-logo.orange {
        color: #fbbc05
    }
    .google-logo.green {
        color: #34a853
    }
    `;

    let MultiAction;
    let UpdateObject;
    let UpdateFeatureGeometry;
    let UpdateFeatureAddress;
    let OpeningHour;

    const _SCRIPT_VERSION = GM_info.script.version.toString(); // pull version from header
    const _SCRIPT_NAME = GM_info.script.name;
    const _IS_BETA_VERSION = /Beta/i.test(_SCRIPT_NAME); //  enables dev messages and unique DOM options if the script is called "... Beta"
    const _BETA_VERSION_STR = _IS_BETA_VERSION ? 'Beta' : ''; // strings to differentiate DOM elements between regular and beta script
    const _PNH_DATA = { USA: {}, CAN: {} };
    const _CATEGORY_LOOKUP = {};
    const _DEFAULT_HOURS_TEXT = 'Paste hours here';
    const _MAX_CACHE_SIZE = 25000;
    let _wordVariations;
    let _resultsCache = {};
    let _initAlreadyRun = false; // This is used to skip a couple things if already run once.  This could probably be handled better...
    let _textEntryValues = null; // Store the values entered in text boxes so they can be re-added when the banner is reassembled.

    // vars for cat-name checking
    let _hospitalPartMatch;
    let _hospitalFullMatch;
    let _animalPartMatch;
    let _animalFullMatch;
    let _schoolPartMatch;
    let _schoolFullMatch;

    let _attributionEl;
    let _placesService;

    // Userlists
    let _wmephDevList;
    let _wmephBetaList;

    let _shortcutParse;
    let _modifKey = 'Alt+';

    // Whitelisting vars
    let _venueWhitelist;
    const _WL_BUTTON_TEXT = 'WL';
    const _WL_LOCAL_STORE_NAME = 'WMEPH-venueWhitelistNew';
    const _WL_LOCAL_STORE_NAME_COMPRESSED = 'WMEPH-venueWhitelistCompressed';

    // Dupe check vars
    let _dupeLayer;
    let _dupeIDList = [];
    let _dupeHNRangeList;
    let _dupeHNRangeDistList;

    // Web search Window forming:
    let _searchResultsWindowSpecs = `"resizable=yes, top=${Math.round(window.screen.height * 0.1)}, left=${
        Math.round(window.screen.width * 0.3)}, width=${Math.round(window.screen.width * 0.7)}, height=${Math.round(window.screen.height * 0.8)}"`;
    const _SEARCH_RESULTS_WINDOW_NAME = '"WMEPH Search Results"';
    let _wmephMousePosition;
    let _cloneMaster = null;

    // Banner Buttons objects
    let _buttonBanner;
    let _buttonBanner2;
    let _servicesBanner;
    let _dupeBanner;

    let _disableHighlightTest = false; // Set to true to temporarily disable highlight checks immediately when venues change.

    const _USER = {
        ref: null,
        rank: null,
        name: null,
        isBetaUser: false,
        isDevUser: false
    };
    const _SETTING_IDS = {
        sfUrlWarning: 'SFURLWarning', // Warning message for first time using localized storefinder URL.
        gLinkWarning: 'GLinkWarning' // Warning message for first time using Google search to not to use the Google info itself.
    };
    const _URLS = {
        forum: 'https://www.waze.com/forum/posting.php?mode=reply&f=819&t=239985',
        usaPnh: 'https://docs.google.com/spreadsheets/d/1-f-JTWY5UnBx-rFTa4qhyGMYdHBZWNirUTOgn222zMY/edit#gid=0',
        placesWiki: 'https://wazeopedia.waze.com/wiki/USA/Places',
        restAreaWiki: 'https://wazeopedia.waze.com/wiki/USA/Rest_areas#Adding_a_Place',
        uspsWiki: 'https://wazeopedia.waze.com/wiki/USA/Places/Post_office',
        uspsLocationFinder: 'https://tools.usps.com/find-location.htm'
    };
    let _userLanguage;
    // lock levels are offset by one
    const _LOCK_LEVEL_2 = 1;
    const _LOCK_LEVEL_4 = 3;

    // An enum to help clarify flag severity levels
    const _SEVERITY = {
        GREEN: 0,
        BLUE: 1,
        YELLOW: 2,
        RED: 3,
        // 4 isn't used anymore
        PINK: 5
        // TODO: There is also 'lock' and 'lock1' severity. Add those here? Also investigate 'adLock' severity (is it still useful in WME???).
    };
    let _catTransWaze2Lang; // pulls the category translations
    let _tempPNHURL = '';
    const EV_PAYMENT_METHOD = {
        APP: 'APP',
        CREDIT: 'CREDIT',
        DEBIT: 'DEBIT',
        MEMBERSHIP_CARD: 'MEMBERSHIP_CARD',
        ONLENE_PAYMENT: 'ONLINE_PAYMENT',
        PLUG_IN_AUTO_CHARGER: 'PLUG_IN_AUTO_CHARGE',
        OTHER: 'OTHER'
    };
    // Common payment types found at: https://wazeopedia.waze.com/wiki/USA/Places/EV_charging_station
    const COMMON_EV_PAYMENT_METHODS = {
        'Blink Charging': [EV_PAYMENT_METHOD.APP, EV_PAYMENT_METHOD.MEMBERSHIP_CARD, EV_PAYMENT_METHOD.PLUG_IN_AUTO_CHARGER, EV_PAYMENT_METHOD.OTHER],
        ChargePoint: [EV_PAYMENT_METHOD.APP, EV_PAYMENT_METHOD.CREDIT, EV_PAYMENT_METHOD.MEMBERSHIP_CARD],
        'Electrify America': [EV_PAYMENT_METHOD.APP, EV_PAYMENT_METHOD.CREDIT, EV_PAYMENT_METHOD.DEBIT, EV_PAYMENT_METHOD.MEMBERSHIP_CARD, EV_PAYMENT_METHOD.PLUG_IN_AUTO_CHARGER],
        EVgo: [EV_PAYMENT_METHOD.APP, EV_PAYMENT_METHOD.CREDIT, EV_PAYMENT_METHOD.DEBIT, EV_PAYMENT_METHOD.PLUG_IN_AUTO_CHARGER],
        SemaConnect: [EV_PAYMENT_METHOD.APP, EV_PAYMENT_METHOD.MEMBERSHIP_CARD, EV_PAYMENT_METHOD.OTHER],
        Tesla: [EV_PAYMENT_METHOD.PLUG_IN_AUTO_CHARGER]
    };
    const _WME_SERVICES_ARRAY = ['VALLET_SERVICE', 'DRIVETHROUGH', 'WI_FI', 'RESTROOMS', 'CREDIT_CARDS', 'RESERVATIONS', 'OUTSIDE_SEATING',
        'AIR_CONDITIONING', 'PARKING_FOR_CUSTOMERS', 'DELIVERIES', 'TAKE_AWAY', 'CURBSIDE_PICKUP', 'WHEELCHAIR_ACCESSIBLE', 'DISABILITY_PARKING'];
    const _COLLEGE_ABBREVIATIONS = 'USF|USFSP|UF|UCF|UA|UGA|FSU|UM|SCP|FAU|FIU';
    // Change place.name to title case
    const _TITLECASE_SETTINGS = {
        ignoreWords: 'an|and|as|at|by|for|from|hhgregg|in|into|of|on|or|the|to|with'.split('|'),
        // eslint-disable-next-line max-len
        capWords: '3M|AAA|AMC|AOL|AT&T|ATM|BBC|BLT|BMV|BMW|BP|CBS|CCS|CGI|CISCO|CJ|CNG|CNN|CVS|DHL|DKNY|DMV|DSW|EMS|ER|ESPN|FCU|FCUK|FDNY|GNC|H&M|HP|HSBC|IBM|IHOP|IKEA|IRS|JBL|JCPenney|KFC|LLC|MBNA|MCA|MCI|NBC|NYPD|PDQ|PNC|TCBY|TNT|TV|UPS|USA|USPS|VW|XYZ|ZZZ'.split('|'),
        specWords: 'd\'Bronx|iFix|ExtraMile|ChargePoint|EVgo|SemaConnect'.split('|')
    };
    let _customStoreFinder = false; // switch indicating place-specific custom store finder url
    let _customStoreFinderLocal = false; // switch indicating place-specific custom store finder url with localization option (GPS/addr)
    let _customStoreFinderURL = ''; // switch indicating place-specific custom store finder url
    let _customStoreFinderLocalURL = ''; // switch indicating place-specific custom store finder url with localization option (GPS/addr)
    let _updateURL;

    // Split out state-based data
    let _psStateIx;
    let _psState2LetterIx;
    let _psRegionIx;
    let _psGoogleFormStateIx;
    let _psDefaultLockLevelIx;
    // var _ps_requirePhone_ix;
    // var _ps_requireURL_ix;
    let _psAreaCodeIx;
    let _stateDataTemp;
    let _areaCodeList = '800,822,833,844,855,866,877,888'; //  include toll free non-geographic area codes

    let _ixBank;
    let _ixATM;
    let _ixOffices;
    let _layer;

    const _UPDATED_FIELDS = {
        name: {
            updated: false,
            selector: '#venue-edit-general wz-text-input[name="name"]',
            shadowSelector: '#id',
            tab: 'general'
        },
        aliases: {
            updated: false,
            selector: '#venue-edit-general > div.aliases.form-group > wz-list',
            tab: 'general'
        },
        address: {
            updated: false,
            selector: '#venue-edit-general div.address-edit-view div.full-address-container',
            tab: 'general'
        },
        categories: {
            updated: false,
            selector: '#venue-edit-general > div.categories-control.form-group > wz-card',
            shadowSelector: 'div',
            tab: 'general'
        },
        description: {
            updated: false,
            selector: '#venue-edit-general wz-textarea[name="description"]',
            shadowSelector: '#id',
            tab: 'general'
        },
        lockRank: {
            updated: false,
            selector: '#venue-edit-general > div.lock-edit',
            tab: 'general'
        },
        externalProvider: {
            updated: false,
            selector: '#venue-edit-general > div.external-providers-control.form-group > wz-list',
            tab: 'general'
        },
        brand: { updated: false, selector: '.venue .brand .select2-container', tab: 'general' },
        url: {
            updated: false,
            selector: '#venue-url',
            shadowSelector: '#id',
            tab: 'more-info'
        },
        phone: {
            updated: false,
            selector: '#venue-phone',
            shadowSelector: '#id',
            tab: 'more-info'
        },
        openingHours: {
            updated: false,
            selector: '#venue-edit-more-info div.opening-hours.form-group > wz-list',
            tab: 'more-info'
        },
        cost: {
            updated: false,
            selector: '#venue-edit-more-info wz-select[name="costType"]',
            shadowSelector: 'div.select-box',
            tab: 'more-info'
        },
        canExit: { updated: false, selector: '.venue label[for="can-exit-checkbox"]', tab: 'more-info' },
        hasTBR: { updated: false, selector: '.venue label[for="has-tbr"]', tab: 'more-info' },
        lotType: { updated: false, selector: '#venue-edit-more-info > form > div:nth-child(1) > wz-radio-group', tab: 'more-info' },
        parkingSpots: {
            updated: false,
            selector: '#venue-edit-more-info wz-select[name="estimatedNumberOfSpots"]',
            shadowSelector: '#select-wrapper > div',
            tab: 'more-info'
        },
        lotElevation: { updated: false, selector: '.venue .lot-checkbox', tab: 'more-info' },
        evNetwork: { updated: false, selector: '', tab: 'general' },
        evPaymentMethods: {
            updated: false,
            selector: '#venue-edit-general > div.charging-station-controls div.wz-multiselect > wz-card',
            shadowSelector: 'div',
            tab: 'general'
        },
        evCostType: {
            updated: false,
            selector: '#venue-edit-general > div.charging-station-controls > wz-select',
            shadowSelector: '#select-wrapper > div > div',
            tab: 'general'
        },

        getFieldProperties() {
            return Object.keys(this)
                .filter(key => this[key].hasOwnProperty('updated'))
                .map(key => this[key]);
        },
        getUpdatedTabNames() {
            return _.uniq(this.getFieldProperties()
                .filter(prop => prop.updated)
                .map(prop => prop.tab));
        },
        // checkAddedNode(addedNode) {
        //     this.getFieldProperties()
        //         .filter(prop => prop.updated && addedNode.querySelector(prop.selector))
        //         .forEach(prop => {
        //             $(prop.selector).css({ 'background-color': '#dfd' });
        //             $(`a[href="#venue-edit-${prop.tab}"]`).css({ 'background-color': '#dfd' });
        //         });
        // },
        reset() {
            this.clearEditPanelHighlights();
            this.getFieldProperties().forEach(prop => {
                prop.updated = false;
            });
        },
        init() {
            ['VALLET_SERVICE', 'DRIVETHROUGH', 'WI_FI', 'RESTROOMS', 'CREDIT_CARDS', 'RESERVATIONS', 'OUTSIDE_SEATING', 'AIR_CONDITIONING',
                'PARKING_FOR_CUSTOMERS', 'DELIVERIES', 'TAKE_AWAY', 'WHEELCHAIR_ACCESSIBLE', 'DISABILITY_PARKING', 'CURBSIDE_PICKUP', 'CARPOOL_PARKING',
                'EV_CHARGING_STATION', 'CAR_WASH', 'SECURITY', 'AIRPORT_SHUTTLE']
                .forEach(service => {
                    const propName = `services_${service}`;
                    this[propName] = { updated: false, selector: `.venue label[for="service-checkbox-${service}"]`, tab: 'more-info' };
                });

            // 5/24/2019 (mapomatic) This observer doesn't seem to work anymore.  I've added the updateEditPanelHighlights
            // function that can be called after harmonizePlaceGo runs.

            // const observer = new MutationObserver(mutations => {
            //     mutations.forEach(mutation => {
            //         // Mutation is a NodeList and doesn't support forEach like an array
            //         for (let i = 0; i < mutation.addedNodes.length; i++) {
            //             const addedNode = mutation.addedNodes[i];
            //             // Only fire up if it's a node
            //             if (addedNode.nodeType === Node.ELEMENT_NODE) {
            //                 _UPDATED_FIELDS.checkAddedNode(addedNode);
            //             }
            //         }
            //     });
            // });
            // observer.observe(document.getElementById('edit-panel'), { childList: true, subtree: true });

            W.selectionManager.events.register('selectionchanged', null, () => errorHandler(() => this.reset()));
        },
        getTabElement(tabName) {
            let tabText;
            if (tabName === 'more-info') {
                tabText = 'More info';
            } else if (tabName === 'general') {
                tabText = 'General';
            } else {
                return null;
            }
            const tabElements = document.querySelector('#edit-panel div.venue-edit-section > wz-tabs')?.shadowRoot?.querySelectorAll('.wz-tab-label');
            if (tabElements) {
                return [...tabElements].filter(elem => elem.textContent === tabText)[0];
            }
            return null;
        },
        clearEditPanelHighlights() {
            this.getFieldProperties().filter(prop => prop.updated).forEach(prop => {
                if (prop.shadowSelector) {
                    $(document.querySelector(prop.selector)?.shadowRoot?.querySelector(prop.shadowSelector)).css('background-color', '');
                } else {
                    $(prop.selector).css({ 'background-color': '' });
                }
                $(this.getTabElement(prop.tab)).css({ 'background-color': '' });
            });
        },
        // Highlight fields in the editor panel that have been updated by WMEPH.
        updateEditPanelHighlights() {
            // This setTimeout is necessary to get some highlights to work.
            setTimeout(() => {
                this.getFieldProperties().filter(prop => prop.updated).forEach(prop => {
                    if (prop.shadowSelector) {
                        $(document.querySelector(prop.selector)?.shadowRoot?.querySelector(prop.shadowSelector)).css('background-color', '#dfd');
                    } else {
                        $(prop.selector).css({ 'background-color': '#dfd' });
                    }
                    $(this.getTabElement(prop.tab)).css({ 'background-color': '#dfd' });
                });
            }, 100);
        },
        checkNewAttributes(newAttributes, venue) {
            const checkAttribute = name => {
                if (newAttributes.hasOwnProperty(name)
                    && JSON.stringify(venue.attributes[name]) !== JSON.stringify(newAttributes[name])) {
                    _UPDATED_FIELDS[name].updated = true;
                }
            };
            checkAttribute('categories');
            checkAttribute('name');
            checkAttribute('openingHours');
            checkAttribute('description');
            checkAttribute('aliases');
            checkAttribute('url');
            checkAttribute('phone');
            checkAttribute('lockRank');
        }
    };

    // KB Shortcut object
    const _SHORTCUT = {
        allShortcuts: {}, // All the shortcuts are stored in this array
        add(shortcutCombo, callback, opt) {
            // Provide a set of default options
            const defaultOptions = {
                type: 'keydown', propagate: false, disableInInput: false, target: document, keycode: false
            };
            if (!opt) {
                opt = defaultOptions;
            } else {
                Object.keys(defaultOptions).forEach(dfo => {
                    if (typeof opt[dfo] === 'undefined') { opt[dfo] = defaultOptions[dfo]; }
                });
            }
            let ele = opt.target;
            if (typeof opt.target === 'string') { ele = document.getElementById(opt.target); }
            // var ths = this;
            shortcutCombo = shortcutCombo.toLowerCase();
            // The function to be called at keypress
            // eslint-disable-next-line func-names
            const func = function keyPressFunc(e) {
                e = e || window.event;
                if (opt.disableInInput) { // Don't enable shortcut keys in Input, Textarea fields
                    let element;
                    if (e.target) {
                        element = e.target;
                    } else if (e.srcElement) {
                        element = e.srcElement;
                    }
                    if (element.nodeType === 3) { element = element.parentNode; }
                    if (element.tagName === 'INPUT' || element.tagName === 'TEXTAREA') { return; }
                }
                // Find Which key is pressed
                let code;
                if (e.keyCode) {
                    code = e.keyCode;
                } else if (e.which) {
                    code = e.which;
                }
                let character = String.fromCharCode(code).toLowerCase();
                if (code === 188) { character = ','; } // If the user presses , when the type is onkeydown
                if (code === 190) { character = '.'; } // If the user presses , when the type is onkeydown
                const keys = shortcutCombo.split('+');
                // Key Pressed - counts the number of valid keypresses - if it is same as the number of keys, the shortcut function is invoked
                let kp = 0;
                // Work around for stupid Shift key bug created by using lowercase - as a result the shift+num combination was broken
                const shiftNums = {
                    '`': '~',
                    1: '!',
                    2: '@',
                    3: '#',
                    4: '$',
                    5: '%',
                    6: '^',
                    7: '&',
                    8: '*',
                    9: '(',
                    0: ')',
                    '-': '_',
                    '=': '+',
                    ';': ':',
                    '\'': '"',
                    ',': '<',
                    '.': '>',
                    '/': '?',
                    '\\': '|'
                };
                // Special Keys - and their codes
                const specialKeys = {
                    esc: 27,
                    escape: 27,
                    tab: 9,
                    space: 32,
                    return: 13,
                    enter: 13,
                    backspace: 8,
                    scrolllock: 145,
                    // eslint-disable-next-line camelcase
                    scroll_lock: 145,
                    scroll: 145,
                    capslock: 20,
                    // eslint-disable-next-line camelcase
                    caps_lock: 20,
                    caps: 20,
                    numlock: 144,
                    // eslint-disable-next-line camelcase
                    num_lock: 144,
                    num: 144,
                    pause: 19,
                    break: 19,
                    insert: 45,
                    home: 36,
                    delete: 46,
                    end: 35,
                    pageup: 33,
                    // eslint-disable-next-line camelcase
                    page_up: 33,
                    pu: 33,
                    pagedown: 34,
                    // eslint-disable-next-line camelcase
                    page_down: 34,
                    pd: 34,
                    left: 37,
                    up: 38,
                    right: 39,
                    down: 40,
                    f1: 112,
                    f2: 113,
                    f3: 114,
                    f4: 115,
                    f5: 116,
                    f6: 117,
                    f7: 118,
                    f8: 119,
                    f9: 120,
                    f10: 121,
                    f11: 122,
                    f12: 123
                };
                const modifiers = {
                    shift: { wanted: false, pressed: false },
                    ctrl: { wanted: false, pressed: false },
                    alt: { wanted: false, pressed: false },
                    meta: { wanted: false, pressed: false } // Meta is Mac specific
                };
                if (e.ctrlKey) { modifiers.ctrl.pressed = true; }
                if (e.shiftKey) { modifiers.shift.pressed = true; }
                if (e.altKey) { modifiers.alt.pressed = true; }
                if (e.metaKey) { modifiers.meta.pressed = true; }
                for (let i = 0; i < keys.length; i++) {
                    const k = keys[i];
                    // Modifiers
                    if (k === 'ctrl' || k === 'control') {
                        kp++;
                        modifiers.ctrl.wanted = true;
                    } else if (k === 'shift') {
                        kp++;
                        modifiers.shift.wanted = true;
                    } else if (k === 'alt') {
                        kp++;
                        modifiers.alt.wanted = true;
                    } else if (k === 'meta') {
                        kp++;
                        modifiers.meta.wanted = true;
                    } else if (k.length > 1) { // If it is a special key
                        if (specialKeys[k] === code) { kp++; }
                    } else if (opt.keycode) {
                        if (opt.keycode === code) { kp++; }
                    } else if (character === k) { // The special keys did not match
                        kp++;
                    } else if (shiftNums[character] && e.shiftKey) { // Stupid Shift key bug created by using lowercase
                        character = shiftNums[character];
                        if (character === k) { kp++; }
                    }
                }

                if (kp === keys.length && modifiers.ctrl.pressed === modifiers.ctrl.wanted && modifiers.shift.pressed === modifiers.shift.wanted
                    && modifiers.alt.pressed === modifiers.alt.wanted && modifiers.meta.pressed === modifiers.meta.wanted) {
                    callback(e);
                    if (!opt.propagate) { // Stop the event
                        // e.cancelBubble is supported by IE - this will kill the bubbling process.
                        e.cancelBubble = true;
                        e.returnValue = false;
                        // e.stopPropagation works in Firefox.
                        if (e.stopPropagation) {
                            e.stopPropagation();
                            e.preventDefault();
                        }

                        // 5/19/2019 (MapOMatic) Not sure if this return value is necessary.
                        // eslint-disable-next-line consistent-return
                        return false;
                    }
                }
            };
            this.allShortcuts[shortcutCombo] = { callback: func, target: ele, event: opt.type };
            // Attach the function with the event
            if (ele.addEventListener) {
                ele.addEventListener(opt.type, func, false);
            } else if (ele.attachEvent) {
                ele.attachEvent(`on${opt.type}`, func);
            } else {
                ele[`on${opt.type}`] = func;
            }
        },
        // Remove the shortcut - just specify the shortcut and I will remove the binding
        remove(shortcutCombo) {
            shortcutCombo = shortcutCombo.toLowerCase();
            const binding = this.allShortcuts[shortcutCombo];
            delete (this.allShortcuts[shortcutCombo]);
            if (!binding) { return; }
            const type = binding.event;
            const ele = binding.target;
            const { callback } = binding;
            if (ele.detachEvent) {
                ele.detachEvent(`on${type}`, callback);
            } else if (ele.removeEventListener) {
                ele.removeEventListener(type, callback, false);
            } else {
                ele[`on${type}`] = false;
            }
        }
    }; // END Shortcut function

    function errorHandler(callback, ...args) {
        try {
            callback(...args);
        } catch (ex) {
            console.error(`${_SCRIPT_NAME}:`, ex);
        }
    }

    function isNullOrWhitespace(str) {
        return !str?.trim().length;
    }

    function getSelectedVenue() {
        const features = WazeWrap.getSelectedFeatures();
        // Be sure to check for features.length === 1, in case multiple venues are currently selected.
        return features.length === 1 && features[0].model.type === 'venue' ? features[0].model : undefined;
    }

    function getVenueLonLat(venue) {
        const pt = venue.geometry.getCentroid();
        return new OpenLayers.LonLat(pt.x, pt.y);
    }

    function isAlwaysOpen(venue) {
        const hours = venue.attributes.openingHours;
        return hours.length === 1 && hours[0].days.length === 7 && hours[0].isAllDay();
    }

    function isEmergencyRoom(venue) {
        return /(?:emergency\s+(?:room|department|dept))|\b(?:er|ed)\b/i.test(venue.attributes.name);
    }

    function isRestArea(venue) {
        return venue.attributes.categories.includes('REST_AREAS') && /rest\s*area/i.test(venue.attributes.name);
    }

    function getPvaSeverity(pvaValue, venue) {
        const isER = pvaValue === 'hosp' && isEmergencyRoom(venue);
        let severity;
        if (pvaValue === '' || pvaValue === '0' || (pvaValue === 'hosp' && !isER)) {
            severity = _SEVERITY.RED;
        } else if (pvaValue === '2') {
            severity = _SEVERITY.BLUE;
        } else if (pvaValue === '3') {
            severity = _SEVERITY.YELLOW;
        } else {
            severity = _SEVERITY.GREEN;
        }
        return severity;
    }

    function addPURWebSearchButton() {
        const purLayerObserver = new MutationObserver(panelContainerChanged);
        purLayerObserver.observe($('#map #panel-container')[0], { childList: true, subtree: true });

        function panelContainerChanged() {
            if (!$('#WMEPH-HidePURWebSearch').prop('checked')) {
                const $panelNav = $('.place-update-edit.panel .categories.small');
                if ($('#PHPURWebSearchButton').length === 0 && $panelNav.length > 0) {
                    const $btn = $('<button>', {
                        class: 'btn btn-primary', id: 'PHPURWebSearchButton', title: 'Search the web for this place.  Do not copy info from 3rd party sources!'
                    }) // NOTE: Don't use btn-block class. Causes conflict with URO+ "Done" button.
                        .css({
                            width: '100%', display: 'block', marginTop: '4px', marginBottom: '4px'
                        })
                        .text('Web Search')
                        .click(() => { openWebSearch(); });
                    $panelNav.after($btn);
                }
            }
        }

        function buildSearchUrl(searchName, address) {
            searchName = searchName
                .replace(/[/]/g, ' ')
                .trim();
            address = address
                .replace(/No street, /, '')
                .replace(/No address/, '')
                .replace(/CR-/g, 'County Rd ')
                .replace(/SR-/g, 'State Hwy ')
                .replace(/US-/g, 'US Hwy ')
                .replace(/ CR /g, ' County Rd ')
                .replace(/ SR /g, ' State Hwy ')
                .replace(/ US /g, ' US Hwy ')
                .replace(/$CR /g, 'County Rd ')
                .replace(/$SR /g, 'State Hwy ')
                .replace(/$US /g, 'US Hwy ')
                .trim();

            searchName = encodeURIComponent(searchName + (address.length > 0 ? `, ${address}` : ''));
            return `http://www.google.com/search?q=${searchName}`;
        }

        function openWebSearch() {
            const newName = $('.place-update-edit.panel .name').first().text();
            const addr = $('.place-update-edit.panel .address').first().text();
            if ($('#WMEPH-WebSearchNewTab').prop('checked')) {
                window.open(buildSearchUrl(newName, addr));
            } else {
                window.open(buildSearchUrl(newName, addr), _SEARCH_RESULTS_WINDOW_NAME, _searchResultsWindowSpecs);
            }
        }
    }

    // This function runs at script load, and splits the category dataset into the searchable categories.
    function makeCatCheckList(categoryData) {
        const headers = categoryData[0].split('|');
        const idIndex = headers.indexOf('pc_wmecat');
        const nameIndex = headers.indexOf('pc_transcat');

        return categoryData.map(entry => {
            const splits = entry.split('|');
            const id = splits[idIndex].trim();
            if (id.length) {
                _CATEGORY_LOOKUP[splits[nameIndex].trim().toUpperCase()] = id;
            }
            return id;
        });
    } // END makeCatCheckList function

    function addSpellingVariants(nameList, spellingVariantList) {
        for (let spellingOneIdx = 0; spellingOneIdx < spellingVariantList.length; spellingOneIdx++) {
            const spellingOne = spellingVariantList[spellingOneIdx];
            const namesToCheck = nameList.filter(name => name.includes(spellingOne));
            for (let spellingTwoIdx = 0; spellingTwoIdx < spellingVariantList.length; spellingTwoIdx++) {
                if (spellingTwoIdx !== spellingOneIdx) {
                    const spellingTwo = spellingVariantList[spellingTwoIdx];
                    namesToCheck.forEach(name => {
                        const newName = name.replace(spellingOne, spellingTwo);
                        if (!nameList.includes(newName)) nameList.push(newName);
                    });
                }
            }
        }
    }

    // This function runs at script load, and builds the search name dataset to compare the WME selected place name to.
    function makeNameCheckList(pnhData) {
        const headers = pnhData[0].split('|');
        const nameIdx = headers.indexOf('ph_name');
        const aliasesIdx = headers.indexOf('ph_aliases');
        const category1Idx = headers.indexOf('ph_category1');
        const searchNameBaseIdx = headers.indexOf('ph_searchnamebase');
        const searchNameMidIdx = headers.indexOf('ph_searchnamemid');
        const searchNameEndIdx = headers.indexOf('ph_searchnameend');
        const disableIdx = headers.indexOf('ph_disable');
        const specCaseIdx = headers.indexOf('ph_speccase');
        const tighten = str => str.toUpperCase().replace(/ AND /g, '').replace(/^THE /g, '').replace(/[^A-Z0-9]/g, '');
        const stripNonAlphaKeepCommas = str => str.toUpperCase().replace(/[^A-Z0-9,]/g, '');

        return pnhData.map(entry => {
            const splits = entry.split('|');
            const specCase = splits[specCaseIdx];

            if (splits[disableIdx] !== '1' || specCase.includes('betaEnable')) {
                let newNameList = [tighten(splits[nameIdx])];

                if (splits[disableIdx] !== 'altName') {
                    // Add any aliases
                    const tempAliases = splits[aliasesIdx];
                    if (!isNullOrWhitespace(tempAliases)) {
                        newNameList = newNameList.concat(tempAliases.replace(/,[^A-Za-z0-9]*/g, ',').split(',').map(alias => tighten(alias)));
                    }
                }

                // The following code sets up alternate search names as outlined in the PNH dataset.
                // Formula, with P = PNH primary; A1, A2 = PNH aliases; B1, B2 = base terms; M1, M2 = mid terms; E1, E2 = end terms
                // Search list will build: P, A, B, PM, AM, BM, PE, AE, BE, PME, AME, BME.
                // Multiple M terms are applied singly and in pairs (B1M2M1E2).  Multiple B and E terms are applied singly (e.g B1B2M1 not used).
                // Any doubles like B1E2=P are purged at the end to eliminate redundancy.
                const nameBaseStr = splits[searchNameBaseIdx];
                if (!isNullOrWhitespace(nameBaseStr)) { // If base terms exist, otherwise only the primary name is matched
                    newNameList = newNameList.concat(stripNonAlphaKeepCommas(nameBaseStr).split(','));

                    const nameMidStr = splits[searchNameMidIdx];
                    if (!isNullOrWhitespace(nameMidStr)) {
                        let pnhSearchNameMid = stripNonAlphaKeepCommas(nameMidStr).split(',');
                        if (pnhSearchNameMid.length > 1) { // if there are more than one mid terms, it adds a permutation of the first 2
                            pnhSearchNameMid = pnhSearchNameMid.concat([pnhSearchNameMid[0] + pnhSearchNameMid[1], pnhSearchNameMid[1] + pnhSearchNameMid[0]]);
                        }
                        const midLen = pnhSearchNameMid.length;
                        // extend the list by adding Mid terms onto the SearchNameBase names
                        for (let extix = 1, len = newNameList.length; extix < len; extix++) {
                            for (let midix = 0; midix < midLen; midix++) {
                                newNameList.push(newNameList[extix] + pnhSearchNameMid[midix]);
                            }
                        }
                    }

                    const nameEndStr = splits[searchNameEndIdx];
                    if (!isNullOrWhitespace(nameEndStr)) {
                        const pnhSearchNameEnd = stripNonAlphaKeepCommas(nameEndStr).split(',');
                        const endLen = pnhSearchNameEnd.length;
                        // extend the list by adding End terms onto all the SearchNameBase & Base+Mid names
                        for (let extix = 1, len = newNameList.length; extix < len; extix++) {
                            for (let endix = 0; endix < endLen; endix++) {
                                newNameList.push(newNameList[extix] + pnhSearchNameEnd[endix]);
                            }
                        }
                    }
                }
                // Clear out any empty entries
                newNameList = newNameList.filter(name => name.length > 1);

                // Next, add extensions to the search names based on the WME place category
                const category = splits[category1Idx].toUpperCase().replace(/[^A-Z0-9]/g, '');
                const appendWords = [];
                if (category === 'HOTEL') {
                    appendWords.push('HOTEL');
                } else if (category === 'BANKFINANCIAL' && !/\bnotABank\b/.test(specCase)) {
                    appendWords.push('BANK', 'ATM');
                } else if (category === 'SUPERMARKETGROCERY') {
                    appendWords.push('SUPERMARKET');
                } else if (category === 'GYMFITNESS') {
                    appendWords.push('GYM');
                } else if (category === 'GASSTATION') {
                    appendWords.push('GAS', 'GASOLINE', 'FUEL', 'STATION', 'GASSTATION');
                } else if (category === 'CARRENTAL') {
                    appendWords.push('RENTAL', 'RENTACAR', 'CARRENTAL', 'RENTALCAR');
                }
                appendWords.forEach(word => { newNameList = newNameList.concat(newNameList.map(name => name + word)); });

                // Add entries for word/spelling variations
                _wordVariations.forEach(variationsList => addSpellingVariants(newNameList, variationsList));

                return _.uniq(newNameList).join('|').replace(/\|{2,}/g, '|').replace(/\|+$/g, '');
            } // END if valid line
            return '00';
        });
    } // END makeNameCheckList

    function clickGeneralTab() {
        // Make sure the General tab is selected before clicking on the external provider element.
        // These selector strings are very specific.  Could probably make them more generalized for robustness.
        const containerSelector = '#edit-panel > div > div.venue-feature-editor > div > div.venue-edit-section > wz-tabs';
        const shadowSelector = 'div > div > div > div > div:nth-child(1)';
        document.querySelector(containerSelector).shadowRoot.querySelector(shadowSelector).click();
    }

    // Whitelist stringifying and parsing
    function saveWhitelistToLS(compress) {
        let wlString = JSON.stringify(_venueWhitelist);
        if (compress) {
            if (wlString.length < 4800000) { // Also save to regular storage as a back up
                localStorage.setItem(_WL_LOCAL_STORE_NAME, wlString);
            }
            wlString = LZString.compressToUTF16(wlString);
            localStorage.setItem(_WL_LOCAL_STORE_NAME_COMPRESSED, wlString);
        } else {
            localStorage.setItem(_WL_LOCAL_STORE_NAME, wlString);
        }
    }
    function loadWhitelistFromLS(decompress) {
        let wlString;
        if (decompress) {
            wlString = localStorage.getItem(_WL_LOCAL_STORE_NAME_COMPRESSED);
            wlString = LZString.decompressFromUTF16(wlString);
        } else {
            wlString = localStorage.getItem(_WL_LOCAL_STORE_NAME);
        }
        _venueWhitelist = JSON.parse(wlString);
    }
    function backupWhitelistToLS(compress) {
        let wlString = JSON.stringify(_venueWhitelist);
        if (compress) {
            wlString = LZString.compressToUTF16(wlString);
            localStorage.setItem(_WL_LOCAL_STORE_NAME_COMPRESSED + Math.floor(Date.now() / 1000), wlString);
        } else {
            localStorage.setItem(_WL_LOCAL_STORE_NAME + Math.floor(Date.now() / 1000), wlString);
        }
    }

    function log(...args) {
        console.log(`WMEPH${_IS_BETA_VERSION ? '-β' : ''}:`, ...args);
    }
    function logDev(...args) {
        if (_USER.isDevUser) {
            console.debug(`WMEPH${_IS_BETA_VERSION ? '-β' : ''} (dev):`, ...args);
        }
    }

    function zoomPlace() {
        const venue = getSelectedVenue();
        if (venue) {
            W.map.moveTo(getVenueLonLat(venue), 7);
        } else {
            W.map.moveTo(_wmephMousePosition, 5);
        }
    }

    function nudgeVenue(venue) {
        const originalGeometry = venue.geometry.clone();
        const moveNegative = Math.random() > 0.5;
        const nudgeDistance = 0.00000001 * (moveNegative ? -1 : 1);
        if (venue.isPoint()) {
            venue.geometry.x += nudgeDistance;
        } else {
            venue.geometry.components[0].components[0].x += nudgeDistance;
        }
        const action = new UpdateFeatureGeometry(venue, W.model.venues, originalGeometry, venue.geometry);
        const mAction = new MultiAction([action], { description: 'Place nudged by WMEPH' });
        W.model.actionManager.add(mAction);
    }

    function sortWithIndex(toSort) {
        for (let i = 0; i < toSort.length; i++) {
            toSort[i] = [toSort[i], i];
        }
        toSort.sort((left, right) => (left[0] < right[0] ? -1 : 1));
        toSort.sortIndices = [];
        for (let j = 0; j < toSort.length; j++) {
            toSort.sortIndices.push(toSort[j][1]);
            // eslint-disable-next-line prefer-destructuring
            toSort[j] = toSort[j][0];
        }
        return toSort;
    }

    function destroyDupeLabels() {
        _dupeLayer.destroyFeatures();
        _dupeLayer.setVisibility(false);
    }

    // When a dupe is deleted, delete the dupe label
    function deleteDupeLabel() {
        setTimeout(() => {
            const actionsList = W.model.actionManager.getActions();
            const lastAction = actionsList[actionsList.length - 1];
            if (typeof lastAction !== 'undefined' && lastAction.hasOwnProperty('object') && lastAction.object.hasOwnProperty('state') && lastAction.object.state === 'Delete') {
                if (_dupeIDList.includes(lastAction.object.attributes.id)) {
                    if (_dupeIDList.length === 2) {
                        destroyDupeLabels();
                    } else {
                        const deletedDupe = _dupeLayer.getFeaturesByAttribute('dupeID', lastAction.object.attributes.id);
                        _dupeLayer.removeFeatures(deletedDupe);
                        _dupeIDList.splice(_dupeIDList.indexOf(lastAction.object.attributes.id), 1);
                    }
                    log('Deleted a dupe');
                }
            }
        }, 20);
    }

    //  Whitelist an item. Returns true if successful. False if not.
    function whitelistAction(itemID, wlKeyName) {
        const venue = getSelectedVenue();
        let addressTemp = venue.getAddress();
        if (addressTemp.hasOwnProperty('attributes')) {
            addressTemp = addressTemp.attributes;
        }
        if (!addressTemp.country) {
            WazeWrap.Alerts.error(_SCRIPT_NAME, 'Whitelisting requires an address. Enter the place\'s address and try again.');
            return false;
        }
        const centroid = venue.attributes.geometry.getCentroid();
        const itemGPS = OpenLayers.Layer.SphericalMercator.inverseMercator(centroid.x, centroid.y);
        if (!_venueWhitelist.hasOwnProperty(itemID)) { // If venue is NOT on WL, then add it.
            _venueWhitelist[itemID] = {};
        }
        _venueWhitelist[itemID][wlKeyName] = { active: true }; // WL the flag for the venue
        _venueWhitelist[itemID].city = addressTemp.city.attributes.name; // Store city for the venue
        _venueWhitelist[itemID].state = addressTemp.state.name; // Store state for the venue
        _venueWhitelist[itemID].country = addressTemp.country.name; // Store country for the venue
        _venueWhitelist[itemID].gps = itemGPS; // Store GPS coords for the venue
        saveWhitelistToLS(true); // Save the WL to local storage
        wmephWhitelistCounter();
        _buttonBanner2.clearWL.active = true;
        return true;
    }

    // Keep track of how many whitelists have been added since the last pull, alert if over a threshold (100?)
    function wmephWhitelistCounter() {
        // eslint-disable-next-line camelcase
        localStorage.WMEPH_WLAddCount = parseInt(localStorage.WMEPH_WLAddCount, 10) + 1;
        if (localStorage.WMEPH_WLAddCount > 50) {
            WazeWrap.Alerts.warning(_SCRIPT_NAME, 'Don\'t forget to periodically back up your Whitelist data using the Pull option in the WMEPH settings tab.');
            // eslint-disable-next-line camelcase
            localStorage.WMEPH_WLAddCount = 2;
        }
    }

    function createObserver() {
        const observer = new MutationObserver(mutations => {
            mutations.forEach(mutation => {
                // Mutation is a NodeList and doesn't support forEach like an array
                for (let i = 0; i < mutation.addedNodes.length; i++) {
                    const addedNode = mutation.addedNodes[i];
                    // Only fire up if it's a node
                    if (addedNode.querySelector && addedNode.querySelector('.tab-scroll-gradient')) {
                        // Normally, scrolling happens inside the tab-content div.  When WMEPH adds stuff outside the venue div, it effectively breaks that
                        // and causes scrolling to occur at the main content div under edit-panel.  That's actually OK, but need to disable a couple
                        // artifacts that "stick around" with absolute positioning.
                        $('#edit-panel .venue').removeClass('separator-line');
                        $('#edit-panel .tab-scroll-gradient').css({ display: 'none' });
                    }
                }
            });
        });
        observer.observe(document.getElementById('edit-panel'), { childList: true, subtree: true });
    }

    function appendServiceButtonIconCss() {
        const cssArray = [
            '.serv-247 { width: 73px; height: 50px; display: inline-block; background: transparent url() top center no-repeat; }',
            '.serv-ac { width: 50px; height: 50px; display: inline-block; background: transparent url() top center no-repeat; }',
            '.serv-credit { width: 73px; height: 50px; display: inline-block; background: transparent url() top center no-repeat; }',
            '.serv-deliveries { width: 86px; height: 50px; display: inline-block; background: transparent url() top center no-repeat; }',
            '.serv-drivethru { width: 78px; height: 50px; display: inline-block; background: transparent url() top center no-repeat; }',
            '.serv-outdoor { width: 73px; height: 50px; display: inline-block; background: transparent url() top center no-repeat; }',
            '.serv-parking { width: 46px; height: 50px; display: inline-block; background: transparent url() top center no-repeat; }',
            '.serv-reservations { width: 55px; height: 50px; display: inline-block; background: transparent url() top center no-repeat; }',
            '.serv-restrooms { width: 49px; height: 50px; display: inline-block; background: transparent url() top center no-repeat; }',
            '.serv-takeaway { width: 34px; height: 50px; display: inline-block; background: transparent url() top center no-repeat; }',
            '.serv-valet { width: 50px; height: 50px; display: inline-block; background: transparent url() top center no-repeat; }',
            '.serv-wheelchair { width: 50px; height: 50px; display: inline-block; background: transparent url() top center no-repeat; }',
            '.serv-wifi { width: 67px; height: 50px; display: inline-block; background: transparent url() top center no-repeat; }',
            '.serv-curbside { width: 65px; height: 65px; display: inline-block; background: transparent url() top center no-repeat; }'
        ];
        $('head').append($('<style>', { type: 'text/css' }).html(cssArray.join('\n')));
    }

    // Function that checks current place against the Harmonization Data.  Returns place data or "NoMatch"
    function harmoList(itemName, state2L, region3L, country, itemCats, item) {
        if (country !== 'USA' && country !== 'CAN') {
            WazeWrap.Alerts.info(_SCRIPT_NAME, 'No PNH data exists for this country.');
            return ['NoMatch'];
        }
        const { pnhNames, pnh: pnhData } = _PNH_DATA[country];
        const pnhHeaders = pnhData[0].split('|');
        const phNameIdx = pnhHeaders.indexOf('ph_name');
        const phCategory1Idx = pnhHeaders.indexOf('ph_category1');
        const phForceCatIdx = pnhHeaders.indexOf('ph_forcecat');
        const phRegionIdx = pnhHeaders.indexOf('ph_region');
        const phOrderIdx = pnhHeaders.indexOf('ph_order');
        const phSpecCaseIdx = pnhHeaders.indexOf('ph_speccase');
        const phSearchNameWordIdx = pnhHeaders.indexOf('ph_searchnameword');
        let approvedRegions; // filled with the regions that are approved for the place, when match is found
        const matchPNHRegionData = []; // array of matched data with regional approval
        let allowMultiMatch = false;
        const pnhOrderNum = [];
        const pnhNameTemp = [];
        let pnhNameMatch = false; // tracks match status

        itemName = itemName.toUpperCase().replace(/ AND /g, ' ').replace(/^THE /g, '');
        const itemNameSpace = ` ${itemName.replace(/[^A-Z0-9 ]/g, ' ').replace(/ {2,}/g, ' ')} `;
        itemName = itemName.replace(/[^A-Z0-9]/g, ''); // Clear all non-letter and non-number characters ( HOLLYIVY PUB #23 -- > HOLLYIVYPUB23 )

        // for each place on the PNH list (skipping headers at index 0)
        for (let pnhIdx = 1, len = pnhNames.length; pnhIdx < len; pnhIdx++) {
            let PNHStringMatch = false;
            const pnhEntry = pnhData[pnhIdx];
            const pnhEntrySplits = pnhEntry.split('|'); // Split the PNH place data into string array

            // Name Matching
            const specCases = pnhEntrySplits[phSpecCaseIdx];
            if (specCases.includes('regexNameMatch')) {
                // Check for regex name matching instead of "standard" name matching.
                const match = specCases.match(/regexNameMatch<>(.+?)<>/i);
                if (match !== null) {
                    const reStr = match[1].replace(/\\/, '\\').replace(/<or>/g, '|');
                    const re = new RegExp(reStr, 'i');
                    PNHStringMatch = re.test(item.attributes.name);
                }
            } else if (specCases.includes('strMatchAny') || pnhEntrySplits[phCategory1Idx] === 'Hotel') {
                // Match any part of WME name with either the PNH name or any spaced names
                allowMultiMatch = true;
                const spaceMatchList = [];
                spaceMatchList.push(pnhEntrySplits[phNameIdx].toUpperCase().replace(/ AND /g, ' ').replace(/^THE /g, '').replace(/[^A-Z0-9 ]/g, ' ').replace(/ {2,}/g, ' '));
                if (pnhEntrySplits[phSearchNameWordIdx] !== '') {
                    spaceMatchList.push(...pnhEntrySplits[phSearchNameWordIdx].toUpperCase().replace(/, /g, ',').split(','));
                }
                for (let nmix = 0; nmix < spaceMatchList.length; nmix++) {
                    if (itemNameSpace.includes(` ${spaceMatchList[nmix]} `)) {
                        PNHStringMatch = true;
                    }
                }
            } else {
                // Split all possible search names for the current PNH entry
                const nameComps = pnhNames[pnhIdx].split('|');

                // Clear non-letter characters for alternate match ( HOLLYIVYPUB23 --> HOLLYIVYPUB )
                const itemNameNoNum = itemName.replace(/[^A-Z]/g, '');

                if (specCases.includes('strMatchStart')) {
                    //  Match the beginning part of WME name with any search term
                    for (let nmix = 0; nmix < nameComps.length; nmix++) {
                        if (itemName.startsWith(nameComps[nmix]) || itemNameNoNum.startsWith(nameComps[nmix])) {
                            PNHStringMatch = true;
                        }
                    }
                } else if (specCases.includes('strMatchEnd')) {
                    //  Match the end part of WME name with any search term
                    for (let nmix = 0; nmix < nameComps.length; nmix++) {
                        if (itemName.endsWith(nameComps[nmix]) || itemNameNoNum.endsWith(nameComps[nmix])) {
                            PNHStringMatch = true;
                        }
                    }
                } else if (nameComps.includes(itemName) || nameComps.includes(itemNameNoNum)) {
                    // full match of any term only
                    PNHStringMatch = true;
                }
            }

            // if a match was found:
            if (PNHStringMatch) { // Compare WME place name to PNH search name list
                logDev(`Matched PNH Order No.: ${pnhEntrySplits[phOrderIdx]}`);

                const PNHPriCat = catTranslate(pnhEntrySplits[phCategory1Idx]); // Primary category of PNH data
                let PNHForceCat = pnhEntrySplits[phForceCatIdx]; // Primary category of PNH data

                // Gas stations only harmonized if the WME place category is already gas station (prevents Costco Gas becoming Costco Store)
                if (itemCats[0] === 'GAS_STATION') {
                    PNHForceCat = '1';
                }

                let PNHMatchProceed = false;
                if (PNHForceCat === '1' && itemCats.indexOf(PNHPriCat) === 0) {
                    // Name and primary category match
                    PNHMatchProceed = true;
                } else if (PNHForceCat === '2' && itemCats.includes(PNHPriCat)) {
                    // Name and any category match
                    PNHMatchProceed = true;
                } else if (PNHForceCat === '0' || PNHForceCat === '') {
                    // Name only match
                    PNHMatchProceed = true;
                }

                if (PNHMatchProceed) {
                    // remove spaces, upper case the approved regions, and split by commas
                    approvedRegions = pnhEntrySplits[phRegionIdx].replace(/ /g, '').toUpperCase().split(',');

                    if (approvedRegions.includes(state2L) || approvedRegions.includes(region3L) // if the WME-selected item matches the state, region
                        || approvedRegions.includes(country) //  OR if the country code is in the data then it is approved for all regions therein
                        || $('#WMEPH-RegionOverride').prop('checked')) { // OR if region override is selected (dev setting)
                        matchPNHRegionData.push(pnhEntry);
                        _buttonBanner.placeMatched = new Flag.PlaceMatched();
                        if (!allowMultiMatch) {
                            // Return the PNH data string array to the main script
                            return matchPNHRegionData;
                        }
                    } else {
                        // PNH match found (once true, stays true)
                        pnhNameMatch = true;

                        // Pull the data line from the PNH data table.  (**Set in array for future multimatch features)
                        // matchPNHData.push(pnhEntry);

                        // temp name for approval return
                        pnhNameTemp.push(pnhEntrySplits[phNameIdx]);

                        // temp order number for approval return
                        pnhOrderNum.push(pnhEntrySplits[phOrderIdx]);
                    }
                }
            }
        } // END loop through PNH places

        // If NO (name & region) match was found:
        if (_buttonBanner.placeMatched) {
            return matchPNHRegionData;
        }
        if (pnhNameMatch) { // if a name match was found but not for region, prod the user to get it approved
            return ['ApprovalNeeded', pnhNameTemp, pnhOrderNum];
        }
        // if no match was found, suggest adding the place to the sheet if it's a chain
        return ['NoMatch'];
    } // END harmoList function

    function onVenuesChanged(venueProxies) {
        deleteDupeLabel();

        const venue = getSelectedVenue();
        if (venueProxies.map(proxy => proxy.attributes.id).includes(venue?.attributes.id)) {
            if ($('#WMEPH_banner').length) {
                const actions = W.model.actionManager.getActions();
                const lastAction = actions[actions.length - 1];
                if (lastAction._venue?.attributes?.id === venue.attributes.id && lastAction._navigationPoint) {
                    harmonizePlaceGo(venue, 'harmonize');
                }
            }

            updateWmephPanel();
        }
    }

    // This should be called after new venues are saved (using venues'objectssynced' event), so the new IDs can be retrieved and used
    // to replace the temporary IDs in the whitelist.  If WME errors during save, this function may not run.  At that point, the
    // temporary IDs can no longer be traced to the new IDs so the WL for those new venues will be orphaned, and the temporary IDs
    // will be removed from the WL store the next time the script starts.
    function syncWL(newVenues) {
        newVenues.forEach(newVenue => {
            const oldID = newVenue._prevID;
            const newID = newVenue.attributes.id;
            if (oldID && newID && _venueWhitelist[oldID]) {
                _venueWhitelist[newID] = _venueWhitelist[oldID];
                delete _venueWhitelist[oldID];
            }
        });
        saveWhitelistToLS(true);
    }

    function toggleXrayMode(enable) {
        localStorage.setItem('WMEPH_xrayMode_enabled', $('#layer-switcher-item_wmeph_x-ray_mode').prop('checked'));

        const commentsLayer = W.map.getLayerByUniqueName('mapComments');
        const gisLayer = W.map.getLayerByUniqueName('__wmeGISLayers');
        const satLayer = W.map.getLayerByUniqueName('satellite_imagery');
        const roadLayer = W.map.roadLayers[0];
        const commentRuleSymb = commentsLayer.styleMap.styles.default.rules[0].symbolizer;
        if (enable) {
            _layer.styleMap.styles.default.rules = _layer.styleMap.styles.default.rules.filter(rule => rule.wmephDefault !== 'default');
            roadLayer.opacity = 0.25;
            satLayer.opacity = 0.25;
            commentRuleSymb.Polygon.strokeColor = '#888';
            commentRuleSymb.Polygon.fillOpacity = 0.2;
            if (gisLayer) gisLayer.setOpacity(0.4);
        } else {
            _layer.styleMap.styles.default.rules = _layer.styleMap.styles.default.rules.filter(rule => rule.wmephStyle !== 'xray');
            roadLayer.opacity = 1;
            satLayer.opacity = 1;
            commentRuleSymb.Polygon.strokeColor = '#fff';
            commentRuleSymb.Polygon.fillOpacity = 0.4;
            if (gisLayer) gisLayer.setOpacity(1);
            initializeHighlights();
            _layer.redraw();
        }
        commentsLayer.redraw();
        roadLayer.redraw();
        satLayer.redraw();
        if (!enable) return;

        const defaultPointRadius = 6;
        const ruleGenerator = (value, symbolizer) => new W.Rule({
            filter: new OpenLayers.Filter.Comparison({
                type: '==',
                value,
                evaluate(feature) {
                    // 2023-03-30 - Check for .model is necessary until .repositoryObject is implemented in production WME.
                    const attr = feature.model ? feature.model.attributes : feature.attributes.repositoryObject?.attributes;
                    return attr?.wmephSeverity === this.value;
                }
            }),
            symbolizer,
            wmephStyle: 'xray'
        });

        const severity0 = ruleGenerator(0, {
            Point: {
                strokeWidth: 1.67,
                strokeColor: '#888',
                pointRadius: 5,
                fillOpacity: 0.25,
                fillColor: 'white',
                zIndex: 0
            },
            Polygon: {
                strokeWidth: 1.67,
                strokeColor: '#888',
                fillOpacity: 0
            }
        });

        const severityLock = ruleGenerator('lock', {
            Point: {
                strokeColor: 'white',
                fillColor: '#080',
                fillOpacity: 1,
                strokeLinecap: 1,
                strokeDashstyle: '4 2',
                strokeWidth: 2.5,
                pointRadius: defaultPointRadius
            },
            Polygon: {
                strokeColor: 'white',
                fillColor: '#0a0',
                fillOpacity: 0.4,
                strokeDashstyle: '4 2',
                strokeWidth: 2.5
            }
        });

        const severity1 = ruleGenerator(1, {
            strokeColor: 'white',
            strokeWidth: 2,
            pointRadius: defaultPointRadius,
            fillColor: '#0055ff'
        });

        const severityLock1 = ruleGenerator('lock1', {
            pointRadius: defaultPointRadius,
            fillColor: '#0055ff',
            strokeColor: 'white',
            strokeLinecap: '1',
            strokeDashstyle: '4 2',
            strokeWidth: 2.5
        });

        const severity2 = ruleGenerator(2, {
            Point: {
                fillColor: '#ca0',
                strokeColor: 'white',
                strokeWidth: 2,
                pointRadius: defaultPointRadius

            },
            Polygon: {
                fillColor: '#ff0',
                strokeColor: 'white',
                strokeWidth: 2,
                fillOpacity: 0.4
            }
        });

        const severity3 = ruleGenerator(3, {
            strokeColor: 'white',
            strokeWidth: 2,
            pointRadius: defaultPointRadius,
            fillColor: '#ff0000'
        });

        const severity4 = ruleGenerator(4, {
            fillColor: '#f42',
            strokeLinecap: 1,
            strokeWidth: 2,
            strokeDashstyle: '4 2'
        });

        const severityHigh = ruleGenerator(5, {
            fillColor: 'black',
            strokeColor: '#f4a',
            strokeLinecap: 1,
            strokeWidth: 4,
            strokeDashstyle: '4 2',
            pointRadius: defaultPointRadius
        });

        const severityAdLock = ruleGenerator('adLock', {
            pointRadius: 12,
            fillColor: 'yellow',
            fillOpacity: 0.4,
            strokeColor: '#000',
            strokeLinecap: 1,
            strokeWidth: 10,
            strokeDashstyle: '4 2'
        });

        _layer.styleMap.styles.default.rules.push(...[severity0, severityLock, severity1,
            severityLock1, severity2, severity3, severity4, severityHigh, severityAdLock]);

        _layer.redraw();
    }

    function initializeHighlights() {
        const ruleGenerator = (value, symbolizer) => new W.Rule({
            filter: new OpenLayers.Filter.Comparison({
                type: '==',
                value,
                evaluate(feature) {
                    // 2023-03-30 - Check for .model is necessary until .repositoryObject is implemented in production WME.
                    const attr = feature.model ? feature.model.attributes : feature.attributes.repositoryObject?.attributes;
                    return attr?.wmephSeverity === this.value;
                }
            }),
            symbolizer,
            wmephStyle: 'default'
        });

        const severity0 = ruleGenerator(0, {
            pointRadius: 5,
            externalGraphic: '',
            label: '',
            strokeWidth: 4,
            strokeColor: '#24ff14',
            fillColor: '#ba85bf'
        });

        const severityLock = ruleGenerator('lock', {
            pointRadius: 5,
            externalGraphic: '',
            label: '',
            strokeColor: '#24ff14',
            strokeLinecap: 1,
            strokeDashstyle: '7 2',
            strokeWidth: 5,
            fillColor: '#ba85bf'
        });

        const severity1 = ruleGenerator(1, {
            strokeColor: '#0055ff',
            strokeWidth: 4,
            externalGraphic: '',
            label: '',
            pointRadius: 7,
            fillColor: '#ba85bf'
        });

        const severityLock1 = ruleGenerator('lock1', {
            pointRadius: 5,
            strokeColor: '#0055ff',
            strokeLinecap: 1,
            strokeDashstyle: '7 2',
            externalGraphic: '',
            label: '',
            strokeWidth: 5,
            fillColor: '#ba85bf'
        });

        const severity2 = ruleGenerator(2, {
            strokeColor: '#ff0',
            strokeWidth: 6,
            externalGraphic: '',
            label: '',
            pointRadius: 8,
            fillColor: '#ba85bf'
        });

        const severity3 = ruleGenerator(3, {
            strokeColor: '#ff0000',
            strokeWidth: 4,
            externalGraphic: '',
            label: '',
            pointRadius: 8,
            fillColor: '#ba85bf'
        });

        const severity4 = ruleGenerator(4, {
            fillColor: 'black',
            fillOpacity: 0.35,
            strokeColor: '#f42',
            strokeLinecap: 1,
            strokeWidth: 13,
            externalGraphic: '',
            label: '',
            strokeDashstyle: '4 2'
        });

        const severityHigh = ruleGenerator(5, {
            pointRadius: 12,
            fillColor: 'black',
            fillOpacity: 0.4,
            strokeColor: '#f4a',
            strokeLinecap: 1,
            strokeWidth: 10,
            externalGraphic: '',
            label: '',
            strokeDashstyle: '4 2'
        });

        const severityAdLock = ruleGenerator('adLock', {
            pointRadius: 1,
            fillColor: 'yellow',
            fillOpacity: 0.4,
            strokeColor: '#000',
            strokeLinecap: 1,
            strokeWidth: 10,
            externalGraphic: '',
            label: '',
            strokeDashstyle: '4 2'
        });

        function plaTypeRuleGenerator(value, symbolizer) {
            return new W.Rule({
                filter: new OpenLayers.Filter.Comparison({
                    type: '==',
                    value,
                    evaluate(feature) {
                        // 2023-03-30 - Check for .model is necessary until .repositoryObject is implemented in production WME.
                        const attr = feature.model ? feature.model.attributes : feature.attributes.repositoryObject?.attributes;

                        if ($('#WMEPH-PLATypeFill').prop('checked') && attr
                            && attr.categoryAttributes && attr.categoryAttributes.PARKING_LOT
                            && attr.categories.includes('PARKING_LOT')) {
                            const type = attr.categoryAttributes.PARKING_LOT.parkingType;
                            return (!type && this.value === 'public') || (type && (type.toLowerCase() === this.value));
                        }
                        return undefined;
                    }
                }),
                symbolizer,
                wmephStyle: 'default'
            });
        }

        const publicPLA = plaTypeRuleGenerator('public', {
            fillColor: '#0000FF',
            fillOpacity: '0.25'
        });
        const restrictedPLA = plaTypeRuleGenerator('restricted', {
            fillColor: '#FFFF00',
            fillOpacity: '0.3'
        });
        const privatePLA = plaTypeRuleGenerator('private', {
            fillColor: '#FF0000',
            fillOpacity: '0.25'
        });

        _layer.styleMap.styles.default.rules.push(...[severity0, severityLock, severity1, severityLock1, severity2,
            severity3, severity4, severityHigh, severityAdLock, publicPLA, restrictedPLA, privatePLA]);
    }

    /**
    * To highlight a place, set the wmephSeverity attribute to the desired highlight level.
    * @param venues {array of venues, or single venue} Venues to check for highlights.
    * @param force {boolean} Force recalculation of highlights, rather than using cached results.
    */
    function applyHighlightsTest(venues, force) {
        if (!_layer) return;

        // Make sure venues is an array, or convert it to one if not.
        if (venues) {
            if (!Array.isArray(venues)) {
                venues = [venues];
            }
        } else {
            venues = [];
        }

        const storedBannButt = _buttonBanner;
        const storedBannServ = _servicesBanner;
        const storedBannButt2 = _buttonBanner2;
        const t0 = performance.now();
        const doHighlight = $('#WMEPH-ColorHighlighting').prop('checked');
        const disableRankHL = $('#WMEPH-DisableRankHL').prop('checked');

        venues.forEach(venue => {
            if (venue && venue.type === 'venue' && venue.attributes) {
                // Highlighting logic would go here
                // Severity can be: 0, 'lock', 1, 2, 3, 4, or 'high'. Set to
                // anything else to use default WME style.
                if (doHighlight && !(disableRankHL && venue.attributes.lockRank > _USER.rank - 1)) {
                    try {
                        const { id } = venue.attributes;
                        let severity;
                        let cachedResult;
                        // eslint-disable-next-line no-cond-assign
                        if (force || !isNaN(id) || ((cachedResult = _resultsCache[id]) === undefined) || (venue.updatedOn > cachedResult.u)) {
                            severity = harmonizePlaceGo(venue, 'highlight');
                            if (isNaN(id)) _resultsCache[id] = { s: severity, u: venue.updatedOn || -1 };
                        } else {
                            severity = cachedResult.s;
                        }
                        venue.attributes.wmephSeverity = severity;
                    } catch (err) {
                        console.error('WMEPH highlight error: ', err);
                    }
                } else {
                    venue.attributes.wmephSeverity = 'default';
                }
            }
        });

        // Trim the cache if it's over the max size limit.
        const keys = Object.keys(_resultsCache);
        if (keys.length > _MAX_CACHE_SIZE) {
            const trimSize = _MAX_CACHE_SIZE * 0.8;
            for (let i = keys.length - 1; i > trimSize; i--) {
                delete _resultsCache[keys[i]];
            }
        }

        const venue = getSelectedVenue();
        if (venue) {
            venue.attributes.wmephSeverity = harmonizePlaceGo(venue, 'highlight');
            _buttonBanner = storedBannButt;
            _servicesBanner = storedBannServ;
            _buttonBanner2 = storedBannButt2;
        }
        logDev(`Ran highlighter in ${Math.round((performance.now() - t0) * 10) / 10} milliseconds.`);
        logDev(`WMEPH cache size: ${Object.keys(_resultsCache).length}`);
    }

    // Set up CH loop
    function bootstrapWmephColorHighlights() {
        if (localStorage.getItem('WMEPH-ColorHighlighting') === '1') {
            // Add listeners
            W.model.venues.on('objectschanged', e => errorHandler(() => {
                if (!_disableHighlightTest) {
                    applyHighlightsTest(e, true);
                    _layer.redraw();
                }
            }));

            // 2023-03-30 - beforefeaturesadded no longer works because data model objects may be reloaded without re-adding map features.
            // The wmephSeverity property is stored in the venue data model object. One workaround to look into would be to
            // store the wmephSeverity in the feature.
            // W.map.venueLayer.events.register('beforefeaturesadded', null, e => errorHandler(() => applyHighlightsTest(e.features.map(f => f.model))));
            W.model.venues.on('objectsadded', venues => {
                applyHighlightsTest(venues);
                _layer.redraw();
            });

            // Clear the cache (highlight severities may need to be updated).
            _resultsCache = {};

            // Apply the colors
            applyHighlightsTest(W.model.venues.getObjectArray());
            _layer.redraw();
        } else {
            // reset the colors to default
            applyHighlightsTest(W.model.venues.getObjectArray());
            _layer.redraw();
        }
    }

    // Change place.name to title case
    function toTitleCaseStrong(str) {
        if (!str) {
            return str;
        }
        str = str.trim();
        const parensParts = str.match(/\(.*?\)/g);
        if (parensParts) {
            for (let i = 0; i < parensParts.length; i++) {
                str = str.replace(parensParts[i], `%${i}%`);
            }
        }

        // Get indexes of Mac followed by a cap, as in MacMillan.
        const macIndexes = [];
        const macRegex = /\bMac[A-Z]/g;
        let macMatch;
        // eslint-disable-next-line no-cond-assign
        while ((macMatch = macRegex.exec(str)) !== null) {
            macIndexes.push(macMatch.index);
        }

        const allCaps = (str === str.toUpperCase());
        // Cap first letter of each word
        str = str.replace(/([A-Za-z\u00C0-\u017F][^\s-/]*) */g, txt => {
            // If first letter is lower case, followed by a cap, then another lower case letter... ignore it.  Example: iPhone
            if (/^[a-z][A-Z0-9][a-z]/.test(txt)) {
                return txt;
            }
            // If word starts with De/Le/La followed by uppercase then lower case, is 5+ characters long... assume it should be like "DeBerry".
            if (/^([dDlL]e|[lL]a)[A-Z][a-zA-Z\u00C0-\u017F]{2,}/.test(txt)) {
                return txt.charAt(0).toUpperCase() + txt.charAt(1).toLowerCase() + txt.charAt(2) + txt.substr(3).toLowerCase();
            }
            return ((txt === txt.toUpperCase()) && !allCaps) ? txt : txt.charAt(0).toUpperCase() + txt.substr(1).toLowerCase();
        })
            // Cap O'Reilley's, L'Amour, D'Artagnan as long as 5+ letters
            .replace(/\b[oOlLdD]'[A-Za-z']{3,}/g, txt => (((txt === txt.toUpperCase()) && !allCaps) ? txt : txt.charAt(0).toUpperCase()
                + txt.charAt(1) + txt.charAt(2).toUpperCase() + txt.substr(3).toLowerCase()))
            // Cap McFarley's, as long as 5+ letters long
            .replace(/\b[mM][cC][A-Za-z']{3,}/g, txt => (((txt === txt.toUpperCase()) && !allCaps) ? txt : txt.charAt(0).toUpperCase()
                + txt.charAt(1).toLowerCase() + txt.charAt(2).toUpperCase() + txt.substr(3).toLowerCase()))
            // anything with an "&" sign, cap the word after &
            .replace(/&\w+/g, txt => (((txt === txt.toUpperCase()) && !allCaps) ? txt : txt.charAt(0) + txt.charAt(1).toUpperCase() + txt.substr(2)))
            // lowercase any from the ignoreWords list
            .replace(/[^ ]+/g, txt => {
                const txtLC = txt.toLowerCase();
                return (_TITLECASE_SETTINGS.ignoreWords.includes(txtLC)) ? txtLC : txt;
            })
            // uppercase any from the capWords List
            .replace(/[^ ]+/g, txt => {
                const txtLC = txt.toUpperCase();
                return (_TITLECASE_SETTINGS.capWords.includes(txtLC)) ? txtLC : txt;
            })
            // preserve any specific words
            .replace(/[^ ]+/g, txt => {
                const txtUC = txt.toUpperCase();
                return _TITLECASE_SETTINGS.specWords.find(specWord => specWord.toUpperCase() === txtUC) || txt;
            })
            // Fix 1st, 2nd, 3rd, 4th, etc.
            .replace(/\b(\d*1)st\b/gi, '$1st')
            .replace(/\b(\d*2)nd\b/gi, '$1nd')
            .replace(/\b(\d*3)rd\b/gi, '$1rd')
            .replace(/\b(\d+)th\b/gi, '$1th');

        // Cap first letter of entire name if it's not something like iPhone or eWhatever.
        if (!/^[a-z][A-Z0-9][a-z]/.test(str)) str = str.charAt(0).toUpperCase() + str.substr(1);
        if (parensParts) {
            for (let i = 0, len = parensParts.length; i < len; i++) {
                str = str.replace(`%${i}%`, parensParts[i]);
            }
        }

        // Fix any Mac... words.
        macIndexes.forEach(idx => {
            str = str.substr(0, idx + 3) + str.substr(idx + 3, 1).toUpperCase() + str.substr(idx + 4);
        });

        return str;
    }

    // normalize phone
    function normalizePhone(s, outputFormat) {
        s = s.replace(/(\d{3}.*)\W+(?:extension|ext|xt|x).*/i, '$1');
        let s1 = s.replace(/\D/g, ''); // remove non-number characters

        // Ignore leading 1, and also don't allow area code or exchange to start with 0 or 1 (***USA/CAN specific)
        let m = s1.match(/^1?([2-9]\d{2})([2-9]\d{2})(\d{4})$/);

        if (!m) { // then try alphanumeric matching
            if (s) { s = s.toUpperCase(); }
            s1 = s.replace(/[^0-9A-Z]/g, '').replace(/^\D*(\d)/, '$1').replace(/^1?([2-9][0-9]{2}[0-9A-Z]{7,10})/g, '$1');
            s1 = replaceLetters(s1);

            // Ignore leading 1, and also don't allow area code or exchange to start with 0 or 1 (***USA/CAN specific)
            m = s1.match(/^([2-9]\d{2})([2-9]\d{2})(\d{4})(?:.{0,3})$/);

            if (!m) {
                return 'badPhone';
            }
        }
        return phoneFormat(outputFormat, m[1], m[2], m[3]);
    }

    // Alphanumeric phone conversion
    function replaceLetters(number) {
        const conversionMap = _({
            2: /A|B|C/,
            3: /D|E|F/,
            4: /G|H|I/,
            5: /J|K|L/,
            6: /M|N|O/,
            7: /P|Q|R|S/,
            8: /T|U|V/,
            9: /W|X|Y|Z/
        });
        number = typeof number === 'string' ? number.toUpperCase() : '';
        return number.replace(/[A-Z]/g, letter => conversionMap.findKey(re => re.test(letter)));
    }

    // Add array of actions to a MultiAction to be executed at once (counts as one edit for redo/undo purposes)
    function executeMultiAction(actions, description) {
        if (actions.length > 0) {
            const mAction = new MultiAction();
            mAction.setModel(W.model);
            mAction._description = description || mAction._description || 'Change(s) made by WMEPH';
            actions.forEach(action => { mAction.doSubAction(action); });
            W.model.actionManager.add(mAction);
        }
    }

    // Split localizer (suffix) part of names, like "SUBWAY - inside Walmart".
    function getNameParts(name) {
        const splits = name.match(/(.*?)(\s+[-(–].*)*$/);
        return { base: splits[1], suffix: splits[2] };
    }

    function addUpdateAction(venue, newAttributes, actions, runHarmonizer = false, dontHighlightFields = false) {
        if (Object.keys(newAttributes).length) {
            if (!dontHighlightFields) {
                _UPDATED_FIELDS.checkNewAttributes(newAttributes, venue);
            }

            const action = new UpdateObject(venue, newAttributes);
            if (actions) {
                actions.push(action);
            } else {
                W.model.actionManager.add(action);
            }
        }
        if (runHarmonizer) harmonizePlaceGo(venue, 'harmonize');
    }

    function setServiceChecked(servBtn, checked, actions) {
        const servID = _WME_SERVICES_ARRAY[servBtn.servIDIndex];
        const checkboxChecked = $(`wz-checkbox[value="${servID}"]`).prop('checked');
        const venue = getSelectedVenue();

        if (checkboxChecked !== checked) {
            _UPDATED_FIELDS[`services_${servID}`].updated = true;
        }
        const toggle = typeof checked === 'undefined';
        let noAdd = false;
        checked = (toggle) ? !servBtn.checked : checked;
        if (checkboxChecked === servBtn.checked && checkboxChecked !== checked) {
            servBtn.checked = checked;
            let services;
            if (actions) {
                for (let i = 0; i < actions.length; i++) {
                    const existingAction = actions[i];
                    if (existingAction.newAttributes && existingAction.newAttributes.services) {
                        ({ services } = existingAction.newAttributes);
                    }
                }
            }
            if (!services) {
                services = venue.attributes.services.slice();
            } else {
                noAdd = services.includes(servID);
            }
            if (checked) {
                services.push(servID);
            } else {
                const index = services.indexOf(servID);
                if (index > -1) {
                    services.splice(index, 1);
                }
            }
            if (!noAdd) {
                addUpdateAction(venue, { services }, actions);
            }
        }
        updateServicesChecks(_servicesBanner);
        if (!toggle) servBtn.active = checked;
    }

    // Normalize url
    function normalizeURL(s, lc, skipBannerActivate, venue, region, wl) {
        const regionsThatWantPLAUrls = ['SER'];

        if ((!s || s.trim().length === 0) && !skipBannerActivate) {
            // Notify that url is missing and provide web search to find website and gather data (provided for all editors)
            const hasOperator = venue.attributes.brand && W.model.categoryBrands.PARKING_LOT.includes(venue.attributes.brand);
            if (!venue.isParkingLot() || (venue.isParkingLot() && (regionsThatWantPLAUrls.includes(region) || hasOperator))) {
                _buttonBanner.urlMissing = new Flag.UrlMissing(venue, wl);
                if (wl.urlWL || (venue.isParkingLot() && !hasOperator)) {
                    _buttonBanner.urlMissing.severity = _SEVERITY.GREEN;
                    _buttonBanner.urlMissing.WLactive = false;
                }
            }
            return s;
        }

        s = s.replace(/ \(.*/g, ''); // remove anything with parentheses after it
        s = s.replace(/ /g, ''); // remove any spaces
        let m = s.match(/^http:\/\/(.*)$/i); // remove http://
        if (m) { [, s] = m; }
        if (lc) { // lowercase the entire domain
            s = s.replace(/[^/]+/i, txt => ((txt === txt.toLowerCase()) ? txt : txt.toLowerCase()));
        } else { // lowercase only the www and com
            s = s.replace(/www\./i, 'www.');
            s = s.replace(/\.com/i, '.com');
        }
        m = s.match(/^(.*)\/pages\/welcome.aspx$/i); // remove unneeded terms
        if (m) { [, s] = m; }
        m = s.match(/^(.*)\/pages\/default.aspx$/i); // remove unneeded terms
        if (m) { [, s] = m; }
        // m = s.match(/^(.*)\/index.html$/i); // remove unneeded terms
        // if (m) { s = m[1]; }
        // m = s.match(/^(.*)\/index.htm$/i); // remove unneeded terms
        // if (m) { s = m[1]; }
        // m = s.match(/^(.*)\/index.php$/i); // remove unneeded terms
        // if (m) { s = m[1]; }
        m = s.match(/^(.*)\/$/i); // remove final slash
        if (m) { [, s] = m; }
        if (!s || s.trim().length === 0 || !/(^https?:\/\/)?\w+\.\w+/.test(s)) s = 'badURL';
        return s;
    } // END normalizeURL function

    // Only run the harmonization if a venue is selected
    function harmonizePlace() {
        // Beta version for approved users only
        if (_IS_BETA_VERSION && !_USER.isBetaUser) {
            WazeWrap.Alerts.error(_SCRIPT_NAME, 'Please sign up to beta-test this script version.<br>Send a PM or Slack-DM to MapOMatic or Tonestertm, or post in the WMEPH forum thread. Thanks.');
            return;
        }
        // Only run if a single place is selected
        const venue = getSelectedVenue();
        if (venue) {
            _UPDATED_FIELDS.reset();
            blurAll(); // focus away from current cursor position
            _disableHighlightTest = true;
            harmonizePlaceGo(venue, 'harmonize');
            _disableHighlightTest = false;
            applyHighlightsTest(venue);
        } else { // Remove duplicate labels
            destroyDupeLabels();
        }
    }

    // Abstract flag classes.  Must be declared outside the "Flag" namespace.
    class FlagBase {
        constructor(active, severity, message) {
            this.active = active;
            this.severity = severity;
            this.message = message;
        }
    }
    class ActionFlag extends FlagBase {
        constructor(active, severity, message, value, title) {
            super(active, severity, message);
            this.value = value;
            this.title = title;
        }

        // 5/19/2019 (mapomatic) This base class action function doesn't seem to be necessary.
        // action() { } // overwrite this
    }
    class WLFlag extends FlagBase {
        WLactive;
        WLtitle;
        WLkeyName;
        constructor(active, severity, message, WLactive, WLtitle, WLkeyName) {
            super(active, severity, message);
            this.WLactive = WLactive;
            this.WLtitle = WLtitle;
            this.WLkeyName = WLkeyName;
        }

        WLaction() {
            const venue = getSelectedVenue();
            if (whitelistAction(venue.attributes.id, this.WLkeyName)) {
                harmonizePlaceGo(venue, 'harmonize');
            }
        }
    }
    class WLActionFlag extends WLFlag {
        value;
        title;
        constructor(active, severity, message, value, title, WLactive, WLtitle, WLkeyName) {
            super(active, severity, message, WLactive, WLtitle, WLkeyName);
            this.value = value;
            this.title = title;
        }

        // 5/19/2019 (mapomatic) This base class action function doesn't seem to be necessary.
        // action() { } // overwrite this
    }

    // Namespace to keep these grouped.
    const Flag = {
        // 2020-10-5 Disabling HN validity checks for now. See note on HnNonStandard flag for details.
        // HnDashRemoved: class extends FlagBase {
        //     constructor() { super(true, SEVERITY.GREEN, 'Dash removed from house number. Verify'); }
        // },
        FullAddressInference: class extends FlagBase {
            constructor(inferredAddress) {
                super(true, _SEVERITY.RED, 'Missing address was inferred from nearby segments. Verify the address and run script again.');
                this.noLock = true;
                this.inferAddress = inferredAddress;
            }

            static eval(venue, addr, actions, highlightOnly, categories) {
                let result = null;
                if (!highlightOnly) {
                    if (!addr.state || !addr.country) {
                        if (W.map.getZoom() < 4) {
                            if ($('#WMEPH-EnableIAZoom').prop('checked')) {
                                W.map.moveTo(getVenueLonLat(venue), 5);
                            } else {
                                WazeWrap.Alerts.error(_SCRIPT_NAME, 'No address and the state cannot be determined. Please zoom in and rerun the script. '
                                    + 'You can enable autozoom for this type of case in the options.');
                            }
                            result = { exit: true }; // Don't bother returning a Flag. This will exit the rest of the harmonizePlaceGo function.
                        } else {
                            let inferredAddress = inferAddress(venue, 7); // Pull address info from nearby segments
                            inferredAddress = inferredAddress.attributes ?? inferredAddress;

                            if (inferredAddress?.state && inferredAddress.country) {
                                if ($('#WMEPH-AddAddresses').prop('checked')) { // update the item's address if option is enabled
                                    updateAddress(venue, inferredAddress, actions);
                                    _UPDATED_FIELDS.address.updated = true;
                                    result = new this(inferredAddress);
                                } else if (!['JUNCTION_INTERCHANGE'].includes(categories[0])) {
                                    _buttonBanner.cityMissing = new Flag.CityMissing();
                                }
                            } else { //  if the inference doesn't work...
                                WazeWrap.Alerts.error(_SCRIPT_NAME, 'This place has no address data and the address cannot be inferred from nearby segments. Please edit the address and run WMEPH again.');
                                result = { exit: true }; // Don't bother returning a Flag. This will exit the rest of the harmonizePlaceGo function.
                            }
                        }
                    }
                } else if (!addr.state || !addr.country) { // only highlighting
                    result = { exit: true };
                    if (venue.attributes.adLocked) {
                        result.severity = 'adLock';
                    } else {
                        const cat = venue.attributes.categories;
                        if (containsAny(cat, ['HOSPITAL_MEDICAL_CARE', 'HOSPITAL_URGENT_CARE', 'GAS_STATION'])) {
                            logDev('Unaddressed HUC/GS');
                            result.severity = _SEVERITY.PINK;
                        } else if (cat.includes('JUNCTION_INTERCHANGE')) {
                            result.severity = _SEVERITY.GREEN;
                        } else {
                            result.severity = _SEVERITY.RED;
                        }
                    }
                }
                return result;
            }
        },
        NameMissing: class extends FlagBase {
            constructor() {
                super(true, _SEVERITY.RED, 'Name is missing.');
                this.noLock = true;
            }

            static #venueIsFlaggable(venue, name) {
                return !venue.isResidential()
                    && (!name || !name.replace(/[^A-Za-z0-9]/g, '').length)
                    && !['ISLAND', 'FOREST_GROVE', 'SEA_LAKE_POOL', 'RIVER_STREAM', 'CANAL', 'PARKING_LOT'].includes(venue.attributes.categories[0])
                    && !(venue.isGasStation() && venue.attributes.brand);
            }

            static eval(venue, name) {
                return this.#venueIsFlaggable(venue, name) ? new this() : null;
            }
        },
        GasNameMissing: class extends ActionFlag {
            constructor(venue, brand, highlightOnly) {
                let message;
                if (!highlightOnly) message = `Name is missing. Use "${brand}"?`;
                super(true, _SEVERITY.RED, message, 'Yes', 'Use gas brand as station name');
                this.brand = brand;
                this.venue = venue;
            }

            static #venueIsFlaggable(venue, name, brand) {
                return venue.isGasStation()
                    && isNullOrWhitespace(name)
                    && !isNullOrWhitespace(brand);
            }

            static eval(venue, name, brand, highlightOnly) {
                return this.#venueIsFlaggable(venue, name, brand) ? new this(venue, brand, highlightOnly) : null;
            }

            action() {
                addUpdateAction(this.venue, { name: this.brand }, null, true);
            }
        },
        PlaIsPublic: class extends FlagBase {
            constructor(venue, highlightOnly) {
                super(true, _SEVERITY.GREEN, 'If this does not meet the requirements for a <a href="https://wazeopedia.waze.com/wiki/USA/Places/Parking_lot#Lot_Type" '
                    + 'target="_blank" style="color:5a5a73">public parking lot</a>, change to:<br>');
                if (!highlightOnly) {
                    this.venue = venue;
                    // Add the buttons to the message.
                    this.message += [
                        ['RESTRICTED', 'Restricted'],
                        ['PRIVATE', 'Private']
                    ].map(
                        btnInfo => $('<button>', { class: 'wmeph-pla-lot-type-btn btn btn-default btn-xs wmeph-btn', 'data-lot-type': btnInfo[0] })
                            .text(btnInfo[1])
                            .prop('outerHTML')
                    ).join('');
                }
            }

            static #venueIsFlaggable(venue) {
                return venue.isParkingLot() && venue.attributes.categoryAttributes?.PARKING_LOT?.parkingType === 'PUBLIC';
            }

            static eval(venue, highlightOnly) {
                return this.#venueIsFlaggable(venue) ? new this(venue, highlightOnly) : null;
            }

            postProcess() {
                $('.wmeph-pla-lot-type-btn').click(evt => {
                    const lotType = $(evt.currentTarget).data('lot-type');
                    const categoryAttrClone = JSON.parse(JSON.stringify(this.venue.attributes.categoryAttributes));
                    categoryAttrClone.PARKING_LOT.parkingType = lotType;
                    _UPDATED_FIELDS.lotType.updated = true;
                    addUpdateAction(this.venue, { categoryAttributes: categoryAttrClone }, null, true);
                });
            }
        },
        PlaNameMissing: class extends FlagBase {
            constructor(userNormalizedRank) {
                super(
                    true,
                    _SEVERITY.BLUE,
                    `Name is missing. ${userNormalizedRank < 3 ? 'Request an R3+ lock' : 'Lock to 3+'} to confirm unnamed parking lot.`
                );
                this.noLock = true;
            }

            static #venueIsFlaggable(venue, name) {
                return venue.isParkingLot()
                    && (!name || !name.replace(/[^A-Za-z0-9]/g, '').length)
                    && venue.attributes.lockRank < 2;
            }

            static eval(venue, name, userNormalizedRank) {
                return this.#venueIsFlaggable(venue, name) ? new Flag.PlaNameMissing(userNormalizedRank) : null;
            }
        },
        PlaNameNonStandard: class extends WLFlag {
            constructor() {
                super(
                    true,
                    _SEVERITY.YELLOW,
                    'Parking lot names typically contain words like "Parking", "Lot", and/or "Garage"',
                    true,
                    'Whitelist non-standard PLA name',
                    'plaNameNonStandard'
                );
            }

            static #venueIsFlaggable(venue, wl) {
                if (!wl.plaNameNonStandard && venue.isParkingLot()) {
                    const { name } = venue.attributes;
                    if (name) {
                        const state = venue.getAddress().getStateName();
                        const re = state === 'Quebec' ? /\b(parking|stationnement)\b/i : /\b((park[ -](and|&|'?n'?)[ -]ride)|parking|lot|garage|ramp)\b/i;
                        if (!re.test(name)) {
                            return true;
                        }
                    }
                }
                return false;
            }

            static eval(venue, wl) {
                return this.#venueIsFlaggable(venue, wl) ? new this() : null;
            }
        },
        IndianaLiquorStoreHours: class extends WLFlag {
            constructor() {
                super(
                    true,
                    _SEVERITY.GREEN,
                    'If this is a liquor store, check the hours. As of Feb 2018, liquor stores in Indiana are allowed to be open between noon and 8 pm on Sunday.',
                    true,
                    'Whitelist Indiana liquor store hours',
                    'indianaLiquorStoreHours'
                );
            }

            static eval(venue, name, highlightOnly, wl) {
                let result = null;
                if (!highlightOnly && !wl.indianaLiquorStoreHours
                    && [/\bbeers?\b/, /\bwines?\b/, /\bliquor\b/, /\bspirits\b/].some(re => re.test(name))
                    && !venue.attributes.openingHours.some(entry => entry.days.includes(0))
                    && !venue.isResidential()) {
                    const tempAddr = venue.getAddress();
                    if (tempAddr && tempAddr.getStateName() === 'Indiana') {
                        result = new this();
                    }
                }
                return result;
            }
        },
        HoursOverlap: class extends FlagBase {
            constructor() { super(true, _SEVERITY.RED, 'Overlapping hours of operation. Place might not save.'); }

            static eval(hoursOverlap) {
                return hoursOverlap ? new this() : null;
            }
        },
        UnmappedRegion: class extends WLFlag {
            constructor() { super(true, _SEVERITY.RED, 'This category is usually not mapped in this region.', true, 'Whitelist unmapped category', 'unmappedRegion'); }
        },
        RestAreaName: class extends WLFlag {
            constructor(wl) {
                super(
                    true,
                    wl.restAreaName ? _SEVERITY.GREEN : _SEVERITY.RED,
                    'Rest area name is out of spec. Use the Rest Area wiki button below to view formats.',
                    !wl.restAreaName,
                    'Whitelist rest area name',
                    'restAreaName'
                );
            }

            static #venueIsFlaggable(venue, hasRestAreaCategory) {
                return hasRestAreaCategory && !/^Rest Area.* - /.test(venue.attributes.name);
            }

            static eval(venue, hasRestAreaCategory, wl) {
                return this.#venueIsFlaggable(venue, hasRestAreaCategory, wl) ? new Flag.RestAreaName(wl) : null;
            }
        },
        RestAreaNoTransportation: class extends ActionFlag {
            constructor(venue) {
                super(true, _SEVERITY.YELLOW, 'Rest areas should not use the Transportation category.', 'Remove it?');
                this.venue = venue;
            }

            static #venueIsFlaggable(hasRestAreaCategory, categories) {
                return hasRestAreaCategory && categories.includes('TRANSPORTATION');
            }

            static eval(venue, hasRestAreaCategory, categories) {
                return this.#venueIsFlaggable(hasRestAreaCategory, categories) ? new Flag.RestAreaNoTransportation(venue) : null;
            }

            action() {
                const categories = this.venue.getCategories().slice(); // create a copy
                const index = categories.indexOf('TRANSPORTATION');
                if (index > -1) {
                    categories.splice(index, 1); // remove the category
                    addUpdateAction(this.venue, { categories }, null, true);
                }
            }
        },
        RestAreaGas: class extends FlagBase {
            constructor() { super(true, _SEVERITY.RED, 'Gas stations at Rest Areas should be separate area places.'); }

            static eval(hasRestAreaCategory, categories) {
                return hasRestAreaCategory && categories.includes('GAS_STATION') ? new Flag.RestAreaGas() : null;
            }
        },
        RestAreaScenic: class extends WLActionFlag {
            constructor(venue) {
                super(
                    true,
                    _SEVERITY.GREEN,
                    'Verify that the "Scenic Overlook" category is appropriate for this rest area.  If not: ',
                    'Remove it',
                    'Remove "Scenic Overlook" category.',
                    true,
                    'Whitelist place',
                    'restAreaScenic'
                );
                this.venue = venue;
            }

            static #venueIsFlaggable(hasRestAreaCategory, categories, wl) {
                return !wl.restAreaScenic
                    && hasRestAreaCategory
                    && categories.includes('SCENIC_LOOKOUT_VIEWPOINT');
            }

            static eval(venue, hasRestAreaCategory, categories, wl) {
                return this.#venueIsFlaggable(hasRestAreaCategory, categories, wl) ? new Flag.RestAreaScenic(venue) : null;
            }

            action() {
                const categories = this.venue.getCategories().slice(); // create a copy
                const index = categories.indexOf('SCENIC_LOOKOUT_VIEWPOINT');
                if (index > -1) {
                    categories.splice(index, 1); // remove the category
                    addUpdateAction(this.venue, { categories }, null, true);
                }
            }
        },
        RestAreaSpec: class extends WLActionFlag {
            constructor(venue) {
                super(
                    true,
                    _SEVERITY.RED,
                    'Is this a rest area?',
                    'Yes',
                    'Update with proper categories and services.',
                    true,
                    'Whitelist place',
                    'restAreaSpec'
                );
                this.venue = venue;
            }

            static #venueIsFlaggable(hasRestAreaCategory, name, wl) {
                return !wl.restAreaSpec
                    && !hasRestAreaCategory
                    && (/rest (?:area|stop)|service plaza/i.test(name));
            }

            static eval(venue, hasRestAreaCategory, name, wl) {
                return this.#venueIsFlaggable(hasRestAreaCategory, name, wl) ? new Flag.RestAreaSpec(venue) : null;
            }

            action() {
                const categories = insertAtIndex(this.venue.getCategories(), 'REST_AREAS', 0);
                // make it 24/7
                const openingHours = [new OpeningHour({ days: [1, 2, 3, 4, 5, 6, 0], fromHour: '00:00', toHour: '00:00' })];
                addUpdateAction(this.venue, { categories, openingHours }, null, true);
            }
        },
        EVChargingStationWarning: class extends FlagBase {
            constructor() {
                super(
                    true,
                    _SEVERITY.GREEN,
                    'Please do not delete EV Charging Stations. Be sure you are completely up to date with the latest guidelines in '
                        + '<a href="https://wazeopedia.waze.com/wiki/USA/Places/EV_charging_station" target="_blank">wazeopedia</a>.'
                );
            }

            static eval(venue, highlightOnly) {
                return !highlightOnly && venue.isChargingStation() ? new this() : null;
            }
        },
        EVCSPriceMissing: class extends FlagBase {
            constructor(venue, highlightOnly) {
                super(true, _SEVERITY.BLUE, 'EVCS price: ');
                if (!highlightOnly) {
                    [['FREE', 'Free', 'Free'], ['FEE', 'Paid', 'Paid']].forEach(btnInfo => {
                        this.message += $('<button>', {
                            id: `wmeph_${btnInfo[0]}`,
                            class: 'wmeph-evcs-cost-type-btn btn btn-default btn-xs wmeph-btn',
                            title: btnInfo[2]
                        })
                            .text(btnInfo[1])
                            .css({
                                padding: '3px',
                                height: '20px',
                                lineHeight: '0px',
                                marginRight: '2px',
                                marginBottom: '1px',
                                minWidth: '18px'
                            })
                            .prop('outerHTML');
                    });
                    this.venue = venue;
                    this.noLock = true;
                }
            }

            static #venueIsFlaggable(venue) {
                const evcsAttr = venue.attributes.categoryAttributes?.CHARGING_STATION;
                return venue.isChargingStation()
                    && (!evcsAttr?.costType || evcsAttr.costType === 'COST_TYPE_UNSPECIFIED');
            }

            static eval(venue, highlightOnly) {
                return this.#venueIsFlaggable(venue) ? new this(venue, highlightOnly) : null;
            }

            postProcess() {
                $('.wmeph-evcs-cost-type-btn').click(evt => {
                    const selectedValue = $(evt.currentTarget).attr('id').replace('wmeph_', '');
                    let attrClone;
                    if (this.venue.attributes.categoryAttributes) {
                        attrClone = JSON.parse(JSON.stringify(this.venue.attributes.categoryAttributes));
                    } else {
                        attrClone = {};
                    }
                    attrClone.CHARGING_STATION ??= {};
                    attrClone.CHARGING_STATION.costType = selectedValue;
                    addUpdateAction(this.venue, { categoryAttributes: attrClone }, null, true);
                    _UPDATED_FIELDS.evCostType.updated = true;
                });
            }
        },
        GasMismatch: class extends WLFlag {
            constructor(wl) {
                super(
                    true,
                    _SEVERITY.RED,
                    '<a href="https://wazeopedia.waze.com/wiki/USA/Places/Gas_station#Name" target="_blank" class="red">Gas brand should typically be included in the place name.</a>',
                    !wl.gasMismatch,
                    'Whitelist gas brand / name mismatch',
                    'gasMismatch'
                );
                if (!wl.gasMismatch) {
                    this.noLock = true;
                }
            }

            static #venueIsFlaggable(venue, categories, brand, name) {
                // For gas stations, check to make sure brand exists somewhere in the place name.
                // Remove non - alphanumeric characters first, for more relaxed matching.
                if (categories[0] === 'GAS_STATION' && venue.attributes.brand) {
                    const compressedName = venue.attributes.name.toUpperCase().replace(/[^a-zA-Z0-9]/g, '');
                    const compressedNewName = name.toUpperCase().replace(/[^a-zA-Z0-9]/g, '');
                    // Some brands may have more than one acceptable name, or the brand listed in WME doesn't match what we want to see in the name.
                    // Ideally, this would be addressed in the PNH spreadsheet somehow, but for now hardcoding is the only option.
                    const compressedBrands = [brand.toUpperCase().replace(/[^a-zA-Z0-9]/g, '')];
                    if (brand === 'Diamond Gasoline') {
                        compressedBrands.push('DIAMONDOIL');
                    } else if (brand === 'Murphy USA') {
                        compressedBrands.push('MURPHY');
                    } else if (brand === 'Mercury Fuel') {
                        compressedBrands.push('MERCURY', 'MERCURYPRICECUTTER');
                    } else if (brand === 'Carrollfuel') {
                        compressedBrands.push('CARROLLMOTORFUEL', 'CARROLLMOTORFUELS');
                    }
                    if (compressedBrands.every(compressedBrand => !compressedName.includes(compressedBrand) && !compressedNewName.includes(compressedBrand))) {
                        return true;
                    }
                }
                return false;
            }

            static eval(venue, categories, brand, name, wl) {
                return this.#venueIsFlaggable(venue, categories, brand, name) ? new this(wl) : null;
            }
        },
        GasUnbranded: class extends FlagBase {
            //  Unbranded is not used per wiki
            constructor() {
                super(true, _SEVERITY.RED, '"Unbranded" should not be used for the station brand. Change to the correct brand or '
                    + 'delete the brand.');
                this.noLock = true;
            }

            static eval(venue, brand) {
                return venue.isGasStation() && brand === 'Unbranded' ? new this() : null;
            }
        },
        GasMkPrim: class extends ActionFlag {
            constructor(venue) {
                super(true, _SEVERITY.RED, 'Gas Station should be the primary category', 'Fix', 'Make the Gas Station '
                    + 'category the primary category.');
                this.venue = venue;
                this.noLock = true;
            }

            static #venueIsFlaggable(categories) {
                return categories.includes('GAS_STATION') && categories.indexOf('GAS_STATION') !== 0;
            }

            static eval(venue, categories) {
                return this.#venueIsFlaggable(categories) ? new this(venue) : null;
            }

            action() {
                // Move Gas category to the first position
                const categories = insertAtIndex(this.venue.getCategories(), 'GAS_STATION', 0);
                addUpdateAction(this.venue, { categories }, null, true);
            }
        },
        IsThisAPilotTravelCenter: class extends ActionFlag {
            constructor(venue, newName) {
                super(true, _SEVERITY.GREEN, 'Is this a "Travel Center"?', 'Yes', '');
                this.venue = venue;
                this.newName = newName;
            }

            static eval(venue, highlightOnly, state2L, newName, actions) {
                let result = null;
                if (!highlightOnly && state2L === 'TN') {
                    if (newName.toLowerCase().trim() === 'pilot') {
                        newName = 'Pilot Food Mart';
                        addUpdateAction(venue, { name: newName }, actions);
                    }
                    if (newName.toLowerCase().trim() === 'pilot food mart') {
                        result = new this(venue, newName);
                    }
                }
                return result;
            }

            action() {
                addUpdateAction(this.venue, { name: 'Pilot Travel Center' }, null, true);
            }
        },
        HotelMkPrim: class extends WLActionFlag {
            constructor(venue) {
                super(
                    true,
                    _SEVERITY.RED,
                    'Hotel category is not first',
                    'Fix',
                    'Make the Hotel category the primary category.',
                    true,
                    'Whitelist hotel as secondary category',
                    'hotelMkPrim'
                );
                this.venue = venue;
            }

            action() {
                // Insert/move Hotel category in the first position
                const categories = insertAtIndex(this.venue.attributes.categories.slice(), 'HOTEL', 0);
                addUpdateAction(this.venue, { categories }, null, true);
            }
        },
        ChangeToPetVet: class extends WLActionFlag {
            constructor(venue) {
                super(
                    true,
                    _SEVERITY.RED,
                    'Key words suggest this should be a Pet/Veterinarian category. Change?',
                    'Yes',
                    'Change to Pet/Veterinarian Category',
                    true,
                    'Whitelist Pet/Vet category',
                    'changeHMC2PetVet'
                );
                this.venue = venue;
                this.noLock = true;
            }

            static eval(venue, name, categories, wl) {
                let result = null;
                if (!wl.changeHMC2PetVet) {
                    const testName = name.toLowerCase().replace(/[^a-z]/g, ' ');
                    const testNameWords = testName.split(' ');
                    if ((categories.includes('HOSPITAL_URGENT_CARE') || categories.includes('DOCTOR_CLINIC'))
                        && (containsAny(testNameWords, _animalFullMatch) || _animalPartMatch.some(match => testName.includes(match)))) {
                        result = new this(venue);
                    }
                }
                return result;
            }

            action() {
                let updated = false;
                let categories = _.uniq(this.venue.attributes.categories.slice());
                categories.forEach((cat, idx) => {
                    if (cat === 'HOSPITAL_URGENT_CARE' || cat === 'DOCTOR_CLINIC') {
                        categories[idx] = 'PET_STORE_VETERINARIAN_SERVICES';
                        updated = true;
                    }
                });
                if (updated) {
                    categories = _.uniq(categories);
                }
                addUpdateAction(this.venue, { categories }, null, true);
            }
        },
        NotASchool: class extends WLFlag {
            constructor() {
                super(
                    true,
                    _SEVERITY.RED,
                    'Key words suggest this should not be School category.',
                    true,
                    'Whitelist School category',
                    'changeSchool2Offices'
                );
                this.noLock = true;
            }

            static eval(name, categories, wl) {
                let result = null;
                if (!wl.changeSchool2Offices) {
                    const testName = name.toLowerCase().replace(/[^a-z]/g, ' ');
                    const testNameWords = testName.split(' ');

                    if (categories.includes('SCHOOL')
                        && (containsAny(testNameWords, _schoolFullMatch) || _schoolPartMatch.some(match => testName.includes(match)))) {
                        result = new this();
                    }
                }
                return result;
            }
        },
        PointNotArea: class extends WLActionFlag {
            constructor(venue) {
                super(
                    true,
                    _SEVERITY.RED,
                    'This category should be a point place.',
                    'Change to point',
                    'Change to point place',
                    true,
                    'Whitelist point (not area)',
                    'pointNotArea'
                );
                this.venue = venue;
            }

            action() {
                if (this.venue.getCategories().includes('RESIDENCE_HOME')) {
                    // 7/1/2022 - Not sure if this is necessary? Can residence be converted to area? Either way, updateFeatureGeometry function no longer works.
                    // const centroid = venue.geometry.getCentroid();
                    // updateFeatureGeometry(venue, new OpenLayers.Geometry.Point(centroid.x, centroid.y));
                } else {
                    $('wz-checkable-chip.geometry-type-control-point').click();
                }
                harmonizePlaceGo(this.venue, 'harmonize'); // Rerun the script to update fields and lock
            }
        },
        AreaNotPoint: class extends WLActionFlag {
            constructor(venue) {
                super(
                    true,
                    _SEVERITY.RED,
                    'This category should be an area place.',
                    'Change to area',
                    'Change to Area',
                    true,
                    'Whitelist area (not point)',
                    'areaNotPoint'
                );
                this.venue = venue;
            }

            action() {
                $('wz-checkable-chip.geometry-type-control-area').click();
                // updateFeatureGeometry(venue, venue.getPolygonGeometry());
                harmonizePlaceGo(this.venue, 'harmonize');
            }
        },
        HnMissing: class extends WLActionFlag {
            constructor(venue) {
                super(
                    true,
                    _SEVERITY.RED,
                    'No HN: <input type="text" id="WMEPH-HNAdd" autocomplete="off" style="font-size:0.85em;width:100px;padding-left:2px;color:#000;" > ',
                    'Add',
                    'Add HN to place',
                    true,
                    'Whitelist empty HN',
                    'HNWL'
                );
                this.venue = venue;
                this.noBannerAssemble = true;
                this.badInput = false;
            }

            action() {
                const newHN = $('#WMEPH-HNAdd').val().replace(/\s+/g, '');
                logDev(newHN);
                const hnTemp = newHN.replace(/[^\d]/g, '');
                const hnTempDash = newHN.replace(/[^\d-]/g, '');
                if (hnTemp > 0 && hnTemp < 1000000) {
                    const action = new UpdateObject(this.venue, { houseNumber: hnTempDash });
                    action.wmephDescription = `Changed house # to: ${hnTempDash}`;
                    harmonizePlaceGo(this.venue, 'harmonize', [action]); // Rerun the script to update fields and lock
                    _UPDATED_FIELDS.address.updated = true;
                } else {
                    $('input#WMEPH-HNAdd').css({ backgroundColor: '#FDD' }).attr('title', 'Must be a number between 0 and 1000000');
                    this.badInput = true;
                }
            }
        },
        HnTooManyDigits: class extends WLFlag {
            constructor() {
                super(true, _SEVERITY.YELLOW, 'HN contains more than 6 digits. Please verify.', true, 'Whitelist long HN', 'hnTooManyDigits');
                this.noLock = true;
            }

            static eval(houseNumber, wl) {
                let result = null;
                if (!wl.hnTooManyDigits && houseNumber) {
                    houseNumber = houseNumber.replace(/[^0-9]/g, '');
                    if (houseNumber.length > 6) {
                        result = new this();
                    }
                }
                return result;
            }
        },
        // 2019-5-22 There's an issue in WME where it won't update the address displayed in the side panel
        // when the underlying model is updated.  I changed to the code below for a while, but we've
        // come up with a temporary fix using WW, so using the textbox entry should be OK now.
        // HnMissing: class extends WLActionFlag {
        //     constructor() { super(true, SEVERITY.RED, 'No HN:', 'Edit address', 'Edit address to add HN.'); }

        //     // eslint-disable-next-line class-methods-use-this
        //     action() {
        //         $('.nav-tabs a[href="#venue-edit-general"]').trigger('click');
        //         $('.venue .full-address').click();
        //         $('input.house-number').focus();
        //     }
        // },
        // 2020-10-5 HN's with letters have been allowed since last year.  Currently, RPPs can be saved with a number
        // followed by up to 4 letters but it's not clear if the app actually searches if only 1, 2, or more letters
        // are present.  Other places can have a more flexible HN (up to 15 characters long, total. A single space between
        // the # and letters. Etc)
        // HnNonStandard: class extends WLFlag {
        //     constructor() {
        //         super(true, SEVERITY.RED, 'House number is non-standard.', true,
        //             'Whitelist non-standard HN', 'hnNonStandard');
        //     }
        // },
        HNRange: class extends WLFlag {
            constructor() {
                super(
                    true,
                    _SEVERITY.YELLOW,
                    'House number seems out of range for the street name. Verify.',
                    true,
                    'Whitelist HN range',
                    'HNRange'
                );
            }
        },
        StreetMissing: class extends ActionFlag {
            constructor(primaryCategory) {
                super(true, _SEVERITY.RED, 'No street:', 'Edit address', 'Edit address to add street.');
                if (['SCENIC_LOOKOUT_VIEWPOINT'].includes(primaryCategory)) {
                    this.severity = _SEVERITY.BLUE;
                } else {
                    this.noLock = true;
                }
            }

            static #venueIsFlaggable(venue, addr) {
                return addr.city
                    && (!addr.street || addr.street.isEmpty)
                    && !['BRIDGE', 'ISLAND', 'FOREST_GROVE', 'SEA_LAKE_POOL', 'RIVER_STREAM', 'CANAL',
                        'DAM', 'TUNNEL', 'JUNCTION_INTERCHANGE'].includes(venue.attributes.categories[0])
                    && !venue.attributes.categories.includes('REST_AREAS');
            }

            static eval(venue, addr) {
                return this.#venueIsFlaggable(venue, addr) ? new this(venue.attributes.categories[0]) : null;
            }

            // eslint-disable-next-line class-methods-use-this
            action() {
                clickGeneralTab();
                $('.venue .full-address').click();
                setTimeout(() => {
                    if ($('.empty-street').prop('checked')) {
                        $('.empty-street').click();
                    }
                    setTimeout(() => {
                        const elem = document
                            .querySelector('#venue-edit-general > div:nth-child(1) > div > div > wz-card > form > div:nth-child(2) > div > wz-autocomplete')
                            .shadowRoot.querySelector('#text-input')
                            .shadowRoot.querySelector('#id');
                        elem.focus();
                    }, 100);
                }, 100);
            }
        },
        CityMissing: class extends ActionFlag {
            constructor(isResidential, highlightOnly) {
                super(true, _SEVERITY.RED, 'No city:', 'Edit address', 'Edit address to add city.');
                this.noLock = true;
                if (isResidential && highlightOnly) {
                    this.severity = _SEVERITY.BLUE;
                }
            }

            static #venueIsFlaggable(venue, addr) {
                return (!addr.city || addr.city.attributes.isEmpty)
                    && !['BRIDGE', 'ISLAND', 'FOREST_GROVE', 'SEA_LAKE_POOL', 'RIVER_STREAM', 'CANAL',
                        'DAM', 'TUNNEL', 'JUNCTION_INTERCHANGE'].includes(venue.attributes.categories[0])
                    && !venue.attributes.categories.includes('REST_AREAS');
            }

            static eval(venue, addr, highlightOnly) {
                return this.#venueIsFlaggable(venue, addr) ? new this(venue.attributes.residential, highlightOnly) : null;
            }

            // eslint-disable-next-line class-methods-use-this
            action() {
                clickGeneralTab();
                $('.venue .full-address').click();
                setTimeout(() => {
                    if ($('.empty-city').prop('checked')) {
                        $('.empty-city').click();
                    }
                    setTimeout(() => {
                        const elem = document
                            .querySelector('#venue-edit-general > div:nth-child(1) > div > div > wz-card > form > div:nth-child(4) > wz-autocomplete')
                            .shadowRoot.querySelector('#text-input')
                            .shadowRoot.querySelector('#id');
                        elem.focus();
                    }, 100);
                }, 100);

                $('.city-name').focus();
            }
        },
        BankType1: class extends FlagBase {
            constructor() { super(true, _SEVERITY.RED, 'Clarify the type of bank: the name has ATM but the primary category is Offices'); }
        },
        // TODO: Fix if the name has "(ATM)" or " - ATM" or similar. This flag is not currently catching those.
        BankBranch: class extends ActionFlag {
            constructor(venue) {
                super(true, _SEVERITY.BLUE, 'Is this a bank branch office? ', 'Yes', 'Is this a bank branch?');
                this.venue = venue;
            }

            action() {
                const newAttributes = {};

                const originalCategories = this.venue.getCategories();
                const newCategories = insertAtIndex(originalCategories, ['BANK_FINANCIAL', 'ATM'], 0); // Change to bank and atm cats
                if (!arraysAreEqual(originalCategories, newCategories)) {
                    newAttributes.categories = newCategories;
                }

                // strip ATM from name if present
                const originalName = this.venue.getName();
                const newName = originalName.replace(/[- (]*ATM[- )]*/ig, ' ').replace(/^ /g, '').replace(/ $/g, '');
                if (originalName !== newName) {
                    newAttributes.name = newName;
                    _UPDATED_FIELDS.name.updated = true;
                }

                addUpdateAction(this.venue, newAttributes, null, true);
            }
        },
        StandaloneATM: class extends ActionFlag {
            constructor(venue) {
                super(true, _SEVERITY.YELLOW, 'Or is this a standalone ATM? ', 'Yes', 'Is this a standalone ATM with no bank branch?');
                this.venue = venue;
            }

            action() {
                const newAttributes = {};

                const originalName = this.venue.getName();
                if (!/\bATM\b/i.test(originalName)) {
                    newAttributes.name = `${originalName} ATM`;
                    _UPDATED_FIELDS.name.updated = true;
                }

                const atmCategory = ['ATM'];
                if (!arraysAreEqual(this.venue.getCategories(), atmCategory)) {
                    newAttributes.categories = atmCategory; // Change to ATM only
                }

                addUpdateAction(this.venue, newAttributes, null, true);
            }
        },
        BankCorporate: class extends ActionFlag {
            constructor(venue) {
                super(true, _SEVERITY.BLUE, 'Or is this the bank\'s corporate offices?', 'Yes', 'Is this the bank\'s corporate offices?');
                this.venue = venue;
            }

            action() {
                const newAttributes = {};

                const officesCategory = ['OFFICES'];
                if (!arraysAreEqual(this.venue.getCategories(), officesCategory)) {
                    newAttributes.categories = officesCategory;
                }

                // strip ATM from name if present
                const originalName = this.venue.getName();
                let newName = originalName
                    .replace(/[- (]*atm[- )]*/ig, ' ')
                    .replace(/^ /g, '')
                    .replace(/ $/g, '')
                    .replace(/ {2,}/g, ' ')
                    .replace(/\s*-\s*corporate\s*offices\s*$/i, '');
                const suffix = ' - Corporate Offices';
                if (!newName.endsWith(suffix)) newName += suffix;
                if (originalName !== newName) {
                    newAttributes.name = newName;
                    _UPDATED_FIELDS.name.updated = true;
                }

                addUpdateAction(this.venue, newAttributes, null, true);
            }
        },
        CatPostOffice: class extends FlagBase {
            constructor() {
                super(
                    true,
                    _SEVERITY.GREEN,
                    `The Post Office category is reserved for certain USPS locations. Please be sure to follow <a href="${
                        _URLS.uspsWiki}" style="color:#3a3a3a;" target="_blank">the guidelines</a>.`
                );
            }
        },
        IgnEdited: class extends FlagBase {
            constructor() { super(true, _SEVERITY.YELLOW, 'Last edited by an IGN editor'); }
        },
        WazeBot: class extends ActionFlag {
            constructor(venue) {
                super(
                    true,
                    _SEVERITY.YELLOW,
                    'Edited last by an automated process. Please verify information is correct.',
                    'Nudge',
                    'If no other properties need to be updated, click to nudge the place (force an edit).'
                );
                this.venue = venue;
            }

            action() {
                nudgeVenue(this.venue);
                harmonizePlaceGo(this.venue, 'harmonize');
            }
        },
        ParentCategory: class extends WLFlag {
            constructor() {
                super(
                    true,
                    _SEVERITY.YELLOW,
                    'This parent category is usually not mapped in this region.',
                    true,
                    'Whitelist parent Category',
                    'parentCategory'
                );
            }
        },
        CheckDescription: class extends FlagBase {
            constructor() {
                super(
                    true,
                    _SEVERITY.YELLOW,
                    'Description field already contained info; PNH description was added in front of existing. Check for inconsistency or duplicate info.'
                );
            }
        },
        Overlapping: class extends FlagBase {
            constructor() { super(true, _SEVERITY.YELLOW, 'Place points are stacked up.'); }
        },
        SuspectDesc: class extends WLFlag {
            constructor() { super(true, _SEVERITY.YELLOW, 'Description field might contain copyrighted info.', true, 'Whitelist description', 'suspectDesc'); }
        },
        ResiTypeName: class extends WLFlag {
            constructor() {
                super(
                    true,
                    _SEVERITY.YELLOW,
                    'The place name suggests a residential place or personalized place of work.  Please verify.',
                    true,
                    'Whitelist Residential-type name',
                    'resiTypeName'
                );
            }
        },
        Mismatch247: class extends FlagBase {
            constructor() { super(true, _SEVERITY.YELLOW, 'Hours of operation listed as open 24hrs but not for all 7 days.'); }

            static #venueIsFlaggable(venue) {
                if (venue.attributes.openingHours.length === 1) { // if one set of hours exist, check for partial 24hrs setting
                    const hoursEntry = venue.attributes.openingHours[0];
                    if (hoursEntry.days.length < 7 && /^0?0:00$/.test(hoursEntry.fromHour)
                        && (/^0?0:00$/.test(hoursEntry.toHour) || hoursEntry.toHour === '23:59')) {
                        return true;
                    }
                }
                return false;
            }

            static eval(venue) {
                return this.#venueIsFlaggable(venue) ? new this() : null;
            }
        },
        PhoneInvalid: class extends FlagBase {
            constructor() { super(true, _SEVERITY.YELLOW, 'Phone # is invalid.'); }

            static eval(phone) {
                let result = null;
                if (phone === 'badPhone') {
                    result = new this();
                }
                return result;
            }
        },
        AreaNotPointMid: class extends WLFlag {
            constructor() {
                super(
                    true,
                    _SEVERITY.YELLOW,
                    'This category is usually an area place, but can be a point in some cases. Verify if point is appropriate.',
                    true,
                    'Whitelist area (not point)',
                    'areaNotPoint'
                );
            }
        },
        PointNotAreaMid: class extends WLFlag {
            constructor() {
                super(
                    true,
                    _SEVERITY.YELLOW,
                    'This category is usually a point place, but can be an area in some cases. Verify if area is appropriate.',
                    true,
                    'Whitelist point (not area)',
                    'pointNotArea'
                );
            }
        },
        LongURL: class extends WLActionFlag {
            constructor(venue, placePL) {
                super(
                    true,
                    _SEVERITY.BLUE,
                    'Existing URL doesn\'t match the suggested PNH URL. Use the Website button below to verify that existing URL is valid.  If not:',
                    'Use PNH URL',
                    'Change URL to the PNH standard',
                    true,
                    'Whitelist existing URL',
                    'longURL'
                );
                this.venue = venue;
                this.placePL = placePL;
            }

            action() {
                if (!isNullOrWhitespace(_tempPNHURL)) {
                    addUpdateAction(this.venue, { url: _tempPNHURL }, null, true);
                    _updateURL = true;
                    // // if the category doesn't translate, then pop an alert that will make a forum post to the thread
                    // reportError({
                    //     subject: 'WMEPH URL comparison Error report',
                    //     message: `Error report: URL comparison failed for "${venue.attributes.name}"\nPermalink: ${this.placePL}`
                    // });
                } else {
                    WazeWrap.Alerts.confirm(
                        _SCRIPT_NAME,
                        'WMEPH: URL Matching Error!<br>Click OK to report this error',
                        () => {
                            reportError({
                                subject: 'WMEPH URL comparison Error report',
                                message: `Error report: URL comparison failed for "${this.venue.getName()}"\nPermalink: ${this.placePL}`
                            });
                        },
                        () => { }
                    );
                }
            }
        },
        GasNoBrand: class extends FlagBase {
            constructor(levelToLock) {
                super(true, _SEVERITY.BLUE, `Lock to L${levelToLock + 1}+ to verify no gas brand.`);
                this.noLock = true;
            }

            static eval(venue, brand, levelToLock) {
                // If gas station is missing brand, don't flag if place is locked as high as user can lock it.
                let result = null;
                if (venue.isGasStation() && !brand && venue.attributes.lockRank < levelToLock) {
                    result = new this(levelToLock);
                }
                return result;
            }
        },
        SubFuel: class extends WLFlag {
            constructor() {
                super(
                    true,
                    _SEVERITY.BLUE,
                    'Make sure this place is for the gas station itself and not the main store building.  Otherwise undo and check the categories.',
                    true,
                    'Whitelist no gas brand',
                    'subFuel'
                );
            }
        },
        AddCommonEVPaymentMethods: class extends WLActionFlag {
            // TODO: Instead of passing WL key to super, add a public getter for WL key in all child WLFlag classes. Getter returns a static WL key member.
            // Parent class can then reference that. e.g. for "isWhitelisted()" function.
            static whitelistKey = 'addCommonEVPaymentMethods';
            constructor(venue, missingPaymentMethods, highlightOnly) {
                const stationAttr = venue.attributes.categoryAttributes.CHARGING_STATION;
                super(
                    true,
                    _SEVERITY.BLUE,
                    `These common payment methods for the ${stationAttr.network} network are missing. Verify if they are needed here:`,
                    'Add network payment methods',
                    'Please verify first! If any are not needed, click the WL button and manually add any needed payment methods.',
                    true,
                    'Whitelist common EV payment types',
                    Flag.AddCommonEVPaymentMethods.whitelistKey
                );

                if (!highlightOnly) {
                    this.venue = venue;
                    // Store a copy of the network to check if it has changed in the action() function
                    this.originalNetwork = stationAttr.network;
                    const translations = I18n.translations[I18n.locale].edit.venue.category_attributes.payment_methods;
                    const missingPaymentMethodsList = missingPaymentMethods.map(method => `- ${translations[method]}`).join('<br>');
                    this.message += `<br>${missingPaymentMethodsList}<br>`;
                }
            }

            static eval(venue, highlightOnly, wl) {
                let result = null;
                if (venue.isChargingStation() && !wl[this.whitelistKey]) {
                    const stationAttr = venue.attributes.categoryAttributes.CHARGING_STATION;
                    const network = stationAttr?.network;
                    if (network && COMMON_EV_PAYMENT_METHODS.hasOwnProperty(network)) {
                        const missingMethods = COMMON_EV_PAYMENT_METHODS[network].filter(method => !stationAttr.paymentMethods?.includes(method));
                        if (missingMethods.length) {
                            result = new this(venue, missingMethods, highlightOnly);
                        }
                    }
                }
                return result;
            }

            action() {
                if (!this.venue.isChargingStation()) {
                    WazeWrap.Alerts.info(_SCRIPT_NAME, 'This is no longer a charging station. Please run WMEPH again.', false, false);
                    return;
                }

                const stationAttr = this.venue.attributes.categoryAttributes.CHARGING_STATION;
                const network = stationAttr?.network;
                if (network !== this.originalNetwork) {
                    WazeWrap.Alerts.info(_SCRIPT_NAME, 'EV charging station network has changed. Please run WMEPH again.', false, false);
                    return;
                }

                const newPaymentMethods = stationAttr.paymentMethods?.slice() || [];
                const commonPaymentMethods = COMMON_EV_PAYMENT_METHODS[network];
                commonPaymentMethods.forEach(method => {
                    if (!newPaymentMethods.includes(method)) newPaymentMethods.push(method);
                });

                const categoryAttrClone = JSON.parse(JSON.stringify(this.venue.getCategoryAttributes()));
                if (!categoryAttrClone.CHARGING_STATION) {
                    categoryAttrClone.CHARGING_STATION = {};
                }
                categoryAttrClone.CHARGING_STATION.paymentMethods = newPaymentMethods;

                _UPDATED_FIELDS.evPaymentMethods.updated = true;
                addUpdateAction(this.venue, { categoryAttributes: categoryAttrClone }, null, true);
            }
        },
        RemoveUncommonEVPaymentMethods: class extends WLActionFlag {
            // TODO: Instead of passing WL key to super, add a public getter for WL key in all child WLFlag classes. Getter returns a static WL key member.
            // Parent class can then reference that. e.g. for "isWhitelisted()" function.
            static whitelistKey = 'removeUncommonEVPaymentMethods';
            constructor(venue, extraPaymentMethods, highlightOnly) {
                const stationAttr = venue.attributes.categoryAttributes.CHARGING_STATION;
                super(
                    true,
                    _SEVERITY.BLUE,
                    `These payment methods are uncommon for the ${stationAttr.network} network. Verify if they are needed here:`,
                    'Remove network payment methods',
                    'Please verify first! If any should NOT be removed, click the WL button and manually remove any unneeded payment methods.',
                    true,
                    'Whitelist uncommon EV payment types',
                    Flag.RemoveUncommonEVPaymentMethods.whitelistKey
                );

                if (!highlightOnly) {
                    this.venue = venue;
                    // Store a copy of the network to check if it has changed in the action() function
                    this.originalNetwork = stationAttr.network;
                    const translations = I18n.translations[I18n.locale].edit.venue.category_attributes.payment_methods;
                    const extraPaymentMethodsList = extraPaymentMethods.map(method => `- ${translations[method]}`).join('<br>');
                    this.message += `<br>${extraPaymentMethodsList}<br>`;
                }
            }

            static eval(venue, highlightOnly, wl) {
                let result = null;
                if (venue.isChargingStation() && !wl[this.whitelistKey]) {
                    const stationAttr = venue.attributes.categoryAttributes.CHARGING_STATION;
                    const network = stationAttr?.network;
                    if (network && COMMON_EV_PAYMENT_METHODS.hasOwnProperty(network)) {
                        const extraMethods = stationAttr.paymentMethods?.filter(method => !COMMON_EV_PAYMENT_METHODS[network].includes(method));
                        if (extraMethods?.length) {
                            result = new this(venue, extraMethods, highlightOnly);
                        }
                    }
                }
                return result;
            }

            action() {
                if (!this.venue.isChargingStation()) {
                    WazeWrap.Alerts.info('This is no longer a charging station. Please run WMEPH again.', false, false);
                    return;
                }

                const stationAttr = this.venue.attributes.categoryAttributes.CHARGING_STATION;
                const network = stationAttr?.network;
                if (network !== this.originalNetwork) {
                    WazeWrap.Alerts.info(_SCRIPT_NAME, 'EV charging station network has changed. Please run WMEPH again.', false, false);
                    return;
                }

                const commonPaymentMethods = COMMON_EV_PAYMENT_METHODS[network];
                const newPaymentMethods = (stationAttr.paymentMethods?.slice() || [])
                    .filter(method => commonPaymentMethods.includes(method));

                const categoryAttrClone = JSON.parse(JSON.stringify(this.venue.getCategoryAttributes()));
                if (!categoryAttrClone.CHARGING_STATION) {
                    categoryAttrClone.CHARGING_STATION = {};
                }
                categoryAttrClone.CHARGING_STATION.paymentMethods = newPaymentMethods;

                _UPDATED_FIELDS.evPaymentMethods.updated = true;
                addUpdateAction(this.venue, { categoryAttributes: categoryAttrClone }, null, true);
            }
        },
        AreaNotPointLow: class extends WLFlag {
            constructor() {
                super(
                    true,
                    _SEVERITY.BLUE,
                    'This category is usually an area place, but can be a point in some cases. Verify if point is appropriate.',
                    true,
                    'Whitelist area (not point)',
                    'areaNotPoint'
                );
            }
        },
        PointNotAreaLow: class extends WLFlag {
            constructor() {
                super(
                    true,
                    _SEVERITY.BLUE,
                    'This category is usually a point place, but can be an area in some cases. Verify if area is appropriate.',
                    true,
                    'Whitelist point (not area)',
                    'pointNotArea'
                );
            }
        },
        FormatUSPS: class extends FlagBase {
            constructor() {
                super(
                    true,
                    _SEVERITY.BLUE,
                    `Name the post office according to this region's <a href="${
                        _URLS.uspsWiki}" style="color:#3232e6" target="_blank">standards for USPS post offices</a>`
                );
            }
        },
        MissingUSPSAlt: class extends FlagBase {
            constructor() { super(true, _SEVERITY.BLUE, 'USPS post offices must have an alternate name of "USPS".'); }
        },
        MissingUSPSZipAlt: class extends WLActionFlag {
            constructor(venue, name, wl, highlightOnly) {
                let severity;
                let wlActive;
                let message;
                let zipMatch;

                if (!highlightOnly) {
                    message = `No <a href="${_URLS.uspsWiki}" style="color:#3232e6;" target="_blank">ZIP code alt name</a>: <input type="text" \
id="WMEPH-zipAltNameAdd"autocomplete="off" style="font-size:0.85em;width:65px;padding-left:2px;color:#000;" title="Enter the ZIP code and click Add">`;
                    zipMatch = name.match(/\d{5}/);
                }

                if (wl.missingUSPSZipAlt) {
                    severity = _SEVERITY.GREEN;
                    wlActive = false;
                } else {
                    severity = _SEVERITY.BLUE;
                    wlActive = true;
                }

                super(true, severity, message, 'Add', wlActive, 'Whitelist missing USPS zip alt name', 'missingUSPSZipAlt');
                this.venue = venue;
                this.noBannerAssemble = true;

                // If the zip code appears in the primary name, pre-fill it in the text entry box.
                if (zipMatch) this.suggestedValue = zipMatch;
            }

            static #venueIsFlaggable(isUspsPostOffice, aliases) {
                return isUspsPostOffice
                    && !aliases.some(alias => /\d{5}/.test(alias));
            }

            static eval(venue, isUspsPostOffice, name, aliases, wl, highlightOnly) {
                return this.#venueIsFlaggable(isUspsPostOffice, aliases) ? new this(venue, name, wl, highlightOnly) : null;
            }

            action() {
                const $input = $('input#WMEPH-zipAltNameAdd');
                const zip = $input.val().trim();
                if (zip) {
                    if (/^\d{5}/.test(zip)) {
                        const aliases = [].concat(this.venue.attributes.aliases);
                        // Make sure zip hasn't already been added.
                        if (!aliases.includes(zip)) {
                            aliases.push(zip);
                            addUpdateAction(this.venue, { aliases }, null, true);
                        } else {
                            $input.css({ backgroundColor: '#FDD' }).attr('title', 'Zip code alt name already exists');
                        }
                    } else {
                        $input.css({ backgroundColor: '#FDD' }).attr('title', 'Zip code format error');
                    }
                }
            }

            postProcess() {
                // If pressing enter in the USPS zip code alt entry box...
                $('#WMEPH-zipAltNameAdd').keyup(evt => {
                    if (evt.keyCode === 13 && $(evt.currentTarget).val() !== '') {
                        $('#WMEPH_missingUSPSZipAlt').click();
                    }
                });

                // Prefill zip code text box
                if (this.suggestedValue) {
                    $('input#WMEPH-zipAltNameAdd').val(this.suggestedValue);
                }
            }
        },
        MissingUSPSDescription: class extends WLFlag {
            constructor() {
                super(
                    true,
                    _SEVERITY.BLUE,
                    `The first line of the description for a <a href="${
                        _URLS.uspsWiki}" style="color:#3232e6" target="_blank">USPS post office</a> must be CITY, STATE ZIP, e.g. "Lexington, KY 40511"`,
                    true,
                    'Whitelist missing USPS address line in description',
                    'missingUSPSDescription'
                );
            }
        },
        CatHotel: class extends FlagBase {
            constructor(pnhName) { super(true, _SEVERITY.GREEN, `Check hotel website for any name localization (e.g. ${pnhName} - Tampa Airport).`); }
        },
        LocalizedName: class extends WLFlag {
            constructor() {
                super(true, _SEVERITY.BLUE, 'Place needs localization information', true, 'Whitelist localization', 'localizedName');
            }

            // TODO: This returns a plain object instead of a Flag instance (or null). It doesn't match the pattern of other eval functions. Fix it?
            static eval(name, nameSuffix, specCase, displayNote, wl) {
                const result = { flag: null, showDisplayNote: true };
                const checkLocalization = specCase.match(/^checkLocalization<>(.+)/i);
                if (checkLocalization) {
                    result.showDisplayNote = false;
                    const [, localizationString] = checkLocalization;
                    const localizationRegEx = new RegExp(localizationString, 'g');
                    if (!(name + (nameSuffix || '')).match(localizationRegEx)) {
                        result.flag = new this();
                        if (wl.localizedName) {
                            result.flag.WLactive = false;
                            result.flag.severity = _SEVERITY.GREEN;
                        }
                        if (displayNote) {
                            result.flag.message = displayNote;
                        }
                    }
                }
                return result;
            }
        },
        SpecCaseMessage: class extends FlagBase {
            constructor(message) { super(true, _SEVERITY.GREEN, message); }

            static eval(venue, message, showDisplayNote, specialCases) {
                let result = null;
                if (showDisplayNote && !isNullOrWhitespace(message)) {
                    if (containsAny(specialCases, ['pharmhours'])) {
                        if (!venue.attributes.description.toUpperCase().includes('PHARMACY') || (!venue.attributes.description.toUpperCase().includes('HOURS')
                            && !venue.attributes.description.toUpperCase().includes('HRS'))) {
                            result = new this(message);
                        }
                    } else if (containsAny(specialCases, ['drivethruhours'])) {
                        if (!venue.attributes.description.toUpperCase().includes('DRIVE') || (!venue.attributes.description.toUpperCase().includes('HOURS')
                            && !venue.attributes.description.toUpperCase().includes('HRS'))) {
                            if ($('#service-checkbox-DRIVETHROUGH').prop('checked')) {
                                result = new this(message);
                            }
                        }
                    } else {
                        // 3/23/2023 - This is a temporary solution to add a disambiguator for Tesla chargers.
                        const teslaSC = /tesla supercharger/i;
                        const teslaDC = /tesla destination charger/i;
                        const isTesla = teslaSC.test(message) && teslaDC.test(message);
                        if (isTesla) {
                            message = message.replace(teslaSC, '<button id="wmeph-tesla-supercharger" class="btn wmeph-btn">Tesla SuperCharger</button>');
                            message = message.replace(teslaDC, '<button id="wmeph-tesla-destination-charger" class="btn wmeph-btn">Tesla Destination Charger</button>');
                        }

                        result = new Flag.SpecCaseMessageLow(message); // KEEP THIS LINE (not part of Tesla stuff)

                        if (isTesla) {
                            result.postProcess = () => {
                                $('#wmeph-tesla-supercharger').click(() => {
                                    addUpdateAction(venue, { name: 'Tesla Supercharger' }, null, true);
                                });

                                $('#wmeph-tesla-destination-charger').click(() => {
                                    addUpdateAction(venue, { name: 'Tesla Destination Charger' }, null, true);
                                });
                            };
                            result.severity = _SEVERITY.RED;
                            result.noLock = true;
                        }
                    }
                }
                return result;
            }
        },
        PnhCatMess: class extends ActionFlag {
            constructor(venue, message, categories) {
                super(true, _SEVERITY.GREEN, message, null, null);
                if (categories.includes('HOSPITAL_URGENT_CARE')) {
                    this.value = 'Change to Doctor/Clinic';
                    this.actionType = 'changeToDoctorClinic';
                }
                this.venue = venue;
            }

            static eval(venue, message, categories, highlightOnly) {
                let result = null;
                if (!highlightOnly && !isNullOrWhitespace(message)) {
                    result = new this(venue, message, categories);
                }
                return result;
            }

            action() {
                if (this.actionType === 'changeToDoctorClinic') {
                    const categories = _.uniq(this.venue.attributes.categories.slice());
                    const indexOfHospital = categories.indexOf('HOSPITAL_URGENT_CARE');
                    if (indexOfHospital > -1) {
                        categories[indexOfHospital] = 'DOCTOR_CLINIC';
                        addUpdateAction(this.venue, { categories }, null, true);
                    }
                }
            }
        },
        SpecCaseMessageLow: class extends FlagBase {
            constructor(message) { super(true, _SEVERITY.GREEN, message); }
        },
        ExtProviderMissing: class extends ActionFlag {
            static #categoriesToIgnore = ['BRIDGE', 'TUNNEL', 'JUNCTION_INTERCHANGE', 'NATURAL_FEATURES', 'ISLAND',
                'SEA_LAKE_POOL', 'RIVER_STREAM', 'CANAL', 'SWAMP_MARSH'];

            constructor(venue, isLocked, actions) {
                super(true, _SEVERITY.RED, 'No Google link', 'Nudge', 'If no other properties need to be updated, click to nudge the place (force an edit).');
                this.venue = venue;
                this.value2 = 'Add';
                this.title2 = 'Add a link to a Google place';

                if (isLocked) {
                    let lastUpdated;
                    if (venue.isNew()) {
                        lastUpdated = Date.now();
                    } else if (venue.attributes.updatedOn) {
                        lastUpdated = venue.attributes.updatedOn;
                    } else {
                        lastUpdated = venue.attributes.createdOn;
                    }
                    const weeksSinceLastUpdate = (Date.now() - lastUpdated) / 604800000;
                    if (weeksSinceLastUpdate >= 26 && !venue.isUpdated() && (!actions || actions.length === 0)) {
                        this.severity = _SEVERITY.RED;
                        this.message += ' and place has not been edited for over 6 months. Edit a property (or nudge) and save to reset the 6 month timer: ';
                    } else {
                        this.severity = _SEVERITY.GREEN;
                        this.message += ': ';
                        delete this.value;
                    }
                } else {
                    this.severity = _SEVERITY.GREEN;
                    this.message += ': ';
                    delete this.value;
                }
            }

            static eval(venue, isLocked, categories, userRank, ignoreParkingLots, actions) {
                let result = null;
                if (userRank >= 2 && venue.areExternalProvidersEditable() && !(venue.isParkingLot() && ignoreParkingLots)) {
                    if (!categories.some(cat => this.#categoriesToIgnore.includes(cat))) {
                        const provIDs = venue.attributes.externalProviderIDs;
                        if (!(provIDs && provIDs.length)) {
                            result = new this(venue, isLocked, actions);
                        }
                    }
                }
                return result;
            }

            action() {
                nudgeVenue(this.venue);
                harmonizePlaceGo(this.venue, 'harmonize'); // Rerun the script to update fields and lock
            }

            action2() {
                clickGeneralTab();
                const venueName = this.venue.attributes.name;
                $('wz-button.external-provider-add-new').click();
                setTimeout(() => {
                    clickGeneralTab();
                    setTimeout(() => {
                        const elem = document.querySelector('div.external-provider-edit-form wz-autocomplete').shadowRoot.querySelector('wz-text-input').shadowRoot.querySelector('input');
                        elem.focus();
                        elem.value = venueName;
                        elem.dispatchEvent(new Event('input', { bubbles: true })); // NOTE: jquery trigger('input') and other event calls did not work.
                    }, 100);
                }, 100);
            }
        },
        UrlMissing: class extends WLActionFlag {
            constructor(venue, wl) {
                super(
                    true,
                    _SEVERITY.BLUE,
                    'No URL: <input type="text" id="WMEPH-UrlAdd" autocomplete="off" style="font-size:0.85em;width:100px;padding-left:2px;color:#000;">',
                    'Add',
                    'Add URL to place',
                    true,
                    'Whitelist empty URL',
                    'urlWL'
                );
                this.noBannerAssemble = true;
                this.badInput = false;
                this.venue = venue;
                this.wl = wl;
            }

            action() {
                const newUrl = normalizeURL($('#WMEPH-UrlAdd').val(), true, false, this.venue, null, this.wl);
                if ((!newUrl || newUrl.trim().length === 0) || newUrl === 'badURL') {
                    $('input#WMEPH-UrlAdd').css({ backgroundColor: '#FDD' }).attr('title', 'Invalid URL format');
                    // this.badInput = true;
                } else {
                    logDev(newUrl);
                    addUpdateAction(this.venue, { url: newUrl }, null, true);
                }
            }
        },
        BadAreaCode: class extends WLActionFlag {
            constructor(venue, textValue, outputFormat) {
                super(
                    true,
                    _SEVERITY.BLUE,
                    `Area Code mismatch:<br><input type="text" id="WMEPH-PhoneAdd" autocomplete="off" style="font-size:0.85em;width:100px;padding-left:2px;color:#000;" value="${textValue || ''}">`,
                    'Update',
                    'Update phone #',
                    true,
                    'Whitelist the area code',
                    'aCodeWL'
                );
                this.venue = venue;
                this.outputFormat = outputFormat;
                this.noBannerAssemble = true;
            }

            action() {
                const newPhone = normalizePhone($('#WMEPH-PhoneAdd').val(), this.outputFormat);
                if (newPhone === 'badPhone') {
                    $('input#WMEPH-PhoneAdd').css({ backgroundColor: '#FDD' }).attr('title', 'Invalid phone # format');
                    this.badInput = true;
                } else {
                    this.badInput = false;
                    logDev(newPhone);
                    addUpdateAction(this.venue, { phone: newPhone }, null, true);
                }
            }
        },
        AddRecommendedPhone: class extends WLActionFlag {
            static whitelistKey = 'addRecommendedPhone';
            constructor(venue, phone) {
                super(
                    true,
                    _SEVERITY.BLUE,
                    `Recommended phone #:<br>${phone}`,
                    'Add',
                    'Use recommended chain phone #',
                    true,
                    'Whitelist recommended phone #',
                    Flag.AddRecommendedPhone.whitelistKey
                );
                this.venue = venue;
                this.phone = phone;
            }

            static eval(venue, specCases, outputFormat, wl) {
                let result = null;
                if (!wl[this.whitelistKey]) {
                    const specCaseMatch = specCases.match(/phone<>(.*?)<>/);
                    if (specCaseMatch) {
                        const phone = normalizePhone(specCaseMatch[1], outputFormat);
                        if (phone !== 'badPhone' && phone !== venue.attributes.phone) {
                            result = new this(venue, phone);
                        }
                    }
                }
                return result;
            }

            action() {
                addUpdateAction(this.venue, { phone: this.phone }, null, true);
            }
        },
        PhoneMissing: class extends WLActionFlag {
            static whitelistKey = 'phoneWL';
            constructor(venue, hasOperator, wl, outputFormat, isPLA) {
                super(
                    true,
                    _SEVERITY.BLUE,
                    'No ph#: <input type="text" id="WMEPH-PhoneAdd" autocomplete="off" style="font-size:0.85em;width:100px;padding-left:2px;color:#000;">',
                    'Add',
                    'Add phone to place',
                    true,
                    'Whitelist empty phone',
                    Flag.PhoneMissing.whitelistKey
                );
                this.noBannerAssemble = true;
                this.badInput = false;
                this.outputFormat = outputFormat;
                this.venue = venue;
                if ((isPLA && !hasOperator) || wl[this.WLkeyName]) {
                    this.severity = _SEVERITY.GREEN;
                    this.WLactive = false;
                }
            }

            static get _regionsThatWantPlaPhones() { return ['SER']; }

            static eval(venue, newPhone, wl, region, outputFormat, skipPhoneCheck) {
                let result = null;
                if (!newPhone && !skipPhoneCheck && !wl[this.whitelistKey]) {
                    const hasOperator = venue.attributes.brand && W.model.categoryBrands?.PARKING_LOT?.includes(venue.attributes.brand);
                    const isPLA = venue.isParkingLot();
                    if (!isPLA || (isPLA && (this._regionsThatWantPlaPhones.includes(region) || hasOperator))) {
                        result = new this(venue, hasOperator, wl, outputFormat, isPLA);
                    }
                }
                return result;
            }

            action() {
                const newPhone = normalizePhone($('#WMEPH-PhoneAdd').val(), this.outputFormat);
                if (newPhone === 'badPhone') {
                    $('input#WMEPH-PhoneAdd').css({ backgroundColor: '#FDD' }).attr('title', 'Invalid phone # format');
                    this.badInput = true;
                } else {
                    this.badInput = false;
                    logDev(newPhone);
                    addUpdateAction(this.venue, { phone: newPhone }, null, true);
                }
            }
        },
        NoHours: class extends WLFlag {
            constructor(venue, categories, wl, highlightOnly, actions) {
                let severity;
                let wlActive = true;
                let message;
                const hours = actions?.find(action => action.object === venue && action.newAttributes?.openingHours)?.newAttributes.openingHours
                    || venue.attributes.openingHours;
                if (!hours.length) { // if no hours...
                    if (!highlightOnly) message = Flag.NoHours.#getHoursHtml();
                    if (Flag.NoHours.#noHoursIsOk(categories, wl)) {
                        severity = _SEVERITY.GREEN;
                        wlActive = false;
                    } else {
                        severity = _SEVERITY.BLUE;
                        wlActive = true;
                    }
                } else {
                    if (!highlightOnly) message = Flag.NoHours.#getHoursHtml(true, isAlwaysOpen(venue));
                    severity = _SEVERITY.GREEN;
                    wlActive = false;
                }
                super(true, severity, message, wlActive, 'Whitelist "No hours"', 'noHours');
                this.venue = venue;
                this.hours = hours;
            }

            static #venueIsFlaggable(categories) {
                return !containsAny(categories, ['STADIUM_ARENA', 'CEMETERY', 'TRANSPORTATION', 'FERRY_PIER', 'SUBWAY_STATION',
                    'BRIDGE', 'TUNNEL', 'JUNCTION_INTERCHANGE', 'ISLAND', 'SEA_LAKE_POOL', 'RIVER_STREAM', 'FOREST_GROVE', 'CANAL',
                    'SWAMP_MARSH', 'DAM']);
            }

            static eval(venue, categories, wl, highlightOnly, actions) {
                return this.#venueIsFlaggable(categories) ? new this(venue, categories, wl, highlightOnly, actions) : null;
            }

            static #noHoursIsOk(categories, wl) {
                return wl.noHours
                    || $('#WMEPH-DisableHoursHL').prop('checked')
                    || containsAny(categories, ['SCHOOL', 'CONVENTIONS_EVENT_CENTER',
                        'CAMPING_TRAILER_PARK', 'COTTAGE_CABIN', 'COLLEGE_UNIVERSITY', 'GOLF_COURSE', 'SPORTS_COURT', 'MOVIE_THEATER',
                        'SHOPPING_CENTER', 'RELIGIOUS_CENTER', 'PARKING_LOT', 'PARK', 'PLAYGROUND', 'AIRPORT', 'FIRE_DEPARTMENT', 'POLICE_STATION',
                        'SEAPORT_MARINA_HARBOR', 'FARM', 'SCENIC_LOOKOUT_VIEWPOINT']);
            }

            static #getHoursHtml(hasExistingHours = false, alwaysOpen = false) {
                return $('<span>').append(
                    `${hasExistingHours ? 'Hours' : 'No hours'}:`,
                    !alwaysOpen ? $('<input>', {
                        class: 'btn btn-default btn-xs wmeph-btn',
                        id: 'WMEPH_noHours',
                        title: `Add pasted hours${hasExistingHours ? ' to existing hours' : ''}`,
                        type: 'button',
                        value: 'Add hours',
                        style: 'margin-bottom:4px; margin-right:0px; margin-left:3px;'
                    }) : '',
                    hasExistingHours ? $('<input>', {
                        class: 'btn btn-default btn-xs wmeph-btn',
                        id: 'WMEPH_noHours_2',
                        title: 'Replace existing hours with pasted hours',
                        type: 'button',
                        value: 'Replace all hours',
                        style: 'margin-bottom:4px; margin-right:0px; margin-left:3px;'
                    }) : '',
                    // jquery throws an error when setting autocomplete="off" in a jquery object (must use .autocomplete() function), so just use a string here.
                    // eslint-disable-next-line max-len
                    `<textarea id="WMEPH-HoursPaste" wrap="off" autocomplete="off" style="overflow:auto;width:84%;max-width:84%;min-width:84%;font-size:0.85em;height:24px;min-height:24px;max-height:300px;margin-bottom:-2px;padding-left:3px;color:#AAA;position:relative;z-index:1;">${_DEFAULT_HOURS_TEXT}`
                )[0].outerHTML;
            }

            static #getTitle(parseResult) {
                let title;
                if (parseResult.overlappingHours) {
                    title = 'Overlapping hours.  Check the existing hours.';
                } else if (parseResult.sameOpenAndCloseTimes) {
                    title = 'Open/close times cannot be the same.';
                } else {
                    title = 'Can\'t parse, try again';
                }
                return title;
            }

            applyHours(replaceAllHours) {
                let pasteHours = $('#WMEPH-HoursPaste').val();
                if (pasteHours === _DEFAULT_HOURS_TEXT) {
                    return;
                }
                logDev(pasteHours);
                pasteHours += !replaceAllHours ? `,${getOpeningHours(this.venue).join(',')}` : '';
                $('.nav-tabs a[href="#venue-edit-more-info"]').tab('show');
                const parser = new HoursParser();
                const parseResult = parser.parseHours(pasteHours);
                if (parseResult.hours && !parseResult.overlappingHours && !parseResult.sameOpenAndCloseTimes && !parseResult.parseError) {
                    logDev(parseResult.hours);
                    addUpdateAction(this.venue, { openingHours: parseResult.hours }, null, true);
                    $('#WMEPH-HoursPaste').val(_DEFAULT_HOURS_TEXT);
                } else {
                    log('Can\'t parse those hours');
                    this.severity = _SEVERITY.BLUE;
                    this.WLactive = true;
                    $('#WMEPH-HoursPaste').css({ 'background-color': '#FDD' }).attr({ title: Flag.NoHours.#getTitle(parseResult) });
                }
            }

            onAddHoursClick() {
                this.applyHours();
            }

            onReplaceHoursClick() {
                this.applyHours(true);
            }

            static #getDaysString(days) {
                const dayEnum = {
                    1: 'Mon',
                    2: 'Tue',
                    3: 'Wed',
                    4: 'Thu',
                    5: 'Fri',
                    6: 'Sat',
                    7: 'Sun'
                };
                const dayGroups = [];
                let lastGroup;
                let lastGroupDay = -1;
                days.forEach(day => {
                    if (day !== lastGroupDay + 1) {
                        // Not a consecutive day. Start a new group.
                        lastGroup = [];
                        dayGroups.push(lastGroup);
                    }
                    lastGroup.push(day);
                    lastGroupDay = day;
                });

                // Process the groups into strings
                const groupString = [];
                dayGroups.forEach(group => {
                    if (group.length < 3) {
                        group.forEach(day => {
                            groupString.push(dayEnum[day]);
                        });
                    } else {
                        const firstDay = dayEnum[group[0]];
                        const lastDay = dayEnum[group[group.length - 1]];
                        groupString.push(`${firstDay}–${lastDay}`);
                    }
                });
                if (groupString.length === 1 && groupString[0] === 'Mon–Sun') return 'Every day';
                return groupString.join(', ');
            }

            static #formatAmPm(time24Hrs) {
                const re = /^(\d{1,2}):(\d{2})/;
                const match = time24Hrs.match(re);
                if (match) {
                    let hour = parseInt(match[1], 10);
                    const minute = match[2];
                    let suffix;
                    if (hour === 12 && minute === '00') {
                        return 'noon';
                    }
                    if (hour === 0) {
                        if (minute === '00') {
                            return 'midnight';
                        }
                        hour = 12;
                        suffix = 'am';
                    } else if (hour < 12) {
                        suffix = 'am';
                    } else {
                        suffix = 'pm';
                        if (hour > 12) hour -= 12;
                    }
                    return `${hour}${minute === '00' ? '' : `:${minute}`} ${suffix}`;
                }
                return time24Hrs;
            }

            static #getHoursString(hoursObject) {
                if (hoursObject.isAllDay()) return 'All day';
                const fromHour = this.#formatAmPm(hoursObject.fromHour);
                const toHour = this.#formatAmPm(hoursObject.toHour);
                return `${fromHour}–${toHour}`;
            }

            static #getOrderedDaysArray(hoursObject) {
                const days = hoursObject.days.slice();
                // Change Sunday value from 0 to 7
                const sundayIndex = days.indexOf(0);
                if (sundayIndex > -1) {
                    days.splice(sundayIndex, 1);
                    days.push(7);
                }
                days.sort(); // Maybe not needed, but just in case
                return days;
            }

            static #getHoursStringArray(hoursObjects) {
                const daysWithHours = [];
                const outputArray = hoursObjects.map(hoursObject => {
                    const days = this.#getOrderedDaysArray(hoursObject);
                    daysWithHours.push(...days);

                    // Concatenate the group strings and append hours range
                    const daysString = this.#getDaysString(days);
                    const hoursString = this.#getHoursString(hoursObject);
                    return `${daysString}:&nbsp&nbsp${hoursString}`;
                });

                // Find closed days
                const closedDays = [1, 2, 3, 4, 5, 6, 7].filter(day => !daysWithHours.includes(day));
                if (closedDays.length) {
                    outputArray.push(`${this.#getDaysString(closedDays)}:&nbsp&nbspCLOSED`);
                }
                return outputArray;
            }

            postProcess() {
                if (this.hours.length) {
                    const hoursStringArray = Flag.NoHours.#getHoursStringArray(this.hours);
                    const $hoursTable = $('<div>', {
                        id: 'wmeph-hours-list',
                        style: 'display: inline-block;font-size: 13px;border: 1px solid #aaa;margin: -6px 2px 2px 0px;border-radius: 0px 0px 5px 5px;background-color: #f5f5f5;color: #727272;'
                            + 'padding: 3px 10px 0px 5px !important;z-index: 0;position: relative;min-width: 84%',
                        title: 'Current hours'
                    }).append(
                        hoursStringArray
                            .map((entry, idx) => `<div${idx < hoursStringArray.length - 1 ? ' style="border-bottom: 1px solid #ddd;"' : ''}>${entry}</div>`)
                            .join('')
                    );

                    $('#WMEPH-HoursPaste').after($hoursTable);
                }
                // NOTE: Leave these wrapped in the "() => ..." functions, to make sure "this" is bound properly.
                $('#WMEPH_noHours').click(() => this.onAddHoursClick());
                $('#WMEPH_noHours_2').click(() => this.onReplaceHoursClick());

                // If pasting or dropping into hours entry box
                function resetHoursEntryHeight() {
                    const $sel = $('#WMEPH-HoursPaste');
                    $sel.focus();
                    const oldText = $sel.val();
                    if (oldText === _DEFAULT_HOURS_TEXT) {
                        $sel.val('');
                    }

                    // A small delay to allow window to process pasted text before running.
                    setTimeout(() => {
                        const text = $sel.val();
                        const elem = $sel[0];
                        const lineCount = (text.match(/\n/g) || []).length + 1;
                        const height = lineCount * 18 + 6 + (elem.scrollWidth > elem.clientWidth ? 20 : 0);
                        $sel.css({ height: `${height}px` });
                    }, 0);
                }

                $('#WMEPH-HoursPaste').after($('<i>', {
                    id: 'wmeph-paste-hours-btn',
                    class: 'fa fa-paste',
                    style: 'font-size: 17px;position: relative;vertical-align: top;top: 2px;right: -5px;margin-right: 3px;color: #6c6c6c;cursor: pointer;',
                    title: 'Paste from the clipboard'
                })); // , $('<i>', {
                //     id: 'wmeph-clear-hours-btn',
                //     class: 'fa fa-trash-o',
                //     style: 'font-size: 17px;position: relative;right: -5px;bottom: 6px;color: #6c6c6c;cursor: pointer;margin-left: 5px;',
                //     title: 'Clear pasted hours'
                // }));

                $('#wmeph-paste-hours-btn').click(() => {
                    navigator.clipboard.readText().then(cliptext => {
                        $('#WMEPH-HoursPaste').val(cliptext);
                        resetHoursEntryHeight();
                    }, err => console.error(err));
                });

                // $('#wmeph-clear-hours-btn').click(() => {
                //     $('#WMEPH-HoursPaste').val(null);
                //     resetHoursEntryHeight();
                // });

                $('#WMEPH-HoursPaste')
                    .bind('paste', resetHoursEntryHeight)
                    .bind('drop', resetHoursEntryHeight)
                    .bind('dragenter', evt => {
                        const $control = $(evt.currentTarget);
                        const text = $control.val();
                        if (text === _DEFAULT_HOURS_TEXT) {
                            $control.val('');
                        }
                    }).keydown(evt => {
                        // If pressing enter in the hours entry box then parse the entry, or newline if CTRL or SHIFT.
                        resetHoursEntryHeight();
                        if (evt.keyCode === 13) {
                            if (evt.ctrlKey) {
                                // Simulate a newline event (shift + enter)
                                const target = evt.currentTarget;
                                const text = target.value;
                                const selStart = target.selectionStart;
                                target.value = `${text.substr(0, selStart)}\n${text.substr(target.selectionEnd, text.length - 1)}`;
                                target.selectionStart = selStart + 1;
                                target.selectionEnd = selStart + 1;
                                return true;
                            }
                            if (!(evt.shiftKey || evt.ctrlKey) && $(evt.currentTarget).val().length) {
                                evt.stopPropagation();
                                evt.preventDefault();
                                evt.returnValue = false;
                                evt.cancelBubble = true;
                                $('#WMEPH_noHours').click();
                                return false;
                            }
                        }
                        return true;
                    }).focus(evt => {
                        const target = evt.currentTarget;
                        if (target.value === _DEFAULT_HOURS_TEXT) {
                            target.value = '';
                        }
                        target.style.color = 'black';
                    }).blur(evt => {
                        const target = evt.currentTarget;
                        if (target.value === '') {
                            target.value = _DEFAULT_HOURS_TEXT;
                            target.style.color = '#999';
                        }
                    });
            }
        },
        OldHours: class extends ActionFlag {
            static #categoriesToCheck;
            static #cutoffDateString = '3/15/2020';
            static #cutoffDate = new Date(this.#cutoffDateString);

            constructor(venue, highlightOnly) {
                let message = '';
                const isUnchanged = venue.isUnchanged();
                if (!highlightOnly) {
                    message = `Last updated before ${
                        Flag.OldHours.#cutoffDateString}. Verify hours are correct.`;
                    if (isUnchanged) message += ' If everything is current, nudge this place and save.';
                }
                super(
                    true,
                    isUnchanged ? _SEVERITY.YELLOW : _SEVERITY.GREEN,
                    message,
                    isUnchanged ? 'Nudge' : null
                );
                this.venue = venue;
            }

            static #initializeCategoriesToCheck(catData) {
                const catParentIdx = catData[0].split('|').indexOf('pc_catparent');
                const catNameIdx = catData[0].split('|').indexOf('pc_wmecat');
                const parentCats = ['SHOPPING_AND_SERVICES', 'FOOD_AND_DRINK', 'CULTURE_AND_ENTERTAINEMENT'];
                if (!this.#categoriesToCheck) {
                    this.#categoriesToCheck = catData
                        .map(catRowString => catRowString.split('|'))
                        .filter(catRow => parentCats.includes(catRow[catParentIdx]))
                        .map(catRow => catRow[catNameIdx]);
                    this.#categoriesToCheck.push(...parentCats);
                }
            }

            static #venueIsOld(venue) {
                const lastUpdated = venue.attributes.updatedOn ?? venue.attributes.createdOn;
                return lastUpdated < this.#cutoffDate;
            }

            static #venueIsFlaggable(venue) {
                return !venue.isResidential()
                    && this.#venueIsOld(venue)
                    && venue.attributes.openingHours?.length
                    && venue.attributes.categories.some(cat => this.#categoriesToCheck.includes(cat));
            }

            static eval(venue, catData, highlightOnly) {
                let result = null;
                this.#initializeCategoriesToCheck(catData);
                if (this.#venueIsFlaggable(venue)) {
                    result = new this(venue, highlightOnly);
                }
                return result;
            }

            action() {
                nudgeVenue(this.venue);
                harmonizePlaceGo(this.venue, 'harmonize');
            }
        },
        PlaLotTypeMissing: class extends FlagBase {
            constructor(venue, highlightOnly) {
                super(true, _SEVERITY.RED, 'Lot type: ');
                if (!highlightOnly) {
                    this.noLock = true;
                    this.venue = venue;
                    this.message += [['PUBLIC', 'Public'], ['RESTRICTED', 'Restricted'], ['PRIVATE', 'Private']].map(
                        btnInfo => $('<button>', { class: 'wmeph-pla-lot-type-btn btn btn-default btn-xs wmeph-btn', 'data-lot-type': btnInfo[0] })
                            .text(btnInfo[1])
                            .prop('outerHTML')
                    ).join('');
                }
            }

            static eval(venue, highlightOnly) {
                let result = null;
                if (venue.isParkingLot()) {
                    const catAttr = venue.attributes.categoryAttributes;
                    const parkAttr = catAttr ? catAttr.PARKING_LOT : undefined;
                    if (!parkAttr || !parkAttr.parkingType) {
                        result = new this(venue, highlightOnly);
                    }
                }
                return result;
            }

            postProcess() {
                $('.wmeph-pla-lot-type-btn').click(evt => {
                    const lotType = $(evt.currentTarget).data('lot-type');
                    const categoryAttrClone = JSON.parse(JSON.stringify(this.venue.attributes.categoryAttributes));
                    categoryAttrClone.PARKING_LOT = categoryAttrClone.PARKING_LOT ?? {};
                    categoryAttrClone.PARKING_LOT.parkingType = lotType;
                    _UPDATED_FIELDS.lotType.updated = true;
                    addUpdateAction(this.venue, { categoryAttributes: categoryAttrClone }, null, true);
                });
            }
        },
        PlaCostTypeMissing: class extends FlagBase {
            constructor(venue, highlightOnly) {
                super(true, _SEVERITY.BLUE, 'Parking cost: ');
                if (!highlightOnly) {
                    [['FREE', 'Free', 'Free'], ['LOW', '$', 'Low'], ['MODERATE', '$$', 'Moderate'], ['EXPENSIVE', '$$$', 'Expensive']].forEach(btnInfo => {
                        this.message += $('<button>', { id: `wmeph_${btnInfo[0]}`, class: 'wmeph-pla-cost-type-btn btn btn-default btn-xs wmeph-btn', title: btnInfo[2] })
                            .text(btnInfo[1])
                            .css({
                                padding: '3px',
                                height: '20px',
                                lineHeight: '0px',
                                marginRight: '2px',
                                marginBottom: '1px',
                                minWidth: '18px'
                            })
                            .prop('outerHTML');
                    });
                    this.venue = venue;
                    this.noLock = true;
                }
            }

            static #venueIsFlaggable(venue) {
                const parkingAttr = venue.attributes.categoryAttributes?.PARKING_LOT;
                return venue.isParkingLot()
                    && (!parkingAttr?.costType || parkingAttr.costType === 'UNKNOWN');
            }

            static eval(venue, highlightOnly) {
                return this.#venueIsFlaggable(venue) ? new this(venue, highlightOnly) : null;
            }

            postProcess() {
                $('.wmeph-pla-cost-type-btn').click(evt => {
                    const selectedValue = $(evt.currentTarget).attr('id').replace('wmeph_', '');
                    let attrClone;
                    if (this.venue.attributes.categoryAttributes) {
                        attrClone = JSON.parse(JSON.stringify(this.venue.attributes.categoryAttributes));
                    } else {
                        attrClone = {};
                    }
                    attrClone.PARKING_LOT ??= {};
                    attrClone.PARKING_LOT.costType = selectedValue;
                    addUpdateAction(this.venue, { categoryAttributes: attrClone }, null, true);
                    _UPDATED_FIELDS.cost.updated = true;
                });
            }
        },
        PlaPaymentTypeMissing: class extends ActionFlag {
            constructor() { super(true, _SEVERITY.BLUE, 'Parking isn\'t free. Select payment type(s) from the "More info" tab. ', 'Go there'); }

            static eval(venue) {
                let result = null;
                if (venue.isParkingLot()) {
                    const catAttr = venue.attributes.categoryAttributes;
                    const parkAttr = catAttr ? catAttr.PARKING_LOT : undefined;
                    if (parkAttr && parkAttr.costType && parkAttr.costType !== 'FREE' && parkAttr.costType !== 'UNKNOWN' && (!parkAttr.paymentType || !parkAttr.paymentType.length)) {
                        result = new this();
                    }
                }
                return result;
            }

            // eslint-disable-next-line class-methods-use-this
            action() {
                document.querySelector('#edit-panel wz-tab.venue-edit-tab-more-info').isActive = true;
                // The setTimeout is necessary to allow the previous action to do its thing. A pause isn't needed, just a new thread.
                setTimeout(() => document.querySelector('#venue-edit-more-info wz-select[name="costType"]').scrollIntoView(), 0);
            }
        },
        PlaLotElevationMissing: class extends ActionFlag {
            constructor(venue) {
                super(
                    true,
                    _SEVERITY.BLUE,
                    'No lot elevation. Is it street level?',
                    'Yes',
                    'Click if street level parking only, or select other option(s) in the More Info tab.'
                );
                this.venue = venue;
                this.noLock = true;
            }

            static eval(venue) {
                let result = null;
                if (venue.isParkingLot()) {
                    const catAttr = venue.attributes.categoryAttributes;
                    const parkAttr = catAttr ? catAttr.PARKING_LOT : undefined;
                    if (!parkAttr || !parkAttr.lotType || parkAttr.lotType.length === 0) {
                        result = new this(venue);
                    }
                }
                return result;
            }

            action() {
                const existingAttr = this.venue.attributes.categoryAttributes.PARKING_LOT;
                const newAttr = {};
                if (existingAttr) {
                    Object.keys(existingAttr).forEach(key => {
                        let value = existingAttr[key];
                        if (Array.isArray(value)) value = [].concat(value);
                        newAttr[key] = value;
                    });
                }
                newAttr.lotType = ['STREET_LEVEL'];
                addUpdateAction(this.venue, { categoryAttributes: { PARKING_LOT: newAttr } }, null, true);
            }
        },
        PlaSpaces: class extends FlagBase {
            constructor() {
                super(true, _SEVERITY.GREEN, '# of parking spaces is set to 1-10.<br><b><i>If appropriate</i></b>, select another option:');
                const $btnDiv = $('<div>');
                let btnIdx = 0;
                [
                    ['R_11_TO_30', '11-30'], ['R_31_TO_60', '31-60'], ['R_61_TO_100', '61-100'],
                    ['R_101_TO_300', '101-300'], ['R_301_TO_600', '301-600'], ['R_600_PLUS', '601+']
                ].forEach(btnInfo => {
                    if (btnIdx === 3) $btnDiv.append('<br>');
                    $btnDiv.append(
                        $('<button>', { id: `wmeph_${btnInfo[0]}`, class: 'wmeph-pla-spaces-btn btn btn-default btn-xs wmeph-btn' })
                            .text(btnInfo[1])
                            .css({
                                padding: '3px',
                                height: '20px',
                                lineHeight: '0px',
                                marginTop: '2px',
                                marginRight: '2px',
                                marginBottom: '1px',
                                width: '64px'
                            })
                    );
                    btnIdx++;
                });
                this.suffixMessage = $btnDiv.prop('outerHTML');
            }

            static eval(venue, highlightOnly) {
                let result = null;
                if (!highlightOnly && venue.isParkingLot()) {
                    const catAttr = venue.attributes.categoryAttributes;
                    const parkAttr = catAttr ? catAttr.PARKING_LOT : undefined;
                    if (!parkAttr || !parkAttr.estimatedNumberOfSpots || parkAttr.estimatedNumberOfSpots === 'R_1_TO_10') {
                        result = new this();
                    }
                }
                return result;
            }
        },
        NoPlaStopPoint: class extends ActionFlag {
            constructor(venue) {
                super(true, _SEVERITY.BLUE, 'Entry/exit point has not been created.', 'Add point', 'Add an entry/exit point');
                this.venue = venue;
            }

            static eval(venue) {
                let result = null;
                if (venue.isParkingLot() && (!venue.attributes.entryExitPoints || !venue.attributes.entryExitPoints.length)) {
                    result = new this(venue);
                }
                return result;
            }

            action() {
                $('wz-button.navigation-point-add-new').click();
                harmonizePlaceGo(this.venue, 'harmonize');
            }
        },
        PlaStopPointUnmoved: class extends FlagBase {
            constructor() { super(true, _SEVERITY.BLUE, 'Entry/exit point has not been moved.'); }

            static eval(venue) {
                let result = null;
                const attr = venue.attributes;
                if (venue.isParkingLot() && attr.entryExitPoints && attr.entryExitPoints.length) {
                    const stopPoint = attr.entryExitPoints[0].getPoint();
                    const areaCenter = attr.geometry.getCentroid();
                    if (stopPoint.equals(areaCenter)) {
                        result = new this();
                    }
                }
                return result;
            }
        },
        PlaCanExitWhileClosed: class extends ActionFlag {
            constructor(venue) {
                super(true, _SEVERITY.GREEN, 'Can cars exit when lot is closed? ', 'Yes', '');
                this.venue = venue;
            }

            static #venueIsFlaggable(venue, highlightOnly) {
                return !highlightOnly
                    && venue.isParkingLot()
                    && !venue.attributes.categoryAttributes?.PARKING_LOT?.canExitWhileClosed
                    && ($('#WMEPH-ShowPLAExitWhileClosed').prop('checked') || !(venue.attributes.openingHours.length === 0 || isAlwaysOpen(venue)));
            }

            static eval(venue, highlightOnly) {
                return this.#venueIsFlaggable(venue, highlightOnly) ? new this(venue) : null;
            }

            action() {
                const attrClone = JSON.parse(JSON.stringify(this.venue.attributes.categoryAttributes));
                attrClone.PARKING_LOT = attrClone.PARKING_LOT ?? {};
                attrClone.PARKING_LOT.canExitWhileClosed = true;
                addUpdateAction(this.venue, { categoryAttributes: attrClone }, null, true);
            }
        },
        PlaHasAccessibleParking: class extends ActionFlag {
            constructor(venue) {
                super(true, _SEVERITY.GREEN, 'Does this lot have disability parking? ', 'Yes', '');
                this.venue = venue;
            }

            static eval(venue, highlightOnly) {
                let result = null;
                if (!highlightOnly && venue.isParkingLot() && !(venue.attributes.services?.includes('DISABILITY_PARKING'))) {
                    result = new this(venue);
                }
                return result;
            }

            action() {
                let { services } = this.venue.attributes;
                if (services) {
                    services = [].concat(services);
                } else {
                    services = [];
                }
                services.push('DISABILITY_PARKING');
                addUpdateAction(this.venue, { services }, null, true);
                _UPDATED_FIELDS.services_DISABILITY_PARKING.updated = true;
            }
        },
        AllDayHoursFixed: class extends FlagBase {
            constructor() { super(true, _SEVERITY.GREEN, 'Hours were changed from 00:00-23:59 to "All Day"'); }

            static eval(venue, highlightOnly, actions) {
                const hoursEntries = venue.attributes.openingHours;
                const newHoursEntries = [];
                let updateHours = false;
                let result = null;
                for (let i = 0, len = hoursEntries.length; i < len; i++) {
                    const newHoursEntry = new OpeningHour({
                        days: [].concat(hoursEntries[i].days), fromHour: hoursEntries[i].fromHour, toHour: hoursEntries[i].toHour
                    });
                    if (newHoursEntry.toHour === '23:59' && /^0?0:00$/.test(newHoursEntry.fromHour)) {
                        if (highlightOnly) {
                            // Just return a "placeholder" flag to highlight the place.
                            result = new FlagBase(true, _SEVERITY.YELLOW, 'invalid all day hours');
                            break;
                        } else {
                            updateHours = true;
                            newHoursEntry.toHour = '00:00';
                            newHoursEntry.fromHour = '00:00';
                        }
                    }
                    newHoursEntries.push(newHoursEntry);
                }
                if (updateHours) {
                    addUpdateAction(venue, { openingHours: newHoursEntries }, actions);
                    result = new this();
                }
                return result;
            }
        },
        ResiTypeNameSoft: class extends FlagBase {
            constructor() { super(true, _SEVERITY.GREEN, 'The place name suggests a residential place or personalized place of work. Please verify.'); }
        },
        LocalURL: class extends FlagBase {
            constructor() {
                super(
                    true,
                    _SEVERITY.GREEN,
                    'Some locations for this business have localized URLs, while others use the primary corporate site.'
                      + ' Check if a local URL applies to this location.'
                );
            }

            static #venueIsFlaggable(url, localUrlRegexString) {
                return localUrlRegexString && !(new RegExp(localUrlRegexString, 'i')).test(url);
            }

            static eval(url, localUrlRegexString) {
                return this.#venueIsFlaggable(url, localUrlRegexString) ? new this() : null;
            }
        },
        LockRPP: class extends ActionFlag {
            #defaultLockLevel;

            constructor(venue, defaultLockLevel) {
                let message = 'Lock at <select id="RPPLockLevel">';
                let ddlSelected = false;
                for (let llix = 1; llix < 6; llix++) {
                    if (llix < _USER.rank + 1) {
                        if (!ddlSelected && (defaultLockLevel === llix - 1 || llix === _USER.rank)) {
                            message += `<option value="${llix}" selected="selected">${llix}</option>`;
                            ddlSelected = true;
                        } else {
                            message += `<option value="${llix}">${llix}</option>`;
                        }
                    }
                }
                message += '</select>';
                message = `Current lock: ${parseInt(venue.attributes.lockRank, 10) + 1}. ${message} ?`;
                super(true, _SEVERITY.GREEN, message, 'Lock', 'Lock the residential point');
                this.venue = venue;
                this.defaultLockLevel = defaultLockLevel;
            }

            static #venueIsFlaggable(venue, highlightOnly) {
                // Allow residential point locking by R3+
                return !highlightOnly && venue.isResidential() && (_USER.isDevUser || _USER.isBetaUser || _USER.rank >= 3);
            }

            static eval(venue, highlightOnly, defaultLockLevel) {
                let result = null;
                if (this.#venueIsFlaggable(venue, highlightOnly)) {
                    result = new this(venue, defaultLockLevel);
                }
                return result;
            }

            action() {
                let levelToLock = $('#RPPLockLevel :selected').val() || this.#defaultLockLevel + 1;
                logDev(`RPPlevelToLock: ${levelToLock}`);

                levelToLock -= 1;
                if (this.venue.attributes.lockRank !== levelToLock) {
                    addUpdateAction(this.venue, { lockRank: levelToLock }, null, true);
                    _layer.redraw();
                }
            }
        },
        AddAlias: class extends ActionFlag {
            constructor(venue, specCases, optionalAlias) {
                super(true, _SEVERITY.GREEN, `Is there a ${optionalAlias} at this location?`, 'Yes', `Add ${optionalAlias}`);
                this.venue = venue;
                this.specCases = specCases;
                this.optionalAlias = optionalAlias;
            }

            static eval(venue, specCase, specCases, aliases) {
                let result = null;
                const match = specCase.match(/^optionAltName<>(.+)/i);
                if (match) {
                    const [, optionalAlias] = match;
                    if (!aliases.includes(optionalAlias)) {
                        result = new this(venue, specCases, optionalAlias);
                    }
                }
                return result;
            }

            action() {
                let aliases = insertAtIndex(this.venue.attributes.aliases.slice(), this.optionalAlias, 0);
                if (this.specCases.includes('altName2Desc') && !this.venue.attributes.description.toUpperCase().includes(this.optionalAlias.toUpperCase())) {
                    const description = `${this.optionalAlias}\n${this.venue.attributes.description}`;
                    addUpdateAction(this.venue, { description }, null, false);
                }
                aliases = removeSFAliases(name, aliases);
                addUpdateAction(this.venue, { aliases }, null, true);
            }
        },
        AddCat2: class extends ActionFlag {
            constructor(venue, altCategory) {
                super(true, _SEVERITY.GREEN, `Is there a ${_catTransWaze2Lang[altCategory]} at this location?`, 'Yes', `Add ${_catTransWaze2Lang[altCategory]}`);
                this.altCategory = altCategory;
                this.venue = venue;
            }

            static eval(venue, specCases, newCategories, altCategory) {
                let result = null;
                if (specCases.includes('buttOn_addCat2') && !newCategories.includes(altCategory)) {
                    result = new this(venue, altCategory);
                }
                return result;
            }

            action() {
                const categories = insertAtIndex(this.venue.getCategories(), this.altCategory, 1);
                addUpdateAction(this.venue, { categories }, null, true);
            }
        },
        AddPharm: class extends ActionFlag {
            constructor(venue) {
                super(true, _SEVERITY.GREEN, 'Is there a Pharmacy at this location?', 'Yes', 'Add Pharmacy category');
                this.venue = venue;
            }

            action() {
                const categories = insertAtIndex(this.venue.getCategories(), 'PHARMACY', 1);
                addUpdateAction(this.venue, { categories }, null, true);
            }
        },
        AddSuper: class extends ActionFlag {
            constructor(venue) {
                super(true, _SEVERITY.GREEN, 'Does this location have a supermarket?', 'Yes', 'Add Supermarket category');
                this.venue = venue;
            }

            action() {
                const categories = insertAtIndex(this.venue.getCategories(), 'SUPERMARKET_GROCERY', 1);
                addUpdateAction(this.venue, { categories }, null, true);
            }
        },
        AppendAMPM: class extends ActionFlag {
            constructor(venue) {
                super(true, _SEVERITY.GREEN, 'Is there an ampm at this location?', 'Yes', 'Add ampm to the place');
                this.venue = venue;
            }

            action() {
                const categories = insertAtIndex(this.venue.getCategories(), 'CONVENIENCE_STORE', 1);
                addUpdateAction(this.venue, { name: 'ARCO ampm', url: 'ampm.com', categories }, null, true);
            }
        },
        AddATM: class extends ActionFlag {
            constructor(venue) {
                super(true, _SEVERITY.GREEN, 'ATM at location? ', 'Yes', 'Add the ATM category to this place');
                this.venue = venue;
            }

            action() {
                const categories = insertAtIndex(this.venue.getCategories(), 'ATM', 1); // Insert ATM category in the second position
                addUpdateAction(this.venue, { categories }, null, true);
            }
        },
        AddConvStore: class extends ActionFlag {
            constructor(venue) {
                super(true, _SEVERITY.GREEN, 'Add convenience store category? ', 'Yes', 'Add the Convenience Store category to this place');
                this.venue = venue;
            }

            static #venueIsFlaggable(venue, categories) {
                return venue.isGasStation()
                    && !categories.includes('CONVENIENCE_STORE')
                    && !_buttonBanner.subFuel; // Don't flag if already asking if this is really a gas station
            }

            static eval(venue, categories) {
                return this.#venueIsFlaggable(venue, categories) ? new Flag.AddConvStore(venue) : null;
            }

            action() {
                // Insert C.S. category in the second position
                const categories = insertAtIndex(this.venue.getCategories(), 'CONVENIENCE_STORE', 1);
                addUpdateAction(this.venue, { categories }, null, true);
            }
        },
        IsThisAPostOffice: class extends ActionFlag {
            constructor(venue) {
                super(
                    true,
                    _SEVERITY.GREEN,
                    `Is this a <a href="${_URLS.uspsWiki}" target="_blank" style="color:#3a3a3a">USPS post office</a>? `,
                    'Yes',
                    'Is this a USPS location?'
                );
                this.venue = venue;
            }

            static eval(venue, highlightOnly, countryCode, newCategories, newName) {
                let result = null;
                if (!highlightOnly && countryCode === 'USA' && !newCategories.includes('PARKING_LOT') && !newCategories.includes('POST_OFFICE')) {
                    const cleanName = newName.toUpperCase().replace(/[/\-.]/g, '');
                    if (/\bUSP[OS]\b|\bpost(al)?\s+(service|office)\b/i.test(cleanName)) {
                        result = new this(venue);
                    }
                }
                return result;
            }

            action() {
                const categories = insertAtIndex(this.venue.getCategories(), 'POST_OFFICE', 0);
                addUpdateAction(this.venue, { categories }, null, true);
            }
        },
        ChangeToHospitalUrgentCare: class extends WLActionFlag {
            constructor(venue) {
                super(
                    true,
                    _SEVERITY.GREEN,
                    'If this place provides emergency medical care:',
                    'Change to Hospital / Urgent Care',
                    'Change category to Hospital / Urgent Care',
                    false,
                    'Whitelist category',
                    'changetoHospitalUrgentCare'
                );
                this.venue = venue;
            }

            static eval(venue, highlightOnly) {
                let result = null;
                if (!highlightOnly && venue.attributes.categories.includes('DOCTOR_CLINIC')) {
                    result = new this(venue);
                }
                return result;
            }

            action() {
                let categories = this.venue.getCategories();
                if (!categories.includes('HOSPITAL_MEDICAL_CARE')) {
                    const indexToReplace = categories.indexOf('DOCTOR_CLINIC');
                    if (indexToReplace > -1) {
                        categories = categories.slice(); // create a copy
                        categories[indexToReplace] = 'HOSPITAL_URGENT_CARE';
                    }
                    addUpdateAction(this.venue, { categories });
                }
                harmonizePlaceGo(this.venue, 'harmonize');
            }
        },
        NotAHospital: class extends WLActionFlag {
            constructor(venue) {
                super(
                    true,
                    _SEVERITY.RED,
                    'Key words suggest this location may not be a hospital or urgent care location.',
                    'Change to Doctor / Clinic',
                    'Change category to Doctor / Clinic',
                    true,
                    'Whitelist category',
                    'notAHospital'
                );
                this.venue = venue;
                this.noLock = true;
            }

            static #venueIsFlaggable(categories, name, wl) {
                if (categories.includes('HOSPITAL_URGENT_CARE') && !wl.notAHospital) {
                    const testName = name.toLowerCase().replace(/[^a-z]/g, ' ');
                    const testNameWords = testName.split(' ');
                    return containsAny(testNameWords, _hospitalFullMatch) || _hospitalPartMatch.some(match => testName.includes(match));
                }
                return false;
            }

            static eval(venue, categories, name, wl) {
                return this.#venueIsFlaggable(categories, name, wl) ? new this(venue) : null;
            }

            action() {
                let categories = this.venue.getCategories().slice();
                let updateIt = false;
                if (categories.length) {
                    const idx = categories.indexOf('HOSPITAL_URGENT_CARE');
                    if (idx > -1) {
                        categories[idx] = 'DOCTOR_CLINIC';
                        updateIt = true;
                    }
                    categories = _.uniq(categories);
                } else {
                    categories.push('DOCTOR_CLINIC');
                    updateIt = true;
                }
                if (updateIt) {
                    addUpdateAction(this.venue, { categories });
                }
                harmonizePlaceGo(this.venue, 'harmonize');
            }
        },
        ChangeToDoctorClinic: class extends WLActionFlag {
            constructor(venue) {
                super(
                    true,
                    _SEVERITY.GREEN,
                    'If this place provides non-emergency medical care: ',
                    'Change to Doctor / Clinic',
                    'Change category to Doctor / Clinic',
                    false,
                    'Whitelist category',
                    'changeToDoctorClinic'
                );
                this.venue = venue;
            }

            static eval(venue, newCategories, highlightOnly, pnhNameRegMatch) {
                let result = null;
                if (!highlightOnly && venue.attributes.updatedOn < new Date('3/28/2017').getTime()
                    && ((newCategories.includes('PERSONAL_CARE') && !pnhNameRegMatch) || newCategories.includes('OFFICES'))) {
                    // Show the Change To Doctor / Clinic button for places with PERSONAL_CARE or OFFICES category
                    // The date criteria was added because Doctor/Clinic category was added around then, and it's assumed if the
                    // place has been edited since then, people would have already updated the category.

                    result = new this(venue);
                    result.WLactive = null;
                }
                return result;
            }

            action() {
                let categories = this.venue.getCategories().slice();
                let updateIt = false;
                if (categories.length) {
                    ['OFFICES', 'PERSONAL_CARE'].forEach(cat => {
                        const idx = categories.indexOf(cat);
                        if (idx > -1) {
                            categories[idx] = 'DOCTOR_CLINIC';
                            updateIt = true;
                        }
                    });
                    categories = _.uniq(categories);
                } else {
                    categories.push('DOCTOR_CLINIC');
                    updateIt = true;
                }
                if (updateIt) {
                    addUpdateAction(this.venue, { categories });
                }
                harmonizePlaceGo(this.venue, 'harmonize');
            }
        },
        TitleCaseName: class extends ActionFlag {
            #confirmChange = false;
            #originalName;

            constructor(venue, name, nameSuffix) {
                const titleCaseName = toTitleCaseStrong(name);
                super(true, _SEVERITY.GREEN, '', 'Force Title Case?', `Force title case to: ${titleCaseName}`);
                this.#originalName = name + (nameSuffix || '');
                this.suffixMessage = `<span style="margin-left: 4px;font-size: 14px">&bull; ${titleCaseName}${nameSuffix || ''}</span>`;
                this.noBannerAssemble = true;
                this.venue = venue;
            }

            static #venueIsFlaggable(name) {
                return name !== toTitleCaseStrong(name);
            }

            static eval(venue, name, nameSuffix) {
                return this.#venueIsFlaggable(name) ? new this(venue, name, nameSuffix) : null;
            }

            action() {
                let name = this.venue.getName();
                if (name === this.#originalName || this.#confirmChange) {
                    const parts = getNameParts(this.#originalName);
                    name = toTitleCaseStrong(parts.base);
                    if (parts.base !== name) {
                        addUpdateAction(this.venue, { name: name + (parts.suffix || '') });
                    }
                    harmonizePlaceGo(this.venue, 'harmonize');
                } else {
                    $('button#WMEPH_titleCaseName').text('Are you sure?').after(' The name has changed. This will overwrite the new name.');
                    this.#confirmChange = true;
                }
            }
        },
        SFAliases: class extends FlagBase {
            constructor() { super(true, _SEVERITY.GREEN, 'Unnecessary aliases were removed.'); }
        },
        PlaceMatched: class extends FlagBase {
            constructor() { super(true, _SEVERITY.GREEN, 'Place matched from PNH data.'); }
        },
        PlaceLocked: class extends FlagBase {
            constructor(venue, levelToLock, highlightOnly, actions) {
                super(true, _SEVERITY.GREEN, 'Place locked.');

                if (venue.attributes.lockRank < levelToLock) {
                    if (!highlightOnly) {
                        logDev('Venue locked!');
                        actions.push(new UpdateObject(venue, { lockRank: levelToLock }));
                        _UPDATED_FIELDS.lockRank.updated = true;
                    } else {
                        this.hlLockFlag = true;
                    }
                }
            }

            static eval(venue, lockOK, totalSeverity, levelToLock, highlightOnly, actions) {
                let result = null;
                if (lockOK && totalSeverity < _SEVERITY.YELLOW) {
                    result = new this(venue, levelToLock, highlightOnly, actions);
                }
                return result;
            }
        },
        NewPlaceSubmit: class extends ActionFlag {
            #newPlaceUrl;

            constructor(newPlaceUrl) {
                super(true, _SEVERITY.GREEN, 'No PNH match. If it\'s a chain: ', 'Submit new chain data', 'Submit info for a new chain through the linked form');
                this.#newPlaceUrl = newPlaceUrl;
            }

            action() {
                window.open(this.#newPlaceUrl);
            }
        },
        ApprovalSubmit: class extends ActionFlag {
            #approveRegionURL;

            constructor(approveRegionURL) {
                super(true, _SEVERITY.GREEN, 'PNH data exists but is not approved for this region: ', 'Request approval', 'Request region/country approval of this place');
                this.#approveRegionURL = approveRegionURL;
            }

            action() {
                window.open(this.#approveRegionURL);
            }
        },
        PlaceWebsite: class extends ActionFlag {
            // NOTE: This class is now only used to display the store locator button.
            // It can be updated to remove/change anything that doesn't serve that purpose.
            constructor(venue) {
                super(true, _SEVERITY.GREEN, '', 'Location Finder', 'Look up details about this location on the chain\'s finder web page');
                this.venue = venue;
            }

            // TODO: Currently not fully implemented for all instances. If possible, combine all PlaceWebsite checks into this eval,
            // and remove the previousFlag argument.
            static eval(venue, highlightOnly, countryCode, newCategories, previousFlag) {
                let result = previousFlag;
                // Check for USPS
                if (!highlightOnly && countryCode === 'USA' && !newCategories.includes('PARKING_LOT') && newCategories.includes('POST_OFFICE')) {
                    _customStoreFinderURL = _URLS.uspsLocationFinder;
                    _customStoreFinder = true;
                    result = new this(venue);
                }
                return result;
            }

            action() {
                let openPlaceWebsiteURL;
                // let linkProceed = true;
                if (_updateURL) {
                    // replace WME url with storefinder URLs if they are in the PNH data
                    if (_customStoreFinder) {
                        openPlaceWebsiteURL = _customStoreFinderURL;
                    } else if (_customStoreFinderLocal) {
                        openPlaceWebsiteURL = _customStoreFinderLocalURL;
                    }
                    // If the user has 'never' opened a localized store finder URL, then warn them (just once)
                    if (localStorage.getItem(_SETTING_IDS.sfUrlWarning) === '0' && _customStoreFinderLocal) {
                        WazeWrap.Alerts.confirm(
                            _SCRIPT_NAME,
                            '***Localized store finder sites often show multiple nearby results. Please make sure you pick the right location.<br>Click OK to agree and continue.',
                            () => {
                                localStorage.setItem(_SETTING_IDS.sfUrlWarning, '1'); // prevent future warnings
                                if ($('#WMEPH-WebSearchNewTab').prop('checked')) {
                                    window.open(openPlaceWebsiteURL);
                                } else {
                                    window.open(openPlaceWebsiteURL, _SEARCH_RESULTS_WINDOW_NAME, _searchResultsWindowSpecs);
                                }
                            },
                            () => { }
                        );
                        return;
                    }
                } else {
                    let { url } = this.venue;
                    if (!/^https?:\/\//.test(url)) url = `http://${url}`;
                    openPlaceWebsiteURL = url;
                }
                // open the link depending on new window setting
                // if (linkProceed) {
                if ($('#WMEPH-WebSearchNewTab').prop('checked')) {
                    window.open(openPlaceWebsiteURL);
                } else {
                    window.open(openPlaceWebsiteURL, _SEARCH_RESULTS_WINDOW_NAME, _searchResultsWindowSpecs);
                }
                // }
            }
        }
    }; // END Flag namespace

    function getButtonBanner() {
        // **** Set up banner action buttons.  Structure:
        // active: false until activated in the script
        // severity: determines the color of the banners and whether locking occurs
        // message: The text before the button option
        // value: button text
        // title: tooltip text
        // action: The action that happens if the button is pressed
        // WL terms are for whitelisting

        return {
            evChargingStationWarning: null,
            pnhCatMess: null,
            notAHospital: null,
            notASchool: null,
            hnDashRemoved: null,
            fullAddressInference: null,
            nameMissing: null,
            gasNameMissing: null,
            plaIsPublic: null,
            plaNameMissing: null,
            plaNameNonStandard: null,
            indianaLiquorStoreHours: null,
            hoursOverlap: null,
            unmappedRegion: null,
            restAreaName: null,
            restAreaNoTransportation: null,
            restAreaGas: null,
            restAreaScenic: null,
            restAreaSpec: null,
            gasMismatch: null,
            gasUnbranded: null,
            gasMkPrim: null,
            isThisAPilotTravelCenter: null,
            hotelMkPrim: null,
            changeToPetVet: null,
            pointNotArea: null,
            areaNotPoint: null,
            hnMissing: null,
            hnTooManyDigits: null,
            hnNonStandard: null,
            HNRange: null,
            streetMissing: null,
            cityMissing: null,
            bankType1: null,
            bankBranch: null,
            standaloneATM: null,
            bankCorporate: null,
            catPostOffice: null,
            ignEdited: null,
            wazeBot: null,
            parentCategory: null,
            checkDescription: null,
            overlapping: null,
            suspectDesc: null,
            resiTypeName: null,
            mismatch247: null,
            phoneInvalid: null,
            areaNotPointMid: null,
            pointNotAreaMid: null,
            longURL: null,
            gasNoBrand: null,
            subFuel: null,
            areaNotPointLow: null,
            pointNotAreaLow: null,
            formatUSPS: null,
            missingUSPSAlt: null,
            missingUSPSZipAlt: null,
            missingUSPSDescription: null,
            catHotel: null,
            localizedName: null,
            specCaseMessage: null,
            specCaseMessageLow: null,
            changeToDoctorClinic: null,
            extProviderMissing: null,
            addCommonEVPaymentMethods: null,
            removeUncommonEVPaymentMethods: null,
            urlMissing: null,
            addRecommendedPhone: null,
            badAreaCode: null,
            phoneMissing: null,
            oldHours: null,
            noHours: null,
            evcsPriceMissing: null,
            plaLotTypeMissing: null,
            plaCostTypeMissing: null,
            plaPaymentTypeMissing: null,
            plaLotElevationMissing: null,
            plaSpaces: null,
            noPlaStopPoint: null,
            plaStopPointUnmoved: null,
            plaCanExitWhileClosed: null,
            plaHasAccessibleParking: null,
            allDayHoursFixed: null,
            resiTypeNameSoft: null,
            localURL: null,
            lockRPP: null,
            addAlias: null,
            addCat2: null,
            addPharm: null,
            addSuper: null,
            appendAMPM: null,
            addATM: null,
            addConvStore: null,
            isThisAPostOffice: null,
            titleCaseName: null,
            changeToHospitalUrgentCare: null,
            sfAliases: null,
            placeMatched: null,
            placeLocked: null,
            NewPlaceSubmit: null,
            ApprovalSubmit: null,
            PlaceWebsite: null
        };
    }

    function getServicesBanner() {
        // set up banner action buttons.  Structure:
        // active: false until activated in the script
        // checked: whether the service is already set on the place. Determines grey vs white icon color
        // icon: button icon name
        // value: button text  (Not used for Icons, keep as backup
        // title: tooltip text
        // action: The action that happens if the button is pressed
        return {
            addValet: {
                active: false,
                checked: false,
                icon: 'serv-valet',
                w2hratio: 50 / 50,
                value: 'Valet',
                title: 'Valet service',
                servIDIndex: 0,
                action(actions, checked) {
                    setServiceChecked(this, checked, actions);
                },
                pnhOverride: false,
                actionOn(actions) {
                    this.action(actions, true);
                },
                actionOff(actions) {
                    this.action(actions, false);
                }
            },
            addDriveThru: {
                active: false,
                checked: false,
                icon: 'serv-drivethru',
                w2hratio: 78 / 50,
                value: 'DriveThru',
                title: 'Drive-thru',
                servIDIndex: 1,
                action(actions, checked) {
                    setServiceChecked(this, checked, actions);
                },
                pnhOverride: false,
                actionOn(actions) {
                    this.action(actions, true);
                },
                actionOff(actions) {
                    this.action(actions, false);
                }
            },
            addWiFi: {
                active: false,
                checked: false,
                icon: 'serv-wifi',
                w2hratio: 67 / 50,
                value: 'WiFi',
                title: 'Wi-Fi',
                servIDIndex: 2,
                action(actions, checked) {
                    setServiceChecked(this, checked, actions);
                },
                pnhOverride: false,
                actionOn(actions) {
                    this.action(actions, true);
                },
                actionOff(actions) {
                    this.action(actions, false);
                }
            },
            addRestrooms: {
                active: false,
                checked: false,
                icon: 'serv-restrooms',
                w2hratio: 49 / 50,
                value: 'Restroom',
                title: 'Restrooms',
                servIDIndex: 3,
                action(actions, checked) {
                    setServiceChecked(this, checked, actions);
                },
                pnhOverride: false,
                actionOn(actions) {
                    this.action(actions, true);
                },
                actionOff(actions) {
                    this.action(actions, false);
                }
            },
            addCreditCards: {
                active: false,
                checked: false,
                icon: 'serv-credit',
                w2hratio: 73 / 50,
                value: 'CC',
                title: 'Accepts credit cards',
                servIDIndex: 4,
                action(actions, checked) {
                    setServiceChecked(this, checked, actions);
                },
                pnhOverride: false,
                actionOn(actions) {
                    this.action(actions, true);
                },
                actionOff(actions) {
                    this.action(actions, false);
                }
            },
            addReservations: {
                active: false,
                checked: false,
                icon: 'serv-reservations',
                w2hratio: 55 / 50,
                value: 'Reserve',
                title: 'Reservations',
                servIDIndex: 5,
                action(actions, checked) {
                    setServiceChecked(this, checked, actions);
                },
                pnhOverride: false,
                actionOn(actions) {
                    this.action(actions, true);
                },
                actionOff(actions) {
                    this.action(actions, false);
                }
            },
            addOutside: {
                active: false,
                checked: false,
                icon: 'serv-outdoor',
                w2hratio: 73 / 50,
                value: 'OusideSeat',
                title: 'Outdoor seating',
                servIDIndex: 6,
                action(actions, checked) {
                    setServiceChecked(this, checked, actions);
                },
                pnhOverride: false,
                actionOn(actions) {
                    this.action(actions, true);
                },
                actionOff(actions) {
                    this.action(actions, false);
                }
            },
            addAC: {
                active: false,
                checked: false,
                icon: 'serv-ac',
                w2hratio: 50 / 50,
                value: 'AC',
                title: 'Air conditioning',
                servIDIndex: 7,
                action(actions, checked) {
                    setServiceChecked(this, checked, actions);
                },
                pnhOverride: false,
                actionOn(actions) {
                    this.action(actions, true);
                },
                actionOff(actions) {
                    this.action(actions, false);
                }
            },
            addParking: {
                active: false,
                checked: false,
                icon: 'serv-parking',
                w2hratio: 46 / 50,
                value: 'Customer parking',
                title: 'Parking',
                servIDIndex: 8,
                action(actions, checked) {
                    setServiceChecked(this, checked, actions);
                },
                pnhOverride: false,
                actionOn(actions) {
                    this.action(actions, true);
                },
                actionOff(actions) {
                    this.action(actions, false);
                }
            },
            addDeliveries: {
                active: false,
                checked: false,
                icon: 'serv-deliveries',
                w2hratio: 86 / 50,
                value: 'Delivery',
                title: 'Deliveries',
                servIDIndex: 9,
                action(actions, checked) {
                    setServiceChecked(this, checked, actions);
                },
                pnhOverride: false,
                actionOn(actions) {
                    this.action(actions, true);
                },
                actionOff(actions) {
                    this.action(actions, false);
                }
            },
            addTakeAway: {
                active: false,
                checked: false,
                icon: 'serv-takeaway',
                w2hratio: 34 / 50,
                value: 'Take-out',
                title: 'Take-out',
                servIDIndex: 10,
                action(actions, checked) {
                    setServiceChecked(this, checked, actions);
                },
                pnhOverride: false,
                actionOn(actions) {
                    this.action(actions, true);
                },
                actionOff(actions) {
                    this.action(actions, false);
                }
            },
            addCurbside: {
                active: true,
                checked: false,
                icon: 'serv-curbside',
                w2hratio: 50 / 50,
                value: 'Curbside pickup',
                title: 'Curbside pickup',
                servIDIndex: 11,
                action(actions, checked) {
                    setServiceChecked(this, checked, actions);
                },
                pnhOverride: false,
                actionOn(actions) {
                    this.action(actions, true);
                },
                actionOff(actions) {
                    this.action(actions, false);
                }
            },
            addWheelchair: {
                active: false,
                checked: false,
                icon: 'serv-wheelchair',
                w2hratio: 50 / 50,
                value: 'WhCh',
                title: 'Wheelchair accessible',
                servIDIndex: 12,
                action(actions, checked) {
                    setServiceChecked(this, checked, actions);
                },
                pnhOverride: false,
                actionOn(actions) {
                    this.action(actions, true);
                },
                actionOff(actions) {
                    this.action(actions, false);
                }
            },
            addDisabilityParking: {
                active: false,
                checked: false,
                icon: 'serv-wheelchair',
                w2hratio: 50 / 50,
                value: 'DisabilityParking',
                title: 'Disability parking',
                servIDIndex: 13,
                action(actions, checked) {
                    setServiceChecked(this, checked, actions);
                },
                pnhOverride: false,
                actionOn(actions) {
                    this.action(actions, true);
                },
                actionOff(actions) {
                    this.action(actions, false);
                }
            },
            add247: {
                active: false,
                checked: false,
                icon: 'serv-247',
                w2hratio: 73 / 50,
                value: '247',
                title: 'Hours: Open 24/7',
                action(actions) {
                    if (!_servicesBanner.add247.checked) {
                        const venue = getSelectedVenue();
                        _servicesBanner.add247.checked = true;
                        addUpdateAction(venue, { openingHours: [new OpeningHour({ days: [1, 2, 3, 4, 5, 6, 0], fromHour: '00:00', toHour: '00:00' })] }, actions);
                        // _buttonBanner.noHours = null;
                        // TODO: figure out how to keep the noHours flag without causing an infinite loop when
                        // called from psOn_add247 speccase. Don't call harmonizePlaceGo here.
                    }
                },
                actionOn(actions) {
                    this.action(actions);
                }
            }
        };
    } // END getServicesBanner()

    function getButtonBanner2(venue, placePL) {
        return {
            placesWiki: {
                active: true,
                severity: 0,
                message: '',
                value: 'Places wiki',
                title: 'Open the places Wazeopedia (wiki) page',
                action() {
                    window.open(_URLS.placesWiki);
                }
            },
            restAreaWiki: {
                active: false,
                severity: 0,
                message: '',
                value: 'Rest Area wiki',
                title: 'Open the Rest Area wiki page',
                action() {
                    window.open(_URLS.restAreaWiki);
                }
            },
            clearWL: {
                active: false,
                severity: 0,
                message: '',
                value: 'Clear place whitelist',
                title: 'Clear all Whitelisted fields for this place',
                action() {
                    WazeWrap.Alerts.confirm(
                        _SCRIPT_NAME,
                        'Are you sure you want to clear all whitelisted fields for this place?',
                        () => {
                            delete _venueWhitelist[venue.attributes.id];
                            saveWhitelistToLS(true);
                            harmonizePlaceGo(venue, 'harmonize');
                        },
                        () => { },
                        'Yes',
                        'No'
                    );
                }
            },
            PlaceErrorForumPost: {
                active: true,
                severity: 0,
                message: '',
                value: 'Report script error',
                title: 'Report a script error',
                action() {
                    reportError({
                        subject: 'WMEPH Bug report: Script Error',
                        message: `Script version: ${_SCRIPT_VERSION}${_BETA_VERSION_STR}\nPermalink: ${
                            placePL}\nPlace name: ${venue.attributes.name}\nCountry: ${
                            venue.getAddress().getCountry().name}\n--------\nDescribe the error:  \n `
                    });
                }
            }
        };
    } // END getButtonBanner2()

    // Main script
    function harmonizePlaceGo(item, useFlag, actions) {
        let placePL;
        const itemID = item.attributes.id;

        // Used for collecting all actions to be applied to the model.
        actions = actions || [];

        const highlightOnly = !useFlag.includes('harmonize');

        let totalSeverity = _SEVERITY.GREEN;

        // Whitelist: reset flags
        const wl = {};

        let addr = item.getAddress();
        if (addr.hasOwnProperty('attributes')) {
            addr = addr.attributes;
        }

        _buttonBanner = getButtonBanner();
        let pnhLockLevel;
        if (!highlightOnly) {
            // Uncomment this to test all field highlights.
            // _UPDATED_FIELDS.getFieldProperties().forEach(prop => {
            //     prop.updated = true;
            // });

            // The placePL should only be needed when harmonizing, not when highlighting.
            placePL = getCurrentPL() //  set up external post div and pull place PL
                .replace(/&layers=[^&]+(&?)/g, '$1') // remove Permalink Layers
                .replace(/&s=[^&]+(&?)/g, '$1') // remove Permalink Layers
                .replace(/&update_requestsFilter=[^&]+(&?)/g, '$1') // remove Permalink Layers
                .replace(/&problemsFilter=[^&]+(&?)/g, '$1') // remove Permalink Layers
                .replace(/&mapProblemFilter=[^&]+(&?)/g, '$1') // remove Permalink Layers
                .replace(/&mapUpdateRequestFilter=[^&]+(&?)/g, '$1') // remove Permalink Layers
                .replace(/&venueFilter=[^&]+(&?)/g, '$1'); // remove Permalink Layers

            _buttonBanner2 = getButtonBanner2(item, placePL);
            _servicesBanner = getServicesBanner();

            // Update icons to reflect current WME place services
            updateServicesChecks(_servicesBanner);

            // Setting switch for the Places Wiki button
            if ($('#WMEPH-HidePlacesWiki').prop('checked')) {
                _buttonBanner2.placesWiki.active = false;
            }

            if ($('#WMEPH-HideReportError').prop('checked')) {
                _buttonBanner2.PlaceErrorForumPost.active = false;
            }

            // reset PNH lock level
            pnhLockLevel = -1;
        }

        // If place has hours of 0:00-23:59, highlight yellow or if harmonizing, convert to All Day.
        _buttonBanner.allDayHoursFixed = Flag.AllDayHoursFixed.eval(item, highlightOnly, actions);

        let lockOK = true; // if nothing goes wrong, then place will be locked

        let newCategories = item.attributes.categories.slice();
        const nameParts = getNameParts(item.attributes.name);
        let newNameSuffix = nameParts.suffix;
        let newName = nameParts.base;
        let newAliases = item.attributes.aliases.slice();
        let newDescripion = item.attributes.description;
        let newUrl = item.attributes.url;
        let newPhone = item.attributes.phone;

        let pnhNameRegMatch;
        let result;

        // Some user submitted places have no data in the country, state and address fields.
        result = Flag.FullAddressInference.eval(item, addr, actions, highlightOnly, newCategories);
        if (result?.exit) return result.severity;
        _buttonBanner.fullAddressInference = result;
        const inferredAddress = result?.inferredAddress;
        addr = inferredAddress ?? addr;

        // Check parking lot attributes.
        if (!highlightOnly && item.isParkingLot()) _servicesBanner.addDisabilityParking.active = true;

        _buttonBanner.plaCostTypeMissing = Flag.PlaCostTypeMissing.eval(item, highlightOnly);
        _buttonBanner.plaLotElevationMissing = Flag.PlaLotElevationMissing.eval(item);
        _buttonBanner.plaSpaces = Flag.PlaSpaces.eval(item, highlightOnly);
        _buttonBanner.plaLotTypeMissing = Flag.PlaLotTypeMissing.eval(item, highlightOnly);
        _buttonBanner.noPlaStopPoint = Flag.NoPlaStopPoint.eval(item);
        _buttonBanner.plaStopPointUnmoved = Flag.PlaStopPointUnmoved.eval(item);
        _buttonBanner.plaCanExitWhileClosed = Flag.PlaCanExitWhileClosed.eval(item, highlightOnly);
        _buttonBanner.plaPaymentTypeMissing = Flag.PlaPaymentTypeMissing.eval(item);
        _buttonBanner.plaHasAccessibleParking = Flag.PlaHasAccessibleParking.eval(item, highlightOnly);

        // Check categories that maybe should be Hospital / Urgent Care, or Doctor / Clinic.
        _buttonBanner.changeToHospitalUrgentCare = Flag.ChangeToHospitalUrgentCare.eval(item, highlightOnly);

        // Whitelist breakout if place exists on the Whitelist and the option is enabled

        let itemGPS;
        if (_venueWhitelist.hasOwnProperty(itemID) && (!highlightOnly || (highlightOnly && !$('#WMEPH-DisableWLHL').prop('checked')))) {
            // Enable the clear WL button if any property is true
            Object.keys(_venueWhitelist[itemID]).forEach(wlKey => { // loop thru the venue WL keys
                if (_venueWhitelist[itemID].hasOwnProperty(wlKey) && (_venueWhitelist[itemID][wlKey].active || false)) {
                    if (!highlightOnly) _buttonBanner2.clearWL.active = true;
                    wl[wlKey] = _venueWhitelist[itemID][wlKey];
                }
            });
            if (_venueWhitelist[itemID].hasOwnProperty('dupeWL') && _venueWhitelist[itemID].dupeWL.length > 0) {
                if (!highlightOnly) _buttonBanner2.clearWL.active = true;
                wl.dupeWL = _venueWhitelist[itemID].dupeWL;
            }
            // Update address and GPS info for the place
            if (!highlightOnly) {
                // get GPS lat/long coords from place, call as itemGPS.lat, itemGPS.lon
                if (!itemGPS) {
                    const centroid = item.attributes.geometry.getCentroid();
                    itemGPS = OpenLayers.Layer.SphericalMercator.inverseMercator(centroid.x, centroid.y);
                }
                _venueWhitelist[itemID].city = addr.city.attributes.name; // Store city for the venue
                _venueWhitelist[itemID].state = addr.state.name; // Store state for the venue
                _venueWhitelist[itemID].country = addr.country.name; // Store country for the venue
                _venueWhitelist[itemID].gps = itemGPS; // Store GPS coords for the venue
            }
        }

        // Country restrictions (note that FullAddressInference should guarantee country/state exist if highlightOnly is true)
        if (!addr.country || !addr.state) {
            WazeWrap.Alerts.error(_SCRIPT_NAME, 'Country and/or state could not be determined.  Edit the place address and run WMEPH again.');
            return undefined;
        }
        let countryCode;
        const countryName = addr.country.name;
        const stateName = addr.state.name;
        if (['United States', 'American Samoa', 'Guam', 'Northern Mariana Islands', 'Puerto Rico', 'Virgin Islands (U.S.)'].includes(countryName)) {
            countryCode = 'USA';
        } else if (countryName === 'Canada') {
            countryCode = 'CAN';
        } else {
            if (!highlightOnly) {
                WazeWrap.Alerts.error(_SCRIPT_NAME, `This script is not currently supported in ${countryName}.`);
            }
            return _SEVERITY.RED;
        }

        // Parse state-based data
        let state2L = 'Unknown';
        let region = 'Unknown';
        let gFormState = '';
        let defaultLockLevel = _LOCK_LEVEL_2;
        for (let usdix = 1; usdix < _PNH_DATA.states.length; usdix++) {
            _stateDataTemp = _PNH_DATA.states[usdix].split('|');
            if (stateName === _stateDataTemp[_psStateIx]) {
                state2L = _stateDataTemp[_psState2LetterIx];
                region = _stateDataTemp[_psRegionIx];
                gFormState = _stateDataTemp[_psGoogleFormStateIx];
                if (_stateDataTemp[_psDefaultLockLevelIx].match(/[1-5]{1}/) !== null) {
                    defaultLockLevel = _stateDataTemp[_psDefaultLockLevelIx] - 1; // normalize by -1
                } else if (!highlightOnly) {
                    WazeWrap.Alerts.warning(_SCRIPT_NAME, 'Lock level sheet data is not correct');
                } else {
                    return 3;
                }
                _areaCodeList = `${_areaCodeList},${_stateDataTemp[_psAreaCodeIx]}`;
                break;
            }
            // If State is not found, then use the country
            if (countryName === _stateDataTemp[_psStateIx]) {
                state2L = _stateDataTemp[_psState2LetterIx];
                region = _stateDataTemp[_psRegionIx];
                gFormState = _stateDataTemp[_psGoogleFormStateIx];
                if (_stateDataTemp[_psDefaultLockLevelIx].match(/[1-5]{1}/) !== null) {
                    defaultLockLevel = _stateDataTemp[_psDefaultLockLevelIx] - 1; // normalize by -1
                } else if (!highlightOnly) {
                    WazeWrap.Alerts.warning(_SCRIPT_NAME, 'Lock level sheet data is not correct');
                } else {
                    return 3;
                }
                _areaCodeList = `${_areaCodeList},${_stateDataTemp[_psAreaCodeIx]}`;
                break;
            }
        }
        if (state2L === 'Unknown' || region === 'Unknown') { // if nothing found:
            if (!highlightOnly) {
                /* if (confirm('WMEPH: Localization Error!\nClick OK to report this error')) {
                    // if the category doesn't translate, then pop an alert that will make a forum post to the thread
                    const data = {
                        subject: 'WMEPH Localization Error report',
                        message: `Error report: Localization match failed for "${stateName}".`
                    };
                    if (_PNH_DATA.states.length === 0) {
                        data.message += ' _PNH_DATA.states array is empty.';
                    } else {
                        data.message += ` state2L = ${_stateDataTemp[_psState2LetterIx]}. region = ${_stateDataTemp[_psRegionIx]}`;
                    }
                    reportError(data);
                } */
                WazeWrap.Alerts.confirm(
                    _SCRIPT_NAME,
                    'WMEPH: Localization Error!<br>Click OK to report this error',
                    () => {
                        const data = {
                            subject: 'WMEPH Localization Error report',
                            message: `Error report: Localization match failed for "${stateName}".`
                        };
                        if (_PNH_DATA.states.length === 0) {
                            data.message += ' _PNH_DATA.states array is empty.';
                        } else {
                            data.message += ` state2L = ${_stateDataTemp[_psState2LetterIx]}. region = ${_stateDataTemp[_psRegionIx]}`;
                        }
                        reportError(data);
                    },
                    () => { }
                );
            }
            return 3;
        }

        // Gas station treatment (applies to all including PNH)

        _buttonBanner.isThisAPilotTravelCenter = Flag.IsThisAPilotTravelCenter.eval(item, highlightOnly, state2L, newName, actions);
        if (_buttonBanner.isThisAPilotTravelCenter) newName = _buttonBanner.isThisAPilotTravelCenter.newName;

        _buttonBanner.gasMkPrim = Flag.GasMkPrim.eval(item, newCategories);
        _buttonBanner.addConvStore = Flag.AddConvStore.eval(item, newCategories);

        // Note for Indiana editors to check liquor store hours if Sunday hours haven't been added yet.
        _buttonBanner.indianaLiquorStoreHours = Flag.IndianaLiquorStoreHours.eval(item, newName, highlightOnly, wl);

        const isLocked = item.attributes.lockRank >= (pnhLockLevel > -1 ? pnhLockLevel : defaultLockLevel);

        // Set up a variable (newBrand) to contain the brand. When harmonizing, it may be forced to a new value.
        // Other brand flags should use it since it won't be updated on the actual venue until later.
        let { brand: newBrand } = item.attributes;
        // Clear attributes from residential places
        if (item.attributes.residential) {
            if (!highlightOnly) {
                if (!$('#WMEPH-AutoLockRPPs').prop('checked')) {
                    lockOK = false;
                }
                if (item.attributes.name !== '') { // Set the residential place name to the address (to clear any personal info)
                    logDev('Residential Name reset');
                    actions.push(new UpdateObject(item, { name: '' }));
                    // no field HL
                }
                newCategories = ['RESIDENCE_HOME'];
                // newDescripion = null;
                if (item.attributes.description !== null && item.attributes.description !== '') { // remove any description
                    logDev('Residential description cleared');
                    actions.push(new UpdateObject(item, { description: null }));
                    // no field HL
                }
                // newPhone = null;
                if (item.attributes.phone !== null && item.attributes.phone !== '') { // remove any phone info
                    logDev('Residential Phone cleared');
                    actions.push(new UpdateObject(item, { phone: null }));
                    // no field HL
                }
                // newURL = null;
                if (item.attributes.url !== null && item.attributes.url !== '') { // remove any url
                    logDev('Residential URL cleared');
                    actions.push(new UpdateObject(item, { url: null }));
                    // no field HL
                }
                if (item.attributes.services.length > 0) {
                    logDev('Residential services cleared');
                    actions.push(new UpdateObject(item, { services: [] }));
                    // no field HL
                }
            }
            // NOTE: do not use is2D() function. It doesn't seem to be 100% reliable.
            if (!item.isPoint()) {
                _buttonBanner.pointNotArea = new Flag.PointNotArea(item);
            }
        } else if (item.isParkingLot() || (newName && newName.trim().length)) { // for non-residential places
            // Phone formatting
            let outputPhoneFormat = '({0}) {1}-{2}';
            if (containsAny(['CA', 'CO'], [region, state2L]) && (/^\d{3}-\d{3}-\d{4}$/.test(item.attributes.phone))) {
                outputPhoneFormat = '{0}-{1}-{2}';
            } else if (region === 'SER' && !(/^\(\d{3}\) \d{3}-\d{4}$/.test(item.attributes.phone))) {
                outputPhoneFormat = '{0}-{1}-{2}';
            } else if (region === 'GLR') {
                outputPhoneFormat = '{0}-{1}-{2}';
            } else if (state2L === 'NV') {
                outputPhoneFormat = '{0}-{1}-{2}';
            } else if (countryCode === 'CAN') {
                outputPhoneFormat = '+1-{0}-{1}-{2}';
            }

            _buttonBanner.extProviderMissing = Flag.ExtProviderMissing.eval(
                item,
                isLocked,
                newCategories,
                _USER.rank,
                $('#WMEPH-DisablePLAExtProviderCheck').prop('checked'),
                actions
            );

            // Place Harmonization
            let pnhMatchData;
            if (!highlightOnly) {
                if (item.isParkingLot()) {
                    pnhMatchData = ['NoMatch'];
                } else {
                    // check against the PNH list
                    pnhMatchData = harmoList(newName, state2L, region, countryCode, newCategories, item);

                    if (['NoMatch', 'ApprovalNeeded'].includes(pnhMatchData[0])) {
                        const newURLSubmit = !isNullOrWhitespace(newUrl) ? newUrl : '';
                        let pnhOrderNum = '';
                        let pnhNameTemp = '';
                        let pnhNameTempWeb = '';
                        if (pnhMatchData[0] === 'ApprovalNeeded') {
                            // PNHNameTemp = PNHMatchData[1].join(', ');
                            [, [pnhNameTemp]] = pnhMatchData; // Just do the first match
                            pnhNameTempWeb = encodeURIComponent(pnhNameTemp);
                            pnhOrderNum = pnhMatchData[2].join(',');
                        }

                        // Make PNH submission links
                        let regionFormURL = '';
                        let newPlaceAddon = '';
                        let approvalAddon = '';
                        const approvalMessage = `Submitted via WMEPH. PNH order number ${pnhOrderNum}`;
                        const encodedTempSubmitName = encodeURIComponent(newName);
                        const encodedPlacePL = encodeURIComponent(placePL);
                        const encodedUrlSubmit = encodeURIComponent(newURLSubmit);
                        const suffix = _USER.name + gFormState;
                        switch (region) {
                            case 'NWR': regionFormURL = 'https://docs.google.com/forms/d/1hv5hXBlGr1pTMmo4n3frUx1DovUODbZodfDBwwTc7HE/viewform';
                                newPlaceAddon = `?entry.925969794=${encodedTempSubmitName}&entry.1970139752=${encodedUrlSubmit}&entry.1749047694=${suffix}`;
                                approvalAddon = `?entry.925969794=${pnhNameTempWeb}&entry.50214576=${approvalMessage}&entry.1749047694=${suffix}`;
                                break;
                            case 'SWR': regionFormURL = 'https://docs.google.com/forms/d/1Qf2N4fSkNzhVuXJwPBJMQBmW0suNuy8W9itCo1qgJL4/viewform';
                                newPlaceAddon = `?entry.1497446659=${encodedTempSubmitName}&entry.1970139752=${encodedUrlSubmit}&entry.1749047694=${suffix}`;
                                approvalAddon = `?entry.1497446659=${pnhNameTempWeb}&entry.50214576=${approvalMessage}&entry.1749047694=${suffix}`;
                                break;
                            case 'HI': regionFormURL = 'https://docs.google.com/forms/d/1K7Dohm8eamIKry3KwMTVnpMdJLaMIyDGMt7Bw6iqH_A/viewform';
                                newPlaceAddon = `?entry.1497446659=${encodedTempSubmitName}&entry.1970139752=${encodedUrlSubmit}&entry.1749047694=${suffix}`;
                                approvalAddon = `?entry.1497446659=${pnhNameTempWeb}&entry.50214576=${approvalMessage}&entry.1749047694=${suffix}`;
                                break;
                            case 'PLN': regionFormURL = 'https://docs.google.com/forms/d/1ycXtAppoR5eEydFBwnghhu1hkHq26uabjUu8yAlIQuI/viewform';
                                newPlaceAddon = `?entry.925969794=${encodedTempSubmitName}&entry.1970139752=${encodedUrlSubmit}&entry.1749047694=${suffix}`;
                                approvalAddon = `?entry.925969794=${pnhNameTempWeb}&entry.50214576=${approvalMessage}&entry.1749047694=${suffix}`;
                                break;
                            case 'SCR': regionFormURL = 'https://docs.google.com/forms/d/1KZzLdlX0HLxED5Bv0wFB-rWccxUp2Mclih5QJIQFKSQ/viewform';
                                newPlaceAddon = `?entry.925969794=${encodedTempSubmitName}&entry.1970139752=${encodedUrlSubmit}&entry.1749047694=${suffix}`;
                                approvalAddon = `?entry.925969794=${pnhNameTempWeb}&entry.50214576=${approvalMessage}&entry.1749047694=${suffix}`;
                                break;
                            case 'GLR': regionFormURL = 'https://docs.google.com/forms/d/19btj-Qt2-_TCRlcS49fl6AeUT95Wnmu7Um53qzjj9BA/viewform';
                                newPlaceAddon = `?entry.925969794=${encodedTempSubmitName}&entry.1970139752=${encodedUrlSubmit}&entry.1749047694=${suffix}`;
                                approvalAddon = `?entry.925969794=${pnhNameTempWeb}&entry.50214576=${approvalMessage}&entry.1749047694=${suffix}`;
                                break;
                            case 'SAT': regionFormURL = 'https://docs.google.com/forms/d/1bxgK_20Jix2ahbmUvY1qcY0-RmzUBT6KbE5kjDEObF8/viewform';
                                newPlaceAddon = `?entry.2063110249=${encodedTempSubmitName}&entry.2018912633=${encodedUrlSubmit}&entry.1924826395=${suffix}`;
                                approvalAddon = `?entry.2063110249=${pnhNameTempWeb}&entry.123778794=${approvalMessage}&entry.1924826395=${suffix}`;
                                break;
                            case 'SER': regionFormURL = 'https://docs.google.com/forms/d/1jYBcxT3jycrkttK5BxhvPXR240KUHnoFMtkZAXzPg34/viewform';
                                newPlaceAddon = `?entry.822075961=${encodedTempSubmitName}&entry.1422079728=${encodedUrlSubmit}&entry.1891389966=${suffix}`;
                                approvalAddon = `?entry.822075961=${pnhNameTempWeb}&entry.607048307=${approvalMessage}&entry.1891389966=${suffix}`;
                                break;
                            case 'ATR': regionFormURL = 'https://docs.google.com/forms/d/1v7JhffTfr62aPSOp8qZHA_5ARkBPldWWJwDeDzEioR0/viewform';
                                newPlaceAddon = `?entry.925969794=${encodedTempSubmitName}&entry.1970139752=${encodedUrlSubmit}&entry.1749047694=${suffix}`;
                                approvalAddon = `?entry.925969794=${pnhNameTempWeb}&entry.50214576=${approvalMessage}&entry.1749047694=${suffix}`;
                                break;
                            case 'NER': regionFormURL = 'https://docs.google.com/forms/d/1UgFAMdSQuJAySHR0D86frvphp81l7qhEdJXZpyBZU6c/viewform';
                                newPlaceAddon = `?entry.925969794=${encodedTempSubmitName}&entry.1970139752=${encodedUrlSubmit}&entry.1749047694=${suffix}`;
                                approvalAddon = `?entry.925969794=${pnhNameTempWeb}&entry.50214576=${approvalMessage}&entry.1749047694=${suffix}`;
                                break;
                            case 'NOR': regionFormURL = 'https://docs.google.com/forms/d/1iYq2rd9HRd-RBsKqmbHDIEBGuyWBSyrIHC6QLESfm4c/viewform';
                                newPlaceAddon = `?entry.925969794=${encodedTempSubmitName}&entry.1970139752=${encodedUrlSubmit}&entry.1749047694=${suffix}`;
                                approvalAddon = `?entry.925969794=${pnhNameTempWeb}&entry.50214576=${approvalMessage}&entry.1749047694=${suffix}`;
                                break;
                            case 'MAR': regionFormURL = 'https://docs.google.com/forms/d/1PhL1iaugbRMc3W-yGdqESoooeOz-TJIbjdLBRScJYOk/viewform';
                                newPlaceAddon = `?entry.925969794=${encodedTempSubmitName}&entry.1970139752=${encodedUrlSubmit}&entry.1749047694=${suffix}`;
                                approvalAddon = `?entry.925969794=${pnhNameTempWeb}&entry.50214576=${approvalMessage}&entry.1749047694=${suffix}`;
                                break;
                            case 'CA_EN': regionFormURL = 'https://docs.google.com/forms/d/13JwXsrWPNmCdfGR5OVr5jnGZw-uNGohwgjim-JYbSws/viewform';
                                newPlaceAddon = `?entry_839085807=${encodedTempSubmitName}&entry_1067461077=${encodedUrlSubmit}&entry_318793106=${
                                    _USER.name}&entry_1149649663=${encodedPlacePL}`;
                                approvalAddon = `?entry_839085807=${pnhNameTempWeb}&entry_1125435193=${approvalMessage}&entry_318793106=${
                                    _USER.name}&entry_1149649663=${encodedPlacePL}`;
                                break;
                            case 'QC': regionFormURL = 'https://docs.google.com/forms/d/13JwXsrWPNmCdfGR5OVr5jnGZw-uNGohwgjim-JYbSws/viewform';
                                newPlaceAddon = `?entry_839085807=${encodedTempSubmitName}&entry_1067461077=${encodedUrlSubmit}&entry_318793106=${
                                    _USER.name}&entry_1149649663=${encodedPlacePL}`;
                                approvalAddon = `?entry_839085807=${pnhNameTempWeb}&entry_1125435193=${approvalMessage}&entry_318793106=${
                                    _USER.name}&entry_1149649663=${encodedPlacePL}`;
                                break;
                            default: regionFormURL = '';
                        }
                        const newPlaceUrl = regionFormURL + newPlaceAddon;
                        const approveRegionURL = regionFormURL + approvalAddon;

                        if (pnhMatchData[0] === 'ApprovalNeeded') {
                            _buttonBanner.ApprovalSubmit = new Flag.ApprovalSubmit(approveRegionURL);
                        } else if (pnhMatchData[0] === 'NoMatch') {
                            _buttonBanner.NewPlaceSubmit = new Flag.NewPlaceSubmit(newPlaceUrl);
                        }
                    }
                }
            } else {
                pnhMatchData = ['Highlight'];
            }

            pnhNameRegMatch = false;
            if (pnhMatchData[0] !== 'NoMatch' && pnhMatchData[0] !== 'ApprovalNeeded' && pnhMatchData[0] !== 'Highlight') { // *** Replace place data with PNH data
                pnhNameRegMatch = true;
                let showDispNote = true;
                let updatePNHName = true;
                // Break out the data headers
                const pnhDataHeaders = _PNH_DATA[countryCode].pnh[0].split('|');
                const phNameIdx = pnhDataHeaders.indexOf('ph_name');
                const phAliasesIdx = pnhDataHeaders.indexOf('ph_aliases');
                const phCategory1Idx = pnhDataHeaders.indexOf('ph_category1');
                const phCategory2Idx = pnhDataHeaders.indexOf('ph_category2');
                const phDescriptionIdx = pnhDataHeaders.indexOf('ph_description');
                const phUrlIdx = pnhDataHeaders.indexOf('ph_url');
                const phOrderIdx = pnhDataHeaders.indexOf('ph_order');
                // var ph_notes_ix = _PNH_DATA_headers.indexOf('ph_notes');
                const phSpecCaseIdx = pnhDataHeaders.indexOf('ph_speccase');
                const phStoreFinderUrlIdx = pnhDataHeaders.indexOf('ph_sfurl');
                const phStoreFinderUrlLocalIdx = pnhDataHeaders.indexOf('ph_sfurllocal');
                // var ph_forcecat_ix = _PNH_DATA_headers.indexOf('ph_forcecat');
                const phDisplayNoteIdx = pnhDataHeaders.indexOf('ph_displaynote');

                // Retrieve the data from the PNH line(s)
                let nsMultiMatch = false;
                const orderList = [];
                if (pnhMatchData.length > 1) { // If multiple matches, then
                    let brandParent = -1;
                    let pnhMatchDataHold = pnhMatchData[0].split('|');
                    for (let pmdix = 0; pmdix < pnhMatchData.length; pmdix++) { // For each of the matches,
                        const pmdTemp = pnhMatchData[pmdix].split('|'); // Split the PNH data line
                        orderList.push(pmdTemp[phOrderIdx]); // Add Order number to a list
                        if (pmdTemp[phSpecCaseIdx].match(/brandParent(\d{1})/) !== null) { // If there is a brandParent flag, prioritize by highest match
                            const [, pmdSpecCases] = pmdTemp[phSpecCaseIdx].match(/brandParent(\d{1})/);
                            if (pmdSpecCases > brandParent) { // if the match is more specific than the previous ones:
                                brandParent = pmdSpecCases; // Update the brandParent level
                                pnhMatchDataHold = pmdTemp; // Update the PNH data line
                            }
                        } else { // if any item has no brandParent structure, use highest brandParent match but post an error
                            nsMultiMatch = true;
                        }
                    }
                    pnhMatchData = pnhMatchDataHold;
                } else {
                    pnhMatchData = pnhMatchData[0].split('|'); // Single match just gets direct split
                }

                const priPNHPlaceCat = catTranslate(pnhMatchData[phCategory1Idx]); // translate primary category to WME code

                // if the location has multiple matches, then pop an alert that will make a forum post to the thread
                if (nsMultiMatch) {
                    /* if (confirm('WMEPH: Multiple matches found!\nDouble check the script changes.\nClick OK to report this situation.')) {
                        reportError({
                            subject: `Order Nos. "${orderList.join(', ')}" WMEPH Multiple match report`,
                            message: `Error report: PNH Order Nos. "${orderList.join(', ')}" are ambiguous multiple matches.\n \nExample Permalink: ${placePL}`
                        });
                    } */
                    WazeWrap.Alerts.confirm(
                        _SCRIPT_NAME,
                        'WMEPH: Multiple matches found!<br>Double check the script changes.<br>Click OK to report this situation.',
                        () => {
                            reportError({
                                subject: `Order Nos. "${orderList.join(', ')}" WMEPH Multiple match report`,
                                message: `Error report: PNH Order Nos. "${orderList.join(', ')}" are ambiguous multiple matches.\n \nExample Permalink: ${placePL}`
                            });
                        },
                        () => { }
                    );
                }

                // Check special cases
                let specCases;
                let localURLcheck = '';
                if (phSpecCaseIdx > -1) { // If the special cases column exists
                    specCases = pnhMatchData[phSpecCaseIdx]; // pulls the speccases field from the PNH line
                    if (!isNullOrWhitespace(specCases)) {
                        specCases = specCases.replace(/, /g, ',').split(','); // remove spaces after commas and split by comma
                    }
                    for (let scix = 0; scix < specCases.length; scix++) {
                        let scFlag;
                        const specCase = specCases[scix];
                        let match;

                        /* eslint-disable no-cond-assign */

                        // find any button/message flags in the special case (format: buttOn_xyzXyz, etc.)
                        if (match = specCase.match(/^buttOn_(.+)/i)) {
                            [, scFlag] = match;
                            let flag = null;
                            switch (scFlag) {
                                case 'addCat2':
                                    // flag = new Flag.AddCat2();
                                    break;
                                case 'addPharm':
                                    flag = new Flag.AddPharm(item);
                                    break;
                                case 'addSuper':
                                    flag = new Flag.AddSuper(item);
                                    break;
                                case 'appendAMPM':
                                    flag = new Flag.AppendAMPM(item);
                                    break;
                                case 'addATM':
                                    flag = new Flag.AddATM(item);
                                    break;
                                case 'addConvStore':
                                    flag = new Flag.AddConvStore(item);
                                    break;
                                default:
                                    console.error('WMEPH:', `Could not process specCase value: buttOn_${scFlag}`);
                            }
                            _buttonBanner[scFlag] = flag;
                        } else if (match = specCase.match(/^buttOff_(.+)/i)) {
                            [, scFlag] = match;
                            _buttonBanner[scFlag] = null;
                        } else if (match = specCase.match(/^messOn_(.+)/i)) {
                            [, scFlag] = match;
                            _buttonBanner[scFlag].active = true;
                        } else if (match = specCase.match(/^messOff_(.+)/i)) {
                            [, scFlag] = match;
                            _buttonBanner[scFlag].active = false;
                        } else if (match = specCase.match(/^psOn_(.+)/i)) {
                            [, scFlag] = match;
                            _servicesBanner[scFlag].actionOn(actions);
                            _servicesBanner[scFlag].pnhOverride = true;
                        } else if (match = specCase.match(/^psOff_(.+)/i)) {
                            [, scFlag] = match;
                            _servicesBanner[scFlag].actionOff(actions);
                            _servicesBanner[scFlag].pnhOverride = true;
                        }

                        // If brand is going to be forced, use that.  Otherwise, use existing brand.
                        if (match = /forceBrand<>([^,<]+)/i.exec(pnhMatchData[phSpecCaseIdx])) {
                            [, newBrand] = match;
                        }

                        // parseout localURL data if exists (meaning place can have a URL distinct from the chain URL
                        if (match = specCase.match(/^localURL_(.+)/i)) {
                            [, localURLcheck] = match;
                        }

                        // parse out optional alt-name
                        _buttonBanner.addAlias = Flag.AddAlias.eval(item, specCase, specCases, newAliases);

                        // Gas Station forceBranding
                        if (['GAS_STATION'].includes(priPNHPlaceCat) && (match = specCase.match(/^forceBrand<>(.+)/i))) {
                            const [, forceBrand] = match;
                            if (item.attributes.brand !== forceBrand) {
                                actions.push(new UpdateObject(item, { brand: forceBrand }));
                                _UPDATED_FIELDS.brand.updated = true;
                                logDev('Gas brand updated from PNH');
                            }
                        }

                        // Check Localization
                        let displayNote;
                        if (phDisplayNoteIdx > -1 && !isNullOrWhitespace(pnhMatchData[phDisplayNoteIdx])) {
                            displayNote = pnhMatchData[phDisplayNoteIdx];
                        }
                        result = Flag.LocalizedName.eval(newName, newNameSuffix, specCase, displayNote, wl);
                        if (!result.showDisplayNote) {
                            showDispNote = false;
                        }
                        if (result.flag) {
                            _buttonBanner.localizedName = result.flag;
                        }

                        /* eslint-enable no-cond-assign */

                        // Prevent name change
                        if (specCase.match(/keepName/g) !== null) {
                            updatePNHName = false;
                        }

                        _buttonBanner.addRecommendedPhone = Flag.AddRecommendedPhone.eval(item, pnhMatchData[phSpecCaseIdx], outputPhoneFormat, wl);
                    }
                }

                // If it's a place that also sells fuel, enable the button
                if (pnhMatchData[phSpecCaseIdx] === 'subFuel' && !newName.toUpperCase().includes('GAS') && !newName.toUpperCase().includes('FUEL')) {
                    _buttonBanner.subFuel = new Flag.SubFuel();
                    if (wl.subFuel) {
                        _buttonBanner.subFuel.WLactive = false;
                    }
                }

                // Display any notes for the specific place
                _buttonBanner.specCaseMessage = Flag.SpecCaseMessage.eval(item, pnhMatchData[phDisplayNoteIdx], showDispNote, specCases);

                // Localized Storefinder code:
                _customStoreFinderLocal = false;
                _customStoreFinderLocalURL = '';
                _customStoreFinder = false;
                _customStoreFinderURL = '';
                if (phStoreFinderUrlIdx > -1) { // if the sfurl column exists...
                    if (phStoreFinderUrlLocalIdx > -1 && !isNullOrWhitespace(pnhMatchData[phStoreFinderUrlLocalIdx])) {
                        // I'm not sure why this check for localizedName was here.
                        // if (!_buttonBanner.localizedName) {
                        _buttonBanner.PlaceWebsite = new Flag.PlaceWebsite();
                        _buttonBanner.PlaceWebsite.value = 'Location Finder (L)';
                        // }
                        const tempLocalURL = pnhMatchData[phStoreFinderUrlLocalIdx].replace(/ /g, '').split('<>');
                        let searchStreet = '';
                        let searchCity = '';
                        let searchState = '';
                        if (typeof addr.street.name === 'string') {
                            searchStreet = addr.street.name;
                        }
                        const searchStreetPlus = searchStreet.replace(/ /g, '+');
                        searchStreet = searchStreet.replace(/ /g, '%20');
                        if (typeof addr.city.attributes.name === 'string') {
                            searchCity = addr.city.attributes.name;
                        }
                        const searchCityPlus = searchCity.replace(/ /g, '+');
                        searchCity = searchCity.replace(/ /g, '%20');
                        if (typeof addr.state.name === 'string') {
                            searchState = addr.state.name;
                        }
                        const searchStatePlus = searchState.replace(/ /g, '+');
                        searchState = searchState.replace(/ /g, '%20');

                        const centroid = item.attributes.geometry.getCentroid();
                        if (!itemGPS) itemGPS = OpenLayers.Layer.SphericalMercator.inverseMercator(centroid.x, centroid.y);
                        for (let tlix = 1; tlix < tempLocalURL.length; tlix++) {
                            if (tempLocalURL[tlix] === 'ph_streetName') {
                                _customStoreFinderLocalURL += searchStreet;
                            } else if (tempLocalURL[tlix] === 'ph_streetNamePlus') {
                                _customStoreFinderLocalURL += searchStreetPlus;
                            } else if (tempLocalURL[tlix] === 'ph_cityName') {
                                _customStoreFinderLocalURL += searchCity;
                            } else if (tempLocalURL[tlix] === 'ph_cityNamePlus') {
                                _customStoreFinderLocalURL += searchCityPlus;
                            } else if (tempLocalURL[tlix] === 'ph_stateName') {
                                _customStoreFinderLocalURL += searchState;
                            } else if (tempLocalURL[tlix] === 'ph_stateNamePlus') {
                                _customStoreFinderLocalURL += searchStatePlus;
                            } else if (tempLocalURL[tlix] === 'ph_state2L') {
                                _customStoreFinderLocalURL += state2L;
                            } else if (tempLocalURL[tlix] === 'ph_latitudeEW') {
                                // customStoreFinderLocalURL = customStoreFinderLocalURL + itemGPS[0];
                            } else if (tempLocalURL[tlix] === 'ph_longitudeNS') {
                                // customStoreFinderLocalURL = customStoreFinderLocalURL + itemGPS[1];
                            } else if (tempLocalURL[tlix] === 'ph_latitudePM') {
                                _customStoreFinderLocalURL += itemGPS.lat;
                            } else if (tempLocalURL[tlix] === 'ph_longitudePM') {
                                _customStoreFinderLocalURL += itemGPS.lon;
                            } else if (tempLocalURL[tlix] === 'ph_latitudePMBuffMin') {
                                _customStoreFinderLocalURL += (itemGPS.lat - 0.025).toString();
                            } else if (tempLocalURL[tlix] === 'ph_longitudePMBuffMin') {
                                _customStoreFinderLocalURL += (itemGPS.lon - 0.025).toString();
                            } else if (tempLocalURL[tlix] === 'ph_latitudePMBuffMax') {
                                _customStoreFinderLocalURL += (itemGPS.lat + 0.025).toString();
                            } else if (tempLocalURL[tlix] === 'ph_longitudePMBuffMax') {
                                _customStoreFinderLocalURL += (itemGPS.lon + 0.025).toString();
                            } else if (tempLocalURL[tlix] === 'ph_houseNumber') {
                                _customStoreFinderLocalURL += (item.attributes.houseNumber ? item.attributes.houseNumber : '');
                            } else {
                                _customStoreFinderLocalURL += tempLocalURL[tlix];
                            }
                        }
                        if (_customStoreFinderLocalURL.indexOf('http') !== 0) {
                            _customStoreFinderLocalURL = `http://${_customStoreFinderLocalURL}`;
                        }
                        _customStoreFinderLocal = true;
                    } else if (!isNullOrWhitespace(pnhMatchData[phStoreFinderUrlIdx])) {
                        // I'm not sure why this check for localizedName was here.
                        // if (!_buttonBanner.localizedName) {
                        _buttonBanner.PlaceWebsite = new Flag.PlaceWebsite();
                        // }
                        _customStoreFinderURL = pnhMatchData[phStoreFinderUrlIdx];
                        if (_customStoreFinderURL.indexOf('http') !== 0) {
                            _customStoreFinderURL = `http://${_customStoreFinderURL}`;
                        }
                        _customStoreFinder = true;
                    }
                }

                // Category translations
                let altCategories = pnhMatchData[phCategory2Idx];
                if (altCategories && altCategories.length) { //  translate alt-cats to WME code
                    altCategories = altCategories.replace(/,[^A-Za-z0-9]*/g, ',').split(','); // tighten and split by comma
                    for (let catix = 0; catix < altCategories.length; catix++) {
                        const newAltTemp = catTranslate(altCategories[catix]); // translate altCats into WME cat codes
                        if (newAltTemp === 'ERROR') { // if no translation, quit the loop
                            log(`Category ${altCategories[catix]} cannot be translated.`);
                            return undefined;
                        }
                        altCategories[catix] = newAltTemp; // replace with translated element
                    }
                }

                // name parsing with category exceptions
                if (['HOTEL'].includes(priPNHPlaceCat)) {
                    const nameToCheck = newName + (newNameSuffix || '');
                    if (nameToCheck.toUpperCase() === pnhMatchData[phNameIdx].toUpperCase()) { // If no localization
                        _buttonBanner.catHotel = new Flag.CatHotel(pnhMatchData[phNameIdx]);
                        newName = pnhMatchData[phNameIdx];
                    } else {
                        // Replace PNH part of name with PNH name
                        const splix = newName.toUpperCase().replace(/[-/]/g, ' ').indexOf(pnhMatchData[phNameIdx].toUpperCase().replace(/[-/]/g, ' '));
                        if (splix > -1) {
                            const frontText = newName.slice(0, splix);
                            const backText = newName.slice(splix + pnhMatchData[phNameIdx].length);
                            newName = pnhMatchData[phNameIdx];
                            if (frontText.length > 0) { newName = `${frontText} ${newName}`; }
                            if (backText.length > 0) { newName = `${newName} ${backText}`; }
                            newName = newName.replace(/ {2,}/g, ' ');
                        } else {
                            newName = pnhMatchData[phNameIdx];
                        }
                    }
                    if (altCategories && altCategories.length) { // if PNH alts exist
                        insertAtIndex(newCategories, altCategories, 1); //  then insert the alts into the existing category array after the GS category
                    }
                    if (newCategories.indexOf('HOTEL') !== 0) { // If no HOTEL category in the primary, flag it
                        _buttonBanner.hotelMkPrim = new Flag.HotelMkPrim(item);
                        if (wl.hotelMkPrim) {
                            _buttonBanner.hotelMkPrim.WLactive = false;
                        } else {
                            lockOK = false;
                        }
                    } else if (newCategories.includes('HOTEL')) {
                        // Remove LODGING if it exists
                        const lodgingIdx = newCategories.indexOf('LODGING');
                        if (lodgingIdx > -1) {
                            newCategories.splice(lodgingIdx, 1);
                        }
                    }
                    // If PNH match, set wifi service.
                    if (pnhMatchData && !_servicesBanner.addWiFi.checked) {
                        _servicesBanner.addWiFi.action();
                    }
                    // Set hotel hours to 24/7 for all hotels.
                    if (!_servicesBanner.add247.checked) {
                        _servicesBanner.add247.action();
                    }
                } else if (newCategories.includes('BANK_FINANCIAL') && !pnhMatchData[phSpecCaseIdx].includes('notABank')) {
                    // PNH Bank treatment
                    _ixBank = item.attributes.categories.indexOf('BANK_FINANCIAL');
                    _ixATM = item.attributes.categories.indexOf('ATM');
                    _ixOffices = item.attributes.categories.indexOf('OFFICES');
                    // if the name contains ATM in it
                    if (/\batm\b/ig.test(newName)) {
                        if (_ixOffices === 0) {
                            _buttonBanner.bankType1 = new Flag.BankType1();
                            _buttonBanner.bankBranch = new Flag.BankBranch(item);
                            _buttonBanner.standaloneATM = new Flag.StandaloneATM(item);
                            _buttonBanner.bankCorporate = new Flag.BankCorporate(item);
                        } else if (_ixBank === -1 && _ixATM === -1) {
                            _buttonBanner.bankBranch = new Flag.BankBranch(item);
                            _buttonBanner.standaloneATM = new Flag.StandaloneATM(item);
                        } else if (_ixATM === 0 && _ixBank > 0) {
                            _buttonBanner.bankBranch = new Flag.BankBranch(item);
                        } else if (_ixBank > -1) {
                            _buttonBanner.bankBranch = new Flag.BankBranch(item);
                            _buttonBanner.standaloneATM = new Flag.StandaloneATM(item);
                        }
                        newName = `${pnhMatchData[phNameIdx]} ATM`;
                        newCategories = insertAtIndex(newCategories, 'ATM', 0);
                        // Net result: If the place has ATM cat only and ATM in the name, then it will be green and renamed Bank Name ATM
                    } else if (_ixBank > -1 || _ixATM > -1) { // if no ATM in name but with a banking category:
                        if (_ixOffices === 0) {
                            _buttonBanner.bankBranch = new Flag.BankBranch(item);
                        } else if (_ixBank > -1 && _ixATM === -1) {
                            _buttonBanner.addATM = new Flag.AddATM(item);
                        } else if (_ixATM === 0 && _ixBank === -1) {
                            _buttonBanner.bankBranch = new Flag.BankBranch(item);
                            _buttonBanner.standaloneATM = new Flag.StandaloneATM(item);
                        } else if (_ixBank > 0 && _ixATM > 0) {
                            _buttonBanner.bankBranch = new Flag.BankBranch(item);
                            _buttonBanner.standaloneATM = new Flag.StandaloneATM(item);
                        }
                        newName = pnhMatchData[phNameIdx];
                        // Net result: If the place has Bank category first, then it will be green with PNH name replaced
                    } else { // for PNH match with neither bank type category, make it a bank
                        newCategories = insertAtIndex(newCategories, 'BANK_FINANCIAL', 1);
                        _buttonBanner.standaloneATM = new Flag.StandaloneATM(item);
                        _buttonBanner.bankCorporate = new Flag.BankCorporate(item);
                    }// END PNH bank treatment
                } else if (['GAS_STATION'].includes(priPNHPlaceCat)) { // for PNH gas stations, don't replace existing sub-categories
                    if (altCategories?.length) { // if PNH alts exist
                        insertAtIndex(newCategories, altCategories, 1); //  then insert the alts into the existing category array after the GS category
                    }
                    newName = pnhMatchData[phNameIdx];
                } else if (updatePNHName) { // if not a special category then update the name
                    newName = pnhMatchData[phNameIdx];
                    newCategories = insertAtIndex(newCategories, priPNHPlaceCat, 0);
                    if (altCategories && altCategories.length && !specCases.includes('buttOn_addCat2') && !specCases.includes('optionCat2')) {
                        newCategories = insertAtIndex(newCategories, altCategories, 1);
                    }
                } else if (!updatePNHName) {
                    // Strong title case option for non-PNH places
                    _buttonBanner.titleCaseName = Flag.TitleCaseName.eval(item, newName, newNameSuffix);
                }

                // *** need to add a section above to allow other permissible categories to remain? (optional)

                // Parse URL data
                if (!(localURLcheck && newUrl && (new RegExp(localURLcheck, 'i')).test(newUrl))) {
                    newUrl = pnhMatchData[phUrlIdx];
                }
                newUrl = normalizeURL(newUrl, false, true, item, region, wl);

                _buttonBanner.localURL = Flag.LocalURL.eval(newUrl, localURLcheck);

                // Parse PNH Aliases
                let [newAliasesTemp] = pnhMatchData[phAliasesIdx].match(/([^(]*)/i);
                if (!isNullOrWhitespace(newAliasesTemp)) { // make aliases array
                    newAliasesTemp = newAliasesTemp.replace(/,[^A-za-z0-9]*/g, ','); // tighten up commas if more than one alias.
                    newAliasesTemp = newAliasesTemp.split(','); // split by comma
                }
                if (!specCases.includes('noUpdateAlias') && (!containsAll(newAliases, newAliasesTemp)
                    && newAliasesTemp && newAliasesTemp.length && !specCases.includes('optionName2'))) {
                    newAliases = insertAtIndex(newAliases, newAliasesTemp, 0);
                }

                // Remove unnecessary parent categories
                const catData = _PNH_DATA.USA.categories.map(cat => cat.split('|'));
                const catParentIdx = catData[0].indexOf('pc_catparent');
                const catNameIdx = catData[0].indexOf('pc_wmecat');
                const parentCats = _.uniq(newCategories.map(catName => catData.find(cat => cat[catNameIdx] === catName)[catParentIdx]))
                    .filter(parent => parent.trim(' ').length > 0);
                newCategories = newCategories.filter(cat => !parentCats.includes(cat));

                // update categories if different and no Cat2 option
                if (!matchSets(_.uniq(item.attributes.categories), _.uniq(newCategories))) {
                    if (!specCases.includes('optionCat2') && !specCases.includes('buttOn_addCat2')) {
                        logDev(`Categories updated with ${newCategories}`);
                        addUpdateAction(item, { categories: newCategories }, actions);
                    } else { // if second cat is optional
                        logDev(`Primary category updated with ${priPNHPlaceCat}`);
                        newCategories = insertAtIndex(newCategories, priPNHPlaceCat, 0);
                        addUpdateAction(item, { categories: newCategories });
                    }
                }
                // Enable optional 2nd category button
                _buttonBanner.addCat2 = Flag.AddCat2.eval(item, specCases, newCategories, altCategories[0]);

                // Description update
                newDescripion = pnhMatchData[phDescriptionIdx];
                if (!isNullOrWhitespace(newDescripion) && !item.attributes.description.toUpperCase().includes(newDescripion.toUpperCase())) {
                    if (item.attributes.description !== '' && item.attributes.description !== null && item.attributes.description !== ' ') {
                        _buttonBanner.checkDescription = new Flag.CheckDescription();
                    }
                    logDev('Description updated');
                    newDescripion = `${newDescripion}\n${item.attributes.description}`;
                    actions.push(new UpdateObject(item, { description: newDescripion }));
                    _UPDATED_FIELDS.description.updated = true;
                }

                // Special Lock by PNH
                if (specCases.includes('lockAt5')) {
                    pnhLockLevel = 4;
                }
            } else { // if no PNH match found
                // Strong title case option for non-PNH places
                _buttonBanner.titleCaseName = Flag.TitleCaseName.eval(item, newName, newNameSuffix);

                newUrl = normalizeURL(newUrl, true, false, item, region, wl); // Normalize url

                // Generic Bank treatment
                _ixBank = item.attributes.categories.indexOf('BANK_FINANCIAL');
                _ixATM = item.attributes.categories.indexOf('ATM');
                _ixOffices = item.attributes.categories.indexOf('OFFICES');
                // if the name contains ATM in it
                if (newName.match(/\batm\b/ig) !== null) {
                    if (_ixOffices === 0) {
                        _buttonBanner.bankType1 = new Flag.BankType1();
                        _buttonBanner.bankBranch = new Flag.BankBranch(item);
                        _buttonBanner.standaloneATM = new Flag.StandaloneATM(item);
                        _buttonBanner.bankCorporate = new Flag.BankCorporate(item);
                    } else if (_ixBank === -1 && _ixATM === -1) {
                        _buttonBanner.bankBranch = new Flag.BankBranch(item);
                        _buttonBanner.standaloneATM = new Flag.StandaloneATM(item);
                    } else if (_ixATM === 0 && _ixBank > 0) {
                        _buttonBanner.bankBranch = new Flag.BankBranch(item);
                    } else if (_ixBank > -1) {
                        _buttonBanner.bankBranch = new Flag.BankBranch(item);
                        _buttonBanner.standaloneATM = new Flag.StandaloneATM(item);
                    }
                    // Net result: If the place has ATM cat only and ATM in the name, then it will be green
                } else if (_ixBank > -1 || _ixATM > -1) { // if no ATM in name:
                    if (_ixOffices === 0) {
                        _buttonBanner.bankBranch = new Flag.BankBranch(item);
                    } else if (_ixBank > -1 && _ixATM === -1) {
                        _buttonBanner.addATM = new Flag.AddATM(item);
                    } else if (_ixATM === 0 && _ixBank === -1) {
                        _buttonBanner.bankBranch = new Flag.BankBranch(item);
                        _buttonBanner.standaloneATM = new Flag.StandaloneATM(item);
                    } else if (_ixBank > 0 && _ixATM > 0) {
                        _buttonBanner.bankBranch = new Flag.BankBranch(item);
                        _buttonBanner.standaloneATM = new Flag.StandaloneATM(item);
                    }
                    // Net result: If the place has Bank category first, then it will be green
                } // END generic bank treatment
            } // END PNH match/no-match updates

            // Category/Name-based Services, added to any existing services:
            const catData = _PNH_DATA[countryCode].categories;
            const catNames = _PNH_DATA[countryCode].categoryNames;
            const catDataHeaders = catData[0].split('|');
            const catDataKeys = catData[1].split('|');
            let catDataTemp;

            if (!highlightOnly) {
                // Update name:
                if ((newName + (newNameSuffix || '')) !== item.attributes.name) {
                    logDev('Name updated');
                    actions.push(new UpdateObject(item, { name: newName + (newNameSuffix || '') }));
                    // actions.push(new UpdateObject(item, { name: newName }));
                    _UPDATED_FIELDS.name.updated = true;
                }

                // Update aliases
                newAliases = removeSFAliases(newName, newAliases);
                if (newAliases.some(alias => !item.attributes.aliases.includes(alias)) || newAliases.length !== item.attributes.aliases.length) {
                    logDev('Alt Names updated');
                    actions.push(new UpdateObject(item, { aliases: newAliases }));
                    _UPDATED_FIELDS.aliases.updated = true;
                }

                // PNH specific Services:

                const servHeaders = [];
                const servKeys = [];
                for (let jjj = 0; jjj < catDataHeaders.length; jjj++) {
                    const servHeaderCheck = catDataHeaders[jjj].match(/^ps_/i); // if it's a service header
                    if (servHeaderCheck) {
                        servHeaders.push(jjj);
                        servKeys.push(catDataKeys[jjj]);
                    }
                }

                if (newCategories.length > 0) {
                    for (let iii = 0; iii < catNames.length; iii++) {
                        if (newCategories.includes(catNames[iii])) {
                            catDataTemp = catData[iii].split('|');
                            for (let psix = 0; psix < servHeaders.length; psix++) {
                                if (!_servicesBanner[servKeys[psix]].pnhOverride) {
                                    if (catDataTemp[servHeaders[psix]] !== '') {
                                        // This section of code previously checked for values of "1", "2", and state/region codes.
                                        // A value of "2" or a state/region code would auto-add the service.  However, it was
                                        // felt that this was a problem since it is difficult to prove that every place in a
                                        // category would *always* offer a specific service.  So now, any value entered in the
                                        // spreadsheet cell will only display the service button, not turn it on.
                                        _servicesBanner[servKeys[psix]].active = true;
                                    }
                                }
                            }
                        }
                    }
                }
            }

            const isPoint = item.isPoint();
            // NOTE: do not use is2D() function. It doesn't seem to be 100% reliable.
            const isArea = !isPoint;
            let maxPointSeverity = _SEVERITY.GREEN;
            let maxAreaSeverity = _SEVERITY.RED;
            let highestCategoryLock = -1;

            for (let ixPlaceCat = 0; ixPlaceCat < newCategories.length; ixPlaceCat++) {
                const category = newCategories[ixPlaceCat];
                const ixPNHCat = catNames.indexOf(category);
                if (ixPNHCat > -1) {
                    catDataTemp = catData[ixPNHCat].split('|');
                    // CH_DATA_headers
                    // pc_area    pc_regpoint    pc_regarea    pc_lock1    pc_lock2    pc_lock3    pc_lock4    pc_lock5    pc_rare    pc_parent    pc_message
                    let pvaPoint = catDataTemp[catDataHeaders.indexOf('pc_point')];
                    let pvaArea = catDataTemp[catDataHeaders.indexOf('pc_area')];
                    const regPoint = catDataTemp[catDataHeaders.indexOf('pc_regpoint')].replace(/,[^A-za-z0-9]*/g, ',').split(',');
                    const regArea = catDataTemp[catDataHeaders.indexOf('pc_regarea')].replace(/,[^A-za-z0-9]*/g, ',').split(',');
                    if (regPoint.includes(state2L) || regPoint.includes(region) || regPoint.includes(countryCode)) {
                        pvaPoint = '1';
                        pvaArea = '';
                    } else if (regArea.includes(state2L) || regArea.includes(region) || regArea.includes(countryCode)) {
                        pvaPoint = '';
                        pvaArea = '1';
                    }

                    // If Post Office and VPO or CPU is in the name, always a point.
                    if (newCategories.includes('POST_OFFICE') && /\b(?:cpu|vpo)\b/i.test(item.attributes.name)) {
                        pvaPoint = '1';
                        pvaArea = '';
                    }

                    const pointSeverity = getPvaSeverity(pvaPoint, item);
                    const areaSeverity = getPvaSeverity(pvaArea, item);

                    if (isPoint && pointSeverity > _SEVERITY.GREEN) {
                        maxPointSeverity = Math.max(pointSeverity, maxPointSeverity);
                    } else if (isArea) {
                        maxAreaSeverity = Math.min(areaSeverity, maxAreaSeverity);
                    }

                    // display any messages regarding the category
                    const catMessage = catDataTemp[catDataHeaders.indexOf('pc_message')];
                    _buttonBanner.pnhCatMess = Flag.PnhCatMess.eval(item, catMessage, newCategories, highlightOnly);
                    // Unmapped categories
                    const catRare = catDataTemp[catDataHeaders.indexOf('pc_rare')].replace(/,[^A-Za-z0-9}]+/g, ',').split(',');
                    if (catRare.includes(state2L) || catRare.includes(region) || catRare.includes(countryCode)) {
                        if (catDataTemp[0] === 'OTHER' && ['GLR', 'NER', 'NWR', 'PLN', 'SCR', 'SER', 'NOR', 'HI', 'SAT'].includes(region)) {
                            if (!isLocked) {
                                _buttonBanner.unmappedRegion = new Flag.UnmappedRegion();
                                _buttonBanner.unmappedRegion.WLactive = false;
                                _buttonBanner.unmappedRegion.severity = _SEVERITY.BLUE;
                                _buttonBanner.unmappedRegion.message = 'The "Other" category should only be used if no other category applies.  Manually lock the place to override this flag.';
                                lockOK = false;
                            }
                        } else {
                            _buttonBanner.unmappedRegion = new Flag.UnmappedRegion();
                            if (wl.unmappedRegion) {
                                _buttonBanner.unmappedRegion.WLactive = false;
                                _buttonBanner.unmappedRegion.severity = _SEVERITY.GREEN;
                            } else {
                                lockOK = false;
                            }
                        }
                    }
                    // Parent Category
                    const catParent = catDataTemp[catDataHeaders.indexOf('pc_parent')].replace(/,[^A-Za-z0-9}]+/g, ',').split(',');
                    if (catParent.includes(state2L) || catParent.includes(region) || catParent.includes(countryCode)) {
                        _buttonBanner.parentCategory = new Flag.ParentCategory();
                        if (wl.parentCategory) {
                            _buttonBanner.parentCategory.WLactive = false;
                        }
                    }
                    // Set lock level
                    for (let lockix = 1; lockix < 6; lockix++) {
                        const catLockTemp = catDataTemp[catDataHeaders.indexOf(`pc_lock${lockix}`)].replace(/,[^A-Za-z0-9}]+/g, ',').split(',');
                        if (lockix - 1 > highestCategoryLock && (catLockTemp.includes(state2L) || catLockTemp.includes(region)
                            || catLockTemp.includes(countryCode))) {
                            highestCategoryLock = lockix - 1; // Offset by 1 since lock ranks start at 0
                        }
                    }
                }
            }

            if (highestCategoryLock > -1) {
                defaultLockLevel = highestCategoryLock;
            }

            if (isPoint) {
                if (maxPointSeverity === _SEVERITY.RED) {
                    _buttonBanner.areaNotPoint = new Flag.AreaNotPoint(item);
                    if (wl.areaNotPoint || item.attributes.lockRank >= defaultLockLevel) {
                        _buttonBanner.areaNotPoint.WLactive = false;
                        _buttonBanner.areaNotPoint.severity = _SEVERITY.GREEN;
                    } else {
                        lockOK = false;
                    }
                } else if (maxPointSeverity === _SEVERITY.YELLOW) {
                    _buttonBanner.areaNotPointMid = new Flag.AreaNotPointMid();
                    if (wl.areaNotPoint || item.attributes.lockRank >= defaultLockLevel) {
                        _buttonBanner.areaNotPointMid.WLactive = false;
                        _buttonBanner.areaNotPointMid.severity = _SEVERITY.GREEN;
                    } else {
                        lockOK = false;
                    }
                } else if (maxPointSeverity === _SEVERITY.BLUE) {
                    _buttonBanner.areaNotPointLow = new Flag.AreaNotPointLow();
                    if (wl.areaNotPoint || item.attributes.lockRank >= defaultLockLevel) {
                        _buttonBanner.areaNotPointLow.WLactive = false;
                        _buttonBanner.areaNotPointLow.severity = 0;
                    }
                }
            } else if (maxAreaSeverity === _SEVERITY.RED) {
                _buttonBanner.pointNotArea = new Flag.PointNotArea(item);
                if (wl.pointNotArea || item.attributes.lockRank >= defaultLockLevel) {
                    _buttonBanner.pointNotArea.WLactive = false;
                    _buttonBanner.pointNotArea.severity = 0;
                } else {
                    lockOK = false;
                }
            } else if (maxAreaSeverity === _SEVERITY.YELLOW) {
                _buttonBanner.pointNotAreaMid = new Flag.PointNotAreaMid();
                if (wl.pointNotArea || item.attributes.lockRank >= defaultLockLevel) {
                    _buttonBanner.pointNotAreaMid.WLactive = false;
                    _buttonBanner.pointNotAreaMid.severity = 0;
                } else {
                    lockOK = false;
                }
            } else if (maxAreaSeverity === _SEVERITY.BLUE) {
                _buttonBanner.pointNotAreaLow = new Flag.PointNotAreaLow();
                if (wl.pointNotArea || item.attributes.lockRank >= defaultLockLevel) {
                    _buttonBanner.pointNotAreaLow.WLactive = false;
                    _buttonBanner.pointNotAreaLow.severity = 0;
                }
            }

            const anpNone = _COLLEGE_ABBREVIATIONS.split('|');
            for (let cii = 0; cii < anpNone.length; cii++) {
                const anpNoneRE = new RegExp(`\\b${anpNone[cii]}\\b`, 'g');
                if (newName.match(anpNoneRE) !== null && _buttonBanner.areaNotPointLow) {
                    _buttonBanner.areaNotPointLow.severity = 0;
                    _buttonBanner.areaNotPointLow.WLactive = false;
                }
            }

            _buttonBanner.noHours = Flag.NoHours.eval(item, newCategories, wl, highlightOnly, actions);
            _buttonBanner.mismatch247 = Flag.Mismatch247.eval(item);

            const hoursOverlap = venueHasOverlappingHours(item);
            if (!hoursOverlap) {
                const tempHours = item.attributes.openingHours.slice();
                for (let ohix = 0; ohix < item.attributes.openingHours.length; ohix++) {
                    if (tempHours[ohix].days.length === 2 && tempHours[ohix].days[0] === 1 && tempHours[ohix].days[1] === 0) {
                        // separate hours
                        logDev('Correcting M-S entry...');
                        tempHours.push(new OpeningHour({ days: [0], fromHour: tempHours[ohix].fromHour, toHour: tempHours[ohix].toHour }));
                        tempHours[ohix].days = [1];
                        addUpdateAction(item, { openingHours: tempHours }, actions);
                    }
                }
            }

            _buttonBanner.hoursOverlap = Flag.HoursOverlap.eval(hoursOverlap);
            _buttonBanner.oldHours = Flag.OldHours.eval(item, catData, highlightOnly);

            if (!highlightOnly) {
                // Highlight 24/7 button if hours are set that way, and add button for all places
                if (isAlwaysOpen(item)) {
                    _servicesBanner.add247.checked = true;
                }
                _servicesBanner.add247.active = true;
            }

            // URL updating
            _updateURL = true;
            if (newUrl !== item.attributes.url && !isNullOrWhitespace(newUrl)) {
                if (pnhNameRegMatch && item.attributes.url !== null && item.attributes.url !== '' && newUrl !== 'badURL') { // for cases where there is an existing URL in the WME place, and there is a PNH url on queue:
                    let newURLTemp = normalizeURL(newUrl, true, false, item, null, wl); // normalize
                    const itemURL = normalizeURL(item.attributes.url, true, false, item, null, wl);
                    newURLTemp = newURLTemp.replace(/^www\.(.*)$/i, '$1'); // strip www
                    const itemURLTemp = itemURL.replace(/^www\.(.*)$/i, '$1'); // strip www
                    if (newURLTemp !== itemURLTemp) { // if formatted URLs don't match, then alert the editor to check the existing URL
                        _buttonBanner.longURL = new Flag.LongURL(item, placePL);
                        if (wl.longURL) {
                            _buttonBanner.longURL.severity = _SEVERITY.GREEN;
                            _buttonBanner.longURL.WLactive = false;
                        }
                        if (!highlightOnly && _updateURL && itemURL !== item.attributes.url) { // Update the URL
                            logDev('URL formatted');
                            actions.push(new UpdateObject(item, { url: itemURL }));
                            _UPDATED_FIELDS.url.updated = true;
                        }
                        _updateURL = false;
                        _tempPNHURL = newUrl;
                    }
                }
                if (!highlightOnly && _updateURL && newUrl !== 'badURL' && newUrl !== item.attributes.url) { // Update the URL
                    logDev('URL updated');
                    actions.push(new UpdateObject(item, { url: newUrl }));
                    _UPDATED_FIELDS.url.updated = true;
                }
            }

            let normalizedPhone;
            if (newPhone) {
                normalizedPhone = normalizePhone(newPhone, outputPhoneFormat);
                if (normalizedPhone !== 'badPhone') newPhone = normalizedPhone;
            }

            _buttonBanner.phoneInvalid = Flag.PhoneInvalid.eval(normalizedPhone, outputPhoneFormat);
            _buttonBanner.phoneMissing = Flag.PhoneMissing.eval(item, newPhone, wl, region, outputPhoneFormat, !!_buttonBanner.addRecommendedPhone);

            // Check if valid area code  #LOC# USA and CAN only
            if (!wl.aCodeWL && (countryCode === 'USA' || countryCode === 'CAN')) {
                if (newPhone !== null && newPhone.match(/[2-9]\d{2}/) !== null) {
                    const areaCode = newPhone.match(/[2-9]\d{2}/)[0];
                    if (!_areaCodeList.includes(areaCode)) {
                        _buttonBanner.badAreaCode = new Flag.BadAreaCode(item, newPhone, outputPhoneFormat);
                    }
                }
            }
            if (!highlightOnly && newPhone !== item.attributes.phone) {
                logDev('Phone updated');
                actions.push(new UpdateObject(item, { phone: newPhone }));
                _UPDATED_FIELDS.phone.updated = true;
            }

            // Post Office check
            _buttonBanner.isThisAPostOffice = Flag.IsThisAPostOffice.eval(item, highlightOnly, countryCode, newCategories, newName);
            _buttonBanner.PlaceWebsite = Flag.PlaceWebsite.eval(item, highlightOnly, countryCode, newCategories, _buttonBanner.PlaceWebsite);

            const isUspsPostOffice = countryCode === 'USA' && !newCategories.includes('PARKING_LOT') && newCategories.includes('POST_OFFICE');
            _buttonBanner.missingUSPSZipAlt = Flag.MissingUSPSZipAlt.eval(item, isUspsPostOffice, newName, newAliases, wl, highlightOnly);
            if (isUspsPostOffice) {
                if (!highlightOnly) {
                    _buttonBanner.NewPlaceSubmit = null;
                    if (item.attributes.url !== 'usps.com') {
                        actions.push(new UpdateObject(item, { url: 'usps.com' }));
                        _UPDATED_FIELDS.url.updated = true;
                        _buttonBanner.urlMissing = null;
                    }
                }

                let postOfficeRegEx;
                if (state2L === 'KY' || (state2L === 'NY' && addr.city && ['Queens', 'Bronx', 'Manhattan', 'Brooklyn', 'Staten Island'].includes(addr.city.attributes.name))) {
                    postOfficeRegEx = /^post office \d{5}( [-–](?: cpu| vpo)?(?: [a-z0-9]+){1,})?$/i;
                } else {
                    postOfficeRegEx = /^post office [-–](?: cpu| vpo)?(?: [a-z0-9]+){1,}$/i;
                }
                newName = newName.trimLeft().replace(/ {2,}/, ' ');
                if (newNameSuffix) {
                    newNameSuffix = newNameSuffix.trimRight().replace(/\bvpo\b/i, 'VPO').replace(/\bcpu\b/i, 'CPU').replace(/ {2,}/, ' ');
                }
                const nameToCheck = newName + (newNameSuffix || '');
                if (!postOfficeRegEx.test(nameToCheck)) {
                    _buttonBanner.formatUSPS = new Flag.FormatUSPS();
                    lockOK = false;
                } else if (!highlightOnly) {
                    if (nameToCheck !== item.attributes.name) {
                        actions.push(new UpdateObject(item, { name: nameToCheck }));
                    }
                    _buttonBanner.catPostOffice = new Flag.CatPostOffice();
                }
                if (!newAliases.some(alias => alias.toUpperCase() === 'USPS')) {
                    if (!highlightOnly) {
                        newAliases.push('USPS');
                        actions.push(new UpdateObject(item, { aliases: newAliases }));
                        _UPDATED_FIELDS.aliases.updated = true;
                    } else {
                        _buttonBanner.missingUSPSAlt = new Flag.MissingUSPSAlt();
                    }
                }

                const descr = item.attributes.description;
                const lines = descr.split('\n');
                if (lines.length < 1 || !/^.{2,}, [A-Z]{2}\s{1,2}\d{5}$/.test(lines[0])) {
                    _buttonBanner.missingUSPSDescription = new Flag.MissingUSPSDescription();
                    if (wl.missingUSPSDescription) {
                        _buttonBanner.missingUSPSDescription.severity = _SEVERITY.GREEN;
                        _buttonBanner.missingUSPSDescription.WLactive = false;
                    }
                }
            } // END Post Office check
        } // END if (!residential && has name)

        _buttonBanner.gasMismatch = Flag.GasMismatch.eval(item, newCategories, newBrand, newName, wl);

        // Check EV charging stations:
        _buttonBanner.evChargingStationWarning = Flag.EVChargingStationWarning.eval(item, highlightOnly);
        _buttonBanner.addCommonEVPaymentMethods = Flag.AddCommonEVPaymentMethods.eval(item, highlightOnly, wl);
        _buttonBanner.removeUncommonEVPaymentMethods = Flag.RemoveUncommonEVPaymentMethods.eval(item, highlightOnly, wl);
        _buttonBanner.evcsPriceMissing = Flag.EVCSPriceMissing.eval(item, highlightOnly);

        // Name check
        _buttonBanner.nameMissing = Flag.NameMissing.eval(item, newName);
        _buttonBanner.plaNameMissing = Flag.PlaNameMissing.eval(item, newName, _USER.rank);
        _buttonBanner.plaNameNonStandard = Flag.PlaNameNonStandard.eval(item, wl);
        _buttonBanner.gasNameMissing = Flag.GasNameMissing.eval(item, newName, newBrand, highlightOnly);

        _buttonBanner.plaIsPublic = Flag.PlaIsPublic.eval(item, highlightOnly);

        // House number / HN check
        let currentHN = item.attributes.houseNumber;
        // Check to see if there's an action that is currently updating the house number.
        const updateHnAction = actions && actions.find(action => action.newAttributes && action.newAttributes.houseNumber);
        if (updateHnAction) currentHN = updateHnAction.newAttributes.houseNumber;
        // Use the inferred address street if currently no street.
        const hasStreet = item.attributes.streetID || (inferredAddress && inferredAddress.street);

        if (hasStreet && (!currentHN || currentHN.replace(/\D/g, '').length === 0)) {
            if (!['BRIDGE', 'ISLAND', 'FOREST_GROVE', 'SEA_LAKE_POOL', 'RIVER_STREAM', 'CANAL',
                'DAM', 'TUNNEL', 'JUNCTION_INTERCHANGE'].includes(item.attributes.categories[0])
                && !item.attributes.categories.includes('REST_AREAS')) {
                _buttonBanner.hnMissing = new Flag.HnMissing(item);
                if (state2L === 'PR' || ['SCENIC_LOOKOUT_VIEWPOINT'].includes(item.attributes.categories[0])) {
                    _buttonBanner.hnMissing.severity = _SEVERITY.GREEN;
                    _buttonBanner.hnMissing.WLactive = false;
                } else if (item.isParkingLot()) {
                    _buttonBanner.hnMissing.WLactive = false;
                    if (item.attributes.lockRank < 2) {
                        lockOK = false;
                        let msgAdd;
                        if (_USER.rank < 3) {
                            msgAdd = 'Request an R3+ lock to confirm no HN.';
                        } else {
                            msgAdd = 'Lock to R3+ to confirm no HN.';
                        }
                        _buttonBanner.hnMissing.suffixMessage = msgAdd;
                        _buttonBanner.hnMissing.severity = _SEVERITY.BLUE;
                    } else {
                        _buttonBanner.hnMissing.severity = _SEVERITY.GREEN;
                    }
                } else if (wl.HNWL) {
                    _buttonBanner.hnMissing.severity = _SEVERITY.GREEN;
                    _buttonBanner.hnMissing.WLactive = false;
                } else {
                    lockOK = false;
                }
            }
        } else if (currentHN) {
            _buttonBanner.hnTooManyDigits = Flag.HnTooManyDigits.eval(currentHN, wl);

            // 2020-10-5 Disabling HN validity checks for now. See the note on the HnNonStandard flag object for more details.

            // let hnOK = false;
            // let updateHNflag = false;
            // const hnTemp = currentHN.replace(/[^\d]/g, ''); // Digits only
            // const hnTempDash = currentHN.replace(/[^\d-]/g, ''); // Digits and dashes only
            // if (hnTemp < 1000000 && state2L === 'NY' && addr.city.attributes.name === 'Queens' && hnTempDash.match(/^\d{1,4}-\d{1,4}$/g) !== null) {
            //     updateHNflag = true;
            //     // hnOK = true;
            // }
            // if (hnTemp === currentHN && hnTemp < 1000000) { //  general check that HN is 6 digits or less, & that it is only [0-9]
            //     hnOK = true;
            // }
            // if (state2L === 'HI' && hnTempDash.match(/^\d{1,2}-\d{1,4}$/g) !== null) {
            //     if (hnTempDash === hnTempDash.match(/^\d{1,2}-\d{1,4}$/g)[0]) {
            //         hnOK = true;
            //     }
            // }

            // if (!hnOK) {
            //     _buttonBanner.hnNonStandard = new Flag.HnNonStandard();
            //     if (_wl.hnNonStandard) {
            //         _buttonBanner.hnNonStandard.WLactive = false;
            //         _buttonBanner.hnNonStandard.severity = SEVERITY.GREEN;
            //     } else {
            //         lockOK = false;
            //     }
            // }
            // if (updateHNflag) {
            //     _buttonBanner.hnDashRemoved = new Flag.HnDashRemoved();
            //     if (!highlightOnly) {
            //         actions.push(new UpdateObject(item, { houseNumber: hnTemp }));
            //         _UPDATED_FIELDS.address.updated = true;
            //     } else if (highlightOnly) {
            //         if (item.attributes.residential) {
            //             _buttonBanner.hnDashRemoved.severity = SEVERITY.RED;
            //         } else {
            //             _buttonBanner.hnDashRemoved.severity = SEVERITY.BLUE;
            //         }
            //     }
            // }
        }

        _buttonBanner.cityMissing = Flag.CityMissing.eval(item, addr, highlightOnly);
        _buttonBanner.streetMissing = Flag.StreetMissing.eval(item, addr);
        _buttonBanner.notAHospital = Flag.NotAHospital.eval(item, newCategories, newName, wl);

        // CATEGORY vs. NAME checks
        _buttonBanner.changeToPetVet = Flag.ChangeToPetVet.eval(item, newName, newCategories, wl);
        _buttonBanner.changeToDoctorClinic = Flag.ChangeToDoctorClinic.eval(item, newCategories, highlightOnly, pnhNameRegMatch);
        _buttonBanner.notASchool = Flag.NotASchool.eval(newName, newCategories, wl);

        // Some cats don't need PNH messages and url/phone severities
        if (['BRIDGE', 'FOREST_GROVE', 'DAM', 'TUNNEL', 'CEMETERY'].includes(item.attributes.categories[0])) {
            _buttonBanner.NewPlaceSubmit = null;
            if (_buttonBanner.phoneMissing) {
                _buttonBanner.phoneMissing.severity = _SEVERITY.GREEN;
                _buttonBanner.phoneMissing.WLactive = false;
            }
            if (_buttonBanner.urlMissing) {
                _buttonBanner.urlMissing.severity = _SEVERITY.GREEN;
                _buttonBanner.urlMissing.WLactive = false;
            }
        } else if (['ISLAND', 'SEA_LAKE_POOL', 'RIVER_STREAM', 'CANAL', 'JUNCTION_INTERCHANGE', 'SCENIC_LOOKOUT_VIEWPOINT'].includes(item.attributes.categories[0])) {
            // Some cats don't need PNH messages and url/phone messages
            _buttonBanner.NewPlaceSubmit = null;
            _buttonBanner.phoneMissing = null;
            _buttonBanner.urlMissing = null;
        }

        // *** Rest Area parsing
        // check rest area name against standard formats or if has the right categories
        const oldName = item.attributes.name;
        const hasRestAreaCategory = newCategories.includes('REST_AREAS');
        _buttonBanner.restAreaSpec = Flag.RestAreaSpec.eval(item, hasRestAreaCategory, newName, wl);
        _buttonBanner.restAreaScenic = Flag.RestAreaScenic.eval(item, hasRestAreaCategory, newCategories, wl);
        _buttonBanner.restAreaNoTransportation = Flag.RestAreaNoTransportation.eval(item, hasRestAreaCategory, newCategories);
        _buttonBanner.restAreaGas = Flag.RestAreaGas.eval(hasRestAreaCategory, newCategories);
        _buttonBanner.restAreaName = Flag.RestAreaName.eval(item, hasRestAreaCategory, wl);

        if (hasRestAreaCategory) {
            if (item.isPoint()) { // needs to be area
                _buttonBanner.areaNotPoint = new Flag.AreaNotPoint(item);
            }
            _buttonBanner.pointNotArea = null;
            _buttonBanner.unmappedRegion = null;

            if (!highlightOnly && oldName.match(/^Rest Area.* - /) !== null) {
                const newSuffix = newNameSuffix.replace(/Mile/i, 'mile');
                if (newName + newSuffix !== item.attributes.name) {
                    actions.push(new UpdateObject(item, { name: newName + newSuffix }));
                    _UPDATED_FIELDS.name.updated = true;
                    logDev('Lower case "mile"');
                } else {
                    // The new name matches the original name, so the only change would have been to capitalize "Mile", which
                    // we don't want. So remove any previous name-change action.  Note: this feels like a hack and is probably
                    // a fragile workaround.  The name shouldn't be capitalized in the first place, unless necessary.
                    for (let i = 0; i < actions.length; i++) {
                        const action = actions[i];
                        if (action.newAttributes.name) {
                            actions.splice(i, 1);
                            _UPDATED_FIELDS.name.updated = false;
                            break;
                        }
                    }
                }
            }

            // switch to rest area wiki button
            if (!highlightOnly) {
                _buttonBanner2.restAreaWiki.active = true;
                _buttonBanner2.placesWiki.active = false;
            }

            if (_buttonBanner.urlMissing) {
                _buttonBanner.urlMissing.WLactive = false;
                _buttonBanner.urlMissing.severity = _SEVERITY.GREEN;
            }
            if (_buttonBanner.phoneMissing) {
                _buttonBanner.phoneMissing.severity = _SEVERITY.GREEN;
                _buttonBanner.phoneMissing.WLactive = false;
            }
        }

        // update Severity for banner messages
        Object.keys(_buttonBanner).forEach(key => {
            if (_buttonBanner[key] && _buttonBanner[key].active) {
                totalSeverity = Math.max(_buttonBanner[key].severity, totalSeverity);
            }
        });

        // Place locking
        // final formatting of desired lock levels
        let levelToLock;
        if (pnhLockLevel !== -1 && !highlightOnly) {
            logDev(`PNHLockLevel: ${pnhLockLevel}`);
            levelToLock = pnhLockLevel;
        } else {
            levelToLock = defaultLockLevel;
        }
        if (region === 'SER') {
            if (newCategories.includes('COLLEGE_UNIVERSITY') && newCategories.includes('PARKING_LOT')) {
                levelToLock = _LOCK_LEVEL_4;
            } else if (item.isPoint() && newCategories.includes('COLLEGE_UNIVERSITY') && (!newCategories.includes('HOSPITAL_MEDICAL_CARE')
                || !newCategories.includes('HOSPITAL_URGENT_CARE'))) {
                levelToLock = _LOCK_LEVEL_4;
            }
        }

        if (levelToLock > (_USER.rank - 1)) { levelToLock = (_USER.rank - 1); } // Only lock up to the user's level

        // Brand checking (be sure to check this after determining if brand will be forced, when harmonizing)
        _buttonBanner.gasNoBrand = Flag.GasNoBrand.eval(item, newBrand, levelToLock);
        _buttonBanner.gasUnbranded = Flag.GasUnbranded.eval(item, newBrand);

        // If no Google link and severity would otherwise allow locking, ask if user wants to lock anyway.
        if (!isLocked && _buttonBanner.extProviderMissing && _buttonBanner.extProviderMissing.active
            && _buttonBanner.extProviderMissing.severity <= _SEVERITY.YELLOW) {
            _buttonBanner.extProviderMissing.severity = _SEVERITY.RED;
            totalSeverity = _SEVERITY.RED;
            if (lockOK) {
                _buttonBanner.extProviderMissing.value = `Lock anyway? (${levelToLock + 1})`;
                _buttonBanner.extProviderMissing.title = 'If no Google link exists, lock this place.\nIf there is still no Google link after '
                    + '6 months from the last update date, it will turn red as a reminder to search again.';
                _buttonBanner.extProviderMissing.action = () => {
                    addUpdateAction(item, { lockRank: levelToLock }, null, true);
                };
            }
        }

        if (!highlightOnly) {
            // Update the lockOK value if "noLock" is set on any flag.
            lockOK &&= !Object.keys(_buttonBanner).some(key => _buttonBanner[key]?.noLock);
            logDev(`Severity: ${totalSeverity}; lockOK: ${lockOK}`);
        }

        _buttonBanner.placeLocked = Flag.PlaceLocked.eval(item, lockOK, totalSeverity, levelToLock, highlightOnly, actions);

        // IGN check
        if (!item.attributes.residential) {
            const updatedBy = W.model.users.getObjectById(item.attributes.updatedBy);
            if (updatedBy && /^ign_/i.test(updatedBy.userName)) {
                _buttonBanner.ignEdited = new Flag.IgnEdited();
            }
        }

        // waze_maint_bot check
        const updatedById = item.attributes.updatedBy ? item.attributes.updatedBy : item.attributes.createdBy;
        const updatedBy = W.model.users.getObjectById(updatedById);
        const updatedByName = updatedBy ? updatedBy.userName : null;
        const botNamesAndIDs = [
            '^waze-maint', '^105774162$',
            '^waze3rdparty$', '^361008095$',
            '^WazeParking1$', '^338475699$',
            '^admin$', '^-1$',
            '^avsus$', '^107668852$'
        ];

        const botRegEx = new RegExp(botNamesAndIDs.join('|'), 'i');
        if (item.isUnchanged() && !item.attributes.residential && updatedById && (botRegEx.test(updatedById.toString())
            || (updatedByName && botRegEx.test(updatedByName)))) {
            _buttonBanner.wazeBot = new Flag.WazeBot(item);
        }

        // RPP Locking option for R3+
        _buttonBanner.lockRPP = Flag.LockRPP.eval(item, highlightOnly, defaultLockLevel);

        // Turn off unnecessary buttons
        if (newCategories.includes('PHARMACY')) {
            if (_buttonBanner.addPharm) _buttonBanner.addPharm = null;
        }
        if (newCategories.includes('SUPERMARKET_GROCERY')) {
            if (_buttonBanner.addSuper) _buttonBanner.addSuper = null;
        }

        // Final alerts for non-severe locations
        if (!item.attributes.residential && totalSeverity < _SEVERITY.RED) {
            const nameShortSpace = newName.toUpperCase().replace(/[^A-Z ']/g, '');
            if (nameShortSpace.includes('\'S HOUSE') || nameShortSpace.includes('\'S HOME') || nameShortSpace.includes('\'S WORK')) {
                if (!containsAny(newCategories, ['RESTAURANT', 'DESSERT', 'BAR']) && !pnhNameRegMatch) {
                    _buttonBanner.resiTypeNameSoft = new Flag.ResiTypeNameSoft();
                }
            }
            if (['HOME', 'MY HOME', 'HOUSE', 'MY HOUSE', 'PARENTS HOUSE', 'CASA', 'MI CASA', 'WORK', 'MY WORK', 'MY OFFICE',
                'MOMS HOUSE', 'DADS HOUSE', 'MOM', 'DAD'].includes(nameShortSpace)) {
                _buttonBanner.resiTypeName = new Flag.ResiTypeName();
                if (wl.resiTypeName) {
                    _buttonBanner.resiTypeName.WLactive = false;
                }
                _buttonBanner.resiTypeNameSoft = null;
            }
            if (item.attributes.description.toLowerCase().includes('google') || item.attributes.description.toLowerCase().includes('yelp')) {
                _buttonBanner.suspectDesc = new Flag.SuspectDesc();
                if (wl.suspectDesc) {
                    _buttonBanner.suspectDesc.WLactive = false;
                }
            }
        }

        // Return severity for highlighter (no dupe run))
        if (highlightOnly) {
            // get severities from the banners
            totalSeverity = _SEVERITY.GREEN;
            Object.keys(_buttonBanner).forEach(tempKey => {
                if (_buttonBanner[tempKey] && _buttonBanner[tempKey].active) { //  If the particular message is active
                    if (_buttonBanner[tempKey].hasOwnProperty('WLactive')) {
                        if (_buttonBanner[tempKey].WLactive) { // If there's a WL option, enable it
                            totalSeverity = Math.max(_buttonBanner[tempKey].severity, totalSeverity);
                        }
                    } else {
                        totalSeverity = Math.max(_buttonBanner[tempKey].severity, totalSeverity);
                    }
                }
            });

            // Special case flags
            if (item.attributes.lockRank === 0
                && item.attributes.categories.some(cat => ['HOSPITAL_MEDICAL_CARE', 'HOSPITAL_URGENT_CARE', 'GAS_STATION'].includes(cat))) {
                totalSeverity = _SEVERITY.PINK;
            }

            if (totalSeverity === _SEVERITY.GREEN && _buttonBanner.placeLocked?.hlLockFlag) {
                totalSeverity = 'lock';
            }
            if (totalSeverity === 1 && _buttonBanner.placeLocked?.hlLockFlag) {
                totalSeverity = 'lock1';
            }
            if (item.attributes.adLocked) {
                totalSeverity = 'adLock';
            }

            return totalSeverity;
        }

        // *** Below here is for harmonization only.  HL ends in previous step.

        // Run nearby duplicate place finder function
        _dupeHNRangeList = [];
        _dupeBanner = {};
        if (newName.replace(/[^A-Za-z0-9]/g, '').length > 0 && !item.attributes.residential && !isEmergencyRoom(item) && !isRestArea(item)) {
            // don't zoom and pan for results outside of FOV
            let duplicateName = findNearbyDuplicate(newName, newAliases, item, !$('#WMEPH-DisableDFZoom').prop('checked'));
            if (duplicateName[1]) {
                _buttonBanner.overlapping = new Flag.Overlapping();
            }
            [duplicateName] = duplicateName;
            if (duplicateName.length) {
                if (duplicateName.length + 1 !== _dupeIDList.length && _USER.isDevUser) {
                    // If there's an issue with the data return, allow an error report
                    WazeWrap.Alerts.confirm(
                        _SCRIPT_NAME,
                        'WMEPH: Dupefinder Error!<br>Click OK to report this',
                        () => {
                            // if the category doesn't translate, then pop an alert that will make a forum post to the thread
                            reportError({
                                subject: 'WMEPH Bug report DupeID',
                                message: `Script version: ${_SCRIPT_VERSION}${_BETA_VERSION_STR}\nPermalink: ${placePL}\nPlace name: ${
                                    item.attributes.name}\nCountry: ${addr.country.name}\n--------\nDescribe the error:\nDupeID mismatch with dupeName list`
                            });
                        },
                        () => { }
                    );
                } else {
                    const wlAction = dID => {
                        const wlKey = 'dupeWL';
                        if (!_venueWhitelist.hasOwnProperty(itemID)) { // If venue is NOT on WL, then add it.
                            _venueWhitelist[itemID] = { dupeWL: [] };
                        }
                        if (!_venueWhitelist[itemID].hasOwnProperty(wlKey)) { // If dupeWL key is not in venue WL, then initialize it.
                            _venueWhitelist[itemID][wlKey] = [];
                        }
                        _venueWhitelist[itemID].dupeWL.push(dID); // WL the id for the duplicate venue
                        _venueWhitelist[itemID].dupeWL = _.uniq(_venueWhitelist[itemID].dupeWL);
                        // Make an entry for the opposite item
                        if (!_venueWhitelist.hasOwnProperty(dID)) { // If venue is NOT on WL, then add it.
                            _venueWhitelist[dID] = { dupeWL: [] };
                        }
                        if (!_venueWhitelist[dID].hasOwnProperty(wlKey)) { // If dupeWL key is not in venue WL, then initialize it.
                            _venueWhitelist[dID][wlKey] = [];
                        }
                        _venueWhitelist[dID].dupeWL.push(itemID); // WL the id for the duplicate venue
                        _venueWhitelist[dID].dupeWL = _.uniq(_venueWhitelist[dID].dupeWL);
                        saveWhitelistToLS(true); // Save the WL to local storage
                        wmephWhitelistCounter();
                        _buttonBanner2.clearWL.active = true;
                        _dupeBanner[dID].active = false;
                        harmonizePlaceGo(item, 'harmonize');
                    };
                    for (let ijx = 1; ijx < duplicateName.length + 1; ijx++) {
                        _dupeBanner[_dupeIDList[ijx]] = {
                            active: true,
                            severity: _SEVERITY.YELLOW,
                            message: duplicateName[ijx - 1],
                            WLactive: false,
                            WLvalue: _WL_BUTTON_TEXT,
                            WLtitle: 'Whitelist Duplicate',
                            WLaction: wlAction
                        };
                        if (_venueWhitelist.hasOwnProperty(itemID) && _venueWhitelist[itemID].hasOwnProperty('dupeWL')
                            && _venueWhitelist[itemID].dupeWL.includes(_dupeIDList[ijx])) {
                            // if the dupe is on the whitelist then remove it from the banner
                            _dupeBanner[_dupeIDList[ijx]].active = false;
                        } else {
                            // Otherwise, activate the WL button
                            _dupeBanner[_dupeIDList[ijx]].WLactive = true;
                        }
                    } // END loop for duplicate venues
                }
            }
        }

        // Check HN range (this depends on the returned dupefinder data, so has to run after it)
        if (_dupeHNRangeList.length > 3) {
            let dhnix;
            const dupeHNRangeListSorted = [];
            sortWithIndex(_dupeHNRangeDistList);
            for (dhnix = 0; dhnix < _dupeHNRangeList.length; dhnix++) {
                dupeHNRangeListSorted.push(_dupeHNRangeList[_dupeHNRangeDistList.sortIndices[dhnix]]);
            }
            // Calculate HN/distance ratio with other venues
            // var sumHNRatio = 0;
            const arrayHNRatio = [];
            for (dhnix = 0; dhnix < dupeHNRangeListSorted.length; dhnix++) {
                arrayHNRatio.push(Math.abs((parseInt(item.attributes.houseNumber, 10) - dupeHNRangeListSorted[dhnix]) / _dupeHNRangeDistList[dhnix]));
            }
            sortWithIndex(arrayHNRatio);
            // Examine either the median or the 8th index if length is >16
            const arrayHNRatioCheckIX = Math.min(Math.round(arrayHNRatio.length / 2), 8);
            if (arrayHNRatio[arrayHNRatioCheckIX] > 1.4) {
                _buttonBanner.HNRange = new Flag.HNRange();
                if (wl.HNRange) {
                    _buttonBanner.HNRange.WLactive = false;
                    _buttonBanner.HNRange.active = false;
                }
                if (arrayHNRatio[arrayHNRatioCheckIX] > 5) {
                    _buttonBanner.HNRange.severity = _SEVERITY.RED;
                }
                // show stats if HN out of range
                logDev(`HNs: ${dupeHNRangeListSorted}`);
                logDev(`Distances: ${_dupeHNRangeDistList}`);
                logDev(`arrayHNRatio: ${arrayHNRatio}`);
                logDev(`HN Ratio Score: ${arrayHNRatio[Math.round(arrayHNRatio.length / 2)]}`);
            }
        }

        executeMultiAction(actions);

        if (!highlightOnly) {
            // Update icons to reflect current WME place services
            updateServicesChecks(_servicesBanner);

            // Add green highlighting to edit panel fields that have been updated by WMEPH
            _UPDATED_FIELDS.updateEditPanelHighlights();

            // Assemble the banners
            assembleBanner(); // Make Messaging banners
        }

        // showOpenPlaceWebsiteButton();
        // showSearchButton();

        // Highlighting will return a value, but no need to return a value here (for end of harmonization).
        // Adding this line to satisfy eslint.
        return undefined;
    } // END harmonizePlaceGo function

    // Set up banner messages
    function assembleBanner() {
        const venue = getSelectedVenue();
        if (!venue) return;
        logDev('Building banners');
        let dupesFound = 0;
        let rowData;
        let $rowDiv;
        let rowDivs = [];
        let totalSeverity = _SEVERITY.GREEN;

        const func = elem => ({ id: elem.getAttribute('id'), val: elem.value });
        _textEntryValues = $('#WMEPH_banner input[type="text"]').toArray().map(func);
        _textEntryValues = _textEntryValues.concat($('#WMEPH_banner textarea').toArray().map(func));

        // Setup duplicates banners
        $rowDiv = $('<div class="banner-row yellow">');
        Object.keys(_dupeBanner).forEach(tempKey => {
            rowData = _dupeBanner[tempKey];
            if (rowData.active) {
                dupesFound += 1;
                const $dupeDiv = $('<div class="dupe">').appendTo($rowDiv);
                $dupeDiv.append($('<span style="margin-right:4px">').html(`&bull; ${rowData.message}`));
                if (rowData.value) {
                    // Nothing happening here yet.
                }
                if (rowData.WLactive && rowData.WLaction) { // If there's a WL option, enable it
                    totalSeverity = Math.max(rowData.severity, totalSeverity);
                    $dupeDiv.append($('<button>', {
                        class: 'btn btn-success btn-xs wmephwl-btn',
                        id: `WMEPH_WL${tempKey}`,
                        title: rowData.WLtitle
                    }).text(rowData.WLvalue));
                }
            }
        });
        if (dupesFound) { // if at least 1 dupe
            $rowDiv.prepend(`Possible duplicate${dupesFound > 1 ? 's' : ''}:`);
            rowDivs.push($rowDiv);
        }

        // Build banners above the Services
        Object.keys(_buttonBanner).forEach(tempKey => {
            rowData = _buttonBanner[tempKey];
            if (rowData && rowData.active) { //  If the particular message is active
                $rowDiv = $('<div class="banner-row">');
                if (rowData.severity === _SEVERITY.RED) {
                    $rowDiv.addClass('red');
                } else if (rowData.severity === _SEVERITY.YELLOW) {
                    $rowDiv.addClass('yellow');
                } else if (rowData.severity === _SEVERITY.BLUE) {
                    $rowDiv.addClass('blue');
                } else if (rowData.severity === _SEVERITY.GREEN) {
                    $rowDiv.addClass('gray');
                }
                if (rowData.divId) {
                    $rowDiv.attr('id', rowData.divId);
                }
                if (rowData.message && rowData.message.length) {
                    $rowDiv.append($('<span>').css({ 'margin-right': '4px' }).append(`&bull; ${rowData.message}`));
                }
                if (rowData.value) {
                    $rowDiv.append($('<button>', {
                        class: 'btn btn-default btn-xs wmeph-btn',
                        id: `WMEPH_${tempKey}`,
                        title: rowData.title || ''
                    }).css({ 'margin-right': '4px' }).html(rowData.value));
                }
                if (rowData.value2) {
                    $rowDiv.append($('<button>', {
                        class: 'btn btn-default btn-xs wmeph-btn',
                        id: `WMEPH_${tempKey}_2`,
                        title: rowData.title2 || ''
                    }).css({ 'margin-right': '4px' }).html(rowData.value2));
                }
                if (rowData.WLactive) {
                    if (rowData.WLaction) { // If there's a WL option, enable it
                        totalSeverity = Math.max(rowData.severity, totalSeverity);
                        $rowDiv.append(
                            $('<button>', { class: 'btn btn-success btn-xs wmephwl-btn', id: `WMEPH_WL${tempKey}`, title: rowData.WLtitle })
                                .text('WL')
                        );
                    }
                } else {
                    totalSeverity = Math.max(rowData.severity, totalSeverity);
                }
                if (rowData.suffixMessage) {
                    $rowDiv.append($('<div>').css({ 'margin-top': '2px' }).append(rowData.suffixMessage));
                }

                rowDivs.push($rowDiv);
            }
        });

        if ($('#WMEPH-ColorHighlighting').prop('checked')) {
            venue.attributes.wmephSeverity = totalSeverity;
        }

        if ($('#WMEPH_banner').length === 0) {
            $('<div id="WMEPH_banner">').prependTo('#wmeph-panel');
        } else {
            $('#WMEPH_banner').empty();
        }
        let bgColor;
        switch (totalSeverity) {
            case _SEVERITY.BLUE:
                bgColor = 'rgb(50, 50, 230)'; // blue
                break;
            case _SEVERITY.YELLOW:
                bgColor = 'rgb(217, 173, 42)'; // yellow
                break;
            case _SEVERITY.RED:
                bgColor = 'rgb(211, 48, 48)'; // red
                break;
            default:
                bgColor = 'rgb(36, 172, 36)'; // green
        }
        $('#WMEPH_banner').css({ 'background-color': bgColor }).append(rowDivs);

        assembleServicesBanner();

        //  Build general banners (below the Services)
        rowDivs = [];
        Object.keys(_buttonBanner2).forEach(tempKey => {
            const banner2RowData = _buttonBanner2[tempKey];
            if (banner2RowData.active) { //  If the particular message is active
                $rowDiv = $('<div>');
                $rowDiv.append(banner2RowData.message);
                if (banner2RowData.action) {
                    $rowDiv.append(` <input class="btn btn-info btn-xs wmeph-btn" id="WMEPH_${tempKey}" title="${
                        banner2RowData.title}" style="" type="button" value="${banner2RowData.value}">`);
                }
                rowDivs.push($rowDiv);
                totalSeverity = Math.max(_buttonBanner2[tempKey].severity, totalSeverity);
            }
        });

        if ($('#WMEPH_tools').length === 0) {
            $('#WMEPH_services').after($('<div id="WMEPH_tools">').css({
                // 'background-color': '#eee',
                color: 'black',
                'font-size': '15px',
                // padding: '0px 4px 4px 4px',
                'margin-left': '6px',
                'margin-right': 'auto'
            }));
        } else {
            $('#WMEPH_tools').empty();
        }
        $('#WMEPH_tools').append(rowDivs);

        // Set up Duplicate onclicks
        if (dupesFound) {
            setupButtons(_dupeBanner);
        }
        // Setup bannButt onclicks
        setupButtons(_buttonBanner);

        // Setup bannButt2 onclicks
        setupButtons(_buttonBanner2);

        // Add click handlers for parking lot helper buttons.
        $('.wmeph-pla-spaces-btn').click(evt => {
            const selectedVenue = getSelectedVenue();
            const selectedValue = $(evt.currentTarget).attr('id').replace('wmeph_', '');
            const existingAttr = selectedVenue.attributes.categoryAttributes.PARKING_LOT;
            const newAttr = {};
            if (existingAttr) {
                Object.keys(existingAttr).forEach(prop => {
                    let value = existingAttr[prop];
                    if (Array.isArray(value)) value = [].concat(value);
                    newAttr[prop] = value;
                });
            }
            newAttr.estimatedNumberOfSpots = selectedValue;
            _UPDATED_FIELDS.parkingSpots.updated = true;
            addUpdateAction(selectedVenue, { categoryAttributes: { PARKING_LOT: newAttr } }, null, true);
        });

        // If pressing enter in the HN entry box, add the HN
        $('#WMEPH-HNAdd').keyup(evt => {
            if (evt.keyCode === 13 && $('#WMEPH-HNAdd').val() !== '') {
                $('#WMEPH_hnMissing').click();
            }
        });

        // If pressing enter in the phone entry box, add the phone
        $('#WMEPH-PhoneAdd').keyup(evt => {
            if (evt.keyCode === 13 && $('#WMEPH-PhoneAdd').val() !== '') {
                $('#WMEPH_phoneMissing').click();
                $('#WMEPH_badAreaCode').click();
            }
        });

        // If pressing enter in the URL entry box, add the URL
        $('#WMEPH-UrlAdd').keyup(evt => {
            if (evt.keyCode === 13 && $('#WMEPH-UrlAdd').val() !== '') {
                $('#WMEPH_urlMissing').click();
            }
        });

        // Format "no hours" section and hook up button events.
        $('#WMEPH_WLnoHours').css({ 'vertical-align': 'top' });

        if (_textEntryValues) {
            _textEntryValues.forEach(entry => $(`#${entry.id}`).val(entry.val));
        }

        // Allow flags to do any additional work (hook up events, etc);
        Object.keys(_buttonBanner)
            .map(key => _buttonBanner[key])
            .filter(flag => flag?.active)
            .forEach(flag => {
                flag.postProcess?.();
            });

        processGoogleLinks(venue);
    } // END assemble Banner function

    async function processGoogleLinks(venue) {
        const promises = venue.attributes.externalProviderIDs.map(link => fetchGoogleLinkInfo(link.attributes.uuid));
        const googleResults = await Promise.all(promises);
        $('#wmeph-google-link-info').remove();
        // Compare to venue to make sure a different place hasn't been selected since the results were requested.
        if (googleResults.length && venue === getSelectedVenue()) {
            const $bannerDiv = $('<div>', { id: 'wmeph-google-link-info' });
            const googleLogoLetter = (letter, colorClass) => $('<span>', { class: 'google-logo' }).addClass(colorClass).text(letter);
            $bannerDiv.append(
                $('<div>', {
                    class: 'banner-row gray',
                    style: 'padding-top: 4px;color: #646464;padding-left: 8px;'
                }).text(' Links').prepend(
                    googleLogoLetter('G', 'blue'),
                    googleLogoLetter('o', 'red'),
                    googleLogoLetter('o', 'orange'),
                    googleLogoLetter('g', 'blue'),
                    googleLogoLetter('l', 'green'),
                    googleLogoLetter('e', 'red')
                ).prepend(
                    $('<i>', {
                        id: 'wmeph-ext-prov-jump',
                        title: 'Jump to external providers section',
                        class: 'fa fa-level-down',
                        style: 'font-size: 15px;float: right;color: cadetblue;cursor: pointer;padding-left: 6px;'
                    })
                )
            );
            venue.attributes.externalProviderIDs.forEach(link => {
                const result = googleResults.find(r => r.uuid === link.attributes.uuid);
                if (result) {
                    const linkStyle = 'margin-left: 5px;text-decoration: none;color: cadetblue;';
                    let $nameSpan;
                    const $row = $('<div>', { class: 'banner-row', style: 'border-top: 1px solid #ccc;' }).append(
                        $('<table>', { style: 'width: 100%' }).append(
                            $('<tbody>').append(
                                $('<tr>').append(
                                    $('<td>').append(
                                        '&bull;',
                                        $nameSpan = $('<span>', {
                                            class:
                                            'wmeph-google-place-name',
                                            style: 'margin-left: 3px;font-weight: normal;'
                                        }).text(`${result.name}`)
                                    ),
                                    $('<td>', { style: 'text-align: right;font-weight: 500;padding: 2px 2px 2px 0px;min-width: 65px;' }).append(
                                        result.website ? [$('<a>', {
                                            style: linkStyle,
                                            href: result.website,
                                            target: '_blank',
                                            title: 'Open the place\'s website, according to Google'
                                        }).append(
                                            $('<i>', {
                                                class: 'fa fa-external-link',
                                                style: 'font-size: 16px;position: relative;top: 1px;'
                                            })
                                        ),
                                        $('<span>', {
                                            style: 'text-align: center;margin-left: 8px;margin-right: 4px;color: #c5c5c5;cursor: default;'
                                        }).text('|')] : null,
                                        $('<a>', {
                                            style: linkStyle,
                                            href: result.url,
                                            target: '_blank',
                                            title: 'Open the place in Google Maps'
                                        }).append(
                                            $('<i>', {
                                                class: 'fa fa-map-o',
                                                style: 'font-size: 16px;'
                                            })
                                        )
                                    )
                                )
                            )
                        )
                    );

                    if (result.business_status === 'CLOSED_PERMANENTLY') {
                        $nameSpan.append(' [CLOSED]');
                        $row.addClass('red');
                        $row.attr('title', 'Google indicates this linked place is permanently closed. Please verify.');
                    } else if (result.business_status === 'CLOSED_TEMPORARILY') {
                        $nameSpan.append(' [TEMPORARILY&nbsp;CLOSED]');
                        $row.addClass('yellow');
                        $row.attr('title', 'Google indicates this linked place is TEMPORARILY closed. Please verify.');
                    } else if (googleResults.find(otherResult => otherResult !== result && otherResult.uuid === result.uuid)) {
                        $nameSpan.append(' [DUPLICATE]');
                        $row.css('background-color', '#fde5c8');
                        $row.attr('title', 'This place is linked more than once. Please remove extra links.');
                    } else {
                        $row.addClass('lightgray');
                    }

                    $bannerDiv.append($row);
                }
            });
            $('#WMEPH_banner').append($bannerDiv);
            $('#wmeph-ext-prov-jump').click(() => {
                const extProvSelector = '#venue-edit-general > div.external-providers-control.form-group';
                document.querySelector('#edit-panel wz-tab.venue-edit-tab-general').isActive = true;
                setTimeout(() => {
                    document.querySelector(extProvSelector).scrollIntoView({ behavior: 'smooth' });
                    setTimeout(() => {
                        $(extProvSelector).addClass('highlight');
                        setTimeout(() => {
                            $(extProvSelector).removeClass('highlight');
                        }, 1500);
                    }, 250);
                }, 0);
            });
        }
    }

    const _googleResults = {};

    function fetchGoogleLinkInfo(uuid) {
        const refreshInterval = 5 * 60 * 1000; // silently refresh data if it's over 5 minutes old
        const staleLimit = 15 * 60 * 1000; // require new data if it's over 15 minutes old
        if (_googleResults.hasOwnProperty(uuid)) {
            const result = _googleResults[uuid];
            const age = Date.now() - result.timestamp;
            if (age < staleLimit) {
                if (age > refreshInterval) {
                    // Refresh the data in the background.
                    fetchGooglePlace(uuid);
                }
                return Promise.resolve(result);
            }
        }
        return fetchGooglePlace(uuid);
    }

    function fetchGooglePlace(uuid) {
        logDev(`fetching ${uuid}`);
        return new Promise(resolve => {
            _placesService.getDetails({
                placeId: uuid,
                fields: ['website', 'business_status', 'url', 'name']
            }, googleResult => {
                googleResult.uuid = uuid;
                googleResult.timestamp = Date.now();
                _googleResults[uuid] = googleResult;
                resolve(googleResult);
            });
        });
    }

    function assembleServicesBanner() {
        const venue = getSelectedVenue();
        if (venue && !$('#WMEPH-HideServicesButtons').prop('checked')) {
            // setup Add Service Buttons for suggested services
            const rowDivs = [];
            if (!venue.isResidential()) {
                const $rowDiv = $('<div>');
                const servButtHeight = '27';
                const buttons = [];
                Object.keys(_servicesBanner).forEach(tempKey => {
                    const rowData = _servicesBanner[tempKey];
                    if (rowData.active) { //  If the particular service is active
                        const $input = $('<input>', {
                            class: rowData.icon,
                            id: `WMEPH_${tempKey}`,
                            type: 'button',
                            title: rowData.title
                        }).css(
                            {
                                border: 0,
                                'background-size': 'contain',
                                height: '27px',
                                width: `${Math.ceil(servButtHeight * rowData.w2hratio).toString()}px`
                            }
                        );
                        buttons.push($input);
                        if (!rowData.checked) {
                            $input.css({ '-webkit-filter': 'opacity(.3)', filter: 'opacity(.3)' });
                        } else {
                            $input.css({ color: 'green' });
                        }
                        $rowDiv.append($input);
                    }
                });
                if ($rowDiv.length) {
                    $rowDiv.prepend('<span class="control-label" title="Verify all Place services before saving">Services (select any that apply):</span><br>');
                }
                rowDivs.push($rowDiv);
            }
            if ($('#WMEPH_services').length === 0) {
                $('#WMEPH_banner').after($('<div id="WMEPH_services">').css({
                    color: 'black',
                    'font-size': '15px',
                    'margin-left': '6px'
                }));
            } else {
                $('#WMEPH_services').empty();
            }
            $('#WMEPH_services').append(rowDivs);

            // Setup bannServ onclicks
            if (!venue.isResidential()) {
                setupButtons(_servicesBanner);
            }
        }
    }

    // Button onclick event handler
    function setupButtons(b) {
        Object.keys(b).forEach(tempKey => { // Loop through the banner possibilities
            if (b[tempKey] && b[tempKey].active) { //  If the particular message is active
                if (b[tempKey].action && b[tempKey].value) { // If there is an action, set onclick
                    buttonAction(b, tempKey);
                }
                if (b[tempKey].action2 && b[tempKey].value2) { // If there is an action2, set onclick
                    buttonAction2(b, tempKey);
                }
                // If there's a WL option, set up onclick
                if (b[tempKey].WLactive && b[tempKey].WLaction) {
                    buttonWhitelist(b, tempKey);
                }
            }
        });
    }

    function buttonAction(b, bKey) {
        const button = document.getElementById(`WMEPH_${bKey}`);
        button.onclick = () => {
            b[bKey].action();
            if (!b[bKey].noBannerAssemble) assembleBanner();
        };
        return button;
    }
    function buttonAction2(b, bKey) {
        const button = document.getElementById(`WMEPH_${bKey}_2`);
        button.onclick = () => {
            b[bKey].action2();
            if (!b[bKey].noBannerAssemble) assembleBanner();
        };
        return button;
    }
    function buttonWhitelist(b, bKey) {
        const button = document.getElementById(`WMEPH_WL${bKey}`);
        button.onclick = () => {
            if (bKey.match(/^\d{5,}/) !== null) {
                b[bKey].WLaction(bKey);
            } else {
                b[bKey].WLaction();
            }
            b[bKey].WLactive = false;
            b[bKey].severity = _SEVERITY.GREEN;
        };
        return button;
    }

    // Helper functions for getting/setting checkbox checked state.
    function isChecked(id) {
        // We could use jquery here, but I assume native is faster.
        return document.getElementById(id).checked;
    }
    function setCheckbox(id, checkedState) {
        if (isChecked(id) !== checkedState) { $(`#${id}`).click(); }
    }
    function setCheckboxes(ids, checkedState) {
        ids.forEach(id => {
            setCheckbox(id, checkedState);
        });
    }

    function onCopyClicked() {
        const venue = getSelectedVenue();
        const attr = venue.attributes;
        _cloneMaster = {};
        _cloneMaster.addr = venue.getAddress();
        if (_cloneMaster.addr.hasOwnProperty('attributes')) {
            _cloneMaster.addr = _cloneMaster.addr.attributes;
        }
        _cloneMaster.houseNumber = attr.houseNumber;
        _cloneMaster.url = attr.url;
        _cloneMaster.phone = attr.phone;
        _cloneMaster.description = attr.description;
        _cloneMaster.services = attr.services;
        _cloneMaster.openingHours = attr.openingHours;
        _cloneMaster.isPLA = venue.isParkingLot();
        logDev('Place Cloned');
    }

    function onPasteClicked() {
        clonePlace();
    }

    function onCheckAllCloneClicked() {
        setCheckboxes(['WMEPH_CPhn', 'WMEPH_CPstr', 'WMEPH_CPcity', 'WMEPH_CPurl', 'WMEPH_CPph', 'WMEPH_CPserv',
            'WMEPH_CPdesc', 'WMEPH_CPhrs'], true);
    }

    function onCheckAddrCloneClicked() {
        setCheckboxes(['WMEPH_CPhn', 'WMEPH_CPstr', 'WMEPH_CPcity'], true);
        setCheckboxes(['WMEPH_CPurl', 'WMEPH_CPph', 'WMEPH_CPserv', 'WMEPH_CPdesc', 'WMEPH_CPhrs'], false);
    }

    function onCheckNoneCloneClicked() {
        setCheckboxes(['WMEPH_CPhn', 'WMEPH_CPstr', 'WMEPH_CPcity', 'WMEPH_CPurl', 'WMEPH_CPph', 'WMEPH_CPserv',
            'WMEPH_CPdesc', 'WMEPH_CPhrs'], false);
    }

    // WMEPH Clone Tool
    function showCloneButton() {
        if (!$('#clonePlace').length) {
            $('#wmeph-run-panel').append(
                $('<div>', { style: 'margin-bottom: 5px' }),
                $('<input>', {
                    class: 'btn btn-warning btn-xs wmeph-btn',
                    id: 'clonePlace',
                    title: 'Copy place info',
                    type: 'button',
                    value: 'Copy',
                    style: 'font-weight: normal'
                }).click(onCopyClicked),
                $('<input>', {
                    class: 'btn btn-warning btn-xs wmeph-btn',
                    id: 'pasteClone',
                    title: 'Apply the Place info. (Ctrl-Alt-O)',
                    type: 'button',
                    value: 'Paste (for checked boxes):',
                    style: 'font-weight: normal; margin-left: 3px;'
                }).click(onPasteClicked),
                '<br>',
                createCloneCheckbox('wmeph-run-panel', 'WMEPH_CPhn', 'HN'),
                createCloneCheckbox('wmeph-run-panel', 'WMEPH_CPstr', 'Str'),
                createCloneCheckbox('wmeph-run-panel', 'WMEPH_CPcity', 'City'),
                createCloneCheckbox('wmeph-run-panel', 'WMEPH_CPurl', 'URL'),
                createCloneCheckbox('wmeph-run-panel', 'WMEPH_CPph', 'Ph'),
                '<br>',
                createCloneCheckbox('wmeph-run-panel', 'WMEPH_CPdesc', 'Desc'),
                createCloneCheckbox('wmeph-run-panel', 'WMEPH_CPserv', 'Serv'),
                createCloneCheckbox('wmeph-run-panel', 'WMEPH_CPhrs', 'Hrs'),
                $('<input>', {
                    class: 'btn btn-info btn-xs wmeph-btn',
                    id: 'checkAllClone',
                    title: 'Check all',
                    type: 'button',
                    value: 'All',
                    style: 'font-weight: normal'
                }).click(onCheckAllCloneClicked),
                $('<input>', {
                    class: 'btn btn-info btn-xs wmeph-btn',
                    id: 'checkAddrClone',
                    title: 'Check address',
                    type: 'button',
                    value: 'Addr',
                    style: 'font-weight: normal; margin-left: 3px;'
                }).click(onCheckAddrCloneClicked),
                $('<input>', {
                    class: 'btn btn-info btn-xs wmeph-btn',
                    id: 'checkNoneClone',
                    title: 'Check none',
                    type: 'button',
                    value: 'None',
                    style: 'font-weight: normal; margin-left: 3px;'
                }).click(onCheckNoneCloneClicked),
                '<br>'
            );
        }
        const venue = getSelectedVenue();
        updateElementEnabledOrVisible($('#pasteClone'), venue?.isApproved() && venue.arePropertiesEditable());
    }

    function onPlugshareSearchClick() {
        const venue = getSelectedVenue();
        const olPoint = venue.attributes.geometry.getCentroid();
        const point = WazeWrap.Geometry.ConvertTo4326(olPoint.x, olPoint.y);
        const url = `https://www.plugshare.com/?latitude=${point.lat}&longitude=${point.lon}&spanLat=.005&spanLng=.005`;
        if ($('#WMEPH-WebSearchNewTab').prop('checked')) {
            window.open(url);
        } else {
            window.open(url, 'WMEPH - PlugShare Search', _searchResultsWindowSpecs);
        }
    }

    function onOpenWebsiteClick() {
        const venue = getSelectedVenue();
        let { url } = venue.attributes;
        if (url.match(/^http/i) === null) {
            url = `http://${url}`;
        }
        try {
            if ($('#WMEPH-WebSearchNewTab').prop('checked')) {
                window.open(url);
            } else {
                window.open(url, _SEARCH_RESULTS_WINDOW_NAME, _searchResultsWindowSpecs);
            }
        } catch (ex) {
            console.error(ex);
            WazeWrap.Alerts.error(_SCRIPT_NAME, 'Possible invalid URL. Check the place\'s Website field.');
        }
    }

    function onGoogleSearchClick() {
        const venue = getSelectedVenue();
        const addr = venue.getAddress();
        if (addr.hasState()) {
            const url = buildGLink(venue.attributes.name, addr, venue.attributes.houseNumber);
            if ($('#WMEPH-WebSearchNewTab').prop('checked')) {
                window.open(url);
            } else {
                window.open(url, _SEARCH_RESULTS_WINDOW_NAME, _searchResultsWindowSpecs);
            }
        } else {
            WazeWrap.Alerts.error(_SCRIPT_NAME, 'The state and country haven\'t been set for this place yet.  Edit the address first.');
        }
    }

    function updateElementEnabledOrVisible($elem, props) {
        if (props.hasOwnProperty('visible')) {
            if (props.visible) {
                $elem.show();
            } else {
                $elem.hide();
            }
        }
        if (props.hasOwnProperty('enabled')) {
            $elem.prop('disabled', !props.enabled);
        }
    }

    // Catch PLs and reloads that have a place selected already and limit attempts to about 10 seconds
    function updateWmephPanel(clearBanner = false) {
        logDev('updateWmephPanel');

        const venue = getSelectedVenue();

        if (!venue) {
            $('#wmeph-panel').remove();
            return;
        }

        if (!venue.isApproved() || !venue.arePropertiesEditable()) {
            clearBanner = true;
        }

        if (clearBanner) {
            $('#WMEPH_banner').remove();
            $('#WMEPH_services').remove();
            $('#WMEPH_tools').remove();
        }

        let $wmephPanel;
        let $wmephRunPanel;
        let $runButton;
        let $websiteButton;
        let $googleSearchButton;
        let $plugshareSearchButton;

        if (!$('#wmeph-panel').length) {
            const devVersSuffix = _IS_BETA_VERSION ? '-β' : '';
            $wmephPanel = $('<div>', { id: 'wmeph-panel' });
            $wmephRunPanel = $('<div>', { id: 'wmeph-run-panel' });
            $runButton = $('<input>', {
                class: 'btn btn-primary wmeph-fat-btn',
                id: 'runWMEPH',
                title: `Run WMEPH${devVersSuffix} on Place`,
                type: 'button',
                value: `Run WMEPH${devVersSuffix}`
            }).click(harmonizePlace);
            $websiteButton = $('<input>', {
                class: 'btn btn-success btn-xs wmeph-fat-btn',
                id: 'WMEPHurl',
                title: 'Open place URL',
                type: 'button',
                value: 'Website'
            }).click(onOpenWebsiteClick);
            $googleSearchButton = $('<input>', {
                class: 'btn btn-danger btn-xs wmeph-fat-btn',
                id: 'wmephSearch',
                title: 'Search the web for this place.  Do not copy info from 3rd party sources!',
                type: 'button',
                value: 'Google'
            }).click(onGoogleSearchClick);
            $plugshareSearchButton = $('<input>', {
                class: 'btn btn-xs btn-danger wmeph-fat-btn',
                id: 'wmephPlugShareSearch',
                title: 'Open PlugShare website',
                type: 'button',
                value: 'PS',
                style: 'background-color: #003ca6; box-shadow:0 2px 0 #5075b9;'
            }).click(onPlugshareSearchClick);

            $('#edit-panel > .contents').prepend(
                $wmephPanel.append(
                    $wmephRunPanel.append(
                        $runButton,
                        $websiteButton,
                        $googleSearchButton,
                        $plugshareSearchButton
                    )
                )
            );
        } else {
            $wmephPanel = $('#wmeph-panel');
            $wmephRunPanel = $('#wmeph-run-panel');
            $runButton = $('#runWMEPH');
            $websiteButton = $('#WMEPHurl');
            $googleSearchButton = $('#wmephSearch');
            $plugshareSearchButton = $('#wmephPlugShareSearch');
        }

        updateElementEnabledOrVisible($runButton, { enabled: venue.isApproved() && venue.arePropertiesEditable() });
        updateElementEnabledOrVisible($websiteButton, { enabled: venue.attributes.url?.trim().length, visible: !venue.isResidential() });
        updateElementEnabledOrVisible($googleSearchButton, { enabled: !venue.isResidential(), visible: !venue.isResidential() });
        updateElementEnabledOrVisible($plugshareSearchButton, { visible: venue.isChargingStation() });

        if (localStorage.getItem('WMEPH-EnableCloneMode') === '1') {
            showCloneButton();
        }
        // If the user selects a place in the dupe list, don't clear the labels yet
        if (_dupeIDList.includes(venue.attributes.id)) {
            destroyDupeLabels();
        }
    }

    // Find field divs
    // function getPanelFields() {
    //     let panelFieldsList = $('.form-control');
    //     for (let pfix = 0; pfix < panelFieldsList.length; pfix++) {
    //         const pfa = panelFieldsList[pfix].name;
    //         if (pfa === 'name') {
    //             _PANEL_FIELDS.name = pfix;
    //         }
    //         if (pfa === 'lockRank') {
    //             _PANEL_FIELDS.lockRank = pfix;
    //         }
    //         if (pfa === 'description') {
    //             _PANEL_FIELDS.description = pfix;
    //         }
    //         if (pfa === 'url') {
    //             _PANEL_FIELDS.url = pfix;
    //         }
    //         if (pfa === 'phone') {
    //             _PANEL_FIELDS.phone = pfix;
    //         }
    //         if (pfa === 'brand') {
    //             _PANEL_FIELDS.brand = pfix;
    //         }
    //     }
    //     const placeNavTabs = $('.nav');
    //     for (let pfix = 0; pfix < placeNavTabs.length; pfix++) {
    //         const pfa = placeNavTabs[pfix].innerHTML;
    //         if (pfa.includes('venue-edit')) {
    //             panelFieldsList = placeNavTabs[pfix].children;
    //             _PANEL_FIELDS.navTabsIX = pfix;
    //             break;
    //         }
    //     }
    //     for (let pfix = 0; pfix < panelFieldsList.length; pfix++) {
    //         const pfa = panelFieldsList[pfix].innerHTML;
    //         if (pfa.includes('venue-edit-general')) {
    //             _PANEL_FIELDS.navTabGeneral = pfix;
    //         }
    //         if (pfa.includes('venue-edit-more')) {
    //             _PANEL_FIELDS.navTabMore = pfix;
    //         }
    //     }
    // }

    // Function to clone info from a place
    function clonePlace() {
        log('Cloning info...');
        if (_cloneMaster !== null && _cloneMaster.hasOwnProperty('url')) {
            const venue = getSelectedVenue();
            const cloneItems = {};
            let updateItem = false;
            if (isChecked('WMEPH_CPurl')) {
                cloneItems.url = _cloneMaster.url;
                updateItem = true;
            }
            if (isChecked('WMEPH_CPph')) {
                cloneItems.phone = _cloneMaster.phone;
                updateItem = true;
            }
            if (isChecked('WMEPH_CPdesc')) {
                cloneItems.description = _cloneMaster.description;
                updateItem = true;
            }
            if (isChecked('WMEPH_CPserv') && venue.isParkingLot() === _cloneMaster.isPLA) {
                cloneItems.services = _cloneMaster.services;
                updateItem = true;
            }
            if (isChecked('WMEPH_CPhrs')) {
                cloneItems.openingHours = _cloneMaster.openingHours;
                updateItem = true;
            }
            if (updateItem) {
                addUpdateAction(new UpdateObject(venue, cloneItems));
                logDev('Item details cloned');
            }

            const copyStreet = isChecked('WMEPH_CPstr');
            const copyCity = isChecked('WMEPH_CPcity');
            const copyHn = isChecked('WMEPH_CPhn');

            if (copyStreet || copyCity || copyHn) {
                const originalAddress = venue.getAddress();
                const itemRepl = {
                    street: copyStreet ? _cloneMaster.addr.street : originalAddress.attributes.street,
                    city: copyCity ? _cloneMaster.addr.city : originalAddress.attributes.city,
                    state: copyCity ? _cloneMaster.addr.state : originalAddress.attributes.state,
                    country: copyCity ? _cloneMaster.addr.country : originalAddress.attributes.country,
                    houseNumber: copyHn ? _cloneMaster.addr.houseNumber : originalAddress.attributes.houseNumber
                };
                updateAddress(venue, itemRepl);
                logDev('Item address cloned');
            }
        } else {
            log('Please copy a place');
        }
    }

    // Formats "hour object" into a string.
    function formatOpeningHour(hourEntry) {
        const dayNames = ['Sunday', 'Monday', 'Tuesday', 'Wednesday', 'Thursday', 'Friday', 'Saturday'];
        const hours = `${hourEntry.fromHour}-${hourEntry.toHour}`;
        return hourEntry.days.map(day => `${dayNames[day]} ${hours}`).join(', ');
    }
    // Pull natural text from opening hours
    function getOpeningHours(venue) {
        return venue && venue.attributes.openingHours && venue.attributes.openingHours.map(formatOpeningHour);
    }

    function venueHasOverlappingHours(venue) {
        const hours = venue.attributes.openingHours;
        if (hours.length < 2) {
            return false;
        }

        for (let day2Ch = 0; day2Ch < 7; day2Ch++) { // Go thru each day of the week
            const daysObj = [];
            for (let hourSet = 0; hourSet < hours.length; hourSet++) { // For each set of hours
                if (hours[hourSet].days.includes(day2Ch)) { // pull out hours that are for the current day, add 2400 if it goes past midnight, and store
                    const fromHourTemp = hours[hourSet].fromHour.replace(/:/g, '');
                    let toHourTemp = hours[hourSet].toHour.replace(/:/g, '');
                    if (toHourTemp <= fromHourTemp) {
                        toHourTemp = parseInt(toHourTemp, 10) + 2400;
                    }
                    daysObj.push([fromHourTemp, toHourTemp]);
                }
            }
            if (daysObj.length > 1) { // If there's multiple hours for the day, check them for overlap
                for (let hourSetCheck2 = 1; hourSetCheck2 < daysObj.length; hourSetCheck2++) {
                    for (let hourSetCheck1 = 0; hourSetCheck1 < hourSetCheck2; hourSetCheck1++) {
                        if (daysObj[hourSetCheck2][0] > daysObj[hourSetCheck1][0] && daysObj[hourSetCheck2][0] < daysObj[hourSetCheck1][1]) {
                            return true;
                        }
                        if (daysObj[hourSetCheck2][1] > daysObj[hourSetCheck1][0] && daysObj[hourSetCheck2][1] < daysObj[hourSetCheck1][1]) {
                            return true;
                        }
                    }
                }
            }
        }
        return false;
    }

    // Duplicate place finder  ###bmtg
    function findNearbyDuplicate(selectedVenueName, selectedVenueAliases, selectedVenue, recenterOption) {
        // Helper function to prep a name for comparisons.
        const formatName = name => name.toUpperCase().replace(/ AND /g, '').replace(/^THE /g, '').replace(/[^A-Z0-9]/g, '');

        // Remove any previous search labels
        _dupeLayer.destroyFeatures();

        const mapExtent = W.map.getExtent();
        const padFrac = 0.15; // how much to pad the zoomed window

        // generic terms to skip if it's all that remains after stripping numbers
        let noNumSkip = 'BANK|ATM|HOTEL|MOTEL|STORE|MARKET|SUPERMARKET|GYM|GAS|GASOLINE|GASSTATION|CAFE|OFFICE|OFFICES'
            + '|CARRENTAL|RENTALCAR|RENTAL|SALON|BAR|BUILDING|LOT';
        noNumSkip = `${noNumSkip}|${_COLLEGE_ABBREVIATIONS}`.split('|');
        const allowedTwoLetters = ['BP', 'DQ', 'BK', 'BW', 'LQ', 'QT', 'DB', 'PO'];

        // Make the padded extent
        mapExtent.left += padFrac * (mapExtent.right - mapExtent.left);
        mapExtent.right -= padFrac * (mapExtent.right - mapExtent.left);
        mapExtent.bottom += padFrac * (mapExtent.top - mapExtent.bottom);
        mapExtent.top -= padFrac * (mapExtent.top - mapExtent.bottom);
        let outOfExtent = false;
        let overlappingFlag = false;

        // Initialize the coordinate extents for duplicates
        const selectedCentroid = selectedVenue.geometry.getCentroid();
        let minLon = selectedCentroid.x;
        let minLat = selectedCentroid.y;
        let maxLon = minLon;
        let maxLat = minLat;

        // Label stuff for display
        const labelFeatures = [];
        const dupeNames = [];
        let labelColorIX = 0;
        const labelColorList = ['#3F3'];

        // Name formatting for the WME place name
        const selectedVenueNameRF = formatName(selectedVenueName);
        let currNameList = [];
        if (selectedVenueNameRF.length > 2 || allowedTwoLetters.includes(selectedVenueNameRF)) {
            currNameList.push(selectedVenueNameRF);
        } else {
            currNameList.push('PRIMNAMETOOSHORT_PJZWX');
        }

        const selectedVenueAttr = selectedVenue.attributes;

        // Clear non-letter characters for alternate match ( HOLLYIVYPUB23 --> HOLLYIVYPUB )
        const itemNameNoNum = selectedVenueNameRF.replace(/[^A-Z]/g, '');
        if (((itemNameNoNum.length > 2 && !noNumSkip.includes(itemNameNoNum)) || allowedTwoLetters.includes(itemNameNoNum))
            && !selectedVenueAttr.categories.includes('PARKING_LOT')) {
            // only add de-numbered name if anything remains
            currNameList.push(itemNameNoNum);
        }

        if (selectedVenueAliases.length > 0) {
            for (let aliix = 0; aliix < selectedVenueAliases.length; aliix++) {
                // Format name
                const aliasNameRF = formatName(selectedVenueAliases[aliix]);
                if ((aliasNameRF.length > 2 && !noNumSkip.includes(aliasNameRF)) || allowedTwoLetters.includes(aliasNameRF)) {
                    // only add de-numbered name if anything remains
                    currNameList.push(aliasNameRF);
                }
                // Clear non-letter characters for alternate match ( HOLLYIVYPUB23 --> HOLLYIVYPUB )
                const aliasNameNoNum = aliasNameRF.replace(/[^A-Z]/g, '');
                if (((aliasNameNoNum.length > 2 && !noNumSkip.includes(aliasNameNoNum)) || allowedTwoLetters.includes(aliasNameNoNum))
                    && !selectedVenueAttr.categories.includes('PARKING_LOT')) {
                    // only add de-numbered name if anything remains
                    currNameList.push(aliasNameNoNum);
                }
            }
        }
        currNameList = _.uniq(currNameList); //  remove duplicates

        let selectedVenueAddr = selectedVenue.getAddress();
        selectedVenueAddr = selectedVenueAddr.attributes || selectedVenueAddr;
        const selectedVenueHN = selectedVenueAttr.houseNumber;

        const selectedVenueAddrIsComplete = selectedVenueAddr.street !== null && selectedVenueAddr.street.name !== null
            && selectedVenueHN && selectedVenueHN.match(/\d/g) !== null;

        const venues = W.model.venues.getObjectArray();
        const selectedVenueId = selectedVenueAttr.id;

        _dupeIDList = [selectedVenueId];
        _dupeHNRangeList = [];
        _dupeHNRangeDistList = [];

        // Get the list of dupes that have been whitelisted.
        const selectedVenueWL = _venueWhitelist[selectedVenueId];
        const whitelistedDupes = selectedVenueWL && selectedVenueWL.dupeWL ? selectedVenueWL.dupeWL : [];

        const excludePLADupes = $('#WMEPH-ExcludePLADupes').prop('checked');
        let randInt = 100;
        // For each place on the map:
        venues.forEach(testVenue => {
            if ((!excludePLADupes || (excludePLADupes && !(selectedVenue.isParkingLot() || testVenue.isParkingLot())))
                && !isEmergencyRoom(testVenue)) {
                const testVenueAttr = testVenue.attributes;
                const testVenueId = testVenueAttr.id;

                // Check for overlapping PP's
                const testCentroid = testVenue.geometry.getCentroid();
                const pt2ptDistance = selectedCentroid.distanceTo(testCentroid);
                if (selectedVenue.isPoint() && testVenue.isPoint() && pt2ptDistance < 2 && selectedVenueId !== testVenueId) {
                    overlappingFlag = true;
                }

                const testVenueHN = testVenueAttr.houseNumber;
                let testVenueAddr = testVenue.getAddress();
                testVenueAddr = testVenueAddr.attributes || testVenueAddr;

                // get HNs for places on same street
                if (selectedVenueAddrIsComplete && testVenueAddr.street !== null && testVenueAddr.street.name !== null
                    && testVenueHN && testVenueHN !== '' && testVenueId !== selectedVenueId
                    && selectedVenueAddr.street.name === testVenueAddr.street.name && testVenueHN < 1000000) {
                    _dupeHNRangeList.push(parseInt(testVenueHN, 10));
                    _dupeHNRangeDistList.push(pt2ptDistance);
                }

                // Check for duplicates
                // don't do res, the point itself, new points or no name
                if (!whitelistedDupes.includes(testVenueId) && _dupeIDList.length < 6 && pt2ptDistance < 800
                    && !testVenue.isResidential() && testVenueId !== selectedVenueId && !testVenue.isNew()
                    && testVenueAttr.name !== null && testVenueAttr.name.length > 1) {
                    // If item has a complete address and test venue does, and they are different, then no dupe
                    let suppressMatch = false;
                    if (selectedVenueAddrIsComplete && testVenueAddr.street !== null && testVenueAddr.street.name !== null
                        && testVenueHN && testVenueHN.match(/\d/g) !== null) {
                        if (selectedVenueAttr.lockRank > 0 && testVenueAttr.lockRank > 0) {
                            if (selectedVenueAttr.houseNumber !== testVenueHN
                                || selectedVenueAddr.street.name !== testVenueAddr.street.name) {
                                suppressMatch = true;
                            }
                        } else if (selectedVenueHN !== testVenueHN
                            && selectedVenueAddr.street.name !== testVenueAddr.street.name) {
                            suppressMatch = true;
                        }
                    }

                    if (!suppressMatch) {
                        let testNameList;
                        // Reformat the testPlace name
                        const strippedTestName = formatName(testVenueAttr.name)
                            .replace(/\s+[-(].*$/, ''); // Remove localization text
                        if ((strippedTestName.length > 2 && !noNumSkip.includes(strippedTestName))
                            || allowedTwoLetters.includes(strippedTestName)) {
                            testNameList = [strippedTestName];
                        } else {
                            testNameList = [`TESTNAMETOOSHORTQZJXS${randInt}`];
                            randInt++;
                        }

                        const testNameNoNum = strippedTestName.replace(/[^A-Z]/g, ''); // Clear non-letter characters for alternate match
                        if (((testNameNoNum.length > 2 && !noNumSkip.includes(testNameNoNum)) || allowedTwoLetters.includes(testNameNoNum))
                            && !testVenueAttr.categories.includes('PARKING_LOT')) { //  only add de-numbered name if at least 2 chars remain
                            testNameList.push(testNameNoNum);
                        }

                        // primary name matching loop
                        let nameMatch = false;
                        for (let tnlix = 0; tnlix < testNameList.length; tnlix++) {
                            for (let cnlix = 0; cnlix < currNameList.length; cnlix++) {
                                if ((testNameList[tnlix].includes(currNameList[cnlix]) || currNameList[cnlix].includes(testNameList[tnlix]))) {
                                    nameMatch = true;
                                    break;
                                }
                            }
                            if (nameMatch) { break; } // break if a match found
                        }

                        let altNameMatch = -1;
                        if (!nameMatch && testVenueAttr.aliases.length > 0) {
                            for (let aliix = 0; aliix < testVenueAttr.aliases.length; aliix++) {
                                const aliasNameRF = formatName(testVenueAttr.aliases[aliix]);
                                if ((aliasNameRF.length > 2 && !noNumSkip.includes(aliasNameRF)) || allowedTwoLetters.includes(aliasNameRF)) {
                                    testNameList = [aliasNameRF];
                                } else {
                                    testNameList = [`ALIASNAMETOOSHORTQOFUH${randInt}`];
                                    randInt++;
                                }
                                const aliasNameNoNum = aliasNameRF.replace(/[^A-Z]/g, ''); // Clear non-letter characters for alternate match ( HOLLYIVYPUB23 --> HOLLYIVYPUB )
                                if (((aliasNameNoNum.length > 2 && !noNumSkip.includes(aliasNameNoNum)) || allowedTwoLetters.includes(aliasNameNoNum))
                                    && !testVenueAttr.categories.includes('PARKING_LOT')) { //  only add de-numbered name if at least 2 characters remain
                                    testNameList.push(aliasNameNoNum);
                                } else {
                                    testNameList.push(`111231643239${randInt}`); //  just to keep track of the alias in question, always add something.
                                    randInt++;
                                }
                            }
                            for (let tnlix = 0; tnlix < testNameList.length; tnlix++) {
                                for (let cnlix = 0; cnlix < currNameList.length; cnlix++) {
                                    if ((testNameList[tnlix].includes(currNameList[cnlix]) || currNameList[cnlix].includes(testNameList[tnlix]))) {
                                        // get index of that match (half of the array index with floor)
                                        altNameMatch = Math.floor(tnlix / 2);
                                        break;
                                    }
                                }
                                if (altNameMatch > -1) { break; } // break from the rest of the alts if a match found
                            }
                        }
                        // If a match was found:
                        if (nameMatch || altNameMatch > -1) {
                            _dupeIDList.push(testVenueAttr.id); // Add the item to the list of matches
                            _dupeLayer.setVisibility(true); // If anything found, make visible the dupe layer

                            const labelText = nameMatch ? testVenueAttr.name : `${testVenueAttr.aliases[altNameMatch]} (Alt)`;
                            logDev(`Possible duplicate found. WME place: ${selectedVenueName} / Nearby place: ${labelText}`);

                            // Reformat the name into multiple lines based on length
                            const labelTextBuild = [];
                            let maxLettersPerLine = Math.round(2 * Math.sqrt(labelText.replace(/ /g, '').length / 2));
                            maxLettersPerLine = Math.max(maxLettersPerLine, 4);
                            let startIX = 0;
                            let endIX = 0;
                            while (endIX !== -1) {
                                endIX = labelText.indexOf(' ', endIX + 1);
                                if (endIX - startIX > maxLettersPerLine) {
                                    labelTextBuild.push(labelText.substr(startIX, endIX - startIX));
                                    startIX = endIX + 1;
                                }
                            }
                            labelTextBuild.push(labelText.substr(startIX)); // Add last line
                            let labelTextReformat = labelTextBuild.join('\n');
                            // Add photo icons
                            if (testVenueAttr.images.length) {
                                labelTextReformat = `${labelTextReformat} `;
                                for (let phix = 0; phix < testVenueAttr.images.length; phix++) {
                                    if (phix === 3) {
                                        labelTextReformat = `${labelTextReformat}+`;
                                        break;
                                    }
                                    labelTextReformat = `${labelTextReformat}\u25A3`; // add photo icons
                                }
                            }

                            const lonLat = getVenueLonLat(testVenue);
                            if (!mapExtent.containsLonLat(lonLat)) {
                                outOfExtent = true;
                            }
                            minLat = Math.min(minLat, lonLat.lat);
                            minLon = Math.min(minLon, lonLat.lon);
                            maxLat = Math.max(maxLat, lonLat.lat);
                            maxLon = Math.max(maxLon, lonLat.lon);

                            labelFeatures.push(new OpenLayers.Feature.Vector(
                                testCentroid,
                                {
                                    labelText: labelTextReformat,
                                    fontColor: '#fff',
                                    strokeColor: labelColorList[labelColorIX % labelColorList.length],
                                    labelAlign: 'cm',
                                    pointRadius: 25,
                                    dupeID: testVenueId
                                }
                            ));
                            dupeNames.push(labelText);
                        }
                        labelColorIX++;
                    }
                }
            }
        });

        // Add a marker for the working place point if any dupes were found
        if (_dupeIDList.length > 1) {
            const lonLat = getVenueLonLat(selectedVenue);
            if (!mapExtent.containsLonLat(lonLat)) {
                outOfExtent = true;
            }
            minLat = Math.min(minLat, lonLat.lat);
            minLon = Math.min(minLon, lonLat.lon);
            maxLat = Math.max(maxLat, lonLat.lat);
            maxLon = Math.max(maxLon, lonLat.lon);
            // Add photo icons
            let currentLabel = 'Current';
            if (selectedVenueAttr.images.length > 0) {
                for (let ciix = 0; ciix < selectedVenueAttr.images.length; ciix++) {
                    currentLabel = `${currentLabel} `;
                    if (ciix === 3) {
                        currentLabel = `${currentLabel}+`;
                        break;
                    }
                    currentLabel = `${currentLabel}\u25A3`; // add photo icons
                }
            }
            labelFeatures.push(new OpenLayers.Feature.Vector(
                selectedCentroid,
                {
                    labelText: currentLabel,
                    fontColor: '#fff',
                    strokeColor: '#fff',
                    labelAlign: 'cm',
                    pointRadius: 25,
                    dupeID: selectedVenueId
                }
            ));
            _dupeLayer.addFeatures(labelFeatures);
        }

        if (recenterOption && dupeNames.length > 0 && outOfExtent) { // then rebuild the extent to include the duplicate
            const padMult = 1.0;
            mapExtent.left = minLon - (padFrac * padMult) * (maxLon - minLon);
            mapExtent.right = maxLon + (padFrac * padMult) * (maxLon - minLon);
            mapExtent.bottom = minLat - (padFrac * padMult) * (maxLat - minLat);
            mapExtent.top = maxLat + (padFrac * padMult) * (maxLat - minLat);
            W.map.getOLMap().zoomToExtent(mapExtent);
        }
        return [dupeNames, overlappingFlag];
    } // END findNearbyDuplicate function

    // Functions to infer address from nearby segments
    function inferAddress(venue, maxRecursionDepth) {
        let distanceToSegment;
        let foundAddresses = [];
        let i;
        // Ignore pedestrian boardwalk, stairways, runways, and railroads
        const IGNORE_ROAD_TYPES = [10, 16, 18, 19];
        let inferredAddress = {
            country: null,
            city: null,
            state: null,
            street: null
        };
        let n;
        let orderedSegments = [];
        const segments = W.model.segments.getObjectArray();
        let stopPoint;

        // Make sure a place is selected and segments are loaded.
        if (!(venue && segments.length)) {
            return undefined;
        }

        const getFCRank = FC => {
            const typeToFCRank = {
                3: 0, // freeway
                6: 1, // major
                7: 2, // minor
                2: 3, // primary
                1: 4, // street
                20: 5, // PLR
                8: 6 // dirt
            };
            return typeToFCRank[FC] || 100;
        };

        const hasStreetName = segment => {
            if (!segment || segment.type !== 'segment') return false;
            const addr = segment.getAddress();
            return !(addr.isEmpty() || addr.isEmptyStreet());
        };

        const findClosestNode = () => {
            const closestSegment = orderedSegments[0].segment;
            let distanceA;
            let distanceB;
            const nodeA = W.model.nodes.getObjectById(closestSegment.attributes.fromNodeID);
            const nodeB = W.model.nodes.getObjectById(closestSegment.attributes.toNodeID);
            if (nodeA && nodeB) {
                const pt = stopPoint.getPoint ? stopPoint.getPoint() : stopPoint;
                distanceA = pt.distanceTo(nodeA.attributes.geometry);
                distanceB = pt.distanceTo(nodeB.attributes.geometry);
                return distanceA < distanceB ? nodeA.attributes.id : nodeB.attributes.id;
            }
            return undefined;
        };

        const findConnections = (startingNodeID, recursionDepth) => {
            let newNode;

            // Limit search depth to avoid problems.
            if (recursionDepth > maxRecursionDepth) {
                return;
            }

            // Populate variable with segments connected to starting node.
            const connectedSegments = orderedSegments.filter(seg => [seg.fromNodeID, seg.toNodeID].includes(startingNodeID));

            // Check connected segments for address info.
            const keys = Object.keys(connectedSegments);
            for (let idx = 0; idx < keys.length; idx++) {
                const k = keys[idx];
                if (hasStreetName(connectedSegments[k].segment)) {
                    // Address found, push to array.
                    foundAddresses.push({
                        depth: recursionDepth,
                        distance: connectedSegments[k].distance,
                        segment: connectedSegments[k].segment
                    });
                    break;
                } else {
                    // If not found, call function again starting from the other node on this segment.
                    const attr = connectedSegments[k].segment.attributes;
                    newNode = attr.fromNodeID === startingNodeID ? attr.toNodeID : attr.fromNodeID;
                    findConnections(newNode, recursionDepth + 1);
                }
            }
        };

        const { entryExitPoints } = venue.attributes;
        if (entryExitPoints.length) {
            // Get the primary stop point, if one exists.  If none, get the first point.
            stopPoint = entryExitPoints.find(pt => pt.isPrimary()) || entryExitPoints[0];
        } else {
            // If no stop points, just use the venue's centroid.
            stopPoint = venue.geometry.getCentroid();
        }

        // Go through segment array and calculate distances to segments.
        for (i = 0, n = segments.length; i < n; i++) {
            // Make sure the segment is not an ignored roadType.
            if (!IGNORE_ROAD_TYPES.includes(segments[i].attributes.roadType)) {
                distanceToSegment = (stopPoint.getPoint ? stopPoint.getPoint() : stopPoint).distanceTo(segments[i].geometry);
                // Add segment object and its distanceTo to an array.
                orderedSegments.push({
                    distance: distanceToSegment,
                    fromNodeID: segments[i].attributes.fromNodeID,
                    segment: segments[i],
                    toNodeID: segments[i].attributes.toNodeID
                });
            }
        }

        // Sort the array with segments and distance.
        orderedSegments = _.sortBy(orderedSegments, 'distance');

        // Check closest segment for address first.
        if (hasStreetName(orderedSegments[0].segment)) {
            inferredAddress = orderedSegments[0].segment.getAddress();
        } else {
            // If address not found on closest segment, try to find address through branching method.
            findConnections(findClosestNode(), 1);
            if (foundAddresses.length > 0) {
                // If more than one address found at same recursion depth, look at FC of segments.
                if (foundAddresses.length > 1) {
                    foundAddresses.forEach(element => {
                        element.fcRank = getFCRank(element.segment.attributes.roadType);
                    });
                    foundAddresses = _.sortBy(foundAddresses, 'fcRank');
                    foundAddresses = _.filter(foundAddresses, {
                        fcRank: foundAddresses[0].fcRank
                    });
                }

                // If multiple segments with same FC, Use address from segment with address that is closest by connectivity.
                if (foundAddresses.length > 1) {
                    foundAddresses = _.sortBy(foundAddresses, 'depth');
                    foundAddresses = _.filter(foundAddresses, {
                        depth: foundAddresses[0].depth
                    });
                }

                // If more than one of the closest segments by connectivity has the same FC, look for
                // closest segment geometrically.
                if (foundAddresses.length > 1) {
                    foundAddresses = _.sortBy(foundAddresses, 'distance');
                }
                console.debug(foundAddresses[0].streetName, foundAddresses[0].depth);
                inferredAddress = foundAddresses[0].segment.getAddress();
            } else {
                // Default to closest if branching method fails.
                // Go through sorted segment array until a country, state, and city have been found.
                const closestElem = orderedSegments.find(element => hasStreetName(element.segment));
                inferredAddress = closestElem ? closestElem.segment.getAddress() || inferredAddress : inferredAddress;
            }
        }
        return inferredAddress;
    } // END inferAddress function

    /**
     * Updates the address for a place.
     * @param feature {WME Venue Object} The place to update.
     * @param address {Object} An object containing the country, state, city, and street
     * @param actions {Array of actions} Optional. If performing multiple actions at once.
     * objects.
     */
    function updateAddress(feature, address, actions) {
        let newAttributes;
        if (feature && address) {
            newAttributes = {
                countryID: address.country.id,
                stateID: address.state.id,
                cityName: address.city.attributes.name,
                emptyCity: address.city.hasName() ? null : true,
                streetName: address.street.name,
                emptyStreet: address.street.isEmpty ? true : null
            };
            const multiAction = new MultiAction([], { description: 'Update venue address' });
            multiAction.setModel(W.model);
            multiAction.doSubAction(new UpdateFeatureAddress(feature, newAttributes));
            if (address.hasOwnProperty('houseNumber')) {
                multiAction.doSubAction(new UpdateObject(feature, { houseNumber: address.houseNumber }));
            }
            if (actions) {
                actions.push(multiAction);
            } else {
                W.model.actionManager.add(multiAction);
            }
            logDev('Address inferred and updated');
        }
    }

    // Build a Google search url based on place name and address
    function buildGLink(searchName, addr, HN) {
        let searchHN = '';
        let searchStreet = '';
        let searchCity = '';
        searchName = searchName.replace(/\//g, ' ');
        if (!addr.isEmptyStreet()) {
            searchStreet = `${addr.getStreetName()}, `
                .replace(/CR-/g, 'County Rd ')
                .replace(/SR-/g, 'State Hwy ')
                .replace(/US-/g, 'US Hwy ')
                .replace(/ CR /g, ' County Rd ')
                .replace(/ SR /g, ' State Hwy ')
                .replace(/ US /g, ' US Hwy ')
                .replace(/$CR /g, 'County Rd ')
                .replace(/$SR /g, 'State Hwy ')
                .replace(/$US /g, 'US Hwy ');
            if (HN && searchStreet !== '') {
                searchHN = `${HN} `;
            }
        }
        const city = addr.getCity();
        if (city && !city.isEmpty()) {
            searchCity = `${city.getName()}, `;
        }

        searchName = searchName + (searchName ? ', ' : '') + searchHN + searchStreet
            + searchCity + addr.getStateName();
        return `http://www.google.com/search?q=${encodeURIComponent(searchName)}`;
    }

    // WME Category translation from Natural language to object language  (Bank / Financial --> BANK_FINANCIAL)
    function catTranslate(natCategories) {
        const catNameUpper = natCategories.trim().toUpperCase();
        if (_CATEGORY_LOOKUP.hasOwnProperty(catNameUpper)) {
            return _CATEGORY_LOOKUP[catNameUpper];
        }

        // if the category doesn't translate, then pop an alert that will make a forum post to the thread
        // Generally this means the category used in the PNH sheet is not close enough to the natural language categories used inside the WME translations
        /* if (confirm('WMEPH: Category Error!\nClick OK to report this error')) {
            reportError({
                subject: 'WMEPH Bug report: no tns',
                message: `Error report: Category "${natCategories}" was not found in the PNH categories sheet.`
            });
        } */
        WazeWrap.Alerts.confirm(
            _SCRIPT_NAME,
            'WMEPH: Category Error!<br>Click OK to report this error',
            () => {
                reportError({
                    subject: 'WMEPH Bug report: no tns',
                    message: `Error report: Category "${natCategories}" was not found in the PNH categories sheet.`
                });
            },
            () => { }
        );
        return 'ERROR';
    } // END catTranslate function

    // compares two arrays to see if equal, regardless of order
    function matchSets(array1, array2) {
        if (array1.length !== array2.length) { return false; } // compare lengths
        for (let i = 0; i < array1.length; i++) {
            if (!array2.includes(array1[i])) {
                return false;
            }
        }
        return true;
    }

    // function that checks if all elements of target are in array:source
    function containsAll(source, target) {
        if (typeof (target) === 'string') { target = [target]; } // if a single string, convert to an array
        for (let ixx = 0; ixx < target.length; ixx++) {
            if (!source.includes(target[ixx])) {
                return false;
            }
        }
        return true;
    }

    // function that checks if any element of target are in source
    function containsAny(source, target) {
        if (typeof source === 'string') { source = [source]; } // if a single string, convert to an array
        if (typeof target === 'string') { target = [target]; } // if a single string, convert to an array
        return source.some(tt => target.includes(tt));
    }

    /**
     * Copies an array, inserts an item or array of items at a specified index, and removes any duplicates.
     * Can be used to move the position of an item in an array.
     *
     * @param {Array} sourceArray Original array. This array is not modified.
     * @param {*} toInsert Item or array of items to insert.
     * @param {Number} atIndex The index to insert at.
     * @return {Array} An array with the new item(s) inserted.
     */
    function insertAtIndex(sourceArray, toInsert, atIndex) {
        const sourceCopy = sourceArray.slice();
        if (!Array.isArray(toInsert)) toInsert = [toInsert];
        sourceCopy.splice(atIndex, 0, ...toInsert);
        return _.uniq(sourceCopy);
    }

    function arraysAreEqual(array1, array2) {
        return array1.legth === array2.length && array1.every((item, index) => item === array2[index]);
    }

    // Function to remove unnecessary aliases
    function removeSFAliases(nName, nAliases) {
        const newAliasesUpdate = [];
        nName = nName.toUpperCase().replace(/'/g, '').replace(/-/g, ' ').replace(/\/ /g, ' ').replace(/ \//g, ' ').replace(/ {2,}/g, ' ');
        for (let naix = 0; naix < nAliases.length; naix++) {
            if (!nName.startsWith(nAliases[naix].toUpperCase().replace(/'/g, '').replace(/-/g, ' ').replace(/\/ /g, ' ').replace(/ \//g, ' ').replace(/ {2,}/g, ' '))) {
                newAliasesUpdate.push(nAliases[naix]);
            } else {
                _buttonBanner.sfAliases = new Flag.SFAliases();
            }
        }
        return newAliasesUpdate;
    }

    // used for phone reformatting
    function phoneFormat(format, ...rest) {
        return format.replace(/{(\d+)}/g, (name, number) => (typeof rest[number] !== 'undefined' ? rest[number] : null));
    }

    function initSettingsCheckbox(settingID) {
        // Associate click event of new checkbox to call saveSettingToLocalStorage with proper ID
        $(`#${settingID}`).click(() => { saveSettingToLocalStorage(settingID); });
        // Load Setting for Local Storage, if it doesn't exist set it to NOT checked.
        // If previously set to 1, then trigger "click" event.
        if (!localStorage.getItem(settingID)) {
            // logDev(settingID + ' not found.');
        } else if (localStorage.getItem(settingID) === '1') {
            $(`#${settingID}`).prop('checked', true);
        }
    }

    // This routine will create a checkbox in the #PlaceHarmonizer tab and will load the setting
    //        settingID:  The #id of the checkbox being created.
    //  textDescription:  The description of the checkbox that will be use
    function createSettingsCheckbox($div, settingID, textDescription) {
        const $checkbox = $('<input>', { type: 'checkbox', id: settingID });
        $div.append(
            $('<div>', { class: 'controls-container' }).css({ paddingTop: '2px' }).append(
                $checkbox,
                $('<label>', { for: settingID }).text(textDescription).css({ whiteSpace: 'pre-line' })
            )
        );
        return $checkbox;
    }

    function onKBShortcutModifierKeyClick() {
        const $modifKeyCheckbox = $('#WMEPH-KBSModifierKey');
        const $shortcutInput = $('#WMEPH-KeyboardShortcut');
        const $warn = $('#PlaceHarmonizerKBWarn');
        const modifKeyNew = $modifKeyCheckbox.prop('checked') ? 'Ctrl+' : 'Alt+';

        _shortcutParse = parseKBSShift($shortcutInput.val());
        $warn.empty(); // remove any warning
        _SHORTCUT.remove(_modifKey + _shortcutParse);
        _modifKey = modifKeyNew;
        _SHORTCUT.add(_modifKey + _shortcutParse, harmonizePlace);
        $('#PlaceHarmonizerKBCurrent').empty().append(
            `<span style="font-weight:bold">Current shortcut: ${_modifKey}${_shortcutParse}</span>`
        );
    }

    function onKBShortcutChange() {
        const keyId = 'WMEPH-KeyboardShortcut';
        const $warn = $('#PlaceHarmonizerKBWarn');
        const $key = $(`#${keyId}`);
        const oldKey = localStorage.getItem(keyId);
        const newKey = $key.val();

        $warn.empty(); // remove old warning
        if (newKey.match(/^[a-z]{1}$/i) !== null) { // If a single letter...
            _shortcutParse = parseKBSShift(oldKey);
            const shortcutParseNew = parseKBSShift(newKey);
            _SHORTCUT.remove(_modifKey + _shortcutParse);
            _shortcutParse = shortcutParseNew;
            _SHORTCUT.add(_modifKey + _shortcutParse, harmonizePlace);
            $(localStorage.setItem(keyId, newKey));
            $('#PlaceHarmonizerKBCurrent').empty().append(`<span style="font-weight:bold">Current shortcut: ${_modifKey}${_shortcutParse}</span>`);
        } else { // if not a letter then reset and flag
            $key.val(oldKey);
            $warn.append('<p style="color:red">Only letters are allowed<p>');
        }
    }

    function setCheckedByDefault(id) {
        if (localStorage.getItem(id) === null) {
            localStorage.setItem(id, '1');
        }
    }

    // User pref for KB Shortcut:
    function initShortcutKey() {
        const $current = $('#PlaceHarmonizerKBCurrent');
        const defaultShortcutKey = _IS_BETA_VERSION ? 'S' : 'A';
        const shortcutID = 'WMEPH-KeyboardShortcut';
        let shortcutKey = localStorage.getItem(shortcutID);
        const $shortcutInput = $(`#${shortcutID}`);

        // Set local storage to default if none
        if (shortcutKey === null || !/^[a-z]{1}$/i.test(shortcutKey)) {
            localStorage.setItem(shortcutID, defaultShortcutKey);
            shortcutKey = defaultShortcutKey;
        }
        $shortcutInput.val(shortcutKey);

        if (localStorage.getItem('WMEPH-KBSModifierKey') === '1') { // Change modifier key code if checked
            _modifKey = 'Ctrl+';
        }
        _shortcutParse = parseKBSShift(shortcutKey);
        if (!_initAlreadyRun) _SHORTCUT.add(_modifKey + _shortcutParse, harmonizePlace);
        $current.empty().append(`<span style="font-weight:bold">Current shortcut: ${_modifKey}${_shortcutParse}</span>`);

        $('#WMEPH-KBSModifierKey').click(onKBShortcutModifierKeyClick);

        // Upon change of the KB letter:
        $shortcutInput.change(onKBShortcutChange);
    }

    function onWLMergeClick() {
        const $wlToolsMsg = $('#PlaceHarmonizerWLToolsMsg');
        const $wlInput = $('#WMEPH-WLInput');

        $wlToolsMsg.empty();
        if ($wlInput.val() === 'resetWhitelist') {
            /* if (confirm('***Do you want to reset all Whitelist data?\nClick OK to erase.')) {
                // if the category doesn't translate, then pop an alert that will make a forum post to the thread
                _venueWhitelist = { '1.1.1': { Placeholder: {} } }; // Populate with a dummy place
                saveWhitelistToLS(true);
            } */
            WazeWrap.Alerts.confirm( // if the category doesn't translate, then pop an alert that will make a forum post to the thread
                _SCRIPT_NAME,
                '***Do you want to reset all Whitelist data?<br>Click OK to erase.',
                () => {
                    _venueWhitelist = { '1.1.1': { Placeholder: {} } }; // Populate with a dummy place
                    saveWhitelistToLS(true);
                },
                () => { }
            );
        } else { // try to merge uncompressed WL data
            let wlStringToMerge = validateWLS($('#WMEPH-WLInput').val());
            if (wlStringToMerge) {
                log('Whitelists merged!');
                _venueWhitelist = mergeWL(_venueWhitelist, wlStringToMerge);
                saveWhitelistToLS(true);
                $wlToolsMsg.append('<p style="color:green">Whitelist data merged<p>');
                $wlInput.val('');
            } else { // try compressed WL
                wlStringToMerge = validateWLS(LZString.decompressFromUTF16($('#WMEPH-WLInput').val()));
                if (wlStringToMerge) {
                    log('Whitelists merged!');
                    _venueWhitelist = mergeWL(_venueWhitelist, wlStringToMerge);
                    saveWhitelistToLS(true);
                    $wlToolsMsg.append('<p style="color:green">Whitelist data merged<p>');
                    $wlInput.val('');
                } else {
                    $wlToolsMsg.append('<p style="color:red">Invalid Whitelist data<p>');
                }
            }
        }
    }

    function onWLPullClick() {
        $('#WMEPH-WLInput').val(
            LZString.decompressFromUTF16(
                localStorage.getItem(_WL_LOCAL_STORE_NAME_COMPRESSED)
            )
        );
        $('#PlaceHarmonizerWLToolsMsg').empty().append(
            '<p style="color:green">To backup the data, copy & paste the text in the box to a safe location.<p>'
        );
        localStorage.setItem('WMEPH_WLAddCount', 1);
    }

    function onWLStatsClick() {
        const currWLData = JSON.parse(
            LZString.decompressFromUTF16(
                localStorage.getItem(_WL_LOCAL_STORE_NAME_COMPRESSED)
            )
        );
        const countryWL = {};
        const stateWL = {};
        const entries = Object.keys(currWLData).filter(key => key !== '1.1.1');

        $('#WMEPH-WLInputBeta').val('');
        entries.forEach(venueKey => {
            const country = currWLData[venueKey].country || 'None';
            const state = currWLData[venueKey].state || 'None';
            countryWL[country] = countryWL[country] + 1 || 1;
            stateWL[state] = stateWL[state] + 1 || 1;
        });

        const getSectionDiv = (title, list) => $('<div>', { style: 'margin-bottom: 10px;' }).append(
            $('<div>', { style: 'font-weight: bold; text-decoration: underline' }).text(title),
            Object.keys(list).map(key => $('<div>').text(`${key}: ${list[key]}`))
        );

        $('#PlaceHarmonizerWLToolsMsg').empty().append(
            $('<div>', { style: 'margin-bottom: 10px;' }).text(`Number of WL places: ${entries.length}`),
            getSectionDiv('States', stateWL),
            getSectionDiv('Countries', countryWL)
        );
    }

    function onWLStateFilterClick() {
        const $wlInput = $('#WMEPH-WLInput');
        const stateToRemove = $wlInput.val().trim();
        let msgColor;
        let msgText;

        if (stateToRemove.length < 2) {
            msgColor = 'red';
            msgText = 'Invalid state. Enter the state name in the "Whitelist string" box above, '
                + 'exactly as it appears in the Stats output.';
        } else {
            const currWLData = JSON.parse(
                LZString.decompressFromUTF16(
                    localStorage.getItem(_WL_LOCAL_STORE_NAME_COMPRESSED)
                )
            );
            const venuesToRemove = Object.keys(currWLData).filter(
                venueKey => venueKey !== '1.1.1' && (currWLData[venueKey].state === stateToRemove
                    || (!currWLData[venueKey].state && stateToRemove === 'None'))
            );
            if (venuesToRemove.length > 0) {
                if (localStorage.WMEPH_WLAddCount === '1') {
                    /* if (confirm(`Are you sure you want to clear all whitelist data for ${
                        stateToRemove}? This CANNOT be undone. Press OK to delete, cancel to preserve the data.`)) {
                        backupWhitelistToLS(true);
                        venuesToRemove.forEach(venueKey => {
                            delete _venueWhitelist[venueKey];
                        });
                        saveWhitelistToLS(true);
                        msgColor = 'green';
                        msgText = `${venuesToRemove.length} items removed from WL`;
                        $wlInput.val('');
                    } else {
                        msgColor = 'blue';
                        msgText = 'No changes made';
                    } */
                    WazeWrap.Alerts.confirm(
                        _SCRIPT_NAME,
                        `Are you sure you want to clear all whitelist data for ${stateToRemove}? This CANNOT be undone. `
                        + 'Press OK to delete, cancel to preserve the data.',
                        () => {
                            backupWhitelistToLS(true);
                            venuesToRemove.forEach(venueKey => {
                                delete _venueWhitelist[venueKey];
                            });
                            saveWhitelistToLS(true);
                            $wlInput.val('');
                            $('#PlaceHarmonizerWLToolsMsg').empty().append($('<p>').css({ color: 'green' }).text(`${venuesToRemove.length} items removed from WL`));
                        },
                        () => { $('#PlaceHarmonizerWLToolsMsg').empty().append($('<p>').css({ color: 'blue' }).text('No changes made')); }
                    );
                    return;
                } // else {
                msgColor = 'red';
                msgText = 'Please backup your WL using the Pull button before removing state data';
                // }
            } else {
                msgColor = 'red';
                msgText = `No data for "${stateToRemove}". Use the state name exactly as listed in the Stats`;
            }
        }
        $('#PlaceHarmonizerWLToolsMsg').empty().append($('<p>').css({ color: msgColor }).text(msgText));
    }

    function onWLShareClick() {
        window.open(`https://docs.google.com/forms/d/1k_5RyOq81Fv4IRHzltC34kW3IUbXnQqDVMogwJKFNbE/viewform?entry.1173700072=${_USER.name}`);
    }

    // settings tab
    function initWmephTab() {
        // Enable certain settings by default if not set by the user:
        setCheckedByDefault('WMEPH-ColorHighlighting');
        setCheckedByDefault('WMEPH-ExcludePLADupes');
        setCheckedByDefault('WMEPH-DisablePLAExtProviderCheck');

        // Initialize settings checkboxes
        initSettingsCheckbox('WMEPH-WebSearchNewTab');
        initSettingsCheckbox('WMEPH-DisableDFZoom');
        initSettingsCheckbox('WMEPH-EnableIAZoom');
        initSettingsCheckbox('WMEPH-HidePlacesWiki');
        initSettingsCheckbox('WMEPH-HideReportError');
        initSettingsCheckbox('WMEPH-HideServicesButtons');
        initSettingsCheckbox('WMEPH-HidePURWebSearch');
        initSettingsCheckbox('WMEPH-ExcludePLADupes');
        initSettingsCheckbox('WMEPH-ShowPLAExitWhileClosed');
        if (_USER.isDevUser || _USER.isBetaUser || _USER.rank >= 2) {
            initSettingsCheckbox('WMEPH-DisablePLAExtProviderCheck');
            initSettingsCheckbox('WMEPH-AddAddresses');
            initSettingsCheckbox('WMEPH-EnableCloneMode');
            initSettingsCheckbox('WMEPH-AutoLockRPPs');
        }
        initSettingsCheckbox('WMEPH-ColorHighlighting');
        initSettingsCheckbox('WMEPH-DisableHoursHL');
        initSettingsCheckbox('WMEPH-DisableRankHL');
        initSettingsCheckbox('WMEPH-DisableWLHL');
        initSettingsCheckbox('WMEPH-PLATypeFill');
        initSettingsCheckbox('WMEPH-KBSModifierKey');

        if (_USER.isDevUser) {
            initSettingsCheckbox('WMEPH-RegionOverride');
        }

        // Turn this setting on one time.
        if (!_initAlreadyRun) {
            const runOnceDefaultIgnorePlaGoogleLinkChecks = localStorage.getItem('WMEPH-runOnce-defaultToOff-plaGoogleLinkChecks');
            if (!runOnceDefaultIgnorePlaGoogleLinkChecks) {
                const $chk = $('#WMEPH-DisablePLAExtProviderCheck');
                if (!$chk.prop('checked')) { $chk.trigger('click'); }
            }
            localStorage.setItem('WMEPH-runOnce-defaultToOff-plaGoogleLinkChecks', true);
        }

        initShortcutKey();

        if (localStorage.getItem('WMEPH_WLAddCount') === null) {
            localStorage.setItem('WMEPH_WLAddCount', 2); // Counter to remind of WL backups
        }

        // Reload Data button click event
        $('#WMEPH-ReloadDataBtn').click(async() => {
            $('#WMEPH-ReloadDataBtn').attr('disabled', true);
            await downloadPnhData();
            $('#WMEPH-ReloadDataBtn').attr('disabled', false);
        });

        // WL button click events
        $('#WMEPH-WLMerge').click(onWLMergeClick);
        $('#WMEPH-WLPull').click(onWLPullClick);
        $('#WMEPH-WLStats').click(onWLStatsClick);
        $('#WMEPH-WLStateFilter').click(onWLStateFilterClick);
        $('#WMEPH-WLShare').click(onWLShareClick);

        // Color highlighting
        $('#WMEPH-ColorHighlighting').click(bootstrapWmephColorHighlights);
        $('#WMEPH-DisableHoursHL').click(bootstrapWmephColorHighlights);
        $('#WMEPH-DisableRankHL').click(bootstrapWmephColorHighlights);
        $('#WMEPH-DisableWLHL').click(bootstrapWmephColorHighlights);
        $('#WMEPH-PLATypeFill').click(() => applyHighlightsTest(W.model.venues.getObjectArray()));

        _initAlreadyRun = true;
    }

    async function addWmephTab() {
        // Set up the CSS
        GM_addStyle(_CSS);

        const $container = $('<div>');
        const $reloadDataBtn = $('<div style="margin-bottom:6px; text-align:center;"><div style="position:relative; display:inline-block; width:75%"><input id="WMEPH-ReloadDataBtn" style="min-width:90px; width:50%" class="btn btn-success wmeph-fat-btn" type="button" title="Refresh Data" value="Refresh Data"/><div class="checkmark draw"></div></div></div>');
        const $navTabs = $(
            '<ul class="nav nav-tabs"><li class="active"><a data-toggle="tab" href="#sidepanel-harmonizer">Harmonize</a></li>'
            + '<li><a data-toggle="tab" href="#sidepanel-highlighter">HL / Scan</a></li>'
            + '<li><a data-toggle="tab" href="#sidepanel-wltools">WL Tools</a></li>'
            + '<li><a data-toggle="tab" href="#sidepanel-pnh-moderators">Moderators</a></li></ul>'
        );
        const $tabContent = $('<div class="tab-content">');
        const $harmonizerTab = $('<div class="tab-pane wmeph-pane active" id="sidepanel-harmonizer"></div>');
        const $highlighterTab = $('<div class="tab-pane wmeph-pane" id="sidepanel-highlighter"></div>');
        const $wlToolsTab = $('<div class="tab-pane wmeph-pane" id="sidepanel-wltools"></div>');
        const $moderatorsTab = $('<div class="tab-pane wmeph-pane" id="sidepanel-pnh-moderators"></div>');
        $tabContent.append($harmonizerTab, $highlighterTab, $wlToolsTab, $moderatorsTab);
        $container.append($reloadDataBtn, $navTabs, $tabContent);

        // Harmonizer settings
        createSettingsCheckbox($harmonizerTab, 'WMEPH-WebSearchNewTab', 'Open URL & Search Results in new tab instead of new window');
        createSettingsCheckbox($harmonizerTab, 'WMEPH-DisableDFZoom', 'Disable zoom & center for duplicates');
        createSettingsCheckbox($harmonizerTab, 'WMEPH-EnableIAZoom', 'Enable zoom & center for places with no address');
        createSettingsCheckbox($harmonizerTab, 'WMEPH-HidePlacesWiki', 'Hide "Places Wiki" button in results banner');
        createSettingsCheckbox($harmonizerTab, 'WMEPH-HideReportError', 'Hide "Report script error" button in results banner');
        createSettingsCheckbox($harmonizerTab, 'WMEPH-HideServicesButtons', 'Hide services buttons in results banner');
        createSettingsCheckbox($harmonizerTab, 'WMEPH-HidePURWebSearch', 'Hide "Web Search" button on PUR popups');
        createSettingsCheckbox($harmonizerTab, 'WMEPH-ExcludePLADupes', 'Exclude parking lots when searching for duplicate places');
        createSettingsCheckbox($harmonizerTab, 'WMEPH-ShowPLAExitWhileClosed', 'Always ask if cars can exit parking lots');
        if (_USER.isDevUser || _USER.isBetaUser || _USER.rank >= 2) {
            createSettingsCheckbox($harmonizerTab, 'WMEPH-DisablePLAExtProviderCheck', 'Disable check for "Google place link" on Parking Lot Areas');
            createSettingsCheckbox($harmonizerTab, 'WMEPH-AddAddresses', 'Add detected address fields to places with no address');
            createSettingsCheckbox($harmonizerTab, 'WMEPH-EnableCloneMode', 'Enable place cloning tools');
            createSettingsCheckbox($harmonizerTab, 'WMEPH-AutoLockRPPs', 'Lock residential place points to region default');
        }

        $harmonizerTab.append('<hr class="wmeph-hr" align="center" width="100%">');

        // Add Letter input box
        const $phShortcutDiv = $('<div id="PlaceHarmonizerKB">');
        // eslint-disable-next-line max-len
        $phShortcutDiv.append('<div id="PlaceHarmonizerKBWarn"></div>Shortcut Letter (a-Z): <input type="text" maxlength="1" id="WMEPH-KeyboardShortcut" style="width: 30px;padding-left:8px"><div id="PlaceHarmonizerKBCurrent"></div>');
        createSettingsCheckbox($phShortcutDiv, 'WMEPH-KBSModifierKey', 'Use Ctrl instead of Alt'); // Add Alt-->Ctrl checkbox

        if (_USER.isDevUser) { // Override script regionality (devs only)
            $phShortcutDiv.append('<hr class="wmeph-hr" align="center" width="100%"><p>Dev Only Settings:</p>');
            createSettingsCheckbox($phShortcutDiv, 'WMEPH-RegionOverride', 'Disable Region Specificity');
        }

        $harmonizerTab.append(
            $phShortcutDiv,
            '<hr class="wmeph-hr" align="center" width="100%">',
            `<div><a href="${_URLS.placesWiki}" target="_blank">Open the WME Places Wiki page</a></div>`,
            `<div><a href="${_URLS.forum}" target="_blank">Submit script feedback & suggestions</a></div>`,
            '<hr class="wmeph-hr" align="center" width="95%">'
        );

        // Highlighter settings
        $highlighterTab.append('<p>Highlighter Settings:</p>');
        createSettingsCheckbox($highlighterTab, 'WMEPH-ColorHighlighting', 'Enable color highlighting of map to indicate places needing work');
        createSettingsCheckbox($highlighterTab, 'WMEPH-DisableHoursHL', 'Disable highlighting for missing hours');
        createSettingsCheckbox($highlighterTab, 'WMEPH-DisableRankHL', 'Disable highlighting for places locked above your rank');
        createSettingsCheckbox($highlighterTab, 'WMEPH-DisableWLHL', 'Disable Whitelist highlighting (shows all missing info regardless of WL)');
        createSettingsCheckbox($highlighterTab, 'WMEPH-PLATypeFill', 'Fill parking lots based on type (public=blue, restricted=yellow, private=red)');
        if (_USER.isDevUser || _USER.isBetaUser || _USER.rank >= 3) {
            // createSettingsCheckbox($highlighterTab 'WMEPH-UnlockedRPPs','Highlight unlocked residential place points');
        }

        // Scanner settings
        // $highlighterTab.append('<hr align="center" width="90%">');
        // $highlighterTab.append('<p>Scanner Settings (coming !soon)</p>');
        // createSettingsCheckbox($highlighterTab, 'WMEPH-PlaceScanner','Placeholder, under development!');

        // Whitelisting settings
        const phWLContentHtml = $(
            '<div id="PlaceHarmonizerWLTools">Whitelist string: <input onClick="this.select();" type="text" id="WMEPH-WLInput" style="width:100%;padding-left:1px;display:block">'
            + '<div style="margin-top:3px;">'
            + '<input class="btn btn-success btn-xs wmeph-fat-btn" id="WMEPH-WLMerge" title="Merge the string into your existing Whitelist" type="button" value="Merge">'
            + '<input class="btn btn-success btn-xs wmeph-fat-btn" id="WMEPH-WLPull" title="Pull your existing Whitelist for backup or sharing" type="button" value="Pull">'
            + '<input class="btn btn-success btn-xs wmeph-fat-btn" id="WMEPH-WLShare" title="Share your Whitelist to a public Google sheet" type="button" value="Share your WL">'
            + '</div>'
            + '<div style="margin-top:12px;">'
            + '<input class="btn btn-info btn-xs wmeph-fat-btn" id="WMEPH-WLStats" title="Display WL stats" type="button" value="Stats">'
            + '<input class="btn btn-danger btn-xs wmeph-fat-btn" id="WMEPH-WLStateFilter" title="Remove all WL items for a state.  Enter the state in the \'Whitelist string\' box." '
            + '     type="button" value="Remove data for 1 State">'
            + '</div>'
            + '</div>'
            + '<div id="PlaceHarmonizerWLToolsMsg" style="margin-top:10px;"></div>'
        );
        $wlToolsTab.append(phWLContentHtml);

        const pnhModerators = {
            ATR: ['cotero2002'],
            GLR: ['JustinS83', 'wxw777'],
            HI: ['Nacron'],
            MAR: ['ct13', 'jr1982jr'],
            NER: ['JayWazin', 'SNYOWL'],
            NOR: ['Joyriding', 'ehcool68', 'PesachZ'],
            NWR: ['SkyviewGuru', 'dmee92'],
            PLN: ['ehepner1977', 'dmee92'],
            SAT: ['whathappened15', 'Luke6270'],
            SCR: ['jm6087', 'sketch'],
            SER: ['willdanneriv', 'Ardan74'],
            SWR: ['tonestertm']
        };

        $moderatorsTab.append(
            $('<div>', { style: 'margin-bottom: 10px;' }).text('Moderators are responsible for reviewing chain submissions for their region.'
                + ' If you have questions or suggestions regarding a chain, please contact any of your regional moderators.'),
            $('<table>').append(
                Object.keys(pnhModerators).map(region => $('<tr>').append(
                    $('<td>', { class: 'wmeph-mods-table-cell title' }).append(
                        $('<div>').text(region)
                    ),
                    $('<td>', { class: 'wmeph-mods-table-cell' }).append(
                        $('<div>').text(pnhModerators[region].join(', '))
                    )
                ))
            )
        );

        const { tabLabel, tabPane } = W.userscripts.registerSidebarTab('WMEPH');
        tabLabel.innerHTML = `<span title="WME Place Harmonizer">WMEPH${_IS_BETA_VERSION ? '-β' : ''}</span>`;
        tabPane.innerHTML = $container.html();
        await W.userscripts.waitForElementConnected(tabPane);
        // Fix tab content div spacing.
        $(tabPane).parent().css({ width: 'auto', padding: '8px !important' });
        $('.wmeph-pane').css({ width: 'auto', padding: '8px !important' });
        initWmephTab();
    }

    function createCloneCheckbox(divID, settingID, textDescription) {
        const $checkbox = $('<input>', {
            type: 'checkbox',
            id: settingID
        }).click(() => saveSettingToLocalStorage(settingID))
            .prop('checked', localStorage.getItem(settingID) === '1');

        const $label = $('<label>', { for: settingID, style: 'margin-left: 2px; font-weight: normal' }).text(textDescription);

        return $('<span>', { style: 'margin-right: 6px;' }).append($checkbox, $label);
    }

    // Function to add Shift+ to upper case KBS
    function parseKBSShift(kbs) {
        return (/^[A-Z]{1}$/g.test(kbs) ? 'Shift+' : '') + kbs;
    }

    // Save settings prefs
    function saveSettingToLocalStorage(settingID) {
        localStorage.setItem(settingID, $(`#${settingID}`).prop('checked') ? '1' : '0');
    }

    // This function validates that the inputted text is a JSON
    function validateWLS(jsonString) {
        try {
            const objTry = JSON.parse(jsonString);
            if (objTry && typeof objTry === 'object' && objTry !== null) {
                return objTry;
            }
        } catch (e) {
            // do nothing
        }
        return false;
    }

    // This function merges and updates venues from object wl2 into wl1
    function mergeWL(wl1, wl2) {
        let wlVenue1;
        let wlVenue2;
        Object.keys(wl2).forEach(venueKey => {
            if (wl1.hasOwnProperty(venueKey)) { // if the wl2 venue is in wl1, then update any keys
                wlVenue1 = wl1[venueKey];
                wlVenue2 = wl2[venueKey];
                // loop thru the venue WL keys
                Object.keys(wlVenue2).forEach(wlKey => {
                    // Only update if the wl2 key is active
                    if (wlVenue2.hasOwnProperty(wlKey) && wlVenue2[wlKey].active) {
                        // if the key is in the wl1 venue and it is active, then push any array data onto the key
                        if (wlVenue1.hasOwnProperty(wlKey) && wlVenue1[wlKey].active) {
                            if (wlVenue1[wlKey].hasOwnProperty('WLKeyArray')) {
                                wl1[venueKey][wlKey].WLKeyArray = insertAtIndex(wl1[venueKey][wlKey].WLKeyArray, wl2[venueKey][wlKey].WLKeyArray, 100);
                            }
                        } else {
                            // if the key isn't in the wl1 venue, or if it's inactive, then copy the wl2 key across
                            wl1[venueKey][wlKey] = wl2[venueKey][wlKey];
                        }
                    }
                }); // END subLoop for venue keys
            } else { // if the venue doesn't exist in wl1, then add it
                wl1[venueKey] = wl2[venueKey];
            }
        });
        return wl1;
    }

    // Get services checkbox status
    function getServicesChecks(venue) {
        const servArrayCheck = [];
        for (let wsix = 0; wsix < _WME_SERVICES_ARRAY.length; wsix++) {
            if (venue.attributes.services.includes(_WME_SERVICES_ARRAY[wsix])) {
                servArrayCheck[wsix] = true;
            } else {
                servArrayCheck[wsix] = false;
            }
        }
        return servArrayCheck;
    }

    function updateServicesChecks() {
        const venue = getSelectedVenue();
        if (venue) {
            if (!_servicesBanner) return;
            const servArrayCheck = getServicesChecks(venue);
            let wsix = 0;
            Object.keys(_servicesBanner).forEach(keys => {
                if (_servicesBanner.hasOwnProperty(keys)) {
                    _servicesBanner[keys].checked = servArrayCheck[wsix]; // reset all icons to match any checked changes
                    _servicesBanner[keys].active = _servicesBanner[keys].active || servArrayCheck[wsix]; // display any manually checked non-active icons
                    wsix++;
                }
            });
            // Highlight 24/7 button if hours are set that way, and add button for all places
            if (isAlwaysOpen(venue)) {
                _servicesBanner.add247.checked = true;
            }
            _servicesBanner.add247.active = true;
        }
    }

    // Focus away from the current cursor focus, to set text box changes
    function blurAll() {
        const tmp = document.createElement('input');
        document.body.appendChild(tmp);
        tmp.focus();
        document.body.removeChild(tmp);
    }

    // Pulls the item PL
    function getCurrentPL() {
        // Return the current PL

        // 5/22/2019 (mapomatic)
        // I'm not sure what this was supposed to do.  Maybe an attempt to wait until the PL
        // was available when loading WME from PL with a place pre-selected and auto-run WMEPH
        // is turned on?  Whatever the purpose was, it won't work properly because it'll return
        // undefined, and the calling code is expecting a value.

        // if ($('.WazeControlPermalink').length === 0) {
        //     log('Waiting for PL div');
        //     setTimeout(getCurrentPL, 500);
        //     return;
        // }

        let pl = '';
        let elem = $('.WazeControlPermalink .permalink');
        if (elem.length && elem.attr('href').length) {
            pl = $('.WazeControlPermalink .permalink').attr('href');
        } else {
            elem = $('.WazeControlPermalink');
            if (elem.length && elem.children('.fa-link').length) {
                pl = elem.children('.fa-link')[0].href;
            }
        }
        return pl;
    }

    // Sets up error reporting
    function reportError() {
        window.open('https://www.waze.com/forum/viewtopic.php?t=239985', '_blank');
    }

    function updateUserInfo() {
        _USER.ref = W.loginManager.user;
        _USER.name = _USER.ref.userName;
        _USER.rank = _USER.ref.rank + 1; // get editor's level (actual level)
        if (!_wmephBetaList || _wmephBetaList.length === 0) {
            if (_IS_BETA_VERSION) {
                WazeWrap.Alerts.warning(_SCRIPT_NAME, 'Beta user list access issue.  Please post in the GHO or PM/DM MapOMatic about this message.  Script should still work.');
            }
            _USER.isBetaUser = false;
            _USER.isDevUser = false;
        } else {
            const lcName = _USER.name.toLowerCase();
            _USER.isDevUser = _wmephDevList.includes(lcName);
            _USER.isBetaUser = _wmephBetaList.includes(lcName);
        }
        if (_USER.isDevUser) {
            _USER.isBetaUser = true; // dev users are beta users
        }
    }

    async function placeHarmonizerInit() {
        updateUserInfo();
        logDev('placeHarmonizerInit'); // Be sure to update User info before calling logDev()

        // Check for script updates.
        checkWmephVersion();
        setInterval(checkWmephVersion, VERSION_CHECK_MINUTES * 60 * 1000);

        // Set up Google place info service.
        _attributionEl = document.createElement('div');
        _placesService = new google.maps.places.PlacesService(_attributionEl);

        _layer = W.map.venueLayer;

        // Add CSS stuff here
        const css = [
            '.wmeph-mods-table-cell { border: solid 1px #bdbdbd; padding-left: 3px; padding-right: 3px; }',
            '.wmeph-mods-table-cell.title { font-weight: bold; }'
        ].join('\n');
        $('head').append(`<style type="text/css">${css}</style>`);

        MultiAction = require('Waze/Action/MultiAction');
        UpdateObject = require('Waze/Action/UpdateObject');
        UpdateFeatureGeometry = require('Waze/Action/UpdateFeatureGeometry');
        UpdateFeatureAddress = require('Waze/Action/UpdateFeatureAddress');
        OpeningHour = require('Waze/Model/Objects/OpeningHour');

        // For debugging purposes.  May be removed when no longer needed.
        unsafeWindow.PNH_DATA = _PNH_DATA;

        // Append a form div for submitting to the forum, if it doesn't exist yet:
        const tempDiv = document.createElement('div');
        tempDiv.id = 'WMEPH_formDiv';
        tempDiv.style.display = 'none';
        $('body').append(tempDiv);

        _userLanguage = I18n.locale;

        // Array prototype extensions (for Firefox fix)
        // 5/22/2019 (mapomatic) I'm guessing these aren't necessary anymore.  If no one reports any errors after a while, these lines may be deleted.
        // Array.prototype.toSet = function () { return this.reduce(function (e, t) { return e[t] = !0, e; }, {}); };
        // Array.prototype.first = function () { return this[0]; };
        // Array.prototype.isEmpty = function () { return 0 === this.length; };

        appendServiceButtonIconCss();
        _UPDATED_FIELDS.init();
        addPURWebSearchButton();

        // Create duplicatePlaceName layer
        _dupeLayer = W.map.getLayerByUniqueName('__DuplicatePlaceNames');
        if (!_dupeLayer) {
            const lname = 'WMEPH Duplicate Names';
            const style = new OpenLayers.Style({
                label: '${labelText}',
                labelOutlineColor: '#333',
                labelOutlineWidth: 3,
                labelAlign: '${labelAlign}',
                fontColor: '${fontColor}',
                fontOpacity: 1.0,
                fontSize: '20px',
                fontWeight: 'bold',
                labelYOffset: -30,
                labelXOffset: 0,
                fill: false,
                strokeColor: '${strokeColor}',
                strokeWidth: 10,
                pointRadius: '${pointRadius}'
            });
            _dupeLayer = new OpenLayers.Layer.Vector(lname, { displayInLayerSwitcher: false, uniqueName: '__DuplicatePlaceNames', styleMap: new OpenLayers.StyleMap(style) });
            _dupeLayer.setVisibility(false);
            W.map.addLayer(_dupeLayer);
        }

        if (localStorage.getItem('WMEPH-featuresExamined') === null) {
            localStorage.setItem('WMEPH-featuresExamined', '0'); // Storage for whether the User has pressed the button to look at updates
        }

        createObserver();

        const xrayMode = localStorage.getItem('WMEPH_xrayMode_enabled') === 'true';
        WazeWrap.Interface.AddLayerCheckbox('Display', 'WMEPH x-ray mode', xrayMode, toggleXrayMode);
        if (xrayMode) setTimeout(() => toggleXrayMode(true), 2000); // Give other layers time to load before enabling.

        // Whitelist initialization
        if (validateWLS(LZString.decompressFromUTF16(localStorage.getItem(_WL_LOCAL_STORE_NAME_COMPRESSED))) === false) { // If no compressed WL string exists
            if (validateWLS(localStorage.getItem(_WL_LOCAL_STORE_NAME)) === false) { // If no regular WL exists
                _venueWhitelist = { '1.1.1': { Placeholder: {} } }; // Populate with a dummy place
                saveWhitelistToLS(false);
                saveWhitelistToLS(true);
            } else { // if regular WL string exists, then transfer to compressed version
                localStorage.setItem('WMEPH-OneTimeWLBU', localStorage.getItem(_WL_LOCAL_STORE_NAME));
                loadWhitelistFromLS(false);
                saveWhitelistToLS(true);
                WazeWrap.Alerts.info(_SCRIPT_NAME, 'Whitelists are being converted to a compressed format.  If you have trouble with your WL, please submit an error report.');
            }
        } else {
            loadWhitelistFromLS(true);
        }

        if (_USER.name === 'ggrane') {
            _searchResultsWindowSpecs = `"resizable=yes, top=${
                Math.round(window.screen.height * 0.1)}, left=${
                Math.round(window.screen.width * 0.3)}, width=${
                Math.round(window.screen.width * 0.86)}, height=${
                Math.round(window.screen.height * 0.8)}"`;
        }

        // Settings setup
        if (!localStorage.getItem(_SETTING_IDS.gLinkWarning)) { // store settings so the warning is only given once
            localStorage.setItem(_SETTING_IDS.gLinkWarning, '0');
        }
        if (!localStorage.getItem(_SETTING_IDS.sfUrlWarning)) { // store settings so the warning is only given once
            localStorage.setItem(_SETTING_IDS.sfUrlWarning, '0');
        }

        WazeWrap.Events.register('mousemove', W.map, e => errorHandler(() => {
            const wmEvts = (W.map.events) ? W.map.events : W.map.getMapEventsListener();
            _wmephMousePosition = W.map.getLonLatFromPixel(wmEvts.getMousePosition(e));
        }));

        // Add zoom shortcut
        _SHORTCUT.add('Control+Alt+Z', zoomPlace);

        // Add Color Highlighting shortcut
        _SHORTCUT.add('Control+Alt+h', () => {
            $('#WMEPH-ColorHighlighting').trigger('click');
        });

        await addWmephTab(); // initialize the settings tab

        // Event listeners
        W.selectionManager.events.register('selectionchanged', null, () => {
            logDev('selectionchanged');
            errorHandler(updateWmephPanel, true);
        });
        W.model.venues.on('objectssynced', () => errorHandler(destroyDupeLabels));
        W.model.venues.on('objectssynced', e => errorHandler(() => syncWL(e)));
        W.model.venues.on('objectschanged', venues => errorHandler(onVenuesChanged, venues));

        // Remove any temporary ID values (ID < 0) from the WL store at startup.
        let removedWLCount = 0;
        Object.keys(_venueWhitelist).forEach(venueID => {
            if (venueID < 0) {
                delete _venueWhitelist[venueID];
                removedWLCount += 1;
            }
        });
        if (removedWLCount > 0) {
            saveWhitelistToLS(true);
            logDev(`Removed ${removedWLCount} venues with temporary ID's from WL store`);
        }

        _catTransWaze2Lang = I18n.translations[_userLanguage].venues.categories; // pulls the category translations

        // Split out state-based data
        const _stateHeaders = _PNH_DATA.states[0].split('|');
        _psStateIx = _stateHeaders.indexOf('ps_state');
        _psState2LetterIx = _stateHeaders.indexOf('ps_state2L');
        _psRegionIx = _stateHeaders.indexOf('ps_region');
        _psGoogleFormStateIx = _stateHeaders.indexOf('ps_gFormState');
        _psDefaultLockLevelIx = _stateHeaders.indexOf('ps_defaultLockLevel');
        // ps_requirePhone_ix = _stateHeaders.indexOf('ps_requirePhone');
        // ps_requireURL_ix = _stateHeaders.indexOf('ps_requireURL');
        _psAreaCodeIx = _stateHeaders.indexOf('ps_areacode');

        // Set up Run WMEPH button once place is selected
        updateWmephPanel();

        // Setup highlight colors
        initializeHighlights();

        W.model.venues.on('objectschanged', () => errorHandler(() => {
            if ($('#WMEPH_banner').length > 0) {
                updateServicesChecks();
                assembleServicesBanner();
            }
        }));

        log('Starting Highlighter');
        bootstrapWmephColorHighlights();
    } // END placeHarmonizer_init function

    function waitForReady() {
        return new Promise(resolve => {
            function loop() {
                if (W && W.loginManager && W.loginManager.user && W.map && WazeWrap && WazeWrap.Ready && W.model.categoryBrands.PARKING_LOT && require) {
                    resolve();
                } else {
                    setTimeout(loop, 100);
                }
            }
            loop();
        });
    }

    async function placeHarmonizerBootstrap() {
        log('Waiting for WME and WazeWrap...');
        await waitForReady();
        WazeWrap.Interface.ShowScriptUpdate(_SCRIPT_NAME, _SCRIPT_VERSION, _SCRIPT_UPDATE_MESSAGE);
        await placeHarmonizerInit();
    }

    const SPREADSHEET_ID = '1pBz4l4cNapyGyzfMJKqA4ePEFLkmz2RryAt1UV39B4g';
    const SPREADSHEET_RANGE = '2019.01.20.001!A2:L';
    const API_KEY = 'YTJWNVBVRkplbUZUZVVObU1YVXpSRVZ3ZW5OaFRFSk1SbTR4VGxKblRURjJlRTFYY3pOQ2NXZElPQT09';
    const BETA_URL = 'YUhSMGNITTZMeTluY21WaGMzbG1iM0pyTG05eVp5OWxiaTl6WTNKcGNIUnpMekk0TmpnNUxYZHRaUzF3YkdGalpTMW9ZWEp0YjI1cGVtVnlMV0psZEdFPQ==';
    const BETA_META_URL = 'YUhSMGNITTZMeTluY21WaGMzbG1iM0pyTG05eVp5OXpZM0pwY0hSekx6STROamc1TFhkdFpTMXdiR0ZqWlMxb1lYSnRiMjVwZW1WeUxXSmxkR0V2WTI5a1pTOVhUVVVsTWpCUWJHRmpaU1V5TUVoaGNtMXZibWw2WlhJbE1qQkNaWFJoTG0xbGRHRXVhbk09';
    const PROD_URL = 'https://gf.qytechs.cn/scripts/28690-wme-place-harmonizer/code/WME%20Place%20Harmonizer.user.js';
    const PROD_META_URL = 'https://gf.qytechs.cn/scripts/28690-wme-place-harmonizer/code/WME%20Place%20Harmonizer.meta.js';
    const dec = s => atob(atob(s));
    let _lastVersionChecked = '0';
    const VERSION_CHECK_MINUTES = 60; // How frequently to check for script updates, in minutes.

    function checkWmephVersion() {
        try {
            let url = _IS_BETA_VERSION ? dec(BETA_META_URL) : PROD_META_URL;
            GM_xmlhttpRequest({
                url,
                onload(res) {
                    try {
                        const latestVersion = res.responseText.match(/@version\s+(.*)/)[1];
                        if (latestVersion > _SCRIPT_VERSION && latestVersion > (_lastVersionChecked || '0')) {
                            _lastVersionChecked = latestVersion;
                            url = _IS_BETA_VERSION ? dec(BETA_URL) : PROD_URL;
                            WazeWrap.Alerts.info(
                                _SCRIPT_NAME,
                                `<a href="${url}" target = "_blank">Version ${
                                    latestVersion}</a> is available.<br>Update now to get the latest features and fixes.`,
                                true,
                                false
                            );
                        }
                    } catch (ex) {
                        console.error('WMEPH upgrade version check:', ex);
                    }
                },
                onerror(res) {
                    // Silently fail with an error message in the console.
                    console.error('WMEPH upgrade version check:', res);
                }
            });
        } catch (ex) {
            // Silently fail with an error message in the console.
            console.error('WMEPH upgrade version check:', ex);
        }
    }
    function getSpreadsheetUrl(id, range, key) {
        return `https://sheets.googleapis.com/v4/spreadsheets/${id}/values/${range}?${dec(key)}`;
    }
    function downloadPnhData() {
        log('PNH data download started...');
        return new Promise((resolve, reject) => {
            // TODO change the _PNH_DATA cache to use an object so we don't have to rely on ugly array index lookups.
            const processData1 = (data, colIdx) => data.filter(row => row.length >= colIdx + 1).map(row => row[colIdx]);
            const url = getSpreadsheetUrl(SPREADSHEET_ID, SPREADSHEET_RANGE, API_KEY);

            $.getJSON(url).done(res => {
                const { values } = res;
                if (values[0][0].toLowerCase() === 'obsolete') {
                    WazeWrap.Alerts.error(_SCRIPT_NAME, 'You are using an outdated version of WMEPH that doesn\'t work anymore. Update or disable the script.');
                    return;
                }

                // This needs to be performed before makeNameCheckList() is called.
                _wordVariations = processData1(values, 11).slice(1).map(row => row.toUpperCase().replace(/[^A-z0-9,]/g, '').split(','));

                _PNH_DATA.USA.pnh = processData1(values, 0);
                _PNH_DATA.USA.pnhNames = makeNameCheckList(_PNH_DATA.USA.pnh);

                _PNH_DATA.states = processData1(values, 1);

                _PNH_DATA.CAN.pnh = processData1(values, 2);
                _PNH_DATA.CAN.pnhNames = makeNameCheckList(_PNH_DATA.CAN.pnh);

                _PNH_DATA.USA.categories = processData1(values, 3);
                _PNH_DATA.USA.categoryNames = makeCatCheckList(_PNH_DATA.USA.categories);

                // For now, Canada uses some of the same settings as USA.
                _PNH_DATA.CAN.categories = _PNH_DATA.USA.categories;
                _PNH_DATA.CAN.categoryNames = _PNH_DATA.USA.categoryNames;

                const WMEPHuserList = processData1(values, 4)[1].split('|');
                const betaix = WMEPHuserList.indexOf('BETAUSERS');
                _wmephDevList = [];
                _wmephBetaList = [];
                for (let ulix = 1; ulix < betaix; ulix++) _wmephDevList.push(WMEPHuserList[ulix].toLowerCase().trim());
                for (let ulix = betaix + 1; ulix < WMEPHuserList.length; ulix++) _wmephBetaList.push(WMEPHuserList[ulix].toLowerCase().trim());

                const processTermsCell = (termsValues, colIdx) => processData1(termsValues, colIdx)[1]
                    .toLowerCase().split('|').map(value => value.trim());
                _hospitalPartMatch = processTermsCell(values, 5);
                _hospitalFullMatch = processTermsCell(values, 6);
                _animalPartMatch = processTermsCell(values, 7);
                _animalFullMatch = processTermsCell(values, 8);
                _schoolPartMatch = processTermsCell(values, 9);
                _schoolFullMatch = processTermsCell(values, 10);

                log('PNH data download completed');
                resolve();
            }).fail(res => {
                const message = res.responseJSON && res.responseJSON.error ? res.responseJSON.error : 'See response error message above.';
                console.error('WMEPH failed to load spreadsheet:', message);
                reject();
            });
        });
    }

    function devTestCode() {
        if (W.loginManager.user.userName === 'MapOMatic') {
            // test code here
        }
    }

    async function bootstrap() {
        // Quit if another version of WMEPH is already running.
        if (unsafeWindow.wmephRunning) {
            WazeWrap.Alerts.error(_SCRIPT_NAME, 'Multiple versions of Place Harmonizer are turned on.  Only one will be enabled.');
            return;
        }
        unsafeWindow.wmephRunning = 1;
        // Start downloading the PNH spreadsheet data in the background.  Starts the script once data is ready.
        await downloadPnhData();
        await placeHarmonizerBootstrap();
        devTestCode();
    }

    bootstrap();
})();

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址