WME Place Harmonizer

Harmonizes, formats, and locks a selected place

当前为 2019-05-17 提交的版本,查看 最新版本

您需要先安装一个扩展,例如 篡改猴Greasemonkey暴力猴,之后才能安装此脚本。

You will need to install an extension such as Tampermonkey to install this script.

您需要先安装一个扩展,例如 篡改猴暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴Userscripts ,之后才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。

您需要先安装用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name        WME Place Harmonizer
// @namespace   WazeUSA
// @version     1.3.145
// @description Harmonizes, formats, and locks a selected place
// @author      WMEPH Development Group
// @include     /^https:\/\/(www|beta)\.waze\.com\/(?!user\/)(.{2,6}\/)?editor\/?.*$/
// @require     https://greasyfork.org/scripts/24851-wazewrap/code/WazeWrap.js
// @require     https://greasyfork.org/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 I18n */
/* global $ */
/* global W */
/* global GM_info */
/* global require */
/* global performance */
/* global OL */
/* global _ */
/* global Node */
/* global WazeWrap */
/* global unsafeWindow */
/* global LZString */
/* global Promise */

/* eslint-disable */
(function () {
    'use strict';

    // Quit if another version of WMEPH is already running.
    if (unsafeWindow.wmephRunning) {
        alert('Multiple versions of Place Harmonizer are turned on.  Only one will be enabled.');
        return;
    } else {
        unsafeWindow.wmephRunning = 1;
    }

    // Script update info
    const _WHATS_NEW_LIST = [  // New in this version
        '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.',
        '1.3.143: 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.',
        '1.3.138: NEW: Added "ramp" to list of recognized words for parking lots.',
        '1.3.138: FIXED: "Is this a doctor/clinic" flag will only display if Office or Personal Care place was last edited before 3/28/2017',
        '1.3.138: FIXED: Removed feature that would hide suggested category buttons for Shopping / Services (feature is now in PIE).'
    ];
    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;} '
    ];
    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 _PNH_DATA = { USA: {}, CAN: {} };
    const _CATEGORY_LOOKUP = {};
    const _DEFAULT_HOURS_TEXT = 'Paste Hours Here';
    const _MAX_CACHE_SIZE = 25000;
    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.
    var hospitalPartMatch, hospitalFullMatch, animalPartMatch, animalFullMatch, schoolPartMatch, schoolFullMatch;  // vars for cat-name checking
    var WMEPHdevList, WMEPHbetaList;  // Userlists
    var devVersStr = _IS_DEV_VERSION ? 'Beta' : '';  // strings to differentiate DOM elements between regular and beta script
    var WMEServicesArray = ['VALLET_SERVICE', 'DRIVETHROUGH', 'WI_FI', 'RESTROOMS', 'CREDIT_CARDS', 'RESERVATIONS', 'OUTSIDE_SEATING', 'AIR_CONDITIONING', 'PARKING_FOR_CUSTOMERS', 'DELIVERIES', 'TAKE_AWAY', 'WHEELCHAIR_ACCESSIBLE', 'DISABILITY_PARKING'];
    var collegeAbbreviations = 'USF|USFSP|UF|UCF|UA|UGA|FSU|UM|SCP|FAU|FIU';
    var shortcutParse, modifKey = 'Alt+';
    var venueWhitelist, venueWhitelistStr, WLSToMerge, wlKeyName, wlButtText = 'WL';  // Whitelisting vars
    var WLlocalStoreName = 'WMEPH-venueWhitelistNew';
    var WLlocalStoreNameCompressed = 'WMEPH-venueWhitelistCompressed';
    var _dupeLayer, dupeIDList = [], dupeHNRangeList, dupeHNRangeIDList, dupeHNRangeDistList;
    // Web search Window forming:
    var 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) + '"';
    var searchResultsWindowName = '"WMEPH Search Results"';
    var WMEPHmousePosition;
    var cloneMaster = null;
    var bannButt, bannButt2, bannServ, bannDupl;  // Banner Buttons objects
    var RPPLockString = 'Lock?';
    var panelFields = {};  // the fields for the sidebar
    var MultiAction = require('Waze/Action/MultiAction');
    var UpdateObject = require('Waze/Action/UpdateObject');
    var UpdateFeatureGeometry = require('Waze/Action/UpdateFeatureGeometry');
    var UpdateFeatureAddress = require('Waze/Action/UpdateFeatureAddress');
    var OpeningHour = require("Waze/Model/Objects/OpeningHour");
    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'
    };
    var userLanguage;
    // lock levels are offset by one
    var lockLevel2 = 1, lockLevel4 = 3;
    var defaultLockLevel = lockLevel2, PNHLockLevel;
    var PMUserList = { // user names and IDs for PM functions
        SER: { approvalActive: true, modID: '17083181', modName: 'itzwolf' },
        WMEPH: { approvalActive: true, modID: '2647925', modName: 'MapOMatic' }
    };
    var severityButt = 0;  // error tracking to determine banner color (action buttons)
    var duplicateName = '';
    var catTransWaze2Lang;  // pulls the category translations
    var itemID, newName, optionalAlias, newURL, tempPNHURL = '', newPhone;
    var newAliases = [], newAliasesTemp = [], newCategories = [];
    // 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('|'),
        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('|')
    };
    var newPlaceURL, approveRegionURL;
    var customStoreFinder = false;  // switch indicating place-specific custom store finder url
    var customStoreFinderLocal = false;  // switch indicating place-specific custom store finder url with localization option (GPS/addr)
    var customStoreFinderURL = '';  // switch indicating place-specific custom store finder url
    var customStoreFinderLocalURL = '';  // switch indicating place-specific custom store finder url with localization option (GPS/addr)
    var updateURL;

    // Split out state-based data
    var ps_state_ix;
    var ps_state2L_ix;
    var ps_region_ix;
    var ps_gFormState_ix;
    var ps_defaultLockLevel_ix;
    //var ps_requirePhone_ix;
    //var ps_requireURL_ix;
    var ps_areacode_ix;
    var stateDataTemp, areaCodeList = '800,822,833,844,855,866,877,888';  //  include toll free non-geographic area codes
    var ixBank, ixATM, ixOffices;
    var layer;

    var _updatedFields = {
        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: function () {
            return Object.keys(this).filter(key => this[key] && this[key].updated);
        },
        getUpdatedTabs: function () {
            var tabs = [];
            this.getFieldProperties().forEach(propName => {
                var prop = this[propName];
                if (prop.updated && tabs.indexOf(prop.tab) === -1) {
                    tabs.push(prop.tab);
                }
            });
            return tabs;
        },
        checkAddedNode: function (addedNode) {
            this.getFieldProperties().forEach(propName => {
                var prop = this[propName];
                if (prop.updated && addedNode.querySelector(prop.selector)) {
                    $(prop.selector).css({ 'background-color': '#dfd' });
                    $('a[href="#landmark-edit-' + prop.tab + '"]').css({ 'background-color': '#dfd' });
                }
            });
        },
        reset: function () {
            this.getFieldProperties().forEach(propName => { this[propName].updated = false; });
        },
        init: function () {
            ['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 => {
                    var propName = 'services_' + service;
                    this[propName] = { updated: false, selector: '.landmark label[for="service-checkbox-' + service + '"]', tab: 'more-info' };
                });

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

            W.selectionManager.events.register('selectionchanged', null, () => errorHandler(() => this.reset()));
        }
    };

    // KB Shortcut object
    var shortcut = {
        'all_shortcuts': {}, //All the shortcuts are stored in this array
        'add': function (shortcut_combination, callback, opt) {
            //Provide a set of default options
            var default_options = { 'type': 'keydown', 'propagate': false, 'disable_in_input': false, 'target': document, 'keycode': false };
            if (!opt) { opt = default_options; }
            else {
                for (var dfo in default_options) {
                    if (typeof opt[dfo] === 'undefined') { opt[dfo] = default_options[dfo]; }
                }
            }
            var ele = opt.target;
            if (typeof opt.target === 'string') { ele = document.getElementById(opt.target); }
            // var ths = this;
            shortcut_combination = shortcut_combination.toLowerCase();
            //The function to be called at keypress
            var func = function (e) {
                e = e || window.event;
                if (opt.disable_in_input) { //Don't enable shortcut keys in Input, Textarea fields
                    var 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
                var code;
                if (e.keyCode) { code = e.keyCode; }
                else if (e.which) { code = e.which; }
                var 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
                var keys = shortcut_combination.split('+');
                //Key Pressed - counts the number of valid keypresses - if it is same as the number of keys, the shortcut function is invoked
                var kp = 0;
                //Work around for stupid Shift key bug created by using lowercase - as a result the shift+num combination was broken
                var shift_nums = {
                    '`': '~', '1': '!', '2': '@', '3': '#', '4': '$', '5': '%', '6': '^', '7': '&',
                    '8': '*', '9': '(', '0': ')', '-': '_', '=': '+', ';': ':', '\'': '"', ',': '<', '.': '>', '/': '?', '\\': '|'
                };
                //Special Keys - and their codes
                var special_keys = {
                    '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
                };
                var 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 (var i = 0; i < keys.length; i++) {
                    var 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 (special_keys[k] === code) { kp++; }
                    } else if (opt.keycode) {
                        if (opt.keycode === code) { kp++; }
                    } else { //The special keys did not match
                        if (character === k) { kp++; }
                        else {
                            if (shift_nums[character] && e.shiftKey) { //Stupid Shift key bug created by using lowercase
                                character = shift_nums[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();
                        }
                        return false;
                    }
                }
            };
            this.all_shortcuts[shortcut_combination] = { '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': function (shortcut_combination) {
            shortcut_combination = shortcut_combination.toLowerCase();
            var binding = this.all_shortcuts[shortcut_combination];
            delete (this.all_shortcuts[shortcut_combination]);
            if (!binding) { return; }
            var type = binding.event;
            var ele = binding.target;
            var callback = binding.callback;
            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 getHoursHtml(label, defaultText) {
        defaultText = defaultText || _DEFAULT_HOURS_TEXT;
        return 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">' +
            '<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 + '</textarea>';
    }

    function getSelectedVenue() {
        let venue;
        let features = W.selectionManager.getSelectedFeatures();
        if (features.length && features[0].model.type === 'venue') {
            venue = features[0].model;
        }
        return venue;
    }

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

    function nudgeVenue(venue) {
        // Use an exact clone of the original geometry to force an edit without actually changing anything.
        let 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 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.indexOf('REST_AREAS') > -1 && /rest\s*area/i.test(venue.attributes.name);
    }

    function getPvaSeverity(pvaValue, venue) {
        var isER = pvaValue === 'hosp' && isEmergencyRoom(venue);
        return (pvaValue === '' || pvaValue === '0' || (pvaValue === 'hosp' && !isER)) ? 3 : (pvaValue === '2') ? 1 : (pvaValue === '3') ? 2 : 0;
    }

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

        function panelContainerChanged() {
            if (!$('#WMEPH-HidePURWebSearch').prop('checked')) {
                var $panelNav = $('.place-update-edit.panel .categories.small');
                if ($('#PHPURWebSearchButton').length === 0 && $panelNav.length > 0) {
                    var $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() {
            var newName = $('.place-update-edit.panel .name').first().text();
            var addr = $('.place-update-edit.panel .address').first().text();
            if ($('#WMEPH-WebSearchNewTab').prop('checked')) {
                window.open(buildSearchUrl(newName, addr));
            } else {
                window.open(buildSearchUrl(newName, addr), searchResultsWindowName, searchResultsWindowSpecs);
            }
        }
    }

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

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

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

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

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

                if (splits[disableIdx] !== 'altName') {
                    // Add any aliases
                    let tempAliases = splits[aliasesIdx];
                    if (tempAliases !== '' && tempAliases !== '0' && 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.
                let nameBaseStr = splits[searchNameBaseIdx];
                if (nameBaseStr !== '0' && nameBaseStr !== '') {   // If base terms exist, otherwise only the primary name is matched
                    newNameList = newNameList.concat(stripNonAlphaKeepCommas(nameBaseStr).split(','));

                    let nameMidStr = splits[searchNameMidIdx];
                    if (nameMidStr !== '0' && 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]]);
                        }
                        let 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]);
                            }
                        }
                    }

                    let nameEndStr = splits[searchNameEndIdx];
                    if (nameEndStr !== '0' && nameEndStr !== '') {
                        let pnhSearchNameEnd = stripNonAlphaKeepCommas(nameEndStr).split(',');
                        let endLen = pnhSearchNameEnd.length;
                        for (let extix = 1, len = newNameList.length; extix < len; extix++) {  // extend the list by adding End terms onto all the SearchNameBase & Base+Mid names
                            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
                let category = splits[category1Idx].toUpperCase().replace(/[^A-Z0-9]/g, '');
                let 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)));
                return _.uniq(newNameList).join('|').replace(/\|{2,}/g, '|').replace(/\|+$/g, '');
            } else { // END if valid line
                return '00';
            }
        });
    }  // END makeNameCheckList

    // Whitelist stringifying and parsing
    function saveWL_LS(compress) {
        venueWhitelistStr = JSON.stringify(venueWhitelist);
        if (compress) {
            if (venueWhitelistStr.length < 4800000) {  // Also save to regular storage as a back up
                localStorage.setItem(WLlocalStoreName, venueWhitelistStr);
            }
            venueWhitelistStr = LZString.compressToUTF16(venueWhitelistStr);
            localStorage.setItem(WLlocalStoreNameCompressed, venueWhitelistStr);
        } else {
            localStorage.setItem(WLlocalStoreName, venueWhitelistStr);
        }
    }
    function loadWL_LS(decompress) {
        if (decompress) {
            venueWhitelistStr = localStorage.getItem(WLlocalStoreNameCompressed);
            venueWhitelistStr = LZString.decompressFromUTF16(venueWhitelistStr);
        } else {
            venueWhitelistStr = localStorage.getItem(WLlocalStoreName);
        }
        venueWhitelist = JSON.parse(venueWhitelistStr);
    }
    function backupWL_LS(compress) {
        venueWhitelistStr = JSON.stringify(venueWhitelist);
        if (compress) {
            venueWhitelistStr = LZString.compressToUTF16(venueWhitelistStr);
            localStorage.setItem(WLlocalStoreNameCompressed + Math.floor(Date.now() / 1000), venueWhitelistStr);
        } else {
            localStorage.setItem(WLlocalStoreName + Math.floor(Date.now() / 1000), venueWhitelistStr);
        }
    }

    // Removes duplicate strings from string array
    function uniq(a) {
        var seen = {};
        return a.filter(item => seen.hasOwnProperty(item) ? false : (seen[item] = true));
    }  // END uniq function

    function phlog(m) {
        if ('object' === typeof m) {
            //m = JSON.stringify(m);
        }
        console.log('WMEPH' + (_IS_DEV_VERSION ? '-β' : '') + ': ' + m);
    }
    function phlogdev(msg, obj) {
        if (_USER.isDevUser) {
            console.log('WMEPH' + (_IS_DEV_VERSION ? '-β' : '') + ': ' + msg, (obj ? obj : ''));
        }
    }

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

    function nudgeVenue(venue) {
        let 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 (var i = 0; i < toSort.length; i++) {
            toSort[i] = [toSort[i], i];
        }
        toSort.sort((left, right) => left[0] < right[0] ? -1 : 1);
        toSort.sortIndices = [];
        for (var j = 0; j < toSort.length; j++) {
            toSort.sortIndices.push(toSort[j][1]);
            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() {
        //phlog('Clearing dupe label...');
        setTimeout(() => {
            var actionsList = W.model.actionManager.getActions();
            var lastAction = actionsList[actionsList.length - 1];
            if ('undefined' !== typeof lastAction && lastAction.hasOwnProperty('object') && lastAction.object.hasOwnProperty('state') && lastAction.object.state === 'Delete') {
                if (dupeIDList.indexOf(lastAction.object.attributes.id) > -1) {
                    if (dupeIDList.length === 2) {
                        _dupeLayer.destroyFeatures();
                        _dupeLayer.setVisibility(false);
                    } else {
                        var deletedDupe = _dupeLayer.getFeaturesByAttribute('dupeID', lastAction.object.attributes.id);
                        _dupeLayer.removeFeatures(deletedDupe);
                        dupeIDList.splice(dupeIDList.indexOf(lastAction.object.attributes.id), 1);
                    }
                    phlog('Deleted a dupe');
                }
            }
            /*
            else if ('undefined' !== typeof lastAction && lastAction.hasOwnProperty('feature') && lastAction.feature.hasOwnProperty('state') && lastAction.object.state === 'Update' &&
            lastAction.hasOwnProperty('newGeometry') ) {
                // update position of marker
            }
            */
        }, 20);
    }

    //  Whitelist an item
    function whitelistAction(itemID, wlKeyName) {
        var venue = getSelectedVenue();
        var addressTemp = venue.getAddress();
        if (addressTemp.hasOwnProperty('attributes')) {
            addressTemp = addressTemp.attributes;
        }
        var itemGPS = OL.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
        saveWL_LS(true);  // Save the WL to local storage
        WMEPH_WLCounter();
        bannButt2.clearWL.active = true;
    }

    // Keep track of how many whitelists have been added since the last pull, alert if over a threshold (100?)
    function WMEPH_WLCounter() {
        localStorage.WMEPH_WLAddCount = parseInt(localStorage.WMEPH_WLAddCount) + 1;
        if (localStorage.WMEPH_WLAddCount > 50) {
            alert('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() {
        let observer = new MutationObserver(mutations => {
            mutations.forEach(mutation => {
                // Mutation is a NodeList and doesn't support forEach like an array
                for (var i = 0; i < mutation.addedNodes.length; i++) {
                    var 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() {
        let 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') {
            alert('No PNH data exists for this country.');
            return ['NoMatch'];
        } else {
            let pnhData = _PNH_DATA[country].pnh;
            let pnhNames = _PNH_DATA[country].pnhNames;
            let pnhHeaders = pnhData[0].split('|');
            let ph_name_ix = pnhHeaders.indexOf('ph_name');
            let ph_category1_ix = pnhHeaders.indexOf('ph_category1');
            let ph_forcecat_ix = pnhHeaders.indexOf('ph_forcecat');
            let ph_region_ix = pnhHeaders.indexOf('ph_region');
            let ph_order_ix = pnhHeaders.indexOf('ph_order');
            let ph_speccase_ix = pnhHeaders.indexOf('ph_speccase');
            let ph_searchnameword_ix = pnhHeaders.indexOf('ph_searchnameword');
            let PNHPriCat;  // Primary category of PNH data
            let PNHForceCat;  // Primary category of PNH data
            let approvedRegions;  // filled with the regions that are approved for the place, when match is found
            let matchPNHRegionData = [];  // array of matched data with regional approval
            let specCases, nmix, allowMultiMatch = false;
            let PNHOrderNum = [];
            let PNHNameTemp = [];
            let PNHNameMatch = false;  // tracks match status

            itemName = itemName.toUpperCase().replace(/ AND /g, ' ').replace(/^THE /g, '');
            let 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;
                let pnhEntry = pnhData[pnhIdx];
                let pnhEntrySplits = pnhEntry.split('|');  // Split the PNH place data into string array

                // Name Matching
                specCases = pnhEntrySplits[ph_speccase_ix];
                if (specCases.indexOf('regexNameMatch') > -1) {
                    // Check for regex name matching instead of "standard" name matching.
                    var match = specCases.match(/regexNameMatch<>(.+?)<>/i);
                    if (match !== null) {
                        let reStr = match[1].replace(/\\/, '\\').replace(/<or>/g, '|');
                        var re = new RegExp(reStr, 'i');
                        PNHStringMatch = re.test(item.attributes.name);
                    }
                } else {
                    if (specCases.indexOf('strMatchAny') > -1 || pnhEntrySplits[ph_category1_ix] === 'Hotel') {  // Match any part of WME name with either the PNH name or any spaced names
                        allowMultiMatch = true;
                        var spaceMatchList = [];
                        spaceMatchList.push(pnhEntrySplits[ph_name_ix].toUpperCase().replace(/ AND /g, ' ').replace(/^THE /g, '').replace(/[^A-Z0-9 ]/g, ' ').replace(/ {2,}/g, ' '));
                        if (pnhEntrySplits[ph_searchnameword_ix] !== '') {
                            spaceMatchList.push.apply(spaceMatchList, pnhEntrySplits[ph_searchnameword_ix].toUpperCase().replace(/, /g, ',').split(','));
                        }
                        for (nmix = 0; nmix < spaceMatchList.length; nmix++) {
                            if (itemNameSpace.includes(' ' + spaceMatchList[nmix] + ' ')) {
                                PNHStringMatch = true;
                            }
                        }
                    } else {
                        let nameComps = pnhNames[pnhIdx].split('|');  // splits all possible search names for the current PNH entry
                        let itemNameNoNum = itemName.replace(/[^A-Z]/g, '');  // Clear non-letter characters for alternate match ( HOLLYIVYPUB23 --> HOLLYIVYPUB )
                        if (specCases.indexOf('strMatchStart') > -1) {  //  Match the beginning part of WME name with any search term
                            for (nmix = 0; nmix < nameComps.length; nmix++) {
                                if (itemName.startsWith(nameComps[nmix]) || itemNameNoNum.startsWith(nameComps[nmix])) {
                                    PNHStringMatch = true;
                                }
                            }
                        } else if (specCases.indexOf('strMatchEnd') > -1) {  //  Match the end part of WME name with any search term
                            for (nmix = 0; nmix < nameComps.length; nmix++) {
                                if (itemName.endsWith(nameComps[nmix]) || itemNameNoNum.endsWith(nameComps[nmix])) {
                                    PNHStringMatch = true;
                                }
                            }
                        } else {  // full match of any term only
                            if (nameComps.indexOf(itemName) > -1 || nameComps.indexOf(itemNameNoNum) > -1) {
                                PNHStringMatch = true;
                            }
                        }
                    }
                }
                // if a match was found:
                if (PNHStringMatch) {  // Compare WME place name to PNH search name list
                    phlogdev('Matched PNH Order No.: ' + pnhEntrySplits[ph_order_ix]);

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

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

                    if (PNHMatchProceed) {
                        approvedRegions = pnhEntrySplits[ph_region_ix].replace(/ /g, '').toUpperCase().split(',');  // remove spaces, upper case the approved regions, and split by commas
                        if (approvedRegions.indexOf(state2L) > -1 || approvedRegions.indexOf(region3L) > -1 ||  // if the WME-selected item matches the state, region
                            approvedRegions.indexOf(country) > -1 ||  //  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);
                            bannButt.placeMatched = new Flag.PlaceMatched();
                            if (!allowMultiMatch) {
                                return matchPNHRegionData;  // Return the PNH data string array to the main script
                            }
                        } else {
                            PNHNameMatch = true;  // PNH match found (once true, stays true)
                            //matchPNHData.push(pnhEntry);  // Pull the data line from the PNH data table.  (**Set in array for future multimatch features)
                            PNHNameTemp.push(pnhEntrySplits[ph_name_ix]);  // temp name for approval return
                            PNHOrderNum.push(pnhEntrySplits[ph_order_ix]);  // temp order number for approval return
                        }
                    }
                }
            }  // END loop through PNH places

            // If NO (name & region) match was found:
            if (bannButt.placeMatched) {
                return matchPNHRegionData;
            } else if (PNHNameMatch) {  // if a name match was found but not for region, prod the user to get it approved
                bannButt.ApprovalSubmit = new Flag.ApprovalSubmit(region3L, PNHOrderNum, PNHNameTemp, placePL);
                return ['ApprovalNeeded', PNHNameTemp, PNHOrderNum];
            } else {  // if no match was found, suggest adding the place to the sheet if it's a chain
                bannButt.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.
        let venue = getSelectedVenue();
        if ($('#WMEPH_banner').length > 0 && venue) {
            var actions = W.model.actionManager.getActions();
            var 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 => {
            var oldID = newVenue._prevID;
            var newID = newVenue.attributes.id;
            if (oldID && newID && venueWhitelist[oldID]) {
                venueWhitelist[newID] = venueWhitelist[oldID];
                delete venueWhitelist[oldID];
            }
        });
        saveWL_LS(true);
    }

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

        let commentsLayer = W.map.getLayerByUniqueName('mapComments');
        let gisLayer = W.map.getLayerByUniqueName('__wmeGISLayers');
        let 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');
            W.map.roadLayers[0].opacity = 0.25;
            W.map.baseLayer.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');
            W.map.roadLayers[0].opacity = 1;
            W.map.baseLayer.opacity = 1;
            commentRuleSymb.Polygon.strokeColor = '#fff';
            commentRuleSymb.Polygon.fillOpacity = 0.4;
            if (gisLayer) gisLayer.setOpacity(1);
            initializeHighlights();
            layer.redraw();
        }
        commentsLayer.redraw();
        W.map.roadLayers[0].redraw();
        W.map.baseLayer.redraw();
        if (!enable) return;

        let defaultPointRadius = 6;
        var ruleGenerator = function (value, symbolizer) {
            return new W.Rule({
                filter: new OL.Filter.Comparison({
                    type: '==',
                    value: value,
                    evaluate: function (venue) {
                        return venue && venue.model && venue.model.attributes.wmephSeverity === this.value;
                    }
                }),
                symbolizer: symbolizer,
                wmephStyle: 'xray'
            });
        };

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

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

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

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

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

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

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

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

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

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

        Array.prototype.push.apply(layer.styleMap.styles['default'].rules, [severity0, severityLock, severity1, severityLock1, severity2, severity3, severity4, severityHigh, severityAdLock]);

        layer.redraw();
    }

    function initializeHighlights() {
        var ruleGenerator = function (value, symbolizer) {
            return new W.Rule({
                filter: new OL.Filter.Comparison({
                    type: '==',
                    value: value,
                    evaluate: function (venue) {
                        return venue && venue.model && venue.model.attributes.wmephSeverity === this.value;
                    }
                }),
                symbolizer: symbolizer,
                wmephStyle: 'default'
            });
        };

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

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

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

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

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

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

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

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

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

        function plaTypeRuleGenerator(value, symbolizer) {
            return new W.Rule({
                filter: new OL.Filter.Comparison({
                    type: '==',
                    value: value,
                    evaluate: function (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.indexOf('PARKING_LOT') > -1) {
                            var type = venue.model.attributes.categoryAttributes.PARKING_LOT.parkingType;
                            return (!type && this.value === 'public') || (type && (type.toLowerCase() === this.value));
                        }
                    }
                }),
                symbolizer: symbolizer,
                wmephStyle: 'default'
            });
        }

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

        Array.prototype.push.apply(layer.styleMap.styles['default'].rules, [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;
        venues = venues ? _.isArray(venues) ? venues : [venues] : [];
        let storedBannButt = bannButt, storedBannServ = bannServ, storedBannButt2 = bannButt2;
        let t0 = performance.now();
        let doHighlight = $('#WMEPH-ColorHighlighting').prop('checked');
        let disableRankHL = $('#WMEPH-DisableRankHL').prop('checked');

        _.each(venues, 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 {
                        let id = venue.attributes.id;
                        let severity;
                        let cachedResult;
                        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.
        let keys = Object.keys(_resultsCache);
        if (keys.length > _MAX_CACHE_SIZE) {
            let trimSize = _MAX_CACHE_SIZE * 0.8;
            for (let i = keys.length - 1; i > trimSize; i--) {
                delete _resultsCache[keys[i]];
            }
        }

        let venue = getSelectedVenue();
        if (venue) {
            venue.attributes.wmephSeverity = harmonizePlaceGo(venue, 'highlight');
            bannButt = storedBannButt;
            bannServ = storedBannServ;
            bannButt2 = storedBannButt2;
        }
        phlogdev('Ran highlighter in ' + Math.round((performance.now() - t0) * 10) / 10 + ' milliseconds.');
        //phlogdev('WMEPH cache size: ' + Object.keys(_resultsCache).length);

        //layer.redraw();
    }

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

            //W.model.venues.on('objectsadded', e => errorHandler(() => applyHighlightsTest(e)));
            W.map.landmarkLayer.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();
        let 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.
        let macIndexes = [];
        let macRegex = /\bMac[A-Z]/g;
        let macMatch;
        while ((macMatch = macRegex.exec(str)) !== null) {
            macIndexes.push(macMatch.index);
        }

        let allCaps = (str === str.toUpperCase());
        // Cap first letter of each word
        str = str.replace(/([A-Za-z\u00C0-\u017F][^\s-\/]*) */g, function (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, function (txt) {
                return ((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, function (txt) {
                return ((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, function (txt) {
                return ((txt === txt.toUpperCase()) && !allCaps) ? txt : txt.charAt(0) + txt.charAt(1).toUpperCase() + txt.substr(2);
            })
            // lowercase any from the ignoreWords list
            .replace(/[^ ]+/g, function (txt) {
                let txtLC = txt.toLowerCase();
                return (_TITLECASE_SETTINGS.ignoreWords.indexOf(txtLC) > -1) ? txtLC : txt;
            })
            // uppercase any from the capWords List
            .replace(/[^ ]+/g, function (txt) {
                let txtLC = txt.toUpperCase();
                return (_TITLECASE_SETTINGS.capWords.indexOf(txtLC) > -1) ? txtLC : txt;
            })
            // preserve any specific words
            .replace(/[^ ]+/g, function (txt) {
                let 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') {
            bannButt.phoneMissing = Flag.PhoneMissing.eval(item, _wl, region, outputFormat);
            return s;
        }
        s = s.replace(/(\d{3}.*)\W+(?:extension|ext|xt|x).*/i, '$1');
        var s1 = s.replace(/\D/g, '');  // remove non-number characters
        var m = s1.match(/^1?([2-9]\d{2})([2-9]\d{2})(\d{4})$/);  // Ignore leading 1, and also don't allow area code or exchange to start with 0 or 1 (***USA/CAN specific)
        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);
            m = s1.match(/^([2-9]\d{2})([2-9]\d{2})(\d{4})(?:.{0,3})$/);  // Ignore leading 1, and also don't allow area code or exchange to start with 0 or 1 (***USA/CAN specific)
            if (!m) {
                if (returnType === 'inputted') {
                    return 'badPhone';
                } else {
                    bannButt.phoneInvalid = new Flag.PhoneInvalid();
                    return s;
                }
            } else {
                return String.plFormat(outputFormat, m[1], m[2], m[3]);
            }
        } else {
            return String.plFormat(outputFormat, m[1], m[2], m[3]);
        }
    }

    // Alphanumeric phone conversion
    function replaceLetters(number) {
        var 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, function (match) {
            return conversionMap.findKey(function (re) {
                return re.test(match);
            });
        });
    }

    // 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) {
            var m_action = new MultiAction();
            m_action.setModel(W.model);
            m_action._description = description || m_action._description || 'Change(s) made by WMEPH';
            actions.forEach(action => { m_action.doSubAction(action); });
            W.model.actionManager.add(m_action);
        }
    }

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

    function addUpdateAction(venue, updateObj, actions) {
        var action = new UpdateObject(venue, updateObj);
        if (actions) {
            actions.push(action);
        } else {
            W.model.actionManager.add(action);
        }
    }

    function setServiceChecked(servBtn, checked, actions) {
        var servID = WMEServicesArray[servBtn.servIDIndex];
        var checkboxChecked = $('#service-checkbox-' + servID).prop('checked');
        let venue = getSelectedVenue();

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

    // Normalize url
    function normalizeURL(s, lc, skipBannerActivate, venue, region) {
        var 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)
            let hasOperator = venue.attributes.brand && W.model.venues.categoryBrands.PARKING_LOT.indexOf(venue.attributes.brand) !== -1;
            if (!venue.isParkingLot() || (venue.isParkingLot() && (regionsThatWantPLAUrls.indexOf(region) > -1 || hasOperator))) {
                bannButt.urlMissing = new Flag.UrlMissing();
                if (_wl.urlWL || (venue.isParkingLot() && !hasOperator)) {
                    bannButt.urlMissing.severity = 0;
                    bannButt.urlMissing.WLactive = false;
                }
            }
            //bannButt.webSearch.active = true;  // Activate websearch button
            return s;
        }

        s = s.replace(/ \(.*/g, '');  // remove anything with parentheses after it
        s = s.replace(/ /g, '');  // remove any spaces
        var m = s.match(/^http:\/\/(.*)$/i);  // remove http://
        if (m) { s = m[1]; }
        if (lc) {  // lowercase the entire domain
            s = s.replace(/[^\/]+/i, function (txt) { // lowercase the domain
                return (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[1]; }
        m = s.match(/^(.*)\/pages\/default.aspx$/i);  // remove unneeded terms
        if (m) { s = m[1]; }
        // 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[1]; }

        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) {
            alert('Please sign up to beta-test this script version.\nSend 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
        let venue = getSelectedVenue();
        if (venue) {
            _updatedFields.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;
        }
        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() {
            let venue = getSelectedVenue();
            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;
        }
        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) {
                let result = {};
                if (!addr.state || !addr.country) {
                    if (W.map.getZoom() < 4) {
                        if ($('#WMEPH-EnableIAZoom').prop('checked')) {
                            W.map.moveTo(getVenueLonLat(venue), 5);
                        } else {
                            alert('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 = WMEPH_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;
                                _updatedFields.address.updated = true;
                                result.flag = new Flag.FullAddressInference();
                                result.noLock = true;
                                //                                 let hn = venue.attributes.houseNumber;
                                //                                 if (hn && hn.replace(/[^0-9A-Za-z]/g,'').length > 0 ) {
                                //                                     result.flag = new Flag.FullAddressInference();
                                //                                     result.noLock = true;
                                //                                 }
                            } else {
                                if (['JUNCTION_INTERCHANGE'].indexOf(newCategories[0]) === -1) {
                                    bannButt.cityMissing = new Flag.CityMissing();
                                    result.noLock = true;
                                }
                            }
                        } else {  //  if the inference doesn't work...
                            alert('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 {
                        let cat = venue.attributes.categories;
                        if (containsAny(cat, ['HOSPITAL_MEDICAL_CARE', 'HOSPITAL_URGENT_CARE', 'GAS_STATION'])) {
                            phlogdev('Unaddressed HUC/GS');
                            result = 5;
                        } else if (cat.indexOf('JUNCTION_INTERCHANGE') > -1) {
                            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) {
                let result = { flag: null };
                if (!wl.plaNameNonStandard) {
                    let name = venue.attributes.name;
                    let state = venue.getAddress().getStateName();
                    let 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');
            }
        },
        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?'); }
            action() {
                let ix = newCategories.indexOf('TRANSPORTATION');
                if (ix > -1) {
                    let venue = getSelectedVenue();
                    newCategories.splice(ix, 1);
                    _updatedFields.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');
            }
            action() {
                var ix = newCategories.indexOf('SCENIC_LOOKOUT_VIEWPOINT');
                if (ix > -1) {
                    let venue = getSelectedVenue();
                    newCategories.splice(ix, 1);
                    _updatedFields.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');
            }
            action() {
                let venue = getSelectedVenue();
                let actions = [];
                // update categories according to spec
                newCategories = insertAtIX(newCategories, 'REST_AREAS', 0);
                actions.push(new UpdateObject(venue, { categories: newCategories }));
                _updatedFields.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' })] }));
                _updatedFields.openingHours.updated = true;

                bannServ.add247.checked = true;
                bannServ.addParking.actionOn(actions);  // add parking service
                bannServ.addWheelchair.actionOn(actions);  // add parking service
                bannButt.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 {
            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) {
                let result = { flag: null };
                if (venue.isGasStation() && venue.attributes.brand === 'Unbranded') {  //  Unbranded is not used per wiki
                    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.'); }
            action() {
                let venue = getSelectedVenue();
                newCategories = insertAtIX(newCategories, 'GAS_STATION', 0);  // Insert/move Gas category in the first position
                _updatedFields.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) {
                let result = { flag: null, newName: 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 }));
                        _updatedFields.name.updated = true;
                    }
                    if (result.newName.toLowerCase().trim() === 'pilot food mart') {
                        result.flag = new Flag.IsThisAPilotTravelCenter();
                    }
                }
                return result;
            }
            action() {
                let venue = getSelectedVenue();
                _updatedFields.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'); }
            action() {
                let venue = getSelectedVenue();
                newCategories = insertAtIX(newCategories, 'HOTEL', 0);  // Insert/move Hotel category in the first position
                _updatedFields.categories.updated = true;
                addUpdateAction(venue, { categories: newCategories });
                harmonizePlaceGo(venue, 'harmonize');
            }
        },
        ChangeToPetVet: class extends WLActionFlag {
            constructor() { super(true, 3, 'This looks like it should be a Pet/Veterinarian category. Change?', 'Yes', 'Change to Pet/Veterinarian Category', true, 'Whitelist PetVet category', 'changeHMC2PetVet'); }
            action() {
                let venue = getSelectedVenue();
                let idx = newCategories[newCategories.indexOf('HOSPITAL_MEDICAL_CARE')];
                if (idx === -1) idx = newCategories[newCategories.indexOf('HOSPITAL_URGENT_CARE')];
                if (idx > -1) {
                    newCategories[idx] = 'PET_STORE_VETERINARIAN_SERVICES';
                    _updatedFields.categories.updated = true;
                    addUpdateAction(venue, { categories: newCategories });
                }
                harmonizePlaceGo(venue, 'harmonize');  // Rerun the script to update fields and lock
            }
        },
        ChangeSchool2Offices: class extends WLActionFlag {
            constructor() { super(true, 3, 'This doesn\'t look like it should be School category.', 'Change to Office', 'Change to Offices Category', true, 'Whitelist School category', 'changeSchool2Offices'); }
            action() {
                let venue = getSelectedVenue();
                newCategories[newCategories.indexOf('SCHOOL')] = 'OFFICES';
                _updatedFields.categories.updated = true;
                addUpdateAction(venue, { categories: newCategories });
                harmonizePlaceGo(venue, 'harmonize');  // Rerun the script to update fields and lock
            }
        },
        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'); }
            action() {
                let venue = getSelectedVenue();
                if (venue.attributes.categories.indexOf('RESIDENCE_HOME') > -1) {
                    let centroid = venue.geometry.getCentroid();
                    updateFeatureGeometry(venue, new OL.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'); }
            action() {
                let venue = getSelectedVenue();
                updateFeatureGeometry(venue, venue.getPolygonGeometry());
                harmonizePlaceGo(venue, 'harmonize');
            }
        },
        // 2019-3-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'm commenting out the code to allow entering the HN in the 
        // banner and replacing with an "Edit address" button.  If Waze addresses the issue, we can revert the
        // change.
        // 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() {
        //         let newHN = $('#WMEPH-HNAdd').val().replace(/ +/g, '');
        //         phlogdev(newHN);
        //         var hnTemp = newHN.replace(/[^\d]/g, '');
        //         var hnTempDash = newHN.replace(/[^\d-]/g, '');
        //         if (hnTemp > 0 && hnTemp < 1000000) {
        //             let 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
        //             _updatedFields.address.updated = true;
        //         } else {
        //             $('input#WMEPH-HNAdd').css({ backgroundColor: '#FDD' }).attr('title', 'Must be a number between 0 and 1000000');
        //             this.badInput = true;
        //         }

        //     }
        // },
        HnMissing: class extends WLActionFlag {
            constructor() { super(true, 3, 'No HN:', 'Edit address', 'Edit address to add HN.'); }
            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.'); }
            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.'); }
            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?'); }
            action() {
                let venue = getSelectedVenue();
                newCategories = ['BANK_FINANCIAL', 'ATM'];  // Change to bank and atm cats
                var 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) _updatedFields.name.updated = true;
                _updatedFields.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?'); }
            action() {
                let venue = getSelectedVenue();
                if (newName.indexOf('ATM') === -1) {
                    newName = newName + ' ATM';
                    _updatedFields.name.updated = true;
                }
                newCategories = ['ATM'];  // Change to ATM only
                W.model.actionManager.add(new UpdateObject(venue, { name: newName, categories: newCategories }));
                _updatedFields.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?'); }
            action() {
                let venue = getSelectedVenue();
                newCategories = ['OFFICES'];  // Change to offices category
                var 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) _updatedFields.name.updated = true;
                _updatedFields.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).'); }
            action() {
                let 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() { 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'); }
            action() {
                let venue = getSelectedVenue();
                if (tempPNHURL !== '') {
                    W.model.actionManager.add(new UpdateObject(venue, { url: tempPNHURL }));
                    _updatedFields.url.updated = true;
                    harmonizePlaceGo(venue, 'harmonize');
                    updateURL = true;
                } else {
                    if (confirm('WMEPH: URL Matching 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
                        reportError({
                            subject: 'WMEPH URL comparison Error report',
                            message: 'Error report: URL comparison failed for "' + venue.attributes.name + '"\nPermalink: ' + placePL
                        });
                    }
                }
            }
        },
        GasNoBrand: class extends FlagBase {
            constructor() { super(true, 1, 'Lock to region standards to verify no gas brand.'); }
            static eval(venue) {
                let result = { flag: null };
                if (venue.isGasStation() && !venue.attributes.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;
            }
            action() {
                let $input = $('input#WMEPH-zipAltNameAdd');
                let zip = $input.val().trim();
                if (zip) {
                    if (/^\d{5}/.test(zip)) {
                        let venue = getSelectedVenue();
                        let aliases = [].concat(venue.attributes.aliases);
                        // Make sure zip hasn't already been added.
                        if (aliases.indexOf(zip) === -1) {
                            aliases.push(zip);
                            W.model.actionManager.add(new UpdateObject(venue, { aliases: 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'); }
        },
        SpecCaseMessage: class extends FlagBase {
            constructor(message) { super(true, 0, message); }
        },
        PnhCatMess: class extends FlagBase {
            constructor(message) { super(true, 0, message); }
        },
        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';
            }
            action() {
                let venue = getSelectedVenue();
                nudgeVenue(venue);
                harmonizePlaceGo(venue, 'harmonize');  // Rerun the script to update fields and lock
            }
            action2() {
                let venue = getSelectedVenue();
                $('div.external-providers-view a').focus().click();
                setTimeout(function () {
                    $('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;
            }
            action() {
                let venue = getSelectedVenue();
                let 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 }));
                    _updatedFields.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 ? textValue : '') + '">', 'Update', 'Update phone #', true, 'Whitelist the area code', 'aCodeWL');
                this.outputFormat = outputFormat;
                this.noBannerAssemble = true;
            }
            action() {
                let venue = getSelectedVenue();
                let 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 }));
                    _updatedFields.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) {
                let hasOperator = venue.attributes.brand && W.model.venues.categoryBrands.PARKING_LOT.indexOf(venue.attributes.brand) !== -1;
                let isPLA = venue.isParkingLot();
                let flag = null;
                if (!isPLA || (isPLA && (this._regionsThatWantPlaPhones.indexOf(region) > -1 || hasOperator))) {
                    flag = new Flag.PhoneMissing(venue, hasOperator, wl, outputFormat, isPLA);
                }
                return flag;
            }
            action() {
                let 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 }));
                    _updatedFields.phone.updated = true;
                    harmonizePlaceGo(this.venue, 'harmonize');
                }
            }
        },
        NoHours: class extends WLFlag {
            constructor() { super(true, 1, getHoursHtml('No hours'), true, 'Whitelist "No hours"', 'noHours'); }
            getTitle(parseResult) {
                let title;
                if (parseResult.overlappingHours) {
                    title = 'Overlapping hours.  Check the existing hours.';
                } else if (parseResult.sameOpenAndCloseTimes) {
                    title = 'Open/close times cannot be the same.';
                } else {
                    title = 'Can\'t parse, try again';
                }
                return title;
            }
            applyHours(replaceAllHours) {
                let 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');
                var parser = new HoursParser();
                var 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 }));
                    _updatedFields.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) {
                let result = { flag: null };
                if (venue.isParkingLot()) {
                    let catAttr = venue.attributes.categoryAttributes;
                    let parkAttr = catAttr ? catAttr.PARKING_LOT : undefined;
                    if (!parkAttr || !parkAttr.parkingType) {
                        result.flag = new Flag.PlaLotTypeMissing();
                        if (hpMode.harmFlag) {
                            result.noLock = true;
                            [['PUBLIC', 'Public'], ['RESTRICTED', 'Restricted'], ['PRIVATE', 'Private']].forEach(btnInfo => {
                                result.flag.message +=
                                    $('<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');
                            });
                        }
                    }
                }
                return result;
            }
        },
        PlaCostTypeMissing: class extends FlagBase {
            constructor() { super(true, 1, 'Parking cost: '); }
            static eval(venue, hpMode) {
                let result = { flag: null };
                if (venue.isParkingLot()) {
                    let catAttr = venue.attributes.categoryAttributes;
                    let 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) {
                let result = { flag: null };
                if (venue.isParkingLot()) {
                    let catAttr = venue.attributes.categoryAttributes;
                    let 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;
            }
            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) {
                let result = { flag: null };
                if (venue.isParkingLot()) {
                    let catAttr = venue.attributes.categoryAttributes;
                    let parkAttr = catAttr ? catAttr.PARKING_LOT : undefined;
                    if (!parkAttr || !parkAttr.lotType || parkAttr.lotType.length === 0) {
                        result.flag = new Flag.PlaLotElevationMissing();
                        result.noLock = true;
                    }
                }
                return result;
            }
            action() {
                let venue = getSelectedVenue();
                let existingAttr = venue.attributes.categoryAttributes.PARKING_LOT;
                let newAttr = {};
                if (existingAttr) {
                    for (let prop in existingAttr) {
                        let value = existingAttr[prop];
                        if (Array.isArray(value)) value = [].concat(value);
                        newAttr[prop] = 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:');
                let $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) {
                let result = { flag: null };
                if (hpMode.harmFlag && venue.isParkingLot()) {
                    let catAttr = venue.attributes.categoryAttributes;
                    let 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) {
                let result = { flag: null };
                if (venue.isParkingLot() && (!venue.attributes.entryExitPoints || !venue.attributes.entryExitPoints.length)) {
                    result.flag = new Flag.NoPlaStopPoint();
                }
                return result;
            }
            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) {
                let result = { flag: null };
                let attr = venue.attributes;
                if (venue.isParkingLot() && attr.entryExitPoints && attr.entryExitPoints.length) {
                    let stopPoint = attr.entryExitPoints[0].getPoint();
                    let 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) {
                let result = { flag: null };
                if (hpMode.harmFlag && venue.isParkingLot()) {
                    let catAttr = venue.attributes.categoryAttributes;
                    let 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;
            }
            action() {
                let venue = getSelectedVenue();
                let existingAttr = venue.attributes.categoryAttributes.PARKING_LOT;
                let newAttr = {};
                if (existingAttr) {
                    for (let prop in existingAttr) {
                        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) {
                let result = { flag: null };
                if (hpMode.harmFlag && venue.isParkingLot()) {
                    let services = venue.attributes.services;
                    if (!(services && services.indexOf('DISABILITY_PARKING') > -1)) {
                        result.flag = new Flag.PlaHasAccessibleParking();
                    }
                }
                return result;
            }
            action() {
                let venue = getSelectedVenue();
                let services = venue.attributes.services;
                if (services) {
                    services = [].concat(services);
                } else {
                    services = [];
                }
                services.push('DISABILITY_PARKING');
                //bannServ.addDisabilityParking.on();
                W.model.actionManager.add(new UpdateObject(venue, { 'services': services }));
                _updatedFields.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) {
                let hoursEntries = venue.attributes.openingHours;
                let newHoursEntries = [];
                let updateHours = false;
                let flag = null;
                for (let i = 0, len = hoursEntries.length; i < len; i++) {
                    let 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);
                    _updatedFields.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() {
                let venue = getSelectedVenue();
                let RPPlevelToLock = $('#RPPLockLevel :selected').val() || defaultLockLevel + 1;
                phlogdev('RPPlevelToLock: ' + 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) + 1) + '. ' + RPPLockString + ' ?';
            }
        },
        AddAlias: class extends ActionFlag {
            constructor() { super(true, 0, 'Is ' + optionalAlias + ' at this location?', 'Yes', 'Add ' + optionalAlias); }
            action() {
                let venue = getSelectedVenue();
                newAliases = insertAtIX(newAliases, optionalAlias, 0);
                if (specCases.indexOf('altName2Desc') > -1 && venue.attributes.description.toUpperCase().indexOf(optionalAlias.toUpperCase()) === -1) {
                    newDescripion = optionalAlias + '\n' + newDescripion;
                    W.model.actionManager.add(new UpdateObject(venue, { description: newDescripion }));
                    _updatedFields.description.updated = true;
                }
                newAliases = removeSFAliases(newName, newAliases);
                W.model.actionManager.add(new UpdateObject(venue, { aliases: newAliases }));
                _updatedFields.aliases.updated = true;
                harmonizePlaceGo(venue, 'harmonize');
            }
        },
        AddCat2: class extends ActionFlag {
            constructor() { super(false, 0, '', 'Yes', ''); }
            action() {
                let venue = getSelectedVenue();
                newCategories.push(this.altCategory);
                W.model.actionManager.add(new UpdateObject(venue, { categories: newCategories }));
                _updatedFields.categories.updated = true;
                harmonizePlaceGo(venue, 'harmonize');
            }
        },
        AddPharm: class extends ActionFlag {
            constructor() { super(false, 0, 'Is there a Pharmacy at this location?', 'Yes', 'Add Pharmacy category'); }
            action() {
                let venue = getSelectedVenue();
                newCategories = insertAtIX(newCategories, 'PHARMACY', 1);
                W.model.actionManager.add(new UpdateObject(venue, { categories: newCategories }));
                _updatedFields.categories.updated = true;
                bannButt.addPharm.active = false;  // reset the display flag
            }
        },
        AddSuper: class extends ActionFlag {
            constructor() { super(false, 0, 'Does this location have a supermarket?', 'Yes', 'Add Supermarket category'); }
            action() {
                let venue = getSelectedVenue();
                newCategories = insertAtIX(newCategories, 'SUPERMARKET_GROCERY', 1);
                W.model.actionManager.add(new UpdateObject(venue, { categories: newCategories }));
                _updatedFields.categories.updated = true;
                bannButt.addSuper.active = false;  // reset the display flag
            }
        },
        AppendAMPM: class extends ActionFlag {
            constructor() { super(false, 0, 'Is there an ampm at this location?', 'Yes', 'Add ampm to the place'); }
            action() {
                let 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 }));
                _updatedFields.name.updated = true;
                _updatedFields.url.updated = true;
                _updatedFields.categories.updated = true;
                bannButt.appendAMPM.active = false;  // reset the display flag
                bannButt.addConvStore.active = false;  // also reset the addConvStore display flag
            }
        },
        AddATM: class extends ActionFlag {
            constructor() { super(false, 0, 'ATM at location? ', 'Yes', 'Add the ATM category to this place'); }
            action() {
                let venue = getSelectedVenue();
                newCategories = insertAtIX(newCategories, 'ATM', 1);  // Insert ATM category in the second position
                W.model.actionManager.add(new UpdateObject(venue, { categories: newCategories }));
                _updatedFields.categories.updated = true;
                bannButt.addATM.active = false;   // reset the display flag
            }
        },
        AddConvStore: class extends ActionFlag {
            constructor() { super(false, 0, 'Add convenience store category? ', 'Yes', 'Add the Convenience Store category to this place'); }
            action() {
                let 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 }));
                _updatedFields.categories.updated = true;
                bannButt.addConvStore.active = false;   // reset the display flag
            }
        },
        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) {
                let result = { flag: null };
                let 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;
            }
            action() {
                let venue = getSelectedVenue();
                newCategories = insertAtIX(newCategories, 'POST_OFFICE', 0);
                W.model.actionManager.add(new UpdateObject(venue, { categories: newCategories }));
                _updatedFields.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) {
                let result = { flag: null };
                if (hpMode.harmFlag && venue.attributes.categories.indexOf('DOCTOR_CLINIC') > -1) {
                    result.flag = new Flag.ChangeToHospitalUrgentCare(0, 'If this place provides emergency medical care:');
                }
                return result;
            }
            action() {
                let idx = newCategories.indexOf('HOSPITAL_MEDICAL_CARE');
                let venue = getSelectedVenue();
                if (idx === -1) idx = newCategories.indexOf('DOCTOR_CLINIC');
                if (idx > -1) {
                    newCategories[idx] = 'HOSPITAL_URGENT_CARE';
                    _updatedFields.categories.updated = true;
                    addUpdateAction(venue, { categories: newCategories });
                }
                harmonizePlaceGo(venue, 'harmonize');  // Rerun the script to update fields and lock
            }
        },
        ChangeToDoctorClinic: class extends WLActionFlag {
            constructor() { super(true, 0, '', 'Change to Doctor / Clinic', 'Change category to Doctor / Clinic', false, 'Whitelist category', 'changeToDoctorClinic'); }
            action() {
                let venue = getSelectedVenue();
                let newCategories = _.clone(venue.attributes.categories);
                let updateIt = false;
                if (newCategories.length) {
                    ['HOSPITAL_MEDICAL_CARE', 'HOSPITAL_URGENT_CARE', 'OFFICES', 'PERSONAL_CARE'].forEach(cat => {
                        let idx = newCategories.indexOf(cat);
                        if (idx > -1) {
                            newCategories[idx] = 'DOCTOR_CLINIC';
                            updateIt = true;
                        }
                    });
                    newCategories = _.uniq(newCategories);
                } else {
                    newCategories.push('DOCTOR_CLINIC');
                    updateIt = true;
                }
                if (updateIt) {
                    _updatedFields.categories.updated = true;
                    W.model.actionManager.add(new UpdateObject(venue, { categories: newCategories }));
                }
                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() {
                let venue = getSelectedVenue();
                let newName = venue.attributes.name;
                if (newName === this.originalName || this.confirmChange) {
                    let parts = getNameParts(this.originalName);
                    newName = toTitleCaseStrong(parts.base);
                    if (parts.base !== newName) {
                        W.model.actionManager.add(new UpdateObject(venue, { name: newName + (parts.suffix || '') }));
                        _updatedFields.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.');
                    bannButt.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'); }
            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 (PMUserList.hasOwnProperty(this.region) && PMUserList[this.region].approvalActive) {
                    var 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'
                    };
                    forumPMInputs['address_list[u][' + PMUserList[this.region].modID + ']'] = 'to';  // Sends a PM to the regional mod instead of the submission form
                    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'); }
            action() {
                let openPlaceWebsiteURL, 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) {
                        linkProceed = false;
                        if (confirm('***Localized store finder sites often show multiple nearby results. Please make sure you pick the right location.\nClick OK to agree and continue.')) {  // if the category doesn't translate, then pop an alert that will make a forum post to the thread
                            localStorage.setItem(_SETTING_IDS.sfUrlWarning, '1');  // prevent future warnings
                            linkProceed = true;
                        }
                    }
                } else {
                    let url = getSelectedVenue().url;
                    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, searchResultsWindowName, searchResultsWindowSpecs);
                    }
                }
            }
        }
    }; // END Flag namespace

    function getBannButt() {
        return {
            hnDashRemoved: null,
            fullAddressInference: null,
            nameMissing: null,
            //The buttons are appended in the code...
            plaIsPublic: null,
            plaNameMissing: null,
            plaNameNonStandard: null,
            indianaLiquorStoreHours: null,
            hoursOverlap: null,
            unmappedRegion: null,
            restAreaName: null,
            restAreaNoTransportation: null,
            restAreaGas: null,
            restAreaScenic: null,
            restAreaSpec: null,
            // if the gas brand and name don't match
            gasMismatch: null,
            gasUnbranded: null,
            gasMkPrim: null,
            isThisAPilotTravelCenter: null,
            hotelMkPrim: null,
            changeToPetVet: null,
            changeSchool2Offices: 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,
            pnhCatMess: null,
            specCaseMessageLow: null,
            changeToHospitalUrgentCare: 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: new Flag.AddCat2(), // special case flag
            addPharm: new Flag.AddPharm(), // special case flag
            addSuper: new Flag.AddSuper(), // special case flag
            appendAMPM: new Flag.AppendAMPM(), // special case flag
            addATM: new Flag.AddATM(), // special case flag
            addConvStore: new Flag.AddConvStore(), // special case flag
            isThisAPostOffice: null,
            STC: null,
            sfAliases: null,
            placeMatched: null,
            placeLocked: null,
            NewPlaceSubmit: null,
            ApprovalSubmit: null,
            PlaceWebsite: null
        };  // END bannButt definitions
    }

    // Main script
    function harmonizePlaceGo(item, useFlag, actions) {
        actions = actions || []; // Used for collecting all actions to be applied to the model.

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

        if (useFlag.indexOf('harmonize') > -1) {
            hpMode.harmFlag = true;
            phlog('Running script on selected place...');
        }
        if (useFlag.indexOf('highlight') > -1) {
            hpMode.hlFlag = true;
        }
        if (useFlag.indexOf('scan') > -1) {
            hpMode.scanFlag = true;
        }

        var placePL = getItemPL();  //  set up external post div and pull place PL
        // https://www.waze.com/editor/?env=usa&lon=-80.60757&lat=28.17850&layers=1957&zoom=4&segments=86124344&update_requestsFilter=false&problemsFilter=false&mapProblemFilter=0&mapUpdateRequestFilter=0&venueFilter=1
        placePL = placePL.replace(/\&layers=[^\&]+(\&?)/g, '$1');  // remove Permalink Layers
        placePL = placePL.replace(/\&s=[^\&]+(\&?)/g, '$1');  // remove Permalink Layers
        placePL = placePL.replace(/\&update_requestsFilter=[^\&]+(\&?)/g, '$1');  // remove Permalink Layers
        placePL = placePL.replace(/\&problemsFilter=[^\&]+(\&?)/g, '$1');  // remove Permalink Layers
        placePL = placePL.replace(/\&mapProblemFilter=[^\&]+(\&?)/g, '$1');  // remove Permalink Layers
        placePL = placePL.replace(/\&mapUpdateRequestFilter=[^\&]+(\&?)/g, '$1');  // remove Permalink Layers
        placePL = placePL.replace(/\&venueFilter=[^\&]+(\&?)/g, '$1');  // remove Permalink Layers
        var region, state2L;
        var gFormState = '';
        var PNHOrderNum = '', PNHNameTemp = '', PNHNameTempWeb = '';
        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
        };

        // **** 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

        bannButt = getBannButt();

        if (hpMode.harmFlag) {
            bannButt2 = {
                placesWiki: {
                    active: true, severity: 0, message: '', value: 'Places wiki', title: 'Open the places Wazeopedia (wiki) page',
                    action: function () {
                        window.open(_URLS.placesWiki);
                    }
                },
                restAreaWiki: {
                    active: false, severity: 0, message: '', value: 'Rest Area wiki', title: 'Open the Rest Area wiki page',
                    action: function () {
                        window.open(_URLS.restAreaWiki);
                    }
                },
                clearWL: {
                    active: false, severity: 0, message: '', value: 'Clear place whitelist', title: 'Clear all Whitelisted fields for this place',
                    action: function () {
                        if (confirm('Are you sure you want to clear all whitelisted fields for this place?')) {  // misclick check
                            delete venueWhitelist[itemID];
                            saveWL_LS(true);
                            harmonizePlaceGo(item, 'harmonize');  // rerun the script to check all flags again
                        }
                    }
                },  // END placesWiki definition
                PlaceErrorForumPost: {
                    active: true, severity: 0, message: '', value: 'Report script error', title: 'Report a script error',
                    action: function () {
                        reportError({
                            subject: 'WMEPH Bug report: Script Error',
                            message: 'Script version: ' + _SCRIPT_VERSION + devVersStr + '\nPermalink: ' + placePL + '\nPlace name: ' + item.attributes.name + '\nCountry: ' + addr.country.name + '\n--------\nDescribe the error:  \n '
                        });
                    }
                }

            };  // END bannButt2 definitions

            // 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
            bannServ = {
                addValet: {  // append optional Alias to the name
                    active: false, checked: false, icon: 'serv-valet', w2hratio: 50 / 50, value: 'Valet', title: 'Valet service', servIDIndex: 0,
                    action: function (actions, checked) {
                        setServiceChecked(this, checked, actions);
                    },
                    pnhOverride: false,
                    actionOn: function (actions) {
                        this.action(actions, true);
                    },
                    actionOff: function (actions) {
                        this.action(actions, false);
                    }
                },
                addDriveThru: {  // append optional Alias to the name
                    active: false, checked: false, icon: 'serv-drivethru', w2hratio: 78 / 50, value: 'DriveThru', title: 'Drive-thru', servIDIndex: 1,
                    action: function (actions, checked) {
                        setServiceChecked(this, checked, actions);
                    },
                    pnhOverride: false,
                    actionOn: function (actions) {
                        this.action(actions, true);
                    },
                    actionOff: function (actions) {
                        this.action(actions, false);
                    }
                },
                addWiFi: {  // append optional Alias to the name
                    active: false, checked: false, icon: 'serv-wifi', w2hratio: 67 / 50, value: 'WiFi', title: 'Wi-Fi', servIDIndex: 2,
                    action: function (actions, checked) {
                        setServiceChecked(this, checked, actions);
                    },
                    pnhOverride: false,
                    actionOn: function (actions) {
                        this.action(actions, true);
                    },
                    actionOff: function (actions) {
                        this.action(actions, false);
                    }
                },
                addRestrooms: {  // append optional Alias to the name
                    active: false, checked: false, icon: 'serv-restrooms', w2hratio: 49 / 50, value: 'Restroom', title: 'Restrooms', servIDIndex: 3,
                    action: function (actions, checked) {
                        setServiceChecked(this, checked, actions);
                    },
                    pnhOverride: false,
                    actionOn: function (actions) {
                        this.action(actions, true);
                    },
                    actionOff: function (actions) {
                        this.action(actions, false);
                    }
                },
                addCreditCards: {  // append optional Alias to the name
                    active: false, checked: false, icon: 'serv-credit', w2hratio: 73 / 50, value: 'CC', title: 'Accepts credit cards', servIDIndex: 4,
                    action: function (actions, checked) {
                        setServiceChecked(this, checked, actions);
                    },
                    pnhOverride: false,
                    actionOn: function (actions) {
                        this.action(actions, true);
                    },
                    actionOff: function (actions) {
                        this.action(actions, false);
                    }
                },
                addReservations: {  // append optional Alias to the name
                    active: false, checked: false, icon: 'serv-reservations', w2hratio: 55 / 50, value: 'Reserve', title: 'Reservations', servIDIndex: 5,
                    action: function (actions, checked) {
                        setServiceChecked(this, checked, actions);
                    },
                    pnhOverride: false,
                    actionOn: function (actions) {
                        this.action(actions, true);
                    },
                    actionOff: function (actions) {
                        this.action(actions, false);
                    }
                },
                addOutside: {  // append optional Alias to the name
                    active: false, checked: false, icon: 'serv-outdoor', w2hratio: 73 / 50, value: 'OusideSeat', title: 'Outdoor seating', servIDIndex: 6,
                    action: function (actions, checked) {
                        setServiceChecked(this, checked, actions);
                    },
                    pnhOverride: false,
                    actionOn: function (actions) {
                        this.action(actions, true);
                    },
                    actionOff: function (actions) {
                        this.action(actions, false);
                    }
                },
                addAC: {  // append optional Alias to the name
                    active: false, checked: false, icon: 'serv-ac', w2hratio: 50 / 50, value: 'AC', title: 'Air conditioning', servIDIndex: 7,
                    action: function (actions, checked) {
                        setServiceChecked(this, checked, actions);
                    },
                    pnhOverride: false,
                    actionOn: function (actions) {
                        this.action(actions, true);
                    },
                    actionOff: function (actions) {
                        this.action(actions, false);
                    }
                },
                addParking: {  // append optional Alias to the name
                    active: false, checked: false, icon: 'serv-parking', w2hratio: 46 / 50, value: 'Customer parking', title: 'Parking', servIDIndex: 8,
                    action: function (actions, checked) {
                        setServiceChecked(this, checked, actions);
                    },
                    pnhOverride: false,
                    actionOn: function (actions) {
                        this.action(actions, true);
                    },
                    actionOff: function (actions) {
                        this.action(actions, false);
                    }
                },
                addDeliveries: {  // append optional Alias to the name
                    active: false, checked: false, icon: 'serv-deliveries', w2hratio: 86 / 50, value: 'Delivery', title: 'Deliveries', servIDIndex: 9,
                    action: function (actions, checked) {
                        setServiceChecked(this, checked, actions);
                    },
                    pnhOverride: false,
                    actionOn: function (actions) {
                        this.action(actions, true);
                    },
                    actionOff: function (actions) {
                        this.action(actions, false);
                    }
                },
                addTakeAway: {  // append optional Alias to the name
                    active: false, checked: false, icon: 'serv-takeaway', w2hratio: 34 / 50, value: 'Take-out', title: 'Take-out', servIDIndex: 10,
                    action: function (actions, checked) {
                        setServiceChecked(this, checked, actions);
                    },
                    pnhOverride: false,
                    actionOn: function (actions) {
                        this.action(actions, true);
                    },
                    actionOff: function (actions) {
                        this.action(actions, false);
                    }
                },
                addWheelchair: {  // add service
                    active: false, checked: false, icon: 'serv-wheelchair', w2hratio: 50 / 50, value: 'WhCh', title: 'Wheelchair accessible', servIDIndex: 11,
                    action: function (actions, checked) {
                        setServiceChecked(this, checked, actions);
                    },
                    pnhOverride: false,
                    actionOn: function (actions) {
                        this.action(actions, true);
                    },
                    actionOff: function (actions) {
                        this.action(actions, false);
                    }
                },
                addDisabilityParking: {
                    active: false, checked: false, icon: 'serv-wheelchair', w2hratio: 50 / 50, value: 'DisabilityParking', title: 'Disability parking', servIDIndex: 12,
                    action: function (actions, checked) {
                        setServiceChecked(this, checked, actions);
                    },
                    pnhOverride: false,
                    actionOn: function (actions) {
                        this.action(actions, true);
                    },
                    actionOff: function (actions) {
                        this.action(actions, false);
                    }
                },
                add247: {  // add 24/7 hours
                    active: false, checked: false, icon: 'serv-247', w2hratio: 73 / 50, value: '247', title: 'Hours: Open 24\/7',
                    action: function (actions) {
                        if (!bannServ.add247.checked) {
                            let venue = getSelectedVenue();
                            addUpdateAction(venue, { openingHours: [new OpeningHour({ days: [1, 2, 3, 4, 5, 6, 0], fromHour: '00:00', toHour: '00:00' })] }, actions);
                            bannServ.add247.checked = true;
                            bannButt.noHours = null;
                        }
                    },
                    actionOn: function (actions) {
                        this.action(actions);
                    }
                }
            };  // END bannServ definitions

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

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

            if ($('#WMEPH-HideReportError').prop('checked')) {
                bannButt2.PlaceErrorForumPost.active = false;
            }
            //                 // provide Google search link to places
            //                 if (_USER.isDevUser || _USER.isBetaUser || _USER.rank > 1) {  // enable the link for all places, for R2+ and betas
            //                     bannButt.webSearch.active = true;
            //                 }
            // reset PNH lock level
            PNHLockLevel = -1;
        }

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

        var lockOK = true;  // if nothing goes wrong, then place will be locked
        var categories = item.attributes.categories;
        newCategories = categories.slice(0);
        var nameParts = getNameParts(item.attributes.name);
        var newNameSuffix = nameParts.suffix;
        newName = nameParts.base;
        newAliases = item.attributes.aliases.slice(0);
        var brand = item.attributes.brand;
        var newDescripion = item.attributes.description;
        newURL = item.attributes.url;
        var newURLSubmit = '';
        if (newURL !== null && newURL !== '') {
            newURLSubmit = newURL;
        }
        newPhone = item.attributes.phone;
        let addr = item.getAddress();
        if (addr.hasOwnProperty('attributes')) {
            addr = addr.attributes;
        }
        var PNHNameRegMatch;

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

        let result;
        // Check parking lot attributes.
        if (hpMode.harmFlag && item.isParkingLot()) bannServ.addDisabilityParking.active = true;
        result = Flag.PlaCostTypeMissing.eval(item, hpMode);
        bannButt.plaCostTypeMissing = result.flag;
        if (result.noLock) lockOK = false;
        result = Flag.PlaLotElevationMissing.eval(item);
        bannButt.plaLotElevationMissing = result.flag;
        if (result.noLock) lockOK = false;
        result = Flag.PlaSpaces.eval(item, hpMode);
        bannButt.plaSpaces = result.flag;
        result = Flag.PlaLotTypeMissing.eval(item, hpMode);
        bannButt.plaLotTypeMissing = result.flag;
        if (result.noLock) lockOK = false;
        bannButt.noPlaStopPoint = Flag.NoPlaStopPoint.eval(item).flag;
        bannButt.plaStopPointUnmoved = Flag.PlaStopPointUnmoved.eval(item).flag;
        bannButt.plaCanExitWhileClosed = Flag.PlaCanExitWhileClosed.eval(item, hpMode).flag;
        bannButt.plaPaymentTypeMissing = Flag.PlaPaymentTypeMissing.eval(item).flag;
        bannButt.plaHasAccessibleParking = Flag.PlaHasAccessibleParking.eval(item, hpMode).flag;

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

        if (hpMode.harmFlag && item.attributes.categories.indexOf('HOSPITAL_URGENT_CARE') > -1) {
            //bannButt.changeToDoctorClinic.active = true;
            //bannButt.changeToDoctorClinic.severity = 0;
        }


        // Whitelist breakout if place exists on the Whitelist and the option is enabled
        itemID = item.attributes.id;
        var itemGPS;
        if (venueWhitelist.hasOwnProperty(itemID) && (hpMode.harmFlag || (hpMode.hlFlag && !$('#WMEPH-DisableWLHL').prop('checked')))) {
            // Enable the clear WL button if any property is true
            for (var WLKey in venueWhitelist[itemID]) {  // loop thru the venue WL keys
                if (venueWhitelist[itemID].hasOwnProperty(WLKey) && (venueWhitelist[itemID][WLKey].active || false)) {
                    if (hpMode.harmFlag) bannButt2.clearWL.active = true;
                    _wl[WLKey] = venueWhitelist[itemID][WLKey];
                }
            }
            if (venueWhitelist[itemID].hasOwnProperty('dupeWL') && venueWhitelist[itemID].dupeWL.length > 0) {
                if (hpMode.harmFlag) bannButt2.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) itemGPS = OL.Layer.SphericalMercator.inverseMercator(item.attributes.geometry.getCentroid().x, item.attributes.geometry.getCentroid().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)) {
            alert('Country and/or state could not be determined.  Edit the place address and run WMEPH again.');
            return;
        }
        var countryName = addr.country.name;
        var 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) {
                alert('At present this script is not supported in this country.');
            }
            return 3;
        }

        // Parse state-based data
        state2L = 'Unknown'; region = 'Unknown';
        for (var usdix = 1; usdix < _PNH_DATA.states.length; usdix++) {
            stateDataTemp = _PNH_DATA.states[usdix].split('|');
            if (stateName === stateDataTemp[ps_state_ix]) {
                state2L = stateDataTemp[ps_state2L_ix];
                region = stateDataTemp[ps_region_ix];
                gFormState = stateDataTemp[ps_gFormState_ix];
                if (stateDataTemp[ps_defaultLockLevel_ix].match(/[1-5]{1}/) !== null) {
                    defaultLockLevel = stateDataTemp[ps_defaultLockLevel_ix] - 1;  // normalize by -1
                } else {
                    if (hpMode.harmFlag) {
                        alert('Lock level sheet data is not correct');
                    } else if (hpMode.hlFlag) {
                        return '3';
                    }
                }
                areaCodeList = areaCodeList + ',' + stateDataTemp[ps_areacode_ix];
                break;
            }
            // If State is not found, then use the country
            if (countryName === stateDataTemp[ps_state_ix]) {
                state2L = stateDataTemp[ps_state2L_ix];
                region = stateDataTemp[ps_region_ix];
                gFormState = stateDataTemp[ps_gFormState_ix];
                if (stateDataTemp[ps_defaultLockLevel_ix].match(/[1-5]{1}/) !== null) {
                    defaultLockLevel = stateDataTemp[ps_defaultLockLevel_ix] - 1;  // normalize by -1
                } else {
                    if (hpMode.harmFlag) {
                        alert('Lock level sheet data is not correct');
                    } else if (hpMode.hlFlag) {
                        return '3';
                    }
                }
                areaCodeList = areaCodeList + ',' + stateDataTemp[ps_areacode_ix];
                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
                    let 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[ps_state2L_ix] + '. region = ' + stateDataTemp[ps_region_ix];
                    }
                    reportError(data);
                }
            }
            return 3;
        }

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

        // Brand checking
        result = Flag.GasNoBrand.eval(item);
        bannButt.gasNoBrand = result.flag;
        if (result.noLock) lockOK = false;

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

        result = Flag.IsThisAPilotTravelCenter.eval(item, hpMode, state2L, newName, actions);
        bannButt.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 }));
                _updatedFields.name.updated = true;
            }

            // Add convenience store category to station
            if (newCategories.indexOf('CONVENIENCE_STORE') === -1 && !bannButt.subFuel) {
                bannButt.addConvStore.active = true;
            }
        }  // END Gas Station Checks

        // Note for Indiana editors to check liquor store hours if Sunday hours haven't been added yet.
        var tempAddr = item.getAddress();
        if (hpMode.harmFlag && tempAddr && tempAddr.getStateName() === 'Indiana' && !item.isResidential() &&
            [/\bbeers?\b/, /\bwines?\b/, /\bliquor\b/, /\bspirits\b/].some(re => re.test(newName)) && !item.attributes.openingHours.some(entry => entry.days.indexOf(0) > -1)) {
            if (!_wl.indianaLiquorStoreHours) bannButt.indianaLiquorStoreHours = new Flag.IndianaLiquorStoreHours();
        }

        var isLocked = item.attributes.lockRank >= (PNHLockLevel > -1 ? PNHLockLevel : defaultLockLevel);

        // 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()) {
                bannButt.pointNotArea = new Flag.PointNotArea();
            }
        } else if (item.isParkingLot() || (newName && newName.trim().length > 0)) {  // for non-residential places
            if (_USER.rank >= 2 && item.areExternalProvidersEditable() && !(item.isParkingLot() && $('#WMEPH-DisablePLAExtProviderCheck').prop('checked'))) {
                if (!newCategories.some(c => ['BRIDGE', 'TUNNEL', 'JUNCTION_INTERCHANGE', 'NATURAL_FEATURES', 'ISLAND', 'SEA_LAKE_POOL', 'RIVER_STREAM', 'CANAL', 'SWAMP_MARSH'].indexOf(c) > -1)) {
                    var provIDs = item.attributes.externalProviderIDs;
                    if (!provIDs || provIDs.length === 0) {
                        var lastUpdated = item.isNew() ? Date.now() : item.attributes.updatedOn ? item.attributes.updatedOn : item.attributes.createdOn;
                        var weeksSinceLastUpdate = (Date.now() - lastUpdated) / 604800000;
                        bannButt.extProviderMissing = new Flag.ExtProviderMissing();
                        if (isLocked && weeksSinceLastUpdate >= 26 && !item.isUpdated() && (!actions || actions.length === 0)) {
                            bannButt.extProviderMissing.severity = 3;
                            bannButt.extProviderMissing.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 if (!isLocked) {
                            bannButt.extProviderMissing.severity = 0;  // This will be changed to 3 later if the user does not choose to lock the place.
                            bannButt.extProviderMissing.message += ': ';
                            delete bannButt.extProviderMissing.value;
                            //delete bannButt.extProviderMissing.action;
                        } else {
                            bannButt.extProviderMissing.severity = 0;
                            bannButt.extProviderMissing.message += ': ';
                            delete bannButt.extProviderMissing.value;
                            //delete bannButt.extProviderMissing.action;
                        }
                    }
                }
            }

            // Place Harmonization
            var PNHMatchData;
            if (hpMode.harmFlag) {
                if (item.isParkingLot()) {
                    PNHMatchData = ['NoMatch'];
                } else {
                    PNHMatchData = harmoList(newName, state2L, region, _countryCode, newCategories, item, placePL);  // check against the PNH list
                }
            } 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;
                var showDispNote = true;
                var updatePNHName = true;
                // Break out the data headers
                var _PNH_DATA_headers;
                _PNH_DATA_headers = _PNH_DATA[_countryCode].pnh[0].split('|');
                var ph_name_ix = _PNH_DATA_headers.indexOf('ph_name');
                var ph_aliases_ix = _PNH_DATA_headers.indexOf('ph_aliases');
                var ph_category1_ix = _PNH_DATA_headers.indexOf('ph_category1');
                var ph_category2_ix = _PNH_DATA_headers.indexOf('ph_category2');
                var ph_description_ix = _PNH_DATA_headers.indexOf('ph_description');
                var ph_url_ix = _PNH_DATA_headers.indexOf('ph_url');
                var ph_order_ix = _PNH_DATA_headers.indexOf('ph_order');
                // var ph_notes_ix = _PNH_DATA_headers.indexOf('ph_notes');
                var ph_speccase_ix = _PNH_DATA_headers.indexOf('ph_speccase');
                var ph_sfurl_ix = _PNH_DATA_headers.indexOf('ph_sfurl');
                var ph_sfurllocal_ix = _PNH_DATA_headers.indexOf('ph_sfurllocal');
                // var ph_forcecat_ix = _PNH_DATA_headers.indexOf('ph_forcecat');
                var ph_displaynote_ix = _PNH_DATA_headers.indexOf('ph_displaynote');

                // Retrieve the data from the PNH line(s)
                var nsMultiMatch = false, orderList = [];
                if (PNHMatchData.length > 1) { // If multiple matches, then
                    var brandParent = -1, pmdTemp, pmdSpecCases, PNHMatchDataHold = PNHMatchData[0].split('|');
                    for (var pmdix = 0; pmdix < PNHMatchData.length; pmdix++) {  // For each of the matches,
                        pmdTemp = PNHMatchData[pmdix].split('|');  // Split the PNH data line
                        orderList.push(pmdTemp[ph_order_ix]);  // Add Order number to a list
                        if (pmdTemp[ph_speccase_ix].match(/brandParent(\d{1})/) !== null) {  // If there is a brandParent flag, prioritize by highest match
                            pmdSpecCases = pmdTemp[ph_speccase_ix].match(/brandParent(\d{1})/)[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
                }

                var priPNHPlaceCat = catTranslate(PNHMatchData[ph_category1_ix]);  // 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 + ''
                        });
                    }
                }

                // Check special cases
                var specCases, scFlag, localURLcheck = '';
                if (ph_speccase_ix > -1) {  // If the special cases column exists
                    specCases = PNHMatchData[ph_speccase_ix];  // pulls the speccases field from the PNH line
                    if (specCases !== '0' && specCases !== '') {
                        specCases = specCases.replace(/, /g, ',').split(',');  // remove spaces after commas and split by comma
                    }
                    for (var scix = 0; scix < specCases.length; scix++) {
                        // find any button/message flags in the special case (format: buttOn_xyzXyz, etc.)
                        if (specCases[scix].match(/^buttOn_/g) !== null) {
                            scFlag = specCases[scix].match(/^buttOn_(.+)/i)[1];
                            if (scFlag !== 'addCat2' || item.attributes.categories.indexOf(catTranslate(PNHMatchData[ph_category2_ix])) === -1) {
                                bannButt[scFlag].active = true;
                            }
                        } else if (specCases[scix].match(/^buttOff_/g) !== null) {
                            scFlag = specCases[scix].match(/^buttOff_(.+)/i)[1];
                            bannButt[scFlag].active = false;
                        } else if (specCases[scix].match(/^messOn_/g) !== null) {
                            scFlag = specCases[scix].match(/^messOn_(.+)/i)[1];
                            bannButt[scFlag].active = true;
                        } else if (specCases[scix].match(/^messOff_/g) !== null) {
                            scFlag = specCases[scix].match(/^messOff_(.+)/i)[1];
                            bannButt[scFlag].active = false;
                        } else if (specCases[scix].match(/^psOn_/g) !== null) {
                            scFlag = specCases[scix].match(/^psOn_(.+)/i)[1];
                            bannServ[scFlag].actionOn(actions);
                            bannServ[scFlag].pnhOverride = true;
                        } else if (specCases[scix].match(/^psOff_/g) !== null) {
                            scFlag = specCases[scix].match(/^psOff_(.+)/i)[1];
                            bannServ[scFlag].actionOff(actions);
                            bannServ[scFlag].pnhOverride = true;
                        }
                        // parseout localURL data if exists (meaning place can have a URL distinct from the chain URL
                        if (specCases[scix].match(/^localURL_/g) !== null) {
                            localURLcheck = specCases[scix].match(/^localURL_(.+)/i)[1];
                        }
                        // parse out optional alt-name
                        if (specCases[scix].match(/^optionAltName<>(.+)/g) !== null) {
                            optionalAlias = specCases[scix].match(/^optionAltName<>(.+)/i)[1];
                            if (newAliases.indexOf(optionalAlias) === -1) {
                                bannButt.addAlias = new Flag.AddAlias();
                            }
                        }
                        // Gas Station forceBranding
                        if (['GAS_STATION'].indexOf(priPNHPlaceCat) > -1 && specCases[scix].match(/^forceBrand<>(.+)/i) !== null) {
                            var forceBrand = specCases[scix].match(/^forceBrand<>(.+)/i)[1];
                            if (item.attributes.brand !== forceBrand) {
                                actions.push(new UpdateObject(item, { brand: forceBrand }));
                                _updatedFields.brand.updated = true;
                                phlogdev('Gas brand updated from PNH');
                            }
                        }
                        // Check Localization
                        if (specCases[scix].match(/^checkLocalization<>(.+)/i) !== null) {
                            updatePNHName = false;
                            var baseName = specCases[scix].match(/^checkLocalization<>(.+)/i)[1];
                            var baseNameRE = new RegExp(baseName, 'g');
                            if ((newName + (newNameSuffix ? newNameSuffix : '')).match(baseNameRE) === null) {
                                bannButt.localizedName = new Flag.LocalizedName();
                                if (_wl.localizedName) {
                                    bannButt.localizedName.WLactive = false;
                                }
                                //bannButt.PlaceWebsite.value = 'Place Website';
                                if (ph_displaynote_ix > -1 && PNHMatchData[ph_displaynote_ix] !== '0' && PNHMatchData[ph_displaynote_ix] !== '') {
                                    bannButt.localizedName.message = PNHMatchData[ph_displaynote_ix];
                                }
                            }
                            showDispNote = false;
                        }

                        // Prevent name change
                        if (specCases[scix].match(/keepName/g) !== null) {
                            updatePNHName = false;
                        }
                    }
                }

                // If it's a place that also sells fuel, enable the button
                if (PNHMatchData[ph_speccase_ix] === 'subFuel' && newName.toUpperCase().indexOf('GAS') === -1 && newName.toUpperCase().indexOf('FUEL') === -1) {
                    bannButt.subFuel = new Flag.SubFuel();
                    if (_wl.subFuel) {
                        bannButt.subFuel.WLactive = false;
                    }
                }

                // Display any notes for the specific place
                if (showDispNote && ph_displaynote_ix > -1 && PNHMatchData[ph_displaynote_ix] !== '0' && PNHMatchData[ph_displaynote_ix] !== '') {
                    if (containsAny(specCases, ['pharmhours'])) {
                        if (item.attributes.description.toUpperCase().indexOf('PHARMACY') === -1 || (item.attributes.description.toUpperCase().indexOf('HOURS') === -1 && item.attributes.description.toUpperCase().indexOf('HRS') === -1)) {
                            bannButt.specCaseMessage = new Flag.SpecCaseMessage(PNHMatchData[ph_displaynote_ix]);
                        }
                    } else if (containsAny(specCases, ['drivethruhours'])) {
                        if (item.attributes.description.toUpperCase().indexOf('DRIVE') === -1 || (item.attributes.description.toUpperCase().indexOf('HOURS') === -1 && item.attributes.description.toUpperCase().indexOf('HRS') === -1)) {
                            if ($('#service-checkbox-' + 'DRIVETHROUGH').prop('checked')) {
                                bannButt.specCaseMessage = new Flag.SpecCaseMessage(PNHMatchData[ph_displaynote_ix]);
                            } else {
                                bannButt.specCaseMessageLow = new Flag.SpecCaseMessageLow(PNHMatchData[ph_displaynote_ix]);
                            }
                        }
                    } else {
                        bannButt.specCaseMessageLow = new Flag.SpecCaseMessageLow(PNHMatchData[ph_displaynote_ix]);
                    }
                }

                // Localized Storefinder code:
                customStoreFinderLocal = false;
                customStoreFinderLocalURL = '';
                customStoreFinder = false;
                customStoreFinderURL = '';
                if (ph_sfurl_ix > -1) {  // if the sfurl column exists...
                    if (ph_sfurllocal_ix > -1 && PNHMatchData[ph_sfurllocal_ix] !== '' && PNHMatchData[ph_sfurllocal_ix] !== '0') {
                        if (!bannButt.localizedName) {
                            bannButt.PlaceWebsite = new Flag.PlaceWebsite();
                            bannButt.PlaceWebsite.value = 'Location Finder (L)';
                        }
                        var tempLocalURL = PNHMatchData[ph_sfurllocal_ix].replace(/ /g, '').split('<>');
                        var searchStreet = '', searchCity = '', searchState = '';
                        if ('string' === typeof addr.street.name) {
                            searchStreet = addr.street.name;
                        }
                        var searchStreetPlus = searchStreet.replace(/ /g, '+');
                        searchStreet = searchStreet.replace(/ /g, '%20');
                        if ('string' === typeof addr.city.attributes.name) {
                            searchCity = addr.city.attributes.name;
                        }
                        var searchCityPlus = searchCity.replace(/ /g, '+');
                        searchCity = searchCity.replace(/ /g, '%20');
                        if ('string' === typeof addr.state.name) {
                            searchState = addr.state.name;
                        }
                        var searchStatePlus = searchState.replace(/ /g, '+');
                        searchState = searchState.replace(/ /g, '%20');

                        for (var tlix = 1; tlix < tempLocalURL.length; tlix++) {
                            if (tempLocalURL[tlix] === 'ph_streetName') {
                                customStoreFinderLocalURL = customStoreFinderLocalURL + searchStreet;
                            } else if (tempLocalURL[tlix] === 'ph_streetNamePlus') {
                                customStoreFinderLocalURL = customStoreFinderLocalURL + searchStreetPlus;
                            } else if (tempLocalURL[tlix] === 'ph_cityName') {
                                customStoreFinderLocalURL = customStoreFinderLocalURL + searchCity;
                            } else if (tempLocalURL[tlix] === 'ph_cityNamePlus') {
                                customStoreFinderLocalURL = customStoreFinderLocalURL + searchCityPlus;
                            } else if (tempLocalURL[tlix] === 'ph_stateName') {
                                customStoreFinderLocalURL = customStoreFinderLocalURL + searchState;
                            } else if (tempLocalURL[tlix] === 'ph_stateNamePlus') {
                                customStoreFinderLocalURL = customStoreFinderLocalURL + searchStatePlus;
                            } else if (tempLocalURL[tlix] === 'ph_state2L') {
                                customStoreFinderLocalURL = 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') {
                                if (!itemGPS) itemGPS = OL.Layer.SphericalMercator.inverseMercator(item.attributes.geometry.getCentroid().x, item.attributes.geometry.getCentroid().y);
                                customStoreFinderLocalURL = customStoreFinderLocalURL + itemGPS.lat;
                            } else if (tempLocalURL[tlix] === 'ph_longitudePM') {
                                if (!itemGPS) itemGPS = OL.Layer.SphericalMercator.inverseMercator(item.attributes.geometry.getCentroid().x, item.attributes.geometry.getCentroid().y);
                                customStoreFinderLocalURL = customStoreFinderLocalURL + itemGPS.lon;
                            } else if (tempLocalURL[tlix] === 'ph_latitudePMBuffMin') {
                                if (!itemGPS) itemGPS = OL.Layer.SphericalMercator.inverseMercator(item.attributes.geometry.getCentroid().x, item.attributes.geometry.getCentroid().y);
                                customStoreFinderLocalURL = customStoreFinderLocalURL + (itemGPS.lat - 0.15).toString();
                            } else if (tempLocalURL[tlix] === 'ph_longitudePMBuffMin') {
                                if (!itemGPS) itemGPS = OL.Layer.SphericalMercator.inverseMercator(item.attributes.geometry.getCentroid().x, item.attributes.geometry.getCentroid().y);
                                customStoreFinderLocalURL = customStoreFinderLocalURL + (itemGPS.lon - 0.15).toString();
                            } else if (tempLocalURL[tlix] === 'ph_latitudePMBuffMax') {
                                if (!itemGPS) itemGPS = OL.Layer.SphericalMercator.inverseMercator(item.attributes.geometry.getCentroid().x, item.attributes.geometry.getCentroid().y);
                                customStoreFinderLocalURL = customStoreFinderLocalURL + (itemGPS.lat + 0.15).toString();
                            } else if (tempLocalURL[tlix] === 'ph_longitudePMBuffMax') {
                                if (!itemGPS) itemGPS = OL.Layer.SphericalMercator.inverseMercator(item.attributes.geometry.getCentroid().x, item.attributes.geometry.getCentroid().y);
                                customStoreFinderLocalURL = customStoreFinderLocalURL + (itemGPS.lon + 0.15).toString();
                            } else if (tempLocalURL[tlix] === 'ph_houseNumber') {
                                customStoreFinderLocalURL = customStoreFinderLocalURL + (item.attributes.houseNumber ? item.attributes.houseNumber : '');
                            } else {
                                customStoreFinderLocalURL = customStoreFinderLocalURL + tempLocalURL[tlix];
                            }
                        }
                        if (customStoreFinderLocalURL.indexOf('http') !== 0) {
                            customStoreFinderLocalURL = 'http:\/\/' + customStoreFinderLocalURL;
                        }
                        customStoreFinderLocal = true;
                    } else if (PNHMatchData[ph_sfurl_ix] !== '' && PNHMatchData[ph_sfurl_ix] !== '0') {
                        if (!bannButt.localizedName) {
                            bannButt.PlaceWebsite = new Flag.PlaceWebsite();
                        }
                        customStoreFinderURL = PNHMatchData[ph_sfurl_ix];
                        if (customStoreFinderURL.indexOf('http') !== 0) {
                            customStoreFinderURL = 'http:\/\/' + customStoreFinderURL;
                        }
                        customStoreFinder = true;
                    }
                }

                // Category translations
                var altCategories = PNHMatchData[ph_category2_ix];
                if (altCategories !== '0' && altCategories !== '') {  //  translate alt-cats to WME code
                    altCategories = altCategories.replace(/,[^A-Za-z0-9]*/g, ',').split(',');  // tighten and split by comma
                    for (var catix = 0; catix < altCategories.length; catix++) {
                        var 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;
                        } else {
                            altCategories[catix] = newAltTemp;  // replace with translated element
                        }
                    }
                }

                // name parsing with category exceptions
                if (['HOTEL'].indexOf(priPNHPlaceCat) > -1) {
                    var nameToCheck = newName + (newNameSuffix ? newNameSuffix : '');
                    if (nameToCheck.toUpperCase() === PNHMatchData[ph_name_ix].toUpperCase()) {  // If no localization
                        bannButt.catHotel = new Flag.CatHotel(PNHMatchData[ph_name_ix]);
                        newName = PNHMatchData[ph_name_ix];
                    } else {
                        // Replace PNH part of name with PNH name
                        var splix = newName.toUpperCase().replace(/[-\/]/g, ' ').indexOf(PNHMatchData[ph_name_ix].toUpperCase().replace(/[-\/]/g, ' '));
                        if (splix > -1) {
                            var frontText = newName.slice(0, splix);
                            var backText = newName.slice(splix + PNHMatchData[ph_name_ix].length);
                            newName = PNHMatchData[ph_name_ix];
                            if (frontText.length > 0) { newName = frontText + ' ' + newName; }
                            if (backText.length > 0) { newName = newName + ' ' + backText; }
                            newName = newName.replace(/ {2,}/g, ' ');
                        } else {
                            newName = PNHMatchData[ph_name_ix];
                        }
                    }
                    if (altCategories !== '0' && altCategories !== '') {  // 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
                        bannButt.hotelMkPrim = new Flag.HotelMkPrim();
                        if (_wl.hotelMkPrim) {
                            bannButt.hotelMkPrim.WLactive = false;
                        } else {
                            lockOK = false;
                        }
                    } else if (newCategories.indexOf('HOTEL') > -1) {
                        // Remove LODGING if it exists
                        var lodgingIdx = newCategories.indexOf('LODGING');
                        if (lodgingIdx > -1) {
                            newCategories.splice(lodgingIdx, 1);
                        }
                    }
                    // If PNH match, set wifi service.
                    if (PNHMatchData && !bannServ.addWiFi.checked) {
                        bannServ.addWiFi.action();
                    }
                    // Set hotel hours to 24/7 for all hotels.
                    if (!bannServ.add247.checked) {
                        bannServ.add247.action();
                    }
                } else if (newCategories.indexOf('BANK_FINANCIAL') > -1 && PNHMatchData[ph_speccase_ix].indexOf('notABank') === -1) {
                    // 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 (newName.match(/\batm\b/ig) !== null) {
                        if (ixOffices === 0) {
                            bannButt.bankType1 = new Flag.BankType1();
                            bannButt.bankBranch = new Flag.BankBranch();
                            bannButt.standaloneATM = new Flag.StandaloneATM();
                            bannButt.bankCorporate = new Flag.BankCorporate();
                        } else if (ixBank === -1 && ixATM === -1) {
                            bannButt.bankBranch = new Flag.BankBranch();
                            bannButt.standaloneATM = new Flag.StandaloneATM();
                        } else if (ixATM === 0 && ixBank > 0) {
                            bannButt.bankBranch = new Flag.BankBranch();
                        } else if (ixBank > -1) {
                            bannButt.bankBranch = new Flag.BankBranch();
                            bannButt.standaloneATM = new Flag.StandaloneATM();
                        }
                        newName = PNHMatchData[ph_name_ix] + ' 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) {
                            bannButt.bankBranch = new Flag.BankBranch();
                        } else if (ixBank > -1 && ixATM === -1) {
                            bannButt.addATM.active = true;
                        } else if (ixATM === 0 && ixBank === -1) {
                            bannButt.bankBranch = new Flag.BankBranch();
                            bannButt.standaloneATM = new Flag.StandaloneATM();
                        } else if (ixBank > 0 && ixATM > 0) {
                            bannButt.bankBranch = new Flag.BankBranch();
                            bannButt.standaloneATM = new Flag.StandaloneATM();
                        }
                        newName = PNHMatchData[ph_name_ix];
                        // 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);
                        bannButt.standaloneATM = new Flag.StandaloneATM();
                        bannButt.bankCorporate = new Flag.BankCorporate();
                    }// END PNH bank treatment
                } else if (['GAS_STATION'].indexOf(priPNHPlaceCat) > -1) {  // for PNH gas stations, don't replace existing sub-categories
                    if (altCategories !== '0' && altCategories !== '') {  // 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
                        bannButt.gasMkPrim = new Flag.GasMkPrim();
                        lockOK = false;
                    } else {
                        newName = PNHMatchData[ph_name_ix];
                    }
                } else if (updatePNHName) {  // if not a special category then update the name
                    newName = PNHMatchData[ph_name_ix];
                    newCategories = insertAtIX(newCategories, priPNHPlaceCat, 0);
                    if (altCategories !== '0' && altCategories !== '' && specCases.indexOf('buttOn_addCat2') === -1 && specCases.indexOf('optionCat2') === -1) {
                        newCategories = insertAtIX(newCategories, altCategories, 1);
                    }
                } else if (!updatePNHName) {
                    // Strong title case option for non-PNH places
                    var titleCaseName = toTitleCaseStrong(newName);
                    if (newName !== titleCaseName) {
                        bannButt.STC = new Flag.STC();
                        bannButt.STC.suffixMessage = '<span style="margin-left: 4px;font-size: 14px">&bull; ' + titleCaseName + (newNameSuffix || '') + '</span>';
                        bannButt.STC.title += titleCaseName;
                        bannButt.STC.originalName = newName + (newNameSuffix || '');
                    }
                }

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

                // Parse URL data
                var 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[ph_url_ix], false, true, item, region);
                            bannButt.localURL = new Flag.LocalURL();
                        }
                    } else {
                        newURL = normalizeURL(PNHMatchData[ph_url_ix], false, true, item, region);
                        bannButt.localURL = new Flag.LocalURL();
                    }
                } else {
                    newURL = normalizeURL(PNHMatchData[ph_url_ix], false, true, item, region);
                }
                // Parse PNH Aliases
                newAliasesTemp = PNHMatchData[ph_aliases_ix].match(/([^\(]*)/i)[0];
                if (newAliasesTemp !== '0' && 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.indexOf('noUpdateAlias') === -1 && (!containsAll(newAliases, newAliasesTemp) && newAliasesTemp !== '0' && newAliasesTemp !== '' && specCases.indexOf('optionName2') === -1)) {
                    newAliases = insertAtIX(newAliases, newAliasesTemp, 0);
                }
                // Enable optional alt-name button
                if (bannButt.addAlias) {
                    bannButt.addAlias.message = 'Is there a ' + optionalAlias + ' at this location?';
                    bannButt.addAlias.title = 'Add ' + optionalAlias;
                }

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

                // update categories if different and no Cat2 option
                if (!matchSets(_.uniq(item.attributes.categories), _.uniq(newCategories))) {
                    if (specCases.indexOf('optionCat2') === -1 && specCases.indexOf('buttOn_addCat2') === -1) {
                        phlogdev('Categories updated with ' + newCategories);
                        actions.push(new UpdateObject(item, { categories: newCategories }));
                        //W.model.actionManager.add(new UpdateObject(item, { categories: newCategories }));
                        _updatedFields.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 }));
                        _updatedFields.categories.updated = true;
                    }
                }
                // Enable optional 2nd category button
                if (specCases.indexOf('buttOn_addCat2') > -1 && newCategories.indexOf(catTransWaze2Lang[altCategories[0]]) === -1) {
                    let altCat = altCategories[0];
                    bannButt.addCat2.message = 'Is there a ' + catTransWaze2Lang[altCat] + ' at this location?';
                    bannButt.addCat2.title = 'Add ' + catTransWaze2Lang[altCat];
                    bannButt.addCat2.altCategory = altCat;
                }

                // Description update
                newDescripion = PNHMatchData[ph_description_ix];
                if (newDescripion !== null && newDescripion !== '0' && item.attributes.description.toUpperCase().indexOf(newDescripion.toUpperCase()) === -1) {
                    if (item.attributes.description !== '' && item.attributes.description !== null && item.attributes.description !== ' ') {
                        bannButt.checkDescription = new Flag.CheckDescription();
                    }
                    phlogdev('Description updated');
                    newDescripion = newDescripion + '\n' + item.attributes.description;
                    actions.push(new UpdateObject(item, { description: newDescripion }));
                    _updatedFields.description.updated = true;
                }

                // Special Lock by PNH
                if (specCases.indexOf('lockAt5') > -1) {
                    PNHLockLevel = 4;
                }

            } else {  // if no PNH match found
                if (PNHMatchData[0] === 'ApprovalNeeded') {
                    //PNHNameTemp = PNHMatchData[1].join(', ');
                    PNHNameTemp = PNHMatchData[1][0];  // Just do the first match
                    PNHNameTempWeb = encodeURIComponent(PNHNameTemp);
                    PNHOrderNum = PNHMatchData[2].join(',');
                }

                // Strong title case option for non-PNH places
                var titleCaseName = toTitleCaseStrong(newName);
                if (newName !== titleCaseName) {
                    bannButt.STC = new Flag.STC();
                    bannButt.STC.suffixMessage = '<span style="margin-left: 4px;font-size: 14px">&bull; ' + titleCaseName + (newNameSuffix || '') + '</span>';
                    bannButt.STC.title += titleCaseName;
                    bannButt.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) {
                        bannButt.bankType1 = new Flag.BankType1();
                        bannButt.bankBranch = new Flag.BankBranch();
                        bannButt.standaloneATM = new Flag.StandaloneATM();
                        bannButt.bankCorporate = new Flag.BankCorporate();
                    } else if (ixBank === -1 && ixATM === -1) {
                        bannButt.bankBranch = new Flag.BankBranch();
                        bannButt.standaloneATM = new Flag.StandaloneATM();
                    } else if (ixATM === 0 && ixBank > 0) {
                        bannButt.bankBranch = new Flag.BankBranch();
                    } else if (ixBank > -1) {
                        bannButt.bankBranch = new Flag.BankBranch();
                        bannButt.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) {
                        bannButt.bankBranch = new Flag.BankBranch();
                    } else if (ixBank > -1 && ixATM === -1) {
                        bannButt.addATM.active = true;
                    } else if (ixATM === 0 && ixBank === -1) {
                        bannButt.bankBranch = new Flag.BankBranch();
                        bannButt.standaloneATM = new Flag.StandaloneATM();
                    } else if (ixBank > 0 && ixATM > 0) {
                        bannButt.bankBranch = new Flag.BankBranch();
                        bannButt.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:
            var CH_DATA = _PNH_DATA[_countryCode].categories;
            var CH_NAMES = _PNH_DATA[_countryCode].categoryNames;
            var CH_DATA_headers = CH_DATA[0].split('|');
            var CH_DATA_keys = CH_DATA[1].split('|');
            var CH_DATA_list = CH_DATA[2].split('|');
            var CH_DATA_Temp;

            if (hpMode.harmFlag) {
                // Update name:
                if ((newName + (newNameSuffix ? newNameSuffix : '')) !== item.attributes.name) {
                    phlogdev('Name updated');
                    actions.push(new UpdateObject(item, { name: newName + (newNameSuffix ? newNameSuffix : '') }));
                    //actions.push(new UpdateObject(item, { name: newName }));
                    _updatedFields.name.updated = true;
                }

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

                // Make PNH submission links
                var regionFormURL = '';
                var newPlaceAddon = '';
                var approvalAddon = '';
                var approvalMessage = 'Submitted via WMEPH. PNH order number ' + PNHOrderNum;
                var tempSubmitName_encoded = encodeURIComponent(newName);
                var placePL_encoded = encodeURIComponent(placePL);
                var newURLSubmit_encoded = encodeURIComponent(newURLSubmit);
                let suffix = _USER.name + gFormState;
                switch (region) {
                    case 'NWR': regionFormURL = 'https://docs.google.com/forms/d/1hv5hXBlGr1pTMmo4n3frUx1DovUODbZodfDBwwTc7HE/viewform';
                        newPlaceAddon = '?entry.925969794=' + tempSubmitName_encoded + '&entry.1970139752=' + newURLSubmit_encoded + '&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=' + tempSubmitName_encoded + '&entry.1970139752=' + newURLSubmit_encoded + '&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=' + tempSubmitName_encoded + '&entry.1970139752=' + newURLSubmit_encoded + '&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=' + tempSubmitName_encoded + '&entry.1970139752=' + newURLSubmit_encoded + '&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=' + tempSubmitName_encoded + '&entry.1970139752=' + newURLSubmit_encoded + '&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=' + tempSubmitName_encoded + '&entry.1970139752=' + newURLSubmit_encoded + '&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=' + tempSubmitName_encoded + '&entry.1970139752=' + newURLSubmit_encoded + '&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=' + tempSubmitName_encoded + '&entry.2018912633=' + newURLSubmit_encoded + '&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=' + tempSubmitName_encoded + '&entry.1422079728=' + newURLSubmit_encoded + '&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=' + tempSubmitName_encoded + '&entry.1970139752=' + newURLSubmit_encoded + '&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=' + tempSubmitName_encoded + '&entry.1970139752=' + newURLSubmit_encoded + '&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=' + tempSubmitName_encoded + '&entry.1970139752=' + newURLSubmit_encoded + '&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=' + tempSubmitName_encoded + '&entry.1970139752=' + newURLSubmit_encoded + '&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=' + tempSubmitName_encoded + '&entry_1067461077=' + newURLSubmit_encoded + '&entry_318793106=' + _USER.name + '&entry_1149649663=' + placePL_encoded;
                        approvalAddon = '?entry_839085807=' + PNHNameTempWeb + '&entry_1125435193=' + approvalMessage + '&entry_318793106=' + _USER.name + '&entry_1149649663=' + placePL_encoded;
                        break;
                    case 'QC': regionFormURL = 'https://docs.google.com/forms/d/13JwXsrWPNmCdfGR5OVr5jnGZw-uNGohwgjim-JYbSws/viewform';
                        newPlaceAddon = '?entry_839085807=' + tempSubmitName_encoded + '&entry_1067461077=' + newURLSubmit_encoded + '&entry_318793106=' + _USER.name + '&entry_1149649663=' + placePL_encoded;
                        approvalAddon = '?entry_839085807=' + PNHNameTempWeb + '&entry_1125435193=' + approvalMessage + '&entry_318793106=' + _USER.name + '&entry_1149649663=' + placePL_encoded;
                        break;
                    default: regionFormURL = '';
                }
                newPlaceURL = regionFormURL + newPlaceAddon;
                approveRegionURL = regionFormURL + approvalAddon;


                // PNH specific Services:

                var servHeaders = [], servKeys = [], servList = [], servHeaderCheck;
                for (var jjj = 0; jjj < CH_DATA_headers.length; jjj++) {
                    servHeaderCheck = CH_DATA_headers[jjj].match(/^ps_/i);  // if it's a service header
                    if (servHeaderCheck) {
                        servHeaders.push(jjj);
                        servKeys.push(CH_DATA_keys[jjj]);
                        servList.push(CH_DATA_list[jjj]);
                    }
                }

                if (newCategories.length > 0) {
                    for (var iii = 0; iii < CH_NAMES.length; iii++) {
                        if (newCategories.indexOf(CH_NAMES[iii]) > -1) {
                            CH_DATA_Temp = CH_DATA[iii].split('|');
                            for (var psix = 0; psix < servHeaders.length; psix++) {
                                if (!bannServ[servKeys[psix]].pnhOverride) {
                                    if (CH_DATA_Temp[servHeaders[psix]] === '1') {  // These are automatically added to all countries/regions (if auto setting is on)
                                        bannServ[servKeys[psix]].active = true;
                                        if ($('#WMEPH-EnableServices').prop('checked')) {
                                            // Automatically enable new services
                                            bannServ[servKeys[psix]].actionOn(actions);
                                        }
                                    } else if (CH_DATA_Temp[servHeaders[psix]] === '2') {  // these are never automatically added but shown
                                        bannServ[servKeys[psix]].active = true;
                                    } else if (CH_DATA_Temp[servHeaders[psix]] !== '') {  // check for state/region auto add
                                        bannServ[servKeys[psix]].active = true;
                                        if ($('#WMEPH-EnableServices').prop('checked')) {
                                            var servAutoRegion = CH_DATA_Temp[servHeaders[psix]].replace(/,[^A-za-z0-9]*/g, ',').split(',');
                                            // if the sheet data matches the state, region, or username then auto add
                                            if (servAutoRegion.indexOf(state2L) > -1 || servAutoRegion.indexOf(region) > -1 || servAutoRegion.indexOf(_USER.name) > -1) {
                                                bannServ[servKeys[psix]].actionOn(actions);
                                            }
                                        }
                                    }
                                }
                            }
                        }
                    }
                }
            }


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

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

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

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

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

                    // display any messaged regarding the category
                    let pc_message = '';
                    if (newCategories.indexOf('HOSPITAL_MEDICAL_CARE') === -1) {
                        pc_message = CH_DATA_Temp[CH_DATA_headers.indexOf('pc_message')];
                    }
                    if (pc_message && pc_message !== '0' && pc_message !== '') {
                        bannButt.pnhCatMess = new Flag.PnhCatMess(pc_message);
                    }
                    // Unmapped categories
                    let pc_rare = CH_DATA_Temp[CH_DATA_headers.indexOf('pc_rare')].replace(/,[^A-Za-z0-9}]+/g, ',').split(',');
                    if (pc_rare.indexOf(state2L) > -1 || pc_rare.indexOf(region) > -1 || pc_rare.indexOf(_countryCode) > -1) {
                        if (CH_DATA_Temp[0] === 'OTHER' && ['GLR', 'NER', 'NWR', 'PLN', 'SCR', 'SER', 'NOR', 'HI', 'SAT'].indexOf(region) > -1) {
                            if (!isLocked) {
                                bannButt.unmappedRegion = new Flag.UnmappedRegion();
                                bannButt.unmappedRegion.WLactive = false;
                                bannButt.unmappedRegion.severity = 1;
                                bannButt.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 {
                            bannButt.unmappedRegion = new Flag.UnmappedRegion();
                            if (_wl.unmappedRegion) {
                                bannButt.unmappedRegion.WLactive = false;
                                bannButt.unmappedRegion.severity = 0;
                            } else {
                                lockOK = false;
                            }
                        }
                    }
                    // Parent Category
                    let pc_parent = CH_DATA_Temp[CH_DATA_headers.indexOf('pc_parent')].replace(/,[^A-Za-z0-9}]+/g, ',').split(',');
                    if (pc_parent.indexOf(state2L) > -1 || pc_parent.indexOf(region) > -1 || pc_parent.indexOf(_countryCode) > -1) {
                        bannButt.parentCategory = new Flag.ParentCategory();
                        if (_wl.parentCategory) {
                            bannButt.parentCategory.WLactive = false;
                        }
                    }
                    // Set lock level
                    for (var lockix = 1; lockix < 6; lockix++) {
                        let pc_lockTemp = CH_DATA_Temp[CH_DATA_headers.indexOf('pc_lock' + lockix)].replace(/,[^A-Za-z0-9}]+/g, ',').split(',');
                        if (lockix - 1 > highestCategoryLock && (pc_lockTemp.indexOf(state2L) > -1 || pc_lockTemp.indexOf(region) > -1 || pc_lockTemp.indexOf(_countryCode) > -1)) {
                            highestCategoryLock = lockix - 1;  // Offset by 1 since lock ranks start at 0
                        }
                    }
                }
            }

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

            if (isPoint) {
                if (maxPointSeverity === 3) {
                    bannButt.areaNotPoint = new Flag.AreaNotPoint();
                    if (_wl.areaNotPoint || item.attributes.lockRank >= defaultLockLevel) {
                        bannButt.areaNotPoint.WLactive = false;
                        bannButt.areaNotPoint.severity = 0;
                    } else {
                        lockOK = false;
                    }
                } else if (maxPointSeverity === 2) {
                    bannButt.areaNotPointMid = new Flag.AreaNotPointMid();
                    if (_wl.areaNotPoint || item.attributes.lockRank >= defaultLockLevel) {
                        bannButt.areaNotPointMid.WLactive = false;
                        bannButt.areaNotPointMid.severity = 0;
                    } else {
                        lockOK = false;
                    }
                } else if (maxPointSeverity === 1) {
                    bannButt.areaNotPointLow = new Flag.AreaNotPointLow();
                    if (_wl.areaNotPoint || item.attributes.lockRank >= defaultLockLevel) {
                        bannButt.areaNotPointLow.WLactive = false;
                        bannButt.areaNotPointLow.severity = 0;
                    }
                }
            } else {
                if (maxAreaSeverity === 3) {
                    bannButt.pointNotArea = new Flag.PointNotArea();
                    if (_wl.pointNotArea || item.attributes.lockRank >= defaultLockLevel) {
                        bannButt.pointNotArea.WLactive = false;
                        bannButt.pointNotArea.severity = 0;
                    } else {
                        lockOK = false;
                    }
                } else if (maxAreaSeverity === 2) {
                    bannButt.pointNotAreaMid = new Flag.PointNotAreaMid();
                    if (_wl.pointNotArea || item.attributes.lockRank >= defaultLockLevel) {
                        bannButt.pointNotAreaMid.WLactive = false;
                        bannButt.pointNotAreaMid.severity = 0;
                    } else {
                        lockOK = false;
                    }
                } else if (maxAreaSeverity === 1) {
                    bannButt.pointNotAreaLow = new Flag.PointNotAreaLow();
                    if (_wl.pointNotArea || item.attributes.lockRank >= defaultLockLevel) {
                        bannButt.pointNotAreaLow.WLactive = false;
                        bannButt.pointNotAreaLow.severity = 0;
                    }
                }
            }

            var anpNone = collegeAbbreviations.split('|'), anpNoneRE;
            for (var cii = 0; cii < anpNone.length; cii++) {
                anpNoneRE = new RegExp('\\b' + anpNone[cii] + '\\b', 'g');
                if (newName.match(anpNoneRE) !== null && bannButt.areaNotPointLow) {
                    bannButt.areaNotPointLow.severity = 0;
                    bannButt.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'])) {
                    bannButt.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'])) {
                        bannButt.noHours.WLactive = false;
                        bannButt.noHours.severity = 0;
                    }
                }
            } else {
                if (item.attributes.openingHours.length === 1) {  // if one set of hours exist, check for partial 24hrs setting
                    var 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')) {
                        bannButt.mismatch247 = new Flag.Mismatch247();
                    }
                }
                bannButt.noHours = new Flag.NoHours();
                bannButt.noHours.severity = 0;
                bannButt.noHours.WLactive = false;
                bannButt.noHours.message = getHoursHtml('Hours');
            }
            if (!checkHours(item.attributes.openingHours)) {
                bannButt.hoursOverlap = new Flag.HoursOverlap();
                bannButt.noHours = new Flag.NoHours();
            } else {
                var tempHours = item.attributes.openingHours.slice(0);
                for (var 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)) {
                    bannServ.add247.checked = true;
                }
                bannServ.add247.active = true;
            }

            // URL updating
            updateURL = true;
            if (newURL !== item.attributes.url && newURL !== '' && newURL !== '0') {
                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:
                    var newURLTemp = normalizeURL(newURL, true, false, item);  // normalize
                    var itemURL = normalizeURL(item.attributes.url, true, false, item);
                    newURLTemp = newURLTemp.replace(/^www\.(.*)$/i, '$1');  // strip www
                    var 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
                        bannButt.longURL = new Flag.LongURL();
                        if (_wl.longURL) {
                            bannButt.longURL.severity = 0;
                            bannButt.longURL.WLactive = false;
                        }
                        //bannButt.PlaceWebsite.value = 'Place Website';
                        if (hpMode.harmFlag && updateURL && itemURL !== item.attributes.url) {  // Update the URL
                            phlogdev('URL formatted');
                            actions.push(new UpdateObject(item, { url: itemURL }));
                            _updatedFields.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 }));
                    _updatedFields.url.updated = true;
                }
            }

            // Phone formatting
            var 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) {
                    var areaCode = newPhone.match(/[2-9]\d{2}/)[0];
                    if (areaCodeList.indexOf(areaCode) === -1) {
                        bannButt.badAreaCode = new Flag.BadAreaCode(newPhone, outputFormat);
                    }
                }
            }
            if (hpMode.harmFlag && newPhone !== item.attributes.phone) {
                phlogdev('Phone updated');
                actions.push(new UpdateObject(item, { phone: newPhone }));
                _updatedFields.phone.updated = true;
            }

            // Post Office check
            if (_countryCode === 'USA' && newCategories.indexOf('PARKING_LOT') === -1) {
                if (newCategories.indexOf('POST_OFFICE') === -1) {
                    bannButt.isThisAPostOffice = Flag.IsThisAPostOffice.eval(item, newName).flag;
                } else {
                    var re;
                    if (hpMode.harmFlag) {
                        customStoreFinderURL = 'https://tools.usps.com/go/POLocatorAction.action';
                        customStoreFinder = true;
                        bannButt.PlaceWebsite = new Flag.PlaceWebsite();
                        bannButt.NewPlaceSubmit = null;
                        if (item.attributes.url !== 'usps.com') {
                            actions.push(new UpdateObject(item, { url: 'usps.com' }));
                            _updatedFields.url.updated = true;
                            bannButt.urlMissing = null;
                        }
                    }
                    if (state2L === 'KY' || (state2L === 'NY' && addr.city && ['Queens', 'Bronx', 'Manhattan', 'Brooklyn', 'Staten Island'].indexOf(addr.city.attributes.name) > -1)) {
                        re = /^post office \d{5}( [-–](?: cpu| vpo)?(?: [a-z]+){1,})?$/i;
                    } else {
                        re = /^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,}/, ' ');
                    }
                    var nameToCheck = newName + (newNameSuffix || '');
                    if (!re.test(nameToCheck)) {
                        bannButt.formatUSPS = new Flag.FormatUSPS();
                        lockOK = false;
                    } else {
                        if (hpMode.harmFlag) {
                            if (nameToCheck !== item.attributes.name) {
                                actions.push(new UpdateObject(item, { name: nameToCheck }));
                            }
                            bannButt.catPostOffice = new Flag.CatPostOffice();
                        }
                    }
                    if (!newAliases.some(alias => alias.toUpperCase() === 'USPS')) {
                        if (hpMode.harmFlag) {
                            newAliases.push('USPS');
                            actions.push(new UpdateObject(item, { aliases: newAliases }));
                            _updatedFields.aliases.updated = true;
                        } else {
                            bannButt.missingUSPSAlt = new Flag.MissingUSPSAlt();
                        }
                    }
                    if (!newAliases.some(alias => /\d{5}/.test(alias))) {
                        bannButt.missingUSPSZipAlt = new Flag.MissingUSPSZipAlt();
                        if (_wl.missingUSPSZipAlt) {
                            bannButt.missingUSPSZipAlt.severity = 0;
                            bannButt.missingUSPSZipAlt.WLactive = false;
                        }
                        // If the zip code appears in the primary name, pre-fill it in the text entry box.
                        var zipMatch = newName.match(/\d{5}/);
                        if (zipMatch) {
                            bannButt.missingUSPSZipAlt.suggestedValue = zipMatch;
                        }
                        // Note: Started work on a Google api lookup to get the zip, but decided it's probably
                        // not worth it since it would need to be verified by the user anyway.
                        //var coords = item.geometry.getCentroid().transform(W.map.getProjection(), W.map.displayProjection);
                        //var url = 'https://maps.googleapis.com/maps/api/geocode/json?latlng=' + coords.y + ',' + coords.x;
                    }
                    var descr = item.attributes.description;
                    var lines = descr.split('\n');
                    if (lines.length < 1 || !/^.{2,}, [A-Z]{2}\s{1,2}\d{5}$/.test(lines[0])) {
                        bannButt.missingUSPSDescription = new Flag.MissingUSPSDescription();
                        if (_wl.missingUSPSDescription) {
                            bannButt.missingUSPSDescription.severity = 0;
                            bannButt.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) {
            var brand = item.attributes.brand;  // If brand is going to be forced, use that.  Otherwise, use existing brand.
            if (PNHMatchData && PNHMatchData[ph_speccase_ix]) {
                var re = /forceBrand<>([^,<]+)/i;
                var match = re.exec(PNHMatchData[ph_speccase_ix]);
                if (match) {
                    brand = match[1];
                }
            }
            var compressedName = item.attributes.name.toUpperCase().replace(/[^a-zA-Z0-9]/g, '');
            var 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.
            var compressedBrands = [brand.toUpperCase().replace(/[^a-zA-Z0-9]/g, '')];
            if (brand === 'Diamond Gasoline') {
                compressedBrands.push('DIAMONDOIL');
            } else if (brand === 'Murphy USA') {
                compressedBrands.push('MURPHY');
            } else if (brand === 'Mercury Fuel') {
                compressedBrands.push('MERCURY', 'MERCURYPRICECUTTER');
            } else if (brand === 'Carrollfuel') {
                compressedBrands.push('CARROLLMOTORFUEL', 'CARROLLMOTORFUELS');
            }
            if (compressedBrands.every(compressedBrand => compressedName.indexOf(compressedBrand) === -1 && compressedNewName.indexOf(compressedBrand) === -1)) {
                bannButt.gasMismatch = new Flag.GasMismatch();
                if (_wl.gasMismatch) {
                    bannButt.gasMismatch.WLactive = false;
                } else {
                    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;
                    bannButt.plaNameMissing = new Flag.PlaNameMissing();
                }
            } else if (['ISLAND', 'FOREST_GROVE', 'SEA_LAKE_POOL', 'RIVER_STREAM', 'CANAL'].indexOf(item.attributes.categories[0]) === -1) {
                bannButt.nameMissing = new Flag.NameMissing();
                lockOK = false;
            }
        }

        bannButt.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') {
            bannButt.plaIsPublic = new Flag.PlaIsPublic();
            // Add the buttons to the message.
            [
                ['RESTRICTED', 'Restricted'],
                ['PRIVATE', 'Private']
            ].forEach(btnInfo => {
                bannButt.plaIsPublic.message +=
                    $('<button>', { class: 'wmeph-pla-lot-type-btn btn btn-default btn-xs wmeph-btn', 'data-lot-type': btnInfo[0] })
                        .text(btnInfo[1])
                        .prop('outerHTML');
            });
        }

        // House number / HN check
        var currentHN = item.attributes.houseNumber;
        // Check to see if there's an action that is currently updating the house number.
        var 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.
        var 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('|').indexOf(item.attributes.categories[0]) === -1) {
                bannButt.hnMissing = new Flag.HnMissing(item);
                if (state2L === 'PR') {
                    bannButt.hnMissing.severity = 0;
                } else {
                    if (item.isParkingLot()) {
                        bannButt.hnMissing.WLactive = false;
                        if (item.attributes.lockRank < 2) {
                            lockOK = false;
                            var msgAdd;
                            if (_USER.rank < 3) {
                                msgAdd = 'Request an R3+ lock to confirm no HN.';
                            } else {
                                msgAdd = 'Lock to R3+ to confirm no HN.';
                            }
                            bannButt.hnMissing.suffixMessage = msgAdd;
                            bannButt.hnMissing.severity = 1;
                        } else {
                            bannButt.hnMissing.severity = 0;
                        }
                    } else if (_wl.HNWL) {
                        bannButt.hnMissing.severity = 0;
                        bannButt.hnMissing.WLactive = false;
                    } else {
                        lockOK = false;
                    }
                }
            }
        } else if (currentHN) {
            var hnOK = false, updateHNflag = false;
            var hnTemp = currentHN.replace(/[^\d]/g, '');  // Digits only
            var 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) {
                bannButt.hnNonStandard = new Flag.HnNonStandard();
                if (_wl.hnNonStandard) {
                    bannButt.hnNonStandard.WLactive = false;
                    bannButt.hnNonStandard.severity = 0;
                } else {
                    lockOK = false;
                }
            }
            if (updateHNflag) {
                bannButt.hnDashRemoved = new Flag.HnDashRemoved();
                if (hpMode.harmFlag) {
                    actions.push(new UpdateObject(item, { houseNumber: hnTemp }));
                    _updatedFields.address.updated = true;
                } else if (hpMode.hlFlag) {
                    if (item.attributes.residential) {
                        bannButt.hnDashRemoved.severity = 3;
                    } else {
                        bannButt.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('|').indexOf(item.attributes.categories[0]) === -1) {
            bannButt.cityMissing = new Flag.CityMissing();
            if (item.attributes.residential && hpMode.hlFlag) {
                bannButt.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('|').indexOf(item.attributes.categories[0]) === -1) {
            bannButt.streetMissing = new Flag.StreetMissing();
            lockOK = false;
        }

        // CATEGORY vs. NAME checks
        var testName = newName.toLowerCase().replace(/[^a-z]/g, ' ');
        var testNameWords = testName.split(' ');
        // Hopsital vs. Name filter
        if ((newCategories.indexOf('HOSPITAL_URGENT_CARE') > -1 || newCategories.indexOf('HOSPITAL_MEDICAL_CARE') > -1) && hospitalPartMatch.length > 0) {
            var hpmMatch = false;
            if (containsAny(testNameWords, animalFullMatch)) {
                bannButt.changeToPetVet = new Flag.ChangeToPetVet();
                if (_wl.changeToPetVet) {
                    bannButt.changeToPetVet.WLactive = false;
                } else {
                    lockOK = false;
                }
                bannButt.pnhCatMess = null;
            } else if (containsAny(testNameWords, hospitalFullMatch)) {
                bannButt.changeToDoctorClinic = new Flag.ChangeToDoctorClinic();
                bannButt.changeToDoctorClinic.message = 'Keywords suggest this location may not be a hospital or urgent care location.';
                if (_wl.changeToDoctorClinic) {
                    bannButt.changeToDoctorClinic.WLactive = false;
                    bannButt.changeToDoctorClinic.severity = 0;
                } else {
                    bannButt.changeToDoctorClinic.WLactive = true;
                    lockOK = false;
                    bannButt.changeToDoctorClinic.severity = 3;
                }
                bannButt.pnhCatMess = null;
            } else {
                for (var apmix = 0; apmix < animalPartMatch.length; apmix++) {
                    if (testName.indexOf(animalPartMatch[apmix]) > -1) {
                        bannButt.changeToPetVet.active = true;
                        if (_wl.changeToPetVet) {
                            bannButt.changeToPetVet.WLactive = false;
                        } else {
                            lockOK = false;
                        }
                        hpmMatch = true;  // don't run the human check if animal is found.
                        bannButt.pnhCatMess = null;
                        break;
                    }
                }
                if (!hpmMatch) {  // don't run the human check if animal is found.
                    for (var hpmix = 0; hpmix < hospitalPartMatch.length; hpmix++) {
                        if (testName.indexOf(hospitalPartMatch[hpmix]) > -1) {
                            if (_wl.changeToDoctorClinic && bannButt.changeToDoctorClinic) {
                                bannButt.changeToDoctorClinic.WLactive = false;
                            } else {
                                lockOK = false;
                            }
                            hpmMatch = true;
                            bannButt.pnhCatMess = null;
                            break;
                        }
                    }
                }
                if (!hpmMatch && !bannButt.changeToDoctorClinic) {
                    bannButt.changeToDoctorClinic = new Flag.ChangeToDoctorClinic();
                }
            }
        }  // END HOSPITAL/Name check

        // School vs. Name filter
        if (newCategories.indexOf('SCHOOL') > -1 && schoolPartMatch.length > 0) {
            if (containsAny(testNameWords, schoolFullMatch)) {
                bannButt.changeSchool2Offices = new Flag.ChangeSchool2Offices();
                if (_wl.changeSchool2Offices) {
                    bannButt.changeSchool2Offices.WLactive = false;
                } else {
                    lockOK = false;
                }
                bannButt.pnhCatMess = null;
            } else {
                for (var schix = 0; schix < schoolPartMatch.length; schix++) {
                    if (testName.indexOf(schoolPartMatch[schix]) > -1) {
                        bannButt.changeSchool2Offices = new Flag.ChangeSchool2Offices();
                        if (_wl.changeSchool2Offices) {
                            bannButt.changeSchool2Offices.WLactive = false;
                        } else {
                            lockOK = false;
                        }
                        bannButt.pnhCatMess = null;
                        break;
                    }
                }
            }
        }  // END SCHOOL/Name check

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

        // 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.
        if (hpMode.harmFlag && item.attributes.updatedOn < new Date('3/28/2017').getTime()
            && ((newCategories.indexOf('PERSONAL_CARE') > -1 && !PNHNameRegMatch) || newCategories.indexOf('OFFICES') > -1)) {
            bannButt.changeToDoctorClinic = new Flag.ChangeToDoctorClinic();
            bannButt.changeToDoctorClinic.message = 'If this place provides non-emergency medical care: ';
            bannButt.changeToDoctorClinic.severity = 0;
            bannButt.changeToDoctorClinic.WLactive = null;
        }

        // *** Rest Area parsing
        // check rest area name against standard formats or if has the right categories
        var restAreaCatIndex = categories.indexOf('REST_AREAS');
        var oldName = item.attributes.name;
        if (/rest area/i.test(oldName) || /rest stop/i.test(oldName) || /service plaza/i.test(oldName) ||
            (restAreaCatIndex > -1)) {
            if (restAreaCatIndex > -1) {

                if (categories.indexOf('SCENIC_LOOKOUT_VIEWPOINT') > -1) {
                    if (!_wl.restAreaScenic) bannButt.restAreaScenic = new Flag.RestAreaScenic();
                }
                if (categories.indexOf('TRANSPORTATION') > -1) {
                    bannButt.restAreaNoTransportation = new Flag.RestAreaNoTransportation();
                }
                if (item.isPoint()) {  // needs to be area
                    bannButt.areaNotPoint = new Flag.AreaNotPoint();
                }
                bannButt.pointNotArea = null;
                bannButt.unmappedRegion = null;

                if (categories.indexOf('GAS_STATION') > -1) {
                    bannButt.restAreaGas = new Flag.RestAreaGas();
                }

                if (oldName.match(/^Rest Area.* \- /) === null) {
                    bannButt.restAreaName = new Flag.RestAreaName();
                    if (_wl.restAreaName) {
                        bannButt.restAreaName.WLactive = false;
                    }
                } else {
                    if (hpMode.harmFlag) {
                        var newSuffix = newNameSuffix.replace(/Mile/i, 'mile');
                        if (newName + newSuffix !== item.attributes.name) {
                            actions.push(new UpdateObject(item, { name: newName + newSuffix }));
                            _updatedFields.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 (var i = 0; i < actions.length; i++) {
                                var action = actions[i];
                                if (action.newAttributes.name) {
                                    actions.splice(i, 1);
                                    _updatedFields.name.updated = false;
                                    break;
                                }
                            }
                        }
                    }
                }

                // switch to rest area wiki button
                if (hpMode.harmFlag) {
                    bannButt2.restAreaWiki.active = true;
                    bannButt2.placesWiki.active = false;
                }

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

        // update Severity for banner messages
        for (var bannKey in bannButt) {
            if (bannButt[bannKey] && bannButt[bannKey].active) {
                severityButt = Math.max(bannButt[bannKey].severity, severityButt);
            }
        }

        if (hpMode.harmFlag) {
            phlogdev('Severity: ' + severityButt + '; lockOK: ' + lockOK);
        }
        // Place locking
        // final formatting of desired lock levels
        var hlLockFlag = false, levelToLock;
        if (PNHLockLevel !== -1 && hpMode.harmFlag) {
            phlogdev('PNHLockLevel: ' + PNHLockLevel);
            levelToLock = PNHLockLevel;
        } else {
            levelToLock = defaultLockLevel;
        }
        if (region === 'SER') {
            if (newCategories.indexOf('COLLEGE_UNIVERSITY') > -1 && newCategories.indexOf('PARKING_LOT') > -1) {
                levelToLock = lockLevel4;
            } else if (item.isPoint() && newCategories.indexOf('COLLEGE_UNIVERSITY') > -1 && (newCategories.indexOf('HOSPITAL_MEDICAL_CARE') === -1 || newCategories.indexOf('HOSPITAL_URGENT_CARE') === -1)) {
                levelToLock = lockLevel4;
            }
        }

        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 (bannButt.gasNoBrand) {
            if (item.attributes.lockRank >= levelToLock) {
                bannButt.gasNoBrand = null;
            } else {
                bannButt.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 && bannButt.extProviderMissing && bannButt.extProviderMissing.active && bannButt.extProviderMissing.severity <= 2) {
            bannButt.extProviderMissing.severity = 3;
            severityButt = 3;
            if (lockOK) {
                bannButt.extProviderMissing.value = 'Lock anyway? (' + (levelToLock + 1) + ')';
                bannButt.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.';
                bannButt.extProviderMissing.action = function () {
                    var action = new UpdateObject(item, { 'lockRank': levelToLock });
                    W.model.actionManager.add(action);
                    _updatedFields.lock.updated = true;
                    harmonizePlaceGo(item, 'harmonize');
                };
            }
        }

        if (lockOK && severityButt < 2) {
            // Campus project exceptions
            if (item.attributes.lockRank < levelToLock) {
                if (hpMode.harmFlag) {
                    phlogdev('Venue locked!');
                    actions.push(new UpdateObject(item, { lockRank: levelToLock }));
                    _updatedFields.lock.updated = true;
                } else if (hpMode.hlFlag) {
                    hlLockFlag = true;
                }
            }
            bannButt.placeLocked = new Flag.PlaceLocked();
        }

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

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

        if (item.isUnchanged() && !item.attributes.residential && updatedById && (re.test(updatedById.toString()) || (updatedByName && re.test(updatedByName)))) {
            bannButt.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">';
                var ddlSelected = false;
                for (var 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>';
                bannButt.lockRPP = new Flag.LockRPP();
                bannButt.lockRPP.message = 'Current lock: ' + (parseInt(item.attributes.lockRank) + 1) + '. ' + RPPLockString + ' ?';
            }
        }

        // Turn off unnecessary buttons
        if (item.attributes.categories.indexOf('PHARMACY') > -1) {
            bannButt.addPharm.active = false;
        }
        if (item.attributes.categories.indexOf('SUPERMARKET_GROCERY') > -1) {
            bannButt.addSuper.active = false;
        }

        // Final alerts for non-severe locations
        if (!item.attributes.residential && severityButt < 3) {
            var nameShortSpace = newName.toUpperCase().replace(/[^A-Z \']/g, '');
            if (nameShortSpace.indexOf('\'S HOUSE') > -1 || nameShortSpace.indexOf('\'S HOME') > -1 || nameShortSpace.indexOf('\'S WORK') > -1) {
                if (!containsAny(newCategories, ['RESTAURANT', 'DESSERT', 'BAR']) && !PNHNameRegMatch) {
                    bannButt.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'].indexOf(nameShortSpace) > -1) {
                bannButt.resiTypeName = new Flag.ResiTypeName();
                if (_wl.resiTypeName) {
                    bannButt.resiTypeName.WLactive = false;
                }
                bannButt.resiTypeNameSoft = null;
            }
            if (item.attributes.description.toLowerCase().indexOf('google') > -1 || item.attributes.description.toLowerCase().indexOf('yelp') > -1) {
                bannButt.suspectDesc = new Flag.SuspectDesc();
                if (_wl.suspectDesc) {
                    bannButt.suspectDesc.WLactive = false;
                }
            }
        }

        // Return severity for highlighter (no dupe run))
        if (hpMode.hlFlag) {
            // get severities from the banners
            severityButt = 0;
            for (var tempKey in bannButt) {
                if (bannButt[tempKey] && bannButt[tempKey].active) {  //  If the particular message is active
                    if (bannButt[tempKey].hasOwnProperty('WLactive')) {
                        if (bannButt[tempKey].WLactive) {  // If there's a WL option, enable it
                            severityButt = Math.max(bannButt[tempKey].severity, severityButt);
                            //                                if ( bannButt[tempKey].severity > 0) {
                            //                                    phlogdev('Issue with '+item.attributes.name+': '+tempKey);
                            //                                    phlogdev('Severity: '+bannButt[tempKey].severity);
                            //                                }
                        }
                    } else {
                        severityButt = Math.max(bannButt[tempKey].severity, severityButt);
                        //                            if ( bannButt[tempKey].severity > 0) {
                        //                                phlogdev('Issue with '+item.attributes.name+': '+tempKey);
                        //                                phlogdev('Severity: '+bannButt[tempKey].severity);
                        //                            }
                    }
                }

            }

            // Special case flags
            if (item.attributes.lockRank === 0 && (item.attributes.categories.indexOf('HOSPITAL_MEDICAL_CARE') > -1 || item.attributes.categories.indexOf('HOSPITAL_URGENT_CARE') > -1 || 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 = [];
        bannDupl = {};
        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]) {
                bannButt.overlapping = new Flag.Overlapping();
            }
            duplicateName = duplicateName[0];
            if (duplicateName.length > 0) {
                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 + devVersStr + '\nPermalink: ' + placePL + '\nPlace name: ' + item.attributes.name + '\nCountry: ' + addr.country.name + '\n--------\nDescribe the error:\nDupeID mismatch with dupeName list'
                        });
                    }
                } else {
                    var wlAction = function (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);
                        saveWL_LS(true);  // Save the WL to local storage
                        WMEPH_WLCounter();
                        bannButt2.clearWL.active = true;
                        bannDupl[dID].active = false;
                        harmonizePlaceGo(item, 'harmonize');
                    };
                    for (var ijx = 1; ijx < duplicateName.length + 1; ijx++) {
                        bannDupl[dupeIDList[ijx]] = {
                            active: true, severity: 2, message: duplicateName[ijx - 1],
                            WLactive: false, WLvalue: wlButtText, WLtitle: 'Whitelist Duplicate',
                            WLaction: wlAction
                        };
                        if (venueWhitelist.hasOwnProperty(itemID) && venueWhitelist[itemID].hasOwnProperty('dupeWL') && venueWhitelist[itemID].dupeWL.indexOf(dupeIDList[ijx]) > -1) {  // if the dupe is on the whitelist then remove it from the banner
                            bannDupl[dupeIDList[ijx]].active = false;
                        } else {  // Otherwise, activate the WL button
                            bannDupl[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) {
            var dhnix, 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;
            var arrayHNRatio = [];
            for (dhnix = 0; dhnix < dupeHNRangeListSorted.length; dhnix++) {
                arrayHNRatio.push(Math.abs((parseInt(item.attributes.houseNumber) - dupeHNRangeListSorted[dhnix]) / dupeHNRangeDistList[dhnix]));
            }
            sortWithIndex(arrayHNRatio);
            // Examine either the median or the 8th index if length is >16
            var arrayHNRatioCheckIX = Math.min(Math.round(arrayHNRatio.length / 2), 8);
            if (arrayHNRatio[arrayHNRatioCheckIX] > 1.4) {
                bannButt.HNRange = new Flag.HNRange();
                if (_wl.HNRange) {
                    bannButt.HNRange.WLactive = false;
                    bannButt.HNRange.active = false;
                }
                if (arrayHNRatio[arrayHNRatioCheckIX] > 5) {
                    bannButt.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(bannServ);
        }

        if (bannButt.lockRPP) bannButt.lockRPP.message = 'Current lock: ' + (parseInt(item.attributes.lockRank) + 1) + '. ' + RPPLockString + ' ?';

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

        showOpenPlaceWebsiteButton();
        showSearchButton();
    }  // END harmonizePlaceGo function

    // Set up banner messages
    function assembleBanner() {
        let venue = getSelectedVenue();
        if (!venue) return;
        phlogdev('Building banners');
        var dupesFound = 0;
        var rowData;
        var $rowDiv;
        var rowDivs = [];
        severityButt = 0;

        let func = elem => { return { 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(bannDupl).forEach(tempKey => {
            rowData = bannDupl[tempKey];
            if (rowData.active) {
                dupesFound += 1;
                let $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(bannButt).forEach(tempKey => {
            rowData = bannButt[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();
        }
        var 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);
        //$('#select2-drop').css({display:'none'});

        assembleServicesBanner();

        //  Build general banners (below the Services)
        rowDivs = [];
        Object.keys(bannButt2).forEach(tempKey => {
            var rowData = bannButt2[tempKey];
            if (rowData.active) {  //  If the particular message is active
                $rowDiv = $('<div>');
                $rowDiv.append(rowData.message);
                if (rowData.action) {
                    $rowDiv.append(' <input class="btn btn-info btn-xs wmeph-btn" id="WMEPH_' + tempKey + '" title="' + rowData.title + '" style="" type="button" value="' + rowData.value + '">');
                }
                rowDivs.push($rowDiv);
                severityButt = Math.max(bannButt2[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);
        //$('#select2-drop').css({display:'none'});


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

        // Setup bannButt2 onclicks
        setupButtons(bannButt2);

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

        // Add click handlers for parking lot helper buttons.
        $('.wmeph-pla-spaces-btn').click(function () {
            let venue = getSelectedVenue();
            var selectedValue = $(this).attr('id').replace('wmeph_', '');
            var existingAttr = venue.attributes.categoryAttributes.PARKING_LOT;
            var newAttr = {};
            if (existingAttr) {
                for (var prop in existingAttr) {
                    var value = existingAttr[prop];
                    if (Array.isArray(value)) value = [].concat(value);
                    newAttr[prop] = value;
                }
            }
            newAttr.estimatedNumberOfSpots = selectedValue;
            W.model.actionManager.add(new UpdateObject(venue, { 'categoryAttributes': { PARKING_LOT: newAttr } }));
            harmonizePlaceGo(venue, 'harmonize');
        });
        $('.wmeph-pla-lot-type-btn').click(function () {
            let venue = getSelectedVenue();
            var selectedValue = $(this).data('lot-type');
            var existingAttr = venue.attributes.categoryAttributes.PARKING_LOT;
            var newAttr = {};
            if (existingAttr) {
                for (var prop in existingAttr) {
                    var value = existingAttr[prop];
                    if (Array.isArray(value)) value = [].concat(value);
                    newAttr[prop] = value;
                }
            }
            newAttr.parkingType = selectedValue;
            W.model.actionManager.add(new UpdateObject(venue, { 'categoryAttributes': { PARKING_LOT: newAttr } }));
            harmonizePlaceGo(venue, 'harmonize');
        });

        $('.wmeph-pla-cost-type-btn').click(function () {
            let venue = getSelectedVenue();
            var selectedValue = $(this).attr('id').replace('wmeph_', '');
            var existingAttr = venue.attributes.categoryAttributes.PARKING_LOT;
            var newAttr = {};
            if (existingAttr) {
                for (var prop in existingAttr) {
                    var value = existingAttr[prop];
                    if (Array.isArray(value)) value = [].concat(value);
                    newAttr[prop] = value;
                }
            }
            newAttr.costType = selectedValue;
            W.model.actionManager.add(new UpdateObject(venue, { 'categoryAttributes': { PARKING_LOT: newAttr } }));
            harmonizePlaceGo(venue, 'harmonize');
        });

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

        // If pressing enter in the phone entry box, add the phone
        $('#WMEPH-PhoneAdd').keyup(function (event) {
            if (event.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(function (event) {
            if (event.keyCode === 13 && $('#WMEPH-UrlAdd').val() !== '') {
                $('#WMEPH_urlMissing').click();
            }
        });

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

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

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

            }, 100);
        }
        $('#WMEPH-HoursPaste')
            .bind('paste', resetHoursEntryHeight)
            .bind('drop', resetHoursEntryHeight)
            .keydown(resetHoursEntryHeight)
            .bind('dragenter', function () {
                var text = $('#WMEPH-HoursPaste').val();
                if (text === _DEFAULT_HOURS_TEXT) {
                    $('#WMEPH-HoursPaste').val('');
                }
            });

        // If pressing enter in the hours entry box, parse the entry
        $('#WMEPH-HoursPaste').keydown(function (event) {
            if (event.keyCode === 13) {
                if (event.ctrlKey) {
                    // Simulate a newline event (shift + enter)
                    var text = this.value;
                    var selStart = this.selectionStart;
                    this.value = text.substr(0, selStart) + '\n' + text.substr(this.selectionEnd, text.length - 1);
                    this.selectionStart = selStart + 1;
                    this.selectionEnd = selStart + 1;
                    return true;
                } else if (!(event.shiftKey || event.ctrlKey) && $('#WMEPH-HoursPaste').val() !== '') {
                    event.stopPropagation();
                    event.preventDefault();
                    event.returnValue = false;
                    event.cancelBubble = true;
                    $('#WMEPH_noHours').click();
                    return false;
                }
            }
        });
        $('#WMEPH-HoursPaste').focus(function () {
            if (this.value === _DEFAULT_HOURS_TEXT) {
                this.value = '';
            }
            this.style.color = 'black';
        }).blur(function () {
            if (this.value === '') {
                this.value = _DEFAULT_HOURS_TEXT;
                this.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 (bannButt.noHours) {
            $('#WMEPH_noHours').click(() => bannButt.noHours.addHoursAction());
            $('#WMEPH_noHours_2').click(() => bannButt.noHours.replaceHoursAction());
        }

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

    function assembleServicesBanner() {
        let venue = getSelectedVenue();
        if (venue && !$('#WMEPH-HideServicesButtons').prop('checked')) {
            // setup Add Service Buttons for suggested services
            var rowDivs = [];
            if (!venue.isResidential()) {
                var $rowDiv = $('<div>');
                var servButtHeight = '27';
                var buttons = [];
                Object.keys(bannServ).forEach(tempKey => {
                    var rowData = bannServ[tempKey];
                    if (rowData.active) {  //  If the particular service is active
                        var $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(bannServ);
            }
        }
    }

    // Button onclick event handler
    function setupButtons(b) {
        for (var tempKey in b) {  // 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);
                }
            }
        }
    }  // END setupButtons function

    function buttonAction(b, bKey) {
        var button = document.getElementById('WMEPH_' + bKey);
        button.onclick = function () {
            b[bKey].action();
            if (!b[bKey].noBannerAssemble) assembleBanner();
        };
        return button;
    }
    function buttonAction2(b, bKey) {
        var button = document.getElementById('WMEPH_' + bKey + '_2');
        button.onclick = function () {
            b[bKey].action2();
            if (!b[bKey].noBannerAssemble) assembleBanner();
        };
        return button;
    }
    function buttonWhitelist(b, bKey) {
        var button = document.getElementById('WMEPH_WL' + bKey);
        button.onclick = function () {
            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() {
        let betaDelay = 100;
        setTimeout(() => {
            let venue = getSelectedVenue();
            if ($('#WMEPH_runButton').length === 0) {
                $('<div id="WMEPH_runButton">').prependTo('.contents');
            }
            if ($('#runWMEPH').length === 0) {
                let devVersSuffix = _IS_DEV_VERSION ? '-β' : '';
                let 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);
            }
            let btn = document.getElementById('runWMEPH');
            if (btn !== null) {
                btn.onclick = function () {
                    harmonizePlace();
                };
            } else {
                setTimeout(bootstrapRunButton, 100);
            }
            showOpenPlaceWebsiteButton();
            showSearchButton();
        }, betaDelay);
    }  // END displayRunButton funtion

    // Displays the Open Place Website button.
    function showOpenPlaceWebsiteButton() {
        let venue = getSelectedVenue();
        if (venue) {
            var openPlaceWebsiteURL = venue.attributes.url;
            if (openPlaceWebsiteURL && openPlaceWebsiteURL.replace(/[^A-Za-z0-9]/g, '').length > 2) {
                if ($('#WMEPHurl').length === 0) {
                    let strButt1 = '<input class="btn btn-success btn-xs wmeph-fat-btn" id="WMEPHurl" title="Open place URL" type="button" value="Website">';
                    $('#runWMEPH').after(strButt1);
                    let btn = document.getElementById('WMEPHurl');
                    if (btn !== null) {
                        btn.onclick = function () {
                            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, searchResultsWindowName, searchResultsWindowSpecs);
                            }
                        };
                    } else {
                        setTimeout(bootstrapRunButton, 100);
                    }
                }
            } else {
                if ($('#WMEPHurl').length > 0) {
                    $('#WMEPHurl').remove();
                }
            }
        }
    }

    function showSearchButton() {
        let venue = getSelectedVenue();
        if (venue && $('#wmephSearch').length === 0) {
            let 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);
            let btn = document.getElementById('wmephSearch');
            if (btn !== null) {
                btn.onclick = function () {
                    let addr = venue.getAddress();
                    if (addr.hasState()) {
                        let url = buildGLink(venue.attributes.name, addr.attributes, venue.attributes.houseNumber);
                        if ($('#WMEPH-WebSearchNewTab').prop('checked')) {
                            window.open(url);
                        } else {
                            window.open(url, searchResultsWindowName, searchResultsWindowSpecs);
                        }
                    } else {
                        alert('The state and country haven\'t been set for this place yet.  Edit the address first.');
                    }
                };
            } else {
                setTimeout(bootstrapRunButton, 100);
            }
        }
    }

    // WMEPH Clone Tool
    function displayCloneButton() {
        var betaDelay = 80;
        if (_IS_DEV_VERSION) { betaDelay = 300; }
        setTimeout(() => {
            if ($('#WMEPH_runButton').length === 0) {
                $('<div id="WMEPH_runButton">').prependTo('.contents');
            }
            let strButt1, btn;
            let venue = getSelectedVenue();
            if (venue) {
                showOpenPlaceWebsiteButton();
                if ($('#clonePlace').length === 0) {
                    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);
                }
                btn = document.getElementById('clonePlace');
                if (btn !== null) {
                    btn.onclick = function () {
                        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 = function () {
                        clonePlace(getSelectedVenue());
                    };
                } else {
                    setTimeout(bootstrapRunButton, 100);
                }
                btn = document.getElementById('checkAllClone');
                if (btn !== null) {
                    btn.onclick = function () {
                        if (!$('#WMEPH_CPhn').prop('checked')) { $('#WMEPH_CPhn').trigger('click'); }
                        if (!$('#WMEPH_CPstr').prop('checked')) { $('#WMEPH_CPstr').trigger('click'); }
                        if (!$('#WMEPH_CPcity').prop('checked')) { $('#WMEPH_CPcity').trigger('click'); }
                        if (!$('#WMEPH_CPurl').prop('checked')) { $('#WMEPH_CPurl').trigger('click'); }
                        if (!$('#WMEPH_CPph').prop('checked')) { $('#WMEPH_CPph').trigger('click'); }
                        if (!$('#WMEPH_CPserv').prop('checked')) { $('#WMEPH_CPserv').trigger('click'); }
                        if (!$('#WMEPH_CPdesc').prop('checked')) { $('#WMEPH_CPdesc').trigger('click'); }
                        if (!$('#WMEPH_CPhrs').prop('checked')) { $('#WMEPH_CPhrs').trigger('click'); }
                    };
                } else {
                    setTimeout(bootstrapRunButton, 100);
                }
                btn = document.getElementById('checkAddrClone');
                if (btn !== null) {
                    btn.onclick = function () {
                        if (!$('#WMEPH_CPhn').prop('checked')) { $('#WMEPH_CPhn').trigger('click'); }
                        if (!$('#WMEPH_CPstr').prop('checked')) { $('#WMEPH_CPstr').trigger('click'); }
                        if (!$('#WMEPH_CPcity').prop('checked')) { $('#WMEPH_CPcity').trigger('click'); }
                        if ($('#WMEPH_CPurl').prop('checked')) { $('#WMEPH_CPurl').trigger('click'); }
                        if ($('#WMEPH_CPph').prop('checked')) { $('#WMEPH_CPph').trigger('click'); }
                        if ($('#WMEPH_CPserv').prop('checked')) { $('#WMEPH_CPserv').trigger('click'); }
                        if ($('#WMEPH_CPdesc').prop('checked')) { $('#WMEPH_CPdesc').trigger('click'); }
                        if ($('#WMEPH_CPhrs').prop('checked')) { $('#WMEPH_CPhrs').trigger('click'); }
                    };
                } else {
                    setTimeout(bootstrapRunButton, 100);
                }
                btn = document.getElementById('checkNoneClone');
                if (btn !== null) {
                    btn.onclick = function () {
                        if ($('#WMEPH_CPhn').prop('checked')) { $('#WMEPH_CPhn').trigger('click'); }
                        if ($('#WMEPH_CPstr').prop('checked')) { $('#WMEPH_CPstr').trigger('click'); }
                        if ($('#WMEPH_CPcity').prop('checked')) { $('#WMEPH_CPcity').trigger('click'); }
                        if ($('#WMEPH_CPurl').prop('checked')) { $('#WMEPH_CPurl').trigger('click'); }
                        if ($('#WMEPH_CPph').prop('checked')) { $('#WMEPH_CPph').trigger('click'); }
                        if ($('#WMEPH_CPserv').prop('checked')) { $('#WMEPH_CPserv').trigger('click'); }
                        if ($('#WMEPH_CPdesc').prop('checked')) { $('#WMEPH_CPdesc').trigger('click'); }
                        if ($('#WMEPH_CPhrs').prop('checked')) { $('#WMEPH_CPhrs').trigger('click'); }
                    };
                } 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) {
                let 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() {
        var panelFieldsList = $('.form-control'), pfa;
        for (var pfix = 0; pfix < panelFieldsList.length; pfix++) {
            pfa = panelFieldsList[pfix].name;
            if (pfa === 'name') {
                panelFields.name = pfix;
            }
            if (pfa === 'lockRank') {
                panelFields.lockRank = pfix;
            }
            if (pfa === 'description') {
                panelFields.description = pfix;
            }
            if (pfa === 'url') {
                panelFields.url = pfix;
            }
            if (pfa === 'phone') {
                panelFields.phone = pfix;
            }
            if (pfa === 'brand') {
                panelFields.brand = pfix;
            }
        }
        var placeNavTabs = $('.nav');
        for (pfix = 0; pfix < placeNavTabs.length; pfix++) {
            pfa = placeNavTabs[pfix].innerHTML;
            if (pfa.indexOf('landmark-edit') > -1) {
                panelFieldsList = placeNavTabs[pfix].children;
                panelFields.navTabsIX = pfix;
                break;
            }
        }
        for (pfix = 0; pfix < panelFieldsList.length; pfix++) {
            pfa = panelFieldsList[pfix].innerHTML;
            if (pfa.indexOf('landmark-edit-general') > -1) {
                panelFields.navTabGeneral = pfix;
            }
            if (pfa.indexOf('landmark-edit-more') > -1) {
                panelFields.navTabMore = pfix;
            }
        }
    }

    // Function to clone info from a place
    function clonePlace() {
        phlog('Cloning info...');
        if (cloneMaster !== null && cloneMaster.hasOwnProperty('url')) {
            let venue = getSelectedVenue();
            var cloneItems = {};
            var updateItem = false;
            if ($('#WMEPH_CPhn').prop('checked')) {
                cloneItems.houseNumber = cloneMaster.houseNumber;
                updateItem = true;
            }
            if ($('#WMEPH_CPurl').prop('checked')) {
                cloneItems.url = cloneMaster.url;
                updateItem = true;
            }
            if ($('#WMEPH_CPph').prop('checked')) {
                cloneItems.phone = cloneMaster.phone;
                updateItem = true;
            }
            if ($('#WMEPH_CPdesc').prop('checked')) {
                cloneItems.description = cloneMaster.description;
                updateItem = true;
            }
            if ($('#WMEPH_CPserv').prop('checked') && venue.isParkingLot() === cloneMaster.isPLA) {
                cloneItems.services = cloneMaster.services;
                updateItem = true;
            }
            if ($('#WMEPH_CPhrs').prop('checked')) {
                cloneItems.openingHours = cloneMaster.openingHours;
                updateItem = true;
            }
            if (updateItem) {
                W.model.actionManager.add(new UpdateObject(venue, cloneItems));
                phlogdev('Item details cloned');
            }

            var copyStreet = $('#WMEPH_CPstr').prop('checked');
            var copyCity = $('#WMEPH_CPcity').prop('checked');

            if (copyStreet || copyCity) {
                var originalAddress = venue.getAddress();
                var 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) {
        var dayNames = ['Sunday', 'Monday', 'Tuesday', 'Wednesday', 'Thursday', 'Friday', 'Saturday'];
        var 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;
        }
        var daysObj, fromHourTemp, toHourTemp;
        for (var day2Ch = 0; day2Ch < 7; day2Ch++) {  // Go thru each day of the week
            daysObj = [];
            for (var hourSet = 0; hourSet < hoursObj.length; hourSet++) {  // For each set of hours
                if (hoursObj[hourSet].days.indexOf(day2Ch) > -1) {  // pull out hours that are for the current day, add 2400 if it goes past midnight, and store
                    fromHourTemp = hoursObj[hourSet].fromHour.replace(/\:/g, '');
                    toHourTemp = hoursObj[hourSet].toHour.replace(/\:/g, '');
                    if (toHourTemp <= fromHourTemp) {
                        toHourTemp = parseInt(toHourTemp) + 2400;
                    }
                    daysObj.push([fromHourTemp, toHourTemp]);
                }
            }
            if (daysObj.length > 1) {  // If there's multiple hours for the day, check them for overlap
                for (var hourSetCheck2 = 1; hourSetCheck2 < daysObj.length; hourSetCheck2++) {
                    for (var 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(itemName, itemAliases, item, recenterOption) {
        dupeIDList = [item.attributes.id];
        dupeHNRangeList = [];
        dupeHNRangeIDList = [];
        dupeHNRangeDistList = [];
        var venueList = W.model.venues.objects, currNameList = [], testNameList = [], testVenueAtt, testName, testNameNoNum, itemNameRF, aliasNameRF, aliasNameNoNum;
        var wlDupeMatch = false, wlDupeList = [], nameMatch = false, altNameMatch = -1, aliix, cnlix, tnlix, randInt = 100;
        var outOfExtent = false, mapExtent = W.map.getExtent(), padFrac = 0.15;  // how much to pad the zoomed window
        // Initialize the cooridnate extents for duplicates
        var minLon = item.geometry.getCentroid().x, minLat = item.geometry.getCentroid().y;
        var maxLon = minLon, maxLat = minLat;
        // genericterms to skip if it's all that remains after stripping numbers
        var noNumSkip = 'BANK|ATM|HOTEL|MOTEL|STORE|MARKET|SUPERMARKET|GYM|GAS|GASOLINE|GASSTATION|CAFE|OFFICE|OFFICES|CARRENTAL|RENTALCAR|RENTAL|SALON|BAR|BUILDING|LOT';
        noNumSkip = noNumSkip + '|' + collegeAbbreviations;
        noNumSkip = noNumSkip.split('|');
        // Make the padded extent
        mapExtent.left = mapExtent.left + padFrac * (mapExtent.right - mapExtent.left);
        mapExtent.right = mapExtent.right - padFrac * (mapExtent.right - mapExtent.left);
        mapExtent.bottom = mapExtent.bottom + padFrac * (mapExtent.top - mapExtent.bottom);
        mapExtent.top = mapExtent.top - padFrac * (mapExtent.top - mapExtent.bottom);

        var allowedTwoLetters = ['BP', 'DQ', 'BK', 'BW', 'LQ', 'QT', 'DB', 'PO'];

        var labelFeatures = [], dupeNames = [], labelText, labelTextReformat, pt, textFeature, labelColorIX = 0;
        var labelColorList = ['#3F3'];
        // Name formatting for the WME place name
        itemNameRF = itemName.toUpperCase().replace(/ AND /g, '').replace(/^THE /g, '').replace(/[^A-Z0-9]/g, '');  // Format name
        if (itemNameRF.length > 2 || allowedTwoLetters.indexOf(itemNameRF) > -1) {
            currNameList.push(itemNameRF);
        } else {
            currNameList.push('PRIMNAMETOOSHORT_PJZWX');
        }
        var itemNameNoNum = itemNameRF.replace(/[^A-Z]/g, '');  // Clear non-letter characters for alternate match ( HOLLYIVYPUB23 --> HOLLYIVYPUB )
        if (((itemNameNoNum.length > 2 && noNumSkip.indexOf(itemNameNoNum) === -1) || allowedTwoLetters.indexOf(itemNameNoNum) > -1) && item.attributes.categories.indexOf('PARKING_LOT') === -1) {  //  only add de-numbered name if anything remains
            currNameList.push(itemNameNoNum);
        }
        if (itemAliases.length > 0) {
            for (aliix = 0; aliix < itemAliases.length; aliix++) {
                aliasNameRF = itemAliases[aliix].toUpperCase().replace(/ AND /g, '').replace(/^THE /g, '').replace(/[^A-Z0-9]/g, '');  // Format name
                if ((aliasNameRF.length > 2 && noNumSkip.indexOf(aliasNameRF) === -1) || allowedTwoLetters.indexOf(aliasNameRF) > -1) {  //  only add de-numbered name if anything remains
                    currNameList.push(aliasNameRF);
                }
                aliasNameNoNum = aliasNameRF.replace(/[^A-Z]/g, '');  // Clear non-letter characters for alternate match ( HOLLYIVYPUB23 --> HOLLYIVYPUB )
                if (((aliasNameNoNum.length > 2 && noNumSkip.indexOf(aliasNameNoNum) === -1) || allowedTwoLetters.indexOf(aliasNameNoNum) > -1) && item.attributes.categories.indexOf('PARKING_LOT') === -1) {  //  only add de-numbered name if anything remains
                    currNameList.push(aliasNameNoNum);
                }
            }
        }
        currNameList = uniq(currNameList);  //  remove duplicates

        // Remove any previous search labels and move the layer above the places layer
        _dupeLayer.destroyFeatures();
        //var vecLyrPlaces = W.map.getLayersBy('uniqueName','landmarks')[0];
        //W.map.setLayerZIndex(_dupeLayer, 70);
        //_dupeLayer.setZIndex(parseInt(vecLyrPlaces.getZIndex())+3);  // Move layer to just on top of Places layer

        if (venueWhitelist.hasOwnProperty(item.attributes.id)) {
            if (venueWhitelist[item.attributes.id].hasOwnProperty('dupeWL')) {
                wlDupeList = venueWhitelist[item.attributes.id].dupeWL;
            }
        }

        var overlappingFlag = false;
        var addrItem = item.getAddress(), itemCompAddr = false;
        if (addrItem.hasOwnProperty('attributes')) {
            addrItem = addrItem.attributes;
        }
        if (addrItem.street !== null && addrItem.street.name !== null && item.attributes.houseNumber && item.attributes.houseNumber.match(/\d/g) !== null) {
            itemCompAddr = true;
        }

        for (var venix in venueList) {  // for each place on the map:
            if (venueList.hasOwnProperty(venix)) {  // hOP filter
                nameMatch = false;
                altNameMatch = -1;
                testVenueAtt = venueList[venix].attributes;
                var excludePLADupes = $('#WMEPH-ExcludePLADupes').prop('checked');
                if ((!excludePLADupes || (excludePLADupes && !(item.isParkingLot() || venueList[venix].isParkingLot()))) && !isEmergencyRoom(venueList[venix])) {

                    var pt2ptDistance = item.geometry.getCentroid().distanceTo(venueList[venix].geometry.getCentroid());
                    if (item.isPoint() && venueList[venix].isPoint() && pt2ptDistance < 2 && item.attributes.id !== testVenueAtt.id) {
                        overlappingFlag = true;
                    }
                    wlDupeMatch = false;
                    if (wlDupeList.length > 0 && wlDupeList.indexOf(testVenueAtt.id) > -1) {
                        wlDupeMatch = true;
                    }

                    // get HNs for places on same street
                    var addrDupe = venueList[venix].getAddress();
                    if (addrDupe.hasOwnProperty('attributes')) {
                        addrDupe = addrDupe.attributes;
                    }
                    if (itemCompAddr && addrDupe.street !== null && addrDupe.street.name !== null && testVenueAtt.houseNumber && testVenueAtt.houseNumber !== '' &&
                        venix !== item.attributes.id && addrItem.street.name === addrDupe.street.name && testVenueAtt.houseNumber < 1000000) {
                        dupeHNRangeList.push(parseInt(testVenueAtt.houseNumber));
                        dupeHNRangeIDList.push(testVenueAtt.id);
                        dupeHNRangeDistList.push(pt2ptDistance);
                    }


                    // Check for duplicates
                    if (!wlDupeMatch && dupeIDList.length < 6 && pt2ptDistance < 800 && !testVenueAtt.residential && venix !== item.attributes.id && 'string' === typeof testVenueAtt.id && testVenueAtt.name !== null && testVenueAtt.name.length > 1) {  // don't do res, the point itself, new points or no name
                        // If item has a complete address and test venue does, and they are different, then no dupe
                        var suppressMatch = false;
                        if (itemCompAddr && addrDupe.street !== null && addrDupe.street.name !== null && testVenueAtt.houseNumber && testVenueAtt.houseNumber.match(/\d/g) !== null) {
                            if (item.attributes.lockRank > 0 && testVenueAtt.lockRank > 0) {
                                if (item.attributes.houseNumber !== testVenueAtt.houseNumber || addrItem.street.name !== addrDupe.street.name) {
                                    suppressMatch = true;
                                }
                            } else {
                                if (item.attributes.houseNumber !== testVenueAtt.houseNumber && addrItem.street.name !== addrDupe.street.name) {
                                    suppressMatch = true;
                                }
                            }
                        }


                        if (!suppressMatch) {
                            //Reformat the testPlace name
                            testName = testVenueAtt.name.toUpperCase().replace(/\s+[-\(].*$/, '').replace(/ AND /g, '').replace(/^THE /g, '').replace(/[^A-Z0-9]/g, '');  // Format test name
                            if ((testName.length > 2 && noNumSkip.indexOf(testName) === -1) || allowedTwoLetters.indexOf(testName) > -1) {
                                testNameList = [testName];
                            } else {
                                testNameList = ['TESTNAMETOOSHORTQZJXS' + randInt];
                                randInt++;
                            }

                            testNameNoNum = testName.replace(/[^A-Z]/g, '');  // Clear non-letter characters for alternate match
                            if (((testNameNoNum.length > 2 && noNumSkip.indexOf(testNameNoNum) === -1) || allowedTwoLetters.indexOf(testNameNoNum) > -1) && testVenueAtt.categories.indexOf('PARKING_LOT') === -1) {  //  only add de-numbered name if at least 2 chars remain
                                testNameList.push(testNameNoNum);
                            }
                            // primary name matching loop

                            for (tnlix = 0; tnlix < testNameList.length; tnlix++) {
                                for (cnlix = 0; cnlix < currNameList.length; cnlix++) {
                                    if ((testNameList[tnlix].indexOf(currNameList[cnlix]) > -1 || currNameList[cnlix].indexOf(testNameList[tnlix]) > -1)) {
                                        nameMatch = true;
                                        break;
                                    }
                                }
                                if (nameMatch) { break; }  // break if a match found
                            }
                            if (!nameMatch && testVenueAtt.aliases.length > 0) {
                                for (aliix = 0; aliix < testVenueAtt.aliases.length; aliix++) {
                                    aliasNameRF = testVenueAtt.aliases[aliix].toUpperCase().replace(/ AND /g, '').replace(/^THE /g, '').replace(/[^A-Z0-9]/g, '');  // Format name
                                    if ((aliasNameRF.length > 2 && noNumSkip.indexOf(aliasNameRF) === -1) || allowedTwoLetters.indexOf(aliasNameRF) > -1) {
                                        testNameList = [aliasNameRF];
                                    } else {
                                        testNameList = ['ALIASNAMETOOSHORTQOFUH' + randInt];
                                        randInt++;
                                    }
                                    aliasNameNoNum = aliasNameRF.replace(/[^A-Z]/g, '');  // Clear non-letter characters for alternate match ( HOLLYIVYPUB23 --> HOLLYIVYPUB )
                                    if (((aliasNameNoNum.length > 2 && noNumSkip.indexOf(aliasNameNoNum) === -1) || allowedTwoLetters.indexOf(aliasNameNoNum) > -1) && testVenueAtt.categories.indexOf('PARKING_LOT') === -1) {  //  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 (tnlix = 0; tnlix < testNameList.length; tnlix++) {
                                    for (cnlix = 0; cnlix < currNameList.length; cnlix++) {
                                        if ((testNameList[tnlix].indexOf(currNameList[cnlix]) > -1 || currNameList[cnlix].indexOf(testNameList[tnlix]) > -1)) {
                                            // 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(testVenueAtt.id);  // Add the item to the list of matches
                                _dupeLayer.setVisibility(true);  // If anything found, make visible the dupe layer
                                if (nameMatch) {
                                    labelText = testVenueAtt.name;  // Pull duplicate name
                                } else {
                                    labelText = testVenueAtt.aliases[altNameMatch] + ' (Alt)';  // Pull duplicate alt name
                                }
                                phlogdev('Possible duplicate found. WME place: ' + itemName + ' / Nearby place: ' + labelText);

                                // Reformat the name into multiple lines based on length
                                var startIX = 0, endIX = 0, labelTextBuild = [], maxLettersPerLine = Math.round(2 * Math.sqrt(labelText.replace(/ /g, '').length / 2));
                                maxLettersPerLine = Math.max(maxLettersPerLine, 4);
                                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
                                labelTextReformat = labelTextBuild.join('\n');
                                // Add photo icons
                                if (testVenueAtt.images.length > 0) {
                                    labelTextReformat = labelTextReformat + ' ';
                                    for (var phix = 0; phix < testVenueAtt.images.length; phix++) {
                                        if (phix === 3) {
                                            labelTextReformat = labelTextReformat + '+';
                                            break;
                                        }
                                        //labelTextReformat = labelTextReformat + '\u25A3';  // add photo icons
                                        labelTextReformat = labelTextReformat + '\u25A3';  // add photo icons
                                    }
                                }

                                const lonLat = getVenueLonLat(venueList[venix])
                                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);

                                textFeature = new OL.Feature.Vector(
                                    venueList[venix].geometry.getCentroid(),
                                    {
                                        labelText: labelTextReformat,
                                        fontColor: '#fff',
                                        strokeColor: labelColorList[labelColorIX % labelColorList.length],
                                        labelAlign: 'cm',
                                        pointRadius: 25,
                                        dupeID: testVenueAtt.id
                                    }
                                );
                                labelFeatures.push(textFeature);
                                //_dupeLayer.addFeatures(labelFeatures);
                                dupeNames.push(labelText);
                            }
                            labelColorIX++;
                        }
                    }
                }
            }
        }
        // Add a marker for the working place point if any dupes were found
        if (dupeIDList.length > 1) {
            const lonLat = getVenueLonLat(item);
            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
            var currentLabel = 'Current';
            if (item.attributes.images.length > 0) {
                for (var ciix = 0; ciix < item.attributes.images.length; ciix++) {
                    currentLabel = currentLabel + ' ';
                    if (ciix === 3) {
                        currentLabel = currentLabel + '+';
                        break;
                    }
                    currentLabel = currentLabel + '\u25A3';  // add photo icons
                }
            }
            textFeature = new OL.Feature.Vector(
                item.geometry.getCentroid(),
                {
                    labelText: currentLabel,
                    fontColor: '#fff',
                    strokeColor: '#fff',
                    labelAlign: 'cm',
                    pointRadius: 25,
                    dupeID: item.attributes.id
                }
            );
            labelFeatures.push(textFeature);
            _dupeLayer.addFeatures(labelFeatures);
        }

        if (recenterOption && dupeNames.length > 0 && outOfExtent) {  // then rebuild the extent to include the duplicate
            var 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.zoomToExtent(mapExtent);
        }
        return [dupeNames, overlappingFlag];
    }  // END Dupefinder function

    // On selection of new item:
    function checkSelection() {
        let venue = getSelectedVenue();
        if (venue && venue.isApproved()) {
            displayRunButton();
            getPanelFields();
            if ($('#WMEPH-EnableCloneMode').prop('checked')) {
                displayCloneButton();
            }
            if ((localStorage.getItem('WMEPH-AutoRunOnSelect') === '1') && venue.arePropertiesEditable()) {
                setTimeout(harmonizePlace, 200);
            }
            for (var 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;
                }
            }
        }
        // If the selection is anything else, clear the labels
        _dupeLayer.destroyFeatures();
        _dupeLayer.setVisibility(false);

    }  // END checkSelection function

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

        let venue = getSelectedVenue();

        var findClosestNode = function () {
            var closestSegment = orderedSegments[0].segment,
                distanceA,
                distanceB,
                nodeA = W.model.nodes.getObjectById(closestSegment.attributes.fromNodeID),
                nodeB = W.model.nodes.getObjectById(closestSegment.attributes.toNodeID);
            if (nodeA && nodeB) {
                var 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;
            }
        };

        var findConnections = function (startingNodeID, recursionDepth) {
            var connectedSegments,
                k,
                newNode;

            // Limit search depth to avoid problems.
            if (recursionDepth > MAX_RECURSION_DEPTH) {
                //console.debug('Max recursion depth reached');
                return;
            }

            // Populate variable with segments connected to starting node.
            connectedSegments = _.where(orderedSegments, {
                fromNodeID: startingNodeID
            });
            connectedSegments = connectedSegments.concat(_.where(orderedSegments, {
                toNodeID: startingNodeID
            }));

            //console.debug('Looking for connections at node ' + startingNodeID);

            // Check connected segments for address info.
            for (k in connectedSegments) {
                if (connectedSegments.hasOwnProperty(k)) {
                    if (hasStreetName(connectedSegments[k].segment)) {
                        // Address found, push to array.
                        /*
                            console.debug('Address found on connnected segment ' +
                            connectedSegments[k].segment.attributes.id +
                            '. Recursion depth: ' + recursionDepth);
                            */
                        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.
                        //console.debug('Address not found on connected segment ' + connectedSegments[k].segment.attributes.id);
                        newNode = connectedSegments[k].segment.attributes.fromNodeID === startingNodeID ?
                            connectedSegments[k].segment.attributes.toNodeID :
                            connectedSegments[k].segment.attributes.fromNodeID;
                        findConnections(newNode, recursionDepth + 1);
                    }
                }
            }
        };

        var getFCRank = function (FC) {
            var typeToFCRank = {
                3: 0, // freeway
                6: 1, // major
                7: 2, // minor
                2: 3, // primary
                1: 4, // street
                20: 5, // PLR
                8: 6 // dirt
            };
            if (FC && !isNaN(FC)) {
                return typeToFCRank[FC] || 100;
            }
        };

        var hasStreetName = function (segment) {
            return segment && segment.type === 'segment' && segment.getAddress().getStreetName() !== 'No street';
        };

        // phlogdev('No address data, gathering ', 2);

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

        if (venue.isPoint()) {
            stopPoint = venue.geometry;
        } else {
            var entryExitPoints = venue.attributes.entryExitPoints;
            if (entryExitPoints.length) {
                stopPoint = entryExitPoints[0];
            } else {
                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.indexOf(segments[i].attributes.roadType) === -1) {
                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) {
                    _.each(foundAddresses, function (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.
                var closestElem = _.find(orderedSegments, function (element) { return 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) {
        var 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
            };
            var 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) {
        var searchHN = '', searchStreet = '', searchCity = '';
        searchName = searchName.replace(/\//g, ' ');
        if ('string' === typeof addr.street.name && addr.street.name !== null && addr.street.name !== '') {
            searchStreet = addr.street.name + ', ';
        }
        searchStreet = searchStreet.replace(/CR-/g, 'County Rd ');
        searchStreet = searchStreet.replace(/SR-/g, 'State Hwy ');
        searchStreet = searchStreet.replace(/US-/g, 'US Hwy ');
        searchStreet = searchStreet.replace(/ CR /g, ' County Rd ');
        searchStreet = searchStreet.replace(/ SR /g, ' State Hwy ');
        searchStreet = searchStreet.replace(/ US /g, ' US Hwy ');
        searchStreet = searchStreet.replace(/$CR /g, 'County Rd ');
        searchStreet = searchStreet.replace(/$SR /g, 'State Hwy ');
        searchStreet = searchStreet.replace(/$US /g, 'US Hwy ');
        if ('string' === typeof HN && searchStreet !== '') {
            searchHN = HN + ' ';
        }
        if ('string' === typeof addr.city.attributes.name && addr.city.attributes.name !== '') {
            searchCity = addr.city.attributes.name + ', ';
        }

        searchName = searchName + (searchName ? ', ' : '') + searchHN + searchStreet + searchCity + addr.state.name;
        return 'http://www.google.com/search?q=' + encodeURIComponent(searchName);
    } // END buildGLink function

    // WME Category translation from Natural language to object language  (Bank / Financial --> BANK_FINANCIAL)
    function catTranslate(natCategories) {
        var 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.'
            });
        }
        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 (var i = 0; i < array1.length; i++) {
            if (array2.indexOf(array1[i]) === -1) {
                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 (var ixx = 0; ixx < target.length; ixx++) {
            if (source.indexOf(target[ixx]) === -1) {
                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.indexOf(tt) > -1);
    }

    // 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
        var arrayNew = array1.slice(0);  // 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
            var arrayTemp = arrayNew.splice(ix);  // split and hold the first part
            arrayNew.push.apply(arrayNew, array2);  // add the insert
            arrayNew.push.apply(arrayNew, 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) {
        var newAliasesUpdate = [];
        nName = nName.toUpperCase().replace(/'/g, '').replace(/-/g, ' ').replace(/\/ /g, ' ').replace(/ \//g, ' ').replace(/ {2,}/g, ' ');
        for (var 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 {
                bannButt.sfAliases = new Flag.SFAliases();
            }
        }
        return newAliasesUpdate;
    }

    function initSettingsCheckbox(settingID) {
        //Associate click event of new checkbox to call saveSettingToLocalStorage with proper ID
        $('#' + settingID).click(function () { 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) {
        let $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() {
        let $modifKeyCheckbox = $('#WMEPH-KBSModifierKey');
        let $shortcutInput = $('#WMEPH-KeyboardShortcut');
        let $warn = $('#PlaceHarmonizerKBWarn');
        let modifKeyNew;

        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, function () { harmonizePlace(); });
        $('#PlaceHarmonizerKBCurrent').empty().append('<span style="font-weight:bold">Current shortcut: ' + modifKey + shortcutParse + '</span>');
    }

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

        $warn.empty();  // remove old warning
        if (newKey.match(/^[a-z]{1}$/i) !== null) {  // If a single letter...
            shortcutParse = parseKBSShift(oldKey);
            let shortcutParseNew = parseKBSShift(newKey);
            shortcut.remove(modifKey + shortcutParse);
            shortcutParse = shortcutParseNew;
            shortcut.add(modifKey + shortcutParse, function () { 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() {
        let $current = $('#PlaceHarmonizerKBCurrent');
        let defaultShortcutKey = _IS_DEV_VERSION ? 'S' : 'A';
        let shortcutID = 'WMEPH-KeyboardShortcut';
        let shortcutKey = localStorage.getItem(shortcutID);
        let $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, function () { 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() {
        let $wlToolsMsg = $('#PlaceHarmonizerWLToolsMsg');
        let $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
                saveWL_LS(true);
            }
        } else {  // try to merge uncompressed WL data
            WLSToMerge = validateWLS($('#WMEPH-WLInput').val());
            if (WLSToMerge) {
                phlog('Whitelists merged!');
                venueWhitelist = mergeWL(venueWhitelist, WLSToMerge);
                saveWL_LS(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);
                    saveWL_LS(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(WLlocalStoreNameCompressed)));
        $('#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() {
        let currWLData = JSON.parse(LZString.decompressFromUTF16(localStorage.getItem(WLlocalStoreNameCompressed)));
        let countryWL = {};
        let stateWL = {};
        let entries = Object.keys(currWLData).filter(key => key !== '1.1.1');

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

        let countryString = '';
        for (var countryKey in countryWL) {
            countryString = countryString + '<br>' + countryKey + ': ' + countryWL[countryKey];
        }
        let stateString = '';
        for (var stateKey in stateWL) {
            stateString = stateString + '<br>' + stateKey + ': ' + stateWL[stateKey];
        }

        $('#PlaceHarmonizerWLToolsMsg').empty().append('<p style="color:black">Number of WL places: ' + entries.length + '</p><p><span style="font-weight:bold;"><u>States</u></span>' + stateString + '</p><p><span style="font-weight:bold;"><u>Countries</u></span>' + countryString + '<p>');
        //localStorage.setItem('WMEPH_WLAddCount', 1);
    }

    function onWLStateFilterClick() {
        let $wlToolsMsg = $('#PlaceHarmonizerWLToolsMsg');
        let $wlInput = $('#WMEPH-WLInput');
        let stateToRemove = $wlInput.val().trim();
        let msg = '';

        if (stateToRemove.length < 2) {
            msg = '<p style="color:red">Invalid state. Enter the state name in the "Whitelist string" box above, exactly as it appears in the Stats output.<p>';
        } else {
            var currWLData, venueToRemove = [];
            currWLData = JSON.parse(LZString.decompressFromUTF16(localStorage.getItem(WLlocalStoreNameCompressed)));

            Object.keys(currWLData).filter(venueKey => venueKey !== '1.1.1').forEach(venueKey => {
                if (currWLData[venueKey].state === stateToRemove || (!currWLData[venueKey].state && stateToRemove === 'None')) {
                    venueToRemove.push(venueKey);
                }
            });
            if (venueToRemove.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.')) {
                        backupWL_LS(true);
                        for (var ixwl = 0; ixwl < venueToRemove.length; ixwl++) {
                            delete venueWhitelist[venueToRemove[ixwl]];
                        }
                        saveWL_LS(true);
                        msg = '<p style="color:green">' + venueToRemove.length + ' items removed from WL<p>';
                        $wlInput.val('');
                    } else {
                        msg = '<p style="color:blue">No changes made<p>';
                    }
                } else {
                    msg = '<p style="color:red">Please backup your WL using the Pull button before removing state data<p>';
                }
            } else {
                msg = '<p style="color:red">No data for that state. Use the state name exactly as listed in the Stats<p>';
            }
        }
        $wlToolsMsg.empty().append(msg);
    }

    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-AutoRunOnSelect');
        }
        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) {
            var runOnceDefaultIgnorePlaGoogleLinkChecks = localStorage.getItem('WMEPH-runOnce-defaultToOff-plaGoogleLinkChecks');
            if (!runOnceDefaultIgnorePlaGoogleLinkChecks) {
                var $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(bootstrapWMEPH_CH);
        $('#WMEPH-DisableHoursHL').click(bootstrapWMEPH_CH);
        $('#WMEPH-DisableRankHL').click(bootstrapWMEPH_CH);
        $('#WMEPH-DisableWLHL').click(bootstrapWMEPH_CH);
        $('#WMEPH-PLATypeFill').click(() => applyHighlightsTest(W.model.venues.getObjectArray()));

        _initAlreadyRun = true;
    }

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

        var $container = $('<div id="wmephtab" class="active" style="padding-top: 5px;">');
        var $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>');
        var $tabContent = $('<div class="tab-content" style="padding:5px;">');
        var $harmonizerTab = $('<div class="tab-pane active" id="sidepanel-harmonizer"></div>');
        var $highlighterTab = $('<div class="tab-pane" id="sidepanel-highlighter"></div>');
        var $wlToolsTab = $('<div class="tab-pane" id="sidepanel-wltools"></div>');
        var $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');
            createSettingsCheckbox($harmonizerTab, 'WMEPH-AutoRunOnSelect', 'Automatically run the script when selecting a place');
        }

        $harmonizerTab.append('<hr align="center" width="90%">');

        // Add Letter input box
        var $phShortcutDiv = $('<div id="PlaceHarmonizerKB">');
        $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 align="center" width="90%"><p>Dev Only Settings:</p>');
            createSettingsCheckbox($phShortcutDiv, 'WMEPH-RegionOverride', 'Disable Region Specificity');
        }

        $harmonizerTab.append($phShortcutDiv);

        $harmonizerTab.append('<hr align="center" width="95%"><p><a href="' +
            _URLS.placesWiki + '" target="_blank">Open the WME Places Wiki page</a><p><a href="' +
            _URLS.forum + '" target="_blank">Submit script feedback & suggestions</a></p><hr align="center" width="95%">Recent updates:<ul>' +
            _WHATS_NEW_LIST.map(i => '<li>' + i + '</li>').join('') + '</ul>');

        // 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
        let 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': ['SkyviewGuru'],
            'PLN': ['bretmcvey', 'dmee92', 'ehepner1977'],
            'SAT': ['crazycaveman', 'whathappened15', 'xanderb'],
            'SCR': ['jm6087'],
            'SER': ['driving79', 'fjsawicki', 'itzwolf'],
            'SWR': ['tonestrtm']
        }

        $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.'),
            Object.keys(pnhModerators).map(region =>
                $('<div>', { style: 'margin-bottom: 10px; font-weight: bold;' }).text(region).append(
                    $('<div>', { style: 'font-weight: normal;' }).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 {
            var objTry = JSON.parse(jsonString);
            if (objTry && typeof objTry === 'object' && objTry !== null) {
                return objTry;
            }
        } catch (e) { }
        return false;
    }

    // This function merges and updates venues from object vWL_2 into vWL_1
    function mergeWL(vWL_1, vWL_2) {
        var venueKey, WLKey, vWL_1_Venue, vWL_2_Venue;
        for (venueKey in vWL_2) {
            if (vWL_2.hasOwnProperty(venueKey)) {  // basic filter
                if (vWL_1.hasOwnProperty(venueKey)) {  // if the vWL_2 venue is in vWL_1, then update any keys
                    vWL_1_Venue = vWL_1[venueKey];
                    vWL_2_Venue = vWL_2[venueKey];
                    for (WLKey in vWL_2_Venue) {  // loop thru the venue WL keys
                        if (vWL_2_Venue.hasOwnProperty(WLKey) && vWL_2_Venue[WLKey].active) {  // Only update if the vWL_2 key is active
                            if (vWL_1_Venue.hasOwnProperty(WLKey) && vWL_1_Venue[WLKey].active) {  // if the key is in the vWL_1 venue and it is active, then push any array data onto the key
                                if (vWL_1_Venue[WLKey].hasOwnProperty('WLKeyArray')) {
                                    vWL_1[venueKey][WLKey].WLKeyArray = insertAtIX(vWL_1[venueKey][WLKey].WLKeyArray, vWL_2[venueKey][WLKey].WLKeyArray, 100);
                                }
                            } else {  // if the key isn't in the vWL_1 venue, or if it's inactive, then copy the vWL_2 key across
                                vWL_1[venueKey][WLKey] = vWL_2[venueKey][WLKey];
                            }
                        }
                    } // END subLoop for venue keys
                } else {  // if the venue doesn't exist in vWL_1, then add it
                    vWL_1[venueKey] = vWL_2[venueKey];
                }
            }
        }
        return vWL_1;
    }

    // Get services checkbox status
    function getServicesChecks(venue) {
        var servArrayCheck = [];
        for (var wsix = 0; wsix < WMEServicesArray.length; wsix++) {
            if (venue.attributes.services.indexOf(WMEServicesArray[wsix]) > -1) {
                servArrayCheck[wsix] = true;
            } else {
                servArrayCheck[wsix] = false;
            }
        }
        return servArrayCheck;
    }

    function updateServicesChecks() {
        let venue = getSelectedVenue();
        if (venue) {
            if (!bannServ) return;
            var servArrayCheck = getServicesChecks(venue), wsix = 0;
            for (var keys in bannServ) {
                if (bannServ.hasOwnProperty(keys)) {
                    bannServ[keys].checked = servArrayCheck[wsix];  // reset all icons to match any checked changes
                    bannServ[keys].active = bannServ[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)) {
                bannServ.add247.checked = true;
            }
            bannServ.add247.active = true;
        }
    }

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

    // Pulls the item PL
    function getItemPL() {
        // Append a form div if it doesn't exist yet:
        if ($('#WMEPH_formDiv').length === 0) {
            var tempDiv = document.createElement('div');
            tempDiv.id = 'WMEPH_formDiv';
            tempDiv.style.display = 'inline';
            $('.WazeControlPermalink').append(tempDiv);
        }
        // Return the current PL
        if ($('.WazeControlPermalink').length === 0) {
            phlog('Waiting for PL div');
            setTimeout(getItemPL, 500);
            return;
        }
        if ($('.WazeControlPermalink .permalink').attr('href').length > 0) {
            return $('.WazeControlPermalink .permalink').attr('href');
        } else 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 (PMUserList.hasOwnProperty('WMEPH') && PMUserList.WMEPH.approvalActive) {
            data['address_list[u][' + PMUserList.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) {
        var form = document.createElement('form');
        form.target = '_blank';
        form.action = url;
        form.method = 'post';
        form.style.display = 'none';
        for (var k in data) {
            if (data.hasOwnProperty(k)) {
                var 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 {OL.Geometry} The new geometry for the place.
         */
    function updateFeatureGeometry(place, newGeometry) {
        var oldGeometry,
            model = W.model.venues,
            wmeUpdateFeatureGeometry = require('Waze/Action/UpdateFeatureGeometry');
        if (place && place.CLASS_NAME === 'Waze.Feature.Vector.Landmark' &&
            newGeometry && (newGeometry instanceof OL.Geometry.Point ||
                newGeometry instanceof OL.Geometry.Polygon)) {
            oldGeometry = place.attributes.geometry;
            W.model.actionManager.add(
                new wmeUpdateFeatureGeometry(place, model, oldGeometry, newGeometry));
        }
    }

    function placeHarmonizer_init() {
        // For debugging purposes.  May be removed when no longer needed.
        unsafeWindow.PNH_DATA = _PNH_DATA;

        _USER.ref = W.loginManager.user;
        _USER.name = _USER.ref.userName;
        _USER.rank = _USER.ref.normalizedLevel;  // get editor's level (actual level)
        userLanguage = I18n.locale;

        // Array prototype extensions (for Firefox fix)
        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();
        _updatedFields.init();
        addPURWebSearchButton();

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

        let xrayMode = localStorage.getItem('WMEPH_xrayMode_enabled') === 'true' ? true : false;
        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(WLlocalStoreNameCompressed))) === false) {  // If no compressed WL string exists
            if (validateWLS(localStorage.getItem(WLlocalStoreName)) === false) {  // If no regular WL exists
                venueWhitelist = { '1.1.1': { Placeholder: {} } }; // Populate with a dummy place
                saveWL_LS(false);
                saveWL_LS(true);
            } else {  // if regular WL string exists, then transfer to compressed version
                localStorage.setItem('WMEPH-OneTimeWLBU', localStorage.getItem(WLlocalStoreName));
                loadWL_LS(false);
                saveWL_LS(true);
                alert('Whitelists are being converted to a compressed format.  If you have trouble with your WL, please submit an error report.');
            }
        } else {
            loadWL_LS(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');
        }

        W.map.events.register('mousemove', W.map, e => errorHandler(() => {
            WMEPHmousePosition = W.map.getLonLatFromPixel(W.map.events.getMousePosition(e));
        }));

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

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

        // Add Autorun shortcut
        if (_USER.name === 'bmtg') {
            shortcut.add('Control+Alt+u', function () {
                $('#WMEPH-AutoRunOnSelect').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.
        var removedWLCount = 0;
        Object.keys(venueWhitelist).forEach(venueID => {
            if (venueID < 0) {
                delete venueWhitelist[venueID];
                removedWLCount += 1;
            }
        });
        if (removedWLCount > 0) {
            saveWL_LS(true);
            phlogdev('Removed ' + removedWLCount + ' venues with temporary ID\'s from WL store');
        }

        if (WMEPHbetaList.length === 0 || 'undefined' === typeof WMEPHbetaList) {
            if (_IS_DEV_VERSION) {
                alert('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 {
            let lcName = _USER.name.toLowerCase();
            _USER.isDevUser = WMEPHdevList.indexOf(lcName) > -1;
            _USER.isBetaUser = WMEPHbetaList.indexOf(lcName) > -1;
        }
        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
        let _stateHeaders = _PNH_DATA.states[0].split('|');
        ps_state_ix = _stateHeaders.indexOf('ps_state');
        ps_state2L_ix = _stateHeaders.indexOf('ps_state2L');
        ps_region_ix = _stateHeaders.indexOf('ps_region');
        ps_gFormState_ix = _stateHeaders.indexOf('ps_gFormState');
        ps_defaultLockLevel_ix = _stateHeaders.indexOf('ps_defaultLockLevel');
        //ps_requirePhone_ix = _stateHeaders.indexOf('ps_requirePhone');
        //ps_requireURL_ix = _stateHeaders.indexOf('ps_requireURL');
        ps_areacode_ix = _stateHeaders.indexOf('ps_areacode');

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

        /**
         * Generates highlighting rules and applies them to the map.
         */
        layer = W.map.landmarkLayer;

        // Setup highlight colors
        initializeHighlights();

        // used for phone reformatting
        if (!String.plFormat) {
            String.plFormat = function (format) {
                var args = Array.prototype.slice.call(arguments, 1);
                return format.replace(/{(\d+)}/g, function (name, number) {
                    return typeof args[number] !== 'undefined' ? args[number] : null;
                });
            };
        }

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

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

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

    function callAjaxAsync(url) {
        return new Promise((resolve, reject) => {
            $.ajax({
                type: 'GET',
                url: url,
                jsonp: 'callback', data: { alt: 'json-in-script' }, dataType: 'jsonp',
                success: resolve,
                error: reject
            });
        });
    }

    const SPREADSHEET_ID = '1pBz4l4cNapyGyzfMJKqA4ePEFLkmz2RryAt1UV39B4g';
    const SPREADSHEET_RANGE = '2019.01.20.001!A2:K';
    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)}`;
        let processData = (response, fieldName) => response.feed.entry.map(entry => entry[fieldName].$t);
        //TODO change the _PNH_DATA cache to use an object so we don't have to rely on ugly array index lookups.
        let 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') {
                alert('You are using an outdated version of WMEPH that doesn\'t work anymore. Update or disable the script.');
                return;
            }

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

            var WMEPHuserList = processData1(values, 4)[1].split('|');
            var betaix = WMEPHuserList.indexOf('BETAUSERS');
            WMEPHdevList = [];
            WMEPHbetaList = [];
            for (var ulix = 1; ulix < betaix; ulix++) WMEPHdevList.push(WMEPHuserList[ulix].toLowerCase().trim());
            for (ulix = betaix + 1; ulix < WMEPHuserList.length; ulix++) WMEPHbetaList.push(WMEPHuserList[ulix].toLowerCase().trim());

            let processTermsCell = (values, colIdx) => processData1(values, 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);

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

    // Start downloading the PNH spreadsheet data in the background.  Starts the script once data is ready.
    downloadPnhData();

})();