// ==UserScript== // @author DanielOnDiordna // @name Drone route planner // @category Layer // @version 1.1.3.20210724.002500 // @updateURL https://raw.githubusercontent.com/IITC-CE/Community-plugins/master/dist/DanielOnDiordna/drone-route-planner.meta.js // @downloadURL https://raw.githubusercontent.com/IITC-CE/Community-plugins/master/dist/DanielOnDiordna/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 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 = '

Select all and press CTRL+C to copy it.

' + ''; window.dialog({ html: html, id: self.pluginname + '-dialog-export-data', dialogClass: 'ui-dialog-' + self.pluginname + '-export', title: self.title + ' Copy', width: 600 }); }; self.pastedata = function(appenddata) { if (!appenddata && self.linecount() !== 0 && !confirm('Are you sure you want to replace all lines?')) return; var promptAction = prompt('Press CTRL+V to paste (' + (appenddata?'append':'replace') + ' ' + self.title + ' route data).', ''); if (promptAction === null || promptAction === '') return; try { var data = JSON.parse(promptAction); if (!appenddata) self.layer.clearLayers(); self.import(data); self.storelines(); alert('Import Successful.'); } catch(e) { alert('Import failed'); } }; self.importfile = function() { if (window.requestFile === undefined) return; window.requestFile(function(filename, content) { try { var data = JSON.parse(content); self.import(data); self.storelines(); alert('Import Successful.'); } catch(e) { alert('Import failed'); } }); }; self.savefile = function() { if(typeof android !== 'undefined' && android && android.saveFile) { android.saveFile('IITC-' + self.id + '-route.json', 'text/plain', localStorage[self.localstoragelayer]); } }; self.copydatahacked = function() { if (typeof android !== 'undefined' && android && android.shareString) { android.shareString(localStorage[self.localstoragehacked]); return; } var data = JSON.parse(localStorage[self.localstoragehacked]); var html = '

Select all and press CTRL+C to copy it.

