// ==UserScript== // @author DanielOnDiordna // @name Quick Draw Links // @category Layer // @version 0.0.9.20210724.002500 // @updateURL https://raw.githubusercontent.com/IITC-CE/Community-plugins/master/dist/DanielOnDiordna/quick-draw-links.meta.js // @downloadURL https://raw.githubusercontent.com/IITC-CE/Community-plugins/master/dist/DanielOnDiordna/quick-draw-links.user.js // @description [danielondiordna-0.0.9.20210724.002500] Quickly draw and move links from portal to portal on the map. Show crosslinks, for links on the map, as well as for drawn links. Store/Restore your projects. Added great circle support. Added fields layer. Export list of used portals with link count. Integrated Spectrum Colorpicker 1.8.1 // @id quick-draw-links@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.quickdrawlinks = function() {}; var self = window.plugin.quickdrawlinks; self.id = 'quickdrawlinks'; self.title = 'Quick Draw Links'; self.version = '0.0.9.20210724.002500'; self.author = 'DanielOnDiordna'; self.changelog = ` Changelog: version 0.0.1.20181101.224000 - added a new button to add bookmarks to portals connected to crosslinks version 0.0.2.20190530.121700 - closedialog fix for smartphone - fixed in/out direction for drawn links in portal overview version 0.0.3.20191120.220400 - open side pane fix for smartphone version 0.0.4.20200613.180700 - linkexists now checks both directions version 0.0.5.20201230.204600 - replaced L.geodesicPolyline by L.geoJson (added sourcecode https://github.com/springmeyer/arc.js) version 0.0.6.20210117.001400 - fixed color storage for setStyle compatibility issues with L.geoJson version 0.0.7.20210118.220700 - rewritten some code for placing the buttons; buttons now also visible on desktop above the status bar - updated plugin wrapper and userscript header formatting to match IITC-CE coding version 0.0.7.20210119.185800 - fixed button width for smartphone version 0.0.8.20210119.220700 - integrated Spectrum Colorpicker 1.8.1 plugin code, no need anymore for the separate plugin - removed drawtools integration version 0.0.9.20210127.225400 - removed icon button titles for mobile users - enabled toggle function for icon buttons - new active icon button indicators version 0.0.9.20210421.190200 - minor fix for IITC CE where runHooks iitcLoaded is executed before addHook is defined in this plugin version 0.0.9.20210724.002500 - prevent double plugin setup on hook iitcLoaded `; self.namespace = 'window.plugin.' + self.id + '.'; self.pluginname = 'plugin-' + self.id; self.panename = 'plugin-' + self.id; self.localstoragesettings = self.pluginname + '-settings'; self.localstoragelayer = self.pluginname + '-layer'; self.localstoragetitlecache = self.pluginname + '-titlecache'; self.localstoragedata = self.pluginname + '-data'; self.markerLayer = undefined; self.iconstyle = 'link'; self.startguid = undefined; self.startpos = undefined; self.multistartpos = undefined; self.guidpos = {}; self.titlecache = {}; self.isSmartphone = null; self.settings = {}; self.settings.hidebuttons = false; self.settings.drawcolor = '#E27000'; self.settings.greatcirclecolor = '#FF0000'; self.settings.fieldcolor = '#E27000'; self.settings.crosslinkbookmarkcolor = '#FF0000'; self.settings.showcrosslinks = 0; // 0 = Links, 1 = Drawn, 2 = Both self.settings.showlinkdirection = false; self.settings.fieldexistinglinks = false; self.requestid = undefined; self.selectedlink = undefined; self.movelinksposition = undefined; self.copylinksposition = undefined; self.linkstyle = [ '10,5,5,5,5,5,5,5,100%', ]; self.crosslinklayerdisabled = false; self.highlightlinkoptions = { color: "#C33", opacity: 1, weight: 5, fill: false, dashArray: "1,6", radius: 18, }; // 120 x 60 - 4 buttons, 2 rows: link, move, star, copy self.menuicons = ""; // 350 x 50 - 7 buttons: trash, move, swap, zoom, list, confirm, cancel self.linkmenubuttonsicon = ""; self.restoresettings = function() { if (typeof localStorage[self.localstoragesettings] === 'string') { var settings = JSON.parse(localStorage[self.localstoragesettings]); if (typeof settings === 'object' && settings instanceof Object) { if (typeof settings.hidebuttons === 'boolean') self.settings.hidebuttons = settings.hidebuttons; if (typeof settings.drawcolor === 'string') self.settings.drawcolor = settings.drawcolor; if (typeof settings.greatcirclecolor === 'string') self.settings.greatcirclecolor = settings.greatcirclecolor; if (typeof settings.fieldcolor === 'string') self.settings.fieldcolor = settings.fieldcolor; if (typeof settings.crosslinkbookmarkcolor === 'string') self.settings.crosslinkbookmarkcolor = settings.crosslinkbookmarkcolor; if (typeof settings.showcrosslinks === 'number') self.settings.showcrosslinks = settings.showcrosslinks; if (typeof settings.showlinkdirection === 'boolean') self.settings.showlinkdirection = settings.showlinkdirection; if (typeof settings.fieldexistinglinks === 'boolean') self.settings.fieldexistinglinks = settings.fieldexistinglinks; } } }; self.storesettings = function() { localStorage[self.localstoragesettings] = JSON.stringify(self.settings); }; self.linkicon = function(zoom,iconstyle) { if (iconstyle instanceof Object) iconstyle = iconstyle.icon; if (!iconstyle) iconstyle = self.iconstyle; var icon; switch(iconstyle) { case 'link': icon = ''; break; case 'move': icon = ''; break; case 'copy': icon = ''; break; case 'star': icon = ''; break; default: iconstyle = 'link'; icon = ''; // icon = '//storage.googleapis.com/ingress.com/img/map_icons/marker_images/lp-origin-ornament.png'; } self.iconstyle = iconstyle; var size = Math.max(30,zoom * 4); return L.icon({ iconUrl: icon, iconSize: [size, size], // size of the icon iconAnchor: [size/2, size/2], // point of the icon which will correspond to marker's location zIndexOffset: -1000 }); }; self.zoomadjust = function(zoom) { if (self.markerLayer === undefined) return; self.markerLayer.setIcon(self.linkicon(zoom)); }; self.clearall = function() { self.moveselectedlinkstartposition = undefined; self.moveselectedlinkendposition = undefined; self.movelinksposition = undefined; self.multistartpos = undefined; self.copylinksposition = undefined; self.closedialog(); self.deactivatebuttons(); }; self.addMarker = function(guid,iconstyle) { if (self.removeMarker() == iconstyle.icon) { self.clearall(); return; } if (self.drawnItems._map === null) return; self.activatebutton(iconstyle.icon); var portal = window.portals[guid]; var startpos = portal.getLatLng(); self.markerLayer = L.marker([startpos.lat, startpos.lng], {icon: self.linkicon(map.getZoom(),iconstyle), iconstyle: iconstyle.icon}).addTo(map); self.markerLayer.on('click', function(e) { self.clearall(); self.removeMarker(); }); self.startguid = guid; self.startpos = startpos; map.on('zoomend', function() { self.zoomadjust(map.getZoom()); }); }; self.multistartlinks = function() { if (window.selectedPortal === null) return; self.addMarker(window.selectedPortal,{icon:'star'}); self.multistartpos = self.startpos; }; self.removeMarker = function() { if (self.markerLayer === undefined) return false; // remove marker let iconstyle = self.markerLayer.options.iconstyle; map.removeLayer(self.markerLayer); self.markerLayer = undefined; return iconstyle; }; self.removelink = function(link) { if (self.markerLayer !== undefined) return; // disable remove when drawing a new line self.stophighlightlink(); var drawlayer = self.getDrawlayer(); drawlayer.removeLayer(link); self.updategreatcircleslayer(); self.updatefieldslayer(); self.storelinks(); // all other event types - assume anything could have been modified and re-check all links self.checkAllLinksForCrosslinks(); return true; }; self.linkexists = function(latLngs) { var drawlayer = self.getDrawlayer(); var linkexists = false; drawlayer.eachLayer( function(layer) { //if (layer instanceof L.GeodesicPolyline && layer.getLatLngs().length === 2) { if (layer instanceof L.GeoJSON && layer.getLatLngs().length === 2) { var existinglatLngs = layer.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)){ linkexists = true; } } }); //console.log('link exists: ' + linkexists); return linkexists; }; self.distanceBetween = function(startLatLng,endLatLng) { // source: Arc 1.7.0 // How far between portals. let R = 6367; // km let lat1 = startLatLng.lat; let lon1 = startLatLng.lng; let lat2 = endLatLng.lat; let lon2 = endLatLng.lng; let dLat = (lat2 - lat1) * Math.PI / 180; let dLon = (lon2 - lon1) * Math.PI / 180; lat1 = lat1 * Math.PI / 180; lat2 = lat2 * Math.PI / 180; let a = Math.sin(dLat / 2) * Math.sin(dLat / 2) + Math.sin(dLon / 2) * Math.sin(dLon / 2) * Math.cos(lat1) * Math.cos(lat2); let c = 2 * Math.atan2(Math.sqrt(a), Math.sqrt(1 - a)); let d = R * c; d = Math.round(d * 1000) / 1000; return d; }; /* self.Coord = function(c, e) { // source: Arc 1.7.0 let D2R = Math.PI / 180; let R2D = 180 / Math.PI; this.lon = c; this.lat = e; this.x = D2R * c; this.y = D2R * e; }; self.Coord.prototype.view = function() { return String(this.lon).slice(0, 4) + "," + String(this.lat).slice(0, 4); }; self.Coord.prototype.antipode = function() { return new Coord(0 > this.lon ? 180 + this.lon : -1 * (180 - this.lon),-1 * this.lat); }; */ self.drawline = function(latLngs,color) { if (latLngs[0].lat === latLngs[1].lat && latLngs[0].lng === latLngs[1].lng) return; // no 0 length lines if (self.linkexists(latLngs)) return; // no double lines function randomguid() { function s4() { return Math.floor((1 + Math.random()) * 0x10000) .toString(16) .substring(1); } return s4() + s4() + '-' + s4() + '-' + s4() + '-' + s4() + '-' + s4() + s4() + s4(); } var drawlayer = self.getDrawlayer(); // Draw method: /* self.lineOptions = { stroke: true, color: self.settings.drawcolor, weight: 4, opacity: 0.8, fill: false, clickable: true }; var lineoptions = self.lineOptions; var link = L.geodesicPolyline(latLngs, L.extend({}, lineoptions, {color:color, dashArray: (self.settings.showlinkdirection?self.linkstyle:null)})); */ // Arc method: var startCoord = new self.arc.Coord(latLngs[0].lng,latLngs[0].lat); var stopCoord = new self.arc.Coord(latLngs[1].lng,latLngs[1].lat); var lineoptions = self.lineOptions; var myStyle = L.extend({}, lineoptions, {color:color, dashArray: (self.settings.showlinkdirection?self.linkstyle:null)}); var distance = self.distanceBetween(latLngs[0],latLngs[1]); var gc = new self.arc.GreatCircle(startCoord,stopCoord); //var gc = new window.plugin.arcs.arc.GreatCircle(startCoord,stopCoord); //console.log('==================== gc new arcs:',gcnew,gc); var geojson_feature = gc.Arc(Math.round(distance)).json(); var link = L.geoJson(geojson_feature,{ style: myStyle }); // additions to Arc, to make it compatible for Quick Draw: link.getLatLngs = function() { return [L.latLng(latLngs[0]),L.latLng(latLngs[1])]; }; link.options = L.extend({}, myStyle); link.on('click',function(e) { self.linkmenu(link); }); drawlayer.addLayer(link); self.updategreatcircleslayer(); self.updatefieldslayer(); link.options.guid = randomguid(); // create a fake guid for cross link support // we can just test the new layer in this case self.testAllLinksAgainstLayer(link); return link; }; self.addline = function() { if (!self.startpos || window.selectedPortal === null) return; var startpos = self.startpos; var endpos = window.portals[window.selectedPortal].getLatLng(); var latLngs = [new L.LatLng(startpos.lat,startpos.lng),new L.LatLng(endpos.lat,endpos.lng)]; self.drawline(latLngs,self.settings.drawcolor); self.storelinks(); }; 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.moveline = function(index) { // 0 = start, 1 = end if (window.selectedPortal === null) return; var guid = window.selectedPortal; var portal = window.portals[guid]; var drawlayer = self.getDrawlayer(); var latLngs = self.clonelatlngs(self.selectedlink.getLatLngs()); var newll = portal.getLatLng(); latLngs[index] = new L.LatLng(newll.lat,newll.lng); if ((latLngs[0].lat === latLngs[1].lat && latLngs[0].lng === latLngs[1].lng) || self.linkexists(latLngs)) { // remove line if start and end match or if line already exists drawlayer.removeLayer(self.selectedlink); self.selectedlink = undefined; } else { //self.selectedlink.setLatLngs(latLngs); var color = self.selectedlink.options.color; drawlayer.removeLayer(self.selectedlink); self.selectedlink = self.drawline(latLngs,color); } self.updategreatcircleslayer(); self.updatefieldslayer(); self.storelinks(); // all other event types - assume anything could have been modified and re-check all links self.checkAllLinksForCrosslinks(); }; self.movelines = function() { if (window.selectedPortal === null) return; if (!self.movelinksposition) return; var guid = window.selectedPortal; var portal = window.portals[guid]; var newll = portal.getLatLng(); var drawlayer = self.getDrawlayer(); var movefromposition = self.movelinksposition; var movecnt = 0; drawlayer.eachLayer( function(layer) { //if (layer instanceof L.GeodesicPolyline && layer.getLatLngs().length === 2) { if (layer instanceof L.GeoJSON && layer.getLatLngs().length === 2) { var moved = false; var latLngs = self.clonelatlngs(layer.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) { movecnt++; if ((latLngs[0].lat === latLngs[1].lat && latLngs[0].lng === latLngs[1].lng) || self.linkexists(latLngs)) { // remove line if start and end match or if line already exists drawlayer.removeLayer(layer); } else { //layer.setLatLngs(latLngs); var color = layer.options.color; drawlayer.removeLayer(layer); self.drawline(latLngs,color); } } } }); if (movecnt > 0) { self.updategreatcircleslayer(); self.updatefieldslayer(); self.storelinks(); } // all other event types - assume anything could have been modified and re-check all links self.checkAllLinksForCrosslinks(); }; self.copylines = function() { if (window.selectedPortal === null) return; if (!self.copylinksposition) return; var guid = window.selectedPortal; var portal = window.portals[guid]; var newll = portal.getLatLng(); var drawlayer = self.getDrawlayer(); var copyfromposition = self.copylinksposition; var copycnt = 0; drawlayer.eachLayer( function(layer) { //if (layer instanceof L.GeodesicPolyline && layer.getLatLngs().length === 2) { if (layer instanceof L.GeoJSON && layer.getLatLngs().length === 2) { var latLngs = self.clonelatlngs(layer.getLatLngs()); var startpos; var endpos; if (latLngs[0].lat === copyfromposition.lat && latLngs[0].lng === copyfromposition.lng) { startpos = newll; endpos = new L.LatLng(latLngs[1].lat,latLngs[1].lng); } else if (latLngs[1].lat === copyfromposition.lat && latLngs[1].lng === copyfromposition.lng) { startpos = new L.LatLng(latLngs[0].lat,latLngs[0].lng); endpos = newll; } if (startpos && endpos) { latLngs = [new L.LatLng(startpos.lat,startpos.lng),new L.LatLng(endpos.lat,endpos.lng)]; self.drawline(latLngs,layer.options.color); copycnt++; } } }); if (copycnt > 0) { self.updategreatcircleslayer(); self.updatefieldslayer(); self.storelinks(); } // all other event types - assume anything could have been modified and re-check all links self.checkAllLinksForCrosslinks(); }; self.storetitles = function() { // only save titles for stored positions var savetitles = {}; var drawlayer = self.getDrawlayer(); drawlayer.eachLayer( function(layer) { //if (layer instanceof L.GeodesicPolyline && layer.getLatLngs().length === 2) { if (layer instanceof L.GeoJSON && layer.getLatLngs().length === 2) { var latLngs = layer.getLatLngs(); for (var cnt = 0; cnt < latLngs.length; cnt++) { var guid = self.getguid(latLngs[cnt]); var title = self.gettitle(latLngs[cnt]); if (title && guid) { savetitles[latLngs[cnt].lat + ',' + latLngs[cnt].lng] = {title:title,guid:guid}; } } } }); localStorage[self.localstoragetitlecache] = JSON.stringify(savetitles); }; self.restoretitles = function() { // restore stored titles try { var dataStr = localStorage[self.localstoragetitlecache]; if (dataStr === undefined) return; var data = JSON.parse(dataStr); $.each(data, function(pos,item) { if (item instanceof Object) { self.guidpos[pos] = item.guid; self.titlecache[item.guid] = item.title; } }); } catch(e) { console.warn(self.id + ': failed to load data from localStorage: '+e); } }; self.gettitle = function(position) { var guid = self.getguid(position); if (!guid) return undefined; return self.titlecache[guid]; }; self.getguid = function(position) { if (!(position instanceof Object) || !position.lat || !position.lng) return undefined; return self.guidpos[position.lat + ',' + position.lng]; }; self.deactivatebuttons = function() { console.log('deactivatebuttons'); $('.screenlinkicon, .titlelinkicon').css('background-position-y','top'); $('.screenmoveicon, .titlemoveicon').css('background-position-y','top'); $('.screenstaricon, .titlestaricon').css('background-position-y','top'); $('.screencopyicon, .titlecopyicon').css('background-position-y','top'); }; self.activatebutton = function(icon) { console.log('activatebutton',icon); $('.screen' + icon + 'icon, .title' + icon + 'icon').css('background-position-y','bottom'); }; self.onPortalSelected = function() { // remove the sidebar button self.stophighlightlink(); $('.quickdrawbutton').remove(); if (window.selectedPortal === null) { $('#quickdrawmenuselectedportal').hide(); return; } if (self.drawnItems._map === null) return; $('#quickdrawmenuselectedportal').show(); var guid = window.selectedPortal; var portal = window.portals[guid]; var latlng = portal.getLatLng(); if (portal.options.data.title) { self.guidpos[latlng.lat + ',' + latlng.lng] = guid; self.titlecache[guid] = portal.options.data.title; self.storetitles(); } if (self.markerLayer !== undefined) { // there is a start marker if (self.startguid !== guid) { // another portal then the start portal is selected if (self.selectedlink && self.moveselectedlinkstartposition) { self.moveline(0); self.moveselectedlinkstartposition = undefined; // remove marker self.removeMarker(); } else if (self.selectedlink && self.moveselectedlinkendposition) { self.moveline(1); self.moveselectedlinkendposition = undefined; // remove marker self.removeMarker(); } else if (self.movelinksposition) { // move all links connected to movelinksposition to new portal self.movelines(); self.movelinksposition = undefined; // remove marker self.removeMarker(); } else if (self.copylinksposition) { // copy all links connected to copylinksposition to new portal self.copylines(); self.copylinksposition = undefined; // remove marker self.removeMarker(); } else { self.addline(); // remove marker let iconstyle = self.removeMarker(); // set marker for new portal self.addMarker(guid,{icon:iconstyle}); // repeat link or star if (self.multistartpos) { // star self.startpos = self.multistartpos; } } } } setTimeout(function() { // the sidebar is constructed after firing the hook // add button to portal details window: $('.quickdrawbutton').remove(); if (window.selectedPortal === null) return; var guid = window.selectedPortal; var portalposition = window.portals[guid].getLatLng(); // find links connected to selectedportal var linkcount = self.linkcount(portalposition); if (linkcount > 0) { // copybutton let titledescription = ''; if (!self.isSmartphone) titledescription = ' title="Click to copy all links from this portal to another portal"'; let onclickaction = self.namespace + 'clearall(); ' + self.namespace + 'copylinks(); return false;' if (!self.settings.hidebuttons) { $('#updatestatus').prepend(''); } $('#portaldetails > .title').prepend(''); } if (true) { // starbutton let titledescription = ''; if (!self.isSmartphone) titledescription = ' title="Click to start making multiple links to this portal"'; let onclickaction = self.namespace + 'clearall(); ' + self.namespace + 'multistartlinks(); return false;' let styleactivebutton = (self.markerLayer != undefined && self.markerLayer.options.iconstyle == 'star'?' style="background-position-y: bottom;"':'' ); if (!self.settings.hidebuttons) { $('#updatestatus').prepend(''); } $('#portaldetails > .title').prepend(''); } if (linkcount > 0) { // movebutton let titledescription = ''; if (!self.isSmartphone) titledescription = ' title="Click to move all links from this portal to another portal"'; let onclickaction = self.namespace + 'clearall(); ' + self.namespace + 'movelinks(); return false;' if (!self.settings.hidebuttons) { $('#updatestatus').prepend(''); } $('#portaldetails > .title').prepend(''); } if (true) { // linkbutton let titledescription = ''; if (!self.isSmartphone) titledescription = ' title="Click to start a link from this portal, click another portal to create a link"'; let onclickaction = self.namespace + 'clearall(); ' + self.namespace + 'addMarker(\'' + guid + '\',{icon:\'link\'}); return false;' let styleactivebutton = (self.markerLayer != undefined && self.markerLayer.options.iconstyle == 'link'?' style="background-position-y: bottom;"':'' ); if (!self.settings.hidebuttons) { $('#updatestatus').prepend(''); } $('#portaldetails > .title').prepend(''); } }, 0); }; self.setalloutgoing = function(position) { if (!position) return; var drawlayer = self.getDrawlayer(); var swapcount = 0; var outcount = 0; drawlayer.eachLayer( function(layer) { //if (layer instanceof L.GeodesicPolyline && layer.getLatLngs().length === 2) { if (layer instanceof L.GeoJSON && layer.getLatLngs().length === 2) { var latLngs = layer.getLatLngs(); if (position.lat === latLngs[1].lat && position.lng === latLngs[1].lng) { // set incoming link to outgoing link self.selectedlink = layer; self.swapstartend(); swapcount++; } else if (position.lat === latLngs[0].lat && position.lng === latLngs[0].lng) { outcount++; } } }); alert('Total Drawn Links connected to portal: ' + (swapcount + outcount) + "\n" + 'links changed to outgoing: ' + swapcount); }; self.setallincoming = function(position) { if (!position) return; var drawlayer = self.getDrawlayer(); var swapcount = 0; var incount = 0; drawlayer.eachLayer( function(layer) { //if (layer instanceof L.GeodesicPolyline && layer.getLatLngs().length === 2) { if (layer instanceof L.GeoJSON && layer.getLatLngs().length === 2) { var latLngs = layer.getLatLngs(); if (position.lat === latLngs[0].lat && position.lng === latLngs[0].lng) { // set outgoing link to incoming link self.selectedlink = layer; self.swapstartend(); swapcount++; } else if (position.lat === latLngs[1].lat && position.lng === latLngs[1].lng) { incount++; } } }); alert('Total Drawn Links connected to portal: ' + (swapcount + incount) + "\n" + 'links changed to incoming: ' + swapcount); }; self.linkcount = function(position) { var drawlayer = self.getDrawlayer(); var linkcount = 0; drawlayer.eachLayer( function(layer) { //if (layer instanceof L.GeodesicPolyline && layer.getLatLngs().length === 2) { if (layer instanceof L.GeoJSON && layer.getLatLngs().length === 2) { var latLngs = layer.getLatLngs(); if (!position || position.lat === latLngs[0].lat && position.lng === latLngs[0].lng || position.lat === latLngs[1].lat && position.lng === latLngs[1].lng) { linkcount++; } } }); return linkcount; }; self.storelinks = function() { var data = []; self.drawnItems.eachLayer( function(layer) { if (layer instanceof L.GeoJSON && layer.getLatLngs().length === 2) { var item = {}; item.type = 'polyline'; item.latLngs = layer.getLatLngs(); item.color = layer.options.color; data.push(item); } }); localStorage[self.localstoragelayer] = JSON.stringify(data); self.storetitles(); }; self.load = 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.import = function(data) { if (!data.layerdata && !data.titledata) data.layerdata = data; // support some backwards compatiblity if (data.layerdata) { $.each(data.layerdata, function(index,item) { switch(item.type) { case 'polyline': self.drawline(item.latLngs,item.color); break; default: console.warn(self.id + ': unknown import layer type "'+item.type); break; } }); } if (data.titledata) { localStorage[self.localstoragetitlecache] = JSON.stringify(data.titledata); self.restoretitles(); } }; // copy start // copied from Cross Links plugin ////////// self.greatCircleArcIntersect = function(a0,a1,b0,b1) { // based on the formula at http://williams.best.vwh.net/avform.htm#Int // method: // check to ensure no line segment is zero length - if so, cannot cross // check to see if either of the lines start/end at the same point. if so, then they cannot cross // check to see if the line segments overlap in longitude. if not, no crossing // if overlap, clip each line to the overlapping longitudes, then see if latitudes cross // anti-meridian handling. this code will not sensibly handle a case where one point is // close to -180 degrees and the other +180 degrees. unwrap coordinates in this case, so one point // is beyond +-180 degrees. this is already true in IITC // FIXME? if the two lines have been 'unwrapped' differently - one positive, one negative - it will fail // zero length line tests if (a0.equals(a1)) return false; if (b0.equals(b1)) return false; // lines have a common point if (a0.equals(b0) || a0.equals(b1)) return false; if (a1.equals(b0) || a1.equals(b1)) return false; // check for 'horizontal' overlap in lngitude if (Math.min(a0.lng,a1.lng) > Math.max(b0.lng,b1.lng)) return false; if (Math.max(a0.lng,a1.lng) < Math.min(b0.lng,b1.lng)) return false; // ok, our two lines have some horizontal overlap in longitude // 1. calculate the overlapping min/max longitude // 2. calculate each line latitude at each point // 3. if latitudes change place between overlapping range, the lines cross // class to hold the pre-calculated maths for a geodesic line // TODO: move this outside this function, so it can be pre-calculated once for each line we test var GeodesicLine = function(start,end) { var d2r = Math.PI/180.0; var r2d = 180.0/Math.PI; // maths based on http://williams.best.vwh.net/avform.htm#Int if (start.lng == end.lng) { throw 'Error: cannot calculate latitude for meridians'; } // only the variables needed to calculate a latitude for a given longitude are stored in 'this' this.lat1 = start.lat * d2r; this.lat2 = end.lat * d2r; this.lng1 = start.lng * d2r; this.lng2 = end.lng * d2r; var dLng = this.lng1-this.lng2; var sinLat1 = Math.sin(this.lat1); var sinLat2 = Math.sin(this.lat2); var cosLat1 = Math.cos(this.lat1); var cosLat2 = Math.cos(this.lat2); this.sinLat1CosLat2 = sinLat1*cosLat2; this.sinLat2CosLat1 = sinLat2*cosLat1; this.cosLat1CosLat2SinDLng = cosLat1*cosLat2*Math.sin(dLng); }; GeodesicLine.prototype.isMeridian = function() { return this.lng1 == this.lng2; }; GeodesicLine.prototype.latAtLng = function(lng) { lng = lng * Math.PI / 180; //to radians var lat; // if we're testing the start/end point, return that directly rather than calculating // 1. this may be fractionally faster, no complex maths // 2. there's odd rounding issues that occur on some browsers (noticed on IITC MObile) for very short links - this may help if (lng == this.lng1) { lat = this.lat1; } else if (lng == this.lng2) { lat = this.lat2; } else { lat = Math.atan ( (this.sinLat1CosLat2*Math.sin(lng-this.lng2) - this.sinLat2CosLat1*Math.sin(lng-this.lng1)) / this.cosLat1CosLat2SinDLng); } return lat * 180 / Math.PI; // return value in degrees }; // calculate the longitude of the overlapping region var leftLng = Math.max( Math.min(a0.lng,a1.lng), Math.min(b0.lng,b1.lng) ); var rightLng = Math.min( Math.max(a0.lng,a1.lng), Math.max(b0.lng,b1.lng) ); // calculate the latitudes for each line at left + right longitudes // NOTE: need a special case for meridians - as GeodesicLine.latAtLng method is invalid in that case var aLeftLat, aRightLat; if (a0.lng == a1.lng) { // 'left' and 'right' now become 'top' and 'bottom' (in some order) - which is fine for the below intersection code aLeftLat = a0.lat; aRightLat = a1.lat; } else { var aGeo = new GeodesicLine(a0,a1); aLeftLat = aGeo.latAtLng(leftLng); aRightLat = aGeo.latAtLng(rightLng); } var bLeftLat, bRightLat; if (b0.lng == b1.lng) { // 'left' and 'right' now become 'top' and 'bottom' (in some order) - which is fine for the below intersection code bLeftLat = b0.lat; bRightLat = b1.lat; } else { var bGeo = new GeodesicLine(b0,b1); bLeftLat = bGeo.latAtLng(leftLng); bRightLat = bGeo.latAtLng(rightLng); } // if both a are less or greater than both b, then lines do not cross if (aLeftLat < bLeftLat && aRightLat < bRightLat) return false; if (aLeftLat > bLeftLat && aRightLat > bRightLat) return false; // latitudes cross between left and right - so geodesic lines cross return true; }; self.testPolyLine = function (polyline, link,closed) { var a = link.getLatLngs(); var b = polyline.getLatLngs(); for (var i=0;i' + displaylength + ''; rows.push({length:length,text:direction + ' ' + title + ' - ' + displaylength}); } rows.sort( function(a, b) { if (a.length === b.length) { return 0; } else { return (a.length > b.length) ? 1 : -1; } }); let rowstext = []; for (let cnt = 0; cnt < rows.length; cnt++) { rowstext.push(rows[cnt].text); } return rowstext.join('
\n') + '
\n'; }; self.overviewConnected = function() { var drawlayer = self.getDrawlayer(); if (!drawlayer) return; if (!window.selectedPortal) return; var portal = window.portals[window.selectedPortal]; var portalLinks = { in: [], out: [] }; $.each(window.links, function(linkguid,link) { if (link.options.data.oGuid === window.selectedPortal) portalLinks.out.push(link); if (link.options.data.dGuid === window.selectedPortal) portalLinks.in.push(link); }); var position = portal.getLatLng(); var drawnLinks = { in: [], out: [] }; drawlayer.eachLayer( function(layer) { //if (layer instanceof L.GeodesicPolyline && layer.getLatLngs().length === 2) { if (layer instanceof L.GeoJSON && layer.getLatLngs().length === 2) { var latLngs = layer.getLatLngs(); if (position.lat === latLngs[0].lat && position.lng === latLngs[0].lng) { // portal position matches first draw links position (base) drawnLinks.out.push(layer); } else if (position.lat === latLngs[1].lat && position.lng === latLngs[1].lng) { // portal position matches second draw links position (target) drawnLinks.in.push(layer); } } }); var html = '
' + '< Main menu' + '
' + 'Selected portal (refresh):
\n' + '' + portal.options.data.title + '
\n' + '
\n' + 'Drawn links: ' + (drawnLinks.in.length + drawnLinks.out.length) + ' (' + drawnLinks.out.length + ' out, ' + drawnLinks.in.length + ' in)
\n' + self.linkedportalshtml(drawnLinks.out,position,true) + self.linkedportalshtml(drawnLinks.in,position,true) + '
\n' + 'Existing links: ' + (portalLinks.in.length + portalLinks.out.length) + ' (' + portalLinks.out.length + ' out, ' + portalLinks.in.length + ' in)
\n' + self.linkedportalshtml(portalLinks.in,position,false) + self.linkedportalshtml(portalLinks.out,position,false) + '
'; if (window.useAndroidPanes()) window.show("map"); window.dialog({ html: html, id: self.pluginname + '-dialog', dialogClass: 'ui-dialog-quickdrawlinks', title: self.title + ' Overview', width: 400 }); }; self.showAll = function() { var drawlayer = self.getDrawlayer(); if (Object.keys(drawlayer._layers).length === 0) { alert('No drawn links to display'); return; } map.fitBounds(drawlayer.getBounds()); }; self.showLink = function(link) { var drawlayer = self.getDrawlayer(); if (!link) return; map.fitBounds(link.getBounds()); }; self.setDrawColor = function(color) { self.settings.drawcolor = color; self.storesettings(); }; self.closedialog = function() { $(".ui-dialog-content").dialog("close"); }; self.removeselectedlink = function() { if (!self.selectedlink) return; self.removelink(self.selectedlink); self.selectedlink = undefined; self.closedialog(); }; self.setSelectedLinkColor = function(color) { if (!self.selectedlink) return; self.selectedlink.setStyle({color:color}); self.selectedlink.options.color = color; // added for L.geoJson compatibility self.storelinks(); }; self.updateAllLinkStyle = function() { var drawlayer = self.getDrawlayer(); drawlayer.eachLayer( function(layer) { //if (layer instanceof L.GeodesicPolyline && layer.getLatLngs().length === 2) { if (layer instanceof L.GeoJSON && layer.getLatLngs().length === 2) { layer.setStyle({dashArray: (self.settings.showlinkdirection?self.linkstyle:null)}); } }); }; self.findPortalGuid = 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.selectportalposition = function(position) { if (Object.keys(window.portals).length === 0) return; // cancel while no portals loaded yet; prevents an error inside the IITC core var guid = self.getguid(position); if (!guid) guid = self.findPortalGuid(position); if (!guid) return false; if (guid === window.selectedPortal) { var visibleBounds = map.getBounds(); if (!visibleBounds.contains(position)) window.map.setView(position, map.getZoom()); return true; } if (window.portals[guid]) { window.renderPortalDetails(guid); } else { self.requestid = guid; window.portalDetail.request(guid); return false; } return true; }; self.selectlinkstart = function() { if (!self.selectedlink) return; self.selectportalposition(self.selectedlink.getLatLngs()[0]); }; self.selectlinkend = function() { if (!self.selectedlink) return; self.selectportalposition(self.selectedlink.getLatLngs()[1]); }; self.movelinkstart = function() { if (!self.selectedlink) return; var position = self.selectedlink.getLatLngs()[0]; if (self.selectportalposition(position)) { self.moveselectedlinkstartposition = position; self.moveselectedlinkendposition = undefined; self.addMarker(window.selectedPortal,{icon:'move'}); self.closedialog(); } else { alert('Portal not found'); } }; self.movelinkend = function() { if (!self.selectedlink) return; var position = self.selectedlink.getLatLngs()[1]; if (self.selectportalposition(position)) { self.moveselectedlinkstartposition = undefined; self.moveselectedlinkendposition = position; self.addMarker(window.selectedPortal,{icon:'move'}); self.closedialog(); } else { alert('Portal not found'); } }; self.swapstartend = function() { if (!self.selectedlink) return; var latLngs = self.clonelatlngs(self.selectedlink.getLatLngs()); var latLng0 = new L.LatLng(latLngs[0].lat,latLngs[0].lng); var latLng1 = new L.LatLng(latLngs[1].lat,latLngs[1].lng); latLngs[0] = latLng1; latLngs[1] = latLng0; //self.selectedlink.setLatLngs(latLngs); var drawlayer = self.getDrawlayer(); var color = self.selectedlink.options.color; drawlayer.removeLayer(self.selectedlink); self.selectedlink = self.drawline(latLngs,color); self.storelinks(); }; self.linkstarttitle = function() { if (!self.selectedlink) return; var title = self.gettitle(self.selectedlink.getLatLngs()[0]); return (title?title:'unknown'); }; self.linkendtitle = function() { if (!self.selectedlink) return; var title = self.gettitle(self.selectedlink.getLatLngs()[1]); return (title?title:'unknown'); }; self.linklength = function(link) { if (!link) link = self.selectedlink; if (!link) return ''; var latlngs = link.getLatLngs(); //var length_old = L.latLng(latlngs[0].lat,latlngs[0].lng).distanceTo([latlngs[1].lat,latlngs[1].lng]); var length = self.distanceBetween(latlngs[0],latlngs[1]) * 1000; return length < 100000 ? Math.round(length) + 'm' : Math.round(length/1000) + 'km'; }; self.drawgreatcircle = function(linklayer) { var drawPolylineLines = window.map.hasLayer(self.greatcircleslayer); if (!drawPolylineLines) return; //var vincenty_ellipsoid = { a: 6378137, b: 6356752.3142, f: 1/298.257223563 }; // WGS-84 var vincenty_ellipsoid = { a: 6367000, b: 6367000, f: 0 }; // Sphere // From Leaflet.Geodesic (https://github.com/henrythasler/Leaflet.Geodesic/) var vincenty_inverse = function (p1, p2) { var φ1 = p1.lat.toRadians(), λ1 = p1.lng.toRadians(); var φ2 = p2.lat.toRadians(), λ2 = p2.lng.toRadians(); var a = vincenty_ellipsoid.a, b = vincenty_ellipsoid.b, f = vincenty_ellipsoid.f; var L = λ2 - λ1; var tanU1 = (1-f) * Math.tan(φ1), cosU1 = 1 / Math.sqrt((1 + tanU1*tanU1)), sinU1 = tanU1 * cosU1; var tanU2 = (1-f) * Math.tan(φ2), cosU2 = 1 / Math.sqrt((1 + tanU2*tanU2)), sinU2 = tanU2 * cosU2; var λ = L, λʹ, iterations = 0; var sinλ, cosλ, σ, sinσ, cosσ, cosSqα, cos2σM; do { sinλ = Math.sin(λ); cosλ = Math.cos(λ); var sinSqσ = (cosU2*sinλ) * (cosU2*sinλ) + (cosU1*sinU2-sinU1*cosU2*cosλ) * (cosU1*sinU2-sinU1*cosU2*cosλ); sinσ = Math.sqrt(sinSqσ); if (sinσ==0) return 0; // co-incident points cosσ = sinU1*sinU2 + cosU1*cosU2*cosλ; σ = Math.atan2(sinσ, cosσ); var sinα = cosU1 * cosU2 * sinλ / sinσ; cosSqα = 1 - sinα*sinα; cos2σM = cosσ - 2*sinU1*sinU2/cosSqα; if (isNaN(cos2σM)) cos2σM = 0; // equatorial line: cosSqα=0 (§6) var C = f/16*cosSqα*(4+f*(4-3*cosSqα)); λʹ = λ; λ = L + (1-C) * f * sinα * (σ + C*sinσ*(cos2σM+C*cosσ*(-1+2*cos2σM*cos2σM))); } while (Math.abs(λ-λʹ) > 1e-12 && ++iterations<100); if (iterations>=100) { console.log('Formula failed to converge. Altering target position.'); return this._vincenty_inverse(p1, {lat: p2.lat, lng:p2.lng-0.01}); // throw new Error('Formula failed to converge'); } var uSq = cosSqα * (a*a - b*b) / (b*b); var A = 1 + uSq/16384*(4096+uSq*(-768+uSq*(320-175*uSq))); var B = uSq/1024 * (256+uSq*(-128+uSq*(74-47*uSq))); var Δσ = B*sinσ*(cos2σM+B/4*(cosσ*(-1+2*cos2σM*cos2σM) - B/6*cos2σM*(-3+4*sinσ*sinσ)*(-3+4*cos2σM*cos2σM))); var s = b*A*(σ-Δσ); var fwdAz = Math.atan2(cosU2*sinλ, cosU1*sinU2-sinU1*cosU2*cosλ); var revAz = Math.atan2(cosU1*sinλ, -sinU1*cosU2+cosU1*sinU2*cosλ); s = Number(s.toFixed(3)); // round to 1mm precision return { distance: s, initialBearing: fwdAz.toDegrees(), finalBearing: revAz.toDegrees() }; }; //From Leaflet.Geodesic (https://github.com/henrythasler/Leaflet.Geodesic/) var vincenty_direct = function (p1, initialBearing, distance, wrap) { var φ1 = p1.lat.toRadians(), λ1 = p1.lng.toRadians(); var α1 = initialBearing.toRadians(); var s = distance; var a = vincenty_ellipsoid.a, b = vincenty_ellipsoid.b, f = vincenty_ellipsoid.f; var sinα1 = Math.sin(α1); var cosα1 = Math.cos(α1); var tanU1 = (1-f) * Math.tan(φ1), cosU1 = 1 / Math.sqrt((1 + tanU1*tanU1)), sinU1 = tanU1 * cosU1; var σ1 = Math.atan2(tanU1, cosα1); var sinα = cosU1 * sinα1; var cosSqα = 1 - sinα*sinα; var uSq = cosSqα * (a*a - b*b) / (b*b); var A = 1 + uSq/16384*(4096+uSq*(-768+uSq*(320-175*uSq))); var B = uSq/1024 * (256+uSq*(-128+uSq*(74-47*uSq))); var σ = s / (b*A), σʹ, iterations = 0; var cos2σM; var sinσ; var cosσ; do { cos2σM = Math.cos(2*σ1 + σ); sinσ = Math.sin(σ); cosσ = Math.cos(σ); var Δσ = B*sinσ*(cos2σM+B/4*(cosσ*(-1+2*cos2σM*cos2σM)- B/6*cos2σM*(-3+4*sinσ*sinσ)*(-3+4*cos2σM*cos2σM))); σʹ = σ; σ = s / (b*A) + Δσ; } while (Math.abs(σ-σʹ) > 1e-12 && ++iterations); var x = sinU1*sinσ - cosU1*cosσ*cosα1; var φ2 = Math.atan2(sinU1*cosσ + cosU1*sinσ*cosα1, (1-f)*Math.sqrt(sinα*sinα + x*x)); var λ = Math.atan2(sinσ*sinα1, cosU1*cosσ - sinU1*sinσ*cosα1); var C = f/16*cosSqα*(4+f*(4-3*cosSqα)); var L = λ - (1-C) * f * sinα * (σ + C*sinσ*(cos2σM+C*cosσ*(-1+2*cos2σM*cos2σM))); var λ2; if (wrap) λ2 = (λ1+L+3*Math.PI)%(2*Math.PI) - Math.PI; // normalise to -180...+180 else λ2 = (λ1+L); // do not normalize var revAz = Math.atan2(sinα, -x); return { lat: φ2.toDegrees(), lng: λ2.toDegrees(), finalBearing: revAz.toDegrees() }; }; var drawLink = function(a, b, style, layerGroup) { var poly = L.geodesicPolyline([a, b], style); poly.addTo(layerGroup); }; var extendEdge = function(a,b,layerGroup) { if(!a || !b) return; var inverse = vincenty_inverse(a,b); var maxLinkDistance = 6881280; var maxLinkToAnchor = maxLinkDistance - inverse.distance; if(maxLinkToAnchor < 0) return; var direct = vincenty_direct(b, inverse.finalBearing, maxLinkToAnchor, true); var c = new L.LatLng(direct.lat, direct.lng); drawLink(b, c, { color: self.settings.greatcirclecolor, opacity: 0.9, weight: 1, clickable: false, smoothFactor: 1, dashArray: null //[6, 4], }, layerGroup); }; var vertices = linklayer.getLatLngs(); $.each(vertices, function(idx, vertex) { var previousVertex = idx === 0 ? null : vertices[idx - 1]; var nextVertex = idx === (vertices.length - 1) ? null : vertices[idx + 1]; extendEdge(previousVertex, vertex, self.greatcircleslayer); extendEdge(nextVertex, vertex, self.greatcircleslayer); }); }; self.updategreatcircleslayer = function() { if (!self.layeractive('Quick Draw Great Circles')) return; self.greatcircleslayer.clearLayers(); var drawlayer = self.getDrawlayer(); if (!drawlayer) return; drawlayer.eachLayer(function(layer) { //if (layer instanceof L.GeodesicPolyline && layer.getLatLngs().length === 2) { if (layer instanceof L.GeoJSON && layer.getLatLngs().length === 2) { self.drawgreatcircle(layer); } }); }; self.removegreatcircles = function() { var drawlayer = self.getDrawlayer(); self.greatcircleslayer.clearLayers(); // erase all }; 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.updatefieldslayer = function() { if (!self.layeractive('Quick Draw Fields')) return; // draw triangles self.fieldslayer.clearLayers(); var drawlayer = self.getDrawlayer(); if (!drawlayer) return; var layers = []; drawlayer.eachLayer(function(layer) { //if (layer instanceof L.GeodesicPolyline && layer.getLatLngs().length === 2) { if (layer instanceof L.GeoJSON && layer.getLatLngs().length === 2) { layers.push(layer); } }); function findcornermatch(corner,layers,startcnt) { var cnt; for (cnt = startcnt; cnt < layers.length; cnt++) { var latlngs = layers[cnt].getLatLngs(); if (corner[0].lat === latlngs[0].lat && corner[0].lng === latlngs[0].lng) { // matching corner found return [0,cnt,0]; } else if (corner[0].lat === latlngs[1].lat && corner[0].lng === latlngs[1].lng) { // matching corner found return [0,cnt,1]; } else if (corner[1].lat === latlngs[0].lat && corner[1].lng === latlngs[0].lng) { // matching corner found return [1,cnt,0]; } else if (corner[1].lat === latlngs[1].lat && corner[1].lng === latlngs[1].lng) { // matching corner found return [1,cnt,1]; } } return false; } function findlinkmatch(corner1,corner2,layers,startcnt) { var cnt; for (cnt = startcnt; cnt < layers.length; cnt++) { var latlngs = layers[cnt].getLatLngs(); if (corner1.lat === latlngs[0].lat && corner1.lng === latlngs[0].lng && corner2.lat === latlngs[1].lat && corner2.lng === latlngs[1].lng) { return cnt; } else if (corner1.lat === latlngs[1].lat && corner1.lng === latlngs[1].lng && corner2.lat === latlngs[0].lat && corner2.lng === latlngs[0].lng) { return cnt; } } return false; } var polygonOptions = { stroke: false, color: null, weight: 4, opacity: 0.5, clickable: false, fill: true, fillColor: self.settings.fieldcolor, fillOpacity: 0.2 }; var cnt; if (self.settings.fieldexistinglinks) { var drawncorners = {}; for (cnt = 0; cnt < layers.length; cnt++) { var layerlatlngs = layers[cnt].getLatLngs(); drawncorners[layerlatlngs[0].lat] = 1; drawncorners[layerlatlngs[0].lng] = 1; drawncorners[layerlatlngs[1].lat] = 1; drawncorners[layerlatlngs[1].lng] = 1; } // include all drawn links for current player faction that match a corner of drawn links var playerteam = (PLAYER.team === 'RESISTANCE'?'R':'E'); for (var guid in window.links) { var link = window.links[guid]; if (link.options.data.team === playerteam) { var linklatlngs = link.getLatLngs(); if (drawncorners[linklatlngs[0].lat] || drawncorners[linklatlngs[0].lng] || drawncorners[linklatlngs[1].lat] || drawncorners[linklatlngs[1].lng]) { layers.push(link); } } } } // to do: find 3 links with 3 matching corners // step 1: test every link, start and end point, for a match with any other (next) start or end point for (cnt = 0; cnt < layers.length; cnt++) { var link1latlngs = layers[cnt].getLatLngs(); var cornermatch; cornermatch = findcornermatch(link1latlngs,layers,cnt + 1); // sourcecorner,matchcnt,matchcorner do { if (cornermatch instanceof Array) { // matching corner found: layers[cnt].getLatLngs()[cornermatch[0]] === layers[cornermatch[1]].getLatLngs()[cornermatch[2]] // step 2: test every (next) link for a match with the 2 opposite corners of the 2 matching links var link2latlngs = layers[cornermatch[1]].getLatLngs(); var linkmatch = findlinkmatch(link1latlngs[1 - cornermatch[0]],link2latlngs[1 - cornermatch[2]],layers,cornermatch[1] + 1); if (linkmatch) { // draw a triangle polygon //console.log(link1latlngs[0],link1latlngs[1],link2latlngs[1 - cornermatch[2]]); /* var trianglejson = { "type": "Polygon", "coordinates": [ [ [link1latlngs[0].lat, link1latlngs[0].lng], [link1latlngs[1].lat, link1latlngs[1].lng], [link2latlngs[1 - cornermatch[2]].lat, link2latlngs[1 - cornermatch[2]].lng], [link1latlngs[0].lat, link1latlngs[0].lng] ] ] }; */ var latLngs = [link1latlngs[0],link1latlngs[1],link2latlngs[1 - cornermatch[2]]]; var layer = L.geodesicPolygon(latLngs,polygonOptions); self.fieldslayer.addLayer(layer).bringToBack(); // link1latlngs[0],link1latlngs[1],link2latlngs[1 - cornermatch[2]] } // check for another match cornermatch = findcornermatch(link1latlngs,layers,cornermatch[1] + 1); // sourcecorner,matchcnt,matchcorner } } while (cornermatch instanceof Array); } }; self.linkindex = function(link) { var drawlayer = self.getDrawlayer(); if (!link) link = self.selectedlink; if (!link) return -1; var links = []; var selectedindex = -1; var selectedlatlngs = link.getLatLngs(); drawlayer.eachLayer( function(layer) { //if (layer instanceof L.GeodesicPolyline && layer.getLatLngs().length === 2) { if (layer instanceof L.GeoJSON && layer.getLatLngs().length === 2) { links.push(layer); var latLngs = layer.getLatLngs(); if (selectedlatlngs && selectedlatlngs[0].lat === latLngs[0].lat && selectedlatlngs[0].lng === latLngs[0].lng && selectedlatlngs[1].lat === latLngs[1].lat && selectedlatlngs[1].lng === latLngs[1].lng) { selectedindex = links.length - 1; } } }); return selectedindex; }; self.highlightlinkbylatlng = function(lat0,lng0,lat1,lng1) { self.stophighlightlink(); var latlng0 = L.latLng(lat0, lng0); var latlng1 = L.latLng(lat1, lng1); self.highlightlink = L.layerGroup().addTo(map); L.circleMarker(latlng0, self.highlightlinkoptions) .addTo(self.highlightlink); L.circleMarker(latlng1, self.highlightlinkoptions) .addTo(self.highlightlink); L.geodesicPolyline([latlng0, latlng1], self.highlightlinkoptions) .addTo(self.highlightlink); }; self.stophighlightlink = function() { if (self.highlightlink) map.removeLayer(self.highlightlink); self.highlightlink = null; }; self.selectlinkbylatlng = function(lat0,lng0,lat1,lng1) { var drawlayer = self.getDrawlayer(); var linkfound = false; drawlayer.eachLayer( function(layer) { //if (layer instanceof L.GeodesicPolyline && layer.getLatLngs().length === 2) { if (layer instanceof L.GeoJSON && layer.getLatLngs().length === 2) { var latLngs = layer.getLatLngs(); if (lat0 === latLngs[0].lat && lng0 === latLngs[0].lng && lat1 === latLngs[1].lat && lng1 === latLngs[1].lng) { self.selectedlink = layer; linkfound = true; } } }); return linkfound; }; self.selectlink = function(offset = 0) { self.stophighlightlink(); // place links in a reference array, find the one that matches the current link var drawlayer = self.getDrawlayer(); var links = []; var selectedindex = -1; var selectedlatlngs = (self.selectedlink?self.selectedlink.getLatLngs():null); drawlayer.eachLayer( function(layer) { //if (layer instanceof L.GeodesicPolyline && layer.getLatLngs().length === 2) { if (layer instanceof L.GeoJSON && layer.getLatLngs().length === 2) { links.push(layer); var latLngs = layer.getLatLngs(); if (selectedlatlngs && selectedlatlngs[0].lat === latLngs[0].lat && selectedlatlngs[0].lng === latLngs[0].lng && selectedlatlngs[1].lat === latLngs[1].lat && selectedlatlngs[1].lng === latLngs[1].lng) { selectedindex = links.length - 1; } } }); if (links.length === 0) return; if (selectedindex === -1) { selectedindex = 0; // select first if none selected } else { selectedindex += offset; } // fix offset within range while (selectedindex < 0) { selectedindex += links.length; } while (selectedindex >= links.length) { selectedindex -= links.length; } self.selectedlink = links[selectedindex]; }; self.selectpreviouslinkmenu = function() { self.selectlink(-1); self.linkmenu(self.selectedlink); }; self.selectnextlinkmenu = function() { self.selectlink(1); self.linkmenu(self.selectedlink); }; self.linkmenu = function(line) { if (self.markerLayer !== undefined) return; // disable menu when drawing a new line if (!line) { self.selectlink(); } else { self.stophighlightlink(); self.selectedlink = line; } var html = '
' + '< Main menu' + '
' + '
' + '
' + '
' + '' + '
' + '
' + ' ' + '' + '
' + '
' + '
' + '
' + '
' + '
Start:
' + '
' + self.linkstarttitle() + '
' + '
' + '
' + '' + ' ' + ' ' + '
' + '
' + '
End:
' + '
' + self.linkendtitle() + '
' + '
' + '
' + ' ' + ' ' + '
' + '
' + (self.linkindex(line) + 1) + '/' + self.linkcount() + '
'; if (window.useAndroidPanes()) window.show('map'); window.dialog({ html: html, id: self.pluginname + '-dialog', dialogClass: 'ui-dialog-quickdrawlinks', width: 350, title: 'Edit Link (' + self.linklength() + ')' }); // need to initialise the 'spectrum' color picker $('#quickdrawlinks_linkcolor').spectrum({ flat: false, showInput: true, showButtons: true, showPalette: true, showSelectionPalette: true, allowEmpty: false, 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'] ], change: function(color) { self.setSelectedLinkColor(color.toHexString()); }, color: self.selectedlink.options.color, }); }; self.createURL = function() { var drawlayer = self.getDrawlayer(); if (self.linkcount() === 0) { alert('No drawn links'); return; } var data = []; drawlayer.eachLayer( function(layer) { //if (layer instanceof L.GeodesicPolyline && layer.getLatLngs().length === 2) { if (layer instanceof L.GeoJSON && layer.getLatLngs().length === 2) { var latLngs = layer.getLatLngs(); data.push([latLngs[0].lat,latLngs[0].lng,latLngs[1].lat,latLngs[1].lng].join(',')); } }); var mappos = map.getCenter(); var baseurl = window.location.protocol + "//" + window.location.host; var URL = baseurl + "/intel?" + "ll=" + mappos.lat + "," + mappos.lng + "&z=" + map.getZoom() + "&pls=" + data.join('_'); var html = '

