WME LiveMap Alerts Overlay

Overlay alerts from the Waze LiveMap

Versión del día 19/09/2017. Echa un vistazo a la versión más reciente.

Tendrás que instalar una extensión para tu navegador como Tampermonkey, Greasemonkey o Violentmonkey si quieres utilizar este script.

Necesitarás instalar una extensión como Tampermonkey o Violentmonkey para instalar este script.

Necesitarás instalar una extensión como Tampermonkey o Violentmonkey para instalar este script.

Necesitarás instalar una extensión como Tampermonkey o Userscripts para instalar este script.

Necesitará instalar una extensión como Tampermonkey para instalar este script.

Necesitarás instalar una extensión para administrar scripts de usuario si quieres instalar este script.

(Ya tengo un administrador de scripts de usuario, déjame instalarlo)

Necesitará instalar una extensión como Stylus para instalar este estilo.

Necesitará instalar una extensión como Stylus para instalar este estilo.

Necesitará instalar una extensión como Stylus para instalar este estilo.

Necesitará instalar una extensión del gestor de estilos de usuario para instalar este estilo.

Necesitará instalar una extensión del gestor de estilos de usuario para instalar este estilo.

Necesitará instalar una extensión del gestor de estilos de usuario para instalar este estilo.

(Ya tengo un administrador de estilos de usuario, déjame instalarlo)

// ==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.7
// @grant               none
// @namespace           https://greatest.deepsurf.us/users/110192
// ==/UserScript==

//------------------------------------------------------------------------------------------------
function bootstrapLiveMapAlerts()
{
    setTimeout(initializeLiveMapAlerts, 999);
}

//--------------------------------------------------------------------------------------------------------
function checkLayerNum()
{
    var lmaoLayer = null;
    for(i=0; i<Waze.map.layers.length; i++)
      {
         if(Waze.map.layers[i].uniqueName == '__livemap_alerts') lmaoLayer = i;
      }
    //console.log('WME LMAO: layer number = ' + lmaoLayer);
    return lmaoLayer;
}

//--------------------------------------------------------------------------------------------------------
function getBounds()
{
   	var alertBounds = Waze.map.getExtent();
    alertBounds.transform(new OpenLayers.Projection("EPSG:900913"),new OpenLayers.Projection("EPSG:4326"));
    //console.log('WME LMAO: Current bounds = Left ' + alertBounds.left + ', Right ' + alertBounds.right + ', Bottom ' + alertBounds.bottom + ', Top ' + alertBounds.top);//verify transform
    return alertBounds;
}

//--------------------------------------------------------------------------------------------------------
function getRoutingServer(){
    var server = Waze.location.code;
    //console.log('WME LMAO: server = ' + server);
    switch(server){
        case 'usa':
           var routingURL = '//www.waze.com/rtserver/web/TGeoRSS';
             break;
        case 'row':
           var routingURL = '//www.waze.com/row-rtserver/web/TGeoRSS';
            break;
        case 'il':
           var routingURL = '//www.waze.com/il-rtserver/web/TGeoRSS';
            break;
        default:
            var routingURL = '//www.waze.com/rtserver/web/TGeoRSS';
    };
    return routingURL;
}

//--------------------------------------------------------------------------------------------------------
function getFeatureYOffset(){
    var yOffset = -30;
    return yOffset;
}

//--------------------------------------------------------------------------------------------------------
function getLiveMapAlerts(){
    if (!LiveMapAlerts_Layer.getVisibility()) {
        return;
    }
    LiveMapAlerts_Layer.destroyFeatures();
    var alertBounds = getBounds();
    var url = getRoutingServer();
	var data = {
      left: alertBounds.left,
      right: alertBounds.right,
      bottom: alertBounds.top,
      top: alertBounds.bottom
    };
    $.ajax({
        dataType: "json",
        url: url,
        data: data,
        success: function(json) {
            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;
                    //console.log("WME LMAO: " + alertData[i].type, alertData[i].location.x, alertData[i].location.y);
                    addImage(lat, long, type, subType, alertData[i]);
                }
            }
            catch(e) {
                //console.log('WME LMAO: No alerts in view');
            }
        }
    });
}

