NOTICE: By continued use of this site you understand and agree to the binding Terms of Service and Privacy Policy.
// ==UserScript== // @name Geocaching Map Enhancements // @version 0.9.0 // @author JRI // @oujs:author JRI // @namespace inge.org.uk/userscripts // @description Adds extra maps and grid reference search to Geocaching.com, along with several other enhancements. // @include https://www.geocaching.com/* // @license MIT; http://www.opensource.org/licenses/mit-license.php // @copyright 2011-18, James Inge (http://geo.inge.org.uk/) // @attribution GeoNames (http://www.geonames.org/) // @attribution Postcodes.io (https://postcodes.io/) // @attribution Chris Veness (http://www.movable-type.co.uk/scripts/latlong-gridref.html) // @grant GM_xmlhttpRequest // @grant GM.xmlHttpRequest // @connect geograph.org.uk // @connect channel-islands.geographs.org // @connect geo-en.hlipp.de // @connect api.geonames.org // @connect api.postcodes.io // @connect www.geocaching.com // @icon http://gcgpx.cz/gme/GeocachingMap48.png // @icon64 http://gcgpx.cz/gme/GeocachingMap64.png // @updateURL http://gcgpx.cz/gme/GeocachingMapEnhancements.meta.js // @downloadURL https://openuserjs.org/install/Vylda/Geocaching_Map_Enhancements.user.js // ==/UserScript== /* jshint multistr: true */ /* global $, amplify, DMM, FileReader, GM, GM_xmlhttpRequest, Groundspeak, L, LatLon, mapLatLng, MapSettings */ (function () { "use strict"; var gmeResources = { parameters: { // Defaults version: "0.9.0", versionMsg: "Icons fix, API key, Default hill shader", brightness: 1, // Default brightness for maps (0-1), can be overridden by custom map parameters. filterFinds: false, // True filters finds out of list searches. follow: false, // Locator widget follows current location (moving map mode) labels: "codes", // Label caches on the map with their GC code. Or "names" to use long name. measure: "metric", // Or "imperial" - used for the scale indicators osgbSearch: true, // Enhance search box with OSGB grid references, zooming, etc. (may interfere with postal code searches) defaultMap: "OpenStreetMap", defaultHillShading: "Hillshading", maps: [ // {alt:"Readable Name", tileUrl: "URL template including {s} (subdomain) and either {q} (quadkey) or {x},{y},{z} (Google/TMS tile coordinates + zoom)", subdomains: "0123", minZoom: 0, maxZoom: 24, attribution: "Copyright message (HTML allowed)", name: "shortname", overlay:false } { alt: "OpenStreetMap", tileUrl: "https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png", name: "osm", subdomains: "abc" }, { alt: "OpenCycleMap", tileUrl: "https://tile.thunderforest.com/cycle/{z}/{x}/{y}.png", name: "ocm", apiKey: 'OpenCycleMap', apiKeyQuery: "apikey={apikey}" }, { alt: "Bing Maps", tileUrl: "https://ecn.t{s}.tiles.virtualearth.net/tiles/r{q}?g=864&mkt=en-gb&lbl=l1&stl=h&shading=hill&n=z", subdomains: "0123", minZoom: 1, maxZoom: 20, attribution: "<a href=\'https://www.bing.com/maps/\'>Bing</a> map data copyright Microsoft and its suppliers", name: "bingmap", ignore: true }, { alt: "Bing Aerial View", tileUrl: "https://ecn.t{s}.tiles.virtualearth.net/tiles/a{q}?g=737&n=z", subdomains: "0123", minZoom: 1, maxZoom: 20, attribution: "<a href=\'https://www.bing.com/maps/\'>Bing</a> map data copyright Microsoft and its suppliers", name: "bingaerial" }, { alt: "Google Maps", tileUrl: "https://mt.google.com/vt?&x={x}&y={y}&z={z}", name: "googlemaps", attribution: "<a href=\'https://maps.google.com/\'>Google</a> Maps", subdomains: "1234", tileSize: 256, maxZoom: 22 }, { alt: "Google Satellite", tileUrl: "https://mt.google.com/vt?lyrs=s&x={x}&y={y}&z={z}", name: "googlemapssat", attribution: "<a href=\'https://maps.google.com/\'>Google</a> Maps Satellite", subdomains: "1234", tileSize: 256, maxZoom: 22 }, { alt: "Freemap Slovakia Hiking", tileUrl: "http://t{s}.freemap.sk/T/{z}/{x}/{y}.jpeg", attribution: "Map © <a href='http://www.freemap.sk/'>Freemap Slovakia</a>, data © <a href='http://openstreetmap.org'>OpenStreetMap</a> contributors", subdomains: "1234", minZoom: 8, maxZoom: 16, ignore: true }, { alt: "Freemap Slovakia Bicycle", tileUrl: "http://t{s}.freemap.sk/C/{z}/{x}/{y}.jpeg", attribution: "Map © <a href='http://www.freemap.sk/'>Freemap Slovakia</a>, data © <a href='http://openstreetmap.org'>OpenStreetMap</a> contributors", subdomains: "1234", minZoom: 8, maxZoom: 16, ignore: true }, { alt: "Freemap Slovakia Car", tileUrl: "http://t{s}.freemap.sk/A/{z}/{x}/{y}.jpeg", attribution: "Map © <a href='http://www.freemap.sk/'>Freemap Slovakia</a>, data © <a href='http://openstreetmap.org'>OpenStreetMap</a> contributors", subdomains: "1234", minZoom: 8, maxZoom: 16, ignore: true }, { alt: "Mapy.cz - Turistická", tileUrl: "https://m{s}.mapserver.mapy.cz/turist-m/{z}-{x}-{y}", minZoom: 5, maxZoom: 18, subdomains: "1234", attribution: "© <a href='//www.seznam.cz' target='_blank'>Seznam.cz, a.s.</a>, © <a href='http://www.openstreetmap.org/copyright'>OpenStreetMap</a>, © NASA" }, { alt: "Hillshading", tileUrl: "http://{s}.tiles.wmflabs.org/hillshading/{z}/{x}/{y}.png", subdomains: "abc", attribution: "Hillshading by <a href=\'https://wiki.openstreetmap.org/wiki/Hike_%26_Bike_Map\'>Colin Marquardt</a> from NASA SRTM data", overlay: true } ], apiKeys: {}, }, css: { main: '.leaflet-control-gme,.leaflet-control-zoomwarning {border-radius:7px; filter: progid:DXImageTransform.Microsoft.gradient(startColorStr="#3F000000",EndColorStr="#3F000000"); padding:5px;z-index:8;}\ .leaflet-control-gme {display: inline-block; padding: 0; background: rgba(0, 0, 0, 0.2); box-shadow: 0 0 8px rgba(0, 0, 0, 0.4);}\ .gme-control-scale {bottom:5em !important;margin-left:13px !important; left: 385px;}\ .gme-left {left: 385px; margin-left:13px !important;}\ div.gme-identify-layer {margin-top:-1em;margin-left:1em;padding-left:0.1em;font-weight:bold;background:rgba(255,255,255,0.57);}\ #gme_caches table { margin-top: 0.5em; }\ .GME_search_list { border: 1px solid #679300; border-radius: 7px; padding: 0.5em; }\ div.GME_search_results { margin-right: -65px; }\ .GME_search_results.hidden { display: none; }\ .groundspeak-control-findmylocation { border: 1px solid #888; border-radius: 5px; box-shadow: 0 0 8px rgba(0, 0, 0, 0.4); padding:0; background:rgba(255,255,255,0.8);}\ .groundspeak-control-findmylocation a { padding: 3px; }\ .gme-button { display: inline-block; box-sizing: content-box; -moz-box-sizing: content-box; padding:3px; vertical-align:middle; background:url(https://www.geocaching.com/map/css/themes/images/icons-18-black.png) no-repeat #eee; background-color: rgba(255,255,255,0.8); border: 1px solid #888; border-right:0; height:19px; width:19px; text-decoration: none; }\ .gme-button-l { border-bottom-left-radius:5px; border-top-left-radius:5px; }\ .gme-button-r { border-right: 1px solid #888; border-bottom-right-radius: 5px; border-top-right-radius:5px; margin-right:0.5em;}\ .gme-button:hover { background-color: #fff; }\ .gme-button-active {border:solid 3px #02b; padding:1px 0 1px 1px; background-color:#fff;}\ .gme-button-active:hover {border-color:#63f;filter:alpha(opacity=100);}\ span.gme-button, .gme-button-wide { padding-left:5px; padding-right:5px; font-size:12px; font-weight:bold; width:auto; background-image:none; color: #424242; }\ .GME_home { background-size: 18px 18px; background-position: center;background-image: url(\'data:image/svg+xml,%3Csvg xmlns="http://www.w3.org/2000/svg" x="0px" y="0px" width="50" height="50" viewBox="0 0 30 30" style="fill:%23000000;"%3E%3Cpath d="M 15 2 A 1 1 0 0 0 14.300781 2.2851562 L 3.3925781 11.207031 A 1 1 0 0 0 3.3554688 11.236328 L 3.3183594 11.267578 L 3.3183594 11.269531 A 1 1 0 0 0 3 12 A 1 1 0 0 0 4 13 L 5 13 L 5 24 C 5 25.105 5.895 26 7 26 L 23 26 C 24.105 26 25 25.105 25 24 L 25 13 L 26 13 A 1 1 0 0 0 27 12 A 1 1 0 0 0 26.681641 11.267578 L 26.666016 11.255859 A 1 1 0 0 0 26.597656 11.199219 L 25 9.8925781 L 25 6 C 25 5.448 24.552 5 24 5 L 23 5 C 22.448 5 22 5.448 22 6 L 22 7.4394531 L 15.677734 2.2675781 A 1 1 0 0 0 15 2 z M 18 15 L 22 15 L 22 23 L 18 23 L 18 15 z"%3E%3C/path%3E%3C/svg%3E\')}\ .GME_config { background-size: 18px 18px; background-position: center;background-image: url(\'data:image/svg+xml,%3Csvg xmlns="http://www.w3.org/2000/svg" x="0px" y="0px" width="50" height="50" viewBox="0 0 32 32" style="fill:%23000000;"%3E%3Cg id="surface1"%3E%3Cpath style=" " d="M 21.5 2.5 L 21.5 3.90625 C 20.664063 4.054688 19.886719 4.371094 19.21875 4.84375 L 18.1875 3.875 L 16.8125 5.34375 L 17.8125 6.28125 C 17.363281 6.9375 17.050781 7.6875 16.90625 8.5 L 15.5 8.5 L 15.5 10.5 L 16.90625 10.5 C 17.050781 11.328125 17.378906 12.085938 17.84375 12.75 L 16.78125 13.78125 L 18.21875 15.21875 L 19.25 14.15625 C 19.914063 14.621094 20.671875 14.949219 21.5 15.09375 L 21.5 16.5 L 23.5 16.5 L 23.5 15.09375 C 24.3125 14.949219 25.0625 14.636719 25.71875 14.1875 L 26.65625 15.1875 L 28.125 13.8125 L 27.15625 12.78125 C 27.628906 12.113281 27.945313 11.335938 28.09375 10.5 L 29.5 10.5 L 29.5 8.5 L 28.09375 8.5 C 27.949219 7.671875 27.621094 6.914063 27.15625 6.25 L 28.09375 5.3125 L 26.6875 3.90625 L 25.75 4.84375 C 25.085938 4.378906 24.328125 4.050781 23.5 3.90625 L 23.5 2.5 Z M 22.5 5.8125 C 24.554688 5.8125 26.1875 7.445313 26.1875 9.5 C 26.1875 11.554688 24.554688 13.1875 22.5 13.1875 C 20.445313 13.1875 18.8125 11.554688 18.8125 9.5 C 18.8125 7.445313 20.445313 5.8125 22.5 5.8125 Z M 9.53125 11.71875 L 7.6875 12.46875 L 8.40625 14.28125 C 7.453125 14.851563 6.640625 15.648438 6.0625 16.59375 L 4.28125 15.875 L 3.53125 17.71875 L 5.3125 18.4375 C 5.179688 18.964844 5.09375 19.523438 5.09375 20.09375 C 5.09375 20.664063 5.179688 21.21875 5.3125 21.75 L 3.53125 22.46875 L 4.28125 24.3125 L 6.0625 23.59375 C 6.640625 24.554688 7.445313 25.359375 8.40625 25.9375 L 7.6875 27.71875 L 9.53125 28.46875 L 10.25 26.6875 C 10.78125 26.820313 11.332031 26.90625 11.90625 26.90625 C 12.476563 26.90625 13.035156 26.820313 13.5625 26.6875 L 14.28125 28.46875 L 16.125 27.71875 L 15.40625 25.9375 C 16.351563 25.359375 17.148438 24.546875 17.71875 23.59375 L 19.53125 24.3125 L 20.28125 22.46875 L 18.46875 21.75 C 18.601563 21.21875 18.6875 20.664063 18.6875 20.09375 C 18.6875 19.523438 18.601563 18.964844 18.46875 18.4375 L 20.28125 17.71875 L 19.53125 15.875 L 17.71875 16.59375 C 17.148438 15.652344 16.351563 14.851563 15.40625 14.28125 L 16.125 12.46875 L 14.28125 11.71875 L 13.5625 13.53125 C 13.03125 13.398438 12.476563 13.3125 11.90625 13.3125 C 11.335938 13.3125 10.78125 13.398438 10.25 13.53125 Z M 11.90625 15.3125 C 14.570313 15.3125 16.6875 17.429688 16.6875 20.09375 C 16.6875 22.757813 14.570313 24.90625 11.90625 24.90625 C 9.242188 24.90625 7.09375 22.757813 7.09375 20.09375 C 7.09375 17.429688 9.242188 15.3125 11.90625 15.3125 Z "%3E%3C/path%3E%3C/g%3E%3C/svg%3E\')}\ .GME_route { background-size: 18px 18px; background-position: center;background-image: url(\'data:image/svg+xml,%3Csvg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"%3E%3Cpath d="M12 0c-4.198 0-8 3.403-8 7.602 0 4.198 3.469 9.21 8 16.398 4.531-7.188 8-12.2 8-16.398 0-4.199-3.801-7.602-8-7.602zm0 11c-1.657 0-3-1.343-3-3s1.343-3 3-3 3 1.343 3 3-1.343 3-3 3z"/%3E%3C/svg%3E\');background-repeat: no-repeat; background-color: #eee;}\ .GME_hide { background-size: 18px 18px; background-position: center;background-image: url(\'data:image/svg+xml,%3Csvg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"%3E%3Cpath d="M19.604 2.562l-3.346 3.137c-1.27-.428-2.686-.699-4.243-.699-7.569 0-12.015 6.551-12.015 6.551s1.928 2.951 5.146 5.138l-2.911 2.909 1.414 1.414 17.37-17.035-1.415-1.415zm-6.016 5.779c-3.288-1.453-6.681 1.908-5.265 5.206l-1.726 1.707c-1.814-1.16-3.225-2.65-4.06-3.66 1.493-1.648 4.817-4.594 9.478-4.594.927 0 1.796.119 2.61.315l-1.037 1.026zm-2.883 7.431l5.09-4.993c1.017 3.111-2.003 6.067-5.09 4.993zm13.295-4.221s-4.252 7.449-11.985 7.449c-1.379 0-2.662-.291-3.851-.737l1.614-1.583c.715.193 1.458.32 2.237.32 4.791 0 8.104-3.527 9.504-5.364-.729-.822-1.956-1.99-3.587-2.952l1.489-1.46c2.982 1.9 4.579 4.327 4.579 4.327z"/%3E%3C/svg%3E\');background-repeat: no-repeat; background-color: #eee;}\ .GME_route.gme-button-active {background-image: url(\'data:image/svg+xml,%3Csvg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"%3E%3Cpath d="M11.6,11C10.1,10.8,9,9.5,9,8c0-1.7,1.3-3,3-3c1.6,0,2.9,1.2,3,2.8L19,4c-1.4-2.4-4.1-4-7-4C7.8,0,4,3.4,4,7.6c0,2.3,1.1,4.9,2.8,7.9L11.6,11z"/%3E%3Cpath d="M7.8,17.3c1.2,2,2.6,4.2,4.2,6.7c4.5-7.2,8-12.2,8-16.4c0-0.5-0.1-1-0.2-1.5L7.8,17.3z"/%3E%3Cline stroke-linecap="round" stroke="black" x1="2.1" y1="21.3" x2="21.9" y2="2.7"/%3E%3C/svg%3E\'); }\ .GME_hide.gme-button-active { background-image: url(\'data:image/svg+xml,%3Csvg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"%3E%3Cpath d="M12.015 7c4.751 0 8.063 3.012 9.504 4.636-1.401 1.837-4.713 5.364-9.504 5.364-4.42 0-7.93-3.536-9.478-5.407 1.493-1.647 4.817-4.593 9.478-4.593zm0-2c-7.569 0-12.015 6.551-12.015 6.551s4.835 7.449 12.015 7.449c7.733 0 11.985-7.449 11.985-7.449s-4.291-6.551-11.985-6.551zm-.015 3c-2.209 0-4 1.792-4 4 0 2.209 1.791 4 4 4s4-1.791 4-4c0-2.208-1.791-4-4-4z"/%3E%3C/svg%3E\'); }\ .gme-button-refresh-labels { background-position: -320px 4px;}\ .gme-button-clear-labels { background-position: -69px 4px;}\ span.gme-distance-container { display: none; }\ span.gme-distance-container.show { display: inline-block; }\ .GME_info { background-size: 16px 16px; background-position: center;background-image: url(\'data:image/svg+xml,%3C%3Fxml version="1.0" encoding="iso-8859-1"%3F%3E%3Csvg version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px" viewBox="0 0 32 32" %3E%3Cpath style="fill:%23030104;" d="M10,16c1.105,0,2,0.895,2,2v8c0,1.105-0.895,2-2,2H8v4h16v-4h-1.992c-1.102,0-2-0.895-2-2L20,12H8 v4H10z"/%3E%3Ccircle style="fill:%23030104;" cx="16" cy="4" r="4"/%3E%3C/svg%3E\')}\ .GME_info.gme-button-active {}\ #GME_loc, a.gme-button.leaflet-active {outline: none;}\ .leaflet-control-zoomwarning { top: 94px; }\ .leaflet-control-zoomwarning a { filter: progid:DXImageTransform.Microsoft.gradient(startColorStr="#BFC80000",EndColorStr="#BFC80000"); background-color:rgba(200,0,0,0.75); margin-left: -4px; background-position: -502px 2px;height:14px;width:14px; border-color: #b00; box-shadow: 0 0 8px rgba(0, 0, 0, 0.4); }\ .leaflet-control-zoomwarning a:hover { background-color:rgba(230,0,0,0.75); }\ .gme-event { cursor: pointer; }\ .gme-modalDialog {position: fixed; top: 0; right: 0; bottom: 0; left: 0; background: rgba(0,0,0,0.5); z-index: 1000; opacity:.5; -webkit-transition: opacity 400ms ease-in; -moz-transition: opacity 400ms ease-in; transition: opacity 400ms ease-in; pointer-events: none; display:none; }\ .gme-modalDialog:target, .gme-modalDialog.gme-targetted { opacity:1; display:block; pointer-events: auto; }\ .gme-modalDialog > div { position: relative; margin: 4% 12.5%; height: 30em; max-height: 75%; padding: 0 0 13px 0; border: 1px solid #000; border-radius: 10px; background: #fff; background: -moz-linear-gradient(#fff, #999); background: -webkit-linear-gradient(#fff, #999); background: -o-linear-gradient(#fff, #999); }\ .gme-modalDialog header { color: #eee; background: none #454545; font-size: 15px; text-align: center; border-top-left-radius: 10px; padding: 0.5em 0; font-weight: bold; text-shadow: none; height: auto; min-height: auto; min-width: auto; }\ .gme-modal-content { position: absolute; top: 3.5em; left: 0.75em; right: 0.75em; bottom: 0.5em; overflow: auto; }\ .gme-modal-content > .leaflet-control-gme { position: absolute; left: 0.5em; bottom: 0.5em; top: auto; }\ .gme-close-dialog { background: #606061; color: #fff; line-height: 25px; position: absolute; right: -12px; text-align: center; top: -10px; width: 24px; text-decoration: none; font-weight: bold; -webkit-border-radius: 12px; -moz-border-radius: 12px; border-radius: 12px; -moz-box-shadow: 1px 1px 3px #000; -webkit-box-shadow: 1px 1px 3px #000; box-shadow: 1px 1px 3px #000; }\ .gme-close-dialog:hover { background: #00d9ff; }\ #searchtabs li a { padding: 1em 0.5em; }\ @media print { #search { display: none !important}}\ .tab-switcher { position: relative; font-family: Arial, sans-serif; font-size: 14px; }\ .gme-tab { float: left; }\ .gme-tab-label { border-radius: 8px 8px 0 0; border: 1px solid #ccc; color: #454545; background: #ddd; display: block; position: relative; margin-left: 15px; padding: 3px 0; font-weight: bold; z-index: 0; }\ .gme-tab-label:after { border-bottom: 1px solid #ccc; border-bottom-left-radius: 8px; border-left: 1px solid #ccc; box-shadow: -2px 2px 0 #ddd; bottom: -8px; content: ""; display: inline-block; height: 8px; left: 9px; position: relative; width: 8px; z-index: 3;}\ .gme-tab-label:before { border-bottom: 1px solid #ccc; border-bottom-right-radius: 8px; border-right: 1px solid #ccc; box-shadow: 2px 2px 0 #ddd; bottom: -8px; content: ""; display: inline-block; height: 8px; left: -9px; position: relative; width: 8px; z-index: 3; }\ .gme-tab-label:hover { cursor: pointer; }\ .gme-tab-content { position: absolute; top: 25px; bottom: 3.5em; left: 0; right: 0; padding: 0.5em; background: #000; border: 1px solid #ccc; border-radius: 8px; color: #555; z-index: 1; opacity: 0; overflow: auto; }\ .gme-tab-content ul { margin: 0.5em 0; }\ .gme-tab input[type=radio] { display: none; }\ .gme-tab input[type=radio]:checked ~ .gme-tab-content { z-index: 2; opacity: 1; background: #fff; color: #454545; }\ .gme-tab input[type=radio]:checked ~ .gme-tab-label { background: #fff; color: #454545; border-bottom: 1px solid #fff; z-index: 3; }\ .gme-tab input[type=radio]:checked ~ .gme-tab-label:after { box-shadow: -2px 2px 0 #fff; }\ .gme-tab input[type=radio]:checked ~ .gme-tab-label:before { box-shadow: 2px 2px 0 #fff; }\ .gme-fieldgroup { position: relative; border: 1px solid #ccc; border-radius: 6px; background: #eee; margin: 0.5em 0 1.5em; padding: 0.5em; }\ .gme-fieldgroup h3 { position:absolute; top: -0.7em; left: 0.5em; padding: 0 0.5em; background: #eee; border-top: 1px solid #ccc; border-radius: 6px; z-index:1; display:inline-block; font-weight: bold; font-size: 12px; }\ .gme-fieldgroup ul { margin: 0.5em 0; padding: 0; }\ .gme-fieldgroup li { display: inline-block; margin: 0 -1px -1px 0; background: #ddd; border: 1px solid #ccc; border-radius: 6px; padding: 0 0.5em; }\ .gme-xhair { cursor: crosshair; }\ .map-button-container { margin-right: 5em; }\ .leaflet-control-layers-toggle { background-size: 75%; background-image: url(\'data:image/svg+xml,%3Csvg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" viewBox="0 0 26 26"%3E%3Cdefs%3E%3Cstyle%3E.diamond%7Bopacity:0.8;fill:url(%23radialGradient);%7D.diamnondPath%7Bfill:url(%23linearGradient);%7D%3C/style%3E%3CradialGradient id="radialGradient" cx="170.89" cy="960.99" r="10.16" gradientTransform="translate(-161.12 -1073.84) scale(1.02 1.13)" gradientUnits="userSpaceOnUse"%3E%3Cstop offset="0.16" stop-color="gray"/%3E%3Cstop offset="1" stop-color="%23fff"/%3E%3C/radialGradient%3E%3ClinearGradient id="linearGradient" x1="13" x2="13" y2="15" gradientUnits="userSpaceOnUse"%3E%3Cstop offset="0" stop-color="%23b3b3b3"/%3E%3Cstop offset="1" stop-color="%234d4d4d"/%3E%3C/linearGradient%3E%3Csymbol id="layer" data-name="Nový symbol…" viewBox="0 0 26 15"%3E%3Cpolygon class="diamond" points="13.52 0 0 6.92 13.52 15 26 6.92 13.52 0"/%3E%3Cpath class="diamnondPath" d="M13.5,1.13,24.06,7,13.5,13.82,2.06,7,13.5,1.13m0-1.13L0,6.92,13.52,15,26,6.92,13.52,0Z"/%3E%3C/symbol%3E%3C/defs%3E%3Cuse width="26" height="15" transform="translate(0 11)" xlink:href="%23layer"/%3E%3Cuse width="26" height="15" transform="translate(0 5.57) scale(1 0.98)" xlink:href="%23layer"/%3E%3Cuse width="26" height="15" transform="translate(0) scale(1 0.98)" xlink:href="%23layer"/%3E%3C/svg%3E\'); }\ #GME_apikeys_field label { display: block; }\ #GME_apikeys_field label input { width: 75%; }', drag: '#cacheDetails .cacheImage { border: solid 1px #ccc; border-radius: 7px; padding-left: 5px; }\ .moveable { cursor: move; box-shadow: 0 1px 4px rgba(102, 51, 255, 0.3); }' }, env: { dragdrop: (document.createElement('span').draggable !== undefined), geolocation: !!navigator.geolocation, init: [], loggedin: (!!document.getElementById("ctl00_uxLoginStatus_divSignedIn") || !!document.getElementById("uxLoginStatus_divSignedIn")), page: "default", storage: false, xhr: (typeof GM_xmlhttpRequest === 'function') ? 'GM' : ((typeof GM === 'object' && typeof GM.xmlHttpRequest === 'function') ? 'GM4' : '') }, html: { config: '<section class="gme-tab">\ <input type="radio" name="gme-tab-row" id="gme-tab-maps" checked />\ <label class="gme-tab-label" for="gme-tab-maps">Map display</label>\ <div class="gme-tab-content">\ <div class="gme-fieldgroup">\ <h3>Maps to show in selector widget</h3>\ <ul id="GME_mapfields"></ul>\ <label>Default map source: <select name="GME_map_default" id="GME_map_default"></select></label>\ <label>Default hill shading source: <select name="GME_hill_shade_default" id="GME_hill_shade_default"></select></label>\ </div>\ <div class="gme-fieldgroup">\ <h3>Map API key</h3>\ <div id="GME_apikeys_field"></dov>\ </div>\ </div>\ </section>\ <section class="gme-tab">\ <input type="radio" name="gme-tab-row" id="gme-tab-manage" />\ <label class="gme-tab-label" for="gme-tab-manage">Manage maps</label>\ <div class="gme-tab-content">\ <div class="gme-fieldgroup">\ <h3>Add map sources</h3>\ <label>Mapsource: <input type="text" name="GME_map_custom" id="GME_map_custom"> </label>\ <div class="leaflet-control-gme"><button type="button" id="GME_custom_add" class="gme-button gme-button-wide gme-button-l gme-button-r" title="Add custom map source">Add</button> <a href="#GME_format" title="Map source format info" class="gme-button gme-button-wide gme-button-l">Mapsource format info</a><button type="button" id="GME_custom_export" title="Export custom map source JSON" class="gme-button gme-button-wide gme-button-r">Export custom maps</button></div>\ </div>\ <div class="gme-fieldgroup">\ <h3>Remove map sources</h3>\ <ul id="GME_mapfields_del"></ul>\ </div>\ </div>\ </section>\ <section class="gme-tab">\ <input type="radio" name="gme-tab-row" id="gme-tab-other" />\ <label class="gme-tab-label" for="gme-tab-other">Other</label>\ <div class="gme-tab-content">\ <div class="gme-fieldgroup">\ <h3>Miscellaneous settings</h3>\ <ul>\ <li><label title="Only list unfound caches in search"><input type="checkbox" name="GME_filterFinds" id="GME_filterFinds" /> Filter finds</label></li>\ <li><label><input type="checkbox" checked="checked" name="GME_osgbSearch" id="GME_osgbSearch" /> Enhance search</label></li>\ <li><label title="Location widget constantly updates position"><input type="checkbox" name="GME_follow" id="GME_follow" /> FollowMe Mode</label></li>\ </ul>\ <label>Labels:\ <select name="GME_labelStyle" id="GME_labelStyle">\ <option value="names">Names</option>\ <option value="codes" selected="selected">Codes</option>\ </select>\ </label>\ <label>Scale:\ <select name="GME_measure" id="GME_measure">\ <option value="metric" selected="selected">Metric</option>\ <option value="imperial">Imperial</option>\ </select>\ </label>\ <label>Map brightness:\ <input type="range" name="GME_brightness" id="GME_brightness" value="100" min="0" max="100" />\ </label>\ </div>\ </div>\ </section>\ <section class="gme-tab">\ <input type="radio" name="gme-tab-row" id="gme-tab-about" />\ <label class="gme-tab-label" for="gme-tab-about">About</label>\ <div class="gme-tab-content">\ <div class="gme-fieldgroup">\ <h3>Geocaching Map Enhancements</h3><br />\ <p>v<span id="GME_version"></span> © 2011-2018 James Inge. Geocaching Map Enhancements is licensed for reuse under the <a target="_blank" rel="noopener noreferrer" href="http://www.opensource.org/licenses/mit-license.php">MIT License</a>. For documentation, see <a target="_blank" rel="noopener noreferrer" href="http://geo.inge.org.uk/gme.htm">http://geo.inge.org.uk/gme.htm</a></p>\ <p>Elevation and reverse geocoding data provided by <a target="_blank" rel="noopener noreferrer" href="http://www.geonames.org/">GeoNames</a> and used under a <a target="_blank" rel="noopener noreferrer" href="https://creativecommons.org/licenses/by/3.0/">Creative Commons Attribution 3.0</a> (CC-BY) License.</p>\ <p>Grid reference manipulation is adapted from code © 2005-2014 Chris Veness (<a target="_blank" rel="noopener noreferrer" href="http://www.movable-type.co.uk/scripts/latlong-gridref.html">www.movable-type.co.uk/scripts/latlong-gridref.html</a>, used under a <a target="_blank" rel="noopener noreferrer" href="https://creativecommons.org/licenses/by/3.0/">Creative Commons Attribution 3.0</a> (CC-BY) License.</p>\ <p>Photos provided by Geograph are copyright their respective owners - hover mouse over thumbnails or click through for attribution details. They may be re-used under a <a target="_blank" rel="noopener noreferrer" href="https://creativecommons.org/licenses/by-sa/2.0/">Creative Commons Attribution-ShareAlike 2.0</a> (CC-BY-SA) License.</p>\ </div>\ </div>\ </section>\ <div class="leaflet-control-gme">\ <a href="#" class="gme-button gme-button-wide gme-button-l" rel="back" title="Cancel">Cancel</a><button type="button" class="gme-button gme-button-wide" id="GME_default" title="Reset to defaults">Defaults</button><button type="button" class="gme-button gme-button-wide gme-button-r" id="GME_set" title="Confirm settings">Save</button>\ </div>', customInfo: '<p>Custom mapsources can be added by supplying entering a <a rel="external" href="http://www.json.org/">JSON</a> configuration string that tells GME what to call the map, where to find it, and how it is set up. e.g.</p>\ <p><code>{"alt":"OS NPE (GB only)","tileUrl":"https://ooc.openstreetmap.org/npe/{z}/{x}/{y}.png", "minZoom":6, "maxZoom": 15, "attribution": "OpenStreetMap NPE" }</code></p>\ <p>The <code>"alt"</code> and <code>"tileUrl"</code> parameters are mandatory. <code>"tileUrl"</code> can contain {x}, {y} and {z} for Google-style coordinate systems (also works with TMS systems like Eniro, but needs the <code>"scheme":"tms"</code> parameter), or {q} for Bing-style quadkeys. GME can also connect with WMS servers, in which case a <code>"layers"</code> parameter is required.</p>\ <p>The other parameters are the same as those used by the <a rel="external" href="http://leafletjs.com/reference-versions.html">Leaflet API</a>, with the addition of a <code>"overlay":true</code> option, that makes the mapsource appear as a selectable overlay.</p>\ <ul><li><a rel="external" href="http://geo.inge.org.uk/gme_config.htm">Detailed documentation</a></li><li><a rel="external" href="http://geo.inge.org.uk/gme_maps.htm">More mapsource examples</a></li></ul>', search: '<input type="text" placeholder="Address, coordinates, GC-code, keyword, etc." id="SearchBox_Text" title="Jump to a specific zoom level by typing zoom then a number. Zoom 1 shows the whole world, maxiumum zoom is normally 18-22. To search using a British National Grid reference, just type it in the search box and hit the button! You can use 2, 4, 6, 8 or 10-digit grid refs with the 2-letter prefix but no spaces in the number (e.g. SU12344225) or absolute grid refs with a comma but no prefix (e.g. 439668,1175316)." />\ <button id="SearchBox_OS" title="Search">Search</button>\ <div class="GME_search_results hidden">\ <h3 class="GME_search_heading">GeoNames search results</h3>\ <ul class="GME_search_list"></ul>\ <p>Or try the <a href="#" class="GME_link_GSSearch">Geocaching.com search</a>\ </div>' }, script: { common: function () { var that = this, callbackCount = 0, load_count = 0, JSONP; function setEnv() { // The script waits for the Leaflet API to load, and will abort if it does not find it after a minute. var maxTries = 60, wait = 1000; switch (gmeConfig.env.page) { case "seek": if (typeof $ === "function") { gmeInit(gmeConfig.env.init); load(); return; } break; case "hide": maxTries = Infinity; wait = 3000; if (window.map !== null && window.map !== undefined && typeof L === "object" && typeof $ === "function") { gmeInit(gmeConfig.env.init); window.setTimeout(load, 500); return; } break; case "type": if (window.map !== null && window.map !== undefined && typeof L === "object" && typeof $ === "function") { gmeInit(gmeConfig.env.init); load(); reload(); $(".cache-type-selector button").click(reload); return; } break; case "maps": // Wait for the map to load and the default map selector to be added. if (typeof L === "object" && typeof $ === "function" && window.MapSettings && window.MapSettings.Map && window.MapSettings.Map._loaded && $(".leaflet-control-layers").length > 0) { gmeInit(gmeConfig.env.init); window.setTimeout(load, 500); return; } break; default: if (typeof L === "object" && typeof $ === "function") { gmeInit(gmeConfig.env.init); window.setTimeout(load, 500); return; } break; } if (load_count < maxTries) { window.setTimeout(setEnv, wait); load_count++; console.log("GME: Waiting for map API to load: " + load_count + "..."); } } function gmeInit(scriptArray) { // Init routines that need either JQuery or Leaflet API, so must be run from load() rather than on script insertion. var initScripts = { "config": function () { if (gmeConfig.env.storage) { setConfig(); $("#GME_set").bind("click", storeSettings); $("#GME_default").bind("click", setDefault); $("#GME_custom_add").bind("click", addCustom); $("#GME_custom_export").bind("click", exportCustom); $("li.li-user ul").append("<li class='li-settings'><a class='icon-settings' id='gme-config-link' href='#GME_config' title='Configure Geocaching Map Enhancements extension'>Geocaching Map Enhancements</a></li>"); } }, "drop": function () { $.fn.filterNode = function (name) { return this.find("*").filter(function () { return this.nodeName === name; }); }; L.GME_dropHandler = L.Control.extend(dropHandlerObj); }, "map": function () { bounds_GB = new L.LatLngBounds(new L.LatLng(49, -9.5), new L.LatLng(62, 2.3)); bounds_IE = new L.LatLngBounds(new L.LatLng(51.2, -12.2), new L.LatLng(55.73, -5.366)); bounds_NI = new L.LatLngBounds(new L.LatLng(54, -8.25), new L.LatLng(55.73, -5.25)); bounds_CI = new L.LatLngBounds(new L.LatLng(49.1, -2.8), new L.LatLng(49.8, -1.8)); bounds_DE = new L.LatLngBounds(new L.LatLng(47.24941, 5.95459), new L.LatLng(55.14121, 14.89746)); L.GME_DistLine = L.Polyline.extend(polylineObj); L.GME_QuadkeyLayer = L.TileLayer.extend(quadkeyLayerObj); L.GME_complexLayer = L.TileLayer.extend(complexLayerObj); L.GME_genericLayer = genericLayerFn; }, "widget": function () { L.GME_Widget = L.Control.extend(widgetControlObj); if (window.Groundspeak && Groundspeak.Map && Groundspeak.Map.Control && Groundspeak.Map.Control.FindMyLocation) { L.GME_FollowMyLocationControl = Groundspeak.Map.Control.FindMyLocation.extend(locationControlObj); } L.GME_ZoomWarning = L.Control.extend(zoomWarningObj); if (L.LatLng.prototype.toUrl === undefined) { L.LatLng.prototype.toUrl = function () { return this.lat.toFixed(6) + "," + this.lng.toFixed(6); }; } if ($.fancybox === undefined) { console.info("GME: Fetching Fancybox"); $("head").append("<link rel='stylesheet' type='text/css' href='https://cdnjs.cloudflare.com/ajax/libs/fancybox/2.1.5/jquery.fancybox.min.css'><script type='text/javascript' src='https://cdnjs.cloudflare.com/ajax/libs/fancybox/2.1.5/jquery.fancybox.min.js'></script>"); } } }, j; for (j = 0; j < scriptArray.length; j++) { if (initScripts.hasOwnProperty(scriptArray[j]) && typeof initScripts[scriptArray[j]] === "function") { initScripts[scriptArray[j]](); } } console.log("GME init: " + scriptArray.join()); } function b64encode(str) { if (typeof window.btoa === "function") { return btoa(encodeURIComponent(str)); } else { return encodeURIComponent(str); } } function b64decode(str) { if (typeof window.atob === "function") { return decodeURIComponent(window.atob(str)); } else { return decodeURIComponent(str); } } function DMM(ll) { var latDeg = ll.lat < 0 ? Math.ceil(ll.lat) : Math.floor(ll.lat), lngDeg = ll.lng < 0 ? Math.ceil(ll.lng) : Math.floor(ll.lng); return (ll.lat < 0 ? "S" : "N") + Math.abs(latDeg) + " " + (60 * Math.abs((ll.lat - latDeg))).toFixed(3) + (ll.lng < 0 ? " W" : " E") + Math.abs(lngDeg) + " " + (60 * Math.abs((ll.lng - lngDeg))).toFixed(3); } function formatDistance(dist) { var formatted = 0; if (that.parameters.measure === "metric") { if (dist > 10000) { formatted = Math.round(dist / 1000) + " km"; } else { if (dist > 1000) { formatted = (dist / 1000).toFixed(1) + " km"; } else { formatted = Math.round(dist) + " m"; } } } else { if (dist > 16093.44) { formatted = Math.round(dist / 1609.344) + " mi"; } else { if (dist > 1609.344) { formatted = (dist / 1609.344).toFixed(1) + " mi"; } else { formatted = Math.round(dist * 3.2808) + " ft"; } } } return formatted; } function htmlEntities(text) { return text .replace(/&/g, "&") .replace(/\"/g, """) .replace(/'/g, "'") .replace(/</g, "<") .replace(/>/g, ">"); } function validCoords(c1, c2) { var lat, lng; if (c1 === undefined) { return false; } if (c1.hasOwnProperty("lat") && c1.hasOwnProperty("lng")) { lat = c1.lat; lng = c1.lng; } else { if (c2 !== undefined) { lat = c1; lng = c2; } } if (lat !== null && lng !== null && !isNaN(+lat) && !isNaN(+lng) && lat >= -90 && lat <= 90) { return true; } return false; } function parseCoords(text) { var lat = 0, lng = 0, num = 0, c = text.replace(/[^\-SsWw0-9\.\s]/g, " ").trim().match(/^([S\-])?\s*(\d{1,2}(\.\d*){0,1}|\.\d*)(\s+(\d{0,2}(\.\d*){0,1})){0,1}(\s+(\d{0,2}(\.\d*){0,1})){0,1}\s*([S\-])?\s+([W\-])?\s*(\d{1,3}(\.\d*){0,1}|\.\d*)(\s+(\d{0,2}(\.\d*){0,1})){0,1}(\s+(\d{0,2}(\.\d*){0,1})){0,1}\s*([W\-])?$/i); if (c) { num = (c[2] ? 1 : 0) + (c[5] ? 1 : 0) + (c[8] ? 1 : 0) + (c[12] ? 1 : 0) + (c[15] ? 1 : 0) + (c[18] ? 1 : 0); switch (num) { case 6: break; case 4: if (c[15] === undefined) { c[15] = c[12]; c[12] = c[8]; c[8] = undefined; } break; case 2: if (c[12] === undefined && c[5]) { c[12] = c[5]; c[5] = undefined; } break; default: alert("Couldnt understand coordinates"); return false; } if (c[2] !== undefined) { lat = +c[2]; } if (c[5] !== undefined) { lat += c[5] / 60; } if (c[8] !== undefined) { lat += c[8] / 3600; } if (c[1] !== undefined || c[10] !== undefined) { lat *= -1; } if (c[12] !== undefined) { lng = +c[12]; } if (c[15] !== undefined) { lng += c[15] / 60; } if (c[18] !== undefined) { lng += c[18] / 3600; } if (c[11] !== undefined || c[20] !== undefined) { lng *= -1; } } if (validCoords(lat, lng)) { return { lat: lat, lng: lng }; } alert("Invalid coordinates"); return false; } function getHomeCoords() { var c, h = document.getElementById("ctl00_ContentBody_lnkPrintDirectionsSimple"); if (window.MapSettings && MapSettings.User && validCoords(MapSettings.User.Home)) { return new L.LatLng(MapSettings.User.Home.lat, MapSettings.User.Home.lng); } if (validCoords(window.homeLat, window.homeLon)) { return new L.LatLng(window.homeLat, window.homeLon); } if (h && h.href) { c = h.href.match(/(?:saddr=)(-?\d{1,2}\.\d*),(-?\d{1,3}\.\d*)/); if (c !== null && c.length === 3 && validCoords(c[1], c[2])) { return new L.LatLng(c[1], c[2]); } } return false; } function validURL(url) { return (/^(http|https|ftp)\:\/\/([a-zA-Z0-9\.\-]+(\:[a-zA-Z0-9\.&%\$\-]+)*@)*((25[0-5]|2[0-4][0-9]|[0-1]{1}[0-9]{2}|[1-9]{1}[0-9]{1}|[1-9])\.(25[0-5]|2[0-4][0-9]|[0-1]{1}[0-9]{2}|[1-9]{1}[0-9]{1}|[1-9]|0)\.(25[0-5]|2[0-4][0-9]|[0-1]{1}[0-9]{2}|[1-9]{1}[0-9]{1}|[1-9]|0)\.(25[0-5]|2[0-4][0-9]|[0-1]{1}[0-9]{2}|[1-9]{1}[0-9]{1}|[0-9])|localhost|([a-zA-Z0-9\-]+\.)*[a-zA-Z0-9\-]+\.(com|edu|gov|int|mil|net|org|biz|arpa|info|name|pro|aero|coop|museum|[a-zA-Z]{2}))(\:[0-9]+)*(\/($|[a-zA-Z0-9\.\,\?'\\\+&%\$#\=~_\-]+))*$/).test(url); } if (window.console === undefined) { var logFn = function (text) { }; window.console = { error: logFn, log: logFn, info: logFn, warn: logFn }; } if (gmeConfig.env.xhr) { JSONP = function (url, id) { console.log("GME: Using GM_xhr to fetch " + url); var s = document.getElementById("gme_jsonp_node"); if (!s) { s = document.createElement("script"); s.id = "gme_jsonp_node"; document.documentElement.firstChild.appendChild(s); } s.type = "text/x-gme-jsonp"; s.text = url; s.setAttribute("data-gme-callback", id); document.dispatchEvent(new Event("GME_XHR_event")); }; document.addEventListener("GME_XHR_callback", function (e) { var s = document.getElementById("gme_jsonp_node"), callback = s.getAttribute("data-gme-callback"); if (typeof window[callback] === "function") { try { window[callback](JSON.parse(s.text)); } catch (e) { console.error("Error processing JSON callback " + callback + ": " + e); } } else { console.warn("Unexpected request to JSON callback handler: Couldn't find callback function " + callback); } return false; }); } else { JSONP = function (url, id) { console.log("GME: Using JSONP to fetch " + url); if (validURL(url)) { var s = document.createElement("script"); s.type = "text/javascript"; if (id) { s.id = id; } s.src = url; document.documentElement.firstChild.appendChild(s); } }; } gmeConfig.env.home = getHomeCoords(); this.parameters = gmeConfig.parameters; this.getVersion = function () { return gmeConfig.parameters.version; }; this.getGeograph = function (coords) { var callprefix = "GME_geograph_callback", call, host = ""; function searchLink(coords) { // URIs for website search pages. if (coords === undefined) { return false; } var host = ""; if (bounds_GB.contains(coords) || bounds_IE.contains(coords)) { host = "https://geograph.org.uk/"; } if (bounds_CI.contains(coords)) { host = "https://www.geograph.org.gg/"; } if (bounds_DE.contains(coords)) { host = "https://geo-en.hlipp.de/"; } return host ? [host, "search.php?location=", coords.toUrl()].join("") : false; } function makeCallback(callname) { callbackCount++; return function (json) { var html, i, p; if (json.items && json.items.length > 0) { html = ["<h3>Geograph images near ", DMM(coords), "</h3><p>"].join(""); for (i = json.items.length - 1; i >= 0; i--) { p = json.items[i]; html += ["<a target='_blank' rel='noopener noreferrer' href='", encodeURI(p.link), "' style='margin-right:0.5em;' title='", htmlEntities(p.title) + " by " + htmlEntities(p.author), "'>", p.thumbTag, "</a>"].join(""); } html += ["</p><p><a target='_blank' rel='noopener noreferrer' href='", searchLink(coords), "'>Search for more photos nearby on Geograph</a></p><p style='font-size:90%;'>Geograph photos are copyrighted by their owners and available under a <a href='https://creativecommons.org/licenses/by-sa/2.0/'>Creative Commons licence</a>. Hover mouse over thumbnails for more details, or click through for full images.</p>"].join(""); $.fancybox(html); } else { $.fancybox(["<p>No photos found nearby. <a target='_blank' rel='noopener noreferrer' href='", searchLink(coords), "'>Search on Geograph</a></p>"].join("")); } $("#" + callname).remove(); if (window[callname] !== undefined) { delete window[callname]; } }; } if (validCoords(coords) && this.isGeographAvailable(coords)) { if (!bounds_CI.contains(coords) && (bounds_GB.contains(coords) || bounds_IE.contains(coords))) { host = "https://api.geograph.org.uk/"; call = callprefix + callbackCount; window[call] = makeCallback(call); JSONP(host + "syndicator.php?key=geo.inge.org.uk&location=" + coords.toUrl() + "&format=JSON&callback=" + call, call); } else { window.open(searchLink(coords), "_blank"); } } else { console.error("GME: Bad coordinates to getGeograph"); } }; this.getHeight = function (coords) { var callprefix = "GME_height_callback", call; function makeCallback(callname) { callbackCount++; return function (json) { if (typeof json.astergdem === "number" && typeof json.lat === "number" && typeof json.lng === "number") { var h, m; if (json.astergdem === -9999) { m = "<p><strong>Spot Height</strong><br/>(Ocean)</p>"; } else { h = that.parameters.measure === "metric" ? json.astergdem + " m" : Math.round(json.astergdem * 3.2808) + " ft"; m = ["<p><strong>Spot Height</strong><br/>Approx ", h, " above sea level</p>"].join(""); } $.fancybox(m); } $("#" + callname).remove(); if (window[callname] !== undefined) { delete window[callname]; } }; } if (validCoords(coords)) { call = callprefix + callbackCount; window[call] = makeCallback(call); JSONP(["http://api.geonames.org/astergdemJSON?lat=", coords.lat, "&lng=", coords.lng, "&username=gme&callback=", call].join(""), call); } else { console.error("GME: Bad coordinates to getHeight"); } }; this.isGeographAvailable = function (coords) { return bounds_GB.contains(coords) || bounds_DE.contains(coords) || bounds_IE.contains(coords) || bounds_CI.contains(coords); }; this.isInUK = function (coords) { if (bounds_GB.contains(coords)) { if (bounds_IE.contains(coords)) { if (bounds_NI.contains(coords)) { return true; } return false; } return true; } return false; }; if (gmeConfig.env.geolocation) { this.seekHere = function () { function hereCallback(pos) { that.seekByLatLng({ lat: pos.coords.latitude, lng: pos.coords.longitude }); $("#GME_hereSub").val("Go"); } function hereError(err) { if (err.code === 2) { alert("Current location not available"); } if (err.code === 3) { alert("Timed out finding current location"); } $("#GME_hereSub").val("Go"); } $("#GME_hereSub").val("Waiting for location..."); navigator.geolocation.getCurrentPosition(hereCallback, hereError, { timeout: 60000, maximumAge: 30000 }); return false; }; } this.seekByLatLng = function (latlng) { if (validCoords(latlng)) { var url = ["https://www.geocaching.com/seek/nearest.aspx?origin_lat=", latlng.lat, "&origin_long=", latlng.lng, that.parameters.filterFinds ? "&f=1" : ""].join(""); window.open(url, "_blank"); } else { console.error("GME: Invalid coordinates for search"); } }; }, config: function () { function addSources(json) { function setSrc(src) { if (src.alt && src.tileUrl) { var m = that.parameters.maps.concat(src); that.parameters.maps = m; return 1; } alert("Map source must include at least \"alt\" and \"tileUrl\" parameters"); return 0; } var i, updated = 0; if (json.length === undefined) { updated += setSrc(json); } else { for (i = 0; i < json.length; i++) { updated += setSrc(json[i]); } } if (updated > 0) { setConfig(); $("#gme-tab-maps")[0].checked = true; } } function addCustom() { try { var n = JSON.parse(document.getElementById("GME_map_custom").value); addSources(n); } catch (e) { alert("Map source string must be valid JSON."); return; } } function exportCustom() { $.fancybox($("<p/>").text(JSON.stringify(that.parameters.maps)).html()); } function setDefault() { if (localStorage.GME_custom) { delete localStorage.GME_custom; } if (localStorage.GME_parameters) { delete localStorage.GME_parameters; } if (localStorage.GME_cache) { delete localStorage.GME_cache; } refresh(); } function refresh(config) { var dest = "https://www.geocaching.com/map/#", mapLink = document.getElementById("map_linkto"), uri; if (config) { dest += "GME_config"; } if (mapLink) { uri = mapLink.value; if (uri) { dest += uri.replace(/^http:\/\/coord.info\/map/, ""); } document.location.href = dest; } else { document.location.hash = ""; } window.location.reload(false); return false; } function setConfig() { var i, mapfields = "", shadingSelect = "<option value=''>none</option>", mapfields_del = "", mapselect = "", alt = "", overlay, sel, overSel, allMaps = that.parameters.maps, allApiKeys = that.parameters.apiKeys; for (i = 0; i < allMaps.length; i++) { alt = allMaps[i].alt; overlay = allMaps[i].overlay; if (!overlay) { mapselect += "<option value='" + htmlEntities(alt) + "'>" + htmlEntities(alt) + "</option>"; } else { shadingSelect += "<option value='" + htmlEntities(alt) + "'>" + htmlEntities(alt) + "</option>"; } mapfields += "<li><label><input type='checkbox' " + (allMaps[i].ignore ? "" : "checked='checked' ") + "name='" + htmlEntities(alt) + "' id='checkbox-" + i + "' /> " + htmlEntities(alt) + (overlay ? " (Overlay)" : "") + "</label></li>"; } // API keys const mapsWithApiKey = allMaps.filter((map) => map.apiKey); mapsWithApiKey.forEach((map) => { const existingApiKey = allApiKeys && allApiKeys[map.apiKey] ? allApiKeys[map.apiKey] : ""; const label = document.createElement("label"); label.innerHTML = `${map.apiKey} `; const input = document.createElement("input"); input.type = "text"; input.name = map.apiKey; input.value = existingApiKey; label.appendChild(input); $("#GME_apikeys_field").append(label); }); if (allMaps.length > 0) { for (i = 0; i < allMaps.length; i++) { alt = allMaps[i].alt; mapfields_del += "<li><label><input type='checkbox' name='" + htmlEntities(alt) + "' id='checkbox-del-" + i + "' /> " + htmlEntities(alt) + (allMaps[i].overlay ? " (Overlay)" : "") + "</label></li>"; } } else { mapfields_del = "< No custom maps installed >"; } $("#GME_mapfields").html(mapfields); $("#GME_mapfields_del").html(mapfields_del); $("#GME_map_default").html(mapselect); $("#GME_hill_shade_default").html(shadingSelect); sel = $("#GME_map_default").children(); for (i = sel.length - 1; i > -1; i--) { if (sel[i].value === that.parameters.defaultMap) { sel[i].selected = "selected"; } } overSel = $("#GME_hill_shade_default").children(); for (i = overSel.length - 1; i > -1; i--) { if (overSel[i].value === that.parameters.defaultHillShade) { overSel[i].selected = "selected"; } } $("#GME_filterFinds").attr("checked", that.parameters.filterFinds); $("#GME_osgbSearch").attr("checked", that.parameters.osgbSearch); $("#GME_follow").attr("checked", that.parameters.follow); $("#GME_labelStyle").val(that.parameters.labels); $("#GME_measure").val(that.parameters.measure); $("#GME_brightness").val(that.parameters.brightness * 100); $("#GME_version").html(that.parameters.version); } function storeSettings() { var i, j, list, apiKeys; that.parameters.defaultMap = $("#GME_map_default")[0].value; that.parameters.defaultHillShade = $("#GME_hill_shade_default")[0].value; list = $("#GME_mapfields input"); for (i = list.length - 1; i >= 0; i--) { for (j = that.parameters.maps.length - 1; j >= 0; j--) { if (that.parameters.maps[j].alt === list[i].name) { that.parameters.maps[j].ignore = !list[i].checked; } } } for (j = that.parameters.maps.length - 1; j >= 0; j--) { if (that.parameters.maps[j].alt === that.parameters.defaultMap) { that.parameters.maps[j].ignore = false; } } list = $("#GME_mapfields_del input"); for (i = list.length - 1; i >= 0; i--) { if (list[i].checked === true) { for (j = that.parameters.maps.length - 1; j >= 0; j--) { if (that.parameters.maps[j].alt === list[i].name) { that.parameters.maps.splice(j, 1); break; } } } } if (!that.parameters.apiKeys) { that.parameters.apiKeys = {}; } apiKeys = $("#GME_apikeys_field label input"); for (var ak = 0; ak < apiKeys.length; ak++) { const field = apiKeys[ak]; if (field.value) { that.parameters.apiKeys[field.name] = field.value; } else if (that.parameters.apiKeys[field.name]) { delete that.parameters.apiKeys[field.name]; } } that.parameters.brightness = $("#GME_brightness").val() / 100; that.parameters.filterFinds = $("#GME_filterFinds")[0].checked ? true : false; that.parameters.follow = $("#GME_follow")[0].checked ? true : false; that.parameters.labels = $("#GME_labelStyle")[0].value; that.parameters.measure = $("#GME_measure")[0].value; that.parameters.osgbSearch = $("#GME_osgbSearch")[0].checked ? true : false; localStorage.setItem("GME_parameters", JSON.stringify(that.parameters)); refresh(); } }, cssTransitionsFix: function () { // <bugfix> // Work around bug that breaks JQuery Mobile dialog boxes in Opera 12. if (window.$ && $.support) { $.support.cssTransitions = false; } // </bugfix> }, dist: function () { $("#lblDistFromHome").parent().append("<br/><span id='gme-dist'><a href='#' id='gme-dist-link'>Check distance from here</a></span>"); $("#gme-dist-link").click(function () { var there = new LatLon(mapLatLng.lat, mapLatLng.lng), rose = [[22.5, 67.5, 112.5, 157.5, 202.5, 247.5, 292.5, 337.5], ["N", "NE", "E", "SE", "S", "SW", "W", "NW"]], watcher; function found(pos) { var here = new LatLon(pos.coords.latitude, pos.coords.longitude), bearing = here.bearingTo(there), dir = "N", i; for (i = 0; i < 8; i++) { if (bearing < rose[0][i]) { dir = rose[1][i]; break; } } $("#gme-dist").html("<img style='vertical-align:text-bottom' alt='" + dir + "' src='/images/icons/compass/" + dir + ".gif'> " + dir + " " + formatDistance(here.distanceTo(there) * 1000) + " from here at bearing " + Math.round(bearing) + "°"); } function lost() { if (watcher) { navigator.geolocation.clearWatch(watcher); } alert("GME couldn't detect your location.\nDisable FollowMe mode in Geocaching Map Enhancements if this error pops up repeatedly."); } if (that.parameters.follow) { watcher = navigator.geolocation.watchPosition(found, lost, { timeout: 60000, maximumAge: 30000 }); } else { navigator.geolocation.getCurrentPosition(found, lost, { timeout: 60000, maximumAge: 30000 }); } return false; }); }, drag: function () { that.dragStart = function (event) { function GME_formatLOC(wpts) { return wpts ? ['<?xml version="1.0" encoding="UTF-8"?>\n<loc version="1.0" src="Geocaching Map Enhancements v' + that.getVersion() + '">' + wpts.join('\n') + '</loc>'].join('\n') : null; } function GME_formatLOC_wpt(id, desc, coords, type, link) { if (id && desc && coords) { var t = "Geocache", l = link ? ('\n\t<link text="' + link.desc + '">' + link.href + '</link>') : ""; switch (type) { case "Original Coordinates": t = type; break; case 217: t = "Parking Area"; break; case 218: t = "Question to Answer"; break; case 219: t = "Stages of a Multicache"; break; case 220: t = "Final Location"; break; case 221: t = "Trailhead"; break; case 452: t = "Reference Point"; break; } return ('<waypoint>\n\t<name id="' + id + '"><![CDATA[' + desc + ']]></name>\n\t<coord lat="' + coords.lat + '" lon="' + coords.lng + '"/>\n\t<type>' + t + '</type>' + l + '\n</waypoint>'); } console.warn("GME: Missing cache data - id:", id, "desc:", desc, "coords:", coords); return null; } var c, dataURI, dt, i, locfmt, id = $("#ctl00_ContentBody_CoordInfoLinkControl1_uxCoordInfoCode")[0].innerHTML, loc = [GME_formatLOC_wpt(id, cache_coords.primary[0].name, cache_coords.primary[0], cache_coords.primary[0].type, { desc: "Cache Details", href: "https://coord.info/" + id })]; for (i = cache_coords.additional.length - 1; i >= 0; i--) { c = cache_coords.additional[i]; loc.push(GME_formatLOC_wpt(c.pf + id.slice(2), [c.name, $("#awpt_" + c.pf).parent().parent().next().children()[2].innerHTML.trim()].join(" "), c, c.type)); } if (cache_coords.primary[0].isUserDefined) { loc.push(GME_formatLOC_wpt("GO" + id.slice(2), cache_coords.primary[0].name, { lat: cache_coords.primary[0].oldLatLng[0], lng: cache_coords.primary[0].oldLatLng[1] }, "Original Coordinates", { desc: "Cache Details", href: "https://coord.info/" + id })); } locfmt = GME_formatLOC(loc); dataURI = "data:application/xml-loc," + encodeURIComponent(locfmt); dt = event.originalEvent.dataTransfer; if (window.DataTransfer !== undefined && dt.constructor === window.DataTransfer) { dt.setData("application/gme-cache-coords", JSON.stringify(cache_coords)); dt.setData("application/xml-loc", locfmt); dt.setData("text/x-moz-url", dataURI + "\nGME_waypoints.loc"); dt.setData("DownloadURL", "application/xml-loc:GME_waypoints.loc:" + dataURI); } dt.setData("text/uri-list", dataURI); dt.setData("Text", locfmt); dt.effectAllowed = "copy"; dt.setDragImage($('a[title="About Cache Types"] img')[0], 0, 0); }; }, drop: function () { var dropHandlerObj = { onAdd: function (map) { var container = $(map.getContainer()); this._map = map; container.on("drop", this.drop(map)); container.on("dragover", this.dragOver); return document.createElement("div"); }, onRemove: function (map) { var container = $(map.getContainer()); container.off("drop", this.drop(map)); container.off("dragover", this.dragOver); }, drop: function (map) { return function (e) { function typeToIcon(t) { var j, type = t; for (j = wptTypes.length - 1; j >= 0; j--) { type = type.replace(wptTypes[j][0], wptTypes[j][1]); } return type; } function parseLOC(text) { var i, l, w, t, len, lat, lng, name, points = { primary: [], additional: [] }, wpts = $($.parseXML(text)).find("waypoint"); for (i = 0, len = wpts.length; i < len; i++) { w = $(wpts[i]); lat = w.find("coord").attr("lat"); lng = w.find("coord").attr("lon"); name = w.find("name").attr("id") + ": " + w.find("name").text().trim(); if (isNaN(+lat) || isNaN(+lng) || lat < -90 || lat > 90) { return false; } t = w.find("type").text(); if (/Geocache/i.test(t)) { points.primary.push({ lat: lat, lng: lng, name: name, type: 2 }); } else { l = points.primary.length; if (l && /Original Coordinates/i.test(t)) { points.primary[l - 1].oldLatLng = [lat, lng]; points.primary[l - 1].isUserDefined = true; } else { points.additional.push({ lat: lat, lng: lng, name: name, type: typeToIcon(t) }); } } } return (points.additional.length + points.primary.length > 0) ? points : false; } function readLOC(e) { var data = e.target.result, pts = parseLOC(data); if (pts) { console.info("GME: Received LOC file"); GME_displayPoints(pts, map, "dragdrop"); } } function parseGPX(text) { var d, i, j, k, w, r, t, lat, lng, len, n, name = "", poly, type, points = { primary: [], additional: [], routes: [] }, gpx = $($.parseXML(text)), wpts = gpx.find("wpt"), tracks = gpx.find("trk"), segs, routes = gpx.find("rte"); for (i = 0, len = wpts.length; i < len; i++) { w = $(wpts[i]); lat = w.attr("lat"); lng = w.attr("lon"); n = w.filterNode("name"); d = w.filterNode("desc"); name = n.length > 0 ? n[0].textContent : "Point " + i; name += (n.length > 0 && d.length > 0) ? " : " : ""; name += d.length > 0 ? d[0].textContent : ""; if (isNaN(+lat) || isNaN(+lng) || lat < -90 || lat > 90) { return false; } t = w.find("sym").text(); if (/Geocache/i.test(t)) { t = w.filterNode("groundspeak:type"); if (t.length > 0) { type = t[0].textContent; } else { type = "Geocache"; } } else { type = t; } points[/Geocache/i.test(t) ? "primary" : "additional"].push({ lat: lat, lng: lng, name: name, type: typeToIcon(type) }); } for (i = routes.length - 1; i >= 0; i--) { poly = []; r = $(routes[i]); n = r.filterNode("name"); name = n.length > 0 ? n[0].textContent : "Route " + i; wpts = r.find("rtept"); for (j = wpts.length - 1; j >= 0; j--) { w = $(wpts[j]); poly.push(new L.LatLng(w.attr("lat"), w.attr("lon"))); } points.routes.push({ name: name, points: poly }); } for (i = tracks.length - 1; i >= 0; i--) { poly = []; r = $(tracks[i]); segs = r.find("trkseg"); for (j = segs.length - 1; j >= 0; j--) { n = r.filterNode("name"); name = [(n.length > 0) ? n[0].textContent : "Track " + i, " segment ", j].join(""); wpts = $(segs[j]).find("trkpt"); for (k = wpts.length - 1; k >= 0; k--) { w = $(wpts[k]); poly.push(new L.LatLng(w.attr("lat"), w.attr("lon"))); } points.routes.push({ name: name, points: poly }); } } return (points.additional.length + points.primary.length + points.routes.length > 0) ? points : false; } function readGPX(e) { var data = e.target.result, pts = parseGPX(data); if (pts) { console.info("GME: Received GPX file"); GME_displayPoints(pts, map, "dragdrop"); } } e.stopPropagation(); e.preventDefault(); var i, data, dt = e.originalEvent.dataTransfer, file, files = dt.files, pts, reader; try { data = dt.getData("application/gme-cache-coords"); if (data) { console.info("GME: Received GME data"); GME_displayPoints(JSON.parse(data), map, "dragdrop"); return; } data = dt.getData("text/plain"); if (data) { pts = parseLOC(data); if (pts) { console.info("GME: Received LOC text"); GME_displayPoints(pts, map, "dragdrop"); return; } pts = parseGPX(data); if (pts) { console.info("GME: Received GPX text"); GME_displayPoints(pts, map, "dragdrop"); return; } } } catch (E) { console.warn("GME: Drop: " + E); } for (i = files.length - 1; i >= 0; i--) { file = files[i]; if (/application\/xml-loc/.test(file.type) || /\.loc$/i.test(file.name)) { reader = new FileReader(); reader.onload = readLOC; reader.readAsText(file); } else { if (/application\/xml-gpx/.test(file.type) || /\.gpx$/i.test(file.name)) { reader = new FileReader(); reader.onload = readGPX; reader.readAsText(file); } else { console.warn("GME: Dropped file not recognised: " + file.name + ", (type: " + file.type + ")"); } } } }; }, dragOver: function (e) { var dt = e.originalEvent.dataTransfer; function contains(array, value) { if (array.indexOf) { return array.indexOf(value) >= 0; } if (array.contains) { return array.contains(value); } console.warn("GME: couldn\'t determine type of dragged data. Accepting anyway."); return true; } if (dt && dt.types) { try { if (contains(dt.types, "application/gme-cache-coords") || contains(dt.types, "application/xml-gpx") || contains(dt.types, "application/xml-loc") || contains(dt.types, "text/plain") || contains(dt.types, "Files")) { e.preventDefault(); return false; } } catch (e) { console.error("GME: dragOver:", e); } } return false; } }; }, loadDefault: function () { if (typeof window.$ === "function") { gmeInit(gmeConfig.env.init); } }, labels: function () { function GME_load_labels(control, div) { function labelHandler() { var action = this.getAttribute("data-gme-action"), cache = this.getAttribute("data-gme-cache"); switch (action) { case "panTo": if (control.labels.labels[cache]) { control._map.panTo(control.labels.labels[cache][2]); } break; case "refresh": control.labels.refresh(); break; case "clear": control.labels.removeLabels(); break; case "auto": control.labels.toggleAuto(); break; case "show": control.labels.toggleShow(); break; } return false; } L.GME_identifyLayer = L.Class.extend({ initialize: function (latlng, options) { L.Util.setOptions(this, options); this._latlng = latlng; }, onAdd: function (map) { this._map = map; this._el = L.DomUtil.create("div", "gme-identify-layer leaflet-zoom-hide"); this._el.innerHTML = this.options.label; this._el.title = this.options.desc; this._el.style.position = "absolute"; map.getPanes().overlayPane.appendChild(this._el); map.on("viewreset", this._reset, this); this._reset(); }, onRemove: function (map) { map.getPanes().overlayPane.removeChild(this._el); map.off("viewreset", this._reset, this); }, options: { label: "Cache", desc: "Long cache name" }, setPosition: function (ll) { this._latlng = ll; this._reset(); }, _reset: function () { if (this._map) { var pos = this._map.latLngToLayerPoint(this._latlng); L.DomUtil.setPosition(this._el, pos); } } }); control.labels = { showLabels: false, autoUpdate: false, labels: {}, labelLayer: new L.LayerGroup(), clearLabels: function () { control._map.removeLayer(control.labels.labelLayer); }, displayLabels: function () { control._map.addLayer(control.labels.labelLayer); }, refresh: function () { if (!(window.MapSettings && MapSettings.MapLayers && MapSettings.MapLayers.UTFGrid)) { return; } var i, coords, tiles = $(".leaflet-tile-pane .leaflet-layer img[src*='geocaching.com/map.png']"); for (i = tiles.length - 1; i >= 0; i--) { coords = tiles[i].src.match(/x=(\d+)&y=(\d+)&z=(\d+)/); if (coords && !MapSettings.MapLayers.UTFGrid._cache.hasOwnProperty(([coords[3], coords[1], coords[2]].join("_")))) { MapSettings.MapLayers.UTFGrid._loadTile(coords[3], coords[1], coords[2]); } } setTimeout(control.labels.refreshLabels, 500); }, refreshLabels: function () { var c, p, q, r, tile, tilepos, tileref, gridref, zoom = control._map.getZoom(); if (!(window.MapSettings && MapSettings.MapLayers && MapSettings.MapLayers.UTFGrid)) { return; } for (p in MapSettings.MapLayers.UTFGrid._cache) { if (MapSettings.MapLayers.UTFGrid._cache.hasOwnProperty(p)) { tileref = p.split("_"); if (tileref.length === 3) { tile = $("img[src*='x=" + tileref[1] + "&y=" + tileref[2] + "&z=" + tileref[0] + "']")[0]; if (tile) { tilepos = L.DomUtil.getPosition(tile); if (MapSettings.MapLayers.UTFGrid._cache[p]) { for (q in MapSettings.MapLayers.UTFGrid._cache[p].data) { if (MapSettings.MapLayers.UTFGrid._cache[p].data.hasOwnProperty(q)) { for (r in MapSettings.MapLayers.UTFGrid._cache[p].data[q]) { if (MapSettings.MapLayers.UTFGrid._cache[p].data[q].hasOwnProperty(r)) { c = MapSettings.MapLayers.UTFGrid._cache[p].data[q][r]; if (!control.labels.labels[c.i]) { gridref = q.match(/\((\d+), (\d+)\)/); if (gridref) { control.labels.labels[c.i] = [c.i, c.n, control._map.layerPointToLatLng(tilepos.add(new L.Point(4 * gridref[1], 4 * gridref[2]))), zoom]; control.labels.labels[c.i][4] = new L.GME_identifyLayer(control.labels.labels[c.i][2], (that.parameters.labels === "names") ? { label: c.n, desc: c.i } : { label: c.i, desc: c.n }); } } else { if (zoom > control.labels.labels[c.i][3]) { gridref = q.match(/\((\d+), (\d+)\)/); if (gridref) { control.labels.labels[c.i][2] = control._map.layerPointToLatLng(tilepos.add(new L.Point(4 * gridref[1], 4 * gridref[2]))); control.labels.labels[c.i][3] = zoom; control.labels.labels[c.i][4].setPosition(control.labels.labels[c.i][2]); } } } control.labels.labelLayer.addLayer(control.labels.labels[c.i][4]); } } } } } } } } } control.labels.updateCachePanel(); if (control.labels.showLabels) { control.labels.clearLabels(); control.labels.displayLabels(); } }, removeLabels: function () { $("#gme_cachelist").html(""); control.labels.labelLayer.clearLayers(); control.labels.labels = {}; }, toggleAuto: function () { if (control.labels.autoUpdate) { control.labels.autoUpdate = false; control._map.off("moveend", control.labels.refresh); $(".gme-button-labels-auto").removeClass("gme-button-active"); } else { control.labels.autoUpdate = true; $(".gme-button-labels-auto").addClass("gme-button-active"); control._map.on("moveend", control.labels.refresh); control.labels.refresh(); } }, toggleShow: function () { if (control.labels.showLabels) { control.labels.showLabels = false; $(".gme-button-labels-show").removeClass("gme-button-active"); control.labels.clearLabels(); } else { control.labels.showLabels = true; $(".gme-button-labels-show").addClass("gme-button-active"); control.labels.refresh(); } }, updateCachePanel: function () { var i, j, sortorder = [], html = ""; for (i in control.labels.labels) { if (control.labels.labels.hasOwnProperty(i)) { sortorder.push(i); } } sortorder.sort(); j = sortorder.length; for (i = 0; i < j; i++) { html += "<tr><td><a href='https://coord.info/" + sortorder[i] + "' target='_blank' rel='noopener noreferrer'>" + control.labels.labels[sortorder[i]][1] + "</a></td><td class='gme-cache-code'> " + sortorder[i] + "</td><td><a class='gme-event' title='Pan map to cache location' data-gme-action='panTo' data-gme-cache='" + sortorder[i] + "'><img src='../images/silk/map.png' width='16' height='16' alt='Pan' /></a></td></tr>"; } $("#gme_cachelist").html(html); } }; $("#searchtabs ul").append("<li id='gme_caches_button'><a href='#gme_caches' title='GME Cache Label List' id='gme_caches_link'>GME</a></li>"); $("#searchtabs li").css("width", 100 / $("#searchtabs li").length + "%"); $("#pqlink").html("PQs"); $("#clistButton").html("GCVote"); document.getElementById("pqlink").innerHTML = "PQs"; $(div).append("<div id='gme_caches'>\ <div class='leaflet-control-gme'>\ <a title='Refresh cache labels' class='gme-event gme-button-refresh-labels gme-button gme-button-l' data-gme-action='refresh'></a><a title='Empty cache list and remove labels from map' class='gme-event gme-button gme-button-clear-labels' data-gme-action='clear'></a><a class='gme-event gme-button gme-button-wide gme-button-labels-show' data-gme-action='show'>Show labels</a><a class='gme-event gme-button gme-button-r gme-button-wide gme-button-labels-auto' data-gme-action='auto'>Auto update</a>\ </div>\ <table><tbody id='gme_cachelist'><tr><td colspan='3'>Hit the refresh button above to populate the list.</td></tr></tbody></table></div>"); $("#gme_caches").css("display", "none").on("click", ".gme-event", labelHandler); $("#gme-labels-show").on("change", control.labels.toggleShow); $("#gme-labels-auto").on("change", control.labels.toggleAuto); } }, loadHide: function () { function load() { window.GME_control = new L.GME_Widget().addTo(map); GME_control._layerControl = GME_load_map(map); if (gmeConfig.env.dragdrop) { map.addControl(new L.GME_dropHandler()); } } window.setTimeout(setEnv, 3000); }, loadListing: function () { var cache_coords = {}, mapLink = document.getElementById("ctl00_ContentBody_uxViewLargerMap"); function load() { var parkUrl = "", label = "", i, parking, uri = "&pop="; if (L.LatLng.prototype.toUrl === undefined) { L.LatLng.prototype.toUrl = function () { var obj = this; if (!(obj instanceof L.LatLng)) { return false; } return [L.Util.formatNum(obj.lat, 5), L.Util.formatNum(obj.lng, 5)].join(","); }; } $("#map_canvas").replaceWith("<div style=\'width: 325px; height: 325px; position: relative;\' id=\'map_canvas2\'></div>"); if (gmeConfig.env.dragdrop) { $("#cacheDetails .cacheImage").hover(function (e) { $("#cacheDetails .cacheImage").addClass("moveable"); }, function (e) { $("#cacheDetails .cacheImage").removeClass("moveable"); }); $("#cacheDetails .cacheImage").attr("draggable", "true").on("dragstart", that.dragStart); $("#cacheDetails .cacheImage a").removeAttr("href"); } window.GME_Map = new L.Map("map_canvas2", { center: new L.LatLng(mapLatLng.lat, mapLatLng.lng), zoom: 14 }); GME_Map.addControl(new L.control.scale()); GME_load_map(GME_Map); cache_coords = { primary: [mapLatLng], additional: [] }; if (cmapAdditionalWaypoints && cmapAdditionalWaypoints.length > 0) { cache_coords.additional = cmapAdditionalWaypoints; if (gmeConfig.env.home) { for (i = cmapAdditionalWaypoints.length - 1; i >= 0; i--) { if (cmapAdditionalWaypoints[i].hasOwnProperty("editurl")) { delete cache_coords.additional[i].editurl; } parking = cmapAdditionalWaypoints[i]; if (parking.type === 217 || parking.type === 221) { label = parking.type === 217 ? "Parking Area" : "Trailhead"; parkUrl = `https://www.google.com/maps/dir/${gmeConfig.env.home.toUrl()}/${parking.lat},${parking.lng}/`; $("#awpt_" + parking.pf)[0].parentNode.parentNode.children[1].innerHTML += `<a target="_blank" rel="noopener noreferrer" href="${parkUrl}"><img width="16" height="16" title="[GME] Directions to ${label}" alt="${label}" src="https://www.geocaching.com/images/icons/16/directions.png" /></a>`; } } } } if (gmeConfig.env.dragdrop) { GME_Map.addControl(new L.GME_dropHandler()); } if (cache_coords.primary[0].oldLatLng || cache_coords.primary.length + cache_coords.additional.length > 1) { uri += b64encode(JSON.stringify(cache_coords)); mapLink.href = mapLink.href.replace("http:", "https:") + uri; $('#ctl00_ContentBody_MapLinks_MapLinks a[href*="geocaching.com"]').attr("href", function (i, val) { return val + uri; }); } GME_displayPoints(cache_coords, GME_Map, "listing"); } setEnv(); }, loadMap: function () { function load() { function goSearch(e) { if (e.type === "click" || (e.which || e.keyCode) === 13) { e.preventDefault(); e.stopImmediatePropagation(); return GME_control.search($.trim($("#SearchBox_Text").val() || "")); } } var jsonURI; GME_load_map(MapSettings.Map); if (gmeConfig.env.dragdrop) { MapSettings.Map.addControl(new L.GME_dropHandler()); } window.GME_control = GME_load_widget(MapSettings.Map); GME_load_labels(GME_control, "#scroller"); if (gmeConfig.parameters.osgbSearch) { $("#SearchBox_OS").on("click keypress", goSearch); $(".SearchBox").on("keydown", goSearch); $("#search p")[0].innerHTML = "Search by <span style='cursor:help;' title='Enhanced by Geonames'>Address</span>, Coordinates, GC-code,<br/><span style='cursor:help;' title='Jump to a specific zoom level by typing zoom then a number. Zoom 1 shows the whole world, maxiumum zoom is normally 18-22.'>zoom</span> or <span style='cursor:help;' title='To search using a British National Grid reference, just type it in the search box and hit the button! You can use 2, 4, 6, 8 or 10-digit grid refs with the 2-letter prefix but no spaces in the number (e.g. SU12344225) or absolute grid refs with a comma but no prefix (e.g. 439668,1175316).'>Grid Ref</span>"; } if (window.pnlOpen === false) { $(".leaflet-control-toolbar,.groundspeak-control-findmylocation,.leaflet-control-scale,.gme-left").css("left", "30px"); } if (gmeConfig.env.storage) { if (localStorage.GME_cache) { try { GME_displayPoints(JSON.parse(b64decode(localStorage.GME_cache)), GME_control._map, "clickthru"); delete localStorage.GME_cache; } catch (e) { console.error("GME Can't pop cache: " + e); } } } } setEnv(); }, loadSeek: function () { function load() { function goGR(e) { if (e.type === "click" || (e.type === "keypress" && (e.which || e.keyCode) === 13)) { e.preventDefault(); e.stopImmediatePropagation(); that.seekGR($.trim($("#grRef").val())); return false; } } function goCoords(e) { var c, coords; if (e.type === "click" || (e.type === "keypress" && (e.which || e.keyCode) === 13)) { e.preventDefault(); e.stopImmediatePropagation(); c = document.getElementById("gme_coords").value; if (c) { coords = parseCoords(c); if (coords) { that.seekByLatLng(coords); return false; } } } } function goGoogle(e) { var q = document.getElementById("gme_google").value.trim(), url; e.preventDefault(); e.stopImmediatePropagation(); if (q) { url = "https://www.google.co.uk/search?q=allintitle%3A" + encodeURIComponent(q) + "+site%3Awww.geocaching.com%2Fgeocache%2F+OR+site%3Awww.geocaching.com%2Fseek%2Fcache_details.aspx"; window.open(url, "_blank"); } return false; } $("#grRef").keypress(goGR); $("#grSub").click(goGR); $("#gme_coords").keypress(goCoords); $("#gme_coords_sub").click(goCoords); $("#GME_hereSub").click(that.seekHere); $("#GME_googleSub").click(goGoogle); } setEnv(); }, loadType: function () { function load() { window.GME_control = new L.GME_Widget(); GME_control._layerControl = GME_get_layerControl(); } function reload() { $($(".leaflet-control-layers")[0]).remove(); try { GME_control.removeFrom(map); } catch (e) { } GME_control.addTo(map); GME_control._layerControl.addTo(map); if (gmeConfig.env.dragdrop) { map.addControl(new L.GME_dropHandler()); } setTimeout(function () { map.eachLayer(function (layer) { if (layer instanceof L.TileLayer) { map.removeLayer(layer); } }); GME_control._layerControl.setDefault(); }, 1000); } window.setTimeout(setEnv, 3000); }, loadTrack: function () { function load() { var caches, coords, i, name; function getLogPoints(layer) { if (layer._latlng && layer._popup && layer._popup._content && ~layer._popup._content.indexOf(name)) { coords = layer._latlng; $(caches[i].parentNode.parentNode.children[0]).append("<br/><a href='#GME_map_anchor'><img src='https://www.geocaching.com/images/silk/map.png' width='16' height='16' alt='Map' class='gme-action' data-gme-ref='" + coords.lat + "," + coords.lng + "' title='Centre map on this geocache'/></a> #" + layer._icon.innerHTML); return; } } GME_load_map(map); map.addControl(new L.GME_Widget()); $("#ctl00_ContentBody_lbHeading").append("<a id=\'GME_map_anchor\'></a>"); caches = $(".TrackableLogTable a[href*=cache_details]"); for (i = caches.length - 1; i >= 0; i--) { name = caches[i].textContent.trim(); map.eachLayer(getLogPoints, this); } $(".TrackableLogTable").on("click", ".gme-action", function (e) { map.panTo(L.latLng(this.getAttribute("data-gme-ref").split(","))); }); } setEnv(); }, map: function () { var bounds_CI, bounds_DE, bounds_GB, bounds_IE, bounds_NI, icons = { marker: '%2BXn3xcWF0NfXpzscjtL4%2BPjrYDD4I4BXTQAiAhF1xWKxg%2BHhYRoYGCCfz0d%2Bv58CgQD5%2FX7y%2BXw0ODhIY2NjFIvFdonITkRgRMQ2NjZO19fXPxkdHUVPTw8AQFVVqKoKm80Gm80GACgWi0gmk1hbW0uvrq5%2BynZ2dlaWlpZ%2BmpychNVqRalUwv7%2B%2FhUR%2FQPgbwCfMcY%2Bnpqakrq7u1Gv13FwcICtra1v2fLy8sHe3t4XIyMjKBQKdHh4eG4YxhMienc2xthMe3v7bxMTEw6n08lSqRSmp6dfc7lcThZFEYZh4Pj4%2BMowjMe34U1fXhmG8fhmH4IgIJfLyVw%2Bn%2B8SBAGKopBhGDEieoMWQURvDMOIKYpCgiCgUCjYuUKhYBEEAaqq1k3TfNEKNsM0zReqqtZvsJUTBKHRaDSgqqoJ4O1DGMBbVVXNRqMBnucbnNvtzpbLZYii2AZA%2Fh8si6LYVqlU4Ha7FW5oaGi3VqtBkiQrx3GPHpIcxz2SJMlaq9Ugy%2FIu19nZua3rOmRZZhaLZYUx1tcKMsb6LBbLiizLTNd1SJL0J5fJZLaq1eo1Ywxer1fkef4lY2zoPTjE8%2FxLr9crchyHSqVynUwmnzMiwsLCwmE%2Bnw94PB4oikInJye1tra2PzRN%2B0sUxVnTNL%2Fy%2BXwfuVwulslk0N%2FffxyPxz9vBwCXy%2FV9IpH43ePxwOVyMYfDIVxeXkY0TYuIooje3l7wPA8AuLi4QCAQ%2BAEAWPNJut3ufz0ej9Vut3%2BwYdVqFaenp3VFUXgA4Jobs7Ozz3K53IP3lMvlMDMz8%2Fxd95sTSZK%2BKZfLpqZpLaGu6ygWi6bFYnl6D0ejUX1%2Bfv7XbDbbEmezWczNzT3b3NzU7mEAsNlsTyuVinl1dXUHapqGUqlkWq3Wr2%2Bv38HRaFQPh8M%2Fp1KpOziVSiEcDv9yuypwq9u3IxAIlBljdlmWcXZ2huvr68rR0VHX%2B3ncPQkgFAotptNpVKtVpNNphEKhxVZ5zd%2Fz3ohEItsdHR0UiUS2P5TTsjIAOJ3OxWAwmHQ6na2rAvgPIb3JdHxMgbEAAAAASUVORK5CYII%3D', tick: '%2F5zBkAAAAHdElNRQfcBREVNTVnAq5wAAAACXBIWXMAAAsSAAALEgHS3X78AAAABGdBTUEAALGPC%2FxhBQAAAnRJREFUeNq9lMlrFEEUh19VdU3P1j0koujFmYzE4B8gCaIXTx6SgODFBUTw4lkv4kHxKuLFs4JHEZQkF0GM%2B4KIATHE4DATBMmiSetktq7NVxM7DB4n03nQdHdtv%2FreRmAbrFAo%2BM4e55477Oad7RAkaTKmq%2FqYrMgE3Q464pJL7kGXAQMRu6ClAwEHUodSjlpUzVhdGtElR5IU6QC%2FF2IltHSEkP3%2BOZ833zQFSZKnsRFGdJnxjOPsciD8GoZA4X5shG06Rga9k14inAtxANbkgvwci%2BAm3WiG830c6k%2Fqkqboq0qlUo3FpZ10%2Bo%2BG1kxLgoRJO9dzwn90F%2F2zvsMHuY0d6HXdkEvyWSyClo56tJA9kXUxbtB43tCEk3nTMMt2vmuXIkkaXz4%2BSxgb00nnnfZSbDcDFIHmBywHTqZwjexaEA9m2Ixv6po%2BjkRvi35xAmP1kqTI4TbdeDZp14mSAPldShR%2BHO3tShBvq4peccY0zXm2k42xPDuCBwuMFfPObNBZw2SxrxVs3HNbErSGh9xl%2FWxEr%2BlT%2Fdf6d%2FA8B13Xmu%2Fl7bwwytj4CZqmL0qVUjXa13XSIGWoVtVlY8xscCOwbQsSQwmKbm3Pqx%2FKEiq82GTnvi1lKYoumqq5IMqiGtwOlBFmcy6cbXeXmvqlXvdM0Fr5W%2FkdGLi6%2FmBd1x7VNgZR13YXkiCf8G%2Bxp4LWMFnu0CSdCG4FYbvQA%2BwuH1sS4zgZlUxkrBeCQRCInJt7DxxGxbzIsT7GalO1lvltrgSrwXLnWtILwcgGhgaOYpE%2FpH3Uw5L5olbUMBLWe%2B7SyEzLTOOJ19VP1cLmPf2%2FmLW%2FylgqETpxl%2BsAAAAASUVORK5CYII%3D' }, wptTypes = [[/Geocache/i, "2"], [/Traditional Cache/i, "2"], [/Multi-cache/i, "3"], [/Virtual Cache/i, "4"], [/Letterbox Hybrid/i, "5"], [/Event Cache/i, "6"], [/Unknown cache/i, "8"], [/Webcam Cache/i, "11"], [/Cache In Trash Out Event/i, "13"], [/Wherigo Cache/i, "1858"], [/Locationless \(Reverse\) Cache/i, "12"], [/Mega-Event Cache/i, "453"], [/GPS Adventures Exhibit/i, "1304"], [/Groundspeak Block Party/i, "4738"], [/Groundspeak HQ/i, "3773"], [/Groundspeak Lost and Found Celebration/i, "3774"], [/Lost and Found Event Cache/i, "3653"], [/Project APE Cache/i, "9"], [/Earthcache/i, "137"], [/Question to Answer/i, "218"], [/Parking Area/i, "217"], [/Stages of a Multicache/i, "219"], [/Final Location/i, "220"], [/Trailhead/i, "221"], [/Reference Point/i, "452"]], polylineObj = { initialize: function (pts, ops) { L.Polyline.prototype.initialize.call(this, pts, ops); this._length = 0; this._markers = L.layerGroup(); this._updateMarkers(); }, addLatLng: function (pt) { L.Polyline.prototype.addLatLng.call(this, pt); var len = this._latlngs.length; this._addMarker(pt, len); if (len > 1) { this._length += this._latlngs[len - 2].distanceTo(this._latlngs[len - 1]); this.fire("gme-length", { length: this._length }); } return this; }, getData: function () { return ((typeof window.btoa === "function") ? "data:application/xml-gpx;base64," : "data:application/xml-gpx,") + b64encode(this.getGPX()); }, getGPX: function () { var name = $(".CommonUsername").attr("title"), author = name ? (name + '</name>\r\n\t\t\t<link href="https://www.geocaching.com/profile/?u=' + name + '"><text>' + name + '\'s profile</text></link>\r\n') : "Geocaching.com user</name>\r\n", date = !!Date.prototype.toISOString ? ["\t\t<time>", new Date().toISOString(), "</time>\r\n"].join("") : "", i, l, gpx = ["<?xml version=\"1.0\" encoding=\"UTF-8\"?>\r\n<gpx creator=\"Geocaching Map Enhancements v", that.getVersion(), "\" xmlns:xsd=\"http://www.w3.org/2001/XMLSchema\" xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\" version=\"1.1\" xsi:schemaLocation=\"http://www.topografix.com/GPX/1/1 http://www.topografix.com/GPX/1/1/gpx.xsd\" xmlns=\"http://www.topografix.com/GPX/1/1\">\r\n\t<metadata>\r\n\t\t<name>GME Export</name>\r\n\t\t<desc>Route file exported from Geocaching.com using Geocaching Map Enhancements.</desc>\r\n\t\t<author>\r\n\t\t\t<name>", author, "\t\t</author>\r\n", date, "\t</metadata>\r\n"].join(""); for (i = 0, l = this._latlngs.length; i < l; i++) { gpx += ["\t<wpt lat=\"", this._latlngs[i].lat, "\" lon=\"", this._latlngs[i].lng, "\">\r\n\t\t<name>P", i, "</name>\r\n\t\t<type>Waypoint</type>\r\n\t</wpt>\r\n"].join(""); } gpx += "\t<rte>\r\n\t\t<name>GME exported route</name>\r\n\t\t<src>Manual entry</src>\r\n\t\t<number>1</number>\r\n"; for (i = 0, l = this._latlngs.length; i < l; i++) { gpx += ["\t\t<rtept lat=\"", this._latlngs[i].lat, "\" lon=\"", this._latlngs[i].lng, "\">\r\n\t\t\t<name>P", i, "</name>\r\n\t\t\t<type>Waypoint</type>\r\n\t\t</rtept>\r\n"].join(""); } gpx += "\t</rte>\r\n</gpx>"; return gpx; }, getLength: function () { return this._length; }, onAdd: function (map) { map.addLayer(this._markers); this._markers.eachLayer(function (a) { a.dragging.enable(); }); return L.Polyline.prototype.onAdd.call(this, map); }, onRemove: function (map) { map.removeLayer(this._markers); return L.Polyline.prototype.onRemove.call(this, map); }, removePt: function (num) { this.spliceLatLngs(num - 1, 1); this._updateMarkers(); }, setLatLngs: function (pts) { L.Polyline.prototype.setLatLngs.call(this, pts); this._updateMarkers(); return this; }, _addMarker: function (pt, num) { var mark = new L.Marker(pt, { icon: new L.Icon({ draggable: "true", iconUrl: icons.marker, iconSize: new L.Point(15, 25), iconAnchor: new L.Point(8, 25) }), zIndexOffset: 99, title: "Route Point #" + num }); mark._routeNum = num; mark.bindPopup(["<p><strong>Route Point #", num, "</strong><br/>Centre: ", pt.toUrl(), "<br/><strong>", DMM(pt), "</strong><br/><span style='float:right;'><a class='gme-event' data-gme-action='removeDistMarker' data-gme-layer='", this._leaflet_id, "' data-gme-ref='", num, "'>Clear</a>, <a class='gme-event' data-gme-layer='", this._leaflet_id, "' data-gme-action='clearDist'>Clear All</a>, <a class='gme-event gme-draggable-gpx' data-gme-action='exportDist' data-gme-layer='", this._leaflet_id, "' draggable='true'>GPX</a></span></p>"].join("")); mark.on("dragend", this._moveMarker, this); this._markers.addLayer(mark); if (mark.dragging) { mark.dragging.enable(); } }, _moveMarker: function (e) { this.spliceLatLngs(e.target._routeNum - 1, 1, e.target.getLatLng()); this._updateLength(); }, _updateLength: function () { var i; this._length = 0; for (i = 1; i < this._latlngs.length; i++) { this._length += this._latlngs[i - 1].distanceTo(this._latlngs[i]); } this.fire("gme-length", { length: this._length }); return this._length; }, _updateMarkers: function () { var i; this._markers.clearLayers(); if (this._latlngs.length > 0) { this._addMarker(this._latlngs[0], 1); for (i = 1; i < this._latlngs.length; i++) { this._addMarker(this._latlngs[i], i + 1); } } this._updateLength(); } }, quadkeyLayerObj = { tile2quad: function (x, y, z) { var i, digit, mask, quad = ""; for (i = z; i > 0; i--) { digit = 0; mask = 1 << (i - 1); if ((x & mask) !== 0) { digit += 1; } if ((y & mask) !== 0) { digit += 2; } quad = quad + digit; } return quad; }, getTileUrl: function (tilePoint) { return L.Util.template(this._url, L.extend({ s: this._getSubdomain(tilePoint), q: this.tile2quad(tilePoint.x, tilePoint.y, this._getZoomForUrl()), z: this._getZoomForUrl() }, this.options)); } }, complexLayerObj = { getTileUrl: function (tilePoint) { return L.Util.template(this._url, L.extend({ s4: tilePoint.x % 4 + 4 * (tilePoint.y % 4), x100: this._getZoomForUrl() <= 13 ? "" : Math.floor(tilePoint.x / 100) + "/", z: this._getZoomForUrl(), x: tilePoint.x, y: tilePoint.y }, this.options)); } }; function GME_displayPoints(plist, map, context) { var bounds = new L.LatLngBounds(), i, p, layers = L.featureGroup(), ll, op, PinIcon = L.Icon.extend({ iconSize: new L.Point(20, 23), iconAnchor: new L.Point(10, 23) }); function checkType(t) { var j; for (j = wptTypes.length - 1; j >= 0; j--) { if (t == wptTypes[j][1]) { return wptTypes[j][1]; } } return 452; } for (i = plist.primary.length - 1; i >= 0; i--) { p = plist.primary[i]; ll = L.latLng(p.lat, p.lng); if (context === "listing" || context === "dragdrop" || p.isUserDefined) { layers.addLayer(L.marker(ll, { icon: new PinIcon({ iconUrl: "/images/wpttypes/pins/" + checkType(p.type) + ".png", iconAnchor: L.point(10, 23) }), clickable: false, zIndexOffset: 98, title: p.name + (p.isUserDefined ? " (Corrected coordinates)" : "") })); if (p.isUserDefined) { layers.addLayer(L.marker(ll, { icon: new PinIcon({ iconSize: new L.Point(28, 23), iconAnchor: L.point(10, 23), iconUrl: icons.tick }), clickable: false, zIndexOffset: 99, title: p.name + " (Corrected coordinates)" })); } } else { bounds.extend(ll); } if (p.isUserDefined) { op = L.latLng(p.oldLatLng[0], p.oldLatLng[1]); layers.addLayer(L.polyline([op, ll], { clickable: false, weight: 3 })); if (context === "listing") { layers.addLayer(L.circleMarker(op, { clickable: false, weight: 3, radius: 6 })); } } } for (i = plist.additional.length - 1; i >= 0; i--) { p = plist.additional[i]; ll = L.latLng(p.lat, p.lng); layers.addLayer(L.marker(ll, { icon: new PinIcon({ iconUrl: "/images/wpttypes/pins/" + checkType(p.type) + ".png", iconAnchor: new L.Point(10, 23) }), title: p.name, clickable: false })); } if (plist.routes) { for (i = plist.routes.length - 1; i >= 0; i--) { if (plist.routes[i].points && plist.routes[i].points.length > 0) { layers.addLayer(L.polyline(plist.routes[i].points)); } } } bounds.extend(layers.getBounds()); if (bounds.isValid()) { switch (context) { case "listing": map.fitBounds(bounds, { padding: [10, 10], maxZoom: 15 }); break; case "clickthru": if (map.getBoundsZoom(bounds) > 15) { map.panTo(bounds.getCenter()).setZoom(15); } else { map.fitBounds(bounds); } break; default: map.panTo(bounds.getCenter()); } } map.addLayer(layers); return bounds; } function genericLayerFn(url, options) { function filterOpts(opts) { /* Remove GME's internal options, so they don't get passed to servers (WMS in particular). */ var opt, filtered = {}, exclude = ["tileUrl", "ignore", "alt"]; for (opt in opts) { if (exclude.indexOf(opt) === -1) { filtered[opt] = options[opt]; } } return filtered; } var filteredOpts = filterOpts(options); if (typeof url === "string") { return (/\{q\}/).test(url) ? (new L.GME_QuadkeyLayer(url, filteredOpts)) : ((/\{s4\}|\{x100\}/).test(url) ? (new L.GME_complexLayer(url, filteredOpts)) : ((/\{x\}/).test(url) ? (new L.TileLayer(url, filteredOpts)) : (new L.TileLayer.WMS(url, filteredOpts)))); } console.error("GME: Bad map source: " + JSON.stringify(options)); return undefined; } function setBrightness(e) { var brightness = this.brightness || that.parameters.brightness; if (brightness < 1) { $(".leaflet-container").css("backgroundColor", "#000"); } else { $(".leaflet-container").css("backgroundColor", "#ddd"); } if (e.layer._url && /^http/.test(e.layer._url) && e.layer.options && !e.layer.options.overlay) { e.layer.setOpacity(brightness); } } function switchLayer(e) { var layer = e.layer; if (layer.options && layer.options.tileUrl && !layer.options.overlay) { this.layersMaxZoom = layer.options.maxZoom; this.layersMinZoom = layer.options.minZoom; if (isNaN(layer.options.maxZoom) || !isNaN(layer.options.minZoom)) { this._zoomBoundLayers[L.stamp(layer)] = layer; this._updateZoomLevels(); } if (this.getZoom() > this.layersMaxZoom) { this.setZoom(this.layersMaxZoom); } if (this.getZoom() < this.layersMinZoom) { this.setZoom(this.layersMinZoom); } this.brightness = e.layer.options.brightness || that.parameters.brightness; } } function GME_get_layerControl(map) { var maps = {}, overlays = {}, allMaps = that.parameters.maps, apiKeys = that.parameters.apiKeys, baseMaps, control, i, layer, src; for (baseMaps = 0, i = 0; i < allMaps.length; i++) { src = allMaps[i]; if (!src.ignore) { let tileUrl = src.tileUrl; if (apiKeys && src.apiKey && src.apiKeyQuery) { const apiKey = apiKeys[src.apiKey]; if (apiKey) { var replacedTemplate = src.apiKeyQuery.replace(/\{apikey\}/, apiKey); const url = new URL(tileUrl); const search = url.search; url.search = search ? `${search}&${replacedTemplate}` : `?${replacedTemplate}`; tileUrl = decodeURI(url.href); } } layer = L.GME_genericLayer(tileUrl, src); if (layer) { if (src.overlay) { overlays[src.alt] = layer; if (src.alt === that.parameters.defaultHillShade) { layer.default = true; } } else { if (src.alt === that.parameters.defaultMap) { layer.default = true; } maps[src.alt] = layer; baseMaps++; } } } } if (baseMaps > 0) { // Only return a new control if we have some basemaps. control = L.control.layers(maps, overlays); control.setDefault = function () { var defLayer, defOverlay, j; for (j in this._layers) { if (this._layers.hasOwnProperty(j) && this._layers[j].layer.default) { if (!this._layers[j].overlay) { defLayer = j; } else { defOverlay = j; } } } if (!defLayer) { defLayer = Object.keys(this._layers)[0]; } if (this._map && defLayer !== undefined) { this._map.addLayer(this._layers[defLayer].layer, true); } if (this._map && defOverlay !== undefined) { this._map.addLayer(this._layers[defOverlay].layer, true); } return defLayer; }; } return control; } function GME_load_map(map) { var control = GME_get_layerControl(), layer; map.on("layeradd", switchLayer); if (document.createElement("div").style.opacity !== undefined) { map.on("layeradd", setBrightness); } /* If we're adding our own map selector control, we need to manually remove any pre-existing map layers. Otherwise, they persist in the background underneath * the layers provided by GME. We check for the _url or _google attribute to distinguish map layers from other Leaflet layers like controls or popups */ if (control) { if (gmeConfig.env.page === "maps" || gmeConfig.env.page === "track" || gmeConfig.env.page === "hide" || gmeConfig.env.page === "hide") { $($(".leaflet-control-layers")[0]).remove(); for (layer in map._layers) { if (map._layers[layer] instanceof L.TileLayer) { if (window.MapSettings !== undefined && MapSettings.MapLayers !== undefined && MapSettings.MapLayers.Geocache === map._layers[layer]) { // Leave geocache layer in place } else { map.removeLayer(map._layers[layer]); } } } } map.addControl(control); control.setDefault(); } } }, osgb: function () { function OSGridToLatLng(E, N) { var a = 6377563.396, b = 6356256.910, F0 = 0.9996012717, lat0 = 49 * Math.PI / 180, lon0 = -2 * Math.PI / 180, N0 = -100000, E0 = 400000, e2 = 1 - (b * b) / (a * a), n = (a - b) / (a + b), n2 = n * n, n3 = n * n2, lat = lat0, lon, M = 0, Ma, Mb, Mc, Md, cosLat, sinLat, nu, nu3, nu5, nu7, rho, eta2, tanLat, tan2lat, tan4lat, tan6lat, secLat, VII, VIII, IX, X, XI, XII, XIIA, dE, dE2, dE3, dE4, dE5, dE6, dE7, tx, ty, tz, rx, ry, rz, s1, sinPhi, cosPhi, sinLambda, cosLambda, eSq, nu2, x1, y1, z1, x2, y2, z2, p, phi, phiP, precision, lambda; do { lat = (N - N0 - M) / (a * F0) + lat; Ma = (1 + n + (5 / 4) * n2 + (5 / 4) * n3) * (lat - lat0); Mb = (3 * n + 3 * n * n + (21 / 8) * n3) * Math.sin(lat - lat0) * Math.cos(lat + lat0); Mc = ((15 / 8) * n2 + (15 / 8) * n3) * Math.sin(2 * (lat - lat0)) * Math.cos(2 * (lat + lat0)); Md = (35 / 24) * n3 * Math.sin(3 * (lat - lat0)) * Math.cos(3 * (lat + lat0)); M = b * F0 * (Ma - Mb + Mc - Md); } while (N - N0 - M >= 0.01); cosLat = Math.cos(lat); sinLat = Math.sin(lat); nu = a * F0 / Math.sqrt(1 - e2 * sinLat * sinLat); rho = a * F0 * (1 - e2) / Math.pow(1 - e2 * sinLat * sinLat, 1.5); eta2 = nu / rho - 1; tanLat = Math.tan(lat); tan2lat = tanLat * tanLat; tan4lat = tan2lat * tan2lat; tan6lat = tan4lat * tan2lat; secLat = 1 / cosLat; nu3 = nu * nu * nu; nu5 = nu3 * nu * nu; nu7 = nu5 * nu * nu; VII = tanLat / (2 * rho * nu); VIII = tanLat / (24 * rho * nu3) * (5 + 3 * tan2lat + eta2 - 9 * tan2lat * eta2); IX = tanLat / (720 * rho * nu5) * (61 + 90 * tan2lat + 45 * tan4lat); X = secLat / nu; XI = secLat / (6 * nu3) * (nu / rho + 2 * tan2lat); XII = secLat / (120 * nu5) * (5 + 28 * tan2lat + 24 * tan4lat); XIIA = secLat / (5040 * nu7) * (61 + 662 * tan2lat + 1320 * tan4lat + 720 * tan6lat); dE = (E - E0); dE2 = dE * dE; dE3 = dE2 * dE; dE4 = dE2 * dE2; dE5 = dE3 * dE2; dE6 = dE4 * dE2; dE7 = dE5 * dE2; lat = lat - VII * dE2 + VIII * dE4 - IX * dE6; lon = lon0 + X * dE - XI * dE3 + XII * dE5 - XIIA * dE7; tx = 446.448; ty = -125.157; tz = 542.060; rx = 7.2819014902652306237205098174164e-7; ry = 1.1974897923405539041670878328241e-6; rz = 4.0826160086234026020206666559563e-6; s1 = 0.9999795106; sinPhi = Math.sin(lat); cosPhi = Math.cos(lat); sinLambda = Math.sin(lon); cosLambda = Math.cos(lon); eSq = (a * a - b * b) / (a * a); nu2 = a / Math.sqrt(1 - eSq * sinPhi * sinPhi); x1 = nu2 * cosPhi * cosLambda; y1 = nu2 * cosPhi * sinLambda; z1 = (1 - eSq) * nu2 * sinPhi; x2 = tx + x1 * s1 - y1 * rz + z1 * ry; y2 = ty + x1 * rz + y1 * s1 - z1 * rx; z2 = tz - x1 * ry + y1 * rx + z1 * s1; a = 6378137; b = 6356752.3142; eSq = (a * a - b * b) / (a * a); p = Math.sqrt(x2 * x2 + y2 * y2); phi = Math.atan2(z2, p * (1 - eSq)); phiP = 2 * Math.PI; precision = 4 / a; while (Math.abs(phi - phiP) > precision) { nu = a / Math.sqrt(1 - eSq * Math.sin(phi) * Math.sin(phi)); phiP = phi; phi = Math.atan2(z2 + eSq * nu * Math.sin(phi), p); } lambda = Math.atan2(y2, x2); return { lat: phi * 180 / Math.PI, lng: lambda * 180 / Math.PI }; } function gridrefLetToNum(letters, numbers) { letters = letters.toUpperCase(); var e, n, l1 = letters.charCodeAt(0) - "A".charCodeAt(0), l2 = letters.charCodeAt(1) - "A".charCodeAt(0); if (l1 > 7) { l1--; } if (l2 > 7) { l2--; } e = ((l1 - 2) % 5) * 5 + (l2 % 5); n = (19 - Math.floor(l1 / 5) * 5) - Math.floor(l2 / 5); e += numbers.slice(0, numbers.length / 2); n += numbers.slice(numbers.length / 2); switch (numbers.length) { case 2: e += "5000"; n += "5000"; break; case 4: e += "500"; n += "500"; break; case 6: e += "50"; n += "50"; break; case 8: e += "5"; n += "5"; break; } return [e, n]; } function parseGR(searchVal) { var ngr, gr = searchVal.match(/^\s*([hnstHNST][A-Ha-hJ-Zj-z])\s*((?:\d\d){1,5})\s*$/); if (gr) { if (gr.length === 3) { if (2 * Math.floor(gr[2].length / 2) === gr[2].length) { ngr = gridrefLetToNum(gr[1], gr[2]); return OSGridToLatLng(ngr[0], ngr[1]); } } return null; } gr = searchVal.match(/^\s*(\d{3,6})\s*,\s*(\d{4,7})\s*$/); if (gr) { if (gr.length === 3) { return OSGridToLatLng(gr[1], gr[2]); } } return null; } }, seek: function () { this.seekGR = function (searchVal) { if (searchVal.length > 0) { var coords = parseGR(searchVal); if (coords !== null) { that.seekByLatLng(coords); } else { alert("Could not recognise grid reference."); } } }; }, widget: function () { var locationControlObj = { onAdd: function (map) { var el, tracking = false, container = L.DomUtil.create("div", "leaflet-control-toolbar groundspeak-control-findmylocation gme-left"); function located(l) { this.panTo(l.latlng); } function click(e) { L.DomEvent.stopPropagation(e); if (tracking) { map.stopLocate(); map.off("locationfound", located); tracking = false; $(".groundspeak-control-findmylocation-lnk").removeClass("gme-button-active"); $("#GME_loc").attr("title", "Follow My Location"); } else { map.on("locationfound", located); map.locate({ enableHighAccuracy: true, watch: true, timeout: 60000 }); tracking = true; $(".groundspeak-control-findmylocation-lnk").addClass("gme-button-active"); $("#GME_loc").attr("title", "Stop following"); } } function click_once(e) { L.DomEvent.stopPropagation(e); this.locate({ setView: true, maxZoom: this.getZoom(), minZoom: this.getZoom(), enableHighAccuracy: true, timeout: 60000 }); } el = document.createElement("a"); el.id = "GME_loc"; el.title = that.parameters.follow ? "Follow My Location" : "Find My Location"; el.className = "groundspeak-control-findmylocation-lnk"; if (that.parameters.follow) { L.DomEvent.addListener(el, "click", click, map); } else { L.DomEvent.addListener(el, "click", click_once, map); } container.appendChild(el); return container; } }, widgetControlObj = { options: { position: "bottomleft" }, onAdd: function (contextmap) { var elem, container = L.DomUtil.create("div", "leaflet-control-gme"), control = this, html = ""; function onPopup(e) { if (e.layer._container && /leaflet-popup/.test(e.layer._container.className)) { $(e.layer._contentNode).on("click", ".gme-event", contextmap, mapHandler); $(e.layer._contentNode).on("dragstart", ".gme-draggable-gpx", { line: control._dist_line }, dragGPXHandler); } } function offPopup(e) { if (e.layer._container && /leaflet-popup/.test(e.layer._container.className)) { $(e.layer._contentNode).off("click", ".gme-event", contextmap, mapHandler); $(e.layer._contentNode).off("dragstart", ".gme-draggable-gpx", { line: control._dist_line }, dragGPXHandler); } } function mapHandler(e) { var action = this.getAttribute("data-gme-action"), c1 = this.getAttribute("data-gme-coords"), c2, coords, data = this.getAttribute("data-gme-ref"), layer = this.getAttribute("data-gme-layer"); e.stopPropagation(); if (action !== "exportDist") { e.preventDefault(e); } if (c1) { c2 = c1.match(/(-?\d{1,2}(\.\d+)?),(-?\d{1,3}(\.\d+)?)/); if (c2 && c2.length === 5 && validCoords(c2[1], c2[3])) { coords = new L.LatLng(c2[1], c2[3]); } } if (action === "clearDist") { control.clearDist(); } if (action === "clearMarkers") { control.clearMarkers(); } if (action === "dropDist" && coords) { control.dropDist(coords); } if (action === "dropMarker" && coords) { control.dropMarker(coords); } if (action === "exportDist") { control.exportDist(this); } if (action === "getGeograph" && coords) { that.getGeograph(coords); } if (action === "getHeight" && coords) { that.getHeight(coords); } if (action === "getPostcode" && coords) { control.getPostcode(coords); } if (action === "panTo" && coords) { e.data.panTo(coords); } if (action === "removeMarker" && data) { control.removeMarker(data); } if (action === "removeDistMarker" && data) { control.removeDistMarker(data); } if (action === "toggleCaches") { control.toggleCaches(); } $(".leaflet-popup-close-button").each(function () { this.click(); }); } function dragGPXHandler(e) { e.originalEvent.dataTransfer.effectAllowed = "copy"; var plain = e.data.line.getGPX(), data = e.data.line.getData(); e.originalEvent.dataTransfer.setData("application/xml-gpx", plain); e.originalEvent.dataTransfer.setData("text/uri-list", data); e.originalEvent.dataTransfer.setData("DownloadURL", "application/xml-gpx:gme-export.gpx:" + data); e.originalEvent.dataTransfer.setData("text/plain", plain); } function widgetHandler(e) { var action = this.getAttribute("data-gme-action"); e.stopPropagation(); if (action === "panToHome") { control.panToHome(); } if (action === "toggleInfo") { control.toggleTool("info"); } if (action === "toggleRoute") { control.toggleTool("route"); } if (action === "toggleCaches") { control.toggleCaches(); } } this._map = contextmap; this._map.infoMode = false; this._map.routeMode = false; this._markers = L.layerGroup().addTo(contextmap); html = "<button type='button' class=\'GME_info gme-button gme-button-l\' title=\'Enable location info tool\' data-gme-action=\'toggleInfo\'></button>"; html += "<button type='button' class=\'GME_hide gme-button\' title=\'Hide caches\' data-gme-action=\'toggleCaches\'></button>"; html += "<button type='button' class=\'GME_route gme-button\' title=\'Enable route tool\' data-gme-action=\'toggleRoute\'></button>"; if (gmeConfig.env.home) { html += "<button type='button' title=\'Go to home location\' class=\'GME_home gme-button\' data-gme-action=\'panToHome\'></button>"; } if (gmeConfig.parameters.osgbSearch) { $(".GME_search_results").on("click", ".gme-event", contextmap, mapHandler); } if (gmeConfig.env.storage) { html += "<a class=\'GME_config gme-button\' title=\'Configure Geocaching Map Enhancements\' href=\'#GME_config\'></a>"; } container.innerHTML = html; $(container.lastChild).addClass("gme-button-r"); container.innerHTML += "<span class=\'gme-button gme-button-l gme-button-r gme-scale-container\' title=\'Approximate width of the full map view\' style=\'cursor:help;\'>Width: <span class=\'gme-scale\'>-</span></span><span class=\'gme-distance-container gme-button gme-button-r\' title=\'Measured distance\'>Route: <span class=\'gme-distance\'>" + formatDistance(0) + "</span></span>"; contextmap.addControl(new L.GME_ZoomWarning()).on("layeradd", onPopup).on("layerremove", offPopup).on("viewreset", this.updateScale, this); $(container).on("click", ".gme-button", this, widgetHandler); $(window).on("resize", this, (function (context) { var t = { timer: null }; return function () { context.updateScale(context._map, t); }; }(this))); return container; }, clearDist: function () { this._dist_line.off("gme-length"); this._map.removeLayer(this._dist_line); delete this._dist_line; $(".gme-distance-container").removeClass("show"); $(".gme-distance").html(formatDistance(0)); $(".gme-scale-container").addClass("gme-button-r"); }, clearMarkers: function () { this._markers.clearLayers(); }, dropDist: function (ll) { if (!validCoords(ll)) { return; } var dist, formatted; if (this._dist_line === undefined) { this._dist_line = new L.GME_DistLine([ll], { clickable: false }); this._dist_line.on("gme-length", function (e) { $(this._map._container).find(".gme-distance").html(formatDistance(e.length)); }); this._map.addLayer(this._dist_line); $(this._map._container).find(".gme-distance-container").addClass("show"); $(this._map._container).find(".gme-scale-container").removeClass("gme-button-r"); } else { this._dist_line.addLatLng(ll); } }, exportDist: function (e) { if (!this._dist_line) { return; } e.download = "ExportedRoute.gpx"; e.href = "data:application/xml-gpx," + encodeURIComponent(this._dist_line.getGPX()); return false; }, dropMarker: function (ll, rad) { if (!validCoords(ll)) { return; } var circle, defaultRadius = 0.161, group, label = "Marker", m = 1000, r, radius, raw, unit = "km"; if (that.parameters.measure !== "metric") { unit = "miles"; defaultRadius = 0.1; m = 1609.344; } radius = defaultRadius; if (!isNaN(rad)) { radius = rad * 1; } else { raw = window.prompt("Radius in " + unit + " [, label]", defaultRadius).match(/([\d]*\.?[\d]*)\s*,?\s*(.*)/); if (raw) { if (raw.length === 3) { radius = raw[1] * 1; label = raw[2] || label; } else { label = raw; } } } if (radius) { radius *= m; if (isNaN(radius)) { radius = 161; } } else { radius = 161; } circle = new L.Circle(ll, radius, { weight: 2 }); group = new L.LayerGroup([circle, new L.CircleMarker(ll, { weight: 2, radius: 3 })]); this._markers.addLayer(group); r = (radius / m).toFixed(3) + " " + unit; circle.bindPopup("<p><strong>" + label + "</strong><br/>Radius: " + r + "<br/>Centre: decimal " + ll.toUrl() + "<br/><strong>" + DMM(ll) + "</strong><br/><span style='float:right;'><a class='gme-event' data-gme-action='removeMarker' data-gme-ref='" + group._leaflet_id + "'>Clear</a>, <a class='gme-event' data-gme-action='clearMarkers'>Clear All</a></span></p>"); }, getPostcode: function (coords) { var that = this, callprefix = "GME_postcode_callback", call; function makeCallback(callname) { callbackCount++; return function (json) { var m; if (json !== undefined && json.status === 200) { if (json.result && json.result.length > 0) { m = "<p>" + json.result[0].postcode + (json.result[0].parish ? (", " + json.result[0].parish) : "") + (json.result[0].admin_ward ? (", " + json.result[0].admin_ward) : "") + "</p>"; } else { m = "<p>No postcode found for this location.<br />Is it within 500m of an occupied building?</p>"; } } else { m = "<p>Error fetching data from postcodes.io</p>"; } if (json.result && !isNaN(json.result[0].latitude) && !isNaN(json.result[0].longitude)) { L.popup().setLatLng({ lat: json.result[0].latitude, lng: json.result[0].longitude }).setContent(m).openOn(that._map); } else { $.fancybox(m); } $("#" + callname).remove(); if (window[callname] !== undefined) { delete window[callname]; } }; } if (validCoords(coords)) { call = callprefix + callbackCount; window[call] = makeCallback(call); JSONP("https://api.postcodes.io/postcodes/lon/" + coords.lng + "/lat/" + coords.lat + "?radius=500&limit=1&callback=" + call, call); } else { console.error("GME: Bad coordinates to getPostcode"); } }, panToHome: function () { if (gmeConfig.env.home) { this._map.panTo(gmeConfig.env.home); return true; } return false; }, removeDistMarker: function (mark) { if (this._dist_line) { this._dist_line.removePt(mark); $(this._map._container).find(".gme-distance").html(formatDistance(this._dist_line.getLength())); } }, removeMarker: function (mark) { this._markers.removeLayer(this._markers._layers[mark]); }, removeMarkers: function (mark) { this._markers.clearLayer(this._markers._layers[mark]); }, showInfo: function (e) { var control = this, popupContent = "<p>", popup = new L.Popup(), i; for (i = 0; i < this.tools.length; i++) { if (this.tools[i].isValid(e.latlng, control._map.getZoom())) { popupContent += this.tools[i].getHTML(e.latlng, control._map.getZoom(), control._map) + " "; } } popupContent += "</p>"; popup.setLatLng(e.latlng); popup.setContent(popupContent); control._map.addLayer(popup); }, tools: [ { name: "Coords", getHTML: function (coords, zoom, map) { var ll = coords.toUrl(); return "<strong>" + DMM(coords) + "</strong><br/>Dec: <a href='geo:" + ll + "?z=" + zoom + "'>" + ll + "</a></br>"; }, isValid: function (coords, zoom) { return true; } }, { name: "List caches", getHTML: function (coords, zoom, map) { return "<a title='List " + (that.parameters.filterFinds ? "unfound " : "") + "caches near point' href='https://www.geocaching.com/seek/nearest.aspx?lat=" + coords.lat + "&lng=" + coords.lng + (that.parameters.filterFinds ? "&f=1" : "") + "' target='_blank' rel='noopener noreferrer'>List caches</a>"; }, isValid: function (coords, zoom) { return true; } }, { name: "Geograph", action: "getGeograph", getHTML: function (coords, zoom, map) { return "<a href='#' title='Show Geograph images near this point' class='gme-event' data-gme-action='getGeograph' data-gme-coords='" + coords.toUrl() + "'>Geograph</a>"; }, isValid: function (coords, zoom) { return bounds_GB.contains(coords) || bounds_DE.contains(coords) || bounds_IE.contains(coords) || bounds_CI.contains(coords); } }, { name: "Directions", getHTML: function (coords, zoom, map) { return "<a title='Launch Google Directions from home to this point' target='_blank' rel='noopener noreferrer' href='https://www.google.com/maps/dir/?api=1&origin=" + gmeConfig.env.home.toUrl() + "&destination=" + coords.toUrl() + "'>Directions</a>"; }, isValid: function (coords, zoom) { return !!gmeConfig.env.home; } }, { name: "Wikimapia", getHTML: function (coords, zoom, map) { var centre = map.getCenter(); return "<a title='Go to wikimapia' target='_blank' rel='noopener noreferrer' href='http://wikimapia.org/#lat=" + centre.lat + "&lon=" + centre.lng + "&z=" + zoom + "'>Wikimapia</a>"; }, isValid: function (coords, zoom) { return true; } }, { name: "Marker", getHTML: function (coords, zoom, map) { return "<a title='Drop route marker onto map' href='#' class='gme-event' data-gme-action='dropMarker' data-gme-coords='" + coords.toUrl() + "'>Marker</a>"; }, isValid: function (coords, zoom) { return true; } }, { name: "MAGIC", getHTML: function (coords, zoom, map) { var b = map.getBounds(); return "<a title='Show MAGIC map of environmentally sensitive areas' target='_blank' rel='noopener noreferrer' href='http://magic.defra.gov.uk/MagicMap.aspx?srs=WGS84&startscale=" + (Math.cos(map.getCenter().lat * L.LatLng.DEG_TO_RAD) * 684090188 * Math.abs(b.getSouthWest().lng - b.getSouthEast().lng)) / map.getSize().x + "&layers=LandBasedSchemes,12,24:HabitatsAndSpecies,38:Designations,6,10,13,16,34,37,40,72,94&box=" + b.toBBoxString().replace(/,/g, ":") + "'>MAGIC</a>"; }, isValid: function (coords, zoom) { return that.isInUK(coords); } }, { name: "Postcode", getHTML: function (coords, zoom, map) { return "<a title='Fetch location data from postcodes.io' href='#' class='gme-event' data-gme-action='getPostcode' data-gme-coords='" + coords.toUrl() + "'>Postcode</a>"; }, isValid: function (coords, zoom) { return that.isInUK(coords); } }, { name: "Height", getHTML: function (coords, zoom, map) { return "<a title='Height of point above sea level' href='#' class='gme-event' data-gme-action='getHeight' data-gme-coords='" + coords.toUrl() + "'>Height</a>"; }, isValid: function (coords, zoom) { return (coords.lat > -65 && coords.lat < 83); } }, { name: "StreetView", getHTML: function (coords, zoom, map) { return "<a title='Launch Google Streetview' target='_blank' rel='noopener noreferrer' href='https://www.google.com/maps/@?api=1&map_action=pano&viewpoint=" + coords.toUrl() + "'>Streetview</a>"; }, isValid: function (coords, zoom) { return true; } }, { name: "MapApp", getHTML: function (coords, zoom, map) { /* Open Bing Maps app if available, otherwise use a cross-platform Google Maps URI */ return "<a title='Launch Bing Maps' href='bingmaps:?cp=" + coords.lat + "~" + coords.lng + "' target='_blank' rel='noopener noreferrer'><a title='Launch Google Maps' href='https://www.google.com/maps/@?api=1&map_action=map¢er=" + coords.toUrl() + "&zoom=" + zoom + "' target='_blank' rel='noopener noreferrer'>Maps</a></a>"; }, isValid: function (coords, zoom) { return true; } } ], showRoute: function (e) { L.DomEvent.stopPropagation(e); this.dropDist(e.latlng); }, toggleCaches: function () { if (window.MapSettings && MapSettings.MapLayers && MapSettings.MapLayers.AddGeocacheLayer && MapSettings.MapLayers.RemoveGeocacheLayer) { if (MapSettings.MapLayers.Geocache) { MapSettings.MapLayers.RemoveGeocacheLayer(); $(".GME_hide").addClass("gme-button-active").attr("title", "Show caches"); } else { MapSettings.MapLayers.AddGeocacheLayer(); $(".GME_hide").removeClass("gme-button-active").attr("title", "Hide caches"); } } }, toggleTool: function (mode) { var that = this, widgets = { info: { on: function () { that._map.on("click contextmenu", that.showInfo, that); $("#map_canvas").addClass("gme-xhair"); $(".GME_info").addClass("gme-button-active").attr("title", "Disable location info tool"); }, off: function () { that._map.off("click contextmenu", that.showInfo, that); $("#map_canvas").removeClass("gme-xhair"); $(".GME_info").removeClass("gme-button-active").attr("title", "Enable location info tool"); } }, none: { on: function () { }, off: function () { } }, route: { on: function () { that._map.on("click contextmenu", that.showRoute, that); $("#map_canvas").addClass("gme-xhair"); $(".GME_route").addClass("gme-button-active").attr("title", "Disable route tool"); }, off: function () { that._map.off("click contextmenu", that.showRoute, that); $("#map_canvas").removeClass("gme-xhair"); $(".GME_route").removeClass("gme-button-active").attr("title", "Enable route tool"); } } }; if (!widgets[mode]) { return; } widgets[this._clickMode].off(); if (mode == this._clickMode) { this._clickMode = "none"; } else { this._clickMode = mode; widgets[mode].on(); } }, search: function (searchVal) { var gr, m, call, callbackPrefix = "GME_search_callback", coords = false, marker, that = this; function searchGS(searchVal) { $(".GME_search_results").addClass("hidden"); $.getJSON("/api/geocode", { q: searchVal }, function (a) { if (a.status === "success") { that._map.panTo(new L.LatLng(a.data.lat, a.data.lng)); } else { alert("Sorry, no results found for " + searchVal); } }); } function makeCallback2(callname) { callbackCount++; return function (json) { var i, j; if (json.geonames && json.geonames.length > 0) { $(".GME_search_list").empty(); for (i = 0, j = json.geonames.length; i < j; i++) { $(".GME_search_list").append("<li><a class='gme-event' data-gme-action='panTo' data-gme-coords='" + json.geonames[i].lat + "," + json.geonames[i].lng + "'>" + json.geonames[i].name + ", " + json.geonames[i].adminName1 + ", " + json.geonames[i].countryCode + "</a></li>"); } $(".GME_search_results").removeClass("hidden"); $(".GME_search_results.ui-collapsible-collapsed a.ui-collapsible-heading-toggle").click(); $(".GME_link_GSSearch").off("click"); $(".GME_link_GSSearch").click(function () { searchGS(searchVal); }); that._map.panTo(new L.LatLng(json.geonames[0].lat, json.geonames[0].lng)); } else { searchGS(searchVal); } $("#" + callname).remove(); if (window[callname] !== undefined) { delete window[callname]; } }; } function makeCallback1(callname) { callbackCount++; return function (json) { var newCall = callbackPrefix + callbackCount; if (json.countryCode) { window[newCall] = makeCallback2(newCall); JSONP("http://api.geonames.org/searchJSON?q=" + encodeURIComponent(searchVal) + "&countryBias=" + json.countryCode + "&maxRows=10&username=gme&callback=" + newCall, newCall); } else { searchGS(searchVal); } $("#" + callname).remove(); if (window[callname] !== undefined) { delete window[callname]; } }; } if (searchVal.length > 0) { m = searchVal.match(/^\s*(?:z|zoom)\s*(\d\d?)\s*$/i); if (m && m.length === 2) { this._map.setZoom(m[1]); return false; } m = searchVal.match(/^\s*(?:p|plot)(?:\s+r\s?(\d*\.?\d*))?\s+(.*)/); if (m && m.length === 3) { coords = parseCoords(m[2]); if (coords) { if (!isNaN(m[1])) { this.dropMarker(L.latLng(coords.lat, coords.lng), m[1]); } else { L.marker(coords, { icon: L.divIcon() }).addTo(this._map).bindPopup(DMM(coords)); } this._map.panTo(coords); return false; } } m = searchVal.match(/^\s*(GC[0123456789ABCDEFGHJKMNOPQRSTVWXYZ]{1,7})\s*$/i); if (m && m.length === 2) { if (gmeConfig.env.loggedin) { this.panToGC(m[1]); $(".GME_search_results").addClass("hidden"); return false; } alert("You must be logged in to allow GME to get cache coordinates"); return false; } gr = parseGR(searchVal); if (gr) { this._map.panTo(new L.LatLng(gr.lat, gr.lng)); } else { call = callbackPrefix + callbackCount; window[call] = makeCallback1(call); JSONP("http://api.geonames.org/countryCodeJSON?lat=" + this._map.getCenter().lat + "&lng=" + this._map.getCenter().lng + "&username=gme&radius=100&callback=" + call, call); } } return false; }, panToGC: function (gc) { var req = new XMLHttpRequest(), map = this._map || e; req.addEventListener("load", function (e) { var r = req.responseText, k = r.indexOf("mapLatLng = {"), c; if (req.status < 400) { try { c = JSON.parse(r.substring(k + 12, r.indexOf("}", k) + 1)); map.panTo(new L.LatLng(c.lat, c.lng)); } catch (e) { console.warn("GME: Couldn't extract cache coordinates:" + e + "\nReceived " + r.length + " bytes, coords at " + k); } } else { if (req.status === 404) { alert("Sorry, cache " + gc + " doesn't seem to exist."); } console.warn("GME: error retrieving cache page to find coords for " + gc + ": " + req.statusText); } }); req.open("GET", "https://www.geocaching.com/geocache/" + gc); req.send(); }, updateScale: function (e, timer) { var map = this._map || e; if (!map.getBounds) { console.warn("updateScale didn't have working map"); return; } function updateMap() { var bound = map.getBounds(); var width = formatDistance(Math.cos(map.getCenter().lat * L.LatLng.DEG_TO_RAD) * 111319.49079327358 * Math.abs(bound.getSouthWest().lng - bound.getSouthEast().lng)); $(this._container).find(".gme-scale").html(width); } if (timer !== undefined) { window.clearTimeout(timer.timer); timer.timer = window.setTimeout(function () { map.whenReady(updateMap); return false; }, 200); } else { map.whenReady(updateMap); } }, _clickMode: "none" }, zoomWarningObj = { options: { position: "topleft" }, onAdd: function (map) { var c = L.DomUtil.create("div", "leaflet-control-zoomwarning gme-left"); function checkZoom() { if (map.getZoom() > map.layersMaxZoom) { map.setZoom(map.layersMaxZoom); } if (map.getZoom() < map.layersMinZoom) { map.setZoom(map.layersMinZoom); } if (this.getZoom() > 18) { c.style.display = "block"; if (typeof amplify === "object" && typeof amplify.store === "function" && amplify.store("ShowPanel") === false) { $(".leaflet-control-zoomwarning").css("left", "30px"); } } else { c.style.display = "none"; } } c.innerHTML = "<a class='gme-button gme-button-l gme-button-r' title='Caches not visible at this zoom level'></a>"; c.style.display = (map.getZoom() > 18) ? "block" : "none"; map.on("zoomend", checkZoom); return c; } }; function GME_load_widget(map) { var control = new L.GME_Widget().addTo(map); $(control._container).addClass("gme-left").css("top", "20px"); $(".groundspeak-control-findmylocation").remove(); if (L.GME_FollowMyLocationControl) { map.addControl(new L.GME_FollowMyLocationControl()); } $(".leaflet-control-scale").addClass("gme-control-scale"); $("a.ToggleSidebar").unbind(); $("a.ToggleSidebar").click(function (a) { a.preventDefault(); if (window.pnlOpen) { window.pnlOpen = false; $(".Sidebar").animate({ left: "-355px" }, 500); $(".leaflet-control-zoom,.leaflet-control-toolbar,.leaflet-control-scale,.gme-left").animate({ left: "30px" }, 500); $(this).removeClass("Open"); } else { window.pnlOpen = true; $(".Sidebar").animate({ left: "0" }, 500); $(".leaflet-control-zoom,.leaflet-control-toolbar,.leaflet-control-scale,.gme-left").animate({ left: "385px" }, 500); $(this).addClass("Open"); } if (typeof amplify === "object" && typeof amplify.store === "function") { amplify.store("ShowPanel", window.pnlOpen); } return false; }); // Trigger reset to update scale and width controls. map.fireEvent("viewreset"); return control; } }, xhr: function (e) { var node = document.getElementById("gme_jsonp_node"), callback = node.getAttribute("data-gme-callback"), url = node.text, details = { "method": "GET", "url": url, "onload": function (response) { var x = response.responseText, call = x.match(/([a-zA-Z_$][0-9a-zA-Z_$]*)\s*\(/), s; if (call && call.length === 2 && call[1] === callback) { s = document.getElementById("gme_jsonp_node"); s.setAttribute("data-gme-callback", callback); s.text = x.substring(x.indexOf("(") + 1, x.lastIndexOf(")")); document.dispatchEvent(new Event("GME_XHR_callback")); } else { console.warn("Received: " + x); } } }; if (gmeResources.env.xhr === 'GM4') { // GreaseMonkey 4+ GM.xmlHttpRequest(details); } else { // Other userscript engines setTimeout(function () { GM_xmlhttpRequest(details); }, 0); } } } }, pageTests = [ ["listing", /\/geocache\/GC|\/seek\/cache_details\.aspx|\/seek\/cache_details2\.aspx/], ["maps", /\/map\//], ["hide", /\/hide\/planning\.aspx/], ["type", /\/hide\/typelocation\.aspx/], ["hide", /\/hide\/waypoints\.aspx/], ["seek", /\/seek\/$|\/seek\/default\.aspx/], ["track", /\/track\/map_gm\.aspx/] ], i, target, target2, targets; function buildScript() { var j, script = ""; for (j = 1; j < arguments.length; j++) { if (typeof arguments[j] === "string" && gmeResources.script.hasOwnProperty(arguments[j])) { script += unwrapFunction(gmeResources.script[arguments[j]]); gmeResources.env.init.push(arguments[j]); } } insertScript( 'var GME;\ (function () {\ "use strict";\ function GeocachingMapEnhancements() {\ var gmeConfig = ' + JSON.stringify({ env: gmeResources.env, parameters: gmeResources.parameters }) + ";" + script + '}\ GME = new GeocachingMapEnhancements();\ console.info("Geocaching Map Enhancements v' + gmeResources.parameters.version + ' loaded.");\ }());', arguments[0] ); } function insertCSS(css) { if (typeof css !== "string") { console.warn("GME: insertCSS not called with string: " + typeof css); return; } var style = document.createElement('style'); style.type = 'text/css'; if (style.styleSheet) { style.styleSheet.cssText = css; } else { style.appendChild(document.createTextNode(css)); } document.documentElement.firstChild.appendChild(style); } function insertPage(div, src, title, back) { if (div && typeof src === 'string') { var d = document.createElement('div'); d.id = div; d.title = title || ''; d.innerHTML = ['<div><a href="#', back || '', '" title="Close" class="gme-close-dialog">X</a><header>', d.title, '</header><div class="gme-modal-content">', src, '</div></div>'].join(''); d.className = 'gme-modalDialog'; document.documentElement.lastChild.appendChild(d); } else { console.warn("GME: insertPage not called with correct parameters."); } } function insertScript(src, id) { console.log("GME: Inserting script: " + id); // console.debug(src); if (typeof src !== "string") { console.warn("GME: insertScript not called with string."); return; } var s = document.createElement("script"); s.type = "text/javascript"; s.text = src; if (id) { s.id = id; } document.documentElement.firstChild.appendChild(s); document.documentElement.firstChild.removeChild(s); } function unwrapFunction(fn) { var text; if (typeof fn === "function") { text = fn.toString(); return text.slice(text.indexOf("{") + 1, text.lastIndexOf("}")); } return fn; } function xhr(e) { var node = document.getElementById("gme_jsonp_node"), callback = node.getAttribute("data-gme-callback"), url = node.text, details = { "method": "GET", "url": url, "onload": function (response) { var x = response.responseText, call = x.match(/([a-zA-Z_$][0-9a-zA-Z_$]*)\s*\(/), s; if (call && call.length === 2 && call[1] === callback) { s = document.getElementById("gme_jsonp_node"); s.setAttribute("data-gme-callback", callback); s.text = x.substring(x.indexOf("(") + 1, x.lastIndexOf(")")); document.dispatchEvent(new Event("GME_XHR_callback")); } else { console.warn("Received: " + x); } } }; if (gmeResources.env.xhr === 'GM4') { GM.xmlHttpRequest(details); } else { setTimeout(function () { GM_xmlhttpRequest(details); }, 0); } } //don't run on frames or iframes if (window.top !== window.self) { return; } if (!(typeof JSON === 'object' && typeof JSON.parse === 'function')) { console.error("Geocaching Map Enhancements requires a browser with JSON support."); return; } if (document.querySelector("head[data-gme-version]")) { console.error("Aborting: GME already running on page: " + document.location); return; } document.documentElement.firstChild.setAttribute("data-gme-version", gmeResources.parameters.version); for (i = 0; i < pageTests.length; i++) { if (pageTests[i][1].test(document.location.pathname)) { gmeResources.env.page = pageTests[i][0]; break; } } try { if (window.localStorage !== undefined && window.localStorage !== null) { gmeResources.env.storage = true; } } catch (e) { /*Potential security exception*/ console.warn("No localStorage capability - GME cannot set configuration"); } if (gmeResources.env.storage) { var a, b, customJSON, GME_custom, paramsJSON, storedParams; /* List of defunct tileUrls to remove from settings */ var blacklist = [ "https://ecn.t{s}.tiles.virtualearth.net/tiles/r{q}?g=737&productSet=mmOS", "https://ecn.t{s}.tiles.virtualearth.net/tiles/r{q}?g=864&productSet=mmCB", "https://otile{s}-s.mqcdn.com/tiles/1.0.0/osm/{z}/{x}/{y}.jpg", "https://otile{s}-s.mqcdn.com/tiles/1.0.0/sat/{z}/{x}/{y}.jpg" ]; try { paramsJSON = localStorage.getItem("GME_parameters"); if (paramsJSON) { try { storedParams = JSON.parse(paramsJSON); if (storedParams.version !== gmeResources.parameters.version) { for (a in gmeResources.parameters) { if (gmeResources.parameters.hasOwnProperty(a)) { if (storedParams[a] === undefined) { storedParams[a] = gmeResources.parameters[a]; } } } alert("Geocaching Map Enhancements has been updated to v" + gmeResources.parameters.version + ". " + gmeResources.parameters.versionMsg); storedParams.version = gmeResources.parameters.version; localStorage.setItem("GME_parameters", JSON.stringify(storedParams)); } if (typeof storedParams.maps === "string") { console.info("GME: Trying to fix corrupted map settings."); storedParams.maps = JSON.parse(storedParams.maps); } gmeResources.parameters = storedParams; } catch (e) { console.warn("GME: Could not parse stored configuration parameters."); } } /* Import old-style custom maps */ customJSON = localStorage.getItem("GME_custom"); if (customJSON) { console.info("GME: Found stored custom settings"); try { GME_custom = JSON.parse(customJSON); if (GME_custom.maps && GME_custom.maps.length > 0) { gmeResources.parameters.maps = gmeResources.parameters.maps.concat(GME_custom.maps); } delete localStorage.GME_custom; } catch (e) { console.warn("GME: Could not parse stored custom maps."); } } /* Remove old-style builtin maps */ if (gmeResources.parameters.includeMaps) { delete gmeResources.parameters.includeMaps; } if (gmeResources.parameters.excludeMaps) { for (a = gmeResources.parameters.excludeMaps.length - 1; a >= 0; a--) { for (b = gmeResources.parameters.maps.length - 1; b >= 0; b--) { if (gmeResources.parameters.maps[b].alt === gmeResources.parameters.excludeMaps[a]) { gmeResources.parameters.maps[b].ignore = true; } } } delete gmeResources.parameters.excludeMaps; } /* Remove broken map sources */ for (a = gmeResources.parameters.maps.length - 1; a >= 0; a--) { for (b = 0; b < blacklist.length; b++) { if (gmeResources.parameters.maps[a].tileUrl === blacklist[b]) { gmeResources.parameters.maps.splice(a, 1); } } } localStorage.setItem("GME_parameters", JSON.stringify(gmeResources.parameters)); } catch (e) { console.error("GME: Bad Exception: " + e); /* Potential security exception. Carry on with default parameters, but block localstorage */ gmeResources.env.storage = false; } } document.addEventListener("GME_XHR_event", xhr); if (!gmeResources.env.geolocation) { gmeResources.script.dist = function () { console.warn("GME: Geolocation not available"); }; } if (!gmeResources.env.dragdrop) { gmeResources.script.drag = function () { console.warn("GME: Drag and Drop not available"); }; gmeResources.script.drop = gmeResources.script.drag; } insertCSS(gmeResources.css.main); if (gmeResources.env.storage) { insertPage('GME_config', gmeResources.html.config, 'Configure GME v' + gmeResources.parameters.version); insertPage('GME_format', gmeResources.html.customInfo, 'Custom Mapsource Format', 'GME_config'); } // <bugfix> // Trixie treats jQuery Mobile dialogs as new page loads, resetting GME's functions if (window.GME !== undefined) { return; } // </bugfix> switch (gmeResources.env.page) { case "listing": // On a geocache listing if (!gmeResources.env.loggedin) { // Not logged in, so no maps or coordinates... console.log("GME: Couldn't detect log-in. Exiting..."); return; } if (gmeResources.env.dragdrop) { insertCSS(gmeResources.css.drag); } buildScript("GME_page_listing", "common", gmeResources.env.storage ? "config" : "", "map", "dist", "drag", "drop", "loadListing"); break; case "seek": // On the Hide & Seek page target2 = document.querySelector(".SeekCacheWidget h4"); targets = document.getElementsByTagName("h5"); for (i = 0; i < targets.length; i++) { if (targets[i].innerHTML.match(/WGS84/)) { target = targets[i]; break; } } if (target && target2) { var grDiv = document.createElement("div"), hereDiv = document.createElement("div"); grDiv.innerHTML = '<h5>Ordnance Survey Grid Reference :</h5><dl><dt>Grid reference : </dt><dd><form name="grForm" id="grForm"><input type="text" class="Text EqualWidthInput" maxlength="50" size="15" name="grRef" id="grRef" placeholder="SU122422"> <input type="submit" class="Button blockWithModalSpinner" name="submitGR" value="Go" id="grSub"></form></dd></dl><h5>Freeform coordinates</h5><dl><dt>Coordinates :</dt><dd><form name="coordForm" id="coordForm"><input type="text" class="Text EqualWidthInput" maxlength="50" size="15" name="gme_coords" id="gme_coords" placeholder="N 51° 10.683′ W 001° 49.604′"/> <input type="submit" class="Button blockWithModalSpinner" name="gme_coords_sub" value="Go" id="gme_coords_sub"/></form></dd></dl>'; hereDiv.innerHTML = '<h4>Where you are...</h4><dl><dt>Use GeoLocation :</dt><dd><form name="hereForm" id="hereForm"><input type="submit" class="Button blockWithModalSpinner" name="GME_hereSub" value="Go" id="GME_hereSub"></form></dd></dl><h4>By keyword...</h4><dl><dt>Google search :</dt><dd><form name="googleForm" id="googleForm"><input type="text" class="Text EqualWidthInput" maxlength="50" size="15" name="gme_google" id="gme_google"/><input type="submit" class="Button blockWithModalSpinner" name="GME_googleSub" value="Go" id="GME_googleSub"></form></dd></dl>'; target.parentNode.insertBefore(grDiv, target); target2.parentNode.insertBefore(hereDiv, target2); buildScript("GME_page_seek", "common", gmeResources.env.storage ? "config" : "", "osgb", "seek", "loadSeek"); } break; case "track": // On a TB tracking map if (!gmeResources.env.loggedin) { // Not logged in, so no maps or coordinates... return; } buildScript("GME_page_track", "common", gmeResources.env.storage ? "config" : "", "map", "widget", "loadTrack"); break; case "maps": // On a Geocaching Maps page // TODO: Detect if the Google Maps API is being used instead of Leaflet, and quit gracefully /* if (document.querySelector("script[src*='//maps.googleapis.com/']")){ console.warn("Geocaching Map Enhancements requires Leaflet Maps to be enabled."); return; f } */ // Check for click-thru cache data in URI var pop = location.search.match(/pop=([A-Za-z0-9+\/=]+)[\?&]?/); if (pop && pop.length === 2) { try { localStorage.setItem("GME_cache", pop[1]); location.search = location.search.replace(/&pop=[A-Za-z0-9+\/=]+[\?&]?/, ""); } catch (e) { console.error(e + "GME couldn't decode click-through data: " + pop[1]); } return; } if (gmeResources.parameters.osgbSearch) { targets = document.getElementsByClassName("SearchBox"); if (targets[0]) { targets[0].innerHTML = gmeResources.html.search; } } buildScript("GME_page_map", "common", gmeResources.env.storage ? "config" : "", "cssTransitionsFix", "map", "widget", "labels", "drop", gmeResources.parameters.osgbSearch ? "osgb" : "", "loadMap"); break; case "type": buildScript("GME_page_type", "common", gmeResources.env.storage ? "config" : "", "map", "widget", "drop", "loadType"); break; case "hide": buildScript("GME_page_hide", "common", gmeResources.env.storage ? "config" : "", "map", "widget", "drop", "loadHide"); break; default: // Somewhere random on the main website if (gmeResources.env.storage) { buildScript("Generic config", "common", "config", "loadDefault"); } } }());