Select all and press CTRL+C to copy it.

' + ''; window.dialog({ html: html, id: 'ui-dialog-quickdrawlinks-export-url', width: 600, dialogClass: 'ui-dialog-quickdrawlinks-export', title: self.title + ' URL' }); }; self.copydata = function() { if (self.linkcount() === 0) { alert('No drawn links'); return; } if (typeof android !== 'undefined' && android && android.shareString) { android.shareString(localStorage[self.localstoragelayer]); return; } var data = {}; data.layerdata = JSON.parse(localStorage[self.localstoragelayer]); data.titledata = JSON.parse(localStorage[self.localstoragetitlecache]); var html = '

Select all and press CTRL+C to copy it.

' + ''; window.dialog({ html: html, id: 'ui-dialog-quickdrawlinks-export-data', width: 600, dialogClass: 'ui-dialog-quickdrawlinks-export', title: self.title + ' Copy' }); }; self.pastedata = function(appenddata) { if (!appenddata && self.linkcount() !== 0 && !confirm('Are you sure you want to replace all data?')) return; var promptAction = prompt('Press CTRL+V to paste (' + (appenddata?'append':'replace') + ' ' + self.title + ' data or URL).', ''); if (promptAction === null || promptAction === '') return; try { // first see if it looks like a URL-format stock intel link, and if so, try and parse out any stock drawn items // from the pls parameter if (promptAction.match(new RegExp("^(https?://)?(www\\.)ingress\\.com/intel.*[?&]pls="))) { //looks like a ingress URL that has drawn items... var items = promptAction.split(/[?&]/); var foundAt = -1; var i; for (i=0; i b.toLowerCase()?1:0)); }); if (layers.length === 0) return 'nothing stored'; var list = []; if (!selectedlayer) selectedlayer = $('#quickdrawlinks_selectlayer option:selected').val(); if (!selectedlayer) selectedlayer = layers[0]; for (var index in layers) { var layername = layers[index]; list.push(''); } return ''; }; self.updatelayerlist = function(selectedlayer) { var newlist = self.projectselectlist(selectedlayer); if (newlist !== $('#quickdrawlinks_selectlayer').html()) $('#quickdrawlinks_selectlayer').replaceWith(newlist); }; self.deletestored = function() { var selectedlayer = $('#quickdrawlinks_selectlayer option:selected').val(); if (!selectedlayer) { alert('Nothing stored to delete'); return; } if (!confirm('Are you sure you want to delete stored data \'' + selectedlayer + '\'?')) return; var storeddata = self.getstoreddata(); delete(storeddata[selectedlayer]); localStorage[self.localstoragedata] = JSON.stringify(storeddata); self.updatelayerlist(); }; self.restore = function(replacelinks) { var selectedlayer = $('#quickdrawlinks_selectlayer option:selected').val(); if (!selectedlayer) { alert('Nothing to restore'); return; } var linkcount = self.linkcount(); if (linkcount !== 0) { if (replacelinks) { if (!confirm('Are you sure you want to replace all drawn links with stored data \'' + selectedlayer + '\'?')) return; } else { if (!confirm('Are you sure you want to add stored links from \'' + selectedlayer + '\' to current drawn links?')) return; } } var storeddata = self.getstoreddata(); if (replacelinks && linkcount) self.removeAll(true); self.import(storeddata[selectedlayer]); self.storelinks(); }; self.renamestored = function() { var selectedlayer = $('#quickdrawlinks_selectlayer option:selected').val(); if (!selectedlayer) { alert('Nothing to rename'); return; } var newname = prompt('Enter a new unique name for this layer:',selectedlayer); if (newname === null || newname === selectedlayer) return; if (newname === '') { alert('Layer name cannot be empty!'); return; } if (newname.match(/['"]/) !== null) { alert('Quote characters are not allowed in the layer name'); return; } var storeddata = self.getstoreddata(); if (storeddata[newname]) { alert('Layer name must be unique! \'' + newname + '\' already in use'); return; } storeddata[newname] = storeddata[selectedlayer]; delete(storeddata[selectedlayer]); localStorage[self.localstoragedata] = JSON.stringify(storeddata); self.updatelayerlist(newname); }; self.listportals = function() { if (self.linkcount() === 0) { alert('No drawn links'); return; } var drawlayer = self.getDrawlayer(); var listportalsindex = {}; var listportals = []; var linklengths = {}; drawlayer.eachLayer( function(layer) { //if (layer instanceof L.GeodesicPolyline && layer.getLatLngs().length === 2) { if (layer instanceof L.GeoJSON && layer.getLatLngs().length === 2) { var latLngs = layer.getLatLngs(); for (var cnt = 0; cnt < latLngs.length; cnt++) { var guid = self.getguid(latLngs[cnt]); var title = self.gettitle(latLngs[cnt]); if (title && guid) { var id = latLngs[cnt].lat + ',' + latLngs[cnt].lng; if (listportalsindex[id] === undefined) { listportalsindex[id] = listportals.length; listportals.push({id:id,title:title,guid:guid,countin:0,countout:0,outindex:[]}); linklengths[listportalsindex[id]] = {}; } if (cnt === 0) { listportals[listportalsindex[id]].countout++; var id1 = latLngs[1].lat + ',' + latLngs[1].lng; listportals[listportalsindex[id]].outindex.push(id1); linklengths[listportalsindex[id]][id1] = self.linklength(layer); } else { listportals[listportalsindex[id]].countin++; } } } } }); var linkout = []; // listportals sort must be AFTER linkout is created, because listportals are used by indexes for (var cnt = 0; cnt < listportals.length; cnt++) { var portalinfo = listportals[cnt]; for (var cnt2 = 0; cnt2 < listportals[cnt].outindex.length; cnt2++) { linkout.push(portalinfo.title + "\t" + listportals[listportalsindex[portalinfo.outindex[cnt2]]].title + "\t" + linklengths[cnt][portalinfo.outindex[cnt2]]); } } linkout.sort(); listportals.sort( function(a, b) { // reverse sort by countin (highest first) if (a.countin === b.countin) { return 0; } else { return (a.countin > b.countin) ? -1 : 1; } }); var baseurl = window.location.protocol + "//" + window.location.host; var linkcount = []; var locations = []; for (var cnt = 0; cnt < listportals.length; cnt++) { var portalinfo = listportals[cnt]; linkcount.push(portalinfo.title + "\t" + portalinfo.countin + "\t" + portalinfo.countout); locations.push(portalinfo.title + "\t" + baseurl + '/intel?ll=' + portalinfo.id + '&pll=' + portalinfo.id + "\t" + portalinfo.guid); } locations.sort(); var html = '

