WME Place Harmonizer

Harmonizes, formats, and locks a selected place

目前为 2020-06-02 提交的版本。查看 最新版本

/* eslint-disable nonblock-statement-body-position, brace-style, curly, radix, no-template-curly-in-string */
// ==UserScript==
// @name        WME Place Harmonizer
// @namespace   WazeUSA
// @version     2020.06.02.001
// @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
// @grant       GM_addStyle
// ==/UserScript==

/* global $ */
/* global W */
/* global GM_info */
/* global OpenLayers */
/* global _ */
/* global WazeWrap */
/* global LZString */
/* global HoursParser */
/* global GM_addStyle */
/* global unsafeWindow */
/* global I18n */
/* global window */
/* global document */
/* global localStorage */
/* global MutationObserver */
/* global performance */
/* global atob */

// Script update info
const _WHATS_NEW_LIST = { // New in this version
    '2020.06.02.001': [
        'Bug fix due to latest WME release.'
        ],
    '2020.03.31.001': [
        'FIXED: WME removed normalized user rank property.'
    ],
    '2020.03.18.001': [
        'Bug fix for X-Ray Mode, caused by WME changes.'
    ],
    '2020.03.15.001': [
        'FIXED: When clicking the "Use PNH URL" button, a window would open to report an error.'
    ],
    '2020.03.14.001': [
        'Minor bug fix for missing street when using localized store locator.'
    ],
    '2020.01.07.001': [
        'Bug fix due to latest WME release.'
    ],
    '2019.10.30.001': [
        'Switch to WazeWrap alerts and event registrations.'
    ],
    '2019.07.25.001': [
        'More bug fixes for latest WME release.'
    ],
    '2019.07.23.001': [
        'Bug fix for latest WME release.'
    ],
    '2019.05.31.001': [
        'Fixed an issue that was preventing WMEPH from running on some places.'
    ],
    '2019.05.28.001': [
        'Some code optimizations.'
    ],
    '2019.05.24.001': [
        'Fix green highlighting of WMEPH-modified fields in the edit panel.',
        'Remove "auto-run on select" option.',
        'Loosen requirements for Scenic Overlook category.'
    ],
    '2019.05.23.001': [
        'Don\'t display WMEPH buttons when multiple places are selected.',
        'A lot of code maintenance/cleanup',
        'New version # format :)'
    ],
    '1.3.146': [
        'FIXED: Moderator table mistakes, and updated its layout to be more compact.'],
    '1.3.145': [
        'NEW: Added a Moderators tab so people can bug moderators more, and me less :D'],
    '1.3.143': [
        'FIXED: HN entry field in WMEPH banner was not working. Replaced with "Edit Address" button.',
        'FIXED: Adding external provider from WMEPH banner would sometimes go to the Category box.'
    ],
    '1.3.142': [
        'FIXED: The "Nudge" buttons do not work in some cases.  After saving, the place is not nudged.'
    ],
    '1.3.141': [
        'FIXED: WMEPH will not run on places where it finds potential duplicate places.'
    ]
};

const _CSS_ARRAY = [
    '#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; }',
    '.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; }',
    '#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 .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_runButton { 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; }',
    '.ui-autocomplete { max-height: 300px;overflow-y: auto;overflow-x: hidden;} ',
    '.wmeph-hr { border-color: #ccc; }'
];

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_DEV_VERSION = /Beta/i.test(_SCRIPT_NAME); //  enables dev messages and unique DOM options if the script is called "... Beta"
const _DEV_VERSION_STR = _IS_DEV_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 _countryCode;
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;

// Userlists
let _wmephDevList;
let _wmephBetaList;

let _shortcutParse;
let _modifKey = 'Alt+';

// Whitelisting vars
let _venueWhitelist;
let _venueWhitelistStr;
let _WLSToMerge;
let _wlKeyName;
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 _rppLockString = 'Lock?';
const _PANEL_FIELDS = {}; // the fields for the sidebar
let _disableHighlightTest = false; // Set to true to temporarily disable highlight checks immediately when venues change.
let _wl = {};
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=215657',
    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'
};
let _userLanguage;
// lock levels are offset by one
const _LOCK_LEVEL_2 = 1;
const _LOCK_LEVEL_4 = 3;
let _defaultLockLevel = _LOCK_LEVEL_2;
let _pnhLockLevel;
const _PM_USER_LIST = { // user names and IDs for PM functions

    SER: {
        approvalActive: true,
        mods: [
            { id: '16888799', name: 'willdanneriv' },
            // { id: '17083181', name: 'itzwolf' },
            { id: '17077334', name: 'ardan74' }
        ]
    },
    WMEPH: {
        approvalActive: true,
        mods: [
            { id: '2647925', name: 'MapOMatic' }
        ]
    }
};
let _severityButt = 0; // error tracking to determine banner color (action buttons)
let _duplicateName = '';
let _catTransWaze2Lang; // pulls the category translations
let _newName;
let _newURL;
let _tempPNHURL = '';
let _newPhone;
let _newAliases = [];
let _newAliasesTemp = [];
let _newCategories = [];
const _WME_SERVICES_ARRAY = ['VALLET_SERVICE', 'DRIVETHROUGH', 'WI_FI', 'RESTROOMS', 'CREDIT_CARDS', 'RESERVATIONS', 'OUTSIDE_SEATING',
    'AIR_CONDITIONING', 'PARKING_FOR_CUSTOMERS', 'DELIVERIES', 'TAKE_AWAY', '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'.split('|')
};
let _newPlaceURL;
let _approveRegionURL;
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: '.landmark .form-control[name="name"]', tab: 'general' },
    aliases: { updated: false, selector: '.landmark .form-control.alias-name', tab: 'general' },
    address: { updated: false, selector: '.landmark .address-edit span.full-address', tab: 'general' },
    categories: { updated: false, selector: '.landmark .categories.controls .select2-container', tab: 'general' },
    description: { updated: false, selector: '.landmark .form-control[name="description"]', tab: 'general' },
    lock: { updated: false, selector: '.landmark .form-control.waze-radio-container', tab: 'general' },
    externalProvider: { updated: false, selector: '.landmark .external-providers-view', tab: 'general' },
    brand: { updated: false, selector: '.landmark .brand .select2-container', tab: 'general' },
    url: { updated: false, selector: '.landmark .form-control[name="url"]', tab: 'more-info' },
    phone: { updated: false, selector: '.landmark .form-control[name="phone"]', tab: 'more-info' },
    openingHours: { updated: false, selector: '.landmark .opening-hours ul', tab: 'more-info' },
    cost: { updated: false, selector: '.landmark .form-control[name="costType"]', tab: 'more-info' },
    canExit: { updated: false, selector: '.landmark label[for="can-exit-checkbox"]', tab: 'more-info' },
    hasTBR: { updated: false, selector: '.landmark label[for="has-tbr"]', tab: 'more-info' },
    lotType: { updated: false, selector: '.landmark .parking-type-option', tab: 'more-info' },
    parkingSpots: { updated: false, selector: '.landmark .form-control[name="estimatedNumberOfSpots"]', tab: 'more-info' },
    lotElevation: { updated: false, selector: '.landmark .lot-checkbox', tab: 'more-info' },

    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="#landmark-edit-${prop.tab}"]`).css({ 'background-color': '#dfd' });
    //         });
    // },
    reset() {
        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', 'CARPOOL_PARKING',
            'EV_CHARGING_STATION', 'CAR_WASH', 'SECURITY', 'AIRPORT_SHUTTLE']
            .forEach(service => {
                const propName = `services_${service}`;
                this[propName] = { updated: false, selector: `.landmark 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()));
    },
    updateEditPanelHighlights() {
        // Highlight fields in the editor panel that have been updated by WMEPH.
        this.getFieldProperties().filter(prop => prop.updated).forEach(prop => {
            $(prop.selector).css({ 'background-color': '#dfd' });
            $(`a[href="#landmark-edit-${prop.tab}"]`).css({ 'background-color': '#dfd' });
        });
    }
};

// KB Shortcut object
const _SHORTCUT = {
    all_shortcuts: {}, // All the shortcuts are stored in this array
    add(shortcutCombo, callback, opt) {
        // Provide a set of default options
        const defaultOptions = {
            type: 'keydown', propagate: false, disable_in_input: 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.disable_in_input) { // 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,
                scroll_lock: 145,
                scroll: 145,
                capslock: 20,
                caps_lock: 20,
                caps: 20,
                numlock: 144,
                num_lock: 144,
                num: 144,
                pause: 19,
                break: 19,
                insert: 45,
                home: 36,
                delete: 46,
                end: 35,
                pageup: 33,
                page_up: 33,
                pu: 33,
                pagedown: 34,
                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.all_shortcuts[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.all_shortcuts[shortcutCombo];
        delete (this.all_shortcuts[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) {
    try {
        callback();
    } catch (ex) {
        console.error(`${_SCRIPT_NAME}:`, ex);
    }
}

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

function getHoursHtml(label, defaultText) {
    defaultText = defaultText || _DEFAULT_HOURS_TEXT;
    return $('<span>').append(
        `${label}:`,
        $('<input>', {
            class: 'btn btn-default btn-xs wmeph-btn',
            id: 'WMEPH_noHours',
            title: 'Add pasted hours to existing',
            type: 'button',
            value: 'Add hours',
            style: 'margin-bottom:4px; margin-right:0px'
        }),
        $('<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'
        }),
        // 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:85%;max-width:85%;min-width:85%;font-size:0.85em;height:24px;min-height:24px;max-height:300px;padding-left:3px;color:#AAA">${defaultText}`
    )[0].outerHTML;
}

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 = 3;
    } else if (pvaValue === '2') {
        severity = 1;
    } else if (pvaValue === '3') {
        severity = 2;
    } else {
        severity = 0;
    }
    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;
                    for (let extix = 1, len = newNameList.length; extix < len; extix++) { // extend the list by adding Mid terms onto the SearchNameBase names
                        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

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

function phlog(msg) {
    console.log(`WMEPH${_IS_DEV_VERSION ? '-β' : ''}:`, msg);
}
function phlogdev(msg) {
    if (_USER.isDevUser) {
        console.log(`WMEPH${_IS_DEV_VERSION ? '-β' : ''} (dev):`, msg);
    }
}

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();
    if (venue.isPoint()) {
        venue.geometry.x += 0.000000001;
    } else {
        venue.geometry.components[0].components[0].x += 0.000000001;
    }
    W.model.actionManager.add(new UpdateFeatureGeometry(venue, W.model.venues, originalGeometry, venue.geometry));
}

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) {
                    _dupeLayer.destroyFeatures();
                    _dupeLayer.setVisibility(false);
                } else {
                    const deletedDupe = _dupeLayer.getFeaturesByAttribute('dupeID', lastAction.object.attributes.id);
                    _dupeLayer.removeFeatures(deletedDupe);
                    _dupeIDList.splice(_dupeIDList.indexOf(lastAction.object.attributes.id), 1);
                }
                phlog('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 itemGPS = OpenLayers.Layer.SphericalMercator.inverseMercator(venue.attributes.geometry.getCentroid().x, venue.attributes.geometry.getCentroid().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() {
    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.');
        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 landmark 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 .landmark').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; }'
    ];
    $('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, placePL) {
    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
            phlogdev(`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
        _buttonBanner.ApprovalSubmit = new Flag.ApprovalSubmit(region3L, pnhOrderNum, pnhNameTemp, placePL);
        return ['ApprovalNeeded', pnhNameTemp, pnhOrderNum];
    }
    // if no match was found, suggest adding the place to the sheet if it's a chain
    _buttonBanner.NewPlaceSubmit = new Flag.NewPlaceSubmit();
    return ['NoMatch'];
} // END harmoList function

function onObjectsChanged() {
    deleteDupeLabel();

    // This is code to handle updating the banner when changes are made external to the script.
    const venue = getSelectedVenue();
    if ($('#WMEPH_banner').length > 0 && venue) {
        const actions = W.model.actionManager.getActions();
        const lastAction = actions[actions.length - 1];
        if (lastAction && lastAction.object && lastAction.object.type === 'venue' && lastAction.attributes && lastAction.attributes.id === venue.attributes.id) {
            if (lastAction.newAttributes && lastAction.newAttributes.entryExitPoints) {
                harmonizePlaceGo(venue, 'harmonize');
            }
        }
    }
}

// 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(venue) {
                return venue && venue.model && venue.model.attributes.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(venue) {
                return venue && venue.model && venue.model.attributes.wmephSeverity === this.value;
            }
        }),
        symbolizer,
        wmephStyle: 'default'
    });

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

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

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

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

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

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

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

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

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

    function plaTypeRuleGenerator(value, symbolizer) {
        return new W.Rule({
            filter: new OpenLayers.Filter.Comparison({
                type: '==',
                value,
                evaluate(venue) {
                    if ($('#WMEPH-PLATypeFill').prop('checked') && venue && venue.model && venue.model.attributes.categories
                        && venue.model.attributes.categoryAttributes && venue.model.attributes.categoryAttributes.PARKING_LOT
                        && venue.model.attributes.categories.includes('PARKING_LOT')) {
                        const type = venue.model.attributes.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;
    }
    phlogdev(`Ran highlighter in ${Math.round((performance.now() - t0) * 10) / 10} milliseconds.`);
    phlogdev(`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();
            }
        }));

        W.map.venueLayer.events.register('beforefeaturesadded', null, e => errorHandler(() => applyHighlightsTest(e.features.map(f => f.model))));

        // 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, returnType, item, region) {
    if (!s && returnType === 'existing') {
        _buttonBanner.phoneMissing = Flag.PhoneMissing.eval(item, _wl, region, outputFormat);
        return s;
    }
    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) {
            if (returnType === 'inputted') {
                return 'badPhone';
            }
            _buttonBanner.phoneInvalid = new Flag.PhoneInvalid();
            return s;
        }
    }
    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, updateObj, actions) {
    const action = new UpdateObject(venue, updateObj);
    if (actions) {
        actions.push(action);
    } else {
        W.model.actionManager.add(action);
    }
}

