WME JumpMaps LV

The script adds in the WME links to third party mapping systems (Google/Open Street Maps/HERE etc.)

スクリプトをインストールするには、Tampermonkey, GreasemonkeyViolentmonkey のような拡張機能のインストールが必要です。

You will need to install an extension such as Tampermonkey to install this script.

スクリプトをインストールするには、TampermonkeyViolentmonkey のような拡張機能のインストールが必要です。

スクリプトをインストールするには、TampermonkeyUserscripts のような拡張機能のインストールが必要です。

このスクリプトをインストールするには、Tampermonkeyなどの拡張機能をインストールする必要があります。

このスクリプトをインストールするには、ユーザースクリプト管理ツールの拡張機能をインストールする必要があります。

(ユーザースクリプト管理ツールは設定済みなのでインストール!)

このスタイルをインストールするには、Stylusなどの拡張機能をインストールする必要があります。

このスタイルをインストールするには、Stylus などの拡張機能をインストールする必要があります。

このスタイルをインストールするには、Stylus tなどの拡張機能をインストールする必要があります。

このスタイルをインストールするには、ユーザースタイル管理用の拡張機能をインストールする必要があります。

このスタイルをインストールするには、ユーザースタイル管理用の拡張機能をインストールする必要があります。

このスタイルをインストールするには、ユーザースタイル管理用の拡張機能をインストールする必要があります。

(ユーザースタイル管理ツールは設定済みなのでインストール!)

このスクリプトの質問や評価の投稿はこちら通報はこちらへお寄せください
// ==UserScript==
// @name WME JumpMaps LV
// @description The script adds in the WME links to third party mapping systems (Google/Open Street Maps/HERE etc.)
// @license MIT
// @match https://*.waze.com/*editor*
// @match https://www.kadastrs.lv/map/*
// @match https://kartes.lgia.gov.lv/*
// @match https://*.balticmaps.eu/*
// @require https://cdnjs.cloudflare.com/ajax/libs/proj4js/2.11.0/proj4.js
// @require https://greasyfork.org/scripts/24851-wazewrap/code/WazeWrap.js
// @require https://unpkg.com/[email protected]/dist/leaflet.js
// @icon 
// @version 6.0
// @author skirda, alexletov, N190392
// @namespace https://greasyfork.org/en/scripts/481079-wme-jumpmaps
// ==/UserScript==

/* global W, WazeWrap */

var wmeJM_IconWME = GM_info.script.icon;
var wmeJM_version = GM_info.script.version;
console.log("WME-JumpMaps (" + wmeJM_version + "): Start");

var wmeJM_countProbe=0;
var wmeJM_countProbe2=0;
var wmeJM_countProbeWM=0;
var wmeJM_countProbeLOC=0;
var wmeJM_debug=false;
var wmeJM_restoreSelected=false;
var wmeJM_around=false;
var wmeJM_hideWindow = false;
var wmeJM_leftOffset = wmeJM_defaultLeftOffset;
var wmeJM_topOffset = wmeJM_defaultTopOffset;
var wmeJM_defaultLeftOffset = '0px';
var wmeJM_defaultTopOffset = '30px';
var wmeJM_showMinimap = __GetLocalStorageItem("WMEJumpMapsShowMinimap", 'bool', true);
var minimapInstance = null;

var wmeJM_Config={};
var wmeJM_Config0 = {
    "_map_WME":    {save:0, title:"Open in WME",						name:"[WME]",	template:'https://www.waze.com/editor/?env=row&zoomLevel={{zoom}}&lat={{lat}}&lon={{lon}}'},
    "_map_LI":     {save:0, title:"Open in LiveMap",					name:"[Live]",	template:'https://www.waze.com/livemap/?zoom={{zoom}}&lon={{lon}}&lat={{lat}}'},
    //-------------------------------------
    "_map_OSM":    {save:1, title:"OpenStreetMap",			    name:"OpenStreetMap",	template:'http://www.openstreetmap.org/#map={{zoom}}/{{lat}}/{{lon}}'},
    "_map_Google": {save:1, title:"Google Maps",				name:"Google",	template:'http://www.google.com/maps/?ll={{lat}}%2C{{lon}}&z={{zoom}}&t=m'},
    "_map_BING":   {save:1, title:"Bing Maps",					name:"Bing",	template:'http://www.bing.com/maps/?v=2&cp={{lat}}~{{lon}}&lvl={{zoom}}&dir=0&sty=h&form=LMLTEW'}, // sty: "h" - ariel, "r" - map
    "_map_HERE":   {save:1, title:"HERE WeGo",					name:"HERE",	template:'https://wego.here.com/?map={{lat}},{{lon}},{{zoom}},normal'}, // "hybrid.day" - ariel, "normal.day" - map
    "_map_APPLE":  {save:1, title:"Apple Maps",					name:"Apple",	template:'https://maps.apple.com/?ll={{lat}},{{lon}}&spn=0.0038614715299516433%2C0.010368359444299813'},
    "_map_MRY":    {save:1, title:"Mapillary",				    name:"Mapillary",	template:'https://www.mapillary.com/app/?lat={{lat}}&lng={{lon}}&z={{zoom}}'},
    "_map_WM":     {save:1, title:"Wikimapia",				    name:"Wikimapia",	template:'http://wikimapia.org/#lang=ru&lat={{lat}}&lon={{lon}}&z={{zoom}}&m=b'},
    "_map_SC":     {save:1, title:"MapCam",					name:"MapCam",	template:'http://mapcam.info/speedcam/?lng={{lon}}&lat={{lat}}&z={{zoom}}&t=OSM'},
    "_map_WMFLAB": {save:1, title:"GeoHack",    		        name:"GeoHack",	template:'https://tools.wmflabs.org/geohack/geohack.php?params={{lat}}_N_{{lon}}_E_scale:{{zoom}}'},
    "_map_OSV":    {save:1, title:"KartaView",			        name:"KartaView",	template:'http://kartaview.org/map/@{{lat}},{{lon}},{{zoom}}z'},
    "_map_RBASE":  {save:1, title:"RadarBase.info",				name:"RadarBase",	template:'https://radarbase.info/map/actual/{{lat}}/{{lon}}/{{zoom}}'},
    "_map_SPRO":   {save:1, title:"satellites.pro",				name:"SatPRO",	template:'https://satellites.pro/#{{lat}},{{lon}},{{zoom}}'},
    "_map_BM":     {save:1, title:"[LV] BalticMaps",			name:"BalticMaps",	template:'https://balticmaps.eu/lv/c___{{lon}}-{{lat}}-{{zoom}}/bl___cl'},
    "_map_LGIA":   {save:1, title:"[LV] LĢIA Kartes",			name:"LĢIA",	template:'https://kartes.lgia.gov.lv/?x={{lat}}&y={{lon}}&zoom={{zoom}}'},
    "_map_KDL":    {save:1, title:"[LV] Kadastrs",				name:"Kadastrs",	template:'https://www.kadastrs.lv/map/di?xy={{lat}},{{lon}}&z={{zoom}}'},
    "_map_LVM":    {save:1, title:"[LV] LVM GEO",	        	name:"LVM GEO",	template:'https://lvmgeo.lvm.lv/?loc={{lat}};{{lon}};{{zoom}}'},
    "_map_CFY":    {save:1, title:"[LV] Citify",         		name:"Citify",	template:'https://citify.eu/lv/?lng={{lon}}&lat={{lat}}&z={{zoom}}'},
    "_map_BIS":    {save:1, title:"[LV] BIS Plānotie būvdarbi",	name:"BIS",	template:'https://bis.gov.lv/bisp/lv/planned_constructions/bismap#x={{lon}}&y={{lat}}&z={{zoom}}'},
    "_map_LVC":    {save:1, title:"[LV] LVC Būvdarbi",			name:"LVC",	template:'https://map.transportdata.gov.lv/public?mode=summer#{{zoom}}/{{lat}}/{{lon}}'},
    "_map_DODLV":  {save:1, title:"[LV] Dodies.lv",				name:"Dodies",	template:'https://vesture.dodies.lv/#m={{zoom}}/{{lat}}/{{lon}}&l=J'},
};

