您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Harmonizes, formats, and locks a selected place
当前为
// ==UserScript== // @name WME Place Harmonizer // @namespace WazeUSA // @version 2024.03.20.001 // @description Harmonizes, formats, and locks a selected place // @author WMEPH Development Group // @include /^https:\/\/(www|beta)\.waze\.com\/(?!user\/)(.{2,6}\/)?editor\/?.*$/ // @require https://gf.qytechs.cn/scripts/24851-wazewrap/code/WazeWrap.js // @require https://gf.qytechs.cn/scripts/37486-wme-utils-hoursparser/code/WME%20Utils%20-%20HoursParser.js // @require https://cdnjs.cloudflare.com/ajax/libs/lz-string/1.4.4/lz-string.min.js // @require https://cdnjs.cloudflare.com/ajax/libs/Turf.js/6.5.0/turf.min.js // @license GNU GPL v3 // @connect gf.qytechs.cn // @grant GM_addStyle // @grant GM_xmlhttpRequest // ==/UserScript== /* global W */ /* global OpenLayers */ /* global _ */ /* global WazeWrap */ /* global LZString */ /* global HoursParser */ /* global I18n */ /* global google */ /* global turf */ /* 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://gf.qytechs.cn/scripts/28690-wme-place-harmonizer/code/WME%20Place%20Harmonizer.user.js'; const BETA_DOWNLOAD_URL = 'YUhSMGNITTZMeTluY21WaGMzbG1iM0pyTG05eVp5OXpZM0pwY0hSekx6STROamc1TFhkdFpTMXdiR0ZqWlMxb1lYSnRiMjVwZW1WeUxXSmxkR0V2WTI5a1pTOVhUVVVsTWpCUWJHRmpaU1V5TUVoaGNtMXZibWw2WlhJbE1qQkNaWFJoTG5WelpYSXVhbk09'; const _pnhModerators = {}; 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.getOLGeometry().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); }); } } } } function uniq(arrayIn) { return [...new Set(arrayIn)]; } // 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 newGeometry = structuredClone(venue.getGeometry()); const moveNegative = Math.random() > 0.5; const nudgeDistance = 0.00000001 * (moveNegative ? -1 : 1); if (venue.isPoint()) { newGeometry.coordinates[0] += nudgeDistance; } else { newGeometry.coordinates[0][0][0] += nudgeDistance; } const action = new UpdateFeatureGeometry(venue, W.model.venues, venue.getGeometry(), newGeometry); 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.getOLGeometry().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) { if (actions.length > 0) { W.model.actionManager.add(new MultiAction(actions)); } } // 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.getOLGeometry(), 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}:  ${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)}:  CLOSED`); } 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?.length) { const stopPoint = attr.entryExitPoints[0].getPoint().coordinates; const areaCenter = turf.centroid(args.venue.getGeometry()).geometry.coordinates; return stopPoint[0] === areaCenter[0] && stopPoint[1] === areaCenter[1]; } 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.getOLGeometry().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.getOLGeometry().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); } else if (!args.highlightOnly && args.categories.includes(CAT.POST_OFFICE)) { Flag.LocationFinder.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(`• ${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(`• ${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( '•', $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 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].getOLGeometry().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.getOLGeometry().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.getOLGeometry().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.getOLGeometry().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.getOLGeometry()); distanceB = pt.distanceTo(nodeB.getOLGeometry()); 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.getOLGeometry().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].getOLGeometry()); // 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 newActions = []; newActions.push(new UpdateFeatureAddress(feature, newAttributes)); if (address.hasOwnProperty('houseNumber')) { newActions.push(new UpdateObject(feature, { houseNumber: address.houseNumber })); } const multiAction = new MultiAction(newActions, { description: 'Update venue address' }); 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); $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).sort().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 && W.model.categoryBrands.PARKING_LOT) { 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 SPREADSHEET_MODERATORS_RANGE = 'Moderators!A1:F'; 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 downloadPnhModerators() { log('PNH moderators download started...'); return new Promise(resolve => { const url = getSpreadsheetUrl(SPREADSHEET_ID, SPREADSHEET_MODERATORS_RANGE, API_KEY); $.getJSON(url).done(res => { const { values } = res; try { values.forEach(regionArray => { const region = regionArray[0]; const mods = regionArray.slice(3); _pnhModerators[region] = mods; }); } catch (ex) { _pnhModerators['?'] = ['Error downloading moderators!']; } // delete Texas region, if it exists delete _pnhModerators.TX; log('PNH moderators 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 moderator list:', message); _pnhModerators['?'] = ['Error downloading moderators!']; resolve(); }); }); } function devTestCode() { if (W.loginManager.user.getUsername() === 'MapOMatic') { // test code here } } 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 downloadPnhModerators(); await placeHarmonizerBootstrap(); devTestCode(); } bootstrap(); })();
QingJ © 2025
镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址