function setServiceChecked(servBtn, checked, actions) {
    const servID = _WME_SERVICES_ARRAY[servBtn.servIDIndex];
    const checkboxChecked = $(`#service-checkbox-${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) {
    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();
            if (_wl.urlWL || (venue.isParkingLot() && !hasOperator)) {
                _buttonBanner.urlMissing.severity = 0;
                _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_DEV_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
        _dupeLayer.destroyFeatures();
    }
}

// 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 {
    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 {
    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.
let Flag = {
    HnDashRemoved: class extends FlagBase {
        constructor() { super(true, 0, 'Dash removed from house number. Verify'); }
    },
    FullAddressInference: class extends FlagBase {
        constructor() { super(true, 3, 'Missing address was inferred from nearby segments. Verify the address and run script again.'); }

        static eval(venue, addr, actions) {
            const result = {};
            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 run the rest of the script
                } else {
                    let inferredAddress = inferAddress(7); // Pull address info from nearby segments
                    if (inferredAddress && inferredAddress.attributes) inferredAddress = inferredAddress.attributes;

                    if (inferredAddress && inferredAddress.state && inferredAddress.country) {
                        if ($('#WMEPH-AddAddresses').prop('checked')) { // update the item's address if option is enabled
                            updateAddress(venue, inferredAddress, actions);
                            result.inferredAddress = inferredAddress;
                            _UPDATED_FIELDS.address.updated = true;
                            result.flag = new Flag.FullAddressInference();
                            result.noLock = true;
                        } else if (!['JUNCTION_INTERCHANGE'].includes(_newCategories[0])) {
                            _buttonBanner.cityMissing = new Flag.CityMissing();
                            result.noLock = true;
                        }
                    } 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 run the rest of the script
                    }
                }
            }
            return result;
        }

        static evalHL(venue, addr) {
            let result = null;
            if (!addr.state || !addr.country) {
                if (venue.attributes.adLocked) {
                    result = 'adLock';
                } else {
                    const cat = venue.attributes.categories;
                    if (containsAny(cat, ['HOSPITAL_MEDICAL_CARE', 'HOSPITAL_URGENT_CARE', 'GAS_STATION'])) {
                        phlogdev('Unaddressed HUC/GS');
                        result = 5;
                    } else if (cat.includes('JUNCTION_INTERCHANGE')) {
                        result = 0;
                    } else {
                        result = 3;
                    }
                }
            }
            return result;
        }
    },
    NameMissing: class extends FlagBase {
        constructor() { super(true, 3, 'Name is missing.'); }
    },
    PlaIsPublic: class extends FlagBase {
        constructor() {
            super(true, 0, '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>');
        }
    },
    PlaNameMissing: class extends FlagBase {
        constructor() {
            super(true, 1, 'Name is missing.');
            this.message += _USER.rank < 3 ? ' Request an R3+ lock to confirm unnamed parking lot.' : ' Lock to 3+ to confirm unnamed parking lot.';
        }
    },
    PlaNameNonStandard: class extends WLFlag {
        constructor() {
            super(true, 2, 'Parking lot names typically contain words like "Parking", "Lot", and/or "Garage"', true, 'Whitelist non-standard PLA name', 'plaNameNonStandard');
        }

        static eval(venue, wl) {
            const result = { flag: null };
            if (!wl.plaNameNonStandard) {
                const { name } = venue.attributes;
                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 (venue.isParkingLot() && name && !re.test(name)) {
                    result.flag = new Flag.PlaNameNonStandard();
                }
            }
            return result;
        }
    },
    IndianaLiquorStoreHours: class extends WLFlag {
        constructor() {
            super(true, 0, '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, hpMode) {
            const result = { flag: null };
            if (hpMode.harmFlag && !_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.flag = new Flag.IndianaLiquorStoreHours();
                }
            }
            return result;
        }
    },
    HoursOverlap: class extends FlagBase {
        constructor() { super(true, 3, 'Overlapping hours of operation. Place might not save.'); }
    },
    UnmappedRegion: class extends WLFlag {
        constructor() { super(true, 3, 'This category is usually not mapped in this region.', true, 'Whitelist unmapped category', 'unmappedRegion'); }
    },
    RestAreaName: class extends WLFlag {
        constructor() { super(true, 3, 'Rest area name is out of spec. Use the Rest Area wiki button below to view formats.', true, 'Whitelist rest area name', 'restAreaName'); }
    },
    RestAreaNoTransportation: class extends ActionFlag {
        constructor() { super(true, 2, 'Rest areas should not use the Transportation category.', 'Remove it?'); }

        // eslint-disable-next-line class-methods-use-this
        action() {
            const ix = _newCategories.indexOf('TRANSPORTATION');
            if (ix > -1) {
                const venue = getSelectedVenue();
                _newCategories.splice(ix, 1);
                _UPDATED_FIELDS.categories.updated = true;
                addUpdateAction(venue, { categories: _newCategories });
                harmonizePlaceGo(venue, 'harmonize');
            }
        }
    },
    RestAreaGas: class extends FlagBase {
        constructor() { super(true, 3, 'Gas stations at Rest Areas should be separate area places.'); }
    },
    RestAreaScenic: class extends WLActionFlag {
        constructor() {
            super(true, 0, 'Verify that the "Scenic Overlook" category is appropriate for this rest area.  If not: ',
                'Remove it', 'Remove "Scenic Overlook" category.', true, 'Whitelist place', 'restAreaScenic');
        }

        // eslint-disable-next-line class-methods-use-this
        action() {
            const ix = _newCategories.indexOf('SCENIC_LOOKOUT_VIEWPOINT');
            if (ix > -1) {
                const venue = getSelectedVenue();
                _newCategories.splice(ix, 1);
                _UPDATED_FIELDS.categories.updated = true;
                addUpdateAction(venue, { categories: _newCategories });
                harmonizePlaceGo(venue, 'harmonize');
            }
        }
    },
    RestAreaSpec: class extends WLActionFlag {
        constructor() {
            super(true, 3, 'Is this a rest area?',
                'Yes', 'Update with proper categories and services.', true, 'Whitelist place', 'restAreaSpec');
        }

        // eslint-disable-next-line class-methods-use-this
        action() {
            const venue = getSelectedVenue();
            const actions = [];
            // update categories according to spec
            _newCategories = insertAtIX(_newCategories, 'REST_AREAS', 0);
            actions.push(new UpdateObject(venue, { categories: _newCategories }));
            _UPDATED_FIELDS.categories.updated = true;

            // make it 24/7
            actions.push(new UpdateObject(venue, {
                openingHours: [new OpeningHour({ days: [1, 2, 3, 4, 5, 6, 0], fromHour: '00:00', toHour: '00:00' })]
            }));
            _UPDATED_FIELDS.openingHours.updated = true;

            _servicesBanner.add247.checked = true;
            _servicesBanner.addParking.actionOn(actions); // add parking service
            _servicesBanner.addWheelchair.actionOn(actions); // add parking service
            _buttonBanner.restAreaSpec.active = false; // reset the display flag

            executeMultiAction(actions);

            _disableHighlightTest = true;
            harmonizePlaceGo(venue, 'harmonize');
            _disableHighlightTest = false;
            applyHighlightsTest(venue);
        }
    },
    GasMismatch: class extends WLFlag {
        constructor() {
            super(true, 3, '<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>',
                true, 'Whitelist gas brand / name mismatch', 'gasMismatch');
        }
    },
    GasUnbranded: class extends FlagBase {
        //  Unbranded is not used per wiki
        constructor() {
            super(true, 3, '"Unbranded" should not be used for the station brand. Change to correct brand or '
                + 'use the blank entry at the top of the brand list.');
        }

        static eval(venue, brand) {
            const result = { flag: null };
            if (venue.isGasStation() && brand === 'Unbranded') {
                result.flag = new Flag.GasUnbranded();
                result.noLock = true;
            }
            return result;
        }
    },
    GasMkPrim: class extends ActionFlag {
        constructor() {
            super(true, 3, 'Gas Station is not the primary category', 'Fix', 'Make the Gas Station '
                + 'category the primary category.');
        }

        // eslint-disable-next-line class-methods-use-this
        action() {
            const venue = getSelectedVenue();
            // Insert/move Gas category in the first position
            _newCategories = insertAtIX(_newCategories, 'GAS_STATION', 0);
            _UPDATED_FIELDS.categories.updated = true;
            addUpdateAction(venue, { categories: _newCategories });
            harmonizePlaceGo(venue, 'harmonize');
        }
    },
    IsThisAPilotTravelCenter: class extends ActionFlag {
        constructor() { super(true, 0, 'Is this a "Travel Center"?', 'Yes', ''); }

        static eval(venue, hpMode, state2L, newName, actions) {
            const result = { flag: null, newName };
            if (hpMode.harmFlag && state2L === 'TN') {
                if (result.newName.toLowerCase().trim() === 'pilot') {
                    result.newName = 'Pilot Food Mart';
                    actions.push(new UpdateObject(venue, { name: result.newName }));
                    _UPDATED_FIELDS.name.updated = true;
                }
                if (result.newName.toLowerCase().trim() === 'pilot food mart') {
                    result.flag = new Flag.IsThisAPilotTravelCenter();
                }
            }
            return result;
        }

        // eslint-disable-next-line class-methods-use-this
        action() {
            const venue = getSelectedVenue();
            _UPDATED_FIELDS.name.updated = true;
            addUpdateAction(venue, { name: 'Pilot Travel Center' });
            harmonizePlaceGo(venue, 'harmonize');
        }
    },
    HotelMkPrim: class extends WLActionFlag {
        constructor() {
            super(true, 3, 'Hotel category is not first', 'Fix', 'Make the Hotel category the primary category.',
                true, 'Whitelist hotel as secondary category', 'hotelMkPrim');
        }

        // eslint-disable-next-line class-methods-use-this
        action() {
            const venue = getSelectedVenue();
            // Insert/move Hotel category in the first position
            const categories = insertAtIX(venue.attributes.categories.slice(), 'HOTEL', 0);
            _UPDATED_FIELDS.categories.updated = true;
            addUpdateAction(venue, { categories });
            harmonizePlaceGo(venue, 'harmonize');
        }
    },
    ChangeToPetVet: class extends WLActionFlag {
        constructor() {
            super(true, 3, 'Key words suggest this should be a Pet/Veterinarian category. Change?', 'Yes', 'Change to Pet/Veterinarian Category',
                true, 'Whitelist Pet/Vet category', 'changeHMC2PetVet');
        }

        static eval(name, categories) {
            const testName = name.toLowerCase().replace(/[^a-z]/g, ' ');
            const testNameWords = testName.split(' ');
            const result = { flag: null, lockOK: true };
            if ((categories.includes('HOSPITAL_URGENT_CARE') || categories.includes('DOCTOR_CLINIC'))
                && (containsAny(testNameWords, _animalFullMatch) || _animalPartMatch.some(match => testName.includes(match)))) {
                if (!_wl.changeHMC2PetVet) {
                    result.flag = new Flag.ChangeToPetVet();
                    result.lockOK = false;
                }
            }
            return result;
        }

        // eslint-disable-next-line class-methods-use-this
        action() {
            const venue = getSelectedVenue();
            let updated = false;
            let categories = _.uniq(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);
                _UPDATED_FIELDS.categories.updated = true;
                addUpdateAction(venue, { categories });
            }
            harmonizePlaceGo(venue, 'harmonize'); // Rerun the script to update fields and lock
        }
    },
    NotASchool: class extends WLFlag {
        constructor() {
            super(true, 3, 'Key words suggest this should not be School category.',
                true, 'Whitelist School category', 'changeSchool2Offices');
        }

        static eval(name, categories) {
            const result = { flag: null, lockOK: true };
            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)))) {
                if (!_wl.changeSchool2Offices) {
                    result.flag = new Flag.NotASchool();
                    result.lockOK = false;
                }
            }
            return result;
        }
    },
    PointNotArea: class extends WLActionFlag {
        constructor() {
            super(true, 3, 'This category should be a point place.', 'Change to point', 'Change to point place',
                true, 'Whitelist point (not area)', 'pointNotArea');
        }

        // eslint-disable-next-line class-methods-use-this
        action() {
            const venue = getSelectedVenue();
            if (venue.attributes.categories.includes('RESIDENCE_HOME')) {
                const centroid = venue.geometry.getCentroid();
                updateFeatureGeometry(venue, new OpenLayers.Geometry.Point(centroid.x, centroid.y));
            } else {
                $('.landmark label.point-btn').click();
            }
            harmonizePlaceGo(venue, 'harmonize'); // Rerun the script to update fields and lock
        }
    },
    AreaNotPoint: class extends WLActionFlag {
        constructor() {
            super(true, 3, 'This category should be an area place.', 'Change to area', 'Change to Area',
                true, 'Whitelist area (not point)', 'areaNotPoint');
        }

        // eslint-disable-next-line class-methods-use-this
        action() {
            const venue = getSelectedVenue();
            updateFeatureGeometry(venue, venue.getPolygonGeometry());
            harmonizePlaceGo(venue, 'harmonize');
        }
    },
    HnMissing: class extends WLActionFlag {
        constructor(venue) {
            super(true, 3,
                '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, '');
            phlogdev(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;
            }
        }
    },
    // 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, 3, 'No HN:', 'Edit address', 'Edit address to add HN.'); }

    //     // eslint-disable-next-line class-methods-use-this
    //     action() {
    //         $('.nav-tabs a[href="#landmark-edit-general"]').trigger('click');
    //         $('.landmark .full-address').click();
    //         $('input.house-number').focus();
    //     }
    // },
    HnNonStandard: class extends WLFlag {
        constructor() {
            super(true, 3, 'House number is non-standard.', true,
                'Whitelist non-standard HN', 'hnNonStandard');
        }
    },
    HNRange: class extends WLFlag {
        constructor() {
            super(true, 2, 'House number seems out of range for the street name. Verify.', true,
                'Whitelist HN range', 'HNRange');
        }
    },
    StreetMissing: class extends ActionFlag {
        constructor() { super(true, 3, 'No street:', 'Edit address', 'Edit address to add street.'); }

        // eslint-disable-next-line class-methods-use-this
        action() {
            $('.nav-tabs a[href="#landmark-edit-general"]').trigger('click');
            $('.landmark .full-address').click();
            if ($('.empty-street').prop('checked')) {
                $('.empty-street').click();
            }
            $('.street-name').focus();
        }
    },
    CityMissing: class extends ActionFlag {
        constructor() { super(true, 3, 'No city:', 'Edit address', 'Edit address to add city.'); }

        // eslint-disable-next-line class-methods-use-this
        action() {
            $('.nav-tabs a[href="#landmark-edit-general"]').trigger('click');
            $('.landmark .full-address').click();
            if ($('.empty-city').prop('checked')) {
                $('.empty-city').click();
            }
            $('.city-name').focus();
        }
    },
    BankType1: class extends FlagBase {
        constructor() { super(true, 3, 'Clarify the type of bank: the name has ATM but the primary category is Offices'); }
    },
    BankBranch: class extends ActionFlag {
        constructor() { super(true, 1, 'Is this a bank branch office? ', 'Yes', 'Is this a bank branch?'); }

        // eslint-disable-next-line class-methods-use-this
        action() {
            const venue = getSelectedVenue();
            _newCategories = ['BANK_FINANCIAL', 'ATM']; // Change to bank and atm cats
            const tempName = _newName.replace(/[- (]*ATM[- )]*/g, ' ').replace(/^ /g, '').replace(/ $/g, ''); // strip ATM from name if present
            _newName = tempName;
            W.model.actionManager.add(new UpdateObject(venue, { name: _newName, categories: _newCategories }));
            if (tempName !== _newName) _UPDATED_FIELDS.name.updated = true;
            _UPDATED_FIELDS.categories.updated = true;
            harmonizePlaceGo(venue, 'harmonize');
        }
    },
    StandaloneATM: class extends ActionFlag {
        constructor() { super(true, 2, 'Or is this a standalone ATM? ', 'Yes', 'Is this a standalone ATM with no bank branch?'); }

        // eslint-disable-next-line class-methods-use-this
        action() {
            const venue = getSelectedVenue();
            if (!_newName.includes('ATM')) {
                _newName += ' ATM';
                _UPDATED_FIELDS.name.updated = true;
            }
            _newCategories = ['ATM']; // Change to ATM only
            W.model.actionManager.add(new UpdateObject(venue, { name: _newName, categories: _newCategories }));
            _UPDATED_FIELDS.categories.updated = true;
            harmonizePlaceGo(venue, 'harmonize');
        }
    },
    BankCorporate: class extends ActionFlag {
        constructor() { super(true, 1, 'Or is this the bank\'s corporate offices?', 'Yes', 'Is this the bank\'s corporate offices?'); }

        // eslint-disable-next-line class-methods-use-this
        action() {
            const venue = getSelectedVenue();
            _newCategories = ['OFFICES']; // Change to offices category
            const tempName = _newName.replace(/[- (]*atm[- )]*/ig, ' ').replace(/^ /g, '').replace(/ $/g, '').replace(/ {2,}/g, ' '); // strip ATM from name if present
            _newName = tempName;
            W.model.actionManager.add(new UpdateObject(venue, { name: `${_newName} - Corporate Offices`, categories: _newCategories }));
            if (_newName !== tempName) _UPDATED_FIELDS.name.updated = true;
            _UPDATED_FIELDS.categories.updated = true;
            harmonizePlaceGo(venue, 'harmonize');
        }
    },
    CatPostOffice: class extends FlagBase {
        constructor() {
            super(true, 0,
                'The Post Office category is reserved for certain USPS locations. Please be sure to follow <a href="https://wazeopedia.waze.com/wiki/USA/Places/Post_Office" style="color:#3a3a3a;" target="_blank">the guidelines</a>.');
        }
    },
    IgnEdited: class extends FlagBase {
        constructor() { super(true, 2, 'Last edited by an IGN editor'); }
    },
    WazeBot: class extends ActionFlag {
        constructor() {
            super(true, 2,
                '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).');
        }

        // eslint-disable-next-line class-methods-use-this
        action() {
            const venue = getSelectedVenue();
            nudgeVenue(venue);
            harmonizePlaceGo(venue, 'harmonize');
        }
    },
    ParentCategory: class extends WLFlag {
        constructor() {
            super(true, 2, 'This parent category is usually not mapped in this region.',
                true, 'Whitelist parent Category', 'parentCategory');
        }
    },
    CheckDescription: class extends FlagBase {
        constructor() {
            super(true, 2,
                '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, 2, 'Place points are stacked up.'); }
    },
    SuspectDesc: class extends WLFlag {
        constructor() { super(true, 2, 'Description field might contain copyrighted info.', true, 'Whitelist description', 'suspectDesc'); }
    },
    ResiTypeName: class extends WLFlag {
        constructor() {
            super(true, 2, '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, 2, 'Hours of operation listed as open 24hrs but not for all 7 days.'); }
    },
    PhoneInvalid: class extends FlagBase {
        constructor() { super(true, 2, 'Phone invalid.'); }
    },
    AreaNotPointMid: class extends WLFlag {
        constructor() {
            super(true, 2, '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, 2, '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(placePL) {
            super(true, 1, '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.placePL = placePL;
        }

        // eslint-disable-next-line class-methods-use-this
        action() {
            const venue = getSelectedVenue();
            if (!isNullOrWhitespace(_tempPNHURL)) {
                W.model.actionManager.add(new UpdateObject(venue, { url: _tempPNHURL }));
                _UPDATED_FIELDS.url.updated = true;
                harmonizePlaceGo(venue, 'harmonize');
                _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 "${venue.attributes.name}"\nPermalink: ${this.placePL}`
                        });
                    },
                    () => { }
                );
            }
        }
    },
    GasNoBrand: class extends FlagBase {
        constructor() {
            super(true, 1, 'Lock to region standards to verify no gas brand.');
        }

        static eval(venue, brand) {
            const result = { flag: null };
            if (venue.isGasStation() && !brand) {
                result.flag = new Flag.GasNoBrand();
                result.noLock = true;
            }
            return result;
        }
    },
    SubFuel: class extends WLFlag {
        constructor() {
            super(true, 1, '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');
        }
    },
    AreaNotPointLow: class extends WLFlag {
        constructor() {
            super(true, 1, '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, 1, '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, 1, 'Name the post office according to this region\'s <a href="https://wazeopedia.waze.com/wiki/USA/Places/Post_Office" style="color:#3232e6" target="_blank"> standards for USPS post offices</a>'); }
    },
    MissingUSPSAlt: class extends FlagBase {
        constructor() { super(true, 1, 'USPS post offices must have an alternate name of "USPS".'); }
    },
    MissingUSPSZipAlt: class extends WLActionFlag {
        constructor() {
            super(true, 1,
                'No <a href="https://wazeopedia.waze.com/wiki/USA/Places/Post_Office" 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">',
                'Add', true, 'Whitelist missing USPS zip alt name', 'missingUSPSZipAlt');
            this.noBannerAssemble = true;
        }

        // eslint-disable-next-line class-methods-use-this
        action() {
            const $input = $('input#WMEPH-zipAltNameAdd');
            const zip = $input.val().trim();
            if (zip) {
                if (/^\d{5}/.test(zip)) {
                    const venue = getSelectedVenue();
                    const aliases = [].concat(venue.attributes.aliases);
                    // Make sure zip hasn't already been added.
                    if (!aliases.includes(zip)) {
                        aliases.push(zip);
                        W.model.actionManager.add(new UpdateObject(venue, { aliases }));
                        harmonizePlaceGo(venue, 'harmonize');
                    } else {
                        $input.css({ backgroundColor: '#FDD' }).attr('title', 'Zip code alt name already exists');
                    }
                } else {
                    $input.css({ backgroundColor: '#FDD' }).attr('title', 'Zip code format error');
                }
            }
        }
    },
    MissingUSPSDescription: class extends WLFlag {
        constructor() {
            super(true, 1, 'The first line of the description for a <a href="https://wazeopedia.waze.com/wiki/USA/Places/Post_Office" 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, 0, `Check hotel website for any name localization (e.g. ${pnhName} - Tampa Airport).`); }
    },
    LocalizedName: class extends WLFlag {
        constructor() {
            super(true, 1, 'Place needs localization information', true, 'Whitelist localization', 'localizedName');
        }

        static eval(name, nameSuffix, specCase, displayNote) {
            const result = { flag: null, updatePnhName: true };
            const match = specCase.match(/^checkLocalization<>(.+)/i);
            if (match) {
                result.updatePnhName = false;
                const [, baseName] = specCase.match(/^checkLocalization<>(.+)/i);
                const baseNameRE = new RegExp(baseName, 'g');
                if ((name + (nameSuffix || '')).match(baseNameRE) === null) {
                    result.flag = new Flag.LocalizedName();
                    if (_wl.localizedName) {
                        result.flag.WLactive = false;
                        result.flag.severity = 0;
                    }
                    if (displayNote) {
                        result.flag.message = displayNote;
                    }
                }
            }
            return result;
        }
    },
    SpecCaseMessage: class extends FlagBase {
        constructor(message) { super(true, 0, message); }
    },
    PnhCatMess: class extends ActionFlag {
        constructor(message) { super(true, 0, message, null, null); }

        static eval(message, categories, hpMode) {
            const result = { flag: null };
            if (hpMode.harmFlag && !isNullOrWhitespace(message)) {
                result.flag = new Flag.PnhCatMess(message);
                if (categories.includes('HOSPITAL_URGENT_CARE')) {
                    result.flag.value = 'Change to Doctor/Clinic';
                    result.flag.actionType = 'changeToDoctorClinic';
                }
            }
            return result;
        }

        action() {
            const venue = getSelectedVenue();
            if (this.actionType === 'changeToDoctorClinic') {
                const categories = _.uniq(venue.attributes.categories.slice());
                const idx = categories.indexOf('HOSPITAL_URGENT_CARE');
                if (idx > -1) {
                    categories[idx] = 'DOCTOR_CLINIC';
                    _UPDATED_FIELDS.categories.updated = true;
                    addUpdateAction(venue, { categories });
                    harmonizePlaceGo(venue, 'harmonize');
                }
            }
        }
    },
    SpecCaseMessageLow: class extends FlagBase {
        constructor(message) { super(true, 0, message); }
    },
    ExtProviderMissing: class extends ActionFlag {
        constructor() {
            super(true, 3, 'No Google link', 'Nudge', 'If no other properties need to be updated, click to nudge the place (force an edit).');
            this.value2 = 'Add';
            this.title2 = 'Add a link to a Google place';
        }

        static eval(venue, isLocked, categories, userRank, ignoreParkingLots, actions) {
            const result = { flag: null };
            if (userRank >= 2 && venue.areExternalProvidersEditable() && !(venue.isParkingLot() && ignoreParkingLots)) {
                const catsToIgnore = ['BRIDGE', 'TUNNEL', 'JUNCTION_INTERCHANGE', 'NATURAL_FEATURES', 'ISLAND',
                    'SEA_LAKE_POOL', 'RIVER_STREAM', 'CANAL', 'SWAMP_MARSH'];
                if (!categories.some(cat => catsToIgnore.includes(cat))) {
                    const provIDs = venue.attributes.externalProviderIDs;
                    if (!(provIDs && provIDs.length)) {
                        result.flag = new Flag.ExtProviderMissing();
                        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)) {
                                result.flag.severity = 3;
                                result.flag.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 {
                                result.flag.severity = 0;
                                result.flag.message += ': ';
                                delete result.flag.value;
                            }
                        } else {
                            result.flag.severity = 0;
                            result.flag.message += ': ';
                            delete result.flag.value;
                        }
                    }
                }
            }
            return result;
        }

        // eslint-disable-next-line class-methods-use-this
        action() {
            const venue = getSelectedVenue();
            nudgeVenue(venue);
            harmonizePlaceGo(venue, 'harmonize'); // Rerun the script to update fields and lock
        }

        // eslint-disable-next-line class-methods-use-this
        action2() {
            const venue = getSelectedVenue();
            $('div.external-providers-view a').focus().click();
            setTimeout(() => {
                $('a[href="#landmark-edit-general"]').click();
                $('.external-providers-view a.add').focus().mousedown();
                $('div.external-providers-view > div > ul > div > li > div > a').last().mousedown();
                setTimeout(() => $('.select2-input').last().focus().val(venue.attributes.name).trigger('input'), 100);
            }, 100);
        }
    },
    UrlMissing: class extends WLActionFlag {
        constructor() {
            super(
                true,
                1,
                '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;
        }

        // eslint-disable-next-line class-methods-use-this
        action() {
            const venue = getSelectedVenue();
            const newUrl = normalizeURL($('#WMEPH-UrlAdd').val(), true, false, venue);
            if ((!newUrl || newUrl.trim().length === 0) || newUrl === 'badURL') {
                $('input#WMEPH-UrlAdd').css({ backgroundColor: '#FDD' }).attr('title', 'Invalid URL format');
                // this.badInput = true;
            } else {
                phlogdev(newUrl);
                W.model.actionManager.add(new UpdateObject(venue, { url: newUrl }));
                _UPDATED_FIELDS.url.updated = true;
                harmonizePlaceGo(venue, 'harmonize');
            }
        }
    },
    BadAreaCode: class extends WLActionFlag {
        constructor(textValue, outputFormat) {
            super(true, 1, `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.outputFormat = outputFormat;
            this.noBannerAssemble = true;
        }

        action() {
            const venue = getSelectedVenue();
            const newPhone = normalizePhone($('#WMEPH-PhoneAdd').val(), this.outputFormat, 'inputted', venue);
            if (newPhone === 'badPhone') {
                $('input#WMEPH-PhoneAdd').css({ backgroundColor: '#FDD' }).attr('title', 'Invalid phone # format');
                this.badInput = true;
            } else {
                this.badInput = false;
                phlogdev(newPhone);
                W.model.actionManager.add(new UpdateObject(venue, { phone: newPhone }));
                _UPDATED_FIELDS.phone.updated = true;
                harmonizePlaceGo(venue, 'harmonize');
            }
        }
    },
    PhoneMissing: class extends WLActionFlag {
        constructor(venue, hasOperator, wl, outputFormat, isPLA) {
            super(true, 1, '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', 'phoneWL');
            this.noBannerAssemble = true;
            this.badInput = false;
            this.outputFormat = outputFormat;
            this.venue = venue;
            if ((isPLA && !hasOperator) || wl[this.WLkeyName]) {
                this.severity = 0;
                this.WLactive = false;
            }
        }

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

        static eval(venue, wl, region, outputFormat) {
            const hasOperator = venue.attributes.brand && W.model.categoryBrands.PARKING_LOT.includes(venue.attributes.brand);
            const isPLA = venue.isParkingLot();
            let flag = null;
            if (!isPLA || (isPLA && (this._regionsThatWantPlaPhones.includes(region) || hasOperator))) {
                flag = new Flag.PhoneMissing(venue, hasOperator, wl, outputFormat, isPLA);
            }
            return flag;
        }

        action() {
            const newPhone = normalizePhone($('#WMEPH-PhoneAdd').val(), this.outputFormat, 'inputted', this.venue);
            if (newPhone === 'badPhone') {
                $('input#WMEPH-PhoneAdd').css({ backgroundColor: '#FDD' }).attr('title', 'Invalid phone # format');
                this.badInput = true;
            } else {
                this.badInput = false;
                phlogdev(newPhone);
                W.model.actionManager.add(new UpdateObject(this.venue, { phone: newPhone }));
                _UPDATED_FIELDS.phone.updated = true;
                harmonizePlaceGo(this.venue, 'harmonize');
            }
        }
    },
    NoHours: class extends WLFlag {
        constructor() { super(true, 1, getHoursHtml('No hours'), true, 'Whitelist "No hours"', 'noHours'); }

        // eslint-disable-next-line class-methods-use-this
        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) {
            const venue = getSelectedVenue();
            let pasteHours = $('#WMEPH-HoursPaste').val();
            if (pasteHours === _DEFAULT_HOURS_TEXT) {
                return;
            }
            phlogdev(pasteHours);
            pasteHours += !replaceAllHours ? `,${getOpeningHours(venue).join(',')}` : '';
            $('.nav-tabs a[href="#landmark-edit-more-info"]').tab('show');
            const parser = new HoursParser();
            const parseResult = parser.parseHours(pasteHours);
            if (parseResult.hours && !parseResult.overlappingHours && !parseResult.sameOpenAndCloseTimes && !parseResult.parseError) {
                phlogdev(parseResult.hours);
                W.model.actionManager.add(new UpdateObject(venue, { openingHours: parseResult.hours }));
                _UPDATED_FIELDS.openingHours.updated = true;
                $('#WMEPH-HoursPaste').val(_DEFAULT_HOURS_TEXT);
                harmonizePlaceGo(venue, 'harmonize');
            } else {
                phlog('Can\'t parse those hours');
                this.severity = 1;
                this.WLactive = true;
                $('#WMEPH-HoursPaste').css({ 'background-color': '#FDD' }).attr({ title: this.getTitle(parseResult) });
            }
        }

        addHoursAction() {
            this.applyHours();
        }

        replaceHoursAction() {
            this.applyHours(true);
        }
    },
    PlaLotTypeMissing: class extends FlagBase {
        constructor() { super(true, 3, 'Lot type: '); }

        static eval(venue, hpMode) {
            const result = { flag: null };
            if (venue.isParkingLot()) {
                const catAttr = venue.attributes.categoryAttributes;
                const parkAttr = catAttr ? catAttr.PARKING_LOT : undefined;
                if (!parkAttr || !parkAttr.parkingType) {
                    result.flag = new Flag.PlaLotTypeMissing();
                    if (hpMode.harmFlag) {
                        result.noLock = true;
                        result.flag.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])
                                .css({
                                    padding: '3px', height: '20px', lineHeight: '0px', marginRight: '2px', marginBottom: '1px'
                                })
                                .prop('outerHTML')
                        ).join('');
                    }
                }
            }
            return result;
        }
    },
    PlaCostTypeMissing: class extends FlagBase {
        constructor() { super(true, 1, 'Parking cost: '); }

        static eval(venue, hpMode) {
            const result = { flag: null };
            if (venue.isParkingLot()) {
                const catAttr = venue.attributes.categoryAttributes;
                const parkAttr = catAttr ? catAttr.PARKING_LOT : undefined;
                if (!parkAttr || !parkAttr.costType || parkAttr.costType === 'UNKNOWN') {
                    result.flag = new Flag.PlaCostTypeMissing();
                    if (hpMode.harmFlag) {
                        [['FREE', 'Free', 'Free'], ['LOW', '$', 'Low'], ['MODERATE', '$$', 'Moderate'], ['EXPENSIVE', '$$$', 'Expensive']].forEach(btnInfo => {
                            result.flag.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');
                        });
                        result.noLock = true;
                    }
                }
            }
            return result;
        }
    },
    PlaPaymentTypeMissing: class extends ActionFlag {
        constructor() { super(true, 1, 'Parking isn\'t free.  Select payment type(s) from the "More info" tab. ', 'Go there'); }

        static eval(venue) {
            const result = { flag: 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.flag = new Flag.PlaPaymentTypeMissing();
                }
            }
            return result;
        }

        // eslint-disable-next-line class-methods-use-this
        action() {
            $('a[href="#landmark-edit-more-info"]').click();
            $('#payment-checkbox-ELECTRONIC_PASS').focus();
        }
    },
    PlaLotElevationMissing: class extends ActionFlag {
        constructor() { super(true, 1, 'No lot elevation. Is it street level?', 'Yes', 'Click if street level parking only, or select other option(s) in the More Info tab.'); }

        static eval(venue) {
            const result = { flag: 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.flag = new Flag.PlaLotElevationMissing();
                    result.noLock = true;
                }
            }
            return result;
        }

        // eslint-disable-next-line class-methods-use-this
        action() {
            const venue = getSelectedVenue();
            const existingAttr = 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'];
            W.model.actionManager.add(new UpdateObject(venue, { categoryAttributes: { PARKING_LOT: newAttr } }));
            harmonizePlaceGo(venue, 'harmonize');
        }
    },
    PlaSpaces: class extends FlagBase {
        constructor() {
            super(true, 0, '# 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, hpMode) {
            const result = { flag: null };
            if (hpMode.harmFlag && 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.flag = new Flag.PlaSpaces();
                }
            }
            return result;
        }
    },
    NoPlaStopPoint: class extends ActionFlag {
        constructor() { super(true, 1, 'Entry/exit point has not been created.', 'Add point', 'Add an entry/exit point'); }

        static eval(venue) {
            const result = { flag: null };
            if (venue.isParkingLot() && (!venue.attributes.entryExitPoints || !venue.attributes.entryExitPoints.length)) {
                result.flag = new Flag.NoPlaStopPoint();
            }
            return result;
        }

        // eslint-disable-next-line class-methods-use-this
        action() {
            $('.navigation-point-view .add-button').click();
            harmonizePlaceGo(getSelectedVenue(), 'harmonize');
        }
    },
    PlaStopPointUnmoved: class extends FlagBase {
        constructor() { super(true, 1, 'Entry/exit point has not been moved.'); }

        static eval(venue) {
            const result = { flag: 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.flag = new Flag.PlaStopPointUnmoved();
                }
            }
            return result;
        }
    },
    PlaCanExitWhileClosed: class extends ActionFlag {
        constructor() { super(true, 0, 'Can cars exit when lot is closed? ', 'Yes', ''); }

        static eval(venue, hpMode) {
            const result = { flag: null };
            if (hpMode.harmFlag && venue.isParkingLot()) {
                const catAttr = venue.attributes.categoryAttributes;
                const parkAttr = catAttr ? catAttr.PARKING_LOT : undefined;
                if (parkAttr && !parkAttr.canExitWhileClosed && ($('#WMEPH-ShowPLAExitWhileClosed').prop('checked') || !(isAlwaysOpen(venue) || venue.attributes.openingHours.length === 0))) {
                    result.flag = new Flag.PlaCanExitWhileClosed();
                }
            }
            return result;
        }

        // eslint-disable-next-line class-methods-use-this
        action() {
            const venue = getSelectedVenue();
            const existingAttr = venue.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.canExitWhileClosed = true;
            W.model.actionManager.add(new UpdateObject(venue, { categoryAttributes: { PARKING_LOT: newAttr } }));
            harmonizePlaceGo(venue, 'harmonize');
        }
    },
    PlaHasAccessibleParking: class extends ActionFlag {
        constructor() { super(true, 0, 'Does this lot have disability parking? ', 'Yes', ''); }

        static eval(venue, hpMode) {
            const result = { flag: null };
            if (hpMode.harmFlag && venue.isParkingLot()) {
                const { services } = venue.attributes;
                if (!(services && services.includes('DISABILITY_PARKING'))) {
                    result.flag = new Flag.PlaHasAccessibleParking();
                }
            }
            return result;
        }

        // eslint-disable-next-line class-methods-use-this
        action() {
            const venue = getSelectedVenue();
            let { services } = venue.attributes;
            if (services) {
                services = [].concat(services);
            } else {
                services = [];
            }
            services.push('DISABILITY_PARKING');
            W.model.actionManager.add(new UpdateObject(venue, { services }));
            _UPDATED_FIELDS.services_DISABILITY_PARKING.updated = true;
            harmonizePlaceGo(venue, 'harmonize');
        }
    },
    AllDayHoursFixed: class extends FlagBase {
        constructor() { super(true, 0, 'Hours were changed from 00:00-23:59 to "All Day"'); }

        static eval(venue, hpMode, actions) {
            const hoursEntries = venue.attributes.openingHours;
            const newHoursEntries = [];
            let updateHours = false;
            let flag = 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 (hpMode.hlFlag) {
                        // Just return a "placeholder" flag to highlight the place.
                        flag = new FlagBase(true, 2, 'invalid all day hours');
                        break;
                    } else if (hpMode.harmFlag) {
                        updateHours = true;
                        newHoursEntry.toHour = '00:00';
                        newHoursEntry.fromHour = '00:00';
                    }
                }
                newHoursEntries.push(newHoursEntry);
            }
            if (updateHours) {
                addUpdateAction(venue, { openingHours: newHoursEntries }, actions);
                _UPDATED_FIELDS.openingHours.updated = true;
                flag = new Flag.AllDayHoursFixed();
            }
            return flag;
        }
    },
    ResiTypeNameSoft: class extends FlagBase {
        constructor() { super(true, 0, 'The place name suggests a residential place or personalized place of work.  Please verify.'); }
    },
    LocalURL: class extends FlagBase {
        constructor() { super(true, 0, 'Some locations for this business have localized URLs, while others use the primary corporate site. Check if a local URL applies to this location.'); }
    },
    LockRPP: class extends ActionFlag {
        constructor() { super(true, 0, 'Lock this residential point?', 'Lock', 'Lock the residential point'); }

        action() {
            const venue = getSelectedVenue();
            let RPPlevelToLock = $('#RPPLockLevel :selected').val() || _defaultLockLevel + 1;
            phlogdev(`RPPlevelToLock: ${RPPlevelToLock}`);

            RPPlevelToLock -= 1;
            W.model.actionManager.add(new UpdateObject(venue, { lockRank: RPPlevelToLock }));
            // no field highlight here
            this.message = `Current lock: ${parseInt(venue.attributes.lockRank, 10) + 1}. ${_rppLockString} ?`;
        }
    },
    AddAlias: class extends ActionFlag {
        constructor(specCases, optionalAlias) {
            super(true, 0, `Is there a ${optionalAlias} at this location?`, 'Yes', `Add ${optionalAlias}`);
            this.specCases = specCases;
            this.optionalAlias = optionalAlias;
        }

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

        action() {
            const venue = getSelectedVenue();
            let aliases = insertAtIX(venue.attributes.aliases.slice(), this.optionalAlias, 0);
            if (this.specCases.includes('altName2Desc') && !venue.attributes.description.toUpperCase().includes(this.optionalAlias.toUpperCase())) {
                const description = `${this.optionalAlias}\n${venue.attributes.description}`;
                addUpdateAction(venue, { description });
                _UPDATED_FIELDS.description.updated = true;
            }
            aliases = removeSFAliases(name, aliases);
            addUpdateAction(venue, { aliases });
            _UPDATED_FIELDS.aliases.updated = true;
            harmonizePlaceGo(venue, 'harmonize');
        }
    },
    AddCat2: class extends ActionFlag {
        constructor() { super(true, 0, '', 'Yes', ''); }

        static eval() {
            const result = { flag: null };
            result.flag = new Flag.AddCat2();
            return result;
        }

        action() {
            const venue = getSelectedVenue();
            _newCategories.push(this.altCategory);
            W.model.actionManager.add(new UpdateObject(venue, { categories: _newCategories }));
            _UPDATED_FIELDS.categories.updated = true;
            harmonizePlaceGo(venue, 'harmonize');
        }
    },
    AddPharm: class extends ActionFlag {
        constructor() { super(true, 0, 'Is there a Pharmacy at this location?', 'Yes', 'Add Pharmacy category'); }

        // eslint-disable-next-line class-methods-use-this
        action() {
            const venue = getSelectedVenue();
            _newCategories = insertAtIX(_newCategories, 'PHARMACY', 1);
            W.model.actionManager.add(new UpdateObject(venue, { categories: _newCategories }));
            _UPDATED_FIELDS.categories.updated = true;
            harmonizePlaceGo(venue, 'harmonize');
        }
    },
    AddSuper: class extends ActionFlag {
        constructor() { super(true, 0, 'Does this location have a supermarket?', 'Yes', 'Add Supermarket category'); }

        // eslint-disable-next-line class-methods-use-this
        action() {
            const venue = getSelectedVenue();
            _newCategories = insertAtIX(_newCategories, 'SUPERMARKET_GROCERY', 1);
            W.model.actionManager.add(new UpdateObject(venue, { categories: _newCategories }));
            _UPDATED_FIELDS.categories.updated = true;
            harmonizePlaceGo(venue, 'harmonize');
        }
    },
    AppendAMPM: class extends ActionFlag {
        constructor() { super(true, 0, 'Is there an ampm at this location?', 'Yes', 'Add ampm to the place'); }

        // eslint-disable-next-line class-methods-use-this
        action() {
            const venue = getSelectedVenue();
            _newCategories = insertAtIX(_newCategories, 'CONVENIENCE_STORE', 1);
            _newName = 'ARCO ampm';
            _newURL = 'ampm.com';
            W.model.actionManager.add(new UpdateObject(venue, { name: _newName, url: _newURL, categories: _newCategories }));
            _UPDATED_FIELDS.name.updated = true;
            _UPDATED_FIELDS.url.updated = true;
            _UPDATED_FIELDS.categories.updated = true;
            _buttonBanner.appendAMPM.active = false; // reset the display flag
            harmonizePlaceGo(venue, 'harmonize');
        }
    },
    AddATM: class extends ActionFlag {
        constructor() { super(true, 0, 'ATM at location? ', 'Yes', 'Add the ATM category to this place'); }

        // eslint-disable-next-line class-methods-use-this
        action() {
            const venue = getSelectedVenue();
            _newCategories = insertAtIX(_newCategories, 'ATM', 1); // Insert ATM category in the second position
            W.model.actionManager.add(new UpdateObject(venue, { categories: _newCategories }));
            _UPDATED_FIELDS.categories.updated = true;
            harmonizePlaceGo(venue, 'harmonize');
        }
    },
    AddConvStore: class extends ActionFlag {
        constructor() { super(true, 0, 'Add convenience store category? ', 'Yes', 'Add the Convenience Store category to this place'); }

        // eslint-disable-next-line class-methods-use-this
        action() {
            const venue = getSelectedVenue();
            _newCategories = insertAtIX(_newCategories, 'CONVENIENCE_STORE', 1); // Insert C.S. category in the second position
            W.model.actionManager.add(new UpdateObject(venue, { categories: _newCategories }));
            _UPDATED_FIELDS.categories.updated = true;
            harmonizePlaceGo(venue, 'harmonize');
        }
    },
    IsThisAPostOffice: class extends ActionFlag {
        constructor() {
            super(true, 0, 'Is this a <a href="https://wazeopedia.waze.com/wiki/USA/Places/Post_Office" target="_blank" style="color:#3a3a3a">USPS post office</a>? ',
                'Yes', 'Is this a USPS location?');
        }

        static eval(venue, newName) {
            const result = { flag: null };
            const cleanName = newName.toUpperCase().replace(/[/\-.]/g, '');
            if (/\bUSP[OS]\b|\bpost(al)?\s+(service|office)\b/i.test(cleanName)) {
                result.flag = new Flag.IsThisAPostOffice();
            }
            return result;
        }

        // eslint-disable-next-line class-methods-use-this
        action() {
            const venue = getSelectedVenue();
            _newCategories = insertAtIX(_newCategories, 'POST_OFFICE', 0);
            W.model.actionManager.add(new UpdateObject(venue, { categories: _newCategories }));
            _UPDATED_FIELDS.categories.updated = true;
            harmonizePlaceGo(venue, 'harmonize');
        }
    },
    ChangeToHospitalUrgentCare: class extends WLActionFlag {
        constructor(severity, message) {
            super(true, severity, message, 'Change to Hospital / Urgent Care', 'Change category to Hospital / Urgent Care',
                false, 'Whitelist category', 'changetoHospitalUrgentCare');
        }

        static eval(venue, hpMode) {
            const result = { flag: null };
            if (hpMode.harmFlag && venue.attributes.categories.includes('DOCTOR_CLINIC')) {
                result.flag = new Flag.ChangeToHospitalUrgentCare(0, 'If this place provides emergency medical care:');
            }
            return result;
        }

        // eslint-disable-next-line class-methods-use-this
        action() {
            let idx = _newCategories.indexOf('HOSPITAL_MEDICAL_CARE');
            const venue = getSelectedVenue();
            if (idx === -1) idx = _newCategories.indexOf('DOCTOR_CLINIC');
            if (idx > -1) {
                _newCategories[idx] = 'HOSPITAL_URGENT_CARE';
                _UPDATED_FIELDS.categories.updated = true;
                addUpdateAction(venue, { categories: _newCategories });
            }
            harmonizePlaceGo(venue, 'harmonize'); // Rerun the script to update fields and lock
        }
    },
    NotAHospital: class extends WLActionFlag {
        constructor() {
            super(true, 3, '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');
        }

        static eval(categories) {
            const result = { flag: null, lockOK: true };
            if (categories.includes('HOSPITAL_URGENT_CARE')) {
                const testName = _newName.toLowerCase().replace(/[^a-z]/g, ' ');
                const testNameWords = testName.split(' ');
                if (containsAny(testNameWords, _hospitalFullMatch) || _hospitalPartMatch.some(match => testName.includes(match))) {
                    if (!_wl.notAHospital) {
                        result.flag = new Flag.NotAHospital();
                        result.lockOK = false;
                    }
                }
            }
            return result;
        }

        // eslint-disable-next-line class-methods-use-this
        action() {
            const venue = getSelectedVenue();
            let categories = venue.attributes.categories.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) {
                _UPDATED_FIELDS.categories.updated = true;
                W.model.actionManager.add(new UpdateObject(venue, { categories }));
            }
            harmonizePlaceGo(venue, 'harmonize'); // Rerun the script to update fields and lock
        }
    },
    ChangeToDoctorClinic: class extends WLActionFlag {
        constructor() {
            super(true, 0, 'If this place provides non-emergency medical care: ', 'Change to Doctor / Clinic', 'Change category to Doctor / Clinic', false,
                'Whitelist category', 'changeToDoctorClinic');
        }

        static eval(venue, categories, hpMode, pnhNameRegMatch) {
            const result = { flag: null, lockOK: true };
            if (hpMode.harmFlag && venue.attributes.updatedOn < new Date('3/28/2017').getTime()
                && ((categories.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.flag = new Flag.ChangeToDoctorClinic();
                result.flag.WLactive = null;
            }
            return result;
        }

        // eslint-disable-next-line class-methods-use-this
        action() {
            const venue = getSelectedVenue();
            let categories = venue.attributes.categories.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) {
                _UPDATED_FIELDS.categories.updated = true;
                W.model.actionManager.add(new UpdateObject(venue, { categories }));
            }
            harmonizePlaceGo(venue, 'harmonize'); // Rerun the script to update fields and lock
        }
    },
    STC: class extends ActionFlag {
        constructor() {
            super(true, 0, '', 'Force Title Case?', 'Force title case to: ');
            this.originalName = null;
            this.confirmChange = false;
            this.noBannerAssemble = true;
        }

        action() {
            const venue = getSelectedVenue();
            let newName = venue.attributes.name;
            if (newName === this.originalName || this.confirmChange) {
                const parts = getNameParts(this.originalName);
                newName = toTitleCaseStrong(parts.base);
                if (parts.base !== newName) {
                    W.model.actionManager.add(new UpdateObject(venue, { name: newName + (parts.suffix || '') }));
                    _UPDATED_FIELDS.name.updated = true;
                }
                harmonizePlaceGo(venue, 'harmonize');
            } else {
                $('button#WMEPH_STC').text('Are you sure?').after(' The name has changed.  This will overwrite the new name.');
                _buttonBanner.STC.confirmChange = true;
            }
        }
    },
    SFAliases: class extends FlagBase {
        constructor() { super(true, 0, 'Unnecessary aliases were removed.'); }
    },
    PlaceMatched: class extends FlagBase {
        constructor() { super(true, 0, 'Place matched from PNH data.'); }
    },
    PlaceLocked: class extends FlagBase {
        constructor() { super(true, 0, 'Place locked.'); }
    },
    NewPlaceSubmit: class extends ActionFlag {
        constructor() {
            super(true, 0, 'No PNH match. If it\'s a chain: ', 'Submit new chain data', 'Submit info for a new chain through the linked form');
        }

        // eslint-disable-next-line class-methods-use-this
        action() {
            window.open(_newPlaceURL);
        }
    },
    ApprovalSubmit: class extends ActionFlag {
        constructor(region, pnhOrderNum, pnhNameTemp, placePL) {
            super(true, 0, 'PNH data exists but is not approved for this region: ', 'Request approval', 'Request region/country approval of this place');
            this.region = region;
            this.pnhOrderNum = pnhOrderNum;
            this.pnhNameTemp = pnhNameTemp;
            this.placePL = placePL;
        }

        action() {
            if (_PM_USER_LIST.hasOwnProperty(this.region) && _PM_USER_LIST[this.region].approvalActive) {
                const forumPMInputs = {
                    subject: `${this.pnhOrderNum} PNH approval for "${this.pnhNameTemp}"`,
                    message: `Please approve "${this.pnhNameTemp}" for the ${this.region} region.  Thanks\n \nPNH order number: ${
                        this.pnhOrderNum}\n \nPermalink: ${this.placePL}\n \nPNH Link: ${_URLS.usaPnh}`,
                    preview: 'Preview',
                    attach_sig: 'on'
                };
                _PM_USER_LIST[this.region].mods.forEach(obj => {
                    forumPMInputs[`address_list[u][${obj.id}]`] = 'to';
                });
                newForumPost('https://www.waze.com/forum/ucp.php?i=pm&mode=compose', forumPMInputs);
            } else {
                window.open(_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() { super(true, 0, '', 'Location Finder', 'Look up details about this location on the chain\'s finder web page'); }

        // eslint-disable-next-line class-methods-use-this
        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 } = getSelectedVenue();
                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 {
        pnhCatMess: null,
        notAHospital: null,
        notASchool: null,
        hnDashRemoved: null,
        fullAddressInference: null,
        nameMissing: 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,
        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,
        urlMissing: null,
        badAreaCode: null,
        phoneMissing: null,
        noHours: 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,
        STC: 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);
            }
        },
        addWheelchair: {
            active: false,
            checked: false,
            icon: 'serv-wheelchair',
            w2hratio: 50 / 50,
            value: 'WhCh',
            title: 'Wheelchair accessible',
            servIDIndex: 11,
            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: 12,
            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();
                    addUpdateAction(venue, { openingHours: [new OpeningHour({ days: [1, 2, 3, 4, 5, 6, 0], fromHour: '00:00', toHour: '00:00' })] }, actions);
                    _servicesBanner.add247.checked = true;
                    _buttonBanner.noHours = null;
                }
            },
            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}${_DEV_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 pnhOrderNum = '';
    let pnhNameTemp = '';
    let pnhNameTempWeb = '';
    let placePL;
    const itemID = item.attributes.id;

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

    const hpMode = {
        harmFlag: false,
        hlFlag: false,
        scanFlag: false
    };

    if (useFlag.includes('harmonize')) {
        hpMode.harmFlag = true;
        phlog('Running script on selected place...');
    }
    if (useFlag.includes('highlight')) {
        hpMode.hlFlag = true;
    }
    // NOTE: scan is not used yet
    // if (useFlag.includes('scan')) {
    //     hpMode.scanFlag = true;
    // }

    _severityButt = 0;

    // Whitelist: reset flags
    _wl = {
        dupeWL: [],
        restAreaName: false,
        restAreaSpec: false,
        restAreaScenic: false,
        unmappedRegion: false,
        gasMismatch: false,
        hotelMkPrim: false,
        changeToOffice: false,
        changeToDoctorClinic: false,
        changeHMC2PetVet: false,
        changeSchool2Offices: false,
        pointNotArea: false,
        areaNotPoint: false,
        HNWL: false,
        hnNonStandard: false,
        HNRange: false,
        parentCategory: false,
        suspectDesc: false,
        resiTypeName: false,
        longURL: false,
        gasNoBrand: false,
        subFuel: false,
        hotelLocWL: false,
        localizedName: false,
        urlWL: false,
        phoneWL: false,
        aCodeWL: false,
        noHours: false,
        nameMissing: false,
        plaNameMissing: false,
        extProviderMissing: false
    };

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

    _buttonBanner = getButtonBanner();

    if (hpMode.harmFlag) {
        // 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, hpMode, actions);

    let lockOK = true; // if nothing goes wrong, then place will be locked
    const { categories } = item.attributes;

    _newCategories = categories.slice();
    const nameParts = getNameParts(item.attributes.name);
    let newNameSuffix = nameParts.suffix;
    _newName = nameParts.base;
    _newAliases = item.attributes.aliases.slice();
    let newDescripion = item.attributes.description;
    _newURL = item.attributes.url;
    let newURLSubmit = '';
    if (_newURL !== null && _newURL !== '') {
        newURLSubmit = _newURL;
    }
    _newPhone = item.attributes.phone;

    let pnhNameRegMatch;

    // Some user submitted places have no data in the country, state and address fields.
    let inferredAddress;
    if (hpMode.harmFlag) {
        const result = Flag.FullAddressInference.eval(item, addr, actions);
        if (result.exit) return undefined;
        _buttonBanner.fullAddressInference = result.flag;
        ({ inferredAddress } = result);
        if (result.inferredAddress) addr = result.inferredAddress;
        if (result.noLock) lockOK = false;
    } else if (hpMode.hlFlag) {
        const result = Flag.FullAddressInference.evalHL(item, addr);
        if (result) return result;
    }

    let result;
    // Check parking lot attributes.
    if (hpMode.harmFlag && item.isParkingLot()) _servicesBanner.addDisabilityParking.active = true;

    result = Flag.PlaCostTypeMissing.eval(item, hpMode);
    _buttonBanner.plaCostTypeMissing = result.flag;
    if (result.noLock) lockOK = false;

    result = Flag.PlaLotElevationMissing.eval(item);
    _buttonBanner.plaLotElevationMissing = result.flag;
    if (result.noLock) lockOK = false;

    result = Flag.PlaSpaces.eval(item, hpMode);
    _buttonBanner.plaSpaces = result.flag;

    result = Flag.PlaLotTypeMissing.eval(item, hpMode);
    _buttonBanner.plaLotTypeMissing = result.flag;
    if (result.noLock) lockOK = false;

    _buttonBanner.noPlaStopPoint = Flag.NoPlaStopPoint.eval(item).flag;
    _buttonBanner.plaStopPointUnmoved = Flag.PlaStopPointUnmoved.eval(item).flag;
    _buttonBanner.plaCanExitWhileClosed = Flag.PlaCanExitWhileClosed.eval(item, hpMode).flag;
    _buttonBanner.plaPaymentTypeMissing = Flag.PlaPaymentTypeMissing.eval(item).flag;
    _buttonBanner.plaHasAccessibleParking = Flag.PlaHasAccessibleParking.eval(item, hpMode).flag;

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

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

    let itemGPS;
    if (_venueWhitelist.hasOwnProperty(itemID) && (hpMode.harmFlag || (hpMode.hlFlag && !$('#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 (hpMode.harmFlag) _buttonBanner2.clearWL.active = true;
                _wl[wlKey] = _venueWhitelist[itemID][wlKey];
            }
        });
        if (_venueWhitelist[itemID].hasOwnProperty('dupeWL') && _venueWhitelist[itemID].dupeWL.length > 0) {
            if (hpMode.harmFlag) _buttonBanner2.clearWL.active = true;
            _wl.dupeWL = _venueWhitelist[itemID].dupeWL;
        }
        // Update address and GPS info for the place
        if (hpMode.harmFlag) {
            // 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
    if (hpMode.harmFlag && (addr.county === null || addr.state === null)) {
        WazeWrap.Alerts.error(_SCRIPT_NAME, 'Country and/or state could not be determined.  Edit the place address and run WMEPH again.');
        return undefined;
    }
    const countryName = addr.country.name;
    const stateName = addr.state.name;
    if (countryName === 'United States') {
        _countryCode = 'USA';
    } else if (countryName === 'Canada') {
        _countryCode = 'CAN';
    } else if (countryName === 'American Samoa') {
        _countryCode = 'USA';
    } else if (countryName === 'Guam') {
        _countryCode = 'USA';
    } else if (countryName === 'Northern Mariana Islands') {
        _countryCode = 'USA';
    } else if (countryName === 'Puerto Rico') {
        _countryCode = 'USA';
    } else if (countryName === 'Virgin Islands (U.S.)') {
        _countryCode = 'USA';
    } else {
        if (hpMode.harmFlag) {
            WazeWrap.Alerts.error(_SCRIPT_NAME, 'At present this script is not supported in this country.');
        }
        return 3;
    }

    // Parse state-based data
    let state2L = 'Unknown';
    let region = 'Unknown';
    let gFormState = '';
    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 (hpMode.harmFlag) {
                WazeWrap.Alerts.warning(_SCRIPT_NAME, 'Lock level sheet data is not correct');
            } else if (hpMode.hlFlag) {
                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 (hpMode.harmFlag) {
                WazeWrap.Alerts.warning(_SCRIPT_NAME, 'Lock level sheet data is not correct');
            } else if (hpMode.hlFlag) {
                return 3;
            }
            _areaCodeList = `${_areaCodeList},${_stateDataTemp[_psAreaCodeIx]}`;
            break;
        }
    }
    if (state2L === 'Unknown' || region === 'Unknown') { // if nothing found:
        if (hpMode.harmFlag) {
            /* 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)

    result = Flag.IsThisAPilotTravelCenter.eval(item, hpMode, state2L, _newName, actions);
    _buttonBanner.isThisAPilotTravelCenter = result.flag;
    _newName = result.newName;

    if (item.isGasStation()) {
        // If no gas station name, replace with brand name
        if (hpMode.harmFlag && (!_newName || _newName.trim().length === 0) && item.attributes.brand) {
            _newName = item.attributes.brand;
            actions.push(new UpdateObject(item, { name: _newName }));
            _UPDATED_FIELDS.name.updated = true;
        }

        // Add convenience store category to station
        if (!_newCategories.includes('CONVENIENCE_STORE') && !_buttonBanner.subFuel) {
            _buttonBanner.addConvStore = new Flag.AddConvStore();
        }
    } // END Gas Station Checks

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

    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 (hpMode.harmFlag) {
            if (!$('#WMEPH-AutoLockRPPs').prop('checked')) {
                lockOK = false;
            }
            if (item.attributes.name !== '') { // Set the residential place name to the address (to clear any personal info)
                phlogdev('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
                phlogdev('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
                phlogdev('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
                phlogdev('Residential URL cleared');
                actions.push(new UpdateObject(item, { url: null }));
                // no field HL
            }
            if (item.attributes.services.length > 0) {
                phlogdev('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();
        }
    } else if (item.isParkingLot() || (_newName && _newName.trim().length)) { // for non-residential places
        _buttonBanner.extProviderMissing = Flag.ExtProviderMissing.eval(item, isLocked, _newCategories, _USER.rank, $('#WMEPH-DisablePLAExtProviderCheck').prop('checked'), actions).flag;

        // Place Harmonization
        let pnhMatchData;
        if (hpMode.harmFlag) {
            if (item.isParkingLot()) {
                pnhMatchData = ['NoMatch'];
            } else {
                // check against the PNH list
                pnhMatchData = harmoList(_newName, state2L, region, _countryCode, _newCategories, item, placePL);
            }
        } else if (hpMode.hlFlag) {
            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();
                                break;
                            case 'addSuper':
                                flag = new Flag.AddSuper();
                                break;
                            case 'appendAMPM':
                                flag = new Flag.AppendAMPM();
                                break;
                            case 'addATM':
                                flag = new Flag.AddATM();
                                break;
                            case 'addConvStore':
                                flag = new Flag.AddConvStore();
                                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
                    result = Flag.AddAlias.eval(specCase, specCases, _newAliases);
                    if (result.flag) {
                        _buttonBanner.addAlias = result.flag;
                    }

                    // 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;
                            phlogdev('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);
                    if (result.flag) {
                        if (!result.updatePnhName) {
                            updatePNHName = false;
                            showDispNote = false;
                        }
                        _buttonBanner.localizedName = result.flag;
                    }

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

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

            // 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
            if (showDispNote && phDisplayNoteIdx > -1 && !isNullOrWhitespace(pnhMatchData[phDisplayNoteIdx])) {
                if (containsAny(specCases, ['pharmhours'])) {
                    if (!item.attributes.description.toUpperCase().includes('PHARMACY') || (!item.attributes.description.toUpperCase().includes('HOURS')
                        && !item.attributes.description.toUpperCase().includes('HRS'))) {
                        _buttonBanner.specCaseMessage = new Flag.SpecCaseMessage(pnhMatchData[phDisplayNoteIdx]);
                    }
                } else if (containsAny(specCases, ['drivethruhours'])) {
                    if (!item.attributes.description.toUpperCase().includes('DRIVE') || (!item.attributes.description.toUpperCase().includes('HOURS')
                        && !item.attributes.description.toUpperCase().includes('HRS'))) {
                        if ($('#service-checkbox-DRIVETHROUGH').prop('checked')) {
                            _buttonBanner.specCaseMessage = new Flag.SpecCaseMessage(pnhMatchData[phDisplayNoteIdx]);
                        } else {
                            _buttonBanner.specCaseMessageLow = new Flag.SpecCaseMessageLow(pnhMatchData[phDisplayNoteIdx]);
                        }
                    }
                } else {
                    _buttonBanner.specCaseMessageLow = new Flag.SpecCaseMessageLow(pnhMatchData[phDisplayNoteIdx]);
                }
            }

            // Localized Storefinder code:
            _customStoreFinderLocal = false;
            _customStoreFinderLocalURL = '';
            _customStoreFinder = false;
            _customStoreFinderURL = '';
            if (phStoreFinderUrlIdx > -1) { // if the sfurl column exists...
                if (phStoreFinderUrlLocalIdx > -1 && !isNullOrWhitespace(pnhMatchData[phStoreFinderUrlLocalIdx])) {
                    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.15).toString();
                        } else if (tempLocalURL[tlix] === 'ph_longitudePMBuffMin') {
                            _customStoreFinderLocalURL += (itemGPS.lon - 0.15).toString();
                        } else if (tempLocalURL[tlix] === 'ph_latitudePMBuffMax') {
                            _customStoreFinderLocalURL += (itemGPS.lat + 0.15).toString();
                        } else if (tempLocalURL[tlix] === 'ph_longitudePMBuffMax') {
                            _customStoreFinderLocalURL += (itemGPS.lon + 0.15).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])) {
                    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
                        phlog(`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
                    insertAtIX(_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();
                    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();
                        _buttonBanner.standaloneATM = new Flag.StandaloneATM();
                        _buttonBanner.bankCorporate = new Flag.BankCorporate();
                    } else if (_ixBank === -1 && _ixATM === -1) {
                        _buttonBanner.bankBranch = new Flag.BankBranch();
                        _buttonBanner.standaloneATM = new Flag.StandaloneATM();
                    } else if (_ixATM === 0 && _ixBank > 0) {
                        _buttonBanner.bankBranch = new Flag.BankBranch();
                    } else if (_ixBank > -1) {
                        _buttonBanner.bankBranch = new Flag.BankBranch();
                        _buttonBanner.standaloneATM = new Flag.StandaloneATM();
                    }
                    _newName = `${pnhMatchData[phNameIdx]} ATM`;
                    _newCategories = insertAtIX(_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();
                    } else if (_ixBank > -1 && _ixATM === -1) {
                        _buttonBanner.addATM = new Flag.AddATM();
                    } else if (_ixATM === 0 && _ixBank === -1) {
                        _buttonBanner.bankBranch = new Flag.BankBranch();
                        _buttonBanner.standaloneATM = new Flag.StandaloneATM();
                    } else if (_ixBank > 0 && _ixATM > 0) {
                        _buttonBanner.bankBranch = new Flag.BankBranch();
                        _buttonBanner.standaloneATM = new Flag.StandaloneATM();
                    }
                    _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 = insertAtIX(_newCategories, 'BANK_FINANCIAL', 1);
                    _buttonBanner.standaloneATM = new Flag.StandaloneATM();
                    _buttonBanner.bankCorporate = new Flag.BankCorporate();
                }// END PNH bank treatment
            } else if (['GAS_STATION'].includes(priPNHPlaceCat)) { // for PNH gas stations, don't replace existing sub-categories
                if (altCategories && altCategories.length) { // if PNH alts exist
                    insertAtIX(_newCategories, altCategories, 1); //  then insert the alts into the existing category array after the GS category
                }
                if (_newCategories.indexOf('GAS_STATION') !== 0) { // If no GS category in the primary, flag it
                    _buttonBanner.gasMkPrim = new Flag.GasMkPrim();
                    lockOK = false;
                } else {
                    _newName = pnhMatchData[phNameIdx];
                }
            } else if (updatePNHName) { // if not a special category then update the name
                _newName = pnhMatchData[phNameIdx];
                _newCategories = insertAtIX(_newCategories, priPNHPlaceCat, 0);
                if (altCategories && altCategories.length && !specCases.includes('buttOn_addCat2') && !specCases.includes('optionCat2')) {
                    _newCategories = insertAtIX(_newCategories, altCategories, 1);
                }
            } else if (!updatePNHName) {
                // Strong title case option for non-PNH places
                const titleCaseName = toTitleCaseStrong(_newName);
                if (_newName !== titleCaseName) {
                    _buttonBanner.STC = new Flag.STC();
                    _buttonBanner.STC.suffixMessage = `<span style="margin-left: 4px;font-size: 14px">&bull; ${titleCaseName}${newNameSuffix || ''}</span>`;
                    _buttonBanner.STC.title += titleCaseName;
                    _buttonBanner.STC.originalName = _newName + (newNameSuffix || '');
                }
            }

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

            // Parse URL data
            let localURLcheckRE;
            if (localURLcheck !== '') {
                if (_newURL !== null || _newURL !== '') {
                    localURLcheckRE = new RegExp(localURLcheck, 'i');
                    if (_newURL.match(localURLcheckRE) !== null) {
                        _newURL = normalizeURL(_newURL, false, true, item, region);
                    } else {
                        _newURL = normalizeURL(pnhMatchData[phUrlIdx], false, true, item, region);
                        _buttonBanner.localURL = new Flag.LocalURL();
                    }
                } else {
                    _newURL = normalizeURL(pnhMatchData[phUrlIdx], false, true, item, region);
                    _buttonBanner.localURL = new Flag.LocalURL();
                }
            } else {
                _newURL = normalizeURL(pnhMatchData[phUrlIdx], false, true, item, region);
            }
            // Parse PNH Aliases
            [_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 = insertAtIX(_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')) {
                    phlogdev(`Categories updated with ${_newCategories}`);
                    actions.push(new UpdateObject(item, { categories: _newCategories }));
                    // W.model.actionManager.add(new UpdateObject(item, { categories: newCategories }));
                    _UPDATED_FIELDS.categories.updated = true;
                } else { // if second cat is optional
                    phlogdev(`Primary category updated with ${priPNHPlaceCat}`);
                    _newCategories = insertAtIX(_newCategories, priPNHPlaceCat, 0);
                    actions.push(new UpdateObject(item, { categories: _newCategories }));
                    _UPDATED_FIELDS.categories.updated = true;
                }
            }
            // Enable optional 2nd category button
            if (specCases.includes('buttOn_addCat2') && !_newCategories.includes(altCategories[0])) {
                const altCat = altCategories[0];
                // TODO - move logic into flag eval
                _buttonBanner.addCat2 = Flag.AddCat2.eval().flag;
                _buttonBanner.addCat2.message = `Is there a ${_catTransWaze2Lang[altCat]} at this location?`;
                _buttonBanner.addCat2.title = `Add ${_catTransWaze2Lang[altCat]}`;
                _buttonBanner.addCat2.altCategory = altCat;
            }

            // 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();
                }
                phlogdev('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
            if (pnhMatchData[0] === 'ApprovalNeeded') {
                // PNHNameTemp = PNHMatchData[1].join(', ');
                [, [pnhNameTemp]] = pnhMatchData; // Just do the first match
                pnhNameTempWeb = encodeURIComponent(pnhNameTemp);
                pnhOrderNum = pnhMatchData[2].join(',');
            }

            // Strong title case option for non-PNH places
            const titleCaseName = toTitleCaseStrong(_newName);
            if (_newName !== titleCaseName) {
                _buttonBanner.STC = new Flag.STC();
                _buttonBanner.STC.suffixMessage = `<span style="margin-left: 4px;font-size: 14px">&bull; ${titleCaseName}${newNameSuffix || ''}</span>`;
                _buttonBanner.STC.title += titleCaseName;
                _buttonBanner.STC.originalName = _newName + (newNameSuffix || '');
            }

            _newURL = normalizeURL(_newURL, true, false, item, region); // 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();
                    _buttonBanner.standaloneATM = new Flag.StandaloneATM();
                    _buttonBanner.bankCorporate = new Flag.BankCorporate();
                } else if (_ixBank === -1 && _ixATM === -1) {
                    _buttonBanner.bankBranch = new Flag.BankBranch();
                    _buttonBanner.standaloneATM = new Flag.StandaloneATM();
                } else if (_ixATM === 0 && _ixBank > 0) {
                    _buttonBanner.bankBranch = new Flag.BankBranch();
                } else if (_ixBank > -1) {
                    _buttonBanner.bankBranch = new Flag.BankBranch();
                    _buttonBanner.standaloneATM = new Flag.StandaloneATM();
                }
                // 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();
                } else if (_ixBank > -1 && _ixATM === -1) {
                    _buttonBanner.addATM = new Flag.AddATM();
                } else if (_ixATM === 0 && _ixBank === -1) {
                    _buttonBanner.bankBranch = new Flag.BankBranch();
                    _buttonBanner.standaloneATM = new Flag.StandaloneATM();
                } else if (_ixBank > 0 && _ixATM > 0) {
                    _buttonBanner.bankBranch = new Flag.BankBranch();
                    _buttonBanner.standaloneATM = new Flag.StandaloneATM();
                }
                // 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 (hpMode.harmFlag) {
            // Update name:
            if ((_newName + (newNameSuffix || '')) !== item.attributes.name) {
                phlogdev('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) {
                phlogdev('Alt Names updated');
                actions.push(new UpdateObject(item, { aliases: _newAliases }));
                _UPDATED_FIELDS.aliases.updated = true;
            }

            // 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 'TX': regionFormURL = 'https://docs.google.com/forms/d/1x7VM7ofPOKVnWOaX7d70OWXpnVKf6Mkadn4dgYxx4ic/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 = '';
            }
            _newPlaceURL = regionFormURL + newPlaceAddon;
            _approveRegionURL = regionFormURL + approvalAddon;


            // 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]] === '1') { // These are automatically added to all countries/regions (if auto setting is on)
                                    _servicesBanner[servKeys[psix]].active = true;
                                    if ($('#WMEPH-EnableServices').prop('checked')) {
                                        // Automatically enable new services
                                        _servicesBanner[servKeys[psix]].actionOn(actions);
                                    }
                                } else if (catDataTemp[servHeaders[psix]] === '2') { // these are never automatically added but shown
                                    _servicesBanner[servKeys[psix]].active = true;
                                } else if (catDataTemp[servHeaders[psix]] !== '') { // check for state/region auto add
                                    _servicesBanner[servKeys[psix]].active = true;
                                    if ($('#WMEPH-EnableServices').prop('checked')) {
                                        const servAutoRegion = catDataTemp[servHeaders[psix]].replace(/,[^A-za-z0-9]*/g, ',').split(',');
                                        // if the sheet data matches the state, region, or username then auto add
                                        if (servAutoRegion.includes(state2L) || servAutoRegion.includes(region)
                                            || servAutoRegion.includes(_USER.name)) {
                                            _servicesBanner[servKeys[psix]].actionOn(actions);
                                        }
                                    }
                                }
                            }
                        }
                    }
                }
            }
        }


        const isPoint = item.isPoint();
        // NOTE: do not use is2D() function. It doesn't seem to be 100% reliable.
        const isArea = !isPoint;
        let maxPointSeverity = 0;
        let maxAreaSeverity = 3;
        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 > 0) {
                    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(catMessage, _newCategories, hpMode).flag;
                // 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 = 1;
                            _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 = 0;
                        } 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 === 3) {
                _buttonBanner.areaNotPoint = new Flag.AreaNotPoint();
                if (_wl.areaNotPoint || item.attributes.lockRank >= _defaultLockLevel) {
                    _buttonBanner.areaNotPoint.WLactive = false;
                    _buttonBanner.areaNotPoint.severity = 0;
                } else {
                    lockOK = false;
                }
            } else if (maxPointSeverity === 2) {
                _buttonBanner.areaNotPointMid = new Flag.AreaNotPointMid();
                if (_wl.areaNotPoint || item.attributes.lockRank >= _defaultLockLevel) {
                    _buttonBanner.areaNotPointMid.WLactive = false;
                    _buttonBanner.areaNotPointMid.severity = 0;
                } else {
                    lockOK = false;
                }
            } else if (maxPointSeverity === 1) {
                _buttonBanner.areaNotPointLow = new Flag.AreaNotPointLow();
                if (_wl.areaNotPoint || item.attributes.lockRank >= _defaultLockLevel) {
                    _buttonBanner.areaNotPointLow.WLactive = false;
                    _buttonBanner.areaNotPointLow.severity = 0;
                }
            }
        } else if (maxAreaSeverity === 3) {
            _buttonBanner.pointNotArea = new Flag.PointNotArea();
            if (_wl.pointNotArea || item.attributes.lockRank >= _defaultLockLevel) {
                _buttonBanner.pointNotArea.WLactive = false;
                _buttonBanner.pointNotArea.severity = 0;
            } else {
                lockOK = false;
            }
        } else if (maxAreaSeverity === 2) {
            _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 === 1) {
            _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;
            }
        }

        // Check for missing hours field
        if (item.attributes.openingHours.length === 0) { // if no hours...
            if (!containsAny(_newCategories, ['STADIUM_ARENA', 'CEMETERY', 'TRANSPORTATION', 'FERRY_PIER', 'SUBWAY_STATION',
                'BRIDGE', 'TUNNEL', 'JUNCTION_INTERCHANGE', 'ISLAND', 'SEA_LAKE_POOL', 'RIVER_STREAM', 'FOREST_GROVE', 'CANAL',
                'SWAMP_MARSH', 'DAM'])) {
                _buttonBanner.noHours = new Flag.NoHours();
                if (_wl.noHours || $('#WMEPH-DisableHoursHL').prop('checked') || containsAny(_newCategories, ['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'])) {
                    _buttonBanner.noHours.WLactive = false;
                    _buttonBanner.noHours.severity = 0;
                }
            }
        } else {
            if (item.attributes.openingHours.length === 1) { // if one set of hours exist, check for partial 24hrs setting
                const hoursEntry = item.attributes.openingHours[0];
                if (hoursEntry.days.length < 7 && /^0?0:00$/.test(hoursEntry.fromHour)
                    && (/^0?0:00$/.test(hoursEntry.toHour) || hoursEntry.toHour === '23:59')) {
                    _buttonBanner.mismatch247 = new Flag.Mismatch247();
                }
            }
            _buttonBanner.noHours = new Flag.NoHours();
            _buttonBanner.noHours.severity = 0;
            _buttonBanner.noHours.WLactive = false;
            _buttonBanner.noHours.message = getHoursHtml('Hours');
        }
        if (!checkHours(item.attributes.openingHours)) {
            _buttonBanner.hoursOverlap = new Flag.HoursOverlap();
            _buttonBanner.noHours = new Flag.NoHours();
        } else {
            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
                    phlogdev('Correcting M-S entry...');
                    tempHours.push(new OpeningHour({ days: [0], fromHour: tempHours[ohix].fromHour, toHour: tempHours[ohix].toHour }));
                    tempHours[ohix].days = [1];
                    actions.push(new UpdateObject(item, { openingHours: tempHours }));
                }
            }
        }

        if (hpMode.harmFlag) {
            // 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); // normalize
                const itemURL = normalizeURL(item.attributes.url, true, false, item);
                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(placePL);
                    if (_wl.longURL) {
                        _buttonBanner.longURL.severity = 0;
                        _buttonBanner.longURL.WLactive = false;
                    }
                    if (hpMode.harmFlag && _updateURL && itemURL !== item.attributes.url) { // Update the URL
                        phlogdev('URL formatted');
                        actions.push(new UpdateObject(item, { url: itemURL }));
                        _UPDATED_FIELDS.url.updated = true;
                    }
                    _updateURL = false;
                    _tempPNHURL = _newURL;
                }
            }
            if (hpMode.harmFlag && _updateURL && _newURL !== 'badURL' && _newURL !== item.attributes.url) { // Update the URL
                phlogdev('URL updated');
                actions.push(new UpdateObject(item, { url: _newURL }));
                _UPDATED_FIELDS.url.updated = true;
            }
        }

        // Phone formatting
        let outputFormat = '({0}) {1}-{2}';
        if (containsAny(['CA', 'CO'], [region, state2L]) && (/^\d{3}-\d{3}-\d{4}$/.test(item.attributes.phone))) {
            outputFormat = '{0}-{1}-{2}';
        } else if (region === 'SER' && !(/^\(\d{3}\) \d{3}-\d{4}$/.test(item.attributes.phone))) {
            outputFormat = '{0}-{1}-{2}';
        } else if (region === 'GLR') {
            outputFormat = '{0}-{1}-{2}';
        } else if (state2L === 'NV') {
            outputFormat = '{0}-{1}-{2}';
        } else if (_countryCode === 'CAN') {
            outputFormat = '+1-{0}-{1}-{2}';
        }
        _newPhone = normalizePhone(item.attributes.phone, outputFormat, 'existing', item, region);

        // 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(_newPhone, outputFormat);
                }
            }
        }
        if (hpMode.harmFlag && _newPhone !== item.attributes.phone) {
            phlogdev('Phone updated');
            actions.push(new UpdateObject(item, { phone: _newPhone }));
            _UPDATED_FIELDS.phone.updated = true;
        }

        // Post Office check
        if (_countryCode === 'USA' && !_newCategories.includes('PARKING_LOT')) {
            if (!_newCategories.includes('POST_OFFICE')) {
                _buttonBanner.isThisAPostOffice = Flag.IsThisAPostOffice.eval(item, _newName).flag;
            } else {
                if (hpMode.harmFlag) {
                    _customStoreFinderURL = 'https://tools.usps.com/go/POLocatorAction.action';
                    _customStoreFinder = true;
                    _buttonBanner.PlaceWebsite = new Flag.PlaceWebsite();
                    _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-z]+){1,})?$/i;
                } else {
                    postOfficeRegEx = /^post office [-–](?: cpu| vpo)?(?: [a-z]+){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 (hpMode.harmFlag) {
                    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 (hpMode.harmFlag) {
                        _newAliases.push('USPS');
                        actions.push(new UpdateObject(item, { aliases: _newAliases }));
                        _UPDATED_FIELDS.aliases.updated = true;
                    } else {
                        _buttonBanner.missingUSPSAlt = new Flag.MissingUSPSAlt();
                    }
                }
                if (!_newAliases.some(alias => /\d{5}/.test(alias))) {
                    _buttonBanner.missingUSPSZipAlt = new Flag.MissingUSPSZipAlt();
                    if (_wl.missingUSPSZipAlt) {
                        _buttonBanner.missingUSPSZipAlt.severity = 0;
                        _buttonBanner.missingUSPSZipAlt.WLactive = false;
                    }
                    // If the zip code appears in the primary name, pre-fill it in the text entry box.
                    const zipMatch = _newName.match(/\d{5}/);
                    if (zipMatch) {
                        _buttonBanner.missingUSPSZipAlt.suggestedValue = zipMatch;
                    }
                }
                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 = 0;
                        _buttonBanner.missingUSPSDescription.WLactive = false;
                    }
                }
            }
        } // END Post Office check
    } // END if (!residential && has 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 (_newCategories[0] === 'GAS_STATION' && item.attributes.brand) {
        const compressedName = item.attributes.name.toUpperCase().replace(/[^a-zA-Z0-9]/g, '');
        const compressedNewName = _newName.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 = [newBrand.toUpperCase().replace(/[^a-zA-Z0-9]/g, '')];
        if (newBrand === 'Diamond Gasoline') {
            compressedBrands.push('DIAMONDOIL');
        } else if (newBrand === 'Murphy USA') {
            compressedBrands.push('MURPHY');
        } else if (newBrand === 'Mercury Fuel') {
            compressedBrands.push('MERCURY', 'MERCURYPRICECUTTER');
        } else if (newBrand === 'Carrollfuel') {
            compressedBrands.push('CARROLLMOTORFUEL', 'CARROLLMOTORFUELS');
        }
        if (compressedBrands.every(compressedBrand => !compressedName.includes(compressedBrand) && !compressedNewName.includes(compressedBrand))) {
            _buttonBanner.gasMismatch = new Flag.GasMismatch();
            if (_wl.gasMismatch) {
                _buttonBanner.gasMismatch.WLactive = false;
            } else {
                lockOK = false;
            }
        }
    }

    // Brand checking (be sure to check this after determining if brand will be forced, when harmonzing)
    result = Flag.GasNoBrand.eval(item, newBrand);
    _buttonBanner.gasNoBrand = result.flag;
    if (result.noLock) lockOK = false;

    result = Flag.GasUnbranded.eval(item, newBrand);
    _buttonBanner.gasUnbranded = result.flag;
    if (result.noLock) lockOK = false;

    // Name check
    if (!item.attributes.residential && (!_newName || _newName.replace(/[^A-Za-z0-9]/g, '').length === 0)) {
        if (item.isParkingLot()) {
            // If it's a parking lot and not locked to R3...
            if (item.attributes.lockRank < 2) {
                lockOK = false;
                _buttonBanner.plaNameMissing = new Flag.PlaNameMissing();
            }
        } else if (!['ISLAND', 'FOREST_GROVE', 'SEA_LAKE_POOL', 'RIVER_STREAM', 'CANAL'].includes(item.attributes.categories[0])) {
            _buttonBanner.nameMissing = new Flag.NameMissing();
            lockOK = false;
        }
    }

    _buttonBanner.plaNameNonStandard = Flag.PlaNameNonStandard.eval(item, _wl).flag;

    // Public parking lot warning message:
    if (item.isParkingLot() && item.attributes.categoryAttributes && item.attributes.categoryAttributes.PARKING_LOT
        && item.attributes.categoryAttributes.PARKING_LOT.parkingType === 'PUBLIC') {
        _buttonBanner.plaIsPublic = new Flag.PlaIsPublic();
        // Add the buttons to the message.
        _buttonBanner.plaIsPublic.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('');
    }

    // 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'.split('|').includes(item.attributes.categories[0])) {
            _buttonBanner.hnMissing = new Flag.HnMissing(item);
            if (state2L === 'PR' || ['SCENIC_LOOKOUT_VIEWPOINT'].includes(item.attributes.categories[0])) {
                _buttonBanner.hnMissing.severity = 0;
                _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 = 1;
                } else {
                    _buttonBanner.hnMissing.severity = 0;
                }
            } else if (_wl.HNWL) {
                _buttonBanner.hnMissing.severity = 0;
                _buttonBanner.hnMissing.WLactive = false;
            } else {
                lockOK = false;
            }
        }
    } else if (currentHN) {
        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 = 0;
            } else {
                lockOK = false;
            }
        }
        if (updateHNflag) {
            _buttonBanner.hnDashRemoved = new Flag.HnDashRemoved();
            if (hpMode.harmFlag) {
                actions.push(new UpdateObject(item, { houseNumber: hnTemp }));
                _UPDATED_FIELDS.address.updated = true;
            } else if (hpMode.hlFlag) {
                if (item.attributes.residential) {
                    _buttonBanner.hnDashRemoved.severity = 3;
                } else {
                    _buttonBanner.hnDashRemoved.severity = 1;
                }
            }
        }
    }

    if ((!addr.city || addr.city.attributes.isEmpty)
        && !'BRIDGE|ISLAND|FOREST_GROVE|SEA_LAKE_POOL|RIVER_STREAM|CANAL|DAM|TUNNEL|JUNCTION_INTERCHANGE'.split('|').includes(item.attributes.categories[0])) {
        _buttonBanner.cityMissing = new Flag.CityMissing();
        if (item.attributes.residential && hpMode.hlFlag) {
            _buttonBanner.cityMissing.severity = 1;
        }
        lockOK = false;
    }
    if (addr.city && (!addr.street || addr.street.isEmpty)
        && !'BRIDGE|ISLAND|FOREST_GROVE|SEA_LAKE_POOL|RIVER_STREAM|CANAL|DAM|TUNNEL|JUNCTION_INTERCHANGE'.split('|').includes(item.attributes.categories[0])) {
        _buttonBanner.streetMissing = new Flag.StreetMissing();
        if (['SCENIC_LOOKOUT_VIEWPOINT'].includes(item.attributes.categories[0])) {
            _buttonBanner.streetMissing.severity = 1;
        } else {
            lockOK = false;
        }
    }

    _buttonBanner.notAHospital = Flag.NotAHospital.eval(_newCategories).flag;

    // CATEGORY vs. NAME checks
    result = Flag.ChangeToPetVet.eval(_newName, _newCategories);
    if (result.flag) {
        _buttonBanner.changeToPetVet = result.flag;
        if (!result.lockOK) lockOK = false;
    }

    result = Flag.ChangeToDoctorClinic.eval(item, _newCategories, hpMode, pnhNameRegMatch);
    if (result.flag) {
        _buttonBanner.changeToPetVet = result.flag;
        if (!result.lockOK) lockOK = false;
    }

    result = Flag.NotASchool.eval(_newName, _newCategories);
    if (result.flag) {
        _buttonBanner.notASchool = result.flag;
        if (!result.lockOK) lockOK = false;
    }

    // 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 = 0;
            _buttonBanner.phoneMissing.WLactive = false;
        }
        if (_buttonBanner.urlMissing) {
            _buttonBanner.urlMissing.severity = 0;
            _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 hasRestAreaCategory = categories.includes('REST_AREAS');
    const oldName = item.attributes.name;
    if (/rest area/i.test(oldName) || /rest stop/i.test(oldName) || /service plaza/i.test(oldName) || hasRestAreaCategory) {
        if (hasRestAreaCategory) {
            if (categories.includes('SCENIC_LOOKOUT_VIEWPOINT')) {
                if (!_wl.restAreaScenic) _buttonBanner.restAreaScenic = new Flag.RestAreaScenic();
            }
            if (categories.includes('TRANSPORTATION')) {
                _buttonBanner.restAreaNoTransportation = new Flag.RestAreaNoTransportation();
            }
            if (item.isPoint()) { // needs to be area
                _buttonBanner.areaNotPoint = new Flag.AreaNotPoint();
            }
            _buttonBanner.pointNotArea = null;
            _buttonBanner.unmappedRegion = null;

            if (categories.includes('GAS_STATION')) {
                _buttonBanner.restAreaGas = new Flag.RestAreaGas();
            }

            if (oldName.match(/^Rest Area.* - /) === null) {
                _buttonBanner.restAreaName = new Flag.RestAreaName();
                if (_wl.restAreaName) {
                    _buttonBanner.restAreaName.WLactive = false;
                }
            } else if (hpMode.harmFlag) {
                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;
                    phlogdev('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 (hpMode.harmFlag) {
                _buttonBanner2.restAreaWiki.active = true;
                _buttonBanner2.placesWiki.active = false;
            }

            // missing address ok
            _buttonBanner.streetMissing = null;
            _buttonBanner.cityMissing = null;
            _buttonBanner.hnMissing = null;
            if (_buttonBanner.urlMissing) {
                _buttonBanner.urlMissing.WLactive = false;
                _buttonBanner.urlMissing.severity = 0;
            }
            if (_buttonBanner.phoneMissing) {
                _buttonBanner.phoneMissing.severity = 0;
                _buttonBanner.phoneMissing.WLactive = false;
            }
        } else if (!_wl.restAreaSpec) {
            _buttonBanner.restAreaSpec = new Flag.RestAreaSpec();
        }
    }

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

    if (hpMode.harmFlag) {
        phlogdev(`Severity: ${_severityButt}; lockOK: ${lockOK}`);
    }
    // Place locking
    // final formatting of desired lock levels
    let levelToLock;
    if (_pnhLockLevel !== -1 && hpMode.harmFlag) {
        phlogdev(`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

    // If gas station is missing brand, don't flag if place is locked.
    if (_buttonBanner.gasNoBrand) {
        if (item.attributes.lockRank >= levelToLock) {
            _buttonBanner.gasNoBrand = null;
        } else {
            _buttonBanner.gasNoBrand.message = `Lock to L${levelToLock + 1}+ to verify no gas brand.`;
        }
    }

    // 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 <= 2) {
        _buttonBanner.extProviderMissing.severity = 3;
        _severityButt = 3;
        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 = () => {
                const action = new UpdateObject(item, { lockRank: levelToLock });
                W.model.actionManager.add(action);
                _UPDATED_FIELDS.lock.updated = true;
                harmonizePlaceGo(item, 'harmonize');
            };
        }
    }

    let hlLockFlag = false;
    if (lockOK && _severityButt < 2) {
        if (item.attributes.lockRank < levelToLock) {
            if (hpMode.harmFlag) {
                phlogdev('Venue locked!');
                actions.push(new UpdateObject(item, { lockRank: levelToLock }));
                _UPDATED_FIELDS.lock.updated = true;
            } else if (hpMode.hlFlag) {
                hlLockFlag = true;
            }
        }
        _buttonBanner.placeLocked = new Flag.PlaceLocked();
    }

    // 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();
    }

    // RPP Locking option for R3+
    if (item.attributes.residential) {
        if (_USER.isDevUser || _USER.isBetaUser || _USER.rank >= 3) { // Allow residential point locking by R3+
            _rppLockString = '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)) {
                        _rppLockString += `<option value="${llix}" selected="selected">${llix}</option>`;
                        ddlSelected = true;
                    } else {
                        _rppLockString += `<option value="${llix}">${llix}</option>`;
                    }
                }
            }
            _rppLockString += '</select>';
            _buttonBanner.lockRPP = new Flag.LockRPP();
            _buttonBanner.lockRPP.message = `Current lock: ${parseInt(item.attributes.lockRank, 10) + 1}. ${_rppLockString} ?`;
        }
    }

    // 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 && _severityButt < 3) {
        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 (hpMode.hlFlag) {
        // get severities from the banners
        _severityButt = 0;
        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
                        _severityButt = Math.max(_buttonBanner[tempKey].severity, _severityButt);
                    }
                } else {
                    _severityButt = Math.max(_buttonBanner[tempKey].severity, _severityButt);
                }
            }
        });

        // Special case flags
        if (item.attributes.lockRank === 0 && (item.attributes.categories.includes('HOSPITAL_MEDICAL_CARE')
            || item.attributes.categories.includes('HOSPITAL_URGENT_CARE') || item.isGasStation())) {
            _severityButt = 5;
        }

        if (_severityButt === 0 && hlLockFlag) {
            _severityButt = 'lock';
        }
        if (_severityButt === 1 && hlLockFlag) {
            _severityButt = 'lock1';
        }
        if (item.attributes.adLocked) {
            _severityButt = 'adLock';
        }

        return _severityButt;
    }

    // *** 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)) {
        if ($('#WMEPH-DisableDFZoom').prop('checked')) { // don't zoom and pan for results outside of FOV
            _duplicateName = findNearbyDuplicate(_newName, _newAliases, item, false);
        } else {
            _duplicateName = findNearbyDuplicate(_newName, _newAliases, item, true);
        }
        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
                /* if (confirm('WMEPH: Dupefinder Error!\nClick 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}${_DEV_VERSION_STR}\nPermalink: ${placePL}\nPlace name: ${
                            item.attributes.name}\nCountry: ${addr.country.name}\n--------\nDescribe the error:\nDupeID mismatch with dupeName list`
                    });
                } */
                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}${_DEV_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 => {
                    _wlKeyName = 'dupeWL';
                    if (!_venueWhitelist.hasOwnProperty(itemID)) { // If venue is NOT on WL, then add it.
                        _venueWhitelist[itemID] = { dupeWL: [] };
                    }
                    if (!_venueWhitelist[itemID].hasOwnProperty(_wlKeyName)) { // If dupeWL key is not in venue WL, then initialize it.
                        _venueWhitelist[itemID][_wlKeyName] = [];
                    }
                    _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(_wlKeyName)) { // If dupeWL key is not in venue WL, then initialize it.
                        _venueWhitelist[dID][_wlKeyName] = [];
                    }
                    _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: 2,
                        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 = 3;
            }
            // show stats if HN out of range
            phlogdev(`HNs: ${dupeHNRangeListSorted}`);
            phlogdev(`Distances: ${_dupeHNRangeDistList}`);
            phlogdev(`arrayHNRatio: ${arrayHNRatio}`);
            phlogdev(`HN Ratio Score: ${arrayHNRatio[Math.round(arrayHNRatio.length / 2)]}`);
        }
    }

    executeMultiAction(actions);

    if (hpMode.harmFlag) {
        // 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();
    }

    if (_buttonBanner.lockRPP) _buttonBanner.lockRPP.message = `Current lock: ${parseInt(item.attributes.lockRank, 10) + 1}. ${_rppLockString} ?`;

    // 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;
    phlogdev('Building banners');
    let dupesFound = 0;
    let rowData;
    let $rowDiv;
    let rowDivs = [];
    _severityButt = 0;

    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
                _severityButt = Math.max(rowData.severity, _severityButt);
                $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 === 3) {
                $rowDiv.addClass('red');
            } else if (rowData.severity === 2) {
                $rowDiv.addClass('yellow');
            } else if (rowData.severity === 1) {
                $rowDiv.addClass('blue');
            } else if (rowData.severity === 0) {
                $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
                    _severityButt = Math.max(rowData.severity, _severityButt);
                    $rowDiv.append(
                        $('<button>', { class: 'btn btn-success btn-xs wmephwl-btn', id: `WMEPH_WL${tempKey}`, title: rowData.WLtitle })
                            .text('WL')
                    );
                }
            } else {
                _severityButt = Math.max(rowData.severity, _severityButt);
            }
            if (rowData.suffixMessage) {
                $rowDiv.append($('<div>').css({ 'margin-top': '2px' }).append(rowData.suffixMessage));
            }

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

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

    if ($('#WMEPH_banner').length === 0) {
        $('<div id="WMEPH_banner">').prependTo('.contents');
    } else {
        $('#WMEPH_banner').empty();
    }
    let bgColor;
    switch (_severityButt) {
        case 1:
            bgColor = 'rgb(50, 50, 230)'; // blue
            break;
        case 2:
            bgColor = 'rgb(217, 173, 42)'; // yellow
            break;
        case 3:
            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);
            _severityButt = Math.max(_buttonBanner2[tempKey].severity, _severityButt);
        }
    });

    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': '4px',
            '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);

    // Prefill zip code text box
    if (_buttonBanner.missingUSPSZipAlt && _buttonBanner.missingUSPSZipAlt.suggestedValue) {
        $('input#WMEPH-zipAltNameAdd').val(_buttonBanner.missingUSPSZipAlt.suggestedValue);
    }

    // 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;
        W.model.actionManager.add(new UpdateObject(selectedVenue, { categoryAttributes: { PARKING_LOT: newAttr } }));
        harmonizePlaceGo(selectedVenue, 'harmonize');
    });

    $('.wmeph-pla-lot-type-btn').click(evt => {
        const selectedVenue = getSelectedVenue();
        const selectedValue = $(evt.currentTarget).data('lot-type');
        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.parkingType = selectedValue;
        W.model.actionManager.add(new UpdateObject(selectedVenue, { categoryAttributes: { PARKING_LOT: newAttr } }));
        harmonizePlaceGo(selectedVenue, 'harmonize');
    });

    $('.wmeph-pla-cost-type-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.costType = selectedValue;
        W.model.actionManager.add(new UpdateObject(selectedVenue, { categoryAttributes: { PARKING_LOT: newAttr } }));
        harmonizePlaceGo(selectedVenue, 'harmonize');
    });

    // 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();
        }
    });

    // 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();
        }
    });

    // 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` });
        }, 100);
    }
    $('#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';
            }
        });

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

    // NOTE: Leave these wrapped in the "() => ..." functions, to make sure "this" is bound properly.
    if (_buttonBanner.noHours) {
        $('#WMEPH_noHours').click(() => _buttonBanner.noHours.addHoursAction());
        $('#WMEPH_noHours_2').click(() => _buttonBanner.noHours.replaceHoursAction());
    }

    if (_textEntryValues) {
        _textEntryValues.forEach(entry => $(`#${entry.id}`).val(entry.val));
    }
} // END assemble Banner function

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(.25)', filter: 'opacity(.25)' });
                    }
                    $rowDiv.append($input);
                }
            });
            if ($rowDiv.length) {
                $rowDiv.prepend('<span class="control-label">Add services:</span><br>');
            }
            rowDivs.push($rowDiv);
        }
        if ($('#WMEPH_services').length === 0) {
            $('#WMEPH_banner').after($('<div id="WMEPH_services">').css({
                'background-color': '#eee',
                color: 'black',
                'font-size': '15px',
                padding: '4px',
                'margin-left': '4px',
                'margin-right': 'auto'
            }));
        } 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 = 0;
        assembleBanner();
    };
    return button;
}