var wmeJM_ArrW2B=[{w:7,b:-2},{w:6,b:-1},{w:5,b:0},{w:4,b:1},{w:3,b:2},{w:2,b:3},{w:1,b:4},{w:0,b:5}];
var wmeJM_ArrW2KDL=[{w:0,r:75000},{w:1,r:50000},{w:2,r:15000},{w:3,r:10000},{w:4,r:5000},{w:5,r:3000},{w:6,r:1000},{w:7,r:750},{w:8,r:500},{w:9,r:200}];

function WmeJM_GetFaviconURL(key, template) {
    try {
        // explicit icon property wins
        if (key && wmeJM_Config[key] && wmeJM_Config[key].icon) {
            return wmeJM_Config[key].icon;
        }
        // derive from template
        if (typeof template === 'string') {
            var m = template.match(/https?:\/\/([^\/]+)/);
            if (m && m[1]) {
                var domain = m[1].replace(/^www\./, '').replace(/^map\./, '');
                //return 'https://www.google.com/s2/favicons?sz=16&domain=' + encodeURIComponent(domain);
                return 'https://icons.duckduckgo.com/ip3/' + encodeURIComponent(domain) + '.ico';
            }
        }
    } catch (e) {
        console.error('WmeJM_GetFaviconURL error', e);
    }
    return '';
}

function cloneConfig(obj)
{
    if (null === obj || "object" != typeof obj) return obj;
    var copy = obj.constructor();
    for (var attr in obj)
    {
        if (obj.hasOwnProperty(attr))
        {
            copy[attr] = cloneConfig(obj[attr]);
        }
    }
    return copy;
}

function CreateID()
{
    return 'WME-JumpMaps-' + wmeJM_version.replace(/\./g,"-");
}

// NEW: Save only checkbox states
function WmeJM_SaveCheckboxStates()
{
    var states = {};
    for(var i in wmeJM_Config) {
        if(["_map_WME","_map_WMEB","_map_LI"].indexOf(i) < 0) {
            states[i] = wmeJM_Config[i].save;
        }
    }
    localStorage.setItem("WMEJumpMapsCheckboxStates", JSON.stringify(states));
}

// NEW: Load only checkbox states
function WmeJM_LoadCheckboxStates()
{
    var states = localStorage.getItem("WMEJumpMapsCheckboxStates");
    if(states) {
        states = JSON.parse(states);
        for(var i in states) {
            if(wmeJM_Config[i]) {
                wmeJM_Config[i].save = states[i];
            }
        }
    }
}

// OLD function - now just returns empty string since we don't save templates
function WmeJM_Config2String()
{
    return "";
}

function getElementsByClassName(classname, node)
{
    if(!node)
        node = document.getElementsByTagName("body")[0];
    var a = [];
    var re = new RegExp('\\b' + classname + '\\b');
    var els = node.getElementsByTagName("*");
    for (var i=0,j=els.length; i<j; i++)
        if (re.test(els[i].className)) a.push(els[i]);
    return a;
}

function wmer_generate_permalink() {
    var wcp=document.getElementsByClassName('WazeControlPermalink');
    for(var i=0; i < wcp.length; ++i)
        for (var j=0; j < wcp[i].getElementsByTagName('a').length;++j)
        {
            var href=wcp[i].getElementsByTagName('a')[j].href;
            if (href.indexOf(".waze.com/") > 0 && href.indexOf("/editor") > 0)
            {
                // kill "/ru/", kill "layers"
                href=href.replace("/ru/","/").replace(/layers=([0-9]+)\&/,"") + "&marker=yes";
                return href;
            }
        }
    return "";
}

// Helper function to extract query string parameter
function __getQueryString(link, name) {
    if (link.indexOf(name + '=') <= 0) return -1;
    var pos = link.indexOf(name + '=') + name.length + 1;
    var len = link.substr(pos).indexOf('&');
    return (len == -1) ? link.substr(pos) : link.substr(pos, len);
}

// Site configuration - maps hostname to site type
const SITE_CONFIG = {
    'www.waze.com': 'waze',
    'editor-beta.waze.com': 'waze',
    'beta.waze.com': 'waze',
    'balticmaps.eu': 'balticmaps',
    'www.kadastrs.lv': 'kdlv'
};

// Get current site type
function WmeJM_GetLocationType() {
    const hostname = location.hostname;

    // Direct match
    if (SITE_CONFIG[hostname]) return SITE_CONFIG[hostname];

    // Google domains (www.google.*)
    if (hostname.startsWith('www.google.')) return 'google';

    if (wmeJM_debug) console.log("WME-JumpMaps: Unknown site type for " + hostname);
    return "";
}