Select all and press CTRL+C to copy it.

' + ''; window.dialog({ html: html, id: 'ui-dialog-quickdrawlinks-export-data', width: 600, dialogClass: 'ui-dialog-quickdrawlinks-export', title: self.title + ' Copy' }); }; self.getDrawlayer = function() { // feature: implement multiple draw layers return self.drawnItems; }; self.getportalguidbylatlng = function(latlng) { for (var guid in window.portals) { var portallatlng = window.portals[guid].getLatLng(); if (latlng.lat === portallatlng.lat && latlng.lng === portallatlng.lng) { return guid; } } return null; }; self.addPortalBookmarkSetup = function() { if (!window.plugin.bookmarks) return; if (self.addPortalBookmark) return; var addPortalBookmark_override = window.plugin.bookmarks.addPortalBookmark.toString(); addPortalBookmark_override = addPortalBookmark_override.replace('window.runHooks','//window.runHooks'); // disable runHooks, just for this plugin eval(self.namespace + 'addPortalBookmark = ' + addPortalBookmark_override + ';'); } self.addcrosslinkbookmarks = function() { if (self.crosslinklayerdisabled) { alert('Quick Draw Cross Links layer is disabled, unable to draw bookmarks'); return; } if (self.linkcount() === 0) { alert('No drawn links, so no cross links, no bookmarks drawn'); return; } var drawlayer = self.getDrawlayer(); // getMapZoomTileParameters(getDataZoomForMapZoom(map.getZoom())).minLinkLength if (Object.keys(self.crosslinkLayerGuids).length === 0) { alert('No crosslinks found: no bookmarks drawn.\n\nBe aware: Crosslinks must be in visible range (and zoom level set at: all portals or all links)!'); return; } var countmissingportals = 0; var countnewbookmarks = 0; var countexistingbookmarks = 0; for (var crosslinkguid in self.crosslinkLayerGuids) { var latlngs = self.crosslinkLayerGuids[crosslinkguid].getLatLngs(); for (var cnt = 0; cnt < latlngs.length; cnt++) { var portalguid = self.getportalguidbylatlng(latlngs[cnt]); if (portalguid === null) { countmissingportals++; } else { var bookmarktitle = window.portals[portalguid].options.data.title || 'crosslink portal ' + portalguid; // draw new bookmark for portalguid var bkmrkData = window.plugin.bookmarks.findByGuid(portalguid); if (!bkmrkData) { var colorbackup; if (window.plugin.bookmarksAddon) { colorbackup = window.plugin.bookmarksAddon.settings.color; window.plugin.bookmarksAddon.settings.color = self.settings.crosslinkbookmarkcolor; } self.addPortalBookmark(portalguid, latlngs[cnt].lat + ',' + latlngs[cnt].lng, bookmarktitle); countnewbookmarks++; window.plugin.bookmarks.addStar(portalguid, latlngs[cnt], bookmarktitle); if (window.plugin.bookmarksAddon) { window.plugin.bookmarksAddon.settings.color = colorbackup; } } else { countexistingbookmarks++; } } } } alert('Visible crosslinks count: ' + Object.keys(self.crosslinkLayerGuids).length + '\nNew bookmarks count: ' + countnewbookmarks + '\nExisting bookmarks count: ' + countexistingbookmarks + '\nSkipped portal count: ' + countmissingportals + '\n\nBe aware: Crosslinks must be in visible range (and zoom level set at: all portals or all links)!'); }; self.storemenu = function() { var html = ''; if (window.useAndroidPanes()) window.show('map'); window.dialog({ html: $('
').append(html), id: self.pluginname + '-dialog', dialogClass: 'ui-dialog-quickdrawlinks', title: 'Quick Draw Store/Restore Projects' }); }; self.backupmenu = function() { var html = '