// Display run button on place sidebar
function displayRunButton() {
    const betaDelay = 100;
    setTimeout(() => {
        if ($('#WMEPH_runButton').length === 0) {
            $('<div id="WMEPH_runButton">').prependTo('.contents');
        }
        if ($('#runWMEPH').length === 0) {
            const devVersSuffix = _IS_DEV_VERSION ? '-β' : '';
            const strButt1 = `<input class="btn btn-primary wmeph-fat-btn" id="runWMEPH" title="Run WMEPH${
                devVersSuffix} on Place" type="button" value="Run WMEPH${devVersSuffix}">`;
            $('#WMEPH_runButton').append(strButt1);
        }
        const btn = document.getElementById('runWMEPH');
        if (btn !== null) {
            btn.onclick = () => {
                harmonizePlace();
            };
        } else {
            setTimeout(bootstrapRunButton, 100);
        }
        showOpenPlaceWebsiteButton();
        showSearchButton();
    }, betaDelay);
}

// Displays the Open Place Website button.
function showOpenPlaceWebsiteButton() {
    const venue = getSelectedVenue();
    if (venue) {
        let openPlaceWebsiteURL = venue.attributes.url;
        if (openPlaceWebsiteURL && openPlaceWebsiteURL.replace(/[^A-Za-z0-9]/g, '').length > 2) {
            if (!$('#WMEPHurl').length) {
                const strButt1 = '<input class="btn btn-success btn-xs wmeph-fat-btn" id="WMEPHurl" title="Open place URL" type="button" value="Website">';
                $('#runWMEPH').after(strButt1);
                const btn = document.getElementById('WMEPHurl');
                if (btn !== null) {
                    btn.onclick = () => {
                        openPlaceWebsiteURL = venue.attributes.url;
                        if (openPlaceWebsiteURL.match(/^http/i) === null) {
                            openPlaceWebsiteURL = `http://${openPlaceWebsiteURL}`;
                        }
                        if ($('#WMEPH-WebSearchNewTab').prop('checked')) {
                            window.open(openPlaceWebsiteURL);
                        } else {
                            window.open(openPlaceWebsiteURL, _SEARCH_RESULTS_WINDOW_NAME, _searchResultsWindowSpecs);
                        }
                    };
                } else {
                    setTimeout(bootstrapRunButton, 100);
                }
            }
        } else if ($('#WMEPHurl').length) {
            $('#WMEPHurl').remove();
        }
    }
}

