Greasy Fork is available in English.

WME KeepMyLayers (Beta)

Resets WME layers to your preferred state plus other fancy stuff. -- Briefly broken for new users, but now fully operational (for most)!

2016-01-16 يوللانغان نەشرى. ئەڭ يېڭى نەشرىنى كۆرۈش.

// ==UserScript==
// @name            WME KeepMyLayers (Beta)
// @description     Resets WME layers to your preferred state plus other fancy stuff. -- Briefly broken for new users, but now fully operational (for most)!
// @namespace       https://greasyfork.org/users/11629-TheLastTaterTot
// @version         0.4.1.2
// @include         https://editor-beta.waze.com/*editor/*
// @include         https://www.waze.com/*editor/*
// @exclude         https://www.waze.com/user/*editor/*
// @author          TheLastTaterTot
// @grant           GM_getValue
// @grant           GM_setValue
// @grant           GM_deleteValue
// @run-at          document-start
// @icon            
// ==/UserScript==

(function () {
    function kmllog(str, _) {
        console.log('WMEKMLayers:', str);
    }

    var kmlPLlayers = '',
        kmlPLhref = false,
        kmlPLhost = false,
        kmlPLpath = false,
        kmlayersBetaChk;

    if (localStorage.WME_KMLSettings) {
        var kmlayersStartupChk = [(localStorage.WME_KMLSettings.indexOf('&x') !== -1), (
            localStorage.WME_KMLSettings.indexOf('&b') !== -1), (localStorage.WME_KMLSettings
            .indexOf('&l') !== -1)];


        if (location.href.lastIndexOf('&kmlayers') === -1) { //not a layers-removal redirect
            //if (!kmlayersStartupChk[0] || kmlayersStartupChk[1] || kmlayersStartupChk[2]) {
            if (!kmlayersStartupChk[0]) { // allow autocheck layers at startup
                kmlPLhref = location.href.match(/&layers=(\d+)/);
                if (kmlPLhref && kmlPLhref[1]) {
                    kmlPLlayers = ('&kmlayers=' + kmlPLhref[1]);
                    kmlPLhref = location.href.replace(kmlPLhref[0], '');
                } // else PL has no layers

                setTimeout(function () {
                    if (localStorage.WME_KeepMyLayers_lV) {
                        localStorage.layerVisibility = localStorage.WME_KeepMyLayers_lV;
                    } // children
                    if (localStorage.WME_KeepMyLayers_lF) {
                        sessionStorage.layerFilters_orig = localStorage.layerFilters; // save a duplicate as backup
                        localStorage.layerFilters = localStorage.WME_KeepMyLayers_lF;
                    }
                }, 0);
            }
        }

        //if (kmlayersBetaChk === true || kmlayersBetaChk === false || kmlayersLangChk === true) {
        if (kmlayersStartupChk[1]) { // beta toggle check //&& location.href.substr(-18).indexOf('&b') === -1
            kmlayersBetaChk = GM_getValue("WMEKMLayers_Beta");

            if (kmlayersBetaChk === true && //if set to beta-editor
                location.host.indexOf('editor-b') === -1) { //if PL is not editor-beta.waze.com
                kmlPLhost = 'https://editor-beta.waze.com';
            } else if (kmlayersBetaChk === false && //if set to production editor
                location.host.indexOf('www.waze') === -1) { //if PL is not www.waze.com
                kmlPLhost = 'http://www.waze.com';
            }
        }

        if (kmlayersStartupChk[2]) { // language check
            if (GM_getValue("WMEKMLayers_Lang") === true) { //if set to filter out language
                if (location.pathname !== '/editor/') {
                    kmlPLpath = '/editor/';
                }
            }
        }

        switch (true) {
            case (!kmlPLhost && !kmlPLpath && !kmlPLhref): //no modifications are necessary
                break;
            default:
                if (!kmlPLhost) kmlPLhost = location.origin;
                if (!kmlPLpath) kmlPLpath = location.pathname;
                if (!kmlPLhref) kmlPLhref = location.href;

                //mllog(location.origin, '/');
                //kmllog(location.pathname, '/');           //kmllog('Redirecting to ' + kmlPLhost + kmlPLpath + location.href.substring(
                //	location.href.indexOf('?'), location.href.length), '/');
                location.replace(kmlPLhost + kmlPLhref.substr(kmlPLhref.indexOf(kmlPLpath)) +
                    kmlPLlayers);
        }
    }

    //==========================================================================
    var resetLocalWithKMLayers = function () {
        if (localStorage.WME_KeepMyLayers_lV) {
            //sessionStorage.layerVisibility_orig = localStorage.layerVisibility; // save a duplicate as backup
            localStorage.layerVisibility = localStorage.WME_KeepMyLayers_lV;
        } // children
        if (localStorage.WME_KeepMyLayers_lF) {
            sessionStorage.layerFilters_orig = localStorage.layerFilters; // save a duplicate as backup
            localStorage.layerFilters = localStorage.WME_KeepMyLayers_lF;
        }
    };

    //==========================================================================
    var KeepMyLayers = function () {
        var kml = Array(10).fill(0), //counter
            kml_W_map_layers = unsafeWindow.Waze.map.layers,
            $ = unsafeWindow.$,
            myKMLayers, kmlVersion = '0.4';

        // ---------------------------------------------------------------------
        var getSavedKMLayers = function () {
            //kmllog('getSavedKMLayers()', '/');
            var myKMLayers;

            if (typeof myKMLayers !== "undefined") {
                if (myKMLayers.visibleInLayersMenu !== undefined) {
                    //kmllog('Replaced myKMLayers in memory with saved from disk.', '/');
                    myKMLayers = JSON.parse(localStorage.WME_KeepMyLayers);
                    myKMLayers.reset = false;
                    return myKMLayers;
                } else {
                    return false;
                }
            } else if (localStorage.WME_KeepMyLayers) {
                //kmllog('Found some saved settings to load!','/');
                myKMLayers = JSON.parse(localStorage.WME_KeepMyLayers);
                myKMLayers.reset = false;

                if (myKMLayers.visibleInLayersMenu !== undefined) { //check that required object key is present
                    /* ~~~~~~ TEMP ~~~~~~~ */
                    if (myKMLayers.visibleInLayersMenu.constructor !== Array) {
                        var tempHoldingVar = JSON.parse(JSON.stringify(myKMLayers.visibleInLayersMenu));
                        myKMLayers.visibleInLayersMenu = [tempHoldingVar];
                        myKMLayers.layerSetNames = ['My default layers'];
                        myKMLayers.idx = 0;
                    }
                    /* ~~~~~~ TEMP ~~~~~~~ */
                    return myKMLayers;
                } else {
                    return false;
                }
            } else { // myKMLayers has not been saved yet
                /* TODO: Call to popup of instructions for saving default layers set will maybe go here */
                return false;
            }
        };


        // W.map.layers.forEach(function(a,i){console.debug('[' + i + ']', a["name"] + ':', a["visibility"])});
        var getWazeMapLayersFromSwitcher = function (wazeMapLayers) { // get OL layers that show up under the WME Layer switcher panel
            var kml_lay = wazeMapLayers.length,
                kml_layerName, kml_layerUniqName,
                kml_layerSwitcher = {
                    accelerator: {},
                    uniqueName: {}
                };

            while (kml_lay--) {
                kml_layerName = wazeMapLayers[kml_lay].accelerator;
                kml_layerUniqName = wazeMapLayers[kml_lay].uniqueName;
                if (kml_layerName !== undefined) { //  accelerator is a marker with high specificity and selectivity of whether the layer appears in the layer switcher menu
                    if (kml_layerUniqName !== undefined) { // uniqueName is used by WME for resetting layers to last PL
                        kml_layerSwitcher.uniqueName[kml_layerUniqName] = kml_lay;
                        kml_layerSwitcher.accelerator[kml_layerName] = kml_lay;
                    } else {
                        kml_layerSwitcher.accelerator[kml_layerName] = kml_lay;
                    }
                }
            }
            return kml_layerSwitcher;
        };

        // -------------------------------------------------------------------------
        var saveKMLayers = function () {
            //kmllog('saveKMLayers()', '/');
            var kml_j, kml_lname,
                kml_layerSwitcher = getWazeMapLayersFromSwitcher(kml_W_map_layers),
                kml_layerNames = Object.keys(kml_layerSwitcher.accelerator),
                kml_numLayers = kml_layerNames.length,
                visibleInLayersMenu = {};


            // Get names of visible menu layers
            for (kml_j = kml_numLayers; kml_j--;) {
                kml_lname = kml_layerNames[kml_j];
                if (kml_W_map_layers[kml_layerSwitcher.accelerator[kml_lname]].getVisibility()) {
                    visibleInLayersMenu[kml_lname] = true;
                }
            }

            myKMLayers = getSavedKMLayers();

            if (!myKMLayers) {
                myKMLayers = {
                    idx: 0,
                    layerSetNames: ["My default layers"],
                    visibleInLayersMenu: []
                };
            }
            if (myKMLayers.idx === 0) {
                // Save to localStorage
                localStorage.WME_KeepMyLayers_lV = localStorage.layerVisibility;
                if (localStorage.layerFilters) {
                    localStorage.WME_KeepMyLayers_lF = localStorage.layerFilters;
                } else {
                    localStorage.WME_KeepMyLayers_lF =
                        '{"update_requests":' + kml_W_map_layers[kml_layerSwitcher.accelerator.toggleUpdateRequests]
                        .visibility + ',"problems":' +
                        kml_W_map_layers[kml_layerSwitcher.accelerator.toggleMapProblems].visibility +
                        ',"mapProblem":0,"mapUpdateRequest":0,"venue":1}';
                }
            }

            myKMLayers.visibleInLayersMenu[myKMLayers.idx] = visibleInLayersMenu; //save only visible

            localStorage.WME_KeepMyLayers = JSON.stringify(myKMLayers);

            var kmlSaved = document.createElement("div");
            kmlSaved.id = "KMLpopup";
            kmlSaved.style.fontWeight = "bold";
            kmlSaved.style.color = "#DD3300";
            kmlSaved.style.position = "absolute";
            kmlSaved.style.display = "block";
            kmlSaved.style.bottom = "-30px";
            kmlSaved.style.right = "14px";
            kmlSaved.innerHTML = "Saved <";
            document.getElementById("layer-switcher-list").appendChild(kmlSaved)
                //$('#layer-switcher-list').after(
                //    '<div id="KMLpopup" style="font-weight: bold; color: #DD3300; position: absolute; display: block; bottom: -30px; right: 14px">Saved &lt;</div>'
                //);

            document.getElementById('iKMLsaveLayers').classList.remove("kml-icn-nsave");
            //document.getElementById('iKMLsaveLayers').className = document.getElementById('iKMLsaveLayers').className.replace(' kml-icn-nsave','');

            setTimeout(function () {
                kmlSaved.remove();
                //document.getElementById('KMLpopup').remove();
            }, 800);

            setTimeout(function () {
                updateKMLayersMenu(myKMLayers)
            }, 10);
        };

        var enableUndoKMLayersReset = function (kml_layerSwitcher) {
            //kmllog('enableUndoKMLayersReset()', '/');

            var convertLayersToObj = function () {
                var getLayersFromPL = function () {
                    kmlayers = location.href.match(/&kmlayers=(\d*)/);
                    if (kmlayers && kmlayers[1]) {
                        return kmlayers[1];
                    }
                };

                var kml_layerVisibility_orig = {},
                    uniqueLayerVal = [1, 2, 4, 8, 16, 32, 64, 128, 256, 512, 1024, 2048],
                    kml_uniqueName = ['satellite_imagery', 'cities', 'roads', 'gps_points',
                'area_managers', 'landmarks', 'speed_cameras', 'problems',
                'update_requests', 'editable_areas', 'live_users', 'place_updates'
            ],
                    kml_lay = uniqueLayerVal.length,
                    kmlayers = getLayersFromPL();

                if (kmlayers !== undefined && kmlayers !== null) {
                    while (kml_lay--) {
                        if (kmlayers > uniqueLayerVal[kml_lay]) {
                            kmlayers = kmlayers - uniqueLayerVal[kml_lay];
                            kml_layerVisibility_orig[kml_uniqueName[kml_lay]] = true;
                        } else {
                            kml_layerVisibility_orig[kml_uniqueName[kml_lay]] = false;
                        }
                    }
                    sessionStorage.layerVisibility_orig = JSON.stringify(
                        kml_layerVisibility_orig);
                    return kml_layerVisibility_orig;
                } else {
                    return JSON.parse(localStorage.WME_KeepMyLayers_lV);
                }
            };

            // the following blocks of code needs cleaning up....
            if (document.getElementById('iKMLtempUndo') !== null &&
                $("a.kml-icn").length < 4) {
                kml[2] = 0; //reset counter

                if (typeof myKMLayers !== "undefined" &&
                    myKMLayers.reset !== undefined && myKMLayers.reset === true) {
                    var kml_layerVisibility_orig = convertLayersToObj(),
                        kml_layerVisibility_kmlayers = JSON.parse(localStorage.WME_KeepMyLayers_lV),
                        kml_layerNames = Object.keys(kml_layerVisibility_orig),
                        kml_numlnames = kml_layerNames.length,
                        kml_toggledLayers = [];

                    kml[3] = 0;
                    while (kml_numlnames--) {
                        if (kml_layerVisibility_kmlayers[kml_layerNames[kml_numlnames]] !== undefined) {
                            if (kml_layerVisibility_orig[kml_layerNames[kml_numlnames]] !==
                                kml_layerVisibility_kmlayers[kml_layerNames[kml_numlnames]]
                            ) {
                                kml_toggledLayers[kml[3]++] = kml_layerNames[kml_numlnames];
                            }
                        }
                    }

                    //TOG = kml_toggledLayers;
                    //KML = myKMLayers;

                    var nToggled = kml_toggledLayers.length;
                    if (nToggled !== 0) {
                        //kmllog('Layer visibilities were reset to preferred default setting.', '/');
                        $('#iKMLtempUndo').wrap(
                            '<a href="javascript:void(0)" class="kml-icn"></a>');
                        $('#iKMLtempUndo').toggleClass("kml-icn-off");

                        document.getElementById("iKMLtempUndo").onclick = function () {
                            var kml_tog = nToggled;
                            while (kml_tog--) {
                                kml_W_map_layers[kml_layerSwitcher.uniqueName[kml_toggledLayers[
                                        kml_tog]]]
                                    .setVisibility(!kml_W_map_layers[kml_layerSwitcher.uniqueName[
                                        kml_toggledLayers[kml_tog]]].getVisibility());
                            }
                        };
                    } // else do nothing, as there is nothing to undo bc layers were not reset
                }
            } else if ($("a.kml-icn").length > 3) {
                kml[2] = 0;
                return true; //button already enabled
            } else if (kml[2]++ < 30) {
                setTimeout(enableUndoKMLayersReset, 200);
            } else {
                console.warn('WMEKMLayers:',
                    'Unable to activate "Undo KMLayers Reset".',
                    'Element #iKMLtempUndo not found on page.');
            }
        };

        //----------------------------------------------------------------------
        /* ["toggleLiveUsers", "toggleRouteCheckerScript", "toggleJunctionAngleInfo",
        "toggleWMEClosestSegment", "toggleRoadClosures", "toggleEditableAreas",
        "toggleUpdateRequests", "toggleMapProblems", "toggleSpeedcameras", "toggleJunctionboxes",
        "togglePlaceupdates", "togglePlaces", "toggleAreaManagers", "toggleGPSpoints",
        "toggleRoads", "toggleCities", "toggleSatelliteImagery"] */
        var checkForReqLayers = function (permalink) {
            switch (true) {
                case permalink.indexOf('&mapUp') !== -1:
                    return "toggleUpdateRequests";
                case permalink.indexOf('&ven') !== -1:
                    return "togglePlaces";
                case permalink.indexOf('&mapPr') !== -1:
                    return "toggleMapProblems";
                case permalink.indexOf('&bigJ') !== -1:
                    return "toggleJunctionboxes";
                case permalink.indexOf('&cam') !== -1:
                    return "toggleSpeedcameras";
                default:
                    return false;
            }
        };

        //----------------------------------------------------------------------
        var resetLayersToSavedKMLayers = function (kml_exclude) {
            //kmllog('resetLayersToSavedKMLayers(' + kml_exclude + ')', '/');
            function resetLayersCatchUp(callback) {
                // check if any W.map.layers are missing compared to saved set
                var maxWait = 20, //4 seconds
                    kml_wait = 0,
                    kml_m = visibleMyKMLayersAccelName.length,
                    kmlWaitAccelName = [];

                while (kml_m--) {
                    // Identify missing layers due to slower loading time
                    if (switcherAccelObj[visibleMyKMLayersAccelName[kml_m]] === undefined) {
                        kmlWaitAccelName.push(visibleMyKMLayersAccelName[kml_m]);
                    }
                }
                //kmllog('Missed setting layers for ' + kmlWaitAccelName);

                var waitForMissedLayers = function () {
                    var kml_layerSwitcher_m = getWazeMapLayersFromSwitcher(kml_W_map_layers),
                        switcherAccelObj_m = kml_layerSwitcher_m.accelerator,
                        kmlWaitAccelName_m = [],
                        kml_m = kmlWaitAccelName.length;

                    //kmllog('start kml_m = ' + kml_m);
                    //kmllog('kml_wait = ' + kml_wait);
                    while (kml_m--) {
                        if (switcherAccelObj_m[kmlWaitAccelName[kml_m]] !== undefined) {

                            kml_W_map_layers[switcherAccelObj_m[kmlWaitAccelName[
                                kml_m]]].setVisibility(
                                visibleMyKMLayersAccelObj[kmlWaitAccelName[kml_m]]);

                            /*kmllog(kmlWaitAccelName[kml_m] + 'visibility set to ' +
                                visibleMyKMLayersAccelObj[kmlWaitAccelName[kml_m]],
                                '/');*/
                        } else {
                            kmlWaitAccelName_m.push(kmlWaitAccelName);
                        }
                    } //while
                    kmlWaitAccelName = kmlWaitAccelName_m;

                    if (kml_wait++ < maxWait && kml_m) {
                        setTimeout(waitForMissedLayers, 200);
                    } else {
                        callback(kml_layerSwitcher_m);
                    }
                };
            }; //resetLayersCatchUp()


            if (typeof myKMLayers !== "undefined" && myKMLayers) {
                var kml_layerSwitcher = getWazeMapLayersFromSwitcher(kml_W_map_layers),
                    switcherAccelObj = kml_layerSwitcher.accelerator,
                    switcherAccelNames = Object.keys(switcherAccelObj),
                    kml_sw = switcherAccelNames.length,
                    kml_lname,
                    visibleMyKMLayersAccelObj = myKMLayers.visibleInLayersMenu[myKMLayers.idx],
                    visibleMyKMLayersAccelName = Object.keys(visibleMyKMLayersAccelObj);

                // Adjust layer visibility if necessary
                if (kml_exclude) visibleMyKMLayersAccelObj[kml_exclude] = true;
                kml[4] = 0;
                while (kml_sw--) {
                    kml_lname = switcherAccelNames[kml_sw];
                    kml_W_map_layers[switcherAccelObj[kml_lname]].setVisibility(!!visibleMyKMLayersAccelObj[kml_lname]);
                }

                myKMLayers.reset = true;

                resetLayersCatchUp(enableUndoKMLayersReset);
                //setTimeout(enableUndoKMLayersReset, 500);
            }
        };

        // ---------------------------------------------------------------------
        var applyAdditionalKMLSettings = function () {
            //kmllog('applyAdditionalKMLSettings()', '/');
            // ['&x', '&b', '&l', '&1', '&5', '&2']; //disable, beta, lang, city, am, roads
            if (localStorage.WME_KMLSettings.indexOf('&1') !== -1) {
                kml_W_map_layers[1].setOpacity(1);
                //kmllog('Opacity of Cities layer increased.', '/');
            } else {
                kml_W_map_layers[1].setOpacity(0.5);
                //kmllog('Opacity of Cities layer at default.', '/');
            }

            if (localStorage.WME_KMLSettings.indexOf('&5') !== -1) {
                kml_W_map_layers[5].setOpacity(0.6);
                //kmllog('Opacity of AM layer decreased.', '/');
            } else {
                kml_W_map_layers[5].setOpacity(1);
                //kmllog('Opacity of AM layer at default.', '/');
            }

            if (localStorage.WME_KMLSettings.indexOf('&2') !== -1) {
                kml_W_map_layers[2].setOpacity(0.8);
                //kmllog('Opacity of Roads layer decreased.', '/');
            } else {
                kml_W_map_layers[2].setOpacity(1);
                //kmllog('Opacity of Roads layer at default.', '/');
            }
        };

        // ---------------------------------------------------------------------
        var runSecondaryKMLayersCheck = function () {
            //kmllog('runSecondaryKMLayersCheck()', '/');
            if (localStorage.WME_KMLSettings === undefined) {
                localStorage.WME_KMLSettings = ':';
            }

            if (localStorage.WME_KMLSettings.indexOf('&x') !== -1) { // user has set WMEKMLayers to be disabled
                console.log('WMEKMLayers:',
                    'Autocheck of KeepMyLayers at startup is disabled (2).');
                return false;
            } else {
                var myKMLayers = getSavedKMLayers();
                //kmllog('Secondary check completed. Any saved layers will be applied (2).', '/');
                if (myKMLayers) {
                    return myKMLayers;
                } else {
                    return false;
                }
            }
        };

        //======================================================================
        myKMLayers = runSecondaryKMLayersCheck();
        //KML = myKMLayers;
        if (myKMLayers) {
            resetLayersToSavedKMLayers(checkForReqLayers(location.href));
        }
        applyAdditionalKMLSettings();

        var kmlayersBetaChk = GM_getValue("WMEKMLayers_Beta"),
            kmlayersLangChk = GM_getValue("WMEKMLayers_Lang");

        //======================================================================
        var userResetOfLayersToSavedKMLayers = function () {
            //kmllog('userResetOfLayersToSavedKMLayers()', '/');
            if (getSavedKMLayers()) {
                resetLayersToSavedKMLayers(false);
            } else {
                //kmllog('Nothing to reset.', '/');
                return false;
            }
        };

        // Dropdown menu for selecting layers
        var updateKMLayersMenu = function (myKMLayers) {

            var numKMLsets = myKMLayers.layerSetNames.length,
                htmlText = '',
                selStatus, kmlset = -1;

            while (++kmlset < numKMLsets) {
                (kmlset === myKMLayers.idx) ? selStatus = 'active ': selStatus = '';

                htmlText += '<li>' +
                    '<a id="liKMLayersSet" value=' + kmlset + ' name="kmlSet_' + kmlset + '" ' +
                    'class = "kml-layers-menu dropdown-item ' + selStatus +
                    '" ' + 'href="javascript:void(0)">' +
                    '<div class="">' +
                    myKMLayers.layerSetNames[kmlset] + ' </div> ' +
                    '</a></li>'
                htmlText +=
                    '<li role="separator" class="divider kml-layers-menu"></li>';

            }
            document.querySelector('ul[class*=kml-layers-menu]').innerHTML = htmlText;

            kmlset = -1;
            while (++kmlset < numKMLsets) {
                document.querySelectorAll('a[class*=kml-layers-menu]')[kmlset].onclick = function () {
                    document.getElementsByName('kmlSet_' + myKMLayers.idx)[0].classList.remove('active');
                    kmlThisName = this.name;
                    //kmllog(kmlThisName)
                    document.getElementsByName(kmlThisName)[0].classList.add('active');
                    myKMLayers.idx = parseInt(document.getElementsByName(kmlThisName)[0].getAttribute("value"));
                    setTimeout(userResetOfLayersToSavedKMLayers, 150);
                };
            }
            //$("ul.kml-layers-menu").html(htmlText);
            //$("a.kml-layers-menu").click(function () {
        };

        //----------------------------------------------------------------------
        var getGMBetaValue = function (callback) {
            setTimeout(function () {
                callback(GM_getValue("WMEKMLayers_Beta"))
            }, 0);
        };

        //----------------------------------------------------------------------
        // Beta Editor Toggle Button
        var toggleKMLToggle = function (e, disableBetaTog) {

            var enableKMLBetaToggle = function (kmlBetaToggleGM) {
                localStorage.WME_KMLSettings = localStorage.WME_KMLSettings.replace('#b', '&b'); //reset
                // kmllog('kmlBetaToggleGM = ' + kmlBetaToggleGM)
                kmlBetaToggleGM = (kmlBetaToggleGM.substr(kmlBetaToggleGM.indexOf('-') + 1) == "true");
                // kmllog('WMEKMLayers_Beta = ' + kmlBetaToggleGM)
                setTimeout(function () {
                    GM_setValue("WMEKMLayers_Beta", kmlBetaToggleGM)
                }, 0);
                document.getElementById('cbKMLtoggle').disabled = false;
                document.getElementById('cbKMLtoggle').checked = kmlBetaToggleGM;
                document.getElementsByClassName('kml-toggle-innerD')[0].className = "kml-toggle-inner";
                //document.getElementsByClassName('kml-toggle-switchD')[0].className = "kml-toggle-switch";

                //setTimeout(function () {
                //    document.getElementsByClassName('kml-toggle')[0].ondblclick = toggleKMLToggle;
                //}, 100);
            };

            var disableKMLBetaToggle = function () {
                var kmlBetaToggleChecked = document.getElementById('cbKMLtoggle').checked,
                    kmlBetaToggleGM = 'disabled-' + kmlBetaToggleChecked;

                //localStorage.WME_KMLSettings = localStorage.WME_KMLSettings.replace('&b', '#b');
                setTimeout(function () {
                    GM_setValue("WMEKMLayers_Beta", kmlBetaToggleGM)
                }, 0);
                //kmllog('WMEKMLayers_Beta = ' + kmlBetaToggleGM)
                document.getElementById('cbKMLtoggle').disabled = true;
                document.getElementById('cbKMLtoggle').checked = kmlBetaToggleChecked;
                document.getElementsByClassName('kml-toggle-inner')[0].className = "kml-toggle-innerD";
                //document.getElementsByClassName('kml-toggle-switch')[0].className = "kml-toggle-switchD";

                //setTimeout(function () {
                //    document.getElementsByClassName('kml-toggle')[0].onclick = toggleKMLToggle;
                //}, 100);
            };
            //----------------------------------------------------
            //kmllog('disableBetaTog = ' + disableBetaTog)
            if (disableBetaTog === undefined || disableBetaTog === null) {
                //kmllog("cbKMLtoggle status: " + document.getElementById('cbKMLtoggle').disabled);
                if (document.getElementById('cbKMLtoggle').disabled) { //enable it
                    //kmllog('Enabling Beta Toggle...')
                    getGMBetaValue(enableKMLBetaToggle);
                } else {
                    //kmllog('Disabling Beta Toggle...')
                    disableKMLBetaToggle();
                }
            } else if (disableBetaTog) {
                disableKMLBetaToggle();
            } else {
                getGMBetaValue(enableKMLBetaToggle);
            }
        };

        var insertKMLBetaToggle = function (kmlToggleArg) {
            //kmllog('insertKMLBetaToggle()', '/');
            var doInsertBetaToggle = function (kmlBetaTogVal) {
                $('#user-details').prepend(
                    '<div class="kml-toggle-container"><div class="kml-toggle">' +
                    '<input type="checkbox" name="kml-toggle" class="kml-toggle-checkbox" id="cbKMLtoggle"' +
                    kmlBetaTogVal + '>' +
                    '<label class="kml-toggle-label" for="cbKMLtoggle">' +
                    '<span class="kml-toggle-inner"></span>' +
                    '<span class="kml-toggle-switch"></span></label></div></div>');

                // Event listeners for beta/prod toggle
                document.getElementById('cbKMLtoggle').onclick = function () {
                    setTimeout(function () {
                        GM_setValue("WMEKMLayers_Beta",
                            document.getElementById('cbKMLtoggle').checked);
                    }, 0)
                };

                document.getElementsByClassName('kml-toggle')[0].ondblclick = toggleKMLToggle;
            }

            //---------------
            if ($('#user-details').length !== 0 && $('#cbKMLtoggle').length === 0) {
                kml[4] = 0; //reset counter
                var kmlBetaTogVal, kmlBetaGMType;

                //kmllog(kmlToggleArg);

                if (kmlToggleArg.constructor === Boolean) {
                    kmlBetaGMType = kmlToggleArg;
                    setTimeout(function () {
                        GM_setValue("WMEKMLayers_Beta", kmlToggleArg)
                    }, 0);
                } else { //insertKMLBetaToggle('disabled')
                    kmlToggleArg = (kmlToggleArg.substr(kmlToggleArg.indexOf('-') + 1) == "true"); //bool  
                    kmlBetaGMType = 'disabled';
                }

                (kmlToggleArg) ? kmlBetaTogVal = 'checked': kmlBetaTogVal = '';
                doInsertBetaToggle(kmlBetaTogVal);

                // Disable toggle?
                if (kmlBetaGMType === 'disabled') {
                    if (localStorage.WME_KMLSettings.indexOf('#b') !== -1) {
                        toggleKMLToggle(null, false); // enable toggle
                    } else {
                        toggleKMLToggle(null, true); // disable toggle
                    }
                }

            } else if ($('#cbKMLtoggle').length > 0) {
                kml[4] = 0;
                return true;
            } else if (kml[4]++ < 20) {
                setTimeout(function () {
                    insertKMLBetaToggle(kmlToggleArg)
                }, 400);
            } else {
                console.warn('WMEKMLayers:',
                    'Unable to insert "Beta-Editor Toggle".',
                    'Element #user-details not found on page.')
            }
        };

        //----------------------------------------------------------------------
        var showKMLPrefsPanel = function () {
            //kmllog('showKMLPrefsPanel()');
            var kmlInitStatus = !!(location.host.indexOf('editor-beta') + 1),
                kmlOptions, kmlSettingsUI;

            // setup checkbox list for UI
            kmlOptions = ['Show WME Beta Editor toggle in the left side-panel', //[0]
                'Remove language specifications from clicked permalinks', //[1]
                'Increase the opacity of city polygons to make them more visible', //[2] layer 1 - 1city
                'Make area manager polygons more translucent', //[3] layer 5 - 5am
                'Add a small amount of transparency to Roads layer', //[4] layer 2 - 2roads
                'Disable checking layers at startup']; //[5] xx

            kmlSettingsUI =
                '<div id="KMLsettings" class="kml-panel-blackout">' +
                '<div class="kml-panel">' +
                '   <i id="iKMLsettings" class="icon-cog icon-4x pull-left fa fa-cog fa-4x fa-pull-left"></i>' +
                '   <h2>KeepMyLayers Preferences</h2>' +
                '   <hr class="kml-panel-hr">' +
                '   <div class="kml-panel-section">';

            var selectedKML = Array(kmlOptions.length),
                allStr = ['b', '&l', '&1', '&5', '&2', '&x'], //beta, lang, city, am, roads, disable
                cbKMLSettingsStr;

            //kmllog('kmlayersBetaChk = ' + kmlayersBetaChk)

            //TODO: Clean the following code up....
            if (localStorage.WME_KMLSettings === undefined) {
                localStorage.WME_KMLSettings = ':';
                cbKMLSettingsStr = ':';
            } else {
                cbKMLSettingsStr = localStorage.WME_KMLSettings;
            }

            //  Setup KMLayers settings panel
            var kml_s, numKmlOpt = kmlOptions.length;

            for (kml_s = 0; kml_s < numKmlOpt; kml_s++) {
                (cbKMLSettingsStr.indexOf(allStr[kml_s]) !== -1) ? selectedKML[kml_s] =
                    ' checked': selectedKML[kml_s] = ' ';
                kmlSettingsUI +=
                    '<div class="controls-container">' +
                    '<input type="checkbox" id="cbKML_' + kml_s +
                    '" class="btn btn-default"' + selectedKML[kml_s] +
                    '></input>' +
                    '<label for="cbKML_' + kml_s + '">' + kmlOptions[kml_s] +
                    '</label>' +
                    '</div>';
            }
            kmlSettingsUI += '</div>' +
                '<div style="font-size: 9pt; font-style: italic; margin-left: 35px; margin-top: 0px; margin-bottom: 0px;">' +
                '   Does not include any of the above settings' +
                '</div>' +
                '<hr class="kml-panel-hr">' +
                '<div class="kml-panel-btn">' +
                '   <div id="KMLnote" style="line-height: 10pt; color: #FF0080; position: absolute; left: 0px; display: inline-block; text-align: left; ' +
                '        width: 300px; vertical-align: middle">' + //line-height: 15px; height: 40px; 
                '' + // notes and layers menu go here
                '   </div>' +
                '   <div style="position: absolute; right: 0px; display: inline-block;">' +
                '       <button id="btnKMLsave" style="width: 85px" class="btn btn-primary kml-panel-btn">Save</button>' +
                '       <button id="btnKMLcancel" class="btn btn-default kml-panel-btn">Cancel</button>' +
                '   </div>' +
                '</div></div></div>';
            $('#map').append(kmlSettingsUI);
            kmlSettingsUI = null;

            // Notice about neding to save default layers
            if (!getSavedKMLayers()) {
                $('#KMLnote').append(
                    '<span style="font-size:10px">' +
                    'To save a default layers setting, select your layers under the Layer Switcher menu and then click ' +
                    '   <div style="color: #FF0080; font-size: 11pt; font-style: normal; margin-right: 4px; display:inline-block" ' +
                    '        class="icon-save fa fa-save kml-icn-nsave"></div> (bottom-right).' +
                    '</span>'
                );
            } else { // Dropdown menu of saved layer settings
                $('#KMLnote').append(
                    '<div class="" style="display: inline-block; vertical-align: middle;">' +
                    '   <select id="selKMLset" class="" style="width: 200px;"></select>' +
                    '   <div style="display: inline-block; vertical-align: middle;">' +
                    '   <div id="addKMLset" style="color: #AAAAAA; margin-left: 4px; display:inline-block; cursor: pointer;" class="icon-plus-sign icon-2x fa fa-plus-circle fa-2x"></div>' +
                    '   <div id="removeKMLset" style="color: #AAAAAA; margin-left: 4px; display:inline-block; cursor: pointer;" class="icon-minus-sign icon-2x fa fa-minus-circle fa-2x"></div>' +
                    '</div></div>' +
                    '');

                var kml_sel, numSets = myKMLayers.layerSetNames.length;

                for (kml_sel = 0; kml_sel < numSets; kml_sel++) {
                    document.getElementById("selKMLset").add(new Option(myKMLayers.layerSetNames[kml_sel]));
                };

                document.getElementById("selKMLset").selectedIndex = myKMLayers.idx;
                document.getElementById("selKMLset").value = myKMLayers.layerSetNames[myKMLayers.idx];

                document.getElementById("selKMLset").onchange = function () {
                    myKMLayers.idx = document.getElementById("selKMLset").selectedIndex;
                    if (Object.keys(myKMLayers.visibleInLayersMenu[myKMLayers.idx]).length === 0) {
                        document.getElementById("iKMLsaveLayers").classList.add('kml-icn-nsave');
                    } else {
                        document.getElementById("iKMLsaveLayers").classList.remove('kml-icn-nsave');
                    }
                };

                document.getElementById("addKMLset").onclick = function () {
                    var newKMLayerSetName = prompt("Please enter a name", ""),
                        newKMLayerOpt = document.createElement('option');
                    newKMLayerOpt.appendChild(document.createTextNode(newKMLayerSetName));
                    document.getElementById("selKMLset").appendChild(newKMLayerOpt);
                    document.getElementById("selKMLset").selectedIndex = myKMLayers.layerSetNames.length;
                    myKMLayers.idx = myKMLayers.layerSetNames.length;
                    myKMLayers.layerSetNames[myKMLayers.idx] = newKMLayerSetName;
                    myKMLayers.visibleInLayersMenu[myKMLayers.idx] = {};
                    document.getElementById("iKMLsaveLayers").classList.add('kml-icn-nsave');
                };

                document.getElementById("removeKMLset").onclick = function () {
                    if (myKMLayers.layerSetNames.length > 1) {
                        myKMLayers.layerSetNames.splice(document.getElementById("selKMLset").selectedIndex, 1)
                        myKMLayers.visibleInLayersMenu.splice(document.getElementById("selKMLset").selectedIndex, 1)
                        document.getElementById("selKMLset").removeChild(document.getElementById("selKMLset").options[document.getElementById("selKMLset").selectedIndex]);
                        myKMLayers.idx = myKMLayers.layerSetNames.length - 1;
                        document.getElementById("selKMLset").selectedIndex = myKMLayers.idx;
                        document.getElementById("selKMLset").value = myKMLayers.layerSetNames[myKMLayers.idx];
                    }
                };
            }

            var applyKMLayersSettings = function () {

                if (document.getElementById("cbKML_0").checked) {
                    insertKMLBetaToggle(kmlInitStatus);
                } else {
                    if (document.getElementsByClassName('kml-toggle-container').length !== 0) {
                        setTimeout(function () {
                            GM_deleteValue("WMEKMLayers_Beta")
                        }, 0);
                        document.getElementsByClassName('kml-toggle-container')[0].remove();
                    }
                }

                if (document.getElementById("cbKML_1").checked) {
                    setTimeout(function () {
                        GM_setValue("WMEKMLayers_Lang", true)
                    }, 0);
                } else {
                    setTimeout(function () {
                        GM_deleteValue("WMEKMLayers_Lang")
                    }, 0);
                }

                var storeStr = ':';

                if ($('#cbKML_5')[0].checked) storeStr += '&x'; //disable KML
                if ($('#cbKML_0')[0].checked) storeStr += '&b'; //beta
                if ($('#cbKML_1')[0].checked) storeStr += '&l'; //lang
                if ($('#cbKML_2')[0].checked) storeStr += '&1'; //city
                if ($('#cbKML_3')[0].checked) storeStr += '&5'; //am
                if ($('#cbKML_4')[0].checked) storeStr += '&2'; //roads

                document.getElementById("KMLsettings").remove();

                localStorage.WME_KMLSettings = storeStr + '&' + kmlVersion;
                localStorage.WME_KeepMyLayers = JSON.stringify(myKMLayers);

                //kmllog(document.getElementById("KMLsettings"))
                setTimeout(applyAdditionalKMLSettings, 0);

                if (myKMLayers && myKMLayers.visibleInLayersMenu) {
                    updateKMLayersMenu(myKMLayers);

                    if (Object.keys(myKMLayers.visibleInLayersMenu[myKMLayers.idx]).length !== 0) {
                        userResetOfLayersToSavedKMLayers();
                    }
                }

            };

            document.getElementById('btnKMLsave').onclick = applyKMLayersSettings;

            document.getElementById('btnKMLcancel').onclick = function () {
                document.getElementById("KMLsettings").remove();
                var tempResetHolder = myKMLayers.reset;
                myKMLayers = getSavedKMLayers();
                myKMLayers.reset = tempResetHolder;
                if (myKMLayers && myKMLayers.visibleInLayersMenu && Object.keys(myKMLayers.visibleInLayersMenu[myKMLayers.idx]).length === 0) {
                    document.getElementById("iKMLsaveLayers").classList.add('kml-icn-nsave');
                } else {
                    document.getElementById("iKMLsaveLayers").classList.remove('kml-icn-nsave');
                }
            };
        };

        //======================================================================
        var initKMLButtons = function () {
            //kmllog('inside initKMLButtons()', '/');

            if (document.getElementById("layer-switcher-list") && document.getElementById("iKMLsaveLayers") === null) {
                kml[6] = 0; //reset counter
                var panelHeight = 350,
                    panelWidth = 540,
                    kmlStyle = document.createElement("style");

                // Create CSS container element
                kmlStyle.type = "text/css";
                kmlStyle.id = "kml-css-container";

                // CSS for KMLayers icons under Layers dropdown menu
                kmlStyle.innerHTML =
                    'div.kml-icn { position: absolute; display: inline-block; vertical-align: middle; ' +
                    '   bottom: 0px; right: 0px; padding: 0; margin: 0px 10px 10px 0px; }\n' +
                    '.kml-icn-nsave { font-weight: 400; color: #D36343 !important; } \n' + //	'.kml-icn-save { position: relative; display: block; bottom: 45px; right: 0px }\n' +
                    '.kml-icn-btn {	position: relative; display: inline-block; padding-left: 4px; padding-right: 4px; }\n' + //right: 30px; top: 10px; 'width: 12px; height: 12px;
                    '.kml-icn:active, .kml-icn:focus, .kml-icn:hover, .kml-icn.active { ' +
                    '   text-decoration: none; background-image: none; outline: 0; ' +
                    '   -webkit-box-shadow: none; box-shadow: none; cursor: pointer }\n' +
                    '.kml-icn-off, .kml-icn-off:hover, .kml-icn-off:focus { ' +
                    '   color: #B1D4DF !important; cursor: default !important; }\n' +
                    'span.kml-icn-btn { padding: 0; margin: 0px 10px 10px 0px; }\n';

                // CSS for settings panel
                kmlStyle.innerHTML +=
                    'div.kml-panel-blackout { position: absolute; top: 0px; left: 0px; width: 100%; height: 100%; ' +
                    '   background: rgba(0,0,0,0.5); z-index: 2000; }\n' +
                    'div.kml-panel-clear { position: absolute; top: 0px; left: 0px; width: 100%; height: 100%; ' +
                    '   background: transparent; z-index: 2001; }\n' +
                    '.kml-panel { position: absolute; left: 50%; top: 50%; transform: translate(-50%, -50%); ' +
                    '   width: ' + panelWidth + 'px; ' + //height: ' + panelHeight + 'px; ' +
                    '   padding: 10px; margin: 0; overflow-y: auto; overflow-x: auto; word-wrap: break-word; ' +
                    '   background-color: white; box-shadow: 0px 5px 20px #555555; ' +
                    '   border: 1px solid #858585; border-radius: 10px; }\n' +
                    '.kml-panel h2 { margin-top:10px; margin-bottom: 10px; font-size: 20pt; font-weight: bold; text-align: left; color: #C0C0C0 }\n' +
                    'div.kml-panel-section {display: block; font-size: 12pt; text-align: left; padding: 0px 5px; }\n' +
                    '.kml-panel-hr { display: block; border: 0; height: 0; border-top: 1px solid rgba(0, 0, 0, 0.1); ' +
                    '   border-bottom: 1px solid rgba(255, 255, 255, 0.3); ' +
                    '   margin-top: 8px; margin-bottom: 12px; }\n' +
                    '.kml-panel-btn { margin: 0px 5px 0px; padding: 0px 15px; display: inline-block; height: 32px; }\n' +
                    'div.kml-panel-btn {display: block; position: relative; padding: 0; width: 480px; margin: auto; ' +
                    '   vertical-align: middle; height: 40px; }\n' +
                    '.kml-panel ul>li {padding-bottom: 4px}';

                // CSS for beta-editor toggle
                kmlStyle.innerHTML +=
                    '.kml-toggle-container { position: absolute; right: 10px; top: 10px; }\n' +
                    '.kml-toggle { position: relative; width: 40px; ' +
                    '   -webkit-user-select: none; -moz-user-select: none; -ms-user-select: none; }\n' +
                    '.kml-toggle-checkbox { display: none; }\n' +
                    '.kml-toggle-label { display: block;  overflow: hidden; cursor: pointer;  border: 2px solid #DADBDC; border-radius: 20px; }\n' +
                    '.kml-toggle-inner { display: block; width: 200%; margin-left: -100%; transition: margin 0.2s ease-in 0s; }\n' +
                    '.kml-toggle-inner:before, .kml-toggle-inner:after {  ' +
                    '   display: block; float: left; width: 50%; height: 19px;  ' +
                    '   padding: 0; line-height: 20px; font-size: 10px; box-sizing: border-box; }\n' +
                    '.kml-toggle-inner:before { content: ""; padding-left: 5px; background-color: #FC6C70; }\n' +
                    '.kml-toggle-inner:after { content: ""; padding-right: 5px; background-color: #93C4D3; }\n' +
                    '.kml-toggle-switch { display: block; width: 22px; margin: 0px;  ' +
                    '   position: absolute; top: 0; bottom: 0; right: 17px; background: #FFFFFF; ' +
                    '   border: 2px solid #DADBDC; border-radius: 20px; transition: all 0.2s ease-in 0s; }\n' +
                    '.kml-toggle-checkbox:checked + .kml-toggle-label .kml-toggle-inner { margin-left: 0; }\n' +
                    '.kml-toggle-checkbox:checked + .kml-toggle-label .kml-toggle-switch { right: 0px; }\n' +
                    '.kml-toggle-switchD { display: block; width: 22px; margin: 1px 0px 1px;  ' +
                    '   position: absolute; top: 0; bottom: 0; right: 17px; background: #FFFFFF; ' +
                    '   border: 2px solid #DADBDC; border-radius: 20px; }\n' +
                    '.kml-toggle-innerD { display: block; width: 200%; margin-left: -100%; }\n' +
                    '.kml-toggle-innerD:before, .kml-toggle-innerD:after {  ' +
                    '   display: block; float: left; width: 50%; height: 20px;  ' +
                    '   padding: 0; line-height: 20px; font-size: 10px; box-sizing: border-box; }\n' +
                    '.kml-toggle-innerD:before { content: ""; padding-left: 5px; background-color: #DADBDC; }\n' +
                    '.kml-toggle-innerD:after { content: ""; padding-right: 5px; background-color: #DADBDC; }\n';

                // CSS for layer set dropdown
                kmlStyle.innerHTML +=
                    '.kml-layers-menu { overflow-x: hidden; overflow-y: auto;}\n' + //                    
                    '.kml-layers-menu.divider { background-color: #CBE1E8 !important; margin: 0px !important; padding: 1px !important; height: 1px; }\n' +
                    '.kml-layers-menu>li>a:active, .kml-layers-menu>li>a.active { background-color: #D4E7ED; }\n' +
                    '.kml-layers-menu>li>a:hover { background-color: #DFECF0; color: #5998C6; }' +
                    '.kml-layers-menu>li>a { font-weight: bold; color: #5998C6; word-wrap: break-word; white-space: normal; padding: 10px 10px 10px 10px;}\n' +
                    '.dropdown.kml-layers:hover .dropdown-menu { display: block; visibility: visible; }' +
                    '.toolbar-button.kml-layers-icon:after, .toolbar-button.kml-layers-icon:before { left: 0px !important; right: 0px !important; width: 10px !important; }' +
                    '.toolbar-button.kml-layers:after, .toolbar-button.kml-layers:before { left: 0px !important; right: 0px !important; width: 19px !important; }';

                document.body.appendChild(kmlStyle);
                kmlStyle = null;

                // Add save layers icon to Layers switcher dropdown panel
                var kmlDiv = document.createElement("div");
                kmlDiv.className = "kml-icn";
                kmlDiv.innerHTML =
                    '<a href="javascript:void(0)" class="kml-icn"><div id="iKMLsaveLayers" class="icon-save fa fa-save kml-icn-btn reload-button" data-toggle="tooltip" title="Save Current Set of Layers"></div></a>' +
                    '<a href="javascript:void(0)" class="kml-icn"><div id="iKMLsettings" class="icon-cog fa fa-cog kml-icn-btn reload-button" data-toggle="tooltip" title="KeepMyLayers Preferences"></div></a>' +
                    '<a href="javascript:void(0)" class="kml-icn"><div id="iKMLresetLayers" class="icon-magic fa fa-magic kml-icn-btn reload-button" data-toggle="tooltip" title="Reset Layers to My Settings"></div></a>' +
                    '<div id="iKMLtempUndo" class="icon-eye-open fa fa-eye kml-icn-btn reload-button kml-icn-off" data-toggle="tooltip" title="Toggle Layers from Permalink"></div>';

                document.getElementById("layer-switcher-list").parentNode.insertBefore(
                    kmlDiv, document.getElementById("layer-switcher-list").nextSibling);

                kmlDiv = document.createElement("div");
                kmlDiv.className = "btn-group dropdown kml-layers toolbar-button ";
                kmlDiv.style.width = "20px";
                kmlDiv.style.verticalAlign = "middle";
                //'<div class="btn-group toolbar-button" style="width: 20px; vertical-align: middle">' +
                kmlDiv.innerHTML =
                    '   <div class="toolbar-button kml-layers dropdown-toggle" style="width: 20px; border-left: 1px solid #A8CAD5;" data-toggle="dropdown">' +
                    '       <div class="toolbar-button kml-layers-icon icon-caret-down icon-large fa fa-caret-down fa-lg" style="color: #58889E; width: 10px; height: 20px; top: 40%; right: 4px;">' +
                    '       </div>' +
                    '   </div>' +
                    '	<ul class="kml-layers-menu kml-layers dropdown-menu pull-right dropdown-menu-right" ' +
                    '		style="position: relative; width: 250px; border-radius:0px; border: 2px solid #DDDDDD; top: 0px; margin: 0; padding: 0">' +
                    '		<li style="padding: 10px">Layers for different editing contexts will go here... You have not saved any yet! To start, selected your default set of layers under the WME Layers menu to the left and then click the red save icon at the bottom corner.</li>' +
                    '	</ul>' +
                    '	</div>'; //'</div>;
                document.getElementById("toolbar").insertBefore(kmlDiv, document.getElementById("layer-switcher"));
                if (myKMLayers) updateKMLayersMenu(myKMLayers);



                //------------------ Setup event listeners -------------------
                // buttons under layer menu
                document.getElementById("iKMLsaveLayers").onclick = saveKMLayers;
                document.getElementById("iKMLsettings").onclick = showKMLPrefsPanel;
                document.getElementById("iKMLresetLayers").onclick = userResetOfLayersToSavedKMLayers;
                $(".kml-icn-btn[data-toggle=tooltip]").tooltip();
                //{	placement: 'bottom'	}

                // doubleclick layer menu shortcut for KMLayers preferences panel
                document.getElementById("layer-switcher-menu").ondblclick = showKMLPrefsPanel;

                // Insert beta/prod toggle
                var kmlBetaToggle = GM_getValue("WMEKMLayers_Beta")
                if (localStorage.WME_KMLSettings.indexOf('&b') !== -1) { //normal beta toggle setting
                    insertKMLBetaToggle(kmlBetaToggle);
                } else if (localStorage.WME_KMLSettings.indexOf('#b') !== -1) { // temporary disable of toggle is active
                    insertKMLBetaToggle('disabled');
                }

                // Make save button red if nothing saved
                //kmllog('Checking for localStorage.WME_KeepMyLayers_lV');
                setTimeout(function () {
                    switch (true) {
                        case !myKMLayers:
                        case !myKMLayers.visibleInLayersMenu:
                        case !Object.keys(myKMLayers.visibleInLayersMenu[myKMLayers.idx]).length:
                            document.getElementById("iKMLsaveLayers").classList.add('kml-icn-nsave');
                            break;
                    }
                    //kmllog('Done checking for localStorage.WME_KeepMyLayers_lV');
                }, 300);

            } else if (document.getElementById("iKMLsaveLayers") !== null) {
                kml[6] = 0;
                return true; //icons already inserted into page
            } else if (kml[6]++ < 20) {
                setTimeout(initKMLButtons, 500);
            } else {
                console.warn('WMEKMLayers:',
                    'Unable to insert WME KeepMyLayers under Layers menu.',
                    'Element #layer-switcher-list not found on page.')
            }
        };
        /*
        var initPermalinkReplacer = function () {
            // make sure the welcome log-in screen is not up
            //document.getElementById("welcome-popup").className.lastIndexOf('hide') === -1
            if (document.getElementsByClassName('WazeControlPermalink')) {
                var kmlAltBetaKey = !!(location.host.indexOf('editor-b') + 1), //if beta, then true
                    kmlPLselector = eval('document.querySelectorAll(\'a[href*="waze.com' +
                        location.pathname + '"]\')'), //NOTE: this returns an array
                    kmlPLElement = eval('document.querySelector(\'a[href*="waze.com' +
                        location.pathname + '"]\')'),
                    kmlPLhref = kmlPLElement.getAttribute("href"),
                    kmlNewPL = kmlPLhref.substr(kmlPLhref.indexOf(location.pathname)),
                    toggledBetaKey = false,
                    numPLsOnPage = kmlPLselector.length,
                    kml_sel, fuckingNode;

                var updateWMEPermalink = function (kmlNode, togSuffix) {
                    if (kmlAltBetaKey) { //if true, then beta-editor, so change PL to prod
                        kmlNode.setAttribute("href",
                            'https://www.waze.com' + kmlNewPL + togSuffix);
                        console.log(kmlNode.getAttribute());
                    } else { // else, production editor
                        kmlNode.setAttribute("href",
                            'https://editor-beta.waze.com' + kmlNewPL + togSuffix);
                        console.log(kmlNode.getAttribute());
                    }
                };

                var fuckingJSConstraints = function (ee) {
                    for (kml_sel = 0; kml_sel < numPLsOnPage; kml_sel++) {
                        fuckingNode = kmlPLselector[kml_sel];
                        this['kml_' + kml_sel] = function (ee) {
                            //kmllog('fuckingNode = ')
                            //kmllog(fuckingNode);
                            if (ee.altKey) {
                                toggledBetaKey = true;
                                updateWMEPermalink(fuckingNode, '&b=0');
                            }
                        };
                    }
                };

                var fmlIHopeThisWorks = new fuckingJSConstraints(),
                    kmlNode, kmlIndex;

                //FUCK = fmlIHopeThisWorks;

                for (kml_sel = 0; kml_sel < numPLsOnPage; kml_sel++) {
                    //kmllog(kml_sel + 'A - kmlPLselector[kml_sel] = ' + kmlPLselector[kml_sel]);
                    kmlIndex = kml_sel;
                    kmlNode = kmlPLselector[kmlIndex];
                    kmlNode.addEventListener('mouseover', function (e) {
                        window.addEventListener("keydown", fmlIHopeThisWorks["kml_" + kmlIndex], false);
                    }, false);

                    kmlNode.addEventListener('mouseout', function (e) {
                        if (toggledBetaKey) {
                            toggledBetaKey = false;
                            updateWMEPermalink(kmlNode, '');
                            window.removeEventListener("keydown", fmlIHopeThisWorks["kml_" + kmlIndex]);
                        }
                    }, false);
                }
                //kmllog('6 - kmlPLselector = ' + kmlPLselector);
            } else {
                setTimeout(initPermalinkReplacer, 500);
            }
                     
        };
        */
        //==========================================================================

        //kmllog('Stepping into initKMLButtons()...', '/');
        initKMLButtons();
        //initPermalinkReplacer();

        //==========================================================================

        window.addEventListener("beforeunload", function () {
            if (localStorage.WME_KMLSettings !== undefined &&
                localStorage.WME_KMLSettings.indexOf('&x') === -1) { // user has NOT set WMEKMLayers to be disabled
                resetLocalWithKMLayers();
            }
        }, false);

        // ~~~ TEMP (clean-up) ~~~
        localStorage.removeItem("WME_KML_Settings");
        GM_deleteValue("WMEKML_Beta");
        GM_deleteValue("WMEKML_Lang");
        //kmllog('Done performing cleanup')

        // ~~~~~~~~~~~~~~~~~~~~~~~
        var kmlNoticePanel = function (kmlPanelType, kmlVersion, kmlNoticeText, kmlForumURL) {
            var kmlNoticeUI = document.createElement("div");
            kmlNoticeUI.id = "divKMLnotice";
            kmlNoticeUI.innerHTML =
                '<i id="iKMLnotice" class="icon-exclamation-sign icon-4x pull-left fa fa-exclamation-circle fa-4x fa-pull-left"></i>' +
                '<h2>WME KMLayers Update</h2>' +
                '<hr class="kml-panel-hr">';
            kmlNoticeUI.innerHTML +=
                '<div class="kml-panel-section" style="font-size: 10pt">' +
                kmlNoticeText +
                '</div>' +
                '<div style="margin-top: 6px; font-size: 10pt">' +
                'For details and screenshots of new features, please visit the forum post <a href="' + kmlForumURL + '" target="_blank">here</a>.' +
                '</div>' +
                '<div style="margin-top:10px; font-size: 8pt;"> ' +
                'Note: This is a one-time alert for WME KMLayers v. ' + kmlVersion +
                '</div>';
            kmlNoticeUI.innerHTML += '<hr class="kml-panel-hr">' +
                '<div style="position: relative; width: 70px; display: block; left: 148px; bottom: 0px; margin: 10px; vertical-align: middle; padding: 0">' +
                '<button id="btnKMLokay" style="width: 70px" class="btn btn-primary kml-panel-btn">OK</button></div>' +
                '</div>';

            switch (kmlPanelType) {
                case "child": //requires also opening the settings panel beforehand
                    kmlNoticeUI.className = "kml-panel";
                    kmlNoticeUI.style.width = "500px";
                    kmlNoticeUI.style.height = "270px";
                    document.getElementsByClassName("kml-panel")[0].appendChild(kmlNoticeUI);
                    break;
                case "blackout":
                    kmlNoticeUI.innerHTML = '<div class="kml-panel" style="width: 420px; padding-left: 15px; padding-right: 15px;">' +
                        kmlNoticeUI.innerHTML;
                    kmlNoticeUI.innerHTML += '</div>';
                    kmlNoticeUI.className = "kml-panel-blackout"
                    document.getElementById("map").appendChild(kmlNoticeUI);
                    break;
                case "clear":
                    kmlNoticeUI.innerHTML = '<div class="kml-panel" style="width: 420px; padding-left: 15px; padding-right: 15px;">' +
                        kmlNoticeUI.innerHTML;
                    kmlNoticeUI.innerHTML += '</div>';
                    kmlNoticeUI.className = "kml-panel-clear"
                    document.getElementById("map").appendChild(kmlNoticeUI);
                    break;
            }

            //document.getElementsByClassName("kml-panel")[0].appendChild(kmlNoticeUI);

            document.getElementById('btnKMLokay').onclick = function () {
                document.getElementById('divKMLnotice').remove();
                localStorage.WME_KMLSettings += '&' + kmlVersion;
            };
        };
        if (localStorage.WME_KMLSettings.indexOf('&') === -1) showKMLPrefsPanel();
        if (localStorage.WME_KMLSettings.indexOf('&' + kmlVersion) === -1) { //Display update notice
            var kmlNoticeText, kmlForumURL = "https://www.waze.com/forum/viewtopic.php?f=819&t=172335&p=1304954#p1304954";

            kmlNoticeText = 'Woohoo! Another update! This update to version 0.4 brings you the following new beta features and bug-fixes:<p>' +
                '<ul><li>Thanks to rickzabel\'s help, the bug that caused part of the URL to repeat itself <i>ad infinitum</i> has been found and squashed</li>' +
                '<li>Double-click on the beta/prod toggle to quickly disable/enable</li>' +
                '<li><div style="display: inline; text-decoration: line-through">Press <b>Alt</b>-key once while hovering the cursor over any PL within the WME window to switch it to the alternative editor (i.e., beta or prod)</div> (disabled for now due to conflict with WME Toolbox)</li>' +
                '<li>Within the preference pane, you can now add additional sets of layers to accommodate various editing contexts </li>' +
                '<li>Double-click on the WME Layer menu icon for easy access to the KMLayers preference pane</li>' +
                '</ul>';

            kmlNoticePanel("blackout", kmlVersion, kmlNoticeText, kmlForumURL);

        }
        //KML = myKMLayers;
    };

    ////////////////////////////////////////////////////////////////////////////
    function waitForWazeMap_KMLayers() {
        var waitCount = 0,
            maxWait = 100; //5 seconds
        //kmllog('Waiting for Waze...', '/');

        var tryInit_KMLayers = function () {
            try {
                if (waitCount < maxWait &&
                    "undefined" !== typeof (unsafeWindow) && unsafeWindow.$ &&
                    unsafeWindow.Waze && unsafeWindow.Waze.map && unsafeWindow.Waze.map.layers
                ) {
                    //kmllog('Initializing...', '/');
                    KeepMyLayers();
                } else if (waitCount++ < maxWait) {
                    setTimeout(tryInit_KMLayers, 50);
                } else {
                    //kmllog('waitCount = ' + waitCount, '/');
                    //kmllog('typeof Waze = ' + typeof(unsafeWindow.Waze), '/');
                    //kmllog('Waze.map = ' + !!unsafeWindow.Waze.map, '/');
                    //kmllog('Waze.map.layers = ' + !!unsafeWindow.Waze.map.layers, '/');
                    console.error('WMEKMLayers:',
                        'KeepMyLayers could not find necessary Waze map objects.');
                }
            } catch (err) {
                console.error(
                    'WMEKMLayers:',
                    'WME KeepMyLayers failed to load',
                    'due to some kind of technical script error. :(');
                console.error(err);
            }
        };
        tryInit_KMLayers();
    };

    ////////////////////////////////////////////////////////////////////////////
    document.onreadystatechange = function () {
        //kmllog('<' + document.readyState + '>', '/');
        if (document.readyState === 'interactive') {
            //kmllog('Inside DOM interactive event interval.', '/');
            setTimeout(waitForWazeMap_KMLayers, 0);

            if (localStorage.WME_KMLSettings !== undefined &&
                localStorage.WME_KMLSettings.indexOf('&x') !== -1) { // user has set WMEKMLayers to be disabled
                console.log("WMEKMLayers:",
                    "Autocheck of KeepMyLayers at startup is disabled (1).");
            } else {
                resetLocalWithKMLayers();
            }
        }
    };
})();