// ==UserScript==
// @name WME Place Harmonizer
// @namespace WazeUSA
// @version 1.3.127
// @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, '%26')
.replace(/[ \/]/g, '%20')
.trim();
address = address
.replace(/No street, /, '')
.replace(/No address/, '')
.replace(/ /g, '%20')
.replace(/CR-/g, 'County%20Rd%20')
.replace(/SR-/g, 'State%20Hwy%20')
.replace(/US-/g, 'US%20Hwy%20')
.replace(/ CR /g, '%20County%20Rd%20')
.replace(/ SR /g, '%20State%20Hwy%20')
.replace(/ US /g, '%20US%20Hwy%20')
.replace(/$CR /g, 'County%20Rd%20')
.replace(/$SR /g, 'State%20Hwy%20')
.replace(/$US /g, 'US%20Hwy%20')
.trim();
return 'http://www.google.com/search?q=' + searchName + (address.length > 0 ? ',%20' + address: '');
}
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) {
var re = new RegExp(match[1].replace(/\\/,'\\'),'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}.*)(?: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) {
if(actions.length > 0) {
var m_action = new MultiAction();
m_action.setModel(W.model);
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 should 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;
if (venue.isParkingLot() && name && !/\b(parking|lot|garage)\b/i.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;
executeMultiAction([new UpdateObject(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;
executeMultiAction([new UpdateObject(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;
executeMultiAction([new UpdateObject(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;
executeMultiAction([new UpdateObject(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;
executeMultiAction([new UpdateObject(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;
executeMultiAction([new UpdateObject(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;
executeMultiAction([new UpdateObject(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() {
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.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 venue = getSelectedVenue();
harmonizePlaceGo(venue, 'harmonize', [new UpdateObject(venue, { houseNumber: hnTempDash })]); // 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(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;
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(hasOperator, wl, outputFormat, isPLA);
}
return flag;
}
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');
}
}
},
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, 'Is there a ' + newCategories[0] + ' at this location?', 'Yes', 'Add ' + newCategories[0]); }
action() {
let venue = getSelectedVenue();
newCategories.push.apply(newCategories,altCategories);
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?'); }
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;
executeMultiAction([new UpdateObject(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) {
if ( hpMode.harmFlag && $('#WMEPH-ConvenienceStoreToGasStations').prop('checked') ) { // Automatic if user has the setting checked
newCategories = insertAtIX(newCategories, 'CONVENIENCE_STORE', 1); // insert the C.S. category
actions.push(new UpdateObject(item, { categories: newCategories }));
_updatedFields.categories.updated = true;
phlogdev('Conv. store category added');
} else { // If not checked, then it will be a banner button
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
}
}
if (item.is2D()) {
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);
}
}
// *** 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;
}
// 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 ) {
bannButt.addCat2.message = 'Is there a ' + catTransWaze2Lang[altCategories[0]] + ' at this location?';
bannButt.addCat2.title = 'Add ' + catTransWaze2Lang[altCategories[0]];
}
// 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);
}
}
}
}
}
}
}
}
// ### remove unnecessary parent categories (Restaurant doesn't need food and drink)
if ( newCategories.indexOf('FOOD_AND_DRINK') > -1 ) {
if (newCategories.indexOf('RESTAURANT') > -1 || newCategories.indexOf('FAST_FOOD') > -1 ) {
newCategories.splice(newCategories.indexOf('FOOD_AND_DRINK'),1); // remove Food/Drink Cat
actions.push(new UpdateObject(item, { categories: newCategories }));
_updatedFields.categories.updated = true;
}
}
}
var isPoint = item.isPoint();
var isArea = item.is2D();
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) {
var USPSStrings = ['USPS','POSTOFFICE','USPOSTALSERVICE','UNITEDSTATESPOSTALSERVICE','USPO','USPOSTOFFICE','UNITEDSTATESPOSTOFFICE','UNITEDSTATESPOSTALOFFICE'];
if ( USPSStrings.some(words => newName.toUpperCase().replace(/[ \/\-\.]/g,'').indexOf(words) > -1) ) {
bannButt.isThisAPostOffice = new Flag.IsThisAPostOffice();
}
} 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');
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();
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, '%26');
searchName = searchName.replace(/[ \/]/g, '%20');
if ('string' === typeof addr.street.name && addr.street.name !== null && addr.street.name !== '') {
searchStreet = addr.street.name + ',%20';
}
searchStreet = searchStreet.replace(/ /g, '%20');
searchStreet = searchStreet.replace(/CR-/g, 'County%20Rd%20');
searchStreet = searchStreet.replace(/SR-/g, 'State%20Hwy%20');
searchStreet = searchStreet.replace(/US-/g, 'US%20Hwy%20');
searchStreet = searchStreet.replace(/ CR /g, '%20County%20Rd%20');
searchStreet = searchStreet.replace(/ SR /g, '%20State%20Hwy%20');
searchStreet = searchStreet.replace(/ US /g, '%20US%20Hwy%20');
searchStreet = searchStreet.replace(/$CR /g, 'County%20Rd%20');
searchStreet = searchStreet.replace(/$SR /g, 'State%20Hwy%20');
searchStreet = searchStreet.replace(/$US /g, 'US%20Hwy%20');
if ('string' === typeof HN && searchStreet !== '') {
searchHN = HN + '%20';
}
if ('string' === typeof addr.city.attributes.name && addr.city.attributes.name !== '') {
searchCity = addr.city.attributes.name + ',%20';
}
searchCity = searchCity.replace(/ /g, '%20');
return 'http://www.google.com/search?q=' + searchName + (searchName ? ',%20' : '') + searchHN + searchStreet + searchCity + addr.state.name;
} // 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-ConvenienceStoreToGasStations');
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-ConvenienceStoreToGasStations','Automatically add "Convenience Store" category to gas stations');
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();
})();