您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
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();
QingJ © 2025
镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址