' + '< Main menu' + 'Export URL' + 'Export data (copy)' + 'Import data (paste)' + 'Append data (paste)' + 'List portals with drawn links' + (window.requestFile !== undefined ? 'Import file' : '') + ((typeof android !== 'undefined' && android && android.saveFile) ? 'Export file' : ''); if (window.useAndroidPanes()) window.show('map'); window.dialog({ html: html, id: self.pluginname + '-dialog', dialogClass: 'ui-dialog-quickdrawlinks', title: self.title + ' Import/Export' }); }; self.menu = function() { var html = '

' + ' Links Great circles
Fields
' + '
\n' + 'Zoom to view all drawn links' + 'Show link menu...' + 'Import/Export data...' + 'Store/Restore Projects...' + 'Delete fully visible links' + 'Remove all links' + (window.plugin.bookmarks?'Add bookmarks for crosslink portals
\n':'') + '
' + 'Show cross links on:
' + ' ' + ' ' + '
' + 'For selected portal:
\n' + 'Overview connected links' + 'Delete connected Drawn Links...' + 'Set all Drawn Links to outgoing' + 'Set all Drawn Links to incoming' + '
' + 'version ' + self.version + ' by ' + self.author + '' + '
'; if (window.useAndroidPanes()) { self.closedialog(); $('
').append(html).appendTo(document.body); } else { window.dialog({ html: $('
').append(html), id: self.pluginname + '-dialog', dialogClass: 'ui-dialog-quickdrawlinks', title: self.title }); } // need to initialise the 'spectrum' color picker var spectrumoptions = { flat: false, showInput: true, showButtons: true, showPalette: true, showSelectionPalette: true, allowEmpty: false, 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'] ]}; $('#quickdrawlinks_color').spectrum($.extend(true, spectrumoptions, { change: function(color) { self.setDrawColor(color.toHexString()); }, color: self.settings.drawcolor, })); $('#greatcircle_color').spectrum($.extend(true, spectrumoptions, { change: function(color) { self.settings.greatcirclecolor = color.toHexString(); self.storesettings(); self.updategreatcircleslayer(); }, color: self.settings.greatcirclecolor, })); $('#field_color').spectrum($.extend(true, spectrumoptions, { change: function(color) { self.settings.fieldcolor = color.toHexString(); self.storesettings(); self.updatefieldslayer(); }, color: self.settings.fieldcolor, })); }; self.onPaneChanged = function(pane) { if (pane === self.panename) self.menu(); else $("#quickdrawlinksdialog").remove(); }; self.setup_arc = function() { // source start: https://github.com/springmeyer/arc.js (Latest commit e30b63b on 6 Nov 2015) var D2R = Math.PI / 180; var R2D = 180 / Math.PI; var Coord = function(lon,lat) { this.lon = lon; this.lat = lat; this.x = D2R * lon; this.y = D2R * lat; }; Coord.prototype.view = function() { return String(this.lon).slice(0, 4) + ',' + String(this.lat).slice(0, 4); }; Coord.prototype.antipode = function() { var anti_lat = -1 * this.lat; var anti_lon = (this.lon < 0) ? 180 + this.lon : (180 - this.lon) * -1; return new Coord(anti_lon, anti_lat); }; var LineString = function() { this.coords = []; this.length = 0; }; LineString.prototype.move_to = function(coord) { this.length++; this.coords.push(coord); }; var Arc = function(properties) { this.properties = properties || {}; this.geometries = []; }; Arc.prototype.json = function() { if (this.geometries.length <= 0) { return {'geometry': { 'type': 'LineString', 'coordinates': null }, 'type': 'Feature', 'properties': this.properties }; } else if (this.geometries.length == 1) { return {'geometry': { 'type': 'LineString', 'coordinates': this.geometries[0].coords }, 'type': 'Feature', 'properties': this.properties }; } else { var multiline = []; for (var i = 0; i < this.geometries.length; i++) { multiline.push(this.geometries[i].coords); } return {'geometry': { 'type': 'MultiLineString', 'coordinates': multiline }, 'type': 'Feature', 'properties': this.properties }; } }; // TODO - output proper multilinestring Arc.prototype.wkt = function() { var wkt_string = ''; var wkt = 'LINESTRING('; var collect = function(c) { wkt += c[0] + ' ' + c[1] + ','; }; for (var i = 0; i < this.geometries.length; i++) { if (this.geometries[i].coords.length === 0) { return 'LINESTRING(empty)'; } else { var coords = this.geometries[i].coords; coords.forEach(collect); wkt_string += wkt.substring(0, wkt.length - 1) + ')'; } } return wkt_string; }; /* * http://en.wikipedia.org/wiki/Great-circle_distance * */ var GreatCircle = function(start,end,properties) { if (!start || start.x === undefined || start.y === undefined) { throw new Error("GreatCircle constructor expects two args: start and end objects with x and y properties"); } if (!end || end.x === undefined || end.y === undefined) { throw new Error("GreatCircle constructor expects two args: start and end objects with x and y properties"); } this.start = start; //new Coord(start.x,start.y); // FIXED source to match plugin this.end = end; //new Coord(end.x,end.y); // FIXED source to match plugin this.properties = properties || {}; var w = this.start.x - this.end.x; var h = this.start.y - this.end.y; var z = Math.pow(Math.sin(h / 2.0), 2) + Math.cos(this.start.y) * Math.cos(this.end.y) * Math.pow(Math.sin(w / 2.0), 2); this.g = 2.0 * Math.asin(Math.sqrt(z)); if (this.g == Math.PI) { throw new Error('it appears ' + start.view() + ' and ' + end.view() + " are 'antipodal', e.g diametrically opposite, thus there is no single route but rather infinite"); } else if (isNaN(this.g)) { throw new Error('could not calculate great circle between ' + start + ' and ' + end); } }; /* * http://williams.best.vwh.net/avform.htm#Intermediate */ GreatCircle.prototype.interpolate = function(f) { var A = Math.sin((1 - f) * this.g) / Math.sin(this.g); var B = Math.sin(f * this.g) / Math.sin(this.g); var x = A * Math.cos(this.start.y) * Math.cos(this.start.x) + B * Math.cos(this.end.y) * Math.cos(this.end.x); var y = A * Math.cos(this.start.y) * Math.sin(this.start.x) + B * Math.cos(this.end.y) * Math.sin(this.end.x); var z = A * Math.sin(this.start.y) + B * Math.sin(this.end.y); var lat = R2D * Math.atan2(z, Math.sqrt(Math.pow(x, 2) + Math.pow(y, 2))); var lon = R2D * Math.atan2(y, x); return [lon, lat]; }; /* * Generate points along the great circle */ GreatCircle.prototype.Arc = function(npoints,options) { var first_pass = []; if (!npoints || npoints <= 2) { first_pass.push([this.start.lon, this.start.lat]); first_pass.push([this.end.lon, this.end.lat]); } else { var delta = 1.0 / (npoints - 1); for (var i = 0; i < npoints; ++i) { var step = delta * i; var pair = this.interpolate(step); first_pass.push(pair); } } /* partial port of dateline handling from: gdal/ogr/ogrgeometryfactory.cpp TODO - does not handle all wrapping scenarios yet */ var bHasBigDiff = false; var dfMaxSmallDiffLong = 0; // from http://www.gdal.org/ogr2ogr.html // -datelineoffset: // (starting with GDAL 1.10) offset from dateline in degrees (default long. = +/- 10deg, geometries within 170deg to -170deg will be splited) var dfDateLineOffset = options && options.offset ? options.offset : 10; var dfLeftBorderX = 180 - dfDateLineOffset; var dfRightBorderX = -180 + dfDateLineOffset; var dfDiffSpace = 360 - dfDateLineOffset; // https://github.com/OSGeo/gdal/blob/7bfb9c452a59aac958bff0c8386b891edf8154ca/gdal/ogr/ogrgeometryfactory.cpp#L2342 for (var j = 1; j < first_pass.length; ++j) { var dfPrevX = first_pass[j-1][0]; var dfX = first_pass[j][0]; var dfDiffLong = Math.abs(dfX - dfPrevX); if (dfDiffLong > dfDiffSpace && ((dfX > dfLeftBorderX && dfPrevX < dfRightBorderX) || (dfPrevX > dfLeftBorderX && dfX < dfRightBorderX))) { bHasBigDiff = true; } else if (dfDiffLong > dfMaxSmallDiffLong) { dfMaxSmallDiffLong = dfDiffLong; } } var poMulti = []; if (bHasBigDiff && dfMaxSmallDiffLong < dfDateLineOffset) { var poNewLS = []; poMulti.push(poNewLS); for (var k = 0; k < first_pass.length; ++k) { var dfX0 = parseFloat(first_pass[k][0]); if (k > 0 && Math.abs(dfX0 - first_pass[k-1][0]) > dfDiffSpace) { var dfX1 = parseFloat(first_pass[k-1][0]); var dfY1 = parseFloat(first_pass[k-1][1]); var dfX2 = parseFloat(first_pass[k][0]); var dfY2 = parseFloat(first_pass[k][1]); if (dfX1 > -180 && dfX1 < dfRightBorderX && dfX2 == 180 && k+1 < first_pass.length && first_pass[k-1][0] > -180 && first_pass[k-1][0] < dfRightBorderX) { poNewLS.push([-180, first_pass[k][1]]); k++; poNewLS.push([first_pass[k][0], first_pass[k][1]]); continue; } else if (dfX1 > dfLeftBorderX && dfX1 < 180 && dfX2 == -180 && k+1 < first_pass.length && first_pass[k-1][0] > dfLeftBorderX && first_pass[k-1][0] < 180) { poNewLS.push([180, first_pass[k][1]]); k++; poNewLS.push([first_pass[k][0], first_pass[k][1]]); continue; } if (dfX1 < dfRightBorderX && dfX2 > dfLeftBorderX) { // swap dfX1, dfX2 var tmpX = dfX1; dfX1 = dfX2; dfX2 = tmpX; // swap dfY1, dfY2 var tmpY = dfY1; dfY1 = dfY2; dfY2 = tmpY; } if (dfX1 > dfLeftBorderX && dfX2 < dfRightBorderX) { dfX2 += 360; } if (dfX1 <= 180 && dfX2 >= 180 && dfX1 < dfX2) { var dfRatio = (180 - dfX1) / (dfX2 - dfX1); var dfY = dfRatio * dfY2 + (1 - dfRatio) * dfY1; poNewLS.push([first_pass[k-1][0] > dfLeftBorderX ? 180 : -180, dfY]); poNewLS = []; poNewLS.push([first_pass[k-1][0] > dfLeftBorderX ? -180 : 180, dfY]); poMulti.push(poNewLS); } else { poNewLS = []; poMulti.push(poNewLS); } poNewLS.push([dfX0, first_pass[k][1]]); } else { poNewLS.push([first_pass[k][0], first_pass[k][1]]); } } } else { // add normally var poNewLS0 = []; poMulti.push(poNewLS0); for (var l = 0; l < first_pass.length; ++l) { poNewLS0.push([first_pass[l][0],first_pass[l][1]]); } } var arc = new Arc(this.properties); for (var m = 0; m < poMulti.length; ++m) { var line = new LineString(); arc.geometries.push(line); var points = poMulti[m]; for (var j0 = 0; j0 < points.length; ++j0) { line.move_to(points[j0]); } } return arc; }; // source arc.js end self.arc = {}; self.arc.Coord = Coord; self.arc.Arc = Arc; self.arc.GreatCircle = GreatCircle; }; // end setup_arc 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.setup_arc(); self.setupColorpickerSpectrum(); self.isSmartphone = window.isSmartphone(); self.restoresettings(); self.lineOptions = { stroke: true, color: self.settings.drawcolor, weight: 4, opacity: 0.8, fill: false, clickable: true }; // START - Great Circles functionality /** Extend Number object with method to convert numeric degrees to radians */ if (typeof Number.prototype.toRadians == 'undefined') { Number.prototype.toRadians = function() { return this * Math.PI / 180; }; } /** Extend Number object with method to convert radians to numeric (signed) degrees */ if (typeof Number.prototype.toDegrees == 'undefined') { Number.prototype.toDegrees = function() { return this * 180 / Math.PI; }; } // END - Great Circles functionality //create a leaflet FeatureGroup to hold drawn items self.drawnItems = new L.FeatureGroup(); self.drawnItems._map = null; window.addLayerGroup(self.title, self.drawnItems,true); self.greatcircleslayer = new L.FeatureGroup(); window.addLayerGroup('Quick Draw Great Circles',self.greatcircleslayer,true); self.fieldslayer = new L.FeatureGroup(); window.addLayerGroup('Quick Draw Fields',self.fieldslayer,true); map.on('layeradd', function(obj) { // show button if(obj.layer === self.drawnItems) { self.onPortalSelected(); } if(obj.layer === self.greatcircleslayer) { self.updategreatcircleslayer(); } if(obj.layer === self.fieldslayer) { self.updatefieldslayer(); } }); map.on('layerremove', function(obj) { // hide button if(obj.layer === self.drawnItems) { self.removeMarker(); self.onPortalSelected(); } }); //window.map.on('zoomend', function(obj) { //self.updategreatcircleslayer(); //}); //load any previously saved items self.load(); self.restoretitles(); self.storelinks(); // overwrite and update data window.addHook('portalSelected', self.onPortalSelected); self.createCrossLinksLayer(); self.addPortalBookmarkSetup(); window.addHook('linkAdded', self.onLinkAdded); window.addHook('linkAdded', function() { if (!self.linktimerid) self.linktimerid = window.setTimeout(function() { self.linktimerid = undefined; self.updatefieldslayer(); },1000); }); window.addHook('mapDataRefreshEnd', self.onMapDataRefreshEnd); window.addHook('portalDetailLoaded', function(data) { if (self.requestid === data.guid) { self.requestid = undefined; window.renderPortalDetails(data.guid); } } ); //add options menu if(window.useAndroidPanes()) { android.addPane(self.panename, self.title, "ic_action_share"); addHook("paneChanged", self.onPaneChanged); } else { $('#toolbox').append('' + self.title + ''); } let titlebuttonwidth = 23; let titlebuttonheight = 23; let screenbuttonwidth = 25; let screenbuttonheight = 25; if (self.isSmartphone) { screenbuttonwidth = 30; screenbuttonheight = 30; } // Place buttons above the status-bar and add buttons to the portal details screen (in front of the portal title) let topoffset = -33; if (self.isSmartphone && window.plugin.miniMap) { // place it above the miniMap, if enabled topoffset = -196; } let leftoffset = 3; $('head').append( ''); 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);