function showSearchButton() {
    const venue = getSelectedVenue();
    if (venue && $('#wmephSearch').length === 0) {
        const strButt1 = '<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">';
        $('#WMEPH_runButton').append(strButt1);
        const btn = document.getElementById('wmephSearch');
        if (btn !== null) {
            btn.onclick = () => {
                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.');
                }
            };
        } else {
            setTimeout(bootstrapRunButton, 100);
        }
    }
}

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

// WMEPH Clone Tool
function displayCloneButton() {
    let betaDelay = 80;
    if (_IS_DEV_VERSION) { betaDelay = 300; }
    setTimeout(() => {
        if ($('#WMEPH_runButton').length === 0) {
            $('<div id="WMEPH_runButton">').prependTo('.contents');
        }
        const venue = getSelectedVenue();
        if (venue) {
            showOpenPlaceWebsiteButton();
            if ($('#clonePlace').length === 0) {
                let strButt1 = '<div style="margin-bottom: 3px;"></div><input class="btn btn-warning btn-xs wmeph-btn" '
                    + 'id="clonePlace" title="Copy place info" type="button" value="Copy" style="font-weight:normal">'
                    + ' <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"><br>';
                $('#WMEPH_runButton').append(strButt1);
                createCloneCheckbox('WMEPH_runButton', 'WMEPH_CPhn', 'HN');
                createCloneCheckbox('WMEPH_runButton', 'WMEPH_CPstr', 'Str');
                createCloneCheckbox('WMEPH_runButton', 'WMEPH_CPcity', 'City');
                createCloneCheckbox('WMEPH_runButton', 'WMEPH_CPurl', 'URL');
                createCloneCheckbox('WMEPH_runButton', 'WMEPH_CPph', 'Ph');
                $('#WMEPH_runButton').append('<br>');
                createCloneCheckbox('WMEPH_runButton', 'WMEPH_CPdesc', 'Desc');
                createCloneCheckbox('WMEPH_runButton', 'WMEPH_CPserv', 'Serv');
                createCloneCheckbox('WMEPH_runButton', 'WMEPH_CPhrs', 'Hrs');
                strButt1 = '<input class="btn btn-info btn-xs wmeph-btn" id="checkAllClone" title="Check all" '
                    + 'type="button" value="All" style="font-weight:normal"> <input class="btn btn-info btn-xs '
                    + 'wmeph-btn" id="checkAddrClone" title="Check Address" type="button" value="Addr" style="font-weight:normal">'
                    + ' <input class="btn btn-info btn-xs wmeph-btn" id="checkNoneClone" title="Check none" '
                    + 'type="button" value="None" style="font-weight:normal"><br>';
                $('#WMEPH_runButton').append(strButt1);
            }
            let btn = document.getElementById('clonePlace');
            if (btn !== null) {
                btn.onclick = () => {
                    _cloneMaster = {};
                    _cloneMaster.addr = venue.getAddress();
                    if (_cloneMaster.addr.hasOwnProperty('attributes')) {
                        _cloneMaster.addr = _cloneMaster.addr.attributes;
                    }
                    _cloneMaster.houseNumber = venue.attributes.houseNumber;
                    _cloneMaster.url = venue.attributes.url;
                    _cloneMaster.phone = venue.attributes.phone;
                    _cloneMaster.description = venue.attributes.description;
                    _cloneMaster.services = venue.attributes.services;
                    _cloneMaster.openingHours = venue.attributes.openingHours;
                    _cloneMaster.isPLA = venue.isParkingLot();
                    phlogdev('Place Cloned');
                };
            } else {
                setTimeout(bootstrapRunButton, 100);
                return;
            }
            btn = document.getElementById('pasteClone');
            if (btn !== null) {
                btn.onclick = () => {
                    clonePlace(getSelectedVenue());
                };
            } else {
                setTimeout(bootstrapRunButton, 100);
            }
            btn = document.getElementById('checkAllClone');

            if (btn !== null) {
                btn.onclick = () => {
                    setCheckboxes(['WMEPH_CPhn', 'WMEPH_CPstr', 'WMEPH_CPcity', 'WMEPH_CPurl', 'WMEPH_CPph', 'WMEPH_CPserv',
                        'WMEPH_CPdesc', 'WMEPH_CPhrs'], true);
                };
            } else {
                setTimeout(bootstrapRunButton, 100);
            }
            btn = document.getElementById('checkAddrClone');
            if (btn !== null) {
                btn.onclick = () => {
                    setCheckboxes(['WMEPH_CPhn', 'WMEPH_CPstr', 'WMEPH_CPcity'], true);
                    setCheckboxes(['WMEPH_CPurl', 'WMEPH_CPph', 'WMEPH_CPserv', 'WMEPH_CPdesc', 'WMEPH_CPhrs'], false);
                };
            } else {
                setTimeout(bootstrapRunButton, 100);
            }
            btn = document.getElementById('checkNoneClone');
            if (btn !== null) {
                btn.onclick = () => {
                    setCheckboxes(['WMEPH_CPhn', 'WMEPH_CPstr', 'WMEPH_CPcity', 'WMEPH_CPurl', 'WMEPH_CPph', 'WMEPH_CPserv',
                        'WMEPH_CPdesc', 'WMEPH_CPhrs'], false);
                };
            } else {
                setTimeout(bootstrapRunButton, 100);
            }
        }
    }, betaDelay);
} // END displayCloneButton funtion