// Extract lat/lon/zoom from current page
function WmeJM_GetLLZ() {
    const href = location.href;
    const locType = WmeJM_GetLocationType();
    let lat = 0, lon = 0, zoom = 0, city = '';

    const extractors = {
        waze: () => {
            const urPos = new OpenLayers.LonLat(W.map.getCenter().lon, W.map.getCenter().lat);
            urPos.transform(new OpenLayers.Projection("EPSG:900913"), new OpenLayers.Projection("EPSG:4326"));
            return {lat: urPos.lat, lon: urPos.lon, zoom: W.map.getZoom()};
        },

        osm: () => {
            const xy = OSM.mapParams();
            return {lat: xy.lat, lon: xy.lon, zoom: xy.zoom};
        },

        google: () => {
            const i2 = href.indexOf("@");
            if (i2 >= 0) {
                const l = href.substr(i2 + 1).split(",");
                const zoomAttr = l[2].replace(/([0-9]+)([zm]+)(\/.*)?\.*/, '$1.$2').split(".");

                if (zoomAttr[1] === "m") {
                    const ArrM2Z = [
                        {z:1,m:51510000},{z:2,m:25755000},{z:3,m:12877500},{z:4,m:6438750},{z:5,m:3219375},
                        {z:6,m:1609687},{z:7,m:804844},{z:8,m:402422},{z:9,m:201211},{z:10,m:100605},
                        {z:11,m:50303},{z:12,m:25151},{z:13,m:12576},{z:14,m:6288},{z:15,m:3144},
                        {z:16,m:1572},{z:17,m:786},{z:18,m:393},{z:19,m:196},{z:20,m:98},
                        {z:21,m:49},{z:22,m:25},{z:23,m:12}
                    ];
                    const z = parseInt(zoomAttr[0]);
                    let calculatedZoom = 1;
                    for (let i = 0; i < ArrM2Z.length - 1; i++) {
                        if (z <= ArrM2Z[i].m && z >= ArrM2Z[i + 1].m) {
                            calculatedZoom = ArrM2Z[i].z;
                            break;
                        }
                    }
                    return {lat: l[0], lon: l[1], zoom: calculatedZoom};
                }
                return {lat: l[0], lon: l[1], zoom: zoomAttr[0]};
            }
            return {
                lat: __getQueryString(href, 'y'),
                lon: __getQueryString(href, 'x'),
                zoom: parseInt(__getQueryString(href, 'z'))
            };
        },

        balticmaps: () => {
            const res = Array.from(href.matchAll(/http(s)?:\/\/balticmaps\.eu\/\S+\/c___(\d+(\.\d+)?)-(\d+(\.\d+)?)-(\d+)\/.*/g));
            return {lat: res[0][2], lon: res[0][4], zoom: res[0][6]};
        },

        lgia: () => {
            const resp = Array.from(href.matchAll(/https:\/\/kartes\.lgia\.gov\.lv\/karte\/\?x=([0-9]*\.[0-9]+)&y=([0-9]*\.[0-9]+)&zoom=([0-9]+)/g));
            return {lat: resp[0][1], lon: resp[0][2], zoom: resp[0][3]};
        },

        kdlv: () => {
            let frmap = null;
            for (let ii = 0; ii < frames.length; ii++) {
                if (typeof frames[ii].esri !== "undefined") {
                    frmap = frames[ii];
                    break;
                }
            }
            if (frmap) {
                frmap.document.getElementById("dijit_form_Button_15").click();
                const urlKdl = frmap.document.getElementById("dijit_Dialog_0").getElementsByTagName("textarea")[0].value;
                frmap.document.getElementsByClassName("dijitDialogCloseIcon")[0].click();
                const ll = __getQueryString(urlKdl, 'xy').split(',');
                return {lat: ll[1], lon: ll[0], zoom: parseInt(__getQueryString(urlKdl, 'z'))};
            }
            return {lat: 0, lon: 0, zoom: 0};
        }
    };

    // Generic query string extractor for mry, wm, sc (lat/lng/z)
    if (['mry', 'sc'].includes(locType)) {
        lat = __getQueryString(href, 'lat');
        lon = __getQueryString(href, 'lng');
        zoom = parseInt(__getQueryString(href, 'z'));
    } else if (locType === 'wm') {
        lat = __getQueryString(href, 'lat');
        lon = __getQueryString(href, 'lon');
        zoom = parseInt(__getQueryString(href, 'z'));
    } else if (extractors[locType]) {
        const coords = extractors[locType]();
        lat = coords.lat;
        lon = coords.lon;
        zoom = coords.zoom;
    }

    if (wmeJM_debug) console.log(`WME-JumpMaps: GetLLZ (${locType}): lat=${lat}, lon=${lon}, zoom=${zoom}`);
    return {lat, lon, zoom, city};
}

// Transform coordinates FROM external sites TO WME
function WmeJM_Convert_Other2WME(llz) {
    if (wmeJM_debug) console.log("WME-JumpMaps: Convert_Other2WME", llz);

    const locType = WmeJM_GetLocationType();
    const LKS92_PROJ = "+proj=tmerc +lat_0=0 +lon_0=24 +k=0.9996 +x_0=500000 +y_0=-6000000 +ellps=GRS80 +towgs84=0,0,0,0,0,0,0 +units=m +no_defs";

    const transformers = {
        lgia: () => {
            const transformed = proj4(proj4(LKS92_PROJ), proj4('EPSG:4326'), [parseFloat(llz.lon), parseFloat(llz.lat)]);
            llz.lon = transformed[0];
            llz.lat = transformed[1];
            llz.zoom = parseInt(llz.zoom, 10) + 7;
        },
        kdlv: () => {
            const transformed = proj4(proj4(LKS92_PROJ), proj4('EPSG:4326'), [parseFloat(llz.lon), parseFloat(llz.lat)]);
            llz.lon = transformed[0];
            llz.lat = transformed[1];
            llz.zoom = 17;
        },
        kadua: () => {
            const transformed = proj4(proj4('EPSG:900913'), proj4('EPSG:4326'), [parseFloat(llz.lat), parseFloat(llz.lon)]);
            llz.lon = transformed[0];
            llz.lat = transformed[1];
        },
        reglt: () => {
            const transformed = proj4(proj4('EPSG:3346'), proj4('EPSG:4326'), [parseFloat(llz.lon), parseFloat(llz.lat)]);
            llz.lon = transformed[0];
            llz.lat = transformed[1];
            for (let i = 0; i < wmeJM_ArrW2KDL.length - 1; i++) {
                if (llz.zoom >= wmeJM_ArrW2KDL[i + 1].r && llz.zoom <= wmeJM_ArrW2KDL[i].r) {
                    llz.zoom = wmeJM_ArrW2KDL[i].w;
                    break;
                }
            }
        }
    };

    if (transformers[locType]) transformers[locType]();

    if (wmeJM_debug) console.log("WME-JumpMaps: Converted:", llz);
    return llz;
}

// Transform coordinates FROM WME TO external sites
function WmeJM_Convert_WME2Other(id, llz) {
    if (wmeJM_debug) console.log(`WME-JumpMaps: Convert_WME2Other(${id})`, llz);

    llz.zoom = this.id == '_map_LI' ? llz.zoom - 1 : llz.zoom;
    const LKS92_PROJ = "+proj=tmerc +lat_0=0 +lon_0=24 +k=0.9996 +x_0=500000 +y_0=-6000000 +ellps=GRS80 +towgs84=0,0,0,0,0,0,0 +units=m +no_defs";

    const transformers = {
        _map_Google: () => {
            if (location.href.indexOf("mapmaker") > 0) llz.zoom++;
        },
        _map_BM: () => {
            const urPos = new OpenLayers.LonLat(llz.lon, llz.lat);
            urPos.transform(new OpenLayers.Projection("EPSG:4326"), new OpenLayers.Projection("EPSG:3857"));
            llz.lat = urPos.lat;
            llz.lon = urPos.lon;
        },
        _map_LGIA: () => {
            const transformed = proj4(proj4('EPSG:4326'), proj4(LKS92_PROJ), [parseFloat(llz.lon), parseFloat(llz.lat)]);
            llz.lon = transformed[0];
            llz.lat = transformed[1];
            llz.zoom = 9;
        },
        _map_KDL: () => {
            const transformed = proj4(proj4("EPSG:4326"), proj4(LKS92_PROJ), [parseFloat(llz.lon), parseFloat(llz.lat)]);
            llz.lon = transformed[1];
            llz.lat = transformed[0];
            let zoom = llz.zoom - 12;
            if (zoom > 7) zoom = 7;
            for (let i = 0; i < wmeJM_ArrW2KDL.length; i++) {
                if (zoom == wmeJM_ArrW2KDL[i].w) {
                    zoom = wmeJM_ArrW2KDL[i].r;
                    break;
                }
            }
            llz.zoom = zoom;
        },
        _map_LVM: () => {
            const transformed = proj4(proj4('EPSG:4326'), proj4(LKS92_PROJ), [parseFloat(llz.lon), parseFloat(llz.lat)]);
            llz.lat = transformed[0];
            llz.lon = transformed[1];
            llz.zoom = 14;
        },
        _map_MRY: () => { llz.zoom--; },
        _map_WMFLAB: () => {
            const convertd2dms = (degrees) => {
                const mydegrees = parseInt(degrees);
                let remaining = degrees - mydegrees;
                const myminutes = parseInt(remaining * 60);
                remaining = (remaining * 60) - myminutes;
                const myseconds = Math.round(remaining * 60 * 10) / 10;
                return {d: mydegrees, m: myminutes, s: myseconds};
            };
            const la = convertd2dms(llz.lat);
            const lo = convertd2dms(llz.lon);
            llz.lat = `${la.d}_${la.m}_${la.s}`;
            llz.lon = `${lo.d}_${lo.m}_${lo.s}`;
            llz.zoom = Math.pow(2, 12 - llz.zoom) * 100000;
        },
        _map_OSV: () => { if (llz.zoom > 18) llz.zoom = 18; },
        _map_KADUA: () => {
            const urPos = new OpenLayers.LonLat(llz.lon, llz.lat);
            urPos.transform(new OpenLayers.Projection("EPSG:4326"), new OpenLayers.Projection("EPSG:900913"));
            llz.lat = urPos.lat;
            llz.lon = urPos.lon;
        },
        _map_REGLT: () => {
            const transformed = proj4(proj4('EPSG:4326'), proj4('EPSG:3346'), [parseFloat(llz.lon), parseFloat(llz.lat)]);
            llz.lon = transformed[1];
            llz.lat = transformed[0];
            let zoom = llz.zoom - 12;
            if (zoom > 7) zoom = 7;
            for (let i = 0; i < wmeJM_ArrW2KDL.length; i++) {
                if (zoom == wmeJM_ArrW2KDL[i].w) {
                    zoom = wmeJM_ArrW2KDL[i].r;
                    break;
                }
            }
            llz.zoom = zoom;
        }
    };

    if (transformers[id]) transformers[id]();

    if (wmeJM_debug) console.log("WME-JumpMaps: Converted to external:", llz);
    return llz;
}

