// ==UserScript==
// @name WME Find Deleted Objects
// @namespace https://greasyfork.org/users/32336-joyriding
// @version 2022.08.13.01
// @description Attempts to show the history of a venue that has been deleted in an area.
// @author Joyriding
// @include https://beta.waze.com/*
// @include https://www.waze.com/editor*
// @include https://www.waze.com/*/editor*
// @exclude https://www.waze.com/user/editor*
// @icon 
// @require https://greasyfork.org/scripts/24851-wazewrap/code/WazeWrap.js
// @contributionURL https://github.com/WazeDev/Thank-The-Authors
// @license GPLv3
// @grant none
// ==/UserScript==
/* global W */
/* global OpenLayers */
/* global I18n */
/* global $ */
/* global WazeWrap */
/* global require */
/* global google */
/* global Backbone */
(function() {
'use strict';
var settings = {};
var _wmeFdoFeatureLayer;
var _wmeFdoMarkerLayer;
var projection=new OpenLayers.Projection("EPSG:900913");
var displayProjection=new OpenLayers.Projection("EPSG:4326");
var requestStatus = [];
var seenVenueIds = [];
var subcategoryToParentCategories = [];
var flags = {
venue: { }
};
var externalProviderDetails = { };
var idMappings = {
users: [],
streets: [],
}
let userCache = [];
var userProgress = [];
let earliestScanDate = parseInt(new Date('2016-12-31').getTime());
let googleMapsApi = {
map: document.createElement('div'),
service: null
}
const DAY_MS = 1000 * 60 * 60 * 24;
const WEEK_MS = DAY_MS * 7;
function bootstrap(tries) {
tries = tries || 1;
if (W && W.map &&
W.model && W.loginManager.user &&
WazeWrap.Ready && $ ) {
init();
} else if (tries < 1000)
setTimeout(function () {bootstrap(tries++);}, 200);
}
function init()
{
let styleElements = getWmeStyles();
var $style = $("<style>");
$style.html([
'<style>',
'#wmeFdoStatusDate { margin-top:10px;margin-bottom:10px; }',
'.spinner-disable { opacity: 0; }',
'.wmeFdoFormInputLabel { display:inline-block;margin-right:10px;}',
'.wmeFdoFormInput { display:inline-block; width:150px;}',
'.wmeFdoButtonArea {margin-top:10px;margin-bottom:10px; }',
'.wmeFdoGoogleBtn { margin-top:10px;margin-bottom:10px; }',
'.wmeFdoRecreate a { color: #b5b5b5; cursor: pointer; text-decoration:none;}',
'.wmeFdoRecreate a:hover { text-decoration:underline; }',
'.wmeFdoResultDetail { position: relative; }',
'.wmeFdoDetailControls {position: absolute;right: 17px;top: 10px;}',
'.wmeFdoFlag a { text-decoration: none; }',
'.wmeFdoFlag { cursor:pointer; opacity:0.2; margin-right:5px;}',
'.wmeFdoFlag:hover { color:red; opacity:.75}',
'.wmeFdoFlag.flagged { color: red; opacity:1}',
'.wmeFdoHistoryContent { padding-left: 14px;padding-right: 14px; }',
'.wmeFdo-element-history-item:not(:last-child) { margin-bottom: 5px; }',
'.wmeFdo-element-history-item.closed .wmeFdo-tx-header {border-radius: 2px;background: #ededed;}',
'.wmeFdo-element-history-item.closed .wmeFdo-tx-header:hover {background: rgba(255, 255, 255, 0.5);}',
'.wmeFdo-element-history-item .wmeFdo-tx-header {display: flex;flex-direction: row; justify-content: space-between;padding: 10px;border: 1px solid #e4e4e4;border-radius: 2px 2px 0px 0px;font-size: 11px;color: #687077;line-height: 14px;background: white;transition: all 0.3s;cursor: pointer;}',
'.wmeFdo-element-history-item .wmeFdo-tx-content {display: block;padding: 10px;padding-left: 22px; background-color: white;border: 1px solid #e4e4e4;border-top: none; font-size: 11px;}',
'.wmeFdo-element-history-item.closed .wmeFdo-tx-content {display: none;}',
'.wmeFdo-element-history-item.wmeFdo-tx-has-related .wmeFdo-tx-changed-attribute:last-child {margin-bottom: 10px;}',
'.wmeFdo-element-history-item.wmeFdo-tx-has-related .wmeFdo-tx-changed-attribute:last-child { margin-bottom: 10px;}',
'.wmeFdo-element-history-item .tx-toggle-closed {width: 30px;height: 30px;text-align: center;}',
'.wmeFdo-element-history-item:not(.tx-has-content) .tx-content, .element-history-item:not(.tx-has-content) .tx-toggle-closed {display: none;}',
'.wmeFdo-element-history-item .tx-toggle-closed {width: 30px;height: 30px;text-align: center;}',
'.wmeFdo-element-history-item.closed .wmeFdo-tx-toggle-closed::after { content: "\\F107";top: 8px;}',
'.wmeFdo-element-history-item .wmeFdo-tx-toggle-closed::after {display: inline-block;font: normal normal normal 14px/1 FontAwesome;font-size: inherit;text-rendering: auto;-webkit-font-smoothing: antialiased;-moz-osx-font-smoothing: grayscale;color: #3d3d3d;display: block;position: relative;top: 8px;content: "\\F106";font-size: 19px;}',
'.historySummaryItem { margin-left:10px; color:#a7a7a7; }',
'.historySummaryItem a, .historySummaryItem a:visited { color:#a7a7a7; }',
'.wmeFdo-place-delete { filter: saturate(17) hue-rotate(65deg); }',
'.wmeFdo-place-delete:hover { filter: brightness(110%) saturate(17) hue-rotate(65deg) !important; }',
'.wmeFdo-place-delete:after {',
'border-top: 3px solid #ffffff;',
'width: 69%;',
'height: 49%;',
'position: absolute;',
'bottom: 6px;',
'left: 8px;',
'content: "";',
'transform: rotate(-45deg);',
'}',
'#wmeFdoSubTabs, .wmeFdoFlag { display: none!important; }',
'.wmeFdoResultTypeVenue { opacity:0.5; margin-left:-3px;margin-right:5px;position:relative;top:3px;' + styleElements.resultTypeVenueStyle + '}',
'.wmeFdoResultTypeCamera { opacity:0.5; margin-left:-3px;margin-right:5px;position:relative;top:3px;' + styleElements.resultTypeCameraStyle + '}',
'.wmeFdoResultTypeArea { opacity:0.5; margin-left:-3px;margin-right:5px;position:relative;top:10px;' + styleElements.resultTypeAreaStyle + '}',
// '.wmeFdoResultTypeResidential { opacity:0.65; margin-left:-3px;margin-right:5px;position:relative;top:4px;' + styleElements.resultTypeResidentialNew + '}',
// '.wmeFdoResultTypeParking { opacity:0.65; margin-left:-1px;margin-right:3px;margin-bottom:6px;position:relative;top:4px;filter: saturate(0%);' + styleElements.resultTypeParkingNew + '}',
'.wmeFdoResultTypeResidential { filter:invert(.35); margin-left:-11px;margin-right:-1px;position:relative;top:-6px;' + styleElements.resultTypeResidentialNew + '}',
'.wmeFdoResultTypeParking { filter:invert(.35); margin-left:-11px;margin-right:-1px;position:relative;top:-6px;' + styleElements.resultTypeParkingNew + '}',
'.wmeFdoResultDeletedNewPlace:after { position:relative;content:"+";color:black;font-size:17px;font-weight:900;text-shadow: 0px 0px 2px white;top:7px;left:8px; }',
'.wmeFdoResultDeletedNewPlaceAlt:after { position:relative;content:"+";color:black;font-size:17px;font-weight:900;top:14px;left:20px; }',
'.wmeFdoResultDeletedFromRequest:after { position:relative;content:"\\002b06";color:black;font-size:13px;font-weight:900;text-shadow: 0px 0px 3px white;top:7px;left:8px; }',
'.wmeFdoResultDeletedFromRequestAlt:after { position:relative;content:"\\002b06";color:black;font-size:13px;font-weight:900;top:14px;left:20px;opacity: .8; }',
'.wmeFdoIconFilter { transform:rotate(90deg);margin-left:-1px;margin-right:3px;margin-bottom:6px;position:relative;top:4px;filter: saturate(0%);' + styleElements.iconFilter + '}',
//'.wmeFdoIconFilterInline { font-size:15px;margin-top:5px; }',
'.wmeFdoIconFilterInline:before { transform:rotate(90deg);filter:saturate(0%);content:"";position:relative;display:inline-block;top:3px;margin-right:10px;' + styleElements.iconFilter + '}',
'.wmeFdoScanAttributesDropdown { font-size:15px;margin-top:5px; }',
'.wmeFdoScanAttributesLabel { margin-left:10px; }',
'#wmeFdoStatusUsersList > li { margin-bottom:10px }',
'.wmeFdoStatusUserDates { width:200px; }',
'.wmeFdoStatusUserLatest { float:left; }',
'.wmeFdoStatusUserEarliest { float:right; }',
'.wmeFdoProgress {width:200px;height:5px;border:1px solid gray;border-radius:3px; }',
'#wmeFdoScanStatusHeader .wmeFdoProgress { margin-top: 5px;margin-left: 27px; }',
'.wmeFdoComplete { width:0%;height:4px;background-color:#5a7990; }',
'.wmeFdoProgressIcon { transform: scale(1.3,-1) rotate(90deg); }',
'.wmeFdoDropdownHeader { width: 80%; }',
'#wmeFdoStatusSummary { margin-bottom:10px; }',
'.wmeFdoHistorySummaryList > li { margin-bottom:2px; }',
'</style>'
].join(' '));
var $section = $('<div id="wmeFdoPanel">');
$section.html(
'<div id="wmeFdoHead">',
'</div>'
);
var $navTabs = $('<ul id="wmeFdoSubTabs" class="nav nav-tabs"><li class="active"><a data-toggle="tab" href="#wmeFdoTabFind">Find</a></li>' +
'<li><a data-toggle="tab" href="#wmeFdoTabFlagged" id="wmeFdoTabFlagEvent">Flagged</a></li>' +
'<li><a data-toggle="tab" href="#wmeFdoTabSettings"><span class="fa fa-gear"></span></a></li></ul>');
var $tabContent = $('<div class="tab-content" style="padding:5px;">');
var $fdoTabFind = $('<div id="wmeFdoTabFind" class="tab-pane active">');
$fdoTabFind.append([
'<h4 style="margin-top:0px;">Find Deleted Objects</h4>',
'<h6 style="margin-top:0px;">' + GM_info.script.version + '</h6>'
].join(' '));
createSettingsCheckbox($fdoTabFind, 'wmeFdoNearbyEditors','Nearby Editors');
$fdoTabFind.append([
'<div class="controls-container" style="padding-top: 2px;"><div class="wmeFdoFormInputLabel">Editor Name</div><input type="text" id="wmeFdoEditorName" class="form-control wmeFdoFormInput"><label for="wmeFdoEditorName"></label></div><br>',
].join(' '));
createSettingsCheckbox($fdoTabFind, 'wmeFdoIncludeSelf','Include Deletes By This Editor');
createSettingsCheckbox($fdoTabFind, 'wmeFdoLimitToScreen','Limit to Screen');
$fdoTabFind.append([
'<div class="controls-container" style="padding-top: 2px;"><div class="wmeFdoFormInputLabel">Find Prior To</div><input type="text" id="wmeFdoFindDate" class="form-control wmeFdoFormInput"><label for="wmeFdoFindDate"></label></div><br>',
].join(' '));
var $fdoTabFlagged = $('<div id="wmeFdoTabFlagged" class="tab-pane">');
$fdoTabFlagged.html('<h4>Flagged Objects<h4>');
let $fdoTabFlaggedResults = $('<div id="wmeFdoFlaggedResults>');
$fdoTabFlagged.append($fdoTabFlaggedResults);
var $fdoTabSettings = $('<div id="wmeFdoTabSettings" class="tab-pane">');
$fdoTabSettings.append([
'<h4>Settings</h4>',
'<div id="wmeFdoSettings">',
'<div class="controls-container" style="padding-top: 2px;">Rate Limit Delay: <input type="text" id="wmeFdoRateLimitDelay" ><label for="wmeFdoRateLimitDelay"></label></div>',
'</div>'
].join(' '));
$('#wmeFdoTabSettings').append('div').append('Reset Settings');
$tabContent.append($fdoTabFind,$fdoTabFlagged,$fdoTabSettings);
$section.append($navTabs, $tabContent);
$section.append($style);
let updateDesc = "<style>#wmeFdoUpdate > li {margin-top:6px;margin-left: -22px;}</style>This update addresses most known issues of the initial release:"
+ "<ul id='wmeFdoUpdate'>"
+ "<li><b>Better Rate Limiting</b><br>Completely rewritten API request layer.</li>"
+ "<li><b>Recreate Places</b><br>External Providers can now be added from Prod WME.</li>"
+ "<li><b>New Status Display</b><br>Shows the list of nearby editors used and the current progress. Completed scans now show correctly.</li>"
+ "<li><b>Filtering</b><br> </li>"
+ "<li><b>Result Type Icons</b><br>Different icons for Parking and Residential places</li>"
+ "</ul>"
+ "<span style='color:red'>Reminder that this is an SM+ script!</span>";
updateDesc = 'Fixes browser freeze when starting a scan.<br><br>';
//WazeWrap.Interface.ShowScriptUpdate("Find Deleted Objects", GM_info.script.version, updateDesc, "https://greasyfork.org/scripts/376861-wme-find-deleted-objects", "https://www.waze.com/forum/viewtopic.php?f=1286&t=275164&p=1922116#p1922116");
new WazeWrap.Interface.Tab('FiDO', $section.html(), initializeSettings);
prepareTab();
WazeWrap.Interface.AddLayerCheckbox("display", "Deleted Objects", settings.Enabled, onLayerCheckboxChanged);
onLayerCheckboxChanged(settings.Enabled);
buildFilterSection('#wmeFdoTabFind');
let buttons = document.createElement('div');
buttons.className = 'wmeFdoButtonArea';
$('#wmeFdoTabFind').append(buttons);
var findButton = document.createElement('button');
findButton.id = 'wmeFdoFindButton';
findButton.textContent = 'Start Scan';
findButton.className = 'btn btn-success center-block';
buttons.append(findButton);
var cancelButton = document.createElement('button');
cancelButton.id = 'wmeFdoCancelButton';
cancelButton.textContent = 'Cancel Scan';
cancelButton.className = 'btn btn-warning center-block hidden';
buttons.append(cancelButton);
findButton.addEventListener('click', function() {
findButton.className = 'btn btn-success center-block hidden';
cancelButton.className = 'btn btn-danger center-block ';
let tab = document.getElementById('wmeFdoTab');
tab.innerHTML = 'FiDO <span class="spinner-enable"><i class="icon-spinner icon-spin fa fa-spinner fa-spin"></i></span>';
$('#wmeFdoTabFind').data("status","running");
settings.Enabled = true;
onLayerCheckboxChanged(settings.Enabled); // TODO: Fix this so the Layer menu checkbox gets updated
clearResults();
beginScanNew();
findObjects();
});
cancelButton.addEventListener('click', function() {
cancelButton.textContent = 'Cancelling...';
cancelButton.className = 'btn btn-warning center-block disabled';
$('#wmeFdoTabFind').data("status","cancelled");
let tab = document.getElementById('wmeFdoTab');
tab.innerHTML = 'FiDO <span class="spinner-disable"><i class="icon-spinner icon-spin fa fa-spinner fa-spin"></i></span>';
cancelScanNew();
cancelScan(0);
});
$("#wmeFdoEditorName").keyup(function(event) {
if (event.keyCode === 13) {
$("#wmeFdoFindButton").click();
}
});
$("#wmeFdoFindDate").keyup(function(event) {
if (event.keyCode === 13) {
$("#wmeFdoFindButton").click();
}
});
$('#wmeFdoNearbyEditors').change(function() {
var settingName = $(this)[0].id.substr(6);
settings[settingName] = this.checked;
saveSettings();
if(this.checked)
{
$("#wmeFdoEditorName").prop('disabled', true);
$("#wmeFdoIncludeSelf").prop('disabled', true);
$("#wmeFdoLimitToScreen").prop('disabled', true);
}
else
{
$("#wmeFdoEditorName").prop('disabled', false);
$("#wmeFdoIncludeSelf").prop('disabled', false);
$("#wmeFdoLimitToScreen").prop('disabled', false);
}
});
let flaggedTab = document.getElementById('wmeFdoTabFlagEvent');
flaggedTab.addEventListener('click', function() {
let flagList = '';
for (var id in flags.venue) {
flagList = flagList + id + '<br>';
console.log(id);
}
let results = document.getElementById('wmeFdoTabFlagged');
results.append(flagList);
});
var $usersDiv = $( '<div id="fdoUsersPanel"></div>');
//var dateSpan = document.createElement('div');
//dateSpan.id = 'wmeFdoStatusDate';
//$('#wmeFdoTabFind').append(dateSpan);
let inputDate = document.getElementById('wmeFdoFindDate');
inputDate.value = formatCurrentDate();
$('#wmeFdoTabFind').append(buildResultsPanel());
console.log($('#wmeFdoFilterResidential')[0].id);
$('#wmeFdoFilterResidential').change(function() {
var settingName = $(this)[0].id.substr(6);
settings[settingName] = this.checked;
saveSettings();
applyFiltersToResults();
});
$('#wmeFdoFilterParkingLot').change(function() {
var settingName = $(this)[0].id.substr(6);
settings[settingName] = this.checked;
saveSettings();
applyFiltersToResults();
});
$('#wmeFdoFilterAnyType').change(function() {
var settingName = $(this)[0].id.substr(6);
settings[settingName] = this.checked;
saveSettings();
applyFiltersToResults();
});
$('#wmeFdoFilterNewRejected').change(function() {
var settingName = $(this)[0].id.substr(6);
settings[settingName] = this.checked;
saveSettings();
applyFiltersToResults();
});
$('#wmeFdoFilterDeleteRequest').change(function() {
var settingName = $(this)[0].id.substr(6);
settings[settingName] = this.checked;
saveSettings();
applyFiltersToResults();
});
initializeSettings();
console.log('FiDO: done');
}
function initializeSettings()
{
loadSettings();
setChecked('wmeFdoIncludeSelf', settings.IncludeSelf);
setChecked('wmeFdoLimitToScreen', settings.LimitToScreen);
setChecked('wmeFdoNearbyEditors', settings.NearbyEditors);
setChecked('wmeFdoFilterResidential', settings.FilterResidential);
setChecked('wmeFdoFilterParkingLot', settings.FilterParkingLot);
setChecked('wmeFdoFilterAnyType', settings.FilterAnyType);
setChecked('wmeFdoFilterNewRejected', settings.FilterNewRejected);
setChecked('wmeFdoFilterDeleteRequest', settings.FilterDeleteRequest);
// if(settings.Enabled)
// W.selectionManager.events.register("selectionchanged", null, drawSelection);
$('.wmeFdoSettingsCheckbox').change(function() {
var settingName = $(this)[0].id.substr(6);
settings[settingName] = this.checked;
saveSettings();
if(this.checked)
{
// W.selectionManager.events.register("selectionchanged", null, drawSelection);
// processVenues(W.model.venues.getObjectArray());
}
else
{
// W.selectionManager.events.unregister("selectionchanged", null, drawSelection);
// wmeFdoSelectedLayer.removeAllFeatures();
// processVenues(W.model.venues.getObjectArray());
}
});
if(settings.NearbyEditors)
{
$("#wmeFdoEditorName").prop('disabled', true);
$("#wmeFdoIncludeSelf").prop('disabled', true);
$("#wmeFdoLimitToScreen").prop('disabled', true);
}
else
{
$("#wmeFdoEditorName").prop('disabled', false);
$("#wmeFdoIncludeSelf").prop('disabled', false);
$("#wmeFdoLimitToScreen").prop('disabled', false);
}
$('.wmeFdoSettingsText').on('input propertychange paste', function() {
var settingName = $(this)[0].id.substr(6);
settings[settingName] = parseInt($(this).val());
saveSettings();
//W.selectionManager.events.unregister("selectionchanged", null, drawSelection);
//processVenues(W.model.venues.getObjectArray());
});
}
function getWmeStyles() {
// Get the sprite icons from the native WME CSS so that we can use our own document structure
let styleElements = { };
let $tempDiv = null;
let tempQuerySelector = null;
let tempComputedStyle = null;
//.form-search .search-result-region .search-result .icon {
//background-image: url(//editor-assets.waze.com/production/img/toolbare2f6b31….png);
$tempDiv = $('<div class="form-search">').append($('<div class="search-result-region">').append($('<div class="search-result">').append($('<div class="icon">'))));
$('body').append($tempDiv);
tempQuerySelector = document.querySelector('.form-search .search-result-region .search-result .icon');
tempComputedStyle = window.getComputedStyle(tempQuerySelector);
styleElements.resultTypeVenueStyle =
`background-image:${tempComputedStyle.getPropertyValue('background-image')};`
+ `background-size:${tempComputedStyle.getPropertyValue('background-size')};`
+ `background-position:${tempComputedStyle.getPropertyValue('background-position')};`
+ `width:${tempComputedStyle.getPropertyValue('width')};`
+ `height:${tempComputedStyle.getPropertyValue('height')};`;
$tempDiv.remove();
//#edit-buttons .camera .item-icon::after {
//background-image: url(//editor-assets.waze.com/production/img/toolbare2f6b31….png);
tempQuerySelector = document.querySelector("#primary-toolbar > div > div.toolbar-group.toolbar-group-drawing > wz-menu > div > wz-menu-item.toolbar-group-item.camera.WazeControlDrawFeature > div > div.item-icon");
tempComputedStyle = window.getComputedStyle(tempQuerySelector,'::after');
styleElements.resultTypeCameraStyle =
`background-image:${tempComputedStyle.getPropertyValue('background-image')};`
+ `background-size:${tempComputedStyle.getPropertyValue('background-size')};`
+ `background-position:${tempComputedStyle.getPropertyValue('background-position')};`
+ `width:${tempComputedStyle.getPropertyValue('width')};`
+ `height:${tempComputedStyle.getPropertyValue('height')};`;
//#edit-buttons .toolbar-group .toolbar-group-item .drawing-controls .drawing-control.polygon:after {
//background-image: url(//editor-assets.waze.com/production/img/toolbare2f6b31….png);
tempQuerySelector = document.querySelector("#primary-toolbar > div > div.toolbar-group.toolbar-group-venues > wz-menu > div > wz-menu-item.toolbar-group-item.car-services.WazeControlDrawFeature > div > div.drawing-controls > wz-basic-tooltip:nth-child(2) > wz-tooltip > wz-tooltip-source > wz-button").shadowRoot.querySelector("button")
tempComputedStyle = window.getComputedStyle(tempQuerySelector,'::after');
styleElements.resultTypeAreaStyle =
`background-image:${tempComputedStyle.getPropertyValue('background-image')};`
+ `background-size:${tempComputedStyle.getPropertyValue('background-size')};`
+ `background-position:${tempComputedStyle.getPropertyValue('background-position')};`
+ `width:${tempComputedStyle.getPropertyValue('width')};`
+ `height:${tempComputedStyle.getPropertyValue('height')};`;
// .toolbar .toolbar-button.add-house-number .item-icon {
//background-image: url(//editor-assets.waze.com/production/img/toolbarcad3e90….png);
//$tempDiv = $('<div class="toolbar">').append($('<div class="toolbar-button add-house-number">').append($('<div class="item-icon">')));
//$('body').append($tempDiv);
//tempQuerySelector = document.querySelector('.toolbar .toolbar-button.add-house-number .item-icon');
//tempComputedStyle = window.getComputedStyle(tempQuerySelector);
//styleElements.resultTypeResidential = `background-image:${tempComputedStyle.getPropertyValue('background-image')};`
// + `background-size:${tempComputedStyle.getPropertyValue('background-size')};`
// + `background-position:${tempComputedStyle.getPropertyValue('background-position')};`
// + `width:${tempComputedStyle.getPropertyValue('width')};`
// + `height:${tempComputedStyle.getPropertyValue('height')};`;
//$tempDiv.remove();
//#edit-buttons .parking-lot .item-icon::after {
//background-image: url(//editor-assets.waze.com/production/img/toolbarcad3e90….png);
//tempQuerySelector = document.querySelector('#edit-buttons .parking-lot .item-icon');
//tempComputedStyle = window.getComputedStyle(tempQuerySelector,'::after');
//styleElements.resultTypeParking = `background-image:${tempComputedStyle.getPropertyValue('background-image')};`
// + `background-size:${tempComputedStyle.getPropertyValue('background-size')};`
// + `background-position:${tempComputedStyle.getPropertyValue('background-position')};`
// + `width:${tempComputedStyle.getPropertyValue('width')};`
// + `height:${tempComputedStyle.getPropertyValue('height')};`;
//#edit-panel .merge-landmarks .merge-item .do-merge:before, .edit-panel .merge-landmarks .merge-item .do-merge:before {
//background-image: url(//editor-assets.waze.com/production/img/buttons756c103….png);
$tempDiv = $('<div id="edit-panel">').append($('<div class="merge-landmarks">').append($('<div class="merge-item">').append($('<div class="do-merge">'))));
$('body').append($tempDiv);
tempQuerySelector = document.querySelector('#edit-panel .merge-landmarks .merge-item .do-merge');
tempComputedStyle = window.getComputedStyle(tempQuerySelector, '::before');
styleElements.iconFilter =
`background-image:${tempComputedStyle.getPropertyValue('background-image')};`
+ `background-size:${tempComputedStyle.getPropertyValue('background-size')};`
+ `background-position:${tempComputedStyle.getPropertyValue('background-position')};`
+ `width:${tempComputedStyle.getPropertyValue('width')};`
+ `height:${tempComputedStyle.getPropertyValue('height')};`;
$tempDiv.remove();
//#edit-panel .merge-landmarks .merge-item .icon.parking_lot:after
$tempDiv = $('<div id="edit-panel">').append($('<div class="merge-landmarks">').append($('<div class="merge-item">').append($('<div class="icon parking_lot">'))));
$('body').append($tempDiv);
tempQuerySelector = document.querySelector('#edit-panel .merge-landmarks .merge-item .icon.parking_lot');
tempComputedStyle = window.getComputedStyle(tempQuerySelector, '::after');
styleElements.resultTypeParkingNew =
`background-image:${tempComputedStyle.getPropertyValue('background-image')};`
+ `background-size:${tempComputedStyle.getPropertyValue('background-size')};`
+ `background-position:${tempComputedStyle.getPropertyValue('background-position')};`
+ `width:${tempComputedStyle.getPropertyValue('width')};`
+ `height:${tempComputedStyle.getPropertyValue('height')};`;
$tempDiv.remove();
//#edit-panel .merge-landmarks .merge-item .icon.parking_lot:after
$tempDiv = $('<div id="edit-panel">').append($('<div class="merge-landmarks">').append($('<div class="merge-item">').append($('<div class="icon residential_home">'))));
$('body').append($tempDiv);
tempQuerySelector = document.querySelector('#edit-panel .merge-landmarks .merge-item .icon.residential_home');
tempComputedStyle = window.getComputedStyle(tempQuerySelector, '::after');
styleElements.resultTypeResidentialNew =
`background-image:${tempComputedStyle.getPropertyValue('background-image')};`
+ `background-size:${tempComputedStyle.getPropertyValue('background-size')};`
+ `background-position:${tempComputedStyle.getPropertyValue('background-position')};`
+ `width:${tempComputedStyle.getPropertyValue('width')};`
+ `height:${tempComputedStyle.getPropertyValue('height')};`;
$tempDiv.remove();
return styleElements;
}
function getWmeStyle(selector) {
let $tempDiv = null;
let tempQuerySelector = null;
let tempComputedStyle = null;
let returnedStyle = null;
//.form-search .search-result-region .search-result .icon {
//background-image: url(//editor-assets.waze.com/production/img/toolbare2f6b31….png);
$tempDiv = $('<div class="form-search">').append($('<div class="search-result-region">').append($('<div class="search-result">').append($('<div class="icon">'))));
$('body').append($tempDiv);
tempQuerySelector = document.querySelector('.form-search .search-result-region .search-result .icon');
tempComputedStyle = window.getComputedStyle(tempQuerySelector);
returnedStyle =
`background-image:${tempComputedStyle.getPropertyValue('background-image')};`
+ `background-size:${tempComputedStyle.getPropertyValue('background-size')};`
+ `background-position:${tempComputedStyle.getPropertyValue('background-position')};`
+ `width:${tempComputedStyle.getPropertyValue('width')};`
+ `height:${tempComputedStyle.getPropertyValue('height')};`;
$tempDiv.remove();
return returnedStyle;
}
//historyContent
//element-history-item
function buildResultsPanel() {
return $('<div id="fdoResultsPanel">')
.append($('<div class="elementHistoryContainer">')
.append($('<div class="wmeFdoHistoryContent">')
.append($('<div class="transactions">')
.append($('<ul id="wmeFdoTransactionList">'))
)
)
);
}
function buildFilterSection(parentDivName) {
let $section = $('<div id="wmeFdoFilterSection">')
.append($('<div class="elementHistoryContainer">')
.append($('<div class="wmeFdoHistoryContent">')
.append($('<div class="transactions">')
.append($('<ul id="wmeFdoFilter">')
.append($('<li id="wmeFdoFilterDetails" class="wmeFdo-element-history-item wmeFdo-tx-has-content wmeFdo-tx-has-related closed">')
.append($('<div id="wmeFdoFilterDetailsHeader" class="wmeFdo-tx-header">')
// .append($('<div class="flex-noshrink wmeFdoIconFilter">')
// )
//.append($('<div class="flex-noshrink">'))
.append($('<div class="wmeFdo-tx-summary wmeFdoDropdownHeader">')
.append($('<div class="wmeFdoIconFilterInline wmeFdoScanAttributesDropdown">Filter Results</div>'))
)
.append($('<div class="flex-noshrink wmeFdo-tx-toggle-closed">')
)
)
.append($('<div class="wmeFdo-tx-content">')
.append($('<div id="wmeFdoFilterDetailsUsers">'))
.append($('<div id="wmeFdoFilterDetailsPlaces">'))
.append($('<div id="wmeFdoFilterDetailsCameras">')) //<ul><li class="tx-changed-attribute">'))
)
)
.append($('<li id="wmeFdoScanStatus" class="wmeFdo-element-history-item wmeFdo-tx-has-content wmeFdo-tx-has-related closed">')
.append($('<div id="wmeFdoScanStatusHeader" class="wmeFdo-tx-header">')
//.append($('<div class="flex-noshrink">'))
.append($('<div class="wmeFdo-tx-summary wmeFdoDropdownHeader">')
.append($('<div class="wmeFdoScanAttributesDropdown">')
.append($('<i class="fa fa-bar-chart wmeFdoProgressIcon" aria-hidden="true"></i>')
).append($('<span class="wmeFdoScanAttributesLabel"><span id="wmeFdoScanCurrentStatus">Status</span></span>'))
).append($('<div class="wmeFdoProgress hidden"><div class="wmeFdoComplete"></div></div>')
))
.append($('<div class="flex-noshrink wmeFdo-tx-toggle-closed">')
)
)
.append($('<div class="wmeFdo-tx-content">')
.append($('<div id="wmeFdoStatusSummary">')
.append($('<div><span>Total Requests: </span><span id="wmeFdoRequests">0<span></div>'))
.append($('<div><span>Requests per second: </span><span id="wmeFdoRequestsPerSecond">0<span></div>'))
)
.append($('<div id="wmeFdoStatusUsers">')
.append($('<ul id="wmeFdoStatusUsersList">')
))
)
)
)
)
)
);
let $parentDiv = $(parentDivName);
$parentDiv.append($section);
// $('#wmeFdoFilterDetailsUsers').append([
// '<div class="controls-container" style="padding-top: 2px;"><div class="wmeFdoFormInputLabel">Editor Name</div><input type="text" id="wmeFdoFilterEditorName" class="form-control wmeFdoFormInput"><label for="wmeFdoFilterEditorName"></label></div><br>',
// ].join(' '));
let $placeTypes = $('<ul>');
let $placeTypeLi = $('<li class="tx-changed-attribute">');
$placeTypes.append($placeTypeLi);
$('#wmeFdoFilterDetailsPlaces').append($placeTypes);
$placeTypeLi.append(['<div class="ca-name">Place Types</div>'].join(' '));
createSettingsCheckbox($placeTypeLi, 'wmeFdoFilterResidential','Residential');
createSettingsCheckbox($placeTypeLi, 'wmeFdoFilterParkingLot','Parking Lots');
createSettingsCheckbox($placeTypeLi, 'wmeFdoFilterAnyType','All Other Types');
let $purTypes = $('<ul>');
let $purTypeLi = $('<li class="tx-changed-attribute">');
$purTypes.append($purTypeLi);
$('#wmeFdoFilterDetailsPlaces').append($purTypes);
$purTypeLi.append(['<div class="ca-name">Deleted From PUR</div>'].join(' '));
createSettingsCheckbox($purTypeLi, 'wmeFdoFilterNewRejected','New Place Rejected PUR');
createSettingsCheckbox($purTypeLi, 'wmeFdoFilterDeleteRequest','Existing Place Delete Request');
// $('#wmeFdoFilterDetailsPlaces').append([
// '<div class="controls-container" style="padding-top: 2px;"><div class="wmeFdoFormInputLabel">Place Name</div><input type="text" id="wmeFdoFilterPlaceName" class="form-control wmeFdoFormInput"><label for="wmeFdoFilterPlaceName"></label></div><br>',
// ].join(' '));
document.getElementById('wmeFdoFilterDetailsHeader').onclick = function () {
toggleFilterDropDown();
}
document.getElementById('wmeFdoScanStatusHeader').onclick = function () {
toggleScanStatusDropDown();
}
}
function onLayerCheckboxChanged(checked) {
settings.Enabled = checked;
if (settings.Enabled && _wmeFdoFeatureLayer != undefined) {
_wmeFdoFeatureLayer.setVisibility(true);
_wmeFdoMarkerLayer.setVisibility(true);
} else if (_wmeFdoFeatureLayer != undefined) {
_wmeFdoFeatureLayer.setVisibility(false);
_wmeFdoMarkerLayer.setVisibility(false);
}
saveSettings();
}
function clearResults() {
if (_wmeFdoFeatureLayer != undefined) {
_wmeFdoFeatureLayer.removeAllFeatures();
_wmeFdoMarkerLayer.clearMarkers();
}
apiEndpointWazeCom = {};
userProgress = [];
userCache = [];
seenVenueIds = [];
requestStatus = {};
document.getElementById('wmeFdoTransactionList').innerHTML = '';
$('#wmeFdoStatusUsersList').empty();
$('#wmeFdoScanStatusHeader .wmeFdoProgress .wmeFdoComplete').css('width','0%');
}
function cancelScan(tries) {
//TODO: Move rest of cancel code into here
tries = tries || 1;
//console.log('incomplete: ' + incompleteRequestCount());
if (incompleteRequestCount() > 0 && tries < 10) {
console.log('Cancel requested, waiting for ' + incompleteRequestCount() + ' requests to complete');
setTimeout(function () {cancelScan(tries++);}, 500);
} else {
// TODO: Change button to Start Scan
let findButton = document.getElementById('wmeFdoFindButton');
let cancelButton = document.getElementById('wmeFdoCancelButton');
findButton.className = 'btn btn-success center-block';
cancelButton.className = 'btn btn-warning center-block hidden';
cancelButton.textContent = 'Cancel Scan';
//$('#wmeFdoStatusDate')[0].innerText = 'Scan Cancelled';
console.log('Cancelled');
}
}
function incompleteRequestCount() {
let incompleteCount = 0;
let failedCount = 0;
Object.keys(requestStatus).forEach(function (url) {
if (requestStatus[url].status != 'successful') {
if (Date.now() - requestStatus[url].lastAttempt < 10000) {
incompleteCount++;
} else {
failedCount++;
}
}
});
return incompleteCount;
}
function defaultRequestStatus() {
return {
status: 'initial',
attempts: 0,
lastAttempt: null
};
}
function requestParams(url) {
return {
url: url,
iteration: 0,
nextTransactionDate: null,
initiated: Date.now()
}
}
async function beginFind(lookFor, transactionLookup) {
// var findCenterPoint = new OL.Geometry.Point(lookFor.locationCenter.coordinates[0], lookFor.locationCenter.coordinates[1]).transform(W.map.displayProjection, W.map.projection);
if ( $('#wmeFdoTabFind').data("status") == "running"
&& transactionLookup.iteration < transactionLookup.maxIterations
)
{
transactionLookup.iteration++;
let requestParamsTransactions = requestParams(getApiUrlTransactions(transactionLookup.userId, transactionLookup.nextTransactionDate));
requestStatus[requestParamsTransactions.url] = defaultRequestStatus();
apiEndpointWazeCom.requests.push(requestParamsTransactions);
getEndpointResult(apiEndpointWazeCom, requestParamsTransactions, data => {
if (apiEndpointWazeCom.isRunning) {
if (typeof(data) == 'undefined') {
console.log('Request failed.');
}
transactionLookup.nextTransactionDate = data.nextTransactionDate
let nextDate = 'Complete';
if (data.nextTransactionDate != null)
{
nextDate = uuidToDate(data.nextTransactionDate);
//$('#wmeFdoStatusDate')[0].innerText = 'Scanning thru ' + nextDate + '...';
$('#wmeFdoScanCurrentStatus')[0].innerText = 'Scanning...';
}
var transactionCount = 0;
data.transactions.forEach(function(transaction) {
transaction.operations.forEach(async function(operation) {
if (operation.objectType == lookFor.objectType || operation.objectType == 'Camera')
{
transactionCount++;
var operationCenterPoint = new OpenLayers.Geometry.Point(operation.objectCentroid.coordinates[0], operation.objectCentroid.coordinates[1]).transform(displayProjection, projection);
//var distance = operationCenterPoint.distanceTo(findCenterPoint);
//if (distance < 1000)
if (!lookFor.limitToScreen || lookFor.searchArea.containsPoint(operationCenterPoint))
{
if (operation.objectType == 'Camera') {
console.log('camera: ' + operation.objectID);
} else {
if (!seenVenueIds.includes(operation.objectID))
{
getDeletedItemsVenue(operation, transactionLookup, lookFor, operationCenterPoint);
} //else { console.log('already seen: ' + operation.objectID); }
}
}
}
});
});
let earliestDate = userCache[transactionLookup.userId].earliestDate - DAY_MS;
let statusDate = 0;
let pctComplete = 100;
if (transactionLookup.nextTransactionDate != null) {
statusDate = uuidToUnixTime(transactionLookup.nextTransactionDate);
let latestDate = userCache[transactionLookup.userId].latestDate + DAY_MS;
//let earliestDate = userCache[transactionLookup.userId].earliestDate - DAY_MS;
let next = uuidToDate(transactionLookup.nextTransactionDate);
let totalTime = latestDate - earliestDate;
let progressTime = latestDate - statusDate;
pctComplete = (progressTime / totalTime) * 100;
// pctComplete = 100 - pctComplete;
let newCompletedMs = userCache[transactionLookup.userId].previousDate - statusDate;
apiEndpointWazeCom.progress.completeMs += newCompletedMs;
userCache[transactionLookup.userId].completeMs += newCompletedMs;
}
if (transactionLookup.nextTransactionDate == null || statusDate < earliestDate) {
apiEndpointWazeCom.users.complete++;
apiEndpointWazeCom.progress.completeMs -= userCache[transactionLookup.userId].completeMs;
apiEndpointWazeCom.progress.completeMs += userCache[transactionLookup.userId].totalMs;
updateStatusUser(transactionLookup.userId, 'Complete');
$('#wmeFdoStatusUser-' + transactionLookup.userId + ' .wmeFdoProgress .wmeFdoComplete').css('width','100%');
endScan("Scanning...", false);
} else {
updateStatusUser(transactionLookup.userId, uuidToDate(transactionLookup.nextTransactionDate));
$('#wmeFdoStatusUser-' + transactionLookup.userId + ' .wmeFdoProgress .wmeFdoComplete').css('width',pctComplete + '%');
userCache[transactionLookup.userId].previousDate = statusDate;
beginFind(lookFor, transactionLookup);
}
}
});
} else {
if (transactionLookup.iteration >= transactionLookup.maxIterations)
{
endScan("Transaction Limit Exceeded", false);
}
}
}
function updateStatusUser(userID, status) {
$('#wmeFdoStatusUser-' + userID + ' .wmeFdoStatusUser').text(status);
let pctComplete = (apiEndpointWazeCom.progress.completeMs / apiEndpointWazeCom.progress.totalMs) * 100;
if (status == 'Complete') {
$('#wmeFdoStatusUser-' + userID).appendTo('#wmeFdoStatusUsersList');
userCache[userID].completeMs = userCache[userID].totalMs;
//pctComplete = 100;
}
$('#wmeFdoScanStatusHeader .wmeFdoProgress .wmeFdoComplete').css('width',pctComplete + '%');
}
async function getDeletedItemsVenue(operation, transactionLookup, lookFor, operationCenterPoint) {
//console.log('element history for ' + data.nextTransactionDate);
let requestParamsElementHistory = requestParams(getApiUrlElementHistory('venue', operation.objectID, null));
requestStatus[requestParamsElementHistory.url] = defaultRequestStatus();
apiEndpointWazeCom.requests.push(requestParamsElementHistory);
getEndpointResult(apiEndpointWazeCom, requestParamsElementHistory, venueHistory => {
if (apiEndpointWazeCom.isRunning) {
venueHistory.transactions.objects.forEach(function(venueTransaction) {
venueTransaction.objects.forEach(function(venueAction) {
if (venueAction.actionType == 'DELETE') {
// console.log(venueAction);
if (venueAction.objectType == 'venue' && venueAction.oldValue != null)
{
let selfDelete = false;
if (transactionLookup.userId == venueTransaction.userID) {
selfDelete = true;
}
if (lookFor.includeSelf || !selfDelete) {
// Skip if searching for editor who deleted it
if (!seenVenueIds.includes(venueAction.objectID))
{
seenVenueIds.push(venueAction.objectID);
let historySummary = getVenueHistory(venueHistory);
var record = document.createElement('div');
record.id=venueTransaction.transactionID;
let username = null;
venueHistory.users.objects.forEach( function(user) {
if (user.id == venueTransaction.userID) {
let userRank = user.rank + 1;
username = user.userName + '(' + userRank + ')';
}
});
record.innerText = username + ": " + venueAction.oldValue.name + " - " + venueAction.objectID;
var details = document.createElement('li');
details.id = 'entry-' + venueAction.objectID;
details.className = 'wmeFdo-element-history-item wmeFdo-tx-has-content wmeFdo-tx-has-related closed';
var txHeader = document.createElement('div');
txHeader.className = 'wmeFdo-tx-header';
let detailsDisplayedOnce = false;
txHeader.onclick = function () {
if (!detailsDisplayedOnce && details.classList.contains('closed')) {
detailsDisplayedOnce = true;
displayExternalProviders(venueAction.objectID, venueItem.externalProviderIDs);
}
toggleDetailsDropDown(details.id, operationCenterPoint);
};
var txTypeDiv = document.createElement('div');
txTypeDiv.className = 'flex-noshrink';
txHeader.append(txTypeDiv);
var txTypeIcon = document.createElement('div');
// class name populated after venueItem is defined
txTypeDiv.append(txTypeIcon);
var txSummary = document.createElement('div');
txSummary.className = 'wmeFdo-tx-summary wmeFdoDropdownHeader';
txHeader.append(txSummary);
var txCollapse = document.createElement('div');
txCollapse.className = 'flex-noshrink wmeFdo-tx-toggle-closed';
txHeader.append(txCollapse);
var txAuthorDate = document.createElement('div');
txAuthorDate.className = 'wmeFdo-tx-author-date';
var txPreview = document.createElement('div');
txPreview.className = 'wmeFdo-tx-preview';
var txName = document.createElement('h4');
//txName.className = 'type';
txAuthorDate.append(txName);
txSummary.append(txAuthorDate);
txSummary.append(txPreview);
var txContent = document.createElement('div');
txContent.className = 'wmeFdo-tx-content wmeFdoResultDetail';
var txContentClass = document.createElement('div');
txContentClass.className = 'main-object-region wmeFdo-tx-changes';
txContent.append(txContentClass);
var txContentInfo = document.createElement('div');
txContentInfo.className = 'related-objects-region wmeFdo-tx-changes';
var txContentData = document.createElement('div');
var controls = document.createElement('div');
controls.className = 'wmeFdoDetailControls';
txContent.append(controls);
txContentInfo.append(txContentData);
txContent.append(txContentInfo);
details.append(txHeader);
details.append(txContent);
var venueItem = venueAction.oldValue;
if (typeof(venueItem.geometry) == 'undefined') {
// History does not contain any details for this place, other than it was deleted.
// TODO: create empty record to at least show that there was something here at one point.
console.log(`No history found for venue id ${venueAction.objectID}`);
} else {
if (typeof(venueItem.categories) != 'undefined' && venueItem.categories[0] == 'RESIDENCE_HOME') {
txTypeIcon.className = 'flex-noshrink wmeFdoResultTypeResidential';
if (historySummary.isRejectedNew) {
txTypeIcon.classList.add('wmeFdoResultDeletedNewPlaceAlt');
} else if (historySummary.isAcceptedDelete) {
txTypeIcon.classList.add('wmeFdoResultDeletedFromRequestAlt');
}
} else if (typeof(venueItem.categories) != 'undefined' && venueItem.categories[0] == 'PARKING_LOT') {
txTypeIcon.className = 'flex-noshrink wmeFdoResultTypeParking';
if (historySummary.isRejectedNew) {
txTypeIcon.classList.add('wmeFdoResultDeletedNewPlaceAlt');
} else if (historySummary.isAcceptedDelete) {
txTypeIcon.classList.add('wmeFdoResultDeletedFromRequestAlt');
}
} else {
txTypeIcon.className = 'flex-noshrink wmeFdoResultTypeVenue';
if (historySummary.isRejectedNew) {
txTypeIcon.classList.add('wmeFdoResultDeletedNewPlace');
} else if (historySummary.isAcceptedDelete) {
txTypeIcon.classList.add('wmeFdoResultDeletedFromRequest');
}
}
if (historySummary.isRejectedNew) {
details.setAttribute("data-pur-new-rejected", "true");
}
if (venueItem.geometry.type == 'Polygon') {
var txTypeArea = document.createElement('div');
txTypeArea.className = 'flex-noshrink wmeFdoResultTypeArea';
txTypeDiv.append(txTypeArea);
}
let streetName = formatFullStreetName(venueHistory, venueItem.streetID);
let address = streetName;
if (typeof(venueItem.houseNumber) != 'undefined') {
address = venueItem.houseNumber + ' ' + address;
}
var venueName = venueItem.name;
if (typeof venueName == 'undefined' || venueName == " " || venueName == "")
{
if (typeof(venueItem.categories) != 'undefined' && venueItem.categories[0] == 'RESIDENCE_HOME') {
let resiStreet = formatStreetName(venueHistory, venueItem.streetID);
if (typeof(venueItem.houseNumber) != 'undefined') {
resiStreet = venueItem.houseNumber + ' ' + resiStreet;
}
venueName = resiStreet;
} else if (typeof(venueItem.categories) != 'undefined') {
venueName = 'unnamed ' + I18n.t("venues.categories." + venueItem.categories[0]);
} else {
venueName = 'unnamed';
//console.log('No category: ' + venueAction.objectID);
}
}
txName.innerText = venueName;
if (historySummary.purDeletedBy != null)
{
username = historySummary.purDeletedBy;
}
txPreview.innerHTML = streetName + '<br>Deleted by ' + username + " " + timeConverterNoTime(venueTransaction.date);
if (historySummary.isAcceptedDelete) {
//txPreview.innerHTML += " via PUR";
details.setAttribute("data-pur-delete-request", "true");
}
var info = document.createElement('div');
info.id = 'details-' + venueAction.objectID;
let flagControl = document.createElement('div');
flagControl.className = 'focus-buttons';
let flag = document.createElement('a');
flag.className = 'wmeFdoFlag';
if (flags['venue'][venueAction.objectID]) { flag.className = 'wmeFdoFlag flagged'; }
flag.innerHTML = '<span class="fa fa-flag"></span>';
flag.onclick = function () {
toggleFlag(flag, 'venue', venueAction.objectID);
};
flagControl.append(flag);
controls.append(flagControl);
var focusButtons = document.createElement('div');
focusButtons.className = 'focus-buttons';
var focus = document.createElement('a');
focus.className = 'focus';
focus.onclick = function () {
var lonLat = new OpenLayers.LonLat(operationCenterPoint.x, operationCenterPoint.y);
W.map.setCenter(lonLat, 6);
highlightPin(pin);
};
focusButtons.append(focus);
controls.append(focusButtons);
let lockRank = venueItem.lockRank +1;
txContentData.append(info);
let searchText = venueItem.name + " " + streetName;
var searchButton = document.createElement('button');
searchButton.textContent = 'Google';
searchButton.className = 'wmeFdoGoogleBtn btn btn-default center-block';
searchButton.addEventListener('click', function() {
window.open('https://www.google.com/search?q=' + encodeURIComponent(searchText),'_blank');
});
let recreateDiv = document.createElement('div');
recreateDiv.className = 'wmeFdoRecreate';
let recreateButton = document.createElement('a');
recreateButton.textContent = 'Recreate Place';
recreateButton.className = 'focus';
recreateButton.addEventListener('click', function() {
recreatePlace(venueItem, venueHistory, operationCenterPoint);
});
recreateDiv.append(recreateButton);
txContentData.append(searchButton);
txContentData.append(recreateDiv);
let panelHTML = '<ul>';
//panelHTML += '<ul>';
let venueIdReference = 'details-venue-id-' + venueAction.objectID;
panelHTML+= '<li class="tx-changed-attribute"><div class="ca-name">Venue ID</div>';
panelHTML += '<div class="ca-description ca-preview" id="' + venueIdReference + '">';
panelHTML += '</div></li>';
if (address != null && address != "" && address != " ") {
panelHTML+= '<li class="tx-changed-attribute"><div class="ca-name">Address</div>';
panelHTML += '<div class="ca-description ca-preview">';
panelHTML += address;
panelHTML += '</div></li>';
}
panelHTML+= '<li class="tx-changed-attribute"><div class="ca-name">Categories</div>';
panelHTML += '<div class="ca-description ca-preview">';
panelHTML += formatCategories(venueItem.categories);
details.setAttribute("data-categories", formatCategoriesNative(venueItem.categories));
panelHTML += '</div></li>';
if (typeof(venueItem.description) != 'undefined') {
panelHTML+= '<li class="tx-changed-attribute"><div class="ca-name">Description</div>';
panelHTML += '<div class="ca-description ca-preview">';
panelHTML += venueItem.description;
panelHTML += '</div></li>';
}
if (typeof(venueItem.url) != 'undefined') {
panelHTML+= '<li class="tx-changed-attribute"><div class="ca-name">Website</div>';
panelHTML += '<div class="ca-description ca-preview">';
panelHTML += venueItem.url;
panelHTML += '</div></li>';
}
if (typeof(venueItem.phone) != 'undefined') {
panelHTML+= '<li class="tx-changed-attribute"><div class="ca-name">Phone</div>';
panelHTML += '<div class="ca-description ca-preview">';
panelHTML += venueItem.phone;
panelHTML += '</div></li>';
}
if (typeof(venueItem.lockRank) != 'undefined') {
panelHTML+= '<li class="tx-changed-attribute"><div class="ca-name">Lock Rank</div>';
panelHTML += '<div class="ca-description ca-preview">';
panelHTML += lockRank;
panelHTML += '</div></li>';
}
if (typeof(venueItem.externalProviderIDs) != 'undefined') {
panelHTML+= '<li class="tx-changed-attribute"><div class="ca-name">External Provider IDs</div>';
panelHTML += '<div class="ca-description ca-preview">';
panelHTML += '<div id="wmeFdoExternalProviders-' + venueAction.objectID + '"></div>';
panelHTML += '</div></li>';
}
panelHTML+= '<li class="tx-changed-attribute"><div class="ca-name">History</div>';
panelHTML += '<div class="ca-description ca-preview">';
panelHTML += historySummary.html;
panelHTML += '</div></li>';
panelHTML += '</ul>';
info.innerHTML = panelHTML;
//info.innerHTML += 'Changed from';
//info.innerHTML += '<span class="ca-value ca-value-old">';
//info.innerHTML += 'Walmart';
//info.innerHTML += '</span>';
//info.innerHTML += '<span>';
//info.innerHTML += 'to';
//info.innerHTML += '</span>';
//info.innerHTML += '<span class="ca-value ca-value-new">';
//info.innerHTML += 'Walmart Supercenter';
//info.innerHTML += '</span>';
$('#wmeFdoTransactionList').append(details);
let venueIdDiv = document.getElementById(venueIdReference);
let venueIdLi = document.getElementById(details.id);
setFilteredState(venueIdLi);
let venueIdDisplay = document.createElement('span');
if (W.loginManager.user.userName == "Joyriding") {
venueIdDisplay.innerHTML = `<a href="https://${window.location.host}${W.Config.api_base}/ElementHistory?objectType=venue&objectID=${venueAction.objectID}" target="_blank">${venueAction.objectID}</a> `;
} else {
venueIdDisplay.innerHTML = venueAction.objectID;
}
venueIdDiv.append(venueIdDisplay);
let pin = displayPin(operation, venueAction.oldValue);
let areaPreview = null;
details.onmouseenter = function () {
highlightPin(pin);
areaPreview = showDeletedArea(operation, venueAction.oldValue);
};
details.onmouseleave = function () {
returnPin(pin);
//removeDeletedArea();
_wmeFdoFeatureLayer.removeFeatures([areaPreview]);
};
}
}
}
}
}
});
});
}
});
}
function toggleFlag (control, objectType, objectID) {
if (flags[objectType][objectID]) {
delete flags[objectType][objectID];
control.className = 'wmeFdoFlag';
} else {
flags[objectType][objectID] = true;
control.className = 'wmeFdoFlag flagged';
}
}
function displayDetails() {
}
async function recreatePlace(origVenue, venueHistory, operationCenterPoint) {
let lonLat = new OpenLayers.LonLat(operationCenterPoint.x, operationCenterPoint.y);
if(!W.map.getExtent().containsLonLat(lonLat)) {
// Move to pin so that the City can be populated correctly
W.map.setCenter(lonLat, 6);
await wait(1000); //TODO: Check to see that cities have loaded before continuing rather than guessing a time delay
}
let wazeActionUpdateFeatureGeometry = require("Waze/Action/UpdateFeatureGeometry");
let wazefeatureVectorLandmark = require("Waze/Feature/Vector/Landmark");
let wazeActionAddLandmark = require("Waze/Action/AddLandmark");
let wazeActionUpdateObject = require('Waze/Action/UpdateObject');
let wazeActionUpdateFeatureAddress = require('Waze/Action/UpdateFeatureAddress');
let wazeOpeningHour = require("Waze/Model/Objects/OpeningHour");
let landmark = new wazefeatureVectorLandmark();
if (origVenue.geometry.type == 'Polygon') {
let points = [];
origVenue.geometry.coordinates[0].forEach( function(coord) {
let point = new OpenLayers.Geometry.Point(coord[0],coord[1]);
point.transform(displayProjection,projection);
points.push(point);
});
points.push(points[0]);
let linearRing = new OpenLayers.Geometry.LinearRing(points);
let polygon = new OpenLayers.Geometry.Polygon(linearRing);
landmark.geometry = polygon;
} else {
let point = new OpenLayers.Geometry.Point(origVenue.geometry.coordinates[0],origVenue.geometry.coordinates[1]);
point.transform(displayProjection,projection);
landmark.geometry = point;
}
if (typeof(origVenue.name) != 'undefined') { landmark.attributes.name = origVenue.name; }
if (typeof(origVenue.aliases) != 'undefined') { landmark.attributes.aliases = origVenue.aliases; }
if (typeof(origVenue.categories) != 'undefined') { landmark.attributes.categories = origVenue.categories; }
if (typeof(origVenue.description) != 'undefined') { landmark.attributes.description = origVenue.description; }
if (typeof(origVenue.lockRank) != 'undefined') { landmark.attributes.lockRank = origVenue.lockRank; }
if (typeof(origVenue.phone) != 'undefined') { landmark.attributes.phone = origVenue.phone; }
if (typeof(origVenue.rank) != 'undefined') { landmark.attributes.rank = origVenue.rank; }
if (typeof(origVenue.residential) != 'undefined') { landmark.attributes.residential = origVenue.residential; }
if (typeof(origVenue.services) != 'undefined') { landmark.attributes.services = origVenue.services; }
if (typeof(origVenue.url) != 'undefined') { landmark.attributes.url = origVenue.url; }
if (origVenue.categories.indexOf('GAS_STATION') > -1 || origVenue.categories.indexOf('PARKING_LOT') > -1) {
if (typeof(origVenue.brand) != 'undefined') { landmark.attributes.brand = origVenue.brand; }
}
if (origVenue.categories.indexOf('PARKING_LOT') > -1)
{
if (typeof(origVenue.categoryAttributes) != 'undefined') { landmark.attributes.categoryAttributes = origVenue.categoryAttributes; }
}
if (typeof(origVenue.entryExitPoints) != 'undefined') { //TODO: and if # entry points > 0
origVenue.entryExitPoints.forEach(function(origEntryExitPoint) {
let point = new OpenLayers.Geometry.Point(origEntryExitPoint.point.coordinates[0],origEntryExitPoint.point.coordinates[1]);
point.transform(displayProjection, projection);
let entryExitPoint = new NavigationPointMEP(point);
entryExitPoint._name = origEntryExitPoint.name;
entryExitPoint._entry = origEntryExitPoint.entry;
entryExitPoint._exit = origEntryExitPoint.exit;
entryExitPoint._isPrimary = origEntryExitPoint.primary;
landmark.attributes.entryExitPoints.push(entryExitPoint);
});
}
W.model.actionManager.add(new wazeActionAddLandmark(landmark));
if (typeof(origVenue.openingHours) != 'undefined') {
let hours = [];
origVenue.openingHours.forEach(function (hourEntry) {
hours.push(new wazeOpeningHour(hourEntry));
});
W.model.actionManager.add(new wazeActionUpdateObject(landmark, { openingHours: hours }));
}
if (typeof(origVenue.streetID) != 'undefined') {
//TODO: Make this a separate function and use for all address building, cache after built by streetID
var addressParts = {
streetName: "",
emptyStreet: false,
cityName: "",
emptyCity: false,
streetID: origVenue.streetID,
stateID: null,
countryID: null,
addressFormShown: false,
editable: true,
fullAddress: "",
ttsLocales: [W.Config.tts.default_locale],
altStreets: new Backbone.Collection,
newAltStreets: new Backbone.Collection
};
venueHistory.streets.objects.forEach(function(street) {
if (street.id == origVenue.streetID) {
if (street.name == "")
addressParts.emptyStreet = true;
addressParts.streetName = street.name;
venueHistory.cities.objects.forEach(function(city) {
if (street.cityID == city.id) {
if (city.name == "")
addressParts.emptyCity = true;
addressParts.cityName = city.name;
addressParts.stateID = city.stateID;
venueHistory.states.objects.forEach(function(state) {
if (city.stateID == state.id) {
addressParts.countryID = state.countryID;
}
});
}
});
}
});
W.model.actionManager.add(new wazeActionUpdateFeatureAddress(landmark, addressParts,{streetIDField: 'streetID'}));
}
if (typeof(origVenue.houseNumber) != 'undefined') {
W.model.actionManager.add(new wazeActionUpdateObject(landmark,{houseNumber: origVenue.houseNumber}));
}
//console.log(landmark);
let foundFeature = null;
W.selectionManager._layers.forEach(function(layer) {
if (layer.featureType == 'venue') {
layer.features.forEach(function(feature) {
if (landmark.geometry.id == feature.geometry.id)
{
foundFeature = feature;
}
});
}
});
if (foundFeature != null) {
W.selectionManager.selectFeature(foundFeature);
W.selectionManager._triggerSelectionChanged()
}
var epm = Backbone.Model.extend({
defaults: {
uuid: null,
name: null,
url: null,
location: null
},
initialize: function() {
if (null === this.get("uuid")) return this.set({
uuid: this.id
})
},
getDetails: function(e) {
this.set({uuid: e});
return this.set({
name: externalProviderDetails[e].name,
location: externalProviderDetails[e].geometry.location,
url: externalProviderDetails[e].url
});
},
_getDetailsFromUuid: function(e) {
return this.set({
name: this.get("uuid"),
location: "",
url: ""
})
},
toJSON: function() {
return this.get("uuid")
}
});
if (typeof(origVenue.externalProviderIDs) != 'undefined') {
let ids = [];
origVenue.externalProviderIDs.forEach(function(externalProviderID) {
let newID = new epm();
newID.getDetails(externalProviderID);
newID.collection = new Backbone.Collection();
ids.push(newID);
}); //10521
W.model.actionManager.add(new wazeActionUpdateObject(landmark,{externalProviderIDs: ids}));
}
}
function displayExternalProviders(venueID, externalProviderIDs) {
if (typeof(externalProviderIDs) != 'undefined') {
externalProviderIDs.forEach(function(externalProviderID) {
let extProv = document.getElementById('wmeFdoExternalProviders-' + venueID);
let div = document.createElement('div');
div.innerHTML = externalProviderID;
extProv.append(div);
getExternalProviderDetail(externalProviderID, function(data) {
let closed = data.permanently_closed;
div.innerHTML = '<a href="' + data.url + '" target="_blank">' + externalProviderID + '</a>';
if (closed) {
div.innerHTML = div.innerHTML + ' <span style="color:red;">closed</span>';
}
});
});
}
}
function getExternalProviderDetail(externalProviderID, callback) {
if (googleMapsApi.service == null) {
googleMapsApi.service = new google.maps.places.PlacesService(googleMapsApi.map);
}
if (typeof(externalProviderDetails[externalProviderID]) == 'undefined') {
let request = {
placeId: externalProviderID,
fields: ['name','url','permanently_closed','geometry']
}
googleMapsApi.service.getDetails(request, function(place, status) {
if (status == google.maps.places.PlacesServiceStatus.OK) {
externalProviderDetails[externalProviderID] = place;
callback(externalProviderDetails[externalProviderID]);
} else {
//TODO: callback with error response
callback(null);
}
});
} else {
callback(externalProviderDetails[externalProviderID]);
}
}
function toggleFilterDropDown() {
let details = document.getElementById('wmeFdoFilterDetails');
let openDropDown = false;
if (details.classList.contains('closed')) {
details.classList.remove('closed');
} else {
details.classList.add('closed');
}
}
function toggleScanStatusDropDown() {
let details = document.getElementById('wmeFdoScanStatus');
let openDropDown = false;
if (details.classList.contains('closed')) {
details.classList.remove('closed');
} else {
details.classList.add('closed');
}
}
function toggleDetailsDropDown(elementId, operationCenterPoint) {
let details = document.getElementById(elementId);
let openDropDown = false;
if (details.classList.contains('closed')) {
openDropDown = true;
}
closeAllDetailsDropDown();
if (openDropDown) {
details.classList.remove('closed');
settings.Enabled = true;
onLayerCheckboxChanged(settings.Enabled); // TODO: Fix this so the Layer menu checkbox gets updated
let lonLat = new OpenLayers.LonLat(operationCenterPoint.x, operationCenterPoint.y);
if(!W.map.getExtent().containsLonLat(lonLat)) {
W.map.setCenter(lonLat, 6);
}
}
}
function openDetailsDropDown(elementId, operationCenterPoint) {
let details = document.getElementById(elementId);
//details.className = 'element-history-item tx-has-content';
details.classList.remove('closed');
details.classList.remove('hidden');
// var lonLat = new OpenLayers.LonLat(operationCenterPoint.x, operationCenterPoint.y);
//W.map.setCenter(operationCenterPoint.toLonLat(), 6);
}
function closeDetailsDropDown(elementId) {
let details = document.getElementById(elementId);
details.classList.add('closed');
}
function closeAllDetailsDropDown() {
let ul = document.getElementById('wmeFdoTransactionList');
let items = ul.getElementsByTagName('li');
for (var i = 0; i < items.length; ++i) {
let elementId = items[i].id;
if (elementId.startsWith('entry-')) {
closeDetailsDropDown(elementId);
}
}
}
function displayPin(transactionOperation, deletedVenue) {
let deletedPlacePt=new OpenLayers.Geometry.Point(transactionOperation.objectCentroid.coordinates[0], transactionOperation.objectCentroid.coordinates[1]);
deletedPlacePt.transform(displayProjection, projection);
let point = new OpenLayers.Geometry.Point(deletedPlacePt.x, deletedPlacePt.y);
let style = {strokeColor: 'black',
strokeWidth: '5',
strokeDashstyle: 'solid',
pointRadius: '5',
fillOpacity: '1'};
let feature = new OpenLayers.Feature.Vector(point, {id:transactionOperation.objectID}, style);
var di = require("Waze/DivIcon");
var iconA = new di("place-update add_venue low map-marker wmeFdo-place-delete");
var lonlatA = new OpenLayers.LonLat(deletedPlacePt.x, deletedPlacePt.y);
// let markerLayer = W.map.getLayerByUniqueName('__wmeFdoMarkerLayer');
let marker = new OpenLayers.Marker(lonlatA, iconA);
marker.id = 1;
marker.events.register('click', marker, function(evt) {
W.selectionManager.unselectAll();
selectTab();
let elementId = 'entry-' + transactionOperation.objectID;
closeAllDetailsDropDown();
openDetailsDropDown(elementId, deletedPlacePt);
document.getElementById(elementId).scrollIntoView();
});
addLayers();
_wmeFdoFeatureLayer.addFeatures([feature]);
_wmeFdoMarkerLayer.addMarker(marker);
return feature;
}
function addLayers() {
if (_wmeFdoFeatureLayer == undefined) {
_wmeFdoFeatureLayer = new OpenLayers.Layer.Vector("_wmeFdoFeatureLayer",{uniqueName: "__wmeFdoFeatureLayer"});
_wmeFdoMarkerLayer = new OpenLayers.Layer.Markers("_wmeFdoMarkerLayer",{uniqueName: "__wmeFdoMarkerLayer"});
W.map.addLayer(_wmeFdoFeatureLayer);
W.map.addLayer(_wmeFdoMarkerLayer);
if (settings.Enabled) {
_wmeFdoFeatureLayer.setVisibility(true);
_wmeFdoMarkerLayer.setVisibility(true);
} else {
_wmeFdoFeatureLayer.setVisibility(false);
_wmeFdoMarkerLayer.setVisibility(false);
}
}
}
function showDeletedArea(transactionOperation, deletedVenue) {
let polygon = null;
if (deletedVenue.geometry.type == 'Polygon') {
let points = [];
deletedVenue.geometry.coordinates[0].forEach( function(coord) {
let point = new OpenLayers.Geometry.Point(coord[0],coord[1]);
point.transform(displayProjection,projection);
points.push(point);
});
points.push(points[0]);
let linearRing = new OpenLayers.Geometry.LinearRing(points);
polygon = new OpenLayers.Geometry.Polygon(linearRing);
}
let style = {strokeColor: 'orange',
strokeWidth: '4',
strokeDashstyle: 'solid',
fillOpacity: '0.2'};
let feature = new OpenLayers.Feature.Vector(polygon, {id:transactionOperation.objectID}, style);
_wmeFdoFeatureLayer.addFeatures([feature]);
return feature;
}
function getParentCategory(category) {
// Build subcategory > parent category mapping once
if (subcategoryToParentCategories.length == 0) {
let index = 0;
Object.keys(W.Config.venues.subcategories).forEach(function(parent) {
let parentName = Object.getOwnPropertyNames(W.Config.venues.subcategories)[index];
subcategoryToParentCategories[parentName] = parentName;
W.Config.venues.subcategories[parentName].forEach(function(subcategory) {
subcategoryToParentCategories[subcategory] = parentName;
});
index++;
});
}
return subcategoryToParentCategories[category];
}
function getCategoryIconClass(category) {
let parentCategory = getParentCategory(category);
// TODO: Lowercase and replace '_' with '-' instead?
switch(parentCategory) {
case 'TRANSPORTATION':
return 'transportation';
break;
case 'PROFESSIONAL_AND_PUBLIC':
return 'professional-and-public';
break;
case 'SHOPPING_AND_SERVICES':
return 'shopping-and-services';
break;
case 'FOOD_AND_DRINK':
return 'food-and-drink';
break;
case 'CULTURE_AND_ENTERTAINEMENT':
return 'culture-and-entertainement';
break;
case 'LODGING':
return 'lodging';
break;
case 'OUTDOORS':
return 'outdoors';
break;
case 'NATURAL_FEATURES':
return 'natural-features';
break;
case 'PARKING_LOT':
return 'parking-lot';
default:
return 'OTHER';
}
// WME CSS Example:
// #edit-buttons .transportation .item-icon::after {
// background-image: url(//editor-assets.waze.com/production/img/toolbare2f6b31….png);
// background-position: -13px -70px;
// width: 12px;
// height: 12px;
// }
}
function formatStreetName(venueHistory, streetId) {
// TODO: replace with address cache lookup/build function
let streetName = null;
venueHistory.streets.objects.forEach(function(street) {
if (street.id == streetId) {
streetName = street.name;
}
});
if (streetName == null) {
streetName = "";
}
return streetName;
}
function formatFullStreetName(venueHistory, streetId) {
// TODO: replace with address cache lookup/build function
let streetName = null;
venueHistory.streets.objects.forEach(function(street) {
if (street.id == streetId) {
streetName = street.name;
venueHistory.cities.objects.forEach(function(city) {
if (street.cityID == city.id) {
if (streetName != '') {
streetName = streetName + ", ";
}
streetName = streetName + city.name;
venueHistory.states.objects.forEach(function(state) {
if (city.stateID == state.id) {
if (city.name != '') {
streetName = streetName + ", ";
}
streetName = streetName + state.name;
}
});
}
});
}
});
if (streetName == null) {
streetName = "";
}
return streetName;
}
function formatCategories(categories) {
let output = ''
categories.forEach(function(category) {
output = output + ", " + I18n.t("venues.categories." + category);
});
output = output.substring(2);
return output;
}
function formatCategoriesNative(categories) {
let output = ''
categories.forEach(function(category) {
output = output + "," + category;
});
output = output.substring(1);
return output;
}
function formatExternalProviders(externalProviderIDs) {
let output = "<ul>";
if (typeof externalProviderIDs != 'undefined') {
externalProviderIDs.forEach(function(externalProviderID) {
output = output + "<li>" + externalProviderID + "</ul>";
});
}
output = output + "</ul>";
return output;
}
function getVenueHistory(venueHistory) {
let historySummary = {
html: null,
isRejectedNew: false,
isAcceptedDelete: false,
purDeletedBy: null
}
let output = "<ul class='wmeFdoHistorySummaryList'>";
let transactionCount = 0;
venueHistory.transactions.objects.forEach(function(venueTransaction) {
transactionCount++;
let username = null;
let usernameHtml = null;
venueHistory.users.objects.forEach( function(user) {
if (user.id == venueTransaction.userID) {
let userRank = user.rank + 1;
username = user.userName + '(' + userRank + ')';
usernameHtml = `<a target="_blank" href="https://${window.location.host}/user/editor/${user.userName}">${username}</a>`;
}
});
let detailCount = 0;
let requestType = venueTransaction.actionType;
let purText = null;
let addedPurNotation = false;
let previousObjectType = null;
venueTransaction.objects.forEach(function(venueSubObject) {
purText = null;
let addDate = false;
if (venueSubObject.objectType == 'venueUpdateRequest') {
if (transactionCount == 2 && previousObjectType == 'venue') {
historySummary.purDeletedBy = username;
}
previousObjectType = venueSubObject.objectType;
if (!addedPurNotation) {
addedPurNotation = true;
requestType = requestType + " (PUR)";
}
if (typeof(venueSubObject.oldValue) != 'undefined') {
let rejected = false;
if (typeof(venueSubObject.oldValue.approve) == 'undefined') {
purText = "(Unknown Action) ";
} else if (venueSubObject.oldValue.approve) {
purText = "Approved ";
} else {
purText = "Rejected ";
rejected = true;
}
if (typeof(venueSubObject.oldValue.type) == 'undefined'
&& (venueSubObject.oldValue.id.includes('-') && !venueSubObject.oldValue.id.includes('.'))
) {
// Older entries did not contain the type, test to see if it's a rejected image
venueSubObject.oldValue.type = 'IMAGE';
}
if (venueSubObject.oldValue.type == 'VENUE') {
//rejectedNew = true;
purText = purText + "New Place";
addDate = true;
if (rejected) {
historySummary.isRejectedNew = true;
}
} else if (venueSubObject.oldValue.type == 'REQUEST') {
if (venueSubObject.oldValue.subType == 'DELETE') {
purText = purText + "Delete Request";
if (!rejected) {
historySummary.isAcceptedDelete = true;
}
} else if (venueSubObject.oldValue.subType == 'UPDATE') {
purText = purText + "Update Request";
}
addDate = true;
} else if (venueSubObject.oldValue.type == 'IMAGE') {
purText += "Photo";
purText = `<a target="_blank" href="https://venue-image.waze.com/${venueSubObject.oldValue.id}">${purText}</a>`;
}
if (addDate) {
let origPurUnixTime = parseInt(venueSubObject.oldValue.id.split('.')[0]);
let newPurDate = timeConverter(origPurUnixTime);
purText = purText + " (" + newPurDate + ")";
}
} else if (typeof(venueSubObject.newValue) != 'undefined') {
purText = "Submitted ";
if (typeof(venueSubObject.newValue.type) != 'undefined') {
if (venueSubObject.newValue.type == 'VENUE') {
//rejectedNew = true;
purText = purText + "New Place";
addDate = true;
} else if (venueSubObject.newValue.type == 'REQUEST') {
if (venueSubObject.newValue.subType == 'DELETE') {
purText = purText + "Delete Request";
} else if (venueSubObject.newValue.subType == 'UPDATE') {
purText = purText + "Update Request";
}
addDate = true;
}
} else if (venueSubObject.newValue.id.includes('-') && !venueSubObject.newValue.id.includes('.')) {
// Older entries did not contain the type, test to see if it's a rejected image
//venueSubObject.oldValue.type = 'IMAGE';
purText += "Photo";
purText = `<li><a target="_blank" href="https://venue-image.waze.com/${venueSubObject.newValue.id}">Rejected Photo</a></li>`;
}
if (addDate) {
let origPurUnixTime = parseInt(venueSubObject.newValue.id.split('.')[0]);
let newPurDate = timeConverter(origPurUnixTime);
purText = purText + " (" + newPurDate + ")";
}
}
}
});
output = output + "<li><div>" + timeConverter(venueTransaction.date) + " " + requestType + " " + usernameHtml + "</div>";
output = output + '<ul class="historySummaryItem">';
if (purText != null) {
output = output + '<li>' + purText + '</li>';
}
if (typeof(venueTransaction.objects[0].newValue) != 'undefined' && typeof(venueTransaction.objects[0].newValue.name) != 'undefined')
{
detailCount++;
output = output + '<li>Name: ' + venueTransaction.objects[0].newValue.name + '</li>';
if (venueTransaction.objects[0].actionType == 'UPDATE') {
output = output + '<li>Was: ' + venueTransaction.objects[0].oldValue.name + '</li>';
}
}
// if (typeof(venueTransaction.objects[0].oldValue) != 'undefined' && venueTransaction.objects[0].oldValue.type == 'IMAGE') {
// output = output + `<li><a target="_blank" href="https://venue-image.waze.com/${venueTransaction.objects[0].oldValue.id}">Rejected Photo</a></li>`;
// }
output = output + '</ul>';
if (detailCount > 0) {
output = output + '</li style="margin-bottom:4px">';
} else {
output = output + "</li>";
}
// console.log(venueTransaction);
});
output = output + "</ul>";
historySummary.html = output;
return historySummary;
}
function applyFiltersToResults() {
let ul = document.getElementById('wmeFdoTransactionList');
let items = ul.getElementsByTagName('li');
for (var i = 0; i < items.length; ++i) {
let elementId = items[i].id;
let element = document.getElementById(elementId);
if (elementId.startsWith('entry-')) {
setFilteredState(element);
}
}
}
function setFilteredState(element) {
let filtered = false;
let filterTypeCount = 0;
if (!settings.FilterResidential) {
filterTypeCount++;
if (element.getAttribute('data-categories') == 'RESIDENCE_HOME') {
filtered = true;
}
}
if (!settings.FilterParkingLot) {
filterTypeCount++;
if (element.getAttribute('data-categories') == 'PARKING_LOT') {
filtered = true;
}
}
if (!settings.FilterAnyType) {
filterTypeCount++;
if (element.getAttribute('data-categories') != 'RESIDENCE_HOME' && element.getAttribute('data-categories') != 'PARKING_LOT') {
filtered = true;
}
}
let allTypesFiltered = false;
if (filterTypeCount == 3) {
allTypesFiltered = true;
}
if (element.getAttribute('data-pur-new-rejected') == "true") {
if (!settings.FilterNewRejected) {
filtered = true;
} else if (allTypesFiltered && filtered) {
filtered = false;
}
}
//
if (element.getAttribute('data-pur-delete-request') == "true") {
if (!settings.FilterDeleteRequest) {
filtered = true;
} else if (allTypesFiltered && filtered) {
filtered = false;
}
}
if (filtered) {
element.classList.add('hidden');
} else {
element.classList.remove('hidden');
}
}
function endScan(reasonText, setEndedFlag) {
if (setEndedFlag) {
var findButton = document.getElementById('wmeFdoFindButton');
findButton.className = 'btn btn-success center-block';
var cancelButton = document.getElementById('wmeFdoCancelButton');
cancelButton.className = 'btn btn-warning center-block hidden';
$('#wmeFdoTabFind').data("status","ended");
let tab = document.getElementById('wmeFdoTab');
tab.innerHTML = 'FiDO <span class="spinner-disable"><i class="icon-spinner icon-spin fa fa-spinner fa-spin"></i></span>';
cancelScanNew();
}
$('#wmeFdoScanCurrentStatus')[0].innerText = reasonText;
}
async function findObjects() {
let editorList = [];
let searchAreaGeometry = getSearchArea();
var lookFor = {
objectType:'Venue',
actionType:'DELETE',
searchArea: searchAreaGeometry,
limitToScreen: settings.LimitToScreen,
includeSelf: settings.IncludeSelf
}
//var centerCoords = W.map.getCenter().clone().transform(W.map.projection, W.map.displayProjection);
if (settings.NearbyEditors) {
lookFor.limitToScreen = true;
lookFor.includeSelf = true;
//TODO: use W.model if zoom <= 3
let editorListScreen = getNearbyEditors();
let featuresList = await getScreenFeatures();
let editorListFeatures = getNearbyEditorsFromFeatures(featuresList);
let editorListTemp = [];
editorListFeatures.forEach(function(userID) {
editorListTemp[userID] = true;
apiEndpointWazeCom.users.total++;
});
Object.keys(editorListTemp).forEach(function(userID) {
editorList.push(userID);
});
processUsers(lookFor, editorList);
} else {
lookFor.limitToScreen = settings.LimitToScreen;
lookFor.includeSelf = settings.IncludeSelf;
Date.prototype.addDays = function(days) {
var date = new Date(this.valueOf());
date.setDate(date.getDate() + days);
return date;
}
let startDate = null;
let startDateInput = document.getElementById('wmeFdoFindDate');
let dateInput = null;
if (Date.parse(startDateInput.value) != null) {
dateInput = new Date(startDateInput.value).addDays(1);
startDate = dateToUuid(dateInput);
dateInput = new Date(dateInput).getTime();
} else {
dateInput = new Date.now();
}
var transactionLookup = {
username:$('#wmeFdoEditorName')[0].value,
userId:null,
nextTransactionDate:startDate,
iteration:0,
maxIterations:100000
};
if (transactionLookup.username == null || transactionLookup.username == "") {
endScan('Please enter an editor name!', true);
} else {
let requestParamsUserProfile = requestParams(getApiUrlUserProfile(transactionLookup.username));
requestStatus[requestParamsUserProfile.url] = defaultRequestStatus();
apiEndpointWazeCom.requests.push(requestParamsUserProfile);
getEndpointResult(apiEndpointWazeCom, requestParamsUserProfile, userProfile => {
if (typeof(userProfile.userID) == 'undefined') {
endScan('Editor name not found!', true);
} else {
apiEndpointWazeCom.users.total++;
addUserCache(userProfile.userID, userProfile.username, userProfile.rank)
if (userProfile.firstEditDate > earliestScanDate) {
userCache[userProfile.userID].earliestDate = userProfile.firstEditDate;
} else {
userCache[userProfile.userID].earliestDate = earliestScanDate;
}
userCache[userProfile.userID].latestDate = dateInput;
editorList.push(userProfile.userID);
}
processUsers(lookFor, editorList);
});
}
}
}
function processUsers(lookFor, editorList) {
$('#wmeFdoScanStatus .wmeFdoProgress').removeClass('hidden');
apiEndpointWazeCom.gotFeatures = true;
editorList.forEach(function (userID) {
Date.prototype.addDays = function(days) {
var date = new Date(this.valueOf());
date.setDate(date.getDate() + days);
return date;
}
let startDate = null;
let startDateInput = document.getElementById('wmeFdoFindDate');
if (Date.parse(startDateInput.value) != null) {
let dateInput = new Date(startDateInput.value).addDays(1);
startDate = dateToUuid(dateInput);
}
var transactionLookup = {
username:null,
userId:userID,
nextTransactionDate:startDate,
iteration:0,
maxIterations:100000
};
let latestDate = timeConverter(userCache[userID].latestDate + DAY_MS);
transactionLookup.nextTransactionDate = dateToUuid(latestDate);
let username = `<a target="_blank" href="https://${window.location.host}/user/editor/${userCache[userID].username}">${userCache[userID].username}(${(userCache[userID].rank + 1)})</a>`;
$('#wmeFdoStatusUsersList').append('<li id="wmeFdoStatusUser-' + userID + '">'
+ username + ': <span class="wmeFdoStatusUser">Not Started</span>'
+ '<div class="wmeFdoProgress"><div class="wmeFdoComplete"></div></div>'
+ '<div class="wmeFdoStatusUserDates">'
+ '<div class="wmeFdoStatusUserLatest">' + timeConverterNoTime(userCache[userID].latestDate) + '</div>'
+ '<div class="wmeFdoStatusUserEarliest">' + timeConverterNoTime(userCache[userID].earliestDate) + '</div>'
+ '<div style="clear:both;"></div>'
+ '</div>'
+ '</li>');
userCache[userID].previousDate = userCache[userID].latestDate + DAY_MS;
let totalMs = (userCache[userID].latestDate + DAY_MS) - (userCache[userID].earliestDate - DAY_MS);
userCache[userID].totalMs = totalMs;
apiEndpointWazeCom.progress.totalMs += userCache[userID].totalMs;
beginFind(lookFor, transactionLookup);
});
}
function getSearchArea() {
var theVenue = null;
var count = 0;
for (var v in W.model.venues.objects) {
if (W.model.venues.objects.hasOwnProperty(v) === false) {
continue;
}
var venue = W.model.venues.objects[v];
if (venue.isPoint() === true) {
continue;
}
if ($.isNumeric(venue.attributes.id) && parseInt(venue.attributes.id) <= 0) {
theVenue = venue;
count++;
}
}
if (count === 0) {
console.log("using screen");
return W.map.getExtent().toGeometry();
}
if (theVenue.geometry.components.length !== 1) {
alert("Can't parse the geometry");
return;
} else {
console.log("using unsaved place area");
return theVenue.geometry.clone();
}
}
function getNearbyEditors() {
$('#wmeFdoScanCurrentStatus')[0].innerText = 'Finding Nearby Editors';
//let earliestDate = parseInt(new Date('2016-12-31').getTime());
let editorList = [];
let editorListReturn = [];
Object.keys(W.model.venues.objects).forEach(function (venueId) {
let venue = W.model.venues.objects[venueId].attributes;
if (typeof(venue.createdOn) != 'undefined' && venue.createdOn > earliestScanDate) {
editorList[venue.createdBy] = venue.createdOn;
}
if (typeof(venue.updatedOn) != 'undefined' && venue.updatedOn > earliestScanDate) {
editorList[venue.updatedBy] = venue.updatedBy;
}
//console.log(W.model.venues.objects[venueId].attributes.createdBy)
});
Object.keys(W.model.segments.objects).forEach(function (segmentId) {
let segment = W.model.segments.objects[segmentId].attributes;
if (typeof(segment.createdOn) != 'undefined' && segment.createdOn > earliestScanDate) {
editorList[segment.createdBy] = segment.createdOn;
}
if (typeof(segment.updatedOn) != 'undefined' && segment.updatedOn > earliestScanDate) {
editorList[segment.updatedBy] = segment.updatedBy;
}
//console.log(W.model.venues.objects[venueId].attributes.createdBy)
});
Object.keys(editorList).forEach(function (editorId) {
if (editorId != 'undefined') {
editorListReturn.push(editorId);
}
});
return editorListReturn;
}
function getNearbyEditorsFromFeatures(featuresList) {
$('#wmeFdoScanCurrentStatus')[0].innerText = 'Finding Nearby Editors';
let editorList = [];
let editorListReturn = [];
featuresList.forEach(function (features) {
if (typeof(features.error) == 'undefined') {
if (typeof(features.venues) != 'undefined') {
features.users.objects.forEach(function (user) {
addUserCache(user.id, user.userName, user.rank);
});
}
if (typeof(features.venues) != 'undefined') {
features.venues.objects.forEach(function (venue) {
if (typeof(venue.createdOn) != 'undefined' && venue.createdOn > earliestScanDate) {
editorList[venue.createdBy] = venue.createdOn;
updateUserCache(venue.createdBy, venue.createdOn);
}
if (typeof(venue.updatedOn) != 'undefined' && venue.updatedOn > earliestScanDate) {
editorList[venue.updatedBy] = venue.updatedBy;
updateUserCache(venue.updatedBy, venue.updatedOn);
}
seenVenueIds.push(venue.id); // Venue exists, no need to check its history
});
}
if (typeof(features.segments) != 'undefined') {
features.segments.objects.forEach(function (segment) {
if (typeof(segment.createdOn) != 'undefined' && segment.createdOn > earliestScanDate) {
editorList[segment.createdBy] = segment.createdOn;
updateUserCache(segment.createdBy, segment.createdOn);
}
if (typeof(segment.updatedOn) != 'undefined' && segment.updatedOn > earliestScanDate) {
editorList[segment.updatedBy] = segment.updatedBy;
updateUserCache(segment.updatedBy, segment.updatedOn);
}
});
}
}
});
Object.keys(editorList).forEach(function (editorId) {
if (editorId != 'undefined') {
editorListReturn.push(editorId);
}
});
return editorListReturn;
}
function addUserCache(userID, userName, rank) {
if (typeof(userCache[userID]) == 'undefined') {
userCache[userID] = defaultUserCache(userID, userName, rank);
}
}
function updateUserCache(userID, actionDate) {
if (typeof(userCache[userID]) == 'undefined') {
} else {
if (actionDate > userCache[userID].latestDate) {
userCache[userID].latestDate = actionDate;
}
if (actionDate < userCache[userID].earliestDate) {
userCache[userID].earliestDate = actionDate;
}
}
}
async function getScreenFeatures() {
return new Promise( async resolve => {
let urlList = [];
let paramsList = [];
let featuresList = [];
let maxDegrees = 0.125;
let minDegrees = 0.0625;
let extent = W.map.getExtent().transform(projection,displayProjection);
//NW: LT - left top
//NE: RT - right top
//SW: LB - left bottom
//SE: RB - right bottom
let geoLT = new OpenLayers.Geometry.Point(extent.left,extent.top);
let geoRB = new OpenLayers.Geometry.Point(extent.right,extent.bottom);
let geoLTSplit = new OpenLayers.Geometry.Point(geoLT.x,geoLT.y);
let geoRBSplit = new OpenLayers.Geometry.Point(geoRB.x,geoRB.y);
let geoTempY = new OpenLayers.Geometry.Point(geoLT.x,geoLT.y);
let xSplit = geoLT.x;
let ySplit = geoLT.y;
//console.log(extent);
//console.log(geoLT);
//console.log(geoRB);
while (ySplit >= geoRB.y) {
geoTempY.y = geoTempY.y - maxDegrees;
ySplit = geoTempY.y;
let degreesFromGeoRBY = geoRB.y - ySplit;
if (ySplit > geoRB.y)
geoRBSplit.y = ySplit
else if (degreesFromGeoRBY <= minDegrees)
geoRBSplit.y = ySplit - (minDegrees - degreesFromGeoRBY);
else
geoRBSplit.y = geoRB.y;
while (xSplit <= geoRB.x) {
geoTempY.x = geoTempY.x + maxDegrees;
xSplit = geoTempY.x;
let degreesFromGeoRBX = geoRB.x - xSplit;
if (xSplit > geoRB.x)
geoRBSplit.x = xSplit
else if (degreesFromGeoRBX <= minDegrees)
geoRBSplit.x = xSplit - (minDegrees - degreesFromGeoRBX);
else
geoRBSplit.x = geoRB.x;
// let bboxGeoLT = geoLTSplit.clone();
// let bboxGeoEB = geoEBSplit.clone();
let bbox = `${geoLTSplit.x},${geoLTSplit.y},${geoRBSplit.x},${geoRBSplit.y}`;
let url = getApiUrlFeatures(bbox);
urlList.push(url);
//console.log(url);
let params = requestParams(url);
requestStatus[url] = defaultRequestStatus();
// let features = null;
// features = await getApiRequest(params);
// featuresList.push(features);
paramsList.push(params);
apiEndpointWazeCom.requests.push(params);
geoLTSplit.x = xSplit;
}
geoLTSplit.y = ySplit;
geoLTSplit.x = geoLT.x;
xSplit = geoLT.x;
geoTempY.x = xSplit;
}
getEndpointResultFromArray(apiEndpointWazeCom, paramsList, featuresList => {
resolve(featuresList);
});
});
}
function getApiUrlUserProfile(username) {
return `https://${window.location.host}${W.Config.api_base}/UserProfile/Profile?username=${username}`;
}
function getApiUrlTransactions(userId, nextTransactionDate) {
let url = `https://${window.location.host}${W.Config.api_base}/UserProfile/Transactions?userID=${userId}`;
if (nextTransactionDate != null) {
url = url + '&till=' + nextTransactionDate;
}
return url;
}
function getApiUrlElementHistory(objectType, objectId, nextTransactionDate) {
let url = `https://${window.location.host}${W.Config.api_base}/ElementHistory?objectType=${objectType}&objectID=${objectId}`;
if (nextTransactionDate != null) {
url = url + '&till=' + nextTransactionDate;
}
return url;
}
function getApiUrlFeatures(bbox) {
let url = `https://${window.location.host}${W.Config.api_base}/Features?language=en-US&bbox=${bbox}&roadTypes=2%2C3%2C4%2C6%2C7%2C8%2C9%2C10%2C11%2C12%2C13%2C14%2C15%2C16%2C17%2C18%2C19%2C20%2C21%2C22&venueLevel=4&venueFilter=3%2C3%2C3`;
return url;
}
async function highlightPin(pin) {
if (pin.onScreen()) {
let layer = W.map.getLayerByUniqueName('__wmeFdoFeatureLayer');
let enableAnimation = false;
if (enableAnimation) {
pin.data.mouseout = false;
pin.style.strokeColor = 'red';
for (var i=100;i>=1;i--) {
await wait(10);
pin.style.strokeWidth = i;
layer.redraw();
if (pin.data.mouseout == true) {
i = 1;
returnPin(pin);
}
}
pin.style.strokeWidth = 4;
pin.style.pointRadius = 5;
pin.style.strokeColor = 'orange';
layer.redraw();
if (pin.data.mouseout == true) {
returnPin(pin);
}
} else {
pin.style.strokeWidth = 4;
pin.style.pointRadius = 5;
pin.style.strokeColor = 'orange';
layer.redraw();
}
}
}
function returnPin(pin) {
let style = {strokeColor: 'black',
strokeWidth: '4',
strokeDashstyle: 'solid',
pointRadius: '4',
fillOpacity: '1'};
if (pin.onScreen()) {
let layer = W.map.getLayerByUniqueName('__wmeFdoFeatureLayer');
pin.data.mouseout = true;
//pin.style.strokeColor = 'black';
//pin.style.strokeWidth = 5;
pin.style = style;
layer.redraw();
}
}
function createSettingsCheckbox($div, settingID, textDescription) {
let $checkbox = $('<input>', {type:'checkbox', id:settingID, class:'wmeFdoSettingsCheckbox'});
$div.append(
$('<div>', {class:'controls-container'}).css({paddingTop:'2px'}).append(
$checkbox,
$('<label>', {for:settingID}).text(textDescription).css({whiteSpace:'pre-line'})
)
);
return $checkbox;
}
function setChecked(checkboxId, checked) {
$('#' + checkboxId).prop('checked', checked);
}
function saveSettings() {
if (localStorage) {
var localsettings = {
Enabled: settings.Enabled,
IncludeSelf: settings.IncludeSelf,
LimitToScreen: settings.LimitToScreen,
NearbyEditors: settings.NearbyEditors,
FilterResidential: settings.FilterResidential,
FilterParkingLot: settings.FilterParkingLot,
FilterAnyType: settings.FilterAnyType,
FilterNewRejected: settings.FilterNewRejected,
FilterDeleteRequest: settings.FilterDeleteRequest
};
localStorage.setItem("wmeFdo_Settings", JSON.stringify(localsettings));
}
}
function loadSettings() {
var loadedSettings = $.parseJSON(localStorage.getItem("wmeFdo_Settings"));
var defaultSettings = {
Enabled: true,
IncludeSelf: true,
LimitToScreen: true,
NearbyEditors: true,
FilterResidential: true,
FilterParkingLot: true,
FilterAnyType: true,
FilterNewRejected: true,
FilterDeleteRequest: true
};
settings = loadedSettings ? loadedSettings : defaultSettings;
for (var prop in defaultSettings) {
if (!settings.hasOwnProperty(prop))
settings[prop] = defaultSettings[prop];
}
}
function prepareTab() {
let tabs = document.getElementById('user-tabs');
let items = tabs.getElementsByTagName('li');
for (var i = 0; i < items.length; ++i) {
let a = items[i].getElementsByTagName('a')[0]
if (a.href.includes('sidepanel-fido')) {
a.id = "wmeFdoTab";
a.innerHTML = 'FiDO <span class="spinner-disable"><i class="icon-spinner icon-spin fa fa-spinner fa-spin"></i></span>';
}
}
}
function selectTab() {
let tabs = document.getElementById('user-tabs');
let items = tabs.getElementsByTagName('li');
for (var i = 0; i < items.length; ++i) {
let a = items[i].getElementsByTagName('a')[0]
if (a.href.includes('sidepanel-fido')) {
items[i].getElementsByTagName('a')[0].click();
}
}
}
function dateToUuid(inputDate) {
let timestamp = parseInt(new Date(inputDate).getTime());
// uuid = "a02102cd-19df-11e9-bd87-1201fde9baf6"
// timestamp = 1547678386643;
let a = timestamp * 10000;
let b = a + 122192928000000000;
let c = b.toString(16);
//if (c.length % 2) {
// c = '0' + c;
//}
let d = c.substring(7) + '-' + c.substring(3,7) + '-1' + c.substring(0,3) + '-bd87-1201fde9baf6';
return d;
}
function get_time_int(uuid_str) {
var uuid_arr = uuid_str.split( '-' ),
time_str = [
uuid_arr[ 2 ].substring( 1 ),
uuid_arr[ 1 ],
uuid_arr[ 0 ]
].join( '' );
return parseInt( time_str, 16 );
};
function uuidToUnixTime(uuid_str) {
var int_time = get_time_int( uuid_str ) - 122192928000000000,
int_millisec = Math.floor( int_time / 10000 );
return int_millisec;
};
function uuidToDate(uuid_str) {
var int_time = get_time_int( uuid_str ) - 122192928000000000,
int_millisec = Math.floor( int_time / 10000 );
//return new Date( int_millisec );
return timeConverter(int_millisec);
};
function formatCurrentDate() {
var today = new Date();
var dd = today.getDate();
var mm = today.getMonth() + 1; //January is 0!
var yyyy = today.getFullYear();
if (dd < 10) {
dd = '0' + dd;
}
if (mm < 10) {
mm = '0' + mm;
}
return yyyy + '-' + mm + '-' + dd;
}
function timeConverter(UNIX_timestamp){
var a = new Date(UNIX_timestamp);
var months = ['Jan','Feb','Mar','Apr','May','Jun','Jul','Aug','Sep','Oct','Nov','Dec'];
var year = a.getFullYear();
var month = months[a.getMonth()];
var date = a.getDate();
if (date.toString().length == 1) {
date = "0" + date;
}
var hour = a.getHours();
if (hour.toString().length == 1) {
hour = "0" + hour;
}
var min = a.getMinutes();
if (min.toString().length == 1) {
min = "0" + min;
}
var sec = a.getSeconds();
if (sec.toString().length == 1) {
sec = "0" + sec;
}
//Dec 08, 2018
var time = month + ' ' + date + ', ' + year + ' ' + hour + ':' + min;
return time;
}
function timeConverterNoTime(UNIX_timestamp){
var a = new Date(UNIX_timestamp);
var months = ['Jan','Feb','Mar','Apr','May','Jun','Jul','Aug','Sep','Oct','Nov','Dec'];
var year = a.getFullYear();
var month = months[a.getMonth()];
var date = a.getDate();
if (date.toString().length == 1) {
date = "0" + date;
}
var hour = a.getHours();
if (hour.toString().length == 1) {
hour = "0" + hour;
}
var min = a.getMinutes();
if (min.toString().length == 1) {
min = "0" + min;
}
var sec = a.getSeconds();
if (sec.toString().length == 1) {
sec = "0" + sec;
}
//Dec 08, 2018
var time = month + ' ' + date + ', ' + year;
return time;
}
async function wait(ms) {
return new Promise(resolve => {
setTimeout(resolve, ms);
});
}
// ------------- Begin Web Request/Rate Limiting -----------
var apiEndpointWazeCom = null;
function defaultEndpointSettings() {
return {
state: 'idle',
isRunning: false,
requests: [],
results: [],
requestInterval: null,
statusInterval: null,
count: 0,
beginTime: Date.now(),
timeout: 100000,
gotFeatures: false,
rateLimit: {
baseDelay: 50,
addedDelay: 0,
stepUp: 20,
stepDown: 10,
lastCheck: Date.now(),
lastAdjusted: Date.now(),
intervalDelay: 5000,
checkInterval: null,
errorCount: 0,
errorCountMax: 50
},
users: {
total: 0,
complete: 0
},
progress: {
totalMs: 0,
completeMs: 0
}
};
}
function defaultUserCache(userID, username, rank) {
return {
userID: userID,
username: username,
rank: rank,
isRunning: false,
currentScanDate: null,
earliestDate: Date.now(),
latestDate: 0,
previousDate: 0,
totalMs: 0,
completeMs: 0
}
}
function defaultUserProgress(userID) {
return {
userID: userID,
username: null,
rank: null,
isRunning: false,
currentScanDate: null,
earliestDate: Date.now(),
previousDate: 0,
latestDate: 0,
totalMs: 0
}
}
function beginScanNew() {
// start api request handler
apiEndpointWazeCom = defaultEndpointSettings();
apiEndpointWazeCom.state = 'running';
apiEndpointWazeCom.isRunning = true;
apiEndpointWazeCom.requestInterval = setInterval(function() { apiHandlerWazeCom(apiEndpointWazeCom) }, getRateLimit(apiEndpointWazeCom));
apiEndpointWazeCom.rateLimit.checkInterval = setInterval(function() { setRateLimitAdjustDown(apiEndpointWazeCom) }, apiEndpointWazeCom.rateLimit.intervalDelay);
apiEndpointWazeCom.statusInterval = setInterval(function() { populateStatus(); }, 1000);
//$('#wmeFdoScanStatus .wmeFdoProgress').removeClass('hidden');
}
function cancelScanNew() {
// end api request handler
apiEndpointWazeCom.state = 'cancelled';
apiEndpointWazeCom.isRunning = false;
clearInterval(apiEndpointWazeCom.requestInterval);
clearInterval(apiEndpointWazeCom.rateLimit.checkInterval);
clearInterval(apiEndpointWazeCom.statusInverval);
$('#wmeFdoScanCurrentStatus')[0].innerText = 'Cancelled';
}
function apiHandlerWazeCom(apiEndpoint) {
if (apiEndpoint.requests.length > 0) {
//console.log("Handler " + apiEndpoint.count);
let requestParams = apiEndpoint.requests.shift();
getApiWebRequest(apiEndpoint, requestParams);
//apiEndpointWazeCom.count++;
}
}
function getApiWebRequest(apiEndpoint, requestParams) {
apiEndpoint.count++;
$.ajax({
type: 'GET',
url: requestParams.url,
success: function(data) {
//requestStatus[requestParams.url].status = 'successful';
if (typeof(data) == 'undefined') {
debugger;
}
apiEndpoint.results[requestParams.url] = data;
},
statusCode: {
406: function() { // Not Acceptable - bbox invalid?
apiEndpoint.results[requestParams.url] = {"error":true,reason:"error"};
},
429: function() { // Rate limit
setRateLimitAdjustUp(apiEndpoint);
apiEndpoint.requests.push(requestParams);
},
500: function() {
//debugger;
apiEndpoint.results[requestParams.url] = {"error":true,reason:"error"};
}
}
});
}
function getRateLimit(apiEndpoint) {
return apiEndpoint.rateLimit.baseDelay + apiEndpoint.rateLimit.addedDelay;
}
function setRateLimitAdjustUp(apiEndpoint) {
let currentTime = Date.now();
if (currentTime - apiEndpoint.rateLimit.lastAdjusted > 1000) {
apiEndpoint.rateLimit.lastAdjusted = currentTime;
apiEndpoint.rateLimit.errorCount++;
apiEndpoint.rateLimit.addedDelay += apiEndpoint.rateLimit.stepUp;
clearInterval(apiEndpoint.requestInterval);
apiEndpoint.requestInterval = setInterval(function() { apiHandlerWazeCom(apiEndpoint) }, getRateLimit(apiEndpoint));
console.log('rate limit adjusted up: ' + getRateLimit(apiEndpoint));
}
}
function setRateLimitAdjustDown(apiEndpoint) {
if (apiEndpoint.rateLimit.addedDelay > 0 && Date.now() - apiEndpoint.rateLimit.lastAdjusted > apiEndpoint.rateLimit.intervalDelay) { //&& apiEndpoint.rateLimit.errorCount <= apiEndpoint.rateLimit.errorCountMax) {
apiEndpoint.rateLimit.addedDelay -= apiEndpoint.rateLimit.stepDown;
clearInterval(apiEndpoint.requestInterval);
apiEndpoint.requestInterval = setInterval(function() { apiHandlerWazeCom(apiEndpoint) }, getRateLimit(apiEndpoint));
console.log('rate limit adjusted down: ' + getRateLimit(apiEndpoint));
}
}
function getEndpointResultFromArray(apiEndpoint, paramsList, callback) {
waitForAllUrlResults(apiEndpoint, paramsList, complete => {
let resultList = [];
paramsList.forEach(function (params) {
getEndpointResult(apiEndpoint, params, result => {
resultList.push(result);
});
});
callback(resultList);
});
}
function populateStatus() {
let runningSeconds = (Date.now() - apiEndpointWazeCom.beginTime) / 1000;
let requestsPerSecond = apiEndpointWazeCom.count / runningSeconds;
//console.log('queued requests: ' + Object.keys(apiEndpoint.requests).length + ', queued results: ' + Object.keys(apiEndpoint.results).length);
//console.log('users: ' + apiEndpoint.users.complete + '/' + apiEndpoint.users.total + ' complete, ' + (apiEndpoint.users.total - apiEndpoint.users.complete) + ' remaining');
//console.log('requests per second: ' + requestsPerSecond);
let remainingUsers = apiEndpointWazeCom.users.total - apiEndpointWazeCom.users.complete;
$('#wmeFdoRequests')[0].innerText = apiEndpointWazeCom.count;
$('#wmeFdoRequestsPerSecond')[0].innerText = Math.round(requestsPerSecond);
if (apiEndpointWazeCom.gotFeatures && remainingUsers == 0) {
endScan('Complete', true);
}
}
async function getEndpointResult(apiEndpoint, params, callback) {
let timeSinceRequest = Date.now() - params.initiated;
//console.log(typeof(apiEndpoint.results[params.url]));
while ( typeof(apiEndpoint.results[params.url]) == 'undefined' && apiEndpoint.isRunning ) {
// removed && timeSinceRequest < apiEndpoint.timeout
//console.log(typeof(apiEndpoint.results[params.url]));
await wait(50);
timeSinceRequest = Date.now() - params.initiated;
//console.log('waiting for: ' + params.url);
}
//if (timeSinceRequest >= apiEndpoint.timeout) {
// console.log('request timeout'); //TODO: handle this condition better
// }
let result = apiEndpoint.results[params.url];
delete apiEndpoint.results[params.url];
callback(result);
}
async function waitForAllUrlResults(apiEndpoint, paramsList, callback) {
let complete = false;
let listCount = paramsList.length;
while (!complete && apiEndpoint.isRunning) {
let foundCount = 0;
await wait(50);
paramsList.forEach(function (params) {
let foundUrl = false;
if (typeof(apiEndpoint.requests[params.url]) == 'undefined' && typeof(apiEndpoint.results[params.url]) != 'undefined') {
foundUrl = true;
foundCount++;
}
});
if (listCount == foundCount) {
complete = true;
}
}
callback(complete);
}
// ------------- End Web Request/Rate Limiting -----------
bootstrap();
})();
class NavigationPointMEP
{
constructor(point){
this._point = point.clone();
this._entry = true;
this._exit = true;
this._isPrimary = true;
this._name = "";
}
with(){
var e = arguments.length > 0 && void 0 !== arguments[0] ? arguments[0] : {};
if(e.point == null)
e.point = this.toJSON().point;
let val = new this.constructor((this.toJSON().point, e.point));
val._entry = this._entry;
val._exit = this._exit;
val._isPrimary = this._isPrimary;
val._name = this._name;
return val;
}
getPoint(){
return this._point.clone();
}
getEntry(){
return this._entry;
}
getExit(){
return this._exit;
}
getName(){
return this._name;
}
isPrimary(){
return this._isPrimary;
}
toJSON(){
return {
point: this._point,
entry: this._entry,
exit: this._exit,
primary: this._isPrimary,
name: this._name
};
}
clone(){
return this.with();
}
}