// Catch PLs and reloads that have a place selected already and limit attempts to about 10 seconds
function bootstrapRunButton(numAttempts) {
    numAttempts = numAttempts || 0;
    if (numAttempts < 10) {
        if (W.selectionManager.getSelectedFeatures().length === 1) {
            const venue = getSelectedVenue();
            if (venue && venue.isApproved()) {
                displayRunButton();
                showOpenPlaceWebsiteButton();
                showSearchButton();
                getPanelFields();
                if (localStorage.getItem('WMEPH-EnableCloneMode') === '1') {
                    displayCloneButton();
                }
            }
        } else {
            setTimeout(bootstrapRunButton(numAttempts + 1), 1000);
        }
    }
}

// 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('landmark-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('landmark-edit-general')) {
            _PANEL_FIELDS.navTabGeneral = pfix;
        }
        if (pfa.includes('landmark-edit-more')) {
            _PANEL_FIELDS.navTabMore = pfix;
        }
    }
}

// Function to clone info from a place
function clonePlace() {
    phlog('Cloning info...');
    if (_cloneMaster !== null && _cloneMaster.hasOwnProperty('url')) {
        const venue = getSelectedVenue();
        const cloneItems = {};
        let updateItem = false;
        if (isChecked('WMEPH_CPhn')) {
            cloneItems.houseNumber = _cloneMaster.houseNumber;
            updateItem = true;
        }
        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) {
            W.model.actionManager.add(new UpdateObject(venue, cloneItems));
            phlogdev('Item details cloned');
        }

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

        if (copyStreet || copyCity) {
            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
            };
            updateAddress(venue, itemRepl);
            phlogdev('Item address cloned');
        }
    } else {
        phlog('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 to check overlapping hours
function checkHours(hoursObj) {
    if (hoursObj.length === 1) {
        return true;
    }

    for (let day2Ch = 0; day2Ch < 7; day2Ch++) { // Go thru each day of the week
        const daysObj = [];
        for (let hourSet = 0; hourSet < hoursObj.length; hourSet++) { // For each set of hours
            if (hoursObj[hourSet].days.includes(day2Ch)) { // pull out hours that are for the current day, add 2400 if it goes past midnight, and store
                const fromHourTemp = hoursObj[hourSet].fromHour.replace(/:/g, '');
                let toHourTemp = hoursObj[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 false;
                    }
                    if (daysObj[hourSetCheck2][1] > daysObj[hourSetCheck1][0] && daysObj[hourSetCheck2][1] < daysObj[hourSetCheck1][1]) {
                        return false;
                    }
                }
            }
        }
    }
    return true;
}

// 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)`;
                        phlogdev(`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

// On selection of new item:
function checkSelection() {
    const venue = getSelectedVenue();
    if (venue && venue.isApproved()) {
        displayRunButton();
        getPanelFields();
        if ($('#WMEPH-EnableCloneMode').prop('checked')) {
            displayCloneButton();
        }
        for (let dvtix = 0; dvtix < _dupeIDList.length; dvtix++) {
            if (venue.attributes.id === _dupeIDList[dvtix]) { // If the user selects a place in the dupe list, don't clear the labels yet
                return;
            }
        }
    } else {
        // Remove the run button div if it's being displayed.
        $('#WMEPH_runButton').remove();
    }
    // If the selection is anything else, clear the labels
    _dupeLayer.destroyFeatures();
    _dupeLayer.setVisibility(false);
}

// Functions to infer address from nearby segments
function inferAddress(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;

    const venue = getSelectedVenue();

    // 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 action = new UpdateFeatureAddress(feature, newAttributes);
        if (actions) {
            actions.push(action);
        } else {
            W.model.actionManager.add(action);
        }
        phlogdev('Address inferred and updated');
    }
} // END updateAddress function

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

// Function that inserts a string or a string array into another string array at index ix and removes any duplicates
function insertAtIX(array1, array2, ix) { // array1 is original string, array2 is the inserted string, at index ix
    const arrayNew = array1.slice(); // slice the input array so it doesn't change
    if (typeof (array2) === 'string') { array2 = [array2]; } // if a single string, convert to an array
    if (typeof (array2) === 'object') { // only apply to inserted arrays
        const arrayTemp = arrayNew.splice(ix); // split and hold the first part
        arrayNew.push(...array2); // add the insert
        arrayNew.push(...arrayTemp); // add the tail end of original
    }
    return _.uniq(arrayNew); // remove any duplicates (so the function can be used to move the position of a string)
}

// 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)) {
        // phlogdev(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_DEV_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
        _WLSToMerge = validateWLS($('#WMEPH-WLInput').val());
        if (_WLSToMerge) {
            phlog('Whitelists merged!');
            _venueWhitelist = mergeWL(_venueWhitelist, _WLSToMerge);
            saveWhitelistToLS(true);
            $wlToolsMsg.append('<p style="color:green">Whitelist data merged<p>');
            $wlInput.val('');
        } else { // try compressed WL
            _WLSToMerge = validateWLS(LZString.decompressFromUTF16($('#WMEPH-WLInput').val()));
            if (_WLSToMerge) {
                phlog('Whitelists merged!');
                _venueWhitelist = mergeWL(_venueWhitelist, _WLSToMerge);
                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-EnableServices');
        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
    }

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

function addWmephTab() {
    // Set up the CSS
    GM_addStyle(_CSS_ARRAY.join('\n'));

    const $container = $('<div class="active">');
    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 active" id="sidepanel-harmonizer"></div>');
    const $highlighterTab = $('<div class="tab-pane" id="sidepanel-highlighter"></div>');
    const $wlToolsTab = $('<div class="tab-pane" id="sidepanel-wltools"></div>');
    const $moderatorsTab = $('<div class="tab-pane" id="sidepanel-pnh-moderators"></div>');
    $tabContent.append($harmonizerTab, $highlighterTab, $wlToolsTab, $moderatorsTab);
    $container.append($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-EnableServices', 'Enable automatic addition of common services');
        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%">',
    );

    $harmonizerTab.append(
        $('<div>').append(
            $('<div>', { style: 'font-weight: bold; margin-bottom: 6px;' }).text('Recent updates'),
            Object.keys(_WHATS_NEW_LIST).map(
                version => $('<div>').append(
                    $('<div>').text(version),
                    $('<ul>', { style: 'margin-left: -23px;' }).append(
                        _WHATS_NEW_LIST[version].map(textLine => $('<li>').text(textLine))
                    )
                )
            )
        )
    );

    // 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', 'nnote'],
        GLR: ['JustinS83'],
        HI: ['Nacron'],
        MAR: ['jr1982jr', 'nzahn1', 'stephenr1966'],
        NER: ['jaywazin', 'SNYOWL'],
        NOR: ['Joyriding', 'PesachZ'],
        NWR: ['dmee92', 'SkyviewGuru'],
        PLN: ['bretmcvey', 'dmee92', 'ehepner1977'],
        SAT: ['crazycaveman', 'whathappened15', 'xanderb'],
        SCR: ['jm6087'],
        SER: ['driving79', 'willdanneriv', 'ardan74', 'itzwolf'],
        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(', '))
                )
            ))
        )
    );

    new WazeWrap.Interface.Tab(`WMEPH${_IS_DEV_VERSION ? '-β' : ''}`, $container.html(), initWmephTab, null);
}

function createCloneCheckbox(divID, settingID, textDescription) {
    $(`#${divID}`).append(`<input type="checkbox" id="${settingID}">${textDescription}</input>&nbsp&nbsp`);
    $(`#${settingID}`).click(() => saveSettingToLocalStorage(settingID));
    if (localStorage.getItem(settingID) === '1') {
        $(`#${settingID}`).trigger('click');
    }
}

// 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 = insertAtIX(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) {
    //     phlog('Waiting for PL div');
    //     setTimeout(getCurrentPL, 500);
    //     return;
    // }

    if ($('.WazeControlPermalink .permalink').attr('href').length > 0) {
        return $('.WazeControlPermalink .permalink').attr('href');
    }
    if ($('.WazeControlPermalink').children('.fa-link').length > 0) {
        return $('.WazeControlPermalink').children('.fa-link')[0].href;
    }
    return '';
}