// Post-load processing (currently empty for all sites)
function WmeJM_PostLoadOtherMaps() {
    if (wmeJM_debug) console.log("WME-JumpMaps: PostLoadOtherMaps");
    // Currently no post-load processing needed for any site
}

// Mouse click handler - jumping mechanism
function WmeJM_clickJumpToMaps()
{
    if(wmeJM_debug) console.log("WME-JumpMaps (" + wmeJM_version + "): WmeJM_clickJumpToMaps("+this.id+")");

    var savedSelectedItems=[];
    if (wmeJM_restoreSelected && !(this.id === '_map_WME' || this.id.indexOf("_map_WME_") >= 0 || this.id === '_map_WMEB'))
    {
        if(wmeJM_debug) console.log("WME-JumpMaps (" + wmeJM_version + "): WmeJM_clickJumpToMaps("+this.id+"): save selected");
        for( var i=0; i < W.selectionManager.getSelectedFeatures().length; ++i)
            savedSelectedItems.push(W.selectionManager.getSelectedFeatures()[i].model);
    }

    var llz=WmeJM_GetLLZ();
    if(wmeJM_debug) console.log("WME-JumpMaps (" + wmeJM_version + "): WmeJM_clickJumpToMaps("+this.id+"): llz="+JSON.stringify(llz));

    if (this.id == '_map_WME' || this.id == '_map_WMEB' || this.id.indexOf("_map_WME_") >= 0)
        llz=WmeJM_Convert_Other2WME(llz);
    else
        llz=WmeJM_Convert_WME2Other(this.id,llz);

    if(wmeJM_debug) console.log("WME-JumpMaps (" + wmeJM_version + "): WmeJM_clickJumpToMaps("+this.id+"): wmeJM_Config["+this.id+"]="+JSON.stringify(wmeJM_Config[this.id]));

    var template=(typeof wmeJM_Config[this.id] !== "undefined")?wmeJM_Config[this.id].template:"";
    if (this.id.indexOf("_map_WME_") >= 0)
        template=wmeJM_Config["_map_WME"].template;
    var url=template.replace("{{city}}",llz.city).replace("{{lon}}",llz.lon).replace("{{lat}}",llz.lat).replace("{{zoom}}",llz.zoom) + ((this.id == '_map_WME' || this.id.indexOf("_map_WME_") >= 0 || this.id == '_map_WMEB')?"&marker=yes":"");

    if(wmeJM_debug) console.log("WME-JumpMaps (" + wmeJM_version + "): WmeJM_clickJumpToMaps("+this.id+"): " + url + ', _url'+this.id);

    if(wmeJM_restoreSelected && !(this.id == '_map_WME' || this.id.indexOf("_map_WME_") >= 0 || this.id == '_map_WMEB')) // restore selections
    {
        if(wmeJM_debug) console.log("WME-JumpMaps (" + wmeJM_version + "): WmeJM_clickJumpToMaps("+this.id+"): restore selected");
        setTimeout(function() {if (savedSelectedItems.length > 0){W.selectionManager.select(savedSelectedItems);savedSelectedItems.length=0;}},50);
    }

    if (this.id.indexOf("_map_WME_") >= 0) // If it's something like that, jmlink - we kilay from the url of lats/longs/zooms...
    {
        if (this.getAttribute("jmfrom") === "mapbys") // If it's something like that, jmlink - we kilay from the url of lats/longs/zooms...
        {
            window.open("http://map.nca.by/map.html?xy="+this.getAttribute("jmlink")+"&z=16",'_url_jm'+this.id);
        }
        url=url.split("&")[0]+"&jmlink="+this.getAttribute("jmlink");
    }

    window.open(url,'_url'+this.id);
}

