// ==UserScript==
// @name WME UR Minus (UR-)
// @namespace https://greasyfork.org/users/gad_m/wme_ur_minus
// @description Adds minimum UR filtering.
// @include /^https:\/\/(www|beta)\.waze\.com\/(?!user\/)(.{2,6}\/)?editor.*$/
// @author gad_m
// @license MIT
// @grant GM_addStyle
// @version 1.33
// @grant GM_xmlhttpRequest
// @connect raw.githubusercontent.com
//noinspection SpellCheckingInspection
// @icon 
// ==/UserScript==
/* global W */
/* global I18n */
/* global jQuery */
/* global OpenLayers */
/* global GM_addStyle */
/* global GM_xmlhttpRequest */
(function() {
'use strict';
console.info('wme-ur-minus: loading...');
let wMethods = {};
let LonLat;
let Projection;
let Point;
let LinearRing;
let Polygon;
const SCRIPT_ID = "ur_minus";
const UR_MINUS_PREFIX = 'wme-ur-minus_';
const USER_PREF_ACTIVE = UR_MINUS_PREFIX + 'user_preferences_active';
const USER_PREF_ALL_URS = UR_MINUS_PREFIX + 'user_preferences_all_urs';
const USER_PREF_NO_COMMENTS = UR_MINUS_PREFIX + 'user_preferences_no_comments';
const USER_PREF_HAS_COMMENTS = UR_MINUS_PREFIX + 'user_preferences_has_comments';
const USER_PREF_LAST_COMMENT_BY_ME = UR_MINUS_PREFIX + 'user_preferences_last_comment_by_me';
const USER_PREF_LAST_COMMENT_BEFORE = UR_MINUS_PREFIX + 'user_preferences_last_comment_before';
const USER_PREF_LAST_COMMENT_BEFORE_NUM_OF_DAYS = UR_MINUS_PREFIX + 'user_preferences_last_comment_before_num_of_days';
const USER_PREF_WITH_COMMENTS_NOT_MINE = UR_MINUS_PREFIX + 'user_preferences_with_comments_not_mine';
const USER_PREF_OUTSIDE_MY_AREAS_UR = UR_MINUS_PREFIX + 'user_preferences_outside_my_areas_ur';
const USER_PREF_INSIDE_OTHERS_MANAGERS_AREAS_UR = UR_MINUS_PREFIX + 'user_preferences_inside_other_managers_ur';
const USER_PREF_ALL_PURS = UR_MINUS_PREFIX + 'user_preferences_all_purs';
const USER_PREF_ABOVE_MY_RANK = UR_MINUS_PREFIX + 'user_preferences_above_my_rank';
const USER_PREF_OUTSIDE_MY_AREAS_PUR = UR_MINUS_PREFIX + 'user_preferences_outside_my_areas_pur';
const USER_PREF_INSIDE_OTHERS_MANAGERS_AREAS_PUR = UR_MINUS_PREFIX + 'user_preferences_inside_other_managers_pur';
const USER_PREF_CHARGING_STATIONS = UR_MINUS_PREFIX + 'user_preferences_charging_stations_pur';
const WME_UR_MINUS_DIV_TOOLTIP_ID = UR_MINUS_PREFIX + 'DivTooltip';
const ALL_USER_PERF_CHECKBOXES_IDS = [USER_PREF_ALL_URS, USER_PREF_NO_COMMENTS, USER_PREF_HAS_COMMENTS, USER_PREF_LAST_COMMENT_BY_ME, USER_PREF_LAST_COMMENT_BEFORE, USER_PREF_WITH_COMMENTS_NOT_MINE, USER_PREF_OUTSIDE_MY_AREAS_UR, USER_PREF_INSIDE_OTHERS_MANAGERS_AREAS_UR, USER_PREF_ALL_PURS, USER_PREF_ABOVE_MY_RANK, USER_PREF_OUTSIDE_MY_AREAS_PUR, USER_PREF_INSIDE_OTHERS_MANAGERS_AREAS_PUR, USER_PREF_CHARGING_STATIONS];
const STORAGE_KEY_MY_MANAGED_AREAS = UR_MINUS_PREFIX + "my-managed-areas";
let urMinus_csrfToken;
if (typeof OpenLayers != 'undefined' && W && W['userscripts'] && W['userscripts']['state'] && W['userscripts']['state']['isReady']) {
console.debug('wme-ur-minus: WME is ready.');
init();
} else {
console.debug('wme-ur-minus: WME is not ready. adding event listener.');
document.addEventListener("wme-ready", init, {
once: true,
});
}
function getCsrfToken(cb) {
console.debug('wme-ur-minus: getCsrfToken()');
let url = "https://" + document.location.host + W['Config'].paths['configurationInfo'];
GM_xmlhttpRequest({
url: url,
headers: {
"Content-Type": "application/json",
"Accept": "application/json, text/javascript, */*; q=0.01" },
onload: function(response) {
if (response.status === 200) {
let cookies = response['responseHeaders'].match(/_csrf_token=(.*); Path/i);
if (cookies && cookies.length >= 2) {
let csrfToken = cookies[1];
console.debug('wme-ur-minus: getCsrfToken() returning: ' + csrfToken);
cb(csrfToken);
} else {
console.error('wme-ur-minus: getCsrfToken() responseHeaders: ' + response['responseHeaders']);
}
} else {
console.error('wme-ur-minus: getCsrfToken() response status: ' + response.status);
}
}
});
}
function init() {
console.info('wme-ur-minus: init()');
document.addEventListener("wme-map-data-loaded", dataLoadedEvent);
//noinspection JSUnresolvedVariable
LonLat = OpenLayers.LonLat.bind(OpenLayers);
//noinspection JSUnresolvedVariable
Projection = OpenLayers.Projection.bind(OpenLayers);
//noinspection JSUnresolvedVariable
Point = OpenLayers.Geometry.Point.bind(OpenLayers);
//noinspection JSUnresolvedVariable
LinearRing = OpenLayers.Geometry.LinearRing.bind(OpenLayers);
//noinspection JSUnresolvedVariable
Polygon = OpenLayers.Geometry.Polygon.bind(OpenLayers);
//noinspection JSUnresolvedVariable
wMethods["getAppRegionCode"] = W['app'].getAppRegionCode;
//noinspection JSUnresolvedVariable
wMethods["waitForElementConnected"] = W['userscripts'].waitForElementConnected.bind(W['userscripts']);
//noinspection JSUnresolvedVariable
wMethods["registerSidebarTab"] = W['userscripts'].registerSidebarTab.bind(W['userscripts']);
//noinspection JSUnresolvedVariable
//wMethods["getLayerByName"] = W['map'].getLayerByName.bind(W['map']);
getCsrfToken(function (val) {
urMinus_csrfToken = val;
afterGettingCsrfToken();
})
}
function afterGettingCsrfToken() {
setTimeout(function () {
handleUI();
processFilters();
}, 1000)
}
function processFilters() {
console.info('wme-ur-minus: processFilters()');
let userPrefActive = localStorage.getItem(USER_PREF_ACTIVE) === "true";
let managedAreas = getManagedAreas();
processMurFilters(userPrefActive, managedAreas);
processVenueFilters(userPrefActive, managedAreas);
addTooltipEvents();
}
function processVenueFilters(userPrefActive, managedAreas) {
console.debug('wme-ur-minus: processVenueFilters()');
if (W.model['venues'] && W.model['venues'].objects) {
let allPlaceUpdateRequestIDs = Object.values(W.model['venues'].objects).filter(venue => venue.attributes['venueUpdateRequests'].length > 0).map(venue => venue.attributes.id);
/*
let allPlaceUpdateRequestIDs = jQuery("div").map(function () {
if (parseInt($(this).attr("data-update-count")) > 0) {
return $(this).attr("data-id");
}
}).get();
*/
let userID = W['loginManager']['user']['id'] || W['loginManager']['user']['attributes']['id'];
let userRank = W['loginManager']['user']['rank'] || W['loginManager']['user']['attributes']['rank'];
let userPrefAllPurs = localStorage.getItem(USER_PREF_ALL_PURS) === "true";
let userPrefAboveMyRank = localStorage.getItem(USER_PREF_ABOVE_MY_RANK) === "true";
let userPrefOutsideMyAreasPur = localStorage.getItem(USER_PREF_OUTSIDE_MY_AREAS_PUR) === "true";
let userPrefInsideOtherManagersAreasPur = localStorage.getItem(USER_PREF_INSIDE_OTHERS_MANAGERS_AREAS_PUR) === "true";
let userPrefChargingStations = localStorage.getItem(USER_PREF_CHARGING_STATIONS) === "true";
let allVenues = W.model['venues'].objects;
console.info('wme-ur-minus: processVenueFilters() processing ' + allPlaceUpdateRequestIDs.length + ' venues update requests...');
allPlaceUpdateRequestIDs.forEach(id => {
let venue = allVenues[id];
let hideBecauseAllPurs = shouldHBecauseAllPurs(userPrefActive, venue, userPrefAllPurs);
let hideBecauseAboveMyRank = shouldHBecauseAboveMyRank(userPrefActive, venue, userPrefAboveMyRank, userRank);
let hideBecauseOutsideMyAreaPur = shouldHideBecauseOutsideMyArea(userPrefActive, venue, userID, userPrefOutsideMyAreasPur, managedAreas["my"]);
let hideBecauseInsideOtherManagersAreasPur = shouldHideBecauseInsideOtherManagersAreas(userPrefActive, venue, userID, userPrefInsideOtherManagersAreasPur, managedAreas);
let hideBecauseChargingStationsPur = shouldHideBecauseChargingStationsPur(userPrefActive, venue, userPrefChargingStations);
let hide = hideBecauseAllPurs || hideBecauseAboveMyRank || hideBecauseOutsideMyAreaPur || hideBecauseInsideOtherManagersAreasPur || hideBecauseChargingStationsPur;
if (hide) {
hideMUR({id:venue['attributes']['id']});
} else {
showMUR({id:venue['attributes']['id']});
}
});
}
}
function processMurFilters(userPrefActive, managedAreas) {
console.debug('wme-ur-minus: processMurFilters()');
if (W.model.mapUpdateRequests && W.model.mapUpdateRequests.objects) {
getUrComments(userPrefActive, function (urWithCommentsResponse) {
let userID = W['loginManager']['user']['id'] || W['loginManager']['user']['attributes']['id'];
let userPrefAllUrs = localStorage.getItem(USER_PREF_ALL_URS) === "true";
let userPrefNoComments = localStorage.getItem(USER_PREF_NO_COMMENTS) === "true";
let userPrefHasComments = localStorage.getItem(USER_PREF_HAS_COMMENTS) === "true";
let userPrefLastCommentByMe = localStorage.getItem(USER_PREF_LAST_COMMENT_BY_ME) === "true";
let now = (new Date()).getTime();
let userPrefLastCommentBefore = localStorage.getItem(USER_PREF_LAST_COMMENT_BEFORE) === "true";
let userPrefLastCommentBeforeNumOfDays = parseInt(localStorage.getItem(USER_PREF_LAST_COMMENT_BEFORE_NUM_OF_DAYS));
let userPrefWithCommentsNotMine = localStorage.getItem(USER_PREF_WITH_COMMENTS_NOT_MINE) === "true";
let userPrefOutsideMyAreasUr = localStorage.getItem(USER_PREF_OUTSIDE_MY_AREAS_UR) === "true";
let userPrefInsideOtherManagersAreasUr = localStorage.getItem(USER_PREF_INSIDE_OTHERS_MANAGERS_AREAS_UR) === "true";
let users = urWithCommentsResponse['users']['objects'];
console.info('wme-ur-minus: processMurFilters() processing ' + Object.keys(W.model.mapUpdateRequests.objects).length + ' map update requests...');
Object.values(W.model.mapUpdateRequests.objects).forEach(mur => {
let murID = mur['attributes']['id'];
console.debug('wme-ur-minus: processMurFilters() checking map update request ' + murID);
let murComments = urWithCommentsResponse['updateRequestSessions']['objects'].find(obj => obj.id === murID);
let hideBecauseAllUrs = shouldHideBecauseAllUrs(userPrefActive, mur, userID, userPrefAllUrs);
let hideBecauseNoComments = shouldHideBecauseNoComments(userPrefActive, mur, murComments, userPrefNoComments);
let hideBecauseHasComments = shouldHideBecauseHasComments(userPrefActive, mur, murComments, userPrefHasComments);
let hideBecauseLastCommentByMe = shouldHideBecauseLastCommentByMe(userPrefActive, mur, userID, userPrefLastCommentByMe, murComments);
let hideBecauseLastCommentTooNew = shouldHideBecauseLastCommentTooNew(userPrefActive, mur, userPrefLastCommentBefore, userPrefLastCommentBeforeNumOfDays, now);
let hideBecauseWithCommentsNotMine = shouldHideBecauseWithCommentsNotMine(userPrefActive, mur, userPrefWithCommentsNotMine, murComments, users, userID);
let hideBecauseOutsideMyAreaUr = shouldHideBecauseOutsideMyArea(userPrefActive, mur, userID, userPrefOutsideMyAreasUr, managedAreas["my"]);
let hideBecauseInsideOtherManagersAreasUr = shouldHideBecauseInsideOtherManagersAreas(userPrefActive, mur, userID, userPrefInsideOtherManagersAreasUr, managedAreas);
let hide = hideBecauseAllUrs || hideBecauseNoComments || hideBecauseHasComments || hideBecauseLastCommentByMe || hideBecauseLastCommentTooNew || hideBecauseWithCommentsNotMine || hideBecauseOutsideMyAreaUr || hideBecauseInsideOtherManagersAreasUr;
if (hide) {
hideMUR({id:murID});
} else {
showMUR({id:murID, murComments:murComments});
}
});
});
}
}
const DUMMY_RESPONSE = {"updateRequestSessions":{"objects":[]},"users":{"objects":[]}};
function getUrComments(userPrefActive, cb) {
console.debug('wme-ur-minus: getUrComments()');
if (userPrefActive) {
let hasCommentsIDs = Object.values(W.model.mapUpdateRequests.objects).filter(mur => mur.attributes['hasComments']).map(mur => mur.attributes.id)
if (hasCommentsIDs.length > 0) {
getUpdateRequestSessions(hasCommentsIDs, function (response) {
if (response) {
console.debug('wme-ur-minus: getUrComments() got ' + response['updateRequestSessions']['objects'].length + ' URs with comments and ' + response['users']['objects'].length + " users");
cb(response);
} else {
console.error('wme-ur-minus: getUrComments() no response');
}
});
} else {
cb(DUMMY_RESPONSE);
}
} else {
cb(DUMMY_RESPONSE);
}
}
function shouldHBecauseAllPurs(userPrefActive, venue, userPrefAllPurs) {
let result = userPrefActive && userPrefAllPurs;
console.debug('wme-ur-minus: shouldHBecauseAllPurs() venue: ' + venue['attributes']['id'] + ": " + result);
return result;
}
function shouldHBecauseAboveMyRank(userPrefActive, venue, userPrefAboveMyRank, userRank) {
let result = false;
if (userPrefActive && userPrefAboveMyRank) {
result = (userRank < venue.attributes['lockRank'] || venue.attributes['adLocked']);
}
console.debug('wme-ur-minus: shouldHBecauseAboveMyRank() venue: ' + venue['attributes']['id'] + ": " + result);
return result;
}
function shouldHideBecauseLastCommentTooNew(userPrefActive, mur, userPrefLastCommentBefore, userPrefLastCommentBeforeNumOfDays, now) {
let result = false;
if (userPrefActive && userPrefLastCommentBefore) {
result = (now - mur.attributes['updatedOn']) < 1000 * 60 * 60 * 24 * userPrefLastCommentBeforeNumOfDays;
}
console.debug('wme-ur-minus: shouldHideBecauseLastCommentTooNew() mur: ' + mur['attributes']['id'] + ": " + result);
return result;
}
function shouldHideBecauseWithCommentsNotMine(userPrefActive, mur, userPrefWithCommentsNotMine, murComments, users, userID) {
let result = false;
if (userPrefActive && userPrefWithCommentsNotMine) {
if (murComments && murComments.comments.length) {
// has comments
result = true;
murComments.comments.forEach(comment => {
if (comment['userID'] === userID) {
console.debug('wme-ur-minus: shouldHideBecauseWithCommentsNotMine() mur: ' + mur['attributes']['id'] + " found 'mine' comment: " + comment.text);
result = false;
}
})
}
}
console.debug('wme-ur-minus: shouldHideBecauseWithCommentsNotMine() mur: ' + mur['attributes']['id'] + ": " + result);
return result;
}
function shouldHideBecauseAllUrs(userPrefActive, mur, userID, userPrefAllUrs) {
let result = userPrefActive && userPrefAllUrs;
console.debug('wme-ur-minus: shouldHideBecauseAllUrs() mur: ' + mur['attributes']['id'] + ": " + result);
return result;
}
function shouldHideBecauseLastCommentByMe(userPrefActive, mur, userID, userPrefLastCommentByMe, murComments) {
let result = false;
if (userPrefActive && userPrefLastCommentByMe) {
if (murComments && murComments.comments && murComments.comments.length > 0) {
result = (userID === murComments.comments[murComments.comments.length-1]['userID']);
}
}
console.debug('wme-ur-minus: shouldHideBecauseLastCommentByMe() mur: ' + mur['attributes']['id'] + ": " + result);
return result;
}
function shouldHideBecauseNoComments(userPrefActive, mur, murComments, userPrefNoComments) {
let result = false;
if (userPrefActive && userPrefNoComments) {
result = !murComments || !murComments.comments || murComments.comments.length === 0;
}
console.debug('wme-ur-minus: shouldHideBecauseNoComments() mur: ' + mur['attributes']['id'] + ": " + result);
return result;
}
function shouldHideBecauseHasComments(userPrefActive, mur, murComments, userPrefHasComments) {
let result = false;
if (userPrefActive && userPrefHasComments) {
result = murComments && murComments.comments && murComments.comments.length > 0;
}
console.debug('wme-ur-minus: shouldHideBecauseHasComments() mur: ' + mur['attributes']['id'] + ": " + result);
return result;
}
function shouldHideBecauseOutsideMyArea(userPrefActive, obj, userID, userPrefOutsideMyAreasUr, myManagedAreas) {
let result = false;
if (userPrefActive && userPrefOutsideMyAreasUr) {
result = true;
Object.values(myManagedAreas).forEach(managedArea => {
let geometry = obj['attributes']['geometry'];
if (geometry instanceof OpenLayers['Geometry']['Polygon']) {
// noinspection JSUnresolvedFunction
geometry = geometry.getCentroid();
}
// noinspection JSUnresolvedFunction
if (managedArea['geometry'].containsPoint(geometry)) {
result = false;
}
});
}
console.debug('wme-ur-minus: shouldHideBecauseOutsideMyArea() obj: ' + obj['attributes']['id'] + ": " + result);
return result;
}
function shouldHideBecauseInsideOtherManagersAreas(userPrefActive, obj, userID, userPrefInsideOtherManagersAreasUr, managedAreas) {
let result = false;
if (userPrefActive && userPrefInsideOtherManagersAreasUr) {
// check if inside others managed area:
Object.values(managedAreas["others"]).forEach(managedArea => {
let geometry = obj['attributes']['geometry'];
if (geometry instanceof OpenLayers['Geometry']['Polygon']) {
// noinspection JSUnresolvedFunction
geometry = geometry.getCentroid();
}
// noinspection JSUnresolvedFunction
if (managedArea['geometry'].containsPoint(geometry)) {
result = true;
// inside others managers area. check if not in mine
Object.values(managedAreas["my"]).forEach(managedArea => {
let geometry = obj['attributes']['geometry'];
if (geometry instanceof OpenLayers['Geometry']['Polygon']) {
// noinspection JSUnresolvedFunction
geometry = geometry.getCentroid();
}
// noinspection JSUnresolvedFunction
if (managedArea['geometry'].containsPoint(geometry)) {
console.debug('wme-ur-minus: shouldHideBecauseInsideOtherManagersAreas() obj: ' + obj['attributes']['id'] + " is in both other and mine areas. Will not hide");
result = false;
}
});
}
});
}
console.debug('wme-ur-minus: shouldHideBecauseInsideOtherManagersAreas() obj: ' + obj['attributes']['id'] + ": " + result);
return result;
}
function shouldHideBecauseChargingStationsPur(userPrefActive, venue, userPrefChargingStations) {
let result = false;
if (userPrefActive && userPrefChargingStations) {
if (venue.attributes.categories.includes("CHARGING_STATION")) {
result = true;
}
}
console.debug('wme-ur-minus: shouldHideBecauseChargingStationsPur() obj: ' + venue['attributes']['id'] + ": " + result);
return result;
}
function hideMUR(params) {
console.debug('wme-ur-minus: hideMUR() id: ' + params.id);
let olID = getOlID(params.id);
if (olID) {
jQuery('image[id="' + olID +'"]').hide();
} else {
console.warn('wme-ur-minus: hideMUR() id: ' + params.id + ' olID not found.');
}
}
function showMUR(params) {
console.debug('wme-ur-minus: showMUR() id: ' + params.id);
let olID = getOlID(params.id);
if (olID) {
let images = jQuery('image[id="' + olID +'"]');
images.show();
if (params.murComments) {
if (images.length === 1) {
addCommentsCounter(images[0], params.murComments);
} else {
console.debug('wme-ur-minus: showMUR() id: ' + params.id + ' not found with olID ' + olID);
}
}
} else {
console.warn('wme-ur-minus: showMUR() id: ' + params.id + ' not found.');
}
}
function getOlID(murID) {
let olID = W.map.mapUpdateRequestsLayer.featureMap.get(murID)?.geometry.id;
if (olID) {
return olID;
} else {
console.warn('wme-ur-minus: getOlID() id: ' + murID + ' not found.');
return null;
}
}
function addCommentsCounter(imageElement, murComments) {
console.debug('wme-ur-minus: addCommentsCounter() id: ' + murComments.id);
let commentsCounter = murComments?.comments?.length || 0;
const uniqueGroupId = `speech-bubble-group-${imageElement.id}`;
const existingGroup = imageElement.parentNode.querySelector(`#${uniqueGroupId}`);
if (existingGroup) {
existingGroup.remove();
}
const groupElement = document.createElementNS('http://www.w3.org/2000/svg', 'g');
groupElement.setAttribute('id', uniqueGroupId);
const imageX = parseFloat(imageElement.getAttribute('x'));
const imageY = parseFloat(imageElement.getAttribute('y'));
const imageWidth = parseFloat(imageElement.getAttribute('width'));
//const imageHeight = parseFloat(imageElement.getAttribute('height'));
const centerX = imageX + imageWidth - 7; // Adjusted to align with the bubble
const centerY = imageY + 7; // Adjusted to center vertically
const circleElement = document.createElementNS('http://www.w3.org/2000/svg', 'circle');
circleElement.setAttribute('cx', centerX);
circleElement.setAttribute('cy', centerY);
circleElement.setAttribute('r', '7'); // Radius to fully cover the three dots
circleElement.setAttribute('fill', '#ffffff'); // Match the speech bubble's white background
circleElement.setAttribute('stroke', '#000000'); // Optional border for visibility
circleElement.setAttribute('stroke-width', '1');
const textElement = document.createElementNS('http://www.w3.org/2000/svg', 'text');
textElement.setAttribute('x', centerX);
textElement.setAttribute('y', centerY);
textElement.setAttribute('dy', '3'); // Fine-tune vertical centering
textElement.setAttribute('font-size', '10'); // Slightly larger for readability
textElement.setAttribute('fill', '#000000'); // Black text
textElement.setAttribute('font-family', 'Arial, sans-serif');
textElement.setAttribute('text-anchor', 'middle'); // Center horizontally
textElement.textContent = commentsCounter.toString();
groupElement.appendChild(circleElement);
groupElement.appendChild(textElement);
imageElement.parentNode.appendChild(groupElement);
}
function handleUI() {
console.info('wme-ur-minus: handleUI()');
let registerSidebarTabResult = wMethods["registerSidebarTab"](SCRIPT_ID);
wMethods["waitForElementConnected"](registerSidebarTabResult['tabLabel']).then(() => {
registerSidebarTabResult['tabLabel'].innerText = "UR-";
});
wMethods["waitForElementConnected"](registerSidebarTabResult['tabPane']).then(() => {
drawUI(registerSidebarTabResult['tabPane']);
});
}
function drawUI(tabPane) {
console.debug('wme-ur-minus: drawUI()');
addUserScriptTab(tabPane);
addTooltips();
addStyle();
// register for new events. to invoke only once on startup, make a short delay
setTimeout(function () {
document.addEventListener("wme-map-data-loaded", processFilters)
}, 1000);
// refresh filters on panel close
document.body.addEventListener('click', function(event) {
if (event.target.classList.contains('close-panel')) {
console.debug('wme-ur-minus: click() on close-panel');
processFilters();
}
});
console.info('wme-ur-minus: drawUI() done');
}
function addStyle() {
GM_addStyle(`.ur-minus-comments-tooltip {
background: white;
border-radius: 50%;
padding: 8px;
border: 1px solid black;
width: 4px;
height: 4px;
position: absolute;
top: -8px;
left: 20px;
justify-content: center;
align-items: center;
display: flex;
font-size: 10px;
}`);
GM_addStyle(`.ur-minus-remove-after-background-image::after {
background-image: none !important;
}`);
}
function addUserScriptTab(tabPane) {
console.debug('wme-ur-minus: addUserScriptTab()');
let aSection = document.createElement('section');
aSection.id = 'close-mur';
aSection.className = 'tab-pane';
let section = document.createElement('p');
aSection.appendChild(section);
let userPrefActive = localStorage.getItem(USER_PREF_ACTIVE) === "true" ? ' checked' : '';
let userPrefActiveStr = userPrefActive ? '':' disabled';
let userPrefAllUrsStr = localStorage.getItem(USER_PREF_ALL_URS) === "true" ? ' checked' : '';
let userPrefNoCommentsStr = localStorage.getItem(USER_PREF_NO_COMMENTS) === "true" ? ' checked' : '';
let userPrefHasCommentsStr = localStorage.getItem(USER_PREF_HAS_COMMENTS) === "true" ? ' checked' : '';
let userPrefLastCommentByMeStr = localStorage.getItem(USER_PREF_LAST_COMMENT_BY_ME) === "true" ? ' checked' : '';
let userPrefLastCommentBefore = localStorage.getItem(USER_PREF_LAST_COMMENT_BEFORE) === "true" ? ' checked' : '';
let userPrefWithCommentsNotMine = localStorage.getItem(USER_PREF_WITH_COMMENTS_NOT_MINE) === "true" ? ' checked' : '';
let userPrefOutsideMyAreasUr = localStorage.getItem(USER_PREF_OUTSIDE_MY_AREAS_UR) === "true" ? ' checked' : '';
let userPrefInsideOtherManagersAreasUr = localStorage.getItem(USER_PREF_INSIDE_OTHERS_MANAGERS_AREAS_UR) === "true" ? ' checked' : '';
let userPrefAllPurs = localStorage.getItem(USER_PREF_ALL_PURS) === "true" ? ' checked' : '';
let userPrefAboveMyRank = localStorage.getItem(USER_PREF_ABOVE_MY_RANK) === "true" ? ' checked' : '';
let userPrefOutsideMyAreasPur = localStorage.getItem(USER_PREF_OUTSIDE_MY_AREAS_PUR) === "true" ? ' checked' : '';
let userPrefInsideOtherManagersAreasPur = localStorage.getItem(USER_PREF_INSIDE_OTHERS_MANAGERS_AREAS_PUR) === "true" ? ' checked' : '';
let userPrefChargingStationsPur = localStorage.getItem(USER_PREF_CHARGING_STATIONS) === "true" ? ' checked' : '';
section.innerHTML = '<b>When active, filter user reports</b><br/>' +
'<br/>Active: <input type="checkbox" id="WME_ur_minus_active"' + userPrefActive + '/><br/>' +
'<br/><b>Hide Map Update Requests:</b><br/>' +
'<input type="checkbox" id="' + USER_PREF_ALL_URS +'"' + userPrefAllUrsStr + userPrefActiveStr + '/> All<br/>' +
'<input type="checkbox" id="' + USER_PREF_HAS_COMMENTS +'"' + userPrefHasCommentsStr + userPrefActiveStr + '/> Have comments<br/>' +
'<input type="checkbox" id="' + USER_PREF_NO_COMMENTS +'"' + userPrefNoCommentsStr + userPrefActiveStr + '/> Have no comments<br/>' +
'<input type="checkbox" id="' + USER_PREF_LAST_COMMENT_BY_ME +'"' + userPrefLastCommentByMeStr + userPrefActiveStr + '/> Last comment by me<br/>' +
'<input type="checkbox" id="' + USER_PREF_LAST_COMMENT_BEFORE + '"' + userPrefLastCommentBefore + userPrefActiveStr + '/> Last comment at least <select id="' + USER_PREF_LAST_COMMENT_BEFORE_NUM_OF_DAYS + '"><option value="1">1</option><option value="2">2</option><option value="3">3</option><option value="4">4</option><option value="5">5</option><option value="6">6</option><option value="7">7</option><option value="8">8</option><option value="9">9</option><option value="10">10</option><option value="12">12</option><option value="14">14</option></select> days old.<br/>' +
'<input type="checkbox" id="' + USER_PREF_WITH_COMMENTS_NOT_MINE +'"' + userPrefWithCommentsNotMine + userPrefActiveStr + '/> With comments - not mine<br/>' +
'<input type="checkbox" id="' + USER_PREF_OUTSIDE_MY_AREAS_UR +'"' + userPrefOutsideMyAreasUr + userPrefActiveStr + '/> Outside my managed area(s)<br/>' +
'<input type="checkbox" id="' + USER_PREF_INSIDE_OTHERS_MANAGERS_AREAS_UR +'"' + userPrefInsideOtherManagersAreasUr + userPrefActiveStr + '/> Inside other managers areas<br/>' +
'<br/><b>Hide Place Update Requests:</b><br/>' +
'<input type="checkbox" id="' + USER_PREF_ALL_PURS + '"' + userPrefAllPurs + userPrefActiveStr + '/> All<br/>' +
'<input type="checkbox" id="' + USER_PREF_ABOVE_MY_RANK + '"' + userPrefAboveMyRank + userPrefActiveStr + '/> Above my rank<br/>' +
'<input type="checkbox" id="' + USER_PREF_OUTSIDE_MY_AREAS_PUR +'"' + userPrefOutsideMyAreasPur + userPrefActiveStr + '/> Outside my managed area(s)<br/>' +
'<input type="checkbox" id="' + USER_PREF_INSIDE_OTHERS_MANAGERS_AREAS_PUR +'"' + userPrefInsideOtherManagersAreasPur + userPrefActiveStr + '/> Inside other managers areas<br/>' +
'<input type="checkbox" id="' + USER_PREF_CHARGING_STATIONS +'"' + userPrefChargingStationsPur + userPrefActiveStr + '/> ' + I18n.lookup("venues.categories.CHARGING_STATION") + '<br/>';
tabPane.append(aSection);
// set values from storage and register events
jQuery("#WME_ur_minus_active").on("change", urMinusActiveChange);
ALL_USER_PERF_CHECKBOXES_IDS.forEach(id => {
jQuery("#" + id).on("change", urMinusUserPerfCheckboxChange);
})
let numOfDaysItem = jQuery("#" + USER_PREF_LAST_COMMENT_BEFORE_NUM_OF_DAYS);
numOfDaysItem.val(localStorage.getItem(USER_PREF_LAST_COMMENT_BEFORE_NUM_OF_DAYS));
numOfDaysItem.on("change", urMinusNumOfDaysChange);
console.info('wme-ur-minus: addUserScriptTab() done');
}
function addTooltipEvents() {
console.debug('wme-ur-minus: addTooltipEvents()');
// mur
//let urLayerId = wMethods["getLayerByName"]("update_requests").id;
let allUpdateRequestsSelector = "div.map-marker.user-generated.map-problem";
jQuery(allUpdateRequestsSelector).on( "mouseenter", {type:"mur"}, urMouseEnter ).on( "mouseleave", urMouseOut );
//console.debug('wme-ur-minus: addTooltipEvents() number of mur: ' + jQuery(allUpdateRequestsSelector).length + " murs: " + Object.keys(W.model.mapUpdateRequests.objects).length);
// pur
let allPlaceUpdateRequests = jQuery("div.map-marker.place-update").filter(function () {
return parseInt(jQuery(this).attr("data-update-count")) > 0;
});
jQuery(allPlaceUpdateRequests).on( "mouseenter", {type:"venue"} , urMouseEnter).on( "mouseleave", urMouseOut );
// mp
//let mpLayerId = wMethods["getLayerByName"]("mapProblems").id;
let allMapProblemsSelector = "div.map-marker:not(.user-generated).map-problem";
jQuery(allMapProblemsSelector).on( "mouseenter", {type:"mp"} , urMouseEnter).on( "mouseleave", urMouseOut );
// segments suggestions
//let ssLayerId = wMethods["getLayerByName"]("segment_suggestions_markers").id;
let allSSSelector = "div.map-marker.segment-suggestion-marker";
jQuery(allSSSelector).on( "mouseenter", {type:"ss"} , urMouseEnter).on( "mouseleave", urMouseOut );
}
function urMouseEnter(event) {
console.debug('wme-ur-minus: urMouseEnter()');
let toolTipDiv = jQuery('#' + WME_UR_MINUS_DIV_TOOLTIP_ID);
if (toolTipDiv.css("visibility") === "hidden") {
toolTipDiv.css('visibility', 'visible');
toolTipDiv.css('top', jQuery(event.currentTarget).offset().top);
toolTipDiv.css('left', jQuery(event.currentTarget).offset().left + 25);
toolTipDiv.css('display', 'block');
if (event.data.type === "mur") {
let murID = this.getAttribute('data-id');
let mur = W.model.mapUpdateRequests.objects[murID]
if (mur) {
toolTipDiv.html(getUrTooltipText(mur));
} else {
console.error('wme-ur-minus: urMouseEnter() mur not found: ' + murID);
}
} else if (event.data.type === "venue") {
let venueID = this.getAttribute('data-id');
let venue = W.model['venues'].objects[venueID]
if (venue) {
toolTipDiv.html(getPurTooltipText(venue));
} else {
console.error('wme-ur-minus: urMouseEnter() mur venue found: ' + venueID);
}
} else if (event.data.type === "mp") {
let mpID = this.getAttribute('data-id');
let mp = W.model['mapProblems'].objects[mpID]
if (mp) {
toolTipDiv.html(getMpTooltipText(mp));
} else {
console.error('wme-ur-minus: urMouseEnter() mp not found: ' + mpID);
}
} else if (event.data.type === "ss") {
let mousePosition = jQuery('.wz-map-ol-control-span-mouse-position').text().split(" ");
let lat = mousePosition[0];
let lon = mousePosition[1];
let ssID;
let candidateDistance = 1000000;
Object.keys(W.model['segmentSuggestions'].objects).forEach(id => {
let segmentSuggestion = W.model['segmentSuggestions'].objects[id];
//let centroid2 = segmentSuggestion.attributes.geometry.getCentroid();
let center = segmentSuggestion.getCenter();
let segmentSuggestionCentroid = convertTo4326(center.x,center.y);
let latDistance = Math.round(Math.abs(lat * 1000000 - segmentSuggestionCentroid.lat * 1000000));
let lonDistance = Math.round(Math.abs(lon * 1000000 - segmentSuggestionCentroid.lon * 1000000));
const expectedGap = Math.pow(2, 24 - W.map.getZoom()) * 1.5; // 14=1000; 15=500; 16=250; 17=125; 18= 64; 19=32; 2^(23-W.map.getZoom())
let maxDistance = Math.max(latDistance, lonDistance)
if (maxDistance < expectedGap && maxDistance < candidateDistance) {
console.debug('wme-ur-minus: urMouseEnter() for ss ID: ' + id + ' max distance is: ' + maxDistance + " less than expected: " + expectedGap + " and less than candidate: " + candidateDistance);
ssID = id;
candidateDistance = maxDistance;
} else {
console.debug('wme-ur-minus: urMouseEnter() for ss ID: ' + id + ' max distance is: ' + maxDistance + " more than expected: " + expectedGap);
}
});
if (ssID) {
let ss = W.model['segmentSuggestions'].objects[ssID]
if (ss) {
toolTipDiv.html(getSsTooltipText(ss));
} else {
console.error('wme-ur-minus: urMouseEnter() ss not found for ID ' + ssID + '. No tooltip will be generated');
}
} else {
console.error('wme-ur-minus: urMouseEnter() No ID was found as candidate. No tooltip will be generated');
}
}
}
}
function getPurTooltipText(venue) {
let attributes = venue.attributes;
console.debug('wme-ur-minus: getPurTooltipText() venue ID: ' + attributes.id);
let venueName = getVenueName(attributes);
let result = '<p style="text-align: center"><b>' + venueName + '</b></p>';
// open
result += "<b><u>Open in:</u></b> ";
result += "<a target='_blank' href='" + getPurURL(attributes) + "'>new tab</a>";
result += "<br/>";
// update type
let updateTypeValue;
let updateType = venue.attributes['venueUpdateRequests'][0].attributes['updateType'];
if (updateType === 'flag') {
let subject = venue.attributes['venueUpdateRequests'][0].attributes['subject'];
updateTypeValue = I18n.lookup("venues.update_requests.panel.flag_title." + subject);
} else {
updateTypeValue = I18n.lookup("venues.update_requests.panel.title." + updateType);
}
result += "<b><u>Reason:</u></b> ";
result += escapeTooltipText(updateTypeValue);
result += "<br/>";
// description
result += "<b><u>Description:</u></b> ";
result += attributes.description?escapeTooltipText(attributes.description):"(none)";
result += "<br/>";
// categories
result += "<b><u>Categories:</u></b> ";
let translatedCategories = attributes.categories.map(category => I18n.lookup("venues.categories." + category));
result += escapeTooltipText(translatedCategories.join(","));
result += "<br/>";
return result;
}
function getVenueName(attributes) {
if (attributes['residential']) {
let houseNumber = attributes['houseNumber'];
let streetID = attributes['streetID'];
if (streetID) {
let streenName = W.model.streets.objects[streetID].name || W.model.streets.objects[streetID].attributes.name;
let cityID = W.model.streets.objects[streetID].cityID || W.model.streets.objects[streetID].attributes.cityID;
let cityName = W.model['cities'].objects[cityID].attributes.name;
return streenName + " " + houseNumber + ", " + cityName;
} else {
// TODO: I18n.lookup("");
return I18n.lookup("edit.venue.no_address");
}
} else {
return attributes['name'];
}
}
function getUrTooltipText(mur) {
let attributes = mur.attributes;
console.debug('wme-ur-minus: getUrTooltipText() murID: ' + attributes.id);
let result = '<p style="text-align: center"><b>' + attributes['typeText'] + '</b></p>';
// open
result += "<b><u>Open in:</u></b> ";
result += "<a target='_blank' href='" + getMurURL(attributes) + "'>new tab</a>";
result += "<br/>";
// ID
result += "<b><u>ID:</u></b> ";
result += attributes.id;
result += "<br/>";
// description
result += "<b><u>Description:</u></b> ";
result += attributes.description?escapeTooltipText(attributes.description):"(none)";
result += "<br/>";
// driveDate
result += "<b><u>Drive Date:</u></b> ";
result += new Date(attributes.driveDate).toLocaleString(I18n.locale, {year: '2-digit', month: '2-digit',day: '2-digit', hour:'numeric', minute:'2-digit'});
result += "<br/>";
// Source
if (attributes.source) {
result += "<b><u>Source:</u></b> ";
result += I18n.lookup(`issue_tracker.filters.UPDATE_REQUESTS.sources.${attributes.source}.title`);
result += "<br/>";
}
return result;
}
function getMpTooltipText(mp) {
let attributes = mp.attributes;
console.debug('wme-ur-minus: getMpTooltipText() murID: ' + attributes.id);
let subType = attributes['subType'];
let texts = I18n.lookup(`problems.types.${subType}`);
let result = '<p style="text-align: center"><b>' + texts['title'] + '</b></p>';
// open
result += "<b><u>Open in:</u></b> ";
result += "<a target='_blank' href='" + getMpURL(attributes) + "'>new tab</a>";
result += "<br/>";
// ID
result += "<b><u>ID:</u></b> ";
result += attributes.id;
result += "<br/>";
// description
result += "<b><u>Description:</u></b> ";
result += texts['description'];
result += "<br/>";
// solution
result += "<b><u>Solution:</u></b> ";
result += texts['solution'];
result += "<br/>";
return result;
}
function getSsTooltipText(ss) {
let attributes = ss.attributes;
console.debug('wme-ur-minus: getSsTooltipText() murID: ' + attributes.id);
let texts = I18n.lookup(`edit.segment_suggestion`);
let result = '<p style="text-align: center"><b>' + texts['title'] + '</b></p>';
// open
result += "<b><u>Open in:</u></b> ";
result += "<a target='_blank' href='" + getSsURL(ss) + "'>new tab</a>";
result += "<br/>";
// ID
result += "<b><u>ID:</u></b> ";
result += attributes.id;
result += "<br/>";
return result;
}
function getPurURL(purAttributes) {
let id = purAttributes.id;
let boundsCenterLonLat = purAttributes.geometry.getBounds().getCenterLonLat();
let x = purAttributes.geometry.x?purAttributes.geometry.x:boundsCenterLonLat.lon;
let y = purAttributes.geometry.y?purAttributes.geometry.y:boundsCenterLonLat.lat;
let convertedTo4326 = convertTo4326(x, y);
return document.location.origin + "/" + I18n.locale + "/editor?env=" + wMethods["getAppRegionCode"]() + "&lon=" + convertedTo4326.lon.toFixed(6) + "&lat=" + convertedTo4326.lat.toFixed(6) + "&zoomLevel=19&venueUpdateRequest=" + id;
}
function getMurURL(murAttributes) {
let id = murAttributes.id;
let x = murAttributes.geometry.x;
let y = murAttributes.geometry.y;
let convertedTo4326 = convertTo4326(x, y);
return document.location.origin + "/" + I18n.locale + "/editor?env=" + wMethods["getAppRegionCode"]() + "&lon=" + convertedTo4326.lon.toFixed(6) + "&lat=" + convertedTo4326.lat.toFixed(6) + "&zoomLevel=19&mapUpdateRequest=" + id;
}
function getMpURL(mpAttributes) {
let id = mpAttributes.id;
let x = mpAttributes.geometry.x;
let y = mpAttributes.geometry.y;
let convertedTo4326 = convertTo4326(x, y);
return document.location.origin + "/" + I18n.locale + "/editor?env=" + wMethods["getAppRegionCode"]() + "&lon=" + convertedTo4326.lon.toFixed(6) + "&lat=" + convertedTo4326.lat.toFixed(6) + "&zoomLevel=19&mapProblem=" + id;
}
function getSsURL(ss) {
let ssAttributes = ss.attributes;
let id = ssAttributes.id;
let center = ss.getCenter();
let convertedTo4326 = convertTo4326(center.x,center.y);
return document.location.origin + "/" + I18n.locale + "/editor?env=" + wMethods["getAppRegionCode"]() + "&lon=" + convertedTo4326.lon.toFixed(6) + "&lat=" + convertedTo4326.lat.toFixed(6) + "&zoomLevel=19&segmentSuggestions=" + id;
}
function convertTo4326(x, y) {
let projI = new Projection("EPSG:900913");
let projE = new Projection("EPSG:4326");
return (new LonLat(x, y)).transform(projI, projE);
}
function escapeTooltipText(str) {
return (str||'').replace('%20', '"').replace('%27', "'").replace('%28', "(").replace('%29', ")");
}
function urMouseOut() {
console.debug('wme-ur-minus: urMouseOut()');
let tooltipDiv = jQuery('#' + WME_UR_MINUS_DIV_TOOLTIP_ID + ':hover');
if (tooltipDiv.length > 0) {
tooltipDiv.on("mouseleave", function () {
jQuery('#' + WME_UR_MINUS_DIV_TOOLTIP_ID).css('visibility', 'hidden');
});
} else {
jQuery('#' + WME_UR_MINUS_DIV_TOOLTIP_ID).css('visibility', 'hidden');
}
}
function urMinusActiveChange() {
console.info('wme-ur-minus: urMinusActiveChange() new value: ' + this.checked);
localStorage.setItem(USER_PREF_ACTIVE, this.checked);
if (this.checked) {
setEnabled();
} else {
setDisabled();
}
processFilters();
}
function setEnabled() {
console.info('wme-ur-minus: setEnabled()');
ALL_USER_PERF_CHECKBOXES_IDS.forEach(id => {
jQuery("#" + id).removeAttr("disabled");
});
}
function setDisabled() {
console.info('wme-ur-minus: setDisabled()');
ALL_USER_PERF_CHECKBOXES_IDS.forEach(id => {
jQuery("#" + id).attr("disabled", "true");
});
jQuery("div.ur-minus-comments-tooltip").remove();
jQuery("div.has-comments").removeClass('ur-minus-remove-after-background-image');
}
function urMinusUserPerfCheckboxChange() {
console.info('wme-ur-minus: urMinusUserPerfCheckboxChange() for ID: "' + this.id+ '" new value: ' + this.checked);
localStorage.setItem(this.id, this.checked);
processFilters();
}
function urMinusNumOfDaysChange() {
console.info('wme-ur-minus: urMinusNumOfDaysChange() new value: ' + this.value);
localStorage.setItem(USER_PREF_LAST_COMMENT_BEFORE_NUM_OF_DAYS, this.value);
processFilters();
}
function addTooltips() {
console.debug('wme-ur-minus: addTooltips()');
let urTooltipDiv = document.createElement('div');
urTooltipDiv.id = WME_UR_MINUS_DIV_TOOLTIP_ID;
urTooltipDiv.setAttribute("style", "position: absolute; visibility: hidden; top: 107px; left: 1040px; z-index: 10000; background-color: aliceblue; border-width: 3px; border-style: solid; border-radius: 10px; box-shadow: silver 5px 5px 10px; padding: 4px; height: auto; width: auto; overflow: auto; max-width: 300px; word-wrap: break-word");
document.body.appendChild(urTooltipDiv);
urTooltipDiv.addEventListener("mouseover", function () {
this.style.visibility = 'visible';
}, false);
urTooltipDiv.addEventListener("mouseout", function () {
this.style.visibility = 'hidden';
}, false);
urTooltipDiv.addEventListener("dblclick", function () {
this.style.visibility = 'hidden';
}, false);
console.debug('wme-ur-minus: addTooltips() done');
}
function getUpdateRequestSessions(ids, cb) {
console.info('wme-ur-minus: getUpdateRequestSessions() for ' + ids.length + ' IDs.');
let url = "https://" + document.location.host + W['Config'].paths['updateRequestSessions'] + "?ids=" + ids.join(",");
fetch(url, {
headers: {
"X-CSRF-Token": urMinus_csrfToken,
"Accept": "application/json"
}
}).then(response => {
if (response.ok && response.status === 200) {
console.info('wme-ur-minus: getUpdateRequestSessions() response OK');
response.json().then(json => cb(json));
} else {
console.error('wme-ur-minus: getUpdateRequestSessions() failed with status: ' + response.status);
}
}).catch(error => {
console.error('wme-ur-minus: getUpdateRequestSessions() failed with error: ' + error);
});
}
function getManagedAreasWme() {
let userID = W['loginManager']['user']['id'] || W['loginManager']['user']['attributes']['id'];
let userName = W['loginManager']['user']['userName'] || W['loginManager']['user']['attributes']['userName'];
console.debug('wme-ur-minus: getManagedAreasWme() for ' + userName);
let my = {};
let others = {};
if (W.model['managedAreas'] && W.model['managedAreas'].objects) {
Object.values(W.model['managedAreas'].objects).forEach(managedArea => {
if ((managedArea['userID'] || managedArea.attributes['userID']) === userID) {
my[managedArea['id']] = managedArea;
} else {
others[managedArea['id'] || managedArea.attributes['id']] = managedArea;
}
});
}
console.info('wme-ur-minus: getManagedAreasWme() for ' + userName + ' returning my ' + Object.keys(my).length + " areas");
console.info('wme-ur-minus: getManagedAreasWme() for ' + userName + ' returning others ' + Object.keys(others).length + " areas");
return {my:my, others:others};
}
function storeAreaGeomenty(managedArea) {
let id = managedArea['id'];
console.debug('wme-ur-minus: storeAreaGeomenty() id: ' + id);
// noinspection JSUnresolvedFunction
let vertices = managedArea['geometry'].getVertices();
let pointsArray = vertices.map(vertice => [vertice['x'],vertice['y']]);
let fromStorage = JSON.parse(localStorage.getItem(STORAGE_KEY_MY_MANAGED_AREAS) || "{}");
fromStorage[id] = pointsArray;
localStorage.setItem(STORAGE_KEY_MY_MANAGED_AREAS, JSON.stringify(fromStorage));
console.info('wme-ur-minus: storeAreaGeomenty() id: ' + id + " done.");
}
function getManagedAreas() {
console.debug('wme-ur-minus: getManagedAreas()');
let my = {};
// get from WME:
let fromWME = getManagedAreasWme();
// get from storage:
let myFromStorage = JSON.parse(localStorage.getItem(STORAGE_KEY_MY_MANAGED_AREAS) || "{}");
// convert ones from storage from string into objects:
Object.entries(myFromStorage).forEach(([id, xyArray]) => {
let pointsArray = xyArray.map(xy => new Point(xy[0], xy[1]));
let linearRing = new LinearRing(pointsArray)
let polygon = new Polygon([linearRing]);
my[id] = { geometry:polygon };
});
// find which of my areas was found in WME but not yet in storage, and store them.
let needToStore = Object.keys(fromWME['my']).filter(key => !myFromStorage[key]);
needToStore.forEach(key => {
console.debug('wme-ur-minus: getManagedAreas() area ' + key + ' found in WME but not in storage.');
storeAreaGeomenty(fromWME['my'][key]);
my[key] = fromWME[key];
});
return {my:my, "others":fromWME["others"]};
}
function dataLoadedEvent() {
console.debug('wme-ur-minus: dataLoadedEvent()');
addTooltipEvents();
}
})();