WME LiveMap Alerts Overlay

Overlay alerts from the Waze LiveMap

Versão de: 16/06/2020. Veja: a última versão.

// ==UserScript==
// @name                WME LiveMap Alerts Overlay
// @author		davielde, MajkiiTelini
// @description         Overlay alerts from the Waze LiveMap
// @include             /^https:\/\/(www|beta)\.waze\.com\/(?!user\/)(.{2,6}\/)?editor.*$/
// @copyright           Many thanks to Timbones and Twister-UK, whose code (WMECH and URO+) provided a large foundation for this work
// @version             0.5.19
// @namespace           https://greasyforks.org/users/110192
// ==/UserScript==

var W;
var OL;
var I18n;
var LiveMapAlerts_Layer;

initializeLiveMapAlerts();

function checkLayerNum()
{
	var lmaoLayer = null;
	for(var i = 0; i < W.map.getLayers().length; i++)
	{
		if(W.map.getLayers()[i].uniqueName == '__livemap_alerts') lmaoLayer = i;
	}
	return lmaoLayer;
}

function getBounds()
{
	var alertBounds = W.map.getExtent();
	alertBounds.transform(new OL.Projection("EPSG:900913"),new OL.Projection("EPSG:4326"));
	return alertBounds;
}

function getRoutingServer(){
	var server = W.app.getAppRegionCode();
	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 getFeatureYOffset(){
	var yOffset = -30;
	return yOffset;
}

function getLiveMapAlerts(){
	if (!LiveMapAlerts_Layer.getVisibility()) {
		return;
	}
	var alertBounds = getBounds();
	var url = getRoutingServer();
	var data = {
		left: alertBounds.left,
		right: alertBounds.right,
		bottom: alertBounds.top,
		top: alertBounds.bottom,
		types: "alerts"
	};
	$.ajax({
		dataType: "json",
		url: url,
		data: data,
		success: function(json) {
			LiveMapAlerts_Layer.destroyFeatures();
			var alertData = json.alerts;
			try {
				for(var i = 0; i < alertData.length; i++) {
					var lat = alertData[i].location.y;
					var long = alertData[i].location.x;
					var type = alertData[i].type;
					var subType = alertData[i].subtype;
					var title = alertData[i].reportDescription;
					addImage(lat, long, type, subType, alertData[i]);
				}
			}
			catch(e) {
			}
		}
	});
}

function addImage(lat, long, type, subType, detail) {
	var coords = OL.Layer.SphericalMercator.forwardMercator(long, lat);
	var point = new OL.Geometry.Point(coords.lon,coords.lat);
	var alertPx = W.map.getPixelFromLonLat(new OL.LonLat(coords.lon,coords.lat));
	var imgRoot = '/assets';
	var icon;
	switch(type){
		case 'ROAD_CLOSED':
			icon = '';break;
		case 'ACCIDENT':
			icon = '';break;
		case 'JAM':
			switch(subType){
				case 'JAM_HEAVY_TRAFFIC':
					icon = '';break;
				case 'JAM_MODERATE_TRAFFIC':
					icon = '';break;
				default:
					icon = '';
			}
			break;
		case 'POLICE':
			icon = '';break;
		case 'HAZARD':
			switch(subType){
				case 'HAZARD_ON_ROAD_CONSTRUCTION':
					icon = '';break;
				default:
					icon = '';
			}
			break;
		case 'CHIT_CHAT':
			icon = '';break;
		default:
			icon = '';
	}
	var attributes = {
		type: detail.type,
		subtype: detail.subtype,
		description: detail.reportDescription,
		street: detail.street,
		city: detail.city,
		near: detail.nearBy,
		reportby: detail.reportBy,
		pubMillis: detail.pubMillis,
		pixel: alertPx
	};
	var alertAgeInMinutes = getAlertAge(detail.pubMillis);
	var alertTransparency;
	if(type == 'ROAD_CLOSED'){
		alertTransparency = 1;
	}
	else if(alertAgeInMinutes < 15){
		alertTransparency = 1;
	}
	else if(alertAgeInMinutes >= 15 && alertAgeInMinutes < 30){
		alertTransparency = 0.8;
	}
	else if(alertAgeInMinutes >= 30 && alertAgeInMinutes < 45){
		alertTransparency = 0.5;
	}
	else if(alertAgeInMinutes >= 45){
		alertTransparency = 0.25;
	}
	else{
		alertTransparency = 1; //"unknown" age
	}
	var style = {
		externalGraphic: icon,
		graphicWidth: 32,
		graphicHeight: 38,
		graphicYOffset: getFeatureYOffset(),
		fillOpacity: alertTransparency,
		title: 'LiveMap',
		cursor: 'help'
	};
	var imageFeature = new OL.Feature.Vector(point, attributes, style);
	LiveMapAlerts_Layer.addFeatures([imageFeature]);
}

function getAlertAge(msValue){
	var d = new Date();
	var ageInMinutes;
	try {
		ageInMinutes = Math.floor((d.getTime() - msValue) / 60000);
	} //milliseconds to minutes
	catch(e){
		ageInMinutes = "unknown";
	}
	return ageInMinutes;
}

function getFriendlyType(rawType){
	var friendlyType;
	switch(rawType){
		case 'JAM':
			friendlyType = 'Jam'; break;
		case 'POLICE': //Firefox
			friendlyType = 'Police'; break;
		case 'POLICEMAN': //Chrome
			friendlyType = 'Police'; break;
		case 'CHIT_CHAT':
			friendlyType = 'Chit Chat'; break;
		case 'TRAFFIC_JAM':
			friendlyType = 'Traffic Jam'; break;
		case 'HAZARD':
			friendlyType = 'Hazard'; break;
		case 'WEATHERHAZARD': //Chrome JSON has all hazards as weather hazard
			friendlyType = 'Hazard'; break;
		case 'ACCIDENT':
			friendlyType = 'Accident'; break;
		case 'CONSTRUCTION':
			friendlyType = 'Construction'; break;
		case 'ROAD_CLOSED':
			friendlyType = 'Road Closed'; break;
		case 'OTHER':
			friendlyType = 'Chit Chat'; break;
		default:
			friendlyType = 'Unknown';
	}
	return friendlyType;
}

function getFriendlySubType(rawSubType){
	var friendlySubType;
	switch(rawSubType){
		case 'POLICE_VISIBLE':
			friendlySubType = 'Visible Police Trap'; break;
		case 'POLICE_HIDING':
			friendlySubType = 'Hidden Police Trap'; break;
		case 'ROAD_CLOSED_HAZARD':
			friendlySubType = 'Hazard Closure'; break;
		case 'ROAD_CLOSED_CONSTRUCTION':
			friendlySubType = 'Construction Closure'; break;
		case 'ROAD_CLOSED_EVENT':
			friendlySubType = 'Closure'; break;
		case 'ACCIDENT_MINOR':
			friendlySubType = 'Minor Accident'; break;
		case 'ACCIDENT_MAJOR':
			friendlySubType = 'Major Accident'; break;
		case 'JAM_MODERATE_TRAFFIC':
			friendlySubType = 'Moderate Traffic'; break;
		case 'JAM_HEAVY_TRAFFIC':
			friendlySubType = 'Heavy Traffic'; break;
		case 'JAM_STAND_STILL_TRAFFIC':
			friendlySubType = 'Stand Still Traffic'; break;
		case 'JAM_LIGHT_TRAFFIC':
			friendlySubType = 'Light Traffic'; break;
		case 'HAZARD_ON_ROAD':
			friendlySubType = 'Hazard on Road'; break;
		case 'HAZARD_ON_SHOULDER':
			friendlySubType = 'Hazard on Shoulder'; break;
		case 'HAZARD_WEATHER':
			friendlySubType = 'Weather Hazard'; break;
		case 'HAZARD_ON_ROAD_OBJECT':
			friendlySubType = 'Object on Road'; break;
		case 'HAZARD_ON_ROAD_POT_HOLE':
			friendlySubType = 'Pothole'; break;
		case 'HAZARD_ON_ROAD_ROAD_KILL':
			friendlySubType = 'Road Kill'; break;
		case 'HAZARD_ON_ROAD_CAR_STOPPED':
			friendlySubType = 'Car Stopped on Road'; break;
		case 'HAZARD_ON_SHOULDER_CAR_STOPPED':
			friendlySubType = 'Car Stopped on Shoulder'; break;
		case 'HAZARD_ON_SHOULDER_ANIMALS':
			friendlySubType = 'Animals on Shoulder'; break;
		case 'HAZARD_ON_SHOULDER_MISSING_SIGN':
			friendlySubType = 'Missing Sign'; break;
		case 'HAZARD_WEATHER_FOG':
			friendlySubType = 'Fog'; break;
		case 'HAZARD_WEATHER_HAIL':
			friendlySubType = 'Hail'; break;
		case 'HAZARD_WEATHER_HEAVY_RAIN':
			friendlySubType = 'Heavy Rain'; break;
		case 'HAZARD_WEATHER_HEAVY_SNOW':
			friendlySubType = 'Heavy Snow'; break;
		case 'HAZARD_WEATHER_FLOOD':
			friendlySubType = 'Flood'; break;
		case 'HAZARD_WEATHER_MONSOON':
			friendlySubType = 'Monsoon'; break;
		case 'HAZARD_WEATHER_TORNADO':
			friendlySubType = 'Tornado'; break;
		case 'HAZARD_WEATHER_HEAT_WAVE':
			friendlySubType = 'Heat Wave'; break;
		case 'HAZARD_WEATHER_HURRICANE':
			friendlySubType = 'Hurricane'; break;
		case 'HAZARD_WEATHER_FREEZING_RAIN':
			friendlySubType = 'Freezing Rain'; break;
		case 'HAZARD_ON_ROAD_LANE_CLOSED':
			friendlySubType = 'Closed Lane'; break;
		case 'HAZARD_ON_ROAD_OIL':
			friendlySubType = 'Fresh Oil Spill'; break;
		case 'HAZARD_ON_ROAD_ICE':
			friendlySubType = 'Recently Formed Ice'; break;
		case 'HAZARD_ON_ROAD_CONSTRUCTION':
			friendlySubType = 'Construction'; break;
		default:
			friendlySubType = '';
	}
	return friendlySubType;
}

function initializeLiveMapAlerts(e)
{
	W = unsafeWindow.W;
	OL = unsafeWindow.OpenLayers;
	I18n = unsafeWindow.I18n;
	if (e && e.user === null) {
		return;
	}
	if (typeof W === "undefined" || typeof W.loginManager === "undefined" || document.querySelector('.list-unstyled.togglers .group') === null) {
		setTimeout(initializeLiveMapAlerts, 100);
		return;
	}
	if (!W.loginManager.user) {
		W.loginManager.events.register("login", null, initializeLiveMapAlerts);
		W.loginManager.events.register("loginStatus", null, initializeLiveMapAlerts);
		if (!W.loginManager.user) {
			return;
		}
	}
	function showAlertPopup(f){
		try {
			var urPanel = document.getElementById('update-request-panel');
			var mpPanel = document.getElementById('problem-edit-panel');
			var conversationPanel = urPanel.children[5];
			if (urPanel.className == 'top-panel panel-shown collapsed' && conversationPanel.style.display == 'block'){
				divLMAO.style.left = '635px';
			}
			else if (urPanel.className == 'top-panel panel-shown' && conversationPanel.style.display == 'block'){
				divLMAO.style.top = '325px';
				divLMAO.style.left = '635px';
			}
			else if (urPanel.className == 'top-panel panel-shown' || mpPanel.className == 'top-panel panel-shown'){
				divLMAO.style.top = '325px';
			}
			else{
				divLMAO.style.top = '175px';
				divLMAO.style.left = '375px';
			}
		}
		catch(e){
		}
		var attributes = f.attributes;
		var alertType = ((attributes.type == "WEATHERHAZARD") ? "HAZARD" : attributes.type);
		var alertSubType = ((!attributes.subtype) ? "" : attributes.subtype);
		var alertDescription = ((!attributes.description) ? "" : attributes.description);
		var alertStreet = ((!attributes.street) ? "" : attributes.street);
		var alertCity = ((!attributes.city) ? "" : attributes.city);
		var alertNear = ((!attributes.near) ? "" : attributes.near);
		var alertReportBy = ((!attributes.reportby) ? "" : attributes.reportby);
		var alertPubMillis = attributes.pubMillis;
		var alertDateTime = new Date(parseInt(alertPubMillis));
		var alertAge = getAlertAge(attributes.pubMillis);
		var displayAge;
		if(alertAge < 60){
			displayAge = alertAge + " min";
		}
		else if(alertAge >= 60 && alertAge < 1440){
			displayAge = Math.floor(alertAge/60) + " hr";
		}
		else if(alertAge >= 1440){
			displayAge = Math.floor(alertAge/1440) + " day";
		}
		else{
			displayAge = " unknown";
		}
		var reportDetail = "<b>LiveMap Alert Details</b>" +
			"<br>TYPE: " + getFriendlyType(alertType) +
			"<br>SUBTYPE: " + getFriendlySubType(alertSubType) +
			"<br>DESCRIPTION: " + alertDescription +
			"<br>STREET: " + alertStreet +
			"<br>CITY: " + alertCity +
			"<br>NEAR: " + alertNear +
			"<br>REPORT BY: " + alertReportBy +
			"<br>DATE: " + alertDateTime.toDateString() +
			"<br>TIME (user local): " + alertDateTime.toLocaleTimeString() +
			"<br>AGE: " + displayAge
		;
		document.getElementById("divLMAO").innerHTML = reportDetail;
		divLMAO.style.visibility = 'visible';
	}

	function hideAlertPopup(){
		divLMAO.style.visibility = 'hidden';
		divLMAO.style.top = '175px';
		divLMAO.style.left = '375px';
	}

	function registerKeyShortcut(actionName, callback, keyName) {
		I18n.translations[I18n.locale].keyboard_shortcuts.groups.default.members[keyName] = actionName;
		W.accelerators.addAction(keyName, {group: 'default'});
		W.accelerators.events.register(keyName, null, callback);
		W.accelerators._registerShortcuts({[""]: keyName});
	}

	var lmaoVisibility = true;
	LiveMapAlerts_Layer = new OL.Layer.Vector("LiveMap Alerts",{
		rendererOptions: { zIndexing: true },
		uniqueName: '__livemap_alerts'
	});
	if (localStorage.WMELiveMapAlerts) {
		var options = JSON.parse(localStorage.getItem("WMELiveMapAlerts"));
		lmaoVisibility = options[0];
	}
	window.addEventListener("beforeunload", function saveLMAOOptions() {
		if (localStorage) {
			var options = [];
			lmaoVisibility = document.getElementById("layer-switcher-item_livemap_alerts").checked;
			options[0] = lmaoVisibility;
			localStorage.setItem("WMELiveMapAlerts", JSON.stringify(options));
		}
	}, false);
	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_alerts';
		checkbox.className = "hydrated";
		checkbox.disabled = !roadGroupSelector.checked;
		checkbox.checked = lmaoVisibility;
		lmaoVisibility = checkbox.checked & roadGroupSelector.checked;
		checkbox.appendChild(document.createTextNode("LiveMap Alerts"));
		toggler.appendChild(checkbox);
		roadGroup.appendChild(toggler);
		checkbox.addEventListener('click', function(e) {
			lmaoVisibility = e.target.checked;
			LiveMapAlerts_Layer.setVisibility(lmaoVisibility);
			getLiveMapAlerts();
		});
		roadGroupSelector.addEventListener('click', function(e) {
			lmaoVisibility = e.target.checked & checkbox.checked;
			LiveMapAlerts_Layer.setVisibility(lmaoVisibility);
			getLiveMapAlerts();
			checkbox.disabled = !e.target.checked;
		});
		registerKeyShortcut("LiveMap Alerts Overlay", function(e) {
			if (!roadGroupSelector.checked.disabled) {
				checkbox.click();
			}
		}, "LiveMapAlertsOverlay");
	}
	W.map.addControl(new OL.Control.DrawFeature(LiveMapAlerts_Layer, OL.Handler.Path));
	W.map.addLayer(LiveMapAlerts_Layer);
	LiveMapAlerts_Layer.setVisibility(lmaoVisibility);
	getLiveMapAlerts();
	var divPopupCheck = document.getElementById('divLMAO');
	if (divPopupCheck === null){
		var divLMAO = document.createElement('div');
		divLMAO.id = "divLMAO";
		divLMAO.style.position = 'absolute';
		divLMAO.style.visibility = 'hidden';
		divLMAO.style.top = '175px';
		divLMAO.style.left = '375px';
		divLMAO.style.zIndex = 1000;
		divLMAO.style.backgroundColor = 'aliceblue';
		divLMAO.style.borderWidth = '3px';
		divLMAO.style.borderStyle = 'ridge';
		divLMAO.style.borderRadius = '10px';
		divLMAO.style.boxShadow = '5px 5px 10px Silver';
		divLMAO.style.padding = '4px';
		document.body.appendChild(divLMAO);
	}
	W.map.olMap.events.register("zoomend", null, getLiveMapAlerts);
	W.map.olMap.events.register("moveend", null, getLiveMapAlerts);
	W.map.olMap.events.register("mousemove", W.map, function(e) {
		hideAlertPopup();
		var position = this.olMap.events.getMousePosition(e)
		var lmaoLayer = checkLayerNum();
		if(W.map.getLayers()[lmaoLayer].features.length > 0){
			var alertFeatures = W.map.getLayers()[lmaoLayer];
			for(var j = 0; j < W.map.getLayers()[lmaoLayer].features.length; j++){
				var alertX = alertFeatures.features[j].attributes.pixel.x;
				var alertY = alertFeatures.features[j].attributes.pixel.y + getFeatureYOffset();
				if(LiveMapAlerts_Layer.getVisibility() === true && position.x > alertX - 10 && position.x < alertX + 10 && position.y > alertY - 10 && position.y < alertY + 20){
					showAlertPopup(alertFeatures.features[j]);
				}
			}
		}
	});
}
长期地址
遇到问题?请前往 GitHub 提 Issues。