// Sets up error reporting
function reportError(data) {
    data.preview = 'Preview';
    data.attach_sig = 'on';
    if (_PM_USER_LIST.hasOwnProperty('WMEPH') && _PM_USER_LIST.WMEPH.approvalActive) {
        data[`address_list[u][${_PM_USER_LIST.WMEPH.modID}]`] = 'to';
        newForumPost('https://www.waze.com/forum/ucp.php?i=pm&mode=compose', data);
    } else {
        data.addbbcode20 = 'to';
        data.notify = 'on';
        newForumPost(`${_URLS.forum}#preview`, data);
    }
} // END reportError function

// Make a populated post on a forum thread
function newForumPost(url, data) {
    const form = document.createElement('form');
    form.target = '_blank';
    form.action = url;
    form.method = 'post';
    form.style.display = 'none';
    Object.keys(data).forEach(k => {
        let input;
        if (k === 'message') {
            input = document.createElement('textarea');
        } else if (k === 'username') {
            input = document.createElement('username_list');
        } else {
            input = document.createElement('input');
        }
        input.name = k;
        input.value = data[k];
        // input.type = 'hidden'; // 2018-07/10 (mapomatic) Not sure if this is required, but was causing an error when setting on the textarea object.
        form.appendChild(input);
    });
    document.getElementById('WMEPH_formDiv').appendChild(form);
    form.submit();
    document.getElementById('WMEPH_formDiv').removeChild(form);
    return true;
} // END newForumPost function

