// ==UserScript==
// @id iitc-plugin-draw-tools@breunigs
// @name iitc: draw tools
// @version 0.2
// @namespace https://github.com/breunigs/ingress-intel-total-conversion
// @updateURL https://raw.github.com/breunigs/ingress-intel-total-conversion/gh-pages/plugins/draw-tools.user.js
// @downloadURL https://raw.github.com/breunigs/ingress-intel-total-conversion/gh-pages/plugins/draw-tools.user.js
// @description Allows you to draw things into the current map so you may plan your next move
// @include http://www.ingress.com/intel*
// @match http://www.ingress.com/intel*
// ==/UserScript==
function wrapper() {
// ensure plugin framework is there, even if iitc is not yet loaded
if(typeof window.plugin !== 'function') window.plugin = function() {};
// PLUGIN START ////////////////////////////////////////////////////////
var DRAW_TOOLS_SHAPE_OPTIONS = {
color: '#BB56FF',
fill: false,
opacity: 1,
weight: 4,
clickable: false
};
// use own namespace for plugin
window.plugin.drawTools = function() {};
window.plugin.drawTools.loadExternals = function() {
var base = 'http://breunigs.github.com/ingress-intel-total-conversion/dist';
//~ var base = 'http://0.0.0.0:8000/dist';
$('head').append('');
load(base+'/leaflet.draw.0.1.6.js').thenRun(window.plugin.drawTools.boot);
// overwrite default Leaflet Marker icon.
L.Icon.Default.imagePath = base + '/images';
}
window.plugin.drawTools.addStyles = function() {
$('head').append('');
}
// renders buttons which are not originally part of Leaflet.draw, such
// as the clear-drawings button.
window.plugin.drawTools.addCustomButtons = function() {
$('.leaflet-control-draw .leaflet-bar-part-bottom').removeClass('leaflet-bar-part-bottom');
var undo = $('⎌')
.click(function() {
var last = null;
window.plugin.drawTools.drawnItems.eachLayer(function(l) {
last = l;
});
if(last) window.plugin.drawTools.drawnItems.removeLayer(last);
}
);
var clear = $('✗')
.click(function() {
window.plugin.drawTools.drawnItems.clearLayers();
}
);
$('.leaflet-control-draw').append(undo).append(clear);
}
// renders the draw control buttons in the top left corner
window.plugin.drawTools.addDrawControl = function() {
var drawControl = new L.Control.Draw({
rectangle: false,
polygon: false,
polyline: {
shapeOptions: DRAW_TOOLS_SHAPE_OPTIONS,
title: 'Add a (poly) line.\n\n'
+ 'Click on the button, then click on the map to\n'
+ 'define the start of the line. Continue click-\n'
+ 'ing to draw the line you want. Click the last\n'
+ 'point of the line (a small white rectangle) to\n'
+ 'finish. Double clicking also works.'
},
circle: {
shapeOptions: DRAW_TOOLS_SHAPE_OPTIONS,
title: 'Add a circle.\n\n'
+ 'Click on the button, then click-AND-HOLD on the\n'
+ 'map where the circle’s center should be. Move\n'
+ 'the mouse to control the radius. Release the mouse\n'
+ 'to finish.'
},
marker: {
title: 'Add a marker.\n\n'
+ 'Click on the button, then click on the map where\n'
+ 'you want the marker to appear. You can drag the\n'
+ 'marker around after it has been placed.'
}
});
map.addControl(drawControl);
plugin.drawTools.addCustomButtons();
}
// hacks into circle to render the radius of the circle while drawing
// and to allow the center of the circle to snap to a nearby portal.
window.plugin.drawTools.enhanceCircle = function() {
// replace _onMouseMove function to also display the radius of the
// circle
L.Circle.Draw.prototype._onMouseMove = function (e) {
var layerPoint = e.layerPoint,
latlng = e.latlng;
this._updateLabelPosition(layerPoint);
if (this._isDrawing) {
var dist = this._startLatLng.distanceTo(latlng);
dist = dist > 1000
? (dist / 1000).toFixed(2) + ' km'
: Math.ceil(dist) + ' m';
this._updateLabelText({
text: 'Release mouse to finish drawing. ',
subtext: 'radius: ' +dist }
);
this._drawShape(latlng);
}
}
// replace _onMouseDown to implement snapping
L.Circle.Draw.prototype._onMouseDown = function (e) {
this._isDrawing = true;
var snapTo = window.plugin.drawTools.getSnapLatLng(e.containerPoint);
this._startLatLng = snapTo || e.latlng;
L.DomEvent
.on(document, 'mouseup', this._onMouseUp, this)
.preventDefault(e.originalEvent);
}
}
// hacks into PolyLine to implement snapping and to remove the polyline
// markers when they are not required anymore for finishing the line.
// Otherwise they get in the way and make it harder to create a closed
// polyline.
window.plugin.drawTools.enhancePolyLine = function() {
// hack in snapping
L.Polyline.Draw.prototype._onClickOld = L.Polyline.Draw.prototype._onClick;
L.Polyline.Draw.prototype._onClick = function(e) {
var cp = map.latLngToContainerPoint(e.target.getLatLng());
var snapTo = window.plugin.drawTools.getSnapLatLng(cp);
if(snapTo) e.target._latlng = snapTo;
return this._onClickOld(e);
}
// remove polyline markers because they get in the way
L.Polyline.Draw.prototype._updateMarkerHandlerOld = L.Polyline.Draw.prototype._updateMarkerHandler;
L.Polyline.Draw.prototype._updateMarkerHandler = function() {
this._updateMarkerHandlerOld();
if (this._markers.length > 1)
this._markerGroup.removeLayer(this._markers[this._markers.length - 2]);
}
}
// given a container point it tries to find the most suitable portal to
// snap to. It takes the CircleMarker’s radius and weight into account.
// Will return null if nothing to snap to or a LatLng instance.
window.plugin.drawTools.getSnapLatLng = function(containerPoint) {
var candidates = [];
$.each(window.portals, function(guid, portal) {
var ll = portal.getLatLng();
var pp = map.latLngToContainerPoint(ll);
var size = portal.options.weight + portal.options.radius;
var dist = pp.distanceTo(containerPoint);
if(dist > size) return true;
candidates.push([dist, ll]);
});
if(candidates.length === 0) return;
candidates = candidates.sort(function(a, b) { return a[0]-b[0]; });
return candidates[0][1];
}
window.plugin.drawTools.boot = function() {
plugin.drawTools.enhanceCircle();
plugin.drawTools.enhancePolyLine();
plugin.drawTools.addStyles();
plugin.drawTools.addDrawControl();
window.plugin.drawTools.drawnItems = new L.LayerGroup();
var drawnItems = window.plugin.drawTools.drawnItems;
map.on('draw:poly-created', function (e) {
drawnItems.addLayer(e.poly);
});
map.on('draw:circle-created', function (e) {
drawnItems.addLayer(e.circ);
});
map.on('draw:marker-created', function (e) {
drawnItems.addLayer(e.marker);
e.marker.dragging.enable();
});
window.layerChooser.addOverlay(drawnItems, 'Drawn Items');
map.addLayer(drawnItems);
}
var setup = window.plugin.drawTools.loadExternals;
// PLUGIN END //////////////////////////////////////////////////////////
if(window.iitcLoaded && typeof setup === 'function') {
setup();
} else {
if(window.bootPlugins)
window.bootPlugins.push(setup);
else
window.bootPlugins = [setup];
}
} // wrapper end
// inject code into site context
var script = document.createElement('script');
script.appendChild(document.createTextNode('('+ wrapper +')();'));
(document.body || document.head || document.documentElement).appendChild(script);