' + ''; window.dialog({ html: html, id: self.pluginname + '-dialog-export-data', dialogClass: 'ui-dialog-' + self.pluginname + '-export', title: self.title + ' Copy', width: 600 }); }; self.pastedatahacked = function(appenddata) { if (!appenddata && !confirm('Are you sure you want to replace all hacked portals?')) return; var promptAction = prompt('Press CTRL+V to paste (' + (appenddata?'append':'replace') + ' ' + self.title + ' hacked data).', ''); if (promptAction === null || promptAction === '') return; try { var data = JSON.parse(promptAction); if (typeof data === 'object' && (data instanceof Array)) { if (!appenddata) self.hacked = []; self.importhacked(data); self.storearray(self.localstoragehacked,self.hacked); alert('Import Successful.'); } else { alert('Import failed, expected an array of guids'); } } catch(e) { alert('Import failed'); } }; self.importhacked = function(data) { self.hacked = data; self.updateHighlighter(); }; self.importfilehacked = function() { if (window.requestFile === undefined) return; window.requestFile(function(filename, content) { try { var data = JSON.parse(content); self.importhacked(data); alert('Import Successful.'); } catch(e) { alert('Import failed'); } }); }; self.savefilehacked = function() { if(typeof android !== 'undefined' && android && android.saveFile) { android.saveFile('IITC-' + self.id + '-hacked.json', 'text/plain', localStorage[self.localstoragehacked]); } }; self.linecount = function(position) { var linecount = 0; self.layer.eachLayer( function(line) { if (line instanceof L.GeodesicPolyline && line.getLatLngs().length === 2) { var latLngs = line.getLatLngs(); if (!position || position.lat === latLngs[0].lat && position.lng === latLngs[0].lng || position.lat === latLngs[1].lat && position.lng === latLngs[1].lng) { linecount++; } } }); return linecount; }; self.hightlight_portal = function(portal) { let guid = portal.options.guid; let params = {opacity: 1.0, fillOpacity: 0.5}; if (self.settings.highlighterhacked && self.portal_hacked(guid)) { params.fillColor = 'red'; params.fillOpacity = 1; } else { params.fillColor = window.COLORS[portal.options.team]; params.fillOpacity = 0.5; } if (guid == window.selectedPortal) { params.color = COLOR_SELECTED_PORTAL; params.opacity = 1; console.log('hightlight_portal selectedPortal ',params.fillColor); } portal.setStyle(params); }; self.highlightportals = { highlight: function(data) { if (!data || !(data instanceof Object)) return; if (data.guid && window.portals[data.guid]) data.portal = window.portals[data.guid]; if (!data || !data.portal || !data.portal.options || !data.portal.options.guid) return; if (window._current_highlighter === self.highlighthackedname) { self.hightlight_portal(data.portal); } }, setSelected: function(active) { self.hightlighteractive = active; if (!active) { //hide layer //map.removeLayer(self.halfrangelayer); } else { //show layer //map.addLayer(self.halfrangelayer); if (self.layer._map) self.layer.bringToBack(); //self.halfrangelayer.bringToBack(); } } }; self.updateHighlighter = function() { if (!self.hightlighteractive) return; for (let guid in window.portals) { self.hightlight_portal(window.portals[guid]); } }; self.drawhalfrangecircles = function(guid,latlng) { if (self.portalcircles[guid]) return; var circleOptions = { stroke: true, color: self.settings.halfrangecolor, weight: 1, opacity: Math.max(self.settings.halfrangeopacitypercent / 100 * 2), fill: true, fillColor: null, // to use the same as 'color' for fill fillOpacity: self.settings.halfrangeopacitypercent / 100, clickable: false, interactive: false }; self.portalcircles[guid] = new L.geodesicCircle(latlng, self.settings.halfrange, circleOptions); self.halfrangelayer.addLayer(self.portalcircles[guid]); }; self.onportalAdded = function(data) { // data = {portal: marker, previousData: previousData} let guid = data.portal.options.guid; let latlng = data.portal.getLatLng(); let visiblebounds = map.getBounds(); if (!self.portalcircles[guid] && visiblebounds.contains(latlng)) { self.drawhalfrangecircles(guid,latlng); } }; self.onportalRemoved = function(data) { // data = {portal: p, data: p.options.data } let guid = data.portal.options.guid; if (self.portalcircles[guid]) { self.halfrangelayer.removeLayer(self.portalcircles[guid]); delete(self.portalcircles[guid]); } }; self.updatehalfrangelayer = function() { if (!self.halfrangelayer._map) return; // only if layer is active let visiblebounds = map.getBounds(); // update color, range and opacity for existing circles, but only if changed // and remove circles if not within bounds: let outofbounds = []; for (const guid in self.portalcircles) { if (visiblebounds.contains(self.portalcircles[guid].getLatLng())) { if (self.settings.halfrangecolor !== self.old_halfrangecolor || self.settings.halfrangeopacitypercent !== self.old_halfrangeopacitypercent) self.portalcircles[guid].setStyle({color: self.settings.halfrangecolor,opacity: self.settings.halfrangeopacitypercent / 100 * 2,fillOpacity: self.settings.halfrangeopacitypercent / 100}); if (self.settings.halfrange !== self.old_halfrange) self.portalcircles[guid].setRadius(self.settings.halfrange); } else { outofbounds.push(guid); } } self.old_halfrangecolor = self.settings.halfrangecolor; self.old_halfrange = self.settings.halfrange; self.old_halfrangeopacitypercent = self.settings.halfrangeopacitypercent; // remove circles not within bounds: for (let cnt = 0; cnt < outofbounds.length; cnt++) { let guid = outofbounds[cnt]; self.halfrangelayer.removeLayer(self.portalcircles[guid]); delete(self.portalcircles[guid]); } // draw new circles within bounds: for (const guid in window.portals) { let latlng = window.portals[guid].getLatLng(); if (visiblebounds.contains(latlng)) { self.drawhalfrangecircles(guid,latlng); } } if (self.halfrangelayer._map) self.halfrangelayer.bringToBack(); }; self.move_off = function() { self.move_active = false; document.getElementById("move").style.backgroundColor = "rgba(255,255,255,.5)"; }; self.move_on = function() { if (!window.selectedPortal || self.move_active) return false; var lines = self.getconnectedlines(); if (lines.length === 0) { alert('No connected lines to move'); return; } self.drone_on(); self.move_active = true; self.move_from_position = window.portals[window.selectedPortal].getLatLng(); document.getElementById("move").style.backgroundColor = "rgba(128,0,255,.5)"; }; self.drone_off = function() { self.drone_active = false; document.getElementById("onoff").style.backgroundColor = "rgba(255,0,0,.5)"; self.dronemarker_remove(); self.move_off(); }; self.portal_hacked = function(guid) { return self.hacked.indexOf(guid) >= 0; }; self.toggle_hacked = function() { if (!window.selectedPortal) return; let index = self.hacked.indexOf(window.selectedPortal); if (index >= 0) { self.hacked.splice(index, 1); } else { self.hacked.push(window.selectedPortal); } self.update_hacked(); self.hightlight_portal(window.portals[window.selectedPortal]); self.storearray(self.localstoragehacked,self.hacked); }; self.update_hacked = function() { if (!window.selectedPortal) { document.getElementById("hacked").style.backgroundColor = "rgba(255,255,255,.5)"; } else if (self.portal_hacked(window.selectedPortal)) { document.getElementById("hacked").style.backgroundColor = "rgba(0,255,0,.5)"; } else { document.getElementById("hacked").style.backgroundColor = "rgba(255,0,0,.5)"; } }; self.drone_on = function() { if (self.drone_active) return false; if (!window.selectedPortal) { // select and zoom to last portal let lastlinelatLngs = undefined; self.layer.eachLayer( function(line) { if (line instanceof L.GeodesicPolyline && line.getLatLngs().length === 2) { lastlinelatLngs = line.getLatLngs(); } }); if (lastlinelatLngs == undefined) return false; var visibleBounds = map.getBounds(); var position = lastlinelatLngs[1]; if (!visibleBounds.contains(position)) window.map.setView(position, map.getZoom()); let guid = self.findPortalGuidbyposition(position); if (window.portals[guid]) { window.renderPortalDetails(guid); } else { window.portalDetail.request(guid); } return false; } self.drone_active = true; document.getElementById("onoff").style.backgroundColor = "rgba(0,255,0,.5)"; self.dronemarker_draw(); }; self.toggle_onoff = function() { if (self.drone_active) { self.drone_off(); } else { self.drone_on(); } }; self.toggle_move = function() { if (self.move_active) { self.move_off(); } else { self.move_on(); } }; self.getdroneicon = function() { var zoom = map.getZoom(); var size = Math.max(30,zoom * 4); let icon = L.icon({ iconUrl: self.drawicon, iconSize: [size, size], // size of the icon iconAnchor: [size/2, size/2] // point of the icon which will correspond to marker's location }); return icon; }; self.showremote = function() { if (!self.layeractive(self.routelayertitle)) return; if ($('.DroneRemote').length) self.hideremote(); if (!self.settings.showremote) return; $('#updatestatus').prepend( '
' + '' + '' + '' + '' + '' + '
'); $('.DroneRemote').css('margin-top',(-132 + self.settings.remote_top_offset) + 'px').css('margin-left',(-50 + self.settings.remote_left_offset) + 'px'); // setup buttons self.drone_off(); self.move_off(); self.update_hacked(); }; self.hideremote = function() { $('.DroneRemote').remove(); }; self.move_remote_end = function() { if (!$('.draggable').length) return; if (!self.isSmartphone) $('.DroneRemote').css('cursor','default').css('pointer-events','none').draggable('destroy'); $('.draggable').remove(); var position = $('.DroneRemote').position(); self.settings.remote_top_offset += position.top - self.remote_top_start; self.settings.remote_left_offset += position.left - self.remote_left_start; self.storesettings(); self.remote_top_start = undefined; self.remote_left_start = undefined; }; self.move_remote_cancel = function() { if (!$('.draggable').length) return; if (!self.isSmartphone) $('.DroneRemote').css('cursor','default').css('pointer-events','none').draggable('destroy'); $('.draggable').remove(); self.hideremote(); self.showremote(); self.remote_top_start = undefined; self.remote_left_start = undefined; }; self.remote_move = function(x,y) { var pos = $('.DroneRemote').position(); pos.left += x * 5; pos.top += y * 5; $('.DroneRemote').css({left: pos.left + 'px', top: pos.top + 'px'}); }; self.move_remote_start = function() { if (!$('.DroneRemote').length) return; self.move_remote_end(); var position = $('.DroneRemote').position(); self.remote_top_start = position.top; self.remote_left_start = position.left; if (self.isSmartphone) { $('.DroneRemote').append( '
' + '
' + ' ' + ' ' + ' ' + ' ' + ' ' + ' ' + ' ' + ' ' + '
' + ' ' + ' OK
' + '
' + ' Cancel' + '
' + '
'); } else { $('.DroneRemote').append( '
' + ' ' + ' OK
' + '
' + ' Cancel' + '
' + '
'); $('.DroneRemote').css('cursor','move').css('pointer-events','auto').draggable(); } }; self.move_remote_reset = function() { self.move_remote_end(); self.settings.remote_top_offset = 0; self.settings.remote_left_offset = 0; self.storesettings(); self.hideremote(); self.showremote(); }; self.drawmarker_redraw = function() { if (self.drawmarker === undefined) return; self.drawmarker.setIcon(self.getdroneicon()); }; self.dronemarker_remove = function() { if (self.drawmarker === undefined) return; // remove marker map.removeLayer(self.drawmarker); self.drawmarker = undefined; }; self.dronemarker_draw = function() { self.dronemarker_remove(); // redraw if (!window.selectedPortal) { self.drone_off(); return; } var guid = window.selectedPortal; if (!window.portals[guid]) return; var portal = window.portals[guid]; self.drawmarkerpos = portal.getLatLng(); self.drawmarkerguid = guid; self.drawmarker = L.marker([self.drawmarkerpos.lat, self.drawmarkerpos.lng], {icon: self.getdroneicon(), interactive: false}).addTo(map); }; self.lineexists = function(latLngs) { var lineexists = false; self.layer.eachLayer( function(line) { if (line instanceof L.GeodesicPolyline && line.getLatLngs().length === 2) { var existinglatLngs = line.getLatLngs(); if ((existinglatLngs[0].lat === latLngs[0].lat && existinglatLngs[0].lng === latLngs[0].lng && existinglatLngs[1].lat === latLngs[1].lat && existinglatLngs[1].lng === latLngs[1].lng) || (existinglatLngs[0].lat === latLngs[1].lat && existinglatLngs[0].lng === latLngs[1].lng && existinglatLngs[1].lat === latLngs[0].lat && existinglatLngs[1].lng === latLngs[0].lng)){ lineexists = true; } } }); return lineexists; }; self.removeline = function(line) { self.layer.removeLayer(line); self.storelines(); }; self.removeAll = function(noconfirm) { if (self.linecount() === 0) { if (!noconfirm) alert('No drawn lines to remove'); return; } if (!noconfirm && !confirm('Remove route? ' + self.linecount() + ' lines')) return; delete localStorage[self.localstoragelayer]; self.layer.eachLayer( function(line) { if (line instanceof L.GeodesicPolyline && line.getLatLngs().length === 2) { self.layer.removeLayer(line); } }); self.storelines(); }; self.drawline = function(latLngs) { if (latLngs[0].lat === latLngs[1].lat && latLngs[0].lng === latLngs[1].lng) return; // no 0 length lines if (self.lineexists(latLngs)) return; // no double lines var length = L.latLng(latLngs[0].lat,latLngs[0].lng).distanceTo([latLngs[1].lat,latLngs[1].lng]); var color = (length <= self.settings.range2radius ? self.settings.linecolor : self.settings.linecolorinvalid); var lineOptions = { stroke: true, color: color, weight: 4, opacity: 0.8, fill: false, clickable: true }; var line = L.geodesicPolyline(latLngs, L.extend({}, lineOptions, {color: color, dashArray: null})); line.on('click',function(e) { self.linemenu(line); }); self.layer.addLayer(line); return length; }; self.addline = function() { if (!self.drawmarker || !window.selectedPortal || !window.portals[window.selectedPortal]) return; var startpos = self.drawmarkerpos; var endpos = window.portals[window.selectedPortal].getLatLng(); var latLngs = [new L.LatLng(startpos.lat,startpos.lng),new L.LatLng(endpos.lat,endpos.lng)]; var length = self.drawline(latLngs); self.storelines(); }; self.moveconnectedlines = function() { if (!self.drawmarker || !window.selectedPortal || !window.portals[window.selectedPortal]) return; var guid = window.selectedPortal; var portal = window.portals[guid]; var newll = portal.getLatLng(); var movefromposition = self.move_from_position; self.layer.eachLayer( function(line) { if (line instanceof L.GeodesicPolyline && line.getLatLngs().length === 2) { var moved = false; var latLngs = self.clonelatlngs(line.getLatLngs()); if (latLngs[0].lat === movefromposition.lat && latLngs[0].lng === movefromposition.lng) { latLngs[0] = new L.LatLng(newll.lat,newll.lng); moved = true; } else if (latLngs[1].lat === movefromposition.lat && latLngs[1].lng === movefromposition.lng) { latLngs[1] = new L.LatLng(newll.lat,newll.lng); moved = true; } if (moved) { if ((latLngs[0].lat === latLngs[1].lat && latLngs[0].lng === latLngs[1].lng) || self.lineexists(latLngs)) { // remove line if start and end match or if line already exists self.layer.removeLayer(line); } else { line.setLatLngs(latLngs); var length = L.latLng(latLngs[0].lat,latLngs[0].lng).distanceTo([latLngs[1].lat,latLngs[1].lng]); var color = (length <= self.settings.range2radius ? self.settings.linecolor : self.settings.linecolorinvalid); line.setStyle({color:color}); } } } }); self.storelines(); }; self.updateLineColors = function() { self.layer.eachLayer( function(line) { if (line instanceof L.GeodesicPolyline && line.getLatLngs().length === 2) { var latLngs = line.getLatLngs() var length = L.latLng(latLngs[0].lat,latLngs[0].lng).distanceTo([latLngs[1].lat,latLngs[1].lng]); var color = (length <= self.settings.range2radius ? self.settings.linecolor : self.settings.linecolorinvalid); line.setStyle({color:color}); } }); self.storelines(); }; self.totalrange = function() { var maxrange = 0; self.layer.eachLayer( function(line) { if (line instanceof L.GeodesicPolyline && line.getLatLngs().length === 2) { var range = 0; if (range > maxrange) maxrange = range; } }); return maxrange < 100000 ? Math.round(maxrange) + 'm' : Math.round(maxrange/1000) + 'km'; }; self.storelines = function() { var data = []; self.layer.eachLayer( function(line) { if (line instanceof L.GeodesicPolyline && line.getLatLngs().length === 2) { var item = {}; item.type = 'polyline'; item.latLngs = line.getLatLngs(); data.push(item); } }); localStorage[self.localstoragelayer] = JSON.stringify(data); }; self.import = function(data) { $.each(data, function(index,item) { switch(item.type) { case 'polyline': self.drawline(item.latLngs); break; default: console.warn(self.id + ': unknown import layer type "'+item.type); break; } }); }; self.restorelines = function() { try { var dataStr = localStorage[self.localstoragelayer]; if (dataStr === undefined) return; var data = JSON.parse(dataStr); self.import(data); } catch(e) { console.warn(self.id + ': failed to load data from localStorage: '+e); } }; self.updatedronerange = function() { // remove range if (self.range1) self.layer.removeLayer(self.range1); if (self.range2) self.layer.removeLayer(self.range2); self.range1 = undefined; self.range2 = undefined; if (!window.selectedPortal || !self.settings.showdronerange) return; // draw range var latLng = window.portals[self.selectedPortalGuid].getLatLng(); var circleOptions = { stroke: true, color: self.settings.range1color, weight: 2, opacity: 1.0, fill: true, fillColor: null, // to use the same as 'color' for fill fillOpacity: 0.2, clickable: false, interactive: false }; var extraOpt = {}; extraOpt.portalguid = self.selectedPortalGuid; extraOpt.title = window.portals[self.selectedPortalGuid].options.data.title; self.range1 = L.geodesicCircle(latLng, self.settings.range1radius, L.extend({},circleOptions,extraOpt)); circleOptions.color = self.settings.range2color; circleOptions.weight = 1; circleOptions.fillOpacity = 0.1; self.range2 = L.geodesicCircle(latLng, self.settings.range2radius, L.extend({},circleOptions,extraOpt)); self.layer.addLayer(self.range2).bringToBack(); self.layer.addLayer(self.range1).bringToBack(); self.layer.bringToBack(); //self.hightlighterlayer.bringToBack(); self.updatehalfrangelayer(); if (self.drawmarker !== undefined) { // there is a start marker if (self.drawmarkerguid !== self.selectedPortalGuid) { // another portal then the start portal is selected if (self.move_active) { self.moveconnectedlines(); self.move_from_position = latLng; } else { self.addline(); } self.dronemarker_draw(); } } }; self.onPortalSelected = function() { if (!self.layer._map) return; if (self.selectedPortalGuid === window.selectedPortal) return; if (self.selectedPortalGuid) { // remove range if (self.range1) self.layer.removeLayer(self.range1); if (self.range2) self.layer.removeLayer(self.range2); self.range1 = undefined; self.range2 = undefined; } self.selectedPortalGuid = window.selectedPortal; self.update_hacked(); if (!self.selectedPortalGuid) { self.drone_off(); return; } // draw range if (!window.portals[self.selectedPortalGuid]) return; self.updatedronerange(); }; self.clonelatlngs = function(latlngs) { var newlatlngs = []; for (var cnt = 0; cnt < latlngs.length; cnt++) { newlatlngs.push(new L.LatLng(latlngs[cnt].lat,latlngs[cnt].lng)); } return newlatlngs; }; self.removeselectedline = function() { if (!self.selectedline) return; self.removeline(self.selectedline); self.selectedline = undefined; self.closemenu(); }; self.linelength = function(line) { if (!line) return ''; var latlngs = line.getLatLngs(); var length = L.latLng(latlngs[0].lat,latlngs[0].lng).distanceTo([latlngs[1].lat,latlngs[1].lng]); return length < 100000 ? Math.round(length) + 'm' : Math.round(length/1000) + 'km'; }; self.layeractive = function(layername) { var layers = window.layerChooser.getLayers().overlayLayers; for (var id in layers) { if (layers[id].name === layername) { if (layers[id].active) { return true; } } } return false; }; self.closemenu = function() { $(".ui-dialog-content").dialog("close"); }; self.getconnectedlines = function() { var lines = []; if (!window.selectedPortal) lines; var position = window.portals[window.selectedPortal].getLatLng(); self.layer.eachLayer( function(line) { if (line instanceof L.GeodesicPolyline && line.getLatLngs().length === 2) { var latLngs = line.getLatLngs(); if (position.lat === latLngs[0].lat && position.lng === latLngs[0].lng || position.lat === latLngs[1].lat && position.lng === latLngs[1].lng) { lines.push(line); } } }); return lines; }; self.deleteconnectedlines = function() { if (!window.selectedPortal) return; var lines = self.getconnectedlines(); if (lines.length === 0) { alert('No connected lines to remove'); return; } if (!confirm('Lines connected to selected portal: '+ lines.length + '\n\nDelete?')) return; for (var i in lines) { self.layer.removeLayer(lines[i]); } self.storelines(); }; self.portaltitle = function(guid) { if (Object.keys(window.portals).length === 0) return '(no portals loaded)'; // cancel while no portals loaded yet; prevents an error inside the IITC core if (!guid) return '(no guid loaded)'; if (!window.portals[guid]) return '(portal not found)'; if (!window.portals[guid].options.data.title) return '(title not found)'; return window.portals[guid].options.data.title; }; self.selectportalbyguid = function(guid) { if (Object.keys(window.portals).length === 0) return; // cancel while no portals loaded yet; prevents an error inside the IITC core if (!guid) return false; if (guid === window.selectedPortal) { var visibleBounds = map.getBounds(); var position = window.portals[guid].getLatLngs(); if (!visibleBounds.contains(position)) window.map.setView(position, map.getZoom()); return true; } if (window.portals[guid]) { window.renderPortalDetails(guid); } else { window.portalDetail.request(guid); return false; } return true; }; self.findPortalGuidbyposition = function(position) { for (var guid in window.portals) { var portalposition = window.portals[guid].getLatLng(); if (portalposition.lat === position.lat && portalposition.lng == position.lng) { return guid; } } return undefined; }; self.linemenu = function(line) { if (!line) return; if (self.drawmarker) return; // disable menu when drawing a new line self.selectedline = line; let latLngs = line.getLatLngs(); let guid_start = self.findPortalGuidbyposition(latLngs[0]); let guid_end = self.findPortalGuidbyposition(latLngs[1]); var html = '
' + '' + self.portaltitle(guid_start) + '
' + '' + self.portaltitle(guid_end) + '
' + 'Delete this line' + '
'; window.dialog({ html: html, id: self.id, dialogClass: 'ui-dialog-' + self.pluginname, title: self.title + ' (' + self.linelength(line) + ')' }).dialog('option', 'buttons', { '< Main menu': function() { self.menu(); }, 'Close': function() { $(this).dialog('close'); }, }); }; self.showAll = function() { if (Object.keys(self.layer._layers).length === 0) { alert('No drawn lines to display'); return; } map.fitBounds(self.layer.getBounds()); }; self.spin_down = function(settings) { let oldvalue = $(settings.object).val(); let spinnervalue = parseFloat($(settings.object).val()); spinnervalue -= settings.step; if (isNaN(spinnervalue) || spinnervalue < settings.min) spinnervalue = settings.min; spinnervalue = parseFloat(parseFloat(spinnervalue).toFixed(settings.decimals)); $(settings.object).val(spinnervalue.toFixed(settings.decimals) + settings.appendstring); if (oldvalue != spinnervalue) settings.changed(spinnervalue); }; self.spin_up = function(settings) { let oldvalue = $(settings.object).val(); let spinnervalue = parseFloat($(settings.object).val()); spinnervalue += settings.step; if (isNaN(spinnervalue) || spinnervalue > settings.max) spinnervalue = settings.max; spinnervalue = parseFloat(parseFloat(spinnervalue).toFixed(settings.decimals)); $(settings.object).val(spinnervalue.toFixed(settings.decimals) + settings.appendstring); if (oldvalue != spinnervalue) settings.changed(spinnervalue); }; self.spinner = function(settings) { if (!$(settings.object).length) return; if (typeof settings.decimals == 'undefined') settings.decimals = 0; if (typeof settings.appendstring == 'undefined') settings.appendstring = '' let spinnervalue = parseFloat($(settings.object).val()); if (isNaN(spinnervalue) || spinnervalue < settings.min) spinnervalue = settings.min; if (spinnervalue > settings.max) spinnervalue = settings.max; spinnervalue = parseFloat(parseFloat(spinnervalue).toFixed(settings.decimals)); $(settings.object).val(spinnervalue.toFixed(settings.decimals) + settings.appendstring); $(settings.object).after('â–¼'); $(settings.object).next().bind("click",function() { self.spin_down(settings); }).css('cursor','hand').css('user-select','none'); $(settings.object).after('â–²'); $(settings.object).next().bind("click",function() { self.spin_up(settings); }).css('cursor','hand').css('user-select','none'); $(settings.object).bind('keydown',function( event ) { if ( event.which == 13 ) { event.preventDefault(); } else if ( event.which == 38 ) { // up self.spin_up(settings); } else if ( event.which == 40 ) { // down self.spin_down(settings); } else { event.preventDefault(); } }); }; self.restoreselectedroute = function(replacelinks) { var selectedroute = $('#' + self.pluginname + '_selectroute option:selected').val(); if (!selectedroute) { alert('Nothing to restore'); return; } var linecount = self.linecount(); if (linecount !== 0) { if (replacelinks) { if (!confirm('Are you sure you want to replace all drawn lines\nwith stored route \'' + selectedroute + '\'?')) return; } else { if (!confirm('Are you sure you want to add stored lines\nfrom route \'' + selectedroute + '\' to current drawn lines?')) return; } } var storeddata = self.getstoreroutes(); if (!storeddata[selectedroute]) return; if (replacelinks && linecount !== 0) self.removeAll(true); self.import(storeddata[selectedroute]); self.storelines(); }; self.getstoreroutes = function() { var data = {}; if (typeof localStorage[self.localstorageroutes] === 'string' && localStorage[self.localstorageroutes] !== '') { try { data = JSON.parse(localStorage[self.localstorageroutes]); if (typeof data !== 'object' || !(data instanceof Object) || (data instanceof Array)) { return {}; } } catch(e) { return {}; } } return data; }; self.storeroute = function(selectedroute) { if (self.linecount() === 0) { alert('There are no drawn lines to store as a route'); return; // nothing to store } var storeddata = self.getstoreroutes(); var newname; if (selectedroute && storeddata[selectedroute]) { if (!confirm('Are you sure you want to replace stored route \'' + selectedroute + '\'?')) return; delete(storeddata[selectedroute]); newname = selectedroute; } else { var newnamecnt = Object.keys(storeddata).length + 1; newname = 'route ' + newnamecnt; while (storeddata[newname]) { newnamecnt++; newname = 'route ' + newnamecnt; } } var routedata = []; self.layer.eachLayer( function(line) { if (line instanceof L.GeodesicPolyline && line.getLatLngs().length === 2) { var item = {}; item.type = 'polyline'; item.latLngs = line.getLatLngs(); routedata.push(item); } }); storeddata[newname] = routedata; localStorage[self.localstorageroutes] = JSON.stringify(storeddata); self.updaterouteselectlist(newname); }; self.deletestoredroute = function() { var selectedroute = $('#' + self.pluginname + '_selectroute option:selected').val(); if (!selectedroute) { alert('Nothing stored to delete'); return; } if (!confirm('Are you sure you want to delete stored route \'' + selectedroute + '\'?')) return; var storeddata = self.getstoreroutes(); delete(storeddata[selectedroute]); localStorage[self.localstorageroutes] = JSON.stringify(storeddata); self.updaterouteselectlist(); }; self.renamestoredroute = function() { var selectedroute = $('#' + self.pluginname + '_selectroute option:selected').val(); if (!selectedroute) { alert('Nothing to rename'); return; } var newname = prompt('Enter a new unique name for this route:',selectedroute); if (newname === null || newname === selectedroute) return; if (newname === '') { alert('Route name cannot be empty!'); return; } if (newname.match(/['"]/) !== null) { alert('Quote characters are not allowed in the route name'); return; } var storeddata = self.getstoreroutes(); if (storeddata[newname]) { alert('Route name must be unique! \'' + newname + '\' already in use'); return; } storeddata[newname] = storeddata[selectedroute]; delete(storeddata[selectedroute]); localStorage[self.localstorageroutes] = JSON.stringify(storeddata); self.updaterouteselectlist(newname); }; self.updaterouteselectlist = function(selectedroute) { var newlist = self.createrouteselectlist(selectedroute); if (newlist !== $('#' + self.pluginname + '_selectroute').html()) $('#' + self.pluginname + '_selectroute').replaceWith(newlist); }; self.createrouteselectlist = function(selectedroute) { var storedroutes = self.getstoreroutes(); var routenames = Object.keys(storedroutes).sort(function(a,b) { return (a.toLowerCase() < b.toLowerCase()?-1:(a.toLowerCase() > b.toLowerCase()?1:0)); }); if (routenames.length === 0) return 'nothing stored'; var list = []; if (!selectedroute) selectedroute = $('#' + self.pluginname + '_selectroute option:selected').val(); if (!selectedroute) selectedroute = routenames[0]; for (var index in routenames) { var routename = routenames[index]; list.push(''); } return ''; }; self.storemenu = function() { var html = '
' + 'Store as new route' + 'Overwrite selected route' + self.createrouteselectlist() + 'Rename selected route' + 'Restore route (add)' + 'Restore route (replace)' + 'Delete selected route' + '
'; window.dialog({ html: html, id: self.id, dialogClass: 'ui-dialog-' + self.pluginname, title: self.title + ' Store/Restore' }).dialog('option', 'buttons', { '< Main menu': function() { self.menu(); }, 'Close': function() { $(this).dialog('close'); }, }); }; self.backupmenu = function() { var html = '
' + 'Export data (copy)' + 'Import data (paste)' + 'Append data (paste)' + (window.requestFile !== undefined ? 'Import file' : '') + ((typeof android !== 'undefined' && android && android.saveFile) ? 'Export file' : '') + '
'; window.dialog({ html: html, id: self.id, dialogClass: 'ui-dialog-' + self.pluginname, title: self.title + ' Route Import/Export' }).dialog('option', 'buttons', { '< Main menu': function() { self.menu(); }, 'Close': function() { $(this).dialog('close'); }, }); }; self.backuphackedmenu = function() { var html = '
' + 'Export hack data (copy)' + 'Import hack data (paste)' + 'Append hack data (paste)' + (window.requestFile !== undefined ? 'Import from file' : '') + ((typeof android !== 'undefined' && android && android.saveFile) ? 'Export to file' : '') + '
'; window.dialog({ html: html, id: self.id, dialogClass: 'ui-dialog-' + self.pluginname, title: self.title + ' Hacked Import/Export' }).dialog('option', 'buttons', { '< Main menu': function() { self.menu(); }, 'Close': function() { $(this).dialog('close'); }, }); }; self.rangemenu = function() { 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 = '
' + 'Range and route setup...' + 'Move remote control' + 'Reset remote control location' + 'Zoom to view total route' + 'Import/Export (' + self.hacked.length + ') hacked portals...' + 'Import/Export route...' + 'Store/restore routes...' + 'Remove route (' + self.linecount() + ' lines)...' + '
' + 'version ' + self.version + ' by ' + self.author + '' + '
'; window.dialog({ html: html, id: self.id, dialogClass: 'ui-dialog-' + self.pluginname, width: 'auto', title: self.title }).dialog('option', 'buttons', { 'About': function() { self.about(); }, 'Close': function() { $(this).dialog('close'); } }); }; self.setupColorpickerSpectrum = function() { // source: https://github.com/bgrins/spectrum // minified with https://www.minifier.org/ // Spectrum Colorpicker v1.8.1 // https://github.com/bgrins/spectrum // Author: Brian Grinstead // License: MIT (function(factory){"use strict";if(typeof define==='function'&&define.amd){define(['jquery'],factory)}else if(typeof exports=="object"&&typeof module=="object"){module.exports=factory(require('jquery'))}else{factory(jQuery)}})(function($,undefined){"use strict";var defaultOpts={beforeShow:noop,move:noop,change:noop,show:noop,hide:noop,color:!1,flat:!1,showInput:!1,allowEmpty:!1,showButtons:!0,clickoutFiresChange:!0,showInitial:!1,showPalette:!1,showPaletteOnly:!1,hideAfterPaletteSelect:!1,togglePaletteOnly:!1,showSelectionPalette:!0,localStorageKey:!1,appendTo:"body",maxSelectionSize:7,cancelText:"cancel",chooseText:"choose",togglePaletteMoreText:"more",togglePaletteLessText:"less",clearText:"Clear Color Selection",noColorSelectedText:"No Color Selected",preferredFormat:!1,className:"",containerClassName:"",replacerClassName:"",showAlpha:!1,theme:"sp-light",palette:[["#ffffff","#000000","#ff0000","#ff8000","#ffff00","#008000","#0000ff","#4b0082","#9400d3"]],selectionPalette:[],disabled:!1,offset:null},spectrums=[],IE=!!/msie/i.exec(window.navigator.userAgent),rgbaSupport=(function(){function contains(str,substr){return!!~(''+str).indexOf(substr)} var elem=document.createElement('div');var style=elem.style;style.cssText='background-color:rgba(0,0,0,.5)';return contains(style.backgroundColor,'rgba')||contains(style.backgroundColor,'hsla')})(),replaceInput=["
","
","
","
"].join(''),markup=(function(){var gradientFix="";if(IE){for(var i=1;i<=6;i++){gradientFix+="
"}} return["
","
","
","
","","
","
","
","
","
","
","
","
","
","
","
","
","
","
","
","
","
",gradientFix,"
","
","
","
","
","","
","
","
","","","
","
","
"].join("")})();function paletteTemplate(p,color,className,opts){var html=[];for(var i=0;i')}else{var cls='sp-clear-display';html.push($('
').append($('').attr('title',opts.noColorSelectedText)).html())}} return"
"+html.join('')+"
"} function hideAll(){for(var i=0;iMath.abs(dragY-oldDragY);shiftMovementDirection=furtherFromX?"x":"y"} var setSaturation=!shiftMovementDirection||shiftMovementDirection==="x";var setValue=!shiftMovementDirection||shiftMovementDirection==="y";if(setSaturation){currentSaturation=parseFloat(dragX/dragWidth)} if(setValue){currentValue=parseFloat((dragHeight-dragY)/dragHeight)} isEmpty=!1;if(!opts.showAlpha){currentAlpha=1} move()},dragStart,dragStop);if(!!initialColor){set(initialColor);updateUI();currentPreferredFormat=opts.preferredFormat||tinycolor(initialColor).format;addColorToSelectionPalette(initialColor)}else{updateUI()} if(flat){show()} function paletteElementClick(e){if(e.data&&e.data.ignore){set($(e.target).closest(".sp-thumb-el").data("color"));move()}else{set($(e.target).closest(".sp-thumb-el").data("color"));move();if(opts.hideAfterPaletteSelect){updateOriginalInput(!0);hide()}else{updateOriginalInput()}} return!1} var paletteEvent=IE?"mousedown.spectrum":"click.spectrum touchstart.spectrum";paletteContainer.on(paletteEvent,".sp-thumb-el",paletteElementClick);initialColorContainer.on(paletteEvent,".sp-thumb-el:nth-child(1)",{ignore:!0},paletteElementClick)} function updateSelectionPaletteFromStorage(){if(localStorageKey&&window.localStorage){try{var oldPalette=window.localStorage[localStorageKey].split(",#");if(oldPalette.length>1){delete window.localStorage[localStorageKey];$.each(oldPalette,function(i,c){addColorToSelectionPalette(c)})}}catch(e){} try{selectionPalette=window.localStorage[localStorageKey].split(";")}catch(e){}}} function addColorToSelectionPalette(color){if(showSelectionPalette){var rgb=tinycolor(color).toRgbString();if(!paletteLookup[rgb]&&$.inArray(rgb,selectionPalette)===-1){selectionPalette.push(rgb);while(selectionPalette.length>maxSelectionSize){selectionPalette.shift()}} if(localStorageKey&&window.localStorage){try{window.localStorage[localStorageKey]=selectionPalette.join(";")}catch(e){}}}} function getUniqueSelectionPalette(){var unique=[];if(opts.showPalette){for(var i=0;iviewWidth&&viewWidth>dpWidth)?Math.abs(offsetLeft+dpWidth-viewWidth):0);offsetTop-=Math.min(offsetTop,((offsetTop+dpHeight>viewHeight&&viewHeight>dpHeight)?Math.abs(dpHeight+inputHeight-extraY):extraY));return{top:offsetTop,bottom:offset.bottom,left:offsetLeft,right:offset.right,width:offset.width,height:offset.height}} function noop(){} function stopPropagation(e){e.stopPropagation()} function bind(func,obj){var slice=Array.prototype.slice;var args=slice.call(arguments,2);return function(){return func.apply(obj,args.concat(slice.call(arguments)))}} function draggable(element,onmove,onstart,onstop){onmove=onmove||function(){};onstart=onstart||function(){};onstop=onstop||function(){};var doc=document;var dragging=!1;var offset={};var maxHeight=0;var maxWidth=0;var hasTouch=('ontouchstart' in window);var duringDragEvents={};duringDragEvents.selectstart=prevent;duringDragEvents.dragstart=prevent;duringDragEvents["touchmove mousemove"]=move;duringDragEvents["touchend mouseup"]=stop;function prevent(e){if(e.stopPropagation){e.stopPropagation()} if(e.preventDefault){e.preventDefault()} e.returnValue=!1} function move(e){if(dragging){if(IE&&doc.documentMode<9&&!e.button){return stop()} var t0=e.originalEvent&&e.originalEvent.touches&&e.originalEvent.touches[0];var pageX=t0&&t0.pageX||e.pageX;var pageY=t0&&t0.pageY||e.pageY;var dragX=Math.max(0,Math.min(pageX-offset.left,maxWidth));var dragY=Math.max(0,Math.min(pageY-offset.top,maxHeight));if(hasTouch){prevent(e)} onmove.apply(element,[dragX,dragY,e])}} function start(e){var rightclick=(e.which)?(e.which==3):(e.button==2);if(!rightclick&&!dragging){if(onstart.apply(element,arguments)!==!1){dragging=!0;maxHeight=$(element).height();maxWidth=$(element).width();offset=$(element).offset();$(doc).on(duringDragEvents);$(doc.body).addClass("sp-dragging");move(e);prevent(e)}}} function stop(){if(dragging){$(doc).off(duringDragEvents);$(doc.body).removeClass("sp-dragging");setTimeout(function(){onstop.apply(element,arguments)},0)} dragging=!1} $(element).on("touchstart mousedown",start)} function throttle(func,wait,debounce){var timeout;return function(){var context=this,args=arguments;var throttler=function(){timeout=null;func.apply(context,args)};if(debounce)clearTimeout(timeout);if(debounce||!timeout)timeout=setTimeout(throttler,wait)}} function inputTypeColorSupport(){return $.fn.spectrum.inputTypeColorSupport()} var dataID="spectrum.id";$.fn.spectrum=function(opts,extra){if(typeof opts=="string"){var returnValue=this;var args=Array.prototype.slice.call(arguments,1);this.each(function(){var spect=spectrums[$(this).data(dataID)];if(spect){var method=spect[opts];if(!method){throw new Error("Spectrum: no such method: '"+opts+"'")} if(opts=="get"){returnValue=spect.get()}else if(opts=="container"){returnValue=spect.container}else if(opts=="option"){returnValue=spect.option.apply(spect,args)}else if(opts=="destroy"){spect.destroy();$(this).removeData(dataID)}else{method.apply(spect,args)}}});return returnValue} return this.spectrum("destroy").each(function(){var options=$.extend({},$(this).data(),opts);var spect=spectrum(this,options);$(this).data(dataID,spect.id)})};$.fn.spectrum.load=!0;$.fn.spectrum.loadOpts={};$.fn.spectrum.draggable=draggable;$.fn.spectrum.defaults=defaultOpts;$.fn.spectrum.inputTypeColorSupport=function inputTypeColorSupport(){if(typeof inputTypeColorSupport._cachedResult==="undefined"){var colorInput=$("")[0];inputTypeColorSupport._cachedResult=colorInput.type==="color"&&colorInput.value!==""} return inputTypeColorSupport._cachedResult};$.spectrum={};$.spectrum.localization={};$.spectrum.palettes={};$.fn.spectrum.processNativeColorInputs=function(){var colorInputs=$("input[type=color]");if(colorInputs.length&&!inputTypeColorSupport()){colorInputs.spectrum({preferredFormat:"hex6"})}};(function(){var trimLeft=/^[\s,#]+/,trimRight=/\s+$/,tinyCounter=0,math=Math,mathRound=math.round,mathMin=math.min,mathMax=math.max,mathRandom=math.random;var tinycolor=function(color,opts){color=(color)?color:'';opts=opts||{};if(color instanceof tinycolor){return color} if(!(this instanceof tinycolor)){return new tinycolor(color,opts)} var rgb=inputToRGB(color);this._originalInput=color;this._r=rgb.r;this._g=rgb.g;this._b=rgb.b;this._a=rgb.a;this._roundA=mathRound(1000*this._a)/1000;this._format=opts.format||rgb.format;this._gradientType=opts.gradientType;if(this._r<1){this._r=mathRound(this._r)} if(this._g<1){this._g=mathRound(this._g)} if(this._b<1){this._b=mathRound(this._b)} this._ok=rgb.ok;this._tc_id=tinyCounter++};tinycolor.prototype={isDark:function(){return this.getBrightness()<128},isLight:function(){return!this.isDark()},isValid:function(){return this._ok},getOriginalInput:function(){return this._originalInput},getFormat:function(){return this._format},getAlpha:function(){return this._a},getBrightness:function(){var rgb=this.toRgb();return(rgb.r*299+rgb.g*587+rgb.b*114)/1000},setAlpha:function(value){this._a=boundAlpha(value);this._roundA=mathRound(1000*this._a)/1000;return this},toHsv:function(){var hsv=rgbToHsv(this._r,this._g,this._b);return{h:hsv.h*360,s:hsv.s,v:hsv.v,a:this._a}},toHsvString:function(){var hsv=rgbToHsv(this._r,this._g,this._b);var h=mathRound(hsv.h*360),s=mathRound(hsv.s*100),v=mathRound(hsv.v*100);return(this._a==1)?"hsv("+h+", "+s+"%, "+v+"%)":"hsva("+h+", "+s+"%, "+v+"%, "+this._roundA+")"},toHsl:function(){var hsl=rgbToHsl(this._r,this._g,this._b);return{h:hsl.h*360,s:hsl.s,l:hsl.l,a:this._a}},toHslString:function(){var hsl=rgbToHsl(this._r,this._g,this._b);var h=mathRound(hsl.h*360),s=mathRound(hsl.s*100),l=mathRound(hsl.l*100);return(this._a==1)?"hsl("+h+", "+s+"%, "+l+"%)":"hsla("+h+", "+s+"%, "+l+"%, "+this._roundA+")"},toHex:function(allow3Char){return rgbToHex(this._r,this._g,this._b,allow3Char)},toHexString:function(allow3Char){return'#'+this.toHex(allow3Char)},toHex8:function(){return rgbaToHex(this._r,this._g,this._b,this._a)},toHex8String:function(){return'#'+this.toHex8()},toRgb:function(){return{r:mathRound(this._r),g:mathRound(this._g),b:mathRound(this._b),a:this._a}},toRgbString:function(){return(this._a==1)?"rgb("+mathRound(this._r)+", "+mathRound(this._g)+", "+mathRound(this._b)+")":"rgba("+mathRound(this._r)+", "+mathRound(this._g)+", "+mathRound(this._b)+", "+this._roundA+")"},toPercentageRgb:function(){return{r:mathRound(bound01(this._r,255)*100)+"%",g:mathRound(bound01(this._g,255)*100)+"%",b:mathRound(bound01(this._b,255)*100)+"%",a:this._a}},toPercentageRgbString:function(){return(this._a==1)?"rgb("+mathRound(bound01(this._r,255)*100)+"%, "+mathRound(bound01(this._g,255)*100)+"%, "+mathRound(bound01(this._b,255)*100)+"%)":"rgba("+mathRound(bound01(this._r,255)*100)+"%, "+mathRound(bound01(this._g,255)*100)+"%, "+mathRound(bound01(this._b,255)*100)+"%, "+this._roundA+")"},toName:function(){if(this._a===0){return"transparent"} if(this._a<1){return!1} return hexNames[rgbToHex(this._r,this._g,this._b,!0)]||!1},toFilter:function(secondColor){var hex8String='#'+rgbaToHex(this._r,this._g,this._b,this._a);var secondHex8String=hex8String;var gradientType=this._gradientType?"GradientType = 1, ":"";if(secondColor){var s=tinycolor(secondColor);secondHex8String=s.toHex8String()} return"progid:DXImageTransform.Microsoft.gradient("+gradientType+"startColorstr="+hex8String+",endColorstr="+secondHex8String+")"},toString:function(format){var formatSet=!!format;format=format||this._format;var formattedString=!1;var hasAlpha=this._a<1&&this._a>=0;var needsAlphaFormat=!formatSet&&hasAlpha&&(format==="hex"||format==="hex6"||format==="hex3"||format==="name");if(needsAlphaFormat){if(format==="name"&&this._a===0){return this.toName()} return this.toRgbString()} if(format==="rgb"){formattedString=this.toRgbString()} if(format==="prgb"){formattedString=this.toPercentageRgbString()} if(format==="hex"||format==="hex6"){formattedString=this.toHexString()} if(format==="hex3"){formattedString=this.toHexString(!0)} if(format==="hex8"){formattedString=this.toHex8String()} if(format==="name"){formattedString=this.toName()} if(format==="hsl"){formattedString=this.toHslString()} if(format==="hsv"){formattedString=this.toHsvString()} return formattedString||this.toHexString()},_applyModification:function(fn,args){var color=fn.apply(null,[this].concat([].slice.call(args)));this._r=color._r;this._g=color._g;this._b=color._b;this.setAlpha(color._a);return this},lighten:function(){return this._applyModification(lighten,arguments)},brighten:function(){return this._applyModification(brighten,arguments)},darken:function(){return this._applyModification(darken,arguments)},desaturate:function(){return this._applyModification(desaturate,arguments)},saturate:function(){return this._applyModification(saturate,arguments)},greyscale:function(){return this._applyModification(greyscale,arguments)},spin:function(){return this._applyModification(spin,arguments)},_applyCombination:function(fn,args){return fn.apply(null,[this].concat([].slice.call(args)))},analogous:function(){return this._applyCombination(analogous,arguments)},complement:function(){return this._applyCombination(complement,arguments)},monochromatic:function(){return this._applyCombination(monochromatic,arguments)},splitcomplement:function(){return this._applyCombination(splitcomplement,arguments)},triad:function(){return this._applyCombination(triad,arguments)},tetrad:function(){return this._applyCombination(tetrad,arguments)}};tinycolor.fromRatio=function(color,opts){if(typeof color=="object"){var newColor={};for(var i in color){if(color.hasOwnProperty(i)){if(i==="a"){newColor[i]=color[i]}else{newColor[i]=convertToPercentage(color[i])}}} color=newColor} return tinycolor(color,opts)};function inputToRGB(color){var rgb={r:0,g:0,b:0};var a=1;var ok=!1;var format=!1;if(typeof color=="string"){color=stringInputToObject(color)} if(typeof color=="object"){if(color.hasOwnProperty("r")&&color.hasOwnProperty("g")&&color.hasOwnProperty("b")){rgb=rgbToRgb(color.r,color.g,color.b);ok=!0;format=String(color.r).substr(-1)==="%"?"prgb":"rgb"}else if(color.hasOwnProperty("h")&&color.hasOwnProperty("s")&&color.hasOwnProperty("v")){color.s=convertToPercentage(color.s);color.v=convertToPercentage(color.v);rgb=hsvToRgb(color.h,color.s,color.v);ok=!0;format="hsv"}else if(color.hasOwnProperty("h")&&color.hasOwnProperty("s")&&color.hasOwnProperty("l")){color.s=convertToPercentage(color.s);color.l=convertToPercentage(color.l);rgb=hslToRgb(color.h,color.s,color.l);ok=!0;format="hsl"} if(color.hasOwnProperty("a")){a=color.a}} a=boundAlpha(a);return{ok:ok,format:color.format||format,r:mathMin(255,mathMax(rgb.r,0)),g:mathMin(255,mathMax(rgb.g,0)),b:mathMin(255,mathMax(rgb.b,0)),a:a}} function rgbToRgb(r,g,b){return{r:bound01(r,255)*255,g:bound01(g,255)*255,b:bound01(b,255)*255}} function rgbToHsl(r,g,b){r=bound01(r,255);g=bound01(g,255);b=bound01(b,255);var max=mathMax(r,g,b),min=mathMin(r,g,b);var h,s,l=(max+min)/2;if(max==min){h=s=0}else{var d=max-min;s=l>0.5?d/(2-max-min):d/(max+min);switch(max){case r:h=(g-b)/d+(g1)t-=1;if(t<1/6)return p+(q-p)*6*t;if(t<1/2)return q;if(t<2/3)return p+(q-p)*(2/3-t)*6;return p} if(s===0){r=g=b=l}else{var q=l<0.5?l*(1+s):l+s-l*s;var p=2*l-q;r=hue2rgb(p,q,h+1/3);g=hue2rgb(p,q,h);b=hue2rgb(p,q,h-1/3)} return{r:r*255,g:g*255,b:b*255}} function rgbToHsv(r,g,b){r=bound01(r,255);g=bound01(g,255);b=bound01(b,255);var max=mathMax(r,g,b),min=mathMin(r,g,b);var h,s,v=max;var d=max-min;s=max===0?0:d/max;if(max==min){h=0}else{switch(max){case r:h=(g-b)/d+(g>1))+720)%360;--results;){hsl.h=(hsl.h+part)%360;ret.push(tinycolor(hsl))} return ret} function monochromatic(color,results){results=results||6;var hsv=tinycolor(color).toHsv();var h=hsv.h,s=hsv.s,v=hsv.v;var ret=[];var modification=1/results;while(results--){ret.push(tinycolor({h:h,s:s,v:v}));v=(v+modification)%1} return ret} tinycolor.mix=function(color1,color2,amount){amount=(amount===0)?0:(amount||50);var rgb1=tinycolor(color1).toRgb();var rgb2=tinycolor(color2).toRgb();var p=amount/100;var w=p*2-1;var a=rgb2.a-rgb1.a;var w1;if(w*a==-1){w1=w}else{w1=(w+a)/(1+w*a)} w1=(w1+1)/2;var w2=1-w1;var rgba={r:rgb2.r*w1+rgb1.r*w2,g:rgb2.g*w1+rgb1.g*w2,b:rgb2.b*w1+rgb1.b*w2,a:rgb2.a*p+rgb1.a*(1-p)};return tinycolor(rgba)};tinycolor.readability=function(color1,color2){var c1=tinycolor(color1);var c2=tinycolor(color2);var rgb1=c1.toRgb();var rgb2=c2.toRgb();var brightnessA=c1.getBrightness();var brightnessB=c2.getBrightness();var colorDiff=(Math.max(rgb1.r,rgb2.r)-Math.min(rgb1.r,rgb2.r)+Math.max(rgb1.g,rgb2.g)-Math.min(rgb1.g,rgb2.g)+Math.max(rgb1.b,rgb2.b)-Math.min(rgb1.b,rgb2.b));return{brightness:Math.abs(brightnessA-brightnessB),color:colorDiff}};tinycolor.isReadable=function(color1,color2){var readability=tinycolor.readability(color1,color2);return readability.brightness>125&&readability.color>500};tinycolor.mostReadable=function(baseColor,colorList){var bestColor=null;var bestScore=0;var bestIsReadable=!1;for(var i=0;i125&&readability.color>500;var score=3*(readability.brightness/125)+(readability.color/500);if((readable&&!bestIsReadable)||(readable&&bestIsReadable&&score>bestScore)||((!readable)&&(!bestIsReadable)&&score>bestScore)){bestIsReadable=readable;bestScore=score;bestColor=tinycolor(colorList[i])}} return bestColor};var names=tinycolor.names={aliceblue:"f0f8ff",antiquewhite:"faebd7",aqua:"0ff",aquamarine:"7fffd4",azure:"f0ffff",beige:"f5f5dc",bisque:"ffe4c4",black:"000",blanchedalmond:"ffebcd",blue:"00f",blueviolet:"8a2be2",brown:"a52a2a",burlywood:"deb887",burntsienna:"ea7e5d",cadetblue:"5f9ea0",chartreuse:"7fff00",chocolate:"d2691e",coral:"ff7f50",cornflowerblue:"6495ed",cornsilk:"fff8dc",crimson:"dc143c",cyan:"0ff",darkblue:"00008b",darkcyan:"008b8b",darkgoldenrod:"b8860b",darkgray:"a9a9a9",darkgreen:"006400",darkgrey:"a9a9a9",darkkhaki:"bdb76b",darkmagenta:"8b008b",darkolivegreen:"556b2f",darkorange:"ff8c00",darkorchid:"9932cc",darkred:"8b0000",darksalmon:"e9967a",darkseagreen:"8fbc8f",darkslateblue:"483d8b",darkslategray:"2f4f4f",darkslategrey:"2f4f4f",darkturquoise:"00ced1",darkviolet:"9400d3",deeppink:"ff1493",deepskyblue:"00bfff",dimgray:"696969",dimgrey:"696969",dodgerblue:"1e90ff",firebrick:"b22222",floralwhite:"fffaf0",forestgreen:"228b22",fuchsia:"f0f",gainsboro:"dcdcdc",ghostwhite:"f8f8ff",gold:"ffd700",goldenrod:"daa520",gray:"808080",green:"008000",greenyellow:"adff2f",grey:"808080",honeydew:"f0fff0",hotpink:"ff69b4",indianred:"cd5c5c",indigo:"4b0082",ivory:"fffff0",khaki:"f0e68c",lavender:"e6e6fa",lavenderblush:"fff0f5",lawngreen:"7cfc00",lemonchiffon:"fffacd",lightblue:"add8e6",lightcoral:"f08080",lightcyan:"e0ffff",lightgoldenrodyellow:"fafad2",lightgray:"d3d3d3",lightgreen:"90ee90",lightgrey:"d3d3d3",lightpink:"ffb6c1",lightsalmon:"ffa07a",lightseagreen:"20b2aa",lightskyblue:"87cefa",lightslategray:"789",lightslategrey:"789",lightsteelblue:"b0c4de",lightyellow:"ffffe0",lime:"0f0",limegreen:"32cd32",linen:"faf0e6",magenta:"f0f",maroon:"800000",mediumaquamarine:"66cdaa",mediumblue:"0000cd",mediumorchid:"ba55d3",mediumpurple:"9370db",mediumseagreen:"3cb371",mediumslateblue:"7b68ee",mediumspringgreen:"00fa9a",mediumturquoise:"48d1cc",mediumvioletred:"c71585",midnightblue:"191970",mintcream:"f5fffa",mistyrose:"ffe4e1",moccasin:"ffe4b5",navajowhite:"ffdead",navy:"000080",oldlace:"fdf5e6",olive:"808000",olivedrab:"6b8e23",orange:"ffa500",orangered:"ff4500",orchid:"da70d6",palegoldenrod:"eee8aa",palegreen:"98fb98",paleturquoise:"afeeee",palevioletred:"db7093",papayawhip:"ffefd5",peachpuff:"ffdab9",peru:"cd853f",pink:"ffc0cb",plum:"dda0dd",powderblue:"b0e0e6",purple:"800080",rebeccapurple:"663399",red:"f00",rosybrown:"bc8f8f",royalblue:"4169e1",saddlebrown:"8b4513",salmon:"fa8072",sandybrown:"f4a460",seagreen:"2e8b57",seashell:"fff5ee",sienna:"a0522d",silver:"c0c0c0",skyblue:"87ceeb",slateblue:"6a5acd",slategray:"708090",slategrey:"708090",snow:"fffafa",springgreen:"00ff7f",steelblue:"4682b4",tan:"d2b48c",teal:"008080",thistle:"d8bfd8",tomato:"ff6347",turquoise:"40e0d0",violet:"ee82ee",wheat:"f5deb3",white:"fff",whitesmoke:"f5f5f5",yellow:"ff0",yellowgreen:"9acd32"};var hexNames=tinycolor.hexNames=flip(names);function flip(o){var flipped={};for(var i in o){if(o.hasOwnProperty(i)){flipped[o[i]]=i}} return flipped} function boundAlpha(a){a=parseFloat(a);if(isNaN(a)||a<0||a>1){a=1} return a} function bound01(n,max){if(isOnePointZero(n)){n="100%"} var processPercent=isPercentage(n);n=mathMin(max,mathMax(0,parseFloat(n)));if(processPercent){n=parseInt(n*max,10)/100} if((math.abs(n-max)<0.000001)){return 1} return(n%max)/parseFloat(max)} function clamp01(val){return mathMin(1,mathMax(0,val))} function parseIntFromHex(val){return parseInt(val,16)} function isOnePointZero(n){return typeof n=="string"&&n.indexOf('.')!=-1&&parseFloat(n)===1} function isPercentage(n){return typeof n==="string"&&n.indexOf('%')!=-1} function pad2(c){return c.length==1?'0'+c:''+c} function convertToPercentage(n){if(n<=1){n=(n*100)+"%"} return n} function convertDecimalToHex(d){return Math.round(parseFloat(d)*255).toString(16)} function convertHexToDecimal(h){return(parseIntFromHex(h)/255)} var matchers=(function(){var CSS_INTEGER="[-\\+]?\\d+%?";var CSS_NUMBER="[-\\+]?\\d*\\.\\d+%?";var CSS_UNIT="(?:"+CSS_NUMBER+")|(?:"+CSS_INTEGER+")";var PERMISSIVE_MATCH3="[\\s|\\(]+("+CSS_UNIT+")[,|\\s]+("+CSS_UNIT+")[,|\\s]+("+CSS_UNIT+")\\s*\\)?";var PERMISSIVE_MATCH4="[\\s|\\(]+("+CSS_UNIT+")[,|\\s]+("+CSS_UNIT+")[,|\\s]+("+CSS_UNIT+")[,|\\s]+("+CSS_UNIT+")\\s*\\)?";return{rgb:new RegExp("rgb"+PERMISSIVE_MATCH3),rgba:new RegExp("rgba"+PERMISSIVE_MATCH4),hsl:new RegExp("hsl"+PERMISSIVE_MATCH3),hsla:new RegExp("hsla"+PERMISSIVE_MATCH4),hsv:new RegExp("hsv"+PERMISSIVE_MATCH3),hsva:new RegExp("hsva"+PERMISSIVE_MATCH4),hex3:/^([0-9a-fA-F]{1})([0-9a-fA-F]{1})([0-9a-fA-F]{1})$/,hex6:/^([0-9a-fA-F]{2})([0-9a-fA-F]{2})([0-9a-fA-F]{2})$/,hex8:/^([0-9a-fA-F]{2})([0-9a-fA-F]{2})([0-9a-fA-F]{2})([0-9a-fA-F]{2})$/}})();function stringInputToObject(color){color=color.replace(trimLeft,'').replace(trimRight,'').toLowerCase();var named=!1;if(names[color]){color=names[color];named=!0}else if(color=='transparent'){return{r:0,g:0,b:0,a:0,format:"name"}} var match;if((match=matchers.rgb.exec(color))){return{r:match[1],g:match[2],b:match[3]}} if((match=matchers.rgba.exec(color))){return{r:match[1],g:match[2],b:match[3],a:match[4]}} if((match=matchers.hsl.exec(color))){return{h:match[1],s:match[2],l:match[3]}} if((match=matchers.hsla.exec(color))){return{h:match[1],s:match[2],l:match[3],a:match[4]}} if((match=matchers.hsv.exec(color))){return{h:match[1],s:match[2],v:match[3]}} if((match=matchers.hsva.exec(color))){return{h:match[1],s:match[2],v:match[3],a:match[4]}} if((match=matchers.hex8.exec(color))){return{a:convertHexToDecimal(match[1]),r:parseIntFromHex(match[2]),g:parseIntFromHex(match[3]),b:parseIntFromHex(match[4]),format:named?"name":"hex8"}} if((match=matchers.hex6.exec(color))){return{r:parseIntFromHex(match[1]),g:parseIntFromHex(match[2]),b:parseIntFromHex(match[3]),format:named?"name":"hex"}} if((match=matchers.hex3.exec(color))){return{r:parseIntFromHex(match[1]+''+match[1]),g:parseIntFromHex(match[2]+''+match[2]),b:parseIntFromHex(match[3]+''+match[3]),format:named?"name":"hex"}} return!1} window.tinycolor=tinycolor})();$(function(){if($.fn.spectrum.load){$.fn.spectrum.processNativeColorInputs()}})}); $('head').append(''); }; // end setupColorpickerSpectrum self.setup = function() { if ('pluginloaded' in self) { console.log('IITC plugin already loaded: ' + self.title + ' version ' + self.version); return; } else { self.pluginloaded = true; } self.restoresettings(); self.setupColorpickerSpectrum(); self.hacked = self.restorearray(self.localstoragehacked); self.layer = new L.FeatureGroup(); window.addLayerGroup(self.routelayertitle, self.layer, true); map.on('layeradd', function(obj) { if (obj.layer === self.layer) { self.showremote(); self.onPortalSelected(); } }); map.on('layerremove', function(obj) { if (obj.layer === self.layer) { self.dronemarker_remove(); self.onPortalSelected(); self.hideremote(); } }); self.restorelines(); self.halfrangelayer = new L.FeatureGroup(); window.addLayerGroup(self.halfrangelayertitle, self.halfrangelayer, true); map.on('layeradd', function(obj) { if (obj.layer === self.halfrangelayer) { self.updatehalfrangelayer(); } }); map.on('layerremove', function(obj) { if (obj.layer === self.halfrangelayer) { } }); //map.addLayer(self.halfrangelayer); window.map.on('moveend', function() { window.setTimeout(self.updatehalfrangelayer); }); window.addHook('mapDataRefreshEnd',function() { window.setTimeout(self.updatehalfrangelayer); }); window.addHook('portalAdded', self.onportalAdded); window.addHook('portalRemoved', self.onportalRemoved); window.addPortalHighlighter(self.highlighthackedname,self.highlightportals); // if hightligheractive: if (window._current_highlighter == self.highlighthackedname) { self.hightlighteractive = true; self.updateHighlighter(); } window.map.on('zoomend', function() { self.drawmarker_redraw(); }); window.addHook('portalSelected', self.onPortalSelected); $('#toolbox').append('' + self.title + ''); $('head').append( ''); self.isSmartphone = window.isSmartphone(); $('head').append(''); $('head').append(''); if (self.layeractive(self.routelayertitle)) { self.showremote(); } console.log('IITC plugin loaded: ' + self.title + ' version ' + self.version); }; var setup = function() { (window.iitcLoaded?self.setup():window.addHook('iitcLoaded',self.setup)); }; setup.info = plugin_info; //add the script info data to the function as a property if(!window.bootPlugins) window.bootPlugins = []; window.bootPlugins.push(setup); // if IITC has already booted, immediately run the 'setup' function if(window.iitcLoaded && typeof setup === 'function') setup(); } // wrapper end // inject code into site context var script = document.createElement('script'); var info = {}; if (typeof GM_info !== 'undefined' && GM_info && GM_info.script) info.script = { version: GM_info.script.version, name: GM_info.script.name, description: GM_info.script.description }; script.appendChild(document.createTextNode('('+ wrapper +')('+JSON.stringify(info)+');')); (document.body || document.head || document.documentElement).appendChild(script);