// Add JumpMaps floating menu to WazeMapEditor
function WmeJM_InsertWMEIcon()
{
    if(wmeJM_debug) console.log("WME-JumpMaps (" + wmeJM_version + "): WmeJM_InsertWMEIcon(): "+(document.getElementById('WME.JumpMaps_' + wmeJM_version)?"found":"none"));
    var nod=document.getElementById('WME.JumpMaps_' + wmeJM_version);
    if(nod)
    {
        // clear exist
        nod.innerHTML="";
    }
    else
    {
        // Create new template
        nod=document.createElement("div");
        nod.setAttribute('id', 'WME.JumpMaps_' + wmeJM_version);
        nod.setAttribute('unselectable', 'on');
        var leftPos = wmeJM_leftOffset;
        var topPos = wmeJM_topOffset;
        nod.setAttribute('style', ' font-size: 12px; color: #fff; padding-left: 20px; position:absolute; top:' + topPos + '; left:' + leftPos + '; display:block; background-color:rgba(0,0,0,.7); visibility:' + (wmeJM_hideWindow ? "hidden":"visible") + ';cursor:pointer;');
    }

    nod.innerHTML="";

    // Load checkbox states
    WmeJM_LoadCheckboxStates();

    var innerHTML="";
    for (let i in wmeJM_Config)
    {
        if (["_map_WME","_map_WMEB","_map_LI"].indexOf(i) < 0)
        {
            if (typeof wmeJM_Config[i] !== "undefined" && wmeJM_Config[i].save === 1)
            {
                // Resolve template for this entry (work with either aLinks or wmeJM_Config)
                const template = (typeof aLinks !== 'undefined' && aLinks[i] && aLinks[i].template)
                ? aLinks[i].template
                : (wmeJM_Config[i] ? wmeJM_Config[i].template : '');

                // Compute favicon URL: prefer central helper if available, otherwise derive from template (DuckDuckGo)
                let iconURL = '';
                if (typeof WmeJM_GetFaviconURL === 'function') {
                    iconURL = WmeJM_GetFaviconURL(i, template) || '';
                } else {
                    if (typeof template === 'string') {
                        const m = template.match(/https?:\/\/([^\/]+)/);
                        if (m && m[1]) {
                            const domain = m[1].replace(/^www\./, '');
                            iconURL = 'https://icons.duckduckgo.com/ip3/' + encodeURIComponent(domain) + '.ico';
                        }
                    }
                }

                const iconHTML = iconURL ? '<img src="' + iconURL + '" style="width:12px;height:12px;vertical-align:middle;margin-right:4px;">' : '';

                innerHTML += `<a id="${i}" style="color: #fff; font-size: 11px" title="${wmeJM_Config[i].title}">${iconHTML}${wmeJM_Config[i].name}</a>&nbsp;<span style="opacity:0.4;">|</span>&nbsp;`;
            }
        }
    }

    var main_site=location.hostname === "www.waze.com";
    nod.innerHTML =
        (innerHTML || "")
        + "<a id='_map_LI' style='color: #fff; font-size: 10px' title='Open in LiveMap'>[Live]</a>&nbsp;"
        + "<a id='_map_AB' tp="+(main_site?'A':'B')+" href='' style='color: #fff; font-size: 10px' title='Open in "+(main_site?"Beta":"Main")+" editor' target='" +CreateID()+ (main_site?"b":"a")+"' id='__map_BETAALPA'>["+ (main_site?"&#946;":"&#945;")+"]</a>&nbsp;"
        + "<a href='https://greasyfork.org/en/scripts/481079-wme-jumpmaps' title='WME-JumpMaps_" + wmeJM_version + "' style='color: #fff; font-size: 10px' target='_blank'>[?]</a>&nbsp;";

    document.getElementById('waze-map-container').parentElement.appendChild(nod);

    if(wmeJM_debug) console.log("WME-JumpMaps (" + wmeJM_version + "): WmeJM_InsertWMEIcon(): innerHTML="+document.getElementById('WME.JumpMaps_' + wmeJM_version).innerHTML);

    for (let i in wmeJM_Config)
    {
        if (document.getElementById(i))
            document.getElementById(i).onclick	= WmeJM_clickJumpToMaps;
    }
    document.getElementById('_map_LI').onclick	= WmeJM_clickJumpToMaps;
    document.getElementById('_map_AB').onclick	= function(){
        var permalink="?"+wmer_generate_permalink().split("?")[1];
        if(wmeJM_debug) console.log("_map_AB.click(), permalink=",permalink);
        if(wmeJM_debug) console.log(this.getAttribute('tp'));
        var main_site=this.getAttribute('tp')==='A';
        this.href=(main_site?"https://beta.waze.com/editor":"https://www.waze.com/editor") + permalink;
        if(wmeJM_debug) console.log(this.href);
    };
}

// Insert button for for jumping to Waze
function WmeJM_InsertIcon()
{
    if(wmeJM_debug) console.log("WME-JumpMaps (" + wmeJM_version + "): WmeJM_InsertIcon()");
    var locType=WmeJM_GetLocationType();
    if(wmeJM_debug) console.log("WME-JumpMaps (" + wmeJM_version + "): WmeJM_InsertIcon(): locType="+locType);

    if(locType == "waze")
        return true;

    var result = false;

    var nod=document.createElement(locType === "mry" || locType === "osm" || locType === "google"?"div":(locType === "NM"|| locType === "kadua"?"button":"span"));
    nod.setAttribute('id', 'WME.JumpMaps_' + wmeJM_version);
    window.nod=nod;

    var clsid=
        {
            "balticmaps"	  : {t:1,c:"map_mb"},
            "kdlv"	  : {t:1,c:"social_networks"},
        };

    if(typeof clsid[locType] === 'undefined')
    {
        WmeJM_PostLoadOtherMaps();
        return true;
    }

    var WazeControlAttribution = null;
    if(wmeJM_debug) console.log("WME-JumpMaps (" + wmeJM_version + "): WmeJM_InsertIcon(): clsid[locType]="+JSON.stringify(clsid[locType]));
    if(clsid[locType])
        WazeControlAttribution = clsid[locType].t == 1?document.getElementById(clsid[locType].c):(clsid[locType].t == 0 || clsid[locType].t == 2?document.getElementsByClassName(clsid[locType].c):document.getElementsByTagName(clsid[locType].c));
    if(wmeJM_debug) console.log("WME-JumpMaps (" + wmeJM_version + "): WmeJM_InsertIcon(): WazeControlAttribution="+(typeof WazeControlAttribution),WazeControlAttribution);

    var found00=false;
    if (WazeControlAttribution)
    {
        if (!clsid[locType].t)
        {
            if (WazeControlAttribution.length > 0)
            {
                found00=true;
            }
            else
            {
                if(wmeJM_debug) console.log("WME-JumpMaps (" + wmeJM_version + "): WmeJM_InsertIcon(): wait 1 ",locType,clsid[locType]);
                setTimeout(function() {WmeJM_InsertIcon();},500,this);
                return false;
            }
        }
        else
        {
            if(wmeJM_debug) console.log("WME-JumpMaps (" + wmeJM_version + "): WmeJM_InsertIcon(): found00=true;");
            found00=true;
        }

        if (!found00)
        {
            if(document.readyState != 'complete' && ++wmeJM_countProbe2 < 5)
            {
                if(wmeJM_debug) console.log("WME-JumpMaps (" + wmeJM_version + "): WmeJM_InsertIcon(): wait 2 ",locType,clsid[locType]);
                setTimeout(function() {WmeJM_InsertIcon();},500,this);
                return false;
            }
        }
    }
    else
    {
        if(wmeJM_debug) console.log("WME-JumpMaps (" + wmeJM_version + "): WmeJM_InsertIcon(): wait 3 ",clsid[locType]);
        setTimeout(function() {WmeJM_InsertIcon();},100,this);
        return false;
    }

    if (found00)
    {
        if(wmeJM_debug) console.log("WME-JumpMaps (" + wmeJM_version + "): WmeJM_InsertIcon(): found '", (clsid[locType].t?WazeControlAttribution:WazeControlAttribution[0]));
        if(wmeJM_debug) console.log("WME-JumpMaps (" + wmeJM_version + "): WmeJM_InsertIcon(): document.readyState=" + document.readyState);

        // Note: Since we removed the icon, these cases may need adjustment
        // The icon was previously used in these switch cases
        switch(locType)
        {
            case "balticmaps":
                {
                    nod.setAttribute('style', 'margin-top:11px; margin-right:5px; position:absolute; top:0px; left:558px; z-index:1; border-radius:20px; box-shadow: 0px 1px 8px 0px rgba(0, 0, 0, 0.2), 0px 3px 4px 0px rgba(0, 0, 0, 0.14), 0px 3px 3px -2px rgba(0, 0, 0, 0.12);');
                    nod.innerHTML = "<div id='_map_WME' style='cursor: pointer;' title='Open to WME'><img width=40 height=40 src='"+wmeJM_IconWME+"'></div>";
                    WazeControlAttribution.parentElement.insertBefore(nod, WazeControlAttribution.nextSibling);
                    break;
                }
            case "kdlv":
                {
                    WazeControlAttribution.insertAdjacentHTML('afterbegin', '<a id="_map_WME" class="waze" target="_blank" title="Open in WME"></a>');
                    document.styleSheets[0].insertRule("#social_networks a.waze { background:url("+wmeJM_IconWME+" );background-size: 100% 100%; right: 356px}", 0);
                    break;
                }
        }

        if (document.getElementById('_map_WME'))
        {
            document.getElementById('_map_WME').onclick	 = WmeJM_clickJumpToMaps;
            result=true;
        }
        if (document.getElementById('_map_WMEB'))
        {
            document.getElementById('_map_WMEB').onclick	 = WmeJM_clickJumpToMaps;
            result=true;
        }
    }
    else
    {
        if(wmeJM_debug) console.log("WME-JumpMaps (" + wmeJM_version + "): WmeJM_InsertIcon(): ELSE typeof WazeControlAttribution="+(typeof WazeControlAttribution)+", clsid[locType].t="+clsid[locType].t +", WazeControlAttribution.length="+WazeControlAttribution.length+" ["+(WazeControlAttribution && (clsid[locType].t || WazeControlAttribution.length >= 1))+"]");
    }

    if (result)
        WmeJM_PostLoadOtherMaps();

    return result;
}

