您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Harmonizes, formats, and locks a selected place
当前为
// ==UserScript== // @name WME Place Harmonizer // @namespace WazeUSA // @version 1.3.135 // @description Harmonizes, formats, and locks a selected place // @author WMEPH Development Group // @include /^https:\/\/(www|beta)\.waze\.com\/(?!user\/)(.{2,6}\/)?editor\/?.*$/ // @require https://gf.qytechs.cn/scripts/24851-wazewrap/code/WazeWrap.js // @require https://gf.qytechs.cn/scripts/37486-wme-utils-hoursparser/code/WME%20Utils%20-%20HoursParser.js // @require https://cdnjs.cloudflare.com/ajax/libs/lz-string/1.4.4/lz-string.min.js // @license GNU GPL v3 // @grant GM_addStyle // ==/UserScript== /* global I18n */ /* global $ */ /* global W */ /* global GM_info */ /* global require */ /* global performance */ /* global OL */ /* global _ */ /* global define */ /* global Node */ /* global WazeWrap */ /* global unsafeWindow */ /* global LZString */ /* global Promise */ (function () { 'use strict'; // Quit if another version of WMEPH is already running. if (unsafeWindow.wmephRunning) { alert('Multiple versions of Place Harmonizer are turned on. Only one will be enabled.'); return; } else { unsafeWindow.wmephRunning = 1; } // Script update info const _WHATS_NEW_LIST = [ // New in this version '1.3.91: NEW - Added a "Google" button to search for place information so you don\'t have to click Run WMEPH first.', '1.3.87: FIXED - X-ray mode doesn\'t stay disabled after refresh.', '1.3.86: FIXED - X-ray mode not being restored after refresh.', '1.3.85: NEW - Place will be green (not blue) for 6mo after locking with a missing a Google link.', '1.3.84: FIXED - Address inference fails in some circumstances.', '1.3.84: FIXED - WL of missing URL flag does not update banner color.', '1.3.84: FIXED - ExtraMile not handled properly with Force Title Case.', '1.3.84: NEW - Added more categories to ignore for missing Google link.', '1.3.84: NEW - Darken the GIS Layers script\'s layer when X-ray mode is enabled.', '1.3.83: FIXED - Disable "No Google link" flag for some natural feature categories.', '1.3.82: NEW - Experimental "X-ray mode"!', '1.3.81: FIXED - WL of "area code mismatch" and/or "HN out of range" doesn\'t update banner color.', '1.3.79: FIXED - Optional category messages not displaying correctly.', '1.3.78: FIXED - WL of "No Hours" and/or "No Ph#" doesn\'t update banner color.' ]; const _CSS_ARRAY = [ '#WMEPH_banner .wmeph-btn { background-color: #fbfbfb; box-shadow: 0 2px 0 #aaa; border: solid 1px #bbb; font-weight:normal; margin-bottom: 2px; margin-right:4px}', '.wmeph-btn, .wmephwl-btn { height:19px; }', '.btn.wmeph-btn { padding: 0px 3px }', '.btn.wmephwl-btn { padding: 0px 1px 0px 2px; height: 18px; box-shadow: 0 2px 0 #b3b3b3;}', '#WMEPH_banner .banner-row { padding:2px 4px; }', '#WMEPH_banner .banner-row.red { color:#b51212; background-color:#f0dcdc; }', '#WMEPH_banner .banner-row.blue { color:#3232e6; background-color:#dcdcf0; }', '#WMEPH_banner .banner-row.yellow { color:#584a04; background-color:#f0f0c2; }', '#WMEPH_banner .banner-row.gray { color:#3a3a3a; background-color:#eeeeee; }', '#WMEPH_banner .banner-row .dupe { padding-left:8px; }', '#WMEPH_banner { background-color:#fff; color:black; font-size:14px; padding-top:8px; padding-bottom:8px; margin-left:4px; margin-right:4px; line-height:18px; margin-top:2px; border: solid 1px #8d8c8c; border-radius: 6px; margin-bottom: 4px;}', '#WMEPH_banner input[type=text] { font-size: 13px !important; height:22px !important; font-family: "Open Sans", Alef, helvetica, sans-serif !important; }', '#WMEPH_banner div:last-child { padding-bottom: 3px !important; }', '#WMEPH_runButton { padding-bottom: 6px; padding-top: 3px; width: 290; color: black; font-size: 15px; margin-right: auto; margin-left: 4px; }', '#WMEPH_tools div { padding-bottom: 2px !important; }', '.wmeph-fat-btn { padding-left:8px; padding-right:8px; padding-top:4px; margin-right:3px; display:inline-block; font-weight:normal; height:24px; }', '.ui-autocomplete { max-height: 300px;overflow-y: auto;overflow-x: hidden;} ' ]; const _SCRIPT_VERSION = GM_info.script.version.toString(); // pull version from header const _SCRIPT_NAME = GM_info.script.name; const _IS_DEV_VERSION = /Beta/i.test(_SCRIPT_NAME); // enables dev messages and unique DOM options if the script is called "... Beta" const _PNH_DATA = { USA: {}, CAN: {} }; const _CATEGORY_LOOKUP = {}; const _DEFAULT_HOURS_TEXT = 'Paste Hours Here'; const _MAX_CACHE_SIZE = 25000; let _resultsCache = {}; let _initAlreadyRun = false; // This is used to skip a couple things if already run once. This could probably be handled better... let _countryCode; let _textEntryValues = null; // Store the values entered in text boxes so they can be re-added when the banner is reassembled. var hospitalPartMatch, hospitalFullMatch, animalPartMatch, animalFullMatch, schoolPartMatch, schoolFullMatch; // vars for cat-name checking var WMEPHdevList, WMEPHbetaList; // Userlists var devVersStr= _IS_DEV_VERSION ? 'Beta' : ''; // strings to differentiate DOM elements between regular and beta script var WMEServicesArray = ['VALLET_SERVICE','DRIVETHROUGH','WI_FI','RESTROOMS','CREDIT_CARDS','RESERVATIONS','OUTSIDE_SEATING','AIR_CONDITIONING','PARKING_FOR_CUSTOMERS','DELIVERIES','TAKE_AWAY','WHEELCHAIR_ACCESSIBLE','DISABILITY_PARKING']; var collegeAbbreviations = 'USF|USFSP|UF|UCF|UA|UGA|FSU|UM|SCP|FAU|FIU'; var shortcutParse, modifKey = 'Alt+'; var venueWhitelist, venueWhitelistStr, WLSToMerge, wlKeyName, wlButtText = 'WL'; // Whitelisting vars var WLlocalStoreName = 'WMEPH-venueWhitelistNew'; var WLlocalStoreNameCompressed = 'WMEPH-venueWhitelistCompressed'; var _dupeLayer, dupeIDList = [], dupeHNRangeList, dupeHNRangeIDList, dupeHNRangeDistList; // Web search Window forming: var searchResultsWindowSpecs = '"resizable=yes, top='+ Math.round(window.screen.height*0.1) +', left='+ Math.round(window.screen.width*0.3) +', width='+ Math.round(window.screen.width*0.7) +', height='+ Math.round(window.screen.height*0.8) +'"'; var searchResultsWindowName = '"WMEPH Search Results"'; var WMEPHmousePosition; var cloneMaster = null; var bannButt, bannButt2, bannServ, bannDupl; // Banner Buttons objects var RPPLockString = 'Lock?'; var panelFields = {}; // the fields for the sidebar var MultiAction = require('Waze/Action/MultiAction'); var UpdateObject = require('Waze/Action/UpdateObject'); var UpdateFeatureGeometry = require('Waze/Action/UpdateFeatureGeometry'); var UpdateFeatureAddress = require('Waze/Action/UpdateFeatureAddress'); let _disableHighlightTest = false; // Set to true to temporarily disable highlight checks immediately when venues change. let _wl = {}; const _USER = { ref: null, rank: null, name: null, isBetaUser: false, isDevUser: false }; const _SETTING_IDS = { sfUrlWarning: 'SFURLWarning', // Warning message for first time using localized storefinder URL. gLinkWarning: 'GLinkWarning' // Warning message for first time using Google search to not to use the Google info itself. }; const _URLS = { forum: 'https://www.waze.com/forum/posting.php?mode=reply&f=819&t=215657', usaPnh: 'https://docs.google.com/spreadsheets/d/1-f-JTWY5UnBx-rFTa4qhyGMYdHBZWNirUTOgn222zMY/edit#gid=0', placesWiki: 'https://wazeopedia.waze.com/wiki/USA/Places', restAreaWiki: 'https://wazeopedia.waze.com/wiki/USA/Rest_areas#Adding_a_Place' }; var userLanguage; // lock levels are offset by one var lockLevel2 = 1, lockLevel4 = 3; var defaultLockLevel = lockLevel2, PNHLockLevel; var PMUserList = { // user names and IDs for PM functions SER: {approvalActive: true, modID: '17083181', modName: 'itzwolf'}, WMEPH: {approvalActive: true, modID: '2647925', modName: 'MapOMatic'} }; var severityButt=0; // error tracking to determine banner color (action buttons) var duplicateName = ''; var catTransWaze2Lang; // pulls the category translations var itemID, newName, optionalAlias, newURL, tempPNHURL = '', newPhone; var newAliases = [], newAliasesTemp = [], newCategories = []; // Change place.name to title case const _TITLECASE_SETTINGS = { ignoreWords: 'an|and|as|at|by|for|from|hhgregg|in|into|of|on|or|the|to|with'.split('|'), capWords: '3M|AAA|AMC|AOL|AT&T|ATM|BBC|BLT|BMV|BMW|BP|CBS|CCS|CGI|CISCO|CJ|CNG|CNN|CVS|DHL|DKNY|DMV|DSW|EMS|ER|ESPN|FCU|FCUK|FDNY|GNC|H&M|HP|HSBC|IBM|IHOP|IKEA|IRS|JBL|JCPenney|KFC|LLC|MBNA|MCA|MCI|NBC|NYPD|PDQ|PNC|TCBY|TNT|TV|UPS|USA|USPS|VW|XYZ|ZZZ'.split('|'), specWords: 'd\'Bronx|iFix|ExtraMile'.split('|') }; var newPlaceURL, approveRegionURL; var customStoreFinder = false; // switch indicating place-specific custom store finder url var customStoreFinderLocal = false; // switch indicating place-specific custom store finder url with localization option (GPS/addr) var customStoreFinderURL = ''; // switch indicating place-specific custom store finder url var customStoreFinderLocalURL = ''; // switch indicating place-specific custom store finder url with localization option (GPS/addr) var updateURL; // Split out state-based data var ps_state_ix; var ps_state2L_ix; var ps_region_ix; var ps_gFormState_ix; var ps_defaultLockLevel_ix; //var ps_requirePhone_ix; //var ps_requireURL_ix; var ps_areacode_ix; var stateDataTemp, areaCodeList = '800,822,833,844,855,866,877,888'; // include toll free non-geographic area codes var ixBank, ixATM, ixOffices; var layer; var _updatedFields = { name: { updated: false, selector: '.landmark .form-control[name="name"]', tab: 'general' }, aliases: {updated: false, selector: '.landmark .form-control.alias-name', tab: 'general' }, address: {updated: false, selector: '.landmark .address-edit span.full-address', tab: 'general'}, categories: {updated: false, selector: '.landmark .categories.controls .select2-container', tab: 'general'}, description: {updated: false, selector: '.landmark .form-control[name="description"]', tab: 'general' }, lock: {updated: false, selector: '.landmark .form-control.waze-radio-container', tab: 'general' }, externalProvider: {updated: false, selector: '.landmark .external-providers-view', tab: 'general' }, brand: {updated: false, selector: '.landmark .brand .select2-container', tab: 'general' }, url: {updated: false, selector: '.landmark .form-control[name="url"]', tab: 'more-info' }, phone: {updated: false, selector: '.landmark .form-control[name="phone"]', tab: 'more-info' }, openingHours: {updated: false, selector: '.landmark .opening-hours ul', tab: 'more-info' }, cost: {updated: false, selector: '.landmark .form-control[name="costType"]', tab: 'more-info' }, canExit: {updated: false, selector: '.landmark label[for="can-exit-checkbox"]', tab: 'more-info' }, hasTBR: {updated: false, selector: '.landmark label[for="has-tbr"]', tab: 'more-info' }, lotType: {updated: false, selector: '.landmark .parking-type-option', tab: 'more-info' }, parkingSpots: {updated: false, selector: '.landmark .form-control[name="estimatedNumberOfSpots"]', tab: 'more-info' }, lotElevation: {updated: false, selector: '.landmark .lot-checkbox', tab: 'more-info'}, getFieldProperties: function() { return Object.keys(this).filter(key => this[key] && this[key].updated); }, getUpdatedTabs: function() { var tabs = []; this.getFieldProperties().forEach(propName => { var prop = this[propName]; if (prop.updated && tabs.indexOf(prop.tab) === -1) { tabs.push(prop.tab); } }); return tabs; }, checkAddedNode: function(addedNode) { this.getFieldProperties().forEach(propName => { var prop = this[propName]; if (prop.updated && addedNode.querySelector(prop.selector)) { $(prop.selector).css({'background-color':'#dfd'}); $('a[href="#landmark-edit-' + prop.tab + '"]').css({'background-color':'#dfd'}); } }); }, reset: function() { this.getFieldProperties().forEach(propName => {this[propName].updated = false;}); }, init: function() { ['VALLET_SERVICE', 'DRIVETHROUGH', 'WI_FI', 'RESTROOMS', 'CREDIT_CARDS', 'RESERVATIONS', 'OUTSIDE_SEATING', 'AIR_CONDITIONING', 'PARKING_FOR_CUSTOMERS', 'DELIVERIES', 'TAKE_AWAY', 'WHEELCHAIR_ACCESSIBLE', 'DISABILITY_PARKING', 'CARPOOL_PARKING', 'EV_CHARGING_STATION', 'CAR_WASH', 'SECURITY', 'AIRPORT_SHUTTLE'].forEach(service => { var propName = 'services_' + service; this[propName] = {updated: false, selector:'.landmark label[for="service-checkbox-' + service + '"]', tab: 'more-info' }; }); var observer = new MutationObserver(mutations => { mutations.forEach(mutation => { // Mutation is a NodeList and doesn't support forEach like an array for (var i = 0; i < mutation.addedNodes.length; i++) { var addedNode = mutation.addedNodes[i]; // Only fire up if it's a node if (addedNode.nodeType === Node.ELEMENT_NODE) { _updatedFields.checkAddedNode(addedNode); } } }); }); observer.observe(document.getElementById('edit-panel'), { childList: true, subtree: true }); W.selectionManager.events.register('selectionchanged', null, () => errorHandler(() => this.reset())); } }; // KB Shortcut object var shortcut = { 'all_shortcuts': {}, //All the shortcuts are stored in this array 'add': function(shortcut_combination, callback, opt) { //Provide a set of default options var default_options = { 'type': 'keydown', 'propagate': false, 'disable_in_input': false, 'target': document, 'keycode': false }; if (!opt) {opt = default_options;} else { for (var dfo in default_options) { if (typeof opt[dfo] === 'undefined') {opt[dfo] = default_options[dfo];} } } var ele = opt.target; if (typeof opt.target === 'string') {ele = document.getElementById(opt.target);} // var ths = this; shortcut_combination = shortcut_combination.toLowerCase(); //The function to be called at keypress var func = function(e) { e = e || window.event; if (opt.disable_in_input) { //Don't enable shortcut keys in Input, Textarea fields var element; if (e.target) {element = e.target;} else if (e.srcElement) {element = e.srcElement;} if (element.nodeType === 3) {element = element.parentNode;} if (element.tagName === 'INPUT' || element.tagName === 'TEXTAREA') {return;} } //Find Which key is pressed var code; if (e.keyCode) {code = e.keyCode;} else if (e.which) {code = e.which;} var character = String.fromCharCode(code).toLowerCase(); if (code === 188) {character = ',';} //If the user presses , when the type is onkeydown if (code === 190) {character = '.';} //If the user presses , when the type is onkeydown var keys = shortcut_combination.split('+'); //Key Pressed - counts the number of valid keypresses - if it is same as the number of keys, the shortcut function is invoked var kp = 0; //Work around for stupid Shift key bug created by using lowercase - as a result the shift+num combination was broken var shift_nums = { '`': '~','1': '!','2': '@','3': '#','4': '$','5': '%','6': '^','7': '&', '8': '*','9': '(','0': ')','-': '_','=': '+',';': ':','\'': '"',',': '<','.': '>','/': '?','\\': '|' }; //Special Keys - and their codes var special_keys = { 'esc': 27,'escape': 27,'tab': 9,'space': 32,'return': 13,'enter': 13,'backspace': 8,'scrolllock': 145, 'scroll_lock': 145,'scroll': 145,'capslock': 20,'caps_lock': 20,'caps': 20,'numlock': 144,'num_lock': 144,'num': 144, 'pause': 19,'break': 19,'insert': 45,'home': 36,'delete': 46,'end': 35,'pageup': 33,'page_up': 33,'pu': 33,'pagedown': 34, 'page_down': 34,'pd': 34,'left': 37,'up': 38,'right': 39,'down': 40,'f1': 112,'f2': 113,'f3': 114,'f4': 115,'f5': 116, 'f6': 117,'f7': 118,'f8': 119,'f9': 120,'f10': 121,'f11': 122,'f12': 123 }; var modifiers = { shift: { wanted: false, pressed: false }, ctrl: { wanted: false, pressed: false }, alt: { wanted: false, pressed: false }, meta: { wanted: false, pressed: false } //Meta is Mac specific }; if (e.ctrlKey) {modifiers.ctrl.pressed = true;} if (e.shiftKey) {modifiers.shift.pressed = true;} if (e.altKey) {modifiers.alt.pressed = true;} if (e.metaKey) {modifiers.meta.pressed = true;} for (var i = 0; i < keys.length; i++) { var k = keys[i]; //Modifiers if (k === 'ctrl' || k === 'control') { kp++; modifiers.ctrl.wanted = true; } else if (k === 'shift') { kp++; modifiers.shift.wanted = true; } else if (k === 'alt') { kp++; modifiers.alt.wanted = true; } else if (k === 'meta') { kp++; modifiers.meta.wanted = true; } else if (k.length > 1) { //If it is a special key if (special_keys[k] === code) {kp++;} } else if (opt.keycode) { if (opt.keycode === code) {kp++;} } else { //The special keys did not match if (character === k) {kp++;} else { if (shift_nums[character] && e.shiftKey) { //Stupid Shift key bug created by using lowercase character = shift_nums[character]; if (character === k) {kp++;} } } } } if (kp === keys.length && modifiers.ctrl.pressed === modifiers.ctrl.wanted && modifiers.shift.pressed === modifiers.shift.wanted && modifiers.alt.pressed === modifiers.alt.wanted && modifiers.meta.pressed === modifiers.meta.wanted) { callback(e); if (!opt.propagate) { //Stop the event //e.cancelBubble is supported by IE - this will kill the bubbling process. e.cancelBubble = true; e.returnValue = false; //e.stopPropagation works in Firefox. if (e.stopPropagation) { e.stopPropagation(); e.preventDefault(); } return false; } } }; this.all_shortcuts[shortcut_combination] = { 'callback': func, 'target': ele, 'event': opt.type }; //Attach the function with the event if (ele.addEventListener) {ele.addEventListener(opt.type, func, false);} else if (ele.attachEvent) {ele.attachEvent('on' + opt.type, func);} else {ele['on' + opt.type] = func;} }, //Remove the shortcut - just specify the shortcut and I will remove the binding 'remove': function(shortcut_combination) { shortcut_combination = shortcut_combination.toLowerCase(); var binding = this.all_shortcuts[shortcut_combination]; delete(this.all_shortcuts[shortcut_combination]); if (!binding) {return;} var type = binding.event; var ele = binding.target; var callback = binding.callback; if (ele.detachEvent) {ele.detachEvent('on' + type, callback);} else if (ele.removeEventListener) {ele.removeEventListener(type, callback, false);} else {ele['on' + type] = false;} } }; // END Shortcut function function errorHandler(callback) { try { callback(); } catch (ex) { console.error(_SCRIPT_NAME + ':', ex); } } function getHoursHtml(label, defaultText){ defaultText = defaultText || _DEFAULT_HOURS_TEXT; return label + ': ' + '<input class="btn btn-default btn-xs wmeph-btn" id="WMEPH_noHours" title="Add pasted hours to existing" type="button" value="Add hours" style="margin-bottom:4px; margin-right:0px"> ' + '<input class="btn btn-default btn-xs wmeph-btn" id="WMEPH_noHours_2" title="Replace existing hours with pasted hours" type="button" value="Replace all hours" style="margin-bottom:4px">' + '<textarea id="WMEPH-HoursPaste" wrap="off" autocomplete="off" style="overflow:auto;width:85%;max-width:85%;min-width:85%;font-size:0.85em;height:24px;min-height:24px;max-height:300px;padding-left:3px;color:#AAA">' + defaultText + '</textarea>'; } function getSelectedVenue(){ let venue; let features = W.selectionManager.getSelectedFeatures(); if (features.length && features[0].model.type === 'venue') { venue = features[0].model; } return venue; } function isAlwaysOpen(venue) { var hours = venue.attributes.openingHours; return hours.length === 1 && hours[0].days.length === 7 && hours[0].fromHour === '00:00' && hours[0].toHour ==='00:00'; } function isEmergencyRoom(venue) { return /(?:emergency\s+(?:room|department|dept))|\b(?:er|ed)\b/i.test(venue.attributes.name); } function isRestArea(venue) { return venue.attributes.categories.indexOf('REST_AREAS') > -1 && /rest\s*area/i.test(venue.attributes.name); } function getPvaSeverity(pvaValue, venue) { var isER = pvaValue === 'hosp' && isEmergencyRoom(venue); return (pvaValue ==='' || pvaValue === '0' || (pvaValue === 'hosp' && !isER)) ? 3 : (pvaValue ==='2') ? 1 : (pvaValue ==='3') ? 2 : 0; } function addPURWebSearchButton() { var purLayerObserver = new MutationObserver(panelContainerChanged); purLayerObserver.observe($('#map #panel-container')[0],{childList: true, subtree: true}); function panelContainerChanged() { if (!$('#WMEPH-HidePURWebSearch').prop('checked')) { var $panelNav = $('.place-update-edit.panel .categories.small'); if ($('#PHPURWebSearchButton').length === 0 && $panelNav.length > 0) { var $btn = $('<button>', {class:'btn btn-primary', id:'PHPURWebSearchButton', title: 'Search the web for this place. Do not copy info from 3rd party sources!' }) //NOTE: Don't use btn-block class. Causes conflict with URO+ "Done" button. .css({width:'100%',display:'block',marginTop:'4px',marginBottom:'4px'}) .text('Web Search') .click(() => { openWebSearch(); }); $panelNav.after($btn); } } } function buildSearchUrl(searchName, address) { searchName = searchName .replace(/[\/]/g, ' ') .trim(); address = address .replace(/No street, /, '') .replace(/No address/, '') .replace(/CR-/g, 'County Rd ') .replace(/SR-/g, 'State Hwy ') .replace(/US-/g, 'US Hwy ') .replace(/ CR /g, ' County Rd ') .replace(/ SR /g, ' State Hwy ') .replace(/ US /g, ' US Hwy ') .replace(/$CR /g, 'County Rd ') .replace(/$SR /g, 'State Hwy ') .replace(/$US /g, 'US Hwy ') .trim(); searchName = encodeURIComponent(searchName + (address.length > 0 ? ', ' + address: '')); return 'http://www.google.com/search?q=' + searchName; } function openWebSearch() { var newName = $('.place-update-edit.panel .name').first().text(); var addr = $('.place-update-edit.panel .address').first().text(); if ( $('#WMEPH-WebSearchNewTab').prop('checked') ) { window.open(buildSearchUrl(newName,addr)); } else { window.open(buildSearchUrl(newName,addr), searchResultsWindowName, searchResultsWindowSpecs); } } } // This function runs at script load, and splits the category dataset into the searchable categories. function makeCatCheckList(categoryData) { let headers = categoryData[0].split('|'); let idIndex = headers.indexOf('pc_wmecat'); let nameIndex = headers.indexOf('pc_transcat'); return categoryData.map(entry => { let splits = entry.split('|'); let id = splits[idIndex].trim(); if (id.length) { _CATEGORY_LOOKUP[splits[nameIndex].trim().toUpperCase()] = id; } return id; }); } // END makeCatCheckList function // This function runs at script load, and builds the search name dataset to compare the WME selected place name to. function makeNameCheckList(pnhData) { let headers = pnhData[0].split('|'); let nameIdx = headers.indexOf('ph_name'); let aliasesIdx = headers.indexOf('ph_aliases'); let category1Idx = headers.indexOf('ph_category1'); let searchNameBaseIdx = headers.indexOf('ph_searchnamebase'); let searchNameMidIdx = headers.indexOf('ph_searchnamemid'); let searchNameEndIdx = headers.indexOf('ph_searchnameend'); let disableIdx = headers.indexOf('ph_disable'); let specCaseIdx = headers.indexOf('ph_speccase'); let tighten = str => str.toUpperCase().replace(/ AND /g, '').replace(/^THE /g, '').replace(/[^A-Z0-9]/g, ''); let stripNonAlphaKeepCommas = str => str.toUpperCase().replace(/[^A-Z0-9,]/g, ''); return pnhData.map(entry => { let splits = entry.split('|'); let specCase = splits[specCaseIdx]; if (splits[disableIdx] !== '1' || specCase.indexOf('betaEnable') > -1 ) { let newNameList = [tighten(splits[nameIdx])]; if (splits[disableIdx] !== 'altName') { // Add any aliases let tempAliases = splits[aliasesIdx]; if ( tempAliases !== '' && tempAliases !== '0' && tempAliases !== '') { newNameList = newNameList.concat(tempAliases.replace(/,[^A-Za-z0-9]*/g, ',').split(',').map(alias => tighten(alias))); } } // The following code sets up alternate search names as outlined in the PNH dataset. // Formula, with P = PNH primary; A1, A2 = PNH aliases; B1, B2 = base terms; M1, M2 = mid terms; E1, E2 = end terms // Search list will build: P, A, B, PM, AM, BM, PE, AE, BE, PME, AME, BME. // Multiple M terms are applied singly and in pairs (B1M2M1E2). Multiple B and E terms are applied singly (e.g B1B2M1 not used). // Any doubles like B1E2=P are purged at the end to eliminate redundancy. let nameBaseStr = splits[searchNameBaseIdx]; if (nameBaseStr !== '0' && nameBaseStr !== '') { // If base terms exist, otherwise only the primary name is matched newNameList = newNameList.concat(stripNonAlphaKeepCommas(nameBaseStr).split(',')); let nameMidStr = splits[searchNameMidIdx]; if (nameMidStr !== '0' && nameMidStr !== '') { let pnhSearchNameMid = stripNonAlphaKeepCommas(nameMidStr).split(','); if (pnhSearchNameMid.length > 1) { // if there are more than one mid terms, it adds a permutation of the first 2 pnhSearchNameMid = pnhSearchNameMid.concat([pnhSearchNameMid[0] + pnhSearchNameMid[1], pnhSearchNameMid[1] + pnhSearchNameMid[0]]); } let midLen = pnhSearchNameMid.length; for (let extix = 1, len = newNameList.length; extix < len; extix++) { // extend the list by adding Mid terms onto the SearchNameBase names for (let midix = 0; midix < midLen; midix++) { newNameList.push(newNameList[extix] + pnhSearchNameMid[midix]); } } } let nameEndStr = splits[searchNameEndIdx]; if (nameEndStr !== '0' && nameEndStr !== '') { let pnhSearchNameEnd = stripNonAlphaKeepCommas(nameEndStr).split(','); let endLen = pnhSearchNameEnd.length; for (let extix = 1, len = newNameList.length; extix < len; extix++) { // extend the list by adding End terms onto all the SearchNameBase & Base+Mid names for (let endix = 0; endix < endLen; endix++) { newNameList.push(newNameList[extix] + pnhSearchNameEnd[endix]); } } } } // Clear out any empty entries newNameList = newNameList.filter(name => name.length > 1); // Next, add extensions to the search names based on the WME place category let category = splits[category1Idx].toUpperCase().replace(/[^A-Z0-9]/g, ''); let appendWords = []; if (category === 'HOTEL') { appendWords.push('HOTEL'); } else if (category === 'BANKFINANCIAL' && !/\bnotABank\b/.test(specCase)) { appendWords.push('BANK','ATM'); } else if (category === 'SUPERMARKETGROCERY') { appendWords.push('SUPERMARKET'); } else if (category === 'GYMFITNESS') { appendWords.push('GYM'); } else if (category === 'GASSTATION') { appendWords.push('GAS','GASOLINE','FUEL','STATION','GASSTATION'); } else if (category === 'CARRENTAL') { appendWords.push('RENTAL','RENTACAR','CARRENTAL','RENTALCAR'); } appendWords.forEach(word => newNameList = newNameList.concat(newNameList.map(name => name + word))); return _.uniq(newNameList).join('|').replace(/\|{2,}/g, '|').replace(/\|+$/g, ''); } else { // END if valid line return '00'; } }); } // END makeNameCheckList // Whitelist stringifying and parsing function saveWL_LS(compress) { venueWhitelistStr = JSON.stringify(venueWhitelist); if (compress) { if (venueWhitelistStr.length < 4800000 ) { // Also save to regular storage as a back up localStorage.setItem(WLlocalStoreName, venueWhitelistStr); } venueWhitelistStr = LZString.compressToUTF16(venueWhitelistStr); localStorage.setItem(WLlocalStoreNameCompressed, venueWhitelistStr); } else { localStorage.setItem(WLlocalStoreName, venueWhitelistStr); } } function loadWL_LS(decompress) { if (decompress) { venueWhitelistStr = localStorage.getItem(WLlocalStoreNameCompressed); venueWhitelistStr = LZString.decompressFromUTF16(venueWhitelistStr); } else { venueWhitelistStr = localStorage.getItem(WLlocalStoreName); } venueWhitelist = JSON.parse(venueWhitelistStr); } function backupWL_LS(compress) { venueWhitelistStr = JSON.stringify(venueWhitelist); if (compress) { venueWhitelistStr = LZString.compressToUTF16(venueWhitelistStr); localStorage.setItem(WLlocalStoreNameCompressed+Math.floor(Date.now() / 1000), venueWhitelistStr); } else { localStorage.setItem(WLlocalStoreName+Math.floor(Date.now() / 1000), venueWhitelistStr); } } // Removes duplicate strings from string array function uniq(a) { var seen = {}; return a.filter(item => seen.hasOwnProperty(item) ? false : (seen[item] = true)); } // END uniq function function phlog(m) { if ('object' === typeof m) { //m = JSON.stringify(m); } console.log('WMEPH' + (_IS_DEV_VERSION ? '-β' : '') + ': ' + m); } function phlogdev(msg, obj) { if (_USER.isDevUser) { console.log('WMEPH' + (_IS_DEV_VERSION ? '-β' : '') + ': ' + msg, (obj ? obj : '')); } } function zoomPlace() { let venue = getSelectedVenue(); if (venue) { W.map.moveTo(venue.geometry.getCentroid().toLonLat(), 7); } else { W.map.moveTo(WMEPHmousePosition, 5); } } function sortWithIndex(toSort) { for (var i = 0; i < toSort.length; i++) { toSort[i] = [toSort[i], i]; } toSort.sort( (left, right) => left[0] < right[0] ? -1 : 1 ); toSort.sortIndices = []; for (var j = 0; j < toSort.length; j++) { toSort.sortIndices.push(toSort[j][1]); toSort[j] = toSort[j][0]; } return toSort; } function destroyDupeLabels(){ _dupeLayer.destroyFeatures(); _dupeLayer.setVisibility(false); } // When a dupe is deleted, delete the dupe label function deleteDupeLabel(){ //phlog('Clearing dupe label...'); setTimeout(() => { var actionsList = W.model.actionManager.getActions(); var lastAction = actionsList[actionsList.length-1]; if ( 'undefined' !== typeof lastAction && lastAction.hasOwnProperty('object') && lastAction.object.hasOwnProperty('state') && lastAction.object.state === 'Delete' ) { if ( dupeIDList.indexOf(lastAction.object.attributes.id) > -1 ) { if (dupeIDList.length === 2) { _dupeLayer.destroyFeatures(); _dupeLayer.setVisibility(false); } else { var deletedDupe = _dupeLayer.getFeaturesByAttribute('dupeID', lastAction.object.attributes.id) ; _dupeLayer.removeFeatures(deletedDupe); dupeIDList.splice(dupeIDList.indexOf(lastAction.object.attributes.id),1); } phlog('Deleted a dupe'); } } /* else if ('undefined' !== typeof lastAction && lastAction.hasOwnProperty('feature') && lastAction.feature.hasOwnProperty('state') && lastAction.object.state === 'Update' && lastAction.hasOwnProperty('newGeometry') ) { // update position of marker } */ },20); } // Whitelist an item function whitelistAction(itemID, wlKeyName) { var venue = getSelectedVenue(); var addressTemp = venue.getAddress(); if ( addressTemp.hasOwnProperty('attributes') ) { addressTemp = addressTemp.attributes; } var itemGPS = OL.Layer.SphericalMercator.inverseMercator(venue.attributes.geometry.getCentroid().x,venue.attributes.geometry.getCentroid().y); if (!venueWhitelist.hasOwnProperty(itemID)) { // If venue is NOT on WL, then add it. venueWhitelist[itemID] = { }; } venueWhitelist[itemID][wlKeyName] = {active: true}; // WL the flag for the venue venueWhitelist[itemID].city = addressTemp.city.attributes.name; // Store city for the venue venueWhitelist[itemID].state = addressTemp.state.name; // Store state for the venue venueWhitelist[itemID].country = addressTemp.country.name; // Store country for the venue venueWhitelist[itemID].gps = itemGPS; // Store GPS coords for the venue saveWL_LS(true); // Save the WL to local storage WMEPH_WLCounter(); bannButt2.clearWL.active = true; } // Keep track of how many whitelists have been added since the last pull, alert if over a threshold (100?) function WMEPH_WLCounter() { localStorage.WMEPH_WLAddCount = parseInt(localStorage.WMEPH_WLAddCount)+1; if (localStorage.WMEPH_WLAddCount > 50) { alert('Don\'t forget to periodically back up your Whitelist data using the Pull option in the WMEPH settings tab.'); localStorage.WMEPH_WLAddCount = 2; } } function createObserver() { let observer = new MutationObserver(mutations => { mutations.forEach(mutation => { // Mutation is a NodeList and doesn't support forEach like an array for (var i = 0; i < mutation.addedNodes.length; i++) { var addedNode = mutation.addedNodes[i]; // Only fire up if it's a node if (addedNode.querySelector && addedNode.querySelector('.tab-scroll-gradient')) { // Normally, scrolling happens inside the tab-content div. When WMEPH adds stuff outside the landmark div, it effectively breaks that // and causes scrolling to occur at the main content div under edit-panel. That's actually OK, but need to disable a couple // artifacts that "stick around" with absolute positioning. $('#edit-panel .landmark').removeClass('separator-line'); $('#edit-panel .tab-scroll-gradient').css({display:'none'}); } } }); }); observer.observe(document.getElementById('edit-panel'), { childList: true, subtree: true }); } function appendServiceButtonIconCss() { let cssArray = [ '.serv-247 { width: 73px; height: 50px; display: inline-block; background: transparent url() top center no-repeat; }', '.serv-ac { width: 50px; height: 50px; display: inline-block; background: transparent url() top center no-repeat; }', '.serv-credit { width: 73px; height: 50px; display: inline-block; background: transparent url() top center no-repeat; }', '.serv-deliveries { width: 86px; height: 50px; display: inline-block; background: transparent url() top center no-repeat; }', '.serv-drivethru { width: 78px; height: 50px; display: inline-block; background: transparent url() top center no-repeat; }', '.serv-outdoor { width: 73px; height: 50px; display: inline-block; background: transparent url() top center no-repeat; }', '.serv-parking { width: 46px; height: 50px; display: inline-block; background: transparent url() top center no-repeat; }', '.serv-reservations { width: 55px; height: 50px; display: inline-block; background: transparent url() top center no-repeat; }', '.serv-restrooms { width: 49px; height: 50px; display: inline-block; background: transparent url() top center no-repeat; }', '.serv-takeaway { width: 34px; height: 50px; display: inline-block; background: transparent url() top center no-repeat; }', '.serv-valet { width: 50px; height: 50px; display: inline-block; background: transparent url() top center no-repeat; }', '.serv-wheelchair { width: 50px; height: 50px; display: inline-block; background: transparent url() top center no-repeat; }', '.serv-wifi { width: 67px; height: 50px; display: inline-block; background: transparent url() top center no-repeat; }' ]; $('head').append( $('<style>', {type:'text/css'}).html(cssArray.join('\n')) ); } // Function that checks current place against the Harmonization Data. Returns place data or "NoMatch" function harmoList(itemName, state2L, region3L, country, itemCats, item, placePL) { if (country !== 'USA' && country !== 'CAN') { alert('No PNH data exists for this country.'); return ['NoMatch']; } else { let pnhData = _PNH_DATA[country].pnh; let pnhNames = _PNH_DATA[country].pnhNames; let pnhHeaders = pnhData[0].split('|'); let ph_name_ix = pnhHeaders.indexOf('ph_name'); let ph_category1_ix = pnhHeaders.indexOf('ph_category1'); let ph_forcecat_ix = pnhHeaders.indexOf('ph_forcecat'); let ph_region_ix = pnhHeaders.indexOf('ph_region'); let ph_order_ix = pnhHeaders.indexOf('ph_order'); let ph_speccase_ix = pnhHeaders.indexOf('ph_speccase'); let ph_searchnameword_ix = pnhHeaders.indexOf('ph_searchnameword'); let PNHPriCat; // Primary category of PNH data let PNHForceCat; // Primary category of PNH data let approvedRegions; // filled with the regions that are approved for the place, when match is found let matchPNHRegionData = []; // array of matched data with regional approval let specCases, nmix, allowMultiMatch = false; let PNHOrderNum = []; let PNHNameTemp = []; let PNHNameMatch = false; // tracks match status itemName = itemName.toUpperCase().replace(/ AND /g, ' ').replace(/^THE /g, ''); let itemNameSpace = ' ' + itemName.replace(/[^A-Z0-9 ]/g, ' ').replace(/ {2,}/g, ' ') + ' '; itemName = itemName.replace(/[^A-Z0-9]/g, ''); // Clear all non-letter and non-number characters ( HOLLYIVY PUB #23 -- > HOLLYIVYPUB23 ) // for each place on the PNH list (skipping headers at index 0) for (let pnhIdx = 1, len = pnhNames.length; pnhIdx < len; pnhIdx++) { let PNHStringMatch = false; let pnhEntry = pnhData[pnhIdx]; let pnhEntrySplits = pnhEntry.split('|'); // Split the PNH place data into string array // Name Matching specCases = pnhEntrySplits[ph_speccase_ix]; if (specCases.indexOf('regexNameMatch') > -1) { // Check for regex name matching instead of "standard" name matching. var match = specCases.match(/regexNameMatch<>(.+?)<>/i); if (match !== null) { let reStr = match[1].replace(/\\/,'\\').replace(/<or>/g,'|'); var re = new RegExp(reStr,'i'); PNHStringMatch = re.test(item.attributes.name); } } else { if (specCases.indexOf('strMatchAny') > -1 || pnhEntrySplits[ph_category1_ix] === 'Hotel') { // Match any part of WME name with either the PNH name or any spaced names allowMultiMatch = true; var spaceMatchList = []; spaceMatchList.push( pnhEntrySplits[ph_name_ix].toUpperCase().replace(/ AND /g, ' ').replace(/^THE /g, '').replace(/[^A-Z0-9 ]/g, ' ').replace(/ {2,}/g, ' ') ); if (pnhEntrySplits[ph_searchnameword_ix] !== '') { spaceMatchList.push.apply( spaceMatchList,pnhEntrySplits[ph_searchnameword_ix].toUpperCase().replace(/, /g,',').split(',') ); } for (nmix=0; nmix<spaceMatchList.length; nmix++) { if ( itemNameSpace.includes(' '+spaceMatchList[nmix]+' ') ) { PNHStringMatch = true; } } } else { let nameComps = pnhNames[pnhIdx].split('|'); // splits all possible search names for the current PNH entry let itemNameNoNum = itemName.replace(/[^A-Z]/g, ''); // Clear non-letter characters for alternate match ( HOLLYIVYPUB23 --> HOLLYIVYPUB ) if (specCases.indexOf('strMatchStart') > -1) { // Match the beginning part of WME name with any search term for (nmix=0; nmix<nameComps.length; nmix++) { if ( itemName.startsWith(nameComps[nmix]) || itemNameNoNum.startsWith(nameComps[nmix]) ) { PNHStringMatch = true; } } } else if (specCases.indexOf('strMatchEnd') > -1) { // Match the end part of WME name with any search term for (nmix=0; nmix<nameComps.length; nmix++) { if ( itemName.endsWith(nameComps[nmix]) || itemNameNoNum.endsWith(nameComps[nmix]) ) { PNHStringMatch = true; } } } else { // full match of any term only if ( nameComps.indexOf(itemName) > -1 || nameComps.indexOf(itemNameNoNum) > -1 ) { PNHStringMatch = true; } } } } // if a match was found: if ( PNHStringMatch ) { // Compare WME place name to PNH search name list phlogdev('Matched PNH Order No.: '+pnhEntrySplits[ph_order_ix]); PNHPriCat = catTranslate(pnhEntrySplits[ph_category1_ix]); PNHForceCat = pnhEntrySplits[ph_forcecat_ix]; if (itemCats[0] === 'GAS_STATION') { // Gas stations only harmonized if the WME place category is already gas station (prevents Costco Gas becoming Costco Store) PNHForceCat = '1'; } let PNHMatchProceed = false; if ( PNHForceCat === '1' && itemCats.indexOf(PNHPriCat) === 0 ) { // Name and primary category match PNHMatchProceed = true; } else if ( PNHForceCat === '2' && itemCats.indexOf(PNHPriCat) > -1 ) { // Name and any category match PNHMatchProceed = true; } else if ( PNHForceCat === '0' || PNHForceCat === '') { // Name only match PNHMatchProceed = true; } if (PNHMatchProceed) { approvedRegions = pnhEntrySplits[ph_region_ix].replace(/ /g, '').toUpperCase().split(','); // remove spaces, upper case the approved regions, and split by commas if (approvedRegions.indexOf(state2L) > -1 || approvedRegions.indexOf(region3L) > -1 || // if the WME-selected item matches the state, region approvedRegions.indexOf(country) > -1 || // OR if the country code is in the data then it is approved for all regions therein $('#WMEPH-RegionOverride').prop('checked')) { // OR if region override is selected (dev setting matchPNHRegionData.push(pnhEntry); bannButt.placeMatched = new Flag.PlaceMatched(); if (!allowMultiMatch) { return matchPNHRegionData; // Return the PNH data string array to the main script } } else { PNHNameMatch = true; // PNH match found (once true, stays true) //matchPNHData.push(pnhEntry); // Pull the data line from the PNH data table. (**Set in array for future multimatch features) PNHNameTemp.push(pnhEntrySplits[ph_name_ix]); // temp name for approval return PNHOrderNum.push(pnhEntrySplits[ph_order_ix]); // temp order number for approval return } } } } // END loop through PNH places // If NO (name & region) match was found: if (bannButt.placeMatched) { return matchPNHRegionData; } else if (PNHNameMatch) { // if a name match was found but not for region, prod the user to get it approved bannButt.ApprovalSubmit = new Flag.ApprovalSubmit(region3L, PNHOrderNum, PNHNameTemp, placePL); return ['ApprovalNeeded', PNHNameTemp, PNHOrderNum]; } else { // if no match was found, suggest adding the place to the sheet if it's a chain bannButt.NewPlaceSubmit = new Flag.NewPlaceSubmit(); return ['NoMatch']; } } } // END harmoList function function onObjectsChanged() { deleteDupeLabel(); // This is code to handle updating the banner when changes are made external to the script. let venue = getSelectedVenue(); if ($('#WMEPH_banner').length > 0 && venue) { var actions = W.model.actionManager.getActions(); var lastAction = actions[actions.length - 1]; if (lastAction && lastAction.object && lastAction.object.type === 'venue' && lastAction.attributes && lastAction.attributes.id === venue.attributes.id) { if (lastAction.newAttributes && lastAction.newAttributes.entryExitPoints) { harmonizePlaceGo(venue, 'harmonize'); } } } } // This should be called after new venues are saved (using venues'objectssynced' event), so the new IDs can be retrieved and used // to replace the temporary IDs in the whitelist. If WME errors during save, this function may not run. At that point, the // temporary IDs can no longer be traced to the new IDs so the WL for those new venues will be orphaned, and the temporary IDs // will be removed from the WL store the next time the script starts. function syncWL(newVenues) { newVenues.forEach(newVenue => { var oldID = newVenue._prevID; var newID = newVenue.attributes.id; if (oldID && newID && venueWhitelist[oldID]) { venueWhitelist[newID] = venueWhitelist[oldID]; delete venueWhitelist[oldID]; } }); saveWL_LS(true); } function toggleXrayMode(enable) { localStorage.setItem('WMEPH_xrayMode_enabled', $('#layer-switcher-item_wmeph_x-ray_mode').prop('checked')); let commentsLayer = W.map.getLayerByUniqueName('mapComments'); let gisLayer = W.map.getLayerByUniqueName('__wmeGISLayers'); let commentRuleSymb = commentsLayer.styleMap.styles.default.rules[0].symbolizer; if (enable) { layer.styleMap.styles['default'].rules = layer.styleMap.styles['default'].rules.filter(rule => rule.wmephDefault !== 'default'); W.map.roadLayers[0].opacity = 0.25; W.map.baseLayer.opacity = 0.25; commentRuleSymb.Polygon.strokeColor = '#888'; commentRuleSymb.Polygon.fillOpacity = 0.2; if (gisLayer) gisLayer.setOpacity(0.4); } else { layer.styleMap.styles['default'].rules = layer.styleMap.styles['default'].rules.filter(rule => rule.wmephStyle !== 'xray'); W.map.roadLayers[0].opacity = 1; W.map.baseLayer.opacity = 1; commentRuleSymb.Polygon.strokeColor = '#fff'; commentRuleSymb.Polygon.fillOpacity = 0.4; if (gisLayer) gisLayer.setOpacity(1); initializeHighlights(); layer.redraw(); } commentsLayer.redraw(); W.map.roadLayers[0].redraw(); W.map.baseLayer.redraw(); if (!enable) return; let defaultPointRadius = 6; var ruleGenerator = function(value, symbolizer) { return new W.Rule({ filter: new OL.Filter.Comparison({ type: '==', value: value, evaluate: function(venue) { return venue && venue.model && venue.model.attributes.wmephSeverity === this.value; } }), symbolizer: symbolizer, wmephStyle: 'xray' }); }; var severity0 = ruleGenerator(0, { Point:{ strokeWidth: 1.67, strokeColor: '#888', pointRadius: 5, fillOpacity: 0.25, fillColor: 'white', zIndex: 0 }, Polygon: { strokeWidth: 1.67, strokeColor: '#888', fillOpacity: 0 } }); var severityLock = ruleGenerator('lock', { Point: { strokeColor: 'white', fillColor: '#080', fillOpacity: 1, strokeLinecap: 1, strokeDashstyle: '4 2', strokeWidth: 2.5, pointRadius: defaultPointRadius }, Polygon: { strokeColor: 'white', fillColor: '#0a0', fillOpacity: 0.4, strokeDashstyle: '4 2', strokeWidth: 2.5 } }); var severity1 = ruleGenerator(1, { strokeColor: 'white', strokeWidth: 2, pointRadius: defaultPointRadius, fillColor: '#0055ff' }); var severityLock1 = ruleGenerator('lock1', { pointRadius: defaultPointRadius, fillColor: '#0055ff', strokeColor: 'white', strokeLinecap: '1', strokeDashstyle: '4 2', strokeWidth: 2.5 }); var severity2 = ruleGenerator(2, { Point: { fillColor: '#ca0', strokeColor: 'white', strokeWidth: 2, pointRadius: defaultPointRadius }, Polygon: { fillColor: '#ff0', strokeColor: 'white', strokeWidth: 2, fillOpacity: 0.4 } }); var severity3 = ruleGenerator(3, { strokeColor: 'white', strokeWidth: 2, pointRadius: defaultPointRadius, fillColor: '#ff0000' }); var severity4 = ruleGenerator(4, { fillColor: '#f42', strokeLinecap: 1, strokeWidth: 2, strokeDashstyle: '4 2' }); var severityHigh = ruleGenerator(5, { fillColor: 'black', strokeColor: '#f4a', strokeLinecap: 1, strokeWidth: 4, strokeDashstyle: '4 2', pointRadius: defaultPointRadius }); var severityAdLock = ruleGenerator('adLock', { pointRadius: 12, fillColor: 'yellow', fillOpacity: 0.4, strokeColor: '#000', strokeLinecap: 1, strokeWidth: 10, strokeDashstyle: '4 2' }); Array.prototype.push.apply(layer.styleMap.styles['default'].rules, [severity0, severityLock, severity1, severityLock1, severity2, severity3, severity4, severityHigh, severityAdLock]); layer.redraw(); } function initializeHighlights() { var ruleGenerator = function(value, symbolizer) { return new W.Rule({ filter: new OL.Filter.Comparison({ type: '==', value: value, evaluate: function(venue) { return venue && venue.model && venue.model.attributes.wmephSeverity === this.value; } }), symbolizer: symbolizer, wmephStyle: 'default' }); }; var severity0 = ruleGenerator(0, { 'pointRadius': '5', 'strokeWidth': '4', 'strokeColor': '#24ff14' }); var severityLock = ruleGenerator('lock', { 'pointRadius': '5', 'strokeColor': '#24ff14', 'strokeLinecap': '1', 'strokeDashstyle': '7 2', 'strokeWidth': '5' }); var severity1 = ruleGenerator(1, { 'strokeColor': '#0055ff', 'strokeWidth': '4', 'pointRadius': '7' }); var severityLock1 = ruleGenerator('lock1', { 'pointRadius': '5', 'strokeColor': '#0055ff', 'strokeLinecap': '1', 'strokeDashstyle': '7 2', 'strokeWidth': '5' }); var severity2 = ruleGenerator(2, { 'strokeColor': '#ff0', 'strokeWidth': '6', 'pointRadius': '8' }); var severity3 = ruleGenerator(3, { 'strokeColor': '#ff0000', 'strokeWidth': '4', 'pointRadius': '8' }); var severity4 = ruleGenerator(4, { 'fillColor': 'black', 'fillOpacity': '0.35', 'strokeColor': '#f42', 'strokeLinecap': '1', 'strokeWidth': '13', 'strokeDashstyle': '4 2' }); var severityHigh = ruleGenerator(5, { 'pointRadius': '12', 'fillColor': 'black', 'fillOpacity': '0.4', 'strokeColor': '#f4a', 'strokeLinecap': '1', 'strokeWidth': '10', 'strokeDashstyle': '4 2' }); var severityAdLock = ruleGenerator('adLock', { 'pointRadius': '12', 'fillColor': 'yellow', 'fillOpacity': '0.4', 'strokeColor': '#000', 'strokeLinecap': '1', 'strokeWidth': '10', 'strokeDashstyle': '4 2' }); function plaTypeRuleGenerator(value, symbolizer) { return new W.Rule({ filter: new OL.Filter.Comparison({ type: '==', value: value, evaluate: function(venue) { if ($('#WMEPH-PLATypeFill').prop('checked') && venue && venue.model && venue.model.attributes.categories && venue.model.attributes.categoryAttributes && venue.model.attributes.categoryAttributes.PARKING_LOT && venue.model.attributes.categories.indexOf('PARKING_LOT') > -1) { var type = venue.model.attributes.categoryAttributes.PARKING_LOT.parkingType; return (!type && this.value === 'public') || (type && (type.toLowerCase() === this.value)); } } }), symbolizer: symbolizer, wmephStyle: 'default' }); } var publicPLA = plaTypeRuleGenerator('public', { fillColor: '#0000FF', fillOpacity: '0.25' }); var restrictedPLA = plaTypeRuleGenerator('restricted', { fillColor: '#FFFF00', fillOpacity: '0.3' }); var privatePLA = plaTypeRuleGenerator('private', { fillColor: '#FF0000', fillOpacity: '0.25' }); Array.prototype.push.apply(layer.styleMap.styles['default'].rules, [severity0, severityLock, severity1, severityLock1, severity2, severity3, severity4, severityHigh, severityAdLock,publicPLA, restrictedPLA, privatePLA]); } /** * To highlight a place, set the wmephSeverity attribute to the desired highlight level. * @param venues {array of venues, or single venue} Venues to check for highlights. * @param force {boolean} Force recalculation of highlights, rather than using cached results. */ function applyHighlightsTest(venues, force) { if (!layer) return; venues = venues ? _.isArray(venues) ? venues : [venues] : []; let storedBannButt = bannButt, storedBannServ = bannServ, storedBannButt2 = bannButt2; let t0 = performance.now(); let doHighlight = $('#WMEPH-ColorHighlighting').prop('checked'); let disableRankHL = $('#WMEPH-DisableRankHL').prop('checked'); _.each(venues, venue => { if (venue && venue.type === 'venue' && venue.attributes) { // Highlighting logic would go here // Severity can be: 0, 'lock', 1, 2, 3, 4, or 'high'. Set to // anything else to use default WME style. if ( doHighlight && !(disableRankHL && venue.attributes.lockRank > _USER.rank - 1)) { try { let id = venue.attributes.id; let severity; let cachedResult; if (force || !isNaN(id) || ((cachedResult = _resultsCache[id]) === undefined) || (venue.updatedOn > cachedResult.u)) { severity = harmonizePlaceGo(venue,'highlight'); if (isNaN(id)) _resultsCache[id] = { s: severity, u: venue.updatedOn || -1 }; } else { severity = cachedResult.s; } venue.attributes.wmephSeverity = severity; } catch (err) { console.error('WMEPH highlight error: ', err); } } else { venue.attributes.wmephSeverity = 'default'; } } }); // Trim the cache if it's over the max size limit. let keys = Object.keys(_resultsCache); if (keys.length > _MAX_CACHE_SIZE) { let trimSize = _MAX_CACHE_SIZE * 0.8; for (let i = keys.length - 1; i > trimSize; i--) { delete _resultsCache[keys[i]]; } } let venue = getSelectedVenue(); if (venue) { venue.attributes.wmephSeverity = harmonizePlaceGo(venue,'highlight'); bannButt = storedBannButt; bannServ = storedBannServ; bannButt2 = storedBannButt2; } phlogdev('Ran highlighter in ' + Math.round((performance.now() - t0)*10)/10 + ' milliseconds.'); //phlogdev('WMEPH cache size: ' + Object.keys(_resultsCache).length); //layer.redraw(); } // Set up CH loop function bootstrapWMEPH_CH() { if ( localStorage.getItem('WMEPH-ColorHighlighting') === '1' ) { // Add listeners W.model.venues.on('objectschanged', e => errorHandler(() => { if (!_disableHighlightTest) { applyHighlightsTest(e, true); layer.redraw(); } })); //W.model.venues.on('objectsadded', e => errorHandler(() => applyHighlightsTest(e))); W.map.landmarkLayer.events.register( 'beforefeaturesadded', null, e => errorHandler(() => applyHighlightsTest(e.features.map(f => f.model))) ); // Clear the cache (highlight severities may need to be updated). _resultsCache = {}; // Apply the colors applyHighlightsTest(W.model.venues.getObjectArray()); layer.redraw(); } else { // reset the colors to default applyHighlightsTest(W.model.venues.getObjectArray()); layer.redraw(); } } // Change place.name to title case function toTitleCaseStrong(str) { if (!str) { return str; } str = str.trim(); let parensParts = str.match(/\(.*?\)/g); if (parensParts) { for (let i=0; i<parensParts.length; i++) { str = str.replace(parensParts[i], '%' + i + '%'); } } // Get indexes of Mac followed by a cap, as in MacMillan. let macIndexes = []; let macRegex = /\bMac[A-Z]/g; let macMatch; while ((macMatch = macRegex.exec(str)) !== null) { macIndexes.push(macMatch.index); } let allCaps = (str === str.toUpperCase()); // Cap first letter of each word str = str.replace(/([A-Za-z\u00C0-\u017F][^\s-\/]*) */g, function(txt) { // If first letter is lower case, followed by a cap, then another lower case letter... ignore it. Example: iPhone if (/^[a-z][A-Z0-9][a-z]/.test(txt)) { return txt; } // If word starts with De/Le/La followed by uppercase then lower case, is 5+ characters long... assume it should be like "DeBerry". if (/^([dDlL]e|[lL]a)[A-Z][a-zA-Z\u00C0-\u017F]{2,}/.test(txt)) { return txt.charAt(0).toUpperCase() + txt.charAt(1).toLowerCase() + txt.charAt(2) + txt.substr(3).toLowerCase(); } return ((txt === txt.toUpperCase()) && !allCaps) ? txt : txt.charAt(0).toUpperCase() + txt.substr(1).toLowerCase(); }) // Cap O'Reilley's, L'Amour, D'Artagnan as long as 5+ letters .replace(/\b[oOlLdD]'[A-Za-z']{3,}/g, function(txt) { return ((txt === txt.toUpperCase()) && !allCaps) ? txt : txt.charAt(0).toUpperCase() + txt.charAt(1) + txt.charAt(2).toUpperCase() + txt.substr(3).toLowerCase(); }) // Cap McFarley's, as long as 5+ letters long .replace(/\b[mM][cC][A-Za-z']{3,}/g, function(txt) { return ((txt === txt.toUpperCase()) && !allCaps) ? txt : txt.charAt(0).toUpperCase() + txt.charAt(1).toLowerCase() + txt.charAt(2).toUpperCase() + txt.substr(3).toLowerCase(); }) // anything with an "&" sign, cap the word after & .replace(/&\w+/g, function(txt) { return ((txt === txt.toUpperCase()) && !allCaps) ? txt : txt.charAt(0) + txt.charAt(1).toUpperCase() + txt.substr(2); }) // lowercase any from the ignoreWords list .replace(/[^ ]+/g, function(txt) { let txtLC = txt.toLowerCase(); return (_TITLECASE_SETTINGS.ignoreWords.indexOf(txtLC) > -1) ? txtLC : txt; }) // uppercase any from the capWords List .replace(/[^ ]+/g, function(txt) { let txtLC = txt.toUpperCase(); return (_TITLECASE_SETTINGS.capWords.indexOf(txtLC) > -1) ? txtLC : txt; }) // preserve any specific words .replace(/[^ ]+/g, function(txt) { let txtUC = txt.toUpperCase(); return _TITLECASE_SETTINGS.specWords.find(specWord => specWord.toUpperCase() === txtUC) || txt; }) // Fix 1st, 2nd, 3rd, 4th, etc. .replace(/\b(\d*1)st\b/gi, '$1st') .replace(/\b(\d*2)nd\b/gi, '$1nd') .replace(/\b(\d*3)rd\b/gi, '$1rd') .replace(/\b(\d+)th\b/gi, '$1th'); // Cap first letter of entire name if it's not something like iPhone or eWhatever. if (!/^[a-z][A-Z0-9][a-z]/.test(str)) str = str.charAt(0).toUpperCase() + str.substr(1); if (parensParts) { for (let i = 0, len = parensParts.length; i < len; i++) { str = str.replace('%' + i + '%', parensParts[i]); } } // Fix any Mac... words. macIndexes.forEach(idx => { str = str.substr(0,idx+3) + str.substr(idx+3,1).toUpperCase() + str.substr(idx+4); }); return str; } // normalize phone function normalizePhone(s, outputFormat, returnType, item, region) { if ( !s && returnType === 'existing' ) { bannButt.phoneMissing = Flag.PhoneMissing.eval(item, _wl, region, outputFormat); return s; } s = s.replace(/(\d{3}.*)\W+(?:extension|ext|xt|x).*/i, '$1'); var s1 = s.replace(/\D/g, ''); // remove non-number characters var m = s1.match(/^1?([2-9]\d{2})([2-9]\d{2})(\d{4})$/); // Ignore leading 1, and also don't allow area code or exchange to start with 0 or 1 (***USA/CAN specific) if (!m) { // then try alphanumeric matching if (s) { s = s.toUpperCase(); } s1 = s.replace(/[^0-9A-Z]/g, '').replace(/^\D*(\d)/,'$1').replace(/^1?([2-9][0-9]{2}[0-9A-Z]{7,10})/g,'$1'); s1 = replaceLetters(s1); m = s1.match(/^([2-9]\d{2})([2-9]\d{2})(\d{4})(?:.{0,3})$/); // Ignore leading 1, and also don't allow area code or exchange to start with 0 or 1 (***USA/CAN specific) if (!m) { if ( returnType === 'inputted' ) { return 'badPhone'; } else { bannButt.phoneInvalid = new Flag.PhoneInvalid(); return s; } } else { return String.plFormat(outputFormat, m[1], m[2], m[3]); } } else { return String.plFormat(outputFormat, m[1], m[2], m[3]); } } // Alphanumeric phone conversion function replaceLetters(number) { var conversionMap = _({ 2: /A|B|C/, 3: /D|E|F/, 4: /G|H|I/, 5: /J|K|L/, 6: /M|N|O/, 7: /P|Q|R|S/, 8: /T|U|V/, 9: /W|X|Y|Z/ }); number = typeof number === 'string' ? number.toUpperCase() : ''; return number.replace(/[A-Z]/g, function(match) { return conversionMap.findKey(function(re) { return re.test(match); }); }); } // Add array of actions to a MultiAction to be executed at once (counts as one edit for redo/undo purposes) function executeMultiAction(actions, description) { if(actions.length > 0) { var m_action = new MultiAction(); m_action.setModel(W.model); m_action._description = description || m_action._description || 'Change(s) made by WMEPH'; actions.forEach( action => {m_action.doSubAction(action);} ); W.model.actionManager.add(m_action); } } // Split localizer (suffix) part of names, like "SUBWAY - inside Walmart". function getNameParts(name) { var splits = name.match(/(.*?)(\s+[-\(–].*)*$/); return {base: splits[1], suffix: splits[2]}; } function addUpdateAction(venue, updateObj, actions) { var action = new UpdateObject(venue, updateObj); if (actions) { actions.push(action); } else { W.model.actionManager.add(action); } } function setServiceChecked(servBtn, checked, actions) { var servID = WMEServicesArray[servBtn.servIDIndex]; var checkboxChecked = $('#service-checkbox-'+servID).prop('checked'); let venue = getSelectedVenue(); if (checkboxChecked !== checked) { _updatedFields['services_' + servID].updated = true; } var toggle = typeof checked === 'undefined'; var noAdd = false; checked = (toggle) ? !servBtn.checked : checked; if (checkboxChecked === servBtn.checked && checkboxChecked !== checked) { servBtn.checked = checked; var services; if (actions) { for (var i=0; i<actions.length; i++ ) { var existingAction = actions[i]; if (existingAction.newAttributes && existingAction.newAttributes.services) { services = existingAction.newAttributes.services; } } } if (!services) { services = venue.attributes.services.slice(0); } else { noAdd = services.indexOf(servID) > -1; } if (checked) { services.push(servID); } else { var index = services.indexOf(servID); if (index > -1) { services.splice(index, 1); } } if (!noAdd) { addUpdateAction(venue, {services:services}, actions); } } updateServicesChecks(bannServ); if (!toggle) servBtn.active = checked; } // Normalize url function normalizeURL(s, lc, skipBannerActivate, venue, region) { var regionsThatWantPLAUrls = ['SER']; if ((!s || s.trim().length === 0) && !skipBannerActivate) { // Notify that url is missing and provide web search to find website and gather data (provided for all editors) let hasOperator = venue.attributes.brand && W.model.venues.categoryBrands.PARKING_LOT.indexOf(venue.attributes.brand) !== -1; if (!venue.isParkingLot() || (venue.isParkingLot() && (regionsThatWantPLAUrls.indexOf(region) > -1 || hasOperator))) { bannButt.urlMissing = new Flag.UrlMissing(); if (_wl.urlWL || (venue.isParkingLot() && !hasOperator)) { bannButt.urlMissing.severity = 0; bannButt.urlMissing.WLactive = false; } } //bannButt.webSearch.active = true; // Activate websearch button return s; } s = s.replace(/ \(.*/g, ''); // remove anything with parentheses after it s = s.replace(/ /g, ''); // remove any spaces var m = s.match(/^http:\/\/(.*)$/i); // remove http:// if (m) { s = m[1]; } if (lc) { // lowercase the entire domain s = s.replace(/[^\/]+/i, function(txt) { // lowercase the domain return (txt === txt.toLowerCase()) ? txt : txt.toLowerCase(); }); } else { // lowercase only the www and com s = s.replace(/www\./i, 'www.'); s = s.replace(/\.com/i, '.com'); } m = s.match(/^(.*)\/pages\/welcome.aspx$/i); // remove unneeded terms if (m) { s = m[1]; } m = s.match(/^(.*)\/pages\/default.aspx$/i); // remove unneeded terms if (m) { s = m[1]; } // m = s.match(/^(.*)\/index.html$/i); // remove unneeded terms // if (m) { s = m[1]; } // m = s.match(/^(.*)\/index.htm$/i); // remove unneeded terms // if (m) { s = m[1]; } // m = s.match(/^(.*)\/index.php$/i); // remove unneeded terms // if (m) { s = m[1]; } m = s.match(/^(.*)\/$/i); // remove final slash if (m) { s = m[1]; } if (!s || s.trim().length === 0 || !/(^https?:\/\/)?\w+\.\w+/.test(s)) s = 'badURL'; return s; } // END normalizeURL function // Only run the harmonization if a venue is selected function harmonizePlace() { // Beta version for approved users only if (_IS_DEV_VERSION && !_USER.isBetaUser) { alert('Please sign up to beta-test this script version.\nSend a PM or Slack-DM to MapOMatic or Tonestertm, or post in the WMEPH forum thread. Thanks.'); return; } // Only run if a single place is selected let venue = getSelectedVenue(); if (venue) { _updatedFields.reset(); blurAll(); // focus away from current cursor position _disableHighlightTest = true; harmonizePlaceGo(venue,'harmonize'); _disableHighlightTest = false; applyHighlightsTest(venue); } else { // Remove duplicate labels _dupeLayer.destroyFeatures(); } } // Abstract flag classes. Must be declared outside the "Flag" namespace. class FlagBase { constructor(active, severity, message) { this.active = active; this.severity = severity; this.message = message; } } class ActionFlag extends FlagBase { constructor(active, severity, message, value, title) { super(active, severity, message); this.value = value; this.title = title; } action() {} // overwrite this } class WLFlag extends FlagBase { constructor(active, severity, message, WLactive, WLtitle, WLkeyName) { super(active, severity, message); this.WLactive = WLactive; this.WLtitle = WLtitle; this.WLkeyName = WLkeyName; } WLaction() { let venue = getSelectedVenue(); whitelistAction(venue.attributes.id, this.WLkeyName); harmonizePlaceGo(venue, 'harmonize'); } } class WLActionFlag extends WLFlag { constructor(active, severity, message, value, title, WLactive, WLtitle, WLkeyName) { super(active, severity, message, WLactive, WLtitle, WLkeyName); this.value = value; this.title = title; } action() {} // overwrite this } // Namespace to keep these grouped. let Flag = { HnDashRemoved: class extends FlagBase { constructor() { super(true, 0, 'Dash removed from house number. Verify'); } }, FullAddressInference: class extends FlagBase { constructor() { super(true, 3, 'Missing address was inferred from nearby segments. Verify the address and run script again.'); } static eval(venue, addr, actions) { let result = {}; if (!addr.state || !addr.country) { if (W.map.getZoom() < 4 ) { if ( $('#WMEPH-EnableIAZoom').prop('checked') ) { W.map.moveTo(venue.geometry.getCentroid().toLonLat(), 5); } else { alert('No address and the state cannot be determined. Please zoom in and rerun the script. You can enable autozoom for this type of case in the options.'); } result.exit = true; // don't run the rest of the script } else { let inferredAddress = WMEPH_inferAddress(7); // Pull address info from nearby segments if (inferredAddress && inferredAddress.attributes) inferredAddress = inferredAddress.attributes; if (inferredAddress && inferredAddress.state && inferredAddress.country ) { if ( $('#WMEPH-AddAddresses').prop('checked') ) { // update the item's address if option is enabled updateAddress(venue, inferredAddress, actions); result.inferredAddress = inferredAddress; _updatedFields.address.updated = true; result.flag = new Flag.FullAddressInference(); result.noLock = true; // let hn = venue.attributes.houseNumber; // if (hn && hn.replace(/[^0-9A-Za-z]/g,'').length > 0 ) { // result.flag = new Flag.FullAddressInference(); // result.noLock = true; // } } else { if (['JUNCTION_INTERCHANGE'].indexOf(newCategories[0]) === -1) { bannButt.cityMissing = new Flag.CityMissing(); result.noLock = true; } } } else { // if the inference doesn't work... alert('This place has no address data and the address cannot be inferred from nearby segments. Please edit the address and run WMEPH again.'); result.exit = true; // don't run the rest of the script } } } return result; } static evalHL(venue, addr) { let result = null; if (!addr.state || !addr.country) { if ( venue.attributes.adLocked ) { result = 'adLock'; } else { let cat = venue.attributes.categories; if ( containsAny(cat, ['HOSPITAL_MEDICAL_CARE','HOSPITAL_URGENT_CARE','GAS_STATION']) ) { phlogdev('Unaddressed HUC/GS'); result = 5; } else if ( cat.indexOf('JUNCTION_INTERCHANGE') > -1 ) { result = 0; } else { result = 3; } } } return result; } }, NameMissing: class extends FlagBase { constructor() { super(true, 3, 'Name is missing.'); } }, PlaIsPublic: class extends FlagBase { constructor() { super(true, 0, 'If this does not meet the requirements for a <a href="https://wazeopedia.waze.com/wiki/USA/Places/Parking_lot#Lot_Type" target="_blank" style="color:5a5a73">public parking lot</a>, change to:<br>'); } }, PlaNameMissing: class extends FlagBase { constructor() { super(true, 1, 'Name is missing.'); this.message += _USER.rank < 3 ? ' Request an R3+ lock to confirm unnamed parking lot.' : ' Lock to 3+ to confirm unnamed parking lot.'; } }, PlaNameNonStandard: class extends WLFlag { constructor() { super(true, 2, 'Parking lot names typically contain "Parking", "Lot", and/or "Garage"', true, 'Whitelist non-standard PLA name', 'plaNameNonStandard'); } static eval(venue, wl) { let result = {flag: null}; if (!wl.plaNameNonStandard) { let name = venue.attributes.name; let state = venue.getAddress().getStateName(); let re = state === 'Quebec' ? /\b(parking|stationnement)\b/i : /\b((park[ -](and|&|'?n'?)[ -]ride)|parking|lot|garage)\b/i; if (venue.isParkingLot() && name && !re.test(name)) { result.flag = new Flag.PlaNameNonStandard(); } } return result; } }, IndianaLiquorStoreHours: class extends WLFlag { constructor() { super(true, 0, 'If this is a liquor store, check the hours. As of Feb 2018, liquor stores in Indiana are allowed to be open between noon and 8 pm on Sunday.', true, 'Whitelist Indiana liquor store hours', 'indianaLiquorStoreHours'); } }, HoursOverlap: class extends FlagBase { constructor() { super(true, 3, 'Overlapping hours of operation. Place might not save.'); } }, UnmappedRegion: class extends WLFlag { constructor() { super(true, 3, 'This category is usually not mapped in this region.', true, 'Whitelist unmapped category', 'unmappedRegion'); } }, RestAreaName: class extends WLFlag { constructor() { super(true, 3, 'Rest area name is out of spec. Use the Rest Area wiki button below to view formats.', true, 'Whitelist rest area name', 'restAreaName'); } }, RestAreaNoTransportation: class extends ActionFlag { constructor() { super(true, 2, 'Rest areas should not use the Transportation category.', 'Remove it?'); } action() { let ix = newCategories.indexOf('TRANSPORTATION'); if (ix > -1) { let venue = getSelectedVenue(); newCategories.splice(ix, 1); _updatedFields.categories.updated = true; addUpdateAction(venue, { categories: newCategories }); harmonizePlaceGo(venue,'harmonize'); } } }, RestAreaGas: class extends FlagBase { constructor() { super(true, 3, 'Gas stations at Rest Areas should be separate area places.'); } }, RestAreaScenic: class extends WLActionFlag { constructor() { super(true, 0, 'Verify that the "Scenic Overlook" category is appropriate for this rest area. If not: ', 'Remove it', 'Remove "Scenic Overlook" category.', true, 'Whitelist place', 'restAreaScenic'); } action() { var ix = newCategories.indexOf('SCENIC_LOOKOUT_VIEWPOINT'); if (ix > -1) { let venue = getSelectedVenue(); newCategories.splice(ix, 1); _updatedFields.categories.updated = true; addUpdateAction(venue, { categories: newCategories }); harmonizePlaceGo(venue,'harmonize'); } } }, RestAreaSpec: class extends WLActionFlag { constructor() { super(true, 3, 'Is this a rest area?', 'Yes', 'Update with proper categories and services.', true, 'Whitelist place', 'restAreaSpec'); } action () { let venue = getSelectedVenue(); let actions = []; // update categories according to spec newCategories = insertAtIX(newCategories,'REST_AREAS',0); actions.push(new UpdateObject(venue, { categories: newCategories })); _updatedFields.categories.updated = true; // make it 24/7 actions.push(new UpdateObject(venue, { openingHours: [{days: [1,2,3,4,5,6,0], fromHour: '00:00', toHour: '00:00'}] })); _updatedFields.openingHours.updated = true; bannServ.add247.checked = true; bannServ.addParking.actionOn(actions); // add parking service bannServ.addWheelchair.actionOn(actions); // add parking service bannButt.restAreaSpec.active = false; // reset the display flag executeMultiAction(actions); _disableHighlightTest = true; harmonizePlaceGo(venue,'harmonize'); _disableHighlightTest = false; applyHighlightsTest(venue); } }, GasMismatch: class extends WLFlag{ constructor() { super(true, 3, '<a href="https://wazeopedia.waze.com/wiki/USA/Places/Gas_station#Name" target="_blank" class="red">Gas brand should typically be included in the place name.</a>', true, 'Whitelist gas brand / name mismatch', 'gasMismatch'); } }, GasUnbranded: class extends FlagBase { constructor() { super(true, 3, '"Unbranded" should not be used for the station brand. Change to correct brand or use the blank entry at the top of the brand list.'); } static eval(venue) { let result = {flag: null}; if (venue.isGasStation() && venue.attributes.brand === 'Unbranded' ) { // Unbranded is not used per wiki result.flag = new Flag.GasUnbranded(); result.noLock = true; } return result; } }, GasMkPrim: class extends ActionFlag { constructor() { super(true, 3, 'Gas Station is not the primary category', 'Fix', 'Make the Gas Station category the primary category.'); } action() { let venue = getSelectedVenue(); newCategories = insertAtIX(newCategories,'GAS_STATION',0); // Insert/move Gas category in the first position _updatedFields.categories.updated = true; addUpdateAction(venue, { categories: newCategories }); harmonizePlaceGo(venue,'harmonize'); } }, IsThisAPilotTravelCenter: class extends ActionFlag { constructor() { super(true, 0, 'Is this a "Travel Center"?', 'Yes', ''); } static eval(venue, hpMode, state2L, newName, actions) { let result = {flag: null, newName: newName}; if (hpMode.harmFlag && state2L === 'TN') { if (result.newName.toLowerCase().trim() === 'pilot') { result.newName = 'Pilot Food Mart'; actions.push(new UpdateObject(venue, { name: result.newName })); _updatedFields.name.updated = true; } if (result.newName.toLowerCase().trim() === 'pilot food mart') { result.flag = new Flag.IsThisAPilotTravelCenter(); } } return result; } action() { let venue = getSelectedVenue(); _updatedFields.name.updated = true; addUpdateAction(venue, { name: 'Pilot Travel Center' }); harmonizePlaceGo(venue, 'harmonize'); } }, HotelMkPrim: class extends WLActionFlag { constructor() { super(true, 3, 'Hotel category is not first', 'Fix', 'Make the Hotel category the primary category.', true, 'Whitelist hotel as secondary category', 'hotelMkPrim'); } action() { let venue = getSelectedVenue(); newCategories = insertAtIX(newCategories, 'HOTEL', 0); // Insert/move Hotel category in the first position _updatedFields.categories.updated = true; addUpdateAction(venue, { categories: newCategories }); harmonizePlaceGo(venue,'harmonize'); } }, ChangeToPetVet: class extends WLActionFlag { constructor() { super(true, 3, 'This looks like it should be a Pet/Veterinarian category. Change?', 'Yes', 'Change to Pet/Veterinarian Category', true, 'Whitelist PetVet category', 'changeHMC2PetVet'); } action() { let venue = getSelectedVenue(); let idx = newCategories[newCategories.indexOf('HOSPITAL_MEDICAL_CARE')]; if (idx === -1) idx = newCategories[newCategories.indexOf('HOSPITAL_URGENT_CARE')]; if (idx > -1) { newCategories[idx] = 'PET_STORE_VETERINARIAN_SERVICES'; _updatedFields.categories.updated = true; addUpdateAction(venue, { categories: newCategories }); } harmonizePlaceGo(venue,'harmonize'); // Rerun the script to update fields and lock } }, ChangeSchool2Offices: class extends WLActionFlag { constructor() { super(true, 3, 'This doesn\'t look like it should be School category.', 'Change to Office', 'Change to Offices Category', true, 'Whitelist School category', 'changeSchool2Offices'); } action() { let venue = getSelectedVenue(); newCategories[newCategories.indexOf('SCHOOL')] = 'OFFICES'; _updatedFields.categories.updated = true; addUpdateAction(venue, { categories: newCategories }); harmonizePlaceGo(venue, 'harmonize'); // Rerun the script to update fields and lock } }, PointNotArea: class extends WLActionFlag { constructor() { super(true, 3, 'This category should be a point place.', 'Change to point', 'Change to point place', true, 'Whitelist point (not area)', 'pointNotArea'); } action() { let venue = getSelectedVenue(); if (venue.attributes.categories.indexOf('RESIDENCE_HOME') > -1){ let centroid = venue.geometry.getCentroid(); updateFeatureGeometry(venue, new OL.Geometry.Point(centroid.x,centroid.y)); } else { $('.landmark label.point-btn').click(); } harmonizePlaceGo(venue, 'harmonize'); // Rerun the script to update fields and lock } }, AreaNotPoint: class extends WLActionFlag { constructor() { super(true, 3, 'This category should be an area place.', 'Change to area', 'Change to Area', true, 'Whitelist area (not point)', 'areaNotPoint'); } action() { let venue = getSelectedVenue(); updateFeatureGeometry(venue, venue.getPolygonGeometry()); harmonizePlaceGo(venue, 'harmonize'); } }, HnMissing: class extends WLActionFlag { constructor(venue) { super(true, 3, 'No HN: <input type="text" id="WMEPH-HNAdd" autocomplete="off" style="font-size:0.85em;width:100px;padding-left:2px;color:#000;">', 'Add', 'Add HN to place', true, 'Whitelist empty HN', 'HNWL'); this.venue = venue; this.noBannerAssemble = true; this.badInput = false; } action() { let newHN = $('#WMEPH-HNAdd').val().replace(/ +/g, ''); phlogdev(newHN); var hnTemp = newHN.replace(/[^\d]/g, ''); var hnTempDash = newHN.replace(/[^\d-]/g, ''); if (hnTemp > 0 && hnTemp < 1000000) { let action = new UpdateObject(this.venue, { houseNumber: hnTempDash }); action.wmephDescription = 'Changed house # to: ' + hnTempDash; harmonizePlaceGo(this.venue, 'harmonize', [action]); // Rerun the script to update fields and lock _updatedFields.address.updated = true; } else { $('input#WMEPH-HNAdd').css({backgroundColor:'#FDD'}).attr('title', 'Must be a number between 0 and 1000000'); this.badInput = true; } } }, HnNonStandard: class extends WLFlag { constructor() { super(true, 3, 'House number is non-standard.', true, 'Whitelist non-standard HN', 'hnNonStandard'); } }, HNRange: class extends WLFlag { constructor() { super(true, 2, 'House number seems out of range for the street name. Verify.', true, 'Whitelist HN range', 'HNRange'); } }, StreetMissing: class extends ActionFlag { constructor() { super(true, 3, 'No street:', 'Edit address', 'Edit address to add street.'); } action() { $('.nav-tabs a[href="#landmark-edit-general"]').trigger('click'); $('.landmark .full-address').click(); if ($('.empty-street').prop('checked')) { $('.empty-street').click(); } $('.street-name').focus(); } }, CityMissing: class extends ActionFlag { constructor() { super(true, 3, 'No city:', 'Edit address', 'Edit address to add city.'); } action() { $('.nav-tabs a[href="#landmark-edit-general"]').trigger('click'); $('.landmark .full-address').click(); if ($('.empty-city').prop('checked')) { $('.empty-city').click(); } $('.city-name').focus(); } }, BankType1: class extends FlagBase { constructor() { super(true, 3, 'Clarify the type of bank: the name has ATM but the primary category is Offices'); } }, BankBranch: class extends ActionFlag { constructor() { super(true, 1, 'Is this a bank branch office? ', 'Yes', 'Is this a bank branch?'); } action() { let venue = getSelectedVenue(); newCategories = ['BANK_FINANCIAL','ATM']; // Change to bank and atm cats var tempName = newName.replace(/[\- (]*ATM[\- )]*/g, ' ').replace(/^ /g,'').replace(/ $/g,''); // strip ATM from name if present newName = tempName; W.model.actionManager.add(new UpdateObject(venue, { name: newName, categories: newCategories })); if (tempName !== newName) _updatedFields.name.updated = true; _updatedFields.categories.updated = true; harmonizePlaceGo(venue, 'harmonize'); } }, StandaloneATM: class extends ActionFlag { constructor() { super(true, 2, 'Or is this a standalone ATM? ', 'Yes', 'Is this a standalone ATM with no bank branch?'); } action() { let venue = getSelectedVenue(); if (newName.indexOf('ATM') === -1) { newName = newName + ' ATM'; _updatedFields.name.updated = true; } newCategories = ['ATM']; // Change to ATM only W.model.actionManager.add(new UpdateObject(venue, { name: newName, categories: newCategories })); _updatedFields.categories.updated = true; harmonizePlaceGo(venue, 'harmonize'); } }, BankCorporate: class extends ActionFlag { constructor() { super(true, 1, 'Or is this the bank\'s corporate offices?', 'Yes', 'Is this the bank\'s corporate offices?'); } action() { let venue = getSelectedVenue(); newCategories = ['OFFICES']; // Change to offices category var tempName = newName.replace(/[\- (]*atm[\- )]*/ig, ' ').replace(/^ /g,'').replace(/ $/g,'').replace(/ {2,}/g,' '); // strip ATM from name if present newName = tempName; W.model.actionManager.add(new UpdateObject(venue, { name: newName + ' - Corporate Offices', categories: newCategories })); if (newName !== tempName) _updatedFields.name.updated = true; _updatedFields.categories.updated = true; harmonizePlaceGo(venue, 'harmonize'); } }, CatPostOffice: class extends FlagBase { constructor() { super(true, 0, 'The Post Office category is reserved for certain USPS locations. Please be sure to follow <a href="https://wazeopedia.waze.com/wiki/USA/Places/Post_Office" style="color:#3a3a3a;" target="_blank">the guidelines</a>.'); } }, IgnEdited: class extends FlagBase { constructor() { super(true, 2, 'Last edited by an IGN editor'); } }, WazeBot: class extends ActionFlag { constructor() { super(true, 2, 'Edited last by an automated process. Please verify information is correct.', 'Nudge', 'If no other properties need to be updated, click to nudge the place (force an edit).'); } action() { let venue = getSelectedVenue(); // Use an exact clone of the original geometry to force an edit without actually changing anything. W.model.actionManager.add(new UpdateFeatureGeometry(venue, W.model.venues, venue.geometry, venue.geometry.clone())); harmonizePlaceGo(venue, 'harmonize'); // Rerun the script to update fields and lock } }, ParentCategory: class extends WLFlag { constructor() { super(true, 2, 'This parent category is usually not mapped in this region.', true, 'Whitelist parent Category','parentCategory'); } }, CheckDescription: class extends FlagBase { constructor() { super(true, 2, 'Description field already contained info; PNH description was added in front of existing. Check for inconsistency or duplicate info.'); } }, Overlapping: class extends FlagBase { constructor() { super(true, 2, 'Place points are stacked up.'); } }, SuspectDesc: class extends WLFlag { constructor() { super(true, 2, 'Description field might contain copyrighted info.', true, 'Whitelist description', 'suspectDesc'); } }, ResiTypeName: class extends WLFlag { constructor() { super(true, 2, 'The place name suggests a residential place or personalized place of work. Please verify.', true, 'Whitelist Residential-type name', 'resiTypeName'); } }, Mismatch247: class extends FlagBase { constructor() { super(true, 2, 'Hours of operation listed as open 24hrs but not for all 7 days.'); } }, PhoneInvalid: class extends FlagBase { constructor() { super(true, 2, 'Phone invalid.'); } }, AreaNotPointMid: class extends WLFlag { constructor() { super(true, 2, 'This category is usually an area place, but can be a point in some cases. Verify if point is appropriate.', true, 'Whitelist area (not point)', 'areaNotPoint'); } }, PointNotAreaMid: class extends WLFlag { constructor() { super(true, 2, 'This category is usually a point place, but can be an area in some cases. Verify if area is appropriate.', true, 'Whitelist point (not area)', 'pointNotArea'); } }, LongURL: class extends WLActionFlag{ constructor() { super(true, 1, 'Existing URL doesn\'t match the suggested PNH URL. Use the Website button below to verify that existing URL is valid. If not:', 'Use PNH URL', 'Change URL to the PNH standard', true, 'Whitelist existing URL', 'longURL'); } action() { let venue = getSelectedVenue(); if (tempPNHURL !== '') { W.model.actionManager.add(new UpdateObject(venue, { url: tempPNHURL })); _updatedFields.url.updated = true; harmonizePlaceGo(venue, 'harmonize'); updateURL = true; } else { if (confirm('WMEPH: URL Matching Error!\nClick OK to report this error') ) { // if the category doesn't translate, then pop an alert that will make a forum post to the thread reportError({ subject: 'WMEPH URL comparison Error report', message: 'Error report: URL comparison failed for "' + venue.attributes.name + '"\nPermalink: ' + placePL }); } } } }, GasNoBrand: class extends FlagBase { constructor() { super(true, 1, 'Lock to region standards to verify no gas brand.'); } static eval(venue) { let result = {flag: null}; if (venue.isGasStation() && !venue.attributes.brand) { result.flag = new Flag.GasNoBrand(); result.noLock = true; } return result; } }, SubFuel: class extends WLFlag { constructor() { super(true, 1, 'Make sure this place is for the gas station itself and not the main store building. Otherwise undo and check the categories.', true, 'Whitelist no gas brand', 'subFuel'); } }, AreaNotPointLow: class extends WLFlag { constructor() { super(true, 1, 'This category is usually an area place, but can be a point in some cases. Verify if point is appropriate.', true, 'Whitelist area (not point)', 'areaNotPoint'); } }, PointNotAreaLow: class extends WLFlag { constructor() { super(true,1, 'This category is usually a point place, but can be an area in some cases. Verify if area is appropriate.', true, 'Whitelist point (not area)', 'pointNotArea'); } }, FormatUSPS: class extends FlagBase { constructor() { super(true, 1, 'Name the post office according to this region\'s <a href="https://wazeopedia.waze.com/wiki/USA/Places/Post_Office" style="color:#3232e6" target="_blank"> standards for USPS post offices</a>'); } }, MissingUSPSAlt: class extends FlagBase { constructor() { super(true, 1, 'USPS post offices must have an alternate name of "USPS".'); } }, MissingUSPSZipAlt: class extends WLActionFlag { constructor() { super(true, 1, 'No <a href="https://wazeopedia.waze.com/wiki/USA/Places/Post_Office" style="color:#3232e6;" target="_blank">ZIP code alt name</a>: ' + '<input type="text" id="WMEPH-zipAltNameAdd" autocomplete="off" style="font-size:0.85em;width:65px;padding-left:2px;color:#000;" title="Enter the ZIP code and click Add">', 'Add', true, 'Whitelist missing USPS zip alt name', 'missingUSPSZipAlt'); this.noBannerAssemble = true; } action() { let $input = $('input#WMEPH-zipAltNameAdd'); let zip = $input.val().trim(); if (zip) { if (/^\d{5}/.test(zip)) { let venue = getSelectedVenue(); let aliases = [].concat(venue.attributes.aliases); // Make sure zip hasn't already been added. if (aliases.indexOf(zip) === -1) { aliases.push(zip); W.model.actionManager.add(new UpdateObject(venue, {aliases: aliases})); harmonizePlaceGo(venue, 'harmonize'); } else { $input.css({backgroundColor: '#FDD'}).attr('title', 'Zip code alt name already exists'); } } else { $input.css({backgroundColor: '#FDD'}).attr('title', 'Zip code format error'); } } } }, MissingUSPSDescription: class extends WLFlag { constructor() { super(true, 1, 'The first line of the description for a <a href="https://wazeopedia.waze.com/wiki/USA/Places/Post_Office" style="color:#3232e6" target="_blank">USPS post office</a> must be CITY, STATE ZIP, e.g. "Lexington, KY 40511"', true, 'Whitelist missing USPS address line in description', 'missingUSPSDescription'); } }, CatHotel: class extends FlagBase { constructor(pnhName) { super(true, 0, 'Check hotel website for any name localization (e.g. '+ pnhName +' - Tampa Airport).'); } }, LocalizedName: class extends WLFlag { constructor() { super(true, 1, 'Place needs localization information', true, 'Whitelist localization', 'localizedName'); } }, SpecCaseMessage: class extends FlagBase { constructor(message) { super(true, 0, message); } }, PnhCatMess: class extends FlagBase { constructor(message) { super(true, 0, message); } }, SpecCaseMessageLow: class extends FlagBase { constructor(message) { super(true, 0, message); } }, ExtProviderMissing: class extends ActionFlag { constructor() { super(true, 3, 'No Google link', 'Nudge', 'If no other properties need to be updated, click to nudge the place (force an edit).'); this.value2 = 'Add'; this.title2 = 'Add a link to a Google place'; } action() { let venue = getSelectedVenue(); // Use an exact clone of the original geometry to force an edit without actually changing anything. W.model.actionManager.add(new UpdateFeatureGeometry(venue, W.model.venues, venue.geometry, venue.geometry.clone())); harmonizePlaceGo(venue,'harmonize'); // Rerun the script to update fields and lock } action2() { let venue = getSelectedVenue(); $('div.external-providers-view a').focus().click(); setTimeout(function() { $('a[href="#landmark-edit-general"]').click(); $('.external-providers-view a.add').focus().mousedown(); $('div.external-providers-view > div > ul > div > li > div > a').last().mousedown(); $('.select2-input').last().focus().val(venue.attributes.name).trigger('input'); }, 100); } }, UrlMissing: class extends WLActionFlag { constructor() { super(true, 1, 'No URL: <input type="text" id="WMEPH-UrlAdd" autocomplete="off" style="font-size:0.85em;width:100px;padding-left:2px;color:#000;">', 'Add', 'Add URL to place', true, 'Whitelist empty URL', 'urlWL'); this.noBannerAssemble = true; this.badInput = false; } action() { let venue = getSelectedVenue(); let newUrl = normalizeURL($('#WMEPH-UrlAdd').val(), true, false, venue); if ((!newUrl || newUrl.trim().length === 0) || newUrl === 'badURL') { $('input#WMEPH-UrlAdd').css({backgroundColor:'#FDD'}).attr('title','Invalid URL format'); //this.badInput = true; } else { phlogdev(newUrl); W.model.actionManager.add(new UpdateObject(venue, { url: newUrl })); _updatedFields.url.updated = true; harmonizePlaceGo(venue, 'harmonize'); } } }, BadAreaCode: class extends WLActionFlag { constructor(textValue, outputFormat) { super(true, 1, 'Area Code mismatch:<br><input type="text" id="WMEPH-PhoneAdd" autocomplete="off" style="font-size:0.85em;width:100px;padding-left:2px;color:#000;" value="' + (textValue ? textValue : '') + '">', 'Update', 'Update phone #', true, 'Whitelist the area code', 'aCodeWL'); this.outputFormat = outputFormat; this.noBannerAssemble = true; } action() { let venue = getSelectedVenue(); let newPhone = normalizePhone($('#WMEPH-PhoneAdd').val(), this.outputFormat, 'inputted', venue); if (newPhone === 'badPhone') { $('input#WMEPH-PhoneAdd').css({backgroundColor: '#FDD'}).attr('title','Invalid phone # format'); this.badInput = true; } else { this.badInput = false; phlogdev(newPhone); W.model.actionManager.add(new UpdateObject(venue, { phone: newPhone })); _updatedFields.phone.updated = true; harmonizePlaceGo(venue, 'harmonize'); } } }, PhoneMissing: class extends WLActionFlag { constructor(venue, hasOperator, wl, outputFormat, isPLA) { super(true, 1, 'No ph#: <input type="text" id="WMEPH-PhoneAdd" autocomplete="off" style="font-size:0.85em;width:100px;padding-left:2px;color:#000;">', 'Add', 'Add phone to place', true, 'Whitelist empty phone', 'phoneWL'); this.noBannerAssemble = true; this.badInput = false; this.outputFormat = outputFormat; this.venue = venue; if ((isPLA && !hasOperator) || wl[this.WLkeyName]) { this.severity = 0; this.WLactive = false; } } static get _regionsThatWantPlaPhones() { return ['SER']; } static eval(venue, wl, region, outputFormat) { let hasOperator = venue.attributes.brand && W.model.venues.categoryBrands.PARKING_LOT.indexOf(venue.attributes.brand) !== -1; let isPLA = venue.isParkingLot(); let flag = null; if (!isPLA || (isPLA && (this._regionsThatWantPlaPhones.indexOf(region) > -1 || hasOperator))) { flag = new Flag.PhoneMissing(venue, hasOperator, wl, outputFormat, isPLA); } return flag; } action() { let newPhone = normalizePhone($('#WMEPH-PhoneAdd').val(), this.outputFormat, 'inputted', this.venue); if (newPhone === 'badPhone') { $('input#WMEPH-PhoneAdd').css({backgroundColor: '#FDD'}).attr('title','Invalid phone # format'); this.badInput = true; } else { this.badInput = false; phlogdev(newPhone); W.model.actionManager.add(new UpdateObject(this.venue, { phone: newPhone })); _updatedFields.phone.updated = true; harmonizePlaceGo(this.venue, 'harmonize'); } } }, NoHours: class extends WLFlag { constructor() { super(true, 1, getHoursHtml('No hours'), true, 'Whitelist "No hours"', 'noHours'); } getTitle(parseResult) { let title; if (parseResult.overlappingHours) { title = 'Overlapping hours. Check the existing hours.'; } else if (parseResult.sameOpenAndCloseTimes) { title = 'Open/close times cannot be the same.'; } else { title = 'Can\'t parse, try again'; } return title; } applyHours(replaceAllHours) { let venue = getSelectedVenue(); let pasteHours = $('#WMEPH-HoursPaste').val(); if (pasteHours === _DEFAULT_HOURS_TEXT) { return; } phlogdev(pasteHours); pasteHours += !replaceAllHours ? ',' + getOpeningHours(venue).join(',') : ''; $('.nav-tabs a[href="#landmark-edit-more-info"]').tab('show'); var parser = new HoursParser(); var parseResult = parser.parseHours(pasteHours); if (parseResult.hours && !parseResult.overlappingHours && !parseResult.sameOpenAndCloseTimes && !parseResult.parseError) { phlogdev(parseResult.hours); W.model.actionManager.add(new UpdateObject(venue, { openingHours: parseResult.hours })); _updatedFields.openingHours.updated = true; $('#WMEPH-HoursPaste').val(_DEFAULT_HOURS_TEXT); harmonizePlaceGo(venue, 'harmonize'); } else { phlog('Can\'t parse those hours'); this.severity = 1; this.WLactive = true; $('#WMEPH-HoursPaste').css({'background-color':'#FDD'}).attr({title: this.getTitle(parseResult)}); } } addHoursAction() { this.applyHours(); } replaceHoursAction() { this.applyHours(true); } }, PlaLotTypeMissing: class extends FlagBase { constructor() { super(true, 3, 'Lot type: '); } static eval(venue, hpMode) { let result = {flag: null}; if (venue.isParkingLot()) { let catAttr = venue.attributes.categoryAttributes; let parkAttr = catAttr ? catAttr.PARKING_LOT : undefined; if (!parkAttr || !parkAttr.parkingType) { result.flag = new Flag.PlaLotTypeMissing(); if (hpMode.harmFlag) { result.noLock = true; [['PUBLIC','Public'],['RESTRICTED','Restricted'],['PRIVATE','Private']].forEach(btnInfo => { result.flag.message += $('<button>', {class: 'wmeph-pla-lot-type-btn btn btn-default btn-xs wmeph-btn', 'data-lot-type':btnInfo[0]}) .text(btnInfo[1]) .css({padding:'3px', height:'20px', lineHeight:'0px', marginRight:'2px', marginBottom:'1px'}) .prop('outerHTML'); }); } } } return result; } }, PlaCostTypeMissing: class extends FlagBase { constructor() { super(true, 1, 'Parking cost: '); } static eval(venue, hpMode) { let result = {flag: null}; if (venue.isParkingLot()) { let catAttr = venue.attributes.categoryAttributes; let parkAttr = catAttr ? catAttr.PARKING_LOT : undefined; if (!parkAttr || !parkAttr.costType || parkAttr.costType === 'UNKNOWN') { result.flag = new Flag.PlaCostTypeMissing(); if (hpMode.harmFlag) { [['FREE','Free','Free'],['LOW','$','Low'],['MODERATE','$$','Moderate'],['EXPENSIVE','$$$','Expensive']].forEach(btnInfo => { result.flag.message += $('<button>', {id: 'wmeph_' + btnInfo[0], class: 'wmeph-pla-cost-type-btn btn btn-default btn-xs wmeph-btn', title: btnInfo[2]}) .text(btnInfo[1]) .css({padding:'3px', height:'20px', lineHeight:'0px', marginRight:'2px', marginBottom:'1px', minWidth:'18px'}) .prop('outerHTML'); }); result.noLock = true; } } } return result; } }, PlaPaymentTypeMissing: class extends ActionFlag { constructor() { super(true, 1, 'Parking isn\'t free. Select payment type(s) from the "More info" tab. ', 'Go there'); } static eval(venue) { let result = {flag: null}; if (venue.isParkingLot()) { let catAttr = venue.attributes.categoryAttributes; let parkAttr = catAttr ? catAttr.PARKING_LOT : undefined; if (parkAttr && parkAttr.costType && parkAttr.costType !== 'FREE' && parkAttr.costType !== 'UNKNOWN' && (!parkAttr.paymentType || !parkAttr.paymentType.length)) { result.flag = new Flag.PlaPaymentTypeMissing(); } } return result; } action() { $('a[href="#landmark-edit-more-info"]').click(); $('#payment-checkbox-ELECTRONIC_PASS').focus(); } }, PlaLotElevationMissing: class extends ActionFlag { constructor() { super(true, 1, 'No lot elevation. Is it street level?', 'Yes', 'Click if street level parking only, or select other option(s) in the More Info tab.'); } static eval(venue) { let result = {flag: null}; if (venue.isParkingLot()) { let catAttr = venue.attributes.categoryAttributes; let parkAttr = catAttr ? catAttr.PARKING_LOT : undefined; if (!parkAttr || !parkAttr.lotType || parkAttr.lotType.length === 0) { result.flag = new Flag.PlaLotElevationMissing(); result.noLock = true; } } return result; } action() { let venue = getSelectedVenue(); let existingAttr = venue.attributes.categoryAttributes.PARKING_LOT; let newAttr = {}; if (existingAttr) { for (let prop in existingAttr) { let value = existingAttr[prop]; if (Array.isArray(value)) value = [].concat(value); newAttr[prop] = value; } } newAttr.lotType = ['STREET_LEVEL']; W.model.actionManager.add(new UpdateObject(venue, {'categoryAttributes': {PARKING_LOT: newAttr}})); harmonizePlaceGo(venue, 'harmonize'); } }, PlaSpaces: class extends FlagBase { constructor() { super(true, 0, '# of parking spaces is set to 1-10.<br><b><i>If appropriate</i></b>, select another option:'); let $btnDiv = $('<div>'); let btnIdx = 0; [['R_11_TO_30','11-30'], ['R_31_TO_60','31-60'], ['R_61_TO_100','61-100'], ['R_101_TO_300','101-300'], ['R_301_TO_600','301-600'], ['R_600_PLUS','601+']].forEach(btnInfo => { if (btnIdx === 3) $btnDiv.append('<br>'); $btnDiv.append( $('<button>', {id: 'wmeph_' + btnInfo[0], class: 'wmeph-pla-spaces-btn btn btn-default btn-xs wmeph-btn'}) .text(btnInfo[1]) .css({padding:'3px', height:'20px', lineHeight:'0px', marginTop:'2px', marginRight:'2px', marginBottom:'1px', width:'64px'}) ); btnIdx++; }); this.suffixMessage = $btnDiv.prop('outerHTML'); } static eval(venue, hpMode) { let result = {flag: null}; if (hpMode.harmFlag && venue.isParkingLot()) { let catAttr = venue.attributes.categoryAttributes; let parkAttr = catAttr ? catAttr.PARKING_LOT : undefined; if (!parkAttr || !parkAttr.estimatedNumberOfSpots || parkAttr.estimatedNumberOfSpots === 'R_1_TO_10') { result.flag = new Flag.PlaSpaces(); } } return result; } }, NoPlaStopPoint: class extends ActionFlag { constructor() { super(true, 1, 'Entry/exit point has not been created.', 'Add point', 'Add an entry/exit point'); } static eval(venue) { let result = {flag: null}; if (venue.isParkingLot() && (!venue.attributes.entryExitPoints || !venue.attributes.entryExitPoints.length)) { result.flag = new Flag.NoPlaStopPoint(); } return result; } action() { $('.navigation-point-view .add-button').click(); harmonizePlaceGo(getSelectedVenue(), 'harmonize'); } }, PlaStopPointUnmoved: class extends FlagBase { constructor() { super(true, 1, 'Entry/exit point has not been moved.'); } static eval(venue) { let result = {flag: null}; let attr = venue.attributes; if (venue.isParkingLot() && attr.entryExitPoints && attr.entryExitPoints.length) { let stopPoint = attr.entryExitPoints[0].getPoint(); let areaCenter = attr.geometry.getCentroid(); if (stopPoint.equals(areaCenter)) { result.flag = new Flag.PlaStopPointUnmoved(); } } return result; } }, PlaCanExitWhileClosed: class extends ActionFlag { constructor() { super(true, 0, 'Can cars exit when lot is closed? ', 'Yes', ''); } static eval(venue, hpMode) { let result = {flag: null}; if (hpMode.harmFlag && venue.isParkingLot()) { let catAttr = venue.attributes.categoryAttributes; let parkAttr = catAttr ? catAttr.PARKING_LOT : undefined; if (parkAttr && !parkAttr.canExitWhileClosed && ($('#WMEPH-ShowPLAExitWhileClosed').prop('checked') || !(isAlwaysOpen(venue) || venue.attributes.openingHours.length === 0))) { result.flag = new Flag.PlaCanExitWhileClosed(); } } return result; } action() { let venue = getSelectedVenue(); let existingAttr = venue.attributes.categoryAttributes.PARKING_LOT; let newAttr = {}; if (existingAttr) { for (let prop in existingAttr) { let value = existingAttr[prop]; if (Array.isArray(value)) value = [].concat(value); newAttr[prop] = value; } } newAttr.canExitWhileClosed = true; W.model.actionManager.add(new UpdateObject(venue, {'categoryAttributes': {PARKING_LOT: newAttr}})); harmonizePlaceGo(venue, 'harmonize'); } }, PlaHasAccessibleParking: class extends ActionFlag { constructor() { super(true, 0, 'Does this lot have disability parking? ', 'Yes', ''); } static eval(venue, hpMode) { let result = {flag: null}; if (hpMode.harmFlag && venue.isParkingLot()) { let services = venue.attributes.services; if (!(services && services.indexOf('DISABILITY_PARKING') > -1)) { result.flag = new Flag.PlaHasAccessibleParking(); } } return result; } action() { let venue = getSelectedVenue(); let services = venue.attributes.services; if (services) { services = [].concat(services); } else { services = []; } services.push('DISABILITY_PARKING'); //bannServ.addDisabilityParking.on(); W.model.actionManager.add(new UpdateObject(venue, {'services': services})); _updatedFields.services_DISABILITY_PARKING.updated = true; harmonizePlaceGo(venue, 'harmonize'); } }, AllDayHoursFixed: class extends FlagBase { constructor() { super(true, 0, 'Hours were changed from 00:00-23:59 to "All Day"'); } static eval(venue, hpMode, actions) { let hoursEntries = venue.attributes.openingHours; let newHoursEntries = []; let updateHours = false; let flag = null; for (let i=0, len=hoursEntries.length; i<len; i++) { let newHoursEntry = {days:[].concat(hoursEntries[i].days), fromHour:hoursEntries[i].fromHour, toHour:hoursEntries[i].toHour}; if (newHoursEntry.toHour === '23:59' && /^0?0:00$/.test(newHoursEntry.fromHour)) { if (hpMode.hlFlag) { // Just return a "placeholder" flag to highlight the place. flag = new FlagBase(true, 2, 'invalid all day hours'); break; } else if (hpMode.harmFlag) { updateHours = true; newHoursEntry.toHour = '00:00'; newHoursEntry.fromHour = '00:00'; } } newHoursEntries.push(newHoursEntry); } if (updateHours) { addUpdateAction(venue, { openingHours: newHoursEntries }, actions); _updatedFields.openingHours.updated = true; flag = new Flag.AllDayHoursFixed(); } return flag; } }, ResiTypeNameSoft: class extends FlagBase { constructor() { super(true, 0, 'The place name suggests a residential place or personalized place of work. Please verify.'); } }, LocalURL: class extends FlagBase { constructor() { super(true, 0, 'Some locations for this business have localized URLs, while others use the primary corporate site. Check if a local URL applies to this location.'); } }, LockRPP: class extends ActionFlag { constructor() { super(true, 0, 'Lock this residential point?', 'Lock', 'Lock the residential point'); } action() { let venue = getSelectedVenue(); let RPPlevelToLock = $('#RPPLockLevel :selected').val() || defaultLockLevel + 1; phlogdev('RPPlevelToLock: '+ RPPlevelToLock); RPPlevelToLock = RPPlevelToLock -1 ; W.model.actionManager.add(new UpdateObject(venue, { lockRank: RPPlevelToLock })); // no field highlight here this.message = 'Current lock: '+ (parseInt(venue.attributes.lockRank) + 1) +'. '+RPPLockString+' ?'; } }, AddAlias: class extends ActionFlag { constructor() { super(true, 0, 'Is ' + optionalAlias + ' at this location?', 'Yes', 'Add ' + optionalAlias); } action() { let venue = getSelectedVenue(); newAliases = insertAtIX(newAliases,optionalAlias,0); if (specCases.indexOf('altName2Desc') > -1 && venue.attributes.description.toUpperCase().indexOf(optionalAlias.toUpperCase()) === -1 ) { newDescripion = optionalAlias + '\n' + newDescripion; W.model.actionManager.add(new UpdateObject(venue, { description: newDescripion })); _updatedFields.description.updated = true; } newAliases = removeSFAliases(newName, newAliases); W.model.actionManager.add(new UpdateObject(venue, { aliases: newAliases })); _updatedFields.aliases.updated = true; harmonizePlaceGo(venue, 'harmonize'); } }, AddCat2: class extends ActionFlag { constructor() { super(false, 0, '', 'Yes', ''); } action() { let venue = getSelectedVenue(); newCategories.push(this.altCategory); W.model.actionManager.add(new UpdateObject(venue, { categories: newCategories })); _updatedFields.categories.updated = true; harmonizePlaceGo(venue, 'harmonize'); } }, AddPharm: class extends ActionFlag { constructor() { super(false, 0, 'Is there a Pharmacy at this location?', 'Yes', 'Add Pharmacy category'); } action() { let venue = getSelectedVenue(); newCategories = insertAtIX(newCategories, 'PHARMACY', 1); W.model.actionManager.add(new UpdateObject(venue, { categories: newCategories })); _updatedFields.categories.updated = true; bannButt.addPharm.active = false; // reset the display flag } }, AddSuper: class extends ActionFlag { constructor() { super(false, 0, 'Does this location have a supermarket?', 'Yes', 'Add Supermarket category'); } action() { let venue = getSelectedVenue(); newCategories = insertAtIX(newCategories, 'SUPERMARKET_GROCERY', 1); W.model.actionManager.add(new UpdateObject(venue, { categories: newCategories })); _updatedFields.categories.updated = true; bannButt.addSuper.active = false; // reset the display flag } }, AppendAMPM: class extends ActionFlag { constructor() { super(false, 0, 'Is there an ampm at this location?', 'Yes', 'Add ampm to the place'); } action() { let venue = getSelectedVenue(); newCategories = insertAtIX(newCategories, 'CONVENIENCE_STORE', 1); newName = 'ARCO ampm'; newURL = 'ampm.com'; W.model.actionManager.add(new UpdateObject(venue, { name: newName, url: newURL, categories: newCategories })); _updatedFields.name.updated = true; _updatedFields.url.updated = true; _updatedFields.categories.updated = true; bannButt.appendAMPM.active = false; // reset the display flag bannButt.addConvStore.active = false; // also reset the addConvStore display flag } }, AddATM: class extends ActionFlag { constructor() { super(false, 0, 'ATM at location? ', 'Yes', 'Add the ATM category to this place'); } action() { let venue = getSelectedVenue(); newCategories = insertAtIX(newCategories,'ATM',1); // Insert ATM category in the second position W.model.actionManager.add(new UpdateObject(venue, { categories: newCategories })); _updatedFields.categories.updated = true; bannButt.addATM.active = false; // reset the display flag } }, AddConvStore: class extends ActionFlag { constructor() { super(false, 0, 'Add convenience store category? ', 'Yes', 'Add the Convenience Store category to this place'); } action() { let venue = getSelectedVenue(); newCategories = insertAtIX(newCategories,'CONVENIENCE_STORE',1); // Insert C.S. category in the second position W.model.actionManager.add(new UpdateObject(venue, { categories: newCategories })); _updatedFields.categories.updated = true; bannButt.addConvStore.active = false; // reset the display flag } }, IsThisAPostOffice: class extends ActionFlag { constructor() { super(true, 0, 'Is this a <a href="https://wazeopedia.waze.com/wiki/USA/Places/Post_Office" target="_blank" style="color:#3a3a3a">USPS post office</a>? ', 'Yes', 'Is this a USPS location?'); } static eval(venue, newName) { let result = {flag: null}; let cleanName = newName.toUpperCase().replace(/[\/\-\.]/g, ''); if (/\bUSP[OS]\b|\bpost(al)?\s+(service|office)\b/i.test(cleanName)) { result.flag = new Flag.IsThisAPostOffice(); } return result; } action() { let venue = getSelectedVenue(); newCategories = insertAtIX(newCategories, 'POST_OFFICE', 0); W.model.actionManager.add(new UpdateObject(venue, { categories: newCategories })); _updatedFields.categories.updated = true; harmonizePlaceGo(venue, 'harmonize'); } }, ChangeToHospitalUrgentCare: class extends WLActionFlag { constructor(severity, message) { super(true, severity, message, 'Change to Hospital / Urgent Care', 'Change category to Hospital / Urgent Care', false, 'Whitelist category', 'changetoHospitalUrgentCare'); } static eval(venue, hpMode) { let result = {flag: null}; if (hpMode.harmFlag && venue.attributes.categories.indexOf('DOCTOR_CLINIC') > -1) { result.flag = new Flag.ChangeToHospitalUrgentCare(0 ,'If this place provides emergency medical care:'); } return result; } action() { let idx = newCategories.indexOf('HOSPITAL_MEDICAL_CARE'); let venue = getSelectedVenue(); if (idx === -1) idx = newCategories.indexOf('DOCTOR_CLINIC'); if (idx > -1) { newCategories[idx] = 'HOSPITAL_URGENT_CARE'; _updatedFields.categories.updated = true; addUpdateAction(venue, { categories: newCategories }); } harmonizePlaceGo(venue, 'harmonize'); // Rerun the script to update fields and lock } }, ChangeToDoctorClinic: class extends WLActionFlag { constructor() { super(true, 0, '', 'Change to Doctor / Clinic', 'Change category to Doctor / Clinic', false, 'Whitelist category', 'changeToDoctorClinic'); } action() { let venue = getSelectedVenue(); let newCategories = _.clone(venue.attributes.categories); let updateIt = false; if (newCategories.length) { ['HOSPITAL_MEDICAL_CARE', 'HOSPITAL_URGENT_CARE', 'OFFICES', 'PERSONAL_CARE'].forEach(cat => { let idx = newCategories.indexOf(cat); if (idx > -1) { newCategories[idx] = 'DOCTOR_CLINIC'; updateIt = true; } }); newCategories = _.uniq(newCategories); } else { newCategories.push('DOCTOR_CLINIC'); updateIt = true; } if (updateIt) { _updatedFields.categories.updated = true; W.model.actionManager.add(new UpdateObject(venue, {categories: newCategories})); } harmonizePlaceGo(venue, 'harmonize'); // Rerun the script to update fields and lock } }, STC: class extends ActionFlag { constructor() { super(true, 0, '', 'Force Title Case?', 'Force title case to: '); this.originalName = null; this.confirmChange = false; this.noBannerAssemble = true; } action() { let venue = getSelectedVenue(); let newName = venue.attributes.name; if (newName === this.originalName || this.confirmChange) { let parts = getNameParts(this.originalName); newName = toTitleCaseStrong(parts.base); if (parts.base !== newName) { W.model.actionManager.add(new UpdateObject(venue, { name: newName + (parts.suffix || '') })); _updatedFields.name.updated = true; } harmonizePlaceGo(venue, 'harmonize'); } else { $('button#WMEPH_STC').text('Are you sure?').after(' The name has changed. This will overwrite the new name.'); bannButt.STC.confirmChange = true; } } }, SFAliases: class extends FlagBase { constructor() { super(true, 0, 'Unnecessary aliases were removed.'); } }, PlaceMatched: class extends FlagBase { constructor() { super(true, 0, 'Place matched from PNH data.'); } }, PlaceLocked: class extends FlagBase { constructor() { super(true, 0, 'Place locked.'); } }, NewPlaceSubmit: class extends ActionFlag { constructor() { super(true, 0, 'No PNH match. If it\'s a chain: ', 'Submit new chain data', 'Submit info for a new chain through the linked form'); } action() { window.open(newPlaceURL); } }, ApprovalSubmit: class extends ActionFlag { constructor(region, pnhOrderNum, pnhNameTemp, placePL) { super(true, 0, 'PNH data exists but is not approved for this region: ', 'Request approval', 'Request region/country approval of this place'); this.region = region; this.pnhOrderNum = pnhOrderNum; this.pnhNameTemp = pnhNameTemp; this.placePL = placePL; } action() { if ( PMUserList.hasOwnProperty(this.region) && PMUserList[this.region].approvalActive ) { var forumPMInputs = { subject: '' + this.pnhOrderNum + ' PNH approval for "' + this.pnhNameTemp + '"', message: 'Please approve "' + this.pnhNameTemp + '" for the ' + this.region + ' region. Thanks\n \nPNH order number: ' + this.pnhOrderNum + '\n \nPermalink: ' + this.placePL + '\n \nPNH Link: ' + _URLS.usaPnh, preview: 'Preview', attach_sig: 'on' }; forumPMInputs['address_list[u]['+PMUserList[this.region].modID+']'] = 'to'; // Sends a PM to the regional mod instead of the submission form newForumPost('https://www.waze.com/forum/ucp.php?i=pm&mode=compose', forumPMInputs); } else { window.open(approveRegionURL); } } }, PlaceWebsite: class extends ActionFlag { // NOTE: This class is now only used to display the store locator button. It can be updated to remove/change anything that doesn't serve that purpose. constructor() { super(true, 0, '', 'Location Finder', 'Look up details about this location on the chain\'s finder web page'); } action() { let openPlaceWebsiteURL, linkProceed = true; if (updateURL) { // replace WME url with storefinder URLs if they are in the PNH data if (customStoreFinder) { openPlaceWebsiteURL = customStoreFinderURL; } else if (customStoreFinderLocal) { openPlaceWebsiteURL = customStoreFinderLocalURL; } // If the user has 'never' opened a localized store finder URL, then warn them (just once) if (localStorage.getItem(_SETTING_IDS.sfUrlWarning) === '0' && customStoreFinderLocal) { linkProceed = false; if (confirm('***Localized store finder sites often show multiple nearby results. Please make sure you pick the right location.\nClick OK to agree and continue.') ) { // if the category doesn't translate, then pop an alert that will make a forum post to the thread localStorage.setItem(_SETTING_IDS.sfUrlWarning, '1'); // prevent future warnings linkProceed = true; } } } else { let url = getSelectedVenue().url; if (!/^https?:\/\//.test(url)) url = 'http://' + url; openPlaceWebsiteURL = url; } // open the link depending on new window setting if (linkProceed) { if ( $('#WMEPH-WebSearchNewTab').prop('checked') ) { window.open(openPlaceWebsiteURL); } else { window.open(openPlaceWebsiteURL, searchResultsWindowName, searchResultsWindowSpecs); } } } } }; // END Flag namespace function getBannButt() { return { hnDashRemoved: null, fullAddressInference: null, nameMissing: null, //The buttons are appended in the code... plaIsPublic: null, plaNameMissing: null, plaNameNonStandard: null, indianaLiquorStoreHours: null, hoursOverlap: null, unmappedRegion: null, restAreaName: null, restAreaNoTransportation: null, restAreaGas: null, restAreaScenic: null, restAreaSpec: null, // if the gas brand and name don't match gasMismatch: null, gasUnbranded: null, gasMkPrim: null, isThisAPilotTravelCenter: null, hotelMkPrim: null, changeToPetVet: null, changeSchool2Offices: null, pointNotArea: null, areaNotPoint: null, hnMissing: null, hnNonStandard: null, HNRange: null, streetMissing: null, cityMissing: null, bankType1: null, bankBranch: null, standaloneATM: null, bankCorporate: null, catPostOffice: null, ignEdited: null, wazeBot: null, parentCategory: null, checkDescription: null, overlapping: null, suspectDesc: null, resiTypeName: null, mismatch247: null, phoneInvalid: null, areaNotPointMid: null, pointNotAreaMid: null, longURL: null, gasNoBrand: null, subFuel: null, areaNotPointLow: null, pointNotAreaLow: null, formatUSPS: null, missingUSPSAlt: null, missingUSPSZipAlt: null, missingUSPSDescription: null, catHotel: null, localizedName: null, specCaseMessage: null, pnhCatMess: null, specCaseMessageLow: null, changeToHospitalUrgentCare: null, changeToDoctorClinic: null, extProviderMissing: null, urlMissing: null, badAreaCode: null, phoneMissing: null, noHours: null, plaLotTypeMissing: null, plaCostTypeMissing: null, plaPaymentTypeMissing: null, plaLotElevationMissing: null, plaSpaces: null, noPlaStopPoint: null, plaStopPointUnmoved: null, plaCanExitWhileClosed: null, plaHasAccessibleParking: null, allDayHoursFixed: null, resiTypeNameSoft: null, localURL: null, lockRPP: null, addAlias: null, addCat2: new Flag.AddCat2(), // special case flag addPharm: new Flag.AddPharm(), // special case flag addSuper: new Flag.AddSuper(), // special case flag appendAMPM: new Flag.AppendAMPM(), // special case flag addATM: new Flag.AddATM(), // special case flag addConvStore: new Flag.AddConvStore(), // special case flag isThisAPostOffice: null, STC: null, sfAliases: null, placeMatched: null, placeLocked: null, NewPlaceSubmit: null, ApprovalSubmit: null, PlaceWebsite: null }; // END bannButt definitions } // Main script function harmonizePlaceGo(item, useFlag, actions) { actions = actions || []; // Used for collecting all actions to be applied to the model. var hpMode = { harmFlag: false, hlFlag: false, scanFlag: false }; if ( useFlag.indexOf('harmonize') > -1 ) { hpMode.harmFlag = true; phlog('Running script on selected place...'); } if ( useFlag.indexOf('highlight') > -1 ) { hpMode.hlFlag = true; } if ( useFlag.indexOf('scan') > -1 ) { hpMode.scanFlag = true; } var placePL = getItemPL(); // set up external post div and pull place PL // https://www.waze.com/editor/?env=usa&lon=-80.60757&lat=28.17850&layers=1957&zoom=4&segments=86124344&update_requestsFilter=false&problemsFilter=false&mapProblemFilter=0&mapUpdateRequestFilter=0&venueFilter=1 placePL = placePL.replace(/\&layers=[^\&]+(\&?)/g, '$1'); // remove Permalink Layers placePL = placePL.replace(/\&s=[^\&]+(\&?)/g, '$1'); // remove Permalink Layers placePL = placePL.replace(/\&update_requestsFilter=[^\&]+(\&?)/g, '$1'); // remove Permalink Layers placePL = placePL.replace(/\&problemsFilter=[^\&]+(\&?)/g, '$1'); // remove Permalink Layers placePL = placePL.replace(/\&mapProblemFilter=[^\&]+(\&?)/g, '$1'); // remove Permalink Layers placePL = placePL.replace(/\&mapUpdateRequestFilter=[^\&]+(\&?)/g, '$1'); // remove Permalink Layers placePL = placePL.replace(/\&venueFilter=[^\&]+(\&?)/g, '$1'); // remove Permalink Layers var region, state2L; var gFormState = ''; var PNHOrderNum = '', PNHNameTemp = '', PNHNameTempWeb = ''; severityButt = 0; // Whitelist: reset flags _wl = { dupeWL: [], restAreaName: false, restAreaSpec: false, restAreaScenic: false, unmappedRegion: false, gasMismatch: false, hotelMkPrim: false, changeToOffice: false, changeToDoctorClinic: false, changeHMC2PetVet: false, changeSchool2Offices: false, pointNotArea: false, areaNotPoint: false, HNWL: false, hnNonStandard: false, HNRange: false, parentCategory: false, suspectDesc: false, resiTypeName: false, longURL: false, gasNoBrand: false, subFuel: false, hotelLocWL: false, localizedName: false, urlWL: false, phoneWL: false, aCodeWL: false, noHours: false, nameMissing: false, plaNameMissing: false, extProviderMissing: false }; // **** Set up banner action buttons. Structure: // active: false until activated in the script // severity: determines the color of the banners and whether locking occurs // message: The text before the button option // value: button text // title: tooltip text // action: The action that happens if the button is pressed // WL terms are for whitelisting bannButt = getBannButt(); if (hpMode.harmFlag) { bannButt2 = { placesWiki: { active: true, severity: 0, message: '', value: 'Places wiki', title: 'Open the places Wazeopedia (wiki) page', action: function() { window.open(_URLS.placesWiki); } }, restAreaWiki: { active: false, severity: 0, message: '', value: 'Rest Area wiki', title: 'Open the Rest Area wiki page', action: function() { window.open(_URLS.restAreaWiki); } }, clearWL: { active: false, severity: 0, message: '', value: 'Clear place whitelist', title: 'Clear all Whitelisted fields for this place', action: function() { if (confirm('Are you sure you want to clear all whitelisted fields for this place?') ) { // misclick check delete venueWhitelist[itemID]; saveWL_LS(true); harmonizePlaceGo(item,'harmonize'); // rerun the script to check all flags again } } }, // END placesWiki definition PlaceErrorForumPost: { active: true, severity: 0, message: '', value: 'Report script error', title: 'Report a script error', action: function() { reportError({ subject: 'WMEPH Bug report: Script Error', message: 'Script version: ' + _SCRIPT_VERSION + devVersStr + '\nPermalink: ' + placePL + '\nPlace name: ' + item.attributes.name + '\nCountry: ' + addr.country.name + '\n--------\nDescribe the error: \n ' }); } } }; // END bannButt2 definitions // set up banner action buttons. Structure: // active: false until activated in the script // checked: whether the service is already set on the place. Determines grey vs white icon color // icon: button icon name // value: button text (Not used for Icons, keep as backup // title: tooltip text // action: The action that happens if the button is pressed bannServ = { addValet: { // append optional Alias to the name active: false, checked: false, icon: 'serv-valet', w2hratio: 50/50, value: 'Valet', title: 'Valet service', servIDIndex: 0, action: function(actions, checked) { setServiceChecked(this, checked, actions); }, pnhOverride: false, actionOn: function(actions) { this.action(actions, true); }, actionOff: function(actions) { this.action(actions, false); } }, addDriveThru: { // append optional Alias to the name active: false, checked: false, icon: 'serv-drivethru', w2hratio: 78/50, value: 'DriveThru', title: 'Drive-thru', servIDIndex: 1, action: function(actions, checked) { setServiceChecked(this, checked, actions); }, pnhOverride: false, actionOn: function(actions) { this.action(actions, true); }, actionOff: function(actions) { this.action(actions, false); } }, addWiFi: { // append optional Alias to the name active: false, checked: false, icon: 'serv-wifi', w2hratio: 67/50, value: 'WiFi', title: 'Wi-Fi', servIDIndex: 2, action: function(actions, checked) { setServiceChecked(this, checked, actions); }, pnhOverride: false, actionOn: function(actions) { this.action(actions, true); }, actionOff: function(actions) { this.action(actions, false); } }, addRestrooms: { // append optional Alias to the name active: false, checked: false, icon: 'serv-restrooms', w2hratio: 49/50, value: 'Restroom', title: 'Restrooms', servIDIndex: 3, action: function(actions, checked) { setServiceChecked(this, checked, actions); }, pnhOverride: false, actionOn: function(actions) { this.action(actions, true); }, actionOff: function(actions) { this.action(actions, false); } }, addCreditCards: { // append optional Alias to the name active: false, checked: false, icon: 'serv-credit', w2hratio: 73/50, value: 'CC', title: 'Accepts credit cards', servIDIndex: 4, action: function(actions, checked) { setServiceChecked(this, checked, actions); }, pnhOverride: false, actionOn: function(actions) { this.action(actions, true); }, actionOff: function(actions) { this.action(actions, false); } }, addReservations: { // append optional Alias to the name active: false, checked: false, icon: 'serv-reservations', w2hratio: 55/50, value: 'Reserve', title: 'Reservations', servIDIndex: 5, action: function(actions, checked) { setServiceChecked(this, checked, actions); }, pnhOverride: false, actionOn: function(actions) { this.action(actions, true); }, actionOff: function(actions) { this.action(actions, false); } }, addOutside: { // append optional Alias to the name active: false, checked: false, icon: 'serv-outdoor', w2hratio: 73/50, value: 'OusideSeat', title: 'Outdoor seating', servIDIndex: 6, action: function(actions, checked) { setServiceChecked(this, checked, actions); }, pnhOverride: false, actionOn: function(actions) { this.action(actions, true); }, actionOff: function(actions) { this.action(actions, false); } }, addAC: { // append optional Alias to the name active: false, checked: false, icon: 'serv-ac', w2hratio: 50/50, value: 'AC', title: 'Air conditioning', servIDIndex: 7, action: function(actions, checked) { setServiceChecked(this, checked, actions); }, pnhOverride: false, actionOn: function(actions) { this.action(actions, true); }, actionOff: function(actions) { this.action(actions, false); } }, addParking: { // append optional Alias to the name active: false, checked: false, icon: 'serv-parking', w2hratio: 46/50, value: 'Customer parking', title: 'Parking', servIDIndex: 8, action: function(actions, checked) { setServiceChecked(this, checked, actions); }, pnhOverride: false, actionOn: function(actions) { this.action(actions, true); }, actionOff: function(actions) { this.action(actions, false); } }, addDeliveries: { // append optional Alias to the name active: false, checked: false, icon: 'serv-deliveries', w2hratio: 86/50, value: 'Delivery', title: 'Deliveries', servIDIndex: 9, action: function(actions, checked) { setServiceChecked(this, checked, actions); }, pnhOverride: false, actionOn: function(actions) { this.action(actions, true); }, actionOff: function(actions) { this.action(actions, false); } }, addTakeAway: { // append optional Alias to the name active: false, checked: false, icon: 'serv-takeaway', w2hratio: 34/50, value: 'Take-out', title: 'Take-out', servIDIndex: 10, action: function(actions, checked) { setServiceChecked(this, checked, actions); }, pnhOverride: false, actionOn: function(actions) { this.action(actions, true); }, actionOff: function(actions) { this.action(actions, false); } }, addWheelchair: { // add service active: false, checked: false, icon: 'serv-wheelchair', w2hratio: 50/50, value: 'WhCh', title: 'Wheelchair accessible', servIDIndex: 11, action: function(actions, checked) { setServiceChecked(this, checked, actions); }, pnhOverride: false, actionOn: function(actions) { this.action(actions, true); }, actionOff: function(actions) { this.action(actions, false); } }, addDisabilityParking: { active: false, checked: false, icon: 'serv-wheelchair', w2hratio: 50/50, value: 'DisabilityParking', title: 'Disability parking', servIDIndex: 12, action: function(actions, checked) { setServiceChecked(this, checked, actions); }, pnhOverride: false, actionOn: function(actions) { this.action(actions, true); }, actionOff: function(actions) { this.action(actions, false); } }, add247: { // add 24/7 hours active: false, checked: false, icon: 'serv-247', w2hratio: 73/50, value: '247', title: 'Hours: Open 24\/7', action: function(actions) { if (!bannServ.add247.checked) { let venue = getSelectedVenue(); addUpdateAction(venue, { openingHours: [{days: [1,2,3,4,5,6,0], fromHour: '00:00', toHour: '00:00'}] }, actions); bannServ.add247.checked = true; bannButt.noHours = null; } }, actionOn: function(actions) { this.action(actions); } } }; // END bannServ definitions // Update icons to reflect current WME place services updateServicesChecks(bannServ); //Setting switch for the Places Wiki button if ( $('#WMEPH-HidePlacesWiki').prop('checked') ) { bannButt2.placesWiki.active = false; } if ( $('#WMEPH-HideReportError').prop('checked') ) { bannButt2.PlaceErrorForumPost.active = false; } // // provide Google search link to places // if (_USER.isDevUser || _USER.isBetaUser || _USER.rank > 1) { // enable the link for all places, for R2+ and betas // bannButt.webSearch.active = true; // } // reset PNH lock level PNHLockLevel = -1; } // If place has hours of 0:00-23:59, highlight yellow or if harmonizing, convert to All Day. bannButt.allDayHoursFixed = Flag.AllDayHoursFixed.eval(item, hpMode, actions); var lockOK = true; // if nothing goes wrong, then place will be locked var categories = item.attributes.categories; newCategories = categories.slice(0); var nameParts = getNameParts(item.attributes.name); var newNameSuffix = nameParts.suffix; newName = nameParts.base; newAliases = item.attributes.aliases.slice(0); var brand = item.attributes.brand; var newDescripion = item.attributes.description; newURL = item.attributes.url; var newURLSubmit = ''; if (newURL !== null && newURL !== '') { newURLSubmit = newURL; } newPhone = item.attributes.phone; let addr = item.getAddress(); if ( addr.hasOwnProperty('attributes') ) { addr = addr.attributes; } var PNHNameRegMatch; // Some user submitted places have no data in the country, state and address fields. let inferredAddress; if (hpMode.harmFlag) { let result = Flag.FullAddressInference.eval(item, addr, actions); if (result.exit) return; bannButt.fullAddressInference = result.flag; inferredAddress = result.inferredAddress; if (result.inferredAddress) addr = result.inferredAddress; if (result.noLock) lockOK = false; } else if (hpMode.hlFlag) { let result = Flag.FullAddressInference.evalHL(item, addr); if (result) return result; } let result; // Check parking lot attributes. if (hpMode.harmFlag && item.isParkingLot()) bannServ.addDisabilityParking.active = true; result = Flag.PlaCostTypeMissing.eval(item, hpMode); bannButt.plaCostTypeMissing = result.flag; if (result.noLock) lockOK = false; result = Flag.PlaLotElevationMissing.eval(item); bannButt.plaLotElevationMissing = result.flag; if (result.noLock) lockOK = false; result = Flag.PlaSpaces.eval(item, hpMode); bannButt.plaSpaces = result.flag; result = Flag.PlaLotTypeMissing.eval(item, hpMode); bannButt.plaLotTypeMissing = result.flag; if (result.noLock) lockOK = false; bannButt.noPlaStopPoint = Flag.NoPlaStopPoint.eval(item).flag; bannButt.plaStopPointUnmoved = Flag.PlaStopPointUnmoved.eval(item).flag; bannButt.plaCanExitWhileClosed = Flag.PlaCanExitWhileClosed.eval(item, hpMode).flag; bannButt.plaPaymentTypeMissing = Flag.PlaPaymentTypeMissing.eval(item).flag; bannButt.plaHasAccessibleParking = Flag.PlaHasAccessibleParking.eval(item, hpMode).flag; // Check categories that maybe should be Hospital / Urgent Care, or Doctor / Clinic. bannButt.changeToHospitalUrgentCare = Flag.ChangeToHospitalUrgentCare.eval(item, hpMode).flag; if (hpMode.harmFlag && item.attributes.categories.indexOf('HOSPITAL_URGENT_CARE') > -1) { //bannButt.changeToDoctorClinic.active = true; //bannButt.changeToDoctorClinic.severity = 0; } // Whitelist breakout if place exists on the Whitelist and the option is enabled itemID = item.attributes.id; var itemGPS; if (venueWhitelist.hasOwnProperty(itemID) && (hpMode.harmFlag || (hpMode.hlFlag && !$('#WMEPH-DisableWLHL').prop('checked')))) { // Enable the clear WL button if any property is true for (var WLKey in venueWhitelist[itemID]) { // loop thru the venue WL keys if ( venueWhitelist[itemID].hasOwnProperty(WLKey) && (venueWhitelist[itemID][WLKey].active || false) ) { if (hpMode.harmFlag) bannButt2.clearWL.active = true; _wl[WLKey] = venueWhitelist[itemID][WLKey]; } } if (venueWhitelist[itemID].hasOwnProperty('dupeWL') && venueWhitelist[itemID].dupeWL.length > 0) { if (hpMode.harmFlag) bannButt2.clearWL.active = true; _wl.dupeWL = venueWhitelist[itemID].dupeWL; } // Update address and GPS info for the place if (hpMode.harmFlag) { // get GPS lat/long coords from place, call as itemGPS.lat, itemGPS.lon if (!itemGPS) itemGPS = OL.Layer.SphericalMercator.inverseMercator(item.attributes.geometry.getCentroid().x,item.attributes.geometry.getCentroid().y); venueWhitelist[itemID].city = addr.city.attributes.name; // Store city for the venue venueWhitelist[itemID].state = addr.state.name; // Store state for the venue venueWhitelist[itemID].country = addr.country.name; // Store country for the venue venueWhitelist[itemID].gps = itemGPS; // Store GPS coords for the venue } } // Country restrictions if (hpMode.harmFlag && (addr.county === null || addr.state === null)) { alert('Country and/or state could not be determined. Edit the place address and run WMEPH again.'); return; } var countryName = addr.country.name; var stateName = addr.state.name; if (countryName === 'United States') { _countryCode = 'USA'; } else if (countryName === 'Canada') { _countryCode = 'CAN'; } else if (countryName === 'American Samoa') { _countryCode = 'USA'; } else if (countryName === 'Guam') { _countryCode = 'USA'; } else if (countryName === 'Northern Mariana Islands') { _countryCode = 'USA'; } else if (countryName === 'Puerto Rico') { _countryCode = 'USA'; } else if (countryName === 'Virgin Islands (U.S.)') { _countryCode = 'USA'; } else { if (hpMode.harmFlag) { alert('At present this script is not supported in this country.'); } return 3; } // Parse state-based data state2L = 'Unknown'; region = 'Unknown'; for (var usdix=1; usdix < _PNH_DATA.states.length; usdix++) { stateDataTemp = _PNH_DATA.states[usdix].split('|'); if (stateName === stateDataTemp[ps_state_ix]) { state2L = stateDataTemp[ps_state2L_ix]; region = stateDataTemp[ps_region_ix]; gFormState = stateDataTemp[ps_gFormState_ix]; if (stateDataTemp[ps_defaultLockLevel_ix].match(/[1-5]{1}/) !== null) { defaultLockLevel = stateDataTemp[ps_defaultLockLevel_ix] - 1; // normalize by -1 } else { if (hpMode.harmFlag) { alert('Lock level sheet data is not correct'); } else if (hpMode.hlFlag) { return '3'; } } areaCodeList = areaCodeList+','+stateDataTemp[ps_areacode_ix]; break; } // If State is not found, then use the country if (countryName === stateDataTemp[ps_state_ix]) { state2L = stateDataTemp[ps_state2L_ix]; region = stateDataTemp[ps_region_ix]; gFormState = stateDataTemp[ps_gFormState_ix]; if (stateDataTemp[ps_defaultLockLevel_ix].match(/[1-5]{1}/) !== null) { defaultLockLevel = stateDataTemp[ps_defaultLockLevel_ix] - 1; // normalize by -1 } else { if (hpMode.harmFlag) { alert('Lock level sheet data is not correct'); } else if (hpMode.hlFlag) { return '3'; } } areaCodeList = areaCodeList+','+stateDataTemp[ps_areacode_ix]; break; } } if (state2L === 'Unknown' || region === 'Unknown') { // if nothing found: if (hpMode.harmFlag) { if (confirm('WMEPH: Localization Error!\nClick OK to report this error') ) { // if the category doesn't translate, then pop an alert that will make a forum post to the thread let data = { subject: 'WMEPH Localization Error report', message: 'Error report: Localization match failed for "' + stateName + '".' }; if (_PNH_DATA.states.length === 0) { data.message += ' _PNH_DATA.states array is empty.'; } else { data.message += ' state2L = ' + stateDataTemp[ps_state2L_ix] + '. region = ' + stateDataTemp[ps_region_ix]; } reportError(data); } } return 3; } // Gas station treatment (applies to all including PNH) // Brand checking result = Flag.GasNoBrand.eval(item); bannButt.gasNoBrand = result.flag; if (result.noLock) lockOK = false; result = Flag.GasUnbranded.eval(item); bannButt.gasUnbranded = result.flag; if (result.noLock) lockOK = false; result = Flag.IsThisAPilotTravelCenter.eval(item, hpMode, state2L, newName, actions); bannButt.isThisAPilotTravelCenter = result.flag; newName = result.newName; if (item.isGasStation()) { // If no gas station name, replace with brand name if (hpMode.harmFlag && (!newName || newName.trim().length === 0) && item.attributes.brand) { newName = item.attributes.brand; actions.push(new UpdateObject(item, {name: newName })); _updatedFields.name.updated = true; } // Add convenience store category to station if (newCategories.indexOf('CONVENIENCE_STORE') === -1 && !bannButt.subFuel) { bannButt.addConvStore.active = true; } } // END Gas Station Checks // Note for Indiana editors to check liquor store hours if Sunday hours haven't been added yet. var tempAddr = item.getAddress(); if (hpMode.harmFlag && tempAddr && tempAddr.getStateName() === 'Indiana' && !item.isResidential() && [/\bbeers?\b/,/\bwines?\b/,/\bliquor\b/,/\bspirits\b/].some(re => re.test(newName)) && !item.attributes.openingHours.some(entry => entry.days.indexOf(0) > -1)) { if (!_wl.indianaLiquorStoreHours) bannButt.indianaLiquorStoreHours = new Flag.IndianaLiquorStoreHours(); } var isLocked = item.attributes.lockRank >= (PNHLockLevel > -1 ? PNHLockLevel : defaultLockLevel); // Clear attributes from residential places if (item.attributes.residential) { if (hpMode.harmFlag) { if ( !$('#WMEPH-AutoLockRPPs').prop('checked') ) { lockOK = false; } if (item.attributes.name !== '') { // Set the residential place name to the address (to clear any personal info) phlogdev('Residential Name reset'); actions.push(new UpdateObject(item, {name: ''})); // no field HL } newCategories = ['RESIDENCE_HOME']; // newDescripion = null; if (item.attributes.description !== null && item.attributes.description !== '') { // remove any description phlogdev('Residential description cleared'); actions.push(new UpdateObject(item, {description: null})); // no field HL } // newPhone = null; if (item.attributes.phone !== null && item.attributes.phone !== '') { // remove any phone info phlogdev('Residential Phone cleared'); actions.push(new UpdateObject(item, {phone: null})); // no field HL } // newURL = null; if (item.attributes.url !== null && item.attributes.url !== '') { // remove any url phlogdev('Residential URL cleared'); actions.push(new UpdateObject(item, {url: null})); // no field HL } if (item.attributes.services.length > 0) { phlogdev('Residential services cleared'); actions.push(new UpdateObject(item, {services: [] })); // no field HL } } // NOTE: do not use is2D() function. It doesn't seem to be 100% reliable. if (!item.isPoint()) { bannButt.pointNotArea = new Flag.PointNotArea(); } } else if (item.isParkingLot() || (newName && newName.trim().length > 0)) { // for non-residential places if (_USER.rank >= 2 && item.areExternalProvidersEditable() && !(item.isParkingLot() && $('#WMEPH-DisablePLAExtProviderCheck').prop('checked'))) { if (!newCategories.some(c => ['BRIDGE','TUNNEL','JUNCTION_INTERCHANGE','NATURAL_FEATURES','ISLAND','SEA_LAKE_POOL','RIVER_STREAM','CANAL','SWAMP_MARSH'].indexOf(c) > -1)) { var provIDs = item.attributes.externalProviderIDs; if (!provIDs || provIDs.length === 0) { var lastUpdated = item.isNew() ? Date.now() : item.attributes.updatedOn ? item.attributes.updatedOn : item.attributes.createdOn; var weeksSinceLastUpdate = (Date.now() - lastUpdated) / 604800000; bannButt.extProviderMissing = new Flag.ExtProviderMissing(); if (isLocked && weeksSinceLastUpdate >= 26 && !item.isUpdated() && (!actions || actions.length === 0)) { bannButt.extProviderMissing.severity = 3; bannButt.extProviderMissing.message += ' and place has not been edited for over 6 months. Edit a property (or nudge) and save to reset the 6 month timer: '; } else if (!isLocked) { bannButt.extProviderMissing.severity = 0; // This will be changed to 3 later if the user does not choose to lock the place. bannButt.extProviderMissing.message += ': '; delete bannButt.extProviderMissing.value; //delete bannButt.extProviderMissing.action; } else { bannButt.extProviderMissing.severity = 0; bannButt.extProviderMissing.message += ': '; delete bannButt.extProviderMissing.value; //delete bannButt.extProviderMissing.action; } } } } // Place Harmonization var PNHMatchData; if (hpMode.harmFlag) { if (item.isParkingLot()) { PNHMatchData = ['NoMatch']; } else { PNHMatchData = harmoList(newName,state2L,region,_countryCode,newCategories,item,placePL); // check against the PNH list } } else if (hpMode.hlFlag) { PNHMatchData = ['Highlight']; } PNHNameRegMatch = false; if (PNHMatchData[0] !== 'NoMatch' && PNHMatchData[0] !== 'ApprovalNeeded' && PNHMatchData[0] !== 'Highlight' ) { // *** Replace place data with PNH data PNHNameRegMatch = true; var showDispNote = true; var updatePNHName = true; // Break out the data headers var _PNH_DATA_headers; _PNH_DATA_headers = _PNH_DATA[_countryCode].pnh[0].split('|'); var ph_name_ix = _PNH_DATA_headers.indexOf('ph_name'); var ph_aliases_ix = _PNH_DATA_headers.indexOf('ph_aliases'); var ph_category1_ix = _PNH_DATA_headers.indexOf('ph_category1'); var ph_category2_ix = _PNH_DATA_headers.indexOf('ph_category2'); var ph_description_ix = _PNH_DATA_headers.indexOf('ph_description'); var ph_url_ix = _PNH_DATA_headers.indexOf('ph_url'); var ph_order_ix = _PNH_DATA_headers.indexOf('ph_order'); // var ph_notes_ix = _PNH_DATA_headers.indexOf('ph_notes'); var ph_speccase_ix = _PNH_DATA_headers.indexOf('ph_speccase'); var ph_sfurl_ix = _PNH_DATA_headers.indexOf('ph_sfurl'); var ph_sfurllocal_ix = _PNH_DATA_headers.indexOf('ph_sfurllocal'); // var ph_forcecat_ix = _PNH_DATA_headers.indexOf('ph_forcecat'); var ph_displaynote_ix = _PNH_DATA_headers.indexOf('ph_displaynote'); // Retrieve the data from the PNH line(s) var nsMultiMatch = false, orderList = []; if (PNHMatchData.length > 1) { // If multiple matches, then var brandParent = -1, pmdTemp, pmdSpecCases, PNHMatchDataHold = PNHMatchData[0].split('|'); for (var pmdix=0; pmdix<PNHMatchData.length; pmdix++) { // For each of the matches, pmdTemp = PNHMatchData[pmdix].split('|'); // Split the PNH data line orderList.push(pmdTemp[ph_order_ix]); // Add Order number to a list if (pmdTemp[ph_speccase_ix].match(/brandParent(\d{1})/) !== null) { // If there is a brandParent flag, prioritize by highest match pmdSpecCases = pmdTemp[ph_speccase_ix].match(/brandParent(\d{1})/)[1]; if (pmdSpecCases > brandParent) { // if the match is more specific than the previous ones: brandParent = pmdSpecCases; // Update the brandParent level PNHMatchDataHold = pmdTemp; // Update the PNH data line } } else { // if any item has no brandParent structure, use highest brandParent match but post an error nsMultiMatch = true; } } PNHMatchData = PNHMatchDataHold; } else { PNHMatchData = PNHMatchData[0].split('|'); // Single match just gets direct split } var priPNHPlaceCat = catTranslate(PNHMatchData[ph_category1_ix]); // translate primary category to WME code // if the location has multiple matches, then pop an alert that will make a forum post to the thread if (nsMultiMatch) { if (confirm('WMEPH: Multiple matches found!\nDouble check the script changes.\nClick OK to report this situation.') ) { reportError({ subject: 'Order Nos. "' + orderList.join(', ') + '" WMEPH Multiple match report', message: 'Error report: PNH Order Nos. "' + orderList.join(', ') + '" are ambiguous multiple matches.\n \nExample Permalink: ' + placePL + '' }); } } // Check special cases var specCases, scFlag, localURLcheck = ''; if (ph_speccase_ix > -1) { // If the special cases column exists specCases = PNHMatchData[ph_speccase_ix]; // pulls the speccases field from the PNH line if (specCases !== '0' && specCases !== '') { specCases = specCases.replace(/, /g, ',').split(','); // remove spaces after commas and split by comma } for (var scix = 0; scix < specCases.length; scix++) { // find any button/message flags in the special case (format: buttOn_xyzXyz, etc.) if ( specCases[scix].match(/^buttOn_/g) !== null ) { scFlag = specCases[scix].match(/^buttOn_(.+)/i)[1]; if (scFlag !== 'addCat2' || item.attributes.categories.indexOf(catTranslate(PNHMatchData[ph_category2_ix])) === -1) { bannButt[scFlag].active = true; } } else if ( specCases[scix].match(/^buttOff_/g) !== null ) { scFlag = specCases[scix].match(/^buttOff_(.+)/i)[1]; bannButt[scFlag].active = false; } else if ( specCases[scix].match(/^messOn_/g) !== null ) { scFlag = specCases[scix].match(/^messOn_(.+)/i)[1]; bannButt[scFlag].active = true; } else if ( specCases[scix].match(/^messOff_/g) !== null ) { scFlag = specCases[scix].match(/^messOff_(.+)/i)[1]; bannButt[scFlag].active = false; } else if ( specCases[scix].match(/^psOn_/g) !== null ) { scFlag = specCases[scix].match(/^psOn_(.+)/i)[1]; bannServ[scFlag].actionOn(actions); bannServ[scFlag].pnhOverride = true; } else if ( specCases[scix].match(/^psOff_/g) !== null ) { scFlag = specCases[scix].match(/^psOff_(.+)/i)[1]; bannServ[scFlag].actionOff(actions); bannServ[scFlag].pnhOverride = true; } // parseout localURL data if exists (meaning place can have a URL distinct from the chain URL if ( specCases[scix].match(/^localURL_/g) !== null ) { localURLcheck = specCases[scix].match(/^localURL_(.+)/i)[1]; } // parse out optional alt-name if ( specCases[scix].match(/^optionAltName<>(.+)/g) !== null ) { optionalAlias = specCases[scix].match(/^optionAltName<>(.+)/i)[1]; if (newAliases.indexOf(optionalAlias) === -1) { bannButt.addAlias = new Flag.AddAlias(); } } // Gas Station forceBranding if ( ['GAS_STATION'].indexOf(priPNHPlaceCat) > -1 && specCases[scix].match(/^forceBrand<>(.+)/i) !== null ) { var forceBrand = specCases[scix].match(/^forceBrand<>(.+)/i)[1]; if (item.attributes.brand !== forceBrand) { actions.push(new UpdateObject(item, { brand: forceBrand })); _updatedFields.brand.updated = true; phlogdev('Gas brand updated from PNH'); } } // Check Localization if ( specCases[scix].match(/^checkLocalization<>(.+)/i) !== null ) { updatePNHName = false; var baseName = specCases[scix].match(/^checkLocalization<>(.+)/i)[1]; var baseNameRE = new RegExp(baseName, 'g'); if ( (newName + (newNameSuffix ? newNameSuffix : '')).match(baseNameRE) === null ) { bannButt.localizedName = new Flag.LocalizedName(); if (_wl.localizedName) { bannButt.localizedName.WLactive = false; } //bannButt.PlaceWebsite.value = 'Place Website'; if (ph_displaynote_ix > -1 && PNHMatchData[ph_displaynote_ix] !== '0' && PNHMatchData[ph_displaynote_ix] !== '') { bannButt.localizedName.message = PNHMatchData[ph_displaynote_ix]; } } showDispNote = false; } // Prevent name change if ( specCases[scix].match(/keepName/g) !== null ) { updatePNHName = false; } } } // If it's a place that also sells fuel, enable the button if ( PNHMatchData[ph_speccase_ix] === 'subFuel' && newName.toUpperCase().indexOf('GAS') === -1 && newName.toUpperCase().indexOf('FUEL') === -1 ) { bannButt.subFuel = new Flag.SubFuel(); if (_wl.subFuel) { bannButt.subFuel.WLactive = false; } } // Display any notes for the specific place if (showDispNote && ph_displaynote_ix > -1 && PNHMatchData[ph_displaynote_ix] !== '0' && PNHMatchData[ph_displaynote_ix] !== '' ) { if ( containsAny(specCases,['pharmhours']) ) { if ( item.attributes.description.toUpperCase().indexOf('PHARMACY') === -1 || ( item.attributes.description.toUpperCase().indexOf('HOURS') === -1 && item.attributes.description.toUpperCase().indexOf('HRS') === -1 ) ) { bannButt.specCaseMessage = new Flag.SpecCaseMessage(PNHMatchData[ph_displaynote_ix]); } } else if ( containsAny(specCases,['drivethruhours']) ) { if ( item.attributes.description.toUpperCase().indexOf('DRIVE') === -1 || ( item.attributes.description.toUpperCase().indexOf('HOURS') === -1 && item.attributes.description.toUpperCase().indexOf('HRS') === -1 ) ) { if ( $('#service-checkbox-'+'DRIVETHROUGH').prop('checked') ) { bannButt.specCaseMessage = new Flag.SpecCaseMessage(PNHMatchData[ph_displaynote_ix]); } else { bannButt.specCaseMessageLow = new Flag.SpecCaseMessageLow(PNHMatchData[ph_displaynote_ix]); } } } else { bannButt.specCaseMessageLow = new Flag.SpecCaseMessageLow(PNHMatchData[ph_displaynote_ix]); } } // Localized Storefinder code: customStoreFinderLocal = false; customStoreFinderLocalURL = ''; customStoreFinder = false; customStoreFinderURL = ''; if (ph_sfurl_ix > -1) { // if the sfurl column exists... if ( ph_sfurllocal_ix > -1 && PNHMatchData[ph_sfurllocal_ix] !== '' && PNHMatchData[ph_sfurllocal_ix] !== '0' ) { if ( !bannButt.localizedName ) { bannButt.PlaceWebsite = new Flag.PlaceWebsite(); bannButt.PlaceWebsite.value = 'Location Finder (L)'; } var tempLocalURL = PNHMatchData[ph_sfurllocal_ix].replace(/ /g,'').split('<>'); var searchStreet = '', searchCity = '', searchState = ''; if ('string' === typeof addr.street.name) { searchStreet = addr.street.name; } var searchStreetPlus = searchStreet.replace(/ /g, '+'); searchStreet = searchStreet.replace(/ /g, '%20'); if ('string' === typeof addr.city.attributes.name) { searchCity = addr.city.attributes.name; } var searchCityPlus = searchCity.replace(/ /g, '+'); searchCity = searchCity.replace(/ /g, '%20'); if ('string' === typeof addr.state.name) { searchState = addr.state.name; } var searchStatePlus = searchState.replace(/ /g, '+'); searchState = searchState.replace(/ /g, '%20'); for (var tlix = 1; tlix<tempLocalURL.length; tlix++) { if (tempLocalURL[tlix] === 'ph_streetName') { customStoreFinderLocalURL = customStoreFinderLocalURL + searchStreet; } else if (tempLocalURL[tlix] === 'ph_streetNamePlus') { customStoreFinderLocalURL = customStoreFinderLocalURL + searchStreetPlus; } else if (tempLocalURL[tlix] === 'ph_cityName') { customStoreFinderLocalURL = customStoreFinderLocalURL + searchCity; } else if (tempLocalURL[tlix] === 'ph_cityNamePlus') { customStoreFinderLocalURL = customStoreFinderLocalURL + searchCityPlus; } else if (tempLocalURL[tlix] === 'ph_stateName') { customStoreFinderLocalURL = customStoreFinderLocalURL + searchState; } else if (tempLocalURL[tlix] === 'ph_stateNamePlus') { customStoreFinderLocalURL = customStoreFinderLocalURL + searchStatePlus; } else if (tempLocalURL[tlix] === 'ph_state2L') { customStoreFinderLocalURL = customStoreFinderLocalURL + state2L; } else if (tempLocalURL[tlix] === 'ph_latitudeEW') { //customStoreFinderLocalURL = customStoreFinderLocalURL + itemGPS[0]; } else if (tempLocalURL[tlix] === 'ph_longitudeNS') { //customStoreFinderLocalURL = customStoreFinderLocalURL + itemGPS[1]; } else if (tempLocalURL[tlix] === 'ph_latitudePM') { if (!itemGPS) itemGPS = OL.Layer.SphericalMercator.inverseMercator(item.attributes.geometry.getCentroid().x,item.attributes.geometry.getCentroid().y); customStoreFinderLocalURL = customStoreFinderLocalURL + itemGPS.lat; } else if (tempLocalURL[tlix] === 'ph_longitudePM') { if (!itemGPS) itemGPS = OL.Layer.SphericalMercator.inverseMercator(item.attributes.geometry.getCentroid().x,item.attributes.geometry.getCentroid().y); customStoreFinderLocalURL = customStoreFinderLocalURL + itemGPS.lon; } else if (tempLocalURL[tlix] === 'ph_latitudePMBuffMin') { if (!itemGPS) itemGPS = OL.Layer.SphericalMercator.inverseMercator(item.attributes.geometry.getCentroid().x,item.attributes.geometry.getCentroid().y); customStoreFinderLocalURL = customStoreFinderLocalURL + (itemGPS.lat-0.15).toString(); } else if (tempLocalURL[tlix] === 'ph_longitudePMBuffMin') { if (!itemGPS) itemGPS = OL.Layer.SphericalMercator.inverseMercator(item.attributes.geometry.getCentroid().x,item.attributes.geometry.getCentroid().y); customStoreFinderLocalURL = customStoreFinderLocalURL + (itemGPS.lon-0.15).toString(); } else if (tempLocalURL[tlix] === 'ph_latitudePMBuffMax') { if (!itemGPS) itemGPS = OL.Layer.SphericalMercator.inverseMercator(item.attributes.geometry.getCentroid().x,item.attributes.geometry.getCentroid().y); customStoreFinderLocalURL = customStoreFinderLocalURL + (itemGPS.lat+0.15).toString(); } else if (tempLocalURL[tlix] === 'ph_longitudePMBuffMax') { if (!itemGPS) itemGPS = OL.Layer.SphericalMercator.inverseMercator(item.attributes.geometry.getCentroid().x,item.attributes.geometry.getCentroid().y); customStoreFinderLocalURL = customStoreFinderLocalURL + (itemGPS.lon+0.15).toString(); } else if (tempLocalURL[tlix] === 'ph_houseNumber') { customStoreFinderLocalURL = customStoreFinderLocalURL + (item.attributes.houseNumber ? item.attributes.houseNumber : ''); } else { customStoreFinderLocalURL = customStoreFinderLocalURL + tempLocalURL[tlix]; } } if ( customStoreFinderLocalURL.indexOf('http') !== 0 ) { customStoreFinderLocalURL = 'http:\/\/' + customStoreFinderLocalURL; } customStoreFinderLocal = true; } else if (PNHMatchData[ph_sfurl_ix] !== '' && PNHMatchData[ph_sfurl_ix] !== '0') { if ( !bannButt.localizedName ) { bannButt.PlaceWebsite = new Flag.PlaceWebsite(); } customStoreFinderURL = PNHMatchData[ph_sfurl_ix]; if ( customStoreFinderURL.indexOf('http') !== 0 ) { customStoreFinderURL = 'http:\/\/' + customStoreFinderURL; } customStoreFinder = true; } } // Category translations var altCategories = PNHMatchData[ph_category2_ix]; if (altCategories !== '0' && altCategories !== '') { // translate alt-cats to WME code altCategories = altCategories.replace(/,[^A-Za-z0-9]*/g, ',').split(','); // tighten and split by comma for (var catix = 0; catix<altCategories.length; catix++) { var newAltTemp = catTranslate(altCategories[catix]); // translate altCats into WME cat codes if (newAltTemp === 'ERROR') { // if no translation, quit the loop phlog('Category ' + altCategories[catix] + 'cannot be translated.'); return; } else { altCategories[catix] = newAltTemp; // replace with translated element } } } // name parsing with category exceptions if (['HOTEL'].indexOf(priPNHPlaceCat) > -1) { var nameToCheck = newName + (newNameSuffix ? newNameSuffix : ''); if (nameToCheck.toUpperCase() === PNHMatchData[ph_name_ix].toUpperCase()) { // If no localization bannButt.catHotel = new Flag.CatHotel(PNHMatchData[ph_name_ix]); newName = PNHMatchData[ph_name_ix]; } else { // Replace PNH part of name with PNH name var splix = newName.toUpperCase().replace(/[-\/]/g,' ').indexOf(PNHMatchData[ph_name_ix].toUpperCase().replace(/[-\/]/g,' ') ); if (splix>-1) { var frontText = newName.slice(0,splix); var backText = newName.slice(splix+PNHMatchData[ph_name_ix].length); newName = PNHMatchData[ph_name_ix]; if (frontText.length > 0) { newName = frontText + ' ' + newName; } if (backText.length > 0) { newName = newName + ' ' + backText; } newName = newName.replace(/ {2,}/g,' '); } else { newName = PNHMatchData[ph_name_ix]; } } if ( altCategories !== '0' && altCategories !== '' ) { // if PNH alts exist insertAtIX(newCategories, altCategories, 1); // then insert the alts into the existing category array after the GS category } if ( newCategories.indexOf('HOTEL') !== 0 ) { // If no HOTEL category in the primary, flag it bannButt.hotelMkPrim = new Flag.HotelMkPrim(); if (_wl.hotelMkPrim) { bannButt.hotelMkPrim.WLactive = false; } else { lockOK = false; } } else if (newCategories.indexOf('HOTEL') > -1) { // Remove LODGING if it exists var lodgingIdx = newCategories.indexOf('LODGING'); if ( lodgingIdx > -1) { newCategories.splice(lodgingIdx,1); } } // If PNH match, set wifi service. if (PNHMatchData && !bannServ.addWiFi.checked) { bannServ.addWiFi.action(); } // Set hotel hours to 24/7 for all hotels. if (!bannServ.add247.checked) { bannServ.add247.action(); } } else if ( newCategories.indexOf('BANK_FINANCIAL') > -1 && PNHMatchData[ph_speccase_ix].indexOf('notABank') === -1 ) { // PNH Bank treatment ixBank = item.attributes.categories.indexOf('BANK_FINANCIAL'); ixATM = item.attributes.categories.indexOf('ATM'); ixOffices = item.attributes.categories.indexOf('OFFICES'); // if the name contains ATM in it if ( newName.match(/\batm\b/ig) !== null ) { if ( ixOffices === 0 ) { bannButt.bankType1 = new Flag.BankType1(); bannButt.bankBranch = new Flag.BankBranch(); bannButt.standaloneATM = new Flag.StandaloneATM(); bannButt.bankCorporate = new Flag.BankCorporate(); } else if ( ixBank === -1 && ixATM === -1 ) { bannButt.bankBranch = new Flag.BankBranch(); bannButt.standaloneATM = new Flag.StandaloneATM(); } else if ( ixATM === 0 && ixBank > 0 ) { bannButt.bankBranch = new Flag.BankBranch(); } else if ( ixBank > -1 ) { bannButt.bankBranch = new Flag.BankBranch(); bannButt.standaloneATM = new Flag.StandaloneATM(); } newName = PNHMatchData[ph_name_ix] + ' ATM'; newCategories = insertAtIX(newCategories, 'ATM', 0); // Net result: If the place has ATM cat only and ATM in the name, then it will be green and renamed Bank Name ATM } else if (ixBank > -1 || ixATM > -1) { // if no ATM in name but with a banking category: if ( ixOffices === 0 ) { bannButt.bankBranch = new Flag.BankBranch(); } else if ( ixBank > -1 && ixATM === -1 ) { bannButt.addATM.active = true; } else if ( ixATM === 0 && ixBank === -1 ) { bannButt.bankBranch = new Flag.BankBranch(); bannButt.standaloneATM = new Flag.StandaloneATM(); } else if ( ixBank > 0 && ixATM > 0 ) { bannButt.bankBranch = new Flag.BankBranch(); bannButt.standaloneATM = new Flag.StandaloneATM(); } newName = PNHMatchData[ph_name_ix]; // Net result: If the place has Bank category first, then it will be green with PNH name replaced } else { // for PNH match with neither bank type category, make it a bank newCategories = insertAtIX(newCategories, 'BANK_FINANCIAL', 1); bannButt.standaloneATM = new Flag.StandaloneATM(); bannButt.bankCorporate = new Flag.BankCorporate(); }// END PNH bank treatment } else if ( ['GAS_STATION'].indexOf(priPNHPlaceCat) > -1 ) { // for PNH gas stations, don't replace existing sub-categories if ( altCategories !== '0' && altCategories !== '' ) { // if PNH alts exist insertAtIX(newCategories, altCategories, 1); // then insert the alts into the existing category array after the GS category } if ( newCategories.indexOf('GAS_STATION') !== 0 ) { // If no GS category in the primary, flag it bannButt.gasMkPrim = new Flag.GasMkPrim(); lockOK = false; } else { newName = PNHMatchData[ph_name_ix]; } } else if (updatePNHName) { // if not a special category then update the name newName = PNHMatchData[ph_name_ix]; newCategories = insertAtIX(newCategories, priPNHPlaceCat,0); if (altCategories !== '0' && altCategories !== '' && specCases.indexOf('buttOn_addCat2') === -1 && specCases.indexOf('optionCat2') === -1) { newCategories = insertAtIX(newCategories,altCategories,1); } } else if (!updatePNHName) { // Strong title case option for non-PNH places var titleCaseName = toTitleCaseStrong(newName); if (newName !== titleCaseName) { bannButt.STC = new Flag.STC(); bannButt.STC.suffixMessage = '<span style="margin-left: 4px;font-size: 14px">• ' + titleCaseName + (newNameSuffix || '') + '</span>'; bannButt.STC.title += titleCaseName; bannButt.STC.originalName = newName + (newNameSuffix || ''); } } // *** need to add a section above to allow other permissible categories to remain? (optional) // Parse URL data var localURLcheckRE; if ( localURLcheck !== '') { if (newURL !== null || newURL !== '') { localURLcheckRE = new RegExp(localURLcheck, 'i'); if ( newURL.match(localURLcheckRE) !== null ) { newURL = normalizeURL(newURL,false, true, item, region); } else { newURL = normalizeURL(PNHMatchData[ph_url_ix],false, true, item, region); bannButt.localURL = new Flag.LocalURL(); } } else { newURL = normalizeURL(PNHMatchData[ph_url_ix],false, true, item, region); bannButt.localURL = new Flag.LocalURL(); } } else { newURL = normalizeURL(PNHMatchData[ph_url_ix],false, true, item, region); } // Parse PNH Aliases newAliasesTemp = PNHMatchData[ph_aliases_ix].match(/([^\(]*)/i)[0]; if (newAliasesTemp !== '0' && newAliasesTemp !== '') { // make aliases array newAliasesTemp = newAliasesTemp.replace(/,[^A-za-z0-9]*/g, ','); // tighten up commas if more than one alias. newAliasesTemp = newAliasesTemp.split(','); // split by comma } if ( specCases.indexOf('noUpdateAlias') === -1 && (!containsAll(newAliases,newAliasesTemp) && newAliasesTemp !== '0' && newAliasesTemp !== '' && specCases.indexOf('optionName2') === -1 )) { newAliases = insertAtIX(newAliases,newAliasesTemp,0); } // Enable optional alt-name button if (bannButt.addAlias) { bannButt.addAlias.message = 'Is there a ' + optionalAlias + ' at this location?'; bannButt.addAlias.title = 'Add ' + optionalAlias; } // Remove unnecessary parent categories let catData = _PNH_DATA.USA.categories.map(cat => cat.split('|')); let catParentIdx = catData[0].indexOf('pc_catparent'); let catNameIdx = catData[0].indexOf('pc_wmecat'); let parentCats = _.uniq(newCategories.map(catName => catData.find(cat => cat[catNameIdx] === catName)[catParentIdx])).filter(parent => parent.trim(' ').length > 0); newCategories = newCategories.filter(cat => parentCats.findIndex(parentCat => cat === parentCat) === -1); // update categories if different and no Cat2 option if ( !matchSets( _.uniq(item.attributes.categories), _.uniq(newCategories) ) ) { if ( specCases.indexOf('optionCat2') === -1 && specCases.indexOf('buttOn_addCat2') === -1 ) { phlogdev('Categories updated with ' + newCategories); actions.push(new UpdateObject(item, { categories: newCategories })); //W.model.actionManager.add(new UpdateObject(item, { categories: newCategories })); _updatedFields.categories.updated = true; } else { // if second cat is optional phlogdev('Primary category updated with ' + priPNHPlaceCat); newCategories = insertAtIX(newCategories, priPNHPlaceCat, 0); actions.push(new UpdateObject(item, { categories: newCategories })); _updatedFields.categories.updated = true; } } // Enable optional 2nd category button if (specCases.indexOf('buttOn_addCat2') > -1 && newCategories.indexOf(catTransWaze2Lang[altCategories[0]]) === -1 ) { let altCat = altCategories[0]; bannButt.addCat2.message = 'Is there a ' + catTransWaze2Lang[altCat] + ' at this location?'; bannButt.addCat2.title = 'Add ' + catTransWaze2Lang[altCat]; bannButt.addCat2.altCategory = altCat; } // Description update newDescripion = PNHMatchData[ph_description_ix]; if (newDescripion !== null && newDescripion !== '0' && item.attributes.description.toUpperCase().indexOf(newDescripion.toUpperCase()) === -1 ) { if ( item.attributes.description !== '' && item.attributes.description !== null && item.attributes.description !== ' ' ) { bannButt.checkDescription = new Flag.CheckDescription(); } phlogdev('Description updated'); newDescripion = newDescripion + '\n' + item.attributes.description; actions.push(new UpdateObject(item, { description: newDescripion })); _updatedFields.description.updated = true; } // Special Lock by PNH if (specCases.indexOf('lockAt5') > -1 ) { PNHLockLevel = 4; } } else { // if no PNH match found if (PNHMatchData[0] === 'ApprovalNeeded') { //PNHNameTemp = PNHMatchData[1].join(', '); PNHNameTemp = PNHMatchData[1][0]; // Just do the first match PNHNameTempWeb = encodeURIComponent(PNHNameTemp); PNHOrderNum = PNHMatchData[2].join(','); } // Strong title case option for non-PNH places var titleCaseName = toTitleCaseStrong(newName); if (newName !== titleCaseName) { bannButt.STC = new Flag.STC(); bannButt.STC.suffixMessage = '<span style="margin-left: 4px;font-size: 14px">• ' + titleCaseName + (newNameSuffix || '') + '</span>'; bannButt.STC.title += titleCaseName; bannButt.STC.originalName = newName + (newNameSuffix || ''); } newURL = normalizeURL(newURL,true,false, item, region); // Normalize url // Generic Bank treatment ixBank = item.attributes.categories.indexOf('BANK_FINANCIAL'); ixATM = item.attributes.categories.indexOf('ATM'); ixOffices = item.attributes.categories.indexOf('OFFICES'); // if the name contains ATM in it if ( newName.match(/\batm\b/ig) !== null ) { if ( ixOffices === 0 ) { bannButt.bankType1 = new Flag.BankType1(); bannButt.bankBranch = new Flag.BankBranch(); bannButt.standaloneATM = new Flag.StandaloneATM(); bannButt.bankCorporate = new Flag.BankCorporate(); } else if ( ixBank === -1 && ixATM === -1 ) { bannButt.bankBranch = new Flag.BankBranch(); bannButt.standaloneATM = new Flag.StandaloneATM(); } else if ( ixATM === 0 && ixBank > 0 ) { bannButt.bankBranch = new Flag.BankBranch(); } else if ( ixBank > -1 ) { bannButt.bankBranch = new Flag.BankBranch(); bannButt.standaloneATM = new Flag.StandaloneATM(); } // Net result: If the place has ATM cat only and ATM in the name, then it will be green } else if (ixBank > -1 || ixATM > -1) { // if no ATM in name: if ( ixOffices === 0 ) { bannButt.bankBranch = new Flag.BankBranch(); } else if ( ixBank > -1 && ixATM === -1 ) { bannButt.addATM.active = true; } else if ( ixATM === 0 && ixBank === -1 ) { bannButt.bankBranch = new Flag.BankBranch(); bannButt.standaloneATM = new Flag.StandaloneATM(); } else if ( ixBank > 0 && ixATM > 0 ) { bannButt.bankBranch = new Flag.BankBranch(); bannButt.standaloneATM = new Flag.StandaloneATM(); } // Net result: If the place has Bank category first, then it will be green } // END generic bank treatment } // END PNH match/no-match updates // Category/Name-based Services, added to any existing services: var CH_DATA = _PNH_DATA[_countryCode].categories; var CH_NAMES = _PNH_DATA[_countryCode].categoryNames; var CH_DATA_headers = CH_DATA[0].split('|'); var CH_DATA_keys = CH_DATA[1].split('|'); var CH_DATA_list = CH_DATA[2].split('|'); var CH_DATA_Temp; if (hpMode.harmFlag) { // Update name: if ((newName + (newNameSuffix ? newNameSuffix : '')) !== item.attributes.name) { phlogdev('Name updated'); actions.push(new UpdateObject(item, { name: newName + (newNameSuffix ? newNameSuffix : '') })); //actions.push(new UpdateObject(item, { name: newName })); _updatedFields.name.updated = true; } // Update aliases newAliases = removeSFAliases(newName, newAliases); if (newAliases !== item.attributes.aliases && newAliases.length !== item.attributes.aliases.length) { phlogdev('Alt Names updated'); actions.push(new UpdateObject(item, { aliases: newAliases })); _updatedFields.aliases.updated = true; } // Make PNH submission links var regionFormURL = ''; var newPlaceAddon = ''; var approvalAddon = ''; var approvalMessage = 'Submitted via WMEPH. PNH order number ' + PNHOrderNum; var tempSubmitName_encoded = encodeURIComponent(newName); var placePL_encoded = encodeURIComponent(placePL); var newURLSubmit_encoded = encodeURIComponent(newURLSubmit); let suffix = _USER.name + gFormState; switch (region) { case 'NWR': regionFormURL = 'https://docs.google.com/forms/d/1hv5hXBlGr1pTMmo4n3frUx1DovUODbZodfDBwwTc7HE/viewform'; newPlaceAddon = '?entry.925969794='+tempSubmitName_encoded+'&entry.1970139752='+newURLSubmit_encoded+'&entry.1749047694='+suffix; approvalAddon = '?entry.925969794='+PNHNameTempWeb+'&entry.50214576='+approvalMessage+'&entry.1749047694='+suffix; break; case 'SWR': regionFormURL = 'https://docs.google.com/forms/d/1Qf2N4fSkNzhVuXJwPBJMQBmW0suNuy8W9itCo1qgJL4/viewform'; newPlaceAddon = '?entry.1497446659='+tempSubmitName_encoded+'&entry.1970139752='+newURLSubmit_encoded+'&entry.1749047694='+suffix; approvalAddon = '?entry.1497446659='+PNHNameTempWeb+'&entry.50214576='+approvalMessage+'&entry.1749047694='+suffix; break; case 'HI': regionFormURL = 'https://docs.google.com/forms/d/1K7Dohm8eamIKry3KwMTVnpMdJLaMIyDGMt7Bw6iqH_A/viewform'; newPlaceAddon = '?entry.1497446659='+tempSubmitName_encoded+'&entry.1970139752='+newURLSubmit_encoded+'&entry.1749047694='+suffix; approvalAddon = '?entry.1497446659='+PNHNameTempWeb+'&entry.50214576='+approvalMessage+'&entry.1749047694='+suffix; break; case 'PLN': regionFormURL = 'https://docs.google.com/forms/d/1ycXtAppoR5eEydFBwnghhu1hkHq26uabjUu8yAlIQuI/viewform'; newPlaceAddon = '?entry.925969794='+tempSubmitName_encoded+'&entry.1970139752='+newURLSubmit_encoded+'&entry.1749047694='+suffix; approvalAddon = '?entry.925969794='+PNHNameTempWeb+'&entry.50214576='+approvalMessage+'&entry.1749047694='+suffix; break; case 'SCR': regionFormURL = 'https://docs.google.com/forms/d/1KZzLdlX0HLxED5Bv0wFB-rWccxUp2Mclih5QJIQFKSQ/viewform'; newPlaceAddon = '?entry.925969794='+tempSubmitName_encoded+'&entry.1970139752='+newURLSubmit_encoded+'&entry.1749047694='+suffix; approvalAddon = '?entry.925969794='+PNHNameTempWeb+'&entry.50214576='+approvalMessage+'&entry.1749047694='+suffix; break; case 'TX': regionFormURL = 'https://docs.google.com/forms/d/1x7VM7ofPOKVnWOaX7d70OWXpnVKf6Mkadn4dgYxx4ic/viewform'; newPlaceAddon = '?entry.925969794='+tempSubmitName_encoded+'&entry.1970139752='+newURLSubmit_encoded+'&entry.1749047694='+suffix; approvalAddon = '?entry.925969794='+PNHNameTempWeb+'&entry.50214576='+approvalMessage+'&entry.1749047694='+suffix; break; case 'GLR': regionFormURL = 'https://docs.google.com/forms/d/19btj-Qt2-_TCRlcS49fl6AeUT95Wnmu7Um53qzjj9BA/viewform'; newPlaceAddon = '?entry.925969794='+tempSubmitName_encoded+'&entry.1970139752='+newURLSubmit_encoded+'&entry.1749047694='+suffix; approvalAddon = '?entry.925969794='+PNHNameTempWeb+'&entry.50214576='+approvalMessage+'&entry.1749047694='+suffix; break; case 'SAT': regionFormURL = 'https://docs.google.com/forms/d/1bxgK_20Jix2ahbmUvY1qcY0-RmzUBT6KbE5kjDEObF8/viewform'; newPlaceAddon = '?entry.2063110249='+tempSubmitName_encoded+'&entry.2018912633='+newURLSubmit_encoded+'&entry.1924826395='+suffix; approvalAddon = '?entry.2063110249='+PNHNameTempWeb+'&entry.123778794='+approvalMessage+'&entry.1924826395='+suffix; break; case 'SER': regionFormURL = 'https://docs.google.com/forms/d/1jYBcxT3jycrkttK5BxhvPXR240KUHnoFMtkZAXzPg34/viewform'; newPlaceAddon = '?entry.822075961='+tempSubmitName_encoded+'&entry.1422079728='+newURLSubmit_encoded+'&entry.1891389966='+suffix; approvalAddon = '?entry.822075961='+PNHNameTempWeb+'&entry.607048307='+approvalMessage+'&entry.1891389966='+suffix; break; case 'ATR': regionFormURL = 'https://docs.google.com/forms/d/1v7JhffTfr62aPSOp8qZHA_5ARkBPldWWJwDeDzEioR0/viewform'; newPlaceAddon = '?entry.925969794='+tempSubmitName_encoded+'&entry.1970139752='+newURLSubmit_encoded+'&entry.1749047694='+suffix; approvalAddon = '?entry.925969794='+PNHNameTempWeb+'&entry.50214576='+approvalMessage+'&entry.1749047694='+suffix; break; case 'NER': regionFormURL = 'https://docs.google.com/forms/d/1UgFAMdSQuJAySHR0D86frvphp81l7qhEdJXZpyBZU6c/viewform'; newPlaceAddon = '?entry.925969794='+tempSubmitName_encoded+'&entry.1970139752='+newURLSubmit_encoded+'&entry.1749047694='+suffix; approvalAddon = '?entry.925969794='+PNHNameTempWeb+'&entry.50214576='+approvalMessage+'&entry.1749047694='+suffix; break; case 'NOR': regionFormURL = 'https://docs.google.com/forms/d/1iYq2rd9HRd-RBsKqmbHDIEBGuyWBSyrIHC6QLESfm4c/viewform'; newPlaceAddon = '?entry.925969794='+tempSubmitName_encoded+'&entry.1970139752='+newURLSubmit_encoded+'&entry.1749047694='+suffix; approvalAddon = '?entry.925969794='+PNHNameTempWeb+'&entry.50214576='+approvalMessage+'&entry.1749047694='+suffix; break; case 'MAR': regionFormURL = 'https://docs.google.com/forms/d/1PhL1iaugbRMc3W-yGdqESoooeOz-TJIbjdLBRScJYOk/viewform'; newPlaceAddon = '?entry.925969794='+tempSubmitName_encoded+'&entry.1970139752='+newURLSubmit_encoded+'&entry.1749047694='+suffix; approvalAddon = '?entry.925969794='+PNHNameTempWeb+'&entry.50214576='+approvalMessage+'&entry.1749047694='+suffix; break; case 'CA_EN': regionFormURL = 'https://docs.google.com/forms/d/13JwXsrWPNmCdfGR5OVr5jnGZw-uNGohwgjim-JYbSws/viewform'; newPlaceAddon = '?entry_839085807='+tempSubmitName_encoded+'&entry_1067461077='+newURLSubmit_encoded+'&entry_318793106='+_USER.name+'&entry_1149649663='+placePL_encoded; approvalAddon = '?entry_839085807='+PNHNameTempWeb+'&entry_1125435193='+approvalMessage+'&entry_318793106='+_USER.name+'&entry_1149649663='+placePL_encoded; break; case 'QC': regionFormURL = 'https://docs.google.com/forms/d/13JwXsrWPNmCdfGR5OVr5jnGZw-uNGohwgjim-JYbSws/viewform'; newPlaceAddon = '?entry_839085807='+tempSubmitName_encoded+'&entry_1067461077='+newURLSubmit_encoded+'&entry_318793106='+_USER.name+'&entry_1149649663='+placePL_encoded; approvalAddon = '?entry_839085807='+PNHNameTempWeb+'&entry_1125435193='+approvalMessage+'&entry_318793106='+_USER.name+'&entry_1149649663='+placePL_encoded; break; default: regionFormURL = ''; } newPlaceURL = regionFormURL + newPlaceAddon; approveRegionURL = regionFormURL + approvalAddon; // PNH specific Services: var servHeaders = [], servKeys = [], servList = [], servHeaderCheck; for (var jjj=0; jjj<CH_DATA_headers.length; jjj++) { servHeaderCheck = CH_DATA_headers[jjj].match(/^ps_/i); // if it's a service header if (servHeaderCheck) { servHeaders.push(jjj); servKeys.push(CH_DATA_keys[jjj]); servList.push(CH_DATA_list[jjj]); } } if (newCategories.length > 0) { for (var iii=0; iii<CH_NAMES.length; iii++) { if (newCategories.indexOf(CH_NAMES[iii]) > -1 ) { CH_DATA_Temp = CH_DATA[iii].split('|'); for (var psix=0; psix<servHeaders.length; psix++) { if ( !bannServ[servKeys[psix]].pnhOverride ) { if (CH_DATA_Temp[servHeaders[psix]] === '1') { // These are automatically added to all countries/regions (if auto setting is on) bannServ[servKeys[psix]].active = true; if ($('#WMEPH-EnableServices').prop('checked')) { // Automatically enable new services bannServ[servKeys[psix]].actionOn(actions); } } else if (CH_DATA_Temp[servHeaders[psix]] === '2') { // these are never automatically added but shown bannServ[servKeys[psix]].active = true; } else if (CH_DATA_Temp[servHeaders[psix]] !== '') { // check for state/region auto add bannServ[servKeys[psix]].active = true; if ($('#WMEPH-EnableServices').prop('checked')) { var servAutoRegion = CH_DATA_Temp[servHeaders[psix]].replace(/,[^A-za-z0-9]*/g, ',').split(','); // if the sheet data matches the state, region, or username then auto add if ( servAutoRegion.indexOf(state2L) > -1 || servAutoRegion.indexOf(region) > -1 || servAutoRegion.indexOf(_USER.name) > -1 ) { bannServ[servKeys[psix]].actionOn(actions); } } } } } } } } } var isPoint = item.isPoint(); // NOTE: do not use is2D() function. It doesn't seem to be 100% reliable. var isArea = !isPoint; var maxPointSeverity = 0; var maxAreaSeverity = 3; var highestCategoryLock = -1; for(var ixPlaceCat=0; ixPlaceCat<newCategories.length; ixPlaceCat++) { var category = newCategories[ixPlaceCat]; var ixPNHCat = CH_NAMES.indexOf(category); if (ixPNHCat>-1) { CH_DATA_Temp = CH_DATA[ixPNHCat].split('|'); // CH_DATA_headers //pc_point pc_area pc_regpoint pc_regarea pc_lock1 pc_lock2 pc_lock3 pc_lock4 pc_lock5 pc_rare pc_parent pc_message var pvaPoint = CH_DATA_Temp[CH_DATA_headers.indexOf('pc_point')]; var pvaArea = CH_DATA_Temp[CH_DATA_headers.indexOf('pc_area')]; var regPoint = CH_DATA_Temp[CH_DATA_headers.indexOf('pc_regpoint')].replace(/,[^A-za-z0-9]*/g, ',').split(','); var regArea = CH_DATA_Temp[CH_DATA_headers.indexOf('pc_regarea')].replace(/,[^A-za-z0-9]*/g, ',').split(','); if (regPoint.indexOf(state2L) > -1 || regPoint.indexOf(region) > -1 || regPoint.indexOf(_countryCode) > -1) { pvaPoint = '1'; pvaArea = ''; } else if (regArea.indexOf(state2L) > -1 || regArea.indexOf(region) > -1 || regArea.indexOf(_countryCode) > -1) { pvaPoint = ''; pvaArea = '1'; } // If Post Office and VPO or CPU is in the name, always a point. if (newCategories.indexOf('POST_OFFICE') > -1 && /\b(?:cpu|vpo)\b/i.test(item.attributes.name)) { pvaPoint = '1'; pvaArea=''; } var pointSeverity = getPvaSeverity(pvaPoint, item); var areaSeverity = getPvaSeverity(pvaArea, item); if (isPoint && pointSeverity > 0) { maxPointSeverity = Math.max(pointSeverity, maxPointSeverity); } else if (isArea) { maxAreaSeverity = Math.min(areaSeverity, maxAreaSeverity); } // display any messaged regarding the category let pc_message = ''; if (newCategories.indexOf('HOSPITAL_MEDICAL_CARE') === -1) { pc_message = CH_DATA_Temp[CH_DATA_headers.indexOf('pc_message')]; } if (pc_message && pc_message !== '0' && pc_message !== '') { bannButt.pnhCatMess = new Flag.PnhCatMess(pc_message); } // Unmapped categories let pc_rare = CH_DATA_Temp[CH_DATA_headers.indexOf('pc_rare')].replace(/,[^A-Za-z0-9}]+/g, ',').split(','); if (pc_rare.indexOf(state2L) > -1 || pc_rare.indexOf(region) > -1 || pc_rare.indexOf(_countryCode) > -1) { if (CH_DATA_Temp[0] === 'OTHER' && ['GLR','NER','NWR','PLN','SCR','SER','NOR','HI','SAT'].indexOf(region) > -1) { if (!isLocked) { bannButt.unmappedRegion = new Flag.UnmappedRegion(); bannButt.unmappedRegion.WLactive = false; bannButt.unmappedRegion.severity = 1; bannButt.unmappedRegion.message = 'The "Other" category should only be used if no other category applies. Manually lock the place to override this flag.'; lockOK = false; } } else { bannButt.unmappedRegion = new Flag.UnmappedRegion(); if (_wl.unmappedRegion) { bannButt.unmappedRegion.WLactive = false; bannButt.unmappedRegion.severity = 0; } else { lockOK = false; } } } // Parent Category let pc_parent = CH_DATA_Temp[CH_DATA_headers.indexOf('pc_parent')].replace(/,[^A-Za-z0-9}]+/g, ',').split(','); if (pc_parent.indexOf(state2L) > -1 || pc_parent.indexOf(region) > -1 || pc_parent.indexOf(_countryCode) > -1) { bannButt.parentCategory = new Flag.ParentCategory(); if (_wl.parentCategory) { bannButt.parentCategory.WLactive = false; } } // Set lock level for (var lockix=1; lockix<6; lockix++) { let pc_lockTemp = CH_DATA_Temp[CH_DATA_headers.indexOf('pc_lock'+lockix)].replace(/,[^A-Za-z0-9}]+/g, ',').split(','); if (lockix - 1 > highestCategoryLock && (pc_lockTemp.indexOf(state2L) > -1 || pc_lockTemp.indexOf(region) > -1 || pc_lockTemp.indexOf(_countryCode) > -1)) { highestCategoryLock = lockix - 1; // Offset by 1 since lock ranks start at 0 } } } } if (highestCategoryLock > -1) { defaultLockLevel = highestCategoryLock; } if (isPoint) { if (maxPointSeverity === 3) { bannButt.areaNotPoint = new Flag.AreaNotPoint(); if (_wl.areaNotPoint || item.attributes.lockRank >= defaultLockLevel) { bannButt.areaNotPoint.WLactive = false; bannButt.areaNotPoint.severity = 0; } else { lockOK = false; } } else if (maxPointSeverity === 2) { bannButt.areaNotPointMid = new Flag.AreaNotPointMid(); if (_wl.areaNotPoint || item.attributes.lockRank >= defaultLockLevel) { bannButt.areaNotPointMid.WLactive = false; bannButt.areaNotPointMid.severity = 0; } else { lockOK = false; } } else if (maxPointSeverity === 1) { bannButt.areaNotPointLow = new Flag.AreaNotPointLow(); if (_wl.areaNotPoint || item.attributes.lockRank >= defaultLockLevel) { bannButt.areaNotPointLow.WLactive = false; bannButt.areaNotPointLow.severity = 0; } } } else { if (maxAreaSeverity === 3) { bannButt.pointNotArea = new Flag.PointNotArea(); if (_wl.pointNotArea || item.attributes.lockRank >= defaultLockLevel) { bannButt.pointNotArea.WLactive = false; bannButt.pointNotArea.severity = 0; } else { lockOK = false; } } else if (maxAreaSeverity === 2) { bannButt.pointNotAreaMid = new Flag.PointNotAreaMid(); if (_wl.pointNotArea || item.attributes.lockRank >= defaultLockLevel) { bannButt.pointNotAreaMid.WLactive = false; bannButt.pointNotAreaMid.severity = 0; } else { lockOK = false; } } else if (maxAreaSeverity === 1) { bannButt.pointNotAreaLow = new Flag.PointNotAreaLow(); if (_wl.pointNotArea || item.attributes.lockRank >= defaultLockLevel) { bannButt.pointNotAreaLow.WLactive = false; bannButt.pointNotAreaLow.severity = 0; } } } var anpNone = collegeAbbreviations.split('|'), anpNoneRE; for (var cii=0; cii<anpNone.length; cii++) { anpNoneRE = new RegExp('\\b'+anpNone[cii]+'\\b', 'g'); if ( newName.match( anpNoneRE) !== null && bannButt.areaNotPointLow ) { bannButt.areaNotPointLow.severity = 0; bannButt.areaNotPointLow.WLactive = false; } } // Check for missing hours field if (item.attributes.openingHours.length === 0) { // if no hours... if (!containsAny(newCategories,['STADIUM_ARENA','CEMETERY','TRANSPORTATION','FERRY_PIER','SUBWAY_STATION', 'BRIDGE','TUNNEL','JUNCTION_INTERCHANGE','ISLAND','SEA_LAKE_POOL','RIVER_STREAM','FOREST_GROVE','CANAL','SWAMP_MARSH','DAM']) ) { bannButt.noHours = new Flag.NoHours(); if (_wl.noHours || $('#WMEPH-DisableHoursHL').prop('checked') || containsAny(newCategories,['SCHOOL','CONVENTIONS_EVENT_CENTER','CAMPING_TRAILER_PARK','COTTAGE_CABIN','COLLEGE_UNIVERSITY','GOLF_COURSE','SPORTS_COURT','MOVIE_THEATER','SHOPPING_CENTER','RELIGIOUS_CENTER','PARKING_LOT','PARK','PLAYGROUND','AIRPORT','FIRE_DEPARTMENT','POLICE_STATION','SEAPORT_MARINA_HARBOR','FARM'])) { bannButt.noHours.WLactive = false; bannButt.noHours.severity = 0; } } } else { if (item.attributes.openingHours.length === 1) { // if one set of hours exist, check for partial 24hrs setting var hoursEntry = item.attributes.openingHours[0]; if (hoursEntry.days.length < 7 && /^0?0:00$/.test(hoursEntry.fromHour) && (/^0?0:00$/.test(hoursEntry.toHour) || hoursEntry.toHour==='23:59' ) ) { bannButt.mismatch247 = new Flag.Mismatch247(); } } bannButt.noHours = new Flag.NoHours(); bannButt.noHours.severity = 0; bannButt.noHours.WLactive = false; bannButt.noHours.message = getHoursHtml('Hours'); } if ( !checkHours(item.attributes.openingHours) ) { bannButt.hoursOverlap = new Flag.HoursOverlap(); bannButt.noHours = new Flag.NoHours(); } else { var tempHours = item.attributes.openingHours.slice(0); for ( var ohix=0; ohix<item.attributes.openingHours.length; ohix++ ) { if ( tempHours[ohix].days.length === 2 && tempHours[ohix].days[0] === 1 && tempHours[ohix].days[1] === 0) { // separate hours phlogdev('Correcting M-S entry...'); tempHours.push({days: [0], fromHour: tempHours[ohix].fromHour, toHour: tempHours[ohix].toHour}); tempHours[ohix].days = [1]; actions.push(new UpdateObject(item, { openingHours: tempHours })); } } } if (hpMode.harmFlag) { // Highlight 24/7 button if hours are set that way, and add button for all places if ( item.attributes.openingHours.length === 1 && item.attributes.openingHours[0].days.length === 7 && item.attributes.openingHours[0].fromHour === '00:00' && item.attributes.openingHours[0].toHour ==='00:00' ) { bannServ.add247.checked = true; } bannServ.add247.active = true; } // URL updating updateURL = true; if (newURL !== item.attributes.url && newURL !== '' && newURL !== '0') { if ( PNHNameRegMatch && item.attributes.url !== null && item.attributes.url !== '' && newURL !== 'badURL') { // for cases where there is an existing URL in the WME place, and there is a PNH url on queue: var newURLTemp = normalizeURL(newURL,true,false, item); // normalize var itemURL = normalizeURL(item.attributes.url,true,false, item); newURLTemp = newURLTemp.replace(/^www\.(.*)$/i,'$1'); // strip www var itemURLTemp = itemURL.replace(/^www\.(.*)$/i,'$1'); // strip www if ( newURLTemp !== itemURLTemp ) { // if formatted URLs don't match, then alert the editor to check the existing URL bannButt.longURL = new Flag.LongURL(); if (_wl.longURL) { bannButt.longURL.severity = 0; bannButt.longURL.WLactive = false; } //bannButt.PlaceWebsite.value = 'Place Website'; if (hpMode.harmFlag && updateURL && itemURL !== item.attributes.url) { // Update the URL phlogdev('URL formatted'); actions.push(new UpdateObject(item, { url: itemURL })); _updatedFields.url.updated = true; } updateURL = false; tempPNHURL = newURL; } } if (hpMode.harmFlag && updateURL && newURL !== 'badURL' && newURL !== item.attributes.url) { // Update the URL phlogdev('URL updated'); actions.push(new UpdateObject(item, { url: newURL })); _updatedFields.url.updated = true; } } // Phone formatting var outputFormat = '({0}) {1}-{2}'; if ( containsAny(['CA','CO'],[region,state2L]) && (/^\d{3}-\d{3}-\d{4}$/.test(item.attributes.phone))) { outputFormat = '{0}-{1}-{2}'; } else if (region === 'SER' && !(/^\(\d{3}\) \d{3}-\d{4}$/.test(item.attributes.phone))) { outputFormat = '{0}-{1}-{2}'; } else if (region === 'GLR') { outputFormat = '{0}-{1}-{2}'; } else if (state2L === 'NV') { outputFormat = '{0}-{1}-{2}'; } else if (_countryCode === 'CAN') { outputFormat = '+1-{0}-{1}-{2}'; } newPhone = normalizePhone(item.attributes.phone, outputFormat, 'existing', item, region); // Check if valid area code #LOC# USA and CAN only if (!_wl.aCodeWL && (_countryCode === 'USA' || _countryCode === 'CAN')) { if (newPhone !== null && newPhone.match(/[2-9]\d{2}/) !== null) { var areaCode = newPhone.match(/[2-9]\d{2}/)[0]; if ( areaCodeList.indexOf(areaCode) === -1 ) { bannButt.badAreaCode = new Flag.BadAreaCode(newPhone, outputFormat); } } } if (hpMode.harmFlag && newPhone !== item.attributes.phone) { phlogdev('Phone updated'); actions.push(new UpdateObject(item, {phone: newPhone})); _updatedFields.phone.updated = true; } // Post Office check if (_countryCode === 'USA' && newCategories.indexOf('PARKING_LOT') === -1) { if (newCategories.indexOf('POST_OFFICE') === -1) { bannButt.isThisAPostOffice = Flag.IsThisAPostOffice.eval(item, newName).flag; } else { var re; if (hpMode.harmFlag) { customStoreFinderURL = 'https://tools.usps.com/go/POLocatorAction.action'; customStoreFinder = true; bannButt.PlaceWebsite = new Flag.PlaceWebsite(); bannButt.NewPlaceSubmit = null; if (item.attributes.url !== 'usps.com') { actions.push(new UpdateObject(item, { url: 'usps.com' })); _updatedFields.url.updated = true; bannButt.urlMissing = null; } } if (state2L === 'KY' || (state2L === 'NY' && addr.city && ['Queens','Bronx','Manhattan','Brooklyn','Staten Island'].indexOf(addr.city.attributes.name) > -1)) { re = /^post office \d{5}( [-–](?: cpu| vpo)?(?: [a-z]+){1,})?$/i; } else { re = /^post office [-–](?: cpu| vpo)?(?: [a-z]+){1,}$/i; } newName = newName.trimLeft().replace(/ {2,}/, ' '); if (newNameSuffix) { newNameSuffix = newNameSuffix.trimRight().replace(/\bvpo\b/i, 'VPO').replace(/\bcpu\b/i, 'CPU').replace(/ {2,}/, ' '); } var nameToCheck = newName + (newNameSuffix || ''); if (!re.test(nameToCheck)) { bannButt.formatUSPS = new Flag.FormatUSPS(); lockOK = false; } else { if (hpMode.harmFlag) { if (nameToCheck !== item.attributes.name) { actions.push(new UpdateObject(item, {name: nameToCheck})); } bannButt.catPostOffice = new Flag.CatPostOffice(); } } if (!newAliases.some(alias => alias.toUpperCase() === 'USPS')) { if (hpMode.harmFlag) { newAliases.push('USPS'); actions.push(new UpdateObject(item, {aliases: newAliases})); _updatedFields.aliases.updated = true; } else { bannButt.missingUSPSAlt = new Flag.MissingUSPSAlt(); } } if ( !newAliases.some(alias => /\d{5}/.test(alias)) ) { bannButt.missingUSPSZipAlt = new Flag.MissingUSPSZipAlt(); if (_wl.missingUSPSZipAlt) { bannButt.missingUSPSZipAlt.severity = 0; bannButt.missingUSPSZipAlt.WLactive = false; } // If the zip code appears in the primary name, pre-fill it in the text entry box. var zipMatch = newName.match(/\d{5}/); if (zipMatch) { bannButt.missingUSPSZipAlt.suggestedValue = zipMatch; } // Note: Started work on a Google api lookup to get the zip, but decided it's probably // not worth it since it would need to be verified by the user anyway. //var coords = item.geometry.getCentroid().transform(W.map.getProjection(), W.map.displayProjection); //var url = 'https://maps.googleapis.com/maps/api/geocode/json?latlng=' + coords.y + ',' + coords.x; } var descr = item.attributes.description; var lines = descr.split('\n'); if (lines.length < 1 || !/^.{2,}, [A-Z]{2}\s{1,2}\d{5}$/.test(lines[0])) { bannButt.missingUSPSDescription = new Flag.MissingUSPSDescription(); if (_wl.missingUSPSDescription) { bannButt.missingUSPSDescription.severity = 0; bannButt.missingUSPSDescription.WLactive = false; } } } } // END Post Office check } // END if (!residential && has name) //For gas stations, check to make sure brand exists somewhere in the place name. Remove non-alphanumeric characters first, for more relaxed matching. if (newCategories[0] === 'GAS_STATION' && item.attributes.brand) { var brand = item.attributes.brand; // If brand is going to be forced, use that. Otherwise, use existing brand. if (PNHMatchData && PNHMatchData[ph_speccase_ix]) { var re = /forceBrand<>([^,<]+)/i; var match = re.exec(PNHMatchData[ph_speccase_ix]); if (match) { brand = match[1]; } } var compressedName = item.attributes.name.toUpperCase().replace(/[^a-zA-Z0-9]/g,''); var compressedNewName = newName.toUpperCase().replace(/[^a-zA-Z0-9]/g,''); // Some brands may have more than one acceptable name, or the brand listed in WME doesn't match what we want to see in the name. // Ideally, this would be addressed in the PNH spreadsheet somehow, but for now hardcoding is the only option. var compressedBrands = [brand.toUpperCase().replace(/[^a-zA-Z0-9]/g,'')]; if (brand === 'Diamond Gasoline') { compressedBrands.push('DIAMONDOIL'); } else if (brand === 'Murphy USA') { compressedBrands.push('MURPHY'); } if (compressedBrands.every(compressedBrand => compressedName.indexOf(compressedBrand) === -1 && compressedNewName.indexOf(compressedBrand) === -1)) { bannButt.gasMismatch = new Flag.GasMismatch(); if (_wl.gasMismatch) { bannButt.gasMismatch.WLactive = false; } else { lockOK = false; } } } // Name check if ( !item.attributes.residential && ( !newName || newName.replace(/[^A-Za-z0-9]/g,'').length === 0 )) { if (item.isParkingLot()) { // If it's a parking lot and not locked to R3... if (item.attributes.lockRank < 2) { lockOK = false; bannButt.plaNameMissing = new Flag.PlaNameMissing(); } }else if ( ['ISLAND','FOREST_GROVE','SEA_LAKE_POOL','RIVER_STREAM','CANAL'].indexOf(item.attributes.categories[0]) === -1 ) { bannButt.nameMissing = new Flag.NameMissing(); lockOK = false; } } bannButt.plaNameNonStandard = Flag.PlaNameNonStandard.eval(item, _wl).flag; // Public parking lot warning message: if (item.isParkingLot() && item.attributes.categoryAttributes && item.attributes.categoryAttributes.PARKING_LOT && item.attributes.categoryAttributes.PARKING_LOT.parkingType === 'PUBLIC') { bannButt.plaIsPublic = new Flag.PlaIsPublic(); // Add the buttons to the message. [ ['RESTRICTED','Restricted'], ['PRIVATE','Private'] ].forEach(btnInfo => { bannButt.plaIsPublic.message += $('<button>', {class: 'wmeph-pla-lot-type-btn btn btn-default btn-xs wmeph-btn', 'data-lot-type':btnInfo[0]}) .text(btnInfo[1]) .prop('outerHTML'); }); } // House number / HN check var currentHN = item.attributes.houseNumber; // Check to see if there's an action that is currently updating the house number. var updateHnAction = actions && actions.find(action => action.newAttributes && action.newAttributes.houseNumber); if (updateHnAction) currentHN = updateHnAction.newAttributes.houseNumber; // Use the inferred address street if currently no street. var hasStreet = item.attributes.streetID || (inferredAddress && inferredAddress.street); if (hasStreet && (!currentHN || currentHN.replace(/\D/g,'').length === 0)) { if ( 'BRIDGE|ISLAND|FOREST_GROVE|SEA_LAKE_POOL|RIVER_STREAM|CANAL|DAM|TUNNEL|JUNCTION_INTERCHANGE'.split('|').indexOf(item.attributes.categories[0]) === -1 ) { bannButt.hnMissing = new Flag.HnMissing(item); if (state2L === 'PR') { bannButt.hnMissing.severity = 0; } else { if (item.isParkingLot()) { bannButt.hnMissing.WLactive = false; if (item.attributes.lockRank < 2) { lockOK = false; var msgAdd; if (_USER.rank < 3) { msgAdd = 'Request an R3+ lock to confirm no HN.'; } else { msgAdd = 'Lock to R3+ to confirm no HN.'; } bannButt.hnMissing.suffixMessage = msgAdd; bannButt.hnMissing.severity = 1; } else { bannButt.hnMissing.severity = 0; } } else if (_wl.HNWL) { bannButt.hnMissing.severity = 0; bannButt.hnMissing.WLactive = false; } else { lockOK = false; } } } } else if (currentHN) { var hnOK = false, updateHNflag = false; var hnTemp = currentHN.replace(/[^\d]/g, ''); // Digits only var hnTempDash = currentHN.replace(/[^\d-]/g, ''); // Digits and dashes only if ( hnTemp < 1000000 && state2L === 'NY' && addr.city.attributes.name === 'Queens' && hnTempDash.match(/^\d{1,4}-\d{1,4}$/g) !== null ) { updateHNflag = true; hnOK = true; } if (hnTemp === currentHN && hnTemp < 1000000) { // general check that HN is 6 digits or less, & that it is only [0-9] hnOK = true; } if (state2L === 'HI' && hnTempDash.match(/^\d{1,2}-\d{1,4}$/g) !== null) { if (hnTempDash === hnTempDash.match(/^\d{1,2}-\d{1,4}$/g)[0]) { hnOK = true; } } if (!hnOK) { bannButt.hnNonStandard = new Flag.HnNonStandard(); if (_wl.hnNonStandard) { bannButt.hnNonStandard.WLactive = false; bannButt.hnNonStandard.severity = 0; } else { lockOK = false; } } if ( updateHNflag ) { bannButt.hnDashRemoved = new Flag.HnDashRemoved(); if (hpMode.harmFlag) { actions.push(new UpdateObject(item, { houseNumber: hnTemp })); _updatedFields.address.updated = true; } else if (hpMode.hlFlag) { if (item.attributes.residential) { bannButt.hnDashRemoved.severity = 3; } else { bannButt.hnDashRemoved.severity = 1; } } } } if ((!addr.city || addr.city.attributes.isEmpty) && 'BRIDGE|ISLAND|FOREST_GROVE|SEA_LAKE_POOL|RIVER_STREAM|CANAL|DAM|TUNNEL|JUNCTION_INTERCHANGE'.split('|').indexOf(item.attributes.categories[0]) === -1 ) { bannButt.cityMissing = new Flag.CityMissing(); if (item.attributes.residential && hpMode.hlFlag) { bannButt.cityMissing.severity = 1; } lockOK = false; } if (addr.city && (!addr.street || addr.street.isEmpty) && 'BRIDGE|ISLAND|FOREST_GROVE|SEA_LAKE_POOL|RIVER_STREAM|CANAL|DAM|TUNNEL|JUNCTION_INTERCHANGE'.split('|').indexOf(item.attributes.categories[0]) === -1 ) { bannButt.streetMissing = new Flag.StreetMissing(); lockOK = false; } // CATEGORY vs. NAME checks var testName = newName.toLowerCase().replace(/[^a-z]/g,' '); var testNameWords = testName.split(' '); // Hopsital vs. Name filter if ((newCategories.indexOf('HOSPITAL_URGENT_CARE') > -1 || newCategories.indexOf('HOSPITAL_MEDICAL_CARE') > -1) && hospitalPartMatch.length > 0) { var hpmMatch = false; if (containsAny(testNameWords,animalFullMatch)) { bannButt.changeToPetVet = new Flag.ChangeToPetVet(); if (_wl.changeToPetVet) { bannButt.changeToPetVet.WLactive = false; } else { lockOK = false; } bannButt.pnhCatMess = null; } else if (containsAny(testNameWords,hospitalFullMatch)) { bannButt.changeToDoctorClinic = new Flag.ChangeToDoctorClinic(); bannButt.changeToDoctorClinic.message = 'Keywords suggest this location may not be a hospital or urgent care location.'; if (_wl.changeToDoctorClinic) { bannButt.changeToDoctorClinic.WLactive = false; bannButt.changeToDoctorClinic.severity = 0; } else { bannButt.changeToDoctorClinic.WLactive = true; lockOK = false; bannButt.changeToDoctorClinic.severity = 3; } bannButt.pnhCatMess = null; } else { for (var apmix=0; apmix<animalPartMatch.length; apmix++) { if (testName.indexOf(animalPartMatch[apmix]) > -1) { bannButt.changeToPetVet.active = true; if (_wl.changeToPetVet) { bannButt.changeToPetVet.WLactive = false; } else { lockOK = false; } hpmMatch = true; // don't run the human check if animal is found. bannButt.pnhCatMess = null; break; } } if (!hpmMatch) { // don't run the human check if animal is found. for (var hpmix=0; hpmix<hospitalPartMatch.length; hpmix++) { if (testName.indexOf(hospitalPartMatch[hpmix]) > -1) { if (_wl.changeToDoctorClinic && bannButt.changeToDoctorClinic) { bannButt.changeToDoctorClinic.WLactive = false; } else { lockOK = false; } hpmMatch = true; bannButt.pnhCatMess = null; break; } } } if (!hpmMatch && !bannButt.changeToDoctorClinic) { bannButt.changeToDoctorClinic = new Flag.ChangeToDoctorClinic(); } } } // END HOSPITAL/Name check // School vs. Name filter if (newCategories.indexOf('SCHOOL') > -1 && schoolPartMatch.length>0) { if (containsAny(testNameWords,schoolFullMatch)) { bannButt.changeSchool2Offices = new Flag.ChangeSchool2Offices(); if (_wl.changeSchool2Offices) { bannButt.changeSchool2Offices.WLactive = false; } else { lockOK = false; } bannButt.pnhCatMess = null; } else { for (var schix=0; schix<schoolPartMatch.length; schix++) { if (testName.indexOf(schoolPartMatch[schix]) > -1) { bannButt.changeSchool2Offices = new Flag.ChangeSchool2Offices(); if (_wl.changeSchool2Offices) { bannButt.changeSchool2Offices.WLactive = false; } else { lockOK = false; } bannButt.pnhCatMess = null; break; } } } } // END SCHOOL/Name check // Some cats don't need PNH messages and url/phone severities if ( ['BRIDGE','FOREST_GROVE','DAM','TUNNEL','CEMETERY'].indexOf(item.attributes.categories[0]) > -1 ) { bannButt.NewPlaceSubmit = null; if (bannButt.phoneMissing) { bannButt.phoneMissing.severity = 0; bannButt.phoneMissing.WLactive = false; } if (bannButt.urlMissing) { bannButt.urlMissing.severity = 0; bannButt.urlMissing.WLactive = false; } } else if ( ['ISLAND','SEA_LAKE_POOL','RIVER_STREAM','CANAL','JUNCTION_INTERCHANGE'].indexOf(item.attributes.categories[0]) > -1 ) { // Some cats don't need PNH messages and url/phone messages bannButt.NewPlaceSubmit = null; bannButt.phoneMissing = null; bannButt.urlMissing = null; } // Show the Change To Doctor / Clinic button for places with PERSONAL_CARE or OFFICES category if (hpMode.harmFlag && ((newCategories.indexOf('PERSONAL_CARE') > -1 && !PNHNameRegMatch) || newCategories.indexOf('OFFICES') > -1)) { bannButt.changeToDoctorClinic = new Flag.ChangeToDoctorClinic(); bannButt.changeToDoctorClinic.message = 'If this place provides non-emergency medical care: '; bannButt.changeToDoctorClinic.severity = 0; bannButt.changeToDoctorClinic.WLactive = null; } // *** Rest Area parsing // check rest area name against standard formats or if has the right categories var restAreaCatIndex = categories.indexOf('REST_AREAS'); var oldName = item.attributes.name; if ( /rest area/i.test(oldName) || /rest stop/i.test(oldName) || /service plaza/i.test(oldName) || ( restAreaCatIndex > -1 ) ) { if ( restAreaCatIndex > -1 ) { if (categories.indexOf('SCENIC_LOOKOUT_VIEWPOINT') > -1) { if (!_wl.restAreaScenic) bannButt.restAreaScenic = new Flag.RestAreaScenic(); } if (categories.indexOf('TRANSPORTATION') > -1) { bannButt.restAreaNoTransportation = new Flag.RestAreaNoTransportation(); } if ( item.isPoint() ) { // needs to be area bannButt.areaNotPoint = new Flag.AreaNotPoint(); } bannButt.pointNotArea = null; bannButt.unmappedRegion = null; if ( categories.indexOf('GAS_STATION') > -1 ) { bannButt.restAreaGas = new Flag.RestAreaGas(); } if ( oldName.match(/^Rest Area.* \- /) === null ) { bannButt.restAreaName = new Flag.RestAreaName(); if (_wl.restAreaName) { bannButt.restAreaName.WLactive = false; } } else { if (hpMode.harmFlag) { var newSuffix = newNameSuffix.replace(/Mile/i, 'mile'); if (newName + newSuffix !== item.attributes.name) { actions.push(new UpdateObject(item, { name: newName + newSuffix })); _updatedFields.name.updated = true; phlogdev('Lower case "mile"'); } else { // The new name matches the original name, so the only change would have been to capitalize "Mile", which // we don't want. So remove any previous name-change action. Note: this feels like a hack and is probably // a fragile workaround. The name shouldn't be capitalized in the first place, unless necessary. for (var i=0; i<actions.length; i++) { var action = actions[i]; if (action.newAttributes.name) { actions.splice(i,1); _updatedFields.name.updated = false; break; } } } } } // switch to rest area wiki button if (hpMode.harmFlag) { bannButt2.restAreaWiki.active = true; bannButt2.placesWiki.active = false; } // missing address ok bannButt.streetMissing = null; bannButt.cityMissing = null; bannButt.hnMissing = null; if (bannButt.urlMissing) { bannButt.urlMissing.WLactive = false; bannButt.urlMissing.severity = 0; } if (bannButt.phoneMissing) { bannButt.phoneMissing.severity = 0; bannButt.phoneMissing.WLactive = false; } //assembleBanner(); } else { if (!_wl.restAreaSpec) bannButt.restAreaSpec = new Flag.RestAreaSpec(); } } // update Severity for banner messages for (var bannKey in bannButt) { if (bannButt[bannKey] && bannButt[bannKey].active) { severityButt = Math.max(bannButt[bannKey].severity, severityButt); } } if (hpMode.harmFlag) { phlogdev('Severity: '+severityButt+'; lockOK: '+lockOK); } // Place locking // final formatting of desired lock levels var hlLockFlag = false, levelToLock; if (PNHLockLevel !== -1 && hpMode.harmFlag) { phlogdev('PNHLockLevel: '+PNHLockLevel); levelToLock = PNHLockLevel; } else { levelToLock = defaultLockLevel; } if (region === 'SER') { if (newCategories.indexOf('COLLEGE_UNIVERSITY') > -1 && newCategories.indexOf('PARKING_LOT') > -1) { levelToLock = lockLevel4; } else if ( item.isPoint() && newCategories.indexOf('COLLEGE_UNIVERSITY') > -1 && (newCategories.indexOf('HOSPITAL_MEDICAL_CARE') === -1 || newCategories.indexOf('HOSPITAL_URGENT_CARE') === -1) ) { levelToLock = lockLevel4; } } if (levelToLock > (_USER.rank - 1)) {levelToLock = (_USER.rank - 1);} // Only lock up to the user's level // If gas station is missing brand, don't flag if place is locked. if (bannButt.gasNoBrand) { if (item.attributes.lockRank >= levelToLock) { bannButt.gasNoBrand = null; } else { bannButt.gasNoBrand.message = 'Lock to L' + (levelToLock + 1) + '+ to verify no gas brand.'; } } // If no Google link and severity would otherwise allow locking, ask if user wants to lock anyway. if (!isLocked && bannButt.extProviderMissing && bannButt.extProviderMissing.active && bannButt.extProviderMissing.severity <= 2) { bannButt.extProviderMissing.severity = 3; severityButt = 3; if (lockOK) { bannButt.extProviderMissing.value = 'Lock anyway? (' + (levelToLock + 1) + ')'; bannButt.extProviderMissing.title = 'If no Google link exists, lock this place.\nIf there is still no Google link after 6 months from the last update date, it will turn red as a reminder to search again.'; bannButt.extProviderMissing.action = function() { var action = new UpdateObject(item, {'lockRank': levelToLock}); W.model.actionManager.add(action); _updatedFields.lock.updated = true; harmonizePlaceGo(item, 'harmonize'); }; } } if ( lockOK && severityButt < 2) { // Campus project exceptions if ( item.attributes.lockRank < levelToLock) { if (hpMode.harmFlag) { phlogdev('Venue locked!'); actions.push(new UpdateObject(item, { lockRank: levelToLock })); _updatedFields.lock.updated = true; } else if (hpMode.hlFlag) { hlLockFlag = true; } } bannButt.placeLocked = new Flag.PlaceLocked(); } //IGN check if (!item.attributes.residential && item.attributes.updatedBy && W.model.users.getObjectById(item.attributes.updatedBy) && W.model.users.getObjectById(item.attributes.updatedBy).userName && W.model.users.getObjectById(item.attributes.updatedBy).userName.match(/^ign_/i) !== null) { bannButt.ignEdited = new Flag.IgnEdited(); } //waze_maint_bot check var updatedById = item.attributes.updatedBy ? item.attributes.updatedBy : item.attributes.createdBy; var updatedBy = W.model.users.getObjectById(updatedById); var updatedByName = updatedBy ? updatedBy.userName : null; var botNamesAndIDs = [ '^waze-maint', '^105774162$', '^waze3rdparty$', '^361008095$', '^WazeParking1$', '^338475699$', '^admin$', '^-1$', '^avsus$', '^107668852$' ]; var re = new RegExp(botNamesAndIDs.join('|'),'i'); if (item.isUnchanged() && !item.attributes.residential && updatedById && (re.test(updatedById.toString()) || (updatedByName && re.test(updatedByName)))) { bannButt.wazeBot = new Flag.WazeBot(); } // RPP Locking option for R3+ if (item.attributes.residential) { if (_USER.isDevUser || _USER.isBetaUser || _USER.rank >= 3) { // Allow residential point locking by R3+ RPPLockString = 'Lock at <select id="RPPLockLevel">'; var ddlSelected = false; for (var llix=1; llix<6; llix++) { if (llix < _USER.rank+1) { if ( !ddlSelected && (defaultLockLevel === llix - 1 || llix === _USER.rank) ) { RPPLockString += '<option value="'+llix+'" selected="selected">'+llix+'</option>'; ddlSelected = true; } else { RPPLockString += '<option value="'+llix+'">'+llix+'</option>'; } } } RPPLockString += '</select>'; bannButt.lockRPP = new Flag.LockRPP(); bannButt.lockRPP.message = 'Current lock: '+ (parseInt(item.attributes.lockRank)+1) +'. '+RPPLockString+' ?'; } } // Turn off unnecessary buttons if (item.attributes.categories.indexOf('PHARMACY') > -1) { bannButt.addPharm.active = false; } if (item.attributes.categories.indexOf('SUPERMARKET_GROCERY') > -1) { bannButt.addSuper.active = false; } // Final alerts for non-severe locations if ( !item.attributes.residential && severityButt < 3) { var nameShortSpace = newName.toUpperCase().replace(/[^A-Z \']/g, ''); if ( nameShortSpace.indexOf('\'S HOUSE') > -1 || nameShortSpace.indexOf('\'S HOME') > -1 || nameShortSpace.indexOf('\'S WORK') > -1) { if ( !containsAny(newCategories,['RESTAURANT','DESSERT','BAR']) && !PNHNameRegMatch ) { bannButt.resiTypeNameSoft = new Flag.ResiTypeNameSoft(); } } if ( ['HOME','MY HOME','HOUSE','MY HOUSE','PARENTS HOUSE','CASA','MI CASA','WORK','MY WORK','MY OFFICE','MOMS HOUSE','DADS HOUSE','MOM','DAD'].indexOf( nameShortSpace ) > -1 ) { bannButt.resiTypeName = new Flag.ResiTypeName(); if (_wl.resiTypeName) { bannButt.resiTypeName.WLactive = false; } bannButt.resiTypeNameSoft = null; } if ( item.attributes.description.toLowerCase().indexOf('google') > -1 || item.attributes.description.toLowerCase().indexOf('yelp') > -1 ) { bannButt.suspectDesc = new Flag.SuspectDesc(); if (_wl.suspectDesc) { bannButt.suspectDesc.WLactive = false; } } } // Return severity for highlighter (no dupe run)) if (hpMode.hlFlag) { // get severities from the banners severityButt = 0; for ( var tempKey in bannButt ) { if ( bannButt[tempKey] && bannButt[tempKey].active ) { // If the particular message is active if ( bannButt[tempKey].hasOwnProperty('WLactive') ) { if ( bannButt[tempKey].WLactive ) { // If there's a WL option, enable it severityButt = Math.max(bannButt[tempKey].severity, severityButt); // if ( bannButt[tempKey].severity > 0) { // phlogdev('Issue with '+item.attributes.name+': '+tempKey); // phlogdev('Severity: '+bannButt[tempKey].severity); // } } } else { severityButt = Math.max(bannButt[tempKey].severity, severityButt); // if ( bannButt[tempKey].severity > 0) { // phlogdev('Issue with '+item.attributes.name+': '+tempKey); // phlogdev('Severity: '+bannButt[tempKey].severity); // } } } } // Special case flags if ( item.attributes.lockRank === 0 && (item.attributes.categories.indexOf('HOSPITAL_MEDICAL_CARE') > -1 || item.attributes.categories.indexOf('HOSPITAL_URGENT_CARE') > -1 || item.isGasStation()) ) { severityButt = 5; } if ( severityButt === 0 && hlLockFlag ) { severityButt = 'lock'; } if ( severityButt === 1 && hlLockFlag ) { severityButt = 'lock1'; } if ( item.attributes.adLocked ) { severityButt = 'adLock'; } return severityButt; } // *** Below here is for harmonization only. HL ends in previous step. // Run nearby duplicate place finder function dupeHNRangeList = []; bannDupl = {}; if (newName.replace(/[^A-Za-z0-9]/g,'').length > 0 && !item.attributes.residential && !isEmergencyRoom(item) && !isRestArea(item)) { if ( $('#WMEPH-DisableDFZoom').prop('checked') ) { // don't zoom and pan for results outside of FOV duplicateName = findNearbyDuplicate(newName, newAliases, item, false); } else { duplicateName = findNearbyDuplicate(newName, newAliases, item, true); } if (duplicateName[1]) { bannButt.overlapping = new Flag.Overlapping(); } duplicateName = duplicateName[0]; if (duplicateName.length > 0) { if (duplicateName.length+1 !== dupeIDList.length && _USER.isDevUser) { // If there's an issue with the data return, allow an error report if (confirm('WMEPH: Dupefinder Error!\nClick OK to report this') ) { // if the category doesn't translate, then pop an alert that will make a forum post to the thread reportError({ subject: 'WMEPH Bug report DupeID', message: 'Script version: ' + _SCRIPT_VERSION + devVersStr + '\nPermalink: ' + placePL + '\nPlace name: ' + item.attributes.name + '\nCountry: ' + addr.country.name + '\n--------\nDescribe the error:\nDupeID mismatch with dupeName list' }); } } else { var wlAction = function(dID) { wlKeyName = 'dupeWL'; if (!venueWhitelist.hasOwnProperty(itemID)) { // If venue is NOT on WL, then add it. venueWhitelist[itemID] = { dupeWL: [] }; } if (!venueWhitelist[itemID].hasOwnProperty(wlKeyName)) { // If dupeWL key is not in venue WL, then initialize it. venueWhitelist[itemID][wlKeyName] = []; } venueWhitelist[itemID].dupeWL.push(dID); // WL the id for the duplicate venue venueWhitelist[itemID].dupeWL = uniq(venueWhitelist[itemID].dupeWL); // Make an entry for the opposite item if (!venueWhitelist.hasOwnProperty(dID)) { // If venue is NOT on WL, then add it. venueWhitelist[dID] = { dupeWL: [] }; } if (!venueWhitelist[dID].hasOwnProperty(wlKeyName)) { // If dupeWL key is not in venue WL, then initialize it. venueWhitelist[dID][wlKeyName] = []; } venueWhitelist[dID].dupeWL.push(itemID); // WL the id for the duplicate venue venueWhitelist[dID].dupeWL = uniq(venueWhitelist[dID].dupeWL); saveWL_LS(true); // Save the WL to local storage WMEPH_WLCounter(); bannButt2.clearWL.active = true; bannDupl[dID].active = false; harmonizePlaceGo(item,'harmonize'); }; for (var ijx=1; ijx<duplicateName.length+1; ijx++) { bannDupl[dupeIDList[ijx]] = { active: true, severity: 2, message: duplicateName[ijx-1], WLactive: false, WLvalue: wlButtText, WLtitle: 'Whitelist Duplicate', WLaction: wlAction }; if ( venueWhitelist.hasOwnProperty(itemID) && venueWhitelist[itemID].hasOwnProperty('dupeWL') && venueWhitelist[itemID].dupeWL.indexOf(dupeIDList[ijx]) > -1 ) { // if the dupe is on the whitelist then remove it from the banner bannDupl[dupeIDList[ijx]].active = false; } else { // Otherwise, activate the WL button bannDupl[dupeIDList[ijx]].WLactive = true; } } // END loop for duplicate venues } } } // Check HN range (this depends on the returned dupefinder data, so has to run after it) if (dupeHNRangeList.length > 3) { var dhnix, dupeHNRangeListSorted = []; sortWithIndex(dupeHNRangeDistList); for (dhnix = 0; dhnix < dupeHNRangeList.length; dhnix++) { dupeHNRangeListSorted.push(dupeHNRangeList[ dupeHNRangeDistList.sortIndices[dhnix] ]); } // Calculate HN/distance ratio with other venues // var sumHNRatio = 0; var arrayHNRatio = []; for (dhnix = 0; dhnix < dupeHNRangeListSorted.length; dhnix++) { arrayHNRatio.push(Math.abs( (parseInt(item.attributes.houseNumber) - dupeHNRangeListSorted[dhnix]) / dupeHNRangeDistList[dhnix] )); } sortWithIndex(arrayHNRatio); // Examine either the median or the 8th index if length is >16 var arrayHNRatioCheckIX = Math.min(Math.round(arrayHNRatio.length/2), 8); if (arrayHNRatio[arrayHNRatioCheckIX] > 1.4) { bannButt.HNRange = new Flag.HNRange(); if (_wl.HNRange) { bannButt.HNRange.WLactive = false; bannButt.HNRange.active = false; } if (arrayHNRatio[arrayHNRatioCheckIX] > 5) { bannButt.HNRange.severity = 3; } // show stats if HN out of range phlogdev('HNs: ' + dupeHNRangeListSorted); phlogdev('Distances: ' + dupeHNRangeDistList); phlogdev('arrayHNRatio: ' + arrayHNRatio); phlogdev('HN Ratio Score: ' + arrayHNRatio[Math.round(arrayHNRatio.length/2)]); } } executeMultiAction(actions); if (hpMode.harmFlag) { // Update icons to reflect current WME place services updateServicesChecks(bannServ); } if (bannButt.lockRPP) bannButt.lockRPP.message = 'Current lock: '+ (parseInt(item.attributes.lockRank)+1) +'. '+RPPLockString+' ?'; // Assemble the banners assembleBanner(); // Make Messaging banners showOpenPlaceWebsiteButton(); showSearchButton(); } // END harmonizePlaceGo function // Set up banner messages function assembleBanner() { let venue = getSelectedVenue(); if (!venue) return; phlogdev('Building banners'); var dupesFound = 0; var rowData; var $rowDiv; var rowDivs = []; severityButt = 0; let func = elem => { return {id: elem.getAttribute('id'), val: elem.value}; }; _textEntryValues = $('#WMEPH_banner input[type="text"]').toArray().map(func); _textEntryValues = _textEntryValues.concat($('#WMEPH_banner textarea').toArray().map(func)); // Setup duplicates banners $rowDiv = $('<div class="banner-row yellow">'); Object.keys(bannDupl).forEach(tempKey => { rowData = bannDupl[tempKey]; if (rowData.active) { dupesFound += 1; let $dupeDiv = $('<div class="dupe">').appendTo($rowDiv); $dupeDiv.append($('<span style="margin-right:4px">').html('• ' + rowData.message)); if (rowData.value) { // Nothing happening here yet. } if (rowData.WLactive && rowData.WLaction) { // If there's a WL option, enable it severityButt = Math.max(rowData.severity, severityButt); $dupeDiv.append( $('<button>', {class:'btn btn-success btn-xs wmephwl-btn', id:'WMEPH_WL' + tempKey, title: rowData.WLtitle}).text(rowData.WLvalue) ); } } }); if (dupesFound) { // if at least 1 dupe $rowDiv.prepend('Possible duplicate' + (dupesFound > 1 ? 's' : '') + ':'); rowDivs.push($rowDiv); } // Build banners above the Services Object.keys(bannButt).forEach(tempKey => { rowData = bannButt[tempKey]; if (rowData && rowData.active) { // If the particular message is active $rowDiv = $('<div class="banner-row">'); if (rowData.severity === 3) { $rowDiv.addClass('red'); } else if (rowData.severity === 2) { $rowDiv.addClass('yellow'); } else if (rowData.severity === 1) { $rowDiv.addClass('blue'); } else if (rowData.severity === 0) { $rowDiv.addClass('gray'); } if (rowData.divId) { $rowDiv.attr('id', rowData.divId); } if (rowData.message && rowData.message.length) { $rowDiv.append($('<span>').css({'margin-right':'4px'}).append('• ' + rowData.message)); } if (rowData.value) { $rowDiv.append( $('<button>', {class:'btn btn-default btn-xs wmeph-btn', id:'WMEPH_' + tempKey, title:rowData.title || ''}).css({'margin-right':'4px'}).html(rowData.value) ); } if (rowData.value2) { $rowDiv.append( $('<button>', {class:'btn btn-default btn-xs wmeph-btn', id:'WMEPH_' + tempKey + '_2', title:rowData.title2 || ''}).css({'margin-right':'4px'}).html(rowData.value2) ); } if (rowData.WLactive) { if (rowData.WLaction) { // If there's a WL option, enable it severityButt = Math.max(rowData.severity, severityButt); $rowDiv.append( $('<button>', {class:'btn btn-success btn-xs wmephwl-btn', id:'WMEPH_WL' + tempKey, title:rowData.WLtitle}).text('WL') ); } } else { severityButt = Math.max(rowData.severity, severityButt); } if (rowData.suffixMessage) { $rowDiv.append( $('<div>').css({'margin-top':'2px'}).append(rowData.suffixMessage) ); } rowDivs.push($rowDiv); } }); if ( $('#WMEPH-ColorHighlighting').prop('checked') ) { venue.attributes.wmephSeverity = severityButt; } if ($('#WMEPH_banner').length === 0 ) { $('<div id="WMEPH_banner">').prependTo('.contents'); } else { $('#WMEPH_banner').empty(); } var bgColor; switch (severityButt) { case 1: bgColor = 'rgb(50, 50, 230)'; // blue break; case 2: bgColor = 'rgb(217, 173, 42)'; // yellow break; case 3: bgColor = 'rgb(211, 48, 48)'; // red break; default: bgColor = 'rgb(36, 172, 36)'; // green } $('#WMEPH_banner').css({'background-color': bgColor}).append(rowDivs); //$('#select2-drop').css({display:'none'}); assembleServicesBanner(); // Build general banners (below the Services) rowDivs = []; Object.keys(bannButt2).forEach(tempKey => { var rowData = bannButt2[tempKey]; if (rowData.active ) { // If the particular message is active $rowDiv = $('<div>'); $rowDiv.append(rowData.message); if (rowData.action) { $rowDiv.append(' <input class="btn btn-info btn-xs wmeph-btn" id="WMEPH_' + tempKey + '" title="' + rowData.title + '" style="" type="button" value="' + rowData.value + '">'); } rowDivs.push($rowDiv); severityButt = Math.max(bannButt2[tempKey].severity, severityButt); } }); if ($('#WMEPH_tools').length === 0 ) { $('#WMEPH_services').after( $('<div id="WMEPH_tools">').css({'background-color': '#eee', 'color': 'black', 'font-size': '15px', 'padding': '0px 4px 4px 4px', 'margin-left': '4px', 'margin-right': 'auto'}) ); } else { $('#WMEPH_tools').empty(); } $('#WMEPH_tools').append(rowDivs); //$('#select2-drop').css({display:'none'}); // Set up Duplicate onclicks if ( dupesFound ) { setupButtons(bannDupl); } // Setup bannButt onclicks setupButtons(bannButt); // Setup bannButt2 onclicks setupButtons(bannButt2); // Prefill zip code text box if (bannButt.missingUSPSZipAlt && bannButt.missingUSPSZipAlt.suggestedValue) { $('input#WMEPH-zipAltNameAdd').val(bannButt.missingUSPSZipAlt.suggestedValue); } // Add click handlers for parking lot helper buttons. $('.wmeph-pla-spaces-btn').click(function() { let venue = getSelectedVenue(); var selectedValue = $(this).attr('id').replace('wmeph_',''); var existingAttr = venue.attributes.categoryAttributes.PARKING_LOT; var newAttr = {}; if (existingAttr) { for (var prop in existingAttr) { var value = existingAttr[prop]; if (Array.isArray(value)) value = [].concat(value); newAttr[prop] = value; } } newAttr.estimatedNumberOfSpots = selectedValue; W.model.actionManager.add(new UpdateObject(venue, {'categoryAttributes': {PARKING_LOT: newAttr}})); harmonizePlaceGo(venue, 'harmonize'); }); $('.wmeph-pla-lot-type-btn').click(function() { let venue = getSelectedVenue(); var selectedValue = $(this).data('lot-type'); var existingAttr = venue.attributes.categoryAttributes.PARKING_LOT; var newAttr = {}; if (existingAttr) { for (var prop in existingAttr) { var value = existingAttr[prop]; if (Array.isArray(value)) value = [].concat(value); newAttr[prop] = value; } } newAttr.parkingType = selectedValue; W.model.actionManager.add(new UpdateObject(venue, {'categoryAttributes': {PARKING_LOT: newAttr}})); harmonizePlaceGo(venue, 'harmonize'); }); $('.wmeph-pla-cost-type-btn').click(function() { let venue = getSelectedVenue(); var selectedValue = $(this).attr('id').replace('wmeph_',''); var existingAttr = venue.attributes.categoryAttributes.PARKING_LOT; var newAttr = {}; if (existingAttr) { for (var prop in existingAttr) { var value = existingAttr[prop]; if (Array.isArray(value)) value = [].concat(value); newAttr[prop] = value; } } newAttr.costType = selectedValue; W.model.actionManager.add(new UpdateObject(venue, {'categoryAttributes': {PARKING_LOT: newAttr}})); harmonizePlaceGo(venue, 'harmonize'); }); // If pressing enter in the HN entry box, add the HN $('#WMEPH-HNAdd').keyup(function(event){ if( event.keyCode === 13 && $('#WMEPH-HNAdd').val() !== '' ){ $('#WMEPH_hnMissing').click(); } }); // If pressing enter in the phone entry box, add the phone $('#WMEPH-PhoneAdd').keyup(function(event){ if( event.keyCode === 13 && $('#WMEPH-PhoneAdd').val() !== '' ){ $('#WMEPH_phoneMissing').click(); $('#WMEPH_badAreaCode').click(); } }); // If pressing enter in the URL entry box, add the URL $('#WMEPH-UrlAdd').keyup(function(event){ if( event.keyCode === 13 && $('#WMEPH-UrlAdd').val() !== '' ){ $('#WMEPH_urlMissing').click(); } }); // If pressing enter in the USPS zip code alt entry box... $('#WMEPH-zipAltNameAdd').keyup(function(event) { if (event.keyCode === 13 && $(this).val() !== '') { $('#WMEPH_missingUSPSZipAlt').click(); } }); // If pasting or dropping into hours entry box function resetHoursEntryHeight() { var $sel = $('#WMEPH-HoursPaste'); $sel.focus(); var oldText = $sel.val(); if (oldText === _DEFAULT_HOURS_TEXT) { $sel.val(''); } // A small delay to allow window to process pasted text before running. setTimeout(() => { var text = $sel.val(); var elem = $sel[0]; var lineCount = (text.match(/\n/g) || []).length + 1; var height = lineCount*18 + 6 + (elem.scrollWidth > elem.clientWidth ? 20 : 0); $sel.css({height:height+'px'}); },100); } $('#WMEPH-HoursPaste') .bind('paste', resetHoursEntryHeight) .bind('drop', resetHoursEntryHeight) .keydown(resetHoursEntryHeight) .bind('dragenter', function() { var text = $('#WMEPH-HoursPaste').val(); if (text === _DEFAULT_HOURS_TEXT) { $('#WMEPH-HoursPaste').val(''); } }); // If pressing enter in the hours entry box, parse the entry $('#WMEPH-HoursPaste').keydown(function(event){ if (event.keyCode === 13) { if (event.ctrlKey) { // Simulate a newline event (shift + enter) var text = this.value; var selStart = this.selectionStart; this.value = text.substr(0, selStart) + '\n' + text.substr(this.selectionEnd, text.length-1); this.selectionStart = selStart+1; this.selectionEnd = selStart+1; return true; } else if(!(event.shiftKey||event.ctrlKey) && $('#WMEPH-HoursPaste').val() !== '' ){ event.stopPropagation(); event.preventDefault(); event.returnValue = false; event.cancelBubble = true; $('#WMEPH_noHours').click(); return false; } } }); $('#WMEPH-HoursPaste').focus(function(){ if (this.value === _DEFAULT_HOURS_TEXT) { this.value = ''; } this.style.color = 'black'; }).blur(function(){ if ( this.value === '') { this.value = _DEFAULT_HOURS_TEXT; this.style.color = '#999'; } }); // Format "no hours" section and hook up button events. $('#WMEPH_WLnoHours').css({'vertical-align':'top'}); // NOTE: Leave these wrapped in the "() => ..." functions, to make sure "this" is bound properly. if (bannButt.noHours) { $('#WMEPH_noHours').click(() => bannButt.noHours.addHoursAction()); $('#WMEPH_noHours_2').click(() => bannButt.noHours.replaceHoursAction()); } if (_textEntryValues) { _textEntryValues.forEach(entry => $('#' + entry.id).val(entry.val)); } } // END assemble Banner function function assembleServicesBanner() { let venue = getSelectedVenue(); if (venue && !$('#WMEPH-HideServicesButtons').prop('checked')) { // setup Add Service Buttons for suggested services var rowDivs = []; if (!venue.isResidential()) { var $rowDiv = $('<div>'); var servButtHeight = '27'; var buttons = []; Object.keys(bannServ).forEach(tempKey => { var rowData = bannServ[tempKey]; if (rowData.active) { // If the particular service is active var $input = $('<input>', {class:rowData.icon, id:'WMEPH_' + tempKey, type:'button', 'title': rowData.title}).css( {border:0, 'background-size':'contain', height:'27px', width: Math.ceil(servButtHeight * rowData.w2hratio).toString()+'px'} ); buttons.push($input); if (!rowData.checked) { $input.css({'-webkit-filter':'opacity(.25)', filter:'opacity(.25)'}); } $rowDiv.append($input); } }); if ($rowDiv.length) { $rowDiv.prepend('<span class="control-label">Add services:</span><br>'); } rowDivs.push($rowDiv); } if ($('#WMEPH_services').length === 0 ) { $('#WMEPH_banner').after( $('<div id="WMEPH_services">').css({'background-color': '#eee', 'color': 'black', 'font-size': '15px', 'padding': '4px', 'margin-left': '4px', 'margin-right': 'auto'}) ); } else { $('#WMEPH_services').empty(); } $('#WMEPH_services').append(rowDivs); // Setup bannServ onclicks if (!venue.isResidential()) { setupButtons(bannServ); } } } // Button onclick event handler function setupButtons(b) { for ( var tempKey in b ) { // Loop through the banner possibilities if ( b[tempKey] && b[tempKey].active ) { // If the particular message is active if (b[tempKey].action && b[tempKey].value) { // If there is an action, set onclick buttonAction(b, tempKey); } if (b[tempKey].action2 && b[tempKey].value2) { // If there is an action2, set onclick buttonAction2(b, tempKey); } // If there's a WL option, set up onclick if ( b[tempKey].WLactive && b[tempKey].WLaction ) { buttonWhitelist(b, tempKey); } } } } // END setupButtons function function buttonAction(b,bKey) { var button = document.getElementById('WMEPH_'+bKey); button.onclick = function() { b[bKey].action(); if (!b[bKey].noBannerAssemble) assembleBanner(); }; return button; } function buttonAction2(b,bKey) { var button = document.getElementById('WMEPH_'+bKey+'_2'); button.onclick = function() { b[bKey].action2(); if (!b[bKey].noBannerAssemble) assembleBanner(); }; return button; } function buttonWhitelist(b,bKey) { var button = document.getElementById('WMEPH_WL'+bKey); button.onclick = function() { if ( bKey.match(/^\d{5,}/) !== null ) { b[bKey].WLaction(bKey); } else { b[bKey].WLaction(); } b[bKey].WLactive = false; b[bKey].severity = 0; assembleBanner(); }; return button; } // Display run button on place sidebar function displayRunButton() { let betaDelay = 100; setTimeout(() => { let venue = getSelectedVenue(); if ($('#WMEPH_runButton').length === 0 ) { $('<div id="WMEPH_runButton">').prependTo('.contents'); } if ($('#runWMEPH').length === 0 ) { let devVersSuffix = _IS_DEV_VERSION ? '-β' : ''; let strButt1 = '<input class="btn btn-primary wmeph-fat-btn" id="runWMEPH" title="Run WMEPH'+devVersSuffix+' on Place" type="button" value="Run WMEPH'+devVersSuffix+'">'; $('#WMEPH_runButton').append(strButt1); } let btn = document.getElementById('runWMEPH'); if (btn !== null) { btn.onclick = function() { harmonizePlace(); }; } else { setTimeout(bootstrapRunButton,100); } if ( venue.attributes.categories.length === 1 && venue.attributes.categories[0] === 'SHOPPING_AND_SERVICES' ) { $('.suggested-categories').remove(); } showOpenPlaceWebsiteButton(); showSearchButton(); }, betaDelay); } // END displayRunButton funtion // Displays the Open Place Website button. function showOpenPlaceWebsiteButton() { let venue = getSelectedVenue(); if (venue) { var openPlaceWebsiteURL = venue.attributes.url; if (openPlaceWebsiteURL && openPlaceWebsiteURL.replace(/[^A-Za-z0-9]/g,'').length > 2) { if ($('#WMEPHurl').length === 0 ) { let strButt1 = '<input class="btn btn-success btn-xs wmeph-fat-btn" id="WMEPHurl" title="Open place URL" type="button" value="Website">'; $('#runWMEPH').after(strButt1); let btn = document.getElementById('WMEPHurl'); if (btn !== null) { btn.onclick = function() { openPlaceWebsiteURL = venue.attributes.url; if (openPlaceWebsiteURL.match(/^http/i) === null) { openPlaceWebsiteURL = 'http:\/\/'+openPlaceWebsiteURL; } if ( $('#WMEPH-WebSearchNewTab').prop('checked') ) { window.open(openPlaceWebsiteURL); } else { window.open(openPlaceWebsiteURL, searchResultsWindowName, searchResultsWindowSpecs); } }; } else { setTimeout(bootstrapRunButton,100); } } } else { if ($('#WMEPHurl').length > 0 ) { $('#WMEPHurl').remove(); } } } } function showSearchButton() { let venue = getSelectedVenue(); if (venue && $('#wmephSearch').length === 0 ) { let strButt1 = '<input class="btn btn-danger btn-xs wmeph-fat-btn" id="wmephSearch" title="Search the web for this place. Do not copy info from 3rd party sources!" type="button" value="Google">'; $('#WMEPH_runButton').append(strButt1); let btn = document.getElementById('wmephSearch'); if (btn !== null) { btn.onclick = function() { let addr = venue.getAddress(); if (addr.hasState()) { let url = buildGLink(venue.attributes.name, addr.attributes, venue.attributes.houseNumber); if ( $('#WMEPH-WebSearchNewTab').prop('checked') ) { window.open(url); } else { window.open(url, searchResultsWindowName, searchResultsWindowSpecs); } } else { alert('The state and country haven\'t been set for this place yet. Edit the address first.'); } }; } else { setTimeout(bootstrapRunButton,100); } } } // WMEPH Clone Tool function displayCloneButton() { var betaDelay = 80; if (_IS_DEV_VERSION) { betaDelay = 300; } setTimeout(() => { if ($('#WMEPH_runButton').length === 0 ) { $('<div id="WMEPH_runButton">').prependTo('.contents'); } let strButt1, btn; let venue = getSelectedVenue(); if (venue) { showOpenPlaceWebsiteButton(); if ($('#clonePlace').length === 0 ) { strButt1 = '<div style="margin-bottom: 3px;"></div><input class="btn btn-warning btn-xs wmeph-btn" id="clonePlace" title="Copy place info" type="button" value="Copy" style="font-weight:normal">'+ ' <input class="btn btn-warning btn-xs wmeph-btn" id="pasteClone" title="Apply the Place info. (Ctrl-Alt-O)" type="button" value="Paste (for checked boxes):" style="font-weight:normal"><br>'; $('#WMEPH_runButton').append(strButt1); createCloneCheckbox('WMEPH_runButton', 'WMEPH_CPhn', 'HN'); createCloneCheckbox('WMEPH_runButton', 'WMEPH_CPstr', 'Str'); createCloneCheckbox('WMEPH_runButton', 'WMEPH_CPcity', 'City'); createCloneCheckbox('WMEPH_runButton', 'WMEPH_CPurl', 'URL'); createCloneCheckbox('WMEPH_runButton', 'WMEPH_CPph', 'Ph'); $('#WMEPH_runButton').append('<br>'); createCloneCheckbox('WMEPH_runButton', 'WMEPH_CPdesc', 'Desc'); createCloneCheckbox('WMEPH_runButton', 'WMEPH_CPserv', 'Serv'); createCloneCheckbox('WMEPH_runButton', 'WMEPH_CPhrs', 'Hrs'); strButt1 = '<input class="btn btn-info btn-xs wmeph-btn" id="checkAllClone" title="Check all" type="button" value="All" style="font-weight:normal">'+ ' <input class="btn btn-info btn-xs wmeph-btn" id="checkAddrClone" title="Check Address" type="button" value="Addr" style="font-weight:normal">'+ ' <input class="btn btn-info btn-xs wmeph-btn" id="checkNoneClone" title="Check none" type="button" value="None" style="font-weight:normal"><br>'; $('#WMEPH_runButton').append(strButt1); } btn = document.getElementById('clonePlace'); if (btn !== null) { btn.onclick = function() { cloneMaster = {}; cloneMaster.addr = venue.getAddress(); if ( cloneMaster.addr.hasOwnProperty('attributes') ) { cloneMaster.addr = cloneMaster.addr.attributes; } cloneMaster.houseNumber = venue.attributes.houseNumber; cloneMaster.url = venue.attributes.url; cloneMaster.phone = venue.attributes.phone; cloneMaster.description = venue.attributes.description; cloneMaster.services = venue.attributes.services; cloneMaster.openingHours = venue.attributes.openingHours; cloneMaster.isPLA = venue.isParkingLot(); phlogdev('Place Cloned'); }; } else { setTimeout(bootstrapRunButton,100); return; } btn = document.getElementById('pasteClone'); if (btn !== null) { btn.onclick = function() { clonePlace(getSelectedVenue()); }; } else { setTimeout(bootstrapRunButton,100); } btn = document.getElementById('checkAllClone'); if (btn !== null) { btn.onclick = function() { if ( !$('#WMEPH_CPhn').prop('checked') ) { $('#WMEPH_CPhn').trigger('click'); } if ( !$('#WMEPH_CPstr').prop('checked') ) { $('#WMEPH_CPstr').trigger('click'); } if ( !$('#WMEPH_CPcity').prop('checked') ) { $('#WMEPH_CPcity').trigger('click'); } if ( !$('#WMEPH_CPurl').prop('checked') ) { $('#WMEPH_CPurl').trigger('click'); } if ( !$('#WMEPH_CPph').prop('checked') ) { $('#WMEPH_CPph').trigger('click'); } if ( !$('#WMEPH_CPserv').prop('checked') ) { $('#WMEPH_CPserv').trigger('click'); } if ( !$('#WMEPH_CPdesc').prop('checked') ) { $('#WMEPH_CPdesc').trigger('click'); } if ( !$('#WMEPH_CPhrs').prop('checked') ) { $('#WMEPH_CPhrs').trigger('click'); } }; } else { setTimeout(bootstrapRunButton,100); } btn = document.getElementById('checkAddrClone'); if (btn !== null) { btn.onclick = function() { if ( !$('#WMEPH_CPhn').prop('checked') ) { $('#WMEPH_CPhn').trigger('click'); } if ( !$('#WMEPH_CPstr').prop('checked') ) { $('#WMEPH_CPstr').trigger('click'); } if ( !$('#WMEPH_CPcity').prop('checked') ) { $('#WMEPH_CPcity').trigger('click'); } if ( $('#WMEPH_CPurl').prop('checked') ) { $('#WMEPH_CPurl').trigger('click'); } if ( $('#WMEPH_CPph').prop('checked') ) { $('#WMEPH_CPph').trigger('click'); } if ( $('#WMEPH_CPserv').prop('checked') ) { $('#WMEPH_CPserv').trigger('click'); } if ( $('#WMEPH_CPdesc').prop('checked') ) { $('#WMEPH_CPdesc').trigger('click'); } if ( $('#WMEPH_CPhrs').prop('checked') ) { $('#WMEPH_CPhrs').trigger('click'); } }; } else { setTimeout(bootstrapRunButton,100); } btn = document.getElementById('checkNoneClone'); if (btn !== null) { btn.onclick = function() { if ( $('#WMEPH_CPhn').prop('checked') ) { $('#WMEPH_CPhn').trigger('click'); } if ( $('#WMEPH_CPstr').prop('checked') ) { $('#WMEPH_CPstr').trigger('click'); } if ( $('#WMEPH_CPcity').prop('checked') ) { $('#WMEPH_CPcity').trigger('click'); } if ( $('#WMEPH_CPurl').prop('checked') ) { $('#WMEPH_CPurl').trigger('click'); } if ( $('#WMEPH_CPph').prop('checked') ) { $('#WMEPH_CPph').trigger('click'); } if ( $('#WMEPH_CPserv').prop('checked') ) { $('#WMEPH_CPserv').trigger('click'); } if ( $('#WMEPH_CPdesc').prop('checked') ) { $('#WMEPH_CPdesc').trigger('click'); } if ( $('#WMEPH_CPhrs').prop('checked') ) { $('#WMEPH_CPhrs').trigger('click'); } }; } else { setTimeout(bootstrapRunButton,100); } } }, betaDelay); } // END displayCloneButton funtion // Catch PLs and reloads that have a place selected already and limit attempts to about 10 seconds function bootstrapRunButton(numAttempts) { numAttempts = numAttempts || 0; if (numAttempts < 10) { if (W.selectionManager.getSelectedFeatures().length === 1) { let venue = getSelectedVenue(); if (venue && venue.isApproved()) { displayRunButton(); showOpenPlaceWebsiteButton(); showSearchButton(); getPanelFields(); if (localStorage.getItem('WMEPH-EnableCloneMode') === '1') { displayCloneButton(); } } } else { setTimeout(bootstrapRunButton(numAttempts + 1),1000); } } } // Find field divs function getPanelFields() { var panelFieldsList = $('.form-control'), pfa; for (var pfix=0; pfix<panelFieldsList.length; pfix++) { pfa = panelFieldsList[pfix].name; if (pfa === 'name') { panelFields.name = pfix; } if (pfa === 'lockRank') { panelFields.lockRank = pfix; } if (pfa === 'description') { panelFields.description = pfix; } if (pfa === 'url') { panelFields.url = pfix; } if (pfa === 'phone') { panelFields.phone = pfix; } if (pfa === 'brand') { panelFields.brand = pfix; } } var placeNavTabs = $('.nav'); for (pfix=0; pfix<placeNavTabs.length; pfix++) { pfa = placeNavTabs[pfix].innerHTML; if (pfa.indexOf('landmark-edit') > -1) { panelFieldsList = placeNavTabs[pfix].children; panelFields.navTabsIX = pfix; break; } } for (pfix=0; pfix<panelFieldsList.length; pfix++) { pfa = panelFieldsList[pfix].innerHTML; if (pfa.indexOf('landmark-edit-general') > -1) { panelFields.navTabGeneral = pfix; } if (pfa.indexOf('landmark-edit-more') > -1) { panelFields.navTabMore = pfix; } } } // Function to clone info from a place function clonePlace() { phlog('Cloning info...'); if (cloneMaster !== null && cloneMaster.hasOwnProperty('url')) { let venue = getSelectedVenue(); var cloneItems = {}; var updateItem = false; if ( $('#WMEPH_CPhn').prop('checked') ) { cloneItems.houseNumber = cloneMaster.houseNumber; updateItem = true; } if ( $('#WMEPH_CPurl').prop('checked') ) { cloneItems.url = cloneMaster.url; updateItem = true; } if ( $('#WMEPH_CPph').prop('checked') ) { cloneItems.phone = cloneMaster.phone; updateItem = true; } if ( $('#WMEPH_CPdesc').prop('checked') ) { cloneItems.description = cloneMaster.description; updateItem = true; } if ( $('#WMEPH_CPserv').prop('checked') && venue.isParkingLot() === cloneMaster.isPLA) { cloneItems.services = cloneMaster.services; updateItem = true; } if ( $('#WMEPH_CPhrs').prop('checked') ) { cloneItems.openingHours = cloneMaster.openingHours; updateItem = true; } if (updateItem) { W.model.actionManager.add(new UpdateObject(venue, cloneItems) ); phlogdev('Item details cloned'); } var copyStreet = $('#WMEPH_CPstr').prop('checked'); var copyCity = $('#WMEPH_CPcity').prop('checked'); if (copyStreet || copyCity) { var originalAddress = venue.getAddress(); var itemRepl = { street: copyStreet ? cloneMaster.addr.street : originalAddress.attributes.street, city: copyCity ? cloneMaster.addr.city : originalAddress.attributes.city, state: copyCity ? cloneMaster.addr.state : originalAddress.attributes.state, country: copyCity ? cloneMaster.addr.country : originalAddress.attributes.country }; updateAddress(venue, itemRepl); phlogdev('Item address cloned'); } } else { phlog('Please copy a place'); } } // Formats "hour object" into a string. function formatOpeningHour(hourEntry) { var dayNames = ['Sunday','Monday','Tuesday','Wednesday','Thursday','Friday','Saturday']; var hours = hourEntry.attributes.fromHour + '-' + hourEntry.attributes.toHour; return hourEntry.attributes.days.map(day => dayNames[day] + ' ' + hours).join(', '); } // Pull natural text from opening hours function getOpeningHours(venue) { return venue && venue.getOpeningHours && venue.getOpeningHours().map(formatOpeningHour); } // function to check overlapping hours function checkHours(hoursObj) { if (hoursObj.length === 1) { return true; } var daysObj, fromHourTemp, toHourTemp; for (var day2Ch=0; day2Ch<7; day2Ch++) { // Go thru each day of the week daysObj = []; for ( var hourSet = 0; hourSet < hoursObj.length; hourSet++ ) { // For each set of hours if (hoursObj[hourSet].days.indexOf(day2Ch) > -1) { // pull out hours that are for the current day, add 2400 if it goes past midnight, and store fromHourTemp = hoursObj[hourSet].fromHour.replace(/\:/g,''); toHourTemp = hoursObj[hourSet].toHour.replace(/\:/g,''); if (toHourTemp <= fromHourTemp) { toHourTemp = parseInt(toHourTemp) + 2400; } daysObj.push([fromHourTemp, toHourTemp]); } } if (daysObj.length > 1) { // If there's multiple hours for the day, check them for overlap for ( var hourSetCheck2 = 1; hourSetCheck2 < daysObj.length; hourSetCheck2++ ) { for ( var hourSetCheck1 = 0; hourSetCheck1 < hourSetCheck2; hourSetCheck1++ ) { if ( daysObj[hourSetCheck2][0] > daysObj[hourSetCheck1][0] && daysObj[hourSetCheck2][0] < daysObj[hourSetCheck1][1] ) { return false; } if ( daysObj[hourSetCheck2][1] > daysObj[hourSetCheck1][0] && daysObj[hourSetCheck2][1] < daysObj[hourSetCheck1][1] ) { return false; } } } } } return true; } // Duplicate place finder ###bmtg function findNearbyDuplicate(itemName, itemAliases, item, recenterOption) { dupeIDList = [item.attributes.id]; dupeHNRangeList = []; dupeHNRangeIDList = []; dupeHNRangeDistList = []; var venueList = W.model.venues.objects, currNameList = [], testNameList = [], testVenueAtt, testName, testNameNoNum, itemNameRF, aliasNameRF, aliasNameNoNum; var wlDupeMatch = false, wlDupeList = [], nameMatch = false, altNameMatch = -1, aliix, cnlix, tnlix, randInt = 100; var outOfExtent = false, mapExtent = W.map.getExtent(), padFrac = 0.15; // how much to pad the zoomed window // Initialize the cooridnate extents for duplicates var minLon = item.geometry.getCentroid().x, minLat = item.geometry.getCentroid().y; var maxLon = minLon, maxLat = minLat; // genericterms to skip if it's all that remains after stripping numbers var noNumSkip = 'BANK|ATM|HOTEL|MOTEL|STORE|MARKET|SUPERMARKET|GYM|GAS|GASOLINE|GASSTATION|CAFE|OFFICE|OFFICES|CARRENTAL|RENTALCAR|RENTAL|SALON|BAR|BUILDING|LOT'; noNumSkip = noNumSkip + '|'+ collegeAbbreviations; noNumSkip = noNumSkip.split('|'); // Make the padded extent mapExtent.left = mapExtent.left + padFrac * (mapExtent.right-mapExtent.left); mapExtent.right = mapExtent.right - padFrac * (mapExtent.right-mapExtent.left); mapExtent.bottom = mapExtent.bottom + padFrac * (mapExtent.top-mapExtent.bottom); mapExtent.top = mapExtent.top - padFrac * (mapExtent.top-mapExtent.bottom); var allowedTwoLetters = ['BP','DQ','BK','BW','LQ','QT','DB','PO']; var labelFeatures = [], dupeNames = [], labelText, labelTextReformat, pt, textFeature, labelColorIX = 0; var labelColorList = ['#3F3']; // Name formatting for the WME place name itemNameRF = itemName.toUpperCase().replace(/ AND /g, '').replace(/^THE /g, '').replace(/[^A-Z0-9]/g, ''); // Format name if ( itemNameRF.length>2 || allowedTwoLetters.indexOf(itemNameRF) > -1 ) { currNameList.push(itemNameRF); } else { currNameList.push('PRIMNAMETOOSHORT_PJZWX'); } var itemNameNoNum = itemNameRF.replace(/[^A-Z]/g, ''); // Clear non-letter characters for alternate match ( HOLLYIVYPUB23 --> HOLLYIVYPUB ) if ( ((itemNameNoNum.length>2 && noNumSkip.indexOf(itemNameNoNum) === -1) || allowedTwoLetters.indexOf(itemNameNoNum) > -1) && item.attributes.categories.indexOf('PARKING_LOT') === -1 ) { // only add de-numbered name if anything remains currNameList.push(itemNameNoNum); } if (itemAliases.length > 0) { for (aliix=0; aliix<itemAliases.length; aliix++) { aliasNameRF = itemAliases[aliix].toUpperCase().replace(/ AND /g, '').replace(/^THE /g, '').replace(/[^A-Z0-9]/g, ''); // Format name if ( (aliasNameRF.length>2 && noNumSkip.indexOf(aliasNameRF) === -1) || allowedTwoLetters.indexOf(aliasNameRF) > -1 ) { // only add de-numbered name if anything remains currNameList.push(aliasNameRF); } aliasNameNoNum = aliasNameRF.replace(/[^A-Z]/g, ''); // Clear non-letter characters for alternate match ( HOLLYIVYPUB23 --> HOLLYIVYPUB ) if ( ((aliasNameNoNum.length>2 && noNumSkip.indexOf(aliasNameNoNum) === -1) || allowedTwoLetters.indexOf(aliasNameNoNum) > -1) && item.attributes.categories.indexOf('PARKING_LOT') === -1 ) { // only add de-numbered name if anything remains currNameList.push(aliasNameNoNum); } } } currNameList = uniq(currNameList); // remove duplicates // Remove any previous search labels and move the layer above the places layer _dupeLayer.destroyFeatures(); //var vecLyrPlaces = W.map.getLayersBy('uniqueName','landmarks')[0]; //W.map.setLayerZIndex(_dupeLayer, 70); //_dupeLayer.setZIndex(parseInt(vecLyrPlaces.getZIndex())+3); // Move layer to just on top of Places layer if ( venueWhitelist.hasOwnProperty(item.attributes.id) ) { if ( venueWhitelist[item.attributes.id].hasOwnProperty('dupeWL') ) { wlDupeList = venueWhitelist[item.attributes.id].dupeWL; } } var overlappingFlag = false; var addrItem = item.getAddress(), itemCompAddr = false; if ( addrItem.hasOwnProperty('attributes') ) { addrItem = addrItem.attributes; } if (addrItem.street !== null && addrItem.street.name !== null && item.attributes.houseNumber && item.attributes.houseNumber.match(/\d/g) !== null) { itemCompAddr = true; } for (var venix in venueList) { // for each place on the map: if (venueList.hasOwnProperty(venix)) { // hOP filter nameMatch = false; altNameMatch = -1; testVenueAtt = venueList[venix].attributes; var excludePLADupes = $('#WMEPH-ExcludePLADupes').prop('checked'); if ((!excludePLADupes || (excludePLADupes && !(item.isParkingLot() || venueList[venix].isParkingLot()))) && !isEmergencyRoom(venueList[venix])) { var pt2ptDistance = item.geometry.getCentroid().distanceTo(venueList[venix].geometry.getCentroid()); if ( item.isPoint() && venueList[venix].isPoint() && pt2ptDistance < 2 && item.attributes.id !== testVenueAtt.id ) { overlappingFlag = true; } wlDupeMatch = false; if (wlDupeList.length>0 && wlDupeList.indexOf(testVenueAtt.id) > -1) { wlDupeMatch = true; } // get HNs for places on same street var addrDupe = venueList[venix].getAddress(); if ( addrDupe.hasOwnProperty('attributes') ) { addrDupe = addrDupe.attributes; } if (itemCompAddr && addrDupe.street !== null && addrDupe.street.name !== null && testVenueAtt.houseNumber && testVenueAtt.houseNumber !== '' && venix !== item.attributes.id && addrItem.street.name === addrDupe.street.name && testVenueAtt.houseNumber < 1000000) { dupeHNRangeList.push(parseInt(testVenueAtt.houseNumber)); dupeHNRangeIDList.push(testVenueAtt.id); dupeHNRangeDistList.push(pt2ptDistance); } // Check for duplicates if ( !wlDupeMatch && dupeIDList.length<6 && pt2ptDistance < 800 && !testVenueAtt.residential && venix !== item.attributes.id && 'string' === typeof testVenueAtt.id && testVenueAtt.name !== null && testVenueAtt.name.length>1 ) { // don't do res, the point itself, new points or no name // If item has a complete address and test venue does, and they are different, then no dupe var suppressMatch = false; if ( itemCompAddr && addrDupe.street !== null && addrDupe.street.name !== null && testVenueAtt.houseNumber && testVenueAtt.houseNumber.match(/\d/g) !== null ) { if ( item.attributes.lockRank > 0 && testVenueAtt.lockRank > 0 ) { if ( item.attributes.houseNumber !== testVenueAtt.houseNumber || addrItem.street.name !== addrDupe.street.name ) { suppressMatch = true; } } else { if ( item.attributes.houseNumber !== testVenueAtt.houseNumber && addrItem.street.name !== addrDupe.street.name ) { suppressMatch = true; } } } if ( !suppressMatch ) { //Reformat the testPlace name testName = testVenueAtt.name.toUpperCase().replace(/\s+[-\(].*$/,'').replace(/ AND /g, '').replace(/^THE /g, '').replace(/[^A-Z0-9]/g, ''); // Format test name if ( (testName.length>2 && noNumSkip.indexOf(testName) === -1) || allowedTwoLetters.indexOf(testName) > -1 ) { testNameList = [testName]; } else { testNameList = ['TESTNAMETOOSHORTQZJXS'+randInt]; randInt++; } testNameNoNum = testName.replace(/[^A-Z]/g, ''); // Clear non-letter characters for alternate match if ( ((testNameNoNum.length>2 && noNumSkip.indexOf(testNameNoNum) === -1) || allowedTwoLetters.indexOf(testNameNoNum) > -1) && testVenueAtt.categories.indexOf('PARKING_LOT') === -1 ) { // only add de-numbered name if at least 2 chars remain testNameList.push(testNameNoNum); } // primary name matching loop for (tnlix=0; tnlix < testNameList.length; tnlix++) { for (cnlix=0; cnlix < currNameList.length; cnlix++) { if ( (testNameList[tnlix].indexOf(currNameList[cnlix]) > -1 || currNameList[cnlix].indexOf(testNameList[tnlix]) > -1) ) { nameMatch = true; break; } } if (nameMatch) {break;} // break if a match found } if (!nameMatch && testVenueAtt.aliases.length > 0) { for (aliix=0; aliix<testVenueAtt.aliases.length; aliix++) { aliasNameRF = testVenueAtt.aliases[aliix].toUpperCase().replace(/ AND /g, '').replace(/^THE /g, '').replace(/[^A-Z0-9]/g, ''); // Format name if ( (aliasNameRF.length>2 && noNumSkip.indexOf(aliasNameRF) === -1) || allowedTwoLetters.indexOf(aliasNameRF) > -1 ) { testNameList = [aliasNameRF]; } else { testNameList = ['ALIASNAMETOOSHORTQOFUH'+randInt]; randInt++; } aliasNameNoNum = aliasNameRF.replace(/[^A-Z]/g, ''); // Clear non-letter characters for alternate match ( HOLLYIVYPUB23 --> HOLLYIVYPUB ) if (((aliasNameNoNum.length>2 && noNumSkip.indexOf(aliasNameNoNum) === -1) || allowedTwoLetters.indexOf(aliasNameNoNum) > -1) && testVenueAtt.categories.indexOf('PARKING_LOT') === -1) { // only add de-numbered name if at least 2 characters remain testNameList.push(aliasNameNoNum); } else { testNameList.push('111231643239'+randInt); // just to keep track of the alias in question, always add something. randInt++; } } for (tnlix=0; tnlix < testNameList.length; tnlix++) { for (cnlix=0; cnlix < currNameList.length; cnlix++) { if ( (testNameList[tnlix].indexOf(currNameList[cnlix]) > -1 || currNameList[cnlix].indexOf(testNameList[tnlix]) > -1) ) { // get index of that match (half of the array index with floor) altNameMatch = Math.floor(tnlix/2); break; } } if (altNameMatch > -1) {break;} // break from the rest of the alts if a match found } } // If a match was found: if ( nameMatch || altNameMatch > -1 ) { dupeIDList.push(testVenueAtt.id); // Add the item to the list of matches _dupeLayer.setVisibility(true); // If anything found, make visible the dupe layer if (nameMatch) { labelText = testVenueAtt.name; // Pull duplicate name } else { labelText = testVenueAtt.aliases[altNameMatch] + ' (Alt)'; // Pull duplicate alt name } phlogdev('Possible duplicate found. WME place: ' + itemName + ' / Nearby place: ' + labelText); // Reformat the name into multiple lines based on length var startIX=0, endIX=0, labelTextBuild = [], maxLettersPerLine = Math.round(2*Math.sqrt(labelText.replace(/ /g,'').length/2)); maxLettersPerLine = Math.max(maxLettersPerLine,4); while (endIX !== -1) { endIX = labelText.indexOf(' ', endIX+1); if (endIX - startIX > maxLettersPerLine) { labelTextBuild.push( labelText.substr(startIX,endIX-startIX) ); startIX = endIX+1; } } labelTextBuild.push( labelText.substr(startIX) ); // Add last line labelTextReformat = labelTextBuild.join('\n'); // Add photo icons if (testVenueAtt.images.length > 0 ) { labelTextReformat = labelTextReformat + ' '; for (var phix=0; phix<testVenueAtt.images.length; phix++) { if (phix===3) { labelTextReformat = labelTextReformat + '+'; break; } //labelTextReformat = labelTextReformat + '\u25A3'; // add photo icons labelTextReformat = labelTextReformat + '\u25A3'; // add photo icons } } pt = venueList[venix].geometry.getCentroid(); if ( !mapExtent.containsLonLat(pt.toLonLat()) ) { outOfExtent = true; } minLat = Math.min(minLat, pt.y); minLon = Math.min(minLon, pt.x); maxLat = Math.max(maxLat, pt.y); maxLon = Math.max(maxLon, pt.x); textFeature = new OL.Feature.Vector( pt, {labelText: labelTextReformat, fontColor: '#fff', strokeColor: labelColorList[labelColorIX%labelColorList.length], labelAlign: 'cm', pointRadius: 25 , dupeID: testVenueAtt.id } ); labelFeatures.push(textFeature); //_dupeLayer.addFeatures(labelFeatures); dupeNames.push(labelText); } labelColorIX++; } } } } } // Add a marker for the working place point if any dupes were found if (dupeIDList.length>1) { pt = item.geometry.getCentroid(); if ( !mapExtent.containsLonLat(pt.toLonLat()) ) { outOfExtent = true; } minLat = Math.min(minLat, pt.y); minLon = Math.min(minLon, pt.x); maxLat = Math.max(maxLat, pt.y); maxLon = Math.max(maxLon, pt.x); // Add photo icons var currentLabel = 'Current'; if (item.attributes.images.length > 0 ) { for (var ciix=0; ciix<item.attributes.images.length; ciix++) { currentLabel = currentLabel + ' '; if (ciix===3) { currentLabel = currentLabel + '+'; break; } currentLabel = currentLabel + '\u25A3'; // add photo icons } } textFeature = new OL.Feature.Vector( pt, {labelText: currentLabel, fontColor: '#fff', strokeColor: '#fff', labelAlign: 'cm', pointRadius: 25 , dupeID: item.attributes.id} ); labelFeatures.push(textFeature); _dupeLayer.addFeatures(labelFeatures); } if (recenterOption && dupeNames.length>0 && outOfExtent) { // then rebuild the extent to include the duplicate var padMult = 1.0; mapExtent.left = minLon - (padFrac*padMult) * (maxLon-minLon); mapExtent.right = maxLon + (padFrac*padMult) * (maxLon-minLon); mapExtent.bottom = minLat - (padFrac*padMult) * (maxLat-minLat); mapExtent.top = maxLat + (padFrac*padMult) * (maxLat-minLat); W.map.zoomToExtent(mapExtent); } return [dupeNames, overlappingFlag]; } // END Dupefinder function // On selection of new item: function checkSelection() { let venue = getSelectedVenue(); if (venue && venue.isApproved()) { displayRunButton(); getPanelFields(); if ( $('#WMEPH-EnableCloneMode').prop('checked') ) { displayCloneButton(); } if ((localStorage.getItem('WMEPH-AutoRunOnSelect') === '1') && venue.arePropertiesEditable()) { setTimeout(harmonizePlace,200); } for (var dvtix=0; dvtix<dupeIDList.length; dvtix++) { if (venue.attributes.id === dupeIDList[dvtix]) { // If the user selects a place in the dupe list, don't clear the labels yet return; } } } // If the selection is anything else, clear the labels _dupeLayer.destroyFeatures(); _dupeLayer.setVisibility(false); } // END checkSelection function // Functions to infer address from nearby segments function WMEPH_inferAddress(MAX_RECURSION_DEPTH) { var distanceToSegment, foundAddresses = [], i, // Ignore pedestrian boardwalk, stairways, runways, and railroads IGNORE_ROAD_TYPES = [10, 16, 18, 19], inferredAddress = { country: null, city: null, state: null, street: null }, //MAX_RECURSION_DEPTH = 8, n, orderedSegments = [], segments = W.model.segments.getObjectArray(), stopPoint; let venue = getSelectedVenue(); var findClosestNode = function () { var closestSegment = orderedSegments[0].segment, distanceA, distanceB, nodeA = W.model.nodes.getObjectById(closestSegment.attributes.fromNodeID), nodeB = W.model.nodes.getObjectById(closestSegment.attributes.toNodeID); if (nodeA && nodeB) { var pt = stopPoint.getPoint ? stopPoint.getPoint() : stopPoint; distanceA = pt.distanceTo(nodeA.attributes.geometry); distanceB = pt.distanceTo(nodeB.attributes.geometry); return distanceA < distanceB ? nodeA.attributes.id : nodeB.attributes.id; } }; var findConnections = function (startingNodeID, recursionDepth) { var connectedSegments, k, newNode; // Limit search depth to avoid problems. if (recursionDepth > MAX_RECURSION_DEPTH) { //console.debug('Max recursion depth reached'); return; } // Populate variable with segments connected to starting node. connectedSegments = _.where(orderedSegments, { fromNodeID: startingNodeID }); connectedSegments = connectedSegments.concat(_.where(orderedSegments, { toNodeID: startingNodeID })); //console.debug('Looking for connections at node ' + startingNodeID); // Check connected segments for address info. for (k in connectedSegments) { if (connectedSegments.hasOwnProperty(k)) { if (hasStreetName(connectedSegments[k].segment)) { // Address found, push to array. /* console.debug('Address found on connnected segment ' + connectedSegments[k].segment.attributes.id + '. Recursion depth: ' + recursionDepth); */ foundAddresses.push({ depth: recursionDepth, distance: connectedSegments[k].distance, segment: connectedSegments[k].segment }); break; } else { // If not found, call function again starting from the other node on this segment. //console.debug('Address not found on connected segment ' + connectedSegments[k].segment.attributes.id); newNode = connectedSegments[k].segment.attributes.fromNodeID === startingNodeID ? connectedSegments[k].segment.attributes.toNodeID : connectedSegments[k].segment.attributes.fromNodeID; findConnections(newNode, recursionDepth + 1); } } } }; var getFCRank = function (FC) { var typeToFCRank = { 3: 0, // freeway 6: 1, // major 7: 2, // minor 2: 3, // primary 1: 4, // street 20: 5, // PLR 8: 6 // dirt }; if (FC && !isNaN(FC)) { return typeToFCRank[FC] || 100; } }; var hasStreetName = function (segment) { return segment && segment.type === 'segment' && segment.getAddress().getStreetName() !== 'No street'; }; // phlogdev('No address data, gathering ', 2); // Make sure a place is selected and segments are loaded. if (!(venue && segments.length)) { return; } if (venue.isPoint()) { stopPoint = venue.geometry; } else { var entryExitPoints = venue.attributes.entryExitPoints; if (entryExitPoints.length) { stopPoint = entryExitPoints[0]; } else { stopPoint = venue.geometry.getCentroid(); } } // Go through segment array and calculate distances to segments. for (i = 0, n = segments.length; i < n; i++) { // Make sure the segment is not an ignored roadType. if (IGNORE_ROAD_TYPES.indexOf(segments[i].attributes.roadType) === -1) { distanceToSegment = (stopPoint.getPoint ? stopPoint.getPoint() : stopPoint).distanceTo(segments[i].geometry); // Add segment object and its distanceTo to an array. orderedSegments.push({ distance: distanceToSegment, fromNodeID: segments[i].attributes.fromNodeID, segment: segments[i], toNodeID: segments[i].attributes.toNodeID }); } } // Sort the array with segments and distance. orderedSegments = _.sortBy(orderedSegments, 'distance'); // Check closest segment for address first. if (hasStreetName(orderedSegments[0].segment)) { inferredAddress = orderedSegments[0].segment.getAddress(); } else { // If address not found on closest segment, try to find address through branching method. findConnections(findClosestNode(), 1); if (foundAddresses.length > 0) { // If more than one address found at same recursion depth, look at FC of segments. if (foundAddresses.length > 1) { _.each(foundAddresses, function (element) { element.fcRank = getFCRank( element.segment.attributes.roadType); }); foundAddresses = _.sortBy(foundAddresses, 'fcRank'); foundAddresses = _.filter(foundAddresses, { fcRank: foundAddresses[0].fcRank }); } // If multiple segments with same FC, Use address from segment with address that is closest by connectivity. if (foundAddresses.length > 1) { foundAddresses = _.sortBy(foundAddresses, 'depth'); foundAddresses = _.filter(foundAddresses, { depth: foundAddresses[0].depth }); } // If more than one of the closest segments by connectivity has the same FC, look for // closest segment geometrically. if (foundAddresses.length > 1) { foundAddresses = _.sortBy(foundAddresses, 'distance'); } console.debug(foundAddresses[0].streetName, foundAddresses[0].depth); inferredAddress = foundAddresses[0].segment.getAddress(); } else { // Default to closest if branching method fails. // Go through sorted segment array until a country, state, and city have been found. var closestElem = _.find(orderedSegments, function (element) {return hasStreetName(element.segment); }); inferredAddress = closestElem ? closestElem.segment.getAddress() || inferredAddress : inferredAddress; } } return inferredAddress; } // END inferAddress function /** * Updates the address for a place. * @param feature {WME Venue Object} The place to update. * @param address {Object} An object containing the country, state, city, and street * @param actions {Array of actions} Optional. If performing multiple actions at once. * objects. */ function updateAddress(feature, address, actions) { var newAttributes; if (feature && address) { newAttributes = { countryID: address.country.id, stateID: address.state.id, cityName: address.city.attributes.name, emptyCity: address.city.hasName() ? null : true, streetName: address.street.name, emptyStreet: address.street.isEmpty ? true : null }; var action = new UpdateFeatureAddress(feature, newAttributes); if(actions) { actions.push(action); } else { W.model.actionManager.add(action); } phlogdev('Address inferred and updated'); } } // END updateAddress function // Build a Google search url based on place name and address function buildGLink(searchName,addr,HN) { var searchHN = '', searchStreet = '', searchCity = ''; searchName = searchName.replace(/\//g, ' '); if ('string' === typeof addr.street.name && addr.street.name !== null && addr.street.name !== '') { searchStreet = addr.street.name + ', '; } searchStreet = searchStreet.replace(/CR-/g, 'County Rd '); searchStreet = searchStreet.replace(/SR-/g, 'State Hwy '); searchStreet = searchStreet.replace(/US-/g, 'US Hwy '); searchStreet = searchStreet.replace(/ CR /g, ' County Rd '); searchStreet = searchStreet.replace(/ SR /g, ' State Hwy '); searchStreet = searchStreet.replace(/ US /g, ' US Hwy '); searchStreet = searchStreet.replace(/$CR /g, 'County Rd '); searchStreet = searchStreet.replace(/$SR /g, 'State Hwy '); searchStreet = searchStreet.replace(/$US /g, 'US Hwy '); if ('string' === typeof HN && searchStreet !== '') { searchHN = HN + ' '; } if ('string' === typeof addr.city.attributes.name && addr.city.attributes.name !== '') { searchCity = addr.city.attributes.name + ', '; } searchName = searchName + (searchName ? ', ' : '') + searchHN + searchStreet + searchCity + addr.state.name; return 'http://www.google.com/search?q=' + encodeURIComponent(searchName); } // END buildGLink function // WME Category translation from Natural language to object language (Bank / Financial --> BANK_FINANCIAL) function catTranslate(natCategories) { var catNameUpper = natCategories.trim().toUpperCase(); if (_CATEGORY_LOOKUP.hasOwnProperty(catNameUpper)) { return _CATEGORY_LOOKUP[catNameUpper]; } // if the category doesn't translate, then pop an alert that will make a forum post to the thread // Generally this means the category used in the PNH sheet is not close enough to the natural language categories used inside the WME translations if (confirm('WMEPH: Category Error!\nClick OK to report this error') ) { reportError({ subject: 'WMEPH Bug report: no tns', message: 'Error report: Category "' + natCategories + '" was not found in the PNH categories sheet.' }); } return 'ERROR'; } // END catTranslate function // compares two arrays to see if equal, regardless of order function matchSets(array1, array2) { if (array1.length !== array2.length) {return false;} // compare lengths for (var i = 0; i < array1.length; i++) { if (array2.indexOf(array1[i]) === -1) { return false; } } return true; } // function that checks if all elements of target are in array:source function containsAll(source, target) { if (typeof(target) === 'string') { target = [target]; } // if a single string, convert to an array for (var ixx = 0; ixx < target.length; ixx++) { if ( source.indexOf(target[ixx]) === -1 ) { return false; } } return true; } // function that checks if any element of target are in source function containsAny(source, target) { if (typeof(source) === 'string') { source = [source]; } // if a single string, convert to an array if (typeof(target) === 'string') { target = [target]; } // if a single string, convert to an array return source.some(tt => target.indexOf(tt) > -1); } // Function that inserts a string or a string array into another string array at index ix and removes any duplicates function insertAtIX(array1, array2, ix) { // array1 is original string, array2 is the inserted string, at index ix var arrayNew = array1.slice(0); // slice the input array so it doesn't change if (typeof(array2) === 'string') { array2 = [array2]; } // if a single string, convert to an array if (typeof(array2) === 'object') { // only apply to inserted arrays var arrayTemp = arrayNew.splice(ix); // split and hold the first part arrayNew.push.apply(arrayNew, array2); // add the insert arrayNew.push.apply(arrayNew, arrayTemp); // add the tail end of original } return uniq(arrayNew); // remove any duplicates (so the function can be used to move the position of a string) } // Function to remove unnecessary aliases function removeSFAliases(nName, nAliases) { var newAliasesUpdate = []; nName = nName.toUpperCase().replace(/'/g,'').replace(/-/g,' ').replace(/\/ /g,' ').replace(/ \//g,' ').replace(/ {2,}/g,' '); for (var naix=0; naix<nAliases.length; naix++) { if ( !nName.startsWith( nAliases[naix].toUpperCase().replace(/'/g,'').replace(/-/g,' ').replace(/\/ /g,' ').replace(/ \//g,' ').replace(/ {2,}/g,' ') ) ) { newAliasesUpdate.push(nAliases[naix]); } else { bannButt.sfAliases = new Flag.SFAliases(); } } return newAliasesUpdate; } function initSettingsCheckbox(settingID) { //Associate click event of new checkbox to call saveSettingToLocalStorage with proper ID $('#' + settingID).click(function() {saveSettingToLocalStorage(settingID);}); //Load Setting for Local Storage, if it doesn't exist set it to NOT checked. //If previously set to 1, then trigger "click" event. if (!localStorage.getItem(settingID)) { //phlogdev(settingID + ' not found.'); } else if (localStorage.getItem(settingID) === '1') { $('#' + settingID).prop('checked', true); } } // This routine will create a checkbox in the #PlaceHarmonizer tab and will load the setting // settingID: The #id of the checkbox being created. // textDescription: The description of the checkbox that will be use function createSettingsCheckbox($div, settingID, textDescription) { let $checkbox = $('<input>', {type:'checkbox', id:settingID}); $div.append( $('<div>', {class:'controls-container'}).css({paddingTop:'2px'}).append( $checkbox, $('<label>', {for:settingID}).text(textDescription).css({whiteSpace:'pre-line'}) ) ); return $checkbox; } function onKBShortcutModifierKeyClick() { let $modifKeyCheckbox = $('#WMEPH-KBSModifierKey'); let $shortcutInput = $('#WMEPH-KeyboardShortcut'); let $warn = $('#PlaceHarmonizerKBWarn'); let modifKeyNew; modifKeyNew = $modifKeyCheckbox.prop('checked') ? 'Ctrl+' : 'Alt+'; shortcutParse = parseKBSShift($shortcutInput.val()); $warn.empty(); // remove any warning shortcut.remove(modifKey + shortcutParse); modifKey = modifKeyNew; shortcut.add(modifKey + shortcutParse, function() { harmonizePlace(); }); $('#PlaceHarmonizerKBCurrent').empty().append('<span style="font-weight:bold">Current shortcut: '+modifKey+shortcutParse+'</span>'); } function onKBShortcutChange() { let keyId = 'WMEPH-KeyboardShortcut'; let $warn = $('#PlaceHarmonizerKBWarn'); let $key = $('#' + keyId); let oldKey = localStorage.getItem(keyId); let newKey = $key.val(); $warn.empty(); // remove old warning if (newKey.match(/^[a-z]{1}$/i) !== null) { // If a single letter... shortcutParse = parseKBSShift(oldKey); let shortcutParseNew = parseKBSShift(newKey); shortcut.remove(modifKey + shortcutParse); shortcutParse = shortcutParseNew; shortcut.add(modifKey + shortcutParse, function() { harmonizePlace(); }); $(localStorage.setItem(keyId, newKey) ); $('#PlaceHarmonizerKBCurrent').empty().append('<span style="font-weight:bold">Current shortcut: '+modifKey+shortcutParse+'</span>'); } else { // if not a letter then reset and flag $key.val(oldKey); $warn.append('<p style="color:red">Only letters are allowed<p>'); } } function setCheckedByDefault(id) { if (localStorage.getItem(id) === null) { localStorage.setItem(id, '1'); } } // User pref for KB Shortcut: function initShortcutKey() { let $current = $('#PlaceHarmonizerKBCurrent'); let defaultShortcutKey = _IS_DEV_VERSION ? 'S' : 'A'; let shortcutID = 'WMEPH-KeyboardShortcut'; let shortcutKey = localStorage.getItem(shortcutID); let $shortcutInput = $('#'+shortcutID); // Set local storage to default if none if (shortcutKey === null || !/^[a-z]{1}$/i.test(shortcutKey)) { localStorage.setItem(shortcutID, defaultShortcutKey); shortcutKey = defaultShortcutKey; } $shortcutInput.val(shortcutKey); if ( localStorage.getItem('WMEPH-KBSModifierKey') === '1' ) { // Change modifier key code if checked modifKey = 'Ctrl+'; } shortcutParse = parseKBSShift(shortcutKey); if (!_initAlreadyRun) shortcut.add(modifKey + shortcutParse, function() { harmonizePlace(); }); $current.empty().append('<span style="font-weight:bold">Current shortcut: '+modifKey+shortcutParse+'</span>'); $('#WMEPH-KBSModifierKey').click(onKBShortcutModifierKeyClick); // Upon change of the KB letter: $shortcutInput.change(onKBShortcutChange); } function onWLMergeClick() { let $wlToolsMsg = $('#PlaceHarmonizerWLToolsMsg'); let $wlInput = $('#WMEPH-WLInput'); $wlToolsMsg.empty(); if ($wlInput.val() === 'resetWhitelist') { if (confirm('***Do you want to reset all Whitelist data?\nClick OK to erase.') ) { // if the category doesn't translate, then pop an alert that will make a forum post to the thread venueWhitelist = { '1.1.1': { Placeholder: { } } }; // Populate with a dummy place saveWL_LS(true); } } else { // try to merge uncompressed WL data WLSToMerge = validateWLS($('#WMEPH-WLInput').val()); if (WLSToMerge) { phlog('Whitelists merged!'); venueWhitelist = mergeWL(venueWhitelist,WLSToMerge); saveWL_LS(true); $wlToolsMsg.append('<p style="color:green">Whitelist data merged<p>'); $wlInput.val(''); } else { // try compressed WL WLSToMerge = validateWLS( LZString.decompressFromUTF16($('#WMEPH-WLInput').val()) ); if (WLSToMerge) { phlog('Whitelists merged!'); venueWhitelist = mergeWL(venueWhitelist,WLSToMerge); saveWL_LS(true); $wlToolsMsg.append('<p style="color:green">Whitelist data merged<p>'); $wlInput.val(''); } else { $wlToolsMsg.append('<p style="color:red">Invalid Whitelist data<p>'); } } } } function onWLPullClick() { $('#WMEPH-WLInput').val( LZString.decompressFromUTF16(localStorage.getItem(WLlocalStoreNameCompressed)) ); $('#PlaceHarmonizerWLToolsMsg').empty().append('<p style="color:green">To backup the data, copy & paste the text in the box to a safe location.<p>'); localStorage.setItem('WMEPH_WLAddCount', 1); } function onWLStatsClick() { let currWLData = JSON.parse( LZString.decompressFromUTF16( localStorage.getItem(WLlocalStoreNameCompressed) ) ); let countryWL = {}; let stateWL = {}; let entries = Object.keys(currWLData).filter(key => key !== '1.1.1'); $('#WMEPH-WLInputBeta').val(''); entries.forEach(venueKey => { let country = currWLData[venueKey].country || 'None'; let state = currWLData[venueKey].state || 'None'; countryWL[country] = countryWL[country] + 1 || 1; stateWL[state] = stateWL[state] + 1 || 1; }); let countryString = ''; for (var countryKey in countryWL) { countryString = countryString + '<br>' + countryKey + ': ' + countryWL[countryKey]; } let stateString = ''; for (var stateKey in stateWL) { stateString = stateString + '<br>' + stateKey + ': ' + stateWL[stateKey]; } $('#PlaceHarmonizerWLToolsMsg').empty().append('<p style="color:black">Number of WL places: '+ entries.length +'</p><p><span style="font-weight:bold;"><u>States</u></span>'+ stateString +'</p><p><span style="font-weight:bold;"><u>Countries</u></span>'+ countryString + '<p>'); //localStorage.setItem('WMEPH_WLAddCount', 1); } function onWLStateFilterClick() { let $wlToolsMsg = $('#PlaceHarmonizerWLToolsMsg'); let $wlInput = $('#WMEPH-WLInput'); let stateToRemove = $wlInput.val().trim(); let msg = ''; if ( stateToRemove.length < 2 ) { msg = '<p style="color:red">Invalid state. Enter the state name in the "Whitelist string" box above, exactly as it appears in the Stats output.<p>'; } else { var currWLData, venueToRemove = []; currWLData = JSON.parse( LZString.decompressFromUTF16( localStorage.getItem(WLlocalStoreNameCompressed) ) ); Object.keys(currWLData).filter(venueKey => venueKey !== '1.1.1').forEach(venueKey => { if ( currWLData[venueKey].state === stateToRemove || (!currWLData[venueKey].state && stateToRemove === 'None') ) { venueToRemove.push(venueKey); } }); if (venueToRemove.length > 0) { if (localStorage.WMEPH_WLAddCount === '1') { if (confirm('Are you sure you want to clear all whitelist data for '+stateToRemove+'? This CANNOT be undone. Press OK to delete, cancel to preserve the data.') ) { backupWL_LS(true); for (var ixwl=0; ixwl<venueToRemove.length; ixwl++) { delete venueWhitelist[venueToRemove[ixwl]]; } saveWL_LS(true); msg = '<p style="color:green">'+venueToRemove.length+' items removed from WL<p>'; $wlInput.val(''); } else { msg = '<p style="color:blue">No changes made<p>'; } } else { msg = '<p style="color:red">Please backup your WL using the Pull button before removing state data<p>'; } } else { msg = '<p style="color:red">No data for that state. Use the state name exactly as listed in the Stats<p>'; } } $wlToolsMsg.empty().append(msg); } function onWLShareClick() { window.open('https://docs.google.com/forms/d/1k_5RyOq81Fv4IRHzltC34kW3IUbXnQqDVMogwJKFNbE/viewform?entry.1173700072='+_USER.name); } // settings tab function initWmephTab() { // Enable certain settings by default if not set by the user: setCheckedByDefault('WMEPH-ColorHighlighting'); setCheckedByDefault('WMEPH-ExcludePLADupes'); setCheckedByDefault('WMEPH-DisablePLAExtProviderCheck'); // Initialize settings checkboxes initSettingsCheckbox('WMEPH-WebSearchNewTab'); initSettingsCheckbox('WMEPH-DisableDFZoom'); initSettingsCheckbox('WMEPH-EnableIAZoom'); initSettingsCheckbox('WMEPH-HidePlacesWiki'); initSettingsCheckbox('WMEPH-HideReportError'); initSettingsCheckbox('WMEPH-HideServicesButtons'); initSettingsCheckbox('WMEPH-HidePURWebSearch'); initSettingsCheckbox('WMEPH-ExcludePLADupes'); initSettingsCheckbox('WMEPH-ShowPLAExitWhileClosed'); if (_USER.isDevUser || _USER.isBetaUser || _USER.rank >= 2) { initSettingsCheckbox('WMEPH-DisablePLAExtProviderCheck'); initSettingsCheckbox('WMEPH-EnableServices'); initSettingsCheckbox('WMEPH-AddAddresses'); initSettingsCheckbox('WMEPH-EnableCloneMode'); initSettingsCheckbox('WMEPH-AutoLockRPPs'); initSettingsCheckbox('WMEPH-AutoRunOnSelect'); } initSettingsCheckbox('WMEPH-ColorHighlighting'); initSettingsCheckbox('WMEPH-DisableHoursHL'); initSettingsCheckbox('WMEPH-DisableRankHL'); initSettingsCheckbox('WMEPH-DisableWLHL'); initSettingsCheckbox('WMEPH-PLATypeFill'); initSettingsCheckbox('WMEPH-KBSModifierKey'); if (_USER.isDevUser) { initSettingsCheckbox('WMEPH-RegionOverride'); } // Turn this setting on one time. if (!_initAlreadyRun) { var runOnceDefaultIgnorePlaGoogleLinkChecks = localStorage.getItem('WMEPH-runOnce-defaultToOff-plaGoogleLinkChecks'); if (!runOnceDefaultIgnorePlaGoogleLinkChecks) { var $chk = $('#WMEPH-DisablePLAExtProviderCheck'); if (!$chk.prop('checked')) { $chk.trigger('click'); } } localStorage.setItem('WMEPH-runOnce-defaultToOff-plaGoogleLinkChecks', true); } initShortcutKey(); if (localStorage.getItem('WMEPH_WLAddCount') === null) { localStorage.setItem('WMEPH_WLAddCount', 2); // Counter to remind of WL backups } // WL button click events $('#WMEPH-WLMerge').click(onWLMergeClick); $('#WMEPH-WLPull').click(onWLPullClick); $('#WMEPH-WLStats').click(onWLStatsClick); $('#WMEPH-WLStateFilter').click(onWLStateFilterClick); $('#WMEPH-WLShare').click(onWLShareClick); // Color highlighting $('#WMEPH-ColorHighlighting').click(bootstrapWMEPH_CH); $('#WMEPH-DisableHoursHL').click(bootstrapWMEPH_CH); $('#WMEPH-DisableRankHL').click(bootstrapWMEPH_CH); $('#WMEPH-DisableWLHL').click(bootstrapWMEPH_CH); $('#WMEPH-PLATypeFill').click(() => applyHighlightsTest(W.model.venues.getObjectArray())); _initAlreadyRun = true; } function addWmephTab() { // Set up the CSS GM_addStyle(_CSS_ARRAY.join('\n')); var $container = $('<div id="wmephtab" class="active" style="padding-top: 5px;">'); var $navTabs = $('<ul class="nav nav-tabs"><li class="active"><a data-toggle="tab" href="#sidepanel-harmonizer">Harmonize</a></li>' + '<li><a data-toggle="tab" href="#sidepanel-highlighter">HL \/ Scan</a></li>' + '<li><a data-toggle="tab" href="#sidepanel-wltools">WL Tools</a></li></ul>'); var $tabContent = $('<div class="tab-content" style="padding:5px;">'); var $harmonizerTab = $('<div class="tab-pane active" id="sidepanel-harmonizer"></div>'); var $highlighterTab = $('<div class="tab-pane" id="sidepanel-highlighter"></div>'); var $wlToolsTab = $('<div class="tab-pane" id="sidepanel-wltools"></div>'); $tabContent.append($harmonizerTab, $highlighterTab, $wlToolsTab); $container.append($navTabs, $tabContent); //Harmonizer settings createSettingsCheckbox($harmonizerTab, 'WMEPH-WebSearchNewTab','Open URL & Search Results in new tab instead of new window'); createSettingsCheckbox($harmonizerTab, 'WMEPH-DisableDFZoom','Disable zoom & center for duplicates'); createSettingsCheckbox($harmonizerTab, 'WMEPH-EnableIAZoom','Enable zoom & center for places with no address'); createSettingsCheckbox($harmonizerTab, 'WMEPH-HidePlacesWiki','Hide "Places Wiki" button in results banner'); createSettingsCheckbox($harmonizerTab, 'WMEPH-HideReportError','Hide "Report script error" button in results banner'); createSettingsCheckbox($harmonizerTab, 'WMEPH-HideServicesButtons','Hide services buttons in results banner'); createSettingsCheckbox($harmonizerTab, 'WMEPH-HidePURWebSearch','Hide "Web Search" button on PUR popups'); createSettingsCheckbox($harmonizerTab, 'WMEPH-ExcludePLADupes','Exclude parking lots when searching for duplicate places'); createSettingsCheckbox($harmonizerTab, 'WMEPH-ShowPLAExitWhileClosed','Always ask if cars can exit parking lots'); if (_USER.isDevUser || _USER.isBetaUser || _USER.rank >= 2) { createSettingsCheckbox($harmonizerTab, 'WMEPH-DisablePLAExtProviderCheck','Disable check for "Google place link" on Parking Lot Areas'); createSettingsCheckbox($harmonizerTab, 'WMEPH-EnableServices','Enable automatic addition of common services'); createSettingsCheckbox($harmonizerTab, 'WMEPH-AddAddresses','Add detected address fields to places with no address'); createSettingsCheckbox($harmonizerTab, 'WMEPH-EnableCloneMode','Enable place cloning tools'); createSettingsCheckbox($harmonizerTab, 'WMEPH-AutoLockRPPs','Lock residential place points to region default'); createSettingsCheckbox($harmonizerTab, 'WMEPH-AutoRunOnSelect','Automatically run the script when selecting a place'); } $harmonizerTab.append('<hr align="center" width="90%">'); // Add Letter input box var $phShortcutDiv = $('<div id="PlaceHarmonizerKB">'); $phShortcutDiv.append('<div id="PlaceHarmonizerKBWarn"></div>Shortcut Letter (a-Z): <input type="text" maxlength="1" id="WMEPH-KeyboardShortcut" style="width: 30px;padding-left:8px"><div id="PlaceHarmonizerKBCurrent"></div>'); createSettingsCheckbox($phShortcutDiv, 'WMEPH-KBSModifierKey', 'Use Ctrl instead of Alt'); // Add Alt-->Ctrl checkbox if (_USER.isDevUser) { // Override script regionality (devs only) $phShortcutDiv.append('<hr align="center" width="90%"><p>Dev Only Settings:</p>'); createSettingsCheckbox($phShortcutDiv, 'WMEPH-RegionOverride','Disable Region Specificity'); } $harmonizerTab.append($phShortcutDiv); $harmonizerTab.append('<hr align="center" width="95%"><p><a href="' + _URLS.placesWiki + '" target="_blank">Open the WME Places Wiki page</a><p><a href="' + _URLS.forum + '" target="_blank">Submit script feedback & suggestions</a></p><hr align="center" width="95%">Recent updates:<ul>'+ _WHATS_NEW_LIST.map(i => '<li>'+i+'</li>').join('')+'</ul>'); // Highlighter settings $highlighterTab.append('<p>Highlighter Settings:</p>'); createSettingsCheckbox($highlighterTab, 'WMEPH-ColorHighlighting','Enable color highlighting of map to indicate places needing work'); createSettingsCheckbox($highlighterTab, 'WMEPH-DisableHoursHL','Disable highlighting for missing hours'); createSettingsCheckbox($highlighterTab, 'WMEPH-DisableRankHL','Disable highlighting for places locked above your rank'); createSettingsCheckbox($highlighterTab, 'WMEPH-DisableWLHL','Disable Whitelist highlighting (shows all missing info regardless of WL)'); createSettingsCheckbox($highlighterTab, 'WMEPH-PLATypeFill','Fill parking lots based on type (public=blue, restricted=yellow, private=red)'); if (_USER.isDevUser || _USER.isBetaUser || _USER.rank >= 3) { //createSettingsCheckbox($highlighterTab 'WMEPH-UnlockedRPPs','Highlight unlocked residential place points'); } // Scanner settings //$highlighterTab.append('<hr align="center" width="90%">'); //$highlighterTab.append('<p>Scanner Settings (coming !soon)</p>'); //createSettingsCheckbox($highlighterTab, 'WMEPH-PlaceScanner','Placeholder, under development!'); // Whitelisting settings let phWLContentHtml = $('<div id="PlaceHarmonizerWLTools">Whitelist string: <input onClick="this.select();" type="text" id="WMEPH-WLInput" style="width:100%;padding-left:1px;display:block">'+ '<div style="margin-top:3px;">'+ '<input class="btn btn-success btn-xs wmeph-fat-btn" id="WMEPH-WLMerge" title="Merge the string into your existing Whitelist" type="button" value="Merge">'+ '<input class="btn btn-success btn-xs wmeph-fat-btn" id="WMEPH-WLPull" title="Pull your existing Whitelist for backup or sharing" type="button" value="Pull">'+ '<input class="btn btn-success btn-xs wmeph-fat-btn" id="WMEPH-WLShare" title="Share your Whitelist to a public Google sheet" type="button" value="Share your WL">'+ '</div>'+ '<div style="margin-top:12px;">'+ '<input class="btn btn-info btn-xs wmeph-fat-btn" id="WMEPH-WLStats" title="Display WL stats" type="button" value="Stats">'+ '<input class="btn btn-danger btn-xs wmeph-fat-btn" id="WMEPH-WLStateFilter" title="Remove all WL items for a state. Enter the state in the \'Whitelist string\' box." type="button" value="Remove data for 1 State">'+ '</div>'+ '</div>'+ '<div id="PlaceHarmonizerWLToolsMsg" style="margin-top:10px;"></div>'); $wlToolsTab.append(phWLContentHtml); new WazeWrap.Interface.Tab('WMEPH' + (_IS_DEV_VERSION ? '-β' : ''), $container.html(), initWmephTab, null); } function createCloneCheckbox(divID, settingID, textDescription) { $('#' + divID).append('<input type="checkbox" id="' + settingID + '">'+ textDescription +'</input>  '); $('#' + settingID).click(() => saveSettingToLocalStorage(settingID)); if (localStorage.getItem(settingID) === '1') { $('#' + settingID).trigger('click'); } } //Function to add Shift+ to upper case KBS function parseKBSShift(kbs) { return (/^[A-Z]{1}$/g.test(kbs) ? 'Shift+' : '') + kbs; } // Save settings prefs function saveSettingToLocalStorage(settingID) { localStorage.setItem(settingID, $('#' + settingID).prop('checked') ? '1' : '0'); } // This function validates that the inputted text is a JSON function validateWLS(jsonString) { try { var objTry = JSON.parse(jsonString); if (objTry && typeof objTry === 'object' && objTry !== null) { return objTry; } } catch (e) { } return false; } // This function merges and updates venues from object vWL_2 into vWL_1 function mergeWL(vWL_1,vWL_2) { var venueKey, WLKey, vWL_1_Venue, vWL_2_Venue; for (venueKey in vWL_2) { if (vWL_2.hasOwnProperty(venueKey)) { // basic filter if (vWL_1.hasOwnProperty(venueKey)) { // if the vWL_2 venue is in vWL_1, then update any keys vWL_1_Venue = vWL_1[venueKey]; vWL_2_Venue = vWL_2[venueKey]; for (WLKey in vWL_2_Venue) { // loop thru the venue WL keys if (vWL_2_Venue.hasOwnProperty(WLKey) && vWL_2_Venue[WLKey].active) { // Only update if the vWL_2 key is active if ( vWL_1_Venue.hasOwnProperty(WLKey) && vWL_1_Venue[WLKey].active ) { // if the key is in the vWL_1 venue and it is active, then push any array data onto the key if (vWL_1_Venue[WLKey].hasOwnProperty('WLKeyArray')) { vWL_1[venueKey][WLKey].WLKeyArray = insertAtIX(vWL_1[venueKey][WLKey].WLKeyArray,vWL_2[venueKey][WLKey].WLKeyArray,100); } } else { // if the key isn't in the vWL_1 venue, or if it's inactive, then copy the vWL_2 key across vWL_1[venueKey][WLKey] = vWL_2[venueKey][WLKey]; } } } // END subLoop for venue keys } else { // if the venue doesn't exist in vWL_1, then add it vWL_1[venueKey] = vWL_2[venueKey]; } } } return vWL_1; } // Get services checkbox status function getServicesChecks(venue) { var servArrayCheck = []; for (var wsix=0; wsix<WMEServicesArray.length; wsix++) { if (venue.attributes.services.indexOf(WMEServicesArray[wsix]) > -1) { servArrayCheck[wsix] = true; } else { servArrayCheck[wsix] = false; } } return servArrayCheck; } function updateServicesChecks() { let venue = getSelectedVenue(); if (venue) { if (!bannServ) return; var servArrayCheck = getServicesChecks(venue), wsix=0; for (var keys in bannServ) { if (bannServ.hasOwnProperty(keys)) { bannServ[keys].checked = servArrayCheck[wsix]; // reset all icons to match any checked changes bannServ[keys].active = bannServ[keys].active || servArrayCheck[wsix]; // display any manually checked non-active icons wsix++; } } // Highlight 24/7 button if hours are set that way, and add button for all places if ( venue.attributes.openingHours.length === 1 && venue.attributes.openingHours[0].days.length === 7 && /^0?0:00$/.test(venue.attributes.openingHours[0].fromHour) && /^0?0:00$/.test(venue.attributes.openingHours[0].toHour) ) { bannServ.add247.checked = true; } bannServ.add247.active = true; } } // Focus away from the current cursor focus, to set text box changes function blurAll() { var tmp = document.createElement('input'); document.body.appendChild(tmp); tmp.focus(); document.body.removeChild(tmp); } // Pulls the item PL function getItemPL() { // Append a form div if it doesn't exist yet: if ( $('#WMEPH_formDiv').length ===0 ) { var tempDiv = document.createElement('div'); tempDiv.id = 'WMEPH_formDiv'; tempDiv.style.display = 'inline'; $('.WazeControlPermalink').append(tempDiv); } // Return the current PL if ($('.WazeControlPermalink').length === 0) { phlog('Waiting for PL div'); setTimeout(getItemPL, 500); return; } if ( $('.WazeControlPermalink .permalink').attr('href').length > 0 ) { return $('.WazeControlPermalink .permalink').attr('href'); } else if ( $('.WazeControlPermalink').children('.fa-link').length > 0 ) { return $('.WazeControlPermalink').children('.fa-link')[0].href; } return ''; } // Sets up error reporting function reportError(data) { data.preview = 'Preview'; data.attach_sig = 'on'; if (PMUserList.hasOwnProperty('WMEPH') && PMUserList.WMEPH.approvalActive) { data['address_list[u]['+PMUserList.WMEPH.modID+']'] = 'to'; newForumPost('https://www.waze.com/forum/ucp.php?i=pm&mode=compose', data); } else { data.addbbcode20 = 'to'; data.notify = 'on'; newForumPost(_URLS.forum + '#preview', data); } } // END reportError function // Make a populated post on a forum thread function newForumPost(url, data) { var form = document.createElement('form'); form.target = '_blank'; form.action = url; form.method = 'post'; form.style.display = 'none'; for (var k in data) { if (data.hasOwnProperty(k)) { var input; if (k === 'message') { input = document.createElement('textarea'); } else if (k === 'username') { input = document.createElement('username_list'); } else { input = document.createElement('input'); } input.name = k; input.value = data[k]; //input.type = 'hidden'; // 2018-07/10 (mapomatic) Not sure if this is required, but was causing an error when setting on the textarea object. form.appendChild(input); } } document.getElementById('WMEPH_formDiv').appendChild(form); form.submit(); document.getElementById('WMEPH_formDiv').removeChild(form); return true; } // END newForumPost function /** * Updates the geometry of a place. * @param place {Waze venue object} The place to update. * @param newGeometry {OL.Geometry} The new geometry for the place. */ function updateFeatureGeometry(place, newGeometry) { var oldGeometry, model = W.model.venues, wmeUpdateFeatureGeometry = require('Waze/Action/UpdateFeatureGeometry'); if (place && place.CLASS_NAME === 'Waze.Feature.Vector.Landmark' && newGeometry && (newGeometry instanceof OL.Geometry.Point || newGeometry instanceof OL.Geometry.Polygon)) { oldGeometry = place.attributes.geometry; W.model.actionManager.add( new wmeUpdateFeatureGeometry(place, model, oldGeometry, newGeometry)); } } function placeHarmonizer_init() { // For debugging purposes. May be removed when no longer needed. unsafeWindow.PNH_DATA = _PNH_DATA; _USER.ref = W.loginManager.user; _USER.name = _USER.ref.userName; _USER.rank = _USER.ref.normalizedLevel; // get editor's level (actual level) userLanguage = I18n.locale; // Array prototype extensions (for Firefox fix) Array.prototype.toSet = function () { return this.reduce(function (e, t) {return e[t] = !0, e;}, {}); }; Array.prototype.first = function () { return this[0]; }; Array.prototype.isEmpty = function () { return 0 === this.length; }; appendServiceButtonIconCss(); _updatedFields.init(); addPURWebSearchButton(); // Create duplicatePlaceName layer _dupeLayer = W.map.getLayerByUniqueName('__DuplicatePlaceNames'); if (!_dupeLayer) { var lname = 'WMEPH Duplicate Names'; var style = new OL.Style({ label : '${labelText}', labelOutlineColor: '#333', labelOutlineWidth: 3, labelAlign: '${labelAlign}', fontColor: '${fontColor}', fontOpacity: 1.0, fontSize: '20px', labelYOffset: -30, labelXOffset: 0, fontWeight: 'bold', fill: false, strokeColor: '${strokeColor}', strokeWidth: 10, pointRadius: '${pointRadius}' }); _dupeLayer = new OL.Layer.Vector(lname, { displayInLayerSwitcher: false, uniqueName: '__DuplicatePlaceNames', styleMap: new OL.StyleMap(style) }); _dupeLayer.setVisibility(false); W.map.addLayer(_dupeLayer); } if ( localStorage.getItem('WMEPH-featuresExamined') === null ) { localStorage.setItem('WMEPH-featuresExamined', '0'); // Storage for whether the User has pressed the button to look at updates } createObserver(); let xrayMode = localStorage.getItem('WMEPH_xrayMode_enabled') === 'true' ? true : false; WazeWrap.Interface.AddLayerCheckbox('Display', 'WMEPH x-ray mode', xrayMode, toggleXrayMode); if (xrayMode) setTimeout(() => toggleXrayMode(true), 2000); // Give other layers time to load before enabling. // Whitelist initialization if ( validateWLS( LZString.decompressFromUTF16(localStorage.getItem(WLlocalStoreNameCompressed)) ) === false ) { // If no compressed WL string exists if ( validateWLS(localStorage.getItem(WLlocalStoreName)) === false ) { // If no regular WL exists venueWhitelist = { '1.1.1': { Placeholder: { } } }; // Populate with a dummy place saveWL_LS(false); saveWL_LS(true); } else { // if regular WL string exists, then transfer to compressed version localStorage.setItem('WMEPH-OneTimeWLBU', localStorage.getItem(WLlocalStoreName)); loadWL_LS(false); saveWL_LS(true); alert('Whitelists are being converted to a compressed format. If you have trouble with your WL, please submit an error report.'); } } else { loadWL_LS(true); } if (_USER.name === 'ggrane') { searchResultsWindowSpecs = '"resizable=yes, top='+ Math.round(window.screen.height*0.1) +', left='+ Math.round(window.screen.width*0.3) +', width='+ Math.round(window.screen.width*0.86) +', height='+ Math.round(window.screen.height*0.8) +'"'; } // Settings setup if (!localStorage.getItem(_SETTING_IDS.gLinkWarning)) { // store settings so the warning is only given once localStorage.setItem(_SETTING_IDS.gLinkWarning, '0'); } if (!localStorage.getItem(_SETTING_IDS.sfUrlWarning)) { // store settings so the warning is only given once localStorage.setItem(_SETTING_IDS.sfUrlWarning, '0'); } W.map.events.register('mousemove', W.map, e => errorHandler(() => { WMEPHmousePosition = W.map.getLonLatFromPixel( W.map.events.getMousePosition(e) ); })); // Add zoom shortcut shortcut.add('Control+Alt+Z', () => zoomPlace()); // Add Color Highlighting shortcut shortcut.add('Control+Alt+h', function() { $('#WMEPH-ColorHighlighting').trigger('click'); }); // Add Autorun shortcut if (_USER.name === 'bmtg') { shortcut.add('Control+Alt+u', function() { $('#WMEPH-AutoRunOnSelect').trigger('click'); }); } addWmephTab(); // initialize the settings tab // Event listeners W.selectionManager.events.registerPriority('selectionchanged', this, () => errorHandler(checkSelection)); W.model.venues.on('objectssynced', () => errorHandler(destroyDupeLabels)); W.model.venues.on('objectssynced', e => errorHandler(() => syncWL(e))); W.model.venues.on('objectschanged', () => errorHandler(onObjectsChanged)); // Remove any temporary ID values (ID < 0) from the WL store at startup. var removedWLCount = 0; Object.keys(venueWhitelist).forEach(venueID => { if (venueID < 0) { delete venueWhitelist[venueID]; removedWLCount += 1; } }); if (removedWLCount > 0) { saveWL_LS(true); phlogdev('Removed ' + removedWLCount + ' venues with temporary ID\'s from WL store'); } if (WMEPHbetaList.length === 0 || 'undefined' === typeof WMEPHbetaList) { if (_IS_DEV_VERSION) { alert('Beta user list access issue. Please post in the GHO or PM/DM MapOMatic about this message. Script should still work.'); } _USER.isBetaUser = false; _USER.isDevUser = false; } else { let lcName = _USER.name.toLowerCase(); _USER.isDevUser = WMEPHdevList.indexOf(lcName) > -1; _USER.isBetaUser = WMEPHbetaList.indexOf(lcName) > -1; } if (_USER.isDevUser) { _USER.isBetaUser = true; // dev users are beta users } catTransWaze2Lang = I18n.translations[userLanguage].venues.categories; // pulls the category translations // Split out state-based data let _stateHeaders = _PNH_DATA.states[0].split('|'); ps_state_ix = _stateHeaders.indexOf('ps_state'); ps_state2L_ix = _stateHeaders.indexOf('ps_state2L'); ps_region_ix = _stateHeaders.indexOf('ps_region'); ps_gFormState_ix = _stateHeaders.indexOf('ps_gFormState'); ps_defaultLockLevel_ix = _stateHeaders.indexOf('ps_defaultLockLevel'); //ps_requirePhone_ix = _stateHeaders.indexOf('ps_requirePhone'); //ps_requireURL_ix = _stateHeaders.indexOf('ps_requireURL'); ps_areacode_ix = _stateHeaders.indexOf('ps_areacode'); // Set up Run WMEPH button once place is selected bootstrapRunButton(); /** * Generates highlighting rules and applies them to the map. */ layer = W.map.landmarkLayer; // Setup highlight colors initializeHighlights(); // used for phone reformatting if (!String.plFormat) { String.plFormat = function(format) { var args = Array.prototype.slice.call(arguments, 1); return format.replace(/{(\d+)}/g, function(name, number) { return typeof args[number] !== 'undefined' ? args[number] : null; }); }; } W.model.venues.on('objectschanged', () => errorHandler(() => { if ($('#WMEPH_banner').length > 0) { updateServicesChecks(); assembleServicesBanner(); } })); phlog('Starting Highlighter'); bootstrapWMEPH_CH(); } // END placeHarmonizer_init function function placeHarmonizer_bootstrap() { if ( W && W.loginManager && W.loginManager.user && W.map && WazeWrap.Interface) { placeHarmonizer_init(); } else { phlog('Waiting for WME map and login...'); setTimeout(function () { placeHarmonizer_bootstrap(); }, 200); } } function callAjaxAsync(url) { return new Promise((resolve, reject) => { $.ajax({ type: 'GET', url: url, jsonp: 'callback', data: { alt: 'json-in-script' }, dataType: 'jsonp', success: resolve, error: reject }); }); } function downloadPnhData() { let processData = (response, fieldName) => response.feed.entry.map(entry => entry[fieldName].$t); Promise.all([ callAjaxAsync('https://spreadsheets.google.com/feeds/list/1-f-JTWY5UnBx-rFTa4qhyGMYdHBZWNirUTOgn222zMY/o6q7kx/public/values').then(response => { _PNH_DATA.USA.pnh = processData(response, 'gsx$pnhdata'); _PNH_DATA.USA.pnhNames = makeNameCheckList(_PNH_DATA.USA.pnh); }), callAjaxAsync('https://spreadsheets.google.com/feeds/list/1-f-JTWY5UnBx-rFTa4qhyGMYdHBZWNirUTOgn222zMY/ov3dubz/public/values').then(response => { _PNH_DATA.USA.categories = processData(response, 'gsx$pcdata'); _PNH_DATA.USA.categoryNames = makeCatCheckList(_PNH_DATA.USA.categories); }), callAjaxAsync('https://spreadsheets.google.com/feeds/list/1-f-JTWY5UnBx-rFTa4qhyGMYdHBZWNirUTOgn222zMY/os2g2ln/public/values').then(response => { _PNH_DATA.states = processData(response, 'gsx$psdata'); }), callAjaxAsync('https://spreadsheets.google.com/feeds/list/1TIxQZVLUbAJ8iH6LPTkJsvqFb_DstrHpKsJbv1W1FZs/o4ghhas/public/values').then(response => { _PNH_DATA.CAN.pnh = processData(response, 'gsx$pnhdata'); _PNH_DATA.CAN.pnhNames = makeNameCheckList(_PNH_DATA.CAN.pnh); }), callAjaxAsync('https://spreadsheets.google.com/feeds/list/1pDmenZA-3FOTvhlCq9yz1dnemTmS9l_njZQbu_jLVMI/op17piq/public/values').then(response => { let entry = response.feed.entry[0]; let processEntryField = entryField => entryField.$t.toLowerCase().replace(/ \|/g,'|').replace(/\| /g,'|').split('|'); hospitalPartMatch = processEntryField(entry.gsx$hmchp); hospitalFullMatch = processEntryField(entry.gsx$hmchf); animalPartMatch = processEntryField(entry.gsx$hmcap); animalFullMatch = processEntryField(entry.gsx$hmcaf); schoolPartMatch = processEntryField(entry.gsx$schp); schoolFullMatch = processEntryField(entry.gsx$schf); }), callAjaxAsync('https://spreadsheets.google.com/feeds/list/1L82mM8Xg-MvKqK3WOfsMhFEGmVM46lA8BVcx8qwgmA8/ofblgob/public/values').then(response => { var WMEPHuserList = response.feed.entry[0].gsx$phuserlist.$t.split('|'); var betaix = WMEPHuserList.indexOf('BETAUSERS'); WMEPHdevList = []; WMEPHbetaList = []; for (var ulix=1; ulix<betaix; ulix++) WMEPHdevList.push(WMEPHuserList[ulix].toLowerCase().trim()); for (ulix=betaix+1; ulix<WMEPHuserList.length; ulix++) WMEPHbetaList.push(WMEPHuserList[ulix].toLowerCase().trim()); }) ]).then(() => { // For now, Canada uses some of the same settings as USA. _PNH_DATA.CAN.categories = _PNH_DATA.USA.categories; _PNH_DATA.CAN.categoryNames = _PNH_DATA.USA.categoryNames; placeHarmonizer_bootstrap(); }); // Start the script } // Start downloading the PNH spreadsheet data in the background. Starts the script once data is ready. downloadPnhData(); })();
QingJ © 2025
镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址