WAZEPT Livemap Routes

Displays the toll price, the number of alerts and jams (and the total length of jams) for the alternative routes on the Waze Live Map.

// ==UserScript==
// @name         WAZEPT Livemap Routes
// @namespace    http://tampermonkey.net/
// @version      2025.05.18
// @description  Displays the toll price, the number of alerts and jams (and the total length of jams) for the alternative routes on the Waze Live Map.
// @author       J0N4S13
// @include      /^https:\/\/www.waze.com\/.*live-map/
// @grant        none
// ==/UserScript==

(function() {
    'use strict';

    const imgSpeed = '';
    const imgToll = '';
    const imgAlert = '';
    const imgJam = '';
    const imgArea = '';

    let env = "";

    function haversine(lat1, lon1, lat2, lon2) {
        const R = 6371; // Earth radius in km
        const toRad = x => x * Math.PI / 180;
        const dLat = toRad(lat2 - lat1);
        const dLon = toRad(lon2 - lon1);
        const a = Math.sin(dLat/2) * Math.sin(dLat/2) +
              Math.cos(toRad(lat1)) * Math.cos(toRad(lat2)) *
              Math.sin(dLon/2) * Math.sin(dLon/2);
        const c = 2 * Math.atan2(Math.sqrt(a), Math.sqrt(1-a));
        return R * c;
    }

    function totalJamLength(jams) {
        let total = 0;
        if (!Array.isArray(jams)) return 0;
        jams.forEach(jam => {
            if (Array.isArray(jam.line)) {
                for (let i = 0; i < jam.line.length - 1; i++) {
                    const p1 = jam.line[i], p2 = jam.line[i+1];
                    total += haversine(p1.y, p1.x, p2.y, p2.x);
                }
            }
        });
        return total;
    }

    const currencyData = {};
    currencyData.USD = { symbol: "$", digits: 2 };
    currencyData.EUR = { symbol: "€", digits: 2 };
    currencyData.JPY = { symbol: "¥", digits: 0 };
    currencyData.GBP = { symbol: "£", digits: 2 };
    currencyData.AUD = { symbol: "$", digits: 2 };
    currencyData.CAD = { symbol: "$", digits: 2 };
    currencyData.CHF = { symbol: "Fr.", digits: 2 };
    currencyData.CNY = { symbol: "¥", digits: 1 };
    currencyData.SEK = { symbol: "kr", digits: 2 };
    currencyData.NOK = { symbol: "kr", digits: 2 };
    currencyData.KRW = { symbol: "₩", digits: 0 };
    currencyData.INR = { symbol: "₹", digits: 2 };
    currencyData.BRL = { symbol: "R$", digits: 2 };
    currencyData.RUB = { symbol: "₽", digits: 2 };

    function formatTollPrice(price, currency){
        if(!price || !currency) return null;
        const cd = currencyData[currency];
        if(!cd) return `${price} ${currency}`;
        return `${price.toLocaleString(undefined, {
            minimumFractionDigits: cd.digits,
            maximumFractionDigits: cd.digits,
        })} ${cd.symbol}`;
    }

    const origFetch = window.fetch;

    window.fetch = function(...args) {
        if (args[0].includes("/live-map/api/user-drive")) {
            const params = new URL(args[0]).searchParams;
            env = params.get('geo_env');
        }
        return origFetch.apply(this, args).then(response => {
            if (response.url.includes("/live-map/api/user-drive")) {
                response.clone().json().then(function(data) {
                    const infoArr = [];
                    (data["alternatives"] || []).forEach(function(route, index) {
                        const tollInfo = route?.response?.tollPriceInfo;
                        const tollText = (tollInfo && Object.keys(tollInfo).length > 0)
                        ? formatTollPrice(tollInfo?.tollPrice, tollInfo?.tollPriceCurrencyCode)
                            : null;
                        const jamsArr = Array.isArray(route?.response?.jams) ? route.response.jams : [];
                        const jams = jamsArr.length;
                        let jamLength = 0.0;
                        if(jams > 0)
                        {
                            if(env == "na")
                                jamLength = (totalJamLength(jamsArr) / 1.60934);
                            else
                                jamLength = totalJamLength(jamsArr);
                        }
                        const alerts = Array.isArray(route?.response?.alerts) ? route.response.alerts.length : 0;
                        const areasArr = Array.isArray(route?.response?.areas) ? route.response.areas : [];
                        const areasText = areasArr.length > 0 ? areasArr.join(', ') : null;
                        let avarageSpeed = 0.0;
                        if(env == "na")
                            avarageSpeed = (route.response.totalLength / 1609.34) / (route.response.totalSeconds / 3600);
                        else
                            avarageSpeed = (route.response.totalLength / 1000) / (route.response.totalSeconds / 3600);

                        infoArr[index] = { tollText, jams, jamLength, alerts, areasText, avarageSpeed };
                    });

                    document.querySelectorAll('.wm-routes-item-desktop__details').forEach(function(el, idx) {
                        if (typeof infoArr[idx] !== 'undefined') {
                            if (!el.querySelector('.custom-info-div')) {
                                const infoDiv = document.createElement('div');
                                infoDiv.className = 'custom-info-div';
                                infoDiv.style.display = 'flex';
                                infoDiv.style.flexDirection = 'column'; // Para alinhar colunas e área em "stack"
                                infoDiv.style.gap = '4px';
                                infoDiv.style.marginTop = '6px';

                                // Coluna 1
                                let col1 = '';
                                col1 += `<img src="${imgAlert}" alt="Alerts" style="width:21px;height:21px;vertical-align:middle;"> <b>${infoArr[idx].alerts}</b><br>`;
                                if (infoArr[idx].jams > 0) {
                                    col1 += `<img src="${imgJam}" alt="Jams" style="width:21px;height:21px;vertical-align:middle;"> <b>${infoArr[idx].jams} (${infoArr[idx].jamLength.toFixed(2)} ${env == 'na' ? 'mi' : 'km'})</b><br>`;
                                } else {
                                    col1 += `<img src="${imgJam}" alt="Jams" style="width:21px;height:21px;vertical-align:middle;"> <b>0</b><br>`;
                                }

                                // Coluna 2
                                let col2 = '';
                                col2 += `<img src="${imgSpeed}" alt="AverageSpeed" style="width:21px;height:21px;vertical-align:middle;"> <b>${infoArr[idx].avarageSpeed.toFixed(2)} ${env == 'na' ? 'mi/h' : 'km/h'}</b><br>`;
                                if (infoArr[idx].tollText != null) {
                                    col2 += `<img src="${imgToll}" alt="Tolls" style="width:21px;height:21px;vertical-align:middle;"> <b>${infoArr[idx].tollText}</b><br>`;
                                }

                                // Montar as colunas
                                const columnsDiv = document.createElement('div');
                                columnsDiv.style.display = 'flex';
                                columnsDiv.style.gap = '20px';
                                columnsDiv.innerHTML = `
                <div style="flex:1; min-width:120px;">${col1}</div>
                <div style="flex:1; min-width:120px;">${col2}</div>
            `;
                                infoDiv.appendChild(columnsDiv);

                                // Área ocupa a largura toda, se existir
                                if (infoArr[idx].areasText) {
                                    const areaDiv = document.createElement('div');
                                    areaDiv.innerHTML = `<img src="${imgArea}" alt="Areas" style="width:21px;height:21px;vertical-align:middle;"> <b>${infoArr[idx].areasText}</b>`;
                                    infoDiv.appendChild(areaDiv);
                                }

                                el.appendChild(infoDiv);
                            }
                        }
                    });


                });
            }
            return response;
        });
    };
})();