//--------------------------------------------------------------------------------------------------------
function addImage(lat, long, type, subType, detail) {
	var coords = OpenLayers.Layer.SphericalMercator.forwardMercator(long, lat);
	var point = new OpenLayers.Geometry.Point(coords.lon,coords.lat);
	var alertPx = Waze.map.getPixelFromLonLat(new OpenLayers.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);
	if(type == 'ROAD_CLOSED'){
		var alertTransparency = 1;
	}
	else if(alertAgeInMinutes < 15){
		var alertTransparency = 1;
	}
	else if(alertAgeInMinutes >= 15 && alertAgeInMinutes < 30){
		var alertTransparency = .8;
	}
	else if(alertAgeInMinutes >= 30 && alertAgeInMinutes < 45){
		var alertTransparency = .5;
	}
	else if(alertAgeInMinutes >= 45){
		var alertTransparency = .25;
	}
	else{
		var alertTransparency = 1; //"unknown" age
	}
	var style = {
		externalGraphic: icon,
		graphicWidth: 32,
		graphicHeight: 38,
		graphicYOffset: getFeatureYOffset(),
		fillOpacity: alertTransparency,
		title: 'LiveMap',
		cursor: 'help'
	};
	var imageFeature = new OpenLayers.Feature.Vector(point, attributes, style);
	LiveMapAlerts_Layer.addFeatures([imageFeature]);
	//console.log('WME LMAO: Added alert at ' + lat + ',' + long);
}

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

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

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

