187 lines
6.8 KiB
JavaScript
187 lines
6.8 KiB
JavaScript
// ==UserScript==
|
|
// @id iitc-plugin-portal-level-numbers@rongou
|
|
// @name IITC plugin: Portal Level Numbers
|
|
// @category Layer
|
|
// @version 0.1.5.@@DATETIMEVERSION@@
|
|
// @namespace https://github.com/jonatkins/ingress-intel-total-conversion
|
|
// @updateURL @@UPDATEURL@@
|
|
// @downloadURL @@DOWNLOADURL@@
|
|
// @description [@@BUILDNAME@@-@@BUILDDATE@@] Show portal level numbers on map.
|
|
// @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 ////////////////////////////////////////////////////////
|
|
|
|
// use own namespace for plugin
|
|
window.plugin.portalLevelNumbers = function() {
|
|
};
|
|
|
|
window.plugin.portalLevelNumbers.ICON_SIZE = 12;
|
|
window.plugin.portalLevelNumbers.MOBILE_SCALE = 1.5;
|
|
|
|
window.plugin.portalLevelNumbers.levelLayers = {};
|
|
window.plugin.portalLevelNumbers.levelLayerGroup = null;
|
|
|
|
window.plugin.portalLevelNumbers.setupCSS = function() {
|
|
$("<style>")
|
|
.prop("type", "text/css")
|
|
.html(".plugin-portal-level-numbers {\
|
|
font-size: 10px;\
|
|
color: #FFFFBB;\
|
|
font-family: monospace;\
|
|
text-align: center;\
|
|
text-shadow: 0 0 0.5em black, 0 0 0.5em black, 0 0 0.5em black;\
|
|
pointer-events: none;\
|
|
-webkit-text-size-adjust:none;\
|
|
}")
|
|
.appendTo("head");
|
|
}
|
|
|
|
window.plugin.portalLevelNumbers.removeLabel = function(guid) {
|
|
var previousLayer = window.plugin.portalLevelNumbers.levelLayers[guid];
|
|
if(previousLayer) {
|
|
window.plugin.portalLevelNumbers.levelLayerGroup.removeLayer(previousLayer);
|
|
delete plugin.portalLevelNumbers.levelLayers[guid];
|
|
}
|
|
}
|
|
|
|
window.plugin.portalLevelNumbers.addLabel = function(guid,latLng) {
|
|
// remove old layer before updating
|
|
window.plugin.portalLevelNumbers.removeLabel(guid);
|
|
|
|
// add portal level to layers
|
|
var p = window.portals[guid];
|
|
var levelNumber = p.options.level;
|
|
var level = L.marker(latLng, {
|
|
icon: L.divIcon({
|
|
className: 'plugin-portal-level-numbers',
|
|
iconSize: [window.plugin.portalLevelNumbers.ICON_SIZE, window.plugin.portalLevelNumbers.ICON_SIZE],
|
|
html: levelNumber
|
|
}),
|
|
guid: guid
|
|
});
|
|
plugin.portalLevelNumbers.levelLayers[guid] = level;
|
|
level.addTo(plugin.portalLevelNumbers.levelLayerGroup);
|
|
}
|
|
|
|
window.plugin.portalLevelNumbers.updatePortalLabels = function() {
|
|
|
|
var SQUARE_SIZE = L.Browser.mobile ? (window.plugin.portalLevelNumbers.ICON_SIZE + 3) * window.plugin.portalLevelNumbers.MOBILE_SCALE
|
|
: (window.plugin.portalLevelNumbers.ICON_SIZE + 3);
|
|
|
|
// as this is called every time layers are toggled, there's no point in doing it when the layer is off
|
|
if (!map.hasLayer(window.plugin.portalLevelNumbers.levelLayerGroup)) {
|
|
return;
|
|
}
|
|
|
|
var portalPoints = {};
|
|
|
|
for (var guid in window.portals) {
|
|
var p = window.portals[guid];
|
|
if (p._map && p.options.data.level !== undefined) { // only consider portals added to the map, and that have a level set
|
|
var point = map.project(p.getLatLng());
|
|
portalPoints[guid] = point;
|
|
}
|
|
}
|
|
|
|
// for efficient testing of intersection, group portals into buckets based on the defined rectangle size
|
|
var buckets = {};
|
|
for (var guid in portalPoints) {
|
|
var point = portalPoints[guid];
|
|
|
|
var bucketId = L.point([Math.floor(point.x/(SQUARE_SIZE*2)),Math.floor(point.y/SQUARE_SIZE*2)]);
|
|
// the guid is added to four buckets. this way, when testing for overlap we don't need to test
|
|
// all 8 buckets surrounding the one around the particular portal, only the bucket it is in itself
|
|
var bucketIds = [bucketId, bucketId.add([1,0]), bucketId.add([0,1]), bucketId.add([1,1])];
|
|
for (var i in bucketIds) {
|
|
var b = bucketIds[i].toString();
|
|
if (!buckets[b]) buckets[b] = {};
|
|
buckets[b][guid] = true;
|
|
}
|
|
}
|
|
|
|
var coveredPortals = {};
|
|
|
|
for (var bucket in buckets) {
|
|
var bucketGuids = buckets[bucket];
|
|
for (var guid in bucketGuids) {
|
|
var point = portalPoints[guid];
|
|
// the bounds used for testing are twice as wide as the rectangle. this is so that there's no left/right
|
|
// overlap between two different portals text
|
|
var southWest = point.subtract([SQUARE_SIZE, SQUARE_SIZE]);
|
|
var northEast = point.add([SQUARE_SIZE, SQUARE_SIZE]);
|
|
var largeBounds = L.bounds(southWest, northEast);
|
|
|
|
for (var otherGuid in bucketGuids) {
|
|
// do not check portals already marked as covered
|
|
if (guid != otherGuid && !coveredPortals[otherGuid]) {
|
|
var otherPoint = portalPoints[otherGuid];
|
|
|
|
if (largeBounds.contains(otherPoint)) {
|
|
// another portal is within the rectangle - remove if it has not a higher level
|
|
if (portals[guid].options.level > portals[otherGuid].options.level) continue;
|
|
else coveredPortals[guid] = true;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
for (var guid in coveredPortals) {
|
|
delete portalPoints[guid];
|
|
}
|
|
|
|
// remove any not wanted
|
|
for (var guid in window.plugin.portalLevelNumbers.levelLayers) {
|
|
if (!(guid in portalPoints)) {
|
|
window.plugin.portalLevelNumbers.removeLabel(guid);
|
|
}
|
|
}
|
|
|
|
// and add those we do
|
|
for (var guid in portalPoints) {
|
|
window.plugin.portalLevelNumbers.addLabel(guid, portals[guid].getLatLng());
|
|
}
|
|
}
|
|
|
|
// as calculating portal marker visibility can take some time when there's lots of portals shown, we'll do it on
|
|
// a short timer. this way it doesn't get repeated so much
|
|
window.plugin.portalLevelNumbers.delayedUpdatePortalLabels = function(wait) {
|
|
|
|
if (window.plugin.portalLevelNumbers.timer === undefined) {
|
|
window.plugin.portalLevelNumbers.timer = setTimeout ( function() {
|
|
window.plugin.portalLevelNumbers.timer = undefined;
|
|
window.plugin.portalLevelNumbers.updatePortalLabels();
|
|
}, wait*1000);
|
|
|
|
}
|
|
}
|
|
|
|
var setup = function() {
|
|
|
|
window.plugin.portalLevelNumbers.setupCSS();
|
|
|
|
window.plugin.portalLevelNumbers.levelLayerGroup = new L.LayerGroup();
|
|
window.addLayerGroup('Portal Levels', window.plugin.portalLevelNumbers.levelLayerGroup, true);
|
|
|
|
window.addHook('requestFinished', function() { setTimeout(function(){window.plugin.portalLevelNumbers.delayedUpdatePortalLabels(3.0);},1); });
|
|
window.addHook('mapDataRefreshEnd', function() { window.plugin.portalLevelNumbers.delayedUpdatePortalLabels(0.5); });
|
|
window.map.on('overlayadd overlayremove', function() { setTimeout(function(){window.plugin.portalLevelNumbers.delayedUpdatePortalLabels(1.0);},1); });
|
|
|
|
}
|
|
|
|
// PLUGIN END //////////////////////////////////////////////////////////
|
|
|
|
@@PLUGINEND@@
|