Provide access to all closure details within the closures list
// ==UserScript==
// @name WME Closure Details
// @namespace http://www.tomputtemans.com/
// @description Provide access to all closure details within the closures list
// @include /^https:\/\/(www|beta)\.waze\.com\/(?!user\/)(.{2,6}\/)?editor.*$/
// @icon 
// @version 1.3.0
// @grant none
// ==/UserScript==
/* global W, $, I18n */
var styleElement; // Style element to reuse whenever it gets removed by the WME (user login, for example)
async function onWmeReady(e) {
if (e && e.user === null) {
return;
}
if (!W.loginManager.user) {
W.loginManager.events.register('login', null, onWmeReady);
W.loginManager.events.register('loginStatus', null, onWmeReady);
// Double check as event might have triggered already
if (!W.loginManager.user) {
return;
}
}
console.log('Initialize script');
observeContentsPane();
applyStyles();
}
function observeContentsPane() {
function handleMutations(mutations) {
mutations.forEach(function(mutation) {
var closureBlocks = mutation.target.querySelectorAll('.closure-item');
var selectedIDs = W.selectionManager.getSelectedFeatures().filter(function(obj) {
return obj.layer?.name == 'segments';
}).map(function(obj) {
return obj.attributes?.wazeFeature?.id;
});
var selectedClosures = W.model.roadClosures.getObjectArray().filter(function(closure) {
return selectedIDs.indexOf(closure.attributes.segID) != -1;
});
for (var i = 0; i < closureBlocks.length; i++) {
let closureBlock = closureBlocks[i];
closureBlock.addEventListener('click', removeAllTooltips);
if (closureBlock.querySelector('.menu-initiator')) { // To be replaced with optional chaining somewhere in the future (https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Operators/Optional_chaining)
closureBlock.querySelector('.menu-initiator').addEventListener('click', removeAllTooltips);
}
var buttons = closureBlock.querySelectorAll('a');
for (var j = 0; j < buttons.length; j++) {
buttons[j].addEventListener('click', removeAllTooltips);
}
var matchedClosure = selectedClosures.find(function(closure) {
return getLocalizedTime(closure.attributes.startDate) == getTimeFromBlock(closureBlock.querySelector('.start-date')) &&
getLocalizedTime(closure.attributes.endDate) == getTimeFromBlock(closureBlock.querySelector('.end-date'));
});
if (matchedClosure) {
var description;
if (matchedClosure.attributes.reason !== '') {
description = '<strong class="description">' + matchedClosure.attributes.reason + '</strong>';
} else {
description = '<em class="description">No description set</em>';
}
description += '<table class="details"><tbody>';
if (matchedClosure.attributes.provider) {
description += '<tr><th>Provided by</th><td>' + matchedClosure.attributes.provider + '</td></tr>';
}
description += '<tr><th>Created by</th><td>' + getUsername(matchedClosure.attributes.createdBy) + '</td></tr><tr><th>Created on</th><td>' + I18n.l('time.formats.long', matchedClosure.attributes.createdOn) + '</td></tr>';
if (matchedClosure.attributes.updatedBy) {
description += '<tr><th>Updated by</th><td>' + getUsername(matchedClosure.attributes.updatedBy) + '</td></tr>';
}
if (matchedClosure.attributes.updatedOn) {
description += '<tr><th>Updated on</th><td>' + I18n.l('time.formats.long', matchedClosure.attributes.updatedOn) + '</td></tr>';
}
description += '<tr><td colspan="2" style="text-align:center"><em>' + (matchedClosure.attributes.permanent ? 'Ignores traffic' : 'Listens to traffic') + '</em></td></tr></tbody></table></div>';
$(closureBlock).tooltip({
placement: 'right',
trigger: 'hover',
html: true,
title: description
});
}
}
});
if (document.querySelector('.contents .closures')) {
(new MutationObserver(handleMutations)).observe(document.querySelector('.contents .closures'), {
childList: true
});
}
}
(new MutationObserver(handleMutations)).observe(document.querySelector('.contents'), {
childList: true
});
}
function removeAllTooltips() {
var tooltips = document.querySelectorAll('.tooltip');
if (tooltips.length > 0) {
for (var tooltip of tooltips) {
tooltip.parentNode.removeChild(tooltip);
}
}
}
function getTimeFromBlock(node) {
return node.querySelector('.date').textContent + ' ' + node.querySelector('.time').textContent;
}
function getLocalizedTime(date) {
var splitDate = date.split(' ');
return I18n.l('date.formats.default', splitDate[0]) + ' ' + splitDate[1];
}
function getUsername(id) {
var user = W.model.users.getObjectById(id);
if (user) {
return user.attributes.userName;
} else {
return id + ' (user not loaded)';
}
}
function applyStyles() {
if (!styleElement) {
styleElement = document.createElement('style');
styleElement.textContent = `
div.tooltip {
z-index: 998 !important;
}
.tooltip .tooltip-inner {
width: 250px;
max-width: 400px;
}
.tooltip-inner .description {
font-size: 120%;
}
.tooltip-inner .details {
border-width: 0;
text-align: left;
width: 100%;
}
.tooltip-inner .details th {
text-align: right;
padding-right: 0.4em;
color: #fff;
}
`;
}
if (!styleElement.parentNode) {
document.head.appendChild(styleElement);
}
}
function onWmeInitialized() {
if (W.userscripts?.state?.isReady) {
console.log('W is ready and in "wme-ready" state. Proceeding with initialization.');
onWmeReady();
} else {
console.log('W is ready, but not in "wme-ready" state. Adding event listener.');
document.addEventListener('wme-ready', onWmeReady, { once: true });
}
}
function bootstrap() {
if (!W) {
console.log('W is not available. Adding event listener.');
document.addEventListener('wme-initialized', onWmeInitialized, { once: true });
} else {
onWmeInitialized();
}
}
bootstrap();