WME Easy ShortCuts

Muestra y gestiona ShortCuts simples para acciones de WME (incluyendo Hazards y Places) evitando choques con ShortCuts nativos.

您需要先安装一个扩展,例如 篡改猴Greasemonkey暴力猴,之后才能安装此脚本。

You will need to install an extension such as Tampermonkey to install this script.

您需要先安装一个扩展,例如 篡改猴暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴Userscripts ,之后才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。

您需要先安装用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name         WME Easy ShortCuts
// @namespace    https://greasyfork.org/en/users/mincho77
// @version      1.1.2
// @author       mincho77
// @description  Muestra y gestiona ShortCuts simples para acciones de WME (incluyendo Hazards y Places) evitando choques con ShortCuts nativos.
// @license      MIT
// @match        https://www.waze.com/editor*
// @match        https://www.waze.com/*/editor*
// @include      https://beta.waze.com/*
// @include      https://www.waze.com/editor*
// @include      https://www.waze.com/*/editor*
// @exclude      https://www.waze.com/user/editor*
// @grant        GM_xmlhttpRequest
// @grant        unsafeWindow
// @run-at       document-end
// @require      https://greasyfork.org/scripts/24851-wazewrap/code/WazeWrap.js
// ==/UserScript==

(function () 
{
    'use strict';
    // Capa para mostrar el análisis de curvas
    let curveAnalysisLayer = null;

    const MAIN_TAB_ICON_BASE64 = "";
    /**************************************************************************
    //Nombre: initEasyShortCuts
    //Fecha modificación: 2025-11-15
    //Hora: 11:30
    //Autor: mincho77
    //Entradas: Ninguna
    //Salidas: Ninguna
    //Prerrequisitos si existen: WazeWrap cargado y editor inicializado
    //Descripción: Punto de arranque del script. Inicializa el registro de ShortCuts,
    //             configura el modo híbrido y arranca observadores de menú.
    **************************************************************************/
    function initEasyShortCuts(sdkInitializer) 
    { // <-- Ahora acepta el inicializador del SDK
        ShortcutRegistry.init();
        HazardAndPlaceActions.registerAll();
        KeyboardListener.attach(); 
        MenuDecorator.start();
        ScriptLauncher.start();

        let wmeSDK = null;
        if (sdkInitializer) 
        {
            try 
            {
                //
                wmeSDK = sdkInitializer({ scriptId: 'WMEEasyShortCuts', scriptName: 'WME Easy ShortCuts' });
                logInfo('WME SDK inicializado correctamente.');
            }
            catch (e)
            {
                logWarn('Falló la llamada a getWmeSdk: ' + e.message);
            }
        }
        else
        {
            // Esto no debería pasar con el nuevo bootstrap, pero es una buena comprobación
            logWarn('WME SDK no se pasó a init. El analizador de curvas estará desactivado.');
        }        
        // Pasar el SDK al analizador
        CurveAnalyzer.setSDK(wmeSDK); //
        SchoolZoneVisualizer.setSDK(wmeSDK);
        logInfo('WME Easy ShortCuts inicializado');
    }

    /**************************************************************************
    //Nombre: logInfo
    //Fecha modificación: 2025-11-15
    //Hora: 11:30
    //Autor: mincho77
    //Entradas: msg (string)
    //Salidas: Ninguna
    //Prerrequisitos si existen: Consola disponible
    //Descripción: Wrapper simple para logs informativos del script.
    **************************************************************************/
    function logInfo(msg) 
    {
        // Prefijo para facilitar filtrado en la consola
        console.info('[EasyShortCuts] ' + msg);
    }

    /**************************************************************************
    //Nombre: logWarn
    //Fecha modificación: 2025-11-15
    //Hora: 11:30
    //Autor: mincho77
    //Entradas: msg (string)
    //Salidas: Ninguna
    //Prerrequisitos si existen: Consola disponible
    //Descripción: Wrapper simple para logs de advertencia del script.
    **************************************************************************/
    function logWarn(msg) 
    {
        console.warn('[EasyShortCuts] ' + msg);
    }//logWarn

    function safeGetElementById(id) 
    {
        if (typeof id !== 'string' || !id.trim()) {
            return null;
        }
        return document.getElementById(id);
    }

    function getMenuLabelCandidates(action) 
    {
        if (!action) 
        {
            return [];
        }
        const values = [];
        function pushCandidate(val) 
        {
            if (!val) {
                return;
            }
            const normalized = ('' + val).trim();
            if (!normalized) {
                return;
            }
            if (values.indexOf(normalized) === -1) {
                values.push(normalized);
            }
        }
        if (action.menuLabel) {
            pushCandidate(action.menuLabel);
        }
        if (Array.isArray(action.menuLabels)) {
            action.menuLabels.forEach(pushCandidate);
        }
        if (Array.isArray(action.menuAltLabels)) {
            action.menuAltLabels.forEach(pushCandidate);
        }
        pushCandidate(action.label);
        return values;
    }

    // ---------------------------------------------------------------------
    //  Módulo: Estilos compartidos (modal + launcher propio)
    // ---------------------------------------------------------------------

    const UIStyles = (function () {
        const STYLE_ID = 'esc-shared-styles';

        function ensure() {
            if (safeGetElementById(STYLE_ID)) {
                return;
            }

            const style = document.createElement('style');
            style.id = STYLE_ID;
            style.textContent = '\
#esc-shortcuts-modal {\
    position: fixed;\
    top: 0;\
    left: 0;\
    width: 100%;\
    height: 100%;\
    background: rgba(0, 0, 0, 0.4);\
    z-index: 99999;\
    display: flex;\
    align-items: center;\
    justify-content: center;\
}\
#esc-shortcuts-modal .esc-modal {\
    background: #fff;\
    border-radius: 10px;\
    box-shadow: 0 10px 30px rgba(0, 0, 0, 0.25);\
    width: 950px;\
    max-height: 85vh;\
    display: flex;\
    flex-direction: column;\
    padding: 18px 24px;\
    font-family: "Rubik", "Open Sans", sans-serif;\
    animation: esc-modal-in 0.2s ease-out;\
}\
#esc-shortcuts-modal .esc-modal__header {\
    display: flex;\
    align-items: center;\
    justify-content: space-between;\
    margin-bottom: 8px;\
}\
#esc-shortcuts-modal .esc-modal__header h2 {\
    margin: 0;\
    font-size: 20px;\
}\
#esc-shortcuts-modal .esc-close-btn {\
    background: transparent;\
    border: none;\
    font-size: 22px;\
    cursor: pointer;\
    color: #666;\
}\
#esc-shortcuts-modal .esc-close-btn:hover {\
    color: #000;\
}\
#esc-shortcuts-modal .esc-modal__body {\
    overflow: auto;\
    flex: 1;\
    margin-top: 8px;\
}\
#esc-shortcuts-modal table {\
    width: 100%;\
    border-collapse: collapse;\
    font-size: 13px;\
}\
#esc-shortcuts-modal thead th {\
    position: sticky;\
    top: 0;\
    background: #f7f7f7;\
    padding: 6px;\
    border-bottom: 1px solid #ddd;\
    text-align: left;\
}\
#esc-shortcuts-modal tbody td {\
    padding: 6px;\
    border-bottom: 1px solid #f0f0f0;\
    vertical-align: top;\
}\
#esc-shortcuts-modal .esc-combo-control {\
    display: flex;\
    gap: 6px;\
    align-items: center;\
}\
#esc-shortcuts-modal .esc-state-toggle-btn {\
    border: 1px solid #d1d5db;\
    background: #fff;\
    color: #111;\
    padding: 4px 10px;\
    border-radius: 4px;\
    cursor: pointer;\
    display: inline-flex;\
    align-items: center;\
    gap: 4px;\
    font-size: 12px;\
}\
#esc-shortcuts-modal .esc-state-toggle-btn:focus {\
    outline: 2px solid #2563eb;\
    outline-offset: 2px;\
}\
#esc-shortcuts-modal .esc-state-toggle-icon {\
    font-size: 18px;\
    line-height: 1;\
}\
#esc-shortcuts-modal .esc-state-toggle-label {\
    font-size: 12px;\
    font-weight: 500;\
}\
#esc-shortcuts-modal .esc-state-cell-na {\
    color: #9ca3af;\
    font-style: italic;\
    font-size: 12px;\
}\
#esc-shortcuts-modal .esc-combo-input {\
    width: 120px; \
    padding: 5px 8px; \
    border-radius: 4px;\
    border: 1px solid #c5c5c5;\
    font-size: 13px;\
}\
#esc-shortcuts-modal .esc-combo-input:disabled {\
    background: #f0f0f0;\
    color: #555;\
}\
#esc-shortcuts-modal .esc-icon-hint {\
    font-size: 12px;\
    color: #777;\
}\
#esc-shortcuts-modal .esc-btn {\
    border: none;\
    border-radius: 4px;\
    padding: 6px 12px;\
    font-size: 13px;\
    cursor: pointer;\
}\
#esc-shortcuts-modal .esc-btn-primary {\
    background: #4c89ff;\
    color: #fff;\
}\
#esc-shortcuts-modal .esc-btn-secondary {\
    background: #ececec;\
    color: #333;\
}\
#esc-shortcuts-modal .esc-btn-ghost {\
    background: transparent;\
    color: #555;\
    border: 1px dashed #bbb;\
}\
#esc-shortcuts-modal .esc-btn:disabled {\
    opacity: 0.5;\
    cursor: default;\
}\
#esc-shortcuts-modal .esc-modal__footer {\
    margin-top: 12px;\
    display: flex;\
    align-items: center;\
    justify-content: space-between;\
    gap: 12px;\
}\
#esc-shortcuts-modal .esc-footer-buttons {\
    display: flex;\
    gap: 8px;\
}\
#esc-shortcuts-modal .esc-status {\
    font-size: 12px;\
    color: #666;\
    min-height: 18px;\
}\
#esc-shortcuts-modal .esc-status--success {\
    color: #0a7f44;\
}\
#esc-shortcuts-modal .esc-status--error {\
    color: #c0392b;\
}\
#esc-shortcuts-modal .esc-input-error {\
    border-color: #c0392b !important;\
}\
@keyframes esc-modal-in {\
    from {\
        opacity: 0;\
        transform: translateY(10px);\
    }\
    to {\
        opacity: 1;\
        transform: translateY(0);\
    }\
}\
#esc-launcher-btn {\
    position: fixed;\
    bottom: 60px;\
    left: 19%;\
    top: auto;\
    width: 44px;\
    height: 44px;\
    z-index: 99990;\
    background: #ffffff;\
    border: 1px solid rgba(60, 64, 67, 0.2);\
    border-radius: 50%;\
    padding: 0;\
    box-shadow: 0 4px 12px rgba(0, 0, 0, 0.15);\
    display: flex;\
    align-items: center;\
    justify-content: center;\
    cursor: pointer;\
    font-family: "Rubik", "Open Sans", sans-serif;\
    transition: transform 0.15s ease, box-shadow 0.15s ease;\
}\
#esc-launcher-btn:hover {\
    transform: translateY(-1px);\
    box-shadow: 0 6px 20px rgba(0, 0, 0, 0.2);\
}\
#esc-launcher-btn .esc-launcher__icon {\
    width: 28px;\
    height: 28px;\
    border-radius: 50%;\
    background: #f3f4f6;\
    display: flex;\
    align-items: center;\
    justify-content: center;\
    overflow: hidden;\
}\
#esc-launcher-btn .esc-launcher__icon img {\
    width: 20px;\
    height: 20px;\
}\
#esc-launcher-btn .esc-launcher__label {\
    display: none;\
}\
#esc-launcher-btn::after {\
    content: attr(data-tooltip);\
    position: absolute;\
    top: 50%;\
    transform: translateY(-50%);\
    left: 52px;\
    background: rgba(17, 24, 39, 0.9);\
    color: #fff;\
    padding: 3px 8px;\
    border-radius: 4px;\
    font-size: 11px;\
    opacity: 0;\
    pointer-events: none;\
    transition: opacity 0.15s ease;\
    white-space: nowrap;\
    font-family: "Rubik", "Open Sans", sans-serif;\
}\
#esc-launcher-btn:hover::after {\
    opacity: 1;\
}\
';
            document.head.appendChild(style);
        }

        return {
            ensure
        };
    })();


    // ---------------------------------------------------------------------
    //  Módulo: Configuración de Curvas (Datos + Interfaz + Control)
    // ---------------------------------------------------------------------
    const CurveSettings = (function () {
        const STORAGE_KEY = 'esc_curve_settings';
        
        // Lista COMPLETA de tipos de vía conducibles en Waze
        const roadTypes = {
            3: 'Autopista (Freeway)',
            6: 'Carretera Mayor (Major)',
            7: 'Carretera Menor (Minor)',
            4: 'Rampa',
            2: 'Calle Primaria (Primary)',
            1: 'Calle (Street)',
            8: 'Off-road / No mantenida',
            20: 'Vía de Estacionamiento', // <--- NUEVO
            22: 'Calle Estrecha (Narrow)' // <--- NUEVO
        };

        // Configuración por defecto para todos los tipos
        const defaultSettings = {
            3: { angle: 115, sampleDist: 45 },  // Autopista: mayor distancia por alta velocidad
            6: { angle: 115, sampleDist: 35 },  // Carretera Mayor: distancia estándar
            7: { angle: 115, sampleDist: 35 },  // Carretera Menor: distancia estándar
            4: { angle: 100, sampleDist: 30 },  // Rampa: reducida, curvas más cerradas
            2: { angle: 90, sampleDist: 35 },   // Calle Primaria: distancia estándar
            1: { angle: 90, sampleDist: 30 },   // Calle: reducida, velocidades más bajas
            8: { angle: 120, sampleDist: 20 },  // Off-road: muy reducida, curvas cerradas
            20: { angle: 90, sampleDist: 20 },  // Parking: muy reducida, velocidad muy baja
            22: { angle: 90, sampleDist: 25 }   // Narrow: reducida moderadamente
        };

        let currentSettings = loadSettings();

        function loadSettings() {
            try {
                const saved = localStorage.getItem(STORAGE_KEY);
                if (!saved) return JSON.parse(JSON.stringify(defaultSettings));
                const parsed = JSON.parse(saved);
                const merged = JSON.parse(JSON.stringify(defaultSettings));
                // Fusionar guardados con defaults para asegurar que existan todas las llaves
                Object.keys(merged).forEach(k => { 
                    if(parsed[k]) merged[k].angle = parsed[k].angle; 
                });
                return merged;
            } catch (e) {
                return JSON.parse(JSON.stringify(defaultSettings));
            }
        }

        function saveSettings(newSettings) {
            currentSettings = newSettings;
            localStorage.setItem(STORAGE_KEY, JSON.stringify(currentSettings));
            logInfo('Configuración de curvas guardada.');
            if (typeof CurveAnalyzer !== 'undefined' && CurveAnalyzer.isActive()) {
                CurveAnalyzer.refresh(); 
            }
        }

        function getSettingForType(roadTypeId) {
            // Si el tipo no está en la lista, devolvemos un valor muy alto para que se marque siempre (o nunca, según prefieras)
            // Usamos 180 para que sea evidente si se nos escapó algún tipo raro
            return currentSettings[roadTypeId] || { angle: 180 };
        }

        function showPopup() {
            UIStyles.ensure();

            const overlay = document.createElement('div');
            overlay.className = 'esc-modal-backdrop';
            overlay.style.cssText = "position:fixed;top:0;left:0;width:100%;height:100%;background:rgba(0,0,0,0.5);z-index:99999;display:flex;justify-content:center;align-items:center;";
            overlay.addEventListener('click', (e) => { if(e.target === overlay) document.body.removeChild(overlay); });

            const modal = document.createElement('div');
            modal.className = 'esc-modal';
            modal.style.cssText = "background:#fff;padding:20px;border-radius:8px;width:480px;max-width:95%;box-shadow:0 4px 20px rgba(0,0,0,0.3);font-family:'Rubik',sans-serif;display:flex;flex-direction:column;gap:15px;";

            const header = document.createElement('div');
            header.style.cssText = "display:flex;justify-content:space-between;align-items:center;border-bottom:1px solid #eee;padding-bottom:15px;";
            
            const title = document.createElement('h2');
            title.textContent = 'Panel de Curvas';
            title.style.margin = '0';
            title.style.fontSize = '24px';
            title.style.fontWeight = 'bold';

            const btnToggle = document.createElement('button');
            function updateBtnState() {
                const active = (typeof CurveAnalyzer !== 'undefined' && CurveAnalyzer.isActive());
                btnToggle.textContent = active ? '🟢 Análisis: ACTIVADO' : '🔴 Análisis: DESACTIVADO';
                btnToggle.style.cssText = active 
                    ? "background:#fff; color:#28a745; border:1px solid #28a745; padding: 5px 15px; font-weight:bold; border-radius:4px; cursor:pointer;"
                    : "background:#fff; color:#dc3545; border:1px solid #dc3545; padding: 5px 15px; font-weight:bold; border-radius:4px; cursor:pointer;";
            }
            btnToggle.onclick = () => {
                if (typeof CurveAnalyzer !== 'undefined') {
                    CurveAnalyzer.toggleCurveAnalysis();
                    updateBtnState();
                }
            };
            updateBtnState();

            header.appendChild(title);
            header.appendChild(btnToggle);
            modal.appendChild(header);

            const tableContainer = document.createElement('div');
            tableContainer.style.cssText = "max-height:60vh;overflow-y:auto;";
            
            const table = document.createElement('table');
            table.style.width = '100%';
            table.style.borderCollapse = 'collapse';
            
            const thead = document.createElement('thead');
            thead.innerHTML = `
                <tr style="background:#f8f9fa;text-align:left;font-size:13px;color:#555;">
                    <th style="padding:10px;">Tipo de Vía</th>
                    <th style="padding:10px;">Ángulo Máximo (°)</th>
                </tr>`;
            table.appendChild(thead);

            const tbody = document.createElement('tbody');
            // Ordenamos para que aparezcan en orden de importancia (Freeway primero)
            const order = [3, 6, 7, 4, 2, 1, 8, 20, 22]; 
            
            order.forEach(typeId => {
                const config = currentSettings[typeId] || { angle: 135 };
                const label = roadTypes[typeId] || 'Desconocido (' + typeId + ')';
                
                const row = document.createElement('tr');
                row.style.borderBottom = '1px solid #f0f0f0';
                row.innerHTML = `
                    <td style="padding:10px;">${label}</td>
                    <td style="padding:10px;">
                        <input type="number" id="angle-${typeId}" value="${config.angle}" min="1" max="180" style="width:80px;padding:5px;border:1px solid #ccc;border-radius:4px;"> °
                    </td>
                `;
                tbody.appendChild(row);
            });
            table.appendChild(tbody);
            tableContainer.appendChild(table);
            modal.appendChild(tableContainer);

            const footer = document.createElement('div');
            footer.style.cssText = "display:flex;justify-content:flex-end;gap:10px;margin-top:10px;padding-top:10px;border-top:1px solid #eee;";

            const btnSave = document.createElement('button');
            btnSave.textContent = 'Guardar y Aplicar';
            btnSave.className = 'esc-btn esc-btn-primary';
            btnSave.onclick = () => {
                const newConf = JSON.parse(JSON.stringify(currentSettings));
                Object.keys(roadTypes).forEach(typeId => {
                    const el = safeGetElementById(`angle-${typeId}`);
                    if (el) {
                         const a = parseInt(el.value) || 180;
                         newConf[typeId] = { angle: a };
                    }
                });
                saveSettings(newConf);
                document.body.removeChild(overlay);
            };

            footer.appendChild(btnSave);
            modal.appendChild(footer);

            overlay.appendChild(modal);
            document.body.appendChild(overlay);
        }

        return {
            get: getSettingForType,
            showConfig: showPopup
        };
    })();//curvesettings

        //**************************************************************************
        //Nombre: getCurveAnalyzerStyle
        //Fecha modificación: 2025-11-19
        //Hora: 17:00
        //Autor: mincho77
        //Entradas: feature (opcional)
        //Salidas: Objeto de estilo para la capa de curvas
        //Prerrequisitos si existen: Librería de mapa cargada
        //Descripción: Define el estilo visual de la capa del CurveAnalyzer
        //**************************************************************************
    function getCurveAnalyzerStyle(feature) 
    {
        return {
            color: '#ff0000',
            weight: 4,
            opacity: 0.9
        };
    }

 // ---------------------------------------------------------------------
    //  Módulo: Analizador de Curvas (Rumbo Acumulado - Método Profesional)
    // ---------------------------------------------------------------------
    const CurveAnalyzer = (function () {
        let wmeSDK = null;
        let isAnalyerActive = false;
        const LAYER_NAME = 'ESC_Curve_Analysis_Layer';
        let layerExists = false;

        function projectLatLonToMercator(lon, lat) {
            const rMajor = 6378137;
            const x = rMajor * (lon * Math.PI / 180);
            const y = rMajor * Math.log(Math.tan((Math.PI / 4) + (lat * Math.PI / 360)));
            return { x: x, y: y };
        }

        // Calcula el rumbo (azimuth) en grados (0-360) entre dos puntos
        function getBearing(p1, p2) {
            const dx = p2.x - p1.x;
            const dy = p2.y - p1.y;
            let theta = Math.atan2(dx, dy); // Usamos dx, dy para que 0 sea Norte
            let bearing = theta * (180 / Math.PI);
            if (bearing < 0) bearing += 360;
            return bearing;
        }

        // Calcula la diferencia angular absoluta entre dos rumbos (0-180)
        function getAngleDiff(b1, b2) {
            let diff = Math.abs(b1 - b2);
            if (diff > 180) diff = 360 - diff;
            return diff;
        }

        function getDistance(p1, p2) {
            return Math.sqrt(Math.pow(p2.x - p1.x, 2) + Math.pow(p2.y - p1.y, 2));
        }

        // Encuentra un punto a cierta distancia (aprox) hacia atrás o adelante en la línea
        function findPointAtDistance(points, startIndex, targetDist, direction) {
            let currentDist = 0;
            let i = startIndex;
            
            while (currentDist < targetDist) {
                let nextI = i + direction;
                if (nextI < 0 || nextI >= points.length) return points[i]; // Fin de línea
                
                currentDist += getDistance(points[i], points[nextI]);
                i = nextI;
            }
            return points[i];
        }

     //**************************************************************************
//Nombre: ensureLayer
//Fecha modificación: 2025-11-19
//Hora: 17:10
//Autor: mincho77
//Entradas: Ninguna
//Salidas: Ninguna
//Prerrequisitos si existen: wmeSDK.Map disponible
//Descripción: Crea (una sola vez) la capa de análisis de curvas en el mapa.
//**************************************************************************
async function ensureLayer() {
    // Si ya se creó antes, no hacemos nada
    if (layerExists) {
        return;
    }

    // Sin SDK no hay capa
    if (!wmeSDK || !wmeSDK.Map || !wmeSDK.Map.addLayer) {
        logWarn('CurveAnalyzer: SDK de mapa no disponible; no se puede crear la capa.');
        return;
    }

        try {
        const layerConfig = {
            layerName: LAYER_NAME,
            title: 'Curve Analyzer',
            type: 'feature',
            visible: isAnalyerActive, // o true si quieres que siempre se vea
            zIndex: 310
        };

        await wmeSDK.Map.addLayer(layerConfig);
        layerExists = true;
        logInfo('CurveAnalyzer: capa creada: ' + LAYER_NAME);
    } catch (e) {
        logWarn('CurveAnalyzer: error creando capa: ' + e.message);
    }
}
       //**************************************************************************
//Nombre: analyzeAndDrawCurves
//Fecha modificación: 2025-11-19
//Hora: 17:25
//Autor: mincho77
//Entradas: Ninguna
//Salidas: Dibuja features de puntos peligrosos en la capa de curvas
//Prerrequisitos si existen: wmeSDK inicializado, capa creada por ensureLayer
//Descripción: Analiza todos los segmentos visibles y marca puntos con
//             ángulo interno menor al umbral configurado por tipo de vía.
//**************************************************************************
async function analyzeAndDrawCurves() {
    if (!wmeSDK) return;

    // Asegura que la capa exista (usa wmeSDK.Map.addLayer)
    await ensureLayer();
    if (!layerExists) return;

    // Limpia features anteriores
    try {
        await wmeSDK.Map.removeAllFeaturesFromLayer({ layerName: LAYER_NAME });
    } catch (e) {
        logWarn('CurveAnalyzer: error limpiando capa: ' + e.message);
    }

    logInfo('--- ANÁLISIS DE RUMBO (HEADING) ---');

    let segments = [];
    try {
        segments = await wmeSDK.DataModel.Segments.getAll();
    } catch (e) {
        logWarn('CurveAnalyzer: error obteniendo segmentos: ' + e.message);
        return;
    }

    let segmentsFound = 0;
    const featuresToDraw = [];

    for (const segment of segments) {
        // 1) Geometría
        let coordinates = null;
        if (typeof segment.getOLGeometry === 'function') {
            const olGeometry = segment.getOLGeometry();
            if (!olGeometry || olGeometry.getType() !== 'LineString') continue;
            coordinates = olGeometry.getCoordinates();
        } else if (!segment.getOLGeometry && segment.geometry && segment.geometry.type === 'LineString') {
            coordinates = segment.geometry.coordinates;
        } else {
            continue; // sin línea, sin fiesta
        }

        if (!coordinates || coordinates.length < 3) continue;

        const points = coordinates.map(c => projectLatLonToMercator(c[0], c[1]));

        // 2) Tipo de vía
        let roadType = null;
        if (segment.attributes && typeof segment.attributes.roadType !== 'undefined') {
            roadType = segment.attributes.roadType;
        } else if (typeof segment.roadType !== 'undefined') {
            roadType = segment.roadType;
        }
        if (roadType === null) roadType = 99;
        if (roadType === 17) continue; // tipo 17: ignorado

        // 3) Config de curvas
        const config = CurveSettings.get(roadType);
        const SAMPLEDIST = config.sampleDist || 35;
        const thresholdAngle = config.angle || 140;

        // 3.5) Verificar si el segmento es de un solo sentido y su dirección
        let isOneWay = false;
        let isReverseOnly = false;

        let fwd = null;
        let rev = null;

        // Intentar leer de attributes
        if (segment.attributes) {
            if (segment.attributes.fwdDirection !== undefined) fwd = segment.attributes.fwdDirection;
            if (segment.attributes.revDirection !== undefined) rev = segment.attributes.revDirection;
        }
        
        // Fallback: leer directo del objeto
        if (fwd === null && segment.fwdDirection !== undefined) fwd = segment.fwdDirection;
        if (rev === null && segment.revDirection !== undefined) rev = segment.revDirection;

        if (fwd !== null && rev !== null) {
            // Si solo una dirección está habilitada, es one-way
            isOneWay = (fwd && !rev) || (!fwd && rev);
            // Determinar si el tráfico va solo en dirección reversa
            isReverseOnly = (!fwd && rev);
        }

        // 4) Barrido de puntos intermedios - coleccionamos primero todos los puntos peligrosos
        const dangerousPoints = [];
        for (let i = 1; i < points.length - 1; i++) {
            const current = points[i];
            const prev = findPointAtDistance(points, i, SAMPLEDIST, -1);
            const next = findPointAtDistance(points, i, SAMPLEDIST, 1);
            if (!prev || !next) continue;

            const bearingIn = getBearing(prev, current);
            const bearingOut = getBearing(current, next);
            const turnAngle = getAngleDiff(bearingIn, bearingOut);
            const internalAngle = 180 - turnAngle;

            // Rectas casi perfectas
            if (internalAngle > 175) continue;

            // Marca cada punto "peligroso" según umbral
            if (internalAngle < thresholdAngle) {
                dangerousPoints.push({
                    index: i,
                    angle: internalAngle,
                    coordinate: coordinates[i]
                });
            }
        }

        // 5) Agrupar puntos peligrosos en curvas consecutivas
        // Un segmento largo puede tener varias curvas separadas por tramos rectos
        const curves = [];
        let currentCurve = [];
        
        for (let i = 0; i < dangerousPoints.length; i++) {
            if (currentCurve.length === 0) {
                // Empezar nueva curva
                currentCurve.push(dangerousPoints[i]);
            } else {
                // Verificar si este punto es consecutivo al anterior
                const lastPoint = currentCurve[currentCurve.length - 1];
                const gap = dangerousPoints[i].index - lastPoint.index;
                
                if (gap <= 5) {  // Si están cerca (máximo 5 puntos de separación), misma curva
                    currentCurve.push(dangerousPoints[i]);
                } else {
                    // Hay un gap grande, terminar curva actual y empezar nueva
                    curves.push([...currentCurve]);
                    currentCurve = [dangerousPoints[i]];
                }
            }
        }
        
        // No olvidar la última curva
        if (currentCurve.length > 0) {
            curves.push(currentCurve);
        }

        // 6) Para cada curva individual, seleccionar puntos a mostrar
        for (const curve of curves) {
            if (curve.length === 0) continue;
            
            const indicesToShow = [];
            
            if (isOneWay) {
                // Si es de un solo sentido, mostramos solo el punto de inicio de ESTA curva
                // Si es reverse, el "inicio" para el conductor es el último punto de la curva
                if (isReverseOnly) {
                    indicesToShow.push(curve.length - 1); // Último punto de la curva (inicio en reverse)
                } else {
                    indicesToShow.push(0); // Primer punto de la curva (inicio en forward)
                }
            } else if (curve.length <= 3) {
                // Si es doble sentido y la curva tiene 3 o menos puntos, mostramos todos
                indicesToShow.push(...curve.map((_, idx) => idx));
            } else {
                // Si es doble sentido y la curva tiene más de 3 puntos
                indicesToShow.push(0); // Primer punto (inicio en forward)
                const middleIdx = Math.floor(curve.length / 2);
                indicesToShow.push(middleIdx); // Punto del medio
                indicesToShow.push(curve.length - 1); // Último punto (inicio en reverse)
            }

            // Creamos las features solo para los puntos seleccionados de ESTA curva
            for (const idx of indicesToShow) {
                const point = curve[idx];
                const angleText = point.angle.toFixed(0);

                const pointFeature = {
                    type: 'Feature',
                    id: 'curve_' + segment.id + '_' + point.index,
                    geometry: {
                        type: 'Point',
                        coordinates: point.coordinate
                    },
                    properties: {
                        angle: angleText
                    },
                    style: {
                        pointRadius: 6,
                        fillColor: '#FF0000',
                        fillOpacity: 0.8,
                        strokeColor: '#FFFFFF',
                        strokeWidth: 2,
                        label: angleText + '\u00B0',
                        fontColor: 'black',
                        fontSize: '14px',
                        fontWeight: 'bold',
                        fontFamily: 'Arial, sans-serif',
                        labelOutlineColor: 'white',
                        labelOutlineWidth: 3,
                        labelAlign: 'cm',
                        labelYOffset: -20
                    }
                };

                // Debug fino por si acaso
                console.log('PUSHING curve feature:', pointFeature);
                featuresToDraw.push(pointFeature);
                segmentsFound++;
            }
        }
    }

    // 5) Dibujar en la capa
    if (featuresToDraw.length > 0) {
        //console.log('ADDING features to layer', LAYER_NAME, featuresToDraw);
        await wmeSDK.Map.addFeaturesToLayer({
            layerName: LAYER_NAME,
            features: featuresToDraw
        });
    }

    logInfo('Análisis (Rumbo) completado. ' + segmentsFound + ' puntos.');
}
        async function clearAnalysis() {
            if (layerExists && wmeSDK) {
                try { await wmeSDK.Map.removeAllFeaturesFromLayer({ layerName: LAYER_NAME }); } catch (e) {}
            }
        }

        function toggleCurveAnalysis() {
            if (!wmeSDK) return;
            isAnalyerActive = !isAnalyerActive;
            if (isAnalyerActive) {
                logInfo('Análisis: ACTIVADO');
                analyzeAndDrawCurves();
            } else {
                logInfo('Análisis: DESACTIVADO');
                clearAnalysis();
            }
        }
        
        function isActive() { return isAnalyerActive; }
        function refresh() { if (isAnalyerActive) analyzeAndDrawCurves(); }
        function setSDK(sdk) { wmeSDK = sdk; }

        return { toggleCurveAnalysis, setSDK, isActive, refresh };
    })();//curveanalyzer

    // ---------------------------------------------------------------------
    //  Módulo: Visualizador de Zonas Escolares (Places)
    // ---------------------------------------------------------------------
    const SchoolZoneVisualizer = (function () {
        let wmeSDK = null;
        let isActive = false;
        const LAYER_NAME = 'ESC_School_Highlights';
        let layerExists = false;

        // Categorías de Waze que indican zona escolar
        const SCHOOL_CATEGORIES = ['SCHOOL', 'COLLEGE_UNIVERSITY', 'KINDERGARDEN'];

        async function ensureLayer() {
            if (layerExists || !wmeSDK) return;
            try {
                const layerConfig = {
                    layerName: LAYER_NAME,
                    title: 'Zonas Escolares',
                    type: 'feature',
                    visible: isActive,
                    zIndex: 320, // Un poco por encima de las curvas
                    styleRules: [{
                        style: {
                            strokeColor: '#FF0000',              // Rojo brillante para el borde
                            strokeOpacity: 1.0,
                            strokeWidth: 8,                      // Borde más grueso
                            fill: true,
                            fillColor: '#FFFF00',                // Amarillo brillante
                            fillOpacity: 0.7,                    // 70% opacidad
                            pointRadius: 15,
                            label: "🏫 ZONA ESCOLAR",
                            fontColor: "#FF0000",                // Texto rojo brillante
                            labelOutlineColor: "#FFFFFF",        // Contorno blanco para contraste
                            labelOutlineWidth: 4,     
                            fontSize: "14px",                    // Fuente más grande
                            fontWeight: "bold",       
                            labelYOffset: -22
                        }
                    }]
                };
                await wmeSDK.Map.addLayer(layerConfig);
                layerExists = true;
                logInfo('SchoolZoneVisualizer: capa creada.');
            } catch (e) {
                logWarn('SchoolZoneVisualizer: error creando capa: ' + e.message);
            }
        }

        async function highlightSchools() {
            console.log('[DEBUG SchoolZone] 1. Iniciando highlightSchools');
            if (!wmeSDK) {
                console.error('[DEBUG SchoolZone] ERROR: wmeSDK no está disponible');
                return;
            }
            console.log('[DEBUG SchoolZone] 2. SDK disponible, creando capa...');
            await ensureLayer();
            if (!layerExists) {
                console.error('[DEBUG SchoolZone] ERROR: No se pudo crear la capa');
                return;
            }
            console.log('[DEBUG SchoolZone] 3. Capa creada exitosamente');

            // Limpiar cualquier marca anterior
            try { await wmeSDK.Map.removeAllFeaturesFromLayer({ layerName: LAYER_NAME }); } catch (e) {}

            logInfo('--- BUSCANDO ZONAS ESCOLARES ---');
            
            // Obtener todos los lugares (venues)
            let venues = [];
            console.log('[DEBUG SchoolZone] 4. Intentando obtener venues...');
            if (wmeSDK.DataModel.Venues && typeof wmeSDK.DataModel.Venues.getAll === 'function') {
                console.log('[DEBUG SchoolZone] 5. Usando wmeSDK.DataModel.Venues.getAll()');
                venues = await wmeSDK.DataModel.Venues.getAll();
            } else if (typeof W !== 'undefined' && W.model && W.model.venues) {
                console.log('[DEBUG SchoolZone] 5. Fallback: Usando W.model.venues.getObjectArray()');
                venues = W.model.venues.getObjectArray();
            } else {
                console.error('[DEBUG SchoolZone] ERROR: No se pudo obtener venues por ningún método');
            }
            console.log('[DEBUG SchoolZone] 6. Total de venues obtenidos:', venues.length);

            const featuresToDraw = [];
            let totalVenuesChecked = 0;
            let schoolsFound = 0;

            for (const venue of venues) {
                totalVenuesChecked++;
                // Obtener categorías
                const cats = venue.attributes ? venue.attributes.categories : venue.categories;
                if (!cats || cats.length === 0) continue;

                // Verificar si alguna categoría es escolar
                const isSchool = cats.some(c => SCHOOL_CATEGORIES.includes((c || '').toUpperCase()));
                if (isSchool) {
                    schoolsFound++;
                    const venueName = venue.attributes?.name || venue.name || 'sin nombre';
                    console.log('[DEBUG SchoolZone] 7. Escuela encontrada:', venueName, '| Categorías:', cats);
                }
                if (!isSchool) continue;

                // Obtener geometría
                let geometry = null;
                if (typeof venue.getOLGeometry === 'function') {
                    const olGeo = venue.getOLGeometry();
                    if (olGeo) geometry = olGeo;
                } else if (!venue.getOLGeometry && venue.geometry) {
                    geometry = venue.geometry;
                }

                if (!geometry) continue;

                let featureGeo = null;
                let isPoint = false;

                if (geometry.getType && typeof geometry.getType === 'function') {
                    const type = geometry.getType();
                    const coords = geometry.getCoordinates();
                    if (type === 'Point') {
                        featureGeo = { type: 'Point', coordinates: coords };
                        isPoint = true;
                    } else if (type === 'Polygon') {
                        featureGeo = { type: 'Polygon', coordinates: coords };
                    } else if (type === 'MultiPolygon') {
                        featureGeo = { type: 'MultiPolygon', coordinates: coords };
                    }
                } else {
                    featureGeo = geometry;
                    if (featureGeo && featureGeo.type === 'Point') isPoint = true;
                }

                if (!featureGeo) continue;

                const schoolFeature = {
                    type: 'Feature',
                    id: 'school_highlight_' + (venue.attributes ? venue.attributes.id : venue.id),
                    geometry: featureGeo,
                    properties: {
                        type: 'school_zone'
                    }
                };
                console.log('[DEBUG SchoolZone] 8. Feature creada:', schoolFeature);
                featuresToDraw.push(schoolFeature);
            }

            console.log('[DEBUG SchoolZone] 9. Resumen - Venues revisados:', totalVenuesChecked, '| Escuelas encontradas:', schoolsFound, '| Features a dibujar:', featuresToDraw.length);
            
            if (featuresToDraw.length > 0) {
                console.log('[DEBUG SchoolZone] 10. Agregando features a la capa:', LAYER_NAME);
                try {
                    await wmeSDK.Map.addFeaturesToLayer({
                        layerName: LAYER_NAME,
                        features: featuresToDraw
                    });
                    console.log('[DEBUG SchoolZone] 11. Features agregadas exitosamente ✓');
                    logInfo('Marcadas ' + featuresToDraw.length + ' zonas escolares.');
                } catch (e) {
                    console.error('[DEBUG SchoolZone] ERROR al agregar features:', e);
                }
            } else {
                console.warn('[DEBUG SchoolZone] No hay features para dibujar');
                logInfo('No se encontraron zonas escolares en el área visible.');
            }
        }

        async function clearHighlights() {
            if (layerExists && wmeSDK) {
                try { await wmeSDK.Map.removeAllFeaturesFromLayer({ layerName: LAYER_NAME }); } catch (e) {}
            }
        }

        function toggle() {
            console.log('[DEBUG SchoolZone] Toggle llamado. Estado actual:', isActive);
            if (!wmeSDK) {
                console.error('[DEBUG SchoolZone] ERROR: SDK no disponible en toggle');
                logWarn('Visualizador: SDK no listo.');
                return;
            }
            isActive = !isActive;
            console.log('[DEBUG SchoolZone] Nuevo estado:', isActive);
            if (isActive) {
                logInfo('Visualizador Escuelas: ACTIVADO');
                highlightSchools();
            } else {
                logInfo('Visualizador Escuelas: DESACTIVADO');
                clearHighlights();
            }
        }

        function setSDK(sdk) { wmeSDK = sdk; }

        // Agregamos 'isActive' para que el botón sepa si ponerse verde o rojo
        return { 
            toggle, 
            setSDK, 
            isActive: function() { return isActive; } 
        };
    })();



    // ---------------------------------------------------------------------
    //  Módulo: Normalización de teclas y combos
    // ---------------------------------------------------------------------
    const KeyUtils = (function () {
        const ARROW_MAP = {
            ArrowUp: 'UP',
            ArrowDown: 'DOWN',
            ArrowLeft: 'LEFT',
            ArrowRight: 'RIGHT'
        };
        const ARROW_NORMALIZED_MAP = {
            ARROWUP: 'UP',
            ARROWDOWN: 'DOWN',
            ARROWLEFT: 'LEFT',
            ARROWRIGHT: 'RIGHT',
            UP: 'UP',
            DOWN: 'DOWN',
            LEFT: 'LEFT',
            RIGHT: 'RIGHT'
        };
        const SHIFTED_DIGIT_MAP = {
            '!': '1',
            '@': '2',
            '#': '3',
            '$': '4',
            '%': '5',
            '^': '6',
            '&': '7',
            '*': '8',
            '(': '9',
            ')': '0'
        };

        /**************************************************************************
        //Nombre: normalizeComboFromEvent
        //Fecha modificación: 2025-11-15
        //Hora: 11:30
        //Autor: mincho77
        //Entradas: evt (KeyboardEvent)
        //Salidas: string con el combo normalizado o null si no aplica
        //Prerrequisitos si existen: Evento de teclado válido
        //Descripción: Convierte un KeyboardEvent en una cadena tipo
        //             "ALT+SHIFT+Z" o "UP". Ignora eventos repetidos.
        **************************************************************************/
        function normalizeComboFromEvent(evt, options) {
            if (!evt || evt.repeat) {
                return null;
            }

            // Evitar capturar ShortCuts cuando el usuario está escribiendo
            const opts = options || {};
            const allowEditableTargets = !!opts.allowEditableTargets;
            if (!allowEditableTargets) {
                const target = evt.target;
                if (target) {
                    const tagName = (target.tagName || '').toLowerCase();
                    const isEditable = target.isContentEditable;
                    if (tagName === 'input' || tagName === 'textarea' || isEditable) {
                        return null;
                    }
                }
            }

            const parts = [];
            if (evt.ctrlKey) {
                parts.push('CTRL');
            }
            if (evt.altKey) {
                parts.push('ALT');
            }
            if (evt.shiftKey) {
                parts.push('SHIFT');
            }
            if (evt.metaKey) {
                parts.push('CMD');
            }

            let key = evt.key;
            if (!key) {
                return null;
            }

            if (ARROW_MAP[key]) {
                key = ARROW_MAP[key];
            } else if (key.length === 1) {
                // Si es un símbolo de número con SHIFT (por ejemplo !, @, #),
                // lo mapeamos al dígito correspondiente para combos tipo CTRL+SHIFT+1
                if (evt.shiftKey && SHIFTED_DIGIT_MAP[key]) {
                    key = SHIFTED_DIGIT_MAP[key];
                } else {
                    key = key.toUpperCase();
                }
            } else {
                // Normalizar algunos casos especiales si fueran necesarios
                key = key.toUpperCase();
            }

            parts.push(key);
            return parts.join('+');
        }

        function normalizeComboString(comboText) {
            if (!comboText) {
                return '';
            }

            const rawParts = comboText.split('+');
            const modifierFlags = {
                CTRL: false,
                ALT: false,
                SHIFT: false,
                CMD: false
            };
            let keyPart = '';

            rawParts.forEach(function (part) {
                const trimmed = part.trim();
                if (!trimmed) {
                    return;
                }
                const upper = trimmed.toUpperCase();
                if (upper === 'CTRL' || upper === 'CONTROL' || upper === 'STRG') {
                    modifierFlags.CTRL = true;
                    return;
                }
                if (upper === 'ALT' || upper === 'OPTION') {
                    modifierFlags.ALT = true;
                    return;
                }
                if (upper === 'SHIFT') {
                    modifierFlags.SHIFT = true;
                    return;
                }
                if (upper === 'CMD' || upper === 'COMMAND' || upper === 'META') {
                    modifierFlags.CMD = true;
                    return;
                }

                if (ARROW_NORMALIZED_MAP[upper]) {
                    keyPart = ARROW_NORMALIZED_MAP[upper];
                    return;
                }

                keyPart = upper.length === 1 ? upper : upper;
            });

            const ordered = [];
            if (modifierFlags.CTRL) {
                ordered.push('CTRL');
            }
            if (modifierFlags.ALT) {
                ordered.push('ALT');
            }
            if (modifierFlags.SHIFT) {
                ordered.push('SHIFT');
            }
            if (modifierFlags.CMD) {
                ordered.push('CMD');
            }
            if (keyPart) {
                ordered.push(keyPart);
            }

            return ordered.join('+');
        }

        return {
            normalizeComboFromEvent,
            normalizeComboString
        };
    })();

    // ---------------------------------------------------------------------
    //  Módulo: Persistencia de overrides (localStorage simple)
    // ---------------------------------------------------------------------

    const ShortcutsStorage = (function () {
        const STORAGE_KEY = 'wme_easy_shortcuts_user_combos';
        let comboCache = null;

        function ensureLoaded() {
            if (comboCache !== null) {
                return;
            }

            comboCache = {};
            try {
                const raw = window.localStorage.getItem(STORAGE_KEY);
                if (!raw) {
                    return;
                }
                const parsed = JSON.parse(raw);
                if (parsed && typeof parsed === 'object') {
                    comboCache = parsed;
                }
            } catch (err) {
                logWarn('ShortcutsStorage: error leyendo overrides -> ' + err.message);
                comboCache = {};
            }
        }

        function persist() {
            if (!comboCache || Object.keys(comboCache).length === 0) {
                window.localStorage.removeItem(STORAGE_KEY);
                return;
            }

            try {
                window.localStorage.setItem(STORAGE_KEY, JSON.stringify(comboCache));
            } catch (err) {
                logWarn('ShortcutsStorage: error guardando overrides -> ' + err.message);
            }
        }

        function getOverride(actionId) {
            if (!actionId) {
                return undefined;
            }
            ensureLoaded();
            if (Object.prototype.hasOwnProperty.call(comboCache, actionId)) {
                return comboCache[actionId];
            }
            return undefined;
        }

        function setOverride(actionId, combo) {
            if (!actionId) {
                return;
            }
            ensureLoaded();
            comboCache[actionId] = combo;
            persist();
        }

        function clearOverride(actionId) {
            if (!actionId) {
                return;
            }
            ensureLoaded();
            if (Object.prototype.hasOwnProperty.call(comboCache, actionId)) {
                delete comboCache[actionId];
                persist();
            }
        }

        function resetAll() {
            comboCache = {};
            window.localStorage.removeItem(STORAGE_KEY);
        }

        return {
            getOverride,
            setOverride,
            clearOverride,
            resetAll
        };
    })();

    // ---------------------------------------------------------------------
    //  Módulo: Registro híbrido de ShortCuts
    // ---------------------------------------------------------------------

    const ShortcutRegistry = (function () {
        // Combos nativos reservados (tomados de la ayuda de ShortCuts de WME)
        // Formato: igual que devuelve KeyUtils.normalizeComboFromEvent
        const BUILTIN_RESERVED_COMBOS = new Set([
            // General
            'Z',              // Undo
            'SHIFT+Z',        // Redo
            'S',              // Save
            'F',              // Focus search bar
            'ALT+SHIFT+R',    // Refresh map data
            ']',              // Issue tracker next
            '[',              // Issue tracker previous

            // Drawing
            'I',              // New road
            'O',              // New roundabout
            'J',              // Junction box
            'P',              // Pedestrian path
            'N',              // Map note
            'B',              // Restricted area
            'G',              // Residential place
            'X',              // Speed camera

            // Editing
            'M',              // Toggle multiple selection
            'W',              // Allow all turns
            'Q',              // Disallow all turns
            'R',              // Toggle direction
            'E',              // Edit address
            'H',              // House numbers
            'C',              // Closures
            'T',              // Restrictions
            'A',              // Select entire street
            'ALT+P',          // Copy permalink
            'ALT+SHIFT+P',    // Set permalink to URL
            'UP',             // Increase elevation
            'DOWN',           // Decrease elevation
            'D',              // Delete vertex

            // Layers
            'L',              // Toggle layer switcher
            'SHIFT+D',        // Highlight segments with no name
            'SHIFT+W',        // Close Street View
            'SHIFT+I',        // Toggle satellite imagery
            'SHIFT+G',        // Toggle GPS points
            'SHIFT+R',        // Toggle roads
            'SHIFT+M',        // Toggle map notes
            'SHIFT+A',        // Toggle area managers
            'SHIFT+B',        // Toggle junction boxes
            'SHIFT+H',        // Toggle house numbers
            'SHIFT+P',        // Toggle map problems
            'SHIFT+U',        // Toggle update requests
            'SHIFT+E',        // Toggle editable areas
            'SHIFT+C',        // Toggle road closures
            'SHIFT+V',        // Toggle online editors
            'SHIFT+Z',        // Show disallowed turns
            'SHIFT+L',        // Toggle places

            // Map navigation
            'SHIFT+UP',       // Zoom in
            'SHIFT+DOWN',     // Zoom out
            'LEFT',           // Pan left
            'RIGHT'           // Pan right
        ]);

        // Acciones registradas por el script
        const actionsById = new Map();
        const actionsByCombo = new Map();

        function init() {
            // En modo híbrido, partimos de la lista reservada estática.
            // Si en el futuro usamos el SDK para leer ShortCuts reales, se puede
            // expandir esta función sin tocar el resto del módulo.
            logInfo('ShortcutRegistry iniciado con modo híbrido estático.');
        }

        function isComboReserved(combo, actionId) {
            if (!combo) {
                return false;
            }
            const owner = actionsByCombo.get(combo);
            if (owner && owner.id !== actionId) {
                return true;
            }
            return BUILTIN_RESERVED_COMBOS.has(combo);
        }

        function registerAction(def) {
            if (!def || !def.id) {
                return;
            }

            const existing = actionsById.get(def.id);
            if (existing) {
                logWarn('Acción ya registrada: ' + def.id);
                return;
            }

            const action = Object.assign({}, def);
            action.defaultCombo = def.combo || null;

            actionsById.set(action.id, action);

            const overrideCombo = ShortcutsStorage.getOverride(action.id);
            if (typeof overrideCombo !== 'undefined') {
                const result = setActionCombo(action.id, overrideCombo, { skipStorage: true });
                if (!result.ok) {
                    logWarn('Override inválido para "' + action.label + '": ' + result.reason);
                    action.combo = action.defaultCombo || null;
                    registerKeyboardCombo(action);
                }
                return;
            }

            registerKeyboardCombo(action);
        }

        function registerKeyboardCombo(action) {
            if (!action || !action.handler || typeof action.handler !== 'function') {
                return;
            }
            if (!action.combo) {
                return;
            }

            if (isComboReserved(action.combo, action.id)) {
                logWarn('Combo reservado o en uso, se omite registro de teclado para: ' + action.label + ' [' + action.combo + ']');
                return;
            }

            actionsByCombo.set(action.combo, action);
            logInfo('Registrado ShortCut personalizado: ' + action.label + ' [' + action.combo + ']');
        }

        function unregisterKeyboardCombo(action) {
            if (!action || !action.combo) {
                return;
            }
            const owner = actionsByCombo.get(action.combo);
            if (owner && owner.id === action.id) {
                actionsByCombo.delete(action.combo);
            }
        }

        function setActionCombo(actionId, requestedCombo, options) {
            const action = actionsById.get(actionId);
            if (!action) {
                return { ok: false, reason: 'Acción no encontrada' };
            }

            const normalized = KeyUtils.normalizeComboString(requestedCombo || '');
            const newCombo = normalized || null;
            const previousCombo = action.combo || null;

            if (previousCombo === newCombo) {
                return { ok: true, changed: false, combo: newCombo };
            }

            if (action.handler && newCombo && isComboReserved(newCombo, action.id)) {
                return {
                    ok: false,
                    reason: 'ShortCut ya usado por otro script o reservado por WME.'
                };
            }

            unregisterKeyboardCombo(action);
            action.combo = newCombo;
            registerKeyboardCombo(action);

            const skipStorage = options && options.skipStorage;
            if (!skipStorage) {
                if (action.defaultCombo === newCombo) {
                    ShortcutsStorage.clearOverride(action.id);
                } else {
                    ShortcutsStorage.setOverride(action.id, newCombo);
                }
            }

            return { ok: true, changed: true, combo: newCombo };
        }

        function resetAllCombos() {
            actionsById.forEach(function (action) {
                unregisterKeyboardCombo(action);
                action.combo = action.defaultCombo || null;
                registerKeyboardCombo(action);
            });
            ShortcutsStorage.resetAll();
        }

        function getActionById(actionId) {
            return actionsById.get(actionId) || null;
        }

        function findActionByCombo(combo) {
            return actionsByCombo.get(combo) || null;
        }

        function getAllActions() {
            return Array.from(actionsById.values());
        }

        return {
            init,
            registerAction,
            findActionByCombo,
            getAllActions,
            isComboReserved,
            setActionCombo,
            resetAllCombos,
            getActionById
        };
    })();

    // ---------------------------------------------------------------------
    //  Módulo: Listener global de teclado (solo para combos personalizados)
    // ---------------------------------------------------------------------

    const KeyboardListener = (function () {
        function onKeyDown(evt) {
            const combo = KeyUtils.normalizeComboFromEvent(evt);
            if (!combo) {
                return;
            }

            const action = ShortcutRegistry.findActionByCombo(combo);
            if (!action || !action.handler) {
                return;
            }

            // LOGS: Combo detectado y acción a ejecutar
            logInfo('Combo detectado: ' + combo);
            logInfo('Ejecutando acción: ' + action.id + ' [' + action.label + ']');

            // Si el ShortCut es nuestro, evitamos que llegue al editor
            evt.preventDefault();
            evt.stopPropagation();

            try {
                action.handler();
            } catch (e) {
                logWarn('Error ejecutando acción "' + action.label + '": ' + e.message);
            }
        }

        function attach() {
            document.addEventListener('keydown', onKeyDown, true);
            logInfo('KeyboardListener adjuntado.');
        }

        return {
            attach
        };
    })();

    // ---------------------------------------------------------------------
    //  Módulo: Acciones de Hazards y Places (DOM puro)
    // ---------------------------------------------------------------------

    const HazardAndPlaceActions = (function () {
        // Definición de acciones: algunas son nativas (solo para mostrar hint),
        // otras son personalizadas (ejecutadas por este script).

        const ACTION_DEFS = [
            // --- Hazards ---
            {
                id: 'hazard-speed-bump-native',
                group: 'Peligros',
                label: 'Reductor de velocidad (nativo)',
                icon: 'Speed bump',
                menuLabel: 'Speed bump',
                menuAltLabels: ['Reductor de velocidad', 'Resalto'],
                combo: 'Z',        // ShortCut nativo de WME
                type: 'native',
                handler: null
            },
            {
                id: 'hazard-sharp-curve',
                group: 'Peligros',
                label: 'Curva pronunciada',
                icon: 'Sharp curve',
                menuLabel: 'Sharp curve',
                menuAltLabels: ['Curva pronunciada', 'Curva peligrosa'],
                combo: 'CTRL+SHIFT+1',    // ShortCut personalizado
                type: 'custom',
                handler: function () {
                    triggerHazardByLabels(['Sharp curve', 'Curva pronunciada', 'Curva peligrosa']);
                }
            },
            {
                id: 'hazard-analyze-curves-unified', // ID único
                group: 'Peligros',
                label: 'Analizar Curvas (Config)',   // Este texto saldrá en azul
                icon: 'Sharp curve',
                menuLabel: 'Analizar Curvas',
                combo: 'CTRL+SHIFT+A',              // El atajo que activa/desactiva
                type: 'custom',
                stateControl: {
                    label: 'Curvas',
                    tooltip: 'Análisis de curvas',
                    toggle: function () {
                        CurveAnalyzer.toggleCurveAnalysis();
                    },
                    getState: function () {
                        return typeof CurveAnalyzer !== 'undefined' && CurveAnalyzer.isActive();
                    }
                },

                // 1. Esto pasa cuando presionas las TECLAS (On/Off)
                handler: function () {
                    CurveAnalyzer.toggleCurveAnalysis();
                },
                
                // 2. Esto pasa cuando haces CLICK en el texto de la lista (Abre Panel)
                configHandler: function () {
                    CurveSettings.showConfig();
                }
            },
            {
                id: 'hazard-complex-intersection',
                group: 'Peligros',
                label: 'Intersección compleja',
                icon: 'Complex intersection',
                menuLabel: 'Complex intersection',
                menuAltLabels: ['Intersección compleja'],
                combo: 'CTRL+SHIFT+2',
                type: 'custom',
                handler: function () {
                    triggerHazardByLabels(['Complex intersection', 'Intersección compleja']);
                }
            },
            {
                id: 'hazard-multiple-lanes',
                group: 'Peligros',
                label: 'Convergencia de carriles',
                icon: 'Multiple lanes merging',
                menuLabel: 'Multiple lanes merging',
                menuAltLabels: ['Convergencia de carriles', 'Carriles se unen'],
                combo: 'CTRL+SHIFT+3',
                type: 'custom',
                handler: function () {
                    triggerHazardByLabels(['Multiple lanes merging', 'Convergencia de carriles', 'Carriles se unen']);
                }
            },
            {
                id: 'hazard-tollbooth',
                group: 'Peligros',
                label: 'Peaje',
                icon: 'Tollbooth',
                menuLabel: 'Tollbooth',
                menuAltLabels: ['Peaje'],
                combo: 'CTRL+SHIFT+4',
                type: 'custom',
                handler: function () {
                    triggerHazardByLabels(['Tollbooth', 'Peaje']);
                }
            },
            {
                id: 'hazard-school-zone',
                group: 'Peligros',
                label: 'Zona escolar',
                icon: 'School zone',
                menuLabel: 'School zone',
                menuAltLabels: ['Zona escolar', 'School zone'],
                combo: 'CTRL+SHIFT+5', 
                type: 'custom',
                // Botón para togglear la visualización de zonas escolares
                stateControl: {
                    label: 'Ver Zona Escolar',
                    tooltip: 'Activar/Desactivar visualización de zonas escolares',
                    toggle: function () {
                        SchoolZoneVisualizer.toggle();
                    },
                    getState: function () {
                        return SchoolZoneVisualizer.isActive();
                    }
                },
                handler: function () {
                    return triggerHazardByLabels(['School zone', 'Zona escolar']);
                }
            },
            // --- Peligros Faltantes ---
            {
                id: 'hazard-ground-level-crossing',
                group: 'Peligros',
                label: 'Cruce Ferroviario',
                icon: 'Ground level crossing',
                menuLabel: 'Ground level crossing',
                menuAltLabels: ['Paso a nivel', 'Cruce ferroviario', 'Railroad crossing'],
                combo: 'CTRL+SHIFT+7', // ShortCut personalizado
                type: 'custom',
                handler: function () {
                    // Esta función busca el item directamente en el menú de Peligros
                    return triggerHazardByLabels(['Ground level crossing', 'Paso a nivel', 'Cruce ferroviario', 'Railroad crossing']);
                }
            },
         /*  {
                id: 'hazard-narrow-bridge',
                group: 'Peligros',
                label: 'Puente angosto',
                icon: 'Narrow bridge',
                menuLabel: 'Narrow bridge',
                menuAltLabels: ['Puente angosto', 'Puente estrecho'],
                combo: 'CTRL+SHIFT+8', // ShortCut personalizado
                type: 'custom',
                handler: function () {
                    // Esta función busca el item directamente en el menú de Peligros
                    return triggerHazardByLabels(['Narrow bridge', 'Puente angosto', 'Puente estrecho']);
                }
            },*/
            // --- Cámaras (Grupo Nuevo) ---
            {
                id: 'hazard-speed-camera-native',
                group: 'Camaras',
                label: 'Cámara de velocidad (nativo)',
                icon: 'Speed camera',
                menuLabel: 'Speed camera',
                menuAltLabels: ['Cámara de velocidad', 'Control de velocidad'],
                combo: 'X', // ShortCut nativo de WME
                type: 'native',
                handler: null
            },
            {
                id: 'hazard-redlight-camera',
                group: 'Camaras',
                label: 'Cámara de semáforo',
                icon: 'Red light camera',
                menuLabel: 'Red light camera',
                menuAltLabels: ['Cámara de semáforo', 'Cámara de luz roja'],
                combo: 'CTRL+SHIFT+9', // ShortCut personalizado
                type: 'custom',
                handler: function () {
                    // Llama a la nueva función: triggerHazardSubMenuByLabels(['SubMenuIngles', 'SubMenuEsp'], ['ItemIngles', 'ItemEsp'])
                    return triggerHazardSubMenuByLabels(
                        ['Camera', 'Cámara'],
                        ['Red light camera', 'Cámara de semáforo', 'Cámara de luz roja']
                    );
                }
            },
            {
                id: 'hazard-fake-camera',
                group: 'Camaras',
                label: 'Cámara falsa',
                icon: 'Fake camera',
                menuLabel: 'Fake camera',
                menuAltLabels: ['Cámara falsa', 'Cámara disuasoria'],
                combo: 'CTRL+SHIFT+6', // ShortCut personalizado
                type: 'custom',
                handler: function () {
                    return triggerHazardSubMenuByLabels(
                        ['Camera', 'Cámara'],
                        ['Fake camera', 'Cámara falsa', 'Cámara disuasoria']
                    );
                }
            },
        ];

        function registerAll() {
            ACTION_DEFS.forEach(function (def) {
                ShortcutRegistry.registerAction(def);
            });
        }

        // --- Helpers DOM ---

        function normalizeLabelInput(labels) {
            const normalized = [];
            function push(val) {
                if (!val) {
                    return;
                }
                const text = ('' + val).trim();
                if (!text) {
                    return;
                }
                if (normalized.indexOf(text) === -1) {
                    normalized.push(text);
                }
            }

            if (Array.isArray(labels)) {
                labels.forEach(push);
            } else {
                push(labels);
            }

            return normalized;
        }

        function findFirstMatchingMenuItem(labels, scope) {
            const normalizedLabels = normalizeLabelInput(labels);
            for (let i = 0; i < normalizedLabels.length; i += 1) {
                const candidate = findMenuItemByText(normalizedLabels[i], scope);
                if (candidate) {
                    return candidate;
                }
            }
            return null;
        }

        function ensureAddMenuOpen() {
            // Nuevo enfoque: por ahora asumimos que el usuario abre el menú Add manualmente.
            // Este helper solo valida que exista un <wz-menu> en el DOM.
            const wzMenu = document.querySelector('wz-menu');
            if (wzMenu) {
                logInfo('ensureAddMenuOpen: wz-menu ya presente, usando menú actual.');
                return true;
            }

            logWarn('ensureAddMenuOpen: no hay wz-menu abierto. Abre el menú Add manualmente.');
            return false;
        }

        function findMenuItemByText(labelText, root) {
            if (!labelText) {
                return null;
            }

            const scope = root || document;

            // Primer intento: elementos clásicos con role="menuitem"
            const menuItems = scope.querySelectorAll('div[role="menuitem"], button[role="menuitem"], li[role="menuitem"]');
            for (let i = 0; i < menuItems.length; i += 1) {
                const item = menuItems[i];
                const txt = (item.textContent || '').trim();
                if (!txt) {
                    continue;
                }

                if (txt === labelText || txt.startsWith(labelText + '\n')) {
                    return item;
                }
            }

            // Segundo intento: estructura nueva de WME con <wz-menu-item> y .itemLabel--kXZjU
            const labelNodes = scope.querySelectorAll('wz-menu-item .itemLabel--kXZjU');
            for (let j = 0; j < labelNodes.length; j += 1) {
                const labelNode = labelNodes[j];
                const txt = (labelNode.textContent || '').trim();
                if (!txt) {
                    continue;
                }

                if (txt === labelText || txt.startsWith(labelText + '\n')) {
                    // Intentamos devolver el contenedor clicable; si no existe, devolvemos el propio labelNode
                    const hostItem = labelNode.closest('wz-menu-item');
                    return hostItem || labelNode;
                }
            }

            return null;
        }

        function triggerHazardByLabels(labels) {
            const labelList = normalizeLabelInput(labels);
            logInfo('triggerHazardByLabel: inicio para labels="' + labelList.join(', ') + '"');
            if (!ensureAddMenuOpen()) {
                logWarn('No se encontró un menú Add abierto para Hazards (wz-menu ausente).');
                return;
            }

            const menu = document.querySelector('wz-menu');
            if (!menu) {
                logWarn('triggerHazardByLabel: no se encontró el contenedor wz-menu.');
                return;
            }

            const hazardSubMenu = menu.querySelector('wz-menu-sub-menu[sub-menu-title="Hazard"]');
            logInfo('triggerHazardByLabel: submenú Hazard -> ' + (hazardSubMenu ? 'encontrado' : 'NO encontrado'));
            if (!hazardSubMenu) {
                logWarn('No se encontró el submenú Hazard dentro de wz-menu.');
                return;
            }

            const target = findFirstMatchingMenuItem(labelList, hazardSubMenu);
            logInfo('triggerHazardByLabel: resultado búsqueda hazard -> ' + (target ? 'encontrado' : 'NO encontrado'));
            if (!target) {
                logWarn('No se encontró el hazard con etiquetas: ' + labelList.join(', '));
                return;
            }

            logInfo('triggerHazardByLabel: haciendo click en hazard (etiquetas evaluadas: ' + labelList.join(', ') + ').');
            target.click();
        }

        function triggerPlaceCategoryByLabels(labels) {
            const labelList = normalizeLabelInput(labels);
            logInfo('triggerPlaceCategoryByLabel: inicio para labels="' + labelList.join(', ') + '"');
            if (!ensureAddMenuOpen()) {
                logWarn('No se encontró un menú Add abierto para Places (wz-menu ausente).');
                return;
            }

            const menu = document.querySelector('wz-menu');
            if (!menu) {
                logWarn('triggerPlaceCategoryByLabel: no se encontró el contenedor wz-menu.');
                return;
            }

            const placeSubMenu = menu.querySelector('wz-menu-sub-menu[sub-menu-title="Place"]');
            logInfo('triggerPlaceCategoryByLabel: submenú Place -> ' + (placeSubMenu ? 'encontrado' : 'NO encontrado'));
            if (!placeSubMenu) {
                logWarn('No se encontró el submenú Place dentro de wz-menu.');
                return;
            }

            // 1) Intentar primero si es un submenú (por ejemplo "Car Services")
            let target = null;
            labelList.some(function (label) {
                target = placeSubMenu.querySelector('wz-menu-sub-menu[sub-menu-title="' + label + '"]');
                return !!target;
            });
            if (target) {
                logInfo('triggerPlaceCategoryByLabel: haciendo click en submenú (label: ' + (labelList[0] || '?') + ').');
                target.click();
                return;
            }

            // 2) Si no es submenú, buscarlo como ítem normal dentro de Place
            target = findFirstMatchingMenuItem(labelList, placeSubMenu);
            logInfo('triggerPlaceCategoryByLabel: resultado búsqueda categoría -> ' + (target ? 'encontrado' : 'NO encontrado'));
            if (!target) {
                logWarn('No se encontró la categoría Place con etiquetas: ' + labelList.join(', '));
                return;
            }

            logInfo('triggerPlaceCategoryByLabel: haciendo click en categoría (etiquetas evaluadas: ' + labelList.join(', ') + ').');
            target.click();
        }

        function triggerHazardSubMenuByLabels(subMenuLabels, itemLabels) {
            const subMenuLabelList = normalizeLabelInput(subMenuLabels);
            const itemLabelList = normalizeLabelInput(itemLabels);
            logInfo('triggerHazardSubMenuByLabels: inicio para subMenú="' + subMenuLabelList.join(', ') + '", items="' + itemLabelList.join(', ') + '"');

            if (!ensureAddMenuOpen()) {
                logWarn('triggerHazardSubMenuByLabels: no se encontró un menú Add abierto.');
                return;
            }

            const menu = document.querySelector('wz-menu');
            if (!menu) {
                logWarn('triggerHazardSubMenuByLabels: no se encontró el contenedor wz-menu.');
                return;
            }

            const hazardSubMenu = menu.querySelector('wz-menu-sub-menu[sub-menu-title="Hazard"]');
            if (!hazardSubMenu) {
                logWarn('triggerHazardSubMenuByLabels: no se encontró el submenú Hazard.');
                return;
            }

            let targetSubMenu = null;
            for (let i = 0; i < subMenuLabelList.length; i++) {
                targetSubMenu = hazardSubMenu.querySelector('wz-menu-sub-menu[sub-menu-title="' + subMenuLabelList[i] + '"]');
                if (targetSubMenu) {
                    if (typeof targetSubMenu.click === 'function') {
                        targetSubMenu.click();
                    }
                    break;
                }
            }

            if (!targetSubMenu) {
                logWarn('triggerHazardSubMenuByLabels: submenú no encontrado (' + subMenuLabelList.join(', ') + ').');
                return;
            }

            const targetItem = findFirstMatchingMenuItem(itemLabelList, targetSubMenu);
            if (!targetItem) {
                logWarn('triggerHazardSubMenuByLabels: item no encontrado (' + itemLabelList.join(', ') + ').');
                return;
            }

            logInfo('triggerHazardSubMenuByLabels: haciendo click en item.');
            targetItem.click();
        }

        return {
            registerAll,
            triggerHazardByLabels,
            triggerPlaceCategoryByLabels,
            triggerHazardSubMenuByLabels
        };
    })();

    // ---------------------------------------------------------------------
    //  Módulo: Decorador de menús (añade [shortcut] o ND al texto)
    // ---------------------------------------------------------------------

    const MenuDecorator = (function () {
        let intervalId = null;

        function decorateOnce() {
            const allActions = ShortcutRegistry.getAllActions();
            if (!allActions || allActions.length === 0) {
                return;
            }

            allActions.forEach(function (act) {
                const comboText = act.combo || 'ND';
                const labelCandidates = getMenuLabelCandidates(act);
                if (!labelCandidates || labelCandidates.length === 0) {
                    return;
                }

                const menuItems = document.querySelectorAll('div[role="menuitem"], button[role="menuitem"], li[role="menuitem"]');
                for (let i = 0; i < menuItems.length; i += 1) {
                    const item = menuItems[i];
                    const txt = (item.textContent || '').trim();
                    if (!txt) {
                        continue;
                    }

                    const matches = labelCandidates.some(function (candidate) {
                        return txt === candidate || txt.startsWith(candidate + '\n');
                    });

                    if (matches) {
                        let span = item.querySelector('.pln-shortcut-hint');
                        if (!span) {
                            span = document.createElement('span');
                            span.className = 'pln-shortcut-hint';
                            item.appendChild(span);
                        }
                        span.textContent = ' [' + comboText + ']';
                    }
                }
            });
        }

        function start() {
            if (intervalId !== null) {
                return;
            }

            // Decoramos periódicamente porque los menús aparecen y desaparecen.
            intervalId = window.setInterval(decorateOnce, 1000);
            logInfo('MenuDecorator iniciado.');
        }

        return {
            start
        };
    })();

    // ---------------------------------------------------------------------
    //  Módulo: UI de configuración de ShortCuts (entrada en lista de scripts)
    // ---------------------------------------------------------------------

    const ShortcutsConfigUI = (function () {
        const MODAL_ID = 'esc-shortcuts-modal';
        let escHandler = null;
        let statusTimer = null;
        let stateWatcherTimer = null;
        let stateWatchers = [];

        function attachEscListener() {
            if (escHandler) {
                return;
            }
            escHandler = function (evt) {
                if (evt.key === 'Escape') {
                    closeManager();
                }
            };
            document.addEventListener('keydown', escHandler, true);
        }

        function closeManager() {
            const overlay = safeGetElementById(MODAL_ID);
            if (overlay && overlay.parentElement) {
                overlay.parentElement.removeChild(overlay);
            }
            if (escHandler) {
                document.removeEventListener('keydown', escHandler, true);
                escHandler = null;
            }
            stopStateWatcher();
            stateWatchers.length = 0;
            if (statusTimer) {
                window.clearTimeout(statusTimer);
                statusTimer = null;
            }
        }

        function handleComboInputKeydown(evt) {
            if (evt.key === 'Tab') {
                return;
            }

            evt.preventDefault();
            evt.stopPropagation();

            if (evt.key === 'Escape') {
                evt.target.blur();
                return;
            }

            if (evt.key === 'Backspace' || evt.key === 'Delete') {
                evt.target.value = '';
                return;
            }

            const combo = KeyUtils.normalizeComboFromEvent(evt, { allowEditableTargets: true });
            if (combo) {
                evt.target.value = combo;
            }
        }

        function renderTableRows(tbody, actions) {
            tbody.innerHTML = '';
            stateWatchers.length = 0;
            const sorted = actions.slice().sort(function (a, b) {
                const groupA = (a.group || 'General').toLowerCase();
                const groupB = (b.group || 'General').toLowerCase();
                if (groupA !== groupB) {
                    return groupA.localeCompare(groupB);
                }
                return (a.label || a.id).localeCompare(b.label || b.id);
            });

            sorted.forEach(function (action) {
                const row = document.createElement('tr');
                row.setAttribute('data-action-id', action.id);

                const stateCell = document.createElement('td');
                const stateInfo = action.stateControl;
                if (stateInfo && typeof stateInfo.toggle === 'function' && typeof stateInfo.getState === 'function') {
                    const toggleBtn = document.createElement('button');
                    toggleBtn.type = 'button';
                    toggleBtn.className = 'esc-state-toggle-btn';
                    const icon = document.createElement('span');
                    icon.className = 'esc-state-toggle-icon';
                    icon.textContent = '✔';
                    toggleBtn.appendChild(icon);
                    const labelSpan = document.createElement('span');
                    labelSpan.className = 'esc-state-toggle-label';
                    labelSpan.textContent = stateInfo.label || action.label || 'Estado';
                    toggleBtn.appendChild(labelSpan);

                    function updateStateIcon() {
                        let active = false;
                        try {
                            active = !!stateInfo.getState();
                        } catch (err) {
                            active = false;
                            logWarn('esc: no se pudo leer el estado de ' + action.id + ' -> ' + err.message);
                        }
                        icon.style.color = active ? '#16a34a' : '#dc2626';
                        toggleBtn.dataset.state = active ? 'active' : 'inactive';
                        const targetLabel = stateInfo.tooltip || stateInfo.label || action.label || 'estado';
                        toggleBtn.title = (active ? 'Desactivar ' : 'Activar ') + targetLabel;
                    }

                    toggleBtn.addEventListener('click', function () {
                        try {
                            stateInfo.toggle();
                        } catch (err) {
                            logWarn('esc: error alternando ' + action.id + ' -> ' + err.message);
                        }
                        updateStateIcon();
                    });

                    stateWatchers.push({
                        refresh: updateStateIcon
                    });

                    updateStateIcon();
                    stateCell.appendChild(toggleBtn);
                } else {
                    stateCell.textContent = 'N/A';
                    stateCell.className = 'esc-state-cell-na';
                }
                row.appendChild(stateCell);

                const groupCell = document.createElement('td');
                groupCell.textContent = action.group || 'General';
                row.appendChild(groupCell);

                const labelCell = document.createElement('td');
                // --- INICIO MODIFICACIÓN: Texto Cliqueable ---
                if (action.configHandler && typeof action.configHandler === 'function') {
                    // Si tiene configuración, creamos un enlace
                    const link = document.createElement('span'); // Usamos span con estilo de link
                    link.textContent = action.label || action.id;
                    link.style.cssText = "color:#007bff; text-decoration:underline; cursor:pointer; font-weight:500;";
                    link.title = "Click para abrir configuración";
                    link.onclick = function(e) {
                        e.preventDefault(); // Evitar comportamientos extraños
                        action.configHandler(); // Ejecutar la función de abrir panel
                    };
                    labelCell.appendChild(link);
                } else {
                    // Comportamiento normal
                    labelCell.textContent = action.label || action.id;
                }
                // --- FIN MODIFICACIÓN ---
                row.appendChild(labelCell);

                const comboCell = document.createElement('td');
                const comboControl = document.createElement('div');
                comboControl.className = 'esc-combo-control';
                const input = document.createElement('input');
                input.type = 'text';
                input.className = 'esc-combo-input';
                input.dataset.actionId = action.id;
                input.value = action.combo || '';
                input.placeholder = action.defaultCombo ? action.defaultCombo : 'Sin ShortCut';

                if (!action.handler || typeof action.handler !== 'function') {
                    input.disabled = true;
                    input.title = 'ShortCut nativo / informativo (no editable).';
                } else {
                    input.title = 'Haz click y presiona el nuevo ShortCut (Backspace para limpiar).';
                    input.addEventListener('keydown', handleComboInputKeydown);
                }

                comboControl.appendChild(input);

                const restoreBtn = document.createElement('button');
                restoreBtn.type = 'button';
                restoreBtn.className = 'esc-btn esc-btn-ghost';
                restoreBtn.textContent = 'Predeterminado';
                restoreBtn.title = 'Restaurar ShortCut original';
                if (!action.defaultCombo || !action.handler || typeof action.handler !== 'function') {
                    restoreBtn.disabled = true;
                }
                restoreBtn.addEventListener('click', function () {
                    if (restoreBtn.disabled) {
                        return;
                    }
                    input.value = action.defaultCombo || '';
                });
                comboControl.appendChild(restoreBtn);

                comboCell.appendChild(comboControl);
                row.appendChild(comboCell);

                const defaultCell = document.createElement('td');
                defaultCell.textContent = action.defaultCombo || 'ND';
                row.appendChild(defaultCell);

                const typeCell = document.createElement('td');
                typeCell.textContent = action.type === 'native' ? 'Nativo' : 'Custom';
                row.appendChild(typeCell);

                tbody.appendChild(row);
            });
        }

        function refreshStateIndicators() {
            if (!stateWatchers.length) {
                return;
            }
            stateWatchers.forEach(function (watcher) {
                try {
                    watcher.refresh();
                } catch (err) {
                    logWarn('esc: error actualizando estado -> ' + err.message);
                }
            });
        }

        function startStateWatcher() {
            if (!stateWatchers.length) {
                stopStateWatcher();
                return;
            }
            stopStateWatcher();
            refreshStateIndicators();
            stateWatcherTimer = window.setInterval(refreshStateIndicators, 1200);
        }

        function stopStateWatcher() {
            if (stateWatcherTimer) {
                window.clearInterval(stateWatcherTimer);
                stateWatcherTimer = null;
            }
        }

        function showStatus(modal, message, type) {
            if (!modal) {
                return;
            }
            const statusNode = modal.querySelector('.esc-status');
            if (!statusNode) {
                return;
            }

            statusNode.textContent = message || '';
            statusNode.className = 'esc-status';
            if (type === 'success') {
                statusNode.classList.add('esc-status--success');
            } else if (type === 'error') {
                statusNode.classList.add('esc-status--error');
            }

            if (statusTimer) {
                window.clearTimeout(statusTimer);
            }
            if (message) {
                statusTimer = window.setTimeout(function () {
                    statusNode.textContent = '';
                    statusNode.className = 'esc-status';
                }, 4500);
            }
        }

        function applyChanges(modal) {
            const inputs = modal.querySelectorAll('.esc-combo-input');
            let changed = 0;
            const errors = [];

            inputs.forEach(function (input) {
                input.classList.remove('esc-input-error');
                if (input.disabled) {
                    return;
                }

                const actionId = input.dataset.actionId;
                const action = ShortcutRegistry.getActionById(actionId);
                if (!action) {
                    return;
                }

                const result = ShortcutRegistry.setActionCombo(actionId, input.value || '');
                if (!result.ok) {
                    errors.push(action.label + ': ' + result.reason);
                    input.classList.add('esc-input-error');
                    return;
                }
                if (result.changed) {
                    changed += 1;
                }
                input.value = result.combo || '';
            });

            if (errors.length > 0) {
                showStatus(modal, 'No se aplicaron algunos cambios:\n- ' + errors.join('\n- '), 'error');
                return;
            }

            if (changed === 0) {
                showStatus(modal, 'Sin cambios por guardar.', 'success');
            } else {
                showStatus(modal, 'Se guardaron ' + changed + ' cambio(s).', 'success');
            }
        }

        function handleResetAll(modal) {
            const confirmed = window.confirm('¿Restablecer todos los ShortCuts a sus valores originales?');
            if (!confirmed) {
                return;
            }

            ShortcutRegistry.resetAllCombos();
            const inputs = modal.querySelectorAll('.esc-combo-input');
            inputs.forEach(function (input) {
                const action = ShortcutRegistry.getActionById(input.dataset.actionId);
                input.classList.remove('esc-input-error');
                if (action) {
                    input.value = action.combo || '';
                }
            });
            showStatus(modal, 'Se restablecieron los ShortCuts originales.', 'success');
        }

        function openManager() {
            const actions = ShortcutRegistry.getAllActions();
            if (!actions || actions.length === 0) {
                window.alert('WME Easy ShortCuts\n\nNo hay acciones registradas todavía.');
                return;
            }

            UIStyles.ensure();
            closeManager();

            const overlay = document.createElement('div');
            overlay.id = MODAL_ID;
            overlay.className = 'esc-modal-backdrop';
            overlay.addEventListener('click', function (evt) {
                if (evt.target === overlay) {
                    closeManager();
                }
            });

            const modal = document.createElement('div');
            modal.className = 'esc-modal';
            overlay.appendChild(modal);

            const header = document.createElement('div');
            header.className = 'esc-modal__header';
            const title = document.createElement('h2');
            title.textContent = 'WME Easy ShortCuts';
            header.appendChild(title);
            const closeBtn = document.createElement('button');
            closeBtn.type = 'button';
            closeBtn.className = 'esc-close-btn';
            closeBtn.textContent = '×';
            closeBtn.addEventListener('click', closeManager);
            header.appendChild(closeBtn);
            modal.appendChild(header);

            const intro = document.createElement('p');
            intro.className = 'esc-icon-hint';
            intro.textContent = 'Captura un ShortCut haciendo click en el campo y presionando el combo deseado. Backspace lo limpia. Los ShortCuts nativos solo se muestran como referencia.';
            modal.appendChild(intro);

            const body = document.createElement('div');
            body.className = 'esc-modal__body';
            const table = document.createElement('table');
            const thead = document.createElement('thead');
            const headRow = document.createElement('tr');
            ['Estado', 'Grupo', 'Acción', 'ShortCut actual', 'Predeterminado', 'Tipo'].forEach(function (label) {
                const th = document.createElement('th');
                th.textContent = label;
                headRow.appendChild(th);
            });
            thead.appendChild(headRow);
            table.appendChild(thead);
            const tbody = document.createElement('tbody');
            renderTableRows(tbody, actions);
            table.appendChild(tbody);
            body.appendChild(table);
            modal.appendChild(body);

            const footer = document.createElement('div');
            footer.className = 'esc-modal__footer';
            const status = document.createElement('div');
            status.className = 'esc-status';
            footer.appendChild(status);
            const buttons = document.createElement('div');
            buttons.className = 'esc-footer-buttons';

            const resetBtn = document.createElement('button');
            resetBtn.type = 'button';
            resetBtn.className = 'esc-btn esc-btn-secondary';
            resetBtn.textContent = 'Restaurar todo';
            resetBtn.addEventListener('click', function () {
                handleResetAll(modal);
            });
            buttons.appendChild(resetBtn);

            const saveBtn = document.createElement('button');
            saveBtn.type = 'button';
            saveBtn.className = 'esc-btn esc-btn-primary';
            saveBtn.textContent = 'Guardar cambios';
            saveBtn.addEventListener('click', function () {
                applyChanges(modal);
            });
            buttons.appendChild(saveBtn);

            footer.appendChild(buttons);
            modal.appendChild(footer);

            document.body.appendChild(overlay);
            startStateWatcher();
            attachEscListener();
        }

        return {
            openManager
        };
    })();

    // ---------------------------------------------------------------------
    //  Módulo: Lanzador flotante independiente
    // ---------------------------------------------------------------------

    const ScriptLauncher = (function () {
        const BUTTON_ID = 'esc-launcher-btn';
        let attached = false;
        let repositionTimerId = null;

        function findOnlineEditorsBadge() {
            const candidates = Array.from(document.querySelectorAll('div, span, button, wz-button, wz-tooltip'));
            for (let i = 0; i < candidates.length; i += 1) {
                const node = candidates[i];
                const txt = (node.textContent || '').trim().toLowerCase();
                if (!txt) {
                    continue;
                }
                if (txt.includes('online editors')) {
                    return node;
                }
            }
            return null;
        }

        function findNrmlizerBadge() {
            const candidates = Array.from(document.querySelectorAll('div, span, button'));
            for (let i = 0; i < candidates.length; i += 1) {
                const node = candidates[i];
                const txt = (node.textContent || '').trim();
                if (!txt) {
                    continue;
                }
                if (txt.includes('Places NrmliZer') || txt.includes('Places NrmlIZer') || txt.includes('Places Nrmlizer')) {
                    return node;
                }
            }
            return null;
        }

        function getAnchorElement() {
            // 1) Intentar anclar al badge de "online editors" (nuevo comportamiento principal)
            const onlineEditorsBadge = findOnlineEditorsBadge();
            if (onlineEditorsBadge) {
                return onlineEditorsBadge;
            }

            // 2) Si no existe (por ejemplo en pantallas muy pequeñas), usar Places NrmliZer como respaldo
            const nrmlizerBadge = findNrmlizerBadge();
            if (nrmlizerBadge) {
                return nrmlizerBadge;
            }

            // 3) Último recurso: el contenedor del mapa o el panel izquierdo
            const map = safeGetElementById('map') ||
                document.querySelector('#map') ||
                document.querySelector('.WazeMap') ||
                document.querySelector('[data-testid="map-container"]');
            if (map) {
                return map;
            }

            return safeGetElementById('edit-panel') ||
                document.querySelector('#left-panel') ||
                document.querySelector('[data-testid="left-panel"]') ||
                document.querySelector('.left-panel, .side-panel');
        }

        function positionButton() {
            const btn = safeGetElementById(BUTTON_ID);
            if (!btn) {
                return;
            }

            // Posición fija similar al panel de Places NrmliZer
            btn.style.position = 'fixed';
            btn.style.bottom = '60px';
            btn.style.left = '19%';
            btn.style.top = 'auto';
        }

        function createButtonIfNeeded() {
            if (!document.body) {
                return false;
            }

            let container = safeGetElementById(BUTTON_ID);
            if (container) {
                attached = true;
                positionButton();
                return true;
            }

            UIStyles.ensure();

            container = document.createElement('div');
            container.id = BUTTON_ID;
            container.setAttribute('data-tooltip', 'EasyShortCuts');

            const wzBtn = document.createElement('wz-button');
            wzBtn.setAttribute('color', 'clear-icon');
            wzBtn.setAttribute('size', 'md');
            wzBtn.setAttribute('type', 'button');
            wzBtn.setAttribute('aria-label', 'Abrir gestor Easy ShortCuts');
            wzBtn.title = 'Abrir gestor de ShortCuts de WME Easy ShortCuts';

            const iconSpan = document.createElement('span');
            iconSpan.className = 'esc-launcher__icon';
            const img = document.createElement('img');
            img.src = MAIN_TAB_ICON_BASE64;
            img.alt = 'EasyShortCuts icon';
            iconSpan.appendChild(img);

            wzBtn.appendChild(iconSpan);

            wzBtn.addEventListener('click', function (evt) {
                evt.preventDefault();
                evt.stopPropagation();
                ShortcutsConfigUI.openManager();
            });

            container.appendChild(wzBtn);
            document.body.appendChild(container);

            attached = true;
            positionButton();

            window.addEventListener('resize', positionButton);

            logInfo('ScriptLauncher: acceso rápido creado en la interfaz (SDK button, anclado al mapa).');
            return true;
        }

        function start() {
            if (attached && safeGetElementById(BUTTON_ID)) {
                return;
            }

            createButtonIfNeeded();

            if (!repositionTimerId) {
                repositionTimerId = window.setInterval(function () {
                    if (!safeGetElementById(BUTTON_ID)) {
                        createButtonIfNeeded();
                    }
                    positionButton();
                }, 1200);
            }
        }

        return {
            start
        };
    })();
    
    // ---------------------------------------------------------------------
    //  Bootstrap: esperar a que WME/WazeWrap Y EL SDK estén listos
    // ---------------------------------------------------------------------
    let sdkReady = false;
    let wazeWrapReady = false;
    let sdkInitializerFunc = null;

    // Función que se llamará cuando AMBOS estén listos
    function tryInitializeScript() 
    {
        if (sdkReady && wazeWrapReady) 
        {
            logInfo('WazeWrap y WME SDK detectados. Inicializando EasyShortCuts...');
            initEasyShortCuts(sdkInitializerFunc);
        }
    }

    // Vigilante 1: Espera por WazeWrap
    function bootstrapWazeWrap() 
    {
        // Simplificamos la comprobación. Si WazeWrap dice que está listo, confiamos en él.
        // Eliminamos la dependencia estricta de W.map.projection para evitar bloqueos.
        if (typeof WazeWrap === 'undefined' || !WazeWrap.Ready) 
        {
            logInfo('Esperando inicialización de WazeWrap...');
            window.setTimeout(bootstrapWazeWrap, 500);
            return;
        }
        
        logInfo('WazeWrap está listo.');
        wazeWrapReady = true;
        tryInitializeScript();
    }

    // Vigilante 2: Espera por el WME SDK
    function bootstrapSDK() 
    {
        if (typeof unsafeWindow === 'undefined' || typeof unsafeWindow.SDK_INITIALIZED === 'undefined') 
        {
            // Log menos agresivo (debug en lugar de warn) para no llenar la consola mientras carga
            // console.debug('[EasyShortCuts] Esperando WME SDK...');
            window.setTimeout(bootstrapSDK, 500);
            return;
        }

        unsafeWindow.SDK_INITIALIZED.then(function() {
            logInfo('WME SDK inicializado correctamente.');
            sdkInitializerFunc = unsafeWindow.getWmeSdk;
            sdkReady = true;
            tryInitializeScript();
        }).catch(function(err) {
            logWarn('Error crítico al obtener WME SDK: ' + err.message);
            // Aún si el SDK falla, podríamos querer arrancar el resto del script
            // sdkReady = true; tryInitializeScript(); 
        });
    }

    // Iniciar secuencia de arranque
    bootstrapWazeWrap();
    bootstrapSDK();

})();