/**
 * Updates the geometry of a place.
 * @param place {Waze venue object} The place to update.
 * @param newGeometry {OpenLayers.Geometry} The new geometry for the place.
 */
function updateFeatureGeometry(place, newGeometry) {
    let oldGeometry;
    const model = W.model.venues;
    if (place && place.CLASS_NAME === 'Waze.Feature.Vector.Venue' && newGeometry && (newGeometry instanceof OpenLayers.Geometry.Point
        || newGeometry instanceof OpenLayers.Geometry.Polygon)) {
        oldGeometry = place.attributes.geometry;
        W.model.actionManager.add(new UpdateFeatureGeometry(place, model, oldGeometry, newGeometry));
    }
}

function placeHarmonizerInit() {
    _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');

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

    _USER.ref = W.loginManager.user;
    _USER.name = _USER.ref.userName;
    _USER.rank = _USER.ref.rank + 1; // get editor's level (actual level)
    _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');
    });

    addWmephTab(); // initialize the settings tab

    // Event listeners
    W.selectionManager.events.registerPriority('selectionchanged', this, () => errorHandler(checkSelection));
    W.model.venues.on('objectssynced', () => errorHandler(destroyDupeLabels));
    W.model.venues.on('objectssynced', e => errorHandler(() => syncWL(e)));
    W.model.venues.on('objectschanged', () => errorHandler(onObjectsChanged));

    // 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);
        phlogdev(`Removed ${removedWLCount} venues with temporary ID's from WL store`);
    }

    if (!_wmephBetaList || _wmephBetaList.length === 0) {
        if (_IS_DEV_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
    }

    _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
    bootstrapRunButton();

    // Setup highlight colors
    initializeHighlights();

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

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

function placeHarmonizerBootstrap() {
    if (W && W.loginManager && W.loginManager.user && W.map && WazeWrap && WazeWrap.Ready && W.model.categoryBrands.PARKING_LOT && require) {
        placeHarmonizerInit();
    } else {
        phlog('Waiting for WME map and login...');
        setTimeout(placeHarmonizerBootstrap, 200);
    }
}

const SPREADSHEET_ID = '1pBz4l4cNapyGyzfMJKqA4ePEFLkmz2RryAt1UV39B4g';
const SPREADSHEET_RANGE = '2019.01.20.001!A2:L';
const API_KEY = 'YTJWNVBVRkplbUZUZVVObU1YVXpSRVZ3ZW5OaFRFSk1SbTR4VGxKblRURjJlRTFYY3pOQ2NXZElPQT09';

function downloadPnhData() {
    const dec = s => atob(atob(s));
    const getSpreadsheetUrl = (id, range, key) => `https://sheets.googleapis.com/v4/spreadsheets/${id}/values/${range}?${dec(key)}`;

    // 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]);

    $.getJSON(getSpreadsheetUrl(SPREADSHEET_ID, SPREADSHEET_RANGE, API_KEY)).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);

        placeHarmonizerBootstrap();
    }).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);
    });
}

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.
    downloadPnhData();
}

bootstrap();

QingJ © 2025

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