Raw Source
Vylda / Geocache Circles

// ==UserScript==
// @name        Geocache Circles
// @namespace   inge.org.uk/userscripts
// @description Allows you to display a 0.1 mile radius circle around a cache on the Geocaching.com map page
// @match       https://www.geocaching.com/map*
// @author      JRI
// @oujs:author JRI
// @license     MIT; http://www.opensource.org/licenses/mit-license.php
// @copyright   2016-2017, James Inge (http://geo.inge.org.uk/)
// @version     0.0.5
// @icon        https://geo.inge.org.uk/userscripts/circleIcon48.png
// @icon64      https://geo.inge.org.uk/userscripts/circleIcon64.png
// @connect     www.geocaching.com
// @updateURL   http://gcgpx.cz/geocache_circles/Geocache_Circles.meta.js
// @downloadURL https://openuserjs.org/install/Vylda/Geocache_Circles.user.js
// ==/UserScript==

/*jslint browser:true */
/*jshint esversion: 6, undef:true, unused:true */
/*global L, MapSettings, window, console */

(function() {
	"use strict";
	const version = "Geocache Circles v0.0.5";
	const loggedIn = document.getElementById("uxLoginStatus_divSignedIn");
	const template = document.getElementById("cacheDetailsTemplate");
	const script = document.createElement("script");
	const circleIcon = "";

	//delete existing circle 
	function handleCircleRemove(e) {
		const gc = e.detail;
		if (window.MapSettings && MapSettings.Map && window.L && circleStorage[gc]) {
			{
				MapSettings.Map.removeLayer(circleStorage[gc]);
				delete circleStorage[gc];
			}
		}
	}

	function handleCircleRequest(e) {
		/* Fetch coordinates from cache page */
		const req = new XMLHttpRequest();
		const gc = e.detail;

		//if circle exists, delete it
		if (circleStorage[gc] && window.MapSettings && MapSettings.Map && window.L && window.L.Circle) {
			MapSettings.Map.removeLayer(circleStorage[gc]);
			delete circleStorage[gc];
			return;
		}

		req.addEventListener("load", function(ignore) {
			const r = req.responseText;
			const k = r.indexOf("mapLatLng = {");



			if (req.status < 400) {
				try {
					const { lat, lng, name } = JSON.parse(r.substring(k + 12, r.indexOf("}", k) + 1));

					if (typeof lat !== "number" || typeof lng !== "number") {
						// Missing data in JSON string
						console.error("Geocache Circles: no cache coordinates retrieved.");
						return;
					}

					if (window.MapSettings && MapSettings.Map && window.L && window.L.Circle) {
						const ll = new L.LatLng(lat, lng);
						var circle = new L.Circle(ll, 161, { weight: 2 });
						circle.addTo(MapSettings.Map)
							.bindPopup(`<p><strong>${name}</strong><br/>${ll.toUrl()}<br/>
								<a href="https://www.geocaching.com/geocache/${gc}" target="_blank">OpenCache</a><br>
								<a style="cursor: pointer;" onclick="document.dispatchEvent(new CustomEvent('gme_circle_remove', {'detail':'${gc}'}));">Remove circle</a>`);
						circleStorage[gc] = circle;
					} else {
						console.error("Geocache Circles: couldn't find map interface.");
					}
				} catch (err) {
					if (err instanceof SyntaxError) {
						console.warn(`Geocache Circles: Received ${r.length} bytes, coords at ${k} but couldn't extract cache coordinates from data (are you still logged in?):
${err}`);
					} else {
						console.error(`Geocache Circles: couldn't add circle to ssmap: ${err}`);
					}
				}
			} else {
				console.warn(`Geocache Circles: error retrieving cache page to find coords for ${gc}: ${req.statusText}`);
			}
		});
		req.open("GET", `https://www.geocaching.com/geocache/${gc}`);
		req.send();
	}

	// Don't run on frames or iframes
	if (window.top !== window.self) {
		return false;
	}

	// Check feature support
	if (!window.JSON || !window.XMLHttpRequest) {
		console.warn(`${version} requires a browser with support for JSON and XMLHttpRequest`);
		return false;
	}

	if (loggedIn === null) {
		// Warn if not logged in, as coords unavailable.  Don't quit, as user might log in later in a different window, or login detection might have failed.
		console.warn("Geocache Circles may not be able to locate caches as you don't seem to be logged in.");
	}

	if (template) {
		console.info(version);

		// Attach to cache info popup template
		template.textContent = template.textContent.replace(/<div\sclass="links\sClear">/, `<div class="links Clear"> <a class="jri-circle-link" style="cursor: pointer; text-decoration: underline;" onclick="document.dispatchEvent(new CustomEvent('gme_circle_request', {'detail':'{{=gc}}'}));"><img src="${circleIcon}" alt="O" style="vertical-align:middle; margin-right: 0.25em;" width="16" height="16" />Circle</a>&nbsp; `);

		// Add event listener to content script context
		script.type = "text/javascript";
		script.text = `"use strict";
			const circleStorage = {};
			${handleCircleRequest.toString()}
			${handleCircleRemove.toString()}
          document.addEventListener("gme_circle_request", handleCircleRequest, false);
          document.addEventListener("gme_circle_remove", handleCircleRemove, false)`;
		document.documentElement.firstChild.appendChild(script);
		document.documentElement.firstChild.removeChild(script);
	} else {
		// Couldn't find popup template
		console.error(`${version} didn't understand page structure.`);
	}
}());