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.

目前為 2025-05-14 提交的版本,檢視 最新版本

// ==UserScript==
// @name         WAZEPT Livemap Routes
// @namespace    http://tampermonkey.net/
// @version      1.2
// @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';

    // Portagens: SVG do Waze
    const imgToll = '';
    // Alertas: círculo amarelo
    const imgAlert = '';
    // Jams: círculo vermelho
    const imgJam = '';

    // Haversine formula to calculate distance between two lat/lon points in km
    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;
    }

    // Calcula o comprimento total dos jams (em km)
    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 origFetch = window.fetch;

    window.fetch = function(...args) {
        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)
                        ? `${tollInfo.tollPrice} ${tollInfo.tollPriceCurrencyCode}`
                            : null;
                        const jamsArr = Array.isArray(route?.response?.jams) ? route.response.jams : [];
                        const jams = jamsArr.length;
                        const jamLength = totalJamLength(jamsArr);
                        const alerts = Array.isArray(route?.response?.alerts) ? route.response.alerts.length : 0;
                        infoArr[index] = { tollText, jams, jamLength, alerts };
                    });

                    setTimeout(function() {
                        document.querySelectorAll('.wm-routes-item-desktop__details').forEach(function(el, idx) {
                            if (typeof infoArr[idx] !== 'undefined') {
                                const span = el.querySelector("span[dir='auto']");
                                if (span) {
                                    // Evitar duplicação se o script correr mais que uma vez
                                    if (!span.innerHTML.includes(imgToll) && !span.innerHTML.includes(imgAlert) && !span.innerHTML.includes(imgJam)) {
                                        if (infoArr[idx].tollText != null) {
                                            span.innerHTML += `<br><img src="${imgToll}" alt="Tolls" style="width:21px;height:21px;vertical-align:middle;"> <b>${infoArr[idx].tollText}</b>`;
                                        }
                                        span.innerHTML += `<br><img src="${imgAlert}" alt="Alerts" style="width:21px;height:21px;vertical-align:middle;"> <b>${infoArr[idx].alerts}</b>`;
                                        if (infoArr[idx].jams > 0) {
                                            span.innerHTML += `<br><img src="${imgJam}" alt="Jams" style="width:21px;height:21px;vertical-align:middle;"> <b>${infoArr[idx].jams} (${infoArr[idx].jamLength.toFixed(2)} km)</b>`;
                                        } else {
                                            span.innerHTML += `<br><img src="${imgJam}" alt="Jams" style="width:21px;height:21px;vertical-align:middle;"> <b>0</b>`;
                                        }

                                    }
                                }
                            }
                        });
                    }, 500);
                });
            }
            return response;
        });
    };
})();