WME Switch Environment

Switch between Prod and Beta

// ==UserScript==
// @name         WME Switch Environment
// @description  Switch between Prod and Beta
// @namespace    https://greasyfork.org/users/gad_m/wme_switch_environment
// @version      1.0.8
// @author       gad_m
// @license      MIT
// @include 	 /^https:\/\/(www|beta)\.waze\.com\/(?!user\/)(.{2,6}\/)?editor.*$/
// @exclude      https://www.waze.com/user/*editor/*
// @exclude      https://www.waze.com/*/user/*editor/*
// @grant        GM_openInTab
// @icon         
// ==/UserScript==

/* global W */
/* global jQuery */

(function() {

    'use strict';
    console.info('wme-switch-environment: loaded');

    if (typeof W !== 'undefined' && W['userscripts'] && W['userscripts']['state'] && W['userscripts']['state']['isInitialized']) {
        console.debug('wme-switch-environment: WME is initialized.');
        init();
    } else {
        console.debug('wme-switch-environment: WME is not initialized. adding event listener.');
        document.addEventListener("wme-logged-in", init, {
            once: true,
        });
    }

    function init() {
        console.debug('wme-switch-environment: init()');
        addUI();
        addSaveEvent();
    }

    function addUI() {
        addSwitchButton();
        addSwitchClickEvent();
    }

    function addSaveEvent() {
        console.debug('wme-switch-environment: addSaveEvent()...');
        waitForElementConnected("#save-button", 10000).then((saveButton) => {
            jQuery(saveButton).on('click', () => {
                console.debug("wme-switch-environment: 'Save' button clicked...");
                addUI();
            });
        });
    }

    function addSwitchButton() {
        console.debug('wme-switch-environment: addSwitchButton()...');
        waitForElementConnected(".reload-button", 10000).then((reloadButton) => {
            let exiting = jQuery(".switch-button").length;
            if (!exiting) {
                let buttonsContainer = jQuery(".overlay-buttons-container.top");
                let clonedButton = jQuery(reloadButton).clone();
                clonedButton.removeClass('reload-button');
                clonedButton.addClass('switch-button');
                clonedButton.removeAttr('disabled');
                let iconI = jQuery(clonedButton).find("i.w-icon.w-icon-refresh");
                iconI.removeClass('w-icon-refresh');
                iconI.text("β");
                iconI.css("padding-bottom", "10px");
                buttonsContainer.append(clonedButton);
                console.debug('wme-switch-environment: addSwitchButton() done.');
            } else {
                console.debug('wme-switch-environment: addSwitchButton() button exits.');
            }
        }).catch((error) => {
            console.error(error);
        });
    }

    function addSwitchClickEvent() {
        console.debug('wme-switch-environment: addSwitchClickEvent()...');
        waitForElementConnected(".switch-button").then((element) => {
            let switchButton = jQuery(element);
            switchButton.attr('title', 'Switch Prod/Beta with current permalink');
            switchButton.on('click', switchButtonClicked);
            console.debug('wme-switch-environment: addSwitchClickEvent() done.');
            return false;
        }).catch((error) => {
            console.error(error);
        });
    }

    function switchButtonClicked(evt) {
        console.debug('wme-switch-environment: switchButtonClicked() switch button clicked');
        let permalink = jQuery(".permalink");
        let href = (permalink && permalink[0])?permalink[0].getAttribute("href"):null;
        href = enrichWithLayersBitmask(href);
        console.info('wme-switch-environment: switchButtonClicked() setting browser URL to: ' + href);
        href = switchEvn(href);
        if (evt && evt.shiftKey) {
            window.open(href);
        } else if (evt && (evt.metaKey || evt.shiftKey)) {
            GM_openInTab(href);
        } else {
            document.location = href;
        }
    }

    function switchEvn(url) {
        if (url.startsWith("https://www.waze.com")) {
            return url.replace("https://www.waze.com", "https://beta.waze.com");
        } else if (url.startsWith("https://beta.waze.com")) {
            return url.replace("https://beta.waze.com", "https://www.waze.com");
        } else {
            console.info('wme-switch-environment: switchEvn() invalid URL: ' + url + "'");
        }
    }

    function enrichWithLayersBitmask(href) {
        console.debug('wme-switch-environment: enrichWithLayersBitmask() href: ' + href);
        if (href) {
            let hasLayersBitmask = getLayersBitmask(href);
            if (!hasLayersBitmask) {
                let layersBitmask = W.map.getLayersBitmask();
                console.debug('wme-switch-environment: enrichWithLayersBitmask() adding LayersBitmask: ' + layersBitmask);
                href += "&s=" + layersBitmask;
            } else {
                console.debug('wme-switch-environment: enrichWithLayersBitmask() no need to add LayersBitmask');
            }
        }
        console.debug('wme-switch-environment: enrichWithLayersBitmask() returning: ' + href);
        return href;
    }

    function getLayersBitmask(href) {
        let urlSearchParams = new URLSearchParams(href);
        return urlSearchParams.get('s');
    }

    function waitForElementConnected(selector, timeout = 5000) {
        console.debug("wme-switch-environment: waitForElementConnected() selector: '" + selector + "'...");
        return new Promise((resolve, reject) => {
            const element = document.querySelector(selector);
            if (element && document.body.contains(element)) {
                resolve(element);
                return;
            }

            const observer = new MutationObserver((mutations, obs) => {
                const element = document.querySelector(selector);
                if (element && document.body.contains(element)) {
                    resolve(element);
                    obs.disconnect();
                }
            });

            observer.observe(document.body, {
                childList: true,
                subtree: true
            });

            setTimeout(() => {
                observer.disconnect();
                reject(new Error("wme-switch-environment: Element '" + selector + "' not found within the timeout period"));
            }, timeout);
        });
    }

}.call(this));