WME Place Harmonizer

Harmonizes, formats, and locks a selected place

当前为 2023-09-28 提交的版本,查看 最新版本

您需要先安装一个扩展,例如 篡改猴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     2023.09.27.001
// @description Harmonizes, formats, and locks a selected place
// @author      WMEPH Development Group
// @include     /^https:\/\/(www|beta)\.waze\.com\/(?!user\/)(.{2,6}\/)?editor\/?.*$/
// @require     https://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
// @connect     greasyfork.org
// @grant       GM_addStyle
// @grant       GM_xmlhttpRequest
// ==/UserScript==

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

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

(function main() {
    'use strict';

    // Script update info

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

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

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

    const SCRIPT_VERSION = GM_info.script.version.toString(); // pull version from header
    const SCRIPT_NAME = GM_info.script.name;
    const IS_BETA_VERSION = /Beta/i.test(SCRIPT_NAME); //  enables dev messages and unique DOM options if the script is called "... Beta"
    const BETA_VERSION_STR = IS_BETA_VERSION ? 'Beta' : ''; // strings to differentiate DOM elements between regular and beta script
    const PNH_DATA = { USA: {}, CAN: {} };
    const DEFAULT_HOURS_TEXT = 'Paste hours here';
    const MAX_CACHE_SIZE = 25000;
    const PROD_DOWNLOAD_URL = 'https://greasyfork.org/scripts/28690-wme-place-harmonizer/code/WME%20Place%20Harmonizer.user.js';
    const BETA_DOWNLOAD_URL = 'YUhSMGNITTZMeTluY21WaGMzbG1iM0pyTG05eVp5OXpZM0pwY0hSekx6STROamc1TFhkdFpTMXdiR0ZqWlMxb1lYSnRiMjVwZW1WeUxXSmxkR0V2WTI5a1pTOVhUVVVsTWpCUWJHRmpaU1V5TUVoaGNtMXZibWw2WlhJbE1qQkNaWFJoTG5WelpYSXVhbk09';

    let _wordVariations;
    let _resultsCache = {};
    let _initAlreadyRun = false; // This is used to skip a couple things if already run once.  This could probably be handled better...
    let _textEntryValues = null; // Store the values entered in text boxes so they can be re-added when the banner is reassembled.

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

    let _attributionEl;
    let _placesService;

    // Userlists
    let _wmephDevList;
    let _wmephBetaList;

    let _shortcutParse;
    let _modifKey = 'Alt+';

    // Whitelisting vars
    let _venueWhitelist;
    const WL_BUTTON_TEXT = 'WL';
    const WL_LOCAL_STORE_NAME = 'WMEPH-venueWhitelistNew';
    const WL_LOCAL_STORE_NAME_COMPRESSED = 'WMEPH-venueWhitelistCompressed';

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

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

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

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

    const USER = {
        ref: null,
        rank: null,
        name: null,
        isBetaUser: false,
        isDevUser: false
    };
    const SETTING_IDS = {
        sfUrlWarning: 'SFURLWarning', // Warning message for first time using localized storefinder URL.
        gLinkWarning: 'GLinkWarning' // Warning message for first time using Google search to not to use the Google info itself.
    };
    const URLS = {
        forum: 'https://www.waze.com/forum/posting.php?mode=reply&f=819&t=239985',
        usaPnh: 'https://docs.google.com/spreadsheets/d/1-f-JTWY5UnBx-rFTa4qhyGMYdHBZWNirUTOgn222zMY/edit#gid=0',
        placesWiki: 'https://wazeopedia.waze.com/wiki/USA/Places',
        restAreaWiki: 'https://wazeopedia.waze.com/wiki/USA/Rest_areas#Adding_a_Place',
        uspsWiki: 'https://wazeopedia.waze.com/wiki/USA/Places/Post_office'
    };
    class Region {
        static #defaultNewChainRequestEntryIds = ['entry.925969794', 'entry.1970139752', 'entry.1749047694'];
        static #defaultApproveChainRequestEntryIds = ['entry.925969794', 'entry.50214576', 'entry.1749047694'];
        #formId;
        #newChainRequestEntryIds;
        #approveChainRequestEntryIds;

        constructor(formId, newChainRequestEntryIds, approveChainRequestEntryIds) {
            this.#formId = formId;
            this.#newChainRequestEntryIds = newChainRequestEntryIds ?? Region.#defaultNewChainRequestEntryIds;
            this.#approveChainRequestEntryIds = approveChainRequestEntryIds ?? Region.#defaultApproveChainRequestEntryIds;
        }

        #getFormUrl(entryIds, entryValues) {
            const entryValuesUrl = entryValues.map((value, idx) => `${entryIds[idx]}=${value}`).join('&');
            return `https://docs.google.com/forms/d/${this.#formId}/viewform?${entryValuesUrl}`;
        }

        getNewChainFormUrl(entryValues) {
            return this.#getFormUrl(this.#newChainRequestEntryIds, entryValues);
        }

        getApproveChainFormUrl(entryValues) {
            return this.#getFormUrl(this.#approveChainRequestEntryIds, entryValues);
        }
    }
    const REGION_SETTINGS = {
        NWR: new Region('1hv5hXBlGr1pTMmo4n3frUx1DovUODbZodfDBwwTc7HE'),
        SWR: new Region('1Qf2N4fSkNzhVuXJwPBJMQBmW0suNuy8W9itCo1qgJL4'),
        HI: new Region('1K7Dohm8eamIKry3KwMTVnpMdJLaMIyDGMt7Bw6iqH_A', null, ['entry.1497446659', 'entry.50214576', 'entry.1749047694']),
        PLN: new Region('1ycXtAppoR5eEydFBwnghhu1hkHq26uabjUu8yAlIQuI'),
        SCR: new Region('1KZzLdlX0HLxED5Bv0wFB-rWccxUp2Mclih5QJIQFKSQ'),
        GLR: new Region('19btj-Qt2-_TCRlcS49fl6AeUT95Wnmu7Um53qzjj9BA'),
        SAT: new Region(
            '1bxgK_20Jix2ahbmUvY1qcY0-RmzUBT6KbE5kjDEObF8',
            ['entry.2063110249', 'entry.2018912633', 'entry.1924826395'],
            ['entry.2063110249', 'entry.123778794', 'entry.1924826395']
        ),
        SER: new Region(
            '1jYBcxT3jycrkttK5BxhvPXR240KUHnoFMtkZAXzPg34',
            ['entry.822075961', 'entry.1422079728', 'entry.1891389966'],
            ['entry.822075961', 'entry.607048307', 'entry.1891389966']
        ),
        ATR: new Region('1v7JhffTfr62aPSOp8qZHA_5ARkBPldWWJwDeDzEioR0'),
        NER: new Region('1UgFAMdSQuJAySHR0D86frvphp81l7qhEdJXZpyBZU6c'),
        NOR: new Region('1iYq2rd9HRd-RBsKqmbHDIEBGuyWBSyrIHC6QLESfm4c'),
        MAR: new Region('1PhL1iaugbRMc3W-yGdqESoooeOz-TJIbjdLBRScJYOk'),
        CA_EN: new Region(
            '13JwXsrWPNmCdfGR5OVr5jnGZw-uNGohwgjim-JYbSws',
            ['entry_839085807', 'entry_1067461077', 'entry_318793106', 'entry_1149649663'],
            ['entry_839085807', 'entry_1125435193', 'entry_318793106', 'entry_1149649663']
        ),
        QC: new Region(
            '13JwXsrWPNmCdfGR5OVr5jnGZw-uNGohwgjim-JYbSws',
            ['entry_839085807', 'entry_1067461077', 'entry_318793106', 'entry_1149649663'],
            ['entry_839085807', 'entry_1125435193', 'entry_318793106', 'entry_1149649663']
        )
    };
    let _userLanguage;
    // lock levels are offset by one
    const LOCK_LEVEL_2 = 1;
    const LOCK_LEVEL_4 = 3;

    // An enum to help clarify flag severity levels
    const SEVERITY = {
        GREEN: 0,
        BLUE: 1,
        YELLOW: 2,
        RED: 3,
        // 4 isn't used anymore
        PINK: 5
        // TODO: There is also 'lock' and 'lock1' severity. Add those here? Also investigate 'adLock' severity (is it still useful in WME???).
    };

    const CAT = {
        AIRPORT: 'AIRPORT',
        // ART_GALLERY: 'ART_GALLERY',
        // ARTS_AND_CRAFTS: 'ARTS_AND_CRAFTS',
        ATM: 'ATM',
        // BAKERY: 'BAKERY',
        BANK_FINANCIAL: 'BANK_FINANCIAL',
        BAR: 'BAR',
        // BEACH: 'BEACH',
        // BED_AND_BREAKFAST: 'BED_AND_BREAKFAST',
        // BOOKSTORE: 'BOOKSTORE',
        BRIDGE: 'BRIDGE',
        // BUS_STATION: 'BUS_STATION',
        // CAFE: 'CAFE',
        CAMPING_TRAILER_PARK: 'CAMPING_TRAILER_PARK',
        CANAL: 'CANAL',
        // CAR_DEALERSHIP: 'CAR_DEALERSHIP',
        CAR_RENTAL: 'CAR_RENTAL',
        // CAR_SERVICES: 'CAR_SERVICES',
        // CAR_WASH: 'CAR_WASH',
        // CASINO: 'CASINO',
        CHARGING_STATION: 'CHARGING_STATION',
        CEMETERY: 'CEMETERY',
        // CITY_HALL: 'CITY_HALL',
        // CLUB: 'CLUB',
        COLLEGE_UNIVERSITY: 'COLLEGE_UNIVERSITY',
        // CONSTRUCTION_SITE: 'CONSTRUCTION_SITE',
        CONVENIENCE_STORE: 'CONVENIENCE_STORE',
        CONVENTIONS_EVENT_CENTER: 'CONVENTIONS_EVENT_CENTER',
        COTTAGE_CABIN: 'COTTAGE_CABIN',
        // COURTHOUSE: 'COURTHOUSE',
        CULTURE_AND_ENTERTAINEMENT: 'CULTURE_AND_ENTERTAINEMENT',
        // CURRENCY_EXCHANGE: 'CURRENCY_EXCHANGE',
        DAM: 'DAM',
        // DEPARTMENT_STORE: 'DEPARTMENT_STORE',
        DESSERT: 'DESSERT',
        DOCTOR_CLINIC: 'DOCTOR_CLINIC',
        // ELECTRONICS: 'ELECTRONICS',
        // EMBASSY_CONSULATE: 'EMBASSY_CONSULATE',
        // EMERGENCY_SHELTER: 'EMERGENCY_SHELTER',
        // FACTORY_INDUSTRIAL: 'FACTORY_INDUSTRIAL',
        FARM: 'FARM',
        // FASHION_AND_CLOTHING: 'FASHION_AND_CLOTHING',
        // FAST_FOOD: 'FAST_FOOD',
        FERRY_PIER: 'FERRY_PIER',
        FIRE_DEPARTMENT: 'FIRE_DEPARTMENT',
        // FLOWERS: 'FLOWERS',
        FOOD_AND_DRINK: 'FOOD_AND_DRINK',
        // FOOD_COURT: 'FOOD_COURT',
        FOREST_GROVE: 'FOREST_GROVE',
        // FURNITURE_HOME_STORE: 'FURNITURE_HOME_STORE',
        // GAME_CLUB: 'GAME_CLUB',
        // GARAGE_AUTOMOTIVE_SHOP: 'GARAGE_AUTOMOTIVE_SHOP',
        GAS_STATION: 'GAS_STATION',
        // GIFTS: 'GIFTS',
        GOLF_COURSE: 'GOLF_COURSE',
        // GOVERNMENT: 'GOVERNMENT',
        GYM_FITNESS: 'GYM_FITNESS',
        // HARDWARE_STORE: 'HARDWARE_STORE',
        HOSPITAL_MEDICAL_CARE: 'HOSPITAL_MEDICAL_CARE',
        HOSPITAL_URGENT_CARE: 'HOSPITAL_URGENT_CARE',
        // HOSTEL: 'HOSTEL',
        HOTEL: 'HOTEL',
        // ICE_CREAM: 'ICE_CREAM',
        // INFORMATION_POINT: 'INFORMATION_POINT',
        ISLAND: 'ISLAND',
        // JEWELRY: 'JEWELRY',
        JUNCTION_INTERCHANGE: 'JUNCTION_INTERCHANGE',
        // KINDERGARDEN: 'KINDERGARDEN',
        // LAUNDRY_DRY_CLEAN: 'LAUNDRY_DRY_CLEAN',
        // LIBRARY: 'LIBRARY',
        LODGING: 'LODGING',
        // MARKET: 'MARKET',
        // MILITARY: 'MILITARY',
        MOVIE_THEATER: 'MOVIE_THEATER',
        // MUSEUM: 'MUSEUM',
        // MUSIC_STORE: 'MUSIC_STORE',
        // MUSIC_VENUE: 'MUSIC_VENUE',
        NATURAL_FEATURES: 'NATURAL_FEATURES',
        OFFICES: 'OFFICES',
        // ORGANIZATION_OR_ASSOCIATION: 'ORGANIZATION_OR_ASSOCIATION',
        OTHER: 'OTHER',
        // OUTDOORS: 'OUTDOORS',
        PARK: 'PARK',
        PARKING_LOT: 'PARKING_LOT',
        PERSONAL_CARE: 'PERSONAL_CARE',
        PET_STORE_VETERINARIAN_SERVICES: 'PET_STORE_VETERINARIAN_SERVICES',
        // PERFORMING_ARTS_VENUE: 'PERFORMING_ARTS_VENUE',
        PHARMACY: 'PHARMACY',
        // PHOTOGRAPHY: 'PHOTOGRAPHY',
        PLAYGROUND: 'PLAYGROUND',
        // PLAZA: 'PLAZA',
        POLICE_STATION: 'POLICE_STATION',
        // POOL: 'POOL',
        POST_OFFICE: 'POST_OFFICE',
        // PRISON_CORRECTIONAL_FACILITY: 'PRISON_CORRECTIONAL_FACILITY',
        // PROFESSIONAL_AND_PUBLIC: 'PROFESSIONAL_AND_PUBLIC',
        // PROMENADE: 'PROMENADE',
        // RACING_TRACK: 'RACING_TRACK',
        RELIGIOUS_CENTER: 'RELIGIOUS_CENTER',
        RESIDENCE_HOME: 'RESIDENCE_HOME',
        REST_AREAS: 'REST_AREAS',
        RESTAURANT: 'RESTAURANT',
        RIVER_STREAM: 'RIVER_STREAM',
        SCENIC_LOOKOUT_VIEWPOINT: 'SCENIC_LOOKOUT_VIEWPOINT',
        SCHOOL: 'SCHOOL',
        SEA_LAKE_POOL: 'SEA_LAKE_POOL',
        SEAPORT_MARINA_HARBOR: 'SEAPORT_MARINA_HARBOR',
        SHOPPING_AND_SERVICES: 'SHOPPING_AND_SERVICES',
        SHOPPING_CENTER: 'SHOPPING_CENTER',
        // SKI_AREA: 'SKI_AREA',
        // SPORTING_GOODS: 'SPORTING_GOODS',
        SPORTS_COURT: 'SPORTS_COURT',
        STADIUM_ARENA: 'STADIUM_ARENA',
        SUBWAY_STATION: 'SUBWAY_STATION',
        SUPERMARKET_GROCERY: 'SUPERMARKET_GROCERY',
        SWAMP_MARSH: 'SWAMP_MARSH',
        // SWIMMING_POOL: 'SWIMMING_POOL',
        // TAXI_STATION: 'TAXI_STATION',
        // THEATER: 'THEATER',
        // THEME_PARK: 'THEME_PARK',
        // TELECOM: 'TELECOM',
        // TOURIST_ATTRACTION_HISTORIC_SITE: 'TOURIST_ATTRACTION_HISTORIC_SITE',
        // TOY_STORE: 'TOY_STORE',
        // TRAIN_STATION: 'TRAIN_STATION',
        TRANSPORTATION: 'TRANSPORTATION',
        // TRASH_AND_RECYCLING_FACILITIES: 'TRASH_AND_RECYCLING_FACILITIES',
        // TRAVEL_AGENCY: 'TRAVEL_AGENCY',
        TUNNEL: 'TUNNEL'
        // ZOO_AQUARIUM: 'ZOO_AQUARIUM',
    };

    let _catTransWaze2Lang; // pulls the category translations
    const EV_PAYMENT_METHOD = {
        APP: 'APP',
        CREDIT: 'CREDIT',
        DEBIT: 'DEBIT',
        MEMBERSHIP_CARD: 'MEMBERSHIP_CARD',
        ONLENE_PAYMENT: 'ONLINE_PAYMENT',
        PLUG_IN_AUTO_CHARGER: 'PLUG_IN_AUTO_CHARGE',
        OTHER: 'OTHER'
    };
    // Common payment types found at: https://wazeopedia.waze.com/wiki/USA/Places/EV_charging_station
    const COMMON_EV_PAYMENT_METHODS = {
        'Blink Charging': [
            EV_PAYMENT_METHOD.APP,
            EV_PAYMENT_METHOD.MEMBERSHIP_CARD,
            EV_PAYMENT_METHOD.PLUG_IN_AUTO_CHARGER,
            EV_PAYMENT_METHOD.OTHER
        ],
        ChargePoint: [
            EV_PAYMENT_METHOD.APP,
            EV_PAYMENT_METHOD.CREDIT,
            EV_PAYMENT_METHOD.MEMBERSHIP_CARD
        ],
        'Electrify America': [
            EV_PAYMENT_METHOD.APP,
            EV_PAYMENT_METHOD.CREDIT,
            EV_PAYMENT_METHOD.DEBIT,
            EV_PAYMENT_METHOD.MEMBERSHIP_CARD,
            EV_PAYMENT_METHOD.PLUG_IN_AUTO_CHARGER
        ],
        EVgo: [
            EV_PAYMENT_METHOD.APP,
            EV_PAYMENT_METHOD.CREDIT,
            EV_PAYMENT_METHOD.DEBIT,
            EV_PAYMENT_METHOD.MEMBERSHIP_CARD,
            EV_PAYMENT_METHOD.PLUG_IN_AUTO_CHARGER
        ],
        SemaConnect: [
            EV_PAYMENT_METHOD.APP,
            EV_PAYMENT_METHOD.MEMBERSHIP_CARD,
            EV_PAYMENT_METHOD.OTHER
        ],
        Tesla: [
            EV_PAYMENT_METHOD.PLUG_IN_AUTO_CHARGER
        ]
    };
    const WME_SERVICES_ARRAY = ['VALLET_SERVICE', 'DRIVETHROUGH', 'WI_FI', 'RESTROOMS', 'CREDIT_CARDS', 'RESERVATIONS', 'OUTSIDE_SEATING',
        'AIR_CONDITIONING', 'PARKING_FOR_CUSTOMERS', 'DELIVERIES', 'TAKE_AWAY', 'CURBSIDE_PICKUP', 'WHEELCHAIR_ACCESSIBLE', 'DISABILITY_PARKING'];
    const COLLEGE_ABBREVIATIONS = ['USF', 'USFSP', 'UF', 'UCF', 'UA', 'UGA', 'FSU', 'UM', 'SCP', 'FAU', 'FIU'];
    // Change place.name to title case
    const TITLECASE_SETTINGS = {
        ignoreWords: 'an|and|as|at|by|for|from|hhgregg|in|into|of|on|or|the|to|with'.split('|'),
        // eslint-disable-next-line max-len
        capWords: '3M|AAA|AMC|AOL|AT&T|ATM|BBC|BLT|BMV|BMW|BP|CBS|CCS|CGI|CISCO|CJ|CNG|CNN|CVS|DHL|DKNY|DMV|DSW|EMS|ER|ESPN|FCU|FCUK|FDNY|GNC|H&M|HP|HSBC|IBM|IHOP|IKEA|IRS|JBL|JCPenney|KFC|LLC|MBNA|MCA|MCI|NBC|NYPD|PDQ|PNC|TCBY|TNT|TV|UPS|USA|USPS|VW|XYZ|ZZZ'.split('|'),
        specWords: 'd\'Bronx|iFix|ExtraMile|ChargePoint|EVgo|SemaConnect'.split('|')
    };
    const PRIMARY_CATS_TO_IGNORE_MISSING_PHONE_URL = [
        CAT.ISLAND,
        CAT.SEA_LAKE_POOL,
        CAT.RIVER_STREAM,
        CAT.CANAL,
        CAT.JUNCTION_INTERCHANGE,
        CAT.SCENIC_LOOKOUT_VIEWPOINT
    ];
    const PRIMARY_CATS_TO_FLAG_GREEN_MISSING_PHONE_URL = [
        CAT.BRIDGE,
        CAT.FOREST_GROVE,
        CAT.DAM,
        CAT.TUNNEL,
        CAT.CEMETERY
    ];
    const ANY_CATS_TO_FLAG_GREEN_MISSING_PHONE_URL = [CAT.REST_AREAS];
    const REGIONS_THAT_WANT_PLA_PHONE_URL = ['SER'];
    const CHAIN_APPROVAL_PRIMARY_CATS_TO_IGNORE = [
        CAT.POST_OFFICE,
        CAT.BRIDGE,
        CAT.FOREST_GROVE,
        CAT.DAM,
        CAT.TUNNEL,
        CAT.CEMETERY,
        CAT.ISLAND,
        CAT.SEA_LAKE_POOL,
        CAT.RIVER_STREAM,
        CAT.CANAL,
        CAT.JUNCTION_INTERCHANGE,
        CAT.SCENIC_LOOKOUT_VIEWPOINT
    ];
    const CATS_THAT_DONT_NEED_NAMES = [
        CAT.SEA_LAKE_POOL
    ];
    const BAD_URL = 'badURL';
    const BAD_PHONE = 'badPhone';

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

    let _layer;

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

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

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

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

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

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

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

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

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

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

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

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

    function isAlwaysOpen(venue) {
        return is247Hours(venue.attributes.openingHours);
    }

    function is247Hours(openingHours) {
        return openingHours.length === 1 && openingHours[0].days.length === 7 && openingHours[0].isAllDay();
    }

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

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

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

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

        function panelContainerChanged() {
            if (!$('#WMEPH-HidePURWebSearch').prop('checked')) {
                const $panelNav = $('.place-update-edit .place-update > div > span');
                if ($('#PHPURWebSearchButton').length === 0 && $panelNav.length) {
                    const $btn = $('<div>').css({
                        paddingLeft: '15px',
                        paddingBottom: '8px'
                    }).append(
                        $('<button>', {
                            class: 'btn btn-danger', id: 'PHPURWebSearchButton', title: 'Search Google 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({
                                marginTop: '-10px',
                                fontSize: '14px'
                            })
                            .text('Google')
                            .click(() => { openWebSearch(); })
                    );
                    $panelNav.after($btn);
                }
            }
        }

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

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

        function openWebSearch() {
            const nameElem = $('.place-update-edit.panel .name');
            let name = null;
            let addr = null;
            if (nameElem.length) {
                name = $('.place-update-edit.panel .name').first().text();
                addr = $('.place-update-edit.panel .address').first().text();
            } else {
                name = $('.place-update-edit.panel .changes div div')[0].textContent;
                addr = $('.place-update-edit.panel .changes div div')[1].textContent;
            }
            if (!name) return;
            if ($('#WMEPH-WebSearchNewTab').prop('checked')) {
                window.open(buildSearchUrl(name, addr));
            } else {
                window.open(buildSearchUrl(name, addr), SEARCH_RESULTS_WINDOW_NAME, _searchResultsWindowSpecs);
            }
        }
    }

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

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

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

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

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

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

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

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

                // Next, add extensions to the search names based on the WME place category
                const categoryInfo = countryData.categoryInfos.getByName(splits[category1Idx]);
                const appendWords = [];
                if (categoryInfo) {
                    if (categoryInfo.id === CAT.HOTEL) {
                        appendWords.push('HOTEL');
                    } else if (categoryInfo.id === CAT.BANK_FINANCIAL && !/\bnotABank\b/.test(specCase)) {
                        appendWords.push('BANK', 'ATM');
                    } else if (categoryInfo.id === CAT.SUPERMARKET_GROCERY) {
                        appendWords.push('SUPERMARKET');
                    } else if (categoryInfo.id === CAT.GYM_FITNESS) {
                        appendWords.push('GYM');
                    } else if (categoryInfo.id === CAT.GAS_STATION) {
                        appendWords.push('GAS', 'GASOLINE', 'FUEL', 'STATION', 'GASSTATION');
                    } else if (categoryInfo.id === CAT.CAR_RENTAL) {
                        appendWords.push('RENTAL', 'RENTACAR', 'CARRENTAL', 'RENTALCAR');
                    }
                    appendWords.forEach(word => { newNameList = newNameList.concat(newNameList.map(name => name + word)); });
                }

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

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

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

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

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

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

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

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

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

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

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

        // Remove venue from the results cache so it can be updated again.
        delete _resultsCache[venue.attributes.id];
        return true;
    }

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

    function onVenuesChanged(venueProxies) {
        logDev('onVenuesChanged');
        deleteDupeLabel();

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

            updateWmephPanel();
        }
    }

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

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

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

        const defaultPointRadius = 6;
        const ruleGenerator = (value, symbolizer) => new W.Rule({
            filter: new OpenLayers.Filter.Comparison({
                type: '==',
                value,
                evaluate(feature) {
                    const attr = feature.attributes.wazeFeature?._wmeObject?.attributes;
                    return attr?.wmephSeverity === this.value;
                }
            }),
            symbolizer,
            wmephStyle: 'xray'
        });

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

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

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

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

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

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

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

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

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

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

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

        _layer.redraw();
    }

    function initializeHighlights() {
        const ruleGenerator = (value, symbolizer) => new W.Rule({
            filter: new OpenLayers.Filter.Comparison({
                type: '==',
                value,
                evaluate(feature) {
                    const attr = feature.attributes.wazeFeature?._wmeObject?.attributes;
                    return attr?.wmephSeverity === this.value;
                }
            }),
            symbolizer,
            wmephStyle: 'default'
        });

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

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

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

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

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

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

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

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

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

        function plaTypeRuleGenerator(value, symbolizer) {
            return new W.Rule({
                filter: new OpenLayers.Filter.Comparison({
                    type: '==',
                    value,
                    evaluate(feature) {
                        const attr = feature.attributes.wazeFeature?._wmeObject?.attributes;

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

        return str;
    }

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

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

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

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

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

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

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

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

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

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

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

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

    // Normalize url
    function normalizeURL(url, makeLowerCase = true) {
        if (!url?.trim().length) {
            return url;
        }

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

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

    // Abstract flag classes.  Must be declared outside the "Flag" namespace.
    class FlagBase {
        static defaultSeverity = SEVERITY.GREEN;
        static defaultMessage = '';
        static currentFlags;
        #severity;
        #message;
        #noLock;

        get name() { return this.constructor.name; }

        get severity() { return this.#severity ?? this.constructor.defaultSeverity; }
        set severity(value) { this.#severity = value; }

        get message() { return this.#message ?? this.constructor.defaultMessage; }
        set message(value) { this.#message = value; }

        get noLock() { return this.#noLock ?? this.severity > SEVERITY.BLUE; }
        set noLock(value) { this.#noLock = value; }

        constructor() {
            FlagBase.currentFlags.add(this);
        }

        static eval(args) {
            if (this.venueIsFlaggable(args)) {
                const flag = new this(args);
                flag.args = args;
                return flag;
            }
            return null;
        }
    }
    class ActionFlag extends FlagBase {
        static defaultButtonTooltip = '';
        #buttonText;
        #buttonTooltip;

        get buttonText() { return this.#buttonText ?? this.constructor.defaultButtonText; }
        set buttonText(value) { this.#buttonText = value; }
        get buttonTooltip() { return this.#buttonTooltip ?? this.constructor.defaultButtonTooltip; }
        set buttonTooltip(value) { this.#buttonTooltip = value; }

        // 5/19/2019 (mapomatic) This base class action function doesn't seem to be necessary.
        // action() { } // overwrite this
    }
    class WLFlag extends FlagBase {
        static defaultWLTooltip = 'Whitelist this message';
        #showWL;

        get severity() { return this.constructor.isWhitelisted(this.args) ? SEVERITY.GREEN : super.severity; }
        set severity(value) { super.severity = value; }

        get showWL() { return this.#showWL ?? !this.constructor.isWhitelisted(this.args); }
        set showWL(value) { this.#showWL = value; }

        get wlTooltip() { return this.constructor.defaultWLTooltip; }

        WLaction() {
            const venue = getSelectedVenue();
            if (whitelistAction(venue.attributes.id, this.constructor.WL_KEY)) {
                harmonizePlaceGo(venue, 'harmonize');
            }
        }

        static isWhitelisted(args) {
            return !!args.wl[this.WL_KEY];
        }
    }
    class WLActionFlag extends WLFlag {
        static defaultButtonTooltip = '';
        #buttonText;
        #buttonTooltip;

        get buttonText() { return this.#buttonText ?? this.constructor.defaultButtonText; }
        set buttonText(value) { this.#buttonText = value; }

        get buttonTooltip() { return this.#buttonTooltip ?? this.constructor.defaultButtonTooltip; }
        set buttonTooltip(value) { this.#buttonTooltip = value; }
    }

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

            constructor(inferredAddress) {
                super();
                this.inferredAddress = inferredAddress;
            }

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

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

            static venueIsFlaggable(args) {
                return !args.categories.includes(CAT.RESIDENCE_HOME)
                    && (!args.nameBase?.replace(/[^A-Za-z0-9]/g, ''))
                    && ![CAT.ISLAND, CAT.FOREST_GROVE, CAT.SEA_LAKE_POOL, CAT.RIVER_STREAM, CAT.CANAL, CAT.PARKING_LOT].includes(args.categories[0])
                    && !(args.categories.includes(CAT.GAS_STATION) && args.brand);
            }
        },
        GasNameMissing: class extends ActionFlag {
            static defaultSeverity = SEVERITY.RED;
            static defaultButtonText = 'Yes';
            static defaultButtonTooltip = 'Use gas brand as station name';

            get message() { return `Name is missing. Use "${this.args.brand}"?`; }

            static venueIsFlaggable(args) {
                return args.categories.includes(CAT.GAS_STATION)
                    && isNullOrWhitespace(args.nameBase)
                    && !isNullOrWhitespace(args.brand);
            }

            action() {
                addUpdateAction(this.args.venue, { name: this.args.brand }, null, true);
            }
        },
        ClearThisUrl: class extends FlagBase {
            static defaultSeverity = SEVERITY.YELLOW;

            // Use this to highlight yellow any venues that have an invalid value and will be
            // auto-corrected when WMEPH is run.
            static venueIsFlaggable(args) {
                return args.categories.includes(CAT.CHARGING_STATION)
                    && args.url
                    && ['https://www.nissan-europe.com/', 'https://www.eco-movement.com/'].includes(args.url);
            }
        },
        ClearThisPhone: class extends FlagBase {
            static defaultSeverity = SEVERITY.YELLOW;

            // Use this to highlight yellow any venues that have an invalid value and will be
            // auto-corrected when WMEPH is run.
            static venueIsFlaggable(args) {
                return args.categories.includes(CAT.CHARGING_STATION)
                    && args.phone === '+33-1-72676914'; // Nissan Europe ph#
            }
        },
        PlaIsPublic: class extends FlagBase {
            static get defaultMessage() {
                // Add the buttons to the message.
                let msg = '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>';
                msg += [
                    ['RESTRICTED', 'Restricted'],
                    ['PRIVATE', 'Private']
                ].map(
                    btnInfo => $('<button>', { class: 'wmeph-pla-lot-type-btn btn btn-default btn-xs wmeph-btn', 'data-lot-type': btnInfo[0] })
                        .text(btnInfo[1])
                        .prop('outerHTML')
                ).join('');
                return msg;
            }

            static venueIsFlaggable(args) {
                return args.categories.includes(CAT.PARKING_LOT)
                    && args.venue.attributes.categoryAttributes?.PARKING_LOT?.parkingType === 'PUBLIC';
            }

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

            static venueIsFlaggable(args) {
                return args.categories.includes(CAT.PARKING_LOT)
                    && (!args.nameBase?.replace(/[^A-Za-z0-9]/g, '').length)
                    && args.venue.attributes.lockRank < 2;
            }
        },
        PlaNameNonStandard: class extends WLFlag {
            static defaultSeverity = SEVERITY.YELLOW;
            static defaultMessage = 'Parking lot names typically contain words like "Parking", "Lot", and/or "Garage"';
            static WL_KEY = 'plaNameNonStandard';
            static defaultWLTooltip = 'Whitelist non-standard PLA name';

            static venueIsFlaggable(args) {
                if (!this.isWhitelisted(args) && args.venue.isParkingLot()) {
                    const name = args.venue.getName();
                    if (name) {
                        const state = args.venue.getAddress().getStateName();
                        const re = state === 'Quebec' ? /\b(parking|stationnement)\b/i : /\b((park[ -](and|&|'?n'?)[ -]ride)|parking|lot|garage|ramp)\b/i;
                        if (!re.test(name)) {
                            return true;
                        }
                    }
                }
                return false;
            }
        },
        IndianaLiquorStoreHours: class extends WLFlag {
            static defaultMessage = '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.';

            static WL_KEY = 'indianaLiquorStoreHours';
            static defaultWLTooltip = 'Whitelist Indiana liquor store hours';

            static venueIsFlaggable(args) {
                return !args.highlightOnly && !this.isWhitelisted(args)
                    && !args.categories.includes(CAT.RESIDENCE_HOME)
                    && args.addr?.state.getName() === 'Indiana'
                    && /\b(beers?|wines?|liquors?|spirits)\b/i.test(args.nameBase)
                    && !args.openingHours.some(entry => entry.days.includes(0));
            }
        },
        HoursOverlap: class extends FlagBase {
            static defaultSeverity = SEVERITY.RED;
            static defaultMessage = 'Overlapping hours of operation. Place might not save.';

            static venueIsFlaggable(args) {
                return args.hoursOverlap;
            }
        },
        UnmappedRegion: class extends WLFlag {
            static WL_KEY = 'unmappedRegion';
            static defaultWLTooltip = 'Whitelist unmapped category';
            static #regionsToFlagOther = ['HI', 'NER', 'NOR', 'NWR', 'PLN', 'ATR'];

            get noLock() {
                return Flag.UnmappedRegion.#getRareCategoryInfos(this.args)
                    .some(categoryInfo => (categoryInfo.id === CAT.OTHER
                        && Flag.UnmappedRegion.#regionsToFlagOther.includes(this.args.region)
                        && !this.args.isLocked)
                            || !Flag.UnmappedRegion.isWhitelisted(this.args));
            }

            constructor(args) {
                let showWL = true;
                let severity = SEVERITY.GREEN;
                // let noLock = false;
                let message;
                const categoryNames = [];
                let addOtherMessage = false;

                Flag.UnmappedRegion.#getRareCategoryInfos(args)
                    .forEach(categoryInfo => {
                        if (categoryInfo.id === CAT.OTHER) {
                            if (Flag.UnmappedRegion.#regionsToFlagOther.includes(args.region) && !args.isLocked) {
                                addOtherMessage = true;
                                severity = Math.max(severity, SEVERITY.BLUE);
                                showWL = false;
                                // noLock = true;
                            }
                        } else {
                            if (Flag.UnmappedRegion.isWhitelisted(args)) {
                                showWL = false;
                                severity = Math.max(severity, SEVERITY.GREEN);
                            } else {
                                severity = SEVERITY.YELLOW;
                                // noLock = true;
                            }
                            if (!args.highlightOnly) categoryNames.push(categoryInfo.name);
                        }
                    });
                if (!args.highlightOnly) {
                    const messages = [];
                    if (categoryNames.length === 1) {
                        messages.push(`The <b>${categoryNames[0]}</b> category is usually not mapped in this region.`);
                    } else if (categoryNames.length > 1) {
                        messages.push(`These categories are usually not mapped in this region: ${categoryNames.map(name => `<b>${name}</b>`).join(', ')}`);
                    }
                    if (addOtherMessage) {
                        messages.push('The <b>Other</b> category should only be used if no other category applies. '
                            + 'Manually lock the place to override this flag.');
                    }
                    message = messages.join('<br><br>');
                }
                super();
                this.message = message;
                this.severity = severity;
                // this.noLock = noLock;
                this.showWL = showWL;
            }

            static venueIsFlaggable(args) {
                return !args.categories.includes(CAT.REST_AREAS)
                    && !!this.#getRareCategoryInfos(args).length;
            }

            static #getRareCategoryInfos(args) {
                return args.categories
                    .map(cat => args.pnhCategoryInfos.getById(cat))
                    .filter(pnhCategoryInfo => {
                        const rareLocalities = pnhCategoryInfo.rare;
                        if (rareLocalities.includes(args.state2L) || rareLocalities.includes(args.region) || rareLocalities.includes(args.countryCode)) {
                            if (pnhCategoryInfo.id === CAT.OTHER && this.#regionsToFlagOther.includes(args.region)) {
                                if (!args.isLocked) {
                                    return true;
                                }
                            } else {
                                return true;
                            }
                        }
                        return false;
                    });
            }
        },
        RestAreaName: class extends WLFlag {
            static defaultSeverity = SEVERITY.RED;
            static defaultMessage = 'Rest area name is out of spec. Use the Rest Area wiki button below to view formats.';
            static WL_KEY = 'restAreaName';
            static defaultWLTooltip = 'Whitelist rest area name';

            static venueIsFlaggable(args) {
                return args.categories.includes(CAT.REST_AREAS) && !/^Rest Area.* - /.test(args.nameBase + (args.nameSuffix ?? ''));
            }
        },
        RestAreaNoTransportation: class extends ActionFlag {
            static defaultSeverity = SEVERITY.YELLOW;
            static defaultMessage = 'Rest areas should not use the Transportation category.';
            static defaultButtonText = 'Remove it?';

            static venueIsFlaggable(args) {
                return args.categories.includes(CAT.REST_AREAS)
                    && args.categories.includes(CAT.TRANSPORTATION);
            }

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

            static venueIsFlaggable(args) {
                return args.categories.includes(CAT.REST_AREAS) && args.categories.includes(CAT.GAS_STATION);
            }
        },
        RestAreaScenic: class extends WLActionFlag {
            static WL_KEY = 'restAreaScenic';
            static defaultWLTooltip = 'Whitelist place';
            static defaultMessage = 'Verify that the "Scenic Overlook" category is appropriate for this rest area. If not: ';
            static defaultButtonText = 'Remove it';
            static defaultButtonTooltip = 'Remove "Scenic Overlook" category.';

            static venueIsFlaggable(args) {
                return !this.isWhitelisted(args)
                    && args.categories.includes(CAT.REST_AREAS)
                    && args.categories.includes(CAT.SCENIC_LOOKOUT_VIEWPOINT);
            }

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

            static venueIsFlaggable(args) {
                return !this.isWhitelisted(args)
                    && !args.categories.includes(CAT.REST_AREAS)
                    && (/rest (?:area|stop)|service plaza/i.test(args.nameBase));
            }

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

            static venueIsFlaggable(args) {
                return !args.highlightOnly && args.categories.includes(CAT.CHARGING_STATION);
            }
        },
        EVCSPriceMissing: class extends FlagBase {
            static defaultSeverity = SEVERITY.BLUE;
            static get defaultMessage() {
                let msg = 'EVCS price: ';
                [['FREE', 'Free', 'Free'], ['FEE', 'Paid', 'Paid']].forEach(btnInfo => {
                    msg += $('<button>', {
                        id: `wmeph_${btnInfo[0]}`,
                        class: 'wmeph-evcs-cost-type-btn btn btn-default btn-xs wmeph-btn',
                        title: btnInfo[2]
                    })
                        .text(btnInfo[1])
                        .css({
                            padding: '3px',
                            height: '20px',
                            lineHeight: '0px',
                            marginRight: '2px',
                            marginBottom: '1px',
                            minWidth: '18px'
                        })
                        .prop('outerHTML');
                });
                return msg;
            }

            constructor() {
                super();
                this.noLock = true;
            }

            static venueIsFlaggable(args) {
                const evcsAttr = args.venue.attributes.categoryAttributes?.CHARGING_STATION;
                return args.categories.includes(CAT.CHARGING_STATION)
                    && (!evcsAttr?.costType || evcsAttr.costType === 'COST_TYPE_UNSPECIFIED');
            }

            postProcess() {
                $('.wmeph-evcs-cost-type-btn').click(evt => {
                    const selectedValue = $(evt.currentTarget).attr('id').replace('wmeph_', '');
                    let attrClone;
                    if (this.args.venue.attributes.categoryAttributes) {
                        attrClone = JSON.parse(JSON.stringify(this.args.venue.attributes.categoryAttributes));
                    } else {
                        attrClone = {};
                    }
                    attrClone.CHARGING_STATION ??= {};
                    attrClone.CHARGING_STATION.costType = selectedValue;
                    addUpdateAction(this.args.venue, { categoryAttributes: attrClone }, null, true);
                    UPDATED_FIELDS.evCostType.updated = true;
                });
            }
        },
        GasMismatch: class extends WLFlag {
            static defaultSeverity = SEVERITY.RED;
            static WL_KEY = 'gasMismatch';
            static defaultWLTooltip = 'Whitelist gas brand / name mismatch';
            static defaultMessage = '<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>';

            static venueIsFlaggable(args) {
                // For gas stations, check to make sure brand exists somewhere in the place name.
                // Remove non - alphanumeric characters first, for more relaxed matching.
                if (args.categories[0] === CAT.GAS_STATION && args.brand) {
                    const compressedName = (args.nameBase + args.nameSuffix ?? '').toUpperCase().replace(/[^a-zA-Z0-9]/g, '');
                    // Some brands may have more than one acceptable name, or the brand listed in WME doesn't match what we want to see in the name.
                    // Ideally, this would be addressed in the PNH spreadsheet somehow, but for now hardcoding is the only option.
                    const compressedBrands = [args.brand.toUpperCase().replace(/[^a-zA-Z0-9]/g, '')];
                    if (args.brand === 'Diamond Gasoline') {
                        compressedBrands.push('DIAMONDOIL');
                    } else if (args.brand === 'Murphy USA') {
                        compressedBrands.push('MURPHY');
                    } else if (args.brand === 'Mercury Fuel') {
                        compressedBrands.push('MERCURY', 'MERCURYPRICECUTTER');
                    } else if (args.brand === 'Carrollfuel') {
                        compressedBrands.push('CARROLLMOTORFUEL', 'CARROLLMOTORFUELS');
                    }
                    if (!compressedBrands.some(compressedBrand => compressedName.includes(compressedBrand))) {
                        return true;
                    }
                }
                return false;
            }
        },
        GasUnbranded: class extends FlagBase {
            //  Unbranded is not used per wiki
            static defaultSeverity = SEVERITY.RED;
            static defaultMessage = '"Unbranded" should not be used for the station brand. Change to the correct brand or delete the brand.';

            static venueIsFlaggable(args) {
                return args.categories.includes(CAT.GAS_STATION)
                    && args.brand === 'Unbranded';
            }
        },
        GasMkPrim: class extends ActionFlag {
            static defaultSeverity = SEVERITY.RED;
            static defaultMessage = 'Gas Station should be the primary category';
            static defaultButtonText = 'Fix';
            static defaultButtonTooltip = 'Make the Gas Station category the primary category.';

            static venueIsFlaggable(args) {
                return args.categories.indexOf(CAT.GAS_STATION) > 0;
            }

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

            static venueIsFlaggable(args) {
                return !args.highlightOnly
                    && args.state2L === 'TN'
                    && args.nameBase.toLowerCase().trim() === 'pilot food mart';
            }

            action() {
                addUpdateAction(this.args.venue, { name: 'Pilot Travel Center' }, null, true);
            }
        },
        HotelMkPrim: class extends WLActionFlag {
            static defaultSeverity = SEVERITY.RED;
            static defaultMessage = 'Hotel category is not first';
            static defaultButtonText = 'Fix';
            static defaultButtonTooltip = 'Make the Hotel category the primary category.';
            static WL_KEY = 'hotelMkPrim';
            static defaultWLTooltip = 'Whitelist hotel as secondary category';

            static venueIsFlaggable(args) {
                return args.priPNHPlaceCat === CAT.HOTEL
                    && args.categories.indexOf(CAT.HOTEL) !== 0;
            }

            action() {
                // Insert/move Hotel category in the first position
                const categories = insertAtIndex(this.args.venue.attributes.categories.slice(), CAT.HOTEL, 0);
                addUpdateAction(this.args.venue, { categories }, null, true);
            }
        },
        ChangeToPetVet: class extends WLActionFlag {
            static defaultSeverity = SEVERITY.RED;
            static defaultMessage = 'Key words suggest this should be a Pet/Veterinarian category. Change?';
            static defaultButtonText = 'Yes';
            static defaultButtonTooltip = 'Change to Pet/Veterinarian Category';
            static WL_KEY = 'changeHMC2PetVet';
            static defaultWLTooltip = 'Whitelist Pet/Vet category';

            static venueIsFlaggable(args) {
                if (!this.isWhitelisted(args)) {
                    const testName = name.toLowerCase().replace(/[^a-z]/g, ' ');
                    const testNameWords = testName.split(' ');
                    if ((args.categories.includes(CAT.HOSPITAL_URGENT_CARE) || args.categories.includes(CAT.DOCTOR_CLINIC))
                        && (containsAny(testNameWords, _animalFullMatch) || _animalPartMatch.some(match => testName.includes(match)))) {
                        return true;
                    }
                }
                return false;
            }

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

            static venueIsFlaggable(args) {
                if (!this.isWhitelisted(args)) {
                    const testName = args.nameBase.toLowerCase().replace(/[^a-z]/g, ' ');
                    const testNameWords = testName.split(' ');

                    if (args.categories.includes(CAT.SCHOOL)
                        && (containsAny(testNameWords, _schoolFullMatch) || _schoolPartMatch.some(match => testName.includes(match)))) {
                        return true;
                    }
                }
                return false;
            }
        },
        PointNotArea: class extends WLActionFlag {
            static defaultButtonText = 'Change to point';
            static defaultButtonTooltip = 'Change to Point Place';
            static WL_KEY = 'pointNotArea';
            static defaultWLTooltip = 'Whitelist point (not area)';

            get message() {
                if (this.args.maxAreaSeverity === SEVERITY.RED) {
                    return 'This category should be a point place.';
                }
                return 'This category is usually a point place, but can be an area in some cases. Verify if area is appropriate.';
            }

            constructor(args) {
                let severity;
                let showWL = true;

                const makeGreen = Flag.PointNotArea.isWhitelisted(args)
                    || args.venue.attributes.lockRank >= args.defaultLockLevel;

                if (makeGreen) {
                    showWL = false;
                    severity = SEVERITY.GREEN;
                } else {
                    severity = args.maxAreaSeverity;
                }

                super();
                this.showWL = showWL;
                this.severity = severity;
            }

            static venueIsFlaggable(args) {
                return !args.venue.isPoint()
                    && (args.categories.includes(CAT.RESIDENCE_HOME) || (args.maxAreaSeverity > SEVERITY.BLUE && !args.categories.includes(CAT.REST_AREAS)));
            }

            action() {
                if (this.args.venue.isResidential()) {
                    // 7/1/2022 - Not sure if this is necessary? Can residence be converted to area? Either way, updateFeatureGeometry function no longer works.
                    // const centroid = venue.geometry.getCentroid();
                    // updateFeatureGeometry(venue, new OpenLayers.Geometry.Point(centroid.x, centroid.y));
                } else {
                    $('wz-checkable-chip.geometry-type-control-point').click();
                }
                harmonizePlaceGo(this.args.venue, 'harmonize'); // Rerun the script to update fields and lock
            }
        },
        AreaNotPoint: class extends WLActionFlag {
            static defaultButtonText = 'Change to area';
            static defaultButtonTooltip = 'Change to Area Place';
            static WL_KEY = 'areaNotPoint';
            static defaultWLTooltip = 'Whitelist area (not point)';
            static #collegeAbbrRegExps;

            get message() {
                if (this.args.maxPointSeverity === SEVERITY.RED) {
                    return 'This category should be an area place.';
                }
                return 'This category is usually an area place, but can be a point in some cases. Verify if point is appropriate.';
            }

            constructor(args) {
                let severity;
                let showWL = true;

                const makeGreen = Flag.AreaNotPoint.isWhitelisted(args)
                    || args.venue.attributes.lockRank >= args.defaultLockLevel
                    || (args.maxPointSeverity === SEVERITY.BLUE && Flag.AreaNotPoint.#hasCollegeInName(args.nameBase));

                if (makeGreen) {
                    showWL = false;
                    severity = SEVERITY.GREEN;
                } else {
                    severity = args.maxPointSeverity;
                }

                super();
                this.severity = severity;
                this.showWL = showWL;
            }

            static venueIsFlaggable(args) {
                return args.venue.isPoint()
                    && (args.maxPointSeverity > SEVERITY.GREEN || args.categories.includes(CAT.REST_AREAS));
            }

            static #hasCollegeInName(name) {
                if (!this.#collegeAbbrRegExps) {
                    this.#collegeAbbrRegExps = COLLEGE_ABBREVIATIONS.map(abbr => new RegExp(`\\b${abbr}\\b`, 'g'));
                }
                return this.#collegeAbbrRegExps.some(re => re.test(name));
            }

            action() {
                const { venue } = this.args;
                W.model.actionManager.add(new UpdateFeatureGeometry(venue, venue.model.venues, venue.attributes.geometry, venue.getPolygonGeometry()));
                harmonizePlaceGo(venue, 'harmonize');
            }
        },
        HnMissing: class extends WLActionFlag {
            static defaultButtonText = 'Add';
            static defaultButtonTooltip = 'Add HN to place';
            static WL_KEY = 'HNWL';
            static defaultWLTooltip = 'Whitelist empty HN';
            static #CATEGORIES_TO_IGNORE = [CAT.BRIDGE, CAT.ISLAND, CAT.FOREST_GROVE, CAT.SEA_LAKE_POOL, CAT.RIVER_STREAM, CAT.CANAL,
                CAT.DAM, CAT.TUNNEL, CAT.JUNCTION_INTERCHANGE];

            static #TEXTBOX_ID = 'WMEPH-HNAdd';
            noBannerAssemble = true;

            get message() {
                let msg = `No HN: <input type="text" id="${Flag.HnMissing.#TEXTBOX_ID}" autocomplete="off" `
                + 'style="font-size:0.85em;width:100px;padding-left:2px;color:#000;" > ';

                if (this.args.categories.includes(CAT.PARKING_LOT) && this.args.venue.attributes.lockRank < 2) {
                    if (USER.rank < 3) {
                        msg += 'Request an R3+ lock to confirm no HN.';
                    } else {
                        msg += 'Lock to R3+ to confirm no HN.';
                    }
                }
                return msg;
            }

            constructor(args) {
                let showWL = true;
                let severity = SEVERITY.RED;
                let noLock = false;
                if (args.state2L === 'PR' || args.categories[0] === CAT.SCENIC_LOOKOUT_VIEWPOINT) {
                    severity = SEVERITY.GREEN;
                    showWL = false;
                } else if (args.categories.includes(CAT.PARKING_LOT)) {
                    showWL = false;
                    if (args.venue.attributes.lockRank < 2) {
                        noLock = true;
                        severity = SEVERITY.BLUE;
                    } else {
                        severity = SEVERITY.GREEN;
                    }
                } else if (Flag.HnMissing.isWhitelisted(args)) {
                    severity = SEVERITY.GREEN;
                    showWL = false;
                } else {
                    noLock = true;
                }

                super();
                this.severity = severity;
                this.showWL = showWL;
                this.noLock = noLock;
            }

            static venueIsFlaggable(args) {
                return args.hasStreet
                    && (!args.currentHN?.replace(/\D/g, ''))
                    && !this.#CATEGORIES_TO_IGNORE.includes(args.categories[0])
                    && !args.categories.includes(CAT.REST_AREAS);
            }

            static #getTextbox() {
                return $(`#${Flag.HnMissing.#TEXTBOX_ID}`);
            }

            action() {
                const newHN = $('#WMEPH-HNAdd').val().replace(/\s+/g, '');
                logDev(newHN);
                const hnTemp = newHN.replace(/[^\d]/g, '');
                const hnTempDash = newHN.replace(/[^\d-]/g, '');
                if (hnTemp > 0 && hnTemp < 1000000) {
                    const action = new UpdateObject(this.args.venue, { houseNumber: hnTempDash });
                    action.wmephDescription = `Changed house # to: ${hnTempDash}`;
                    harmonizePlaceGo(this.args.venue, 'harmonize', [action]); // Rerun the script to update fields and lock
                    UPDATED_FIELDS.address.updated = true;
                } else {
                    Flag.HnMissing.#getTextbox().css({ backgroundColor: '#FDD' }).attr('title', 'Must be a number between 0 and 1000000');
                }
            }

            postProcess() {
                // If pressing enter in the HN entry box, add the HN
                const textbox = Flag.HnMissing.#getTextbox();
                textbox.keyup(evt => {
                    if (evt.keyCode === 13 && textbox.val()) {
                        this.action();
                    }
                });
            }
        },
        HnTooManyDigits: class extends WLFlag {
            static defaultMessage = 'HN contains more than 6 digits. Please verify.';
            static defaultSeverity = SEVERITY.YELLOW;
            static WL_KEY = 'hnTooManyDigits';
            static defaultWLTooltip = 'Whitelist long HN';

            static venueIsFlaggable(args) {
                return !this.isWhitelisted(args)
                    && args.currentHN?.replace(/[^0-9]/g, '').length > 6;
            }
        },
        // 2020-10-5 HN's with letters have been allowed since last year.  Currently, RPPs can be saved with a number
        // followed by up to 4 letters but it's not clear if the app actually searches if only 1, 2, or more letters
        // are present.  Other places can have a more flexible HN (up to 15 characters long, total. A single space between
        // the # and letters. Etc)
        // HnNonStandard: class extends WLFlag {
        //     constructor() {
        //         super(SEVERITY.RED, 'House number is non-standard.', true,
        //             'Whitelist non-standard HN', 'hnNonStandard');
        //     }
        //
        // BELOW IS COPIED FROM harmonizePlaceGo function. To be included in HN flags if enabled again.
        // 2020-10-5 Disabling HN validity checks for now. See the note on the HnNonStandard flag object for more details.
        // if (hasStreet && (!currentHN || currentHN.replace(/\D/g, '').length === 0)) {

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

        //     if (!hnOK) {
        //         _buttonBanner.hnNonStandard = new Flag.HnNonStandard();
        //         if (_wl.hnNonStandard) {
        //             _buttonBanner.hnNonStandard.WLactive = false;
        //             _buttonBanner.hnNonStandard.severity = SEVERITY.GREEN;
        //         } else {
        //             lockOK = false;
        //         }
        //     }
        //     if (updateHNflag) {
        //         _buttonBanner.hnDashRemoved = new Flag.HnDashRemoved();
        //         if (!highlightOnly) {
        //             actions.push(new UpdateObject(venue, { houseNumber: hnTemp }));
        //             _UPDATED_FIELDS.address.updated = true;
        //         } else if (highlightOnly) {
        //             if (venue.attributes.residential) {
        //                 _buttonBanner.hnDashRemoved.severity = SEVERITY.RED;
        //             } else {
        //                 _buttonBanner.hnDashRemoved.severity = SEVERITY.BLUE;
        //             }
        //         }
        //     }
        // }
        //
        // },
        HNRange: class extends WLFlag {
            static defaultMessage = 'House number seems out of range for the street name. Verify.';
            static defaultSeverity = SEVERITY.YELLOW;
            static WL_KEY = 'HNRange';
            static defaultWLTooltip = 'Whitelist HN range';

            static venueIsFlaggable(args) {
                if (!this.isWhitelisted(args) && _dupeHNRangeList.length > 3) {
                    let dhnix;
                    const dupeHNRangeListSorted = [];
                    sortWithIndex(_dupeHNRangeDistList);
                    for (dhnix = 0; dhnix < _dupeHNRangeList.length; dhnix++) {
                        dupeHNRangeListSorted.push(_dupeHNRangeList[_dupeHNRangeDistList.sortIndices[dhnix]]);
                    }
                    // Calculate HN/distance ratio with other venues
                    // var sumHNRatio = 0;
                    const arrayHNRatio = [];
                    for (dhnix = 0; dhnix < dupeHNRangeListSorted.length; dhnix++) {
                        arrayHNRatio.push(Math.abs((parseInt(args.currentHN, 10) - dupeHNRangeListSorted[dhnix]) / _dupeHNRangeDistList[dhnix]));
                    }
                    sortWithIndex(arrayHNRatio);
                    // Examine either the median or the 8th index if length is >16
                    const arrayHNRatioCheckIX = Math.min(Math.round(arrayHNRatio.length / 2), 8);
                    if (arrayHNRatio[arrayHNRatioCheckIX] > 1.4) {
                        // show stats if HN out of range
                        logDev(`HNs: ${dupeHNRangeListSorted}`);
                        logDev(`Distances: ${_dupeHNRangeDistList}`);
                        logDev(`arrayHNRatio: ${arrayHNRatio}`);
                        logDev(`HN Ratio Score: ${arrayHNRatio[Math.round(arrayHNRatio.length / 2)]}`);
                        return true;
                    }
                }
                return false;
            }
        },
        StreetMissing: class extends ActionFlag {
            static defaultSeverity = SEVERITY.RED;
            static defaultMessage = 'No street:';
            static defaultButtonText = 'Edit address';
            static defaultButtonTooltip = 'Edit address to add street.';

            constructor(args) {
                super();
                if (args.categories[0] === CAT.SCENIC_LOOKOUT_VIEWPOINT) {
                    this.severity = SEVERITY.BLUE;
                }
            }

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

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

            constructor(args) {
                super();
                if (args.categories.includes(CAT.RESIDENCE_HOME) && args.highlightOnly) {
                    this.severity = SEVERITY.BLUE;
                }
            }

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

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

                $('.city-name').focus();
            }
        },
        BankType1: class extends FlagBase {
            static defaultSeverity = SEVERITY.RED;
            static defaultMessage = 'Clarify the type of bank: the name has ATM but the primary category is Offices';

            static venueIsFlaggable(args) {
                return (!args.pnhNameRegMatch || (args.pnhNameRegMatch && args.priPNHPlaceCat === CAT.BANK_FINANCIAL && !args.pnhMatchData[args.phSpecCaseIdx].includes('notABank')))
                    && args.categories[0] === CAT.OFFICES
                    && /\batm\b/i.test(name);
            }
        },
        // TODO: Fix if the name has "(ATM)" or " - ATM" or similar. This flag is not currently catching those.
        BankBranch: class extends ActionFlag {
            static defaultSeverity = SEVERITY.BLUE;
            static defaultMessage = 'Is this a bank branch office? ';
            static defaultButtonText = 'Yes';
            static defaultButtonTooltip = 'Is this a bank branch?';

            static venueIsFlaggable(args) {
                let flaggable = false;
                if (!args.priPNHPlaceCat || (args.priPNHPlaceCat === CAT.BANK_FINANCIAL && !args.pnhMatchData[args.phSpecCaseIdx].includes('notABank'))) {
                    const ixBank = args.categories.indexOf(CAT.BANK_FINANCIAL);
                    const ixATM = args.categories.indexOf(CAT.ATM);
                    const ixOffices = args.categories.indexOf(CAT.OFFICES);

                    if (/\batm\b/ig.test(args.nameBase)) {
                        flaggable = ixOffices === 0
                            || (ixBank === -1 && ixATM === -1)
                            || (ixATM === 0 && ixBank > 0)
                            || (ixBank > -1);
                    } else if (ixBank > -1 || ixATM > -1) {
                        flaggable = ixOffices === 0
                            || (ixATM === 0 && ixBank === -1)
                            || (ixBank > 0 && ixATM > 0);
                    } else if (args.priPNHPlaceCat) {
                        flaggable = ixBank === -1 && !(/\bcorporate offices\b/i.test(args.nameSuffix) && ixOffices === 0);
                    }
                }
                return flaggable;
            }

            action() {
                const newAttributes = {};

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

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

                addUpdateAction(this.args.venue, newAttributes, null, true);
            }
        },
        StandaloneATM: class extends ActionFlag {
            static defaultSeverity = SEVERITY.YELLOW;
            static defaultMessage = 'Or is this a standalone ATM? ';
            static defaultButtonText = 'Yes';
            static defaultButtonTooltip = 'Is this a standalone ATM with no bank branch?';

            static venueIsFlaggable(args) {
                let flaggable = false;
                if (!args.priPNHPlaceCat || (args.priPNHPlaceCat === CAT.BANK_FINANCIAL && !args.pnhMatchData[args.phSpecCaseIdx].includes('notABank'))) {
                    const ixBank = args.categories.indexOf(CAT.BANK_FINANCIAL);
                    const ixATM = args.categories.indexOf(CAT.ATM);
                    const ixOffices = args.categories.indexOf(CAT.OFFICES);

                    if (/\batm\b/ig.test(args.nameBase)) {
                        flaggable = ixOffices === 0
                            || (ixBank === -1 && ixATM === -1)
                            || (ixBank > -1);
                    } else if (ixBank > -1 || ixATM > -1) {
                        flaggable = ixOffices === 0
                            || (ixATM === 0 && ixBank === -1)
                            || (ixBank > 0 && ixATM > 0);
                    } else {
                        flaggable = args.priPNHPlaceCat && !(/\bcorporate offices\b/i.test(args.nameSuffix) && ixOffices === 0);
                    }
                }
                return flaggable;
            }

            action() {
                const newAttributes = {};

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

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

                addUpdateAction(this.args.venue, newAttributes, null, true);
            }
        },
        BankCorporate: class extends ActionFlag {
            static defaultSeverity = SEVERITY.BLUE;
            static defaultMessage = 'Or is this the bank\'s corporate offices?';
            static defaultButtonText = 'Yes';
            static defaultButtonTooltip = 'Is this the bank\'s corporate offices?';

            static venueIsFlaggable(args) {
                let flaggable = false;
                if (!args.priPNHPlaceCat) {
                    flaggable = (/\batm\b/ig.test(args.nameBase) && args.categories.indexOf(CAT.OFFICES) === 0);
                } else if (args.priPNHPlaceCat === CAT.BANK_FINANCIAL && !args.pnhMatchData[args.phSpecCaseIdx].includes('notABank')) {
                    flaggable = !containsAny(args.categories, [CAT.BANK_FINANCIAL, CAT.ATM])
                        && !/\bcorporate offices\b/i.test(args.nameSuffix);
                }
                return flaggable;
            }

            action() {
                const newAttributes = {};

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

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

                addUpdateAction(this.args.venue, newAttributes, null, true);
            }
        },
        CatPostOffice: class extends FlagBase {
            static defaultMessage = `The Post Office category is reserved for certain USPS locations. Please be sure to follow <a href="${
                URLS.uspsWiki}" style="color:#3a3a3a;" target="_blank">the guidelines</a>.`;

            static venueIsFlaggable(args) {
                return !args.highlightOnly && args.isUspsPostOffice;
            }
        },
        IgnEdited: class extends FlagBase {
            static defaultSeverity = SEVERITY.YELLOW;
            static defaultMessage = 'Last edited by an IGN editor';

            static venueIsFlaggable(args) {
                let updatedBy;
                return !args.categories.includes(CAT.RESIDENCE_HOME)
                    && (updatedBy = args.venue.attributes.updatedBy)
                    && /^ign_/i.test(W.model.users.getObjectById(updatedBy)?.userName);
            }
        },
        WazeBot: class extends ActionFlag {
            static defaultSeverity = SEVERITY.YELLOW;
            static defaultMessage = 'Edited last by an automated process. Please verify information is correct.';
            static defaultButtonText = 'Nudge';
            static defaultButtonTooltip = 'If no other properties need to be updated, click to nudge the place (force an edit).';
            static #botIds = [105774162, 361008095, 338475699, -1, 107668852];
            static #botNames = [/^waze-maint/i, /^waze3rdparty$/i, /^WazeParking1$/i, /^admin$/i, /^avsus$/i];

            static venueIsFlaggable(args) {
                let flaggable = args.venue.isUnchanged() && !args.categories.includes(CAT.RESIDENCE_HOME);
                if (flaggable) {
                    const lastUpdatedById = args.venue.attributes.updatedBy ?? args.venue.attributes.createdBy;
                    flaggable = this.#botIds.includes(lastUpdatedById);
                    if (!flaggable) {
                        const lastUpdatedByName = W.model.users.getObjectById(lastUpdatedById)?.userName;
                        flaggable = (this.#botNames.some(botName => botName.test(lastUpdatedByName)));
                    }
                }
                return flaggable;
            }

            action() {
                nudgeVenue(this.args.venue);
                harmonizePlaceGo(this.args.venue, 'harmonize');
            }
        },
        ParentCategory: class extends WLFlag {
            static defaultSeverity = SEVERITY.YELLOW;
            static WL_KEY = 'parentCategory';
            static defaultWLTooltip = 'Whitelist parent Category';

            get message() {
                let msg;
                const badCatInfos = this.args.categories
                    .filter(category => Flag.ParentCategory.categoryIsDisallowedParent(category, this.args))
                    .map(category => this.args.pnhCategoryInfos.getById(category));
                if (badCatInfos.length === 1) {
                    msg = `The <b>${badCatInfos[0].name}</b> parent category is usually not mapped in this region.`;
                } else {
                    msg = 'These parent categories are usually not mapped in this region: ';
                    msg += badCatInfos.map(catInfo => `<b>${catInfo.name}</b>`).join(', ');
                }
                return msg;
            }

            static categoryIsDisallowedParent(category, args) {
                const pnhCategoryInfo = args.pnhCategoryInfos.getById(category);
                const localities = pnhCategoryInfo.disallowedParent;
                return localities.includes(args.state2L) || localities.includes(args.region) || localities.includes(args.countryCode);
            }

            static venueIsFlaggable(args) {
                return args.categories.some(category => this.categoryIsDisallowedParent(category, args));
            }
        },
        CheckDescription: class extends FlagBase {
            static defaultSeverity = SEVERITY.YELLOW;
            static defaultMessage = 'Description field already contained info; PNH description was added in front of existing. Check for inconsistency or duplicate info.';

            static venueIsFlaggable(args) {
                return args.descriptionInserted;
            }
        },
        Overlapping: class extends FlagBase {
            static defaultSeverity = SEVERITY.YELLOW;
            static defaultMessage = 'Place points are stacked up.';
        },
        SuspectDesc: class extends WLFlag {
            static defaultSeverity = SEVERITY.YELLOW;
            static defaultMessage = 'Description field might contain copyrighted info.';
            static WL_KEY = 'suspectDesc';
            static defaultWLTooltip = 'Whitelist description';

            static venueIsFlaggable(args) {
                return !args.venue.isResidential()
                    && args.totalSeverity < SEVERITY.RED
                    && !this.isWhitelisted(args)
                    && /(google|yelp)/i.test(args.description);
            }
        },
        ResiTypeName: class extends WLFlag {
            static defaultMessage = 'The place name suggests a residential place or personalized place of work.  Please verify.';
            static WL_KEY = 'resiTypeName';
            static defaultWLTooltip = 'Whitelist Residential-type name';

            constructor(likelyResidential) {
                super();
                if (likelyResidential) this.severity = SEVERITY.YELLOW;
            }

            // TODO: make this a public method and pass the result to args so args can be passed into vanueIsFlaggable
            static #likelyResidentialName(alphaName) {
                return /^((my|mi|moms|dads)?\s*(home|work|office|casa|house))|(mom|dad)$/i.test(alphaName);
            }

            static #possiblyResidentialName(alphaName, categories) {
                return /('?s|my)\s+(house|home|work)/i.test(alphaName)
                    && !containsAny(categories, [CAT.RESTAURANT, CAT.DESSERT, CAT.BAR]);
            }

            static #isPreflaggable(args) {
                return !args.categories.includes(CAT.RESIDENCE_HOME)
                    && !args.pnhNameRegMatch
                    && !this.isWhitelisted(args)
                    && args.totalSeverity < SEVERITY.RED;
            }

            // TODO
            static #venueIsFlaggable(preflaggable, likelyResidential, alphaName, categories) {
                return preflaggable
                    && (likelyResidential || this.#possiblyResidentialName(alphaName, categories));
            }

            static eval(args) {
                const preflaggable = this.#isPreflaggable(args);
                if (preflaggable) {
                    const alphaName = name.replace(/[^A-Z ]/i, ''); // remove non-alpha characters
                    const likelyResidential = this.#likelyResidentialName(alphaName);
                    if (this.#venueIsFlaggable(preflaggable, likelyResidential, alphaName, args.categories)) return new this(likelyResidential);
                }
                return null;
            }
        },
        Mismatch247: class extends FlagBase {
            static defaultSeverity = SEVERITY.YELLOW;
            static defaultMessage = 'Hours of operation listed as open 24hrs but not for all 7 days.';

            static venueIsFlaggable(args) {
                return args.openingHours.length === 1
                    && args.openingHours[0].days.length < 7
                    && /^0?0:00$/.test(args.openingHours[0].fromHour)
                    && (/^0?0:00$/.test(args.openingHours[0].toHour) || args.openingHours[0].toHour === '23:59');
            }
        },
        PhoneInvalid: class extends FlagBase {
            static defaultSeverity = SEVERITY.YELLOW;
            static defaultMessage = 'Phone # is invalid.';

            static venueIsFlaggable(args) {
                if (!args.phone) return false;
                const normalizedPhone = normalizePhone(args.phone, args.outputPhoneFormat);
                return (args.highlightOnly && normalizedPhone !== args.phone)
                    || (!args.highlightOnly && normalizedPhone === BAD_PHONE);
            }
        },
        UrlMismatch: class extends WLActionFlag {
            static defaultSeverity = SEVERITY.BLUE;
            static defaultMessage = 'Existing URL doesn\'t match the suggested PNH URL. Use the Website button below to verify the existing URL is valid. If not:';
            static defaultButtonText = 'Use PNH URL';
            static defaultButtonTooltip = 'Change URL to the PNH standard';
            static WL_KEY = 'longURL';
            static defaultWLTooltip = 'Whitelist existing URL';

            static venueIsFlaggable(args) {
                // for cases where there is an existing URL in the WME place, and there is a PNH url on queue:
                return !isNullOrWhitespace(args.url)
                    && !isNullOrWhitespace(args.pnhUrl)
                    && args.url !== args.pnhUrl
                    && args.pnhUrl !== BAD_URL;
            }

            action() {
                if (!isNullOrWhitespace(this.args.pnhUrl)) {
                    addUpdateAction(this.args.venue, { url: this.args.pnhUrl }, null, true);
                } else {
                    WazeWrap.Alerts.confirm(
                        SCRIPT_NAME,
                        'URL Matching Error!<br>Click OK to report this error',
                        () => { reportError(); },
                        () => { }
                    );
                }
            }
        },
        GasNoBrand: class extends FlagBase {
            static defaultSeverity = SEVERITY.BLUE;

            get message() { return `Lock to L${this.args.levelToLock + 1}+ to verify no gas brand.`; }

            constructor() {
                super();
                this.noLock = true;
            }

            static venueIsFlaggable(args) {
                // If gas station is missing brand, don't flag if place is locked as high as user can lock it.
                return args.categories.includes(CAT.GAS_STATION)
                    && !args.brand
                    && args.venue.attributes.lockRank < args.levelToLock;
            }
        },
        SubFuel: class extends WLFlag {
            static defaultSeverity = SEVERITY.BLUE;
            static defaultMessage = 'Make sure this place is for the gas station itself and not the main store building. Otherwise undo and check the categories.';
            static WL_KEY = 'subFuel';
            static defaultWLTooltip = 'Whitelist no gas brand';

            static venueIsFlaggable(args) {
                return !this.isWhitelisted(args)
                    && args.specCases.includes('subFuel')
                    && !/\bgas(oline)?\b/i.test(args.venue.attributes.name)
                    && !/\bfuel\b/i.test(args.venue.attributes.name);
            }
        },
        AddCommonEVPaymentMethods: class extends WLActionFlag {
            static defaultSeverity = SEVERITY.BLUE;
            static defaultButtonText = 'Add network payment methods';
            static defaultButtonTooltip = 'Please verify first! If any are not needed, click the WL button and manually add any needed payment methods.';
            static WL_KEY = 'addCommonEVPaymentMethods';
            static defaultWLTooltip = 'Whitelist common EV payment types';

            get message() {
                const stationAttr = this.args.venue.attributes.categoryAttributes.CHARGING_STATION;
                const { network } = stationAttr;
                let msg = `These common payment methods for the ${network} network are missing. Verify if they are needed here:`;
                this.originalNetwork = stationAttr.network;
                const translations = I18n.translations[I18n.locale].edit.venue.category_attributes.payment_methods;
                const list = COMMON_EV_PAYMENT_METHODS[network]
                    .filter(method => !stationAttr.paymentMethods?.includes(method))
                    .map(method => `- ${translations[method]}`).join('<br>');
                msg += `<br>${list}<br>`;
                return msg;
            }

            static venueIsFlaggable(args) {
                if (args.categories.includes(CAT.CHARGING_STATION) && !this.isWhitelisted(args)) {
                    const stationAttr = args.venue.attributes.categoryAttributes.CHARGING_STATION;
                    const network = stationAttr?.network;
                    return !!(COMMON_EV_PAYMENT_METHODS[network]?.some(method => !stationAttr.paymentMethods?.includes(method)));
                }
                return false;
            }

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

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

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

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

                UPDATED_FIELDS.evPaymentMethods.updated = true;
                addUpdateAction(this.args.venue, { categoryAttributes: categoryAttrClone }, null, true);
            }
        },
        RemoveUncommonEVPaymentMethods: class extends WLActionFlag {
            static defaultSeverity = SEVERITY.BLUE;
            static defaultButtonText = 'Remove network payment methods';
            static defaultButtonTooltip = 'Please verify first! If any should NOT be removed, click the WL button and manually remove any unneeded payment methods.';
            static WL_KEY = 'removeUncommonEVPaymentMethods';
            static defaultWLTooltip = 'Whitelist uncommon EV payment types';

            get message() {
                const stationAttr = this.args.venue.attributes.categoryAttributes.CHARGING_STATION;
                const { network } = stationAttr;
                let msg = `These payment methods are uncommon for the ${stationAttr.network} network. Verify if they are needed here:`;
                // Store a copy of the network to check if it has changed in the action() function
                this.originalNetwork = stationAttr.network;
                const translations = I18n.translations[I18n.locale].edit.venue.category_attributes.payment_methods;
                const list = stationAttr.paymentMethods
                    ?.filter(method => !COMMON_EV_PAYMENT_METHODS[network]?.includes(method))
                    .map(method => `- ${translations[method]}`).join('<br>');
                msg += `<br>${list}<br>`;
                return msg;
            }

            static venueIsFlaggable(args) {
                if (args.categories.includes(CAT.CHARGING_STATION) && !this.isWhitelisted(args)) {
                    const stationAttr = args.venue.attributes.categoryAttributes.CHARGING_STATION;
                    const network = stationAttr?.network;
                    return COMMON_EV_PAYMENT_METHODS.hasOwnProperty(network)
                        && !!(stationAttr?.paymentMethods?.some(method => !COMMON_EV_PAYMENT_METHODS[network]?.includes(method)));
                }
                return false;
            }

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

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

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

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

                UPDATED_FIELDS.evPaymentMethods.updated = true;
                addUpdateAction(this.args.venue, { categoryAttributes: categoryAttrClone }, null, true);
            }
        },
        FormatUSPS: class extends FlagBase {
            static defaultSeverity = SEVERITY.YELLOW;
            static defaultMessage = `Name the post office according to this region's <a href="${
                URLS.uspsWiki}" style="color:#3232e6" target="_blank">standards for USPS post offices</a>`;

            static venueIsFlaggable(args) {
                return args.isUspsPostOffice
                    && !this.isNameOk(this.getCleanNameParts(args.nameBase, args.nameSuffix).join(''), args.state2L, args.addr);
            }

            static getCleanNameParts(name, nameSuffix) {
                name = name.trimLeft().replace(/ {2,}/, ' ');
                if (nameSuffix) {
                    nameSuffix = nameSuffix.trimRight().replace(/\bvpo\b/i, 'VPO').replace(/\bcpu\b/i, 'CPU').replace(/ {2,}/, ' ');
                }
                return [name, nameSuffix || ''];
            }

            static isNameOk(name, state2L, addr) {
                return this.#getPostOfficeRegEx(state2L, addr).test(name);
            }

            static #getPostOfficeRegEx(state2L, addr) {
                return state2L === 'KY'
                    || (state2L === 'NY' && ['Queens', 'Bronx', 'Manhattan', 'Brooklyn', 'Staten Island'].includes(addr.city?.attributes.name))
                    ? /^post office \d{5}( [-–](?: cpu| vpo)?(?: [a-z0-9]+){1,})?$/i
                    : /^post office [-–](?: cpu| vpo)?(?: [a-z0-9]+){1,}$/i;
            }
        },
        MissingUSPSAlt: class extends ActionFlag {
            static defaultSeverity = SEVERITY.BLUE;
            static defaultMessage = 'USPS post offices must have an alternate name of "USPS".';
            static defaultButtonText = 'Add it';
            static defaultButtonTooltip = 'Add USPS alternate name';

            static venueIsFlaggable(args) {
                return args.isUspsPostOffice
                    && !args.aliases.some(alias => alias.toUpperCase() === 'USPS');
            }

            action() {
                const aliases = this.args.venue.attributes.aliases.slice();
                if (!aliases.some(alias => alias === 'USPS')) {
                    aliases.push('USPS');
                    addUpdateAction(this.args.venue, { aliases }, null, true);
                } else {
                    harmonizePlaceGo(this.args.venue, 'harmonize');
                }
            }
        },
        MissingUSPSZipAlt: class extends WLActionFlag {
            static defaultSeverity = SEVERITY.BLUE;
            static defaultMessage = `No <a href="${URLS.uspsWiki}" style="color:#3232e6;" target="_blank">ZIP code alt name</a>: <input type="text" \
id="WMEPH-zipAltNameAdd"autocomplete="off" style="font-size:0.85em;width:65px;padding-left:2px;color:#000;" title="Enter the ZIP code and click Add">`;

            static defaultButtonText = 'Add';
            static WL_KEY = 'missingUSPSZipAlt';
            static defaultWLTooltip = 'Whitelist missing USPS zip alt name';
            static #TEXTBOX_ID = 'WMEPH-zipAltNameAdd';
            noBannerAssemble = true;

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

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

            postProcess() {
                // If pressing enter in the USPS zip code alt entry box...
                const $textbox = $(`#${Flag.MissingUSPSZipAlt.#TEXTBOX_ID}`);
                $textbox.keyup(evt => {
                    if (evt.keyCode === 13 && $(evt.currentTarget).val() !== '') {
                        $('#WMEPH_MissingUSPSZipAlt').click();
                    }
                });

                // Prefill zip code text box
                const zipMatch = (this.args.nameBase + (this.args.nameSuffix ?? '')).match(/\d{5}/);
                if (zipMatch) {
                    $textbox.val(zipMatch[0]);
                }
            }
        },
        MissingUSPSDescription: class extends WLFlag {
            static defaultSeverity = SEVERITY.BLUE;
            static defaultMessage = `The first line of the description for a <a href="${
                URLS.uspsWiki}" style="color:#3232e6" target="_blank">USPS post office</a> must be CITY, STATE(2-letter) ZIP, e.g. "Lexington, KY 40511"`;

            static WL_KEY = 'missingUSPSDescription';
            static defaultWLTooltip = 'Whitelist missing USPS address line in description';

            static venueIsFlaggable(args) {
                if (args.isUspsPostOffice) {
                    const lines = args.description?.split('\n');
                    return !lines?.length
                        || !/^.{2,}, [A-Z]{2}\s{1,2}\d{5}$/.test(lines[0]);
                }
                return false;
            }
        },
        CatHotel: class extends FlagBase {
            constructor(args) {
                const pnhName = args.pnhMatchData[args.phNameIdx];
                super(`Check hotel website for any name localization (e.g. ${pnhName} - Tampa Airport).`);
            }

            static venueIsFlaggable(args) {
                return args.priPNHPlaceCat === CAT.HOTEL
                    && (args.nameBase + (args.nameSuffix || '')).toUpperCase() === args.pnhMatchData[args.phNameIdx].toUpperCase();
            }
        },
        LocalizedName: class extends WLFlag {
            static defaultSeverity = SEVERITY.BLUE;
            static WL_KEY = 'localizedName';
            static defaultWLTooltip = 'Whitelist localization';

            get message() { return this.args.displayNote || 'Place needs localization information'; }

            static venueIsFlaggable(args) {
                return args.localizationRegEx
                    && !args.localizationRegEx.test(args.nameBase + (args.nameSuffix || ''));
            }
        },
        SpecCaseMessage: class extends FlagBase {
            static #teslaSC = /tesla supercharger/i;
            static #teslaDC = /tesla destination charger/i;
            static #rivianAN = /<b>rivian adventure network<\/b> charger/i;
            static #rivianW = /<b>rivian waypoints<\/b> charger/i;

            constructor(args) {
                let message = args.pnhMatchData[args.phDisplayNoteIdx];

                // 3/23/2023 - This is a temporary solution to add a disambiguator for Tesla & Rivian chargers.
                let isRivian = false;
                const isTesla = Flag.SpecCaseMessage.#teslaSC.test(message) && Flag.SpecCaseMessage.#teslaDC.test(message);
                if (isTesla) {
                    message = message.replace(
                        Flag.SpecCaseMessage.#teslaSC,
                        '<button id="wmeph-tesla-supercharger" class="btn wmeph-btn">Tesla SuperCharger</button>'
                    );
                    message = message.replace(
                        Flag.SpecCaseMessage.#teslaDC,
                        '<button id="wmeph-tesla-destination-charger" class="btn wmeph-btn">Tesla Destination Charger</button>'
                    );
                } else {
                    isRivian = Flag.SpecCaseMessage.#rivianAN.test(message) && Flag.SpecCaseMessage.#rivianW.test(message);
                    if (isRivian) {
                        message = message.replace(
                            Flag.SpecCaseMessage.#rivianAN,
                            '<button id="wmeph-rivian-adventure-network" class="btn wmeph-btn">Rivian Adventure Network charger</button>'
                        );
                        message = message.replace(
                            Flag.SpecCaseMessage.#rivianW,
                            '<button id="wmeph-rivian-waypoints" class="btn wmeph-btn">Rivian Waypoints charger</button>'
                        );
                    }
                }

                super();
                this.message = message;

                if (isTesla) {
                    this.postProcess = () => {
                        $('#wmeph-tesla-supercharger').click(() => {
                            addUpdateAction(args.venue, { name: 'Tesla Supercharger' }, null, true);
                        });
                        $('#wmeph-tesla-destination-charger').click(() => {
                            addUpdateAction(args.venue, { name: 'Tesla Destination Charger' }, null, true);
                        });
                    };
                    this.severity = SEVERITY.RED;
                } else if (isRivian) {
                    this.postProcess = () => {
                        $('#wmeph-rivian-adventure-network').click(() => {
                            addUpdateAction(args.venue, { name: 'Rivian Adventure Network' }, null, true);
                        });
                        $('#wmeph-rivian-waypoints').click(() => {
                            addUpdateAction(args.venue, { name: 'Rivian Waypoints' }, null, true);
                        });
                    };
                    this.severity = SEVERITY.RED;
                }
            }

            static venueIsFlaggable(args) {
                const message = args.pnhMatchData[args.phDisplayNoteIdx];
                if (args.showDispNote && !isNullOrWhitespace(message)) {
                    if (args.specialCases.pharmhours) {
                        if (!args.description.toUpperCase().includes('PHARMACY') || (!args.description.toUpperCase().includes('HOURS')
                            && !args.description.toUpperCase().includes('HRS'))) {
                            return true;
                        }
                    } else if (args.specialCases.drivethruhours) {
                        if (!args.description.toUpperCase().includes('DRIVE') || (!args.description.toUpperCase().includes('HOURS')
                            && !args.description.toUpperCase().includes('HRS'))) {
                            if ($('#service-checkbox-DRIVETHROUGH').prop('checked')) {
                                return true;
                            }
                        }
                    } else {
                        return true;
                    }
                }
                return false;
            }
        },
        PnhCatMess: class extends ActionFlag {
            constructor(venue, pnhCategoryInfo, categories) {
                super();
                this.message = pnhCategoryInfo.message;
                if (categories.includes(CAT.HOSPITAL_URGENT_CARE)) {
                    this.buttonText = 'Change to Doctor/Clinic';
                    this.actionType = 'changeToDoctorClinic';
                }
                this.venue = venue;
            }

            static #venueIsFlaggable(highlightOnly, pnhCategoryInfo) {
                return !highlightOnly && !isNullOrWhitespace(pnhCategoryInfo.message);
            }

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

            action() {
                if (this.actionType === 'changeToDoctorClinic') {
                    const categories = _.uniq(this.venue.attributes.categories.slice());
                    const indexOfHospital = categories.indexOf(CAT.HOSPITAL_URGENT_CARE);
                    if (indexOfHospital > -1) {
                        categories[indexOfHospital] = CAT.DOCTOR_CLINIC;
                        addUpdateAction(this.venue, { categories }, null, true);
                    }
                }
            }
        },
        ExtProviderMissing: class extends ActionFlag {
            static defaultButtonTooltip = 'If no other properties need to be updated, click to nudge the place (force an edit).';
            static #categoriesToIgnore = [CAT.BRIDGE, CAT.TUNNEL, CAT.JUNCTION_INTERCHANGE, CAT.NATURAL_FEATURES, CAT.ISLAND,
                CAT.SEA_LAKE_POOL, CAT.RIVER_STREAM, CAT.CANAL, CAT.SWAMP_MARSH];

            get message() {
                let msg = 'No Google link';
                msg += this.makeRed()
                    ? ' and place has not been edited for over 6 months. Edit a property (or nudge) and save to reset the 6 month timer: '
                    : ': ';
                return msg;
            }

            get severity() { return this.makeRed() ? SEVERITY.RED : super.severity; }
            set severity(value) { super.severity = value; }

            get buttonText() { return this.makeRed() ? 'Nudge' : ''; }

            set buttonText(value) { super.buttonText = value; }

            constructor() {
                super();
                this.value2 = 'Add';
                this.title2 = 'Add a link to a Google place';
            }

            makeRed() {
                const { venue } = this.args;
                if (this.args.isLocked) {
                    let lastUpdated;
                    if (venue.isNew()) {
                        lastUpdated = Date.now();
                    } else if (venue.attributes.updatedOn) {
                        lastUpdated = venue.attributes.updatedOn;
                    } else {
                        lastUpdated = venue.attributes.createdOn;
                    }
                    const weeksSinceLastUpdate = (Date.now() - lastUpdated) / 604800000;
                    if (weeksSinceLastUpdate >= 26 && !venue.isUpdated() && (!this.args.actions || this.args.actions.length === 0)) {
                        return true;
                    }
                }
                return false;
            }

            static venueIsFlaggable(args) {
                if (USER.rank >= 2 && args.venue.areExternalProvidersEditable() && !(args.categories.includes(CAT.PARKING_LOT) && args.ignoreParkingLots)) {
                    if (!args.categories.some(cat => this.#categoriesToIgnore.includes(cat))) {
                        const provIDs = args.venue.attributes.externalProviderIDs;
                        if (!(provIDs && provIDs.length)) {
                            return true;
                        }
                    }
                }
                return false;
            }

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

            action2() {
                clickGeneralTab();
                const venueName = this.args.venue.attributes.name;
                $('wz-button.external-provider-add-new').click();
                setTimeout(() => {
                    clickGeneralTab();
                    setTimeout(() => {
                        const elem = document.querySelector('div.external-provider-edit-form wz-autocomplete').shadowRoot.querySelector('wz-text-input').shadowRoot.querySelector('input');
                        elem.focus();
                        elem.value = venueName;
                        elem.dispatchEvent(new Event('input', { bubbles: true })); // NOTE: jquery trigger('input') and other event calls did not work.
                    }, 100);
                }, 100);
            }

            preProcess() {
                // If no Google link and severity would otherwise allow locking, ask if user wants to lock anyway.
                const { args } = this;
                if (!args.isLocked && this.severity <= SEVERITY.YELLOW) {
                    this.severity = SEVERITY.RED;
                    args.totalSeverity = SEVERITY.RED;
                    if (args.lockOK) {
                        this.buttonText = `Lock anyway? (${args.levelToLock + 1})`;
                        this.buttonTooltip = '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.';
                        this.action = () => {
                            addUpdateAction(args.venue, { lockRank: args.levelToLock }, null, true);
                        };
                    }
                }
            }
        },
        UrlMissing: class extends WLActionFlag {
            static defaultSeverity = SEVERITY.BLUE;
            static get defaultMessage() {
                return `No URL: <input type="text" id="${Flag.UrlMissing.#TEXTBOX_ID}" autocomplete="off"`
                    + ' style="font-size:0.85em;width:100px;padding-left:2px;color:#000;">';
            }

            static defaultButtonText = 'Add';
            static defaultButtonTooltip = 'Add URL to place';
            static WL_KEY = 'urlWL';
            static defaultWLTooltip = 'Whitelist empty URL';
            static #TEXTBOX_ID = 'WMEPH-UrlAdd';
            noBannerAssemble = true;

            static isWhitelisted(args) {
                return super.isWhitelisted(args)
                    || (args.venue.isParkingLot() && !this.#venueHasOperator(args.venue))
                    || PRIMARY_CATS_TO_FLAG_GREEN_MISSING_PHONE_URL.includes(args.categories[0])
                    || ANY_CATS_TO_FLAG_GREEN_MISSING_PHONE_URL.some(category => args.categories.includes(category));
            }

            static venueIsFlaggable(args) {
                return !args.url?.trim().length
                    && (!args.venue.isParkingLot()
                        || (args.venue.isParkingLot() && (REGIONS_THAT_WANT_PLA_PHONE_URL.includes(args.region) || this.#venueHasOperator(args.venue))))
                    && !PRIMARY_CATS_TO_IGNORE_MISSING_PHONE_URL.includes(args.categories[0]);
            }

            static #venueHasOperator(venue) {
                return venue.attributes.brand && W.model.categoryBrands.PARKING_LOT.includes(venue.attributes.brand);
            }

            static #getTextbox() {
                return $(`#${Flag.UrlMissing.#TEXTBOX_ID}`);
            }

            action() {
                const $textbox = Flag.UrlMissing.#getTextbox();
                const newUrl = normalizeURL($textbox.val());
                if ((!newUrl?.trim().length) || newUrl === BAD_URL) {
                    $textbox.css({ backgroundColor: '#FDD' }).attr('title', 'Invalid URL format');
                } else {
                    logDev(newUrl);
                    addUpdateAction(this.args.venue, { url: newUrl }, null, true);
                }
            }

            postProcess() {
                // If pressing enter in the URL entry box, add the URL
                const textbox = Flag.UrlMissing.#getTextbox();
                textbox.keyup(evt => {
                    if (evt.keyCode === 13 && textbox.val() !== '') {
                        this.action();
                    }
                });
            }
        },
        InvalidUrl: class extends WLFlag {
            static defaultSeverity = SEVERITY.YELLOW;
            static defaultMessage = 'URL appears to be invalid.';
            static WL_KEY = 'invalidUrl';
            static defaultWLTooltip = 'Whitelist bad URL';

            static venueIsFlaggable(args) {
                return args.normalizedUrl === BAD_URL
                    && !this.isWhitelisted(args);
            }
        },
        BadAreaCode: class extends WLActionFlag {
            static defaultSeverity = SEVERITY.YELLOW;
            static defaultButtonText = 'Update';
            static defaultButtonTooltip = 'Update phone #';
            static WL_KEY = 'aCodeWL';
            static defaultWLTooltip = 'Whitelist the area code';
            noBannerAssemble = true;

            get message() {
                return 'Area Code appears to be invalid for this region:<br><input type="text" id="WMEPH-PhoneAdd" autocomplete="off" '
                + `style="font-size:0.85em;width:100px;padding-left:2px;color:#000;" value="${this.args.phone || ''}">`;
            }

            static venueIsFlaggable(args) {
                return args.phone
                    && !this.isWhitelisted(args)
                    && ['USA', 'CAN'].includes(args.countryCode)
                    && !_areaCodeList.includes(args.phone.match(/[2-9]\d{2}/)?.[0]);
            }

            action() {
                const newPhone = normalizePhone($('#WMEPH-PhoneAdd').val(), this.args.outputPhoneFormat);
                if (newPhone === BAD_PHONE) {
                    $('input#WMEPH-PhoneAdd').css({ backgroundColor: '#FDD' }).attr('title', 'Invalid phone # format');
                } else {
                    addUpdateAction(this.args.venue, { phone: newPhone }, null, true);
                }
            }
        },
        AddRecommendedPhone: class extends WLActionFlag {
            static defaultSeverity = SEVERITY.BLUE;
            static defaultButtonText = 'Add';
            static defaultButtonTooltip = 'Add recommended chain phone #';
            static WL_KEY = 'addRecommendedPhone';
            static defaultWLTooltip = 'Whitelist recommended phone #';

            get message() { return `Recommended phone #:<br>${this.args.recommendedPhone}`; }

            static venueIsFlaggable(args) {
                return args.recommendedPhone
                    && !this.isWhitelisted(args)
                    && args.recommendedPhone !== BAD_PHONE
                    && args.recommendedPhone !== normalizePhone(args.phone, args.outputPhoneFormat);
            }

            action() {
                addUpdateAction(this.args.venue, { phone: this.args.recommendedPhone }, null, true);
            }
        },
        PhoneMissing: class extends WLActionFlag {
            static defaultSeverity = SEVERITY.BLUE;
            static defaultMessage = 'No ph#: <input type="text" id="WMEPH-PhoneAdd" autocomplete="off" style="font-size:0.85em;width:100px;padding-left:2px;color:#000;">';
            static defaultButtonText = 'Add';
            static defaultButtonTooltip = 'Add phone to place';
            static WL_KEY = 'phoneWL';
            static defaultWLTooltip = 'Whitelist empty phone';
            noBannerAssemble = true;

            static isWhitelisted(args) {
                return (args.venue.isParkingLot() && !Flag.PhoneMissing.#venueHasOperator(args.venue))
                    || super.isWhitelisted(args)
                    || PRIMARY_CATS_TO_FLAG_GREEN_MISSING_PHONE_URL.includes(args.categories[0])
                    || ANY_CATS_TO_FLAG_GREEN_MISSING_PHONE_URL.some(category => args.categories.includes(category));
            }

            static venueIsFlaggable(args) {
                return !args.phone
                    && !FlagBase.currentFlags.hasFlag(Flag.AddRecommendedPhone)
                    && (!args.venue.isParkingLot()
                        || (args.venue.isParkingLot() && (REGIONS_THAT_WANT_PLA_PHONE_URL.includes(args.region) || this.#venueHasOperator(args.venue))))
                    && !PRIMARY_CATS_TO_IGNORE_MISSING_PHONE_URL.includes(args.categories[0]);
            }

            static #venueHasOperator(venue) {
                return venue.attributes.brand && W.model.categoryBrands.PARKING_LOT.includes(venue.attributes.brand);
            }

            action() {
                const newPhone = normalizePhone($('#WMEPH-PhoneAdd').val(), this.args.outputPhoneFormat);
                if (newPhone === BAD_PHONE || !newPhone) {
                    $('input#WMEPH-PhoneAdd').css({ backgroundColor: '#FDD' }).attr('title', 'Invalid phone # format');
                } else {
                    logDev(newPhone);
                    addUpdateAction(this.args.venue, { phone: newPhone }, null, true);
                }
            }

            // eslint-disable-next-line class-methods-use-this
            postProcess() {
                // TODO: Is this needed???
                // If pressing enter in the phone entry box, add the phone
                $('#WMEPH-PhoneAdd').keyup(evt => {
                    if (evt.keyCode === 13 && $('#WMEPH-PhoneAdd').val() !== '') {
                        $('#WMEPH_PhoneMissing').click();
                        $('#WMEPH_BadAreaCode').click();
                    }
                });
            }
        },
        NoHours: class extends WLFlag {
            static WL_KEY = 'noHours';
            static defaultSeverity = SEVERITY.BLUE;
            static defaultWLTooltip = 'Whitelist "No hours"';

            get message() {
                let msg;
                if (!this.args.openingHours.length) {
                    msg = Flag.NoHours.#getHoursHtml();
                } else {
                    msg = Flag.NoHours.#getHoursHtml(true, isAlwaysOpen(this.args.venue));
                }
                return msg;
            }

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

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

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

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

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

            onAddHoursClick() {
                this.applyHours();
            }

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

            get message() {
                let msg = `Last updated before ${Flag.OldHours.#cutoffDateString}. Verify hours are correct.`;
                if (this.args.venue.isUnchanged()) msg += ' If everything is current, nudge this place and save.';
                return msg;
            }

            get buttonText() {
                return this.args.venue.isUnchanged() ? 'Nudge' : null;
            }

            get severity() {
                return this.args.venue.isUnchanged() ? super.severity : SEVERITY.GREEN;
            }

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

            static #initializeCategoriesToCheck(pnhCategoryInfos) {
                if (!this.#categoriesToCheck) {
                    this.#categoriesToCheck = pnhCategoryInfos
                        .toArray()
                        .filter(pnhCategoryInfo => this.#parentCategoriesToCheck.includes(pnhCategoryInfo.parent))
                        .map(catInfo => catInfo.id);
                    this.#categoriesToCheck.push(...this.#parentCategoriesToCheck);
                }
            }

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

            action() {
                nudgeVenue(this.args.venue);
                harmonizePlaceGo(this.args.venue, 'harmonize');
            }
        },
        PlaLotTypeMissing: class extends FlagBase {
            static defaultSeverity = SEVERITY.RED;
            static get defaultMessage() {
                return `Lot type: ${
                    [['PUBLIC', 'Public'], ['RESTRICTED', 'Restricted'], ['PRIVATE', 'Private']]
                        .map(btnInfo => $('<button>', { class: 'wmeph-pla-lot-type-btn btn btn-default btn-xs wmeph-btn', 'data-lot-type': btnInfo[0] })
                            .text(btnInfo[1])
                            .prop('outerHTML')).join('')
                }`;
            }

            static venueIsFlaggable(args) {
                if (args.categories.includes(CAT.PARKING_LOT)) {
                    const catAttr = args.venue.attributes.categoryAttributes;
                    const parkAttr = catAttr ? catAttr.PARKING_LOT : undefined;
                    if (!parkAttr || !parkAttr.parkingType) {
                        return true;
                    }
                }
                return false;
            }

            postProcess() {
                $('.wmeph-pla-lot-type-btn').click(evt => {
                    const lotType = $(evt.currentTarget).data('lot-type');
                    const categoryAttrClone = JSON.parse(JSON.stringify(this.args.venue.attributes.categoryAttributes));
                    categoryAttrClone.PARKING_LOT = categoryAttrClone.PARKING_LOT ?? {};
                    categoryAttrClone.PARKING_LOT.parkingType = lotType;
                    UPDATED_FIELDS.lotType.updated = true;
                    addUpdateAction(this.args.venue, { categoryAttributes: categoryAttrClone }, null, true);
                });
            }
        },
        PlaCostTypeMissing: class extends FlagBase {
            static defaultSeverity = SEVERITY.BLUE;
            static get defaultMessage() {
                return `Parking cost: ${
                    [['FREE', 'Free', 'Free'], ['LOW', '$', 'Low'], ['MODERATE', '$$', 'Moderate'], ['EXPENSIVE', '$$$', 'Expensive']]
                        .map(btnInfo => $('<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')).join('')
                }`;
            }

            static venueIsFlaggable(args) {
                const parkingAttr = args.venue.attributes.categoryAttributes?.PARKING_LOT;
                return args.categories.includes(CAT.PARKING_LOT)
                    && (!parkingAttr?.costType || parkingAttr.costType === 'UNKNOWN');
            }

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

            static venueIsFlaggable(args) {
                if (args.categories.includes(CAT.PARKING_LOT)) {
                    const catAttr = args.venue.attributes.categoryAttributes;
                    const parkAttr = catAttr ? catAttr.PARKING_LOT : undefined;
                    if (parkAttr && parkAttr.costType && parkAttr.costType !== 'FREE' && parkAttr.costType !== 'UNKNOWN' && (!parkAttr.paymentType || !parkAttr.paymentType.length)) {
                        return true;
                    }
                }
                return false;
            }

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

            static venueIsFlaggable(args) {
                if (args.categories.includes(CAT.PARKING_LOT)) {
                    const catAttr = args.venue.attributes.categoryAttributes;
                    const parkAttr = catAttr ? catAttr.PARKING_LOT : undefined;
                    if (!parkAttr || !parkAttr.lotType || parkAttr.lotType.length === 0) {
                        return true;
                    }
                }
                return false;
            }

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

            static venueIsFlaggable(args) {
                if (!args.highlightOnly && args.categories.includes(CAT.PARKING_LOT)) {
                    const catAttr = args.venue.attributes.categoryAttributes;
                    const parkAttr = catAttr ? catAttr.PARKING_LOT : undefined;
                    if (!parkAttr || !parkAttr.estimatedNumberOfSpots || parkAttr.estimatedNumberOfSpots === 'R_1_TO_10') {
                        return true;
                    }
                }
                return false;
            }
        },
        NoPlaStopPoint: class extends ActionFlag {
            static defaultSeverity = SEVERITY.BLUE;
            static defaultMessage = 'Entry/exit point has not been created.';
            static defaultButtonText = 'Add point';
            static defaultButtonTooltip = 'Add an entry/exit point';

            static venueIsFlaggable(args) {
                return args.categories.includes(CAT.PARKING_LOT)
                    && !args.venue.attributes.entryExitPoints?.length;
            }

            action() {
                $('wz-button.navigation-point-add-new').click();
                harmonizePlaceGo(this.args.venue, 'harmonize');
            }
        },
        PlaStopPointUnmoved: class extends FlagBase {
            static defaultSeverity = SEVERITY.BLUE;
            static defaultMessage = 'Entry/exit point has not been moved.';

            static venueIsFlaggable(args) {
                const attr = args.venue.attributes;
                if (args.venue.isParkingLot() && attr.entryExitPoints && attr.entryExitPoints.length) {
                    const stopPoint = attr.entryExitPoints[0].getPoint();
                    const areaCenter = attr.geometry.getCentroid();
                    if (stopPoint.equals(areaCenter)) {
                        return true;
                    }
                }
                return false;
            }
        },
        PlaCanExitWhileClosed: class extends ActionFlag {
            static defaultMessage = 'Can cars exit when lot is closed? ';
            static defaultButtonText = 'Yes';

            static venueIsFlaggable(args) {
                return !args.highlightOnly
                    && args.categories.includes(CAT.PARKING_LOT)
                    && !args.venue.attributes.categoryAttributes?.PARKING_LOT?.canExitWhileClosed
                    && ($('#WMEPH-ShowPLAExitWhileClosed').prop('checked') || !(args.openingHours.length === 0 || is247Hours(args.openingHours)));
            }

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

            static venueIsFlaggable(args) {
                return !args.highlightOnly
                    && args.categories.includes(CAT.PARKING_LOT)
                    && !(args.venue.attributes.services?.includes('DISABILITY_PARKING'));
            }

            action() {
                const services = this.args.venue.attributes.services?.slice() ?? [];
                services.push('DISABILITY_PARKING');
                addUpdateAction(this.args.venue, { services }, null, true);
                UPDATED_FIELDS.services_DISABILITY_PARKING.updated = true;
            }
        },
        AllDayHoursFixed: class extends FlagBase {
            static defaultSeverity = SEVERITY.YELLOW;
            static defaultMessage = 'Hours were changed from 00:00-23:59 to "All Day"';

            // If highlightOnly, flag place yellow. Running WMEPH on a place will automatically fix the hours, so
            // then this can be green and just display the message.
            get severity() { return this.args.highlightOnly ? super.severity : SEVERITY.GREEN; }

            static venueIsFlaggable(args) {
                return args.almostAllDayHoursEntries.length > 0;
            }
        },
        LocalURL: class extends FlagBase {
            static defaultMessage = 'Some locations for this business have localized URLs, while others use the primary corporate site.'
                + ' Check if a local URL applies to this location.';

            static venueIsFlaggable(args) {
                return args.localUrlRegexString && !(new RegExp(args.localUrlRegexString, 'i')).test(args.url);
            }
        },
        LockRPP: class extends ActionFlag {
            static defaultButtonText = 'Lock';
            static defaultButtonTooltip = 'Lock the residential point';

            get message() {
                let msg = 'Lock at <select id="RPPLockLevel">';
                let ddlSelected = false;
                for (let llix = 1; llix < 6; llix++) {
                    if (llix < USER.rank + 1) {
                        if (!ddlSelected && (this.args.defaultLockLevel === llix - 1 || llix === USER.rank)) {
                            msg += `<option value="${llix}" selected="selected">${llix}</option>`;
                            ddlSelected = true;
                        } else {
                            msg += `<option value="${llix}">${llix}</option>`;
                        }
                    }
                }
                msg += '</select>';
                msg = `Current lock: ${parseInt(this.args.venue.attributes.lockRank, 10) + 1}. ${msg} ?`;
                return msg;
            }

            static venueIsFlaggable(args) {
                // Allow residential point locking by R3+
                return !args.highlightOnly
                    && args.categories.includes(CAT.RESIDENCE_HOME)
                    && (USER.isDevUser || USER.isBetaUser || USER.rank >= 3);
            }

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

                levelToLock -= 1;
                if (this.args.venue.attributes.lockRank !== levelToLock) {
                    addUpdateAction(this.args.venue, { lockRank: levelToLock }, null, true);
                }
            }
        },
        AddAlias: class extends ActionFlag {
            static defaultButtonText = 'Yes';

            get message() { return `Is there a ${this.args.optionalAlias} at this location?`; }
            get buttonTooltip() { return `Add ${this.args.optionalAlias}`; }

            static venueIsFlaggable(args) {
                return args.optionalAlias
                    && !args.aliases.includes(args.optionalAlias);
            }

            action() {
                const attr = this.args.venue.attributes;
                const alias = this.args.optionalAlias;
                let aliases = insertAtIndex(attr.aliases.slice(), alias, 0);
                if (this.args.specCases.includes('altName2Desc') && !attr.description.toUpperCase().includes(alias.toUpperCase())) {
                    const description = `${alias}\n${attr.description}`;
                    addUpdateAction(this.args.venue, { description }, null, false);
                }
                aliases = removeUnnecessaryAliases(name, aliases);
                addUpdateAction(this.args.venue, { aliases }, null, true);
            }
        },
        AddCat2: class extends ActionFlag {
            static defaultButtonText = 'Yes';

            get message() { return `Is there a ${_catTransWaze2Lang[this.altCategory]} at this location?`; }
            get buttonTooltip() { return `Add ${_catTransWaze2Lang[this.altCategory]}`; }

            constructor(venue, altCategory) {
                super();
                this.altCategory = altCategory;
                this.venue = venue;
            }

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

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

            static venueIsFlaggable(args) {
                return args.specialCases.addPharm && !args.categories.includes(CAT.PHARMACY);
            }

            action() {
                const categories = insertAtIndex(this.args.venue.getCategories(), CAT.PHARMACY, 1);
                addUpdateAction(this.args.venue, { categories }, null, true);
            }
        },
        AddSuper: class extends ActionFlag {
            static defaultMessage = 'Does this location have a supermarket?';
            static defaultButtonText = 'Yes';
            static defaultButtonTooltip = 'Add Supermarket category';

            static venueIsFlaggable(args) {
                return args.specialCases.addSuper && !args.categories.includes(CAT.SUPERMARKET_GROCERY);
            }

            action() {
                const categories = insertAtIndex(this.args.venue.getCategories(), CAT.SUPERMARKET_GROCERY, 1);
                addUpdateAction(this.args.venue, { categories }, null, true);
            }
        },
        AppendAMPM: class extends ActionFlag {
            // Only used on the ARCO gas station PNH entry.
            static defaultMessage = 'Is there an ampm at this location?';
            static defaultButtonText = 'Yes';
            static defaultButtonTooltip = 'Add ampm to the place';

            static venueIsFlaggable(args) {
                // No need to check for name/catgory. After the action is run, the name will match the "ARCO ampm"
                // PNH entry, which doesn't have this flag.
                return args.specialCases.appendAMPM;
            }

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

            static venueIsFlaggable(args) {
                let flaggable = false;
                if (args.specialCases.addATM) {
                    flaggable = true;
                } else if (args.pnhMatchData[args.phSpecCaseIdx]?.includes('notABank')) {
                    // do nothing
                } else if (!args.categories.includes(CAT.ATM) && args.categories.includes(CAT.BANK_FINANCIAL)) {
                    if (args.priPNHPlaceCat === CAT.BANK_FINANCIAL) {
                        if ((args.categories.indexOf(CAT.OFFICES) !== 0)) {
                            flaggable = true;
                        }
                    } else {
                        flaggable = true;
                    }
                }
                return flaggable;
            }

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

            static venueIsFlaggable(args) {
                return (args.categories.includes(CAT.GAS_STATION)
                    && !args.categories.includes(CAT.CONVENIENCE_STORE)
                    && !this.currentFlags.hasFlag(Flag.SubFuel)) // Don't flag if already asking if this is really a gas station
                    || args.specialCases.addConvStore;
            }

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

            static venueIsFlaggable(args) {
                return !args.highlightOnly
                    && args.countryCode === 'USA'
                    && !args.venue.isParkingLot()
                    && !args.categories.includes(CAT.POST_OFFICE)
                    && /\bUSP[OS]\b|\bpost(al)?\s+(service|office)\b/i.test(args.nameBase.replace(/[/\-.]/g, ''));
            }

            action() {
                const categories = insertAtIndex(this.args.venue.getCategories(), CAT.POST_OFFICE, 0);
                addUpdateAction(this.args.venue, { categories }, null, true);
            }
        },
        ChangeToHospitalUrgentCare: class extends ActionFlag {
            static defaultMessage = 'If this place provides emergency medical care:';
            static defaultButtonText = 'Change to Hospital / Urgent Care';
            static defaultButtonTooltip = 'Change category to Hospital / Urgent Care';

            static venueIsFlaggable(args) {
                return !args.highlightOnly && args.categories.includes(CAT.DOCTOR_CLINIC);
            }

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

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

            action() {
                let categories = this.args.venue.getCategories().slice();
                let updateIt = false;
                if (categories.length) {
                    const idx = categories.indexOf(CAT.HOSPITAL_URGENT_CARE);
                    if (idx > -1) {
                        categories[idx] = CAT.DOCTOR_CLINIC;
                        updateIt = true;
                    }
                    categories = _.uniq(categories);
                } else {
                    categories.push(CAT.DOCTOR_CLINIC);
                    updateIt = true;
                }
                if (updateIt) {
                    addUpdateAction(this.args.venue, { categories }, null, true);
                } else {
                    harmonizePlaceGo(this.args.venue, 'harmonize');
                }
            }
        },
        ChangeToDoctorClinic: class extends ActionFlag {
            static defaultMessage = 'If this place provides non-emergency medical care: ';
            static defaultButtonText = 'Change to Doctor / Clinic';
            static defaultButtonTooltip = 'Change category to Doctor / Clinic';

            static venueIsFlaggable(args) {
                // 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.
                return !args.highlightOnly
                    && args.venue.attributes.updatedOn < new Date('3/28/2017').getTime()
                    && ((args.categories.includes(CAT.PERSONAL_CARE) && !args.pnhNameRegMatch) || args.categories.includes(CAT.OFFICES));
            }

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

            get message() { return `${this.#titleCaseName}${this.args.nameSuffix || ''}`; }
            get buttonTooltip() { return `Rename to: ${this.#titleCaseName}${this.args.nameSuffix || ''}`; }

            constructor(args) {
                super();
                this.#titleCaseName = titleCase(args.nameBase);
                this.#originalName = args.nameBase + (args.nameSuffix || '');
            }

            static venueIsFlaggable(args) {
                return !args.pnhNameRegMatch && args.nameBase !== titleCase(args.nameBase);
            }

            action() {
                let name = this.args.venue.getName();
                if (name === this.#originalName || this.#confirmChange) {
                    const parts = getNameParts(this.#originalName);
                    name = titleCase(parts.base);
                    if (parts.base !== name) {
                        addUpdateAction(this.args.venue, { name: name + (parts.suffix || '') });
                    }
                    harmonizePlaceGo(this.args.venue, 'harmonize');
                } else {
                    $('button#WMEPH_titleCaseName').text('Are you sure?').after(' The name has changed. This will overwrite the new name.');
                    this.#confirmChange = true;
                }
            }
        },
        SFAliases: class extends FlagBase {
            static defaultMessage = 'Unnecessary aliases were removed.';

            static venueIsFlaggable(args) {
                return args.aliasesRemoved;
            }
        },
        PlaceMatched: class extends FlagBase {
            static defaultMessage = 'Place matched from PNH data.';

            static venueIsFlaggable(args) {
                return args.pnhNameRegMatch;
            }
        },
        PlaceLocked: class extends FlagBase {
            static defaultMessage = 'Place locked.';
            constructor(args) {
                super();

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

            static venueIsFlaggable(args) {
                return args.lockOK && args.totalSeverity < SEVERITY.YELLOW;
            }
        },
        NewPlaceSubmit: class extends ActionFlag {
            static defaultMessage = 'No PNH match. If it\'s a chain: ';
            static defaultButtonText = 'Submit new chain data';
            static defaultButtonTooltip = 'Submit info for a new chain through the linked form';
            #formUrl;

            constructor(args) {
                super();

                // Make PNH submission link
                const encodedName = encodeURIComponent(args.nameBase);
                const encodedPermalink = encodeURIComponent(args.placePL);
                const encodedUrl = encodeURIComponent(args.newUrl?.trim() ?? '');
                const regionSettings = REGION_SETTINGS[args.region];
                let entryValues;
                if (['CA_EN', 'QC'].includes(args.region)) {
                    entryValues = [encodedName, encodedUrl, USER.name, encodedPermalink];
                } else {
                    entryValues = [encodedName, encodedUrl, USER.name + args.gFormState];
                }
                this.#formUrl = regionSettings.getNewChainFormUrl(entryValues);
            }

            static venueIsFlaggable(args) {
                return !args.highlightOnly
                    && args.pnhMatchData[0] === 'NoMatch'
                    && !args.venue.isParkingLot()
                    && !CHAIN_APPROVAL_PRIMARY_CATS_TO_IGNORE.includes(args.categories[0])
                    && !args.categories.includes(CAT.REST_AREAS);
            }

            action() {
                window.open(this.#formUrl);
            }
        },
        ApprovalSubmit: class extends ActionFlag {
            static defaultMessage = 'PNH data exists but is not approved for this region: ';
            static defaultButtonText = 'Request approval';
            static defaultButtonTooltip = 'Request region/country approval of this place';
            #formUrl;

            constructor(args) {
                super();

                const encodedName = encodeURIComponent(args.pnhMatchData[1][0]); // Just do the first match
                const pnhOrderNum = args.pnhMatchData[2].join(',');
                const approvalMessage = `Submitted via WMEPH. PNH order number ${pnhOrderNum}`;
                const encodedPermalink = encodeURIComponent(args.placePL);
                const regionSettings = REGION_SETTINGS[args.region];
                let entryValues;
                if (['CA_EN', 'QC'].includes(args.region)) {
                    entryValues = [encodedName, approvalMessage, USER.name, encodedPermalink];
                } else {
                    entryValues = [encodedName, approvalMessage, USER.name + args.gFormState];
                }
                this.#formUrl = regionSettings.getApproveChainFormUrl(entryValues);
            }

            static venueIsFlaggable(args) {
                return !args.highlightOnly
                    && args.pnhMatchData[0] === 'ApprovalNeeded'
                    && !args.venue.isParkingLot()
                    && !CHAIN_APPROVAL_PRIMARY_CATS_TO_IGNORE.includes(args.categories[0])
                    && !args.categories.includes(CAT.REST_AREAS);
            }

            action() {
                window.open(this.#formUrl);
            }
        },
        LocationFinder: class extends ActionFlag {
            static defaultButtonTooltip = 'Look up details about this location on the chain\'s finder web page.';
            static #USPS_LOCATION_FINDER_URL = 'https://tools.usps.com/find-location.htm';
            #storeFinderUrl;
            #isCustom = false;

            get buttonText() { return `Location Finder${this.isCustom ? ' (L)' : ''}`; }

            constructor(venue, storeFinderUrl, isCustom, addr, state2L, venueGPS) {
                super();
                this.isCustom = isCustom;
                this.venue = venue;
                this.#isCustom = isCustom;
                this.#storeFinderUrl = storeFinderUrl;
                this.#processUrl(venue, addr, state2L, venueGPS);
            }

            static #venueIsFlaggable(highlightOnly, storeFinderUrl) {
                return !highlightOnly && storeFinderUrl;
            }

            // TODO: Can this be put into venueIsFlaggable?
            static eval(args) {
                const isUsps = args.countryCode === 'USA' && !args.categories.includes(CAT.PARKING_LOT) && args.categories.includes(CAT.POST_OFFICE);
                let storeFinderUrl;
                let isCustom = false;
                if (isUsps) {
                    storeFinderUrl = this.#USPS_LOCATION_FINDER_URL;
                } else {
                    let colIndex = args.pnhDataHeaders.indexOf('ph_sfurllocal');
                    storeFinderUrl = args.pnhMatchData[colIndex]?.trim();
                    if (storeFinderUrl) {
                        isCustom = true;
                    } else {
                        colIndex = args.pnhDataHeaders.indexOf('ph_sfurl');
                        storeFinderUrl = args.pnhMatchData[colIndex]?.trim();
                    }
                }

                return this.#venueIsFlaggable(args.highlightOnly, storeFinderUrl)
                    ? new this(args.venue, storeFinderUrl, isCustom, args.addr, args.state2L, args.venueGPS)
                    : null;
            }

            #processUrl(venue, addr, state2L, venueGPS) {
                if (this.#isCustom) {
                    const location = venue.attributes.geometry.getCentroid();
                    const { houseNumber } = venue.attributes;

                    const urlParts = this.#storeFinderUrl.replace(/ /g, '').split('<>');
                    let searchStreet = '';
                    let searchCity = '';
                    let searchState = '';
                    if (typeof addr.street.getName() === 'string') {
                        searchStreet = addr.street.getName();
                    }
                    const searchStreetPlus = searchStreet.replace(/ /g, '+');
                    searchStreet = searchStreet.replace(/ /g, '%20');
                    if (typeof addr.city.getName() === 'string') {
                        searchCity = addr.city.getName();
                    }
                    const searchCityPlus = searchCity.replace(/ /g, '+');
                    searchCity = searchCity.replace(/ /g, '%20');
                    if (typeof addr.state.getName() === 'string') {
                        searchState = addr.state.getName();
                    }
                    const searchStatePlus = searchState.replace(/ /g, '+');
                    searchState = searchState.replace(/ /g, '%20');

                    if (!venueGPS) venueGPS = OpenLayers.Layer.SphericalMercator.inverseMercator(location.x, location.y);
                    this.#storeFinderUrl = '';
                    for (let tlix = 1; tlix < urlParts.length; tlix++) {
                        let part = '';
                        switch (urlParts[tlix]) {
                            case 'ph_streetName':
                                part = searchStreet;
                                break;
                            case 'ph_streetNamePlus':
                                part = searchStreetPlus;
                                break;
                            case 'ph_cityName':
                                part = searchCity;
                                break;
                            case 'ph_cityNamePlus':
                                part = searchCityPlus;
                                break;
                            case 'ph_stateName':
                                part = searchState;
                                break;
                            case 'ph_stateNamePlus':
                                part = searchStatePlus;
                                break;
                            case 'ph_state2L':
                                part = state2L;
                                break;
                            case 'ph_latitudeEW':
                                // customStoreFinderLocalURL = customStoreFinderLocalURL + venueGPS[0];
                                break;
                            case 'ph_longitudeNS':
                                // customStoreFinderLocalURL = customStoreFinderLocalURL + venueGPS[1];
                                break;
                            case 'ph_latitudePM':
                                part = venueGPS.lat;
                                break;
                            case 'ph_longitudePM':
                                part = venueGPS.lon;
                                break;
                            case 'ph_latitudePMBuffMin':
                                part = (venueGPS.lat - 0.025).toString();
                                break;
                            case 'ph_longitudePMBuffMin':
                                part = (venueGPS.lon - 0.025).toString();
                                break;
                            case 'ph_latitudePMBuffMax':
                                part = (venueGPS.lat + 0.025).toString();
                                break;
                            case 'ph_longitudePMBuffMax':
                                part = (venueGPS.lon + 0.025).toString();
                                break;
                            case 'ph_houseNumber':
                                part = houseNumber ?? '';
                                break;
                            default:
                                part = urlParts[tlix];
                        }
                        this.#storeFinderUrl += part;
                    }
                }
                if (!/^https?:\/\//.test(this.#storeFinderUrl)) {
                    this.#storeFinderUrl = `http://${this.#storeFinderUrl}`;
                }
            }

            #openStoreFinderWebsite() {
                if ($('#WMEPH-WebSearchNewTab').prop('checked')) {
                    window.open(this.#storeFinderUrl);
                } else {
                    window.open(this.#storeFinderUrl, SEARCH_RESULTS_WINDOW_NAME, _searchResultsWindowSpecs);
                }
            }

            action() {
                // If the user has 'never' opened a localized store finder URL, then warn them (just once)
                if (localStorage.getItem(SETTING_IDS.sfUrlWarning) === '0' && this.#isCustom) {
                    WazeWrap.Alerts.confirm(
                        SCRIPT_NAME,
                        '***Localized store finder sites often show multiple nearby results. Please make sure you pick the right location.'
                            + '<br>Click OK to agree and continue.',
                        () => {
                            localStorage.setItem(SETTING_IDS.sfUrlWarning, '1'); // prevent future warnings
                            this.#openStoreFinderWebsite();
                        },
                        () => { }
                    );
                    return;
                }
                this.#openStoreFinderWebsite();
            }
        }
    }; // END Flag namespace

    class FlagContainer {
        static #flagOrder = [
            Flag.EVChargingStationWarning,
            Flag.PnhCatMess,
            Flag.NotAHospital,
            Flag.NotASchool,
            Flag.FullAddressInference,
            Flag.NameMissing,
            Flag.GasNameMissing,
            Flag.PlaIsPublic,
            Flag.PlaNameMissing,
            Flag.PlaNameNonStandard,
            Flag.IndianaLiquorStoreHours,
            Flag.HoursOverlap,
            Flag.UnmappedRegion,
            Flag.RestAreaName,
            Flag.RestAreaNoTransportation,
            Flag.RestAreaGas,
            Flag.RestAreaScenic,
            Flag.RestAreaSpec,
            Flag.GasMismatch,
            Flag.GasUnbranded,
            Flag.GasMkPrim,
            Flag.IsThisAPilotTravelCenter,
            Flag.HotelMkPrim,
            Flag.ChangeToPetVet,
            Flag.PointNotArea,
            Flag.AreaNotPoint,
            Flag.HnMissing,
            Flag.HnTooManyDigits,
            Flag.HNRange,
            Flag.StreetMissing,
            Flag.CityMissing,
            Flag.BankType1,
            Flag.BankBranch,
            Flag.StandaloneATM,
            Flag.BankCorporate,
            Flag.CatPostOffice,
            Flag.IgnEdited,
            Flag.WazeBot,
            Flag.ParentCategory,
            Flag.CheckDescription,
            Flag.Overlapping,
            Flag.SuspectDesc,
            Flag.ResiTypeName,
            Flag.PhoneInvalid,
            Flag.UrlMismatch,
            Flag.GasNoBrand,
            Flag.SubFuel,
            Flag.FormatUSPS,
            Flag.MissingUSPSAlt,
            Flag.MissingUSPSZipAlt,
            Flag.MissingUSPSDescription,
            Flag.CatHotel,
            Flag.LocalizedName,
            Flag.SpecCaseMessage,
            Flag.ChangeToDoctorClinic,
            Flag.ExtProviderMissing,
            Flag.AddCommonEVPaymentMethods,
            Flag.RemoveUncommonEVPaymentMethods,
            Flag.UrlMissing,
            Flag.InvalidUrl,
            Flag.AddRecommendedPhone,
            Flag.BadAreaCode,
            Flag.PhoneMissing,
            Flag.OldHours,
            Flag.Mismatch247,
            Flag.NoHours,
            Flag.AllDayHoursFixed,
            Flag.EVCSPriceMissing,
            Flag.PlaLotTypeMissing,
            Flag.PlaCostTypeMissing,
            Flag.PlaPaymentTypeMissing,
            Flag.PlaLotElevationMissing,
            Flag.PlaSpaces,
            Flag.NoPlaStopPoint,
            Flag.PlaStopPointUnmoved,
            Flag.PlaCanExitWhileClosed,
            Flag.PlaHasAccessibleParking,
            Flag.LocalURL,
            Flag.LockRPP,
            Flag.AddAlias,
            Flag.AddCat2,
            Flag.AddPharm,
            Flag.AddSuper,
            Flag.AppendAMPM,
            Flag.AddATM,
            Flag.AddConvStore,
            Flag.IsThisAPostOffice,
            Flag.TitleCaseName,
            Flag.ChangeToHospitalUrgentCare,
            Flag.SFAliases,
            Flag.ClearThisPhone,
            Flag.ClearThisUrl,
            Flag.PlaceMatched,
            Flag.PlaceLocked,
            Flag.NewPlaceSubmit,
            Flag.ApprovalSubmit,
            Flag.LocationFinder
        ];

        static #isIndexed = false;
        #flags = [];

        constructor() {
            FlagContainer.#indexFlags();
        }

        static #indexFlags() {
            if (!this.#isIndexed) {
                let displayIndex = 1;
                this.#flagOrder.forEach(flagClass => (flagClass.displayIndex = displayIndex++));
                this.#isIndexed = true;
            }
        }

        add(flag) {
            if (flag) this.#flags.push(flag);
        }

        remove(flagClass) {
            const idx = this.#flags.indexOf(flagClass);
            if (idx > -1) this.#flags.splice(idx, 1);
        }

        getOrderedFlags() {
            return this.#flags.slice().sort((f1, f2) => {
                const idx1 = f1.constructor.displayIndex;
                const idx2 = f2.constructor.displayIndex;

                if (idx1 > idx2) return 1;
                if (idx1 < idx2) return -1;
                return 0;
            });
        }

        hasFlag(flagClass) {
            return this.#flags.some(flag => flag.constructor === flagClass);
        }
    }

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

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

    function generateNewArgs() {
        return {
            venue: null,
            actions: null,
            highlightOnly: null,
            totalSeverity: SEVERITY.GREEN,
            levelToLock: null,
            lockOK: true,
            isLocked: null,

            // Current venue attributes
            categories: null,
            nameSuffix: null,
            nameBase: null,
            aliases: null,
            description: null,
            url: null,
            phone: null,
            openingHours: null
        };
    }

    // Main script
    function harmonizePlaceGo(venue, useFlag, actions) {
        if (useFlag === 'harmonize') logDev('harmonizePlaceGo: useFlag="harmonize"');

        const venueID = venue.attributes.id;

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

        FlagBase.currentFlags = new FlagContainer();
        const args = generateNewArgs();
        args.venue = venue;
        args.wl = {};
        args.highlightOnly = !useFlag.includes('harmonize');
        args.addr = venue.getAddress();
        args.addr = args.addr.attributes ?? args.addr;
        args.state2L = 'Unknown';
        args.region = 'Unknown';
        args.gFormState = '';
        args.actions = actions;
        args.categories = venue.attributes.categories.slice();
        const nameParts = getNameParts(venue.attributes.name);
        args.nameSuffix = nameParts.suffix;
        args.nameBase = nameParts.base;
        args.aliases = venue.attributes.aliases.slice();
        args.description = venue.attributes.description;
        args.url = venue.attributes.url;
        args.phone = venue.attributes.phone;
        args.openingHours = venue.attributes.openingHours;
        // Set up a variable (newBrand) to contain the brand. When harmonizing, it may be forced to a new value.
        // Other brand flags should use it since it won't be updated on the actual venue until later.
        args.brand = venue.attributes.brand;
        args.showDispNote = true;
        args.hoursOverlap = false;
        args.descriptionInserted = false;
        args.aliasesRemoved = false;
        args.isUspsPostOffice = false;
        args.maxPointSeverity = SEVERITY.GREEN;
        args.maxAreaSeverity = SEVERITY.RED;
        args.specialCases = {
            addPharm: false,
            addSuper: false,
            appendAMPM: false,
            addATM: false,
            addConvStore: false
        };
        args.almostAllDayHoursEntries = [];
        args.defaultLockLevel = LOCK_LEVEL_2;

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

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

            _buttonBanner2 = getButtonBanner2(venue, args.placePL);
            _servicesBanner = getServicesBanner();

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

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

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

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

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

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

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

        // Country restrictions (note that FullAddressInference should guarantee country/state exist if highlightOnly is true)
        if (!args.addr.country || !args.addr.state) {
            WazeWrap.Alerts.error(SCRIPT_NAME, 'Country and/or state could not be determined.  Edit the place address and run WMEPH again.');
            return undefined;
        }

        const countryName = args.addr.country.getName();
        const stateName = args.addr.state.getName();
        if (['United States', 'American Samoa', 'Guam', 'Northern Mariana Islands', 'Puerto Rico', 'Virgin Islands (U.S.)'].includes(countryName)) {
            args.countryCode = 'USA';
        } else if (countryName === 'Canada') {
            args.countryCode = 'CAN';
        } else {
            if (!args.highlightOnly) {
                WazeWrap.Alerts.error(SCRIPT_NAME, `This script is not currently supported in ${countryName}.`);
            }
            return SEVERITY.RED;
        }

        args.pnhCategoryInfos = PNH_DATA[args.countryCode].categoryInfos;

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

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

        if (!args.highlightOnly && args.state2L === 'TN' && args.nameBase.toLowerCase().trim() === 'pilot') {
            // TODO: check what happens here if there's a name suffix.
            args.nameBase = 'Pilot Food Mart';
            addUpdateAction(venue, { name: args.nameBase }, actions);
        }

        // Clear attributes from residential places
        if (venue.attributes.residential) {
            if (!args.highlightOnly) {
                if (!$('#WMEPH-AutoLockRPPs').prop('checked')) {
                    args.lockOK = false;
                }
                if (venue.attributes.name !== '') { // Set the residential place name to the address (to clear any personal info)
                    logDev('Residential Name reset');
                    actions.push(new UpdateObject(venue, { name: '' }));
                    // no field HL
                }
                args.categories = ['RESIDENCE_HOME'];
                if (venue.attributes.description !== null && venue.attributes.description !== '') { // remove any description
                    logDev('Residential description cleared');
                    actions.push(new UpdateObject(venue, { description: null }));
                    // no field HL
                }
                if (venue.attributes.phone !== null && venue.attributes.phone !== '') { // remove any phone info
                    logDev('Residential Phone cleared');
                    actions.push(new UpdateObject(venue, { phone: null }));
                    // no field HL
                }
                if (venue.attributes.url !== null && venue.attributes.url !== '') { // remove any url
                    logDev('Residential URL cleared');
                    actions.push(new UpdateObject(venue, { url: null }));
                    // no field HL
                }
                if (venue.attributes.services.length > 0) {
                    logDev('Residential services cleared');
                    actions.push(new UpdateObject(venue, { services: [] }));
                    // no field HL
                }
            }
        } else if (venue.isParkingLot()
          || (args.nameBase?.trim().length)
          || containsAny(args.categories, CATS_THAT_DONT_NEED_NAMES)) { // for non-residential places
            // Phone formatting
            args.outputPhoneFormat = '({0}) {1}-{2}';
            if (containsAny(['CA', 'CO'], [args.region, args.state2L]) && (/^\d{3}-\d{3}-\d{4}$/.test(venue.attributes.phone))) {
                args.outputPhoneFormat = '{0}-{1}-{2}';
            } else if (args.region === 'SER' && !(/^\(\d{3}\) \d{3}-\d{4}$/.test(venue.attributes.phone))) {
                args.outputPhoneFormat = '{0}-{1}-{2}';
            } else if (args.region === 'GLR') {
                args.outputPhoneFormat = '{0}-{1}-{2}';
            } else if (args.state2L === 'NV') {
                args.outputPhoneFormat = '{0}-{1}-{2}';
            } else if (args.countryCode === 'CAN') {
                args.outputPhoneFormat = '+1-{0}-{1}-{2}';
            }

            args.almostAllDayHoursEntries = args.openingHours.filter(hoursEntry => hoursEntry.toHour === '23:59' && /^0?0:00$/.test(hoursEntry.fromHour));
            if (!args.highlightOnly && args.almostAllDayHoursEntries.length) {
                const newHoursEntries = [];
                args.openingHours.forEach(hoursEntry => {
                    const isInvalid = args.almostAllDayHoursEntries.includes(hoursEntry);
                    const newHoursEntry = new OpeningHour({
                        days: hoursEntry.days.slice(),
                        fromHour: isInvalid ? '00:00' : hoursEntry.fromHour,
                        toHour: isInvalid ? '00:00' : hoursEntry.toHour
                    });
                    newHoursEntries.push(newHoursEntry);
                });
                args.openingHours = newHoursEntries;
                addUpdateAction(venue, { openingHours: args.openingHours }, actions);
            }

            // Place Harmonization
            if (!args.highlightOnly) {
                if (venue.isParkingLot() || venue.isResidential()) {
                    args.pnhMatchData = ['NoMatch'];
                } else {
                    // check against the PNH list
                    args.pnhMatchData = findPnhMatch(args.nameBase, args.state2L, args.region, args.countryCode, args.categories, venue);
                }
            } else {
                args.pnhMatchData = ['Highlight'];
            }

            args.pnhDataHeaders = [];

            args.pnhNameRegMatch = args.pnhMatchData[0] !== 'NoMatch'
                && args.pnhMatchData[0] !== 'ApprovalNeeded'
                && args.pnhMatchData[0] !== 'Highlight';

            if (args.pnhNameRegMatch) { // *** Replace place data with PNH data
                let updatePNHName = true;
                // Break out the data headers
                args.pnhDataHeaders = PNH_DATA[args.countryCode].pnh[0].split('|');
                args.phNameIdx = args.pnhDataHeaders.indexOf('ph_name');
                const phAliasesIdx = args.pnhDataHeaders.indexOf('ph_aliases');
                const phCategory1Idx = args.pnhDataHeaders.indexOf('ph_category1');
                const phCategory2Idx = args.pnhDataHeaders.indexOf('ph_category2');
                const phDescriptionIdx = args.pnhDataHeaders.indexOf('ph_description');
                const phUrlIdx = args.pnhDataHeaders.indexOf('ph_url');
                const phOrderIdx = args.pnhDataHeaders.indexOf('ph_order');
                // var ph_notes_ix = _PNH_DATA_headers.indexOf('ph_notes');
                args.phSpecCaseIdx = args.pnhDataHeaders.indexOf('ph_speccase');
                // var ph_forcecat_ix = _PNH_DATA_headers.indexOf('ph_forcecat');
                args.phDisplayNoteIdx = args.pnhDataHeaders.indexOf('ph_displaynote');

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

                args.priPNHPlaceCat = getCategoryIdFromName(args.pnhMatchData[phCategory1Idx], args.countryCode); // translate primary category to WME code

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

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

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

                        // find any button/message flags in the special case (format: buttOn_xyzXyz, etc.)
                        if (match = specCase.match(/^buttOn_(.+)/i)) {
                            [, scFlag] = match;
                            switch (scFlag) {
                                case 'addCat2':
                                    // flag = new Flag.AddCat2();
                                    break;
                                case 'addPharm':
                                case 'addSuper':
                                case 'appendAMPM':
                                case 'addATM':
                                case 'addConvStore':
                                    args.specialCases[scFlag] = true;
                                    break;
                                default:
                                    console.error('WMEPH:', `Could not process specCase value: buttOn_${scFlag}`);
                            }
                        } else if (match = specCase.match(/^buttOff_(.+)/i)) {
                            [, scFlag] = match;
                            switch (scFlag) {
                                case 'addConvStore':
                                    FlagBase.currentFlags.remove(Flag.AddConvStore);
                                    break;
                                default:
                                    console.error(`WMEPH: Could not process specCase value: buttOff_${scFlag}`);
                            }
                        // } else if (match = specCase.match(/^messOn_(.+)/i)) {
                        //    [, scFlag] = match;
                        //    _buttonBanner[scFlag].active = true;
                        // } else if (match = specCase.match(/^messOff_(.+)/i)) {
                        //    [, scFlag] = match;
                        //    _buttonBanner[scFlag].active = false;
                        } else if (match = specCase.match(/^psOn_(.+)/i)) {
                            [, scFlag] = match;
                            _servicesBanner[scFlag].actionOn(actions);
                            _servicesBanner[scFlag].pnhOverride = true;
                        } else if (match = specCase.match(/^psOff_(.+)/i)) {
                            [, scFlag] = match;
                            _servicesBanner[scFlag].actionOff(actions);
                            _servicesBanner[scFlag].pnhOverride = true;
                        } else if (match = /forceBrand<>([^,<]+)/i.exec(args.pnhMatchData[args.phSpecCaseIdx])) {
                            // If brand is going to be forced, use that.  Otherwise, use existing brand.
                            [, args.brand] = match;
                        } else if (match = specCase.match(/^localURL_(.+)/i)) {
                            // parseout localURL data if exists (meaning place can have a URL distinct from the chain URL
                            [, args.localURLcheck] = match;
                        } else if ([CAT.GAS_STATION].includes(args.priPNHPlaceCat) && (match = specCase.match(/^forceBrand<>(.+)/i))) {
                            // Gas Station forceBranding
                            const [, forceBrand] = match;
                            if (venue.attributes.brand !== forceBrand) {
                                actions.push(new UpdateObject(venue, { brand: forceBrand }));
                                UPDATED_FIELDS.brand.updated = true;
                                logDev('Gas brand updated from PNH');
                            }
                        } else if (match = specCase.match(/^checkLocalization<>(.+)/i)) {
                            args.showDispNote = false;
                            const [, localizationString] = match;
                            args.localizationRegEx = new RegExp(localizationString, 'g');
                        } else if (match = specCase.match(/phone<>(.*?)<>/)) {
                            args.recommendedPhone = normalizePhone(match[0], args.outputPhoneFormat);
                        } else if (/keepName/g.test(specCase)) {
                            // Prevent name change
                            updatePNHName = false;
                        } else if (match = specCase.match(/^optionAltName<>(.+)/i)) {
                            [, args.optionalAlias] = match;
                        }
                        /* eslint-enable no-cond-assign */
                    }
                }
                if (args.phDisplayNoteIdx > -1 && !isNullOrWhitespace(args.pnhMatchData[args.phDisplayNoteIdx])) {
                    args.displayNote = args.pnhMatchData[args.phDisplayNoteIdx];
                }

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

                // name parsing with category exceptions
                if (args.priPNHPlaceCat === CAT.HOTEL) {
                    const nameToCheck = args.nameBase + (args.nameSuffix || '');
                    if (nameToCheck.toUpperCase() === args.pnhMatchData[args.phNameIdx].toUpperCase()) { // If no localization
                        args.nameBase = args.pnhMatchData[args.phNameIdx];
                    } else {
                        // Replace PNH part of name with PNH name
                        const splix = args.nameBase.toUpperCase().replace(/[-/]/g, ' ').indexOf(args.pnhMatchData[args.phNameIdx].toUpperCase().replace(/[-/]/g, ' '));
                        if (splix > -1) {
                            const frontText = args.nameBase.slice(0, splix);
                            const backText = args.nameBase.slice(splix + args.pnhMatchData[args.phNameIdx].length);
                            args.nameBase = args.pnhMatchData[args.phNameIdx];
                            if (frontText.length > 0) { args.nameBase = `${frontText} ${args.nameBase}`; }
                            if (backText.length > 0) { args.nameBase = `${args.nameBase} ${backText}`; }
                            args.nameBase = args.nameBase.replace(/ {2,}/g, ' ');
                        } else {
                            args.nameBase = args.pnhMatchData[args.phNameIdx];
                        }
                    }
                    if (altCategories && altCategories.length) { // if PNH alts exist
                        insertAtIndex(args.categories, altCategories, 1); //  then insert the alts into the existing category array after the GS category
                    }
                    if (args.categories.includes(CAT.HOTEL)) {
                        // Remove LODGING if it exists
                        const lodgingIdx = args.categories.indexOf(CAT.LODGING);
                        if (lodgingIdx > -1) {
                            args.categories.splice(lodgingIdx, 1);
                        }
                    }
                    // If PNH match, set wifi service.
                    if (args.pnhMatchData && !_servicesBanner.addWiFi.checked) {
                        _servicesBanner.addWiFi.action();
                    }
                    // Set hotel hours to 24/7 for all hotels.
                    if (!_servicesBanner.add247.checked) {
                        _servicesBanner.add247.action();
                    }
                } else if (args.priPNHPlaceCat === CAT.BANK_FINANCIAL && !args.pnhMatchData[args.phSpecCaseIdx].includes('notABank')) {
                    if (/\batm\b/ig.test(args.nameBase)) {
                        args.nameBase = `${args.pnhMatchData[args.phNameIdx]} ATM`;
                    } else {
                        args.nameBase = args.pnhMatchData[args.phNameIdx];
                    }
                } else if (args.priPNHPlaceCat === CAT.GAS_STATION) { // for PNH gas stations, don't replace existing sub-categories
                    if (altCategories?.length) { // if PNH alts exist
                        insertAtIndex(args.categories, altCategories, 1); //  then insert the alts into the existing category array after the GS category
                    }
                    args.nameBase = args.pnhMatchData[args.phNameIdx];
                } else if (updatePNHName) { // if not a special category then update the name
                    args.nameBase = args.pnhMatchData[args.phNameIdx];
                    args.categories = insertAtIndex(args.categories, args.priPNHPlaceCat, 0);
                    if (altCategories && altCategories.length && !args.specCases.includes('buttOn_addCat2') && !args.specCases.includes('optionCat2')) {
                        args.categories = insertAtIndex(args.categories, altCategories, 1);
                    }
                } else if (!updatePNHName) {
                    // Strong title case option for non-PNH places
                    Flag.TitleCaseName.eval(venue, args.nameBase, args.nameSuffix);
                }

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

                // Parse URL data
                if (!(args.localURLcheck && args.url && (new RegExp(args.localURLcheck, 'i')).test(args.url))) {
                    args.pnhUrl = normalizeURL(args.pnhMatchData[phUrlIdx]);
                }

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

                // Remove unnecessary parent categories
                const parentCats = _.uniq(args.categories.map(category => args.pnhCategoryInfos.getById(category).parent))
                    .filter(parent => parent.trim().length > 0);
                args.categories = args.categories.filter(cat => !parentCats.includes(cat));

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

                // Description update
                args.description = args.pnhMatchData[phDescriptionIdx];
                if (!isNullOrWhitespace(args.description) && !venue.attributes.description.toUpperCase().includes(args.description.toUpperCase())) {
                    if (!isNullOrWhitespace(venue.attributes.description)) {
                        args.descriptionInserted = true;
                    }
                    logDev('Description updated');
                    args.description = `${args.description}\n${venue.attributes.description}`;
                    actions.push(new UpdateObject(venue, { description: args.description }));
                    UPDATED_FIELDS.description.updated = true;
                }

                // Special Lock by PNH
                if (args.specCases.includes('lockAt5')) {
                    pnhLockLevel = 4;
                }
            } // END PNH match/no-match updates

            const isPoint = venue.isPoint();
            // NOTE: do not use is2D() function. It doesn't seem to be 100% reliable.
            const isArea = !isPoint;
            let highestCategoryLock = -1;
            // Category/Name-based Services, added to any existing services:
            args.categories.forEach(category => {
                const pnhCategoryInfo = args.pnhCategoryInfos.getById(category);

                if (!pnhCategoryInfo) {
                    throw new Error(`WMEPH: Unexpected category: ${category}`);
                }
                let pvaPoint = pnhCategoryInfo.point;
                let pvaArea = pnhCategoryInfo.area;
                if (pnhCategoryInfo.regPoint.includes(args.state2L) || pnhCategoryInfo.regPoint.includes(args.region)
                    || pnhCategoryInfo.regPoint.includes(args.countryCode)) {
                    pvaPoint = '1';
                    pvaArea = '';
                } else if (pnhCategoryInfo.regArea.includes(args.state2L) || pnhCategoryInfo.regArea.includes(args.region)
                    || pnhCategoryInfo.regArea.includes(args.countryCode)) {
                    pvaPoint = '';
                    pvaArea = '1';
                }

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

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

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

                // TODO: Process this flag outside the loop.
                Flag.PnhCatMess.eval(venue, pnhCategoryInfo, args.categories, args.highlightOnly);

                // Set lock level
                for (let lockix = 1; lockix < 6; lockix++) {
                    const categoryLock = pnhCategoryInfo[`lock${lockix}`];
                    if (lockix - 1 > highestCategoryLock && (categoryLock.includes(args.state2L) || categoryLock.includes(args.region)
                        || categoryLock.includes(args.countryCode))) {
                        highestCategoryLock = lockix - 1; // Offset by 1 since lock ranks start at 0
                    }
                }
            });

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

            if (!args.highlightOnly) {
                // Update name:
                if ((args.nameBase + (args.nameSuffix || '')) !== venue.attributes.name) {
                    logDev('Name updated');
                    addUpdateAction(venue, { name: args.nameBase + (args.nameSuffix || '') }, actions);
                }

                // Update aliases
                const tempAliases = removeUnnecessaryAliases(args.nameBase, args.aliases);
                if (tempAliases) {
                    args.aliasesRemoved = true;
                    args.aliases = tempAliases;
                    logDev('Alt Names updated');
                    addUpdateAction(venue, { aliases: args.aliases }, actions);
                }

                // PNH specific Services:
                args.categories.forEach(category => {
                    const pnhCategoryInfo = args.pnhCategoryInfos.getById(category);
                    pnhCategoryInfo.services.forEach(service => {
                        const serviceButton = _servicesBanner[service.pnhKey];
                        if (!serviceButton.pnhOverride) {
                            // This section of code previously checked for values of "1", "2", and state/region codes.
                            // A value of "2" or a state/region code would auto-add the service.  However, it was
                            // felt that this was a problem since it is difficult to prove that every place in a
                            // category would *always* offer a specific service.  So now, any value entered in the
                            // spreadsheet cell will only display the service button, not turn it on.
                            serviceButton.active = true;
                        }
                    });
                });
            }

            args.hoursOverlap = venueHasOverlappingHours(args.openingHours);

            args.isUspsPostOffice = args.countryCode === 'USA' && !args.categories.includes(CAT.PARKING_LOT) && args.categories.includes(CAT.POST_OFFICE);

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

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

                // URL updating
                // Invalid EVCS URL imported from PURs. Clear it.
                if (Flag.ClearThisUrl.venueIsFlaggable(args)) {
                    args.url = null;
                    addUpdateAction(venue, { url: args.url }, actions);
                }
                args.normalizedUrl = normalizeURL(args.url);
                if (args.isUspsPostOffice && args.url !== 'usps.com') {
                    args.url = 'usps.com';
                    addUpdateAction(venue, { url: args.url }, actions);
                } else if (!args.pnhUrl && args.normalizedUrl !== args.url) {
                    if (args.normalizedUrl !== BAD_URL) {
                        args.url = args.normalizedUrl;
                        logDev('URL formatted');
                        addUpdateAction(venue, { url: args.url }, actions);
                    }
                } else if (args.pnhUrl && isNullOrWhitespace(args.url)) {
                    args.url = args.pnhUrl;
                    logDev('URL updated');
                    addUpdateAction(venue, { url: args.url }, actions);
                }

                if (args.phone) {
                    // Invalid EVCS phone # imported from PURs. Clear it.
                    if (Flag.ClearThisPhone.venueIsFlaggable(args)) {
                        args.phone = null;
                    }
                    const normalizedPhone = normalizePhone(args.phone, args.outputPhoneFormat);
                    if (normalizedPhone !== BAD_PHONE) args.phone = normalizedPhone;
                    if (args.phone !== venue.attributes.phone) {
                        logDev('Phone updated');
                        addUpdateAction(venue, { phone: args.phone }, actions);
                    }
                }

                if (args.isUspsPostOffice) {
                    const cleanNameParts = Flag.FormatUSPS.getCleanNameParts(args.nameBase, args.nameSuffix);
                    const nameToCheck = cleanNameParts.join('');
                    if (Flag.FormatUSPS.isNameOk(nameToCheck, args.state2L, args.addr)) {
                        if (nameToCheck !== venue.attributes.name) {
                            [args.nameBase, args.nameSuffix] = cleanNameParts;
                            actions.push(new UpdateObject(venue, { name: nameToCheck }));
                        }
                    }
                }
            }
        } // END if (!residential && has name)

        if (!args.highlightOnly && args.categories.includes(CAT.REST_AREAS)) {
            const oldName = venue.attributes.name;
            if (oldName.match(/^Rest Area.* - /) !== null) {
                const newSuffix = args.nameSuffix.replace(/\bMile\b/i, 'mile');
                if (args.nameBase + newSuffix !== venue.attributes.name) {
                    addUpdateAction(venue, { name: args.nameBase + newSuffix }, actions);
                    logDev('Lower case "mile"');
                }
                // NOTE: I don't know if this else case is needed anymore...
                // else {
                //     // The new name matches the original name, so the only change would have been to capitalize "Mile", which
                //     // we don't want. So remove any previous name-change action.  Note: this feels like a hack and is probably
                //     // a fragile workaround.  The name shouldn't be capitalized in the first place, unless necessary.
                //     for (let i = 0; i < actions.length; i++) {
                //         const action = actions[i];
                //         if (action.newAttributes?.name) {
                //             actions.splice(i, 1);
                //             _UPDATED_FIELDS.name.updated = false;
                //             break;
                //         }
                //     }
                // }
            }

            // switch to rest area wiki button
            _buttonBanner2.restAreaWiki.active = true;
            _buttonBanner2.placesWiki.active = false;
        }

        args.isLocked = venue.attributes.lockRank >= (pnhLockLevel > -1 ? pnhLockLevel : args.defaultLockLevel);
        args.currentHN = venue.attributes.houseNumber;
        // Check to see if there's an action that is currently updating the house number.
        const updateHnAction = actions && actions.find(action => action.newAttributes && action.newAttributes.houseNumber);
        if (updateHnAction) args.currentHN = updateHnAction.newAttributes.houseNumber;
        // Use the inferred address street if currently no street.
        args.hasStreet = venue.attributes.streetID || (inferredAddress && inferredAddress.street);
        args.ignoreParkingLots = $('#WMEPH-DisablePLAExtProviderCheck').prop('checked');

        if (!venue.isResidential() && (venue.isParkingLot() || (args.nameBase?.trim().length))) {
            if (args.pnhNameRegMatch) {
                Flag.HotelMkPrim.eval(args);
                Flag.LocalizedName.eval(args);
                Flag.AddAlias.eval(args);
                Flag.AddRecommendedPhone.eval(args);
                Flag.SubFuel.eval(args);
                Flag.SpecCaseMessage.eval(args);
                Flag.LocalURL.eval(args);
                Flag.UrlMismatch.eval(args);
                Flag.CheckDescription.eval(args);
                Flag.LocationFinder.eval(args);
                Flag.AddPharm.eval(args);
                Flag.AddSuper.eval(args);
                Flag.AppendAMPM.eval(args);
                Flag.PlaceMatched.eval(args);
            }
            Flag.InvalidUrl.eval(args);
            Flag.SFAliases.eval(args);
            Flag.CatHotel.eval(args);
            Flag.ExtProviderMissing.eval(args);
            Flag.NewPlaceSubmit.eval(args);
            Flag.ApprovalSubmit.eval(args);
            Flag.TitleCaseName.eval(args);
            Flag.BankType1.eval(args);
            Flag.BankBranch.eval(args);
            Flag.StandaloneATM.eval(args);
            Flag.BankCorporate.eval(args);
            Flag.AddATM.eval(args);
            Flag.NoHours.eval(args);
            Flag.Mismatch247.eval(args);
            Flag.HoursOverlap.eval(args);
            Flag.OldHours.eval(args);
            Flag.AllDayHoursFixed.eval(args);
            Flag.IsThisAPostOffice.eval(args);
            Flag.MissingUSPSZipAlt.eval(args);
            Flag.FormatUSPS.eval(args);
            Flag.CatPostOffice.eval(args);
            Flag.MissingUSPSDescription.eval(args);
            Flag.MissingUSPSAlt.eval(args);
            Flag.UrlMissing.eval(args);
            Flag.PhoneInvalid.eval(args);
            Flag.PhoneMissing.eval(args);
            Flag.BadAreaCode.eval(args);
            Flag.ParentCategory.eval(args);
            Flag.ClearThisPhone.eval(args);
            Flag.ClearThisUrl.eval(args);
        }
        Flag.UnmappedRegion.eval(args);
        Flag.PlaCostTypeMissing.eval(args);
        Flag.PlaLotElevationMissing.eval(args);
        Flag.PlaSpaces.eval(args);
        Flag.PlaLotTypeMissing.eval(args);
        Flag.NoPlaStopPoint.eval(args);
        Flag.PlaStopPointUnmoved.eval(args);
        Flag.PlaCanExitWhileClosed.eval(args);
        Flag.PlaPaymentTypeMissing.eval(args);
        Flag.PlaHasAccessibleParking.eval(args);
        Flag.ChangeToHospitalUrgentCare.eval(args);
        Flag.IsThisAPilotTravelCenter.eval(args);
        Flag.GasMkPrim.eval(args);
        Flag.AddConvStore.eval(args);
        Flag.IndianaLiquorStoreHours.eval(args);
        Flag.PointNotArea.eval(args);
        Flag.GasMismatch.eval(args);
        Flag.EVChargingStationWarning.eval(args);
        Flag.AddCommonEVPaymentMethods.eval(args);
        Flag.RemoveUncommonEVPaymentMethods.eval(args);
        Flag.EVCSPriceMissing.eval(args);
        Flag.NameMissing.eval(args);
        Flag.PlaNameMissing.eval(args);
        Flag.PlaNameNonStandard.eval(args);
        Flag.GasNameMissing.eval(args);
        Flag.PlaIsPublic.eval(args);
        Flag.HnMissing.eval(args);
        Flag.HnTooManyDigits.eval(args);
        Flag.CityMissing.eval(args);
        Flag.StreetMissing.eval(args);
        Flag.NotAHospital.eval(args);
        Flag.ChangeToPetVet.eval(args);
        Flag.ChangeToDoctorClinic.eval(args);
        Flag.NotASchool.eval(args);

        Flag.RestAreaSpec.eval(args);
        Flag.RestAreaScenic.eval(args);
        Flag.RestAreaNoTransportation.eval(args);
        Flag.RestAreaGas.eval(args);
        Flag.RestAreaName.eval(args);
        Flag.AreaNotPoint.eval(args);

        // update Severity for banner messages
        const orderedFlags = FlagBase.currentFlags.getOrderedFlags();
        orderedFlags.forEach(flag => {
            args.totalSeverity = Math.max(flag.severity, args.totalSeverity);
        });

        // final updating of desired lock levels
        if (pnhLockLevel !== -1 && !args.highlightOnly) {
            logDev(`PNHLockLevel: ${pnhLockLevel}`);
            args.levelToLock = pnhLockLevel;
        } else {
            args.levelToLock = args.defaultLockLevel;
        }
        if (args.region === 'SER') {
            if (args.categories.includes(CAT.COLLEGE_UNIVERSITY) && args.categories.includes(CAT.PARKING_LOT)) {
                args.levelToLock = LOCK_LEVEL_4;
            } else if (venue.isPoint() && args.categories.includes(CAT.COLLEGE_UNIVERSITY) && (!args.categories.includes(CAT.HOSPITAL_MEDICAL_CARE)
                || !args.categories.includes(CAT.HOSPITAL_URGENT_CARE))) {
                args.levelToLock = LOCK_LEVEL_4;
            }
        }

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

        // Brand checking (be sure to check this after determining if brand will be forced, when harmonizing)
        Flag.GasNoBrand.eval(args);
        Flag.GasUnbranded.eval(args);

        Flag.IgnEdited.eval(args);
        Flag.WazeBot.eval(args);
        Flag.LockRPP.eval(args);

        // Allow flags to do any additional work before assigning severity and locks
        orderedFlags.forEach(flag => flag.preProcess?.(args));

        if (!args.highlightOnly) {
            // Update the lockOK value if "noLock" is set on any flag.
            args.lockOK &&= !orderedFlags.some(flag => flag.noLock);
            logDev(`Severity: ${args.totalSeverity}; lockOK: ${args.lockOK}`);
        }

        const placeLockedFlag = Flag.PlaceLocked.eval(args);

        // Turn off unnecessary buttons
        // TODO: handle this in the flag class
        if (args.categories.includes(CAT.PHARMACY)) {
            FlagBase.currentFlags.remove(Flag.AddPharm);
        }
        if (args.categories.includes(CAT.SUPERMARKET_GROCERY)) {
            FlagBase.currentFlags.remove(Flag.AddSuper);
        }

        // Final alerts for non-severe locations
        Flag.ResiTypeName.eval(args);
        Flag.SuspectDesc.eval(args);

        _dupeHNRangeList = [];
        _dupeBanner = {};
        if (!args.highlightOnly) runDuplicateFinder(venue, args.nameBase, args.aliases, args.addr, args.placePL);
        // Check HN range (this depends on the returned dupefinder data, so must run after it)
        Flag.HNRange.eval(args);

        // Return severity for highlighter (no dupe run))
        if (args.highlightOnly) {
            // get severities from the banners
            args.totalSeverity = SEVERITY.GREEN;
            orderedFlags.forEach(flag => {
                args.totalSeverity = Math.max(flag.severity, args.totalSeverity);
            });

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

            if (args.totalSeverity === SEVERITY.GREEN && placeLockedFlag?.hlLockFlag) {
                args.totalSeverity = 'lock';
            }
            if (args.totalSeverity === SEVERITY.BLUE && placeLockedFlag?.hlLockFlag) {
                args.totalSeverity = 'lock1';
            }
            if (venue.attributes.adLocked) {
                args.totalSeverity = 'adLock';
            }

            return args.totalSeverity;
        }

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

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

            assembleBanner();

            executeMultiAction(actions);
        }

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

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

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

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

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

        // Setup duplicates banners
        $rowDiv = $('<div class="banner-row yellow">');
        Object.keys(_dupeBanner).forEach(tempKey => {
            const rowData = _dupeBanner[tempKey];
            if (rowData.active) {
                dupesFound += 1;
                const $dupeDiv = $('<div class="dupe">').appendTo($rowDiv);
                $dupeDiv.append($('<span style="margin-right:4px">').html(`&bull; ${rowData.message}`));
                if (rowData.value) {
                    // Nothing happening here yet.
                }
                if (rowData.WLactive && rowData.WLaction) { // If there's a WL option, enable it
                    totalSeverity = Math.max(rowData.severity, totalSeverity);
                    $dupeDiv.append($('<button>', {
                        class: 'btn btn-success btn-xs wmephwl-btn',
                        id: `WMEPH_WL${tempKey}`,
                        title: rowData.wlTooltip
                    }).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
        flags.forEach(flag => {
            $rowDiv = $('<div class="banner-row">');
            switch (flag.severity) {
                case SEVERITY.RED:
                    $rowDiv.addClass('red');
                    break;
                case SEVERITY.YELLOW:
                    $rowDiv.addClass('yellow');
                    break;
                case SEVERITY.BLUE:
                    $rowDiv.addClass('blue');
                    break;
                case SEVERITY.GREEN:
                    $rowDiv.addClass('gray');
                    break;
                default:
                    throw new Error(`WMEPH: Unexpected severity value while building banner: ${flag.severity}`);
            }
            if (flag.divId) {
                $rowDiv.attr('id', flag.divId);
            }
            if (flag.message && flag.message.length) {
                $rowDiv.append($('<span>').css({ 'margin-right': '4px' }).append(`&bull; ${flag.message}`));
            }
            if (flag.buttonText) {
                $rowDiv.append($('<button>', {
                    class: 'btn btn-default btn-xs wmeph-btn',
                    id: `WMEPH_${flag.name}`,
                    title: flag.title || ''
                }).css({ 'margin-right': '4px' }).html(flag.buttonText));
            }
            if (flag.value2) {
                $rowDiv.append($('<button>', {
                    class: 'btn btn-default btn-xs wmeph-btn',
                    id: `WMEPH_${flag.name}_2`,
                    title: flag.title2 || ''
                }).css({ 'margin-right': '4px' }).html(flag.value2));
            }
            if (flag.showWL) {
                if (flag.WLaction) { // If there's a WL option, enable it
                    totalSeverity = Math.max(flag.severity, totalSeverity);
                    $rowDiv.append(
                        $('<button>', { class: 'btn btn-success btn-xs wmephwl-btn', id: `WMEPH_WL${flag.name}`, title: flag.wlTooltip })
                            .text('WL')
                    );
                }
            } else {
                totalSeverity = Math.max(flag.severity, totalSeverity);
            }
            if (flag.suffixMessage) {
                $rowDiv.append($('<div>').css({ 'margin-top': '2px' }).append(flag.suffixMessage));
            }

            rowDivs.push($rowDiv);
        });

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

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

        assembleServicesBanner();

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

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

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

        // Setup bannButt2 onclicks
        setupButtonsOld(_buttonBanner2);

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

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

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

        // Allow flags to do any additional work (hook up events, etc);
        flags.forEach(flag => {
            flag.postProcess?.();
        });

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

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

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

                    $bannerDiv.append($row);

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

    const _googleResults = {};
    let _googlePlacePtFeature;
    let _googlePlaceLineFeature;
    let _destroyGooglePlacePointTimeoutId;

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

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

    function drawGooglePlacePoint(uuid) {
        if (!uuid) return;
        const link = _googleResults[uuid];
        if (link) {
            const coord = link.geometry.location;
            const poiPt = new OpenLayers.Geometry.Point(coord.lng(), coord.lat());
            poiPt.transform(W.Config.map.projection.remote, W.map.getProjectionObject().projCode);
            const placeGeom = W.selectionManager.getSelectedDataModelObjects()[0].attributes.geometry.getCentroid();
            const placePt = new OpenLayers.Geometry.Point(placeGeom.x, placeGeom.y);
            const ext = W.map.getExtent();
            const lsBounds = new OpenLayers.Geometry.LineString([
                new OpenLayers.Geometry.Point(ext.left, ext.bottom),
                new OpenLayers.Geometry.Point(ext.left, ext.top),
                new OpenLayers.Geometry.Point(ext.right, ext.top),
                new OpenLayers.Geometry.Point(ext.right, ext.bottom),
                new OpenLayers.Geometry.Point(ext.left, ext.bottom)]);
            let lsLine = new OpenLayers.Geometry.LineString([placePt, poiPt]);

            // If the line extends outside the bounds, split it so we don't draw a line across the world.
            const splits = lsLine.splitWith(lsBounds);
            let label = '';
            if (splits) {
                let splitPoints;
                splits.forEach(split => {
                    split.components.forEach(component => {
                        if (component.x === placePt.x && component.y === placePt.y) splitPoints = split;
                    });
                });
                lsLine = new OpenLayers.Geometry.LineString([splitPoints.components[0], splitPoints.components[1]]);
                let distance = WazeWrap.Geometry.calculateDistance([poiPt, placePt]);
                let unitConversion;
                let unit1;
                let unit2;
                if (W.model.isImperial) {
                    distance *= 3.28084;
                    unitConversion = 5280;
                    unit1 = ' ft';
                    unit2 = ' mi';
                } else {
                    unitConversion = 1000;
                    unit1 = ' m';
                    unit2 = ' km';
                }
                if (distance > unitConversion * 10) {
                    label = Math.round(distance / unitConversion) + unit2;
                } else if (distance > 1000) {
                    label = (Math.round(distance / (unitConversion / 10)) / 10) + unit2;
                } else {
                    label = Math.round(distance) + unit1;
                }
            }

            destroyGooglePlacePoint(); // Just in case it still exists.
            _googlePlacePtFeature = new OpenLayers.Feature.Vector(poiPt, { poiCoord: true }, {
                pointRadius: 6,
                strokeWidth: 30,
                strokeColor: '#FF0',
                fillColor: '#FF0',
                strokeOpacity: 0.5
            });
            _googlePlaceLineFeature = new OpenLayers.Feature.Vector(lsLine, {}, {
                strokeWidth: 3,
                strokeDashstyle: '12 8',
                strokeColor: '#FF0',
                label,
                labelYOffset: 45,
                fontColor: '#FF0',
                fontWeight: 'bold',
                labelOutlineColor: '#000',
                labelOutlineWidth: 4,
                fontSize: '18'
            });
            W.map.getLayerByUniqueName('venues').addFeatures([_googlePlacePtFeature, _googlePlaceLineFeature]);
            timeoutDestroyGooglePlacePoint();
        } else {
            fetchGoogleLinkInfo(uuid).then(res => {
                if (res.error || res.apiDisabled) {
                    // API was temporarily disabled.  Ignore for now.
                } else {
                    drawGooglePlacePoint(uuid);
                }
            });
        }
    }

    // Destroy the point after some time, if it hasn't been destroyed already.
    function timeoutDestroyGooglePlacePoint() {
        if (_destroyGooglePlacePointTimeoutId) clearTimeout(_destroyGooglePlacePointTimeoutId);
        _destroyGooglePlacePointTimeoutId = setTimeout(() => destroyGooglePlacePoint(), 4000);
    }

    // Remove the POI point from the map.
    function destroyGooglePlacePoint() {
        if (_googlePlacePtFeature) {
            _googlePlacePtFeature.destroy();
            _googlePlacePtFeature = null;
            _googlePlaceLineFeature.destroy();
            _googlePlaceLineFeature = null;
        }
    }

    function addGoogleLinkHoverEvent($el) {
        $el.hover(() => drawGooglePlacePoint(getGooglePlaceUuidFromElement($el)), () => destroyGooglePlacePoint());
    }

    function getGooglePlaceUuidFromElement($el) {
        return $el.attr('uuid');
    }

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

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

    // Button onclick event handler
    function setupButtons(flags) {
        flags.forEach(flag => { // Loop through the banner possibilities
            if (flag.action && flag.buttonText) { // If there is an action, set onclick
                buttonAction(flag);
            }
            if (flag.action2 && flag.value2) { // If there is an action2, set onclick
                buttonAction2(flag);
            }
            // If there's a WL option, set up onclick
            if (flag.showWL && flag.WLaction) {
                buttonWhitelist(flag);
            }
        });
    }

    function setupButtonsOld(banner) {
        Object.keys(banner).forEach(flagKey => {
            const flag = banner[flagKey];
            if (flag?.active && flag.action && flag.value) {
                buttonActionOld(flagKey, flag);
            }
            if (flag?.WLactive && flag.WLaction) {
                buttonWhitelistOld(flagKey, flag);
            }
        });
    }
    function buttonActionOld(flagKey, flag) {
        const button = document.getElementById(`WMEPH_${flagKey}`);
        button.onclick = () => {
            flag.action();
            if (!flag.noBannerAssemble) harmonizePlaceGo(getSelectedVenue(), 'harmonize');
        };
    }
    function buttonWhitelistOld(flagKey, flag) {
        const button = document.getElementById(`WMEPH_WL${flagKey}`);
        button.onclick = () => {
            if (flagKey.match(/^\d{5,}/) !== null) {
                flag.WLaction(flagKey);
            } else {
                flag.WLaction();
            }
            flag.WLactive = false;
            flag.severity = SEVERITY.GREEN;
        };
        return button;
    }

    function buttonAction(flag) {
        const button = document.getElementById(`WMEPH_${flag.name}`);
        button.onclick = () => {
            flag.action();
            if (!flag.noBannerAssemble) harmonizePlaceGo(getSelectedVenue(), 'harmonize');
        };
        return button;
    }
    function buttonAction2(flag) {
        const button = document.getElementById(`WMEPH_${flag.name}_2`);
        button.onclick = () => {
            flag.action2();
            if (!flag.noBannerAssemble) harmonizePlaceGo(getSelectedVenue(), 'harmonize');
        };
        return button;
    }
    function buttonWhitelist(flag) {
        const button = document.getElementById(`WMEPH_WL${flag.name}`);
        button.onclick = () => {
            if (flag.name.match(/^\d{5,}/) !== null) {
                flag.WLaction(flag.name);
            } else {
                flag.WLaction();
            }
        };
        return button;
    }

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

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

    function onPasteClicked() {
        clonePlace();
    }

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

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

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

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

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

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

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

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

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

        const venue = getSelectedVenue();

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

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

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

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

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

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

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

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

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

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

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

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

    function venueHasOverlappingHours(openingHours) {
        if (openingHours.length < 2) {
            return false;
        }

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

    const NO_NUM_SKIP = ['BANK', 'ATM', 'HOTEL', 'MOTEL', 'STORE', 'MARKET', 'SUPERMARKET', 'GYM', 'GAS', 'GASOLINE',
        'GASSTATION', 'CAFE', 'OFFICE', 'OFFICES', 'CARRENTAL', 'RENTALCAR', 'RENTAL', 'SALON', 'BAR',
        'BUILDING', 'LOT', ...COLLEGE_ABBREVIATIONS];

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

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

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

        // generic terms to skip if it's all that remains after stripping numbers
        const allowedTwoLetters = ['BP', 'DQ', 'BK', 'BW', 'LQ', 'QT', 'DB', 'PO'];

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

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

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

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

        const selectedVenueAttr = selectedVenue.attributes;

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

    // Translation from PNH "natural language" category name to category ID  (Bank / Financial --> BANK_FINANCIAL)
    function getCategoryIdFromName(pnhCategoryName, countryCode) {
        const categoryInfo = PNH_DATA[countryCode].categoryInfos.getByName(pnhCategoryName);

        if (categoryInfo) {
            return categoryInfo.id;
        }

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

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

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

    // function that checks if any element of target are in source
    /**
     * Checks if any element of target are in source
     *
     * @param {Array|string} source Source array.
     * @param {Array|string} target Array of items to check against source.
     * @return {boolean} True if any item in target exists 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(item => target.includes(item));
    }

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

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

    function removeUnnecessaryAliases(venueName, aliases) {
        const newAliases = [];
        let aliasesRemoved = false;
        venueName = venueName.replace(/['=\\/]/i, '');
        venueName = venueName.toUpperCase().replace(/'/g, '').replace(/(-|\/ | \/| {2,})/g, ' ');
        for (let naix = 0; naix < aliases.length; naix++) {
            if (!venueName.startsWith(aliases[naix].toUpperCase().replace(/'/g, '').replace(/(-|\/ | \/| {2,})/g, ' '))) {
                newAliases.push(aliases[naix]);
            } else {
                aliasesRemoved = true;
            }
        }
        return aliasesRemoved ? newAliases : null;
    }

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

    // settings tab
    function initWmephTab() {
        const multicall = (func, names) => names.forEach(name => func(name));

        // Enable certain settings by default if not set by the user:
        multicall(setCheckedByDefault, [
            'WMEPH-ColorHighlighting',
            'WMEPH-ExcludePLADupes',
            'WMEPH-DisablePLAExtProviderCheck'
        ]);

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

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

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

        initShortcutKey();

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

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

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

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

        _initAlreadyRun = true;
    }

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

        // Check for script updates.
        const downloadUrl = IS_BETA_VERSION ? dec(BETA_DOWNLOAD_URL) : PROD_DOWNLOAD_URL;
        let updateMonitor;
        try {
            updateMonitor = new WazeWrap.Alerts.ScriptUpdateMonitor(SCRIPT_NAME, SCRIPT_VERSION, downloadUrl, GM_xmlhttpRequest);
            updateMonitor.start();
        } catch (ex) {
            // Report, but don't stop if ScriptUpdateMonitor fails.
            console.error('WMEPH:', ex);
        }

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

        _layer = W.map.venueLayer;

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

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

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

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

        _userLanguage = I18n.locale;

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

        appendServiceButtonIconCss();
        UPDATED_FIELDS.init();
        addPURWebSearchButton();

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

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

        createObserver();

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

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

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

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

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

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

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

        await addWmephTab(); // initialize the settings tab

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

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

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

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

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

        // Setup highlight colors
        initializeHighlights();

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

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

    function waitForReady() {
        return new Promise(resolve => {
            function loop() {
                if (typeof W === 'object' && W.userscripts?.state.isReady && WazeWrap?.Ready) {
                    resolve();
                } else {
                    setTimeout(loop, 100);
                }
            }
            loop();
        });
    }

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

    const SPREADSHEET_ID = '1pBz4l4cNapyGyzfMJKqA4ePEFLkmz2RryAt1UV39B4g';
    const SPREADSHEET_RANGE = '2019.01.20.001!A2:L';
    const API_KEY = 'YTJWNVBVRkplbUZUZVVObU1YVXpSRVZ3ZW5OaFRFSk1SbTR4VGxKblRURjJlRTFYY3pOQ2NXZElPQT09';
    const dec = s => atob(atob(s));

    class PnhCategoryInfos {
        #categoriesById = {};
        #categoriesByName = {};

        add(categoryInfo) {
            this.#categoriesById[categoryInfo.id] = categoryInfo;
            this.#categoriesByName[categoryInfo.name.toUpperCase()] = categoryInfo;
        }

        getById(id) {
            return this.#categoriesById[id];
        }

        getByName(name) {
            return this.#categoriesByName[name.toUpperCase()];
        }

        toArray() {
            return Object.values(this.#categoriesById);
        }
    }

    function processPnhCategories(categoryDataRows, categoryInfos) {
        let headers;
        let pnhServiceKeys;
        let wmeServiceIds;
        const splitValues = (value => (value.trim() ? value.split(',').map(v => v.trim()) : []));
        categoryDataRows.forEach((row, iRow) => {
            row = row.split('|');
            if (iRow === 0) {
                headers = row;
            } else if (iRow === 1) {
                pnhServiceKeys = row;
            } else if (iRow === 2) {
                wmeServiceIds = row;
            } else {
                const categoryInfo = {
                    services: []
                };
                row.forEach((value, iCol) => {
                    const headerValue = headers[iCol].trim();
                    value = value.trim();
                    switch (headerValue) {
                        case 'pc_wmecat':
                            categoryInfo.id = value;
                            break;
                        case 'pc_transcat':
                            categoryInfo.name = value;
                            break;
                        case 'pc_catparent':
                            categoryInfo.parent = value;
                            break;
                        case 'pc_point':
                            categoryInfo.point = value;
                            break;
                        case 'pc_area':
                            categoryInfo.area = value;
                            break;
                        case 'pc_regpoint':
                            categoryInfo.regPoint = splitValues(value);
                            break;
                        case 'pc_regarea':
                            categoryInfo.regArea = splitValues(value);
                            break;
                        case 'pc_lock1':
                            categoryInfo.lock1 = splitValues(value);
                            break;
                        case 'pc_lock2':
                            categoryInfo.lock2 = splitValues(value);
                            break;
                        case 'pc_lock3':
                            categoryInfo.lock3 = splitValues(value);
                            break;
                        case 'pc_lock4':
                            categoryInfo.lock4 = splitValues(value);
                            break;
                        case 'pc_lock5':
                            categoryInfo.lock5 = splitValues(value);
                            break;
                        case 'pc_rare':
                            categoryInfo.rare = splitValues(value);
                            break;
                        case 'pc_parent':
                            categoryInfo.disallowedParent = splitValues(value);
                            break;
                        case 'pc_message':
                            categoryInfo.messagae = value;
                            break;
                        case 'ps_valet':
                        case 'ps_drivethru':
                        case 'ps_wifi':
                        case 'ps_restrooms':
                        case 'ps_cc':
                        case 'ps_reservations':
                        case 'ps_outside':
                        case 'ps_ac':
                        case 'ps_parking':
                        case 'ps_deliveries':
                        case 'ps_takeaway':
                        case 'ps_wheelchair':
                            if (value) {
                                categoryInfo.services.push({ wmeId: wmeServiceIds[iCol], pnhKey: pnhServiceKeys[iCol] });
                            }
                            break;
                        case '':
                            // ignore blank column
                            break;
                        default:
                            throw new Error(`WMEPH: Unexpected category data from PNH sheet: ${headerValue}`);
                    }
                });
                categoryInfos.add(categoryInfo);
            }
        });
    }

    function processImportedDataColumn(allData, columnIndex) {
        return allData.filter(row => row.length >= columnIndex + 1).map(row => row[columnIndex]);
    }
    function getSpreadsheetUrl(id, range, key) {
        return `https://sheets.googleapis.com/v4/spreadsheets/${id}/values/${range}?${dec(key)}`;
    }
    function downloadPnhData() {
        log('PNH data download started...');
        return new Promise((resolve, reject) => {
            const url = getSpreadsheetUrl(SPREADSHEET_ID, SPREADSHEET_RANGE, API_KEY);

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

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

                PNH_DATA.USA.categoryInfos = new PnhCategoryInfos();
                processPnhCategories(processImportedDataColumn(values, 3), PNH_DATA.USA.categoryInfos);
                PNH_DATA.USA.pnh = processImportedDataColumn(values, 0);
                PNH_DATA.USA.pnhNames = makeNameCheckList(PNH_DATA.USA);
                PNH_DATA.states = processImportedDataColumn(values, 1);

                PNH_DATA.CAN.categoryInfos = PNH_DATA.USA.categoryInfos;
                PNH_DATA.CAN.pnh = processImportedDataColumn(values, 2);
                PNH_DATA.CAN.pnhNames = makeNameCheckList(PNH_DATA.CAN);

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

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

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

    function devTestCode() {
        if (W.loginManager.user.getUsername() === 'MapOMatic') {
            unsafeWindow.UpdateFeatureGeometry = UpdateFeatureGeometry;
            // test code here
            // $('#redo-button').click(harmonizePlace);
            // $('#undo-button').click(harmonizePlace);
        }
    }

    async function bootstrap() {
        // Quit if another version of WMEPH is already running.
        if (unsafeWindow.wmephRunning) {
            // Don't use WazeWrap alerts here. It isn't loaded yet.
            alert('Multiple versions of WME Place Harmonizer are turned on. Only one will be enabled.');
            return;
        }
        unsafeWindow.wmephRunning = 1;
        // Start downloading the PNH spreadsheet data in the background.  Starts the script once data is ready.
        await downloadPnhData();
        await placeHarmonizerBootstrap();
        devTestCode();
    }

    bootstrap();
})();