// ==UserScript==
// @name WME Fix UI Memorial Edition
// @namespace https://greasyfork.org/en/scripts/435828-wme-fix-ui-memorial-edition
// @description Allows alterations to the WME UI to fix things screwed up or ignored by Waze
// @include https://www.waze.com/editor*
// @include https://www.waze.com/*/editor*
// @include https://beta.waze.com/editor*
// @include https://beta.waze.com/*/editor*
// @exclude https://www.waze.com/*user/editor/*
// @supportURL https://www.waze.com/forum/viewtopic.php?t=334618
// @version 1.69.257
// @grant none
// ==/UserScript==
/*
Memorial Edition thanks:
phuz, fuji2086, Timbones, laurenthembprd, jm6087, BeastlyHaz
Original version thanks:
Bellhouse, Twister-UK, Timbones, Dave2084, Rickzabel, Glodenox,
JJohnston84, SAR85, Cardyin, JustinS83, berestovskyy, Sebiseba,
The_Cre8r, ABelter
=======================================================================================================================
Bug fixes - MUST BE CLEARED BEFORE RELEASE
=======================================================================================================================
*/
/* JSHint Directives */
/* globals $: */
/* globals W: true */
/* globals I18n: */
/* globals OpenLayers: true */
/* globals trustedTypes: */
/* globals ResizeObserver: */
/* globals jQuery: */
/* jshint bitwise: false */
/* jshint eqnull: true */
/* jshint esversion: 11 */
/* jshint undef: true */
/* jshint unused: true */
(function ()
{
// global variables
let fumeVersion = "1.69.257";
let fumeDate = "2024-10-05";
const newVersionNotes =
[
"Compatibility updates"
];
let oldVersion;
let prefix = "WMEFUME";
let tabAttempts = 0;
let debug = false;
let wmeFUinitialising = true;
let kineticDragParams;
let yslider;
let layersButton,refreshButton,shareButton;
let zliResizeObserver = null;
let killTurnPopup = false;
let abAlerts = null;
let abAlertBoxStack = [];
let abAlertBoxTickAction = null;
let abAlertBoxCrossAction = null;
let abAlertBoxInUse = false;
const urlSelectableItems =
[
// itemType isArray?
['segments', true],
['venues', true],
['nodes', false],
['mapComments', false],
['cameras', false]
];
function abAlertBoxObj(headericon, title, content, hasCross, tickText, crossText, tickAction, crossAction)
{
this.headericon = headericon;
this.title = title;
this.content = content;
this.hasCross = hasCross;
this.tickText = tickText;
this.crossText = crossText;
this.tickAction = tickAction;
this.crossAction = crossAction;
}
function abCloseAlertBox()
{
document.getElementById('abAlerts').childNodes[0].innerHTML = modifyHTML('');
document.getElementById('abAlerts').childNodes[1].innerHTML = modifyHTML('');
document.getElementById('abAlertTickBtnCaption').innerHTML = modifyHTML('');
document.getElementById('abAlertCrossBtnCaption').innerHTML = modifyHTML('');
abAlertBoxTickAction = null;
abAlertBoxCrossAction = null;
document.getElementById('abAlerts').style.visibility = "hidden";
document.getElementById('abAlertCrossBtn').style.visibility = "hidden";
abAlertBoxInUse = false;
if(abAlertBoxStack.length > 0)
{
abBuildAlertBoxFromStack();
}
}
function abCloseAlertBoxWithTick()
{
if(typeof abAlertBoxTickAction === 'function')
{
abAlertBoxTickAction();
}
abCloseAlertBox();
}
function abCloseAlertBoxWithCross()
{
if(typeof abAlertBoxCrossAction === 'function')
{
abAlertBoxCrossAction();
}
abCloseAlertBox();
}
function abShowAlertBox(headericon, title, content, hasCross, tickText, crossText, tickAction, crossAction)
{
abAlertBoxStack.push(new abAlertBoxObj(headericon, title, content, hasCross, tickText, crossText, tickAction, crossAction));
if(abAlertBoxInUse === false)
{
abBuildAlertBoxFromStack();
}
}
function abBuildAlertBoxFromStack()
{
const tbIcon = "";
abAlertBoxInUse = true;
abAlertBoxTickAction = null;
abAlertBoxCrossAction = null;
var titleContent = '<span style="font-size:14px;padding:2px;">';
//titleContent += '<i class="fa '+abAlertBoxStack[0].headericon+'"> </i> ';
titleContent += '<img src="'+tbIcon+'">';
titleContent += abAlertBoxStack[0].title;
titleContent += '</span>';
document.getElementById('abAlerts').childNodes[0].innerHTML = modifyHTML(titleContent);
document.getElementById('abAlerts').childNodes[1].innerHTML = modifyHTML(abAlertBoxStack[0].content);
document.getElementById('abAlertTickBtnCaption').innerHTML = modifyHTML(abAlertBoxStack[0].tickText);
if(abAlertBoxStack[0].hasCross)
{
document.getElementById('abAlertCrossBtnCaption').innerHTML = modifyHTML(abAlertBoxStack[0].crossText);
document.getElementById('abAlertCrossBtn').style.visibility = "visible";
if(typeof abAlertBoxStack[0].crossAction === "function")
{
abAlertBoxCrossAction = abAlertBoxStack[0].crossAction;
}
}
else
{
document.getElementById('abAlertCrossBtn').style.visibility = "hidden";
}
if(typeof abAlertBoxStack[0].tickAction === "function")
{
abAlertBoxTickAction = abAlertBoxStack[0].tickAction;
}
document.getElementById('abAlerts').style.visibility = "";
abAlertBoxStack.shift();
}
function abInitialiseAlertBox()
{
// create a new div to display script alerts
abAlerts = document.createElement('div');
abAlerts.id = "abAlerts";
abAlerts.style.position = 'fixed';
abAlerts.style.visibility = 'hidden';
abAlerts.style.top = '50%';
abAlerts.style.left = '50%';
abAlerts.style.zIndex = 10000;
abAlerts.style.backgroundColor = 'aliceblue';
abAlerts.style.borderWidth = '3px';
abAlerts.style.borderStyle = 'solid';
abAlerts.style.borderRadius = '10px';
abAlerts.style.boxShadow = '5px 5px 10px Silver';
abAlerts.style.padding = '4px';
abAlerts.style.webkitTransform = "translate(-50%, -50%)";
abAlerts.style.transform = "translate(-50%, -50%)";
var alertsHTML = '<div id="header" style="padding: 4px; background-color:LightBlue; font-weight: bold;">Alert title goes here...</div>';
alertsHTML += '<div id="content" style="padding: 4px; background-color:White; overflow:auto;max-height:500px">Alert content goes here...</div>';
alertsHTML += '<div id="controls" align="center" style="padding: 4px;">';
alertsHTML += '<span id="abAlertTickBtn" style="cursor:pointer;font-size:14px;border:thin outset black;padding:2px 10px 2px 10px;">';
alertsHTML += '<i class="fa fa-check"> </i>';
alertsHTML += '<span id="abAlertTickBtnCaption" style="font-weight: bold;"></span>';
alertsHTML += '</span>';
alertsHTML += ' ';
alertsHTML += '<span id="abAlertCrossBtn" style="cursor:pointer;font-size:14px;border:thin outset black;padding:2px 10px 2px 10px;">';
alertsHTML += '<i class="fa fa-times"> </i>';
alertsHTML += '<span id="abAlertCrossBtnCaption" style="font-weight: bold;"></span>';
alertsHTML += '</span>';
alertsHTML += '</div>';
abAlerts.innerHTML = modifyHTML(alertsHTML);
document.body.appendChild(abAlerts);
}
function SetStyle(elm, style, value, important)
{
if((elm !== null) && (elm != undefined))
{
if(important === true)
{
value += '!important';
}
elm.style[style] = value;
}
}
function modifyHTML(htmlIn)
{
if(typeof trustedTypes === "undefined")
{
return htmlIn;
}
else
{
const escapeHTMLPolicy = trustedTypes.createPolicy("forceInner", {createHTML: (to_escape) => to_escape});
return escapeHTMLPolicy.createHTML(htmlIn);
}
}
function reorderDaysOfWeek(origDOWs)
{
let dows = [];
for(let i = 0; i < 6; ++i)
{
dows.push(origDOWs[i + 1]);
}
dows.push(origDOWs[0]);
return dows;
}
// contrast items
function GetBorderContrast(contrast, isImportant)
{
var retval = 'border: 1px solid ' + ['','lightgrey','grey'][contrast];
if(isImportant === true)
{
retval += '!important';
}
retval += '; ';
return retval;
}
//Mutation Observer for daterangepicker in Restrictions
var RestrictionObserver = new MutationObserver(function(mutations)
{
if (getById('_cbMondayFirst').checked || getById('_cbISODates').checked)
{
mutations.forEach(function(mutation)
{
if ($(mutation.target).hasClass('modal-content'))
{
if (mutation.addedNodes.length > 0)
{
if ($(".datepicker").length > 0)
{
let DRP = $(".datepicker")[0];
let dows = reorderDaysOfWeek($(DRP).data("daterangepicker").locale.daysOfWeek);
if (getById('_cbMondayFirst').checked)
{
if($(DRP).data("daterangepicker").locale.firstDay === 0)
{
$(DRP).data("daterangepicker").locale.firstDay = 1;
$(DRP).data("daterangepicker").locale.daysOfWeek = dows;
}
}
if (getById('_cbISODates').checked)
{
$(DRP).data("daterangepicker").locale.format = "YYYY-MM-DD";
DRP.value = $(DRP).data("daterangepicker").startDate._i + " - " + $(DRP).data("daterangepicker").endDate._i;
}
}
}
}
});
}
});
function tameSegmentTypeMenu()
{
// Remove whitespace from the segment type menu if it's being used
if(getById('_cbTameSegmentTypeMenu').checked)
{
// Check the menu is being shown - this won't be the case in compact mode
let rtm = document.querySelector('[name="roadType"]');
if(rtm != null)
{
if(rtm.childElementCount > 0)
{
let menuDividers = rtm.getElementsByTagName('wz-menu-divider');
let pe;
let sr;
for(pe of menuDividers)
{
SetStyle(pe, "display", "none", false);
}
let menuTitles = rtm.getElementsByTagName('wz-menu-title');
for (pe of menuTitles)
{
sr = pe.shadowRoot.querySelector('.wz-menu-title');
SetStyle(sr, "padding", "0px, 0px, 0px 4px", false);
SetStyle(sr, "height", "100%", false);
SetStyle(sr, "lineHeight", "100%", false);
SetStyle(sr, "minHeight", "auto", false);
}
let menuEntries = rtm.getElementsByTagName('wz-option');
for (pe of menuEntries)
{
sr = pe.shadowRoot.querySelector('.wz-menu-item');
SetStyle(sr, "padding", "0px 0px 0px 24px", false);
SetStyle(sr, "height", "100%", false);
SetStyle(sr, "lineHeight", "100%", false);
SetStyle(sr, "minHeight", "auto", false);
}
}
}
}
}
function tameSegmentElevationMenu()
{
// Remove those options from the elevation menu which no-one is ever likely to need
if(getById('_cbTameElevationMenu').checked)
{
if(document.getElementsByName('level').length > 0)
{
if(document.getElementsByName('level')[0].childElementCount > 0)
{
let menuEntries = document.getElementsByName('level')[0].getElementsByTagName('wz-option');
let localeGround = I18n.lookup('edit.segment.levels')[0];
for (let pe of menuEntries)
{
let level = 0;
if(pe.value != localeGround)
{
level = parseInt(pe.value);
}
if((level > 5) || (level < -5))
{
SetStyle(pe, "display", "none", false);
}
else
{
let sr = pe.shadowRoot.querySelector('.wz-menu-item');
let indent = 44 + (level * 8);
SetStyle(sr, "padding", "0px 0px 0px "+indent+"px", false);
SetStyle(sr, "height", "100%", false);
SetStyle(sr, "lineHeight", "100%", false);
SetStyle(sr, "minHeight", "auto", false);
}
}
}
}
}
}
function hideSegmentPanelLabels()
{
// Hide the labels in the segment edit panel to give more vertical space to the things we need to interact with
if (getById('_cbHideSegmentPanelLabels').checked)
{
let tObj = document.getElementById('edit-panel').getElementsByTagName('wz-tab');
if(tObj.length > 0)
{
tObj = tObj[0].getElementsByTagName('wz-label');
if(tObj.length > 0)
{
for(let l of tObj)
{
if(l.getElementsByTagName('wz-checkbox').length == 0)
{
l.style.display = "none";
}
}
}
}
}
}
function hideRoutingPreferenceDescription()
{
// Remove that slightly annoying "used as" message under the routing option buttons
if(getById('_cbRemoveRoutingReminder').checked)
{
SetStyle(document.querySelector('.routing-road-type-message')?.parentElement, "display", "none", false);
}
}
function compressSegmentPanel()
{
if (getById('_cbCompressSidebar').checked)
{
if(getById('_inpUICompression').value > 0)
{
// Reduce padding enough so that the compact mode segment type selectors stand a reasonable chance
// of fitting onto two lines instead of needing to spill over onto a third...
SetStyle(document.querySelector('wz-tab')?.shadowRoot.querySelector('.wz-tab'), "padding", "2px", false);
// Reduce gap between the "Select entire street" and "Edit house numbers" buttons
SetStyle(document.querySelector('#segment-edit-general'), "gap", "0px", false);
// Reduce gap under the direction and lock level selectors
SetStyle(document.querySelector('[class^="direction-editor"]'), "marginBottom", "0px", false);
SetStyle(document.querySelector('.lock-edit-view'), "marginBottom", "0px", false);
// Reduce height of the speed limit input boxes
let nSLE = document.querySelectorAll('#segment-edit-general .speed-limit-input').length;
if(nSLE > 0)
{
for(let i = 0; i < nSLE; ++i)
{
let sr = document.querySelectorAll('#segment-edit-general .speed-limit-input')[i].shadowRoot;
SetStyle(sr.querySelector('.wz-text-input'), "height", "26px", false);
}
}
// Reduce height of the elevation drop-down - all this just to tweak the height of ONE UI element, thank you
// VERY much shadowroot :-/
if(document.getElementsByName('level')[0] != undefined)
{
let sr = document.getElementsByName('level')[0].shadowRoot;
SetStyle(sr.querySelector('.wz-select'), "height", "20px", false);
SetStyle(sr.querySelector('.selected-value-wrapper'), "height", "20px", false);
SetStyle(sr.querySelector('.select-wrapper'), "height", "20px", false);
SetStyle(sr.querySelector('.select-box'), "height", "20px", false);
}
}
}
}
function addAltSegmentLength()
{
let segLen = document.querySelector('.length-attribute')?.querySelector('.value');
if(segLen !== undefined)
{
if(segLen.touchedByFUME === undefined)
{
let nSegments = W.selectionManager.getSelectedWMEFeatures().length;
let sLength = 0;
while(nSegments > 0)
{
--nSegments;
sLength += W.selectionManager.getSelectedWMEFeatures()[nSegments]._wmeObject.getOLGeometry().getLength();
}
let sAdjLength = AdjustForLatitude(sLength);
let altDist = GetAltDistanceString(sAdjLength);
segLen.innerText += ' / ' + altDist;
segLen.touchedByFUME = true;
}
}
}
// Mutation Observer for segment edit panel
var SegmentObserver = new MutationObserver(function()
{
tameSegmentTypeMenu();
tameSegmentElevationMenu();
hideSegmentPanelLabels();
hideRoutingPreferenceDescription();
compressSegmentPanel();
addAltSegmentLength();
});
//Mutation Observer for daterangepicker in Closures
var ClosureObserver = new MutationObserver(function(mutations)
{
if (getById('_cbMondayFirst').checked)
{
mutations.forEach(function(mutation)
{
if (mutation.target.className == "closures")
{
if (mutation.addedNodes.length > 0)
{
if (mutation.addedNodes[0].firstChild.classList.contains("edit-closure"))
{
if($(".start-date").data("daterangepicker").locale.firstDay === 0)
{
$(".start-date").data("daterangepicker").locale.firstDay = 1;
$(".end-date").data("daterangepicker").locale.firstDay = 1;
let dows = reorderDaysOfWeek($(".start-date").data("daterangepicker").locale.daysOfWeek);
$(".start-date").data("daterangepicker").locale.daysOfWeek = dows;
$(".end-date").data("daterangepicker").locale.daysOfWeek = dows;
}
}
}
}
});
}
});
// Mutation Observer for place edit panel
var PlaceObserver = new MutationObserver(function()
{
// This slightly convoluted bit of code allows us to manipulate
// the entries in the dynamically created drop-down list which
// is generated whenever you start searching for a GMaps place
// to link to a native one
if (getById('_cbFixExternalProviders').checked)
{
// First check that the MO has fired because the user has selected
// a place for editing...
let acMenu = document.getElementsByClassName('external-provider-edit-form')[0];
if(acMenu !== undefined)
{
// ...and then check the "add linked Google place" option has been
// selected, to start the process of generating the dynamic list
let acInner = acMenu.getElementsByTagName('wz-autocomplete')[0];
if(acInner !== undefined)
{
// If so, then we now need to poll the UI to see if there are any
// list items present - this doesn't appear to be possible via a
// MO due to the items being hidden behind shadowroot, hence the
// slightly old-school nature of polling vs event driven here :-/
window.setTimeout(EPObserver, 500);
}
}
}
let compress = getById('_inpUICompression').value;
if(compress > 0)
{
// Also check for the existence of the entry point UI element, so
// we can dive into its shadowroot to deal with its excessive whitespace
if(document.getElementsByClassName('navigation-point-item navigation-point-editable').length > 0)
{
let npOuter = document.getElementsByClassName('navigation-point-item navigation-point-editable')[0];
let npInner = npOuter.shadowRoot.querySelectorAll('.list-item-wrapper');
if(npInner.length > 0)
{
for(let i of npInner)
{
i.style.paddingTop = "4px";
i.style.paddingBottom = "4px";
}
}
}
}
});
// Mutation Observer for issue tracker panel
var ITObserver = new MutationObserver(function()
{
disableUITransitions();
});
var MTEObserver = new MutationObserver(function()
{
checkForMTEDropDown();
});
// To unwind the temporary moving of the notification icons when entering HN edit mode above, we observe changes in the primary
// toolbar in order to detect when the close HN editor button is displayed, and also when the regular toolbar contents are
// displayed. We use the former to set up an onclick handler so that when the close editor button is clicked, we set a flag
// to then trigger the deferred unwinding of the icon move once we detect the toolbar is back to normal...
let doCleanUpAfterHNEdit = false;
var PriToolBarObserver = new MutationObserver(function()
{
if(document.querySelector('wz-button.waze-icon-exit') !== null)
{
// The HN topbar buttons don't exist within the DOM until the user initiates HN editing, so we need to
// perform a seperate application of the button-specific parts of the topbar compression mods
CheckForHNButtons();
}
if(document.querySelector('#delete-button') !== null)
{
isSaveBtnVisible();
}
});
var RulerShownObserver = new MutationObserver(function()
{
let rDi = document.querySelector('.ruler-snackbar-tnum');
if(rDi !== null)
{
RulerDistanceAdjust();
}
});
function GetAltDistanceString(distMetres)
{
let altDist = '';
// Convert into the most appropriate distance that complements
// the natively displayed distance based on whether WME is running
// in imperial or metric...
if(W.model.isImperial === true)
{
// m or km
if(distMetres < 1000)
{
altDist = distMetres.toFixed(1) + " " + GetLocalisedDistanceUnits("m");
}
else
{
altDist = (distMetres / 1000).toFixed(1) + " " + GetLocalisedDistanceUnits("km");
}
}
else
{
// ft or mi
let ft = (distMetres * 3.28084);
if(ft < 5280)
{
altDist = ft.toFixed(1) + " " + GetLocalisedDistanceUnits("ft");
}
else
{
altDist = (ft / 5280).toFixed(1) + " " + GetLocalisedDistanceUnits("mi");
}
}
return altDist;
}
function AdjustForLatitude(distMetres)
{
let lat = W.map.getUnprojectedCenter().lat;
let cosLat = Math.cos((lat / 360) * (Math.PI * 2));
return distMetres * cosLat;
}
function GetLocalisedDistanceUnits(rawUnits)
{
let retval = rawUnits; // default to returning whatever was passed in
let localised = I18n.lookup("measurements.length." + rawUnits);
if(localised !== undefined)
{
let units = localised.split(' ');
if(units.length == 2)
{
retval = units[1];
}
}
return retval;
}
let prevDistance = -1;
function RulerDistanceAdjust()
{
let rDi = document.querySelector('.ruler-snackbar-tnum');
if(rDi !== null)
{
let rLayer = W.map.getLayerByUniqueName("OpenLayers.Handler.Path");
if(rLayer !== undefined)
{
// Get length of the measurement tool line as drawn onscreen
let rDist = AdjustForLatitude(rLayer.features[0].geometry.getLength());
if(rDist != prevDistance)
{
prevDistance = rDist;
rDi.innerText = rDi.innerText.split("\n")[0] + "\n" + GetAltDistanceString(rDist);
}
window.setTimeout(RulerDistanceAdjust, 100);
}
}
}
function isSaveBtnVisible()
{
if(document.querySelector('#save-button').getAttribute("disabled") === '')
{
if(doCleanUpAfterHNEdit === true)
{
doCleanUpAfterHNEdit = false;
shrinkTopBars();
if(getById('_cbMoveUserInfo').checked === true)
{
insertNodeBeforeNode(document.querySelector('.user-toolbar'),getById('left-app-head'));
insertNodeBeforeNode(document.querySelector('wz-user-box'),getById('left-app-head'));
}
}
}
else
{
window.setTimeout(isSaveBtnVisible, 100);
}
}
function CheckForHNButtons()
{
// The shadowroots for the HN topbar buttons get filled in slightly after the topbar mutation observer fires, which means
// we can't do a simple (because when has dealing with the WME UI design EVER been simple...) call to the code that applies
// the appropriate compression settings to those buttons. Instead we call the function, which will return false so long as
// it saw any empty shadowroot containers - i.e. for any buttons which WME hasn't finished rendering. We use this return
// value to then trigger a short duration timeout before trying to apply the compression again, and so on until it works...
if(ApplyTopBarShadowRootWzButtonCompression() === false)
{
window.setTimeout(CheckForHNButtons, 10);
}
}
function PrepForHNEdit()
{
// The modifications made to the topbar layout by the move user info option don't play entirely nicely with the native
// changes made to the topbar content when entering HN edit mode, so to get around this we first need to briefly
// undo the modifications and then immediately reapply them...
if(getById('_cbMoveUserInfo').checked === true)
{
UnmoveUserInfo();
moveUserInfo();
}
// Regardless of whether or not the user info had been moved, we also then need to do some general cleanup to reinstate
// some of the topbar changes that would otherwise get overriden by the transition out of HN mode
doCleanUpAfterHNEdit = true;
}
function checkForMTEDropDown()
{
let tObj = document.querySelector('#sidepanel-mtes');
let nWO = tObj?.querySelectorAll('wz-option').length;
let nSR = 0;
while(nWO)
{
if(tObj.querySelectorAll('wz-option')[nWO - 1].shadowRoot != null)
{
++nSR;
}
--nWO;
}
if(nSR > 0)
{
restyleDropDownEntries();
}
else
{
window.setTimeout(checkForMTEDropDown, 100);
}
}
function EPObserver()
{
// Just a quick sanity check to make sure the list elements we need are
// all still present...
let acMenu = document.getElementsByClassName('external-provider-edit-form')[0];
if(acMenu == undefined) return;
let acInner = acMenu.getElementsByTagName('wz-autocomplete')[0];
if(acInner == undefined) return;
// Now that we're back to the same position we were at when we originally
// started polling, it's time to mess with whichever list items we can
// find here
let acEntries = acInner.shadowRoot.querySelectorAll('wz-menu-item');
if(acEntries.length != 0)
{
for(let i of acEntries)
{
// To accommodate suggestions that wrap onto 3+ lines, we need to remove the
// height styling from the main menu item plus its child elements, so that each
// entry can expand as needed to keep all of the text visible
SetStyle(i, "--wz-menu-option-height", "auto", false);
let wai = i.querySelector('.wz-autocomplete-item');
SetStyle(wai, "height", "auto", false);
SetStyle(wai, "padding-top", "2px", false);
SetStyle(wai, "padding-bottom", "2px", false);
// Most of the restyling fun takes place on this child element within the
// list element
let acText = i.querySelector('.wz-string-wrapper.primary-text');
// Restore some sanity to the entries, so we can see exactly what they
// say and therefore know what it is we're selecting - UI Design 101...
SetStyle(acText, "overflow", "visible", false);
SetStyle(acText, "overflow", "visible", false);
SetStyle(acText, "lineHeight", "100%", false);
SetStyle(acText, "display", "block", false);
SetStyle(acText, "whiteSpace", "normal", false);
// And just for shits and giggles, the late April WME update now places each and every
// character within an entry in its own childnode of that entry, so that styling can be
// applied to each character individually... As this messes up the entry lineheight
// styling which used to be all that was needed to get things looking good here, we now
// ALSO need to override the native styling on these child nodes. What next devs, an
// individual node for each pixel of each character???
let acChars = acText.childNodes;
for(let j of acChars)
{
SetStyle(j, "lineHeight", "100%", false);
}
// We also need to tweak the shadowroot version of the menu item itself, to
// adjust the height style to prevent the item failing to grow tall enough
// to fully accommodate 3+ line entries
let srMenuItems = i.shadowRoot.querySelector('.wz-menu-item');
SetStyle(srMenuItems, "height", "auto", false);
}
}
// Having messed with whichever elements we found this time around, poll again
// to account for the user continuing to type in the search box causing the
// list to be regenerated with differing entries. As we only get this far into
// the function if the list is still visible, as soon as the user finishes their
// search and the list is removed, the polling will stop until the next time
// the MO fires - part-time polling, I can live with that if the alternative is
// leaving the list entries in a sometimes unuseable state due to the odd
// choice of native styles...
window.setTimeout(EPObserver, 100);
}
function newSaveMode()
{
let sm = W.editingMediator.attributes.saveMode;
if(sm === "IDLE")
{
float();
}
}
function init1()
{
logit("Starting init1","debug");
if(W === undefined)
{
window.setTimeout(init1, 100);
return;
}
logit("Initialising...");
if (W.userscripts?.state?.isReady)
{
init1Finalise();
}
else
{
document.addEventListener("wme-ready", init1Finalise, {once: true});
}
}
function init1Finalise()
{
// insert the content as a tab
addMyTab();
}
function checkSelectableItemInURL()
{
let i = urlSelectableItems.length;
let rType = null;
while(i > 0)
{
--i;
let checkFor = urlSelectableItems[i][0];
if (location.search.match(checkFor))
{
rType = urlSelectableItems[i];
break;
}
}
return rType;
}
function searchSelectableItemsInURL(typeStr)
{
return location.search.match(new RegExp("[?&]" + typeStr + "?=([^&]*)"));
}
function getSelectableItemTotalFromURL()
{
let itemType = checkSelectableItemInURL();
let nItems = 0;
if(itemType !== null)
{
let match = searchSelectableItemsInURL(itemType[0]);
nItems = decodeURIComponent(match[1]).split(',').length;
}
return nItems;
}
function init2()
{
logit("Starting init2","debug");
if(W.userscripts === undefined)
{
//go round again if my tab isn't there yet
if (!getById('sidepanel-FixUI'))
{
logit("Waiting for my tab to appear...","warning");
setTimeout(init2, 200);
return;
}
}
// setup event handlers for my controls:
getById('_cbMoveZoomBar').onclick = createZoomBar;
getById('_cbMoveChatIcon').onclick = moveChatIcon;
getById('_cbHighlightInvisible').onclick = highlightInvisible;
getById('_cbDarkenSaveLayer').onclick = darkenSaveLayer;
getById('_cbSwapRoadsGPS').onclick = swapRoadsGPS;
getById('_cbShowMapBlockers').onclick = showMapBlockers;
getById('_cbShrinkTopBars').onclick = shrinkTopBars;
getById('_cbCompressSidebar').onclick = compressSidebar;
getById('_cbCompressLayersMenu').onclick = compressLayersMenu;
getById('_cbLayersColumns').onclick = compressLayersMenu;
getById('_cbRestyleReports').onclick = restyleReports;
getById('_cbNarrowSidePanel').onclick = narrowSidePanel;
getById('_inpUICompression').onchange = applyEnhancements;
getById('_inpUIContrast').onchange = applyEnhancements;
getById('_cbResizeSearchBox').onclick = resizeSearch;
getById('_cbTameSegmentTypeMenu').onclick = tameSegmentTypeMenu;
getById('_cbTameElevationMenu').onclick = tameSegmentElevationMenu;
getById('_cbHideSegmentPanelLabels').onclick = hideSegmentPanelLabels;
getById('_cbRemoveRoutingReminder').onclick = hideRoutingPreferenceDescription;
getById('_inpASX').onchange = shiftAerials;
getById('_inpASX').onwheel = shiftAerials;
getById('_inpASY').onchange = shiftAerials;
getById('_inpASY').onwheel = shiftAerials;
getById('_inpASO').onchange = shiftAerials;
getById('_inpASO').onwheel = shiftAerials;
getById('_inpASXO').onchange = shiftAerials;
getById('_inpASXO').onwheel = shiftAerials;
getById('_inpASYO').onchange = shiftAerials;
getById('_inpASYO').onwheel = shiftAerials;
getById('_inpASOO').onchange = shiftAerials;
getById('_inpASOO').onwheel = shiftAerials;
getById('_resetAS').onclick = function()
{
getById('_inpASX').value = 0;
getById('_inpASY').value = 0;
getById('_inpASXO').value = 0;
getById('_inpASYO').value = 0;
shiftAerials();
};
getById('_inpGSVContrast').onchange = adjustGSV;
getById('_inpGSVBrightness').onchange = adjustGSV;
getById('_cbGSVInvert').onchange = adjustGSV;
getById('_inpGSVWidth').onchange = GSVWidth;
getById('_cbDisableBridgeButton').onchange = disableBridgeButton;
getById('_cbDisablePathButton').onchange = disablePathButton;
getById('_cbDisableHazardMarkers').onchange = disableHazardMarkers;
getById('_btnKillNode').onclick = killNode;
getById('_cbDisableKinetic').onclick = disableKinetic;
getById('_cbDisableScrollZoom').onclick = disableScrollZoom;
getById('_cbDisableZoomAnimation').onclick = disableAnimatedZoom;
getById('_cbDisableUITransitions').onclick = disableUITransitions;
getById('_cbDisableSaveBlocker').onclick = disableSaveBlocker;
getById('_cbColourBlindTurns').onclick = colourBlindTurns;
getById('_cbUnfloatButtons').onclick = unfloatButtons;
getById('_cbMoveUserInfo').onclick = moveUserInfo;
getById('_cbHackGSVHandle').onclick = hackGSVHandle;
getById('street-view-drag-handle').ondblclick = GSVWidthReset;
getById('_cbEnlargeGeoNodes').onclick = enlargeGeoNodes;
getById('_inpEnlargeGeoNodes').onchange = enlargeGeoNodes;
getById('_cbEnlargeGeoHandlesFU').onclick = enlargeGeoHandles;
getById('_inpEnlargeGeoHandles').onchange = enlargeGeoHandles;
getById('_cbEnlargePointMCs').onclick = enlargePointMCs;
getById('_inpEnlargePointMCs').onchange = enlargePointMCs;
//REGISTER WAZE EVENT HOOKS
// events for Aerial Shifter
W.map.events.register("zoomend", null, shiftAerials);
W.map.events.register("moveend", null, shiftAerials);
W.map.getLayerByUniqueName('satellite_imagery').events.register("loadend", null, shiftAerials);
// events to change menu bar color based on map comments checkbox
W.map.events.register("zoomend", null, warnCommentsOff);
W.map.events.register("moveend", null, warnCommentsOff);
// event to remove the overlay that blocks the sidebar UI if you zoom out too far
W.map.events.register("zoomend", null, unblockSidePanel);
// events to adjust the "Search this area" z-index so it gets rendered behind the drop-down menus
W.map.events.register("zoomend", null, moveSearchThisArea);
W.map.events.register("moveend", null, moveSearchThisArea);
// event to re-hack my zoom bar if it's there
W.map.getLayerByUniqueName("BASE_LAYER").events.register("loadend", null, ZLI);
//window resize event to resize layers menu
window.addEventListener('resize', compressLayersMenu, true);
//window resize event to reapply zoombar fix
window.addEventListener('resize', ZLI, true);
//window resize event to resize search box
window.addEventListener('resize', resizeSearch, true);
//anything we might need to do when the mouse moves...
W.map.events.register("mousemove", null, mouseMove);
let tEvt = Object.assign({}, W.editingMediator._events['change:saveMode'][0]);
tEvt.callback = newSaveMode;
W.editingMediator._events['change:saveMode'].push(tEvt);
// event handlers to help with the weird change log visibility problem...
document.querySelector('#save-button').addEventListener('mouseover', saveMouseOver, true);
document.querySelector('#save-button').addEventListener('mouseout', saveMouseOut, true);
//window resize event to refloat the sharing box in the correct location
window.addEventListener('resize', unfloat, true);
//event to catch entering and exiting HN edit mode
W.editingMediator.on('change:editingHouseNumbers', function()
{
if(W.editingMediator.attributes.editingHouseNumbers === true)
{
PrepForHNEdit();
}
/*
//// unfloat
if (getById('_cbUnfloatButtons').checked)
{
if (W.editingMediator.attributes.editingHouseNumbers) unfloat();
if (W.editingMediator.attributes.editingEnabled) unfloat();
}
*/
});
//create Aerial Shifter warning div
var ASwarning = document.createElement('div');
ASwarning.id = "WMEFU_AS";
ASwarning.innerHTML = modifyHTML("Aerials Shifted");
ASwarning.setAttribute('style','top:20px; left:0px; width:100%; position:absolute; z-index:10000; font-size:100px; font-weight:900; color:rgba(255, 255, 0, 0.4); text-align:center; pointer-events:none; display:none;');
getById("WazeMap").appendChild(ASwarning);
loadSettings();
//create Permalink Count div
var WMEPC_div = document.createElement('div');
var WMEPC_div_sub = document.createElement('div');
WMEPC_div.id = "WMEFUPC";
WMEPC_div.classList.add("toolbar-button","toolbar-button-with-icon");
WMEPC_div.title = "Number of selectable map objects in the URL\nClick to reselect them.";
WMEPC_div.style.cursor="pointer";
WMEPC_div_sub.classList.add("item-container","WMEFU-toolbar-button");
WMEPC_div_sub.innerHTML = modifyHTML('<span class="item-icon" style="display:inline-flex"><i style="margin-top:8px" class="fa fa-link WMEFUPCicon"></i> ' + getSelectableItemTotalFromURL() +'</span>');
WMEPC_div.appendChild(WMEPC_div_sub);
insertNodeBeforeNode(WMEPC_div,getById('search'));
WMEPC_div.onclick = PCclicked;
//Create Turn Popup Blocker div
var WMETPB_div = document.createElement('div');
var WMETPB_div_sub = document.createElement('div');
WMETPB_div.id = "WMEFUTPB";
WMETPB_div.classList.add("toolbar-button","toolbar-button-with-icon");
WMETPB_div.title = "Disable/enable the turn arrow popup dialogue";
WMETPB_div_sub.classList.add('item-container',"WMEFU-toolbar-button");
WMETPB_div_sub.innerHTML = modifyHTML('<span class="item-icon fa-stack fa-2x" style="display:inline-flex; font-size:10px !important"><i class="fa fa-comment fa-stack-2x"></i><i class="fa fa-arrow-up fa-inverse fa-stack-1x"></i></span>');
WMETPB_div.appendChild(WMETPB_div_sub);
insertNodeBeforeNode(WMETPB_div,getById('search'));
WMETPB_div.onclick = toggleKillTurnPopup;
addGlobalStyle('.WMEFU-toolbar-button { padding: 0px !important; }');
// overload the window unload function to save my settings
window.addEventListener("beforeunload", saveSettings, false);
// Alert to new version
if (oldVersion != fumeVersion)
{
let releaseNotes = "Version " + fumeVersion + " ("+fumeDate+") release notes<br><br>";
releaseNotes += "<ul>";
for(let i = 0; i < newVersionNotes.length; ++i)
{
releaseNotes += "<li>" + newVersionNotes[i];
}
releaseNotes += "</ul>";
abShowAlertBox('fa-info-circle', 'WME Fix UI Memorial Edition', releaseNotes, false, "OK", "", null, null);
saveSettings();
}
// fix for sidebar display problem in Safari, requested by edsonajj
var isSafari = /^((?!chrome|android).)*safari/i.test(navigator.userAgent);
if (isSafari)
{
addGlobalStyle('.flex-parent { height: 99% !important; }');
}
// stop wobbling status bar
addGlobalStyle('.WazeControlMousePosition { font-family: monospace }');
// move closed node icon below node markers
// apply the settings
shiftAerials();
// Adjust the date string format used in the drives tab...
I18n.translations[I18n.currentLocale()].date.formats.long_with_time = "%a %Y-%m-%d %H:%M";
// ...and similarly in the edit history entries
I18n.translations[I18n.currentLocale()].date.formats.long = "%a %Y-%m-%d, %H:%M";
window.setTimeout(applyAllSettings, 1000);
logit("Initialisation complete");
}
let wasDrawing = null;
function mouseMove()
{
// Temporarily disable the Enlarge geo/junction nodes and Enlarge geo handles options
// when drawing new geometry, to avoid the enlarged nodes/handles on other geometry
// objects getting in the way of the new object being drawn...
let isDrawing = W.editingMediator.attributes.drawing;
if(isDrawing != wasDrawing)
{
enlargeGeoNodes(isDrawing);
enlargeGeoHandles(isDrawing);
wasDrawing = isDrawing;
}
}
function unblockSidePanel()
{
// I can see why the devs thought blocking access to the sidepanel UI might make sense at wider
// zoom levels, given that you can't select any map elements or therefore do any editing.
// The problem is that this blocking applies to ALL of the sidepanel tabs, including those which
// COULD be useful regardless of the zoom level - drives, edit areas and userscripts...
//
// This one-liner simply kills the blocking overlay if it's present, restoring access to the
// sidebar UI at all zoom levels.
document.querySelector(".overlay.editingDisabled")?.remove();
}
function createTabHTML()
{
let innerHTML = "";
innerHTML += '<b title="Shift aerial images layer to match GPS tracks and reduce image opacity">Aerial Shifter</b> ';
innerHTML += '<span class="fa fa-power-off" id="_resetAS" title="Clear X/Y offsets"></span><br>';
innerHTML += '<div>' + I18n.lookup('layer_switcher.togglers.ITEM_SATELLITE_IMAGERY') + '</div>';
innerHTML += '<div style="display:inline-block; padding-left:10px;"><input type="number" id="_inpASX" title="horizontal shift" max=300 min=-300 step=5 style="height:20px; width:50px;text-align:right;"/><b>m</b><span class="fa fa-arrow-right"></span></div>';
innerHTML += '<div id="as2" style="display:inline-block;padding:0 5px;"><input type="number" id="_inpASY" title="vertical shift" max=300 min=-300 step=5 style="height:20px; width:50px;text-align:right;"/><b>m</b><span class="fa fa-arrow-up"></span></div>';
innerHTML += '<div id="as3" style="display:inline-block"><input type="number" id="_inpASO" title="opacity" max=100 min=10 step=10 style="height:20px; width:50px;text-align:right;"/><b>%</b><span class="fa fa-adjust"></span></div>';
innerHTML += '<div>' + I18n.lookup('layer_switcher.togglers.GROUP_IMAGERY') + '</div>';
innerHTML += '<div style="display:inline-block; padding-left:10px;"><input type="number" id="_inpASXO" title="horizontal shift" max=300 min=-300 step=5 style="height:20px; width:50px;text-align:right;"/><b>m</b><span class="fa fa-arrow-right"></span></div>';
innerHTML += '<div id="as4" style="display:inline-block;padding:0 5px;"><input type="number" id="_inpASYO" title="vertical shift" max=300 min=-300 step=5 style="height:20px; width:50px;text-align:right;"/><b>m</b><span class="fa fa-arrow-up"></span></div>';
innerHTML += '<div id="as5" style="display:inline-block"><input type="number" id="_inpASOO" title="opacity" max=100 min=10 step=10 style="height:20px; width:50px;text-align:right;"/><b>%</b><span class="fa fa-adjust"></span></div>';
innerHTML += '<br>';
innerHTML += '<br>';
innerHTML += '<b title="Adjust contrast, brightness, colours & width for Google Street View images">GSV image adjust</b><br>';
innerHTML += '<span title="Contrast"><input type="number" id="_inpGSVContrast" max=200 min=25 step=25 style="height:20px; width:46px;text-align:right;"/><b>%</b><span class="fa fa-adjust"></span></span> ';
innerHTML += '<span title="Brightness"><input type="number" id="_inpGSVBrightness" max=200 min=25 step=25 style="height:20px; width:46px;text-align:right;"/><b>%</b><span class="fa fa-sun-o"></span></span> ';
innerHTML += '<span title="Invert colours"><input type="checkbox" id="_cbGSVInvert"/><span class="fa fa-tint"></span></span> ';
innerHTML += '<span title="Default width"><input type="number" id="_inpGSVWidth" max=90 min=10 step=10 style="height:20px; width:46px;text-align:right;"/><b>%</b><span class="fa fa-arrows-h"></span></span> ';
innerHTML += '<br>';
innerHTML += '<br>';
innerHTML += '<b>UI Enhancements</b><br>';
innerHTML += '<input type="checkbox" id="_cbShrinkTopBars" /> ' +
'<span title="Because we can\'t afford to waste screen space, particularly on\nstuff we didn\'t ask for and don\'t want, like the black bar.\nAnd why does the reload button have a re-do icon?!">Compress/enhance bars above the map</span><br>';
innerHTML += '<input type="checkbox" id="_cbCompressSidebar" /> ' +
'<span title="Because I\'m sick of having to scroll the side panel because of oversized fonts and wasted space">Compress/enhance side panel contents</span><br>';
innerHTML += '<input type="checkbox" id="_cbCompressLayersMenu" /> ' +
'<span title="Because it\'s already too big for small screens and Waze only plan to make it bigger">Compress/enhance layers menu</span><br>';
innerHTML += '<span id="layersColControls"><input type="checkbox" id="_cbLayersColumns" /> ' +
'<span title="Widen the layers menu to 2 columns - particulary for netbook users\nWon\'t work without some compression turned on">Two-column layers menu</span><br></span>';
innerHTML += '<input type="checkbox" id="_cbRestyleReports" /> ' +
'<span title="Another UI element configured for developers with massive screens instead of normal users">Compress/enhance report panels (UR/MP)</span><br>';
innerHTML += '<input type="checkbox" id="_cbNarrowSidePanel" /> ' +
'<span title="If you have a netbook, Waze isn\'t interested in your experience.\nYou need every bit of map space you can get - so have a present from me!">Reduce width of the side panel</span><span title="This will definitely interfere with scripts that rely on a fixed width for their tab contents." style="font-size: 16px; color: red;">⚠</span><br>';
innerHTML += '<br>';
innerHTML += '<b title="Control the amount of compression/enhancment">UI Enhancement controls</b><br>';
innerHTML += '<div style="display:inline-block"><select id="_inpUICompression" title="Compression enhancement" style="height:20px; padding:0px; border-radius=0px;"><option value="2">High</option><option value="1">Low</option><option value="0">None</option></select><span class="fa fa-compress"></span></div> ';
innerHTML += '<div style="display:inline-block"><select id="_inpUIContrast" title="Contrast enhancement" style="height:20px; padding:0px; border-radius=0px;"><option value="2">High</option><option value="1">Low</option><option value="0">None</option></select><span class="fa fa-adjust"></span></div>';
innerHTML += '<br>';
innerHTML += '<button id="_btnKillNode" style = "height: 18px; margin-top: 5px;" title="Hide the junction nodes layer to allow access to Map Comments hidden under nodes.\nThis stays in effect until the page is zoomed/panned/refreshed.">Hide junction nodes</button> ';
innerHTML += '<br><br>';
innerHTML += '<b>UI Fixes/changes</b><br>';
innerHTML += '<input type="checkbox" id="_cbHideSegmentPanelLabels" /> ' +
'<span title="Hide the labels in the segment sidepanel,\nbecause there are more important things to display in that precious space.">Hide segment sidepanel labels</span><br>';
innerHTML += '<input type="checkbox" id="_cbTameSegmentTypeMenu" /> ' +
'<span title="Do away with all the wasted space in the segment type menu,\nso that we can select types without having to scroll.">Tame segment type menu</span><br>';
innerHTML += '<input type="checkbox" id="_cbTameElevationMenu" /> ' +
'<span title="Do away with all the wasted space and unlikely to ever be used option in the elevation menu,\nso that we can select the ones we DO use without having to scroll.">Tame elevation menu</span><br>';
innerHTML += '<input type="checkbox" id="_cbRemoveRoutingReminder" /> ' +
'<span title="Remove the \'Segment will be used as\' message under the Routing buttons.">Remove segment routing message</span><br>';
innerHTML += '<input type="checkbox" id="_cbReEnableSidePanel" /> ' +
'<span title="Re-enable the side panel at wider zoom levels,\nbecause contrary to what the WME devs seem to think,\nthere is quite a lot you can still do there.">Re-enable side panel at wider zooms</span><br>';
innerHTML += '<input type="checkbox" id="_cbResizeSearchBox" /> ' +
'<span title="Allows the search box to use all the dead space in the top bar">Expand search box</span><br>';
innerHTML += '<input type="checkbox" id="_cbMoveZoomBar" /> ' +
'<span title="Because nobody likes a pointless UI change that breaks your workflow,\nimposed by idiots who rarely use the editor and don\'t listen to feedback.\nNO MATTER HOW HARD THEY TRY, I WILL BRING IT BACK!">Re-create zoom bar & move map controls</span><br>';
innerHTML += '<input type="checkbox" id="_cbFixExternalProviders" /> ' +
'<span title="The External Providers interface has a description box that will only show one line of text.\nThis fixes that.">Expand External Provider details for places</span><br>';
innerHTML += '<input type="checkbox" id="_cbMoveChatIcon" /> ' +
'<span title="Here\'s a truly outstanding example of making a stupid change to the UI in order to\ndeal with another stupid change to the UI!\nBecause HQ couldn\'t make the new layers menu auto-hide, they moved the chat icon.\nTick this box to put it back where it belongs.">Move Chat icon back to right</span><br>';
innerHTML += '<input type="checkbox" id="_cbHighlightInvisible" /> ' +
'<span title="Typical WME design - the chat icon changes when you\'re invisible,\nbut the change is practically invisible!\nThis option provides a more obvious highlight.">Highlight invisible mode</span></span><br>';
innerHTML += '<input type="checkbox" id="_cbDarkenSaveLayer" /> ' +
'<span title="It\'s not bad enough they\'ve removed all the contrast to give you eyestrain,\nbut then they blind you every time you save. ">Darken screen overlay when saving</span><br>';
innerHTML += '<input type="checkbox" id="_cbSwapRoadsGPS" /> ' +
'<span title="Guess what? Waze thinks the GPS layer should now be over the segments layer.\nWhy should you have any choice about that?">Move GPS layer below segments layer</span><br>';
innerHTML += '<input type="checkbox" id="_cbShowMapBlockers" /> ' +
'<span title="Some WME elements block access to the map layers. These problems have been reported as bugs.\nUntil they\'re fixed, this functions makes them visible.">Show map-blocking WME bugs</span></span><br>';
innerHTML += '<input type="checkbox" id="_cbDisableBridgeButton" /> ' +
'<span title="The Bridge button is rarely useful, but often used incorrectly.\nIt\'s best to keep it disabled unless you need it.">Disable Bridge button</span><br>';
innerHTML += '<input type="checkbox" id="_cbDisablePathButton" /> ' +
'<span title="The far turn button seems to be an accidental click-magnet, making it all\ntoo easy to accidentally set a path without noticing until after you save...\nUse this option to disable it and avoid the embarrassment">Disable Far Turn button</span><br>';
innerHTML += '<input type="checkbox" id="_cbDisableHazardMarkers" /> ' +
'<span title="The new permanent hazard markers can get a bit in the way, and whilst \nWME allows you to turn them off, it would also be nice to simply dim/deactivate them as per turn arrows.\nThis option lets you do just that...">Disable Hazard Markers</span><br>';
innerHTML += '<input type="checkbox" id="_cbMondayFirst" /> ' +
'<span title="Requests to have calendar items localised with Monday as the first day of the week\ngo back a while. Now you don\'t have to wait for Waze.">Start calendars on Monday</span><br>';
innerHTML += '<input type="checkbox" id="_cbISODates" /> ' +
'<span title="Dates in the Restrictions dialogues are all in American format - MM/DD/YY\nFine if you\' American, confusing as hell for the rest of us!\nThis changes the dates to ISO format, matching the Closures dialogue">ISO dates in Restrictions</span><br>';
innerHTML += '<input type="checkbox" id="_cbDisableKinetic" /> ' +
'<span title="Kinetic panning is a new WME feature: if you release the mouse whilst dragging the map,\nthe map will keep moving. It can be very useful for panning large distances.\nIt can also be very annoying. Now YOU have control.">Disable Kinetic Panning</span><br>';
innerHTML += '<input type="checkbox" id="_cbDisableZoomAnimation" /> ' +
'<span title="Animated zooming is a new WME feature which some would prefer not to have enabled. Click here to express that preference...">Disable Animated Zooming</span><br>';
innerHTML += '<input type="checkbox" id="_cbDisableUITransitions" /> ' +
'<span title="Because life is simply too short to waste time waiting for UI elements to oooooooooze into position">Disable UI Transitions</span><br>';
innerHTML += '<input type="checkbox" id="_cbDisableScrollZoom" /> ' +
'<span title="Zooming with the scroll wheel can be problematic when using an Apple Magic Mouse, which\nscrolls on touch. This will disable scroll-to-zoom.">Disable scroll-to-zoom</span><br>';
innerHTML += '<input type="checkbox" id="_cbDisableSaveBlocker" /> ' +
'<span title="When you hit Save, WME places a blocking element over the map until the save is complete\nThis disables that element, allowing you to pan the map and use GSV whilst a slow save is in progress.">Disable map blocking during save</span><br>';
innerHTML += '<input type="checkbox" id="_cbColourBlindTurns" /> ' +
'<span title="Change green turn arrows blue in order to make them more visible\nfor users with the most common type of colour blindness.">Change green turn arrows to blue</span><br>';
innerHTML += '<input type="checkbox" id="_cbUnfloatButtons" /> ' +
'<span title="Move Layers/Refresh buttons back into the toolbar and Share button into the\nfooter.\nWaze put little enough effort into giving us enough map area to work with,\nand then they drop little button turds all over it!">Remove floating buttons from map area</span><br>';
innerHTML += '<input type="checkbox" id="_cbMoveUserInfo" /> ' +
'<span title="The new user info button is very useful, but it\'s not a map editing control,\nso it shouldn\'t be in the toolbar. The same goes for the notification button.\nThis function moves them both to a sensible location.">Move user info/notification buttons</span><br>';
innerHTML += '<input type="checkbox" id="_cbHackGSVHandle" /> ' +
'<span title="Whilst being able to adjust the GSV width is useful, the drag handle\ninvisibly covers 30 pixels of map and is very easy to drag accidentally.\nThis function transforms it to a button drag control that\'s much less\nlikely to be used by accident.">Minimise GSV drag handle</span><br>';
innerHTML += '<input type="checkbox" id="_cbEnlargeGeoNodes" /> ' +
'<span title="If you\'re getting old, like me, grabbing those little circles is a pain!\nThis control lets you enlarge the geo nodes (and junction nodes for segments),\nwhich define the shapes of segments and place boundaries.">Enlarge geo/junction nodes</span><div style="display:inline-block"> <input type="number" id="_inpEnlargeGeoNodes" title="radius (default=6)" max=12 min=8 step=2 style="height:16px; padding:0 0 0 2px;; border:1px solid; width:37px;"/></div><br>';
innerHTML += '<input type="checkbox" id="_cbEnlargeGeoHandlesFU" /> ' +
'<span title="If you\'re getting old, like me, grabbing those little circles is a pain!\nThis control lets you enlarge the geo handles, used to add geo nodes to segments and place boundaries.">Enlarge geo handles</span><div style="display:inline-block"> <input type="number" id="_inpEnlargeGeoHandles" title="radius (default=4)" max=10 min=6 step=2 style="height:16px; padding:0 0 0 2px;; border:1px solid; width:37px;"/></div><br>';
innerHTML += '<input type="checkbox" id="_cbEnlargePointMCs" /> ' +
'<span title="This control lets you enlarge point map comments, because sometimes they can look a little swamped inamongst the rest of the stuff on show">Enlarge point map comments</span><div style="display:inline-block"> <input type="number" id="_inpEnlargePointMCs" title="scale (default=1)" max=3 min=1 step=0.1 style="height:16px; padding:0 0 0 2px;; border:1px solid; width:37px;"/></div><br>';
innerHTML += '<br>';
innerHTML += '<b><a href="https://www.waze.com/forum/viewtopic.php?t=334618" title="Forum topic" target="_blank"><u>' +
'WME Fix UI Memorial Edition</u></a></b> v' + fumeVersion;
return innerHTML;
}
function addMyTab()
{
logit("Creating tab...");
tabAttempts = 0;
tabCreate();
}
async function tabCreate()
{
let {tabLabel, tabPane} = W.userscripts.registerSidebarTab("FUME");
tabLabel.innerText = "FUME";
tabPane.innerHTML = modifyHTML(createTabHTML());
await W.userscripts.waitForElementConnected(tabPane);
logit("Tab now available...");
createDSASection();
abInitialiseAlertBox();
document.getElementById('abAlertTickBtn').addEventListener('click', abCloseAlertBoxWithTick, true);
document.getElementById('abAlertCrossBtn').addEventListener('click', abCloseAlertBoxWithCross, true);
//pass control to init2
init2();
}
function loadSettings()
{
// Remove old V1 settings if they're still hanging around
if (localStorage.WMEFixUI)
{
localStorage.removeItem("WMEFixUI");
}
var options;
if (localStorage.WMEFUSettings)
{
options = JSON.parse(localStorage.WMEFUSettings);
}
else
{
options = {};
}
oldVersion = (options.oldVersion !== undefined ? options.oldVersion : "0.0");
getById('_cbMoveZoomBar').checked = (options.moveZoomBar !== undefined ? options.moveZoomBar : true);
getById('_cbShrinkTopBars').checked = (options.shrinkTopBars !== undefined ? options.shrinkTopBars : true);
getById('_cbCompressSidebar').checked = ( options.restyleSidePanel !== undefined ? options.restyleSidePanel : true);
getById('_cbRestyleReports').checked = ( options.restyleReports !== undefined ? options.restyleReports : true);
getById('_cbNarrowSidePanel').checked = ( options.narrowSidePanel !== undefined ? options.narrowSidePanel : false);
getById('_inpASX').value = ( options.aerialShiftX !== undefined ? options.aerialShiftX : 0);
getById('_inpASY').value = ( options.aerialShiftY !== undefined ? options.aerialShiftY : 0);
getById('_inpASO').value = ( options.aerialOpacity !== undefined ? options.aerialOpacity : 100);
getById('_inpASXO').value = ( options.aerialShiftXO !== undefined ? options.aerialShiftXO : 0);
getById('_inpASYO').value = ( options.aerialShiftYO !== undefined ? options.aerialShiftYO : 0);
getById('_inpASOO').value = ( options.aerialOpacityO !== undefined ? options.aerialOpacityO : 100);
getById('_cbFixExternalProviders').checked = ( options.fixExternalProviders !== undefined ? options.fixExternalProviders : true);
getById('_inpGSVContrast').value = ( options.GSVContrast !== undefined ? options.GSVContrast : 100);
getById('_inpGSVBrightness').value = ( options.GSVBrightness !== undefined ? options.GSVBrightness : 100);
getById('_cbGSVInvert').checked = ( options.GSVInvert !== undefined ? options.GSVInvert : false);
getById('_inpGSVWidth').value = ( options.GSVWidth !== undefined ? options.GSVWidth : 50);
getById('_cbCompressLayersMenu').checked = ( options.restyleLayersMenu !== undefined ? options.restyleLayersMenu : true);
getById('_cbLayersColumns').checked = ( options.layers2Cols !== undefined ? options.layers2Cols : false);
getById('_cbMoveChatIcon').checked = ( options.moveChatIcon !== undefined ? options.moveChatIcon : true);
getById('_cbHighlightInvisible').checked = ( options.highlightInvisible !== undefined ? options.highlightInvisible : true);
getById('_cbDarkenSaveLayer').checked = ( options.darkenSaveLayer !== undefined ? options.darkenSaveLayer : true);
getById('_inpUIContrast').value = ( options.UIContrast !== undefined ? options.UIContrast : 1);
getById('_inpUICompression').value = ( options.UICompression !== undefined ? options.UICompression : 1);
getById('_cbSwapRoadsGPS').checked = ( options.swapRoadsGPS !== undefined ? options.swapRoadsGPS : true);
getById('_cbShowMapBlockers').checked = ( options.showMapBlockers !== undefined ? options.showMapBlockers : true);
getById('_cbHideSegmentPanelLabels').checked = ( options.hideSegmentPanelLabels !== undefined ? options.hideSegmentPanelLabels : false);
getById('_cbTameSegmentTypeMenu').checked = (options.tameSegmentMenu !== undefined ? options.tameSegmentMenu : false);
getById('_cbTameElevationMenu').checked = (options.tameElevationMenu !== undefined ? options.tameElevationMenu : false);
getById('_cbRemoveRoutingReminder').checked = (options.removeRoutingReminder !== undefined ? options.removeRoutingReminder : false);
getById('_cbReEnableSidePanel').checked = (options.reEnableSidePanel !== undefined ? options.reEnableSidePanel : false);
getById('_cbDisableBridgeButton').checked = ( options.disableBridgeButton !== undefined ? options.disableBridgeButton : true);
getById('_cbDisablePathButton').checked = ( options.disablePathButton !== undefined ? options.disablePathButton : false);
getById('_cbDisableHazardMarkers').checked = ( options.disableHazardMarkers !== undefined ? options.disableHazardMarkers : false);
getById('_cbISODates').checked = ( options.ISODates !== undefined ? options.ISODates : true);
getById('_cbMondayFirst').checked = ( options.mondayFirst !== undefined ? options.mondayFirst : false);
getById('_cbDisableKinetic').checked = ( options.disableKinetic !== undefined ? options.disableKinetic : false);
getById('_cbDisableScrollZoom').checked = ( options.disableScrollZoom !== undefined ? options.disableScrollZoom : false);
getById('_cbDisableZoomAnimation').checked = ( options.disableAnimatedZoom !== undefined ? options.disableAnimatedZoom : false);
getById('_cbDisableUITransitions').checked = ( options.disableUITransitions !== undefined ? options.disableUITransitions : false);
getById('_cbDisableSaveBlocker').checked = ( options.disableSaveBlocker !== undefined ? options.disableSaveBlocker : false);
getById('_cbColourBlindTurns').checked = ( options.colourBlindTurns !== undefined ? options.colourBlindTurns : false);
getById('_cbUnfloatButtons').checked = ( options.unfloatButtons !== undefined ? options.unfloatButtons : false);
getById('_cbMoveUserInfo').checked = ( options.moveUserInfo !== undefined ? options.moveUserInfo : false);
getById('_cbHackGSVHandle').checked = ( options.hackGSVHandle !== undefined ? options.hackGSVHandle : false);
getById('_cbEnlargeGeoNodes').checked = ( options.enlargeGeoNodes !== undefined ? options.enlargeGeoNodes : false);
getById('_inpEnlargeGeoNodes').value = ( options.geoNodeSize !== undefined ? options.geoNodeSize : 8);
getById('_cbEnlargeGeoHandlesFU').checked = ( options.enlargeGeoHandles !== undefined ? options.enlargeGeoHandles : false);
getById('_inpEnlargeGeoHandles').value = ( options.geoHandleSize !== undefined ? options.geoHandleSize : 6);
getById('_cbEnlargePointMCs').checked = ( options.enlargePointMCs !== undefined ? options.enlargePointMCs : false);
getById('_inpEnlargePointMCs').value = ( options.pointMCScale !== undefined ? options.pointMCScale : 1);
getById('_cbResizeSearchBox').checked = ( options.resizeSearchBox !== undefined ? options.resizeSearchBox : false);
}
function saveSettings()
{
if (localStorage)
{
logit("saving options to local storage");
var options = {};
options.oldVersion = fumeVersion;
options.moveZoomBar = getById('_cbMoveZoomBar').checked;
options.shrinkTopBars = getById('_cbShrinkTopBars').checked;
options.restyleSidePanel = getById('_cbCompressSidebar').checked;
options.restyleReports = getById('_cbRestyleReports').checked;
options.narrowSidePanel = getById('_cbNarrowSidePanel').checked;
options.aerialShiftX = getById('_inpASX').value;
options.aerialShiftY = getById('_inpASY').value;
options.aerialOpacity = getById('_inpASO').value;
options.aerialShiftXO = getById('_inpASXO').value;
options.aerialShiftYO = getById('_inpASYO').value;
options.aerialOpacityO = getById('_inpASOO').value;
options.fixExternalProviders = getById('_cbFixExternalProviders').checked;
options.GSVContrast = getById('_inpGSVContrast').value;
options.GSVBrightness = getById('_inpGSVBrightness').value;
options.GSVInvert = getById('_cbGSVInvert').checked;
options.GSVWidth = getById('_inpGSVWidth').value;
options.restyleLayersMenu = getById('_cbCompressLayersMenu').checked;
options.layers2Cols = getById('_cbLayersColumns').checked;
options.moveChatIcon = getById('_cbMoveChatIcon').checked;
options.highlightInvisible = getById('_cbHighlightInvisible').checked;
options.darkenSaveLayer = getById('_cbDarkenSaveLayer').checked;
options.UIContrast = getById('_inpUIContrast').value;
options.UICompression = getById('_inpUICompression').value;
options.swapRoadsGPS = getById('_cbSwapRoadsGPS').checked;
options.showMapBlockers = getById('_cbShowMapBlockers').checked;
options.hideSegmentPanelLabels = getById('_cbHideSegmentPanelLabels').checked;
options.tameSegmentMenu = getById('_cbTameSegmentTypeMenu').checked;
options.tameElevationMenu = getById('_cbTameElevationMenu').checked;
options.removeRoutingReminder = getById('_cbRemoveRoutingReminder').checked;
options.reEnableSidePanel = getById('_cbReEnableSidePanel').checked;
options.disableBridgeButton = getById('_cbDisableBridgeButton').checked;
options.disablePathButton = getById('_cbDisablePathButton').checked;
options.disableHazardMarkers = getById('_cbDisableHazardMarkers').checked;
options.ISODates = getById('_cbISODates').checked;
options.mondayFirst = getById('_cbMondayFirst').checked;
options.disableKinetic = getById('_cbDisableKinetic').checked;
options.disableScrollZoom = getById('_cbDisableScrollZoom').checked;
options.disableAnimatedZoom = getById('_cbDisableZoomAnimation').checked;
options.disableUITransitions = getById('_cbDisableUITransitions').checked;
options.disableSaveBlocker = getById('_cbDisableSaveBlocker').checked;
options.colourBlindTurns = getById('_cbColourBlindTurns').checked;
options.unfloatButtons = getById('_cbUnfloatButtons').checked;
options.moveUserInfo = getById('_cbMoveUserInfo').checked;
options.hackGSVHandle = getById('_cbHackGSVHandle').checked;
options.enlargeGeoNodes = getById('_cbEnlargeGeoNodes').checked;
options.geoNodeSize = getById('_inpEnlargeGeoNodes').value;
options.enlargeGeoHandles = getById('_cbEnlargeGeoHandlesFU').checked;
options.geoHandleSize = getById('_inpEnlargeGeoHandles').value;
options.enlargePointMCs = getById('_cbEnlargePointMCs').checked;
options.pointMCScale = getById('_inpEnlargePointMCs').value;
options.resizeSearchBox = getById('_cbResizeSearchBox').checked;
localStorage.WMEFUSettings = JSON.stringify(options);
}
}
function applyAllSettings()
{
kineticDragParams = W.map.controls.find(control => control.dragPan).dragPan.kinetic;
logit("Applying settings...");
unfloatButtons();
shrinkTopBars();
compressSidebar();
restyleReports();
narrowSidePanel();
warnCommentsOff();
adjustGSV();
GSVWidth();
compressLayersMenu();
moveChatIcon();
highlightInvisible();
darkenSaveLayer();
swapRoadsGPS();
showMapBlockers();
disableBridgeButton();
disablePathButton();
disableHazardMarkers();
disableKinetic();
disableScrollZoom();
disableAnimatedZoom();
disableSaveBlocker();
disableUITransitions();
colourBlindTurns();
createZoomBar();
moveUserInfo();
moveSearchThisArea();
hackGSVHandle();
enlargeGeoNodes(false);
enlargeGeoHandles(false);
enlargePointMCs();
RTCArrowsFix();
hideUnuseableStuff();
resizeSearch();
RestrictionObserver.observe(getById('dialog-container'), { childList: true, subtree: true });
ClosureObserver.observe(getById('edit-panel'), { childList: true, subtree: true });
SegmentObserver.observe(getById('edit-panel .contents'), { childList: true, subtree: true });
PlaceObserver.observe(getById('edit-panel'), { childList: true, subtree: true });
MTEObserver.observe(getById('sidepanel-mtes'), { childList: true, subtree: true });
RTCMarkerObserver.observe(W.map.getLayerByName("closures").div, { childList: true, subtree: true });
OBObserver.observe(getById('overlay-buttons'), { childList: true, subtree: true });
ITObserver.observe(getById('issue-tracker-filter-region'), { childList: true, subtree: true});
PriToolBarObserver.observe(document.querySelector('[class^="primaryToolbar"]'), { childList: true, subtree: true });
RulerShownObserver.observe(getById('overlay-buttons-region'), { childList: true, subtree: true});
document.body.onchange = checkForTippy;
getById('sidepanel-issue-tracker').onchange = checkForIssuesFilter;
wmeFUinitialising = false;
saveSettings();
}
function hasExactlyOneSelectedSegment()
{
let retval = false;
let sf = W.selectionManager.getSelectedDataModelObjects();
if(sf.length === 1)
{
retval = (sf[0].type === "segment");
}
return retval;
}
function checkForTippy()
{
// To handle the "tippy" classed dynamic popup used to show TIO details, we first call this function from the
// body onchange event - the sole purpose of which is to call checkForTippy2() so long as exactly one segment
// is selected. If this latter isn't true, then no turn arrows will be visible, thus the TIO popup can't be
// shown...
if(hasExactlyOneSelectedSegment() === true)
{
let nOpen = document.querySelectorAll('.turn-arrow-state-open').length;
let nClosed = document.querySelectorAll('.turn-arrow-state-closed').length;
if((nOpen + nClosed) === 0)
{
// WME is now sometimes a bit slow to render the arrows...
window.setTimeout(checkForTippy, 100);
}
else
{
while(nOpen)
{
--nOpen;
document.querySelectorAll('.turn-arrow-state-open')[nOpen].addEventListener('mouseenter', checkForTippy1a);
document.querySelectorAll('.turn-arrow-state-open')[nOpen].addEventListener('mouseleave', cancelTippyCheck);
}
while(nClosed)
{
--nClosed;
document.querySelectorAll('.turn-arrow-state-closed')[nClosed].addEventListener('mouseenter', checkForTippy1a);
document.querySelectorAll('.turn-arrow-state-closed')[nClosed].addEventListener('mouseleave', cancelTippyCheck);
}
}
}
}
let tippyIsActive = false;
function checkForTippy1a()
{
let tObj = document.querySelector('.tippy-box');
if(tObj === null)
{
// Wait for the turn popup to appear...
window.setTimeout(checkForTippy1a, 100);
return;
}
// Hang the same mouseenter/leave listeners off the popup as well, so we can also maintain our tweaks if we
// do anything within the popup which causes it to be re-rendered after having moused-off the turn arrow itself
tObj.addEventListener('mouseenter', maintainTippyCheck);
tObj.addEventListener('mouseleave', cancelTippyCheck);
// Now try to apply our tweaks...
tippyIsActive = true;
checkForTippy2();
}
function maintainTippyCheck()
{
tippyIsActive = true;
}
function cancelTippyCheck()
{
tippyIsActive = false;
}
function checkForTippy2()
{
// Once our event listeners have decided we're no longer interested in the popup, exit immediately
// so that we don't re-apply any of the setTimeout() calls
if(tippyIsActive === false)
{
return;
}
// First double-check that the popup IS still visible...
let tObj = document.querySelector('.tippy-box');
if(tObj === null)
{
window.setTimeout(checkForTippy2, 100);
return;
}
// Now check the core popup content has been rendered...
if(tObj.querySelectorAll('.root--r5WBv').length === 0)
{
window.setTimeout(checkForTippy2, 100);
return;
}
if(tObj.querySelector('wz-toggle-switch') !== null)
{
tObj.querySelector('wz-toggle-switch').addEventListener('click', checkForTippy2);
}
// If we get this far it means the stuff we want to mess with is
// present, so, uhh, let's go mess with it :-)
let compress = getById('_inpUICompression').value;
if(compress > 0)
{
restyleTippy();
}
}
var TippyObserver = new MutationObserver(function()
{
// If we detect any changes in the popup contents, treat it almost the same as the original body onchange event
// and go back into checking for the drop-down elements being present prior to restyling
checkForTippy2();
});
function restyleTippy()
{
let tObj = document.querySelector('.tippy-box');
if(tObj === null)
{
window.setTimeout(restyleTippy, 100);
return;
}
// See how many root elements there are in the popup
let n = tObj.querySelectorAll('.root--r5WBv').length;
if(n > 1)
{
// If there's more than one such element in the popup, it means the popup is displaying all the elements
// for an enabled turn, which will also mean the TIO menu should be displayed, so we now wait for that
// to arrive before continuing...
let menuRendered = (document.querySelector('.tippy-box').querySelector('#opcode-select').getBoundingClientRect().width !== 0);
if(menuRendered === false)
{
window.setTimeout(restyleTippy, 100);
return;
}
// Now the menu has been rendered, apply our tweaks to it
let m = tObj.querySelectorAll('wz-option').length;
while(m)
{
--m;
let sr = tObj.querySelectorAll('wz-option')[m].shadowRoot;
let mi = sr.querySelector('.wz-menu-item');
if(mi !== null)
{
SetStyle(mi, "height", "100%", false);
SetStyle(mi, "lineHeight", "130%", false);
SetStyle(mi, "minHeight", "auto", false);
}
}
}
// Compress the root elements...
while(n)
{
--n;
let mi = tObj.querySelectorAll('.root--r5WBv')[n];
if(mi !== null)
{
SetStyle(mi, "minHeight", "auto", false);
}
}
// Having done that, we now set up a mutation observer on the popup, which allows us to detect when its redrawn
// (which would cause the redrawn drop-down to revert to the default styling) without the body onchange event
// triggering - this can occur if they mouse-over one of the other turn arrows on the currently selected segment,
// or if they select a different segment without first deselecting the current one...
TippyObserver.observe(tObj, { childList: true, subtree: true });
}
function checkForIssuesFilter()
{
// To restyle the drop-down menu entries in the issue tracker panel, we need to be aware that, unlike all the other
// drop-downs in WME that exist outside of a dynamic popup (i.e. for TIOs...), the ones within the tracker filtering
// options slideout are only created the first time the options are shown. Once that's happened, it then doesn't
// matter how many times the slideout is slid in or out within that WME session, the DOM remains untouched.
//
// So, to enable restyling of these semi-dynamically created entries, we first wait for the firing of the onchange
// event attached to the filter panel - that tells us WME is starting to update the DOM here. Then we wait for the
// first "wz-option" element within the areas section to gain a "wz-menu-item" classed element in its shadowroot.
// Once we've seen that, we know that the drop-downs not only exist in the main DOM, but also that their shadowroots
// have been populated - as per usual these days, it's the shadowroots where we need to apply the restyling, so until
// WME has finished populating those then we can't do anything...
//
// Once we've seen the required element within the shadowroot, we then simply repeat the general (un)compression
// styling on all of the drop-downs that are now visible within WME - as this is a one-shot action, the slight loss
// of efficiency from reapplying the same styles to the ones we've already dealt with is less of an issue than the
// increased complexity of having the code figure out which ones it does and doesn't need to restyle this time.
//
// And yes, this affords me the ideal opportunity to complain once again about how annoying shadowroots are - in the
// good old days all of this would have been accomplished with a simple addition to the global CSS overrides FUME
// can still use for *some* stuff, yet here we are having to use event handlers and DOM content checking and iterative
// applications of styles to individual elements.
let srProbe = document.querySelector('#areas')?.querySelector('wz-option')?.shadowRoot?.querySelector('.wz-menu-item');
let srExists = ((srProbe != null) && (srProbe != undefined));
if(srExists === true)
{
restyleDropDownEntries();
}
else
{
window.setTimeout(checkForIssuesFilter, 100);
}
}
function applyEnhancements()
{
shrinkTopBars();
compressSidebar();
restyleReports();
compressLayersMenu();
moveUserInfo();
}
function moveSVRecentreIcons()
{
let fname = 'moveSVRecentreIcons';
if (getById('_cbMoveZoomBar').checked)
{
if(getById('WMEFUzoom') === null) return;
// Apply the styling related to the zoombar, so that we can get an accurate read of its
// size/location in a moment...
var styles = "";
//Overall bar
styles += '.olControlPanZoomBar { left: 10px; top: 35px; height: 158px; border: 1px solid #f0f2f2; background-color: #f0f2f2; border-radius: 30px; width: 24px; box-sizing: initial; }';
////styles += '#WMEFUzoom { z-index: 1200 !important;}';
//zoom in/out buttons
styles += '.olButton { background-color: white; border-radius: 30px; width: 24px; height: 24px; cursor: pointer; }';
styles += '.olControlZoomButton { padding: 3px 5px; font-size: 18px; }';
//slider stops
styles += '.yslider-stops { width: 24px; height: 110px; background-color: #f3f3f3; background-image: linear-gradient(90deg, transparent 45%, #dedede 45%, #dedede 55%, transparent 55%), linear-gradient(#dedede 1px, transparent 1px); background-size: 50% 8px; background-repeat: repeat-y; background-position: 6px; }';
//slider
styles += '.slider { position: absolute; font-size: 15px; font-weight: 900; line-height: 1; text-align: center; width: 24px; height: 18px; margin-top: -29px; padding-top: 1px; border: 1px solid lightgrey; border-radius: 10px; background-color: white; cursor: ns-resize; }';
addStyle(prefix + fname,styles);
// Force a re-render of the zoombar, so that the graphics elements which are missing at this point
// get inserted - this is also required to get an accurate size/location read...
ZLI();
// Get the absolute positions/sizes of the newly generated zoombar, the existing element that contains
// the SV and location buttons, and the size of the SV button
var zbBCR = getById('WMEFUzoom').getBoundingClientRect();
var bcBCR = getByClass('bottom.overlay-buttons-container').getBoundingClientRect();
var btnBCR = getByClass('street-view-control').getBoundingClientRect();
// Use this information to calculate what the x/y positions will need to be for the buttons to position
// them correctly below the zoombar. Note that the x position will be negative, as this gets applied
// relative to the parent container in which the button container resides, rather than to the map view
var bcPosX = zbBCR.left - bcBCR.left;
var bcPosY = zbBCR.top + zbBCR.height;
// Also work out how tall the button container will need to be once we've hidden the native zoom
// controls
var bcHeight = (btnBCR.height * 2) + 10;
// Now apply the full set of styling...
styles = "";
// All the stuff for the zoom bar again
styles += '.olControlPanZoomBar { left: 10px; top: 35px; height: 158px; border: 1px solid #f0f2f2; background-color: #f0f2f2; border-radius: 30px; width: 24px; box-sizing: initial; }';
////styles += '#WMEFUzoom { z-index: 1200 !important;}';
styles += '.olButton { background-color: white; border-radius: 30px; width: 24px; height: 24px; cursor: pointer; }';
styles += '.olControlZoomButton { padding: 3px 5px; font-size: 18px; }';
styles += '.yslider-stops { width: 24px; height: 110px; background-color: #f3f3f3; background-image: linear-gradient(90deg, transparent 45%, #dedede 45%, #dedede 55%, transparent 55%), linear-gradient(#dedede 1px, transparent 1px); background-size: 50% 8px; background-repeat: repeat-y; background-position: 6px; }';
styles += '.slider { position: absolute; font-size: 15px; font-weight: 900; line-height: 1; text-align: center; width: 24px; height: 18px; margin-top: -29px; padding-top: 1px; border: 1px solid lightgrey; border-radius: 10px; background-color: white; cursor: ns-resize; }';
// Then the rest of it...
// Hide the native zoom control
styles += '.zoom-bar-container { display: none; }';
// shift UR/MP panel to the right
styles += '.panel.show { margin-left: 55px; }';
// Move the SV/location buttons under the zoom bar
styles += '.bottom.overlay-buttons-container { position: absolute; left: '+bcPosX+'px; top: '+bcPosY+'px; height: '+bcHeight+'px; }';
styles += '.street-view-region { margin-bottom: 8px; }';
// keep the issue tracker filter panel above the relocated ruler/SV/recentre buttons
styles += '#feed-filter-region { z-index: 1; }';
addStyle(prefix + fname,styles);
ZLI();
}
else
{
removeStyle(prefix + fname);
}
}
function createZoomBar()
{
if (getById('_cbMoveZoomBar').checked)
{
// Create the zoombar element and add it to the map view
yslider = new OpenLayers.Control.PanZoomBar({zoomStopHeight:5 , panIcons:false});
yslider.position.x = 10;
yslider.position.y = 35;
yslider.id = 'WMEFUzoom';
W.map.addControl(yslider);
W.map.events.register("zoomend", null, ZLI);
zliResizeObserver = new ResizeObserver(ZLIDeferred);
zliResizeObserver.observe(document.getElementById('street-view-container'));
zliResizeObserver.observe(document.getElementById('sidebar'));
}
else
{
if (yslider)
{
yslider.destroy();
}
W.map.events.unregister("zoomend", null, ZLI);
if(zliResizeObserver !== null)
{
zliResizeObserver.disconnect();
}
}
moveSVRecentreIcons();
}
function ZLIDeferred()
{
// The ResizeObserver attached to the StreetView container fires not only when the container is
// opened or closed, but also when its width is altered by dragging the vertical divider. On
// some systems, the RO events that are triggered by this latter type of resizing are processed
// before the StreetView container has finished redrawing - on such systems, if we were to call
// ZLI directly from the RO event, the zoom bar would end up being trashed again if the user
// resized using the divider, so we add a short delay after the RO event before calling ZLI.
setTimeout(ZLI, 200);
// Likewise for the function used to relocate the SV and recentre icons...
setTimeout(moveSVRecentreIcons, 200);
}
function ZLI()
{
if (yslider)
{
//Need to reset the OpenLayers-created settings from the zoom bar when it's redrawn
//Overall bar
yslider.div.style.left = "";
yslider.div.style.top = "";
//zoom in/out buttons
yslider.buttons[0].style = "";
yslider.buttons[0].innerHTML = modifyHTML("<div class='olControlZoomButton fa fa-plus' ></div>");
yslider.buttons[1].style = "";
yslider.buttons[1].innerHTML = modifyHTML("<div class='olControlZoomButton fa fa-minus' ></div>");
//slider stops
yslider.zoombarDiv.classList.add("yslider-stops");
yslider.zoombarDiv.classList.remove("olButton");
yslider.zoombarDiv.style="";
//slider
yslider.slider.innerHTML = modifyHTML("");
yslider.slider.style = "";
yslider.slider.classList.add("slider");
yslider.moveZoomBar();
//Actually set the ZLI
yslider.slider.innerText = W.map.getZoom();
yslider.slider.title = "Zoom level indicator by WMEFU";
switch (W.map.getZoom())
{
case 4:
case 5:
case 6:
case 7:
case 8:
case 9:
case 10:
case 11:
case 12:
case 13:
yslider.slider.style.background = '#ef9a9a';
yslider.slider.title += "\nCannot permalink any segments at this zoom level";
break;
case 14:
case 15:
yslider.slider.style.background = '#ffe082';
yslider.slider.title += "\nCan only permalink primary or higher at this zoom level";
break;
default:
yslider.slider.style.background = '#ffffff';
yslider.slider.title += "\nCan permalink any segments at this zoom level";
break;
}
if (W.map.getZoom() < 12)
{
// Re-enable the sidepanel UI if the user has opted to do so...
if(getById('_cbReEnableSidePanel').checked === true)
{
if(document.getElementsByClassName('overlay editingDisabled').length > 0)
{
document.getElementsByClassName('overlay editingDisabled')[0].remove();
}
}
// ...and always relocate the warning dialog you get if the feed tab is active, so that
// it sits nicely below the tabs regardless of how many there are.
if (document.getElementsByClassName('zoom-edit-message editingDisabled').length > 0)
{
let eTop = document.getElementById('user-tabs').getBoundingClientRect().height;
eTop = Math.round(eTop) + 'px';
document.getElementsByClassName('zoom-edit-message editingDisabled')[0].style.top = eTop;
document.getElementsByClassName('zoom-edit-message editingDisabled')[0].style.left = '0px';
}
}
}
}
function resizeSearch()
{
let sb = document.querySelector('#search');
if(getById('_cbResizeSearchBox').checked)
{
sb.style.width="100%";
let wcs = window.getComputedStyle(document.querySelector('[class^="primaryToolbar"]'));
let tbAvailable = parseInt(wcs.marginLeft) + parseInt(wcs.marginRight);
if(tbAvailable > 0)
{
sb.style.width = tbAvailable+'px';
}
}
else
{
sb.style.width = "";
}
}
function moveSearchThisArea()
{
if(document.querySelector('div.w-icon-search')?.parentElement?.parentElement !== undefined)
{
document.querySelector('div.w-icon-search').parentElement.parentElement.style.zIndex = "5";
}
}
function UnmoveUserInfo()
{
let ina = document.querySelector('.secondary-toolbar').lastChild;
insertNodeAfterNode(document.querySelector('wz-user-box'), ina);
insertNodeAfterNode(document.querySelector('.user-toolbar'), ina);
}
function moveUserInfo()
{
// Now functioning correctly for prod & beta
let fname = 'moveUserInfo';
let styles = "";
let mStyle = '';
if (getById('_cbMoveUserInfo').checked)
{
// styles += '#user-box-region { margin-left: 5px; }';
styles += '.notifications-button { display: flex; }';
styles += '#app-head aside #left-app-head .waze-logo { width: 50px; }';
styles += '.user-toolbar .notifications-button { padding: 0 4px; }';
styles += '.notifications-box-container { transform: translate3d(300px, 0px, 0px) !important; }';
addStyle(prefix + fname,styles);
insertNodeBeforeNode(document.querySelector('.user-toolbar'),getById('left-app-head'));
insertNodeBeforeNode(document.querySelector('wz-user-box'),getById('left-app-head'));
mStyle = 'translate3d(240px, 0px, 0px)';
//Fix to move control button of Invalidated Camera Mass Eraser
if (getById("_UCME_btn"))
{
getById("advanced-tools").appendChild(getById("_UCME_btn"));
getById('UCME_btn').parentNode.removeChild(getById('UCME_btn'));
}
}
else
{
removeStyle(prefix + fname);
UnmoveUserInfo();
}
unfloat();
// Keep the user profile box aligned to the profile picture
let sr = document.querySelector('wz-user-box');
if(sr !== null) sr = sr.shadowRoot;
if(sr !== null)
{
let mObj = sr.querySelector('wz-menu');
if(mObj !== null)
{
mObj.style.transform = mStyle;
}
}
}
function RemoveTopBarCompression()
{
let a = document.querySelector('wz-header')?.shadowRoot.querySelector('.content-wrapper');
if((a != null) & (a != undefined))
{
a.style.height = "";
}
a = document.querySelector('.restricted-driving-area')?.parentElement;
if((a != null) & (a != undefined))
{
a.style.position = "";
}
}
function ApplyTopBarShadowRootWzButtonCompression()
{
let retval = true;
if (getById('_cbShrinkTopBars').checked)
{
let compress = getById('_inpUICompression').value;
if (compress > 0)
{
const c1 = ['','35px','24px'];
let tbRoot = document.querySelector('#app-head');
if(tbRoot != null)
{
let btnWraps = tbRoot.querySelectorAll('.toolbar-button-wrapper');
for(let i = 0; i < btnWraps.length; ++i)
{
let btnElm = btnWraps[i].querySelector('wz-button');
if(btnElm !== undefined)
{
let srButton = btnElm?.shadowRoot?.querySelector('.wz-button');
if((srButton !== undefined) && (srButton !== null))
{
srButton.style.height = c1[compress];
}
else
{
retval = false;
}
}
}
}
}
}
return retval;
}
function shrinkTopBars()
{
let fname = 'shrinkTopBars';
let styles = "";
RemoveTopBarCompression();
if (getById('_cbShrinkTopBars').checked)
{
let contrast = getById('_inpUIContrast').value;
let compress = getById('_inpUICompression').value;
//always do this stuff
//event mode button
styles += '#mode-switcher-region .title-button .icon { font-size: 13px; font-weight: bold; color: black; }';
//black bar
styles += '#topbar-container { pointer-events: none; }';
styles += '#map #topbar-container .topbar > div { pointer-events: initial; }';
//change toolbar buttons - from JustinS83
$('#mode-switcher-region .title-button .icon').removeClass('w-icon-caret-down');
$('#mode-switcher-region .title-button .icon').addClass('fa fa-calendar');
// HN editing tweaks
styles += '#map-lightbox .content { pointer-events: none; }';
styles += '#map-lightbox .content > div { pointer-events: initial; }';
styles += '#map-lightbox .content .header { pointer-events: none !important; }';
styles += '.toolbar .toolbar-button.add-house-number { background-color: #61cbff; float: right; font-weight: bold; }';
styles += '.waze-icon-exit { background-color: #61cbff; font-weight: bold; }';
// event mode button
styles += '.toolbar.toolbar-mte .add-button { background-color: orange; font-weight: bold; }';
// fix for narrow windows and new toolbar
let nbuttons = 3 + (getById('_cbUnfloatButtons').checked ? 2 : 0);
let minW = nbuttons * ([58,49,33][compress]) + [80,65,55][compress];
styles += '#edit-buttons { min-width: ' + minW + 'px; }';
styles += '#toolbar { padding: 0px ; }';
if (compress > 0)
{
const c1 = ['','35px','24px'];
const c2 = ['','13px','12px'];
//overall menu bar
styles += '#left-app-head { height: ' + c1[compress] + ' !important; }';
styles += '#app-head { height: ' + c1[compress] + '; }';
styles += '#toolbar { height: ' + c1[compress] + ' !important; }';
styles += '.group-title-tooltip-wrap { height: ' + c1[compress] + ' !important; }';
styles += '.restricted-driving-area wz-tooltip-target { height: ' + c1[compress] + ' !important; }';
styles += '.edit-area { height: calc(100% - ' + c1[compress] + '); }';
styles += '#user-toolbar { height: ' + c1[compress] + '; }';
styles += 'wz-user-box { scale: ' + ((parseInt(c1[compress]) * 100) / 36) + '%; }';
styles += '#app-head aside .short-title { font-size: ' + c2[compress] + '; margin-right: 4px; }';
styles += '#app-head aside #debug { padding-right: ' + ['','10px','6px'][compress] + '; line-height: ' + ['','15px','12px'][compress] + '; white-space: nowrap; }';
styles += '.mode-switcher-view .title-button .icon { line-height: ' + c1[compress] + '; }';
styles += '.mode-switcher-view .dropdown-menu { top: ' + c1[compress] + '; }';
styles += '.toolbar { font-size: ' + c2[compress] + '; }';
styles += '.toolbar { height: ' + c1[compress] + ' !important; }';
styles += '.toolbar { gap: 4px; }';
styles += '.toolbar-collection-view { gap: 4px !important; }';
styles += '.toolbar .toolbar-group { margin-right: 0px !important; }';
styles += '#edit-buttons { gap: 2px; }';
styles += '#search { flex: 2 1 auto; }';
// New "thing" popup menu - attempting to override the "top" style added by WME here is a no-go, as that
// appears to get applied to the menu *after* this function is called, so to avoid the need for yet another
// time-based style hack, just alter the top margin of the menu element to give the same effective shift
// needed to keep the top of the menu visible as we reduce the height of the topbar...
// Note: need to use a semi-wildcarded selector in order to apply this fix *only* to the element tagged as
// wz-menu which *also* has a classname starting "menu" (it's one of those OH SO delightful classes with a
// random suffix which means we can't simply do a direct match on it), as simply selecting wz-menu also
// applies this fix to other stuff (such as the "invisible" section of the TIO edit UI, which caused the
// weird and unepected formatting error in earlier versions of FUME...).
styles += 'wz-menu[class^="menu"] { margin-top: ' + ['', '70px', '70px'][compress] + '; }';
//search box
styles += '#search { padding-top: ' + ['','3px','1px'][compress] + ' !important; }';
styles += '.form-search { height: ' + ['','27px','22px'][compress] + '; }';
styles += '.form-search .search-query { height: ' + ['','27px','22px'][compress] + '; font-size: ' + c2[compress] + '; }';
styles += '.form-search .input-wrapper .search-icon { font-size: ' + ['','18px','16px'][compress] + '; left: ' + ['','9px','6px'][compress] + '; }';
styles += '.form-search .search-query { padding-left: ' + ['','34px','24px'][compress] + ';; }';
//edit-buttons section
styles += '#edit-buttons { margin-right: ' + ['','9px','2px'][compress] + '; }';
//toolbar dropdowns
styles += '.toolbar .toolbar-group { margin-right: ' + ['','14px','8px'][compress] + '; padding-top: 0px; height: ' + c1[compress] + '; }';
styles += '.toolbar .group-title { height: ' + ['','34px','24px'][compress] + '; line-height: ' + ['','34px','24px'][compress] + '; }';
styles += '.toolbar .dropdown-menu { top: ' + ['','34px','24px'][compress] + ' !important; left: ' + ['','7px','4px'][compress] + ' !important; }';
styles += 'wz-menu { top: ' + ['','34px','24px'][compress] + ' !important; }';
//toolbar buttons
styles += '.toolbar .toolbar-button { margin-top: ' + ['','3px','1px'][compress] + '; margin-left: 3px; padding-left: ' + ['','10px','5px'][compress] + '; padding-right: ' + ['','10px','5px'][compress] + '; height: ' + ['','27px','22px'][compress] + '; line-height: ' + ['','27px','22px'][compress] + '; }';
styles += '.toolbar .toolbar-button { padding-left: ' + ['','2px','2px'][compress] + '; padding-right: ' + ['','2px','2px'][compress] + '; }';
styles += '.toolbar .toolbar-button .item-container { padding-left: ' + ['','9px','2px'][compress] + '; padding-right: ' + ['','9px','2px'][compress] + '; }';
styles += '.toolbar .item-icon { font-size: ' + ['','22px','20px'][compress] + ' !important; }';
styles += '.toolbar .toolbar-button > .item-icon { top: ' + ['','5px','2px'][compress] + '; }';
styles += '.toolbar .toolbar-separator { height: ' + ['','34px','22px'][compress] + '; }';
styles += '.toolbar-button-wrapper { padding: 0!important; }';
//extra hack for my Permalink Counter button
styles += '.WMEFUPCicon { margin-top: ' + ['','4px !important','2px !important'][compress] + '; }';
//floating buttons
styles += '.overlay-button { height: ' + ['','33px','26px'][compress] + '; width: ' + ['','33px','26px'][compress] + '; font-size: ' + ['','22px','20px'][compress] + '; padding: ' + ['','3px','1px'][compress] + '; }';
styles += '#Info_div { height: ' + ['','33px','26px'][compress] + ' !important; width: ' + ['','33px','26px'][compress] + ' !important; }';
styles += '.zoom-bar-container {width: ' + ['','33px','26px'][compress] + ' !important; }';
styles += '.zoom-bar-container .overlay-button {height: ' + ['','33px','26px'][compress] + ' !important; }';
styles += '#overlay-buttons .overlay-buttons-container > div:last-child { margin-bottom: 0; }';
//layers menu
// styles += '.layer-switcher .toolbar-button { margin-top: ' + ['','1px','0px'][compress] + ' !important; font-size: ' + ['','22px','20px'][compress] + ' !important; height: ' + ['','32px','24px'][compress] + '; }';
//user button
styles += '#user-box-region { margin-right: ' + ['','8px','2px'][compress] + '; }';
styles += '.user-box-avatar { height: ' + ['','32px','23px'][compress] + ' !important; font-size: ' + ['','22px','20px'][compress] + '; }';
styles += '.app .level-icon { width: ' + ['','32px','23px'][compress] + ' !important; height: ' + ['','32px','23px'][compress] + ' !important;}';
//new save menu
styles += '.changes-log-region { top: ' + ['','26px','21px'][compress] + '; }';
//black bar
styles += '.topbar { height: ' + ['','24px','18px'][compress] + '; line-height: ' + ['','24px','18px'][compress] + '; }';
//fix for WME Presets button
styles += '#WMEPresetsDiv > i { height: 100%;}';
// remove the unecessary space to the left of the notification icon
styles += '.secondary-toolbar-spacer { display: none; }';
// All the stuff that can no longer be done via CSS due to shadowroot...
ApplyTopBarShadowRootWzButtonCompression();
let hdrStyle = "height: " + c1[compress] + ";";
hdrStyle += "padding: 0px 16px 0px 16px;";
document.querySelector('wz-header').shadowRoot.querySelector('.content-wrapper').setAttribute("style", hdrStyle);
document.querySelector('#delete-button')?.shadowRoot.querySelector('button')?.setAttribute("style", "height: auto!important;");
document.querySelector('#undo-button')?.shadowRoot.querySelector('button')?.setAttribute("style", "height: auto!important;");
document.querySelector('#redo-button')?.shadowRoot.querySelector('button')?.setAttribute("style", "height: auto!important;");
document.querySelector('#notification-button')?.shadowRoot.querySelector('button')?.setAttribute("style", "height: auto!important;");
document.querySelector('.reload-button')?.shadowRoot.querySelector('button')?.setAttribute("style", "min-width: auto!important;");
document.querySelector('.layer-switcher-button')?.shadowRoot.querySelector('button')?.setAttribute("style", "min-width: auto!important;");
}
else
{
RemoveTopBarCompression();
}
if (contrast > 0)
{
//toolbar dropdown menus
styles += '.toolbar .group-title { color: black; }';
styles += '.toolbar .toolbar-button { border-radius: 8px; ' + GetBorderContrast(contrast, false) + 'color: black; }';
//layers icon - until Waze fix it
styles += '.layer-switcher .waze-icon-layers.toolbar-button{ background-color: white; }';
}
addStyle(prefix + fname,styles);
}
else
{
removeStyle(prefix + fname);
//change toolbar buttons - from JustinS83
$('#mode-switcher-region .title-button .icon').removeClass('fa fa-calendar');
$('#mode-switcher-region .title-button .icon').addClass('fa fa-angle-down');
}
window.dispatchEvent(new Event('resize'));
}
function compressSidebar()
{
let fname = 'compressSidebar';
compressSegmentPanel();
// Apply a permanently active styling fix to enable wrapping in the drives tab,
// to counter the effects of lengthening the datetime string format...
let styles = "";
styles += '.list-item-card-title { white-space: pre-wrap; }';
addStyle(prefix + fname + "_permanent",styles);
// Now go and do the optional styling stuff
styles = "";
if (getById('_cbCompressSidebar').checked)
{
var contrast = getById('_inpUIContrast').value;
var compress = getById('_inpUICompression').value;
//Neuter the top gradient
styles += '#sidebar .tab-scroll-gradient { pointer-events: none; }';
//Nuke the bottom gradient
styles += '#sidebar #links:before { display: none; }';
// Make map comment text always visible
styles += '.map-comment-name-editor .edit-button { display: block !important; }';
// fix the latest layout bug (add closure button at the bottom of the screen) introduced by a WME update...
styles += '.closures-list { height: auto; }';
if (compress > 0)
{
//Lock level
styles += '.lock-level-selector { display: flex; }';
styles += '#edit-panel .lock-edit-view label { line-height: 140% !important; }';
styles += '#edit-panel .lock-edit-view label { height: auto !important; width: auto !important; }';
styles += '#edit-panel .lock-edit-view label { margin-right: 2px !important; }';
styles += '#edit-panel .lock-edit-view label { margin-bottom: 6px !important; }';
//general compression enhancements
styles += '#sidebar { line-height: ' + ['','18px','16px'][compress] + ' !important;}';
styles += '#sidebar .tab-content .tab-pane { padding: ' + ['','8px','1px'][compress] + '; }';
styles += '#sidebar #sidebarContent { font-size: ' + ['','13px','12px'][compress] + '; }';
styles += '#sidebar #advanced-tools { padding: ' + ['','0 9px','0 4px'][compress] + '; }';
styles += '#sidebar .waze-staff-tools { margin-bottom: ' + ['','9px','4px'][compress] + '; height: ' + ['','25px','20px'][compress] + '; }';
styles += '#sidebar .categories-card-content { row-gap: ' + ['','3px','0px'][compress] + '; }';
//Tabs
styles += '#sidebar .nav-tabs { padding-bottom: ' + ['','3px','2px'][compress] + '; }';
styles += '#sidebar #user-info #user-tabs { padding: ' + ['','0 9px','0 4px'][compress] + '; }';
styles += '#sidebar .tabs-container { padding: ' + ['','0 9px','0 4px'][compress] + '; }';
styles += '#sidebar .nav-tabs li a { margin-top: ' + ['','2px','1px'][compress] + '; margin-left: ' + ['','3px','1px'][compress] + '; padding: ' + ['','0 6px','0 2px'][compress] + '; line-height: ' + ['','24px','21px'][compress] + '; height: ' + ['','24px','21px'][compress] + '; }';
styles += '#sidebar .nav-tabs li { flex-grow: 0; }';
//Feed
styles += '.feed-item { margin-bottom: ' + ['','3px','1px'][compress] + '; }';
styles += '.feed-item .inner { padding: ' + ['','5px','0px'][compress] + '; }';
styles += '.feed-item .content .title { margin-bottom: ' + ['','1px','0px'][compress] + '; }';
styles += '.feed-item .motivation { margin-bottom: ' + ['','2px','0px'][compress] + '; }';
//Drives & Areas
styles += '#sidebar .message { margin-bottom: ' + ['','6px','2px'][compress] + '; }';
styles += '#sidebar .result-list .result { padding: ' + ['','6px 17px','2px 9px'][compress] + '; margin-bottom: ' + ['','3px','1px'][compress] + '; }';
styles += '#sidebar .result-list .session { background-color: lightgrey; }';
styles += '#sidebar .result-list .session-available { background-color: white; }';
styles += '#sidebar .result-list .result.selected { background-color: lightgreen; }';
styles += 'div#sidepanel-drives { height: auto !important; }';
//SEGMENT EDIT PANEL
//general changes
//checkbox groups
styles += '#sidebar .controls-container { padding-top: ' + ['','4px','1px'][compress] + '; display: inline-block; font-size: ' + ['','12px','11px'][compress] + '; }';
styles += '#sidebar .controls-container input[type="checkbox"] + label { padding-left: ' + ['','21px','17px'][compress] + ' !important; } }';
//form groups
styles += '#sidebar .form-group { margin-bottom: ' + ['','5px','0px'][compress] + '; }';
//dropdown inputs
styles += '#sidebar .form-control { height: ' + ['','27px','19px'][compress] + '; padding-top: ' + ['','4px','0px'][compress] + '; padding-bottom: ' + ['','4px','0px'][compress] + '; font-size: ' + ['','13px','12px'][compress] + '; color: black; }';
//buttons
styles += '#edit-panel .waze-btn { padding-top: 0px !important; padding-bottom: ' + ['','3px','1px'][compress] + '; height: ' + ['','20px','18px'][compress] + ' !important; line-height: ' + ['','20px','18px'][compress] + ' !important; font-size: ' + ['','13px','12px'][compress] + '; }';
// styles += '#edit-panel .waze-btn { padding-top: ' + ['','3px','0px'][compress] + ' !important; padding-bottom: ' + ['','3px','1px'][compress] + '; height: ' + ['','20px','18px'][compress] + ' !important; line-height: ' + ['','20px','18px'][compress] + ' !important; font-size: ' + ['','13px','12px'][compress] + '; }';
//radio button controls
styles += '.waze-radio-container label { height: ' + ['','19px','16px'][compress] + '; width: ' + ['','19px','16px'][compress] + '; line-height: ' + ['','19px','16px'][compress] + '; font-size: ' + ['','13px','12px'][compress] + '; margin-bottom: ' + ['','3px','1px'][compress] + '; }';
styles += '.waze-radio-container label { width: auto; padding-left: ' + ['','6px','3px'][compress] + ' !important; padding-right: ' + ['','6px','3px'][compress] + ' !important; }';
//text input areas
styles += '#sidebar textarea.form-control { height: auto; }';
styles += '#sidebar textarea { max-width: unset; }';
//specific changes
//Selected segments info
styles += '#edit-panel .selection { padding-top: ' + ['','8px','2px'][compress] + '; padding-bottom: ' + ['','8px','4px'][compress] + '; }';
styles += '#edit-panel .segment .direction-message { margin-bottom: ' + ['','9px','3px'][compress] + '; }';
//Segment details (closure warning)
styles += '#edit-panel .segment .segment-details { padding: ' + ['','10px','5px'][compress] + '; padding-top: 0px; }';
//All control labels
styles += '#edit-panel .control-label { font-size: ' + ['','11px','10px'][compress] + '; margin-bottom: ' + ['','4px','1px'][compress] + '; }';
//Address input
styles += '#edit-panel .address-edit-view { cursor: pointer; margin-bottom: ' + ['','6px','2px'][compress] + '!important; }';
styles += '#edit-panel .address-edit-input { padding: ' + ['','4px','1px'][compress] + '; font-size: ' + ['','13px','12px'][compress] + '; }';
styles += '.tts-button { height: ' + ['','28px','21px'][compress] + '; }';
//alt names
styles += '.alt-street-list { margin-bottom: ' + ['','4px','0px'][compress] + '; }';
styles += '#edit-panel .add-alt-street-form .alt-street { padding-top: ' + ['','13px','3px'][compress] + '; padding-bottom: ' + ['','13px','3px'][compress] + '; }';
styles += '#edit-panel .add-alt-street-form .alt-street .alt-street-delete { top: ' + ['','12px','4px'][compress] + '; }';
styles += '#edit-panel .segment .address-edit-view .address-form .action-buttons { padding-top: ' + ['','11px','6px'][compress] + '; padding-bottom: ' + ['','11px','6px'][compress] + '; margin-top: ' + ['','5px','0px'][compress] + '; height: ' + ['','45px','28px'][compress] + '; }';
styles += '#edit-panel .add-alt-street-form .new-alt-street { padding-top: ' + ['','8px','3px'][compress] + '; padding-bottom: ' + ['','8px','3px'][compress] + '; }';
//restrictions control
styles += '#edit-panel .restriction-list { margin-bottom: ' + ['','5px','0px'][compress] + '; }';
//speed limit controls
styles += '#edit-panel .speed-limit { margin-top: ' + ['','0px','-5px'][compress] + '; margin-bottom: ' + ['','5px','2px'][compress] + ';}';
styles += '#edit-panel .segment .speed-limit label { margin-bottom: ' + ['','3px','1px'][compress] + '; }';
styles += '#edit-panel .segment .speed-limit .form-control { height: ' + ['','23px','19px'][compress] + '; padding-top: ' + ['','4px','2px'][compress] + '; font-size: ' + ['','13px','12px'][compress] + '; width: 5em; margin-left: 0px; }';
styles += '#edit-panel .segment .speed-limit .direction-label { font-size: ' + ['','12px','11px'][compress] + '; line-height: ' + ['','2.0em','1.8em'][compress] + '; }';
styles += '#edit-panel .segment .speed-limit .unit-label { font-size: ' + ['','12px','11px'][compress] + '; line-height: ' + ['','2.0em','1.8em'][compress] + '; margin-left: 0px;}';
styles += '#edit-panel .segment .speed-limit .average-speed-camera { margin-left: 40px; }';
styles += '#edit-panel .segment .speed-limit .average-speed-camera .camera-icon { vertical-align: top; }';
styles += '#edit-panel .segment .speed-limit .verify-buttons { margin-bottom: ' + ['','5px','0px'][compress] + '; }';
//more actions section
styles += '#edit-panel .more-actions { padding-top: ' + ['','6px','2px'][compress] + '; }';
styles += '#edit-panel .more-actions .waze-btn.waze-btn-white { padding-left: 0px; padding-right: 0px; }';
//additional attributes
styles += '#edit-panel .additional-attributes { margin-bottom: ' + ['','3px','1px'][compress] + '; }';
//history items
styles += '.toggleHistory { padding: ' + ['','7px','3px'][compress] + '; }';
styles += '.element-history-item:not(:last-child) { margin-bottom: ' + ['','3px','1px'][compress] + '; }';
styles += '.element-history-item .tx-header { padding: ' + ['','6px','2px'][compress] + '; }';
styles += '.element-history-item .tx-header .tx-author-date { margin-bottom: ' + ['','3px','1px'][compress] + '; }';
styles += '.element-history-item .tx-content { padding: ' + ['','7px 7px 7px 22px','4px 4px 4px 22px'][compress] + '; }';
styles += '.loadMoreContainer { padding: ' + ['','5px 0px','3px 0px'][compress] + '; }';
//closures tab
styles += '.closures-tab wz-button { transform: scale(' + ['','0.85','0.7'][compress] + '); padding: 0px!important; }';
styles += '.closures > div:not(.closures-list) { padding: ' + ['','0px','0px'][compress] + '; }';
styles += 'body { --wz-text-input-height: ' + ['','30px','20px'][compress] + '; }';
styles += 'body { --wz-select-height: ' + ['','30px','20px'][compress] + '; }';
styles += 'input.wz-text-input { height: ' + ['','30px','20px'][compress] + '; }';
styles += '.edit-closure .closure-nodes .closure-node-item .closure-node-control { padding: ' + ['','7px','2px'][compress] + '; }';
//closures list
styles += '.closures-list .add-closure-button { line-height: ' + ['','20px','18px'][compress] + '; }';
styles += '.closures-list .closure-item:not(:last-child) { margin-bottom: ' + ['','6px','2px'][compress] + '; }';
styles += '.closures-list .closure-item .details { padding: ' + ['','5px','0px'][compress] + '; font-size: ' + ['','12px','11px'][compress] + '; }';
styles += '.closures-list .closure-item .buttons { top: ' + ['','7px','4px'][compress] + '; }';
//tweak for Junction Box button
styles += '#edit-panel .junction-actions > button { width: inherit; }';
//PLACE DETAILS
styles += '#edit-panel .navigation-point-list { margin-bottom: ' + ['','4px','0px'][compress] + '; }';
//alert
styles += '#edit-panel .header-alert { margin-bottom: ' + ['','6px','2px'][compress] + '; padding: ' + ['','6px 32px','2px 32px'][compress] + '; }';
//address input
styles += '#edit-panel .full-address { padding-top: ' + ['','4px','1px'][compress] + '; padding-bottom: ' + ['','4px','1px'][compress] + '; font-size: ' + ['','13px','12px'][compress] + '; }';
//alt names
styles += '#edit-panel .aliases-view .list li { margin: ' + ['','12px 0','4px 0'][compress] + '; }';
styles += '#edit-panel .aliases-view .delete { line-height: inherit; }';
//categories
styles += '#edit-panel .categories .select2-search-choice .category { margin: ' + ['','2px 0 2px 4px','1px 0 1px 3px'][compress] + '; height: ' + ['','18px','15px'][compress] + '; line-height: ' + ['','18px','15px'][compress] + '; }';
styles += '#edit-panel .categories .select2-search-field input { height: ' + ['','18px','17px'][compress] + '; }';
styles += '#edit-panel .categories .select2-choices { min-height: ' + ['','26px','19px'][compress] + '; }';
styles += '#edit-panel .categories .select2-container { margin-bottom: 0px; }';
//entry/exit points
styles += '#edit-panel .navigation-point-view .navigation-point-list-item .preview { padding: ' + ['','3px 7px','0px 4px'][compress] + '; font-size: ' + ['','13px','12px'][compress] + '; }';
styles += '#edit-panel .navigation-point-view .add-button { height: ' + ['','28px','18px'][contrast] + '; line-height: ' + ['','17px','16px'][contrast] + '; font-size: ' + ['','13px','12px'][compress] + '; }';
//type buttons
styles += '#sidebar .area-btn, #sidebar .point-btn { display: flex; align-items: center; justify-content: center; height: ' + ['','22px','20px'][compress] + '; line-height: ' + ['','19px','16px'][compress] + '; font-size: ' + ['','13px','12px'][compress] + '; }';
styles += '#sidebar .area-btn:before, #sidebar .point-btn:before { top: 0px; margin-right: 8px; }';
//external providers
styles += '.select2-container { font-size: ' + ['','13px','12px'][compress] + '; }';
styles += '#edit-panel .external-providers-view .external-provider-item { margin-bottom: ' + ['','6px','2px'][compress] + '; }';
styles += '.external-providers-view > div > ul { margin-bottom: ' + ['','4px','0px'][compress] + '; }';
styles += '#edit-panel .external-providers-view .add { padding: ' + ['','3px 12px','1px 9px'][compress] + '; }';
styles += '#edit-panel .waze-btn.waze-btn-smaller { line-height: ' + ['','26px','21px'][compress] + '; }';
//residential toggle
styles += '#edit-panel .toggle-residential { height: ' + ['','27px','22px'][compress] + '; }';
//more info
styles += '.service-checkbox { font-size: ' + ['','13px','12px'][compress] + '; }';
//PARKING LOT SPECIFIC
styles += '.parking-type-option{ display: inline-block; }';
styles += '.payment-checkbox { display: inline-block; min-width: ' + ['','48%','31%'][compress] + '; }';
styles += '.service-checkbox { display: inline-block; min-width: 49%; font-size: ' + ['','12px','11px'][compress] + '; }';
styles += '.lot-checkbox { display: inline-block; min-width: 49%; }';
//MAP COMMENTS
styles += '#sidebar .map-comment-name-editor { padding: ' + ['','10px','5px'][compress] + '; }';
styles += '#sidebar .map-comment-name-editor .edit-button { margin-top: 0px; font-size: ' + ['','13px','12px'][compress] + '; padding-top: ' + ['','3px','1px'][compress] + '; }';
styles += '#sidebar .conversation-view .no-comments { padding: ' + ['','10px 15px','5px 15px'][compress] + '; }';
styles += '#sidebar .map-comment-feature-editor .conversation-view .comment-list { padding-top: ' + ['','8px','1px'][compress] + '; padding-bottom: ' + ['','8px','1px'][compress] + '; }';
styles += '#sidebar .map-comment-feature-editor .conversation-view .comment-list .comment .comment-content { padding: ' + ['','6px 0px','2px 0px'][compress] + '; }';
styles += '#sidebar .conversation-view .comment .text { padding: ' + ['','6px 9px','3px 4px'][compress] + '; font-size: ' + ['','13px','12px'][compress] + '; }';
styles += '#sidebar .conversation-view .new-comment-form { padding-top: ' + ['','10px','5px'][compress] + '; }';
styles += '#sidebar .map-comment-feature-editor .clear-btn { height: ' + ['','26px','19px'][compress] + '; line-height: ' + ['','26px','19px'][compress] + '; }';
//Compression for WME Speedhelper
styles += '.clearfix.controls.speed-limit { margin-top: ' + ['','-4px','-8px'][compress] + '; }';
//Compression for WME Clicksaver
styles += '.rth-btn-container { margin-bottom: ' + ['','2px','-1px'][compress] + '; }';
styles += '#csRoutingTypeContainer { height: ' + ['','23px','16px'][compress] + ' !important; margin-top: ' + ['','-2px','-4px'][compress] + '; }';
styles += '#csElevationButtonsContainer { margin-bottom: ' + ['','2px','-1px'][compress] + ' !important; }';
//tweak for WME Clicksaver tab controls
styles += '#sidepanel-clicksaver .controls-container { width: 100%; }';
//tweak for JAI tab controls
styles += '#sidepanel-ja .controls-container { width: 100%; }';
//tweaks for UR-MP Tracker
styles += '#sidepanel-urt { margin-left: ' + ['','-5px','0px'][compress] + ' !important; }';
styles += '#urt-main-title { margin-top: ' + ['','-5px','0px'][compress] + ' !important; }';
//tweaks for my own panel
styles += '#fuContent { line-height: ' + ['','10px','9px'][compress] + ' !important; }';
// scripts panel
styles += '#user-tabs { padding: 0px !important; }';
}
if (contrast > 0)
{
//contrast enhancements
//general
styles += '#sidebar .form-group { border-top: 1px solid ' + ['','lightgrey','grey'][contrast] + '; }';
styles += '#sidebar .text { color: ' + ['','darkslategrey','black'][contrast] + '; }';
styles += '#sidebar {background-color: #d6ebff; }';
styles += ':root {--background_variant: #d0ffd0; }';
//text colour
styles += '#sidebar { color: black; }';
//advanced tools section
styles += '#sidebar waze-staff-tools { background-color: #c7c7c7; }';
//Tabs
styles += '#sidebar .nav-tabs { ' + GetBorderContrast(contrast, false) + '}';
styles += '#sidebar .nav-tabs li a { ' + GetBorderContrast(contrast, true) + '}';
//Fix the un-noticeable feed refresh button
styles += 'span.fa.fa-repeat.feed-refresh.nav-tab-icon { width: 19px; color: orangered; }';
styles += 'span.fa.fa-repeat.feed-refresh.nav-tab-icon:hover { color: red; font-weight: bold; font-size: 15px; }';
//Feed
styles += '.feed-item { ' + GetBorderContrast(contrast, false) + '}';
styles += '.feed-issue .content .title .type { color: ' + ['','black','black'][contrast] + '; font-weight: bold; }';
styles += '.feed-issue .content .timestamp { color: ' + ['','darkslategrey','black'][contrast] + '; }';
styles += '.feed-issue .content .subtext { color: ' + ['','darkslategrey','black'][contrast] + '; }';
styles += '.feed-item .motivation { font-weight: bold; }';
//Drives & Areas
styles += '#sidebar .result-list .result { ' + GetBorderContrast(contrast, false) + '}';
//Segment edit panel
styles += '#edit-panel .selection { font-size: 13px; }';
styles += '#edit-panel .segment .direction-message { color: orangered; }';
styles += '#edit-panel .address-edit-input { color: black; ' + GetBorderContrast(contrast, false) + '}';
styles += '#sidebar .form-control { ' + GetBorderContrast(contrast, false) + '}';
//radio buttons when disabled
styles += '.waze-radio-container input[type="radio"]:disabled:checked + label { color: black; opacity: 0.7; font-weight:600; }';
//override border for lock levels
styles += '#sidebar .waze-radio-container { border: 0 none !important; }';
styles += '#edit-panel .waze-btn { color: black; ' + GetBorderContrast(contrast, false) + '}';
styles += '.waze-radio-container label { ' + GetBorderContrast(contrast, false) + '}';
//history items
styles += '.toggleHistory { color: black; text-align: center; }';
styles += '.element-history-item .tx-header { color: black; }';
styles += '.element-history-item .tx-header a { color: ' + ['','royalblue','black'][contrast] + '!important; }';
styles += '.element-history-item.closed .tx-header { border-radius: 8px; ' + GetBorderContrast(contrast, false) + '}';
styles += '.loadMoreHistory { ' + GetBorderContrast(contrast, false) + '}';
//closures list
styles += '.closures-list .closure-item .details { border-radius: 8px; ' + GetBorderContrast(contrast, false) + '}';
styles += '.closures-list .closure-item .dates { color: black; }';
styles += '.closures-list .closure-item .dates .date-label { opacity: 1; }';
//Place details
//alert
styles += '#edit-panel .alert-danger { color: red; }';
//address input
styles += '#edit-panel .full-address { color: black; ' + GetBorderContrast(contrast, false) + '}';
styles += '#edit-panel a.waze-link { font-weight: bold; }';
//the almost invisible alternate name link
styles += '#edit-panel .add.waze-link { color: ' + ['','royalblue','black'][contrast] + '!important; }';
//categories
styles += '#edit-panel .categories .select2-search-choice .category { text-transform: inherit; font-weight: bold; background: gray; }';
//entry/exit points
styles += '#edit-panel .navigation-point-view .navigation-point-list-item .preview { ' + GetBorderContrast(contrast, false) + '}';
styles += '#edit-panel .navigation-point-view .add-button { ' + GetBorderContrast(contrast, false) + ' margin-top: 2px; padding: 0 5px; color: ' + ['','royalblue','black'][contrast] + '!important; }';
//type buttons
styles += '#sidebar .point-btn { color: black; ' + GetBorderContrast(contrast, true) + '}';
//external providers
styles += '.select2-container { color: teal; ' + GetBorderContrast(contrast, true) + '}';
styles += '.select2-container .select2-choice { color: black; }';
//residential toggle
styles += '#edit-panel .toggle-residential { font-weight: bold; }';
//COMMENTS
styles += '.map-comment-name-editor { border-color: ' + ['','darkgrey','grey'][contrast] + '; }';
}
//fix for buttons of WME Image Overlay script
styles += '#sidepanel-imageoverlays > div.result-list button { height: 24px; }';
addStyle(prefix + fname,styles);
}
else
{
removeStyle(prefix + fname);
removeStyle(prefix + 'hideHeadlights');
}
restyleDropDownEntries();
}
function restyleDropDownEntries()
{
let compress = getById('_inpUICompression').value;
let enabled = (getById('_cbCompressSidebar').checked);
if((enabled === true) && (compress > 0))
{
compressDropDownEntries();
}
else
{
uncompressDropDownEntries();
}
}
function compressDropDownEntries()
{
let n = document.querySelectorAll('wz-option').length;
while(n)
{
let obj = document.querySelectorAll('wz-option')[n-1];
if(obj != undefined)
{
let mi = obj.shadowRoot.querySelector('.wz-menu-item');
if(mi != null)
{
mi.style.lineHeight = "130%";
mi.style.height = "100%";
mi.style.minHeight = "0px";
}
}
--n;
}
}
function uncompressDropDownEntries()
{
let n = document.querySelectorAll('wz-option').length;
while(n)
{
let obj = document.querySelectorAll('wz-option')[n-1];
if(obj != undefined)
{
let mi = obj.shadowRoot.querySelector('.wz-menu-item');
if(mi != null)
{
mi.style.lineHeight = "var(--wz-menu-option-height, var(--wz-option-height, 40px));";
mi.style.height = "var(--wz-menu-option-height, var(--wz-option-height, 40px))";
}
}
--n;
}
}
function hideUnuseableStuff()
{
if(W?.model?.getTopCountry === undefined)
{
// getTopCountry takes a short while to become available, so keep checking at regular
// intervals until it's there to be used...
setTimeout(hideUnuseableStuff,100);
}
else
{
let fname = 'hideUnuseableStuff';
let styles = "";
// Hide the headlights reminder checkbox for segments in countries that don't use it
if (W?.model?.getTopCountry()?.allowHeadlightsReminderRank === null)
{
styles += '.headlights-reminder { display: none !important; }';
}
// Hide the restricted areas toolbar button for anyone who can't make use of it
/*
//// update as RA option is now in sidepanel "new thing" menu...
if(document.querySelector('wz-button.restricted-driving-area').disabled === true)
{
styles += 'wz-button.restricted-driving-area { display: none !important; }';
}
*/
if(styles !== "")
{
addStyle(prefix + fname,styles);
}
}
}
function compressLayersMenu()
{
let fname = 'compressLayersMenu';
removeStyle(prefix + fname);
let styles = "";
if (getById('_cbCompressLayersMenu').checked)
{
getById('layersColControls').style.opacity = '1';
let contrast = getById('_inpUIContrast').value;
let compress = getById('_inpUICompression').value;
if (compress > 0)
{
//VERTICAL CHANGES
//change menu to autoheight - not working
// styles += '.layer-switcher .menu { height: auto; width: auto; max-height: calc(100% - 26px); overflow-y: scroll }';
//change menu to auto-width
styles += '.layer-switcher .menu { width: auto }';
styles += '.layer-switcher .menu.hide-layer-switcher { left: 100% }';
//menu title
styles += '.layer-switcher .menu > .title { font-size: ' + ['','14px','12px'][compress] + '; padding-bottom: ' + ['','7px','2px'][compress] + '; padding-top: ' + ['','7px','2px'][compress] + ' }';
styles += '.layer-switcher .menu > .title .w-icon-x { font-size: ' + ['','21px','18px'][compress] + ' }';
styles += '.layer-switcher .scrollable { height: calc(100% - ' + ['','39px','29px'][compress] + ') }';
//menu group headers
styles += '.layer-switcher .layer-switcher-toggler-tree-category { padding: ' + ['','5px','2px'][compress] + ' 0; height: ' + ['','30px','20px'][compress] + ' }';
//menu items
styles += '.layer-switcher li { line-height: ' + ['','20px','16px'][compress] + '}';
styles += '.layer-switcher .togglers ul li .wz-checkbox { margin-bottom: ' + ['','3px','0px'][compress] + ' }';
styles += '.wz-checkbox { min-height: ' + ['','20px','16px'][compress] + ' }';
styles += '.wz-checkbox input[type="checkbox"] + label { line-height: ' + ['','20px','16px'][compress] + '; font-size: ' + ['','12px','11px'][compress] + ' }';
styles += '.wz-checkbox input[type="checkbox"] + label:before { font-size: ' + ['','13px','10px'][compress] + '; height: ' + ['','16px','14px'][compress] + '; width: ' + ['','16px','14px'][compress] + '; line-height: ' + ['','12px','11px'][compress] + ' }';
//HORIZONTAL CHANGES
styles += '.layer-switcher .togglers ul { padding-left: ' + ['','19px','12px'][compress] + '; }';
styles += '.layer-switcher .togglers .group { padding: ' + ['','0 8px 0 4px','0 4px 0 2px'][compress] + ' }';
if (getById('_cbLayersColumns').checked)
{
//2 column stuff
styles += '.layer-switcher .scrollable { columns: 2; }';
styles += 'li.group { break-inside: avoid; page-break-inside: avoid; }';
//prevent city names showing up when it should be hidden
styles += ' .layer-switcher ul[class^="collapsible"].collapse-layer-switcher-group { visibility: collapse }';
styles += '.layer-switcher .menu { overflow-x: hidden; overflow-y: scroll; height: auto; max-height: calc(100% - ' + ['','39px','29px'][compress] + ') }';
styles += '.layer-switcher .scrollable { overflow-x: hidden; overflow-y: hidden; height: unset }';
}
// fix from ABelter for layers menu
styles += ' .layer-switcher ul[class^="collapsible"] { max-height: none; }';
}
else
{
//2-columns not available without compression
getById('layersColControls').style.opacity = '0.5';
}
if (contrast > 0)
{
styles += '.controls-container.main.toggler { color: white; background: dimgray }';
styles += '.layer-switcher .toggler.main .label-text { text-transform: inherit }';
//labels
styles += '.layer-switcher .layer-switcher-toggler-tree-category > .label-text { color: black }';
styles += '.wz-checkbox input[type="checkbox"] + label { WME: FU; color: black }';
//group separator
styles += '.layer-switcher .togglers .group { border-bottom: 1px solid ' + ['','lightgrey','grey'][contrast] + ' }';
//column rule
styles += '.layer-switcher .scrollable { column-rule: 1px solid ' + ['','lightgrey','grey'][contrast] + ' }';
}
addStyle(prefix + fname,styles);
}
else
{
getById('layersColControls').style.opacity = '0.5';
removeStyle(prefix + fname);
}
}
function changePassVisibility()
{
ReportObserver.disconnect();
let panelContainer = document.querySelector('#panel-container');
let passes = panelContainer.querySelectorAll('.activeHovSubscriptions');
let pDisp = 'none';
if(panelContainer.querySelector('#_cbCollapsePasses').checked === false)
{
pDisp = '';
}
for(let i = 0; i < passes.length; ++i)
{
passes[i].style.display = pDisp;
}
ReportObserver.observe(document.querySelector('#panel-container'), { childList: true, subtree: true });
}
var ReportObserver = new MutationObserver(function()
{
AddCollapsiblePasses();
});
function AddCollapsiblePasses()
{
let panelContainer = document.querySelector('#panel-container');
if(panelContainer.getBoundingClientRect().width > 0)
{
let passes = panelContainer.querySelectorAll('.activeHovSubscriptions');
if(passes.length > 0)
{
if(panelContainer.querySelector('#_cbCollapsePasses') == null)
{
ReportObserver.disconnect();
let upHeader = panelContainer.querySelector('.reporter-preferences').querySelector('.title');
upHeader.innerHTML += ' | Hide passes ('+passes.length+') <input type="checkbox" id="_cbCollapsePasses" checked/>';
document.getElementById('_cbCollapsePasses').addEventListener('click', changePassVisibility, true);
changePassVisibility();
}
}
}
}
function restyleReports()
{
let fname = 'restyleReports';
let styles = "";
if (getById('_cbRestyleReports').checked)
{
let contrast = getById('_inpUIContrast').value;
let compress = getById('_inpUICompression').value;
// Stops Reject/Approve buttons being partially/completely cut off...
styles += '.place-update-edit .place-update { max-height: 100%; }';
if (compress > 0)
{
//report header
// Remove title text - we know what the panel contains, because we've asked WME to open it...
styles += '#panel-container .main-title { display: none!important; }';
styles += '#panel-container .issue-panel-header { padding: ' + ['','9px 36px','1px 36px'][compress] + '; line-height: ' + ['','19px','17px'][compress] + '; }';
styles += '#panel-container .issue-panel-header .dot { top: ' + ['','15px','7px'][compress] + '; }';
//special treatment for More Information checkboxes (with legends)
styles += '#panel-container .problem-edit .more-info .legend { left: 20px; top: 3px; }';
styles += '#panel-container .more-info input[type="checkbox"] + label { padding-left: 33px !important; }';
// User preferences section
styles += '#panel-container .preferences-container { gap: 0px !important; }';
//report body
styles += '#panel-container .body { line-height: ' + ['','15px','13px'][compress] + '; font-size: ' + ['','13px','12px'][compress] + '; }';
//problem description
styles += '#panel-container .collapsible { padding: ' + ['','9px','3px'][compress] + '; }';
//comments
////styles += '#panel-container .conversation-view .comment .comment-content { padding: ' + ['','6px 9px','2px 3px'][compress] + '; }';
styles += '#panel-container .comment .text { padding: ' + ['','7px 9px','4px 4px'][compress] + '; }';
// Remove padding around comment boxes
styles += '#panel-container wz-list { padding: 0px!important; }';
////styles += '#panel-container .wz-list-item .list-item-wrapper { padding-bottom: 0px!important; padding-top: 0px!important; }';
//new comment entry
styles += '#panel-container .conversation-view .new-comment-form { padding: ' + ['','8px 9px 6px 9px','1px 3px 2px 3px'][compress] + '; }';
//send button
styles += '#panel-container .conversation-view .send-button { padding: ' + ['','4px 16px','2px 12px'][compress] + '; box-shadow: ' + ['','3px 3px 4px 0 #def7ff','3px 2px 4px 0 #def7ff'][compress] + '; }';
//lower buttons
styles += '#panel-container > div > div > div.actions > div > div { padding-top: ' + ['','6px','3px'][compress] + '; }';
styles += '#panel-container .close-details.section { font-size: ' + ['','13px','12px'][compress] + '; line-height: ' + ['','13px','9px'][compress] + '; }';
styles += '#panel-container .problem-edit .actions .controls-container label { height: ' + ['','28px','21px'][compress] + '; line-height: ' + ['','28px','21px'][compress] + '; margin-bottom: ' + ['','5px','2px'][compress] + '; }';
styles += '#panel-container .waze-plain-btn { height: ' + ['','30px','20px'][compress] + '; line-height: ' + ['','30px','20px'][compress] + '; }';
styles += '.panel .navigation { margin-top: ' + ['','6px','2px'][compress] + '; }';
//WMEFP All PM button
styles += '#WMEFP-UR-ALLPM { top: ' + ['','5px','0px'][compress] + ' !important; }';
}
if (contrast > 0)
{
styles += '#panel-container .section { border-bottom: 1px solid ' + ['','lightgrey','grey'][contrast] + '; }';
styles += '#panel-container .close-panel { border-color: ' + ['','lightgrey','grey'][contrast] + '; }';
styles += '#panel-container .main-title { font-weight: 900; }';
styles += '#panel-container .reported { color: ' + ['','darkslategrey','black'][contrast] + '; }';
styles += '#panel-container .date { color: ' + ['','#6d6d6d','#3d3d3d'][contrast] + '; }';
styles += '#panel-container .comment .text { ' + GetBorderContrast(contrast, false) + '}';
styles += '#panel-container .comment-content.reporter .username { color: ' + ['','#159dc6','#107998'][contrast] + '; }';
styles += '#panel-container .conversation-view .new-comment-form textarea { ' + GetBorderContrast(contrast, false) + '}';
styles += '#panel-container .top-section { border-bottom: 1px solid ' + ['','lightgrey','grey'][contrast] + '; }';
styles += '#panel-container .waze-plain-btn { font-weight: 800; color: ' + ['','#159dc6','#107998'][contrast] + '; }';
}
addStyle(prefix + fname,styles);
draggablePanel();
ReportObserver.observe(document.querySelector('#panel-container'), { childList: true, subtree: true });
AddCollapsiblePasses();
}
else
{
removeStyle(prefix + fname);
if (jQuery.ui)
{
if ( $("#panel-container").hasClass('ui-draggable') )
{
$("#panel-container").draggable("destroy");
}
getById("panel-container").style = "";
}
ReportObserver.disconnect();
}
window.dispatchEvent(new Event('resize'));
}
function draggablePanel()
{
if(wmeFUinitialising === true)
{
window.setTimeout(draggablePanel, 500);
}
else
{
if (jQuery.ui)
{
if ($("#panel-container").draggable)
{
$("#panel-container").draggable({ handle: ".issue-panel-header" });
}
}
}
}
function narrowSidePanel()
{
let fname = 'narrowSidePanel';
let styles = "";
if (getById('_cbNarrowSidePanel').checked)
{
//sidebar width
styles += '.row-fluid #sidebar { width: 250px; }';
//map width
styles += '.show-sidebar .row-fluid .fluid-fixed { margin-left: 250px; }';
//user info tweaks
styles += '#sidebar #user-info #user-box { padding: 0 0 5px 0; }';
styles += '#sidebar #user-details { width: 250px; }';
styles += '#sidebar #user-details .user-profile .level-icon { margin: 0; }';
styles += '#sidebar #user-details .user-profile .user-about { max-width: 161px; }';
//gradient bars
styles += '#sidebar .tab-scroll-gradient { width: 220px; }';
styles += '#sidebar #links:before { width: 236px; }';
//feed
styles += '.feed-item .content { max-width: 189px; }';
//segment edit panel
styles += '#edit-panel .more-actions .waze-btn.waze-btn-white { width: 122px; }';
//tweak for WME Bookmarks
styles += '#divBookmarksContent .divName { max-width: 164px; }';
//tweak for WME PH buttons
styles += '#WMEPH_runButton .btn { font-size: 11px; padding: 2px !important; }';
addStyle(prefix + fname, styles);
}
else
{
removeStyle(prefix + fname);
}
compressSidebar();
window.dispatchEvent(new Event('resize'));
}
function shiftAerials()
{
let siLayerNames =
[
"satellite_imagery",
"merged_collection_by_latest_no_candid",
"merged_collection_by_quality_no_candid",
"satellite_pleiades_ortho_rgb",
"satellite_worldview2_ortho_rgb",
"satellite_worldview3_ortho_rgb",
"satellite_geoeye1_ortho_rgb",
"satellite_skysat_ortho_rgb",
"satellite_pneo_ortho_rgb"
];
// calculate meters/pixel for current map view, taking into account how the
// map projection stretches things out the further from the equator you get
let metersPerPixel = W.map.getResolution();
let mapCentre = new OpenLayers.LonLat();
mapCentre.lon = W.map.getCenter().lon;
mapCentre.lat = W.map.getCenter().lat;
mapCentre.transform(new OpenLayers.Projection("EPSG:900913"), new OpenLayers.Projection("EPSG:4326"));
let latAdj = Math.cos(mapCentre.lat * Math.PI / 180);
metersPerPixel *= latAdj;
if(metersPerPixel == 0)
{
metersPerPixel = 0.001;
}
let sLeft = Math.round(getById("_inpASX").value / metersPerPixel) + 'px';
let sTop = Math.round(- getById("_inpASY").value / metersPerPixel) + 'px';
if (getById('_inpASO').value < 10) getById('_inpASO').value = 10;
let sOpa = getById("_inpASO").value/100;
let sLeftO = Math.round(getById("_inpASXO").value / metersPerPixel) + 'px';
let sTopO = Math.round(- getById("_inpASYO").value / metersPerPixel) + 'px';
if (getById('_inpASOO').value < 10) getById('_inpASOO').value = 10;
let sOpaO = getById("_inpASOO").value/100;
if ((getById("_inpASX").value != 0) || (getById("_inpASY").value != 0) || (getById("_inpASXO").value != 0) || (getById("_inpASYO").value != 0))
{
getById("WMEFU_AS").style.display = "block";
}
else
{
getById("WMEFU_AS").style.display = "none";
}
// Apply the shift and opacity to all available imagery layers
for(let i = 0; i < siLayerNames.length; ++i)
{
let siLayer = W.map.getLayersByName(siLayerNames[i]);
if(siLayer.length == 1)
{
let siDiv = siLayer[0].div;
if(i === 0)
{
// Standard layer
siDiv.style.left = sLeft;
siDiv.style.top = sTop;
siDiv.style.opacity = sOpa;
}
else
{
// Additional layers
siDiv.style.left = sLeftO;
siDiv.style.top = sTopO;
siDiv.style.opacity = sOpaO;
}
}
}
}
function ApplyArrowFix(aObj)
{
if(aObj.touchedByFUME === undefined)
{
let rStr = aObj.style.transform;
let rFloat = 0;
if(rStr.indexOf('deg')!=-1)
{
rFloat=parseFloat(rStr.split('(')[1].split('deg')[0]);
}
rFloat += 180.0;
aObj.style.transform="rotate("+rFloat+"deg) scaleX(-1)";
aObj.touchedByFUME = true;
}
}
function RTCArrowsFix()
{
if(W.model.isLeftHand === true)
{
let rtcDiv = W.map.getLayerByName("closures").div;
let fLen = rtcDiv.querySelectorAll('.forward').length;
while(fLen)
{
ApplyArrowFix(rtcDiv.querySelectorAll('.forward')[fLen-1]);
--fLen;
}
let rLen = rtcDiv.querySelectorAll('.backward').length;
while(rLen)
{
ApplyArrowFix(rtcDiv.querySelectorAll('.backward')[rLen-1]);
--rLen;
}
}
}
var RTCMarkerObserver = new MutationObserver(function()
{
RTCArrowsFix();
});
function warnCommentsOff()
{
let fname = 'warnCommentsOff';
if (W.map.getLayerByUniqueName('mapComments').visibility === false)
{
removeStyle(prefix + fname);
addStyle(prefix + fname, '#app-head { --background_default: #FFC107 ; }');
}
else
{
removeStyle(prefix + fname);
}
// extra bit because killNodeLayer will be inactive
getById("_btnKillNode").style.backgroundColor = "";
}
function adjustGSV()
{
let fname = 'adjustGSV';
let styles = "";
let C = getById('_inpGSVContrast');
let B = getById('_inpGSVBrightness');
let I = getById('_cbGSVInvert');
if (C.value < 10) C.value = 10;
if (B.value < 10) B.value = 10;
styles += '.gm-style { filter: contrast(' + C.value + '%) ';
styles += 'brightness(' + B.value + '%) ';
if (I.checked)
{
styles += 'invert(1); }';
}
else
{
styles += 'invert(0); }';
}
removeStyle(prefix + fname);
if ((C.value != 100) || (B.value !=100) || I.checked) addStyle(prefix + fname, styles);
}
function GSVWidth()
{
let fname = 'GSVWidth';
removeStyle(prefix + fname);
let w = getById('_inpGSVWidth').value;
if (w != 50)
{
let styles = "";
styles += '#editor-container #map.street-view-mode #waze-map-container { width: ' + (100-w) + '%; }';
styles += '#editor-container #street-view-container { width: ' + w + '%; }';
styles += '#editor-container #map #street-view-drag-handle { left: ' + (100-w) + '%; }';
addStyle(prefix + fname, styles);
}
window.dispatchEvent(new Event('resize'));
}
function GSVWidthReset()
{
getById('waze-map-container').style = null;
getById('street-view-container').style = null;
getById('street-view-drag-handle').style = null;
// Check for WME Street View Availability
// This can be removed soon - WME SVA no longer remembers GSV width
if (localStorage.WMEStreetViewWidth)
{
localStorage.WMEStreetViewWidth = '';
}
window.dispatchEvent(new Event('resize'));
}
function moveChatIcon()
{
let fname = 'moveChatIcon';
let styles = "";
if (getById('_cbMoveChatIcon').checked)
{
styles += '#chat-overlay { left: inherit !important; right: 60px !important;}';
styles += '#chat-overlay #chat-toggle { right: 0px !important; }';
addStyle(prefix + fname,styles);
}
else
{
removeStyle(prefix + fname);
}
}
function highlightInvisible()
{
let fname = 'highlightInvisible';
let styles = "";
if (getById('_cbHighlightInvisible').checked)
{
styles += '#chat-overlay.visible-false #chat-toggle button { filter: none; background-color: #ff0000c0; }';
addStyle(prefix + fname,styles);
}
else
{
removeStyle(prefix + fname);
}
}
function darkenSaveLayer()
{
let fname = 'darkenSaveLayer';
let styles = "";
if (getById('_cbDarkenSaveLayer').checked)
{
styles += '#popup-overlay { background-color: dimgrey !important; }';
addStyle(prefix + fname,styles);
}
else
{
removeStyle(prefix + fname);
}
}
function swapRoadsGPS()
{
let fname = 'swapRoadsGPS';
let styles = "";
if (getById('_cbSwapRoadsGPS').checked)
{
var rlName = "roads";
var glName = "gps_points";
var roadLayerId = W.map.getLayerByUniqueName(rlName).id;
var GPSLayerId = W.map.getLayerByUniqueName(glName).id;
var roadLayerZ = W.map.getLayerByUniqueName(rlName).getZIndex();
var GPSLayerZ = W.map.getLayerByUniqueName(glName).getZIndex();
logit("Layers identified\n\tRoads: " + roadLayerId + "," + roadLayerZ + "\n\tGPS: " + GPSLayerId + "," + GPSLayerZ, "info");
styles += '#' + roadLayerId.replace(/\./g,"\\2e") + ' { z-index: ' + GPSLayerZ + ' !important; }';
styles += '#' + GPSLayerId.replace(/\./g,"\\2e") + ' { z-index: ' + roadLayerZ + ' !important; }';
addStyle(prefix + fname,styles);
}
else
{
removeStyle(prefix + fname);
}
}
function killNode()
{
getById(W.map.getLayerByUniqueName("nodes").id + "_root").style.display = "none";
getById("_btnKillNode").style.backgroundColor = "yellow";
}
function toggleKillTurnPopup()
{
let fname = 'toggleKillTurnPopup';
if(killTurnPopup === true)
{
getById("WMEFUTPB").style.backgroundColor = "inherit";
killTurnPopup = false;
removeStyle(prefix + fname);
}
else
{
getById("WMEFUTPB").style.backgroundColor = "red";
killTurnPopup = true;
addStyle(prefix + fname,'div[data-theme*="map-tooltip"] { display: none !important; }');
}
}
function showMapBlockers()
{
let fname = 'showMapBlockers';
let styles = "";
if (getById('_cbShowMapBlockers').checked)
{
styles += '.street-view-layer { background-color: rgba(255,0,0,0.3); }';
styles += '.overlay-buttons-container.top { background-color: rgba(255,0,0,0.3); }';
styles += '.overlay-buttons-container.bottom { background-color: rgba(255,0,0,0.3); }';
styles += '#street-view-drag-handle { background-color: rgba(255,0,0,0.3); }';
addStyle(prefix + fname,styles);
fixNodeClosureIcons();
}
else
{
removeStyle(prefix + fname);
}
}
function fixNodeClosureIcons()
{
var closureNodesId = W.map.getLayerByUniqueName('closure_nodes').id;
var SVPinId = W.map.getLayersByName('streetViewPin')[0].id;
addGlobalStyle('div#' + closureNodesId + ' { z-index: 725 !important }');
insertNodeBeforeNode(getById(closureNodesId), getById(SVPinId));
}
function disableBridgeButton()
{
let fname = 'disableBridgeButton';
let styles = "";
if (getById('_cbDisableBridgeButton').checked)
{
styles += '.add-bridge { pointer-events: none; opacity: 0.4; }';
addStyle(prefix + fname,styles);
}
else
{
removeStyle(prefix + fname);
}
}
function disablePathButton()
{
let fname = 'disablePathButton';
let styles = "";
if (getById('_cbDisablePathButton').checked)
{
styles += '.path-icon { pointer-events: none; opacity: 0.4; }';
addStyle(prefix + fname,styles);
}
else
{
removeStyle(prefix + fname);
}
}
function disableHazardMarkers()
{
let fname = 'disableHazardMarkers';
let styles = "";
if (getById('_cbDisableHazardMarkers').checked)
{
styles += '.permanent-hazard-marker { pointer-events: none; opacity: 0.2; }';
styles += '.permanentHazardMarker--Paqzi { pointer-events: none; opacity: 0.2; }';
addStyle(prefix + fname,styles);
}
else
{
removeStyle(prefix + fname);
}
}
function disableKinetic()
{
if (getById('_cbDisableKinetic').checked)
{
W.map.controls.find(control => control.dragPan).dragPan.kinetic = null;
}
else
{
W.map.controls.find(control => control.dragPan).dragPan.kinetic = kineticDragParams;
}
}
function disableAnimatedZoom()
{
if (getById('_cbDisableZoomAnimation').checked)
{
W.map.segmentLayer.map.zoomDuration = 0;
}
else
{
W.map.segmentLayer.map.zoomDuration = 20;
}
}
function disableScrollZoom()
{
var controller = null;
if (W.map.navigationControl)
{
controller = W.map.navigationControl;
}
else if (W.map.controls.find(control => control.CLASS_NAME == 'OpenLayers.Control.Navigation'))
{
controller = W.map.controls.find(control => control.CLASS_NAME == 'OpenLayers.Control.Navigation');
}
else
{
logit('Cannot find zoom wheel controls - please alert script maintainers','error');
}
if (controller !== null)
{
if (getById('_cbDisableScrollZoom').checked)
{
controller.disableZoomWheel();
}
else
{
controller.enableZoomWheel();
}
}
}
function PCclicked()
{
let itemType = checkSelectableItemInURL();
if(itemType !== null)
{
reselectItems(itemType);
}
}
function reselectItems (itemType)
{
let match = searchSelectableItemsInURL(itemType[0]);
if (match)
{
let IDArray = decodeURIComponent(match[1]).split(',');
let objectArray=[];
for (let i=0; i < IDArray.length; i++)
{
let object = W.model[itemType[0]].objects[IDArray[i]];
if (typeof object!='undefined') objectArray.push(object);
}
if (itemType[1])
{
W.selectionManager.setSelectedModels(objectArray);
}
else
{
W.selectionManager.setSelectedModels(objectArray[0]);
}
}
}
function createDSASection()
{
var settingsDiv = null;
settingsDiv = document.querySelector("#sidepanel-prefs > div > form");
if (!settingsDiv)
{
logit("WME Settings div not there yet - looping...","warning");
setTimeout(createDSASection,500);
return;
}
if (localStorage.dontShowAgain)
{
var dontShowAgain = JSON.parse(localStorage.dontShowAgain);
var DSGroup = document.createElement('div');
DSGroup.classList = "form-group";
var DSLabel = document.createElement('label');
DSLabel.classList = "control-label";
DSLabel.innerHTML = modifyHTML("Disabled WME warnings");
DSLabel.title = "This section will not update if you disable a warning\n";
DSLabel.title += "from a WME pop-up. Re-load the page if you need\n";
DSLabel.title += "to re-enable a warning you have just disabled.\n\n";
DSLabel.title += "SECTION ADDED BY WME Fix UI.";
DSGroup.appendChild(DSLabel);
DSGroup.appendChild(document.createElement('br'));
var DSCC = document.createElement('div');
DSCC.classList = "controls-container";
var DSInput;
for (var property in dontShowAgain)
{
DSInput = document.createElement('input');
DSInput.type = "checkbox";
DSInput.id = "WMEFUDScb_" + property.toString();
DSInput.setAttribute("orig", property.toString());
DSInput.checked = dontShowAgain[property];
DSLabel = document.createElement('label');
DSLabel.setAttribute("for", DSInput.id);
DSLabel.innerText = property.toString();
DSCC.appendChild(DSInput);
DSCC.appendChild(DSLabel);
DSCC.appendChild(document.createElement('br'));
DSInput.onclick = DSIclicked;
}
DSGroup.appendChild(DSCC);
settingsDiv.appendChild(DSGroup);
}
}
function DSIclicked (e)
{
var DSA = JSON.parse(localStorage.dontShowAgain);
DSA[e.target.getAttribute("orig")] = e.target.checked;
localStorage.dontShowAgain = JSON.stringify(DSA);
}
// These two event handlers act in concert with the change log restyling carried
// out in disableSaveBlocker(), to allow the change log to continue being shown
// and hidden as expected when the user mouses-over or out of the save button,
// but not for it to then be displayed as soon as an edit is made whilst the
// disable save blocker option is enabled and ctrl+s has been used to save an
// earlier edit. No, I still have NO idea why this combination of FUME setting
// and WME keyboard shortcut causes the log to appear, hence this slightly
// contrived workaround to essentially replicate the visibility changes WME
// ought to be doing itself...
let inSaveDetails = false;
let inSaveButton = false;
function saveMouseOver()
{
let styles = '.changes-log { display: block !important; }';
addStyle(prefix,styles);
inSaveButton = true;
window.setTimeout(addSaveDetailsEventListeners,100);
}
function saveMouseOut()
{
inSaveButton = false;
window.setTimeout(saveMouseOutDeferred, 1000);
}
function addSaveDetailsEventListeners()
{
// Since adding the above fix, it's come to my attention that it has the potential to break
// the ability of the save details popup to remain visible when moused-over. To unbreak
// this without rebreaking the ctrl+s fix, we add similar mouseover/out handlers to the popup,
// noting that as it's dynamically generated whenever the save button is moused-over when
// saves are pending, we have to reapply the handlers each time... The following function is
// therefore called from the buttom mouseover handler above, with a short delay to give WME a
// chance to generate the element before we try dealing with it.
if(document.querySelector('.changes-log') !== null)
{
document.querySelector('.changes-log').addEventListener('mouseover', saveDetailsMouseOver, true);
document.querySelector('.changes-log').addEventListener('mouseout', saveDetailsMouseOut, true);
}
}
function saveDetailsMouseOver()
{
// The mouseover handler simply sets a flag to let the rest of the script know that we're
// in the popup
inSaveDetails = true;
}
function saveDetailsMouseOut()
{
// Whilst the mouseout handler both clears the flag and also calls the mouseout handler for
// the save button (to hide the popup) unless the corresponding flag for the button itself
// hasn't been set by its mouseover handler - i.e. we only want to hide the popup if the
// mouse is over neither the save button or the details popup...
inSaveDetails = false;
if(inSaveButton == false)
{
saveMouseOut();
}
}
function saveMouseOutDeferred()
{
// To account for the order in which the save button mouseout and details popup mouseover
// handlers fire (i.e. in the wrong order to make this fix easy...), the buttom mouseout
// handler is modified to perform most of its processing after a short delay - this gives
// the popup mouseover handler time to fire and set its flag, so that when we get here
// we know that the mouseout event was triggered by movement of the pointer between the
// button and popup, rather than out of the button in some other direction, and therefore
// we should avoid hiding the popup...
if((inSaveDetails == false) && (inSaveButton == false))
{
let styles = '.changes-log { display: none !important; }';
addStyle(prefix,styles);
}
}
function disableSaveBlocker()
{
let fname = 'disableSaveBlocker';
let styles = "";
if (getById('_cbDisableSaveBlocker').checked)
{
styles += '#popup-overlay { z-index: 0 !important; }';
styles += '.changes-log { display: none !important; }';
addStyle(prefix + fname,styles);
}
else
{
removeStyle(prefix + fname);
}
}
function disableUITransitions()
{
let fname = 'disableUITransitions';
let styles = "";
let sliderTrans = "";
// Side panel and main menu fixes we can apply directly via CSS mods...
if (getById('_cbDisableUITransitions').checked)
{
styles += '.collapsible-container { transition: none!important; }';
styles += '#issue-tracker-filter-region { transition: none!important; }';
styles += '.menu { transition: none!important; }';
addStyle(prefix + fname,styles);
sliderTrans = ".wz-slider::before {transition:all 0s linear 0s!important;}";
}
else
{
removeStyle(prefix + fname);
sliderTrans = ".wz-slider::before {transition:all 400ms ease 0s!important;}";
}
// And now the stuff hidden in shadowroots...
// WME adds transition styles both to the slider itself and also to its ::before
// pseudo-element, so we can't simply apply our own style override to the slider
// directly as this doesn't affect the ::before style. To get around this, we
// instead apply the override to the parent element as a new CSS entry to be
// applied to its slider children - this then takes precedence over anything
// defined on the slider itself...
let nTS = document.querySelectorAll('wz-toggle-switch').length;
while(nTS > 0)
{
let tsObj = document.querySelectorAll('wz-toggle-switch')[nTS - 1];
let sr = tsObj.shadowRoot.querySelector('.wz-switch');
if(sr !== null)
{
// If we haven't already set up our CSS entry on this parent element, do
// so now...
if(sr.querySelector('#fume') == null)
{
let sliderStyle = document.createElement('style');
sliderStyle.id = "fume";
sr.insertBefore(sliderStyle,sr.firstChild);
}
// Now we know the parent has our CSS entry, update its contents according to
// whether we're enabling or disabling transition effects
sr.querySelector('#fume').innerHTML = sliderTrans;
}
--nTS;
}
}
function colourBlindTurns()
{
let fname = 'colourBlindTurns';
let styles = "";
if (getById('_cbColourBlindTurns').checked)
{
styles += '.turn-arrow-state-open { filter: hue-rotate(90deg); }';
addStyle(prefix + fname,styles);
}
else
{
removeStyle(prefix + fname);
}
}
function unfloatButtons()
{
//// unfloat
return;
let fname = 'unfloatButtons';
layersButton = getByClass('layer-switcher-button')[0];
refreshButton = getByClass('reload-button')[0];
shareButton = getByClass('share-location-button')[0];
if (getById('_cbUnfloatButtons').checked)
{
unfloat();
//hacks for other scripts
if (getById('Info_div'))
{
getByClass('bottom overlay-buttons-container')[0].appendChild(getById('Info_div'));
getById('Info_div').style.marginTop = '8px';
}
if (getById('BeenHere')) getById('BeenHere').style.top = '310px';
//temporary hack for new button arrangements Map Nav Historic
if (getById('prevIcon')) insertNodeBeforeNode(getById('prevIcon').parentNode,getById('nextIcon').parentNode);
if ( wmeFUinitialising) setTimeout(unfloat,5000);
}
else
{
if (!wmeFUinitialising)
{
float();
layersButton.onmouseover = null;
document.body.onmouseleave = null;
getById('layer-switcher-region').onmouseleave = null;
removeStyle(prefix + fname);
if (getById('Info_div'))
{
getByClass('overlay-buttons-container top')[0].appendChild(getById('Info_div'));
getById('Info_div').style.marginTop = '';
}
if (getById('BeenHere')) getById('BeenHere').style.top = '280px';
}
}
}
function unfloat_ReloadClickHandler()
{
// Clicking on the refresh button essentially just calls the following native function,
// however once we do this then the bloody refresh button loses its onclick hander AGAIN,
// so we need to reinstate it before we go...
W.controller.reloadData();
refreshButton.addEventListener("click", unfloat_ReloadClickHandler);
}
function SLBRelocate()
{
// Once the mutation observer code has decided that the share location popup needs relocation,
// we first wait for it to become visible (usually this is true the first time we get in here,
// but sometimes WME will surprise us by getting here a tad too quickly for the popup).
//
// Once the popup exists, we first hide it (to avoid it briefly appearing in its native position
// before being relocated) and then we wait ever so slightly longer before trying to move it,
// otherwise WME seems to occasionally overwrite our position with the native one again...
let tippy = document.querySelector('.tippy-box');
if(tippy === null)
{
window.setTimeout(SLBRelocate, 100);
}
else
{
tippy.parentElement.style.visibility = "hidden";
window.setTimeout(SLBApplyTransform, 100);
}
}
function SLBApplyTransform()
{
// Finally we get to actually restyle the popup... This is simply a case of replacing the native
// transform (which nudges it a little to tne left and down relative to the top-right corner of the
// map viewport) with our own (which nudges it a little less to the left, but a lot further down,
// based on how far away the share location button is). And then remembering to make it visible
// again, so the user sees it appearing in the desired position as if it was always meant to be :-)
let tippy = document.querySelector('.tippy-box').parentElement;
let tipBCR = tippy.getBoundingClientRect();
let slbBCR = document.querySelector('.share-location-button').getBoundingClientRect();
let tY = (slbBCR.top - tipBCR.bottom) + "px";
tippy.style.transform = "translate(-20px, " + tY + ")";
tippy.style.visibility = "";
}
var OBObserver = new MutationObserver(function()
{
// To check when the share location button popup displays, you'd think we could just use an onclick handler
// on the share location button itself, but no, that would be FAR too easy... Whilst setting an onclick
// does work the first time, the way WME re-renders the button to change it between its selected and
// unselected states causes the handler to be lost, and I couldn't figure out how to get it reapplied
// reliably/efficiently.
//
// So here we are with plan B - a mutation observer attached to the DOM element that contains the button,
// within which we can then check firstly to see if the button has been moved into the bottom bar (if
// it's in its native location then we don't need to relocate the popup), and then if the button is
// active. If it is, then we know the popup is visible and in need of relocation...
/*
//// unfloat
if (getById('_cbUnfloatButtons').checked === true)
{
if(document.querySelector('.share-location-button.overlay-button.overlay-button-active') !== null)
{
SLBRelocate();
}
}
*/
// WME being WME, we also have to deal with it sometimes adding a copy of the share location button, which
// sits atop the original so that visually all looks the same, however this copy doesn't have all of the
// same onclick handlers setup, so clicking on it doesn't actually cause the popup to appear. One such way
// to provoke this second button to be generated is if it's been clicked on at least once, and you then
// enter and leave house number editing mode.
//
// As having this slightly broken button blocking access to the actual button is a bit of an embuggerance,
// we make sure that any time we detect multiple copies appearing in the DOM, we nuke all of the
// imposters, leaving the original available for clicky enjoyment once again.
//
// Have I ever mentioned how much I love what the devs have done with WME? I mean, having to deal with
// random crap like this really, truly, without any question, utterly and completely MAKES my day...
let nSLB = document.querySelectorAll('.share-location-button.overlay-button').length;
if(nSLB > 1)
{
while(nSLB > 1)
{
--nSLB;
document.querySelectorAll('.share-location-button.overlay-button')[nSLB].remove();
}
}
});
function unfloat ()
{
//// unfloat
return;
if (getById('_cbUnfloatButtons').checked === true)
{
let slb = getByClass('share-location-button')[0];
let wcp = getByClass('WazeControlPermalink')[0];
if((slb !== undefined) && (wcp !== undefined))
{
// as we may end up calling this function multiple times, we first refloat so that any changes
// made here will always be applied to the default styles rather than any we've already changed
float();
if(getById('_cbMoveUserInfo').checked === false)
{
insertNodeAfterNode(layersButton, document.querySelector('wz-user-box'));
}
else
{
insertNodeAfterNode(layersButton,getById('save-button').parentElement.parentElement);
}
layersButton.classList.add('toolbar-button');
layersButton.firstChild.classList.add('item-container');
layersButton.firstChild.classList.add('item-icon','w-icon','w-icon-layers');
insertNodeBeforeNode(refreshButton, document.querySelector('.secondary-toolbar'));
refreshButton.classList.add('toolbar-button');
refreshButton.firstChild.classList.add('item-container');
refreshButton.firstChild.classList.add('item-icon','w-icon','w-icon-refresh');
// Something's changed in the latest iteration of WME which means that moving the refresh button
// stops it accepting mouse clicks, so we need to set up a new onclick handler to replicate the
// desired behaviour...
refreshButton.addEventListener("click",unfloat_ReloadClickHandler, {once: true});
var lmBCR = wcp.getBoundingClientRect();
var sbBCR = slb.getBoundingClientRect();
var sbTop = lmBCR.top - sbBCR.top - 3;
var styles = '';
styles += '.share-location-button { position: absolute; top: '+sbTop+'px; height: 18px; }';
styles += '#edit-buttons .overlay-button-disabled { opacity: 0.5; cursor: not-allowed; }';
styles += '.share-location-button-region { display: inline-block; }';
styles += '.w-icon-layers { top: 0px!important; }';
styles += 'div.WazeControlPermalink { padding-right: 64px; }';
styles += 'div.share-location-button-region > div > div > i { line-height: 18px; }';
styles += 'a.w-icon.w-icon-link { line-height:17px; font-size: 20px; }';
// correct button sizing when moved into bottom bar
styles += '.share-location-button { height:24px; width:30px; }';
addStyle(prefix + 'unfloatButtons2',styles);
}
}
}
function float ()
{
// temporary
return;
let elm = getByClass('overlay-buttons-container top')[0];
if(elm !== undefined)
{
elm.appendChild(layersButton);
layersButton.classList.remove('toolbar-button');
layersButton.firstChild.classList.remove('item-container');
layersButton.firstChild.classList.remove('item-icon','w-icon','w-icon-layers');
layersButton.firstChild.classList.add('overlay-button');
layersButton.firstChild.classList.add('w-icon','w-icon-layers');
elm.appendChild(refreshButton);
refreshButton.classList.remove('toolbar-button');
refreshButton.firstChild.classList.remove('item-container');
refreshButton.firstChild.classList.remove('item-icon','w-icon','w-icon-refresh');
refreshButton.firstChild.classList.add('overlay-button');
refreshButton.firstChild.classList.add('w-icon','w-icon-refresh');
elm.appendChild(shareButton);
removeStyle(prefix + 'unfloatButtons2');
}
}
function hackGSVHandle ()
{
let fname = 'hackGSVHandle';
let styles = "";
if (getById('_cbHackGSVHandle').checked)
{
styles += '#editor-container #map.street-view-mode #street-view-drag-handle { height: 29px; background: lightgrey; font-size: 24px; border-radius: 8px; text-align: center; padding-top: 2px; border: 1px black solid; }';
addStyle(prefix + fname,styles);
getById('street-view-drag-handle').classList.add('w-icon','w-icon-round-trip');
getById('street-view-drag-handle').title = 'Double-click to reset\ndefault width.';
}
else
{
removeStyle(prefix + fname);
getById('street-view-drag-handle').removeAttribute('class');
getById('street-view-drag-handle').removeAttribute('title');
}
}
function enlargeGeoNodes (forceOff)
{
let fname = 'enlargeGeoNodes';
removeStyle(prefix + fname);
let styles = "";
if (getById('_inpEnlargeGeoNodes').value < 6) getById('_inpEnlargeGeoNodes').value = 6;
if ((getById('_cbEnlargeGeoNodes').checked) && (forceOff === false))
{
let newStyle = '{ transform-box: fill-box; transform-origin: center; vector-effect: non-scaling-stroke; transform:scale(' + (getById('_inpEnlargeGeoNodes').value / 6) + '); }';
styles += '#' + W.map.getLayerByUniqueName('sketch').id + '_vroot [id^="OpenLayers_Geometry_Point_"][fill-opacity="1"][r="6"] ' + newStyle;
styles += '#' + W.map.venueLayer.id + '_vroot [id^="OpenLayers_Geometry_Point_"][fill-opacity="1"][fill="white"][stroke-opacity="1"][r="6"] ' + newStyle;
styles += '#' + W.map.commentLayer.id + '_vroot [id^="OpenLayers_Geometry_Point_"][fill-opacity="1"][fill="white"][stroke-opacity="1"][r="6"] ' + newStyle;
addStyle(prefix + fname,styles);
}
}
function enlargeGeoHandles (forceOff)
{
let fname = 'enlargeGeoHandles';
removeStyle(prefix + fname);
let styles = "";
if (getById('_inpEnlargeGeoHandles').value < 4) getById('_inpEnlargeGeoHandles').value = 4;
if ((getById('_cbEnlargeGeoHandlesFU').checked) && (forceOff === false))
{
let newStyle = '{ transform-box: fill-box; transform-origin: center; vector-effect: non-scaling-stroke; transform:scale(' + (getById('_inpEnlargeGeoHandles').value / 4) + '); }';
styles += '#' + W.map.getLayerByUniqueName('sketch').id + '_vroot [id^="OpenLayers_Geometry_Point_"][fill-opacity="0.6"][r="4"] ' + newStyle;
styles += '#' + W.map.venueLayer.id + '_vroot [id^="OpenLayers_Geometry_Point_"][fill-opacity="0.6"][stroke-opacity="1"][r="4"]' + newStyle;
styles += '#' + W.map.commentLayer.id + '_vroot [id^="OpenLayers_Geometry_Point_"][fill-opacity="0.6"][stroke-opacity="1"][r="4"]' + newStyle;
addStyle(prefix + fname,styles);
}
}
function enlargePointMCs()
{
let fname = 'enlargePointMCs';
removeStyle(prefix + fname);
let styles = "";
if (getById('_inpEnlargePointMCs').value < 1) getById('_inpEnlargePointMCs').value = 1;
if (getById('_cbEnlargePointMCs').checked)
{
let newStyle = '{ fill: #ffff00; fill-opacity: 0.75; transform-box: fill-box; transform-origin: center; vector-effect: non-scaling-stroke; transform:scale(' + getById('_inpEnlargePointMCs').value + '); }';
let newStyleHover = '{ transform-box: fill-box; transform-origin: center; vector-effect: non-scaling-stroke; transform:scale(' + (0.25 + (getById('_inpEnlargePointMCs').value / 2)) + '); }';
styles += '#' + W.map.commentLayer.id + '_vroot [id^="OpenLayers_Geometry_Point_"][stroke="#ffffff"][r="6"]' + newStyle;
styles += '#' + W.map.commentLayer.id + '_vroot [id^="OpenLayers_Geometry_Point_"][stroke="#ffffff"][r="12"]' + newStyleHover;
addStyle(prefix + fname,styles);
}
}
function addGlobalStyle(css)
{
var head, style;
head = document.getElementsByTagName('head')[0];
if (!head)
{
return;
}
style = document.createElement('style');
style.type = 'text/css';
style.innerHTML = modifyHTML(css);
head.appendChild(style);
}
function addStyle(ID, css)
{
var head, style;
head = document.getElementsByTagName('head')[0];
if (!head)
{
return;
}
removeStyle(ID); // in case it is already there
style = document.createElement('style');
style.type = 'text/css';
style.innerHTML = modifyHTML(css);
style.id = ID;
head.appendChild(style);
}
function removeStyle(ID)
{
var style = document.getElementById(ID);
if (style)
{
style.parentNode.removeChild(style);
}
}
function getByClass(classname, node)
{
if(!node)
{
node = document;
}
let retval = node.querySelector('.' + classname);
return retval;
}
function getById(node)
{
return document.querySelector('#' + node);
}
function insertNodeBeforeNode (insertNode, beforeNode)
{
if((insertNode == null) || (beforeNode == null))
{
logit("null node during insert","error");
}
else
{
beforeNode.parentNode.insertBefore(insertNode,beforeNode);
}
}
function insertNodeAfterNode (insertNode, afterNode)
{
insertNodeBeforeNode (insertNode, afterNode);
insertNodeBeforeNode (afterNode,insertNode);
}
function logit(msg, typ)
{
if (!typ)
{
console.log(prefix + ": " + msg);
}
else
{
switch(typ)
{
case "error":
console.error(prefix + ": " + msg);
break;
case "warning":
console.warn(prefix + ": " + msg);
break;
case "info":
console.info(prefix + ": " + msg);
break;
case "debug":
if (debug)
{
console.warn(prefix + ": " + msg);
}
break;
default:
console.log(prefix + " unknown message type: " + msg);
break;
}
}
}
// Start it running
setTimeout(init1, 200);
})();