Greasy Fork is available in English.

WME LevelReset +

Fork of the original script. The WME LevelReset tool, to make re-locking segments and POI to their appropriate lock level easy & quick. Supports major road types and custom locking rules for specific cities.

// ==UserScript==
// @name         WME LevelReset +
// @version      2024.02.17.001
// @description  Fork of the original script. The WME LevelReset tool, to make re-locking segments and POI to their appropriate lock level easy & quick. Supports major road types and custom locking rules for specific cities.
// @author       Broos Gert '2015, madnut
// @match        https://beta.waze.com/*editor*
// @match        https://www.waze.com/*editor*
// @exclude      https://www.waze.com/*user/*editor/*
// @namespace    https://greasyfork.org/uk/users/160654-waze-ukraine
// @connect      google.com
// @connect      script.googleusercontent.com
// @grant        GM_xmlhttpRequest
// @grant        GM_addStyle
// @icon         
// ==/UserScript==

/* jshint esversion: 11 */
/* global W */
/* global $ */
/* global require */

// initialize LevelReset and do some checks
function LevelReset_bootstrap() {
    if (W?.userscripts?.state.isReady) {
        LevelReset_init();
    } else {
        document.addEventListener("wme-ready", LevelReset_init, {
            once: true
        });
    }
}

function LevelReset_init() {
    // Setting up global variables
    const lrStyle = [
        '.tg { border-collapse: collapse; border-spacing: 0; margin: 0px auto; }',
        '.tg td { border-color: black; border-style: solid; border-width: 1px; overflow: hidden; padding: 2px 2px; word-break: normal; }',
        '.tg .tg-value { text-align: center; vertical-align: top }',
        '.tg .tg-header { background-color: #ecf4ff; border-color: #000000; font-weight: bold; text-align: center; vertical-align: top }',
        '.tg .tg-type { text-align: left; vertical-align: top }',
        ''];
    GM_addStyle(lrStyle.join('\n'));

    const UpdateObject = require("Waze/Action/UpdateObject");
    const VERSION = GM_info.script.version;
    const loader = '';

    const defaultLocks = {
        Street: 1,
        Primary: 1,
        Minor: 2,
        Major: 3,
        Ramp: 4,
        Freeway: 4,
        POI: 1,
        Railroad: 1,
        Private: 1,
        Parking: 1,
        Offroad: 1,
        Narrow: 1,
        Camera: 4,
        RailroadCrossing: 3
    };
    const streets = {
        // fake element to show POI's locks
        90000: {
            typeName: "POI",
            scan: true
        },
        90001: {
            typeName: "Camera",
            scan: true
        },
        90002: {
            typeName: "RailroadCrossing",
            scan: true
        },
        // normal streets
        17: {
            typeName: "Private",
            scan: true
        },
        20: {
            typeName: "Parking",
            scan: true
        },
        8: {
            typeName: "Offroad",
            scan: true
        },
        1: {
            typeName: "Street",
            scan: true
        },
        2: {
            typeName: "Primary",
            scan: true
        },
        7: {
            typeName: "Minor",
            scan: true
        },
        6: {
            typeName: "Major",
            scan: true
        },
        4: {
            typeName: "Ramp",
            scan: true
        },
        3: {
            typeName: "Freeway",
            scan: true
        },
        18: {
            typeName: "Railroad",
            scan: true
        },
        22: {
            typeName: "Narrow",
            scan: true
        }
    };
    let relockObject = {};
    const userlevel = W.loginManager.getUserRank() + 1;
    //const userlevel = 6; // for testing purposes (NOTE: this does not enable you to lock higher!)

    const requestsTimeout = 20000; // in ms
    const rulesHash = "AKfycbzKgUQL7cY6XMBlykq0JzJAPc1B26sKEAnG1RJokA9Wpgf_W0oZJwMKcrhA13LrUbvj";
    let rulesDB = {};

    // Some functions
    function onScreen(obj) {
        if (obj && typeof obj.getOLGeometry === 'function') {
            return (W.map.getExtent().intersectsBounds(obj.getOLGeometry().getBounds()));
        }
        return (false);
    }

    function hasPendingUR(poi) {
        return (poi.attributes.venueUpdateRequests.length > 0);
    }

    function displayHtmlPage(res) {
        if (res.responseText.match(/Authorization needed/) || res.responseText.match(/ServiceLogin/)) {
            alert("LevelReset:\n" +
                "Authorization is required for using this script. This is one time action.\n" +
                "Now you will be redirected to the authorization page, where you'll need to approve request.\n" +
                "After confirmation, please close the page and reload WME.");
        }
        let w = window.open();
        w.document.open();
        w.document.write(res.responseText);
        w.document.close();
        w.location = res.finalUrl;
    }

    function sendHTTPRequest(url, callback) {
        GM_xmlhttpRequest({
            url: url,
            method: 'GET',
            timeout: requestsTimeout,
            onload: function (res) {
                callback(res);
            },
            onreadystatechange: function (res) {
                // fill if needed
            },
            ontimeout: function (res) {
                alert("LevelReset: Sorry, request timeout!");
            },
            onerror: function (res) {
                alert("LevelReset: Sorry, request error!");
            }
        });
    }

    function validateHTTPResponse(res) {
        let result = false,
            displayError = true;
        if (res) {
            switch (res.status) {
                case 200:
                    displayError = false;
                    if (res.responseHeaders.match(/content-type: application\/json/i)) {
                        result = true;
                    } else if (res.responseHeaders.match(/content-type: text\/html/i)) {
                        displayHtmlPage(res);
                    }
                    break;
                default:
                    displayError = false;
                    alert("LevelReset Error: unsupported status code - " + res.status);
                    console.log("LevelReset: " + res.responseHeaders);
                    console.log("LevelReset: " + res.responseText);
                    break;
            }
        } else {
            displayError = false;
            alert("LevelReset error: Response is empty!");
        }

        if (displayError) {
            alert("LevelReset: Error processing request. Response: " + res.responseText);
        }
        return result;
    }

    function getAllLockRules() {
        function requestCallback(res) {
            if (validateHTTPResponse(res)) {
                let out = JSON.parse(res.responseText);
                if (out.result == "success") {
                    initUI(out.rules);
                } else {
                    alert("LevelReset: Error getting locking rules!");
                }
            }
        }

        const url = 'https://script.google.com/macros/s/' + rulesHash + '/exec?func=getAllLockRules';
        sendHTTPRequest(url, requestCallback);
    }

    function initUI(rules) {
        let relockContent = document.createElement('div'),
            relockTitle = document.createElement('wz-overline'),
            relockSubTitle = document.createElement('wz-label'),
            rulesSubTitle = document.createElement('wz-label'),
            relockAllbutton = document.createElement('input'),
            relockSub = document.createElement('p'),
            versionTitle = document.createElement('wz-label'),
            resultsCntr = document.createElement('div'),
            rulesCntr = document.createElement('div'),
            alertCntr = document.createElement('div'),
            hidebutton = document.createElement('div'),
            dotscntr = document.createElement('div'),
            inputDiv1 = document.createElement('div'),
            inputDiv2 = document.createElement('div'),
            includeAllSegments = document.createElement('input'),
            includeAllSegmentsLabel = document.createElement('label'),
            respectRouting = document.createElement('input'),
            respectRoutingLabel = document.createElement('label'),
            percentageLoader = document.createElement('div');

        rulesDB = rules;

        // Begin building
        relockTitle.appendChild(document.createTextNode('Re-lock Segments & POI'));

        // fill tab
        relockSub.innerHTML = 'Your on-screen area is automatically scanned when you load or pan around. Pressing the lock behind each type will relock only those results, or you can choose to relock all.<br/><br/>You can only relock segments lower or equal to your current editor level. Segments locked higher than normal are left alone.';
        relockSub.style.cssText = 'font-size:85%;padding:15px;border:1px solid red;border-radius:5px;position:relative';
        relockSub.id = 'sub';
        hidebutton.style.cssText = 'cursor:pointer;width:16px;height:16px;position:absolute;right:3px;top:3px;background-image:url(\'\');';
        hidebutton.onclick = function () {
            localStorage.msgHide = 1;
            $('#sub').hide('slow');
        };
        dotscntr.style.cssText = 'width:16px;height:16px;margin-left:5px;background:url("' + loader + '");vertical-align:text-top;display:none';
        dotscntr.id = 'dotscntr';
        relockSubTitle.innerHTML = 'Results';
        relockSubTitle.id = 'reshdr';
        rulesSubTitle.innerHTML = 'Active rules';
        versionTitle.innerHTML = 'Version ' + VERSION;
        relockAllbutton.id = 'rlkall';
        relockAllbutton.type = 'button';
        relockAllbutton.value = 'Relock All';
        relockAllbutton.style.cssText = 'margin: 10px 3px 0 0';
        relockAllbutton.onclick = function () {
            relockAll();
        };

        // Also reset higher locked segments?
        includeAllSegments.type = 'checkbox';
        includeAllSegments.name = "name";
        includeAllSegments.value = "value";
        includeAllSegments.checked = (localStorage.getItem('Relock_allSegments') == 'true');
        includeAllSegments.id = "_allSegments";
        includeAllSegments.onclick = function () {
            localStorage.setItem('Relock_allSegments', includeAllSegments.checked.toString());
            scanArea();
            relockShowAlert();
        };
        includeAllSegmentsLabel.htmlFor = "_allSegments";
        includeAllSegmentsLabel.innerHTML = 'Also reset higher locked objects';
        includeAllSegmentsLabel.style.cssText = 'font-size:95%;margin-left:5px;vertical-align:middle';

        // Respect routing road type
        respectRouting.type = 'checkbox';
        respectRouting.name = "name";
        respectRouting.value = "value";
        respectRouting.checked = (localStorage.getItem('Relock_respectRouting') == 'true');
        respectRouting.id = "_respectRouting";
        respectRouting.onclick = function () {
            localStorage.setItem('Relock_respectRouting', respectRouting.checked.toString());
            scanArea();
        };
        respectRoutingLabel.htmlFor = "_respectRouting";
        respectRoutingLabel.innerHTML = 'Respect routing road type';
        respectRoutingLabel.style.cssText = 'font-size:95%;margin-left:5px;vertical-align:middle';

        resultsCntr.style.cssText = 'margin-right:5px;';

        // add results empty list
        $.each(streets, function (key, value) {
            let __cntr = document.createElement('div'),
                __keyLeft = document.createElement('div'),
                __prntRight = document.createElement('div'),
                __cntRight = document.createElement('div'),
                __cleardiv = document.createElement("div"),
                __chkLeft = document.createElement('input'),
                __lblLeft = document.createElement('label');
            let idPrefix = 'Relock_' + value.typeName + '_';

            // Begin building
            __keyLeft.style.cssText = 'float:left';

            __chkLeft.type = 'checkbox';
            __chkLeft.checked = (localStorage.getItem(idPrefix + 'chk') == 'true');
            __chkLeft.id = idPrefix + 'chk';
            __chkLeft.onclick = function () {
                localStorage.setItem(idPrefix + 'chk', __chkLeft.checked.toString());
                scanArea();
            };
            __lblLeft.htmlFor = idPrefix + 'chk';
            __lblLeft.innerHTML = value.typeName;
            __lblLeft.style.cssText = 'margin-bottom:0px;font-weight:normal;';

            __keyLeft.appendChild(__chkLeft);
            __keyLeft.appendChild(__lblLeft);

            __cntRight.style.cssText = 'float:right';
            __cntRight.innerHTML = '-';

            __prntRight.id = idPrefix + 'value';
            __prntRight.style.cssText = 'float:right';
            __prntRight.appendChild(__cntRight);

            __cleardiv.style.cssText = 'clear:both;';

            // Add to stage
            __cntr.appendChild(__keyLeft);
            __cntr.appendChild(__prntRight);
            __cntr.appendChild(__cleardiv);
            resultsCntr.appendChild(__cntr);
        });

        // Alert box
        alertCntr.id = "alertCntr";
        alertCntr.style.cssText = 'border:1px solid #EBCCD1;background-color:#F2DEDE;color:#AC4947;font-weight:bold;font-size:90%;border-radius:5px;padding:10px;margin:5px 5px;display:none';
        alertCntr.innerHTML = 'Watch out for map exceptions, some higher locks are there for a reason!';

        // Rules table
        let rowElm;
        let colElm;

        let tableElm = document.createElement('table');
        tableElm.className = 'tg';

        let bodyElm = document.createElement('tbody');

        const topCountry = W.model.getTopCountry();
        const countryRules = rulesDB[topCountry.attributes.abbr];
        if (countryRules) {
            $.each(countryRules, function (key, value) {
                if (key == "CountryName") return false;

                rowElm = document.createElement('tr');
                rowElm.className = "tg-row";
                rowElm.dataset.name = parseInt(key) === 0 ? 'country' : value.CityName; // need to hard code 'country' to identify later

                colElm = document.createElement('td');
                colElm.className = "tg-header";
                colElm.innerHTML = parseInt(key) === 0 ? countryRules.CountryName : value.CityName;
                colElm.colSpan = 6;
                rowElm.appendChild(colElm);
                tableElm.appendChild(rowElm);

                const maxCol = 3;
                let colIndex = 0;
                rowElm = document.createElement('tr');
                $.each(value.Locks, function (k, v) {
                    if (v) {
                        rowElm.className = "tg-row";
                        rowElm.dataset.name = parseInt(key) === 0 ? 'country' : value.CityName; // need to hard code 'country' to identify later
                        if (colIndex < maxCol) {
                            colElm = document.createElement('td');
                            colElm.className = "tg-type";
                            colElm.innerHTML = k;
                            rowElm.appendChild(colElm);

                            colElm = document.createElement('td');
                            colElm.className = "tg-value";
                            colElm.innerHTML = v;
                            rowElm.appendChild(colElm);

                            colIndex++;
                            if (colIndex == maxCol) {
                                colIndex = 0;
                                tableElm.appendChild(rowElm);
                                rowElm = document.createElement('tr');
                            }
                        }
                    }
                });
                tableElm.appendChild(rowElm);
            });
        }

        tableElm.appendChild(bodyElm);
        rulesCntr.style.cssText = 'font-size:12px';
        rulesCntr.appendChild(tableElm);

        // add to stage
        relockContent.appendChild(relockTitle);
        relockContent.appendChild(versionTitle);

        // Loader bar
        percentageLoader.id = 'percentageLoader';
        percentageLoader.style.cssText = 'width:1px;height:10px;background-color:green;margin-top:10px;border:1px solid:#333333;display:none';

        // only show if user didn't hide it before
        if (localStorage.msgHide != 1) {
            relockSub.appendChild(hidebutton);
            relockContent.appendChild(relockSub);
        }

        inputDiv1.appendChild(respectRouting);
        inputDiv1.appendChild(respectRoutingLabel);
        inputDiv2.appendChild(includeAllSegments);
        inputDiv2.appendChild(includeAllSegmentsLabel);
        relockContent.appendChild(inputDiv1);
        relockContent.appendChild(inputDiv2);

        relockContent.appendChild(alertCntr);
        relockContent.appendChild(relockSubTitle);
        relockContent.appendChild(resultsCntr);
        relockContent.appendChild(relockAllbutton);
        relockContent.appendChild(dotscntr);
        relockContent.appendChild(percentageLoader);
        relockContent.appendChild(rulesSubTitle);
        relockContent.appendChild(rulesCntr);

        const { tabLabel, tabPane } = W.userscripts.registerSidebarTab("sidepanel-relock");

        tabLabel.innerHTML = 'Re - <span class="fa fa-lock" id="lockcolor" style="color:green"></span>';
        tabLabel.title = "Relock segments";

        tabPane.appendChild(relockContent);

        // Do a default scan once at startup
        relockShowAlert();
        scanArea();
    }

    function relock(obj, key) {
        let objects = obj[key];
        let _i = 0;

        // update GUI
        function RunLocal() {
            W.model.actionManager.add(objects[_i]);
            _i++;

            if (_i < objects.length) {
                setTimeout(RunLocal, 1);
                let newWidth = (_i / objects.length) * $('#sidepanel-relockTab').css('width').replace('px', '');
                $('#percentageLoader').show();
                $('#percentageLoader').css('width', newWidth + 'px');
                $('#dotscntr').css('display', 'inline-block');
            } else {
                $('#dotscntr').css('display', 'none');
                $('#percentageLoader').hide();
            }
        }
        RunLocal();
    }

    function relockAll() {
        // only lock "all" until the current editors level is reached, then stop...
        $('#dotscntr').css('display', 'inline-block');

        $.each(relockObject, function (key, value) {
            if (value.length !== 0) {
                // loop trough each segmentType
                let _i = 0;
                let RunLocal5 = function () {
                    W.model.actionManager.add(value[_i]);
                    _i++;

                    // Did not iterate with $.each, so the GUI can update with larger arrays
                    if (_i < value.length) {
                        setTimeout(RunLocal5, 1);
                        let newWidth = (_i / value.length) * $('#sidepanel-relockTab').css('width').replace('px', '');
                        $('#percentageLoader').show();
                        $('#percentageLoader').css('width', newWidth + 'px');
                        $('#dotscntr').css('display', 'inline-block');
                    } else {
                        $('#dotscntr').css('display', 'none');
                        $('#percentageLoader').hide();
                    }
                };
                RunLocal5();
            }
        });
        scanArea();
        $('#dotscntr').hide('slow');
    }

    function relockShowAlert() {
        let includeAllSegments = document.getElementById('_allSegments');
        if (includeAllSegments && includeAllSegments.checked)
            $('#alertCntr').show("fast");
        else
            $('#alertCntr').hide("fast");
    }

    function hideInactiveCities() {
        $('tr.tg-row').each(function (i) {
            let isActive = false;
            for (let j in W.model.cities.objects) {
                if (W.model.cities.objects[j].attributes.name == $(this).data("name")) {
                    isActive = true;
                    break;
                }
            }
            if (isActive || $(this).data("name") == 'country') {
                $(this).show("fast");
            } else {
                $(this).hide("fast");
            }
        });
    }

    function setLockLevel(v, scanObj, desiredLockLevel) {
        let includeAllSegments = document.getElementById('_allSegments');
        let allSegmentsInclude = includeAllSegments.checked && userlevel > 4;
        if (userlevel > desiredLockLevel) {
            if ((v.attributes.lockRank < desiredLockLevel) ||
                (v.attributes.lockRank > desiredLockLevel && allSegmentsInclude)) {
                relockObject[scanObj].push(new UpdateObject(v, {
                    lockRank: desiredLockLevel
                }));
                return true;
            }
        }
        return false;
    }

    function scanArea() {
        let respectRouting = document.getElementById('_respectRouting');
        let relockSubTitle = document.getElementById('reshdr');
        let relockAllbutton = document.getElementById('rlkall');

        if (!(relockSubTitle && relockAllbutton && respectRouting))
            return;

        hideInactiveCities();

        // Object with array of road types, to collect each wrongly locked segment, for later use
        $.each(defaultLocks, function (k, v) {
            relockObject[k] = [];
        });

        let foundBadlocks = false;
        let respectRoutingRoadType = respectRouting.checked;
        let count = 0;

        // Choose country lock settings. If country selection fails
        // or country isn't in this list, WME default values are used.
        let topCountry = W.model.getTopCountry();
        let ABBR = rulesDB[topCountry.attributes.abbr] ? rulesDB[topCountry.attributes.abbr][0].Locks : defaultLocks;
        console.log("LevelReset: ", ABBR);

        // Do a count on how many segments are in need of a correct lock (limit to 150 to save CPU)
        // Count also depends on the users editor level
        const limitCount = 150;
        relockSubTitle.innerHTML = 'Results (limit: ' + limitCount + ')';

        // disable unchecked road types
        $.each(streets, function (key, value) {
            let idPrefix = 'Relock_' + value.typeName + '_chk';
            let chk = document.getElementById(idPrefix);
            value.scan = (chk && chk.checked);
        });

        // ============== POI ===========================
        if (streets["90000"].scan) {
            let scanObj = "POI";
            $.each(W.model.venues.objects, function (k, v) {
                if (count < limitCount && v.type == "venue" && onScreen(v) && v.isGeometryEditable() && !hasPendingUR(v) /*&& !v.isResidential()*/) {

                    let strt = v.attributes.streetID ? W.model.streets.objects[v.attributes.streetID] : null;
                    let cityID = strt ? strt.cityID : null;

                    let desiredLockLevel = (cityID && rulesDB[topCountry.attributes.abbr] && rulesDB[topCountry.attributes.abbr][cityID]) ? rulesDB[topCountry.attributes.abbr][cityID].Locks[scanObj] : ABBR[scanObj];
                    desiredLockLevel--;

                    if (setLockLevel(v, scanObj, desiredLockLevel)) {
                        foundBadlocks = true;
                        count++;
                    }
                }
            });
        }
        // ============== Cameras ===========================
        if (streets["90001"].scan) {
            let scanObj = "Camera";
            $.each(W.model.cameras.objects, function (k, v) {
                if (count < limitCount && v.type == "camera" && onScreen(v) && v.isGeometryEditable()) {

                    let desiredLockLevel = ABBR[scanObj];
                    desiredLockLevel--;

                    if (setLockLevel(v, scanObj, desiredLockLevel)) {
                        foundBadlocks = true;
                        count++;
                    }
                }
            });
        }
        // ============== Railroad Crossings ===========================
        if (streets["90002"].scan) {
            let scanObj = "RailroadCrossing";
            $.each(W.model.railroadCrossings.objects, function (k, v) {
                if (count < limitCount && v.type == "railroadCrossing" && onScreen(v) && v.isGeometryEditable()) {

                    let desiredLockLevel = ABBR[scanObj];
                    desiredLockLevel--;

                    if (setLockLevel(v, scanObj, desiredLockLevel)) {
                        foundBadlocks = true;
                        count++;
                    }
                }
            });
        }
        // ============== Segments ===========================
        $.each(W.model.segments.objects, function (k, v) {
            if (count < limitCount && v.type == "segment" && onScreen(v) && v.isGeometryEditable()) {
                let curStreet = streets[v.attributes.roadType];
                // for changed routing respect the routing type (if enabled)
                if (v.attributes.routingRoadType && respectRoutingRoadType) {
                    curStreet = streets[v.attributes.routingRoadType];
                }
                if (curStreet && curStreet.scan) {
                    let strt = W.model.streets.getObjectById(v.attributes.primaryStreetID);
                    let cityID = strt ? strt.attributes.cityID : null;

                    let stLocks = (cityID && rulesDB[topCountry.attributes.abbr] && rulesDB[topCountry.attributes.abbr][cityID]) ? rulesDB[topCountry.attributes.abbr][cityID].Locks : ABBR;
                    let desiredLockLevel = stLocks[curStreet.typeName] - 1;

                    if (setLockLevel(v, curStreet.typeName, desiredLockLevel)) {
                        foundBadlocks = true;
                        count++;
                    }
                }
            }
        });

        // Build results to users tab panel
        $.each(relockObject, function (key, value) {
            let __lckRight = document.createElement('div');
            let __cntRight = document.createElement('div');
            let idPrefix = 'Relock_' + key + '_value';

            // Begin building
            let __prntRight = document.getElementById(idPrefix);
            __prntRight.innerHTML = '';

            __cntRight.style.cssText = 'float:right';
            __lckRight.style.cssText = 'width:15px;float:right;padding:2px 0 0 8px;cursor:pointer;';

            if (value.length !== 0) {
                __cntRight.innerHTML = '<b>' + value.length + '</b>';
                __lckRight.className = 'fa fa-lock';
                __lckRight.style.cssText += 'color:red;';
                __lckRight.onclick = function () {
                    relock(relockObject, key);
                };
                __prntRight.appendChild(__lckRight);
            } else {
                __cntRight.innerHTML = '-';
            }

            __prntRight.appendChild(__cntRight);
        });

        // Color the small lock icon red, if errors are found, so people can decide what to do...
        if (foundBadlocks) {
            relockAllbutton.removeAttribute('disabled');
            $('#lockcolor').css('color', 'red');
        } else {
            relockAllbutton.setAttribute('disabled', true);
            $('#lockcolor').css('color', 'green');
        }
    }

    getAllLockRules();

    // Register some event listeners
    W.map.events.register("moveend", null, scanArea);
    W.model.actionManager.events.register("afteraction", null, scanArea);
    W.model.actionManager.events.register("afterundoaction", null, scanArea);
    W.model.actionManager.events.register("noActions", null, scanArea);
}

LevelReset_bootstrap();