321 lines
11 KiB
JavaScript
321 lines
11 KiB
JavaScript
// ==UserScript==
|
|
// @id iitc-plugin-cross-links@mcben
|
|
// @name IITC plugin: cross links
|
|
// @category Layer
|
|
// @version 1.1.2.@@DATETIMEVERSION@@
|
|
// @namespace https://github.com/jonatkins/ingress-intel-total-conversion
|
|
// @updateURL @@UPDATEURL@@
|
|
// @downloadURL @@DOWNLOADURL@@
|
|
// @description [@@BUILDNAME@@-@@BUILDDATE@@] EXPERIMENTAL: Checks for existing links that cross planned links. Requires draw-tools plugin.
|
|
// @include https://*.ingress.com/intel*
|
|
// @include http://*.ingress.com/intel*
|
|
// @match https://*.ingress.com/intel*
|
|
// @match http://*.ingress.com/intel*
|
|
// @include https://*.ingress.com/mission/*
|
|
// @include http://*.ingress.com/mission/*
|
|
// @match https://*.ingress.com/mission/*
|
|
// @match http://*.ingress.com/mission/*
|
|
// @grant none
|
|
// ==/UserScript==
|
|
|
|
@@PLUGINSTART@@
|
|
|
|
// PLUGIN START ////////////////////////////////////////////////////////
|
|
|
|
|
|
window.plugin.crossLinks = function() {};
|
|
|
|
|
|
window.plugin.crossLinks.greatCircleArcIntersect = function(a0,a1,b0,b1) {
|
|
// based on the formula at http://williams.best.vwh.net/avform.htm#Int
|
|
|
|
// method:
|
|
// check to ensure no line segment is zero length - if so, cannot cross
|
|
// check to see if either of the lines start/end at the same point. if so, then they cannot cross
|
|
// check to see if the line segments overlap in longitude. if not, no crossing
|
|
// if overlap, clip each line to the overlapping longitudes, then see if latitudes cross
|
|
|
|
// anti-meridian handling. this code will not sensibly handle a case where one point is
|
|
// close to -180 degrees and the other +180 degrees. unwrap coordinates in this case, so one point
|
|
// is beyond +-180 degrees. this is already true in IITC
|
|
// FIXME? if the two lines have been 'unwrapped' differently - one positive, one negative - it will fail
|
|
|
|
// zero length line tests
|
|
if (a0.equals(a1)) return false;
|
|
if (b0.equals(b1)) return false;
|
|
|
|
// lines have a common point
|
|
if (a0.equals(b0) || a0.equals(b1)) return false;
|
|
if (a1.equals(b0) || a1.equals(b1)) return false;
|
|
|
|
|
|
// check for 'horizontal' overlap in lngitude
|
|
if (Math.min(a0.lng,a1.lng) > Math.max(b0.lng,b1.lng)) return false;
|
|
if (Math.max(a0.lng,a1.lng) < Math.min(b0.lng,b1.lng)) return false;
|
|
|
|
|
|
// ok, our two lines have some horizontal overlap in longitude
|
|
// 1. calculate the overlapping min/max longitude
|
|
// 2. calculate each line latitude at each point
|
|
// 3. if latitudes change place between overlapping range, the lines cross
|
|
|
|
|
|
// class to hold the pre-calculated maths for a geodesic line
|
|
// TODO: move this outside this function, so it can be pre-calculated once for each line we test
|
|
var GeodesicLine = function(start,end) {
|
|
var d2r = Math.PI/180.0;
|
|
var r2d = 180.0/Math.PI;
|
|
|
|
// maths based on http://williams.best.vwh.net/avform.htm#Int
|
|
|
|
if (start.lng == end.lng) {
|
|
throw 'Error: cannot calculate latitude for meridians';
|
|
}
|
|
|
|
// only the variables needed to calculate a latitude for a given longitude are stored in 'this'
|
|
this.lat1 = start.lat * d2r;
|
|
this.lat2 = end.lat * d2r;
|
|
this.lng1 = start.lng * d2r;
|
|
this.lng2 = end.lng * d2r;
|
|
|
|
var dLng = this.lng1-this.lng2;
|
|
|
|
var sinLat1 = Math.sin(this.lat1);
|
|
var sinLat2 = Math.sin(this.lat2);
|
|
var cosLat1 = Math.cos(this.lat1);
|
|
var cosLat2 = Math.cos(this.lat2);
|
|
|
|
this.sinLat1CosLat2 = sinLat1*cosLat2;
|
|
this.sinLat2CosLat1 = sinLat2*cosLat1;
|
|
|
|
this.cosLat1CosLat2SinDLng = cosLat1*cosLat2*Math.sin(dLng);
|
|
}
|
|
|
|
GeodesicLine.prototype.isMeridian = function() {
|
|
return this.lng1 == this.lng2;
|
|
}
|
|
|
|
GeodesicLine.prototype.latAtLng = function(lng) {
|
|
lng = lng * Math.PI / 180; //to radians
|
|
|
|
var lat;
|
|
// if we're testing the start/end point, return that directly rather than calculating
|
|
// 1. this may be fractionally faster, no complex maths
|
|
// 2. there's odd rounding issues that occur on some browsers (noticed on IITC MObile) for very short links - this may help
|
|
if (lng == this.lng1) {
|
|
lat = this.lat1;
|
|
} else if (lng == this.lng2) {
|
|
lat = this.lat2;
|
|
} else {
|
|
lat = Math.atan ( (this.sinLat1CosLat2*Math.sin(lng-this.lng2) - this.sinLat2CosLat1*Math.sin(lng-this.lng1))
|
|
/ this.cosLat1CosLat2SinDLng);
|
|
}
|
|
return lat * 180 / Math.PI; // return value in degrees
|
|
}
|
|
|
|
|
|
|
|
// calculate the longitude of the overlapping region
|
|
var leftLng = Math.max( Math.min(a0.lng,a1.lng), Math.min(b0.lng,b1.lng) );
|
|
var rightLng = Math.min( Math.max(a0.lng,a1.lng), Math.max(b0.lng,b1.lng) );
|
|
|
|
// calculate the latitudes for each line at left + right longitudes
|
|
// NOTE: need a special case for meridians - as GeodesicLine.latAtLng method is invalid in that case
|
|
var aLeftLat, aRightLat;
|
|
if (a0.lng == a1.lng) {
|
|
// 'left' and 'right' now become 'top' and 'bottom' (in some order) - which is fine for the below intersection code
|
|
aLeftLat = a0.lat;
|
|
aRightLat = a1.lat;
|
|
} else {
|
|
var aGeo = new GeodesicLine(a0,a1);
|
|
aLeftLat = aGeo.latAtLng(leftLng);
|
|
aRightLat = aGeo.latAtLng(rightLng);
|
|
}
|
|
|
|
var bLeftLat, bRightLat;
|
|
if (b0.lng == b1.lng) {
|
|
// 'left' and 'right' now become 'top' and 'bottom' (in some order) - which is fine for the below intersection code
|
|
bLeftLat = b0.lat;
|
|
bRightLat = b1.lat;
|
|
} else {
|
|
var bGeo = new GeodesicLine(b0,b1);
|
|
bLeftLat = bGeo.latAtLng(leftLng);
|
|
bRightLat = bGeo.latAtLng(rightLng);
|
|
}
|
|
|
|
// if both a are less or greater than both b, then lines do not cross
|
|
|
|
if (aLeftLat < bLeftLat && aRightLat < bRightLat) return false;
|
|
if (aLeftLat > bLeftLat && aRightLat > bRightLat) return false;
|
|
|
|
// latitudes cross between left and right - so geodesic lines cross
|
|
return true;
|
|
}
|
|
|
|
|
|
|
|
window.plugin.crossLinks.testPolyLine = function (polyline, link,closed) {
|
|
|
|
var a = link.getLatLngs();
|
|
var b = polyline.getLatLngs();
|
|
|
|
for (var i=0;i<b.length-1;++i) {
|
|
if (window.plugin.crossLinks.greatCircleArcIntersect(a[0],a[1],b[i],b[i+1])) return true;
|
|
}
|
|
|
|
if (closed) {
|
|
if (window.plugin.crossLinks.greatCircleArcIntersect(a[0],a[1],b[b.length-1],b[0])) return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
window.plugin.crossLinks.onLinkAdded = function (data) {
|
|
if (window.plugin.crossLinks.disabled) return;
|
|
|
|
plugin.crossLinks.testLink(data.link);
|
|
}
|
|
|
|
window.plugin.crossLinks.checkAllLinks = function() {
|
|
if (window.plugin.crossLinks.disabled) return;
|
|
|
|
console.debug("Cross-Links: checking all links");
|
|
plugin.crossLinks.linkLayer.clearLayers();
|
|
plugin.crossLinks.linkLayerGuids = {};
|
|
|
|
$.each(window.links, function(guid, link) {
|
|
plugin.crossLinks.testLink(link);
|
|
});
|
|
}
|
|
|
|
window.plugin.crossLinks.testLink = function (link) {
|
|
if (plugin.crossLinks.linkLayerGuids[link.options.guid]) return;
|
|
|
|
for (var i in plugin.drawTools.drawnItems._layers) { // leaflet don't support breaking out of the loop
|
|
var layer = plugin.drawTools.drawnItems._layers[i];
|
|
if (layer instanceof L.GeodesicPolygon) {
|
|
if (plugin.crossLinks.testPolyLine(layer, link,true)) {
|
|
plugin.crossLinks.showLink(link);
|
|
break;
|
|
}
|
|
} else if (layer instanceof L.GeodesicPolyline) {
|
|
if (plugin.crossLinks.testPolyLine(layer, link)) {
|
|
plugin.crossLinks.showLink(link);
|
|
break;
|
|
}
|
|
}
|
|
};
|
|
}
|
|
|
|
|
|
window.plugin.crossLinks.showLink = function(link) {
|
|
|
|
var poly = L.geodesicPolyline(link.getLatLngs(), {
|
|
color: '#d22',
|
|
opacity: 0.7,
|
|
weight: 5,
|
|
clickable: false,
|
|
dashArray: [8,8],
|
|
|
|
guid: link.options.guid
|
|
});
|
|
|
|
poly.addTo(plugin.crossLinks.linkLayer);
|
|
plugin.crossLinks.linkLayerGuids[link.options.guid]=poly;
|
|
}
|
|
|
|
window.plugin.crossLinks.onMapDataRefreshEnd = function () {
|
|
if (window.plugin.crossLinks.disabled) return;
|
|
|
|
window.plugin.crossLinks.linkLayer.bringToFront();
|
|
|
|
window.plugin.crossLinks.testForDeletedLinks();
|
|
}
|
|
|
|
window.plugin.crossLinks.testAllLinksAgainstLayer = function (layer) {
|
|
if (window.plugin.crossLinks.disabled) return;
|
|
|
|
$.each(window.links, function(guid, link) {
|
|
if (!plugin.crossLinks.linkLayerGuids[link.options.guid])
|
|
{
|
|
if (layer instanceof L.GeodesicPolygon) {
|
|
if (plugin.crossLinks.testPolyLine(layer, link,true)) {
|
|
plugin.crossLinks.showLink(link);
|
|
}
|
|
} else if (layer instanceof L.GeodesicPolyline) {
|
|
if (plugin.crossLinks.testPolyLine(layer, link)) {
|
|
plugin.crossLinks.showLink(link);
|
|
}
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
window.plugin.crossLinks.testForDeletedLinks = function () {
|
|
window.plugin.crossLinks.linkLayer.eachLayer( function(layer) {
|
|
var guid = layer.options.guid;
|
|
if (!window.links[guid]) {
|
|
console.log("link removed");
|
|
plugin.crossLinks.linkLayer.removeLayer(layer);
|
|
delete plugin.crossLinks.linkLayerGuids[guid];
|
|
}
|
|
});
|
|
}
|
|
|
|
window.plugin.crossLinks.createLayer = function() {
|
|
window.plugin.crossLinks.linkLayer = new L.FeatureGroup();
|
|
window.plugin.crossLinks.linkLayerGuids={};
|
|
window.addLayerGroup('Cross Links', window.plugin.crossLinks.linkLayer, true);
|
|
|
|
map.on('layeradd', function(obj) {
|
|
if(obj.layer === window.plugin.crossLinks.linkLayer) {
|
|
delete window.plugin.crossLinks.disabled;
|
|
window.plugin.crossLinks.checkAllLinks();
|
|
}
|
|
});
|
|
map.on('layerremove', function(obj) {
|
|
if(obj.layer === window.plugin.crossLinks.linkLayer) {
|
|
window.plugin.crossLinks.disabled = true;
|
|
window.plugin.crossLinks.linkLayer.clearLayers();
|
|
plugin.crossLinks.linkLayerGuids = {};
|
|
}
|
|
});
|
|
|
|
// ensure 'disabled' flag is initialised
|
|
if (!map.hasLayer(window.plugin.crossLinks.linkLayer)) {
|
|
window.plugin.crossLinks.disabled = true;
|
|
}
|
|
}
|
|
|
|
var setup = function() {
|
|
if (window.plugin.drawTools === undefined) {
|
|
alert("'Cross-Links' requires 'draw-tools'");
|
|
return;
|
|
}
|
|
|
|
// this plugin also needs to create the draw-tools hook, in case it is initialised before draw-tools
|
|
window.pluginCreateHook('pluginDrawTools');
|
|
|
|
window.plugin.crossLinks.createLayer();
|
|
|
|
// events
|
|
window.addHook('pluginDrawTools',function(e) {
|
|
if (e.event == 'layerCreated') {
|
|
// we can just test the new layer in this case
|
|
window.plugin.crossLinks.testAllLinksAgainstLayer(e.layer);
|
|
} else {
|
|
// all other event types - assume anything could have been modified and re-check all links
|
|
window.plugin.crossLinks.checkAllLinks();
|
|
}
|
|
});
|
|
|
|
window.addHook('linkAdded', window.plugin.crossLinks.onLinkAdded);
|
|
window.addHook('mapDataRefreshEnd', window.plugin.crossLinks.onMapDataRefreshEnd);
|
|
|
|
|
|
}
|
|
|
|
// PLUGIN END //////////////////////////////////////////////////////////
|
|
|
|
@@PLUGINEND@@
|