function createMinimap() {
    if (minimapInstance) return; // Already created
    if (!document.getElementById('leaflet-minimap-css')) {
        const link = document.createElement('link');
        link.id = 'leaflet-minimap-css';
        link.rel = 'stylesheet';
        link.href = 'https://unpkg.com/[email protected]/dist/leaflet.css';
        document.head.appendChild(link);
    }
    const minimapContainer = document.createElement('div');
    minimapContainer.id = 'latvia-minimap';
    minimapContainer.style.cssText = `position: absolute;bottom: 65px;left: 10px;width: 220px;height: 145px;z-index: 1000;border: 1px solid #333;border-radius: 8px;box-shadow: 0 4px 12px rgba(0,0,0,0.3);background: white;overflow: hidden;`;
    const header = document.createElement('div');
    header.style.cssText = `background: #007acc;color: white;font-weight: 400;height: 20px;font-size: 13px;text-align: center;cursor: move;user-select: none;font-family: "Rubik";`;
    function updateLocationHeader() {
        const locationElement = document.querySelector('.location-info');
        if (locationElement && locationElement.textContent) {
            header.textContent = locationElement.textContent;
        }
    }
    setTimeout(updateLocationHeader, 1000);
    setInterval(updateLocationHeader, 2000);
    minimapContainer.appendChild(header);
    const mapDiv = document.createElement('div');
    mapDiv.id = 'minimap-leaflet';
    mapDiv.style.cssText = `width: 100%;height: calc(100% - 20px);`;
    minimapContainer.appendChild(mapDiv);
    document.querySelector('#map').appendChild(minimapContainer);
    makeDraggable(minimapContainer, header);
    minimapInstance = minimapContainer;
    setTimeout(() => {
        const minimap = L.map('minimap-leaflet', {zoomControl: false,attributionControl: false,dragging: false,scrollWheelZoom: false,doubleClickZoom: false,boxZoom: false,keyboard: false,touchZoom: false});
        L.tileLayer('https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png', {maxZoom: 18}).addTo(minimap);
        const latviaBounds = [[55.67, 20.97], [58.09, 28.24]];
        minimap.fitBounds(latviaBounds);
        const viewportMarker = L.circleMarker([56.95, 24.1], {radius: 5,color: '#ffffff',fillColor: '#0099ff',fillOpacity: 0.8,weight: 1}).addTo(minimap);
        function updateViewport() {
            try {
                if (!W.map) return;
                const center = W.map.getCenter();
                if (!center) return;
                const fromProj = new OpenLayers.Projection("EPSG:900913");
                const toProj = new OpenLayers.Projection("EPSG:4326");
                const centerPoint = new OpenLayers.LonLat(center.lon, center.lat).transform(fromProj, toProj);
                viewportMarker.setLatLng([centerPoint.lat, centerPoint.lon]);
            } catch (e) {
                console.error('Error updating viewport:', e);
            }
        }
        W.map.events.register('moveend', null, updateViewport);
        W.map.events.register('zoomend', null, updateViewport);
        updateViewport();
        minimap.on('click', function(e) {
            const latlng = e.latlng;
            const fromProj = new OpenLayers.Projection("EPSG:4326");
            const toProj = new OpenLayers.Projection("EPSG:900913");
            const point = new OpenLayers.LonLat(latlng.lng, latlng.lat).transform(fromProj, toProj);
            W.map.setCenter(point);
        });
    }, 500);
}

function destroyMinimap() {
    if (minimapInstance) {
        minimapInstance.remove();
        minimapInstance = null;
    }
}

function toggleMinimap() {
    if (wmeJM_showMinimap) {
        createMinimap();
    } else {
        destroyMinimap();
    }
}

function makeDraggable(element, handle) {
    let pos1 = 0, pos2 = 0, pos3 = 0, pos4 = 0;
    handle.onmousedown = dragMouseDown;
    function dragMouseDown(e) {
        e.preventDefault();
        pos3 = e.clientX;
        pos4 = e.clientY;
        document.onmouseup = closeDragElement;
        document.onmousemove = elementDrag;
    }
    function elementDrag(e) {
        e.preventDefault();
        pos1 = pos3 - e.clientX;
        pos2 = pos4 - e.clientY;
        pos3 = e.clientX;
        pos4 = e.clientY;
        element.style.top = (element.offsetTop - pos2) + "px";
        element.style.left = (element.offsetLeft - pos1) + "px";
        element.style.bottom = 'auto';
        element.style.right = 'auto';
    }
    function closeDragElement() {
        document.onmouseup = null;
        document.onmousemove = null;
    }
}