//--------------------------------------------------------------------------------------------------------
function initializeLiveMapAlerts()
{
    //console.log("WME LMAO: Initializing");
    var lmaoVisibility = true;
    LiveMapAlerts_Layer = new OpenLayers.Layer.Vector("LiveMap Alerts",{
            rendererOptions: { zIndexing: true },
     		uniqueName: '__livemap_alerts'
        }
    );
    //I18n.translations.en.layers.name["__livemap_alerts"] = "LiveMap Alerts";
    // restore saved settings
    if (localStorage.WMELiveMapAlerts) {
        console.log("WME LMAO: loading options");
        var options = JSON.parse(localStorage.getItem("WMELiveMapAlerts"));
        lmaoVisibility = options[0];
    }
    // overload the WME exit function
    saveLMAOOptions = function() {
        if (localStorage) {
            console.log("WME LMAO: saving options");
            var options = [];
            lmaoVisibility = LiveMapAlerts_Layer.visibility;
            options[0] = lmaoVisibility;
            localStorage.setItem("WMELiveMapAlerts", JSON.stringify(options));
        }
    }
    window.addEventListener("beforeunload", saveLMAOOptions, false);
    function showAlertPopup(f){
        //shift popup if UR or MP panel is visible
        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){
            //console.log('WME LMAO: Problem getting detail for UR or MP panel');
        }
        var attributes = f.attributes;
        var alertType = ((attributes.type == "WEATHERHAZARD") ? "HAZARD" : attributes.type);
        var alertSubType = ((attributes.subtype == null) ? "" : attributes.subtype);
        var alertDescription = ((attributes.description == null) ? "" : attributes.description);
        var alertStreet = ((attributes.street == null) ? "" : attributes.street);
        var alertCity = ((attributes.city == null) ? "" : attributes.city);
        var alertNear = ((attributes.near == null) ? "" : attributes.near);
        var alertReportBy = ((attributes.reportby == null) ? "" : attributes.reportby);
        var alertPubMillis = attributes.pubMillis;
        var alertDateTime = new Date(parseInt(alertPubMillis));
        var alertAge = getAlertAge(attributes.pubMillis);
        if(alertAge < 60){
        	var displayAge = alertAge + " min";
        }
        else if(alertAge >= 60 && alertAge < 1440){
        	var displayAge = Math.floor(alertAge/60) + " hr";
        }
        else if(alertAge >= 1440){
        	var displayAge = Math.floor(alertAge/1440) + " day";
        }
        else{
            var 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';
    };
    LiveMapAlerts_Layer.setZIndex(9999);
    Waze.map.addLayer(LiveMapAlerts_Layer);
    var roadGroupSelector = document.getElementById('layer-switcher-group_road');
    if (roadGroupSelector !== null) {
        var roadGroup = roadGroupSelector.parentNode.parentNode.querySelector('.children');
        var toggler = document.createElement('li');
        var togglerContainer = document.createElement('div');
        var checkbox = document.createElement('input');
        var label = document.createElement('label');
        var labelText = document.createElement('span');
        togglerContainer.className = 'controls-container toggler';
        checkbox.type = 'checkbox';
        checkbox.id = 'layer-switcher-item_livemap_alerts';
        checkbox.className = 'toggle';
        checkbox.disabled = !roadGroupSelector.checked;
        checkbox.checked = lmaoVisibility;
        togglerContainer.appendChild(checkbox);
        label.htmlFor = checkbox.id;
        labelText.className = 'label-text';
        labelText.appendChild(document.createTextNode("LiveMap Alerts"));
        label.appendChild(labelText);
        togglerContainer.appendChild(label);
        toggler.appendChild(togglerContainer);
        roadGroup.appendChild(toggler);
        checkbox.addEventListener('click', function(e) {
            LiveMapAlerts_Layer.setVisibility(e.target.checked);
        });
        roadGroupSelector.addEventListener('click', function(e) {
            LiveMapAlerts_Layer.setVisibility(e.target.checked & checkbox.checked);
            checkbox.disabled = !e.target.checked;
        });
    }
    Waze.map.addControl(new OpenLayers.Control.DrawFeature(LiveMapAlerts_Layer, OpenLayers.Handler.Path));
    LiveMapAlerts_Layer.setVisibility(lmaoVisibility);
    var divPopupCheck = document.getElementById('divLMAO');
    if (divPopupCheck == null){
        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);
        //console.log('WME LMAO: Creating popup divLMAO');
    }
    //clear existing LMAO features
    LiveMapAlerts_Layer.destroyFeatures();
    var lmaoLayer = checkLayerNum();
    Waze.map.events.register("mousemove", Waze.map, function(e) {
        hideAlertPopup();
        var position = this.events.getMousePosition(e);
        //console.log('WME LMAO: coords xy = ' + position.x + ' ' + position.y);
        var lmaoLayer = checkLayerNum();
        if(Waze.map.layers[lmaoLayer].features.length > 0){
            //var alertCount = Waze.map.layers[lmaoLayer].features.length;
            //console.log('WME LMAO: Current LiveMap alert count = ' + alertCount);
            var alertFeatures = Waze.map.layers[lmaoLayer];
            for(j=0; j<Waze.map.layers[lmaoLayer].features.length; j++){
                var lmaoLayerVisibility = LiveMapAlerts_Layer.getVisibility();
                var alertX = alertFeatures.features[j].attributes.pixel.x;
                var alertY = alertFeatures.features[j].attributes.pixel.y + getFeatureYOffset();
                if(lmaoLayerVisibility == true && position.x > alertX - 10 && position.x < alertX + 10 && position.y > alertY - 10 && position.y < alertY + 20){
                	//console.log('WME LMAO: hover over alert');
                    showAlertPopup(alertFeatures.features[j]);
                }
            }
        }
    });
    //refresh if user moves map
    Waze.map.events.register("moveend", Waze.map, getLiveMapAlerts);
    window.setTimeout(getLiveMapAlerts(), 500);
}

//--------------------------------------------------------------------------------------------------------------
bootstrapLiveMapAlerts();