617 lines
23 KiB
JavaScript
617 lines
23 KiB
JavaScript
$(function(){
|
|
var query = window.location.search.replace(/^\?/, '').split('&').map(function(v){
|
|
var s = v.split('=');
|
|
var r = {};
|
|
r[s[0]] = s.slice(1).join('=');
|
|
return r;
|
|
}).reduce(function(a, b){
|
|
return a.assign(b);
|
|
});
|
|
|
|
var expectedCallsign;
|
|
if (query.callsign) expectedCallsign = decodeURIComponent(query.callsign);
|
|
var expectedLocator;
|
|
if (query.locator) expectedLocator = query.locator;
|
|
|
|
var protocol = window.location.protocol.match(/https/) ? 'wss' : 'ws';
|
|
|
|
var href = window.location.href;
|
|
var index = href.lastIndexOf('/');
|
|
if (index > 0) {
|
|
href = href.substr(0, index + 1);
|
|
}
|
|
href = href.split("://")[1];
|
|
href = protocol + "://" + href;
|
|
if (!href.endsWith('/')) {
|
|
href += '/';
|
|
}
|
|
var ws_url = href + "ws/";
|
|
|
|
var map;
|
|
var markers = {};
|
|
var rectangles = {};
|
|
var receiverMarker;
|
|
var updateQueue = [];
|
|
|
|
// reasonable default; will be overriden by server
|
|
var retention_time = 2 * 60 * 60 * 1000;
|
|
var strokeOpacity = 0.8;
|
|
var fillOpacity = 0.35;
|
|
var callsign_url = null;
|
|
|
|
var colorKeys = {};
|
|
var colorScale = chroma.scale(['red', 'blue', 'green']).mode('hsl');
|
|
var getColor = function(id){
|
|
if (!id) return "#000000";
|
|
if (!colorKeys[id]) {
|
|
var keys = Object.keys(colorKeys);
|
|
keys.push(id);
|
|
keys.sort(function(a, b) {
|
|
var pa = parseFloat(a);
|
|
var pb = parseFloat(b);
|
|
if (isNaN(pa) || isNaN(pb)) return a.localeCompare(b);
|
|
return pa - pb;
|
|
});
|
|
var colors = colorScale.colors(keys.length);
|
|
colorKeys = {};
|
|
keys.forEach(function(key, index) {
|
|
colorKeys[key] = colors[index];
|
|
});
|
|
reColor();
|
|
updateLegend();
|
|
}
|
|
return colorKeys[id];
|
|
}
|
|
|
|
// when the color palette changes, update all grid squares with new color
|
|
var reColor = function() {
|
|
$.each(rectangles, function(_, r) {
|
|
var color = getColor(colorAccessor(r));
|
|
r.setOptions({
|
|
strokeColor: color,
|
|
fillColor: color
|
|
});
|
|
});
|
|
}
|
|
|
|
var colorMode = 'byband';
|
|
var colorAccessor = function(r) {
|
|
switch (colorMode) {
|
|
case 'byband':
|
|
return r.band;
|
|
case 'bymode':
|
|
return r.mode;
|
|
}
|
|
};
|
|
|
|
$(function(){
|
|
$('#openwebrx-map-colormode').on('change', function(){
|
|
colorMode = $(this).val();
|
|
colorKeys = {};
|
|
filterRectangles(allRectangles);
|
|
reColor();
|
|
updateLegend();
|
|
});
|
|
});
|
|
|
|
var updateLegend = function() {
|
|
var lis = $.map(colorKeys, function(value, key) {
|
|
// fake rectangle to test if the filter would match
|
|
var fakeRectangle = Object.fromEntries([[colorMode.slice(2), key]]);
|
|
var disabled = rectangleFilter(fakeRectangle) ? '' : ' disabled';
|
|
return '<li class="square' + disabled + '" data-selector="' + key + '"><span class="illustration" style="background-color:' + chroma(value).alpha(fillOpacity) + ';border-color:' + chroma(value).alpha(strokeOpacity) + ';"></span>' + key + '</li>';
|
|
});
|
|
$(".openwebrx-map-legend .content").html('<ul>' + lis.join('') + '</ul>');
|
|
}
|
|
|
|
var processUpdates = function(updates) {
|
|
if (typeof(AprsMarker) == 'undefined') {
|
|
updateQueue = updateQueue.concat(updates);
|
|
return;
|
|
}
|
|
updates.forEach(function(update){
|
|
|
|
switch (update.location.type) {
|
|
case 'latlon':
|
|
var pos = new google.maps.LatLng(update.location.lat, update.location.lon);
|
|
var marker;
|
|
var markerClass = google.maps.Marker;
|
|
var aprsOptions = {}
|
|
if (update.location.symbol) {
|
|
markerClass = AprsMarker;
|
|
aprsOptions.symbol = update.location.symbol;
|
|
aprsOptions.course = update.location.course;
|
|
aprsOptions.speed = update.location.speed;
|
|
}
|
|
if (markers[update.callsign]) {
|
|
marker = markers[update.callsign];
|
|
} else {
|
|
marker = new markerClass();
|
|
marker.addListener('click', function(){
|
|
showMarkerInfoWindow(update.callsign, pos);
|
|
});
|
|
markers[update.callsign] = marker;
|
|
}
|
|
marker.setOptions($.extend({
|
|
position: pos,
|
|
map: map,
|
|
title: update.callsign
|
|
}, aprsOptions, getMarkerOpacityOptions(update.lastseen) ));
|
|
marker.lastseen = update.lastseen;
|
|
marker.mode = update.mode;
|
|
marker.band = update.band;
|
|
marker.comment = update.location.comment;
|
|
marker.weather = update.location.weather;
|
|
marker.altitude = update.location.altitude;
|
|
marker.height = update.location.height;
|
|
marker.power = update.location.power;
|
|
marker.gain = update.location.gain;
|
|
marker.device = update.location.device;
|
|
marker.directivity = update.location.directivity;
|
|
|
|
if (expectedCallsign && expectedCallsign == update.callsign) {
|
|
map.panTo(pos);
|
|
showMarkerInfoWindow(update.callsign, pos);
|
|
expectedCallsign = false;
|
|
}
|
|
|
|
if (infowindow && infowindow.callsign && infowindow.callsign == update.callsign) {
|
|
showMarkerInfoWindow(infowindow.callsign, pos);
|
|
}
|
|
break;
|
|
case 'locator':
|
|
var loc = update.location.locator;
|
|
var lat = (loc.charCodeAt(1) - 65 - 9) * 10 + Number(loc[3]);
|
|
var lon = (loc.charCodeAt(0) - 65 - 9) * 20 + Number(loc[2]) * 2;
|
|
var center = new google.maps.LatLng({lat: lat + .5, lng: lon + 1});
|
|
var rectangle;
|
|
// the accessor is designed to work on the rectangle... but it should work on the update object, too
|
|
var color = getColor(colorAccessor(update));
|
|
if (rectangles[update.callsign]) {
|
|
rectangle = rectangles[update.callsign];
|
|
} else {
|
|
rectangle = new google.maps.Rectangle();
|
|
rectangle.addListener('click', function(){
|
|
showLocatorInfoWindow(this.locator, this.center);
|
|
});
|
|
rectangles[update.callsign] = rectangle;
|
|
}
|
|
rectangle.lastseen = update.lastseen;
|
|
rectangle.locator = update.location.locator;
|
|
rectangle.mode = update.mode;
|
|
rectangle.band = update.band;
|
|
rectangle.center = center;
|
|
|
|
rectangle.setOptions($.extend({
|
|
strokeColor: color,
|
|
strokeWeight: 2,
|
|
fillColor: color,
|
|
map: rectangleFilter(rectangle) ? map : undefined,
|
|
bounds:{
|
|
north: lat,
|
|
south: lat + 1,
|
|
west: lon,
|
|
east: lon + 2
|
|
}
|
|
}, getRectangleOpacityOptions(update.lastseen) ));
|
|
|
|
if (expectedLocator && expectedLocator == update.location.locator) {
|
|
map.panTo(center);
|
|
showLocatorInfoWindow(expectedLocator, center);
|
|
expectedLocator = false;
|
|
}
|
|
|
|
if (infowindow && infowindow.locator && infowindow.locator == update.location.locator) {
|
|
showLocatorInfoWindow(infowindow.locator, center);
|
|
}
|
|
break;
|
|
}
|
|
});
|
|
};
|
|
|
|
var clearMap = function(){
|
|
var reset = function(callsign, item) { item.setMap(); };
|
|
$.each(markers, reset);
|
|
$.each(rectangles, reset);
|
|
receiverMarker.setMap();
|
|
markers = {};
|
|
rectangles = {};
|
|
};
|
|
|
|
var reconnect_timeout = false;
|
|
|
|
var config = {}
|
|
|
|
var connect = function(){
|
|
var ws = new WebSocket(ws_url);
|
|
ws.onopen = function(){
|
|
ws.send("SERVER DE CLIENT client=map.js type=map");
|
|
reconnect_timeout = false
|
|
};
|
|
|
|
ws.onmessage = function(e){
|
|
if (typeof e.data != 'string') {
|
|
console.error("unsupported binary data on websocket; ignoring");
|
|
return
|
|
}
|
|
if (e.data.substr(0, 16) == "CLIENT DE SERVER") {
|
|
return
|
|
}
|
|
try {
|
|
var json = JSON.parse(e.data);
|
|
switch (json.type) {
|
|
case "config":
|
|
Object.assign(config, json.value);
|
|
if ('receiver_gps' in config) {
|
|
var receiverPos = {
|
|
lat: config.receiver_gps.lat,
|
|
lng: config.receiver_gps.lon
|
|
};
|
|
if (!map) $.getScript("https://maps.googleapis.com/maps/api/js?key=" + config.google_maps_api_key).done(function(){
|
|
map = new google.maps.Map($('.openwebrx-map')[0], {
|
|
center: receiverPos,
|
|
zoom: 5,
|
|
});
|
|
|
|
$.getScript("static/lib/nite-overlay.js").done(function(){
|
|
nite.init(map);
|
|
setInterval(function() { nite.refresh() }, 10000); // every 10s
|
|
});
|
|
$.getScript('static/lib/AprsMarker.js').done(function(){
|
|
processUpdates(updateQueue);
|
|
updateQueue = [];
|
|
});
|
|
|
|
var $legend = $(".openwebrx-map-legend");
|
|
setupLegendFilters($legend);
|
|
map.controls[google.maps.ControlPosition.LEFT_BOTTOM].push($legend[0]);
|
|
|
|
if (!receiverMarker) {
|
|
receiverMarker = new google.maps.Marker();
|
|
receiverMarker.addListener('click', function() {
|
|
showReceiverInfoWindow(receiverMarker);
|
|
});
|
|
}
|
|
receiverMarker.setOptions({
|
|
map: map,
|
|
position: receiverPos,
|
|
title: config['receiver_name'],
|
|
config: config
|
|
});
|
|
}); else {
|
|
receiverMarker.setOptions({
|
|
map: map,
|
|
position: receiverPos,
|
|
config: config
|
|
});
|
|
}
|
|
}
|
|
if ('receiver_name' in config && receiverMarker) {
|
|
receiverMarker.setOptions({
|
|
title: config['receiver_name']
|
|
});
|
|
}
|
|
if ('map_position_retention_time' in config) {
|
|
retention_time = config.map_position_retention_time * 1000;
|
|
}
|
|
if ('callsign_url' in config) {
|
|
callsign_url = config['callsign_url'];
|
|
}
|
|
break;
|
|
case "update":
|
|
processUpdates(json.value);
|
|
break;
|
|
case 'receiver_details':
|
|
$('.webrx-top-container').header().setDetails(json['value']);
|
|
break;
|
|
default:
|
|
console.warn('received message of unknown type: ' + json['type']);
|
|
}
|
|
} catch (e) {
|
|
// don't lose exception
|
|
console.error(e);
|
|
}
|
|
};
|
|
ws.onclose = function(){
|
|
clearMap();
|
|
if (reconnect_timeout) {
|
|
// max value: roundabout 8 and a half minutes
|
|
reconnect_timeout = Math.min(reconnect_timeout * 2, 512000);
|
|
} else {
|
|
// initial value: 1s
|
|
reconnect_timeout = 1000;
|
|
}
|
|
setTimeout(connect, reconnect_timeout);
|
|
};
|
|
|
|
window.onbeforeunload = function() { //http://stackoverflow.com/questions/4812686/closing-websocket-correctly-html5-javascript
|
|
ws.onclose = function () {};
|
|
ws.close();
|
|
};
|
|
|
|
/*
|
|
ws.onerror = function(){
|
|
console.info("websocket error");
|
|
};
|
|
*/
|
|
};
|
|
|
|
connect();
|
|
|
|
var getInfoWindow = function() {
|
|
if (!infowindow) {
|
|
infowindow = new google.maps.InfoWindow();
|
|
google.maps.event.addListener(infowindow, 'closeclick', function() {
|
|
delete infowindow.locator;
|
|
delete infowindow.callsign;
|
|
});
|
|
}
|
|
delete infowindow.locator;
|
|
delete infowindow.callsign;
|
|
return infowindow;
|
|
}
|
|
|
|
var linkifyCallsign = function(callsign) {
|
|
if ((callsign_url == null) || (callsign_url == ''))
|
|
return callsign;
|
|
else
|
|
return '<a target="callsign_info" href="' +
|
|
callsign_url.replaceAll('{}', callsign.replace(new RegExp('-.*$'), '')) +
|
|
'">' + callsign + '</a>';
|
|
};
|
|
|
|
var distanceKm = function(p1, p2) {
|
|
// Earth radius in km
|
|
var R = 6371.0;
|
|
// Convert degrees to radians
|
|
var rlat1 = p1.lat() * (Math.PI/180);
|
|
var rlat2 = p2.lat() * (Math.PI/180);
|
|
// Compute difference in radians
|
|
var difflat = rlat2-rlat1;
|
|
var difflon = (p2.lng()-p1.lng()) * (Math.PI/180);
|
|
// Compute distance
|
|
d = 2 * R * Math.asin(Math.sqrt(
|
|
Math.sin(difflat/2) * Math.sin(difflat/2) +
|
|
Math.cos(rlat1) * Math.cos(rlat2) * Math.sin(difflon/2) * Math.sin(difflon/2)
|
|
));
|
|
return Math.round(d);
|
|
}
|
|
|
|
var infowindow;
|
|
var showLocatorInfoWindow = function(locator, pos) {
|
|
var infowindow = getInfoWindow();
|
|
infowindow.locator = locator;
|
|
var inLocator = $.map(rectangles, function(r, callsign) {
|
|
return {callsign: callsign, locator: r.locator, lastseen: r.lastseen, mode: r.mode, band: r.band}
|
|
}).filter(rectangleFilter).filter(function(d) {
|
|
return d.locator == locator;
|
|
}).sort(function(a, b){
|
|
return b.lastseen - a.lastseen;
|
|
});
|
|
var distance = receiverMarker?
|
|
" at " + distanceKm(receiverMarker.position, pos) + " km" : "";
|
|
infowindow.setContent(
|
|
'<h3>Locator: ' + locator + distance + '</h3>' +
|
|
'<div>Active Callsigns:</div>' +
|
|
'<ul>' +
|
|
inLocator.map(function(i){
|
|
var timestring = moment(i.lastseen).fromNow();
|
|
var message = linkifyCallsign(i.callsign) + ' (' + timestring + ' using ' + i.mode;
|
|
if (i.band) message += ' on ' + i.band;
|
|
message += ')';
|
|
return '<li>' + message + '</li>'
|
|
}).join("") +
|
|
'</ul>'
|
|
);
|
|
infowindow.setPosition(pos);
|
|
infowindow.open(map);
|
|
};
|
|
|
|
var degToCompass = function(deg) {
|
|
dir = ["N", "NNE", "NE", "ENE", "E", "ESE", "SE", "SSE", "S", "SSW", "SW", "WSW", "W", "WNW", "NW", "NNW"];
|
|
return dir[Math.floor((deg/22.5) + 0.5) % 16];
|
|
}
|
|
|
|
var makeListTitle = function(name) {
|
|
return '<div style="border-bottom:2px solid;"><b>' + name + '</b></div>';
|
|
}
|
|
|
|
var makeListItem = function(name, value) {
|
|
return '<div style="border-bottom:1px dotted;">'
|
|
+ '<span>' + name + '</span>'
|
|
+ '<span style="float:right;">' + value + '</span>'
|
|
+ '</div>';
|
|
}
|
|
|
|
var showMarkerInfoWindow = function(callsign, pos) {
|
|
var infowindow = getInfoWindow();
|
|
infowindow.callsign = callsign;
|
|
var marker = markers[callsign];
|
|
var timestring = moment(marker.lastseen).fromNow();
|
|
var commentString = "";
|
|
var weatherString = "";
|
|
var detailsString = "";
|
|
var distance = "";
|
|
|
|
if (marker.comment) {
|
|
commentString += '<p>' + makeListTitle('Comment') + '<div>' +
|
|
marker.comment + '</div></p>';
|
|
}
|
|
|
|
if (marker.weather) {
|
|
weatherString += '<p>' + makeListTitle('Weather');
|
|
|
|
if (marker.weather.temperature) {
|
|
weatherString += makeListItem('Temperature', marker.weather.temperature.toFixed(2) + ' oC');
|
|
}
|
|
|
|
if (marker.weather.humidity) {
|
|
weatherString += makeListItem('Humidity', marker.weather.humidity + '%');
|
|
}
|
|
|
|
if (marker.weather.barometricpressure) {
|
|
weatherString += makeListItem('Pressure', marker.weather.barometricpressure.toFixed(1) + ' mbar');
|
|
}
|
|
|
|
if (marker.weather.wind && (marker.weather.wind.speed>0)) {
|
|
weatherString += makeListItem('Wind',
|
|
degToCompass(marker.weather.wind.direction) + ' ' +
|
|
marker.weather.wind.speed.toFixed(1) + ' km/h '
|
|
);
|
|
}
|
|
|
|
if (marker.weather.wind.gust && (marker.weather.wind.gust>0)) {
|
|
weatherString += makeListItem('Gusts', marker.weather.wind.gust.toFixed(1) + ' km/h');
|
|
}
|
|
|
|
if (marker.weather.rain && (marker.weather.rain.day>0)) {
|
|
weatherString += makeListItem('Rain',
|
|
marker.weather.rain.hour.toFixed(0) + ' mm/h, ' +
|
|
marker.weather.rain.day.toFixed(0) + ' mm/day'
|
|
// marker.weather.rain.sincemidnight + ' mm since midnight'
|
|
);
|
|
}
|
|
|
|
if (marker.weather.snowfall) {
|
|
weatherString += makeListItem('Snow', marker.weather.snowfall.toFixed(1) + ' cm');
|
|
}
|
|
|
|
weatherString += '</p>';
|
|
}
|
|
|
|
if (marker.altitude || marker.device || marker.power || marker.height || marker.gain || marker.directivity) {
|
|
detailsString += '<p>' + makeListTitle('Details');
|
|
|
|
if (marker.altitude) {
|
|
detailsString += makeListItem('Altitude', marker.altitude.toFixed(0) + ' m');
|
|
}
|
|
|
|
if (marker.device) {
|
|
detailsString += makeListItem('Device',
|
|
marker.device.device + " by " +
|
|
marker.device.manufacturer
|
|
);
|
|
}
|
|
|
|
if (marker.height) {
|
|
detailsString += makeListItem('Height', marker.height.toFixed(0) + ' m');
|
|
}
|
|
|
|
if (marker.power) {
|
|
detailsString += makeListItem('Power', marker.power);
|
|
}
|
|
|
|
if (marker.gain) {
|
|
detailsString += makeListItem('Gain', marker.gain);
|
|
}
|
|
|
|
if (marker.directivity) {
|
|
detailsString += makeListItem('Direction', marker.directivity);
|
|
}
|
|
|
|
detailsString += '</p>';
|
|
}
|
|
|
|
if (receiverMarker) {
|
|
distance = " at " + distanceKm(receiverMarker.position, marker.position) + " km";
|
|
}
|
|
|
|
infowindow.setContent(
|
|
'<h3>' + linkifyCallsign(callsign) + distance + '</h3>' +
|
|
'<div align="center">' + timestring + ' using ' + marker.mode + ( marker.band ? ' on ' + marker.band : '' ) + '</div>' +
|
|
commentString + weatherString + detailsString
|
|
);
|
|
|
|
infowindow.open(map, marker);
|
|
}
|
|
|
|
var showReceiverInfoWindow = function(marker) {
|
|
var infowindow = getInfoWindow()
|
|
infowindow.setContent(
|
|
'<h3>' + marker.config['receiver_name'] + '</h3>' +
|
|
'<div>Receiver location</div>'
|
|
);
|
|
infowindow.open(map, marker);
|
|
}
|
|
|
|
var getScale = function(lastseen) {
|
|
var age = new Date().getTime() - lastseen;
|
|
var scale = 1;
|
|
if (age >= retention_time / 2) {
|
|
scale = (retention_time - age) / (retention_time / 2);
|
|
}
|
|
return Math.max(0, Math.min(1, scale));
|
|
};
|
|
|
|
var getRectangleOpacityOptions = function(lastseen) {
|
|
var scale = getScale(lastseen);
|
|
return {
|
|
strokeOpacity: strokeOpacity * scale,
|
|
fillOpacity: fillOpacity * scale
|
|
};
|
|
};
|
|
|
|
var getMarkerOpacityOptions = function(lastseen) {
|
|
var scale = getScale(lastseen);
|
|
return {
|
|
opacity: scale
|
|
};
|
|
};
|
|
|
|
// fade out / remove positions after time
|
|
setInterval(function(){
|
|
var now = new Date().getTime();
|
|
$.each(rectangles, function(callsign, m) {
|
|
var age = now - m.lastseen;
|
|
if (age > retention_time) {
|
|
delete rectangles[callsign];
|
|
m.setMap();
|
|
return;
|
|
}
|
|
m.setOptions(getRectangleOpacityOptions(m.lastseen));
|
|
});
|
|
$.each(markers, function(callsign, m) {
|
|
var age = now - m.lastseen;
|
|
if (age > retention_time) {
|
|
delete markers[callsign];
|
|
m.setMap();
|
|
return;
|
|
}
|
|
m.setOptions(getMarkerOpacityOptions(m.lastseen));
|
|
});
|
|
}, 1000);
|
|
|
|
var rectangleFilter = allRectangles = function() { return true; };
|
|
|
|
var filterRectangles = function(filter) {
|
|
rectangleFilter = filter;
|
|
$.each(rectangles, function(_, r) {
|
|
r.setMap(rectangleFilter(r) ? map : undefined);
|
|
});
|
|
};
|
|
|
|
var setupLegendFilters = function($legend) {
|
|
$content = $legend.find('.content');
|
|
$content.on('click', 'li', function() {
|
|
var $el = $(this);
|
|
$lis = $content.find('li');
|
|
if ($lis.hasClass('disabled') && !$el.hasClass('disabled')) {
|
|
$lis.removeClass('disabled');
|
|
filterRectangles(allRectangles);
|
|
} else {
|
|
$el.removeClass('disabled');
|
|
$lis.filter(function() {
|
|
return this != $el[0]
|
|
}).addClass('disabled');
|
|
|
|
var key = colorMode.slice(2);
|
|
var selector = $el.data('selector');
|
|
filterRectangles(function(r) {
|
|
return r[key] === selector;
|
|
});
|
|
}
|
|
});
|
|
}
|
|
|
|
});
|