function WmeJM_onWazeTabReady()
{
    document.getElementById("wmejm_cfg_resetConfig").onclick = function(){
        setTimeout(function() {
            if(confirm("Reset config for WME-JumpMaps?"))
            {
                let d=document.getElementById(CreateID());
                d.parentNode.removeChild(d);
                d=document.getElementById("pwmejumpmaps");
                d.parentNode.removeChild(d);

                localStorage.removeItem("WMEJumpMapsCheckboxStates");
                for(var i in wmeJM_Config)	{ wmeJM_Config[i].save = wmeJM_Config0[i].save; }

                localStorage.removeItem("WMEJumpMapsDebug");
                wmeJM_debug=false;

                WmeJM_InsertWMEIcon();
                WmeJM_InitConfig();
            }
        },100,this);
        return false;
    };

    document.getElementById("wmejm_cfg_debug").onclick = function(){wmeJM_debug=this.checked;localStorage.setItem("WMEJumpMapsDebug",wmeJM_debug?"1":"0");};
    document.getElementById("wmejm_cfg_debug").checked = wmeJM_debug;

    document.getElementById("wmejm_cfg_savedsel").onclick = function(){
        wmeJM_restoreSelected=this.checked;
        localStorage.setItem("WMEJumpMapsRestoreSelected",wmeJM_restoreSelected?"1":"0");
    };
    document.getElementById("wmejm_cfg_savedsel").checked = wmeJM_restoreSelected;

    document.getElementById("wmejm_cfg_window_hide").onclick = function(){
        wmeJM_hideWindow=this.checked;
        localStorage.setItem("WMEJumpMapsHideWindow",wmeJM_hideWindow?"1":"0");
        document.getElementById('WME.JumpMaps_' + wmeJM_version).style.visibility = wmeJM_hideWindow ? "hidden":"visible";
    };
    document.getElementById("wmejm_cfg_window_hide").checked = wmeJM_hideWindow;

    document.getElementById("wmejm_cfg_resetWPos").onclick = function(){
        localStorage.setItem("WMEJumpMapsTopOffset", wmeJM_defaultTopOffset);
        localStorage.setItem("WMEJumpMapsLeftOffset", wmeJM_defaultLeftOffset);

        document.getElementById('WME.JumpMaps_' + wmeJM_version).style.left = wmeJM_defaultLeftOffset;
        document.getElementById('WME.JumpMaps_' + wmeJM_version).style.top = wmeJM_defaultTopOffset;

        wmeJM_topOffset = wmeJM_defaultTopOffset;
        wmeJM_leftOffset = wmeJM_defaultLeftOffset;
    };

    // Load checkbox states
    WmeJM_LoadCheckboxStates();

    for(let i in wmeJM_Config)
    {
        if (["_map_WME","_map_WMEB","_map_LI"].indexOf(i) >= 0)
            continue;

        document.getElementById("wmejm_cfg_"+i).checked = wmeJM_Config[i].save;
        document.getElementById("wmejm_cfg_"+i+"_chklab").innerHTML="&nbsp;"+wmeJM_Config[i].title;

        // Visibility handler - only save checkbox state
        document.getElementById("wmejm_cfg_"+i).onchange = function(){
            var id=this.getAttribute('data');
            wmeJM_Config[id].save = this.checked ? 1 : 0;
            WmeJM_SaveCheckboxStates(); // Save only checkbox states
            WmeJM_InsertWMEIcon();
        };
    }

    // ADD MINIMAP HANDLER HERE:
    document.getElementById("wmejm_cfg_minimap").onclick = function(){
        wmeJM_showMinimap=this.checked;
        localStorage.setItem("WMEJumpMapsShowMinimap",wmeJM_showMinimap?"1":"0");
        toggleMinimap();
    };
    wmeJM_showMinimap = __GetLocalStorageItem("WMEJumpMapsShowMinimap", 'bool', true);
    document.getElementById("wmejm_cfg_minimap").checked = wmeJM_showMinimap;
    if (wmeJM_showMinimap) {
        setTimeout(createMinimap, 1000);
    }
}

function WmeJM_InitConfig()
{
    if(wmeJM_debug) console.log("WME-JumpMaps (" + wmeJM_version + "): WmeJM_InitConfig(): "+document.getElementById(CreateID()));
    if(!document.getElementById(CreateID()))
    {
        var srsCtrl = document.createElement('section');
        srsCtrl.id = CreateID();

        var padding="padding:5px 9px";

        // -------------------------------
        var strFormCode = ''
        +'<div class="side-panel-section">'
        +'<h4>WME JumpMaps LV</h4>'
        +'<form class="attributes-form side-panel-section" action="javascript:return false;">'
        +'<div class="form-group">'
        +'<label class="control-label">Map services:</label>'
        +'<div class="controls">';

        // Load checkbox states first
        WmeJM_LoadCheckboxStates();

        for(var i in wmeJM_Config)
        {
            if (["_map_WME","_map_WMEB","_map_LI"].indexOf(i) >= 0)
            {
                continue;
            }
            var id = i;
            var title = wmeJM_Config[id].title;
            var save = wmeJM_Config[id].save;

            // extract base URL from template for hover tooltip
            var baseURL = '';
            if (wmeJM_Config[id].template) {
                var m = wmeJM_Config[id].template.match(/https?:\/\/[^\/]+/);
                if (m) baseURL = m[0];
            }
            // compute favicon for settings panel
            var iconHTML = '';
            var fav = WmeJM_GetFaviconURL(id, wmeJM_Config[id].template);
            if (fav) {
                iconHTML = '<img src="' + fav + '" style="width:16px;height:16px;margin-right:-6px;">';
            }

            strFormCode += ''
                +'<div class="form-group" style="margin-bottom:4px;">'
                +'<div class="control-label" '
                +'style="display:flex; align-items:center; gap:6px; '
                +'font-weight:normal; font-size:14px; line-height:16px;">'
                +'<input data="'+id+'" name="wmejm_cfg_'+id+'" '
                +'id="wmejm_cfg_'+id+'" type="checkbox" ' + (save?'checked':'') + '>'
                + iconHTML
                +'<label id="wmejm_cfg_'+id+'_chklab" for="wmejm_cfg_'+id+'" '
                +'style="cursor:pointer; font-weight:normal; margin:0;" title="' + baseURL + '">'
                + title +
                '</label>'
                +'</div>'
                +'</div>';
        }

        strFormCode += ''
            +'</div>'
            +'</div>';

        // -------------------------------
        strFormCode += ''
            +'<div class="form-group">'
            +'<label class="control-label">Other settings:</label>'
            +'<div class="controls" style="margin:0; line-height:16px;">'
            +'<div style="display:none">'
            +'<input type="checkbox" id="wmejm_cfg_savedsel">'
            +'<label for="wmejm_cfg_savedsel" style="cursor:pointer;margin-left:6px;font-weight:normal;" title="Restore selected">'
            +'<i class="fa fa-clone"></i> Restore selected'
            +'</label><br>'
            +'</div>'
            +'<input type="checkbox" id="wmejm_cfg_minimap">'
            +'<label for="wmejm_cfg_minimap" style="cursor:pointer;margin-left:6px;font-weight:normal;margin-bottom:0px" title="Show minimap">'
            +'<i class="fa fa-map"></i> Show minimap (Latvia)'
            +'</label><br>'
            +'<input type="checkbox" id="wmejm_cfg_debug">'
            +'<label for="wmejm_cfg_debug" style="cursor:pointer;margin-left:6px;font-weight:normal;margin-bottom:0px" title="Debug script">'
            +'<i class="fa fa-bug"></i> Debug script (console logging)'
            +'</label><br>'
            +'<input type="checkbox" id="wmejm_cfg_window_hide">'
            +'<label for="wmejm_cfg_window_hide" style="cursor:pointer;margin-left:6px;font-weight:normal;" title="Hide Window">'
            +'<i class="fa fa-eye-slash"></i> Hide Window'
            +'</label><br>'
            +'<div style="display:none">'
            +'<button id="wmejm_cfg_resetConfig" class="btn btn-default" style="font-size:9px;padding:5px 9px;margin:4px 0;" title="Reset config!">'
            +'<i class="fa fa-recycle"></i> Reset config'
            +'</button><br>'
            +'<button id="wmejm_cfg_resetWPos" class="btn btn-default" style="font-size:9px;padding:5px 9px;" title="Reset window position!">'
            +'<i class="fa fa-recycle"></i> Reset window position'
            +'</button>'
            +'</div>'
            +'</div>'
            +'</div>';

        srsCtrl.className = "tab-pane";
        srsCtrl.innerHTML = strFormCode;
        WazeWrap.Interface.Tab('JumpMaps', strFormCode, WmeJM_onWazeTabReady);
    }
    else
        if(wmeJM_debug) console.log("WME-JumpMaps (" + wmeJM_version + "): WmeJM_InitConfig(): not found '"+CreateID()+"'");
}

