您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Shows road closures (and comments) from Waze Live map in WME
// ==UserScript== // @name WME LiveMap Closures (phuz) // @description Shows road closures (and comments) from Waze Live map in WME // @include https://www.waze.com/editor* // @include https://www.waze.com/*/editor* // @include https://beta.waze.com/* // @exclude https://www.waze.com/*user/editor* // @version 1.16.17 // @namespace https://gf.qytechs.cn/en/users/668704-phuz // @require https://cdnjs.cloudflare.com/ajax/libs/moment.js/2.29.1/moment.min.js // @grant GM_info // @grant GM_addStyle // ==/UserScript== /* global OpenLayers */ /* global W */ /* global I18n */ let rtcCommentLayer; let myCSS = `#rtcCommentContainer { position: absolute; padding: 4em; background: lightgray; border: 1px double black; border-radius: 1ex; z-index: 777; display: block; } table.rtcCommentTable td { padding: 4px; } table.rtcCommentTable th { padding: 4px; } #mydivheader { cursor: move; z-index: 777; position: sticky; background-color: #2f2f2f; color: #FFFFFF; } .modalclose { background: lightgray; z-index: 800; color: #FFFFFF; line-height: 25px; position: absolute; right: -12px; text-align: center; top: -10px; width: 24px; text-decoration: none; font-weight: bold; -webkit-border-radius: 12px; -moz-border-radius: 12px; border-radius: 12px; -moz-box-shadow: 1px 1px 3px #000; -webkit-box-shadow: 1px 1px 3px #000; box-shadow: 1px 1px 3px #000; } .modalclose:hover { background: #00d9ff; text-decoration: none; } hr.myhrline{ margin: 5px; } ` var epsg900913; var epsg4326; var closuresLayer; var uOpenLayers; var uWaze; var lineWidth = [ [4, 5], [5, 6], [6, 7], [7, 8], [8, 9], [10, 12], [12, 14], [14, 16], [15, 17], [16, 18], [17, 19] ]; function drawLine(line) { var linePoints = []; var zoom = uWaze.map.getZoom() - 12; if (zoom >= lineWidth.length) { zoom = lineWidth.length - 1; } var p = new uOpenLayers.Geometry.Point(line[0].x, line[0].y).transform(epsg4326, epsg900913); linePoints.push(p); for (var i = 1; i < line.length - 1; i++) { var lp1 = line[i]; var lp2 = line[i + 1]; var dif_lon = Math.abs(lp1.x - lp2.x); var dif_lat = Math.abs(lp1.y - lp2.y); if (dif_lon < 0.0000001 && dif_lat < 0.0000001) continue; p = new uOpenLayers.Geometry.Point(lp1.x, lp1.y).transform(epsg4326, epsg900913); linePoints.push(p); } p = new uOpenLayers.Geometry.Point(line[line.length - 1].x, line[line.length - 1].y).transform(epsg4326, epsg900913); linePoints.push(p); var lineString = new uOpenLayers.Geometry.LineString(linePoints); var lineFeature = new uOpenLayers.Feature.Vector(lineString, null, { strokeColor: '#000000', strokeDashstyle: 'solid', strokeLinecap: 'round', strokeWidth: lineWidth[zoom][1] }); closuresLayer.addFeatures(lineFeature); lineString = new uOpenLayers.Geometry.LineString(linePoints); lineFeature = new uOpenLayers.Feature.Vector(lineString, null, { strokeColor: '#FF0000', strokeDashstyle: 'solid', strokeLinecap: 'round', strokeWidth: lineWidth[zoom][0] }); closuresLayer.addFeatures(lineFeature); lineString = new uOpenLayers.Geometry.LineString(linePoints); lineFeature = new uOpenLayers.Feature.Vector(lineString, null, { strokeColor: '#FFFFFF', strokeDashstyle: 'dot', strokeLinecap: 'square', strokeWidth: lineWidth[zoom][0] }); closuresLayer.addFeatures(lineFeature); } function getRoutingURL() { var server; if (typeof (uWaze.location) === 'undefined') { server = uWaze.app.getAppRegionCode(); } else { server = uWaze.location.code; } var routingURL = 'https://www.waze.com'; if (~document.URL.indexOf('https://beta.waze.com')) { routingURL = 'https://beta.waze.com'; } switch (server) { case 'usa': routingURL += '/rtserver/web/TGeoRSS'; break; case 'row': routingURL += '/row-rtserver/web/TGeoRSS'; break; case 'il': routingURL += '/il-rtserver/web/TGeoRSS'; break; default: routingURL += '/rtserver/web/TGeoRSS'; } return routingURL; } function requestClosures() { var zoom = uWaze.map.getZoom() - 12; if (zoom >= 0) { if (closuresLayer.getVisibility()) { var extent = uWaze.map.getExtent(); var oh = 500; var pLB = new uOpenLayers.Geometry.Point(extent.left - oh, extent.bottom - oh).transform(epsg900913, epsg4326); var pRT = new uOpenLayers.Geometry.Point(extent.right + oh, extent.top + oh).transform(epsg900913, epsg4326); var data = { ma: "600", mj: "100", mu: "100", types: "traffic,alerts", left: pLB.x, right: pRT.x, bottom: pLB.y, top: pRT.y }; var url = getRoutingURL(); $.ajax({ dataType: "json", url: url, data: data, success: function (json) { if (json.error != undefined) { } else { if (W.map.getLayersByName('rtcCommentLayer').length >= 1) { rtcCommentLayer.clearMarkers(); } closuresLayer.destroyFeatures(); var ids = []; if ("undefined" !== typeof (json.jams)) { var numjams = json.jams.length; var numAlerts = 0; if (json.alerts) { numAlerts = json.alerts.length; } for (var i = 0; i < numjams; i++) { var jam = json.jams[i]; if (jam.delay === -1) { drawLine(jam.line); for (var j = 0; j < numAlerts; j++) { var alerts = json.alerts[j]; if (alerts.uuid == jam.blockingAlertUuid) { if (alerts.comments) { let hasText = false; let comment = [] let timestamp = []; let user = []; for (var k = 0; k < alerts.comments.length; k++) { if (alerts.comments[k].isThumbsUp == false) { comment.push(alerts.comments[k].text); timestamp.push(alerts.comments[k].reportMillis); user.push(alerts.comments[k].reportBy); hasText = true; //build the comment history } } if (hasText) { let x = jam.line[Math.trunc(jam.line.length / 2)].x; let y = jam.line[Math.trunc(jam.line.length / 2)].y; drawCommentMarker(alerts.reportDescription, comment, timestamp, x, y, user); } } } } } } } } } }); } } } function changeLayer() { localStorage.DrawLiveMapClosures = closuresLayer.getVisibility(); requestClosures(); } function liveMapClosures_init() { closuresLayer = new uOpenLayers.Layer.Vector("LiveMap closures", { displayInLayerSwitcher: true, uniqueName: "__DrawLiveMapClosures" }); uWaze.map.addLayer(closuresLayer); W.map.getOLMap().setLayerIndex(closuresLayer, 10); if (localStorage.DrawLiveMapClosures) { closuresLayer.setVisibility(localStorage.DrawLiveMapClosures == "true"); } else { closuresLayer.setVisibility(true); } var roadGroupSelector = document.getElementById('layer-switcher-group_road'); if (roadGroupSelector != null) { var roadGroup = roadGroupSelector.parentNode.parentNode.getElementsByTagName("UL")[0]; var toggler = document.createElement('li'); var checkbox = document.createElement("wz-checkbox"); checkbox.id = 'layer-switcher-item_livemap_closures'; checkbox.className = "hydrated"; checkbox.disabled = !roadGroupSelector.checked; checkbox.checked = closuresLayer.getVisibility(); checkbox.appendChild(document.createTextNode("LiveMap closures")); toggler.appendChild(checkbox); roadGroup.appendChild(toggler); checkbox.addEventListener('click', function (e) { closuresLayer.setVisibility(e.target.checked); }); roadGroupSelector.addEventListener('click', function (e) { closuresLayer.setVisibility(e.target.checked && checkbox.checked); checkbox.disabled = !e.target.checked; }); } var alertsLayer = uWaze.map.getLayerByUniqueName('__livemap_alerts'); if (typeof (alertsLayer) !== "undefined") { var closuresLayerZIdx = closuresLayer.getZIndex(); var alertsLayerZIdx = alertsLayer.getZIndex(); if (closuresLayerZIdx > alertsLayerZIdx) { closuresLayer.setZIndex(alertsLayerZIdx); alertsLayer.setZIndex(closuresLayerZIdx); } } uWaze.map.events.register("zoomend", null, requestClosures); uWaze.map.events.register("moveend", null, requestClosures); uWaze.map.events.register("changelayer", null, changeLayer); requestClosures(); } function liveMapClosures_bootstrap() { uWaze = unsafeWindow.W; uOpenLayers = unsafeWindow.OpenLayers; if (typeof (uOpenLayers) === 'undefined' || typeof (uWaze) === 'undefined' || typeof (uWaze.map) === 'undefined' || document.querySelector('.list-unstyled.togglers .group') === null) { setTimeout(liveMapClosures_bootstrap, 500); } else { epsg900913 = new uOpenLayers.Projection("EPSG:900913"); epsg4326 = new uOpenLayers.Projection("EPSG:4326"); if (!OpenLayers.Icon) { installIcon(); } rtcCommentLayer = new OpenLayers.Layer.Markers('rtcCommentLayer'); W.map.addLayer(rtcCommentLayer); GM_addStyle(myCSS); liveMapClosures_init(); } } //Generate the Advisory markers function drawCommentMarker(title, comments, datetime, x, y, user) { let commentWhite = ''; let commentGreen = ''; let commentYellow = ''; let commentRed = ''; let commentIcon; let lastCommentTime = moment(new Date(parseInt(datetime[datetime.length - 1])), "DD.MM.YYYY").startOf('day'); let timeNow = moment(new Date(Date.now()), "DD.MM.YYYY").startOf('day'); let daysSinceLastMessage = timeNow.diff(lastCommentTime, 'days'); if (daysSinceLastMessage < 4) { commentIcon = commentGreen; } if (daysSinceLastMessage >= 4) { commentIcon = commentYellow; } if (daysSinceLastMessage >= 10) { commentIcon = commentRed; } var size = new OpenLayers.Size(30, 26); var offset = new OpenLayers.Pixel(size.w * 0.5, -size.h); var icon = new OpenLayers.Icon(commentIcon, size, offset); var epsg4326 = new OpenLayers.Projection("EPSG:4326"); //WGS 1984 projection var projectTo = W.map.getProjectionObject(); //The map projection (Spherical Mercator) var lonLat = new OpenLayers.LonLat(x, y).transform(epsg4326, projectTo); var newMarker = new OpenLayers.Marker(lonLat, icon); newMarker.title = title; newMarker.comments = comments; newMarker.timestamp = datetime; newMarker.user = user; newMarker.location = lonLat; newMarker.events.register('click', newMarker, popup); rtcCommentLayer.addMarker(newMarker); } //Generate the Popup function popup(evt) { $("#rtcCommentContainer").remove(); $("#rtcCommentContainer").hide(); var popupHTML; W.map.moveTo(this.location); let user; let htmlString = '<div id="rtcCommentContainer" style="max-width:500px;margin: 1;text-align: center;padding: 5px;z-index: 1100">' + '<a href="#close" id="gmCloseDlgBtn" title="Close" class="modalclose" style="color:#FF0000;">X</a>' + '<table border=1 class="rtcCommentTable"><tr><td colspan=3><div id="mydivheader" style="min-height: 20px;">' + this.title + '</div></td></tr>' htmlString += '<tr><th>Date / Time</th><th>Comment</th><th>By</th>'; for (let i = 0; i < this.comments.length; i++) { if (this.user[i]) { user = '<a target="_blank" href="https://www.waze.com/user/editor/' + this.user[i] + '">' + this.user[i] + '</a>'; } else { user = "<font color=red>Unknown</font>"; } htmlString += '<tr><td width=200 align=right>' + moment(new Date(this.timestamp[i])).format('LLL') + '</td><td align=left>' + this.comments[i] + '</td><td align=center>' + user + '</td></tr>'; } htmlString += '</table></div>' //moment(new Date(this.timestamp[i])).format('LLL') popupHTML = ([htmlString]); $("body").append(popupHTML); //Position the modal based on the position of the click event $("#rtcCommentContainer").css({ left: document.getElementById("user-tabs").offsetWidth + W.map.getPixelFromLonLat(W.map.getCenter()).x - document.getElementById("rtcCommentContainer").clientWidth - 10 }); $("#rtcCommentContainer").css({ top: document.getElementById("left-app-head").offsetHeight + W.map.getPixelFromLonLat(W.map.getCenter()).y - (document.getElementById("rtcCommentContainer").clientHeight / 2) }); $("#rtcCommentContainer").show(); //Add listener for popup's "Close" button $("#gmCloseDlgBtn").click(function () { $("#rtcCommentContainer").remove(); $("#rtcCommentContainer").hide(); }); dragElement(document.getElementById("rtcCommentContainer")); } function dragElement(elmnt) { var pos1 = 0, pos2 = 0, pos3 = 0, pos4 = 0; if (document.getElementById("mydivheader")) { // if present, the header is where you move the DIV from: document.getElementById("mydivheader").onmousedown = dragMouseDown; } else { // otherwise, move the DIV from anywhere inside the DIV: elmnt.onmousedown = dragMouseDown; } function dragMouseDown(e) { e = e || window.event; e.preventDefault(); // get the mouse cursor position at startup: pos3 = e.clientX; pos4 = e.clientY; document.onmouseup = closeDragElement; // call a function whenever the cursor moves: document.onmousemove = elementDrag; } function elementDrag(e) { e = e || window.event; e.preventDefault(); // calculate the new cursor position: pos1 = pos3 - e.clientX; pos2 = pos4 - e.clientY; pos3 = e.clientX; pos4 = e.clientY; // set the element's new position: elmnt.style.top = (elmnt.offsetTop - pos2) + "px"; elmnt.style.left = (elmnt.offsetLeft - pos1) + "px"; } function closeDragElement() { // stop moving when mouse button is released: document.onmouseup = null; document.onmousemove = null; } } function installIcon() { console.log('Installing OpenLayers.Icon'); OpenLayers.Icon = OpenLayers.Class({ url: null, size: null, offset: null, calculateOffset: null, imageDiv: null, px: null, initialize: function (a, b, c, d) { this.url = a; this.size = b || { w: 20, h: 20 }; this.offset = c || { x: -(this.size.w / 2), y: -(this.size.h / 2) }; this.calculateOffset = d; a = OpenLayers.Util.createUniqueID("OL_Icon_"); let div = this.imageDiv = OpenLayers.Util.createAlphaImageDiv(a); $(div.firstChild).removeClass('olAlphaImg'); // LEAVE THIS LINE TO PREVENT WME-HARDHATS SCRIPT FROM TURNING ALL ICONS INTO HARDHAT WAZERS --MAPOMATIC }, destroy: function () { this.erase(); OpenLayers.Event.stopObservingElement(this.imageDiv.firstChild); this.imageDiv.innerHTML = ""; this.imageDiv = null; }, clone: function () { return new OpenLayers.Icon(this.url, this.size, this.offset, this.calculateOffset); }, setSize: function (a) { null !== a && (this.size = a); this.draw(); }, setUrl: function (a) { null !== a && (this.url = a); this.draw(); }, draw: function (a) { OpenLayers.Util.modifyAlphaImageDiv(this.imageDiv, null, null, this.size, this.url, "absolute"); this.moveTo(a); return this.imageDiv; }, erase: function () { null !== this.imageDiv && null !== this.imageDiv.parentNode && OpenLayers.Element.remove(this.imageDiv); }, setOpacity: function (a) { OpenLayers.Util.modifyAlphaImageDiv(this.imageDiv, null, null, null, null, null, null, null, a); }, moveTo: function (a) { null !== a && (this.px = a); null !== this.imageDiv && (null === this.px ? this.display(!1) : ( this.calculateOffset && (this.offset = this.calculateOffset(this.size)), OpenLayers.Util.modifyAlphaImageDiv(this.imageDiv, null, { x: this.px.x + this.offset.x, y: this.px.y + this.offset.y }) )); }, display: function (a) { this.imageDiv.style.display = a ? "" : "none"; }, isDrawn: function () { return this.imageDiv && this.imageDiv.parentNode && 11 != this.imageDiv.parentNode.nodeType; }, CLASS_NAME: "OpenLayers.Icon" }); } liveMapClosures_bootstrap();
QingJ © 2025
镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址