// ==UserScript==
// @author DanielOnDiordna
// @name IITC plugin: Drone route planner
// @category Layer
// @version 1.1.3.20210724.002500
// @updateURL https://softspot.nl/ingress/plugins/iitc-plugin-drone-route-planner.meta.js
// @downloadURL https://softspot.nl/ingress/plugins/iitc-plugin-drone-route-planner.user.js
// @description [danielondiordna-1.1.3.20210724.002500] Plan a route for your drone (manually), draw your route between portals in range. Mark hacked portals. Use the drone half range layer to see if portals are within range. Import/export data functions available. Integrated Spectrum Colorpicker 1.8.1
// @id iitc-plugin-drone-route-planner@danielondiordna
// @namespace https://softspot.nl/ingress/
// @match https://intel.ingress.com/*
// @grant none
// ==/UserScript==
function wrapper(plugin_info) {
// ensure plugin framework is there, even if iitc is not yet loaded
if(typeof window.plugin !== 'function') window.plugin = function() {};
// use own namespace for plugin
window.plugin.dronerouteplanner = function() {};
var self = window.plugin.dronerouteplanner;
self.id = 'dronerouteplanner';
self.title = 'Drone route planner';
self.version = '1.1.3.20210724.002500';
self.author = 'DanielOnDiordna';
self.changelog = `
Changelog:
version 0.0.1.20201106.141900
- version 0.0.1
version 0.0.2.20200613.175800
- added a remote control with interactive buttons
- move all lines connected to selected portal
- delete all lines connected to selected portal
version 0.0.3.20200614.153600
- edit settings, colors, ranges
- store/restore settings
version 0.0.4.20200616.130100
- draggable remote control location (for PC only)
version 0.0.5.20200617.204700
- highlighter for showing half drone range circles
version 0.0.6.20200618.113800
- remote move buttons for mobile users
version 0.0.7.20200618.235400
- home made spinner buttons
- highlighter layer with more accurate circle drawing
version 1.0.0.20200620.005600
- improved remote control move buttons for pc users
- import/export data menu
- improved line menu
- toggle show/hide remote control
version 1.0.1.20200620.192700
- added a highlighter menu with options
version 1.0.2.20200620.233500
- added zoom button on the remote control
- if no portal is selected, the onoff button now zooms and selects the last portal on the route
version 1.0.3.20200621.234800
- added drone hacked button
- added highlighter option for drone hacked portals
version 1.0.4.20200623.171000
- added store/restore routes menu
- added remove all button
- fixed highlighter color for hacked portals
- show total hacked on highlighter menu
- show total lines on Remove all button
version 1.0.5.20200628.231500
- bug fixed: if drone route layer was disabled when iitc starts, the remote and menu would not show
version 1.1.0.20201228.123200
- half range circles removed from highlighter and placed on a layer
version 1.1.1.20210119.234100
- updated plugin wrapper and userscript header formatting to match IITC-CE coding
- integrated Spectrum Colorpicker 1.8.1 plugin code, no need anymore for the separate plugin
- added show drone range toggle option
version 1.1.1.20210421.190200
- minor fix for IITC CE where runHooks iitcLoaded is executed before addHook is defined in this plugin
version 1.1.2.20210523.163900
- fixed a bug when restoring a route
- changed the drone icon state to non-interactive, so portals behind the icon can be selected
- fixed the half range layer display (removed from highlighter)
- moved color settings to range and route setup
- added alpha color picker for half range layer color
- added About window with helpful usage information
version 1.1.3.20210525.203200
- improved drawing of half range layer circles during loading of portals
version 1.1.3.20210724.002500
- prevent double plugin setup on hook iitcLoaded
`;
self.namespace = 'window.plugin.' + self.id + '.';
self.pluginname = 'plugin-' + self.id;
self.routelayertitle = 'Drone routes';
self.halfrangelayertitle = 'Drone half range';
self.highlighthackedname = 'Drone hacked';
self.localstoragesettings = self.pluginname + '-settings';
self.localstoragelayer = self.pluginname + '-layer';
self.localstoragehacked = self.pluginname + '-hacked';
self.localstorageroutes = self.pluginname + '-routes';
self.isSmartphone = false;
self.selectedPortalGuid = undefined;
self.requestportalguid = undefined;
self.layer = undefined;
self.halfrangelayer = undefined;
self.hightlighteractive = false;
self.settings = {};
self.settings.range1radius = 500.0;
self.settings.range2radius = 750.0;
self.settings.range1color = 'yellow';
self.settings.range2color = 'red';
self.settings.linecolor = 'purple';
self.settings.linecolorinvalid = 'red';
self.settings.remote_top_offset = 0;
self.settings.remote_left_offset = 0;
self.settings.halfrangecolor = 'purple';
self.settings.halfrange = 250.0;
self.settings.halfrangeopacitypercent = 10;
self.settings.highlighterhacked = true;
self.settings.showremote = true;
self.settings.showdronerange = true;
self.hacked = [];
self.range1 = undefined;
self.range2 = undefined;
self.drawicon = '';
self.remoteicon = ''; // drone-remote-control-3.png
self.arrowicons = ''; // arrowmatrix_black_200.png
self.drawmarker = undefined;
self.drawmarkerguid = undefined;
self.drawmarkerpos = undefined;
self.selectedline = undefined;
self.remote_top_start = undefined;
self.remote_left_start = undefined;
self.drone_active = false;
self.move_active = false;
self.move_from_position = undefined;
self.portalcircles = {};
self.old_halfrangecolor = undefined;
self.old_halfrange = undefined;
self.old_halfrangeopacitypercent = undefined;
// need to initialise the 'spectrum' color picker
self.colorpickeroptions = {
flat: false,
showInput: true,
showButtons: true,
showPalette: true,
showSelectionPalette: true,
allowEmpty: false,
hideAfterPaletteSelect: true,
palette: [
['#004000','#008000','#00C000'],
['#00FF00','#80FF80','#C0FFC0'],
['#000040','#000080','#0000C0'],
['#4040FF','#8080FF','#C0C0FF'],
['#6A3400','#964A00','#C05F00'],
['#E27000','#FF8309','#FFC287'],
['#a24ac3','#514ac3','#4aa8c3','#51c34a'],
['#c1c34a','#c38a4a','#c34a4a','#c34a6f'],
['#000000','#666666','#bbbbbb','#ffffff']
]};
self.restoresettings = function() {
if (typeof localStorage[self.localstoragesettings] === 'string' && localStorage[self.localstoragesettings] !== '') {
try {
var settings = JSON.parse(localStorage[self.localstoragesettings]);
if (typeof settings === 'object' && settings instanceof Object && !(settings instanceof Array)) {
for (const i in settings) {
if (typeof settings[i] === typeof self.settings[i]) self.settings[i] = settings[i];
}
}
} catch(e) {
return false;
}
}
};
self.storesettings = function() {
localStorage[self.localstoragesettings] = JSON.stringify(self.settings);
};
self.restorearray = function(localstoragename) {
var data = [];
if (typeof localStorage[localstoragename] === 'string' && localStorage[localstoragename] !== '') {
try {
data = JSON.parse(localStorage[localstoragename]);
if (typeof data !== 'object' || !(data instanceof Object) || !(data instanceof Array)) {
return [];
}
} catch(e) {
return [];
}
}
return data;
};
self.storearray = function(localstoragename,data) {
localStorage[localstoragename] = JSON.stringify(data);
};
self.copydata = function() {
if (self.linecount() === 0) {
alert('No drawn lines');
return;
}
if (typeof android !== 'undefined' && android && android.shareString) {
android.shareString(localStorage[self.localstoragelayer]);
return;
}
var data = JSON.parse(localStorage[self.localstoragelayer]);
var html = '
' +
' ' +
' ' +
'Regular range: ' +
' ' +
'Maximum range: ' +
'Color for route lines: ' +
' within maximum range ' +
' out of maximum range ' +
'Half range layer: ' +
' Drone half range: ' +
'
';
window.dialog({
html: html,
id: self.id,
dialogClass: 'ui-dialog-' + self.pluginname,
title: self.title + ' - Range and route setup'
}).dialog('option', 'buttons', {
'< Main menu': function() { self.menu(); },
'Close': function() { $(this).dialog('close'); },
});
self.spinner(
{
object: '#range1spinner',
step: 10,
min: 150,
max: 1000,
appendstring: 'm',
changed: function(value) {
self.settings.range1radius = value;
self.storesettings();
if (self.range1) {
self.range1.setRadius(self.settings.range1radius);
}
self.updateLineColors();
}
}
);
self.spinner(
{
object: '#range2spinner',
step: 10,
min: 160,
max: 1200,
appendstring: 'm',
changed: function(value) {
self.settings.range2radius = value;
self.storesettings();
if (self.range2) {
self.range2.setRadius(self.settings.range2radius);
}
self.updateLineColors();
}
}
);
self.spinner(
{
object: '#halfrangespinner',
step: 10,
min: 150,
max: 1200,
appendstring: 'm',
changed: function(value) {
self.settings.halfrange = value;
self.storesettings();
self.updatehalfrangelayer();
}
}
);
$('#drone_linecolor').spectrum($.extend(true, self.colorpickeroptions, {
change: function(color) { self.settings.linecolor = color.toHexString(); self.storesettings(); self.updateLineColors(); },
color: self.settings.linecolor,
}));
$('#drone_linecolorinvalid').spectrum($.extend(true, self.colorpickeroptions, {
change: function(color) { self.settings.linecolorinvalid = color.toHexString(); self.storesettings(); self.updateLineColors(); },
color: self.settings.linecolorinvalid,
}));
$('#drone_range1color').spectrum($.extend(true, self.colorpickeroptions, {
change: function(color) { self.settings.range1color = color.toHexString(); self.storesettings(); if (self.range1) { self.range1.setStyle({color:self.settings.range1color}); }},
color: self.settings.range1color,
}));
$('#drone_range2color').spectrum($.extend(true, self.colorpickeroptions, {
change: function(color) { self.settings.range2color = color.toHexString(); self.storesettings(); if (self.range1) { self.range2.setStyle({color:self.settings.range2color}); }},
color: self.settings.range2color,
}));
$('#drone_halfrangecolor').spectrum($.extend(true, self.colorpickeroptions, {
showAlpha: true,
change: function(color) { self.settings.halfrangecolor = color.toHexString(); self.settings.halfrangeopacitypercent = color.getAlpha() * 100; self.storesettings(); self.updatehalfrangelayer(); },
color: tinycolor(self.settings.halfrangecolor).setAlpha(self.settings.halfrangeopacitypercent / 100),
}));
};
self.about = function() {
let container = document.createElement('div');
let aboutarea = container.appendChild(document.createElement('div'));
aboutarea.innerHTML = `How to properly use the Drone route planner
This plugin has many features, with 1 basic purpose:
- Mark portals as hacked with your Drone, to keep track of your "Unique Portals Drone" state.
Enable the Drone routes layer, to display a remote control for your Drone (by default at the bottom of the screen).
If a portal is selected, the plugin will draw two range circles around that portal:
- Regular range for your Drone, which is always possible.
- Maximum range for your Drone, which is possible in certain conditions.
You can edit the range distance and colors from the Drone menu. You can also disable it to show no ranges on the map.
With the remote control, you can mark a selected portal as Hacked with button "H".
The button H is red if portal is not yet hacked, and green if it is hacked.
Remember to enable the highlighter "Drone hacked" to show the portals you have marked as hacked.
With the remote control, you can draw a drone route:
1. select a portal
2. switch on the remote control (button turns green) and a Drone icon shows on the selected portal
3. select another portal, within the regular or maximum range for your drone
4. the Drone icon shows on the other portal, and a route link is drawn between the portals
5. you can continue selecting portals to plan a route for your drone
6. when you are finished, switch off the remote control
If you made a mistake, and want to draw your route to another portal, you can enable the move button on the remote control (button turns purple).
If move is enabled, all the route lins connected to the Drone icon portal is moved to another portal you selected.
The trashcan button can be used to delete all routes connected to the selected portal.
Use the looking glass button to zoom the map to see the complete route in view.
The remote control has a menu button ☰
This will show the menu, same as from the normal menu in the portal details dialog.
From the menu there are a lot of extra functions:
- range and route setup to change colors and distances
- edit the half range layer color and distance (if two half range circles touch, your drone can reach that portal)
- enable the move function for the remote control (or reset it's position)
- Zoom to view total route, same as the looking glass button on the remote
- import/export your list of hacked portals
- import/export your routes
- store/restore the routes, so you can draw a new route, or combine stored routes
- remove the current route completely
`.replace(/\n/g,' \n');
let author = container.appendChild(document.createElement('div'));
author.style.fontStyle = 'italic';
author.style.fontSize = 'smaller';
author.textContent = self.title + ' version ' + self.version + ' by ' + self.author;
window.dialog({
html: container,
id: self.id,
width: 'auto',
title: self.title + ' - About'
}).dialog('option', 'buttons', {
'< Main menu': function() { self.menu(); },
'Changelog': function() { alert(self.changelog); },
'Close': function() { $(this).dialog('close'); },
});
};
self.menu = function() {
var html = '