function WmeJM_FakeLoad()
{
    if(wmeJM_debug) console.log("WME-JumpMaps (" + wmeJM_version + "): start WmeJM_FakeLoad(): this"+ this);
    var loctype=WmeJM_GetLocationType();
    if(wmeJM_debug) console.log("WME-JumpMaps (" + wmeJM_version + "): start WmeJM_FakeLoad(): loctype="+loctype);

    if(window.document.getElementById('WME.JumpMaps_' + wmeJM_version)) // if THIS is there, then the other checks have passed
    {
        if(wmeJM_debug) console.log("WME-JumpMaps (" + wmeJM_version + "): WmeJM_FakeLoad(): FOUND WME.JumpMaps_" + wmeJM_version + "!!!. Done");
        return;
    }

    if (loctype === "waze")
    {
        if (typeof Waze === "undefined")
        {
            if(wmeJM_debug) console.log("WME-JumpMaps (" + wmeJM_version + "): WmeJM_FakeLoad(): wait W. Wait 500ms");
            setTimeout(WmeJM_FakeLoad,500);
            return;
        }
        if (typeof W.selectionManager === "undefined")
        {
            if(wmeJM_debug) console.log("WME-JumpMaps (" + wmeJM_version + "): WmeJM_FakeLoad(): wait W.selectionManager. Wait 500ms");
            setTimeout(WmeJM_FakeLoad,500);
            return;
        }
        if (document.getElementsByClassName('olControlAttribution')[0] === null)
        {
            if(wmeJM_debug) console.log("WME-JumpMaps (" + wmeJM_version + "): WmeJM_FakeLoad(): wait waze olControlAttribution. Wait 500ms");
            setTimeout(WmeJM_FakeLoad,500);
        }
        if (!WazeWrap?.Ready) {
            if(wmeJM_debug) console.log("WME-JumpMaps (" + wmeJM_version + "): WmeJM_FakeLoad(): wait WazeWrap. Wait 500ms");
            setTimeout(WmeJM_FakeLoad,500);
            return;
        }
    }

    if (document.readyState != 'complete' && ++wmeJM_countProbe2 < 5)
    {
        if(wmeJM_debug) console.log("WME-JumpMaps (" + wmeJM_version + "): WmeJM_FakeLoad(): document.readyState != 'complete', wmeJM_countProbe="+wmeJM_countProbe2+". Wait 3000ms");
        setTimeout(WmeJM_FakeLoad,3000);
        return;
    }

    // Further Initialization
    if (loctype === "waze")
    {
        // What if we jumped from external and if there is a "jmlink" component in the URL, then center the editor at the specified coordinates (EPSG:900913).
        if (__getQueryString(location.href, "jmlink") != -1)
        {
            var jmlink=__getQueryString(location.href, "jmlink").split(",");
            var urPos=new OpenLayers.LonLat(jmlink[1],jmlink[0]);
            urPos.transform(new OpenLayers.Projection("EPSG:900913"),new OpenLayers.Projection("EPSG:4326"));
            var xy = OpenLayers.Layer.SphericalMercator.forwardMercator(parseFloat(urPos.lon), parseFloat(urPos.lat));
            W.map.setCenter(xy);
        }
        WmeJM_InsertWMEIcon();
        WmeJM_InitConfig();
        WmeJM_initBindKey();
    }
    else
    {
        // Other services...
        if(document.getElementById('WME.JumpMaps_' + wmeJM_version) === null && !WmeJM_InsertIcon())
        {
            if(++wmeJM_countProbe < 8) //  8 attempts
            {
                let cls="";
                if(wmeJM_debug) console.log("WME-JumpMaps (" + wmeJM_version + "): not other found '"+cls+"'. wmeJM_countProbe="+wmeJM_countProbe+". Wait 5000ms");
                setTimeout(WmeJM_FakeLoad,5000);
                return;
            }
        }
        //WmeJM_PostLoadOtherMaps();
    }
}

function __GetLocalStorageItem(Name,Type,Def,Arr)
{
    var tmp0=localStorage.getItem(Name);
    if (tmp0)
    {
        switch(Type)
        {
            case 'string':
                break;
            case 'bool':
                tmp0=(tmp0 === "true" || tmp0 === "1")?true:false;
                break;
            case 'int':
                tmp0=!isNaN(parseInt(tmp0))?parseInt(tmp0):0;
                break;
            case 'arr':
                if (tmp0.length > 0)
                    if(!Arr[tmp0])
                        tmp0=Def;
                break;
        }
    }
    else
        tmp0=Def;
    return tmp0;
}

function WmeJM_clickJumpToMapsArg()
{
    if ((typeof arguments[0]) === "object")
    {
        var o=document.getElementById(arguments[0].id);
        if (typeof o !== "undefined")
        {
            if (arguments[0].save)
                o.click();
        }
    }
}

function WmeJM_initBindKey()
{
    if(wmeJM_debug) console.log("WmeJM_initBindKey()");
    if(!W || !W.model || !I18n || !W.accelerators || !W.model.countries || !W.model.countries.top) {
        setTimeout(WmeJM_initBindKey, 500);
        return;
    }

    var Config =[];

    for(let i in wmeJM_Config)
    {
        Config.push({handler: 'WME-JumpMaps'+i,  title: wmeJM_Config[i].title,  func: WmeJM_clickJumpToMapsArg, key:-1, arg:{id:i,save:wmeJM_Config[i].save}});
    }

    for(let i=0; i < Config.length; ++i)
    {
        // WMEKSRegisterKeyboardShortcut would need to be defined or this function needs adjustment
        // For now, commented out as it's not essential for basic functionality
        // WMEKSRegisterKeyboardShortcut('WME-JumpMaps', 'WME-JumpMaps', Config[i].handler, Config[i].title, Config[i].func, Config[i].key, Config[i].arg);
    }

    // WMEKSLoadKeyboardShortcuts would need to be defined
    // WMEKSLoadKeyboardShortcuts('WME-JumpMaps');

    window.addEventListener("beforeunload", function() {
        // WMEKSSaveKeyboardShortcuts would need to be defined
        // WMEKSSaveKeyboardShortcuts('WME-JumpMaps');
    }, false);
}

// Script launcher
function WmeJM_bootstrap()
{
    console.log("WME-JumpMaps Lite (" + wmeJM_version + "): WmeJM_bootstrap()");

    wmeJM_Config = cloneConfig(wmeJM_Config0);

    wmeJM_debug				= __GetLocalStorageItem("WMEJumpMapsDebug",'bool',false);
    wmeJM_restoreSelected	= __GetLocalStorageItem("WMEJumpMapsRestoreSelected",'bool',false);
    wmeJM_around			= __GetLocalStorageItem("WMEJumpMapsAround",'bool',false);
    wmeJM_hideWindow		= __GetLocalStorageItem("WMEJumpMapsHideWindow",'bool',false);
    wmeJM_topOffset		    = __GetLocalStorageItem("WMEJumpMapsTopOffset",'string', wmeJM_defaultTopOffset);
    wmeJM_leftOffset		= __GetLocalStorageItem("WMEJumpMapsLeftOffset",'string', wmeJM_defaultLeftOffset);

    // Load checkbox states
    WmeJM_LoadCheckboxStates();

    setTimeout(function() {WmeJM_FakeLoad();},(WmeJM_GetLocationType() === "YM")?3000:500,this);
}

WmeJM_bootstrap();