WME Places Name Normalizer

Herramienta de asistencia para normalizar nombres de lugares en Waze Map Editor. Detecta inconsistencias, sugiere correcciones, valida categorías y permite aplicar cambios masivos con un solo click.

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

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

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

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

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

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

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

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

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

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

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

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

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

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

// ==UserScript==
// @name         WME Places Name Normalizer
// @namespace    https://greasyfork.org/en/users/mincho77
// @author       Mincho77
// @version      9.3.1
// @license      MIT
// @description  Herramienta de asistencia para normalizar nombres de lugares en Waze Map Editor. Detecta inconsistencias, sugiere correcciones, valida categorías y permite aplicar cambios masivos con un solo click.
// @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
// @connect      sheets.googleapis.com
// @connect      script.google.com
// @connect      script.googleusercontent.com
// @connect      wme-normalizer-ai.cotalvaro.workers.dev
// @connect      cotalvaro.workers.dev
// @run-at       document-end
// @require      https://greasyfork.org/scripts/24851-wazewrap/code/WazeWrap.js
// ==/UserScript==

// Se establecen las variables de control con valores iniciales claros
let isProcessingActive = false;
let isNormalizationActive = true;
let isResultsPanelOpen = false;
let activeScanning = false; // Bandera para indicar si se encuentra en curso un escaneo
window.__PLN_DECISION_DEBUG_ON = false; // Desactivado por defecto para evitar spam en consola



(function () {
    //window.normalizeWordInternal = normalizeWordInternal;
    // Variables globales básicas
    const SCRIPT_NAME = GM_info.script.name;
    const VERSION = GM_info.script.version.toString();
    // Variable global para el ícono de la pestaña principal
    const MAIN_TAB_ICON_BASE64 = ""
    // === [PLN] Daily Usage Log → Google Sheets (via Apps Script) ===
    // 1) Configure your Apps Script Web App endpoint (POST, Anyone w/ link)
    const PLN_USAGE_LOG = {
        endpoint: 'https://script.google.com/macros/s/AKfycbxkBUFI8RuxrtksAIpvpzmr-ZyeO1huncmG6w-BKKBpcrtXIKEemMAai3zyg1OOYqWJOg/exec',
        timeoutMs: 8000
    };

    // === [PLN] User Tracking → Google Sheets (Separate Sheet) ===
    // 2) Configure your User Tracking Apps Script Web App endpoint
    // NOTA: Debes configurar este endpoint después de crear el Apps Script para usuarios
    const PLN_USER_TRACKING = {
        endpoint: 'https://script.google.com/macros/s/AKfycbxcsZrbJHDX8ic2dcZxt4BmglWdSTt-kfOstGqa65-WGIrrFGwcYAPR6edBmDwGro62/exec', 
        timeoutMs: 20000, // Aumentado a 20 segundos para diagnóstico
        enabled: true
    };

    // === [PLN] Access Control → Validación de lista blanca desde el servidor ===
    // 3) Configure your Access Control Apps Script Web App endpoint
    // Este endpoint valida si un usuario está en la lista blanca (Whitelist)
    const PLN_ACCESS_CONTROL = {
        endpoint: 'https://script.google.com/macros/s/AKfycbxJHNeIJBuroF_Y51po74uYu41VCa9i_6T7kQJv0zWmgg-h0021EaOOetiZVkocz8or/exec', 
        timeoutMs: 10000,
        enabled: true,
        // Estos valores se sobrescriben con lo que diga la hoja "Config"
        defaultMinLevel: 3,
        easyShortcutsMinLevel: 4
    };
    window.PLN_ACCESS_CONTROL = PLN_ACCESS_CONTROL;
    var PLN_ACC_LVL = 2; // Nivel mínimo 3 (rank 2)
    var PLN_ACC_LVL_ES = 4; // Nivel mínimo para EasyShortCuts

    // Basic date helpers (yyyy-mm-dd, local)
    function plnTodayStr() {
        try {
            const d = new Date();
            const y = d.getFullYear();
            const m = String(d.getMonth() + 1).padStart(2, '0');
            const day = String(d.getDate()).padStart(2, '0');
            return `${y}-${m}-${day}`;
        } catch (_) { return ''; }
    }

    // Date + Time helper (yyyy-mm-dd HH:MM:SS, local)
    function plnNowStr() {
        try {
            const d = new Date();
            const y = d.getFullYear();
            const m = String(d.getMonth() + 1).padStart(2, '0');
            const day = String(d.getDate()).padStart(2, '0');
            const h = String(d.getHours()).padStart(2, '0');
            const min = String(d.getMinutes()).padStart(2, '0');
            const s = String(d.getSeconds()).padStart(2, '0');
            return `${y}-${m}-${day} ${h}:${min}:${s}`;
        } catch (_) { return ''; }
    }

    // Persisted list of places where the editor has worked (cities)
    // Stored as a Set in localStorage to be "permanent".
    function plnLoadWorkCities() {
        try {
            const raw = localStorage.getItem('wme_pln_work_cities');
            const arr = raw ? JSON.parse(raw) : [];
            return new Set(Array.isArray(arr) ? arr : []);
        } catch (_) { return new Set(); }
    }
    function plnSaveWorkCities(set) {
        try { localStorage.setItem('wme_pln_work_cities', JSON.stringify(Array.from(set || []))); } catch (_) { }
    }
    function plnAddWorkCity(city) {
        try {
            const c = String(city || '').trim();
            if (!c) return;
            const set = plnLoadWorkCities();
            if (!set.has(c)) {
                set.add(c);
                plnSaveWorkCities(set);
            }
        } catch (_) { }
    }

    // Export helper to register a work city from anywhere (console/UI)
    try {
        window.plnRegisterWorkCity = plnAddWorkCity;
        if (typeof unsafeWindow !== 'undefined') unsafeWindow.plnRegisterWorkCity = plnAddWorkCity;
    } catch (_) { }

    let blinkingPlaces = new Set();
    let blinkState = true;
    let lastBlinkTime = 0;
    let pendingRequests = [];
    const BLINK_INTERVAL = 500; // Tiempo en milisegundos entre cada titilación (0.5 segundos)
    const FORCE_SHEET_RELOAD = true; // Cambia a true para forzar la lectura directa por defecto

    // === Debug helpers for City Apply ===
    const PLN_DEBUG = true; // poner false para silenciar
    // PLN_ACC_LVL declarado arriba
    function plnLog(...args) { if (PLN_DEBUG) console.log('[WME PLN][CityApply]', ...args); }
    function plnWarn(...args) { if (PLN_DEBUG) console.warn('[WME PLN][CityApply]', ...args); }
    function plnErr(...args) {
        console.error('[WME PLN][CityApply]', ...args);
    }

    // Log de guardado global del WME
    window.addEventListener('wme-save-finished', (ev) => {
        plnLog('wme-save-finished', ev && ev.detail);
    });


    // Variables globales para el diccionario de palabras excluidas
    //Permite inicializar el diccionario de palabras intercambiadas
    if (!window.swapWords) {
        const stored = localStorage.getItem("wme_swapWords");
        window.swapWords = stored ? JSON.parse(stored) : [];
    }

    // === Swap Debug Helpers ===
    (function plnSwapDebugInit() {
        try {
            if (window.plnSwapDebug && window.plnSwapExplain) return;
            window.__PLN_SWAP_DEBUG_ON = window.__PLN_SWAP_DEBUG_ON || false;
            window.plnSwapDebug = function (on) {
                window.__PLN_SWAP_DEBUG_ON = !!on;
                try { localStorage.setItem('wme_pln_debug_swap', on ? '1' : '0'); } catch (_) { }
                //console.log('[PLN Swap] debug', on ? 'ON' : 'OFF');
            };
            try { if (typeof unsafeWindow !== 'undefined') unsafeWindow.plnSwapDebug = window.plnSwapDebug; } catch (_) { }
            window.plnSwapExplain = function (name) {
                const prev = window.__PLN_SWAP_DEBUG_ON;
                window.__PLN_SWAP_DEBUG_ON = true;
                console.group('[PLN Swap] TRACE for', name);
                const out = applySwapRules(name);
                //console.log('TRACE result =>', out);
                console.groupEnd();
                window.__PLN_SWAP_DEBUG_ON = prev;
                return out;
            };
            try { if (typeof unsafeWindow !== 'undefined') unsafeWindow.plnSwapExplain = window.plnSwapExplain; } catch (_) { }
        } catch (_) { /* noop */ }
    })();
    // === Decision Debug Helpers (pipeline end‑to‑end) ===
    (function plnDecisionDebugInit() {
        try {
            if (window.plnDecisionDebug && window.plnExplainDecision) return;
            window.__PLN_DECISION_DEBUG_ON = window.__PLN_DECISION_DEBUG_ON || false;

            function noDiacritics(s) {
                return String(s || '')
                    .normalize('NFD')
                    .replace(/[\u0300-\u036f]/g, '')
                    .replace(/\s+/g, ' ')
                    .trim()
                    .toLowerCase();
            }

            window.plnDecisionDebug = function (on) {
                window.__PLN_DECISION_DEBUG_ON = !!on;
                try { localStorage.setItem('wme_pln_debug_decision', on ? '1' : '0'); } catch (_) { }
                //console.log('[PLN Decision] debug', on ? 'ON' : 'OFF');
            };
            try { if (typeof unsafeWindow !== 'undefined') unsafeWindow.plnDecisionDebug = window.plnDecisionDebug; } catch (_) { }

            window.plnExplainDecision = function (name) {
                const origFn = (typeof window.__PLNNormalizeOriginal === 'function') ? window.__PLNNormalizeOriginal
                    : (typeof window.__plnNormalizeOriginal === 'function') ? window.__plnNormalizeOriginal
                        : (typeof window.normalizePlaceName === 'function' ? window.normalizePlaceName : null);

                const input = String(name || '');
                const base = origFn ? origFn(input) : input;
                const baseCap = plnPostSwapCap(base);
                const afterEx = plnApplyExclusions(baseCap);
                const afterSw = applySwapRules(afterEx);
                const renorm = origFn ? origFn(afterSw) : afterSw;
                const finalCap = plnPostSwapCap(renorm);
                const final = plnApplyExclusions(finalCap);

                const equalExact = input === final;
                const noDia = s => String(s || '').normalize('NFD').replace(/[\u0300-\u036f]/g, '').replace(/\s+/g, ' ').trim().toLowerCase();
                const equalNoCaseNoSpacesNoDiacritics = noDia(input) === noDia(final);

                const out = {
                    original: input, baseNormalized: base, baseCap, afterExclusions: afterEx,
                    afterSwap: afterSw, renormalized: renorm, finalSuggested: final,
                    equalExact, equalNoCaseNoSpacesNoDiacritics
                };
                console.group('[PLN Decision] explain', name);
                console.table(out);
                console.groupEnd();
                return out;
            };
            try { if (typeof unsafeWindow !== 'undefined') unsafeWindow.plnExplainDecision = window.plnExplainDecision; } catch (_) { }
        } catch (_) { /* noop */ }
    })();



    // Muestra un mensaje tipo "toast" en la esquina inferior derecha
    function plnToast(message, duration = 3000) {
        try {
            // --- Lógica de creación de elementos ---
            const createToastElement = () => {
                // Estilo Allen: Prioriza el SDK si está disponible.
                if (window.WME_PLN_SDK && window.WME_PLN_SDK.UI && window.WME_PLN_SDK.UI.createElement) {
                    return window.WME_PLN_SDK.UI.createElement('div', {
                        textContent: message,
                        style: {
                            backgroundColor: '#333',
                            color: 'white',
                            padding: '10px 20px',
                            borderRadius: '8px',
                            boxShadow: '0 4px 8px rgba(0,0,0,0.2)',
                            fontFamily: 'sans-serif',
                            fontSize: '14px',
                            opacity: '0',
                            transition: 'opacity 0.3s ease-in-out, transform 0.3s ease-in-out',
                            transform: 'translateY(20px)'
                        }
                    });
                }
                // Fallback al método tradicional si el SDK no está listo.
                else {
                    const toast = document.createElement('div');
                    toast.textContent = message;
                    toast.style.cssText = `
                    background-color: #333; color: white; padding: 10px 20px;
                    border-radius: 8px; box-shadow: 0 4px 8px rgba(0,0,0,0.2);
                    font-family: sans-serif; font-size: 14px; opacity: 0;
                    transition: opacity 0.3s ease-in-out, transform 0.3s ease-in-out;
                    transform: translateY(20px);
                `;
                    return toast;
                }
            };

            // --- Lógica para mostrar y ocultar (sin cambios) ---
            let container = document.getElementById('pln-toast-container');
            if (!container) {
                container = document.createElement('div');
                container.id = 'pln-toast-container';
                container.style.cssText = 'position: fixed; bottom: 20px; right: 20px; z-index: 99999; display: flex; flex-direction: column; gap: 10px;';
                document.body.appendChild(container);
            }

            const toast = createToastElement();
            container.appendChild(toast);

            // Animación de entrada
            setTimeout(() => {
                toast.style.opacity = '1';
                toast.style.transform = 'translateY(0)';
            }, 10);

            // Desaparición automática
            setTimeout(() => {
                toast.style.opacity = '0';
                toast.style.transform = 'translateY(20px)';
                toast.addEventListener('transitionend', () => toast.remove());
            }, duration);

        } catch (e) {
            console.error("Error en plnToast:", e);
            alert(message); // Fallback final
        }
    }

    // Exponer plnToast globalmente para addWordToDictionary
    window.plnToast = plnToast;

    // === Hyphen cleanup: keep all hyphens except a dangling one at the end ===
    function plnShouldPreserveTrailingHyphen(name) {
        // Enforce removal of ALL trailing hyphens, even if they match a replacement rule.
        // This fixes issues like "Texaco ⛽" -> "Texaco -" where the hyphen was preserved.
        return false;
    }

    function plnFixDanglingHyphen(s) {
        try {
            let out = String(s == null ? '' : s);

            // Primero, eliminar guiones al final (lógica original)
            const trailingPattern = /\s*-\s*$/;
            if (trailingPattern.test(out) && !plnShouldPreserveTrailingHyphen(out)) {
                out = out.replace(trailingPattern, '');
            }

            // Limpieza adicional: eliminar otros caracteres especiales comunes al final
            // Esto captura restos de emojis convertidos, puntuación, etc.
            // Caracteres a eliminar: -, ., ,, ;, :, /, \, |, _, +, =, *, ~, `, ', ", etc.
            out = out.replace(/[\-.,;:/\\|_+=*~`'"!?]+\s*$/g, '');

            // normaliza espacios múltiples; NO toca guiones internos
            out = out.replace(/\s{2,}/g, ' ').trim();
            return out;
        } catch (_) { return s; }
    }

    // === Row visibility control (hide only when finalized) ===
    function plnSetRowHidden(row, hidden) {
        try {
            if (!row || row.nodeType !== 1) return;
            row.style.display = hidden ? 'none' : '';
        } catch (_) {/* noop */ }
    }

    function plnMarkRowFinalized(row, reason) {
        try {
            if (!row || row.nodeType !== 1) return;
            // mark as finalized ONLY for explicit reasons (manual-apply, auto-rule)
            const r = String(reason || '').trim();
            row.dataset.normalized = (r && r !== 'ai-preview') ? 'true' : 'false';
            if (r) row.dataset.normalizedReason = r;
            // hide only when it's actually finalized (not preview)
            plnSetRowHidden(row, (r && r !== 'ai-preview'));
        } catch (_) {/* noop */ }
    }

    function plnMaybeHideRow(row) {
        try {
            if (!row || row.nodeType !== 1) return;
            const isManual = row.dataset.manualNormalized === 'true';
            const isNormalized = row.dataset.normalized === 'true';
            const reason = String(row.dataset.normalizedReason || '').trim();
            const isPreview = reason === 'ai-preview' || row.dataset.aiPreview === '1' || row.dataset.aiProcessing === 'true';
            // Only hide if manual OR normalized for a non-preview reason
            if ((isManual) || (isNormalized && !isPreview)) {
                plnSetRowHidden(row, true);
            } else {
                plnSetRowHidden(row, false);
            }
        } catch (_) {/* noop */ }
    }

    /** [PLN][UI] Detección de enlaces de Google Maps dentro de un ámbito (fila/contendedor/document) */
    function plnDetectGoogleMapsLinkInScope(scope) {
        try {
            const root = (scope && scope.nodeType === 1) ? scope : document;

            // 1) Enlaces visibles y botones con data-url
            const nodes = Array.from(root.querySelectorAll('a[href],button[data-url],div[data-url]'));
            const hrefMatches = (u) => {
                const s = String(u || '');
                return /https?:\/\/(www\.)?google\.[^\/]+\/maps/i.test(s)
                    || /https?:\/\/goo\.gl\/maps/i.test(s)
                    || (/google\.[^\/]+\/search\?/i.test(s) && /(cid=|q=place_id:|ftid=)/i.test(s))
                    || /\/maps\/place\//i.test(s);
            };
            if (nodes.some(n => hrefMatches(n.getAttribute('href') || n.getAttribute('data-url') || ''))) return true;

            // 2) Metadatos comunes en DOM (cuando el link no es visible)
            if (root.querySelector('[data-placeid],[data-place-id],[data-google-cid],[data-cid],[data-gcid]')) return true;
        } catch (_) { }
        return false;
    }

    /** [PLN][UI] Detecta vía SDK si el Venue tiene Google Place asociado */
    async function plnHasGooglePlaceAssociation(placeId, scope) {
        try {
            // 0) Heurística DOM primero (más barata)
            if (plnDetectGoogleMapsLinkInScope(scope)) return true;

            // 1) Resolver ID
            let id = placeId;
            if (!id) {
                try {
                    const r = (scope && scope.nodeType === 1) ? scope : null;
                    id = (r && (r.dataset.placeId || r.dataset.venueId)) || null;
                    if (!id && r) {
                        const cand = r.querySelector('[data-place-id],[data-placeid],[data-venue-id],[data-id]');
                        if (cand) id = cand.dataset.placeId || cand.dataset.venueId || cand.getAttribute('data-id');
                    }
                } catch (_) { id = null; }
            }
            if (!id) return false;

            // 2) SDK
            const sdk = await plnEnsureSdkReady(4000);
            if (!sdk?.DataModel?.Venues?.getById) return false;
            const v = await sdk.DataModel.Venues.getById({ venueId: String(id) });
            if (!v) return false;

            const hasGoogle = (val) => {
                try {
                    if (val == null) return false;
                    if (typeof val === 'string') return /google|gplace|place_id|cid/i.test(val);
                    if (Array.isArray(val)) return val.some(hasGoogle);
                    if (typeof val === 'object') {
                        const s = JSON.stringify(val).toLowerCase();
                        return s.includes('google') && (s.includes('place') || s.includes('cid') || s.includes('gmb'));
                    }
                    return false;
                } catch (_) { return false; }
            };

            // Campos comunes observados en builds de WME
            if (hasGoogle(v.externalProviderRefs)) return true;
            if (hasGoogle(v.externalProviders)) return true;
            if (hasGoogle(v.externalRefs)) return true;
            if (hasGoogle(v.providers)) return true;
            if (hasGoogle(v.googlePlaceId || v.googlePlace || v.googlePlaces)) return true;
            if (hasGoogle(v.attributes && v.attributes.externalProviderRefs)) return true;
            if (hasGoogle(v.providerReferences || v.providerIds)) return true;
        } catch (_) { }
        return false;
    }

    /** [PLN][Google] Servicio de autocompletado (sugerir cuando no hay enlace de Google) */
    function plnGetPlaceAutocomplete() {
        try {
            const g = (typeof google !== 'undefined') ? google : null;
            const places = g && g.maps && g.maps.places ? g.maps.places : null;

            // Builds nuevas: AutocompleteSuggestion
            if (places && typeof places.AutocompleteSuggestion === 'function') {
                return new places.AutocompleteSuggestion();
            }
            // Compat: cuentas antiguas
            if (places && typeof places.AutocompleteService === 'function') {
                return new places.AutocompleteService();
            }
        } catch (e) {
            console.warn('[PLN] Error inicializando Google Places autocomplete.', e);
            return null;
        }
        console.warn('[PLN] Google Places autocomplete no disponible en esta sesión.');
        return null;
    }



    // === [PLN] Daily Usage Log core ===
    async function plnResolveCurrentUser() {
        // Try SDK → WazeWrap → WME internal (functions already in this file)
        try {
            if (window.__PLN_DECISION_DEBUG_ON) console.log('[WME PLN] 🕵️‍♂️ plnResolveCurrentUser: Starting detection...');
            let u = await getCurrentEditorViaSdk();
            if (u && window.__PLN_DECISION_DEBUG_ON) console.log('[WME PLN] ✅ User found via SDK:', u);

            if (!u) {
                if (window.__PLN_DECISION_DEBUG_ON) console.log('[WME PLN] ⚠️ User not found via SDK, trying WazeWrap...');
                u = getCurrentEditorViaWazeWrap();
                if (u && window.__PLN_DECISION_DEBUG_ON) console.log('[WME PLN] ✅ User found via WazeWrap:', u);
            }

            if (!u) {
                if (window.__PLN_DECISION_DEBUG_ON) console.log('[WME PLN] ⚠️ User not found via WazeWrap, trying WME Internal...');
                u = getCurrentEditorViaWmeInternal();
                if (u && window.__PLN_DECISION_DEBUG_ON) console.log('[WME PLN] ✅ User found via WME Internal:', u);
            }

            if (u && typeof u.id !== 'undefined') {
                currentGlobalUserInfo = { id: u.id, name: u.name || 'No detectado', privilege: u.privilege || 'N/A' };
                if (window.__PLN_DECISION_DEBUG_ON) console.log('[WME PLN] 🏁 plnResolveCurrentUser: Final result:', currentGlobalUserInfo);
                return currentGlobalUserInfo;
            }
        } catch (e) {
            console.error('[WME PLN] ❌ plnResolveCurrentUser Error:', e);
        }
        console.warn('[WME PLN] ⚠️ plnResolveCurrentUser: Failed to detect user.');
        return { id: null, name: 'No detectado', privilege: 'N/A' };
    }

    function plnHasLoggedToday(userId) {
        try {
            const key = `wme_pln_last_usage_${userId}`;
            const last = localStorage.getItem(key);
            return last === plnTodayStr();
        } catch (_) { return false; }
    }

    function plnMarkLoggedToday(userId) {
        try {
            const key = `wme_pln_last_usage_${userId}`;
            localStorage.setItem(key, plnTodayStr());
        } catch (_) { }
    }

    /**
     * Send a single POST to Apps Script only once per day per user.
     * Payload fields requested:
     *  - date (yyyy-mm-dd), userId, privilege (level), userName
     *  - workCities (permanent list from localStorage)
     */
    function plnLogUsageOncePerDay() {
        try {
            if (!PLN_USAGE_LOG || !PLN_USAGE_LOG.endpoint || !/^https?:\/\//i.test(PLN_USAGE_LOG.endpoint)) {
                // Not configured → nothing to do
                return;
            }
            const u = currentGlobalUserInfo || {};
            if (u.id == null || u.id === 0) return; // Need a user id
            if (plnHasLoggedToday(u.id)) return;    // Already logged today

            const cities = Array.from(plnLoadWorkCities());
            const payload = {
                date: plnTodayStr(),
                userId: u.id,
                privilege: u.privilege || 'N/A',
                userName: u.name || 'No detectado',
                workCities: cities,
                tz: Intl.DateTimeFormat().resolvedOptions().timeZone || 'America/Bogota',
                scriptVersion: (typeof GM_info !== 'undefined' && GM_info.script && GM_info.script.version) ? String(GM_info.script.version) : 'N/A'
            };

            // Send POST via GM_xmlhttpRequest (with visible error toasts)
            makeRequest({
                method: 'POST',
                url: PLN_USAGE_LOG.endpoint,
                data: JSON.stringify(payload),
                headers: {
                    'Content-Type': 'text/plain;charset=utf-8'   // evita preflight/CORS
                },
                timeout: PLN_USAGE_LOG.timeoutMs || 8000,
                onload: function (resp) {
                    try {
                        const status = String(resp && resp.status || '');
                        if (status.startsWith('2')) {
                            plnMarkLoggedToday(u.id);
                            try { plnToast('✅ Uso registrado en nube', 1800); } catch (_) { }
                        } else {
                            const msg = `[PLN] HTTP ${resp && resp.status} al registrar uso`;
                            try { plnToast(`⚠️ ${msg}`, 3000); } catch (_) { }
                            try { console.warn(msg, resp && resp.responseText); } catch (_) { }
                        }
                    } catch (err) {
                        try { plnToast('⚠️ Error inesperado al procesar respuesta', 3000); } catch (_) { }
                        try { console.error('[PLN] onload handler error', err); } catch (_) { }
                    }
                },
                onerror: function (err) {
                    try { plnToast('🚫 Fallo de conexión al registrar uso', 3200); } catch (_) { }
                    try { console.error('[PLN] Error de red al registrar uso:', err); } catch (_) { }
                },
                onabort: function () {
                    try { plnToast('⏹️ Envío cancelado (abort)', 2500); } catch (_) { }
                },
                ontimeout: function () {
                    try { plnToast('⏱️ Tiempo de espera agotado al registrar uso', 3000); } catch (_) { }
                }
            });
        } catch (e) {
            // console.error('[PLN Usage] unexpected error', e);
        }
    }

    /**
     * [PLN] User Activity Tracking - Separate from daily usage log
     * Registers user info in a separate Google Sheet to track:
     * - Username, User ID, Privilege Level
     * - Last usage date (updated every time the script runs)
     * - Total number of times the script has been used
     *
     * Unlike daily log, this runs EVERY TIME the script loads (not just once per day)
     */
    function plnTrackUserActivity() {
        try {
            // Check if tracking is enabled and configured
            if (!PLN_USER_TRACKING || !PLN_USER_TRACKING.enabled) {
                return; // Tracking disabled
            }

            if (!PLN_USER_TRACKING.endpoint || !/^https?:\/\//i.test(PLN_USER_TRACKING.endpoint)) {
                console.warn('[PLN User Tracking] Endpoint no configurado');
                return;
            }

            const u = currentGlobalUserInfo || {};
            if (u.id == null || u.id === 0) {
                console.warn('[PLN User Tracking] No se detectó ID de usuario');
                return;
            }

            // console.log('[PLN User Tracking] 🔍 DEBUG - currentGlobalUserInfo:', currentGlobalUserInfo);
            // console.log('[PLN User Tracking] 🔍 DEBUG - u.id:', u.id, 'tipo:', typeof u.id);
            // console.log('[PLN User Tracking] 🔍 DEBUG - u.privilege:', u.privilege, 'tipo:', typeof u.privilege);

            // WME usa índices base-0 para rank (0-5), convertir a nivel real (1-6)
            let displayLevel = 'N/A';
            if (typeof u.privilege === 'number') {
                displayLevel = u.privilege + 1;
            } else if (u.privilege && u.privilege !== 'N/A') {
                displayLevel = u.privilege;
            }

            // Obtener estadísticas de ediciones del usuario
            // IMPORTANTE: editorStats usa userName como clave, no userId
            const userStats = editorStats[u.name] || { daily_count: 0, total_count: 0 };

            // DEBUG: Verificar que editorStats se esté cargando correctamente
            // console.log('[PLN User Tracking] 🔍 DEBUG - editorStats completo:', editorStats);
            // console.log('[PLN User Tracking] 🔍 DEBUG - Buscando con userName:', u.name);
            // console.log('[PLN User Tracking] 🔍 DEBUG - userStats encontrado:', userStats);
            // console.log('[PLN User Tracking] 🔍 DEBUG - daily_count:', userStats.daily_count, 'total_count:', userStats.total_count);

            const payload = {
                userId: u.id,
                userName: u.name || 'No detectado',
                privilege: displayLevel,
                date: plnNowStr(), // Fecha y hora completa
                scriptVersion: (typeof GM_info !== 'undefined' && GM_info.script && GM_info.script.version)
                    ? String(GM_info.script.version)
                    : 'N/A',
                dailyEdits: userStats.daily_count || 0,  // Ediciones del día
                totalEdits: userStats.total_count || 0   // Ediciones totales
            };

            // console.log('[PLN User Tracking] 📤 Enviando datos:', payload);

            // Send POST request
            makeRequest({
                method: 'POST',
                url: PLN_USER_TRACKING.endpoint,
                data: JSON.stringify(payload),
                headers: {
                    'Content-Type': 'text/plain;charset=utf-8'
                },
                timeout: PLN_USER_TRACKING.timeoutMs || 8000,
                onload: function (resp) {
                    try {
                        const status = String(resp && resp.status || '');
                        if (status.startsWith('2')) {
                            // Success - silent (no toast to avoid spam)
                            // console.log('[PLN User Tracking] ✅ Usuario registrado/actualizado');

                            // Optional: Parse response to see if user was created or updated
                            try {
                                const result = JSON.parse(resp.responseText);
                                if (result.action === 'created') {
                                    console.log('[PLN User Tracking] 🆕 Nuevo usuario registrado');
                                } else if (result.action === 'updated') {
                                    // console.log('[PLN User Tracking] ♻️ Usuario actualizado. Total usos:', result.totalUses);
                                }
                            } catch (_) { }
                        } else {
                            console.warn('[PLN User Tracking] ⚠️ HTTP', resp && resp.status, resp && resp.responseText);
                        }
                    } catch (err) {
                        console.error('[PLN User Tracking] Error procesando respuesta:', err);
                    }
                },
                onerror: function (err) {
                    console.error('[PLN User Tracking] 🚫 Error de conexión:', err);
                },
                onabort: function () {
                    console.warn('[PLN User Tracking] ⏹️ Envío cancelado');
                },
                ontimeout: function () {
                    console.warn('[PLN User Tracking] ⏱️ Timeout al registrar usuario');
                }
            });
        } catch (e) {
            console.error('[PLN User Tracking] Error inesperado:', e);
        }
    }

    // === Debug helper: force a log now (clears today's guard for current user) ===
    try {
        window.plnForceLogNow = function () {
            plnResolveCurrentUser().then(() => {
                try {
                    const uid = (currentGlobalUserInfo && currentGlobalUserInfo.id != null) ? currentGlobalUserInfo.id : 'test';
                    localStorage.removeItem(`wme_pln_last_usage_${uid}`);
                } catch (_) { }
                plnLogUsageOncePerDay();
                try { plnToast('Intentando enviar log ahora...', 2000); } catch (_) { }
            });
        };
        if (typeof unsafeWindow !== 'undefined') unsafeWindow.plnForceLogNow = window.plnForceLogNow;
    } catch (_) { }

    // === Debug helper: force user tracking now ===
    try {
        window.plnForceTrackUser = function () {
            plnResolveCurrentUser().then(() => {
                plnTrackUserActivity();
                try { plnToast('📊 Intentando registrar usuario ahora...', 2000); } catch (_) { }
            });
        };
        if (typeof unsafeWindow !== 'undefined') unsafeWindow.plnForceTrackUser = window.plnForceTrackUser;
    } catch (_) { }

    // ===================================================================
    // [PLN][SDK] Helper
    // ===================================================================
    /** [PLN][SDK] Espera a que el SDK esté 100% listo antes de operar.
     *  - Verifica existencia de getWmeSdk / wmeSDK
     *  - Espera DataModel y los métodos básicos de Venues
     *  - Opcionalmente reintenta si el modelo aún está cargando
     */
    async function plnEnsureSdkReady(maxMs = 5000) {
        const t0 = Date.now();
        // 1) Asegurar instancia
        if (!window.WME_PLN_SDK) {
            try {
                if (typeof getWmeSdk === 'function') {
                    window.WME_PLN_SDK = getWmeSdk({ scriptId: 'WMEPlacesNameInspector', scriptName: 'WME PLN' });
                } else if (typeof wmeSDK !== 'undefined') {
                    window.WME_PLN_SDK = wmeSDK;
                }
            } catch (_) { }
        }
        let sdk = window.WME_PLN_SDK || (typeof wmeSDK !== 'undefined' ? wmeSDK : null);

        // 2) Esperar a que exista DataModel y Venues.getById
        while (Date.now() - t0 < maxMs) {
            try {
                if (!sdk) {
                    if (typeof getWmeSdk === 'function') {
                        window.WME_PLN_SDK = getWmeSdk({ scriptId: 'WMEPlacesNameInspector', scriptName: 'WME PLN' });
                        sdk = window.WME_PLN_SDK;
                    } else if (typeof wmeSDK !== 'undefined') {
                        sdk = wmeSDK;
                    }
                }
                if (sdk?.DataModel?.Venues?.getById) return sdk;

                // Algunas builds exponen ready()
                if (typeof sdk?.ready === 'function') {
                    try { await sdk.ready(); } catch (_) { }
                    if (sdk?.DataModel?.Venues?.getById) return sdk;
                }
            } catch (_) { }
            await new Promise(r => setTimeout(r, 100));
        }
        // Devuelve lo que haya; los callers validan capacidades
        return sdk;
    }
    // === [PLN][AI] Robust JSON parsing (tolerant to extra bytes, logs, or fences) ===
    function plnExtractFirstJsonObject(text) {
        try {
            const s = String(text || '').trim().replace(/^\uFEFF/, '');
            // Strip markdown fences like ```json ... ```
            const fenced = s.match(/```(?:json)?\s*([\s\S]*?)```/i);
            const source = fenced ? fenced[1].trim() : s;

            // Fast path
            try { return JSON.parse(source); } catch (_) { }

            // Try to find the first balanced JSON object { ... }
            let inStr = false, esc = false, depth = 0, start = -1, end = -1;
            for (let i = 0; i < source.length; i++) {
                const ch = source[i];
                if (inStr) {
                    if (esc) { esc = false; continue; }
                    if (ch === '\\') { esc = true; continue; }
                    if (ch === '"') { inStr = false; continue; }
                    continue;
                }
                if (ch === '"') { inStr = true; continue; }
                if (ch === '{') {
                    if (depth === 0) start = i;
                    depth++;
                    continue;
                }
                if (ch === '}') {
                    depth--;
                    if (depth === 0) {
                        end = i;
                        const candidate = source.slice(start, end + 1);
                        try { return JSON.parse(candidate); } catch (_) { /* keep scanning */ }
                    }
                }
                // also tolerate arrays as top-level
                if (ch === '[') {
                    if (depth === 0) start = i;
                    depth++;
                }
                if (ch === ']') {
                    depth--;
                    if (depth === 0) {
                        end = i;
                        const candidate = source.slice(start, end + 1);
                        try { return JSON.parse(candidate); } catch (_) { /* keep scanning */ }
                    }
                }
            }

            // Last fallback: cut at the last closing brace/ bracket
            const lastCurly = source.lastIndexOf('}');
            const lastBracket = source.lastIndexOf(']');
            const cut = Math.max(lastCurly, lastBracket);
            if (cut >= 0) {
                const slice = source.slice(0, cut + 1);
                try { return JSON.parse(slice); } catch (_) { }
            }
        } catch (_) { }
        return null;
    }

    function plnSafeJsonParse(text) {
        try {
            // First attempt
            return JSON.parse(text);
        } catch (err1) {
            const recovered = plnExtractFirstJsonObject(text);
            if (recovered !== null) return recovered;
            return {
                error: 'Respuesta de IA no válida (parser tolerante no pudo recuperar JSON).',
                details: String(err1 && err1.message || err1),
                rawHead: String(text || '').slice(0, 400)
            };
        }
    }

    // Llama al Worker de IA para obtener sugerencias de nombres normalizados.
    async function getAiSuggestions(request) {
        const WORKER_URL = 'https://wme-normalizer-ai.cotalvaro.workers.dev/';
        const DATA_SOURCES = ['foursquare', 'googlePlaces', 'yelp', 'openStreetMap'];

        return new Promise((resolve) => {
            if (typeof makeRequest !== 'function') {
                console.error('[WME PLN][AI] makeRequest no está disponible.', { request });
                resolve({ error: "La función 'makeRequest' no está disponible." });
                return;
            }

            const safeRequest = typeof request === 'object' && request !== null ? request : {};
            const payload = {
                placeName: safeRequest.placeName,
                // Compat: aceptar cityName pero enviar 'city' al Worker
                city: (safeRequest.city || safeRequest.cityName || ''),
                // Compat: aceptar categoryKeys pero enviar 'wazeCategories' al Worker
                wazeCategories: Array.isArray(safeRequest.wazeCategories)
                    ? safeRequest.wazeCategories
                    : (Array.isArray(safeRequest.categoryKeys) ? safeRequest.categoryKeys : []),
                sources: DATA_SOURCES,
                metadata: { requestedAt: new Date().toISOString() }
            };
            console.debug('[WME PLN][AI] → request', payload);
            const requestDebug = { url: WORKER_URL, payload };

            makeRequest({
                method: "POST",
                url: WORKER_URL,
                headers: { "Content-Type": "application/json" },
                data: JSON.stringify(payload),
                timeout: 20000,
                onload: (response) => {
                    const isOk = response.status >= 200 && response.status < 300;
                    if (isOk) {
                        try {
                            const parsed = plnSafeJsonParse(response.responseText);
                            console.debug('[WME PLN][AI] ← response', parsed);
                            resolve(parsed);
                        } catch (e) {
                            console.error('[WME PLN][AI] Error parseando respuesta del agente.', {
                                error: e,
                                responseText: response.responseText,
                                request: requestDebug
                            });
                            resolve({
                                error: "Respuesta de IA no válida.",
                                responseText: response.responseText
                            });
                        }
                    } else {
                        console.error('[WME PLN][AI] El agente respondió con error HTTP.', {
                            status: response.status,
                            responseText: response.responseText,
                            request: requestDebug
                        });
                        try {
                            resolve(plnSafeJsonParse(response.responseText));
                        } catch (_) {
                            resolve({
                                error: `Error HTTP ${response.status}.`,
                                responseText: response.responseText
                            });
                        }
                    }
                },
                onerror: (error) => {
                    console.error('[WME PLN][AI] Error de red al consultar el agente.', {
                        error,
                        request: requestDebug
                    });
                    try { plnToast('🚫 Error de red con el agente IA', 2800); } catch (_) { }
                    resolve({ error: "Error de red.", details: error });
                },
                ontimeout: () => {
                    console.error('[WME PLN][AI] Timeout al consultar el agente.', requestDebug);
                    try { plnToast('⏱️ Timeout consultando el agente IA', 2800); } catch (_) { }
                    resolve({ error: "Timeout.", request: requestDebug });
                }
            });
        });
    }//getAiSuggestions


    /** [PLN][AI] Resolver ciudad real del Venue usando solo el SDK.
     *  Cubre:
     *   1) address.city.name / primaryName
     *   2) address.street.city (si la ciudad viene derivada de la calle)
     *   3) address.streetId -> Streets.getById().city
     *   4) entryPoints[0].street?.city
     *   5) freeformAddress o address.text: último token como heurística
     */
    async function plnResolveCityNameViaSdk(placeId) {
        try {
            if (!window.WME_PLN_SDK && typeof getWmeSdk === "function") {
                window.WME_PLN_SDK = getWmeSdk({ scriptId: "WMEPlacesNameInspector", scriptName: "WME PLN" });
            }
            const sdk = window.WME_PLN_SDK || wmeSDK;
            const venue = await sdk.DataModel.Venues.getById({ venueId: String(placeId) });
            if (!venue) return "Desconocida";

            // 1) Ciudad directa en la dirección
            const directCity = (venue.address && (venue.address.city?.name || venue.address.city?.primaryName));
            if (directCity) return String(directCity);

            // 2) Ciudad desde la calle embebida en la address (WME suele derivarla así)
            const addrStreetCity = venue.address && venue.address.street && (venue.address.street.city?.name || venue.address.street.city?.primaryName);
            if (addrStreetCity) return String(addrStreetCity);

            // 3) address.streetId -> Streets.getById().city
            const streetId = venue.address && (venue.address.streetId || venue.address.street?.id || venue.address.street?.ID || venue.address.street?.streetId);
            if (streetId && sdk.DataModel.Streets && sdk.DataModel.Streets.getById) {
                try {
                    const st = await sdk.DataModel.Streets.getById({ streetId: String(streetId) });
                    const stCity = st && st.city && (st.city.name || st.city.primaryName);
                    if (stCity) return String(stCity);
                } catch (_) { }
            }

            // 4) entryPoints → street.city
            if (Array.isArray(venue.entryPoints) && venue.entryPoints.length) {
                const ep = venue.entryPoints[0];
                const epCity = ep && ep.street && (ep.street.city?.name || ep.street.city?.primaryName);
                if (epCity) return String(epCity);
                const epStreetId = ep && (ep.streetId || ep.street?.id);
                if (epStreetId && sdk.DataModel.Streets && sdk.DataModel.Streets.getById) {
                    try {
                        const st = await sdk.DataModel.Streets.getById({ streetId: String(epStreetId) });
                        const stCity = st && st.city && (st.city.name || st.city.primaryName);
                        if (stCity) return String(stCity);
                    } catch (_) { }
                }
            }

            // 5) Heurística de última coma en la dirección libre
            const free = (venue.address && (venue.address.freeformAddress || venue.address.text)) || '';
            if (free) {
                const last = String(free).split(',').map(s => s.trim()).filter(Boolean).pop();
                if (last) return last;
            }

            return "Desconocida";
        } catch (e) {
            console.warn("[PLN][AI] plnResolveCityNameViaSdk fallo:", e);
            return "Desconocida";
        }
    }

    /** [PLN][AI] Resolver ciudad leyendo el MISMO origen visual del “chulito de ciudad”
     *  Busca en el row cualquier elemento con `title` o texto que empiece por "Ciudad:"
     *  Ej.: title="Ciudad: Bello (derivada de la calle)"  →  "Bello"
     */
    function plnResolveCityFromRowDom(row) {
        try {
            if (!row || row.nodeType !== 1) return '';
            // 1) Revisa elementos con title
            const els = Array.from(row.querySelectorAll('[title]'));
            for (const el of els) {
                const t = String(el.getAttribute('title') || '').trim();
                if (t && /^Ciudad\s*:/.test(t)) {
                    // Extrae lo que va tras "Ciudad:" hasta un paréntesis o fin
                    const m = t.match(/^Ciudad\s*:\s*([^()]+?)(?:\s*\(|$)/i);
                    if (m && m[1]) return m[1].trim();
                }
            }
            // 2) Revisa texto directo en la celda Tipo/Ciudad si existe
            const maybeCell = row.querySelector('td') || null;
            if (maybeCell) {
                const txt = String(maybeCell.textContent || '').trim();
                const m2 = txt.match(/\bCiudad\s*:\s*([^()]+?)(?:\s*\(|$)/i);
                if (m2 && m2[1]) return m2[1].trim();
            }
        } catch (_) { }
        return '';
    }

    /** [PLN][AI] Heurística local de categorías cuando el agente no devuelve nada */
    function plnGuessCategoriesByName(name) {
        try {
            const n = String(name || '').toLowerCase();

            const out = new Set();

            // Transporte / logística / encomiendas
            if (/(transporte|transportes|transportadora|mensajer(ia|ía)|paqueter[ií]a|log[ií]stic|carga|fletes|mudanz)/.test(n)) {
                out.add('TRANSPORTATION');
                out.add('CAR_SERVICES'); // fallback general
            }
            // Gobierno, oficina pública
            if (/(alcald[ií]a|gobernaci[oó]n|secretar[ií]a|oficina\s+(de\s+)?(tr[áa]nsito|hacienda|planeaci[oó]n)|impuest|municipalidad|ayuntamiento)/.test(n)) {
                out.add('GOVERNMENT');
                out.add('CITY_HALL');
            }
            // Religioso
            if (/(parroquia|iglesia|templo|capilla|comunidad\s+cristiana)/.test(n)) out.add('RELIGIOUS_CENTER');
            // Personal care
            if (/(est[ée]tica|peluquer[ií]a|barber[ií]a|manicur|spa|belleza|odontolog|dent|cl[íi]nica\s+dental)/.test(n)) out.add('PERSONAL_CARE');
            // Banco/financiera
            if (/(banco|cooperativa\s+de\s+cr[eé]dito|financier[ao]|caja\s+de\s+compensaci[oó]n)/.test(n)) { out.add('BANK'); out.add('BANK_FINANCIAL'); }
            // Educación
            if (/(colegio|escuela|universidad|institut[oa]|jard[ií]n\s+infantil|kinder|kinderg|preescolar)/.test(n)) { out.add('SCHOOL'); out.add('UNIVERSITY'); }
            // Salud
            if (/(cl[íi]nica|hospital|urgencias|m[eé]dica|doct\.?|odontolog)/.test(n)) { out.add('HOSPITAL'); out.add('DOCTOR_CLINIC'); }

            // Normaliza a array en mayúsculas
            const arr = Array.from(out).map(k => String(k).toUpperCase());
            return arr.length ? arr : [];
        } catch (_) { return []; }
    }

    /** [PLN][AI] Aplicar categorías al Venue usando SOLO el SDK, con rutas de compatibilidad */
    async function plnApplyCategoriesViaSdk(placeId, categoryKeysUpperArray) {
        if (!placeId) throw new Error('placeId requerido');

        // Esperar SDK listo y normalizar ID
        const sdk = await plnEnsureSdkReady(6000);
        const venueIdStr = String(placeId);
        const venueIdNum = Number.isNaN(Number(placeId)) ? null : Number(placeId);
        if (!sdk) throw new Error('SDK no disponible');

        // Normaliza a UPPER y prepara dos formatos: objetos {key} y strings "KEY"
        const keysUpper = (categoryKeysUpperArray || [])
            .filter(Boolean)
            .map(k => String(k).toUpperCase());
        if (!keysUpper.length) return { applied: false, reason: 'No hay categorías para aplicar' };

        const catsObj = keysUpper.map(k => ({ key: k }));
        const catsStr = keysUpper.slice(0);

        // Log de capacidades detectadas para diagnosticar rápidamente
        try {
            const venuePeek = await sdk?.DataModel?.Venues?.getById?.({ venueId: venueIdStr });
            const caps = {
                hasSDK: !!sdk,
                namespaces: Object.keys(sdk || {}),
                hasDataModel: !!sdk?.DataModel,
                hasCommands: !!sdk?.Commands,
                hasOperations: !!sdk?.Operations
            };
            console.debug('[WME PLN][AI] SDK namespaces/caps:', caps);
        } catch (_) { }

        // Helper interno para intentar una llamada y reportar método usado
        async function tryMethod(label, fn) {
            try {
                await fn();
                return { ok: true, method: label };
            } catch (e) {
                console.warn('[WME PLN][AI] Categorías método falló:', label, e);
                return { ok: false, err: e };
            }
        }

        // 1) Ruta directa sobre el objeto Venue
        try {
            const venue = await sdk.DataModel?.Venues?.getById?.({ venueId: venueIdStr });
            if (venue) {
                // Algunas versiones requieren refresh() antes de mutar
                try { if (typeof venue.refresh === 'function') await venue.refresh(); } catch (_) { }

                // Probar con payloads usando {key} y strings y ambos tipos de id
                if (typeof venue.setCategories === 'function') {
                    let r = await tryMethod('venue.setCategories([{key}])', () => venue.setCategories(catsObj));
                    if (r.ok) return { applied: true, method: r.method, payload: catsObj };

                    r = await tryMethod('venue.setCategories(["KEY"])', () => venue.setCategories(catsStr));
                    if (r.ok) return { applied: true, method: r.method, payload: catsStr };

                    if (typeof venue.update === 'function') {
                        r = await tryMethod('venue.update({categories:[{key}]})', () => venue.update({ categories: catsObj }));
                        if (r.ok) return { applied: true, method: r.method, payload: catsObj };

                        r = await tryMethod('venue.update({categories:["KEY"]})', () => venue.update({ categories: catsStr }));
                        if (r.ok) return { applied: true, method: r.method, payload: catsStr };
                    }
                }
                // 1.b update({categories:[{key}]}) y update({categories:["KEY"]})
                if (typeof venue.update === 'function') {
                    let r = await tryMethod('venue.update({categories:[{key}]})', () => venue.update({ categories: catsObj }));
                    if (r.ok) return { applied: true, method: r.method, payload: catsObj };
                    r = await tryMethod('venue.update({categories:["KEY"]})', () => venue.update({ categories: catsStr }));
                    if (r.ok) return { applied: true, method: r.method, payload: catsStr };
                }
            }
        } catch (e1) {
            console.warn('[WME PLN][AI] Acceso directo al Venue falló', e1);
        }

        // 2) DataModel global
        try {
            if (sdk.DataModel?.Venues?.update) {
                let r = await tryMethod('DataModel.Venues.update({categories:[{key}]})',
                    () => sdk.DataModel.Venues.update({ venueId: venueIdStr, categories: catsObj }));
                if (r.ok) return { applied: true, method: r.method, payload: catsObj };

                r = await tryMethod('DataModel.Venues.update({categories:["KEY"]})',
                    () => sdk.DataModel.Venues.update({ venueId: venueIdStr, categories: catsStr }));
                if (r.ok) return { applied: true, method: r.method, payload: catsStr };

                r = await tryMethod('DataModel.Venues.update({id, categories:[{key}]})',
                    () => sdk.DataModel.Venues.update({ id: venueIdStr, categories: catsObj }));
                if (r.ok) return { applied: true, method: r.method, payload: catsObj };

                if (venueIdNum !== null) {
                    r = await tryMethod('DataModel.Venues.update({id:Number, categories:["KEY"]})',
                        () => sdk.DataModel.Venues.update({ id: venueIdNum, categories: catsStr }));
                    if (r.ok) return { applied: true, method: r.method, payload: catsStr };
                }
            }
        } catch (e2) {
            console.warn('[WME PLN][AI] DataModel.Venues.update falló', e2);
        }

        // 3) Operations (algunas versiones exponen esto)
        try {
            if (sdk.Operations?.Venues?.updateCategories) {
                let r = await tryMethod('Operations.Venues.updateCategories([{key}])',
                    () => sdk.Operations.Venues.updateCategories({ venueId: venueIdStr, categories: catsObj }));
                if (r.ok) return { applied: true, method: r.method, payload: catsObj };

                r = await tryMethod('Operations.Venues.updateCategories(["KEY"])',
                    () => sdk.Operations.Venues.updateCategories({ venueId: venueIdStr, categories: catsStr }));
                if (r.ok) return { applied: true, method: r.method, payload: catsStr };
            }
            if (sdk.Operations?.Venues?.setCategories) {
                let r = await tryMethod('Operations.Venues.setCategories([{key}])',
                    () => sdk.Operations.Venues.setCategories({ venueId: venueIdStr, categories: catsObj }));
                if (r.ok) return { applied: true, method: r.method, payload: catsObj };
            }
        } catch (e3) {
            console.warn('[WME PLN][AI] Operations.Venues.updateCategories/setCategories falló', e3);
        }

        // 4) Commands genérico (nombres varían entre builds)
        try {
            if (sdk.Commands?.apply) {
                const idVariants = [venueIdStr].concat(venueIdNum !== null ? [venueIdNum] : []);
                const cmdPayloads = [];
                for (const idv of idVariants) {
                    cmdPayloads.push(['Commands.apply:Venue.SetCategories([{key}])', [{ type: 'Venue.SetCategories', venueId: idv, categories: catsObj }]]);
                    cmdPayloads.push(['Commands.apply:Venue.SetCategories(["KEY"])', [{ type: 'Venue.SetCategories', venueId: idv, categories: catsStr }]]);
                    cmdPayloads.push(['Commands.apply:Venues.Update({categories:[{key}]})', [{ type: 'Venues.Update', venueId: idv, categories: catsObj }]]);
                    cmdPayloads.push(['Commands.apply:SetVenueCategories(["KEY"])', [{ type: 'SetVenueCategories', venueId: idv, categories: catsStr }]]);
                    // otros alias que he visto
                    cmdPayloads.push(['Commands.apply:Venues.SetCategories([{key}])', [{ type: 'Venues.SetCategories', venueId: idv, categories: catsObj }]]);
                    cmdPayloads.push(['Commands.apply:EditVenueCategories', [{ type: 'EditVenueCategories', venueId: idv, categories: catsStr }]]);
                }
                for (const [label, payload] of cmdPayloads) {
                    const r = await tryMethod(label, () => sdk.Commands.apply(payload));
                    if (r.ok) return { applied: true, method: label, payload };
                }
            }
        } catch (e4) {
            console.warn('[WME PLN][AI] Commands.apply falló', e4);
        }

        // Ninguna ruta funcionó
        const err = new Error('Venue SDK inválido');
        err.sdkCaps = {
            hasSDK: !!(await plnEnsureSdkReady(1)),
            hasDataModel: !!(await plnEnsureSdkReady(1))?.DataModel,
            triedRoutes: ['venue.setCategories', 'venue.update', 'DataModel.Venues.update', 'Operations.Venues.updateCategories|setCategories', 'Commands.apply(...) variants']
        };
        console.warn('[WME PLN][AI] Ninguna ruta de SDK funcionó', err.sdkCaps);
        throw err;
    }

    //Maneja la lógica de la UI para una solicitud de IA en una fila específica.
    /** [PLN][AI] Aplicar NOMBRE al Venue usando SOLO el SDK, con fallback UI */
    async function plnApplyNameViaSdk(row, placeId, newName) {
        const name = plnFixDanglingHyphen(String(newName || '').trim());
        if (!name) return { applied: false, reason: 'Nombre vacío' };
        const sdk = await plnEnsureSdkReady(6000);
        if (!sdk) throw new Error('SDK no disponible');

        try {
            const venue = await sdk?.DataModel?.Venues?.getById?.({ venueId: String(placeId) });
            if (!venue) throw new Error('Venue no encontrado por SDK');
            try { if (typeof venue.refresh === 'function') await venue.refresh(); } catch (_) { }
            // Intentos comunes
            if (typeof venue.setPrimaryName === 'function') { await venue.setPrimaryName(name); return { applied: true, method: 'venue.setPrimaryName', value: name }; }
            if (typeof venue.setName === 'function') { await venue.setName(name); return { applied: true, method: 'venue.setName', value: name }; }
            if (typeof venue.update === 'function') { await venue.update({ name }); return { applied: true, method: 'venue.update', value: name }; }
        } catch (e) {
            console.warn('[WME PLN][AI] No se pudo aplicar nombre mediante SDK directo, fallback UI', e);
        }

        // Fallback: dejar el nombre en el textarea de sugerencia para que el flujo normal lo aplique
        try {
            const ta = row && row.querySelector('.replacement-input');
            if (ta) { ta.value = name; return { applied: true, method: 'ui.textarea', value: name }; }
        } catch (_) { }

        // Fallback final: intentar Commands.apply con Venue.Update
        if (sdk?.Commands?.apply) {
            const idv = Number.isNaN(Number(placeId)) ? String(placeId) : Number(placeId);
            try {
                await sdk.Commands.apply([{ type: 'Venue.Update', venueId: idv, name }]);
                return { applied: true, method: 'Commands.apply:Venue.Update', value: name };
            } catch (_) { }
        }

        return { applied: false, reason: 'No se pudo aplicar el nombre' };
    }

    // === Limpieza de guiones colgantes/desalineados (Unicode-safe) ===
    function plnFixDanglingHyphens(s) {
        let out = String(s || '');

        // 1) Guión al final → eliminar (salvo casos bloqueados)
        const trailingPattern = /\s*-\s*$/u;
        if (trailingPattern.test(out) && !plnShouldPreserveTrailingHyphen(out)) {
            out = out.replace(trailingPattern, '');
        }

        // 2) Normalizar espacios en guiones válidos
        out = out.replace(/\s*-\s*/gu, ' - ');

        // 3) Compactar espacios y recortar
        out = out.replace(/\s+/g, ' ').trim();

        return out;
    }
    // === Swap Engine: move configured tokens before/after ===
    // Respeta el arreglo global window.swapWords guardado en localStorage.
    // Cada item puede definir la palabra en `word` | `text` | `token`
    // y la dirección en `position` | `where` | `dir` | `direction` con valores
    // "before" | "after" o "antes" | "despues/después".
    function applySwapRules(originalName) {
        try {
            const DBG = false; // Cambia esto de "!!(window.__PLN_SWAP_DEBUG_ON || localStorage...)" a simplemente false
            let name = String(originalName || '');
            const swaps = (typeof plnCollectSwapRules === 'function')
                ? plnCollectSwapRules()
                : (Array.isArray(window.swapWords) ? window.swapWords : []);

            //if (DBG) console.group('[PLN Swap] applySwapRules', { originalName, swapsCount: Array.isArray(swaps)?swaps.length:0 });
            if (!swaps.length) { if (DBG) { console.warn('[PLN Swap] skip: no swaps configured'); console.groupEnd?.(); } return name; }

            const normalizeSpace = s => s.replace(/\s+/g, ' ').replace(/\s*-\s*/g, ' - ').trim();

            for (const raw of swaps) {
                if (!raw) { if (DBG) console.warn('skip: null item'); continue; }
                const token = String((raw.word || raw.text || raw.token || '').trim());
                if (!token) { if (DBG) console.warn('skip: empty token', raw); continue; }

                let where = String((raw.position || raw.where || raw.dir || raw.direction || '')).toLowerCase();
                if (where === 'antes' || where === 'before' || where === 'pre') where = 'before';
                if (where === 'despues' || where === 'después' || where === 'after' || where === 'post') where = 'after';
                if (where !== 'before' && where !== 'after') { if (DBG) console.warn(`skip [${token}]: invalid position`, raw); continue; }

                const esc = token.replace(/[.*+?^${}()|[\]\\]/g, '\\$&').replace(/\s+/g, '\\s+');
                const SEP = '[\\s,.;:()\\[\\]\\-–—\/]';

                // RegEx para buscar (sin 'g') y para reemplazar (con 'g')
                const reFind = new RegExp(`(?:^|${SEP})${esc}(?=$|${SEP})`, 'iu');
                const reAnywhere = new RegExp(`(?:^|${SEP})${esc}(?=$|${SEP})`, 'giu');
                const reStart = new RegExp(`^\\s*${esc}(?=$|${SEP})`, 'iu');
                const reEnd = new RegExp(`(?:^|${SEP})${esc}\\s*$`, 'iu');

                if (DBG) console.groupCollapsed(`[${token}] → ${where}`);
                if (!reFind.test(name)) {
                    if (DBG) { console.info('no-op: token not present in name', { name, token }); console.groupEnd?.(); }
                    continue;
                }

                if ((where === 'before' && reStart.test(name)) || (where === 'after' && reEnd.test(name))) {
                    if (DBG) { console.info('no-op: already at target edge', { name }); console.groupEnd?.(); }
                    name = normalizeSpace(name);
                    continue;
                }

                const before = name;
                // Eliminar apariciones previas del token como palabra independiente
                name = name.replace(reAnywhere, ' ').replace(/\s{2,}/g, ' ').trim();
                // Colocar en el borde solicitado y normalizar espacios
                name = where === 'before' ? `${token} ${name}`.trim() : `${name} ${token}`.trim();
                name = normalizeSpace(name);
                // if (DBG) console.log('moved', { before, after: name });
                if (DBG) console.groupEnd?.();
            }

            if (DBG) { console.log('result =>', name); console.groupEnd?.(); }
            // 👉 Limpieza final de guiones colgantes
            name = plnFixDanglingHyphens(name);
            return name;
        }
        catch (e) {
            //if (window.__PLN_SWAP_DEBUG_ON) console.error('[PLN Swap] error', e);
            return originalName;
        }
    }

    // --- Post-swap capitalization helpers ---
    function plnCapitalizeStart(str) {
        try { return String(str || '').replace(/^\s*([a-záéíóúñ])/iu, (m, c) => m.replace(c, c.toUpperCase())); } catch { return str; }
    }

    // Capitaliza la primera letra después de " - "
    function plnCapitalizeAfterHyphen(str) {
        try {
            // Capitaliza PALABRAS COMPLETAS después de guion con espacios: " - la" → " - La"
            // Esto sobrescribe STOP words como "la", "el", "de", etc.
            let result = String(str || '').replace(/(\s-\s*)([a-záéíóúñ][\wáéíóúñ]*)/giu, (m, sep, word) => {
                return sep + word.charAt(0).toUpperCase() + word.slice(1).toLowerCase();
            });
            // Capitaliza después de guion sin espacio antes: "word-a" → "word-A"
            result = result.replace(/([^\s])-([a-záéíóúñ][\wáéíóúñ]*)/giu, (m, before, word) => {
                return before + '-' + word.charAt(0).toUpperCase() + word.slice(1).toLowerCase();
            });
            return result;
        } catch (_) { return String(str || ''); }
    }

    // Title-case en español: stopwords en minúscula salvo si van al inicio; respeta siglas
    // REEMPLAZA o AÑADE esta función en tu script

    function plnTitleCaseEs(str) {
        try {
            const STOP = new Set(['de', 'la', 'las', 'el', 'los', 'y', 'e', 'o', 'u', 'un', 'una', 'unos', 'unas', 'a', 'en', 'con', 'tras', 'por', 'al', 'lo']);
            const isAllCaps = w => w.length > 1 && w === w.toUpperCase();
            const cap = w => w.charAt(0).toUpperCase() + w.slice(1).toLowerCase();
            let i = 0;

            return String(str || '').replace(/([\p{L}\p{M}][\p{L}\p{M}\.'’]*)/gu, (m) => {
                const w = m, lw = w.toLowerCase(), atStart = (i === 0);
                i += w.length;

                const excl = (typeof isExcludedWord === 'function') ? isExcludedWord(w) : null;
                if (excl) return excl;
                if (isAllCaps(w)) return w;

                if (STOP.has(lw) && !atStart) {
                    // ✅ **LA MEJORA ESTÁ AQUÍ** ✅
                    // Si la palabra original ya estaba en mayúscula (ej. "La"),
                    // la respetamos y no la convertimos a minúscula.
                    if (w.charAt(0) === w.charAt(0).toUpperCase()) {
                        return w;
                    }
                    return lw; // De lo contrario, sí la convertimos a minúscula.
                }
                return cap(w);
            });
        }
        catch {
            return str;
        }
    }//plnTitleCaseEs

    function plnPostSwapCap(str) {
        let out = String(str || '');
        out = plnTitleCaseEs(out);
        out = plnCapitalizeStart(out);
        out = plnCapitalizeAfterHyphen(out); // <-- asegura mayúscula tras " - "
        return out.trim();
    }

    // === Ocultar filas ya normalizadas / sin acción requerida ===
    // ✅ REEMPLAZA EL BLOQUE ANTERIOR CON ESTE ✅
    function plnPruneNormalizedRowsManager() {
        try {
            if (window.__plnPruneRowsActive) return;
            window.__plnPruneRowsActive = true;

            const HIDE_CLASS = 'pln-hidden-normalized';
            if (!document.getElementById('pln-hide-style')) {
                const st = document.createElement('style');
                st.id = 'pln-hide-style';
                // Oculta las filas solo si el body tiene la clase 'pln-hide-normalized-rows'
                st.textContent = `
                body.pln-hide-normalized-rows tr.${HIDE_CLASS} {
                    display: none !important;
                }
            `;
                //st.textContent = `tr.${HIDE_CLASS}{display:none !important;}`;
                document.head.appendChild(st);
            }

            function expectedOf(s) {
                try {
                    const orig = (typeof window.__plnNormalizeOriginal === 'function') ?
                        window.__plnNormalizeOriginal :
                        (typeof window.normalizePlaceName === 'function' ? window.normalizePlaceName : null);

                    let out = String(s || '');
                    if (orig) out = orig(out);
                    out = plnPostSwapCap(out);
                    out = plnApplyExclusions(out);
                    out = applySwapRules(out);
                    if (orig) out = orig(out);
                    out = plnPostSwapCap(out);
                    out = plnApplyExclusions(out);
                    return out.trim();
                } catch (_) {
                    return String(s || '').trim();
                }
            }

            function processRow(tr) {
                if (!tr || tr.nodeType !== 1) return;
                const tas = tr.querySelectorAll('textarea');
                if (!tas || tas.length === 0) return;

                const current = (tas[0].value || '').trim();
                const suggested = (tas.length > 1 ? tas[1].value : '').trim();
                if (!current) return;

                const expected = expectedOf(current);
                const actionDisabled = Array
                    .from(tr.querySelectorAll('button[disabled], input[disabled], .disabled, [aria-disabled="true"]'))
                    .length > 0;
                const noChange = (suggested && suggested === current) || (current === expected) || actionDisabled;

                if (noChange) tr.classList.add(HIDE_CLASS);
                else tr.classList.remove(HIDE_CLASS);
            }

            function processAll() {
                document.querySelectorAll('tr').forEach(processRow);
            }

            // Exportamos el observador y la función para usarlos externamente
            let debounceTimer;
            window.plnPruneObserver = new MutationObserver(() => {
                clearTimeout(debounceTimer);
                debounceTimer = setTimeout(processAll, 250); // Mantenemos el debounce
            });

            window.plnPruneProcessAll = processAll;

            // YA NO SE OBSERVA document.body AQUÍ

            setTimeout(processAll, 300);
            let ticks = 0;
            const iv = setInterval(() => {
                processAll();
                if (++ticks > 10) clearInterval(iv);
            }, 400);

            window.__plnHideNormalizedRows = processAll;
        } catch (_) { /* noop */ }
    }

    // Llamamos a la función para que prepare las herramientas
    plnPruneNormalizedRowsManager();

    // Exponer para pruebas rápidas en consola
    //  window.__pln_applySwapRulesTest = applySwapRules;
    (function plnAutoSwapSuggestionInputs() {
        try {
            document.addEventListener('input', (ev) => {
                const t = ev.target;
                if (t && t.matches && t.matches('textarea.replacement-input')) {
                    t.dataset.userEdited = '1'; // no pisar ediciones manuales
                }
            }, true);

            function applyOnceTo(el) {
                // DESHABILITADO: Esto causaba normalización doble
                // Los textareas ya tienen los valores correctos desde processPlaceName en renderRow
                // Llamar normalizePlaceName de nuevo aquí causa que los reemplazos se apliquen dos veces
                return;

                /* CÓDIGO ORIGINAL - DESHABILITADO
                if (!el || el.dataset.userEdited === '1') return;
                const tr = el.closest('tr[data-place-id]') || el.closest('tr');
                let currentName = '';
                if (tr) {
                    const tas = tr.querySelectorAll('textarea');
                    if (tas && tas.length) currentName = tas[0].value || '';
                }
                // Usa el normalizador oficial ya parcheado con swap
                const norm = (typeof normalizePlaceName === 'function')
                    ? normalizePlaceName
                    : (typeof window.__plnNormalizeOriginal === 'function' ? window.__plnNormalizeOriginal : null);
                const expected = norm ? norm(currentName) : currentName;
                if (expected && el.value !== expected) el.value = expected;
                */
            }

            let debounceTimer;
            const debouncedHandler = () => {
                // Esta función se asegura de que solo se procesen los textareas una vez.
                document.querySelectorAll('textarea.replacement-input:not([data-pln-processed])').forEach(el => {
                    applyOnceTo(el);
                    el.setAttribute('data-pln-processed', 'true'); // Marcar como procesado
                });
            };

            const obs = new MutationObserver(muts => {
                for (const m of muts) {
                    if (m.addedNodes.length > 0) {
                        clearTimeout(debounceTimer);
                        // Agrupa todas las llamadas en una sola ejecución 100ms después del último cambio.
                        debounceTimer = setTimeout(debouncedHandler, 100);
                        break; // Un solo nodo añadido es suficiente para activar el debounce.
                    }
                }
            });
            obs.observe(document.body, { childList: true, subtree: true });

            window.plnApplySwapToSuggestionInputs = function () {
                document.querySelectorAll('textarea.replacement-input').forEach(applyOnceTo);
            };
        } catch (_) { }
    })();
    // Hook: encadena las reglas de swap al resultado de normalizePlaceName
    (function plnPatchNormalizeForSwap() {
        try {
            if (window.__plnSwapPatched) return;
            let tries = 0;
            const iv = setInterval(() => {
                tries++;
                const fn = (typeof normalizePlaceName === 'function') ? normalizePlaceName : (typeof window !== 'undefined' && typeof window.normalizePlaceName === 'function' ? window.normalizePlaceName : null);
                if (fn) {
                    clearInterval(iv);
                    const original = fn;
                    // Exponer el normalizador original para usarlo post‑swap sin recursión
                    try { window.__plnNormalizeOriginal = original; } catch (_) { }

                    // ✅ CÓDIGO CORRECTO DE REEMPLAZO
                    const patched = function (...args) {
                        const inStr = args && args.length ? String(args[0] || '') : '';

                        // 1. Usar la normalización ORIGINAL (sin processPlaceName que ya aplicó reemplazos)
                        // Esto evita que se apliquen los reemplazos dos veces
                        const baseNormalizada = original(inStr);

                        // 2. Aplicar las reglas de 'swap' (mover palabras) al resultado ya normalizado.
                        const swapped = applySwapRules(baseNormalizada);

                        // 3. Aplicar una capitalización final y limpieza post-swap.
                        const finalCap = plnPostSwapCap(swapped);
                        const finalStr = plnApplyExclusions(finalCap); // Reponer palabras excluidas al final.

                        // (Opcional) Bloque de depuración para trazar el flujo
                        try {
                            const DBG = window.__PLN_DECISION_DEBUG_ON || localStorage.getItem('wme_pln_debug_decision') === '1';
                            if (DBG) {
                                console.group('[PLN Decision] normalizePlaceName patched (CORREGIDO)');
                                console.table({
                                    "Input": inStr,
                                    "1. After original normalize": baseNormalizada,
                                    "2. After applySwapRules": swapped,
                                    "3. Final Result": finalStr
                                });
                                console.groupEnd();
                            }
                        } catch (_) { }

                        // 4. Devolver el resultado final y unificado.
                        return finalStr.trim();
                    };  // Colgar en ambos scopes por seguridad
                    try { window.normalizePlaceName = patched; } catch (_) { }
                    try { normalizePlaceName = patched; } catch (_) { }
                    window.__plnSwapPatched = true;
                    //console.log('[WME PLN] Swap rules hooked into normalizePlaceName.');
                }
                if (tries > 60) { clearInterval(iv); }
            }, 200);
        } catch (_) { /* noop */ }
    })();
    // Variables globales para el panel flotante
    let floatingPanelElement = null;
    let dynamicCategoriesLoaded = false;
    const tempSelectedCategories = new Map(); // Mapa para placeId -> categoryKey seleccionada
    const placesForDuplicateCheckGlobal = []; // Nueva variable global para almacenar datos de lugares para verificar duplicados
    let filterUpdateRequestsOnly = false; // Variable global para el filtro de update requests
    let recommendCategoriesWrapperGlobal = null; // Referencia global al contenedor de checkboxes para crear URs dinámicamente

    const processingPanelDimensions = { width: '400px', height: '200px' };  // Panel pequeño para procesamiento
    const resultsPanelDimensions = { width: '1400px', height: '700px' };    // Panel grande para resultados
    // ✨ OPTIMIZACIÓN: Regex Precompiladas (28-nov-2024)
    const RE_HYPHEN_BETWEEN_LETTERS = /\p{L}-\p{L}/u;
    const RE_ACRONYM_PART = /^[A-ZÁÉÍÓÚÑ0-9.]+$/;
    const RE_ACRONYM_WORD = /^[A-ZÁÉÍÓÚÑ0-9.&]+$/;
    const RE_DIGIT = /\d/;
    const RE_SYMBOL = /[\&.]/;
    const RE_ROMAN = /^M{0,3}(CM|CD|D?C{0,3})(XC|XL|L?X{0,3})(IX|IV|V?I{0,3})$/i;
    const RE_ROMAN_WRAPPED = /^([(\["'¿¡]*)([MDCLXVI]+)([)"'\]°ºª.,;:!?]*)$/i;

    const commonWords = new Set([//Palabras comunes en español que no deberían ser consideradas para normalización
        'es', 'de', 'del', 'el', 'la', 'los', 'las', 'y', 'e',
        'o', 'u', 'un', 'una', 'unos', 'unas', 'a', 'en',
        'con', 'tras', 'por', 'al', 'lo'
    ]);

    // Hacer commonWords accesible globalmente para addWordToDictionary
    window.commonWords = commonWords;
    const tabNames = [//Definir nombres de pestañas cortos antes de la generación de botones
        { label: "Gene", icon: "⚙️" },
        { label: "Espe", icon: "🏷️" },
        { label: "Dicc", icon: "📘" },
        { label: "Reemp", icon: "🔂" }
    ]; let statsPanelElement = null; // Para el panel flotante de estadísticas
    let editorStats = {}; // Para almacenar las estadísticas en memoria
    const STATS_STORAGE_KEY = 'wme_pln_editor_stats'; // Clave para localStorage
    const STATS_ENABLED_KEY = 'wme_pln_stats_enabled'; // Clave para el checkbox de visibilidad

    // Cargar estadísticas inmediatamente al iniciar el script
    loadEditorStats();

    let wmeSDK = null; // Almacena la instancia del SDK de WME.
    //Variable global para almacenar la información del usuario actual
    let currentGlobalUserInfo = { id: null, name: 'Cargando...', privilege: 'N/A' };

    // Exponer para debugging (solo en desarrollo)
    if (typeof window !== 'undefined') {
        window.plnDebugUserInfo = function() {
            return currentGlobalUserInfo;
        };
    }

    // Set para rastrear palabras añadidas localmente por el usuario
    // CORRECCIÓN: Verificar si está vacío para intentar cargar, por si se inicializó antes
    if (!window.userAddedDictionaryWords || window.userAddedDictionaryWords.size === 0) {
        if (!window.userAddedDictionaryWords) window.userAddedDictionaryWords = new Set();

        // Cargar desde localStorage si existe
        try {
            const saved = localStorage.getItem('userAddedDictionaryWords');
            if (saved) {
                const parsed = JSON.parse(saved);
                // Usar add para no perder referencia si ya existía el objeto
                parsed.forEach(w => window.userAddedDictionaryWords.add(w));
                // console.log(`[WME PLN] Cargadas ${parsed.length} palabras de usuario desde localStorage.`);
            }
        } catch (e) {
            console.error('[WME PLN] Error cargando palabras de usuario:', e);
        }
    }

    // ✨ OPTIMIZACIÓN: Mapa de palabras normalizadas a palabras con tilde (28-nov-2024)
    // Permite búsqueda O(1) para corrección de tildes en lugar de iterar sobre el diccionario
    if (!window.normalizedToDictionaryMap) window.normalizedToDictionaryMap = new Map();

    /**
     * Construye el mapa inverso de palabras normalizadas (sin tildes) -> palabras originales
     * Se debe llamar cada vez que se actualiza el diccionario.
     */
    function buildNormalizedDictionaryMap() {
        if (!window.dictionaryWords) return;

        window.normalizedToDictionaryMap.clear();
        window.dictionaryWords.forEach(word => {
            const lowerWord = word.toLowerCase();
            const normalized = removeDiacritics(lowerWord);

            // Solo nos interesa si la palabra tiene tildes y su versión normalizada es diferente
            if (lowerWord !== normalized) {
                // Guardamos la versión normalizada apuntando a la versión con tilde
                // Ej: "camion" -> "camión"
                window.normalizedToDictionaryMap.set(normalized, lowerWord);
            }
        });
        // console.log(`[WME PLN] Mapa de tildes construido. ${window.normalizedToDictionaryMap.size} entradas.`);
    }

    // ✨ OPTIMIZACIÓN: Cache de Lookups Frecuentes (28-nov-2024)
    // Evita repetir búsquedas de las mismas palabras (mejora ~20-25% en rendimiento)
    const wordLookupCache = new Map();
    const CACHE_MAX_SIZE = 500;

    /**
     * Verifica si una palabra está excluida o en el diccionario usando cache LRU
     * @param {string} word - Palabra a verificar (se convertirá a lowercase)
     * @returns {Object} { isExcluded: boolean, isInDictionary: boolean }
     */
    function isWordExcludedOrInDictionary(word) {
        const lowerWord = word.toLowerCase();

        // Intentar obtener del cache
        if (wordLookupCache.has(lowerWord)) {
            return wordLookupCache.get(lowerWord);
        }

        // Si no está en cache, hacer la búsqueda completa
        const result = {
            isExcluded: excludedWords.has(lowerWord),
            isInDictionary: window.dictionaryWords?.has(lowerWord) || false
        };

        // Guardar en cache
        wordLookupCache.set(lowerWord, result);

        // Limitar tamaño del cache (LRU simple: eliminar primera entrada)
        if (wordLookupCache.size > CACHE_MAX_SIZE) {
            const firstKey = wordLookupCache.keys().next().value;
            wordLookupCache.delete(firstKey);
        }

        return result;
    }

    /**
     * Limpia el cache de lookups (llamar cuando se modifica diccionario o palabras excluidas)
     */
    function clearWordLookupCache() {
        wordLookupCache.clear();
    }

    // Exponer clearWordLookupCache globalmente para addWordToDictionary
    window.clearWordLookupCache = clearWordLookupCache;

    // ✨ OPTIMIZACIÓN: Paginación de Listas (28-nov-2024)
    // Solo renderiza palabras visibles en lugar de miles de elementos DOM
    const WORDS_PER_PAGE = 100; // Palabras por página
    let dictionaryCurrentPage = 1;
    let excludedWordsCurrentPage = 1;

    // ✨ OPTIMIZACIÓN: Cache de Similitud (28-nov-2024)
    // Cache para almacenar resultados de Levenshtein entre pares de palabras
    const similarityCache = new Map();
    const SIMILARITY_CACHE_MAX_SIZE = 2000; // Guardar hasta 2000 comparaciones recientes

    //Novedades de cada version del script esto permitirá una pantalla la primera vez que se abra el script
    const myChangelog = {
        [VERSION]: {
            "✨ Novedades": [
                "📋 Agregados logs de diagnóstico detallados para verificar permisos de EasyShortCuts."
            ],
            "🐛 Correcciones": [                   
                "⏰ EasyShortCuts: Implementada espera inteligente (hasta 15s) para asegurar que la configuración del servidor cargue correctamente antes de verificar accesos.",
                "🔧 Solucionado problema donde usuarios con nivel permitido veían el módulo desactivado por carga rápida de configuración."
            ]
        }
    }; // myChangelog



    function checkForOverlappingHours(venueSDKObject) {
        if (!venueSDKObject || !venueSDKObject.openingHours) {
            return false; // No hay horarios que analizar.
        }

        const openingHours = venueSDKObject.openingHours;
        let hasOverlap = false;

        const timeToMinutes = (timeStr) => {
            if (typeof timeStr !== 'string' || !timeStr.includes(':')) return 0;
            const [hours, minutes] = timeStr.split(':').map(Number);
            return hours * 60 + minutes;
        };

        for (const day in openingHours.days) {
            const dayRanges = openingHours.days[day];

            if (Array.isArray(dayRanges) && dayRanges.length > 1) {
                const intervals = dayRanges.map(range => ({
                    start: timeToMinutes(range.from),
                    end: timeToMinutes(range.to)
                }));

                for (let i = 0; i < intervals.length; i++) {
                    for (let j = i + 1; j < intervals.length; j++) {
                        const interval1 = intervals[i];
                        const interval2 = intervals[j];

                        if (interval1.start < interval2.end && interval1.end > interval2.start) {
                            hasOverlap = true;
                            break;
                        }
                    }
                    if (hasOverlap) break;
                }
            }
            if (hasOverlap) break;
        }
        return hasOverlap;
    }// checkForOverlappingHours
    // Verifica si hay ediciones pendientes en el objeto del lugar
    function checkForPendingEdits(venueSDKObject) {
        // Estilo Allen: Ser defensivo. Si no hay objeto o no tiene la propiedad, no hay ediciones pendientes.
        if (!venueSDKObject || !venueSDKObject.venueUpdateRequests) {
            return false;
        }

        // La propiedad 'venueUpdateRequests' es un array. Si tiene uno o más elementos, hay ediciones pendientes.
        if (Array.isArray(venueSDKObject.venueUpdateRequests) && venueSDKObject.venueUpdateRequests.length > 0) {
            return true;
        }

        return false;
    }// checkForPendingEdits


    // --- Línea de referencia (NO la copies, es solo para ubicarte) ---
    // function applySwapRules(originalName) { ...
    // Modified GM_xmlhttpRequest wrapper that tracks requests
    function makeRequest(options) {
        const requestId = Date.now() + Math.random();

        const wrappedOptions = {
            ...options,
            onload: function (response) {
                // Remove this request from pending list
                pendingRequests = pendingRequests.filter(id => id !== requestId);
                if (typeof options.onload === 'function') {
                    options.onload(response);
                }
            },
            onerror: function (error) {
                // Remove this request from pending list
                pendingRequests = pendingRequests.filter(id => id !== requestId);
                if (typeof options.onerror === 'function') {
                    options.onerror(error);
                }
            }
        };

        // Add to pending requests
        pendingRequests.push(requestId);

        // Make the actual request
        return GM_xmlhttpRequest(wrappedOptions);
    }
    // Variables globales para el diccionario de lugares excluidos
    function loadExcludedPlacesFromStorage() {
        try {
            const savedExcludedPlaces = localStorage.getItem("excludedPlacesList");
            if (savedExcludedPlaces) {
                const parsedData = JSON.parse(savedExcludedPlaces);
                excludedPlaces = new Map(parsedData);
                window.excludedPlaces = excludedPlaces;
                console.log("[WME PLN] Lugares excluidos cargados:", excludedPlaces.size);
            } else {
                excludedPlaces = new Map();
                window.excludedPlaces = excludedPlaces;
                console.log("[WME PLN] No se encontraron lugares excluidos guardados.");
            }
        } catch (e) {
            console.error("[WME PLN] Error cargando lugares excluidos desde localStorage:", e);
            excludedPlaces = new Map();
            window.excludedPlaces = excludedPlaces;
        }
        if (typeof window.updateExcludedPlacesCountLabel === 'function') {
            window.updateExcludedPlacesCountLabel();
        }
    }//loadExcludedPlacesFromStorage
    // [PLN] Cerca eléctrica: impedir que reasignaciones borren pares del usuario
    (function () {
        try {
            // Backing stores
            let __plnRW = window.replacementWords || {};
            let __plnRS = (typeof window.replacementSources === 'object' && window.replacementSources) ? window.replacementSources : {};

            // Asegurar objeto de fuentes existente
            if (!window.replacementSources || typeof window.replacementSources !== 'object') window.replacementSources = (__plnRS = {});

            Object.defineProperty(window, 'replacementWords', {
                configurable: true,
                enumerable: true,
                get() { return __plnRW; },
                set(next) {
                    try {
                        const prevMap = __plnRW || {};
                        const prevSrc = __plnRS || {};
                        const incoming = (next && typeof next === 'object') ? { ...next } : {};

                        // Reinyectar SOLO entradas previas cuyo source != 'sheet' y que no estén en el nuevo mapa
                        for (const k in prevMap) {
                            if (prevSrc[k] !== 'sheet' && !(k in incoming)) {
                                incoming[k] = prevMap[k];
                                if (!window.replacementSources || typeof window.replacementSources !== 'object') window.replacementSources = (__plnRS = {});
                                if (window.replacementSources[k] !== 'sheet') window.replacementSources[k] = 'user';
                            }
                        }
                        __plnRW = incoming;
                    } catch (e) { __plnRW = next || {}; }
                }
            });

            Object.defineProperty(window, 'replacementSources', {
                configurable: true,
                enumerable: true,
                get() { return __plnRS; },
                set(next) { __plnRS = (next && typeof next === 'object') ? next : {}; }
            });
        } catch (_) { /* noop */ }
    })();
    // --- Funciones auxiliares para manejar fechas ---
    // Obtiene la fecha actual en formato AAAA-MM-DD
    function getCurrentDateString() {
        const now = new Date();
        const year = now.getFullYear();
        const month = String(now.getMonth() + 1).padStart(2, '0');
        const day = String(now.getDate()).padStart(2, '0');
        return `${year}-${month}-${day}`;
    }
    // Obtiene la semana actual en formato AAAA-WW (ISO 8601)
    function getCurrentISOWeekString() {
        const date = new Date();
        date.setHours(0, 0, 0, 0);
        date.setDate(date.getDate() + 3 - (date.getDay() + 6) % 7);
        const week1 = new Date(date.getFullYear(), 0, 4);
        const weekNumber = 1 + Math.round(((date.getTime() - week1.getTime()) / 86400000 - 3 + (week1.getDay() + 6) % 7) / 7);
        return `${date.getFullYear()}-${String(weekNumber).padStart(2, '0')}`;
    }
    // Obtiene el mes actual en formato AAAA-MM
    function getCurrentMonthString() {
        const now = new Date();
        const year = now.getFullYear();
        const month = String(now.getMonth() + 1).padStart(2, '0');
        return `${year}-${month}`;
    }

    // --- Funciones principales de estadísticas ---
    // Carga las estadísticas desde localStorage
    function loadEditorStats() {
        const savedStats = localStorage.getItem(STATS_STORAGE_KEY);
        if (savedStats) {
            try {
                editorStats = JSON.parse(savedStats);
                if (typeof editorStats !== 'object' || editorStats === null) {
                    editorStats = {};
                }
            }
            catch (e) {
                console.error('[WME PLN Stats] Error al parsear estadísticas desde localStorage:', e);
                editorStats = {};
            }
        }
        else {
            editorStats = {};
        }
    }
    // Guarda las estadísticas en localStorage
    function saveEditorStats() {
        try {
            localStorage.setItem(STATS_STORAGE_KEY, JSON.stringify(editorStats));
        }
        catch (e) {
            console.error('[WME PLN Stats] Error al guardar estadísticas en localStorage:', e);
        }
    }

    // Devuelve la palabra excluida original si existe en la lista (case-insensitive, ignora tildes y espacios)
    function isExcludedWord(word) {
        if (!word || !excludedWords) return null;
        // Normaliza quitando tildes y pasa a minúsculas
        const clean = w => w.trim().toLowerCase().normalize("NFD").replace(/[\u0300-\u036f]/g, "");
        const cleanedWord = clean(word);

        for (const excl of excludedWords) {
            if (clean(excl) === cleanedWord) {
                return excl; // Devuelve la versión guardada, con sus mayúsculas/tildes originales
            }
        }
        return null;
    }// isExcludedWord

    // Aplica las palabras excluidas devolviendo su forma exacta guardada
    function plnApplyExclusions(str) {
        try {
            const reWord = /([\p{L}\p{M}][\p{L}\p{M}\.'’]*)/gu;
            return String(str || '').replace(reWord, (m) => {
                try {
                    const excl = typeof isExcludedWord === 'function' ? isExcludedWord(m) : null;
                    return excl ? excl : m;
                } catch (_) { return m; }
            });
        } catch (_) { return String(str || ''); }
    }

    // FIN: Bloque De Funciones Para Estadísticas

    //Calcula el área de un polígono en metros cuadrados
    // Utiliza la fórmula de Shoelace (fórmula de área de Gauss
    function calculateAreaNoTurf(shape) {
        if (!shape || !shape.geometry) {
            return Infinity; // retorna un valor que no active el titilado
        }

        try {
            // verifica si la geometría es un polígono
            if (shape.geometry.type === 'Polygon') {
                // extrae las coordenadas del primer anillo del polígono
                const coordinates = shape.geometry.coordinates[0]; // primer anillo del polígono

                // usar la fórmula de Shoelace para calcular el área
                let area = 0;
                for (let i = 0; i < coordinates.length - 1; i++) {
                    area += coordinates[i][0] * coordinates[i + 1][1];
                    area -= coordinates[i][1] * coordinates[i + 1][0];
                }

                area = Math.abs(area) / 2;
                //Convierte el área a metros cuadrados
                // Aproximación: 1 grado de latitud es aproximadamente 111.32 km
                // 1 grado de longitud varía según la latitud, pero al nivel del equador es aproximadamente 111.32 km
                // Para simplificar, usamos un valor promedio
                // 1 grado de latitud = 111,319.9 metros
                // 1 grado de longitud = 111,319.9 metros (en el ecuador)
                // Entonces, el área en metros cuadrados es:
                // area * (111319.9 * 111319.9)
                // O simplemente multiplicamos por el cuadrado de la conversión de metros por grado
                // Nota: Esta es una aproximación y no es precisa para áreas grandes o cerca de los polos
                // Para áreas pequeñas, esta aproximación es suficiente
                const metersPerDegree = 111319.9; //aproximadamente 111,319.9 metros por grado
                return area * Math.pow(metersPerDegree, 2);
            }
        }
        catch (error) {
            console.warn("[WME PLN] Error calculating area:", error);
            return Infinity; // Return a value that won't trigger blinking
        }

        return Infinity; // Default return for non-polygon shapes
    }// calculateAreaMeters

    // Modifica esta función para implementar correctamente el procesamiento de titilación
    // Reemplaza completamente la función processingLoop
    function processingLoop() {
        const currentTime = Date.now();

        // Actualizar el estado de titilación cada BLINK_INTERVAL milisegundos
        if (currentTime - lastBlinkTime > BLINK_INTERVAL) {
            blinkState = !blinkState;
            lastBlinkTime = currentTime;

            // Recorrer todos los lugares en la lista de lugares para titilar
            blinkingPlaces.forEach(placeId => {
                // Seleccionar específicamente el elemento con la clase area-value-element
                const areaElement = document.querySelector(`tr[data-place-id="${placeId}"] .area-value-element`);
                if (areaElement) {
                    // Solo aplicar el estilo de titilación al elemento del área
                    areaElement.style.opacity = blinkState ? '1' : '0.3';
                }
            });
        }

        requestAnimationFrame(processingLoop);
    }// processingLoop



    // Registra una edición y actualiza los contadores
    function recordNormalizationEvent() {
        const userId = currentGlobalUserInfo.id;
        const userName = currentGlobalUserInfo.name;

        if (!userId || userId === 0 || userName === 'No detectado') {
            return;
        }

        // Obtiene las estadísticas del usuario o las inicializa si no existen
        let userStats = editorStats[userId];
        if (!userStats) {
            userStats = {
                userName: userName,
                total_count: 0,
                monthly_count: 0,
                monthly_period: "N/A",
                weekly_count: 0,
                weekly_period: "N/A",
                daily_count: 0,
                daily_period: "N/A",
                last_update: 0
            };
            editorStats[userId] = userStats;
        }

        // Obtiene los periodos de tiempo actuales
        const todayStr = getCurrentDateString();
        const weekStr = getCurrentISOWeekString();
        const monthStr = getCurrentMonthString();

        // --- Lógica de reseteo de contadores ---
        // Si la fecha guardada es diferente a la de hoy, resetea el contador diario.
        if (userStats.daily_period !== todayStr) {
            userStats.daily_count = 0;
            userStats.daily_period = todayStr;
        }

        // Si la semana guardada es diferente a la de hoy, resetea el contador semanal.
        if (userStats.weekly_period !== weekStr) {
            userStats.weekly_count = 0;
            userStats.weekly_period = weekStr;
        }

        // Si el mes guardado es diferente al de hoy, resetea el contador mensual.
        if (userStats.monthly_period !== monthStr) {
            userStats.monthly_count = 0;
            userStats.monthly_period = monthStr;
        }

        // --- Incrementar los contadores ---
        userStats.daily_count++;
        userStats.weekly_count++;
        userStats.monthly_count++;
        userStats.total_count++;
        userStats.last_update = Date.now();
        userStats.userName = userName; // Asegurarse de que el nombre esté actualizado

        // Guardar los nuevos datos y actualizar la pantalla
        saveEditorStats();
        updateStatsDisplay();
    }

    // Función modificada para filtrar palabras antes de mostrarlas en la lista de selección
    function openAddSpecialWordPopup(name, listType = "excludeWords") {
        // Dividir el nombre en palabras
        const clean = s => String(s || '').normalize('NFD').replace(/[\u0300-\u036f]/g, '')
            .replace(/\s+/g, ' ').trim().toLowerCase();
        const exclSet = new Set(Array.from(window.excludedWords || []).map(w => clean(w)));

        let words = name.split(/\s+/)
            .filter(word => word.length > 1);

        words = words.filter(word => {
            const lowerWord = word.toLowerCase();

            // Verificar que la palabra no sea común
            if (commonWords.has(lowerWord)) return false;

            // Verificar que no esté ya en palabras excluidas (insensible a mayúsculas/minúsculas)
            if (exclSet.has(clean(word))) {
                return false; // Ya está en especiales
            }

            // Verificar que no esté en el diccionario (insensible a mayúsculas/minúsculas)
            if (window.dictionaryWords && Array.from(window.dictionaryWords).some(dictWord =>
                dictWord.toLowerCase() === lowerWord)) {
                return false;
            }

            // Si la palabra coincide con algún "texto de reemplazo" (columna B) cargado desde Google Sheet,
            // y ese reemplazo proviene de hoja (bloqueado), no mostrarla en la lista
            try {
                if (window.replacementWords) {
                    const isReplacementTargetFromSheet = Object.entries(window.replacementWords)
                        .some(([from, to]) => (to || '').toLowerCase() === lowerWord && (!window.replacementSources || window.replacementSources[from] === 'sheet'));
                    if (isReplacementTargetFromSheet) return false;
                }
            } catch (_) { /* noop */ }

            return true; // Si pasa todos los filtros, mostrar la palabra
        });

        if (words.length === 0) {
            alert("No hay palabras nuevas para agregar. Todas las palabras ya están en el diccionario o en la lista de especiales.");
            return;
        }

    }



    // Muestra los contadores en el panel flotante
    function updateStatsDisplay() {
        if (!statsPanelElement || !currentGlobalUserInfo.id) return;

        const userId = currentGlobalUserInfo.id;
        // Obtiene los datos guardados o valores por defecto si no existen
        const stats = editorStats[userId] || {
            daily_count: 0,
            weekly_count: 0,
            monthly_count: 0,
            total_count: 0
        };

        // Actualiza los elementos de la UI con los valores guardados
        const summaryText = statsPanelElement.querySelector('#stats-summary-text');
        const todayCountSpan = statsPanelElement.querySelector('#stats-count-today');
        const weekCountSpan = statsPanelElement.querySelector('#stats-count-week');
        const monthCountSpan = statsPanelElement.querySelector('#stats-count-month');
        const totalCountSpan = statsPanelElement.querySelector('#stats-count-total');

        if (summaryText) summaryText.textContent = `📊 ${stats.daily_count || 0} Places NrmliZed`;
        if (todayCountSpan) todayCountSpan.textContent = stats.daily_count || 0;
        if (weekCountSpan) weekCountSpan.textContent = stats.weekly_count || 0;
        if (monthCountSpan) monthCountSpan.textContent = stats.monthly_count || 0;
        if (totalCountSpan) totalCountSpan.textContent = stats.total_count || 0;
    }

    // Crea el panel de estadísticas flotante en la interfaz de usuario.
    function createStatsPanel() {
        if (document.getElementById('wme-pln-stats-panel')) return;

        // Contenedor principal del panel
        statsPanelElement = document.createElement('div');
        statsPanelElement.id = 'wme-pln-stats-panel';
        Object.assign(statsPanelElement.style, {
            position: 'fixed',
            bottom: '60px',
            left: '23%', // <-- Ancla el panel a 20px del borde izquierdo
            // Se elimina la propiedad 'transform' que ya no es necesaria
            backgroundColor: 'rgba(45, 45, 45, 0.9)',
            color: 'white',
            padding: '5px 12px',
            borderRadius: '15px',
            fontSize: '13px',
            fontFamily: "'Helvetica Neue', Helvetica, Arial, sans-serif",
            zIndex: '10000',
            cursor: 'pointer',
            display: 'none', // Oculto inicialmente
            border: '1px solid #555',
            boxShadow: '0 2px 10px rgba(0,0,0,0.5)',
            userSelect: 'none',
            whiteSpace: 'nowrap'
        });
        // Vista de resumen (la que siempre está visible)
        const summaryView = document.createElement('div');
        summaryView.id = 'stats-summary-view';
        Object.assign(summaryView.style, {
            display: 'flex',
            alignItems: 'center',
            gap: '6px'
        });
        const summaryText = document.createElement('span');
        summaryText.id = 'stats-summary-text';
        summaryText.textContent = '📊 0 NrmliZer Stats';

        const dropdownArrow = document.createElement('span');
        dropdownArrow.id = 'stats-arrow';
        dropdownArrow.textContent = '▼';
        dropdownArrow.style.fontSize = '10px';

        summaryView.appendChild(summaryText);
        summaryView.appendChild(dropdownArrow);

        // Vista detallada (la que se expande)
        const detailView = document.createElement('div');
        detailView.id = 'stats-detail-view';
        Object.assign(detailView.style, {
            display: 'none',
            marginTop: '8px',
            paddingTop: '8px',
            borderTop: '1px solid #666'
        });

        const list = document.createElement('ul');
        Object.assign(list.style, {
            margin: '0',
            padding: '0',
            listStyle: 'none',
            textAlign: 'left'
        });

        // Crear elementos de la lista
        const items = {
            'Hoy': 'stats-count-today',
            'Esta Semana': 'stats-count-week',
            'Este Mes': 'stats-count-month',
            'Total': 'stats-count-total'
        };

        for (const [label, id] of Object.entries(items)) {
            const listItem = document.createElement('li');
            listItem.style.marginBottom = '4px';

            const countBold = document.createElement('b');
            countBold.id = id;
            countBold.textContent = '0';

            listItem.append(`${label}: `, countBold);
            list.appendChild(listItem);
        }

        detailView.appendChild(list);

        // Ensamblar el panel
        statsPanelElement.appendChild(summaryView);
        statsPanelElement.appendChild(detailView);
        document.body.appendChild(statsPanelElement);

        // Lógica para desplegar/contraer
        statsPanelElement.addEventListener('click', () => {
            const isHidden = detailView.style.display === 'none';
            detailView.style.display = isHidden ? 'block' : 'none';
            dropdownArrow.textContent = isHidden ? '▲' : '▼';
        });

        // Lógica para cerrar al hacer clic fuera
        document.addEventListener('click', (e) => {
            // Ignorar clicks en botones del disclaimer
            const btn = e.target.closest('button');
            if (btn && btn.hasAttribute('data-disclaimer-btn')) return;
            
            if (!statsPanelElement.contains(e.target)) {
                detailView.style.display = 'none';
                dropdownArrow.textContent = '▼';
            }
        }, true);

        toggleStatsPanelVisibility();
    }// createStatsPanel
    // Función para alternar la visibilidad del panel de estadísticas basado en el estado del checkbox.
    function toggleStatsPanelVisibility() {
        if (!statsPanelElement) return;
        const isEnabled = localStorage.getItem(STATS_ENABLED_KEY) === 'true';
        statsPanelElement.style.display = isEnabled ? 'block' : 'none';
    }// toggleStatsPanelVisibility
    // FIN: Bloque De Funciones Para Estadísticas

    // Función que construirá el HTML del changelog
    function getChangelogHtml(versionData) {
        let html = '';
        if (!versionData) return html;

        // Iterar dinámicamente sobre todas las categorías
        Object.keys(versionData).forEach(category => {
            const items = versionData[category];
            if (Array.isArray(items) && items.length > 0) {
                // Usar la categoría como título (incluye emojis si los tiene)
                html += `<h6 style="color: #2196F3; margin-top: 10px; margin-bottom: 5px; font-size: 14px;">${category}</h6>`;
                html += `<ul style="margin-bottom: 10px; list-style-type: disc; margin-left: 20px;">`;
                items.forEach(item => {
                    html += `<li>${item}</li>`;
                });
                html += `</ul>`;
            }
        });
        return html;
    }//getChangelogHtml

    // Función para mostrar el changelog al actualizar el script
    function showChangelogOnUpdate() {
        const LAST_SEEN_VERSION_KEY = `${SCRIPT_NAME}_last_seen_version`;
        const lastSeenVersion = localStorage.getItem(LAST_SEEN_VERSION_KEY);
        const currentScriptVersion = VERSION; // Variable global VERSION
        // Obtener la versión actual del script desde GM_info
        const versionData = myChangelog[currentScriptVersion];
        // Verificar si hay datos de versión y si la versión actual es diferente a la última vista
        if (versionData && currentScriptVersion !== lastSeenVersion) {
            const title = `${SCRIPT_NAME} v${currentScriptVersion}`;
            const bodyHtml = getChangelogHtml(versionData); // Genera el HTML del cuerpo
            // Crear el modal
            const modal = document.createElement("div");
            modal.style.position = "fixed";
            modal.style.top = "50%";
            modal.style.left = "50%";
            modal.style.transform = "translate(-50%, -50%)";
            modal.style.backgroundColor = "#fff";
            modal.style.border = "1px solid #ccc";
            modal.style.borderRadius = "8px";
            modal.style.boxShadow = "0 5px 15px rgba(0,0,0,0.3)";
            modal.style.padding = "20px";
            modal.style.fontFamily = "'Helvetica Neue', Helvetica, Arial, sans-serif";
            modal.style.zIndex = "20000"; // Por encima de casi todo
            modal.style.width = "450px";
            modal.style.maxHeight = "80vh";
            modal.style.overflowY = "auto";
            // Estilos adicionales para el modal
            const modalTitle = document.createElement("h3");
            modalTitle.textContent = title;
            modalTitle.style.marginTop = "0";
            modalTitle.style.marginBottom = "15px";
            modalTitle.style.textAlign = "center";
            modalTitle.style.color = "#333";
            // Crear el cuerpo del modal con el contenido del changelog
            const modalBody = document.createElement("div");
            modalBody.innerHTML = bodyHtml;
            // Estilos para el cuerpo del modal
            const closeButton = document.createElement("button");
            closeButton.textContent = "Entendido";
            closeButton.style.display = "block";
            closeButton.style.margin = "20px auto 0 auto";
            closeButton.style.padding = "10px 20px";
            closeButton.style.backgroundColor = "#007bff";
            closeButton.style.color = "#fff";
            closeButton.style.border = "none";
            closeButton.style.borderRadius = "5px";
            closeButton.style.cursor = "pointer";
            //
            closeButton.addEventListener("click", () => {
                modal.remove();
                localStorage.setItem(LAST_SEEN_VERSION_KEY, currentScriptVersion); // Guarda la versión
            });
            // Añadir todo al modal y al body
            modal.appendChild(modalTitle);
            modal.appendChild(modalBody);
            modal.appendChild(closeButton);
            document.body.appendChild(modal);
        }
        else if (!versionData) {//
            // Si no hay datos de versión, no se hace nada
            localStorage.setItem(LAST_SEEN_VERSION_KEY, currentScriptVersion);
        }
    }//showChangelogOnUpdate

    // Función para mostrar el disclaimer de términos de uso
    function showDisclaimerPopup(callback) {
        // Guardia global para evitar múltiples popups
        if (window.__PLN_DISCLAIMER_SHOWN__) {
            console.log('[WME PLN] Disclaimer ya mostrado, ignorando llamada duplicada');
            return;
        }
        
        const DISCLAIMER_KEY = `wme_pln_disclaimer_v${VERSION}_accepted`;
        const hasAcceptedDisclaimer = localStorage.getItem(DISCLAIMER_KEY);
        
        // Si ya aceptó o rechazó anteriormente, no mostrar
        if (hasAcceptedDisclaimer !== null) {
            if (hasAcceptedDisclaimer === 'true') {
                if (typeof callback === 'function') callback(true);
            } else {
                if (typeof callback === 'function') callback(false);
            }
            return;
        }
        
        // Marcar que el popup está siendo mostrado
        window.__PLN_DISCLAIMER_SHOWN__ = true;

        // Crear el overlay oscuro
        const overlay = document.createElement("div");
        overlay.style.position = "fixed";
        overlay.style.top = "0";
        overlay.style.left = "0";
        overlay.style.width = "100%";
        overlay.style.height = "100%";
        overlay.style.backgroundColor = "rgba(0, 0, 0, 0.6)";
        overlay.style.zIndex = "29999";
        overlay.style.pointerEvents = "none"; // Permite que los clicks pasen al modal

        // Crear el modal
        const modal = document.createElement("div");
        modal.style.position = "fixed";
        modal.style.top = "50%";
        modal.style.left = "50%";
        modal.style.transform = "translate(-50%, -50%)";
        modal.style.backgroundColor = "#fff";
        modal.style.border = "2px solid #2196F3";
        modal.style.borderRadius = "8px";
        modal.style.boxShadow = "0 8px 24px rgba(0,0,0,0.4)";
        modal.style.padding = "25px";
        modal.style.fontFamily = "'Helvetica Neue', Helvetica, Arial, sans-serif";
        modal.style.zIndex = "30000";
        modal.style.width = "600px";
        modal.style.maxHeight = "85vh";
        modal.style.overflowY = "auto";
        modal.style.pointerEvents = "auto"; // El modal sí recibe clicks

        // Título
        const modalTitle = document.createElement("h2");
        modalTitle.innerHTML = "⚠️ Términos de Uso - WME Places Name Normalizer";
        modalTitle.style.marginTop = "0";
        modalTitle.style.marginBottom = "20px";
        modalTitle.style.textAlign = "center";
        modalTitle.style.color = "#2196F3";
        modalTitle.style.fontSize = "20px";

        // Contenido del disclaimer
        const modalBody = document.createElement("div");
        modalBody.style.lineHeight = "1.6";
        modalBody.style.fontSize = "14px";
        modalBody.style.color = "#333";
        modalBody.innerHTML = `
            <div style="font-family: 'Helvetica Neue', Helvetica, Arial, sans-serif; color: #333;">
                <!-- Section: Tool Usage -->
                <div style="margin-bottom: 15px;">
                    <h4 style="color: #2196F3; border-bottom: 1px solid #e0e0e0; padding-bottom: 5px; margin: 0 0 10px 0; font-size: 15px;">
                        🔧 Buen Uso de la Herramienta
                    </h4>
                    <p style="font-size: 13px; line-height: 1.4; margin: 0;">
                        Este script es una herramienta de <strong>asistencia</strong> para la normalización de nombres y categorías. <strong>Su uso no exime al editor de su responsabilidad de revisar detalladamente cada edición.</strong>
                    </p>
                </div>

                <!-- Section: Responsibilities -->
                <div style="margin-bottom: 15px;">
                    <h4 style="color: #2196F3; border-bottom: 1px solid #e0e0e0; padding-bottom: 5px; margin: 0 0 10px 0; font-size: 15px;">
                        📝 Responsabilidades del Editor
                    </h4>
                    <ul style="font-size: 13px; line-height: 1.4; padding-left: 0; margin: 0; list-style-type: none;">
                        <li style="margin-bottom: 4px; display: flex; align-items: flex-start;">
                            <span style="margin-right: 6px;">✅</span>
                            <span><strong>Revisar cuidadosamente</strong> sugerencias antes de aplicar</span>
                        </li>
                        <li style="margin-bottom: 4px; display: flex; align-items: flex-start;">
                            <span style="margin-right: 6px;">✅</span>
                            <span><strong>Verificar</strong> que las categorías sean correctas</span>
                        </li>
                        <li style="margin-bottom: 4px; display: flex; align-items: flex-start;">
                            <span style="margin-right: 6px;">✅</span>
                            <span><strong>Evaluar</strong> ediciones individualmente (no masivas)</span>
                        </li>
                         <li style="margin-bottom: 4px; display: flex; align-items: flex-start;">
                            <span style="margin-right: 6px;">✅</span>
                            <span><strong>Garantizar</strong> la calidad de la información</span>
                        </li>
                    </ul>
                </div>

                <!-- Section: Data Usage -->
                <div style="margin-bottom: 15px;">
                    <h4 style="color: #2196F3; border-bottom: 1px solid #e0e0e0; padding-bottom: 5px; margin: 0 0 10px 0; font-size: 15px;">
                        📊 Uso de Datos
                    </h4>
                     <ul style="font-size: 13px; line-height: 1.4; padding-left: 0; margin: 0; list-style-type: none;">
                        <li style="margin-bottom: 4px; display: flex; align-items: flex-start;">
                            <span style="margin-right: 6px;">📊</span>
                            <span>Registro de <strong>usuario y nivel</strong> para estadísticas</span>
                        </li>
                        <li style="margin-bottom: 4px; display: flex; align-items: flex-start;">
                            <span style="margin-right: 6px;">📊</span>
                            <span><strong>Control de uso</strong> para prevenir abuso</span>
                        </li>
                         <li style="margin-bottom: 4px; display: flex; align-items: flex-start;">
                            <span style="margin-right: 6px;">📊</span>
                            <span>Uso exclusivo para <strong>monitoreo y mejora</strong></span>
                        </li>
                    </ul>
                </div>

                <!-- Warning Block -->
                <div style="background-color: #FFF3E0; border-left: 4px solid #FF9800; padding: 10px 12px; font-size: 13px; margin-bottom: 15px; border-radius: 0 4px 4px 0;">
                    <div style="color: #E65100; font-weight: bold; margin-bottom: 4px;">⚠️ IMPORTANTE: Ediciones Masivas</div>
                    <div style="color: #5d4037;">
                        Las ediciones masivas no supervisadas están <strong>estrictamente prohibidas</strong>. El uso indebido puede resultar en:
                        <ul style="margin: 4px 0 0 0; padding-left: 20px; color: #5d4037;">
                            <li>Pérdida de acceso al script</li>
                            <li>Reporte a la comunidad</li>
                            <li>Sanciones en la cuenta de editor</li>
                        </ul>
                    </div>
                </div>

                <div style="text-align: center; font-size: 14px; font-weight: bold; color: #333; margin-top: 15px; padding-top: 10px; border-top: 1px solid #eee;">
                    ¿Aceptas estos términos de uso y te comprometes a utilizar la herramienta de manera responsable?
                </div>
            </div>
        `;

        // Botones
        const buttonContainer = document.createElement("div");
        buttonContainer.style.display = "flex";
        buttonContainer.style.justifyContent = "center";
        buttonContainer.style.gap = "15px";
        buttonContainer.style.marginTop = "25px";

        const acceptButton = document.createElement("button");
        acceptButton.setAttribute('type', 'button');
        acceptButton.setAttribute('data-disclaimer-btn', 'true');
        acceptButton.textContent = "✅ Aceptar y Continuar";
        acceptButton.style.padding = "12px 30px";
        acceptButton.style.fontSize = "14px";
        acceptButton.style.fontWeight = "bold";
        acceptButton.style.backgroundColor = "#4CAF50";
        acceptButton.style.color = "#fff";
        acceptButton.style.border = "none";
        acceptButton.style.borderRadius = "5px";
        acceptButton.style.cursor = "pointer";
        acceptButton.style.outline = "none";
        acceptButton.style.display = "inline-flex";
        acceptButton.style.alignItems = "center";
        acceptButton.style.justifyContent = "center";
        acceptButton.style.gap = "6px";
        acceptButton.style.lineHeight = "1";
        acceptButton.style.transition = "background-color 0.3s";

        acceptButton.onmouseover = () => acceptButton.style.backgroundColor = "#45a049";
        acceptButton.onmouseout = () => acceptButton.style.backgroundColor = "#4CAF50";

        const rejectButton = document.createElement("button");
        rejectButton.setAttribute('type', 'button');
        rejectButton.setAttribute('data-disclaimer-btn', 'true');
        rejectButton.textContent = "❌ Rechazar";
        rejectButton.style.padding = "12px 30px";
        rejectButton.style.fontSize = "14px";
        rejectButton.style.backgroundColor = "#9E9E9E";
        rejectButton.style.color = "#fff";
        rejectButton.style.border = "none";
        rejectButton.style.borderRadius = "5px";
        rejectButton.style.cursor = "pointer";
        rejectButton.style.outline = "none";
        rejectButton.style.display = "inline-flex";
        rejectButton.style.alignItems = "center";
        rejectButton.style.justifyContent = "center";
        rejectButton.style.gap = "6px";
        rejectButton.style.lineHeight = "1";
        rejectButton.style.transition = "background-color 0.3s";

        rejectButton.onmouseover = () => rejectButton.style.backgroundColor = "#757575";
        rejectButton.onmouseout = () => rejectButton.style.backgroundColor = "#9E9E9E";

        // Handlers para los botones con flag para evitar ejecución múltiple
        let disclaimerHandled = false;
        
        const handleAccept = (e) => {
            if (disclaimerHandled) return;
            disclaimerHandled = true;
            e.stopImmediatePropagation();
            e.stopPropagation();
            e.preventDefault();
            localStorage.setItem(DISCLAIMER_KEY, 'true');
            sendDisclaimerResponse(true);
            try { overlay.remove(); } catch (_) {}
            try { modal.remove(); } catch (_) {}
            plnToast('✅ ¡Gracias por aceptar los términos de uso! El script se cargará ahora...', 3000);
            if (typeof callback === 'function') callback(true);
        };
        
        const handleReject = (e) => {
            if (disclaimerHandled) return;
            disclaimerHandled = true;
            e.stopImmediatePropagation();
            e.stopPropagation();
            e.preventDefault();
            localStorage.setItem(DISCLAIMER_KEY, 'false');
            sendDisclaimerResponse(false);
            try { overlay.remove(); } catch (_) {}
            try { modal.remove(); } catch (_) {}
            plnToast('❌ El script no se ha habilitado porque no aceptaste los términos y condiciones.', 5000);
            if (typeof callback === 'function') callback(false);
        };
        
        // Solo usar click (el mouseup causaba ejecuciones duplicadas)
        acceptButton.addEventListener('click', handleAccept, true);
        rejectButton.addEventListener('click', handleReject, true);

        // Ensamblar el modal
        buttonContainer.appendChild(acceptButton);
        buttonContainer.appendChild(rejectButton);
        modal.appendChild(modalTitle);
        modal.appendChild(modalBody);
        modal.appendChild(buttonContainer);

        // Añadir al DOM
        document.body.appendChild(overlay);
        document.body.appendChild(modal);
    }//showDisclaimerPopup

    // Función para enviar la respuesta del disclaimer al servidor
    function sendDisclaimerResponse(accepted) {
        if (!currentGlobalUserInfo || !currentGlobalUserInfo.id) {
            console.warn('[WME PLN] No se pudo enviar respuesta del disclaimer: usuario no detectado');
            return;
        }

        const payload = {
            userId: currentGlobalUserInfo.id,
            userName: currentGlobalUserInfo.name,
            accepted: accepted,
            date: new Date().toISOString().split('T')[0]
        };

        makeRequest({
            method: 'POST',
            url: PLN_ACCESS_CONTROL.endpoint + '?action=recordDisclaimer',
            data: JSON.stringify(payload),
            headers: { 'Content-Type': 'text/plain;charset=utf-8' },  // Evita CORS preflight
            timeout: PLN_ACCESS_CONTROL.timeoutMs,
            onload: function (response) {
                try {
                    const result = JSON.parse(response.responseText);
                    if (result.success) {
                        console.log(`[WME PLN] Respuesta del disclaimer registrada: ${accepted ? 'Aceptado' : 'Rechazado'}`);
                    } else {
                        console.error('[WME PLN] Error al registrar respuesta del disclaimer:', result.error);
                    }
                } catch (e) {
                    console.error('[WME PLN] Error al procesar respuesta del disclaimer:', e);
                }
            },
            onerror: function (error) {
                console.error('[WME PLN] Error de red al enviar respuesta del disclaimer:', error);
            }
        });
    }//sendDisclaimerResponse

    //Permite inicializar el SDK de WME
    function tryInitializeSDK(finalCallback) {
        let attempts = 0;
        const maxAttempts = 20; // Reduced from 60 to 20
        const intervalTime = 500;
        let sdkAttemptInterval = null;

        // Function to clear interval safely
        function clearSDKInterval() {
            if (sdkAttemptInterval) {
                clearInterval(sdkAttemptInterval);
                sdkAttemptInterval = null;
            }
        }

        // Function to call callback safely
        function safeCallback() {
            clearSDKInterval();
            try {
                if (typeof finalCallback === 'function') {
                    finalCallback();
                }
            } catch (e) {
                console.error("[WME PLN] Error in SDK callback:", e);
            }
        }

        // Function to attempt SDK initialization
        function attempt() {
            // Prefer unsafeWindow.getWmeSdk in TM sandbox. Fallback to page getWmeSdk.
            const getWmeSdkFn = (typeof unsafeWindow !== 'undefined' && typeof unsafeWindow.getWmeSdk === 'function')
                ? unsafeWindow.getWmeSdk
                : (typeof getWmeSdk === 'function' ? getWmeSdk : null);

            if (getWmeSdkFn) {
                clearSDKInterval();
                try {
                    wmeSDK = getWmeSdkFn({
                        scriptId: 'WMEPlacesNameInspector',
                        scriptName: SCRIPT_NAME,
                    });
                    //console.log("[WME PLN] SDK initialized successfully");

                    // === Export stable globals for console and other scripts ===
                    try {
                        window.WME_PLN_SDK = wmeSDK;
                        if (typeof unsafeWindow !== 'undefined') unsafeWindow.WME_PLN_SDK = wmeSDK;
                        window.WME_PLN_getSDK = function () { return window.WME_PLN_SDK; };
                        window.WME_PLN_reinitSDK = function () { tryInitializeSDK(() => { }); };
                        //console.log("[WME PLN] SDK exported: window.WME_PLN_SDK, WME_PLN_getSDK(), WME_PLN_reinitSDK()");
                    } catch (e) {
                        console.warn("[WME PLN] Could not export SDK globals:", e);
                    }
                } catch (e) {
                    console.error("[WME PLN] Error initializing SDK:", e);
                    wmeSDK = null;
                }
                safeCallback();
                return;
            }

            attempts++;
            if (attempts >= maxAttempts) {
                console.warn(`[WME PLN] Could not find getWmeSdk() after ${maxAttempts} attempts.`);
                wmeSDK = null;
                safeCallback();
            }
        }


        sdkAttemptInterval = setInterval(attempt, intervalTime);
        attempt();


        if (!wmeSDK && typeof window.WME_PLN_SDK !== 'undefined' && window.WME_PLN_SDK) {
            try {
                wmeSDK = window.WME_PLN_SDK;
                window.WME_PLN_getSDK = function () { return window.WME_PLN_SDK; };
                window.WME_PLN_reinitSDK = function () { tryInitializeSDK(() => { }); };
                //console.log("[WME PLN] Reused existing window.WME_PLN_SDK");
                safeCallback();
            } catch (e) { /* noop */ }
        }

        // Safety timeout to ensure callback is called
        setTimeout(() => {
            if (sdkAttemptInterval) {
                console.warn("[WME PLN] Safety timeout for SDK initialization");
                safeCallback();
            }
        }, maxAttempts * intervalTime + 1000);
    }//tryInitializeSDK

    // Función para obtener la ciudad de un lugar usando WME API
    function getPlaceCity(venue) {
        try {
            // Intentar obtener la ciudad del venue
            if (venue && venue.getAddress) {
                const address = venue.getAddress();
                if (address && address.city && address.city.name) {
                    plnAddWorkCity(address.city.name);
                    return address.city.name;
                }
            }

            // Intentar obtener la ciudad desde el SDK
            if (wmeSDK && wmeSDK.DataModel && wmeSDK.DataModel.Venues) {
                const venueId = venue.getID();
                const venueSDK = wmeSDK.DataModel.Venues.getById({ venueId });
                if (venueSDK && venueSDK.address && venueSDK.address.city && venueSDK.address.city.name) {
                    plnAddWorkCity(venueSDK.address.city.name);
                    return venueSDK.address.city.name;
                }
            }

            return "Sin Ciudad";
        }
        catch (e) {
            console.warn("[WME PLN] Error al obtener ciudad del lugar:", e);
            return "Sin Ciudad";
        }
    }// getPlaceCity
    // Función segura para obtener un lugar por ID usando el SDK
    function getVenueByIdSafe(placeId) {
        try {
            const id = (typeof placeId === 'string' || typeof placeId === 'number') ? String(placeId) : null;
            if (!id) return null;
            if (wmeSDK && wmeSDK.DataModel && wmeSDK.DataModel.Venues && typeof wmeSDK.DataModel.Venues.getById === 'function') {
                return wmeSDK.DataModel.Venues.getById({ venueId: id }) || null;
            }
        } catch (_) { }
        return null;
    }// getVenueByIdSafe
    //-----------------------------------------------------------------------------------------------------------
    // 1) Con el WME SDK
    // Funciones de obtención de usuario
    async function getCurrentEditorViaSdk() {
        if (!wmeSDK) {
            return null;
        }
        if (!wmeSDK.DataModel || !wmeSDK.DataModel.User || typeof wmeSDK.DataModel.User.getCurrentUser !== 'function') {
            return null;
        }
        try {
            const user = await wmeSDK.DataModel.User.getCurrentUser();
            console.log('[WME PLN][DEBUG] SDK: user obtenido:', user);
            if (user && user.id && user.name) {
                const result = { id: user.id, name: user.name, privilege: user.rank || user.privilege };
                console.log('[WME PLN][DEBUG] SDK: retornando:', result);
                return result;
            } else {
                console.warn('[WME PLN][DEBUG] SDK: getCurrentUser() devolvió datos incompletos o null:', user);
                return null;
            }
        } catch (e) {
            console.error('[WME PLN][DEBUG] SDK ERROR al obtener usuario:', e);
            return null;
        }
    }// getCurrentEditorViaSdk

    // 2) Con WazeWrap
    function getCurrentEditorViaWazeWrap() {
        if (typeof WazeWrap === 'undefined') {
            return null;
        }
        if (!WazeWrap.Login || typeof WazeWrap.Login.getLoggedInUser !== 'function') {
            return null;
        }
        const wrapUser = WazeWrap.Login.getLoggedInUser();
        console.log('[WME PLN][DEBUG] WazeWrap: wrapUser obtenido:', wrapUser);
        if (wrapUser && wrapUser.userId && wrapUser.username) {
            const result = { id: wrapUser.userId, name: wrapUser.username, privilege: wrapUser.rank || wrapUser.privilege };
            console.log('[WME PLN][DEBUG] WazeWrap: retornando:', result);
            return result;
        }
        else {
            console.warn('[WME PLN][DEBUG] WazeWrap: datos incompletos');
            return null;
        }
    }// getCurrentEditorViaWazeWrap

    // 3) Fallback a la API nativa de WME
    function getCurrentEditorViaWmeInternal() {
        if (typeof W === 'undefined' || !W.loginManager) {
            return null;
        }

        // Check W.loginManager.user.attributes first (most reliable)
        if (W.loginManager.user && W.loginManager.user.attributes) {
            const attrs = W.loginManager.user.attributes;
            if (attrs.id) {
                const result = {
                    id: attrs.id,
                    name: attrs.userName || W.loginManager.userName || 'Unknown',
                    privilege: attrs.rank != null ? attrs.rank : (W.loginManager.userPrivilege || 0)
                };
                // console.log('[WME PLN][DEBUG] WME Internal: Found via user.attributes:', result);
                return result;
            }
        }

        if (W.loginManager.userId && W.loginManager.userName) {
            // Priorizar rank sobre userPrivilege
            const rank = (W.loginManager.user && W.loginManager.user.attributes && W.loginManager.user.attributes.rank) || W.loginManager.userPrivilege;
            const result = { id: W.loginManager.userId, name: W.loginManager.userName, privilege: rank };
            console.log('[WME PLN][DEBUG] WME Internal: Found via loginManager props:', result);
            return result;
        }
        else {
            console.warn('[WME PLN][DEBUG] WME Internal: W.loginManager sin userId/userName');
            return null;
        }
    }// getCurrentEditorViaWmeInternal

    // Función para obtener el nivel (rank) del editor actual
    function getEditorRank() {
        try {
            // DEBUG: Ver qué hay en W.loginManager.user
            if (typeof W !== 'undefined' && W.loginManager?.user) {
                // console.log('[WME PLN] 🔍 DEBUG - W.loginManager.user.attributes:', W.loginManager.user.attributes);
                if (W.loginManager.user.attributes) {
                    // console.log('[WME PLN] 🔍 DEBUG - Propiedades de attributes:', Object.keys(W.loginManager.user.attributes));
                    // console.log('[WME PLN] 🔍 DEBUG - attributes.rank:', W.loginManager.user.attributes.rank);
                    // console.log('[WME PLN] 🔍 DEBUG - attributes.userRank:', W.loginManager.user.attributes.userRank);
                }
            }

            // Intento 1: Desde W.loginManager.user.attributes.rank (NUEVO)
            if (typeof W !== 'undefined' && W.loginManager?.user?.attributes?.rank != null) {
                // console.log('[WME PLN] ✅ Rank obtenido desde W.loginManager.user.attributes.rank:', W.loginManager.user.attributes.rank);
                return Number(W.loginManager.user.attributes.rank);
            }

            // Intento 2: Desde el SDK usando State.getUser()
            if (typeof wmeSDK !== 'undefined' && wmeSDK?.State?.getUser) {
                const userSDK = wmeSDK.State.getUser();
                if (userSDK?.rank != null) {
                    console.log('[WME PLN] ✅ Rank obtenido desde wmeSDK.State.getUser():', userSDK.rank);
                    return Number(userSDK.rank);
                }
            }

            // Intento 3: Desde el SDK usando State.User.getUser()
            if (typeof wmeSDK !== 'undefined' && wmeSDK?.State?.User?.getUser) {
                const userSDK = wmeSDK.State.User.getUser();
                if (userSDK?.rank != null) {
                    console.log('[WME PLN] ✅ Rank obtenido desde wmeSDK.State.User.getUser():', userSDK.rank);
                    return Number(userSDK.rank);
                }
            }

            // Intento 4: Desde W.loginManager.user.rank (directo)
            if (typeof W !== 'undefined' && W.loginManager?.user?.rank != null) {
                console.log('[WME PLN] ✅ Rank obtenido desde W.loginManager.user.rank:', W.loginManager.user.rank);
                return Number(W.loginManager.user.rank);
            }

            // Intento 5: Desde W.loginManager.getUser()
            if (typeof W !== 'undefined' && W.loginManager?.getUser) {
                const user = W.loginManager.getUser();
                if (user?.rank != null) {
                    console.log('[WME PLN] ✅ Rank obtenido desde W.loginManager.getUser().rank:', user.rank);
                    return Number(user.rank);
                }
                if (user?.attributes?.rank != null) {
                    console.log('[WME PLN] ✅ Rank obtenido desde W.loginManager.getUser().attributes.rank:', user.attributes.rank);
                    return Number(user.attributes.rank);
                }
            }

            // Intento 6: Desde W.model.users.getObjectById (user ID actual)
            if (typeof W !== 'undefined' && W.model?.users?.getObjectById && W.loginManager?.user?.attributes?.id) {
                const userId = W.loginManager.user.attributes.id;
                const userObj = W.model.users.getObjectById(userId);
                if (userObj?.attributes?.rank != null) {
                    console.log('[WME PLN] ✅ Rank obtenido desde W.model.users.getObjectById().attributes.rank:', userObj.attributes.rank);
                    return Number(userObj.attributes.rank);
                }
            }

            // Intento 7: Desde currentGlobalUserInfo si ya lo tenemos
            if (currentGlobalUserInfo?.rank != null) {
                console.log('[WME PLN] ✅ Rank obtenido desde currentGlobalUserInfo:', currentGlobalUserInfo.rank);
                return Number(currentGlobalUserInfo.rank);
            }

            console.warn('[WME PLN] ⚠️ No se pudo obtener rank del editor desde ninguna fuente disponible');
        } catch (e) {
            console.error('[WME PLN] ❌ Error obteniendo rank del editor:', e);
        }
        return 0; // Por defecto, rank 0 si no se puede obtener
    }// getEditorRank

    //-----------------------------------------------------------------------------------------------------------
    // Función para crear dinámicamente el checkbox de Update Requests (después de detectar rank)
    function createUpdateRequestsCheckbox() {
        try {
            // Usar la referencia global al contenedor
            if (!recommendCategoriesWrapperGlobal) {
                console.warn('[WME PLN] ⚠️ recommendCategoriesWrapperGlobal no está disponible aún');
                return;
            }

            // Obtener rank del usuario
            const editorRank = getEditorRank();
            // console.log('[WME PLN] 🔍 Editor Rank detectado para checkbox URs:', editorRank);

            // Verificar si el checkbox ya existe
            if (document.getElementById('chk-update-requests-only')) {
                console.log('[WME PLN] Checkbox de URs ya existe, no se creará duplicado');
                return;
            }

            // Acceso restringido: Solo editores nivel 4, 5 y 6
            if (editorRank === 4 || editorRank === 5 || editorRank === 6) {
                // console.log(`[WME PLN] ✅ Creando checkbox de URs pendientes (Rank ${editorRank} autorizado)`);

                const updateRequestsContainer = document.createElement('div');
                updateRequestsContainer.style.marginTop = '8px';

                const updateRequestsCheckboxRow = document.createElement("div");
                updateRequestsCheckboxRow.style.display = "flex";
                updateRequestsCheckboxRow.style.alignItems = "center";

                const updateRequestsCheckbox = document.createElement("input");
                updateRequestsCheckbox.type = "checkbox";
                updateRequestsCheckbox.id = "chk-update-requests-only";
                updateRequestsCheckbox.style.marginRight = "8px";

                // Restaurar estado guardado
                const savedUpdateRequestsState = localStorage.getItem("wme_pln_update_requests_only");
                updateRequestsCheckbox.checked = (savedUpdateRequestsState === "true");
                filterUpdateRequestsOnly = updateRequestsCheckbox.checked; // Sincronizar con variable global

                updateRequestsCheckboxRow.appendChild(updateRequestsCheckbox);

                const updateRequestsLabel = document.createElement("label");
                updateRequestsLabel.htmlFor = "chk-update-requests-only";
                updateRequestsLabel.style.fontSize = "16px";
                updateRequestsLabel.style.cursor = "pointer";
                updateRequestsLabel.style.fontWeight = "bold";
                updateRequestsLabel.style.color = "#d32f2f"; // Rojo para destacar
                updateRequestsLabel.innerHTML = `🔔 Solo lugares con URs pendientes`;
                updateRequestsLabel.title = `Solo disponible para editores nivel 4, 5 y 6. Filtra solo lugares con solicitudes de actualización pendientes de aprobación.`;

                updateRequestsCheckboxRow.appendChild(updateRequestsLabel);
                updateRequestsContainer.appendChild(updateRequestsCheckboxRow);
                recommendCategoriesWrapperGlobal.appendChild(updateRequestsContainer);

                // Listener para guardar estado y actualizar variable global
                updateRequestsCheckbox.addEventListener("change", () => {
                    filterUpdateRequestsOnly = updateRequestsCheckbox.checked;
                    localStorage.setItem("wme_pln_update_requests_only", updateRequestsCheckbox.checked ? "true" : "false");
                    console.log('[WME PLN] ✅ Update requests filter activado:', updateRequestsCheckbox.checked);
                });

                console.log('[WME PLN] ✅ Checkbox de URs creado exitosamente');

            } else {
                console.log(`[WME PLN] ❌ Checkbox de URs NO creado. Rank actual: ${editorRank} (Se requiere nivel 4, 5 o 6)`);
                // Asegurar que el filtro esté desactivado si no tiene permisos
                filterUpdateRequestsOnly = false;
            }
        } catch (error) {
            console.error('[WME PLN] ❌ Error al crear checkbox de URs:', error);
        }
    }// createUpdateRequestsCheckbox

    //-----------------------------------------------------------------------------------------------------------
    // Esperar a que la API principal de Waze esté completamente cargada
    async function waitForWazeAPI(callbackPrincipalDelScript) {
        let wAttempts = 0;
        const wMaxAttempts = 40;
        const wInterval = setInterval(async () => {
            wAttempts++;
            if (typeof W !== 'undefined' && W.map && W.loginManager && W.model && W.model.venues && W.userscripts && typeof W.userscripts.registerSidebarTab === 'function') {
                clearInterval(wInterval);
                if (!dynamicCategoriesLoaded) // solo carga las categorías por defecto si no se han cargado aún
                {
                    // INICIO OPTIMIZACIÓN: Carga en background (sin await) para no bloquear la UI
                    loadDynamicCategoriesFromSheet(FORCE_SHEET_RELOAD)
                        .then(() => {
                            dynamicCategoriesLoaded = true; // <-- Marcar como cargado
                            // console.log('[WME PLN] Categorías dinámicas cargadas en segundo plano.');
                        })
                        .catch(error => {
                            console.error("[WME PLN] No se pudieron cargar las categorías dinámicas (background):", error);
                        });
                    // FIN OPTIMIZACIÓN
                }
                // : Esperar a que tryInitializeSDK se complete
                tryInitializeSDK(() => { //
                    // === [PLN] Daily Usage Log: resolve user and log once per day ===
                    // Resolve user info and log daily usage (once per day)
                    plnResolveCurrentUser().then(() => {
                        /*  // Guard: only try to log after we have some user id
                          if (currentGlobalUserInfo && currentGlobalUserInfo.id != null) {
                            plnLogUsageOncePerDay();
                          }*/
                    });
                    // Una vez que el SDK ha intentado inicializarse (exitosamente o no),
                    // y las APIs de WazeWrap y W.loginManager deberían estar cargadas,
                    // llamamos al callback principal del script (createSidebarTab).
                    callbackPrincipalDelScript();
                    // [PLN] Snapshot SOLO-usuario: pares previos cuyo source != 'sheet'
                    const __plnPreUserOnly = {};
                    {
                        const _m = window.replacementWords || {};
                        const _s = window.replacementSources || {};
                        for (const k in _m) {
                            if (_s && _s[k] !== 'sheet') __plnPreUserOnly[k] = _m[k];
                        }
                    }
                    // [PLN] Cargar reemplazos por defecto tras crear la UI y re-render según modo
                    loadReplacementsFromSheet(true).finally(() => {
                        // [PLN] Protector post-loader: reinsertar pares del usuario que el loader haya quitado
                        try {
                            const curMap = window.replacementWords || {};
                            const curSrc = window.replacementSources || {};

                            // Reinsertar SOLO entradas del usuario que falten tras la carga de la hoja
                            for (const k in __plnPreUserOnly) {
                                if (!(k in curMap)) {
                                    curMap[k] = __plnPreUserOnly[k];
                                    if (curSrc && typeof curSrc === 'object') {
                                        curSrc[k] = (curSrc[k] && curSrc[k] !== 'sheet') ? curSrc[k] : 'user';
                                    }
                                }
                            }
                            window.replacementWords = curMap;
                            window.replacementSources = curSrc;

                            // Canoniza (reencamina TO==FROM de hoja), sin borrar locales válidos
                            if (typeof plnCanonicalizeReplacementsBySheet === 'function') plnCanonicalizeReplacementsBySheet();
                        } catch (_) { /* noop */ }

                        // Re-render según modo (evita carreras con RAF)
                        requestAnimationFrame(() => {
                            const _el = document.getElementById("replacementsListUL") || document.querySelector("#replacementsContainer ul");
                            const _sel = document.getElementById("replacementModeSelector");
                            if (_el) {
                                if (_sel && _sel.value === "swapStart" && typeof renderSwapList === "function") {
                                    renderSwapList(_el);
                                }
                                else {
                                    renderReplacementsList(_el);
                                }
                            }
                        });
                    });
                }); //
            }
            else if (wAttempts >= wMaxAttempts) // Si no se ha cargado la API de Waze después de 20 segundos
            {
                clearInterval(wInterval);
                console.error("[WME PLN] Waze API no se cargó completamente después de múltiples intentos."); //
                callbackPrincipalDelScript(); // Llama al callback de todas formas si se agotaron los intentos
            }
        }, 500);
    }//waitforWazeAPI

    //+++++++++Funciones de Georeferenciación y Distancia++++++++++++++
    // Función para calcular la distancia en metros entre dos puntos geográficos (latitud, longitud)
    function calculateDistance(lat1, lon1, lat2, lon2) {
        const earthRadiusMeters = 6371e3; // Radio de la Tierra en metros
        // Convertir latitudes y diferencias de longitudes de grados a radianes
        const lat1Rad = lat1 * Math.PI / 180;// Convertir latitud 1 a radianes
        const lat2Rad = lat2 * Math.PI / 180;// Convertir latitud 2 a radianes
        const deltaLatRad = (lat2 - lat1) * Math.PI / 180;// Convertir diferencia de latitudes a radianes
        const deltaLonRad = (lon2 - lon1) * Math.PI / 180;// Convertir diferencia de longitudes a radianes

        // Fórmula de Haversine para calcular la distancia entre dos puntos en una esfera
        // a = sin²(Δlat/2) + cos(lat1) * cos(lat2) * sin²(Δlon/2)
        const a = Math.sin(deltaLatRad / 2) * Math.sin(deltaLatRad / 2) + Math.cos(lat1Rad) * Math.cos(lat2Rad) * Math.sin(deltaLonRad / 2) * Math.sin(deltaLonRad / 2);//
        // c = 2 * atan2(√a, √(1−a))
        const c = 2 * Math.atan2(Math.sqrt(a), Math.sqrt(1 - a));
        // Calcular la distancia final en metros
        const distanceMeters = earthRadiusMeters * c;
        return distanceMeters;
    }//calculateDistance

    //Función para obtener coordenadas de un lugar
    // --- Reemplaza esta función completa ---

    function getPlaceCoordinates(venueOldModel, venueSDK) {
        let lat = null;
        let lon = null;
        const placeId = venueOldModel ? venueOldModel.getID() : (venueSDK ? venueSDK.id : 'N/A');

        // --- CORRECCIÓN ---
        // PRIORIDAD 1: Usar el método recomendado getOLGeometry() del modelo antiguo, es el más estable.
        if (venueOldModel && typeof venueOldModel.getOLGeometry === 'function') {
            try {
                const geometry = venueOldModel.getOLGeometry();
                if (geometry && typeof geometry.getCentroid === 'function') {
                    const centroid = geometry.getCentroid();
                    if (centroid && typeof centroid.x === 'number' && typeof centroid.y === 'number') {
                        // La geometría de OpenLayers (OL) está en proyección Mercator (EPSG:3857)
                        // Necesitamos transformarla a coordenadas geográficas WGS84 (EPSG:4326)
                        if (typeof OpenLayers !== 'undefined' && OpenLayers.Projection) {
                            const mercatorPoint = new OpenLayers.Geometry.Point(centroid.x, centroid.y);
                            const wgs84Point = mercatorPoint.transform(
                                new OpenLayers.Projection("EPSG:3857"),
                                new OpenLayers.Projection("EPSG:4326")
                            );

                            lat = wgs84Point.y;
                            lon = wgs84Point.x;

                            // Validar que las coordenadas resultantes sean válidas
                            if (typeof lat === 'number' && typeof lon === 'number' && Math.abs(lat) <= 90 && Math.abs(lon) <= 180) {
                                return { lat, lon };
                            }
                        }
                    }
                }
            } catch (e) {
                console.error(`[WME PLN] Error obteniendo coordenadas con getOLGeometry() para ID ${placeId}:`, e);
            }
        }

        // PRIORIDAD 2: Fallback al objeto del SDK si el método anterior falló.
        // Esto es menos ideal porque .geometry está obsoleto, pero sirve como respaldo.
        if (venueSDK && venueSDK.geometry && Array.isArray(venueSDK.geometry.coordinates)) {
            lon = venueSDK.geometry.coordinates[0];
            lat = venueSDK.geometry.coordinates[1];

            if (typeof lat === 'number' && typeof lon === 'number' && Math.abs(lat) <= 90 && Math.abs(lon) <= 180) {
                return { lat, lon };
            }
        }

        // Si todo falló, retornar nulls
        console.warn(`[WME PLN] No se pudieron obtener coordenadas válidas para el ID ${placeId}.`);
        return { lat: null, lon: null };
    }//getPlaceCoordinates

    // Nueva función robusta para agregar datos para verificación de duplicados
    function addPlaceDataForDuplicateCheck(venue, venueSDK, normalizedName) {
        // Usa una variable global para almacenar los datos de lugares para comparar duplicados
        if (typeof window.duplicatePlacesData === "undefined") window.duplicatePlacesData = [];
        const duplicatePlacesData = window.duplicatePlacesData;
        let geometry = null;
        if (venueSDK && venueSDK.geometry && venueSDK.geometry.coordinates) {
            const [lon, lat] = venueSDK.geometry.coordinates;
            duplicatePlacesData.push({ name: normalizedName, lat, lon, venueId: venueSDK.id });
            return;
        }
        else if (venue && typeof venue.getOLGeometry === 'function') {
            geometry = venue.getOLGeometry();
            if (geometry) {
                const lonLat = geometry.getCoordinates();
                const lon = lonLat[0];
                const lat = lonLat[1];
                duplicatePlacesData.push({ name: normalizedName, lat, lon, venueId: venue.getID() });
                return;
            }
        }
        console.warn("[WME_PLN][WARNING] No se pudo obtener geometría válida para el lugar:", venue, venueSDK);
    }

    // Función para detectar nombres duplicados cercanos y generar alertas
    function detectAndAlertDuplicateNames(allScannedPlacesData) {
        const DISTANCE_THRESHOLD_METERS = 50; // Umbral de distancia para considerar "cerca" (en metros)
        const duplicatesGroupedForAlert = new Map(); // Almacenará {normalizedName: [{places}, {places}]}

        // Paso 1: Agrupar por nombre NORMALIZADO y encontrar duplicados cercanos
        allScannedPlacesData.forEach(p1 => {
            if (p1.lat === null || p1.lon === null) return; // Saltar si no tiene coordenadas

            // Buscar otros lugares con el mismo nombre normalizado
            const nearbyMatches = allScannedPlacesData.filter(p2 => {
                if (p2.id === p1.id || p2.lat === null || p2.lon === null) {
                    return false;
                }
                
                // ✅ NUEVA LÓGICA: Si ambos no tienen nombre o tienen el mismo nombre vacío
                const p1HasName = p1.normalized && p1.normalized.trim() !== '';
                const p2HasName = p2.normalized && p2.normalized.trim() !== '';
                
                // Si tienen nombres diferentes, no son duplicados
                if (p1HasName && p2HasName && p1.normalized !== p2.normalized) {
                    return false;
                }
                
                const distance = calculateDistance(p1.lat, p1.lon, p2.lat, p2.lon);
                
                // Si ambos tienen nombre, usar la lógica normal de distancia
                if (p1HasName && p2HasName) {
                    return distance <= DISTANCE_THRESHOLD_METERS;
                }
                
                // ✅ CASO ESPECIAL: Si uno o ambos no tienen nombre
                // Solo considerar duplicados si están MUY cerca (superpuestos)
                // y son del mismo tipo/tamaño aproximado
                if (!p1HasName || !p2HasName) {
                    // Deben estar prácticamente encima (< 5 metros)
                    if (distance > 5) return false;
                    
                    // Verificar que sean del mismo tipo de geometría
                    const p1IsArea = p1.isArea === true;
                    const p2IsArea = p2.isArea === true;
                    
                    // Deben ser ambos áreas o ambos puntos
                    if (p1IsArea !== p2IsArea) return false;
                    
                    // Si son áreas, verificar que sean de tamaño similar
                    // (esto requeriría calcular área, para simplificar asumimos que sí)
                    
                    return true; // Son duplicados sospechosos (mismo lugar, sin nombre, superpuestos)
                }
                
                return false;
            });

            if (nearbyMatches.length > 0) {
                // Si encontramos duplicados cercanos para p1, agruparlos
                const groupKey = (p1.normalized && p1.normalized.trim() !== '') 
                    ? p1.normalized.toLowerCase() 
                    : `__empty_${p1.id}`; // Clave única para lugares sin nombre
                
                if (!duplicatesGroupedForAlert.has(groupKey)) {
                    duplicatesGroupedForAlert.set(groupKey, new Set());
                }
                duplicatesGroupedForAlert.get(groupKey).add(p1); // Añadir p1
                nearbyMatches.forEach(p => duplicatesGroupedForAlert.get(groupKey).add(p)); // Añadir todos sus duplicados
            }
        });
        // Paso 2: Generar el mensaje de alerta final
        if (duplicatesGroupedForAlert.size > 0) {
            let totalNearbyDuplicateGroups = 0; // Para contar la cantidad de "nombres" con duplicados
            const duplicateEntriesHtml = []; // Para almacenar las líneas HTML de la alerta formateadas

            duplicatesGroupedForAlert.forEach((placesSet, normalizedName) => {
                const uniquePlacesInGroup = Array.from(placesSet); // Convertir Set a Array
                if (uniquePlacesInGroup.length > 1) { // Solo si realmente hay más de un lugar en el grupo
                    totalNearbyDuplicateGroups++;


                    // Obtener los números de línea para cada lugar en este grupo
                    const lineNumbers = uniquePlacesInGroup.map(p => {
                        const originalPlaceInInconsistents = allScannedPlacesData.find(item => item.id === p.id);
                        return originalPlaceInInconsistents ? (allScannedPlacesData.indexOf(originalPlaceInInconsistents) + 1) : 'N/A';
                    }).filter(num => num !== 'N/A').sort((a, b) => a - b); // Asegurarse que son números y ordenarlos

                    // Marcar los lugares en `allScannedPlacesData` y guardar sus compañeros
                    uniquePlacesInGroup.forEach(p => {
                        const originalPlaceInInconsistents = allScannedPlacesData.find(item => item.id === p.id);
                        if (originalPlaceInInconsistents) {
                            originalPlaceInInconsistents.isDuplicate = true;

                            // ✨ Guardar información de duplicados compañeros (28-nov-2024)
                            originalPlaceInInconsistents.duplicatePartners = uniquePlacesInGroup
                                .filter(partner => partner.id !== p.id)
                                .map(partner => {
                                    const partnerInList = allScannedPlacesData.find(item => item.id === partner.id);
                                    const line = partnerInList ? (allScannedPlacesData.indexOf(partnerInList) + 1) : null;
                                    return {
                                        id: partner.id,
                                        originalName: partner.original,
                                        line: line
                                    };
                                });
                        }
                    });
                    // Construir la línea para el modal
                    duplicateEntriesHtml.push(`
                        <div style="margin-bottom: 5px; font-size: 15px; text-align: left;">
                            <b>${totalNearbyDuplicateGroups}.</b> Nombre: <b>${normalizedName}</b><br>
                            <span style="font-weight: bold; color: #007bff;">Registros: [${lineNumbers.join("],[")}]</span>
                        </div>
                    `);
                }
            });
            // Solo mostrar la alerta si realmente hay grupos de más de 1 duplicado cercano
            if (duplicateEntriesHtml.length > 0) {
                // Crear el modal
                const modal = document.createElement("div");
                modal.style.position = "fixed";
                modal.style.top = "50%";
                modal.style.left = "50%";
                modal.style.transform = "translate(-50%, -50%)";
                modal.style.background = "#fff";
                modal.style.border = "1px solid #aad";
                modal.style.padding = "28px 32px 20px 32px";
                modal.style.zIndex = "20000"; // Z-INDEX ALTO para asegurar que esté encima
                modal.style.boxShadow = "0 4px 24px rgba(0,0,0,0.18)";
                modal.style.fontFamily = "sans-serif";
                modal.style.borderRadius = "10px";
                modal.style.textAlign = "center";
                modal.style.minWidth = "400px";
                modal.style.maxWidth = "600px";
                modal.style.maxHeight = "80vh"; // Para scroll si hay muchos duplicados
                modal.style.overflowY = "auto"; // Para scroll si hay muchos duplicados

                // Ícono visual
                const iconElement = document.createElement("div");
                iconElement.innerHTML = "⚠️"; // Signo de advertencia
                iconElement.style.fontSize = "38px";
                iconElement.style.marginBottom = "10px";
                modal.appendChild(iconElement);

                // Mensaje principal
                const messageTitle = document.createElement("div");
                messageTitle.innerHTML = `<b>¡Atención! Se encontraron ${duplicateEntriesHtml.length} nombres duplicados.</b>`;
                messageTitle.style.fontSize = "20px";
                messageTitle.style.marginBottom = "8px";
                modal.appendChild(messageTitle);

                const messageExplanation = document.createElement("div");
                messageExplanation.textContent = `Los siguientes grupos de lugares se encuentran a menos de ${DISTANCE_THRESHOLD_METERS}m uno del otro. El algoritmo asume que son el mismo lugar, por favor revisa los registros indicados en el panel flotante:`;
                messageExplanation.style.fontSize = "15px";
                messageExplanation.style.color = "#555";
                messageExplanation.style.marginBottom = "18px";
                messageExplanation.style.textAlign = "left"; // Alinear texto explicativo a la izquierda
                modal.appendChild(messageExplanation);

                // Lista de duplicados
                const duplicatesListDiv = document.createElement("div");
                duplicatesListDiv.style.textAlign = "left"; // Alinear la lista a la izquierda
                duplicatesListDiv.style.paddingLeft = "10px"; // Pequeño padding para los números
                duplicatesListDiv.innerHTML = duplicateEntriesHtml.join('');
                modal.appendChild(duplicatesListDiv);

                // Botón OK
                const buttonWrapper = document.createElement("div");
                buttonWrapper.style.display = "flex";
                buttonWrapper.style.justifyContent = "center";
                buttonWrapper.style.gap = "18px";
                buttonWrapper.style.marginTop = "20px"; // Espacio superior

                const okBtn = document.createElement("button");
                okBtn.textContent = "OK";
                okBtn.style.padding = "7px 18px";
                okBtn.style.background = "#007bff";
                okBtn.style.color = "#fff";
                okBtn.style.border = "none";
                okBtn.style.borderRadius = "4px";
                okBtn.style.cursor = "pointer";
                okBtn.style.fontWeight = "bold";

                okBtn.addEventListener("click", () => modal.remove()); // Cierra el modal

                buttonWrapper.appendChild(okBtn);
                modal.appendChild(buttonWrapper);

                document.body.appendChild(modal); // Añadir el modal al body
            }
        }
    }
    //+++++++++FIN Funciones de Georeferenciación y Distancia++++++++++++++

    // === Compat: parche para Venue.getAddress() (evita firma antigua sin W.model) ===
    // Motivo: "Venue.getAddress() must be called with a W.model as a first argument" (definición antigua deprecada)
    // Estrategia: localizar el prototipo de Venue y envolver getAddress para que, si no recibe modelo, use W.model
    (function patchVenueGetAddressDeprecated() {
        let tries = 0;
        const maxTries = 12;
        const tick = setInterval(() => {
            tries++;
            try {
                if (typeof W === 'undefined' || !W.model || !W.model.venues) return; // esperar a W

                let sampleVenue = null;
                // 1) getObjectArray si existe
                if (W.model.venues && typeof W.model.venues.getObjectArray === 'function') {
                    const arr = W.model.venues.getObjectArray();
                    if (arr && arr.length) sampleVenue = arr[0];
                }
                // 2) repo.objects como fallback
                if (!sampleVenue && W.model.venues && W.model.venues.objects) {
                    const ids = Object.keys(W.model.venues.objects);
                    if (ids.length) sampleVenue = W.model.venues.objects[ids[0]];
                }
                if (!sampleVenue || typeof sampleVenue.getAddress !== 'function') {
                    if (tries >= maxTries) clearInterval(tick);
                    return;
                }

                const proto = Object.getPrototypeOf(sampleVenue);
                if (!proto || typeof proto.getAddress !== 'function' || proto.getAddress.__pln_patched) {
                    if (tries >= maxTries) clearInterval(tick);
                    return;
                }

                const originalGetAddress = proto.getAddress;
                function patchedGetAddress(modelArg) {
                    // Si no se pasa modelo, usar W.model para mantener compatibilidad
                    if (arguments.length === 0 && typeof W !== 'undefined' && W.model) {
                        try { return originalGetAddress.call(this, W.model); } catch (e) { /* cae abajo */ }
                    }
                    return originalGetAddress.apply(this, arguments);
                }
                patchedGetAddress.__pln_patched = true;
                proto.getAddress = patchedGetAddress;
                //console.log('[WME PLN] Compat patch aplicado: Venue.getAddress() acepta llamada sin argumento y usa W.model.');
                clearInterval(tick);
            } catch (e) {
                if (tries >= maxTries) clearInterval(tick);
            }
        }, 500);
    })();

    //**************************************************************************
    //Nombre: plnExtractAddressIds
    //Fecha modificación: 2025-08-10
    //Descripción: SDK‑only. Obtiene countryID y stateID desde sdkVenue.address,
    //             incluso cuando Street/City están vacíos.
    //**************************************************************************
    function plnExtractAddressIds(venueId, sdkVenue) {
        plnLog('extractIds:start', { venueId, hasSdkVenue: !!sdkVenue });
        const out = { countryID: null, stateID: null, streetName: '', houseNumber: '' };
        if (sdkVenue && sdkVenue.address) {
            const addr = sdkVenue.address;
            out.countryID = addr?.country?.id ?? addr?.countryID ?? addr?.countryId ?? null;
            out.stateID = addr?.state?.id ?? addr?.stateID ?? addr?.stateId ?? null;
            out.streetName = addr?.street?.name ?? addr?.streetName ?? '';
            out.houseNumber = addr?.houseNumber ?? '';
        }
        plnLog('extractIds:fromSDK', out);
        return out;
    }

    //**************************************************************************
    //Nombre: plnResolveIdsFromCity
    //Fecha modificación: 2025-08-10
    //Descripción: SDK-only. A partir de cityId intenta obtener stateID y countryID
    //             usando los repositorios del SDK (Cities → States → Countries).
    //**************************************************************************
    function plnResolveIdsFromCity(cityId) {
        const out = { countryID: null, stateID: null };
        try {
            if (!wmeSDK || !wmeSDK.DataModel) return out;
            const cityIdNum = Number(cityId);

            let city = null;
            try {
                if (wmeSDK.DataModel.Cities?.getById) {
                    city = wmeSDK.DataModel.Cities.getById({ cityId: cityIdNum }); // <-- number
                }
            } catch (_) { }
            plnLog('resolveFromCity:city', { requested: cityIdNum, found: !!city });
            if (!city) return out;

            let stateId = city.state?.id ?? city.stateID ?? city.stateId ?? city.attributes?.state?.attributes?.id ?? city.attributes?.state?.id ?? null;
            let countryId = city.country?.id ?? city.countryID ?? city.countryId ?? city.attributes?.country?.attributes?.id ?? city.attributes?.country?.id ?? null;

            if (!countryId && stateId && wmeSDK.DataModel.States?.getById) {
                try {
                    const state = wmeSDK.DataModel.States.getById({ stateId: Number(stateId) }); // <-- number
                    countryId = state?.country?.id ?? state?.countryID ?? state?.countryId ?? null;
                } catch (_) { }
            }

            if (stateId) out.stateID = Number(stateId);
            if (countryId) out.countryID = Number(countryId);
            plnLog('resolveFromCity:result', out);
        } catch (e) {
            plnErr('resolveFromCity:error', e);
        }
        return out;
    }//plnResolveIdsFromCity

    //**************************************************************************
    //Nombre: plnGetBaseVenueId
    //Descripción: Devuelve el ID base (antes del primer punto) como string
    //**************************************************************************
    function plnGetBaseVenueId(id) {
        return String(id).split('.')[0];
    }

    //**************************************************************************
    //Nombre: plnGetEmptyStreetIdForCity
    //Descripción: Obtiene el streetId "vacío" (No street) para una ciudad dada usando solo el SDK
    //             Intenta varias rutas del repositorio y hace pequeños reintentos síncronos.
    //**************************************************************************
    function plnGetEmptyStreetIdForCity(cityId) {
        const cidNum = Number(cityId);

        try {
            if (wmeSDK?.DataModel?.Streets?.getStreet) {
                const st = wmeSDK.DataModel.Streets.getStreet({ cityId: cidNum, streetName: '' }); // <-- number
                if (st && st.id != null) { plnLog('streets:emptyFound', { cityId: cidNum, streetId: Number(st.id) }); return Number(st.id); }
            }
        } catch (_) { }

        try {
            const all = (wmeSDK?.DataModel?.Streets?.getAll?.() || []);
            const found = all.find(s => Number(s?.city?.id) === cidNum && (s?.isEmpty || s?.name === '' || s?.streetName === ''));
            if (found) { plnLog('streets:emptyFound', { cityId: cidNum, streetId: Number(found.id) }); return Number(found.id); }
        } catch (_) { }

        try {
            for (let i = 0; i < 8; i++) {
                const all = (wmeSDK?.DataModel?.Streets?.getAll?.() || []);
                const found = all.find(s => Number(s?.city?.id) === cidNum && (s?.isEmpty || s?.name === '' || s?.streetName === ''));
                if (found) { plnLog('streets:emptyFound', { cityId: cidNum, streetId: Number(found.id) }); return Number(found.id); }
            }
        } catch (_) { }

        return null;
    }//plnGetEmptyStreetIdForCity

    /**************************************************************************
     * Nombre: plnGetVenueCityIdNow
     * Descripción: Lee (sincrónicamente) el cityID actual del venue vía SDK.
     **************************************************************************/
    function plnGetVenueCityIdNow(venueIdStr) {
        try {
            const v = wmeSDK?.DataModel?.Venues?.getById?.({ venueId: String(venueIdStr) });
            const cid = v?.address?.city?.id ?? v?.address?.cityID ?? v?.address?.cityId ?? null;
            return (cid != null) ? Number(cid) : null;
        } catch (_) { return null; }
    }

    /**************************************************************************
     * Nombre: plnWaitVenueCity
     * Descripción: Espera (polling corto) a que el venue muestre el cityID esperado.
     **************************************************************************/
    function plnWaitVenueCity(venueIdStr, expectedCityId, timeoutMs = 1500) {
        return new Promise(resolve => {
            const start = Date.now();
            const target = Number(expectedCityId);
            const tick = setInterval(() => {
                const cid = plnGetVenueCityIdNow(venueIdStr);
                if (cid === target) { clearInterval(tick); return resolve(true); }
                if (Date.now() - start > timeoutMs) { clearInterval(tick); return resolve(false); }
            }, 120);
        });
    }

    /**************************************************************************
     * Nombre: plnFindBridgeCityIdInState
     * Descripción: Busca una ciudad del mismo estado que tenga "No street" (vacía)
     **************************************************************************/
    function plnFindBridgeCityIdInState(stateId) {
        try {
            const all = (wmeSDK?.DataModel?.Streets?.getAll?.() || []);
            const match = all.find(s =>
                (s?.isEmpty || s?.name === '' || s?.streetName === '') &&
                Number(s?.city?.state?.id ?? s?.city?.stateID ?? s?.city?.stateId) === Number(stateId)
            );
            return match?.city?.id != null ? Number(match.city.id) : null;
        } catch (_) { return null; }
    }

    /**************************************************************************
     * Nombre: plnApplyCityOnce
     * Descripción: Aplica una ciudad UNA vez:
     *   - Si existe street vacío en esa ciudad → usa streetId
     *   - Si no, usa country/state/city + emptyStreet:true
     * Devuelve: 'streetId' o { type:'ids', ids } o null si no pudo construir args.
     **************************************************************************/
    function plnApplyCityOnce(venueIdStr, cityIdNum, houseNumber) {
        // Ruta 1: street vacío específico
        const emptyStreetId = plnGetEmptyStreetIdForCity(cityIdNum);
        if (emptyStreetId != null) {
            const args = { venueId: venueIdStr, streetId: Number(emptyStreetId) };
            if (houseNumber) args.houseNumber = houseNumber;
            plnLog('apply:updateAddress(args)', args);
            wmeSDK.DataModel.Venues.updateAddress(args);
            setTimeout(() => { try { plnTryAutoApplyAddressPanel?.(); } catch { } }, 200);
            return 'streetId';
        }

        // Ruta 2: IDs completos con emptyStreet:true
        const ids = plnResolveIdsFromCity(cityIdNum);
        plnLog('apply:fallbackIds', ids);
        if (ids.countryID && ids.stateID) {
            const args2 = {
                venueId: venueIdStr,
                countryID: Number(ids.countryID),
                stateID: Number(ids.stateID),
                cityID: Number(cityIdNum),
                emptyStreet: true
            };
            if (houseNumber) args2.houseNumber = houseNumber;
            plnLog('apply:updateAddress(args2)', args2);
            wmeSDK.DataModel.Venues.updateAddress(args2);
            setTimeout(() => { try { plnTryAutoApplyAddressPanel?.(); } catch { } }, 200);
            return { type: 'ids', ids };
        }

        return null;
    }

    // ===== INICIO: REEMPLAZAR ESTA FUNCIÓN COMPLETA =====
    async function plnApplyCityToVenue(venueId, selectedCityId, selectedCityName) {
        plnLog('apply:start', { venueId, selectedCityId, selectedCityName });
        if (!wmeSDK?.DataModel?.Venues?.updateAddress) {
            plnErr('apply:sdkNotReady');
            return;
        }

        try {
            const venueIdStr = String(venueId);
            const cityIdNum = Number(selectedCityId) || 0;

            // Intento obtener houseNumber (no bloqueante), es una buena práctica mantenerlo.
            let houseNumber = '';
            try {
                const v0 = wmeSDK.DataModel.Venues.getById?.({ venueId: venueIdStr });
                if (v0?.address?.houseNumber) houseNumber = String(v0.address.houseNumber);
            } catch (_) { /* noop */ }

            // MODIFICACIÓN CLAVE: Se elimina la lógica de espera y el "Plan B (bridge)".
            // Simplemente llamamos a la función que aplica la ciudad y confiamos en que funciona.
            const attemptKind = plnApplyCityOnce(venueIdStr, cityIdNum, houseNumber);

            if (attemptKind) {
                // Si attemptKind no es nulo, significa que se pudo construir y enviar la solicitud al SDK.
                // Asumimos el éxito aquí, ya que la espera en la UI es el punto de fallo.
                plnLog('apply:doneWithSDK: optimistic success');

                // El llamado a plnTryAutoApplyAddressPanel ya está dentro de plnApplyCityOnce,
                // por lo que se ejecutará automáticamente.
                return;
            }

            // Si plnApplyCityOnce devuelve null, significa que no pudo encontrar los IDs necesarios.
            // Solo en este caso, lanzamos el error.
            plnErr('apply:noSdkVenueOrAddress', { reason: "Could not resolve IDs for city.", cityIdNum });

        } catch (e) {
            plnErr('apply:sdkBranchError', e);
        }
    }
    // ===== FIN: REEMPLAZAR ESTA FUNCIÓN COMPLETA =====


    // DEBUG: Detectar llamadas a getElementById('') y registrar stack
    (function patchGetElementByIdDebug() {
        try {
            const _origGetById = document.getElementById.bind(document);
            document.getElementById = function (id) {
                if (id === '') {
                    console.warn('[WME PLN][DEBUG] getElementById(\"\") llamada. Stack:', new Error().stack);
                    return null;
                }
                return _origGetById(id);
            };
        } catch (_) { }
    })();

    //**************************************************************************
    //Nombre: plnTryAutoApplyAddressPanel
    //Fecha modificación: 2025-08-09
    //Descripción: Si el editor de Dirección está abierto, pulsa "Apply/Aplicar" para que la UI deje de mostrar "No address".
    //Notas: No guarda en servidor; solo confirma los campos dentro del panel de dirección.
    //**************************************************************************
    function plnTryAutoApplyAddressPanel() {
        plnLog('ui:autoApply:begin');
        try {
            // Buscar botones candidatos con texto Apply/Aplicar que estén visibles
            const btns = Array.from(document.querySelectorAll('button'))
                .filter(b => b && b.offsetParent !== null) // visibles
                .filter(b => /^(apply|aplicar)$/i.test((b.textContent || b.innerText || '').trim()));

            plnLog('ui:autoApply:candidateButtons', btns.length);
            for (const b of btns) {
                // Asegurar que pertenece al editor de dirección: debe haber inputs de calle/house en su contenedor
                const container = (function findContainer(node) {
                    let n = node;
                    for (let i = 0; i < 6 && n; i++) {
                        if (n.querySelector && (
                            n.querySelector('input[placeholder*="street" i]') ||
                            n.querySelector('input[placeholder*="calle" i]') ||
                            n.querySelector('input[placeholder*="house" i]') ||
                            n.querySelector('input[placeholder*="número" i]')
                        )) return n;
                        n = n.parentElement;
                    }
                    return null;
                })(b);

                if (!container) continue; // botón Apply no es del editor de dirección

                // Evitar doble click si el botón está deshabilitado
                if (b.disabled) continue;

                // Log texto del resumen de dirección antes y después
                const beforeSummary = (function () {
                    const el = Array.from(document.querySelectorAll('div,span')).find(x => /no (address|street|city)/i.test((x.textContent || '')));
                    return el ? el.textContent.trim() : null;
                })();
                plnLog('ui:autoApply:beforeClickSummary', beforeSummary);

                // Click seguro
                b.click();
                setTimeout(() => {
                    const afterSummary = (function () {
                        const el = Array.from(document.querySelectorAll('div,span')).find(x => /no (address|street|city)/i.test((x.textContent || '')));
                        return el ? el.textContent.trim() : null;
                    })();
                    plnLog('ui:autoApply:afterClickSummary', afterSummary);
                }, 300);
                //   console.log('[WME PLN] Address panel Apply pulsado automáticamente para reflejar cambios.');
                return; // una sola vez
            }
        } catch (e) {
            console.warn('[WME PLN] No se pudo auto-aplicar el panel de dirección:', e);
        }
    }

    //**************************************************************************
    //Nombre: registerCityModalHelpers
    //Fecha modificación: 2025-08-09
    //Hora: 17:00
    //Autor: mincho77
    //Entradas: ninguna
    //Salidas: ninguna
    //Prerrequisitos si existen: DOM disponible
    //Descripción: Delegado de click para botón 'Aplicar Ciudad' y observador para mostrar ID junto al nombre de ciudad.
    //**************************************************************************
    function registerCityModalHelpers() {
        let _plnCityClickTs = 0; // Allen-style: debounce timestamp for "Aplicar ciudad"
        function findCityContainer(start) {
            let node = start;
            for (let i = 0; i < 8 && node; i++) {
                if (node.querySelector && node.querySelector('input[type="radio"]')) {
                    const hasCityRadios = node.querySelector(
                        'input[type="radio"][name*="city"], input[type="radio"][name^="city-"], input[type="radio"][name^="city_selection"], input[type="radio"][name^="city-selection-"]'
                    );
                    if (hasCityRadios) return node;
                }
                node = node.parentElement;
            }
            return document.body;
        }
        // Añade "(ID: ####)" a cada opción si aún no lo tiene
        function annotateCityIds(container) {
            try {
                const radios = container.querySelectorAll('input[type="radio"][name*="city"], input[type="radio"][name^="city-"], input[type="radio"][name^="city_selection"], input[type="radio"][name^="city-selection-"]');
                radios.forEach(r => {
                    const lbl = container.querySelector(`label[for="${r.id}"]`) || r.closest('label');
                    if (!lbl) return;
                    if (/\(ID:\s*\d+\)/.test(lbl.textContent)) return; // ya está
                    lbl.appendChild(document.createTextNode(` (ID: ${r.value})`));
                });
            } catch (_) { /* noop */ }
        }

        // Resuelve el venueId como string; nunca devuelve número
        function resolveVenueIdString(btn, container, selected) {
            // 1) data-venue-id en botón o contenedor
            let id = btn.getAttribute('data-venue-id') || container.getAttribute('data-venue-id');
            if (id) return String(id);

            // 2) name del radio: city-selection-<venueId>
            const nameAttr = selected?.name || '';
            let m = nameAttr.match(/city[-_]?selection[-_]?([0-9.]+)/i);
            if (m && m[1]) return String(m[1]);

            // 3) atributo id del radio
            const selId = selected?.id || '';
            m = selId.match(/city[-_]?selection[-_]?([0-9.]+)/i);
            if (m && m[1]) return String(m[1]);

            // 4) URL ?venues=186384446. ...
            const h = String(location.href || '');
            m = h.match(/[?&]venues=([0-9.]+)/);
            if (m && m[1]) return String(m[1]);

            return '';
        }

        // Click en "Aplicar ciudad"
        document.addEventListener('click', function (e) {
            const btn = e.target.closest('button');
            if (!btn) return;
            const txt = (btn.textContent || btn.innerText || '').trim().toLowerCase();
            // Ignorar botones del disclaimer
            if (btn.hasAttribute('data-disclaimer-btn')) return;
            plnLog('ui:click', { buttonText: txt });
            if (!/^(aplicar ciudad|apply city)$/i.test(txt)) return;
            // Allen-style: Debounce to avoid double handling when WME/UI fires multiple events
            const _nowTs = Date.now();
            if (_nowTs - _plnCityClickTs < 400) { // 400ms debounce window
                plnLog('ui:debounce');
                return;
            }
            _plnCityClickTs = _nowTs;
            e.preventDefault();
            e.stopImmediatePropagation?.();
            e.stopPropagation?.();

            const container = findCityContainer(btn);
            annotateCityIds(container);

            let selected = container.querySelector(
                'input[type="radio"][name*="city"]:checked, input[type="radio"][name^="city-"]:checked, input[type="radio"][name^="city_selection"]:checked, input[type="radio"][name^="city-selection-"]:checked'
            ) || container.querySelector('input[type="radio"]:checked');

            plnLog('ui:selectedRadio', { exists: !!selected, name: selected && selected.name, value: selected && selected.value });
            if (!selected) { alert('Selecciona una ciudad.'); return; }

            const selectedCityId = Number(selected.value);

            // Lee el nombre mostrado (limpia “(ID: 123) 3.7 km” si existe)
            let selectedCityName = '';
            const label = container.querySelector(`label[for="${selected.id}"]`) || selected.closest('label');
            if (label) {
                selectedCityName = (label.textContent || '').trim();
                selectedCityName = selectedCityName.replace(/\s*\(ID:\s*\d+\)\s*[\d\.,]*\s*(km|m)?/i, '').trim();
            }

            const venueIdStr = resolveVenueIdString(btn, container, selected);
            plnLog('ui:resolvedData', { venueId: venueIdStr, selectedCityId, selectedCityName });
            plnApplyCityToVenue(venueIdStr, selectedCityId, selectedCityName);
            // NUEVO BLOQUE: Cambiar el ícono a chulito en la tabla de resultados
            setTimeout(() => {
                // Busca la fila correspondiente por data-place-id
                const row = document.querySelector(`tr[data-place-id="${venueIdStr}"]`);
                if (row) {
                    const cityStatusIcon = row.querySelector('.city-status-icon');
                    if (cityStatusIcon) {
                        cityStatusIcon.innerHTML = '✅';
                        cityStatusIcon.style.color = 'green';
                        cityStatusIcon.title = `Ciudad: ${selectedCityName}`; // Actualiza el tooltip con el nombre de la ciudad
                    }
                }
            }, 500); // Espera breve para asegurar que el cambio de ciudad se haya aplicado
        }, true);
    }

    try { window.WME_PLN_registerCityModalHelpers = registerCityModalHelpers; } catch (_) { }
    try { registerCityModalHelpers(); } catch (e) { console.warn('[WME PLN] registerCityModalHelpers init error', e); }
    // Permite crear un panel flotante en WME
    function updateScanProgressBar(currentIndex, totalPlaces) {
        if (totalPlaces === 0) // Si no hay lugares, no actualiza la barra de progreso
            return;
        let progressPercent = Math.floor(((currentIndex + 1) / totalPlaces) * 100); // Calcular el porcentaje de progreso
        progressPercent = Math.min(progressPercent, 100);
        const progressBarInnerTab = document.getElementById("progressBarInnerTab"); // Actualizar la barra de progreso
        const progressBarTextTab = document.getElementById("progressBarTextTab"); // Actualizar el texto de la barra de progreso
        if (progressBarInnerTab && progressBarTextTab) // Asegurarse de que los elementos existen antes de intentar actualizarlos
        {
            progressBarInnerTab.style.width = `${progressPercent}%`; // Actualizar el ancho de la barra de progreso
            const currentDisplay = Math.min(currentIndex + 1, totalPlaces); // Mostrar el número actual de lugares procesados
            progressBarTextTab.textContent = `Progreso: ${progressPercent}% (${currentDisplay}/${totalPlaces})`; // Actualizar el texto de la barra de progreso
        }
    }//updateScanProgressBar

    function updateInconsistenciesCount(delta) {
        const resultsCounterDiv = document.querySelector("#wme-place-inspector-panel .results-counter-display");
        if (resultsCounterDiv) {
            // Recalcular basado en filas visibles y total global si es posible
            let currentVisible = 0;
            const tbody = document.querySelector("#wme-place-inspector-panel tbody");
            if (tbody) {
                currentVisible = Array.from(tbody.querySelectorAll('tr')).filter(row => row.style.display !== 'none').length;
            }

            // Usar el global si existe, sino fallback a dataset
            const totalPending = (typeof window.plnAllInconsistents !== 'undefined')
                ? window.plnAllInconsistents.length
                : parseInt(resultsCounterDiv.dataset.totalOriginal || '0', 10);

            const batchSize = 30; // Hardcoded or read from somewhere
            const currentBatchTotal = Math.min(batchSize, totalPending);
            const processedInBatch = currentBatchTotal - currentVisible;
            const realPending = totalPending - processedInBatch;

            if (currentVisible === 0 && totalPending === 0) {
                resultsCounterDiv.innerHTML = `<span style="color: green;">✔</span> Todos los lugares visibles están correctamente normalizados o excluidos.`;
                const outputDiv = document.querySelector("#wme-place-inspector-output");
                if (outputDiv) outputDiv.innerHTML = `<div style='color:green; padding:10px;'>✔ Todos los lugares visibles están correctamente normalizados o excluidos.</div>`;
            } else {
                // "Inconsistencias pendientes: 84. En pantalla: 29 (Normalizados: 1)"
                resultsCounterDiv.innerHTML = `Inconsistencias pendientes: <b style="color: #ff0000;">${realPending}</b>. En pantalla: <b style="color: #ff0000;">${currentVisible}</b> (Normalizados: <b style="color: green;">${processedInBatch}</b>).`;
            }
        }
    }//updateInconsistenciesCount


    // Permite crear un panel flotante en WME
    function escapeRegExp(string) {
        return string.replace(/[.*+?^${}()|[\]\\]/g, '\\$&');
    }//escapeRegExp



    // Función para cargar palabras del diccionario por defecto (Hoja "Dictionary")
    async function loadDictionaryWordsFromSheet(forceReload = false) {
        const SPREADSHEET_ID = "10HSyonrHd__sC0wO0hmOQyBbNRlWF8UFr6kkOuSru00";
        const API_KEY = "AIzaSyAQbvIQwSPNWfj6CcVEz5BmwfNkao533i8";
        const RANGE = "Dictionary!A2:B";

        // usa window.dictionaryWords y window.dictionaryIndex para almacenar las palabras y su índice
        // Si no existen, las inicializa como un Set y un objeto vacío
        if (!window.dictionaryWords) window.dictionaryWords = new Set();
        if (!window.dictionaryIndex) window.dictionaryIndex = {};

        const url = `https://sheets.googleapis.com/v4/spreadsheets/${SPREADSHEET_ID}/values/${RANGE}?key=${API_KEY}`;

        return new Promise((resolve) => {
            if (SPREADSHEET_ID === "TU_SPREADSHEET_ID" || API_KEY === "TU_API_KEY") {
                console.warn('[WME PLN] SPREADSHEET_ID o API_KEY no configurados para el diccionario.');
                resolve();
                return;
            }

            // verifica si hay datos en caché
            // Si hay datos en caché y no se fuerza la recarga, los usa
            // Si no hay datos en caché o se fuerza la recarga, hace la solicitud
            const cachedData = localStorage.getItem("wme_pln_dictionary_cache");
            if (!forceReload && cachedData) {
                try {
                    const { data, timestamp } = JSON.parse(cachedData);
                    // usar caché si tiene menos de 24 horas
                    if (data && timestamp && (Date.now() - timestamp < 24 * 60 * 60 * 1000)) {
                        //console.log('[WME PLN] Usando datos en caché. Tiempo restante para expirar:', ((timestamp + 24 * 60 * 60 * 1000) - Date.now())/1000/60, 'minutos');

                        // console.log('[WME PLN] Usando diccionario en caché');

                        // restaura las palabras y el índice del diccionario desde la caché
                        window.dictionaryWords = new Set(data.words);
                        window.dictionaryIndex = data.index;

                        // Construir mapa de tildes optimizado
                        buildNormalizedDictionaryMap();

                        updateDictionaryWordsCountLabel();

                        resolve();
                        return;
                    }
                } catch (e) {
                    console.warn('[WME PLN] Error al leer caché del diccionario:', e);
                }
            }

            makeRequest({
                method: "GET",
                url: url,
                timeout: 10000,
                onload: function (response) {
                    if (response.status >= 200 && response.status < 300) {
                        try {
                            const data = JSON.parse(response.responseText);
                            let newWordsAdded = 0;

                            if (data.values) {
                                data.values.forEach(row => {
                                    const word = (row[0] || '').trim();
                                    if (word && !window.dictionaryWords.has(word.toLowerCase())) {
                                        window.dictionaryWords.add(word.toLowerCase());
                                        const firstChar = word.charAt(0).toLowerCase();
                                        if (!window.dictionaryIndex[firstChar])
                                            window.dictionaryIndex[firstChar] = [];
                                        window.dictionaryIndex[firstChar].push(word.toLowerCase());
                                        newWordsAdded++;
                                    }
                                });

                                // Cache the dictionary
                                try {
                                    localStorage.setItem("wme_pln_dictionary_cache", JSON.stringify({
                                        data: {
                                            words: Array.from(window.dictionaryWords),
                                            index: window.dictionaryIndex
                                        },
                                        timestamp: Date.now()
                                    }));
                                } catch (e) {
                                    console.warn('[WME PLN] Error al guardar caché del diccionario:', e);
                                }

                                // también guarda en localStorage para uso rápido
                                try {
                                    localStorage.setItem("dictionaryWordsList", JSON.stringify(Array.from(window.dictionaryWords)));
                                } catch (e) {
                                    console.error("[WME PLN] Error guardando diccionario en localStorage:", e);
                                }

                                // Construir mapa de tildes optimizado
                                buildNormalizedDictionaryMap();

                                // ✨ Limpiar palabras propias que ahora están en el diccionario oficial
                                if (window.userAddedDictionaryWords && window.userAddedDictionaryWords.size > 0) {
                                    const wordsToRemove = [];
                                    window.userAddedDictionaryWords.forEach(userWord => {
                                        if (window.dictionaryWords.has(userWord.toLowerCase())) {
                                            wordsToRemove.push(userWord);
                                        }
                                    });

                                    if (wordsToRemove.length > 0) {
                                        wordsToRemove.forEach(word => window.userAddedDictionaryWords.delete(word));
                                        // Guardar la lista actualizada
                                        try {
                                            localStorage.setItem('userAddedDictionaryWords',
                                                JSON.stringify(Array.from(window.userAddedDictionaryWords)));
                                            console.log(`[WME PLN] ${wordsToRemove.length} palabra(s) propia(s) eliminada(s) porque ya están en el diccionario oficial:`, wordsToRemove);
                                        } catch (e) {
                                            console.error('[WME PLN] Error guardando palabras de usuario limpiadas:', e);
                                        }
                                    }
                                }

                                if (window.updateDictionaryWordsCountLabel) {
                                    window.updateDictionaryWordsCountLabel();
                                }

                                //console.log(`[WME PLN] Diccionario cargado: ${newWordsAdded} palabras nuevas añadidas.`);
                            }
                        } catch (e) {
                            console.error('[WME PLN] Error al procesar datos del diccionario:', e);
                        }
                    }
                    resolve();
                },
                // Añade esto en ambas funciones, justo después del try/catch en onload:
                onerror: function (error) {
                    console.error('[WME PLN] Error de red al cargar datos por defecto:', error);
                    //console.log('[WME PLN] URL que falló:', url);
                    resolve(); // Resolver la promesa para no bloquear
                },
                ontimeout: function () {
                    console.error('[WME PLN] Timeout al cargar diccionario');
                    resolve();
                }
            });
        });
    }//loadDictionaryWordsFromSheet

    // Función para cargar palabras especiales/excluidas por defecto
    async function loadExcludedWordsFromSheet(forceReload = false) {
        const SPREADSHEET_ID = "10HSyonrHd__sC0wO0hmOQyBbNRlWF8UFr6kkOuSru00";
        const API_KEY = "AIzaSyAQbvIQwSPNWfj6CcVEz5BmwfNkao533i8";
        // Se prueban varias pestañas candidatas para compatibilidad hacia atrás
        const CANDIDATE_RANGES = [
            "SpecialWords!A2:A",
        ];
        const CACHE_KEY = "wme_pln_excluded_cache";
        const CACHE_TTL_MS = 24 * 60 * 60 * 1000; // 24 horas
        const SYNC_FLAG_KEY = "wme_pln_special_words_synced";

        if (SPREADSHEET_ID === "TU_SPREADSHEET_ID" || API_KEY === "TU_API_KEY") {
            console.warn('[WME PLN] SPREADSHEET_ID o API_KEY no configurados para palabras especiales.');
            return;
        }

        if (!(window.excludedWords instanceof Set)) window.excludedWords = new Set(Array.isArray(window.excludedWords) ? window.excludedWords : []);
        if (!(excludedWordsMap instanceof Map)) excludedWordsMap = new Map();

        // Check if this is the first run (flag not present)
        const isFirstRun = !localStorage.getItem(SYNC_FLAG_KEY);

        if (!forceReload && !isFirstRun) {
            try {
                const cachedRaw = localStorage.getItem(CACHE_KEY);
                if (cachedRaw) {
                    const { timestamp, data } = JSON.parse(cachedRaw);
                    if (timestamp && data && Array.isArray(data) && (Date.now() - timestamp) < CACHE_TTL_MS) {
                        applyExcludedWordsFromSheet(data, false);
                        return;
                    }
                }
            }
            catch (cacheErr) {
                console.warn('[WME PLN] Error leyendo caché de palabras especiales:', cacheErr);
            }
        }

        await new Promise((resolve) => {
            let resolved = false;

            const tryRange = (idx) => {
                if (idx >= CANDIDATE_RANGES.length) {
                    if (!resolved) {
                        resolve();
                        resolved = true;
                    }
                    return;
                }

                const range = CANDIDATE_RANGES[idx];
                const url = `https://sheets.googleapis.com/v4/spreadsheets/${SPREADSHEET_ID}/values/${range}?key=${API_KEY}`;

                makeRequest({
                    method: "GET",
                    url,
                    timeout: 10000,
                    onload: (response) => {
                        if (response.status >= 200 && response.status < 300) {
                            try {
                                const payload = JSON.parse(response.responseText);
                                const values = Array.isArray(payload?.values) ? payload.values : [];
                                const sheetWords = values
                                    .map(row => (row?.[0] || '').trim())
                                    .filter(Boolean);

                                if (sheetWords.length) {
                                    try {
                                        localStorage.setItem(CACHE_KEY, JSON.stringify({ timestamp: Date.now(), data: sheetWords }));
                                    }
                                    catch (cacheErr) {
                                        console.warn('[WME PLN] No se pudo guardar caché de palabras especiales:', cacheErr);
                                    }

                                    applyExcludedWordsFromSheet(sheetWords, isFirstRun);

                                    // If it was first run, mark as synced now
                                    if (isFirstRun) {
                                        try {
                                            localStorage.setItem(SYNC_FLAG_KEY, "true");
                                            console.log('[WME PLN] Primera sincronización de palabras especiales completada. Flag establecido.');
                                        } catch (e) { }
                                    }

                                    if (!resolved) {
                                        resolve();
                                        resolved = true;
                                    }
                                    return;
                                }
                            }
                            catch (parseErr) {
                                console.error('[WME PLN] Error procesando palabras especiales desde Sheets:', parseErr);
                            }
                        }

                        // Intentar con la siguiente pestaña si esta no devolvió datos útiles
                        tryRange(idx + 1);
                    },
                    onerror: () => tryRange(idx + 1),
                    ontimeout: () => tryRange(idx + 1)
                });
            };

            tryRange(0);
        });

        function applyExcludedWordsFromSheet(wordsArray, isFirstRun) {
            try {
                const sheetSet = new Map();
                wordsArray.forEach(word => {
                    const normalizedKey = word.toLowerCase();
                    if (!sheetSet.has(normalizedKey)) sheetSet.set(normalizedKey, word);
                });

                // Logic:
                // If isFirstRun: Overwrite local with sheet (sheetSet becomes the base)
                // If !isFirstRun: Merge sheet into local (keep local additions)

                let combined;
                if (isFirstRun) {
                    // Overwrite: Use only words from sheet
                    combined = new Set(sheetSet.values());
                    console.log('[WME PLN] Primera ejecución: Sobrescribiendo palabras especiales locales con las de la hoja.');
                } else {
                    // Merge: Keep existing local words + add new ones from sheet
                    const existing = window.excludedWords instanceof Set ? Array.from(window.excludedWords) : [];
                    existing.forEach(word => {
                        const key = String(word || '').toLowerCase();
                        if (!sheetSet.has(key)) sheetSet.set(key, word);
                    });
                    combined = new Set(sheetSet.values());
                }

                window.excludedWords = combined;

                if (!(excludedWordsMap instanceof Map)) excludedWordsMap = new Map();
                excludedWordsMap.clear();
                combined.forEach(word => {
                    const firstChar = word.charAt(0).toLowerCase();
                    if (!excludedWordsMap.has(firstChar)) excludedWordsMap.set(firstChar, new Set());
                    excludedWordsMap.get(firstChar).add(word);
                });

                try { saveExcludedWordsToLocalStorage(); } catch (e) { console.warn('[WME PLN] Error guardando palabras especiales tras cargar Sheets:', e); }

                // Refrescar la UI si el panel ya está construido
                requestAnimationFrame(() => {
                    try {
                        const ul = document.getElementById("excludedWordsList");
                        if (ul && typeof renderExcludedWordsList === 'function') renderExcludedWordsList(ul);
                        updateSpecialWordsCountLabel();
                    }
                    catch (uiErr) {
                        console.warn('[WME PLN] No se pudo refrescar la lista de palabras especiales tras recarga:', uiErr);
                    }
                });
            }
            catch (err) {
                console.error('[WME PLN] Error aplicando palabras especiales desde Sheets:', err);
            }
        }
    }// loadExcludedWordsFromSheet
    try { window.loadExcludedWordsFromSheet = loadExcludedWordsFromSheet; } catch (_) { }


    //Función Para Cargar Categorías por defecto
    async function loadDynamicCategoriesFromSheet(forceReload = false) {
        const SPREADSHEET_ID = "10HSyonrHd__sC0wO0hmOQyBbNRlWF8UFr6kkOuSru00";
        const API_KEY = "AIzaSyAQbvIQwSPNWfj6CcVEz5BmwfNkao533i8";
        const RANGE = "Categories!A2:F";
        const CACHE_VERSION = 2; // Increment this when schema changes (added prefixWords)
        window.dynamicCategoryRules = []; // Definimos la variable global para guardar las reglas
        const url = `https://sheets.googleapis.com/v4/spreadsheets/${SPREADSHEET_ID}/values/${RANGE}?key=${API_KEY}`;

        return new Promise((resolve) => {
            if (SPREADSHEET_ID === "TU_SPREADSHEET_ID" || API_KEY === "TU_API_KEY") {
                console.warn('[WME PLN] No se ha configurado SPREADSHEET_ID o API_KEY. Se omitirá la carga de categorías dinámicas.');
                resolve();
                return;
            }

            // Check for cached data first
            const cachedData = localStorage.getItem("wme_pln_categories_cache");
            if (!forceReload && cachedData) {
                try {
                    const { data, timestamp, version } = JSON.parse(cachedData);
                    // Use cache if less than 24 hours old AND version matches
                    if (data && timestamp && version === CACHE_VERSION && (Date.now() - timestamp < 24 * 60 * 60 * 1000)) {
                        //console.log('[WME PLN] Usando categorías en caché. Reconstruyendo RegExp...');

                        // ===================== INICIO DE LA CORRECCIÓN =====================
                        // Se itera sobre los datos de la caché para reconstruir las expresiones regulares
                        const buildKeywordPattern = (keywordRaw) => {
                            const tokens = keywordRaw.split(/\s+/).filter(Boolean);
                            const tokenPatterns = tokens.map(token => {
                                const base = escapeRegExp(token);
                                if (/s$/i.test(token)) return base;
                                if (/[aeiouáéíóú]$/i.test(token)) return `${base}s?`;
                                if (/[nrldct]$/i.test(token)) return `${base}(?:es)?`;
                                return `${base}(?:es|s)?`;
                            });
                            return `\\b${tokenPatterns.join('\\s+')}\\b`;
                        };

                        window.dynamicCategoryRules = data.map(rule => {
                            if (rule.keyword) { // Asegurarse de que la regla tenga keywords
                                const keywords = rule.keyword.split(';').map(k => k.trim()).filter(k => k.length > 0);
                                const regexParts = keywords.map(k => buildKeywordPattern(k));
                                const combinedRegex = new RegExp(`(${regexParts.join('|')})`, 'i');

                                // Devolver la regla con la propiedad compiledRegex correctamente creada
                                return { ...rule, compiledRegex: combinedRegex };
                            }
                            return rule; // Devuelve la regla sin cambios si no tiene keyword
                        });
                        // ===================== FIN DE LA CORRECCIÓN =====================

                        window.dynamicCategoryRules.sort((a, b) => b.keyword.length - a.keyword.length);
                        resolve();
                        return;
                    }
                } catch (e) {
                    console.warn('[WME PLN] Error al leer caché de categorías:', e);
                }
            }

            makeRequest({
                method: "GET",
                url: url,
                timeout: 10000, // Add timeout
                onload: function (response) {
                    if (response.status >= 200 && response.status < 300) {
                        try {
                            const data = JSON.parse(response.responseText);
                            if (data.values) {
                                // El procesamiento de los datos de la API ya era correcto
                                const buildKeywordPattern = (keywordRaw) => {
                                    const tokens = keywordRaw.split(/\s+/).filter(Boolean);
                                    const tokenPatterns = tokens.map(token => {
                                        const base = escapeRegExp(token);
                                        if (/s$/i.test(token)) return base;
                                        if (/[aeiouáéíóú]$/i.test(token)) return `${base}s?`;
                                        if (/[nrldct]$/i.test(token)) return `${base}(?:es)?`;
                                        return `${base}(?:es|s)?`;
                                    });
                                    return `\\b${tokenPatterns.join('\\s+')}\\b`;
                                };

                                window.dynamicCategoryRules = data.values.map(row => {
                                    const keyword = (row[0] || '').toLowerCase().trim();
                                    const keywords = keyword.split(';').map(k => k.trim()).filter(k => k.length > 0);
                                    const regexParts = keywords.map(k => buildKeywordPattern(k));
                                    const combinedRegex = new RegExp(`(${regexParts.join('|')})`, 'i');
                                    return {
                                        keyword: keyword,
                                        categoryKey: row[1] || '',
                                        icon: row[2] || '⚪',
                                        desc_es: row[3] || 'Sin descripción',
                                        desc_en: row[4] || 'No description',
                                        prefixWords: row[5] || '', // Columna F: Palabras para botones
                                        compiledRegex: combinedRegex
                                    };
                                });
                                window.dynamicCategoryRules.sort((a, b) => b.keyword.length - a.keyword.length);

                                // La lógica para guardar en caché también es correcta
                                try {
                                    localStorage.setItem("wme_pln_categories_cache", JSON.stringify({
                                        data: window.dynamicCategoryRules,
                                        timestamp: Date.now(),
                                        version: CACHE_VERSION
                                    }));
                                } catch (e) {
                                    console.warn('[WME PLN] Error al guardar caché de categorías:', e);
                                }

                                //console.log('[WME PLN] Categorías cargadas desde API');
                            }
                        } catch (e) {
                            console.error('[WME PLN] Error al procesar datos de categorías:', e);
                        }
                    } else {
                        console.warn(`[WME PLN] Error HTTP ${response.status} al cargar categorías`);
                    }
                    resolve();
                },
                onerror: function (error) {
                    console.error('[WME PLN] Error de red al cargar categorías:', error);
                    resolve();
                },
                ontimeout: function () {
                    console.error('[WME PLN] Timeout al cargar categorías');
                    resolve();
                }
            });
        });
    }//loadDynamicCategoriesFromSheet

    // === Debug helper: force category cache reload ===
    try {
        window.plnReloadCategories = async function () {
            console.log('[PLN] Limpiando caché de categorías...');
            localStorage.removeItem('wme_pln_categories_cache');
            console.log('[PLN] Recargando categorías desde Google Sheets...');
            await loadDynamicCategoriesFromSheet(true);
            console.log('[PLN] Categorías recargadas. Total de reglas:', window.dynamicCategoryRules?.length || 0);
            // Log a sample to verify prefixWords is present
            if (window.dynamicCategoryRules && window.dynamicCategoryRules.length > 0) {
                console.log('[PLN] Ejemplo de regla (primera):', {
                    keyword: window.dynamicCategoryRules[0].keyword,
                    categoryKey: window.dynamicCategoryRules[0].categoryKey,
                    icon: window.dynamicCategoryRules[0].icon,
                    desc_es: window.dynamicCategoryRules[0].desc_es,
                    prefixWords: window.dynamicCategoryRules[0].prefixWords
                });
            }
            plnToast('✅ Categorías recargadas desde Google Sheets', 2500);
        };
        if (typeof unsafeWindow !== 'undefined') unsafeWindow.plnReloadCategories = window.plnReloadCategories;
    } catch (_) { }

    // Función para encontrar la categoría de un lugar basado en su nombre
    function findCategoryForPlace(placeName) {
        if (!placeName || typeof placeName !== 'string' || !window.dynamicCategoryRules || window.dynamicCategoryRules.length === 0) // Si el nombre del lugar es inválido o no hay reglas de categoría cargadas, devuelve un array vacío de sugerencias.
            return [];
        const lowerCasePlaceName = placeName.toLowerCase();// Convertir el nombre del lugar a minúsculas para comparaciones insensibles a mayúsculas
        const allMatchingRules = []; // Este array almacenará todas las reglas de categoría que coincidan.
        const placeWords = lowerCasePlaceName.split(/\s+/).filter(w => w.length > 0); // Descomponer el nombre del lugar en palabras
        const SIMILARITY_THRESHOLD_FOR_KEYWORDS = 0.95; // Puedes ajustar este umbral (ej. 0.90 para 90% de similitud)
        // PASO 0: Normalizar el nombre del lugar eliminando diacríticos y caracteres especiales
        for (const rule of window.dynamicCategoryRules) {
            if (!rule.compiledRegex) continue; // Si la regla no tiene una expresión regular compilada (lo cual no debería pasar si se cargó correctamente), salta a la siguiente regla.
            // **PASO 1: Búsqueda por Regex Exacta
            if (rule.compiledRegex.test(lowerCasePlaceName)) {
                if (!allMatchingRules.some(mr => mr.categoryKey === rule.categoryKey)) {
                    allMatchingRules.push(rule);
                }
                // Si Ya Añadimos La Regla Por Regex Exacta, Pasar A La Siguiente Regla Para Ahorrar Cálculos De Similitud
                continue;
            }
            // **PASO 2: Búsqueda por Similitud para CADA palabra del lugar vs CADA palabra clave de la regla**
            const ruleKeywords = rule.keyword.split(';').map(k => k.trim().toLowerCase()).filter(k => k.length > 0);
            let foundSimilarityForThisRule = false; // Bandera para saber si ya encontramos una buena similitud para esta regla, para no seguir buscando más palabras clave de la regla.
            for (const pWord of placeWords) // Cada palabra del nombre del lugar
            { // Cada palabra del nombre del lugar
                if (foundSimilarityForThisRule) break; // Si ya encontramos una buena similitud para esta regla, pasamos a la siguiente.
                for (const rKeyword of ruleKeywords) { // Cada palabra clave de la regla
                    // Asegurarse de que rKeyword no sea una expresión regular, sino la palabra literal para Levenshtein
                    const similarity = calculateSimilarity(pWord, rKeyword); // Calcular la similitud entre la palabra del lugar y la palabra clave de la regla
                    if (similarity >= SIMILARITY_THRESHOLD_FOR_KEYWORDS && !allMatchingRules.some(mr => mr.categoryKey === rule.categoryKey)) // Si la similitud es alta y aún no hemos añadido esta categoría
                    {
                        allMatchingRules.push(rule);
                        foundSimilarityForThisRule = true; // Marcamos que ya la encontramos para esta regla
                        break; // Salimos del bucle de rKeyword y pWord
                    }
                }
            }
        }
        //console.log(`[WME PLN][DEBUG] findCategoryForPlace para "${placeName}" devolvió: `, allMatchingRules);
        return allMatchingRules;
    }//findCategoryForPlace

    // Permite obtener el icono de una categoría
    function getWazeLanguage() {
        // 1. Intento principal con el SDK (método recomendado)
        if (wmeSDK && typeof wmeSDK.getWazeLocale === 'function') {
            const locale = wmeSDK.getWazeLocale(); // ej: 'es-419'
            if (locale)
                return locale.split('-')[0].toLowerCase(); // -> 'es'
        }
        // 2. Fallback al objeto global 'W' si el SDK falla
        if (typeof W !== 'undefined' && W.locale)
            return W.locale.split('-')[0].toLowerCase();
        // 3. Último recurso si nada funciona
        return 'es';
    }//getWazeLanguage

    //Permite obtener el icono y descripción de una categoría
    function getCategoryDetails(categoryKey) {
        const lang = getWazeLanguage();
        // 1. Intento con la hoja de Google (window.dynamicCategoryRules)
        if (window.dynamicCategoryRules && window.dynamicCategoryRules.length > 0) {
            const rule = window.dynamicCategoryRules.find(r => r.categoryKey.toUpperCase() === categoryKey.toUpperCase());
            if (rule) {
                const description = (lang === 'es' && rule.desc_es) ? rule.desc_es : rule.desc_en;
                return { icon: rule.icon, description: description };
            }
        }
        // 2. Fallback a la lista interna del script si no se encontró en la hoja
        const hardcodedInfo = getCategoryIcon(categoryKey); // Llama a la función original
        if (hardcodedInfo && hardcodedInfo.icon !== '⚪' && hardcodedInfo.icon !== '❓') {
            // La función original devuelve un título "Español / English", lo separamos.
            const descriptions = hardcodedInfo.title.split(' / ');
            const description = (lang === 'es' && descriptions[0]) ? descriptions[0] : descriptions[1] || descriptions[0];
            return { icon: hardcodedInfo.icon, description: description };
        }
        // 3. Si no se encuentra en ninguna parte, devolver un valor por defecto.
        const defaultDescription = lang === 'es' ? `Categoría no encontrada (${categoryKey})` : `Category not found (${categoryKey})`;
        return { icon: '⚪', description: defaultDescription };
    }//getCategoryDetails

    // Función para eliminar diacríticos de una cadena
    function removeDiacritics(str) {
        return str.normalize("NFD").replace(/[\u0300-\u036f]/g, "");
    }//removeDiacritics

    // Función para validar una palabra excluida
    // Modificación de isValidExcludedWord para hacer verificaciones más estrictas
    function isValidExcludedWord(newWord) {
        if (!newWord) // Si la palabra está vacía, no es válida
            return { valid: false, msg: "La palabra no puede estar vacía." };

        const lowerNewWord = newWord.toLowerCase(); // Convertir a minúsculas para comparaciones insensibles a mayúsculas

        if (newWord.length === 1) // No permitir palabras de un solo caracter
            return { valid: false, msg: "No se permite agregar palabras de un solo caracter." };

        if (/^\d+$/.test(newWord)) // Bloquear números puros
            return { valid: false, msg: "No se permite agregar solo números a la lista." };

        if (/[-']/.test(newWord)) // Permitir palabras con "-" o "'" sin separarlas
            return { valid: true };

        if (/^[^a-zA-Z0-9áéíóúÁÉÍÓÚñÑ]+$/.test(newWord)) // No permitir caracteres especiales solos
            return { valid: false, msg: "No se permite agregar solo caracteres especiales." };

        // VERIFICACIÓN MÁS ESTRICTA DEL DICCIONARIO
        // Verificar tanto coincidencia exacta como insensible a mayúsculas/minúsculas
        if (window.dictionaryWords) {
            // Verificar si la palabra existe en el diccionario (insensible a mayúsculas/minúsculas)
            const dictionaryHasWordInsensitive = Array.from(window.dictionaryWords).some(w => w.toLowerCase() === lowerNewWord);
            if (dictionaryHasWordInsensitive) {
                return { valid: false, msg: "La palabra ya existe en el diccionario (sin considerar mayúsculas/minúsculas). No se puede agregar a especiales." };
            }

            // Verificar coincidencia exacta (esto ya está en código original)
            const dictionaryHasWordExact = Array.from(window.dictionaryWords).some(w => w === newWord);
            if (dictionaryHasWordExact) {
                return { valid: false, msg: "La palabra (con esta capitalización exacta) ya existe en el diccionario. No se puede agregar a especiales." };
            }
        }

        // Verificar si la palabra es una palabra común
        if (commonWords.has(lowerNewWord))
            return { valid: false, msg: "Esa palabra es muy común y no debe agregarse a la lista." };

        // Verificar si la palabra ya está en la lista de excluidas (tanto exacta como insensible a mayúsculas/minúsculas)
        if (excludedWords) {
            // Verificar coincidencia exacta
            if (excludedWords.has(newWord)) {
                return { valid: false, msg: "La palabra (con esta capitalización exacta) ya está en la lista." };
            }
            // Verificar coincidencia insensible a mayúsculas/minúsculas
            const firstChar = lowerNewWord.charAt(0);
            const candidatesForFirstChar = excludedWordsMap.get(firstChar);

            if (candidatesForFirstChar) {
                for (const existingWord of candidatesForFirstChar) {
                    if (existingWord.toLowerCase() === lowerNewWord) {
                        return { valid: false, msg: "Esta palabra ya existe en la lista (con diferente capitalización)." };
                    }
                }
            }


        }

        return { valid: true };
    }
    /*function isValidExcludedWord(newWord)
    {
        if (!newWord) // Si la palabra está vacía, no es válida
            return { valid : false, msg : "La palabra no puede estar vacía." };
        const lowerNewWord = newWord.toLowerCase(); // Convertir a minúsculas para comparaciones insensibles a mayúsculas
        if (newWord.length === 1) // No permitir palabras de un solo caracter
            return { valid: false, msg: "No se permite agregar palabras de un solo caracter." };
        if (/[-']/.test(newWord)) // Permitir palabras con "-" o "'" sin separarlas
            return { valid: true };
        if (/^[^a-zA-Z0-9áéíóúÁÉÍÓÚñÑ]+$/.test(newWord)) // No permitir caracteres especiales solos
            return { valid : false, msg : "No se permite agregar solo caracteres especiales." };
        // La verificación del diccionario ahora será sensible a mayúsculas/minúsculas
        // Si la palabra existe EXACTAMENTE como está en el diccionario, no permitirla como excluida.
        // Si el diccionario tiene 'sos' y la excluida es 'SOS', SÍ se permitirá.
        if (window.dictionaryWords && Array.from(window.dictionaryWords).some(w => w === newWord)) // Comparación sensible a mayúsculas/minúsculas
            return { valid : false, msg :"La palabra (con esta capitalización exacta) ya existe en el diccionario. No se puede agregar a especiales." };
        // Verificar si la palabra es una palabra común
        if (commonWords.has(lowerNewWord))        // No permitir palabras comunes
            return { valid: false, msg: "Esa palabra es muy común y no debe agregarse a la lista." };
       // Verificar si la palabra ya está en la lista de excluidas
        if (excludedWords && Array.from(excludedWords).some(w => w === newWord)) // No permitir duplicados exactos en excluidas
            return { valid : false, msg : "La palabra (con esta capitalización exacta) ya está en la lista." };
        return { valid : true };
    *///isValidExcludeWord

    // La función removeEmoticons con una regex más segura o un paso extraremoveEmoticons solo para emojis (sin afectar números)
    function removeEmoticons(text) {
        if (!text || typeof text !== 'string') {
            return '';
        }
        // Esta es una regex moderna y más robusta que utiliza propiedades de Unicode.
        // \p{Emoji_Presentation}: Coincide con emojis que se muestran como imágenes por defecto (Ej: 😊, 🏨, 🚗).
        // \p{Extended_Pictographic}: Coincide con un conjunto más amplio de símbolos que pueden ser emojis.
        // El flag 'u' es CRUCIAL para que la sintaxis \p{...} funcione.
        const emojiRegex = /[\p{Emoji_Presentation}\p{Extended_Pictographic}]/gu;

        let cleanedText = text.replace(emojiRegex, '');

        // Limpieza final de espacios extra que puedan quedar.
        cleanedText = cleanedText.replace(/\s{2,}/g, ' ').trim();

        // Si al retirar el emoji queda un guion colgando al final, elimínalo.
        // Mejora: Usar regex más robusta que capture guiones con o sin espacios
        cleanedText = cleanedText.replace(/\s*-\s*$/g, '').trim();

        // Evita devolver solo un guion si el emoji dejó el nombre vacío.
        if (/^-\s*$/.test(cleanedText)) {
            cleanedText = '';
        }

        return cleanedText;
    }// removeEmoticons


    // Modify aplicarReemplazosGenerales
    function aplicarReemplazosGenerales(name) {
        if (typeof window.skipGeneralReplacements === "boolean" && window.skipGeneralReplacements)
            return name;
        // Paso 1: Eliminar emoticones al inicio de los reemplazos generales.
        name = removeEmoticons(name);
        const reglas = [
            // Nueva regla: reemplazar | por espacio, guion y espacio
            { buscar: /\|/g, reemplazar: " - " },
            // Nueva regla: reemplazar / por espacio, barra y espacio, eliminando espacios alrededor
            { buscar: /\s*\/\s*/g, reemplazar: " - " },
            // Corrección: Para buscar [P] o [p] literalmente
            { buscar: /\[[Pp]\]/g, reemplazar: "" },
            // 1. Convertir guiones pegados a palabras o con un solo espacio a ' - '
            // Esto convierte "Palabra-Otra" -> "Palabra - Otra"
            // y "Palabra -Otra" -> "Palabra - Otra"
            // y "Palabra- Otra" -> "Palabra - Otra"
            { buscar: /(\p{L}|\p{N})\s*-\s*(\p{L}|\p{N})/gu, reemplazar: "$1 - $2" },
            // 2. Limpiar guiones que no estén entre palabras, convirtiéndolos en un solo ' - '
            // Esto estandariza " -- " a " - ", y " - " a " - ".
            // Asegura que siempre haya un espacio en cada lado del guion si no es un guion interno.
            { buscar: /\s*-\s*/g, reemplazar: " - " },
            { buscar: /\s{2,}/g, reemplazar: ' ' }, // Asegura espacios únicos antes de trim
        ];
        reglas.forEach(regla => { // Itera sobre cada regla de reemplazo
            if (regla.buscar.source === '\\|') // Si la regla es para el carácter '|', usa replaceAll
                name = name.replace(regla.buscar, regla.reemplazar);
            else
                name = name.replace(regla.buscar, regla.reemplazar);
        });
        name = name.replace(/\s{2,}/g, ' ').trim(); // Asegura el recorte final y espacios únicos

        // ****** INICIO DE LA MODIFICACIÓN: Limpieza de guiones dobles ******
        // Eliminar guiones dobles si están seguidos, reemplazándolos por un solo guion
        // Maneja '--', ' - - ', etc. y los convierte a un solo ' - '
        name = name.replace(/\s*-\s*-\s*/g, ' - ');
        name = name.replace(/--/g, '-'); // También para guiones pegados 'a--b' -> 'a-b'
        // ****** FIN DE LA MODIFICACIÓN ******

        return name;
    }// aplicarReemplazosGenerales

    // función auxiliar para capitalizar cada palabra en una frase
    function capitalizeEachWord(phrase) {
        if (!phrase || typeof phrase !== 'string') return "";
        return phrase.split(/\s+/) // Dividir por uno o más espacios
            .map((word, index) => { // Añadir index
                if (word.length === 0) return "";
                // La capitalización de palabras comunes y otras reglas complejas
                // ya las maneja normalizeWordInternal. Aquí solo capitalizamos la primera letra
                // si no es un caso especial.
                // Llamar a normalizeWordInternal para asegurar consistencia
                return normalizeWordInternal(word, index === 0, false); // Pasar isFirstWordInSequence
            })
            .join(' '); // Unir de nuevo con un solo espacio
    }

    //Permite aplicar reglas especiales de capitalización y puntuación a un nombre
    function aplicarReglasEspecialesNombre(newName) {
        // Regla de capitalización después de GUION
        newName = newName.replace(/-(\s*)([^\s]+)/g, (match, spaces, nextWord) => {
            // Esta lógica reutiliza la función principal de normalización para la palabra que sigue al guion.
            // Se pasa 'true' para isFirstWordInSequence, lo que le indica a la función que debe
            // capitalizar palabras comunes como "de", "la", "el", etc.
            const normalizedNextWord = normalizeWordInternal(nextWord, true, false);
            return `-${spaces}${normalizedNextWord}`;
        });
        // Capitalizar Después De Punto
        newName = newName.replace(/\.\s+([a-z])/g, (match, letter) => `. ${letter.toUpperCase()}`);
        // Capitalizar Después De Paréntesis De Apertura
        newName = newName.replace(/(\(\s*)([a-zA-Z])/g, (match, P1, P2) => {
            return P1 + P2.toUpperCase();
        });
        // Asegura que la última letra de una cadena esté en mayúsculas si es una letra sola al final
        newName = newName.replace(/\s([a-zA-Z])$/, (match, letter) => ` ${letter.toUpperCase()}`);
        // Asegurarse de que no haya espacios dobles creados y trim final
        newName = newName.replace(/\s{2,}/g, ' ').trim();
        return newName;
    }//aplicarReglasEspecialesNombre

    // Permite normalizar un nombre de lugar
    function processPlaceName(originalName) {
        //console.log(`[WME PLN] --- INICIANDO ANÁLISIS PARA: "${originalName}" ---`);
        //console.log(`[WME PLN - processPlaceName] Recibido para normalizar: "${originalName}"`); // LOG INICIO

        let processedName = originalName.trim();

        // Primero, reemplazamos el pipe por un espacio, para que las palabras se separen correctamente.
        // Hacemos esto ANTES de dividir en palabras para que normalizeWordInternal las vea por separado.
        processedName = processedName.replace(/\|/g, ' - '); // Reemplaza | por un espacio
        processedName = processedName.replace(/\s{2,}/g, ' ').trim(); // Limpia espacios dobles que puedan generarse

        //console.log(`[WME PLN - processPlaceName] Después de reemplazo de pipe: "${processedName}"`); // LOG PIPE REEMPLAZADO

        // Si el nombre está vacío después de los reemplazos, no hacemos nada más.
        const words = processedName.split(/\s+/).filter(word => word.length > 0);

        //console.log(`[WME PLN - processPlaceName] Palabras extraídas:`, words); // LOG PALABRAS EXTRAÍDAS


        // PASO 1: Normalización palabra por palabra (capitalización, reglas especiales)
        const normalizedWords = words.map((word, index) => {
            if (word === '-') return '-';

            const excl = isExcludedWord(word);
            if (excl) {
                if (excl.endsWith('-') && excl.replace(/-+$/, '').length > 0) {
                    return excl.replace(/-+$/, '');
                }
                return excl;
            }

            const lower = (word || '').toLowerCase();
            if (commonWords.has(lower)) {
                // minúscula, salvo si es la 1ª palabra o si viene justo después de "-" o "("
                const prevIsHyphen = index > 0 && words[index - 1] === '-';
                const prevIsOpenParen = index > 0 && words[index - 1] === '(';
                if (index === 0 || prevIsHyphen || prevIsOpenParen) {
                    return lower.charAt(0).toUpperCase() + lower.slice(1);
                }
                return lower;
            }

            return normalizeWordInternal(word, index === 0, false);
        });
        processedName = normalizedWords.join(" ");
        //console.log(`[WME PLN] [Paso 1] Después de normalizar cada palabra: "${processedName}"`); // LOG PALABRAS NORMALIZADAS

        //console.log(`[WME PLN] [Paso 1] Después de normalizar cada palabra: "${processedName}"`);

        // PASO 2: Aplicar reglas especiales de nombre (capitalización después de guion, etc.)
        // Aquí es donde `aplicarReglasEspecialesNombre` manejará el guion y capitalizará "Bolívar".
        processedName = aplicarReglasEspecialesNombre(processedName);
        //console.log(`[WME PLN] [Paso 2] Después de aplicar reglas especiales: "${processedName}"`);


        // PASO 3: Procesar comillas y paréntesis
        processedName = postProcessQuotesAndParentheses(processedName);
        //console.log(`[WME PLN] [Paso 3] Después de procesar comillas/paréntesis: "${processedName}"`);

        // PASO 4: Aplicar reemplazos definidos por el usuario
        if (typeof replacementWords === 'object' && Object.keys(replacementWords).length > 0) {
            //console.log("[WME PLN] [Paso 4] ANTES de aplicarReemplazosDefinidos:", processedName, replacementWords);
            processedName = aplicarReemplazosDefinidos(processedName, replacementWords);
            //console.log("[WME PLN] [Paso 4]DESPUÉS de aplicarReemplazosDefinidos:", processedName);
        }
        //console.log(`[WME PLN] [Paso 4] DESPUÉS de aplicar reemplazos: "${processedName}"`);

        // PASO 5: Aplicar reemplazos generales (barras, corchetes, etc.)
        // La regla para el pipe '|' en `aplicarReemplazosGenerales` ahora es redundante si solo se busca el pipe,
        // pero es inofensiva si ya lo reemplazamos. Asegúrate de que no haya otras reglas en aplicarReemplazosGenerales
        // que interfieran negativamente.
        processedName = aplicarReemplazosGenerales(processedName); // Esto ya no afectaría el `|`
        //console.log(`[WME PLN] [Paso 5] Después de aplicar reemplazos generales: "${processedName}"`);

        //console.log(`[WME PLN] [Paso 6] Después de corregir tildes: "${processedName}"`);
        // ******************************************************************************

        // PASO FINAL: Mover palabras según la configuración de swap (inicio o final)
        processedName = applySwapMovement(processedName);

        //console.log(`[WME PLN] [Paso 7] Después de applyWordsToStartMovement: "${processedName}"`);

        let finalName = processedName.replace(/\s{2,}/g, ' ').trim();

        const trailingHyphenPattern = /\s*-\s*$/;
        if (trailingHyphenPattern.test(finalName) && !plnShouldPreserveTrailingHyphen(finalName)) {
            finalName = finalName.replace(trailingHyphenPattern, '');
        }

        // Quitar el punto final si existe.
        if (finalName.endsWith('.')) {
            finalName = finalName.slice(0, -1);
        }


        //console.log(`[WME PLN - pPN] Resultado final de pPN: "${finalName}" y processedName: "${processedName}"`); // LOG FINAL

        return finalName;
    }// processPlaceName
    // Permite minimizar el panel de estadísticas
    function minimizeStatsPanel() {
        if (statsPanelElement) {
            statsPanelElement.style.display = 'block';
            statsPanelElement.style.width = '120px';
            statsPanelElement.style.height = '28px';
            statsPanelElement.style.overflow = 'hidden';
            statsPanelElement.style.left = '8px';
            statsPanelElement.style.bottom = '8px';
            statsPanelElement.style.cursor = 'pointer';
            statsPanelElement.querySelector('#stats-summary-view').style.display = 'flex';
            statsPanelElement.querySelector('#stats-detail-view').style.display = 'none';
        }
    }// minimizeStatsPanel

    // Permite maximizar el panel de estadísticas
    function maximizeStatsPanel() {
        if (statsPanelElement) {
            statsPanelElement.style.display = 'block';
            statsPanelElement.style.width = '';
            statsPanelElement.style.height = '';
            statsPanelElement.style.overflow = '';
            statsPanelElement.style.left = '23%';
            statsPanelElement.style.bottom = '60px';
            statsPanelElement.style.cursor = 'pointer';
            statsPanelElement.querySelector('#stats-summary-view').style.display = 'flex';
        }
    }// maximizeStatsPanel
    //Permite normalizar una palabra individual
    // Versión MODIFICADA
    // FUNCIÓN MODIFICADA
    function updateApplyButtonState(row, originalName) {
        // Encontrar los elementos necesarios dentro de la fila
        const inputReplacement = row.querySelector('.replacement-input'); // Usaremos una clase para identificarlo
        const applyButton = row.querySelector('button[title="Aplicar sugerencia"]');
        const applyButtonWrapper = applyButton?.parentElement;
        if (!inputReplacement || !applyButton || !applyButtonWrapper) return;

        const nameIsDifferent = inputReplacement.value.trim() !== originalName.trim();
        const categoryWasChanged = row.dataset.categoryChanged === 'true';
        const addressWasChanged = row.dataset.addressChanged === 'true'; // <-- NUEVA COMPROBACIÓN

        // Si el nombre, la categoría O la dirección han cambiado, habilitar el botón
        if (nameIsDifferent || categoryWasChanged || addressWasChanged) {
            // Habilitar botón
            applyButton.disabled = false;
            applyButton.style.opacity = "1";
            // Quitar el chulo verde de éxito si existe
            const successIcon = applyButtonWrapper.querySelector('span');
            if (successIcon) {
                successIcon.remove();
            }
        }
        else {
            // Deshabilitar botón
            applyButton.disabled = true;
            applyButton.style.opacity = "0.5";
        }
    }// updateApplyButtonState

    //Permite aplicar reemplazos definidos por el usuario a un texto
    function aplicarReemplazosDefinidos(text, replacementRules) {
        let newText = text;

        if (typeof replacementRules !== 'object' || replacementRules === null || Object.keys(replacementRules).length === 0) {
            return newText;
        }

        const sortedFromKeys = Object.keys(replacementRules).sort((a, b) => b.length - a.length);

        for (const fromKey of sortedFromKeys) {
            const toValue = replacementRules[fromKey];

            // CORRECCIÓN: Asegurar que fromKey es una string antes de pasarla a escapeRegExp
            const escapedFromKey = escapeRegExp(String(fromKey));

            let regex;
            const wordCharSet = '[\\p{L}\\p{N}_-]';

            // CORRECCIÓN (28-nov-2024): Usar Lookbehind/Lookahead para evitar coincidencias parciales
            // Esto evita que "ie" coincida dentro de "Barbie" -> "BarbIE"
            // (?<![\p{L}\p{N}_-])  -> No precedido por letra, número, _ o -
            // (?![[\p{L}\p{N}_-])   -> No seguido por letra, número, _ o -

            if (toValue.endsWith(' -')) {
                // Caso especial para reglas que terminan en " -" (ej: "Texaco -")
                // Se mantiene la lógica de grupos para capturar espacios y palabra siguiente
                regex = new RegExp(`(^|[^\\p{L}\\p{N}_\\-])(${escapedFromKey})(\\s+)(${wordCharSet}+)?(?=$|[^\\p{L}\\p{N}_-])`, 'giu');
            }
            else {
                // Caso estándar: Usar Lookaround para límites de palabra estrictos
                regex = new RegExp(`(?<![\\p{L}\\p{N}_-])(${escapedFromKey})(?![\\p{L}\\p{N}_-])`, 'giu');
            }

            // CORRECCIÓN CLAVE: Usar la sintaxis '...args' para capturar todos los argumentos
            // y luego extraerlos de forma robusta.
            newText = newText.replace(regex, (match, ...args) => {
                // El último argumento de `args` es la cadena original completa.
                // El penúltimo argumento de `args` es el offset.
                const originalString = args[args.length - 1]; // Captura el string original
                const offset = args[args.length - 2];       // Captura el offset

                // Los grupos de captura vienen antes del offset y originalString.
                // Reasignar los grupos de captura según el tipo de regex.
                let delimitadorPrevio, matchedFromKey, capturedSpaces, nextWordIfCaptured;

                if (toValue.endsWith(' -')) {
                    // Para la regex con 4 grupos de captura (lógica antigua mantenida para este caso)
                    delimitadorPrevio = args[0]; // p1
                    matchedFromKey = args[1];    // p2
                    capturedSpaces = args[2];    // p3
                    nextWordIfCaptured = args[3]; // p4
                } else {
                    // Para la regex con Lookaround (NUEVA LÓGICA)
                    // match es la palabra completa encontrada (ej: "ie")
                    // No hay grupos de captura extraños al inicio
                    delimitadorPrevio = ""; // No capturamos el delimitador con lookbehind
                    matchedFromKey = match;
                }

                const offsetOfMatchInCurrentText = offset;
                const stringBeingProcessedActual = originalString; // Ya es la cadena correcta

                // --- Lógica Anti-Duplicación de palabra anterior ---
                const textoAntesDelMatch = stringBeingProcessedActual.substring(0, offsetOfMatchInCurrentText + delimitadorPrevio.length);
                const palabrasAntes = textoAntesDelMatch.trim().split(/\s+/);
                const ultimaPalabraAntes = palabrasAntes.length > 0 ? palabrasAntes[palabrasAntes.length - 1] : "";
                const palabrasDelReemplazo = toValue.trim().split(/\s+/);
                const primeraPalabraReemplazo = palabrasDelReemplazo.length > 0 ? palabrasDelReemplazo[0] : "";
                if (ultimaPalabraAntes && primeraPalabraReemplazo) {
                    const semejanza = calculateSimilarity(ultimaPalabraAntes, primeraPalabraReemplazo);
                    if (semejanza > 0.9) {
                        return match;
                    }
                }

                // --- Evitar que reglas que quitan guiones actúen cuando sigue otra palabra ---
                try {
                    const trimmedFromKey = String(fromKey || '').trim();
                    const trimmedToValue = String(toValue || '').trim();
                    const afterMatchSlice = stringBeingProcessedActual.slice(offsetOfMatchInCurrentText + match.length);
                    const nextVisibleChar = afterMatchSlice.replace(/^\s+/, '').charAt(0);
                    if (
                        trimmedFromKey.endsWith('-') &&
                        !trimmedToValue.endsWith('-') &&
                        nextVisibleChar &&
                        /[\p{L}\p{N}]/u.test(nextVisibleChar)
                    ) {
                        return match;
                    }
                } catch (_) { /* noop */ }

                // --- Lógica para evitar auto-reemplazo infinito y duplicación de frases ---
                // (ej: "Terpel" -> "Terpel -", "Tostao" -> "Tostao' Café & Pan")
                if (toValue.toLowerCase().startsWith(fromKey.toLowerCase()) && toValue.length > fromKey.length) {
                    const suffix = toValue.substring(fromKey.length).trim();
                    const textAfterMatchRaw = stringBeingProcessedActual.substring(offsetOfMatchInCurrentText + match.length);

                    // Función local para normalizar (sin tildes, minúsculas, espacios simples)
                    const normalizeForCheck = (s) => {
                        return removeDiacritics(String(s || '')).toLowerCase().replace(/\s+/g, ' ').trim();
                    };

                    const suffixNorm = normalizeForCheck(suffix);
                    const textAfterNorm = normalizeForCheck(textAfterMatchRaw);

                    // 1. Chequeo exacto (starts with)
                    if (textAfterNorm.startsWith(suffixNorm)) {
                        return match;
                    }

                    // 2. Chequeo difuso (Fuzzy) para casos como "Café & Pan" vs "Café y Pan"
                    // Tomamos un substring del texto posterior de longitud similar al sufijo
                    const candidate = textAfterNorm.substring(0, suffixNorm.length + 2); // +2 de margen
                    if (candidate.length >= suffixNorm.length * 0.5) { // Solo si tiene una longitud comparable
                        const similarity = calculateSimilarity(suffixNorm, candidate);
                        // Si es muy similar (> 0.75), asumimos que es lo mismo y no reemplazamos
                        if (similarity > 0.75) {
                            return match;
                        }
                    }
                }

                // --- Lógica específica para el reemplazo que termina en ' -' ---
                // FIX (28-nov-2024): Capitalizar la primera letra de la palabra siguiente al guion
                // Esto asegura que "ara el estadio" con regla "ara" → "Tiendas ARA -"
                // produzca "Tiendas ARA - El Estadio" (no "Tiendas ARA - el Estadio")
                if (toValue.endsWith(' -')) {
                    let capitalizedNext = nextWordIfCaptured || '';
                    if (capitalizedNext && capitalizedNext.length > 0) {
                        capitalizedNext = capitalizedNext.charAt(0).toUpperCase() + capitalizedNext.slice(1);
                    }
                    return delimitadorPrevio + toValue + capitalizedNext;
                }
                // --- Para otros reemplazos que no terminan en ' -' ---
                return delimitadorPrevio + toValue;
            });
        }
        newText = plnFixDanglingHyphens(newText);
        return newText;
    }//aplicarReemplazosDefinidos

    //Permite crear un panel flotante en WME
    function getVisiblePlaces() {
        if (typeof W === 'undefined' || !W.map || !W.model || !W.model.venues) {// Si Waze Map Editor no está completamente cargado, retornar un array vacío
            console.warn('[WME_PLN][WARNING] Waze Map Editor no está completamente cargado.');
            return [];
        }
        // Obtener los lugares visibles en el mapa
        const venues = W.model.venues.objects;
        const visiblePlaces = Object.values(venues).filter(venue => { // Filtrar los lugares que están visibles en el mapa
            const olGeometry = venue.getOLGeometry?.();// Obtener la geometría del lugar
            const bounds = olGeometry?.getBounds?.(); // Obtener los límites del lugar
            return bounds && W.map.getExtent().intersectsBounds(bounds);
        });
        return visiblePlaces;
    }// getVisiblePlaces

    // Devuelve la lista de places sin los excluidos y el total omitido.
    function filterOutExcludedPlaces(placesList) {
        if (!Array.isArray(placesList) || placesList.length === 0)
            return { filtered: [], excludedCount: 0 };

        const filtered = [];
        let excludedCount = 0;

        for (const place of placesList) {
            try {
                const placeId = place?.getID?.();
                if (placeId && excludedPlaces.has(placeId)) {
                    excludedCount++;
                    continue;
                }
            }
            catch (err) {
                console.warn('[WME PLN] Error comprobando lugar excluido:', err);
            }
            filtered.push(place);
        }

        return { filtered, excludedCount };
    }// filterOutExcludedPlaces

    // Filtra la lista de lugares para devolver solo aquellos con update requests pendientes
    function filterByUpdateRequests(placesList) {
        if (!Array.isArray(placesList) || placesList.length === 0) {
            return { filtered: [], filteredOutCount: 0 };
        }

        const filtered = [];
        let filteredOutCount = 0;

        for (const place of placesList) {
            try {
                const placeId = place?.getID?.();
                if (!placeId) {
                    filteredOutCount++;
                    continue;
                }

                // Usar W.model.venues.getObjectById en lugar del SDK
                let venueObj = null;
                if (W?.model?.venues?.getObjectById) {
                    try {
                        venueObj = W.model.venues.getObjectById(placeId);
                    } catch (err) {
                        // Error silencioso
                    }
                }

                // Verificar si tiene update requests pendientes
                // La ruta correcta es: attributes.venueUpdateRequests
                let hasPendingEdits = false;
                if (venueObj?.attributes?.venueUpdateRequests) {
                    const venueUpdateRequests = venueObj.attributes.venueUpdateRequests;
                    if (Array.isArray(venueUpdateRequests) && venueUpdateRequests.length > 0) {
                        hasPendingEdits = true;
                    }
                }

                if (hasPendingEdits) {
                    filtered.push(place);
                } else {
                    filteredOutCount++;
                }
            } catch (err) {
                filteredOutCount++;
            }
        }

        return { filtered, filteredOutCount };
    }// filterByUpdateRequests

    //Permite renderizar los lugares en el panel flotante
    function renderPlacesInFloatingPanel(places, scanStats = {}) {
        // Limpiar la lista global de duplicados antes de llenarla de nuevo
        placesForDuplicateCheckGlobal.length = 0;
        createFloatingPanel("processing"); // Mostrar panel en modo "procesando"
        const maxPlacesToScan = parseInt(document.getElementById("maxPlacesInput")?.value || "100", 10);  //Obtiene el número total de lugares a procesar

        const totalVisibleProvided = typeof scanStats.totalVisibleCount === 'number' ? scanStats.totalVisibleCount : null;
        const excludedProvided = typeof scanStats.excludedCount === 'number' ? scanStats.excludedCount : null;
        const skipExcludedFiltering = scanStats.skipExcludedFiltering === true;

        let workingPlaces = Array.isArray(places) ? [...places] : [];
        const initialVisiblePlacesCount = totalVisibleProvided ?? workingPlaces.length;

        let excludedByInitialFilterCount = excludedProvided ?? 0;

        if (!skipExcludedFiltering) {
            const { filtered, excludedCount } = filterOutExcludedPlaces(workingPlaces);
            workingPlaces = filtered;
            if (excludedProvided === null)
                excludedByInitialFilterCount = excludedCount;
        }
        else if (excludedProvided === null) {
            excludedByInitialFilterCount = Math.max(0, initialVisiblePlacesCount - workingPlaces.length);
        }

        const filteredPlacesCount = workingPlaces.length;
        const limitedPlacesCount = Math.min(filteredPlacesCount, maxPlacesToScan);

        if (filteredPlacesCount > maxPlacesToScan) // Limitar el número de lugares a escanear
            workingPlaces = workingPlaces.slice(0, maxPlacesToScan); // Limitar el número de places a escanear

        places = workingPlaces;
        const lockRankEmojis = ["0️⃣", "1️⃣", "2️⃣", "3️⃣", "4️⃣", "5️⃣", "6️⃣"]; // Definir los emojis de nivel de bloqueo
        // Permite obtener el nombre de la categoría de un lugar, ya sea del modelo antiguo o del SDK

        function getPlaceCategoryName(venueFromOldModel, venueSDKObject) { // Acepta ambos tipos de venue
            let categoryId = null;
            let categoryName = null;
            // Intento 1: Usar el venueSDKObject si está disponible y tiene la info
            if (venueSDKObject) {
                if (venueSDKObject.mainCategory && venueSDKObject.mainCategory.id) {// Si venueSDKObject tiene mainCategory con ID
                    categoryId = venueSDKObject.mainCategory.id; // source = "SDK (mainCategory.id)";
                    //Limpiar comillas aquí
                    if (typeof categoryId === 'string') categoryId = categoryId.replace(/'/g, '');
                    if (venueSDKObject.mainCategory.name) // Si mainCategory tiene nombre
                        categoryName = venueSDKObject.mainCategory.name;// source = "SDK (mainCategory.name)";
                    if (typeof categoryName === 'string') categoryName = categoryName.replace(/'/g, '');
                }
                else if (Array.isArray(venueSDKObject.categories) && venueSDKObject.categories.length > 0) {// Si venueSDKObject tiene un array de categorías y al menos una categoría
                    const firstCategorySDK = venueSDKObject.categories[0]; // source = "SDK (categories[0])";
                    if (typeof firstCategorySDK === 'object' && firstCategorySDK.id) {// Si la primera categoría es un objeto con ID
                        categoryId = firstCategorySDK.id;
                        // Limpiar comillas aquí
                        if (typeof categoryId === 'string') categoryId = categoryId.replace(/'/g, '');

                        if (firstCategorySDK.name)  // Si la primera categoría tiene nombre
                            categoryName = firstCategorySDK.name;
                        if (typeof categoryName === 'string') categoryName = categoryName.replace(/'/g, '');
                    }
                    else if (typeof firstCategorySDK === 'string') // Si la primera categoría es una cadena (nombre de categoría)
                    {
                        categoryName = firstCategorySDK;
                        if (typeof categoryName === 'string') categoryName = categoryName.replace(/'/g, '');
                    }
                }
                else if (venueSDKObject.primaryCategoryID) {
                    categoryId = venueSDKObject.primaryCategoryID;
                    if (typeof categoryName === 'string') categoryName = categoryName.replace(/'/g, '');
                }
            }
            if (categoryName) {// Si se obtuvo el nombre de categoría del SDK

                return categoryName;
            }
            // Intento 2: Usar W.model si no se obtuvo del SDK
            if (!categoryId && venueFromOldModel && venueFromOldModel.attributes && Array.isArray(venueFromOldModel.attributes.categories) && venueFromOldModel.attributes.categories.length > 0)
                categoryId = venueFromOldModel.attributes.categories[0];
            if (!categoryId)// Si no se pudo obtener el ID de categoría de ninguna fuente
                return "Sin categoría";
            let categoryObjWModel = null; // Intentar obtener el objeto de categoría del modelo Waze
            if (typeof W !== 'undefined' && W.model) {// Si Waze Map Editor está disponible
                if (W.model.venueCategories && typeof W.model.venueCategories.getObjectById === "function") // Si venueCategories está disponible en W.model
                    categoryObjWModel = W.model.venueCategories.getObjectById(categoryId);
                if (!categoryObjWModel && W.model.categories && typeof W.model.categories.getObjectById === "function") // Si no se encontró en venueCategories, intentar en categories
                    categoryObjWModel = W.model.categories.getObjectById(categoryId);
            }
            if (categoryObjWModel && categoryObjWModel.attributes && categoryObjWModel.attributes.name) {// Si se encontró el objeto de categoría en W.model
                let nameToReturn = categoryObjWModel.attributes.name;
                //  Limpiar comillas aquí
                if (typeof nameToReturn === 'string') nameToReturn = nameToReturn.replace(/'/g, '');
                return nameToReturn;
            }
            if (typeof categoryId === 'number' || (typeof categoryId === 'string' && categoryId.trim() !== '')) {// Si no se pudo obtener el nombre de categoría de ninguna fuente, devolver el ID
                return `${categoryId}`; // Devuelve el ID si no se encuentra el nombre.
            }
            return "Sin categoría";
        }//getPlaceCategoryName

        // Convierte un lugar de tipo Punto a Área (Polígono)
        // TODO: TEMPORALMENTE DESHABILITADO - La API de WME cambió y necesita investigación
        async function convertPointToArea(venueId) {
            // Mostrar mensaje informativo en lugar de intentar la conversión
            if (typeof plnToast === 'function') {
                plnToast("⚠️ Conversión punto→área temporalmente no disponible (API en revisión)", 3000);
            }
            console.warn("[WME PLN] convertPointToArea: Funcionalidad temporalmente deshabilitada - API de geometría WME en revisión");
            return;
            
            /* CÓDIGO ORIGINAL COMENTADO PARA INVESTIGACIÓN:
            try {
                if (typeof W === 'undefined' || !W.model || !W.model.venues) return;
                const venue = W.model.venues.getObjectById(venueId);
                if (!venue) {
                    console.error("[WME PLN] Venue no encontrado:", venueId);
                    return;
                }

                const olGeometry = venue.getOLGeometry();
                if (!olGeometry) {
                    console.error("[WME PLN] No se pudo obtener la geometría OL del venue");
                    return;
                }
                
                const center = olGeometry.getCentroid();
                const x = center.x;
                const y = center.y;
                const offset = 10;

                // INVESTIGAR: ¿Qué formato espera UpdateFeatureGeometry ahora?
                // - OpenLayers.Geometry.Polygon?
                // - GeoJSON object?
                // - Algo más?
                
            } catch (e) {
                console.error("[WME PLN] Error converting to area", e);
            }
            */
        }

        //Permite obtener el tipo de lugar (área o punto) y su icono
        function getPlaceTypeInfo(venueSDKObject) // <--- AHORA RECIBE venueSDKObject
        {
            let isArea = false;
            let icon = "⊙"; // Icono por defecto para punto
            let title = "Punto"; // Título por defecto para punto

            if (venueSDKObject && venueSDKObject.geometry && venueSDKObject.geometry.type) {
                const geometryType = venueSDKObject.geometry.type;
                if (geometryType === 'Polygon' || geometryType === 'MultiPolygon') {
                    isArea = true;
                    icon = "⭔"; // Icono para área
                    title = "Área"; // Título para área
                }
                // Para otros tipos como 'Point', 'LineString', etc., se mantienen los valores por defecto (Punto).
            }
            return { isArea, icon, title };
        }// getPlaceTypeInfo

        //Permite procesar un lugar y generar un objeto con sus detalles
        function shouldForceSuggestionForReview(word) {
            if (typeof word !== 'string') // Si la palabra no es una cadena, no forzar sugerencia por esta regla
                return false;
            const lowerWord = word.toLowerCase(); // Convertir la palabra a minúsculas para evitar problemas de mayúsculas/minúsculas
            const hasTilde = /[áéíóúÁÉÍÓÚ]/.test(word); // Verificar si la palabra tiene alguna tilde (incluyendo mayúsculas acentuadas)
            if (!hasTilde)  // Si no tiene tilde, no forzar sugerencia por esta regla
                return false; // Si no hay tilde, no forzar sugerencia por esta regla
            const problematicSubstrings = ['c', 's', 'x', 'cc', 'sc', 'cs', 'g', 'j', 'z', 'ñ']; // Lista de patrones de letras/combinaciones que, junto con una tilde, fuerzan la sugerencia (insensible a mayúsculas debido a lowerWord)
            for (const sub of problematicSubstrings) {// Verificar si la palabra contiene alguna de las letras/combinaciones problemáticas
                if (lowerWord.includes(sub))
                    return true; // Tiene tilde y una de las letras/combinaciones problemáticas
            }
            return false; // Tiene tilde, pero no una de las letras/combinaciones problemáticas
        }//shouldForceSuggestionForReview

        // Procesa un lugar y genera un objeto con sus detalles
        async function getPlaceCityInfo(venueFromOldeModel, venueSDKObject, currentCategoryKey = null) {
            let cityName = null;
            let source = "desconocida";

            // --- Función auxiliar para obtener el nombre de la ciudad desde un cityID ---
            function getCityNameFromId(cityId) {
                if (!cityId || typeof W === 'undefined' || !W.model || !W.model.cities || !W.model.cities.getObjectById) {
                    return null;
                }
                const cityObject = W.model.cities.getObjectById(cityId);
                if (cityObject && cityObject.attributes && typeof cityObject.attributes.name === 'string' && cityObject.attributes.name.trim() !== '') {
                    return cityObject.attributes.name.trim();
                }
                return null;
            }

            try {
                // --- INTENTO 1: OBTENER CIUDAD A TRAVÉS DEL streetID (EL MÉTODO MÁS COMÚN) ---
                const streetID = venueFromOldeModel?.attributes?.streetID || venueSDKObject?.address?.street?.id || null;

                if (streetID) {
                    const streetObject = W.model.streets.getObjectById(streetID);
                    if (streetObject && streetObject.attributes && streetObject.attributes.cityID) {
                        cityName = getCityNameFromId(streetObject.attributes.cityID);
                        if (cityName) {
                            source = "derivada de la calle";
                        }
                    }
                }

                // --- INTENTO 2: OBTENER CIUDAD DESDE EL cityID DIRECTO (FALLBACK) ---
                if (!cityName) {
                    const cityID = venueFromOldeModel?.attributes?.cityID || venueSDKObject?.address?.city?.id || null;
                    if (cityID) {
                        cityName = getCityNameFromId(cityID);
                        if (cityName) {
                            source = "explícita";
                        }
                    }
                }

            } catch (e) {
                console.error("[WME PLN] Error crítico en getPlaceCityInfo:", e);
                cityName = null;
            }

            // ✨ LÓGICA ESPECIAL PARA RÍOS/ARROYOS (RIVER_STREAM)
            if (currentCategoryKey === 'RIVER_STREAM') {
                if (cityName) {
                    // ⚠️ Río/Arroyo NO debe tener ciudad
                    return {
                        icon: '⚠️',
                        title: `⚠️ Río/Arroyo no debe tener ciudad (${cityName}) - Click para limpiar`,
                        hasCity: true,
                        hasInvalidCity: true  // Nueva propiedad para identificar este caso
                    };
                } else {
                    // ✅ Río/Arroyo sin ciudad = correcto
                    return {
                        icon: '✅',
                        title: '✓ Río/Arroyo sin ciudad (Correcto)',
                        hasCity: false,
                        hasValidNoCity: true  // Nueva propiedad para identificar este caso
                    };
                }
            }

            // --- Lógica final para devolver el ícono y el título (PARA OTROS TIPOS) ---
            if (cityName) {
                return {
                    icon: '✅',
                    title: `Ciudad: ${cityName} (${source})`,
                    hasCity: true
                };
            } else {
                return {
                    icon: '🚩',
                    title: "Sin ciudad asignada. Haz clic para buscar y asignar la más cercana.",
                    hasCity: false
                };
            }
        }// getPlaceCityInfo

        /*async function getPlaceCityInfo(venueFromOldModel, venueSDKObject)
        {
            let hasExplicitCity = false; // Indica si hay una ciudad explícita definida
            let explicitCityName = null; // Nombre de la ciudad explícita, si se encuentra
            let hasStreetInfo = false; // Indica si hay información de calle disponible
            let cityAssociatedWithStreet = null; // Nombre de la ciudad asociada a la calle, si se encuentra
            // 1. Check for EXPLICIT city  SDK
            if (venueSDKObject && venueSDKObject.address) {
            //console.log("[DEBUG] venueSDKObject.address:", venueSDKObject.address);

                if (venueSDKObject.address.city && typeof venueSDKObject.address.city.name === 'string' && venueSDKObject.address.city.name.trim() !== '') {
                    // Si hay una ciudad explícita en el SDK
                    explicitCityName = venueSDKObject.address.city.name.trim(); // Nombre de la ciudad explícita
                    hasExplicitCity = true; // source = "SDK (address.city.name)";
                //console.log("[DEBUG] Ciudad explícita encontrada en SDK (address.city.name):", explicitCityName);
                } else if (typeof venueSDKObject.address.cityName === 'string' && venueSDKObject.address.cityName.trim() !== '') {
                    // Si hay una ciudad explícita en el SDK (cityName)
                    explicitCityName = venueSDKObject.address.cityName.trim(); // Nombre de la ciudad explícita
                    hasExplicitCity = true; // source = "SDK (address.cityName)";
                //console.log("[DEBUG] Ciudad explícita encontrada en SDK (address.cityName):", explicitCityName);
                } else {
                //console.log("[DEBUG] No se encontró ciudad explícita en SDK.");
                }
            }

            if (!hasExplicitCity && venueFromOldModel && venueFromOldModel.attributes)
            {
                //console.log("[DEBUG] venueFromOldModel.attributes:", venueFromOldModel.attributes);

                    const cityID = venueFromOldModel.attributes.cityID;
                //console.log("[DEBUG] cityID del modelo antiguo:", cityID);

                if (cityID && typeof W !== 'undefined' && W.model && W.model.cities && W.model.cities.getObjectById)
             {
                    //console.log("[DEBUG] Intentando obtener el objeto de ciudad con cityID:", cityID);

                        const cityObject = W.model.cities.getObjectById(cityID); // Obtener el objeto de ciudad del modelo Waze
                    //console.log("[DEBUG] cityObject obtenido:", cityObject);

                        if (cityObject && cityObject.attributes && typeof cityObject.attributes.name === 'string' && cityObject.attributes.name.trim() !== '') {
                            // Si el objeto de ciudad tiene un nombre válido
                            explicitCityName = cityObject.attributes.name.trim(); // Nombre de la ciudad explícita
                            hasExplicitCity = true; // source = "W.model.cities (cityID)";
                        //console.log("[DEBUG] Ciudad explícita encontrada en modelo antiguo (cityID):", explicitCityName);
                        } else {
                        //console.log("[DEBUG] cityObject no tiene un nombre válido.");
                        }
                    }
                    else
                    {
                    //console.log("[DEBUG] cityID no válido o W.model.cities.getObjectById no disponible.");
                    }
            }

            // 2. Check for STREET information (and any city derived from it) // SDK street check
            if (venueSDKObject && venueSDKObject.address)
                if ((venueSDKObject.address.street && typeof venueSDKObject.address.street.name === 'string' && venueSDKObject.address.street.name.trim() !== '') ||
                    (typeof venueSDKObject.address.streetName === 'string' && venueSDKObject.address.streetName.trim() !== ''))
                    hasStreetInfo = true; // source = "SDK (address.street.name or streetName)";
            if (venueFromOldModel && venueFromOldModel.attributes && venueFromOldModel.attributes.streetID)
            {// Old Model street check (if not found via SDK or to supplement)
                hasStreetInfo = true; // Street ID exists in old model
                const streetID = venueFromOldModel.attributes.streetID; // Obtener el streetID del modelo antiguo
                if (typeof W !== 'undefined' && W.model && W.model.streets && W.model.streets.getObjectById)
                {// Si hay un streetID en el modelo antiguo
                    const streetObject = W.model.streets.getObjectById(streetID); // Obtener el objeto de calle del modelo Waze
                    if (streetObject && streetObject.attributes && streetObject.attributes.cityID)
                    {// Si el objeto de calle tiene un cityID asociado
                        const cityIDFromStreet = streetObject.attributes.cityID;// Obtener el cityID de la calle
                        if (W.model.cities && W.model.cities.getObjectById)
                        {// Si W.model.cities está disponible y tiene el método getObjectById
                            const cityObjectFromStreet = W.model.cities.getObjectById(cityIDFromStreet);// Obtener el objeto de ciudad asociado a la calle
                            // Si el objeto de ciudad tiene un nombre válido
                            if (cityObjectFromStreet && cityObjectFromStreet.attributes && typeof cityObjectFromStreet.attributes.name === 'string' && cityObjectFromStreet.attributes.name.trim() !== '')
                                cityAssociatedWithStreet = cityObjectFromStreet.attributes.name.trim(); // Nombre de la ciudad asociada a la calle
                        }
                    }
                }
            }
            // --- 3. Determine icon, title, and returned hasCity based on user's specified logic ---
            let icon;
            let title;
            const returnedHasCityBoolean = hasExplicitCity; // To be returned, indicates if an *explicit* city is set.
            const hasAnyAddressInfo = hasExplicitCity || hasStreetInfo; // Determina si hay alguna información de dirección (ciudad explícita o calle).
            if (hasAnyAddressInfo)
            {// Si hay información de dirección (ciudad explícita o calle)
                if (hasExplicitCity)
                {
                    // Tiene ciudad explícita
                    icon = "🏙️";
                    title = `Ciudad: ${explicitCityName}`;
                }
                else if (cityAssociatedWithStreet)
                {
                    // No tiene ciudad explícita, pero la calle sí está asociada a ciudad
                    icon = "🏙️";
                    title = `Ciudad (por calle): ${cityAssociatedWithStreet}`;
                }
                else {
                    // No hay ciudad explícita ni ciudad por calle
                    icon = "🚫";
                    title = "Sin ciudad asignada";
                }
                return {
                    icon: icon || "❓",
                    title: title || "Info no disponible",
                    hasCity: (hasExplicitCity || !!cityAssociatedWithStreet) // Ahora true si tiene ciudad por calle
                };
            }
            else
            { // No tiene ni ciudad explícita ni información de calle
                icon = "🚫";
                title = "El campo dirección posee inconsistencias"; // Título para "no tiene ciudad ni calle"
            }
            return {
                icon: icon || "❓", // Usar '?' si icon es undefined/null/empty
                title: title || "Info no disponible", // Usar "Info no disponible" si title es undefined/null/empty
                hasCity: returnedHasCityBoolean || false // Asegurarse de que sea un booleano
            };
        }*///getPlaceCityInfo

        //Renderizar barra de progreso en el TAB PRINCIPAL justo después del slice
        const tabOutput = document.querySelector("#wme-normalization-tab-output");
        if (tabOutput) {// Si el tab de salida ya existe, limpiar su contenido
            // Reiniciar el estilo del mensaje en el tab al valor predeterminado
            tabOutput.style.color = "#000";
            tabOutput.style.fontWeight = "normal";
            // Crear barra de progreso visual
            const progressBarWrapperTab = document.createElement("div");
            progressBarWrapperTab.style.margin = "10px 0";
            progressBarWrapperTab.style.marginTop = "10px";
            progressBarWrapperTab.style.height = "18px";
            progressBarWrapperTab.style.backgroundColor = "transparent";
            // Crear el contenedor de la barra de progreso
            const progressBarTab = document.createElement("div");
            progressBarTab.style.height = "100%";
            progressBarTab.style.width = "0%";
            progressBarTab.style.backgroundColor = "#007bff";
            progressBarTab.style.transition = "width 0.2s";
            progressBarTab.id = "progressBarInnerTab";
            progressBarWrapperTab.appendChild(progressBarTab);
            // Crear texto de progreso
            const progressTextTab = document.createElement("div");
            progressTextTab.style.fontSize = "12px";
            progressTextTab.style.marginTop = "5px";
            progressTextTab.id = "progressBarTextTab";
            tabOutput.appendChild(progressBarWrapperTab);
            tabOutput.appendChild(progressTextTab);
        }
        // Asegurar que la barra de progreso en el tab se actualice desde el principio
        const progressBarInnerTab = document.getElementById("progressBarInnerTab"); // Obtener la barra de progreso del tab
        const progressBarTextTab = document.getElementById("progressBarTextTab"); // Obtener el texto de progreso del tab
        if (progressBarInnerTab && progressBarTextTab) {// Si ambos elementos existen, reiniciar su estado
            progressBarInnerTab.style.width = "0%";
            progressBarTextTab.textContent = `Progreso: 0% (0/${places.length})`; // Reiniciar el texto de progreso
        }
        // --- PANEL FLOTANTE: limpiar y preparar salida ---
        const output = document.querySelector("#wme-place-inspector-output");//
        if (!output) {// Si el panel flotante no está disponible, mostrar un mensaje de error
            console.error("[WME_PLN][ERROR]❌ Panel flotante no está disponible");
            return;
        }
        output.innerHTML = ""; // Limpia completamente el contenido del panel flotante
        if (window.WME_PLN_SDK && window.WME_PLN_SDK.UI && typeof window.WME_PLN_SDK.UI.createElement === 'function') {
            // Usar el SDK de WME para crear el div de procesamiento
            const sdkDiv = window.WME_PLN_SDK.UI.createElement('div', {
                style: {
                    display: 'flex',
                    alignItems: 'center',
                    gap: '10px'
                }
            });

            const spinner = window.WME_PLN_SDK.UI.createElement('span', {
                className: 'loader-spinner',
                style: {
                    width: '16px',
                    height: '16px',
                    border: '2px solid #ccc',
                    borderTop: '2px solid #007bff',
                    borderRadius: '50%',
                    animation: 'spin 0.8s linear infinite'
                }
            });

            const infoDiv = window.WME_PLN_SDK.UI.createElement('div');
            const processingText = window.WME_PLN_SDK.UI.createElement('div', {
                id: 'processingText'
            });
            processingText.textContent = 'Procesando lugares visibles';

            const dotsSpan = window.WME_PLN_SDK.UI.createElement('span', { className: 'dots' });
            dotsSpan.textContent = '.';

            processingText.appendChild(dotsSpan);

            const processingStep = window.WME_PLN_SDK.UI.createElement('div', {
                id: 'processingStep',
                style: {
                    fontSize: '13px',
                    color: '#555'
                }
            });
            processingStep.textContent = 'Inicializando escaneo...';

            infoDiv.appendChild(processingText);
            infoDiv.appendChild(processingStep);

            sdkDiv.appendChild(spinner);
            sdkDiv.appendChild(infoDiv);

            output.appendChild(sdkDiv);
        } else {
            // Fallback al HTML tradicional si el SDK no está disponible
            output.innerHTML = "<div style='display:flex; align-items:center; gap:10px;'><span class='loader-spinner' style='width:16px; height:16px; border:2px solid #ccc; border-top:2px solid #007bff; border-radius:50%; animation:spin 0.8s linear infinite;'></span><div><div id='processingText'>Procesando lugares visibles<span class='dots'>.</span></div><div id='processingStep' style='font-size:13px; color:#555;'>Inicializando escaneo...</div></div></div>";
        }
        //output.innerHTML = "<div style='display:flex; align-items:center; gap:10px;'><span class='loader-spinner' style='width:16px; height:16px; border:2px solid #ccc; border-top:2px solid #007bff; border-radius:50%; animation:spin 0.8s linear infinite;'></span><div><div id='processingText'>Procesando lugares visibles<span class='dots'>.</span></div><div id='processingStep' style='font-size:13px; color:#555;'>Inicializando escaneo...</div></div></div>";
        // Asegurar que el panel flotante tenga un alto mínimo
        const processingStepLabel = document.getElementById("processingStep");
        if (processingStepLabel) {
            const infoParts = [`Visibles: ${initialVisiblePlacesCount}`];
            // Agregar información adicional si está disponible
            if (excludedByInitialFilterCount > 0)
                infoParts.push(`Excluidos que serán omitidos: ${excludedByInitialFilterCount}`);
            infoParts.push(`Restantes tras excluir: ${filteredPlacesCount}`);
            infoParts.push(`Se analizarán: ${limitedPlacesCount}`);
            processingStepLabel.textContent = infoParts.join(' · ');
        }
        // Animación de puntos suspensivos
        const dotsSpan = output.querySelector(".dots");
        if (dotsSpan) {// Si el span de puntos existe, iniciar la animación de puntos
            const dotStates = ["", ".", "..", "..."];
            let dotIndex = 0;
            window.processingDotsInterval = setInterval(() => {
                dotIndex = (dotIndex + 1) % dotStates.length;
                dotsSpan.textContent = dotStates[dotIndex];
            }, 500);
        }
        output.style.height = "calc(55vh - 40px)";
        if (!places.length) {// Si no hay places, mostrar mensaje y salir
            output.appendChild(document.createTextNode("No hay places visibles para analizar."));
            const existingOverlay = document.getElementById("scanSpinnerOverlay");
            if (existingOverlay)// Si ya existe un overlay de escaneo, removerlo
                existingOverlay.remove();
            
            // --- FIX: Resetear UI para evitar que se quede pegado ---
            if (window.processingDotsInterval) {
                clearInterval(window.processingDotsInterval);
                window.processingDotsInterval = null;
            }
            if (scanBtn) {
             scanTitle.innerText = "Análisis de Nombres de Places";
             scanBtn.innerHTML = "Escanear";
             scanBtn.disabled = false;
             scanBtn.style.backgroundColor = "#2196F3"; // Color original
             scanBtn.style.cursor = "pointer";
            }
            // ----------------------------------------------------
            return;
        }
        // Procesamiento incremental para evitar congelamiento
        let inconsistents = []; // Array para almacenar inconsistencias encontradas
        let index = 0; // Índice para iterar sobre los lugares
        const scanBtn = document.querySelector("button[type='button']"); // Remover ícono de ✔ previo si existe
        if (scanBtn) {// Si el botón de escaneo existe, remover el ícono de ✔ previo si está presente
            const existingCheck = scanBtn.querySelector("span");
            if (existingCheck) // Si hay un span dentro del botón, removerlo
                existingCheck.remove();
        }
        // --- Sugerencias por palabra global para toda la ejecución ---
        let sugerenciasPorPalabra = {};
        // Convertir excludedWords a array solo una vez al inicio del análisis, seguro ante undefined
        const excludedArray = (typeof excludedWords !== "undefined" && Array.isArray(excludedWords)) ? excludedWords : (typeof excludedWords !== "undefined" ? Array.from(excludedWords) : []);

        async function processNextPlace() {
            // ID del lugar actual que se está procesando
            const currentPlaceForLog = places[index];
            const currentVenueId = currentPlaceForLog ? currentPlaceForLog.getID() : 'ID Desconocido';

            //console.log(`\n[WME PLN - processNextPlace] --- INICIANDO PROCESAMIENTO PARA LUGAR ID: ${currentVenueId} (Índice: ${index}) ---`); // <--- USAR currentVenueId
            //console.log(`[WME PLN - processNextPlace] Total de lugares a procesar: ${places.length}`);

            // Inicialización de variables de estado
            let cityInfo = {
                icon: "❓",
                title: "Información de ciudad no disponible",
                hasCity: false
            };
            let resolvedEditorName = "Desconocido";
            let lastEditorIdForComparison = null;
            let currentLoggedInUserId = currentGlobalUserInfo.id;
            let wasEditedByMe = false;
            let shouldSkipThisPlace = false;
            let skipReasonLog = "";

            // Declaración de avoidMyEdits, typeInfo, areaMeters al inicio
            const avoidMyEdits = document.getElementById("chk-avoid-my-edits")?.checked ?? false; // <-- MOVIDO AQUÍ
            let typeInfo = { isArea: false, icon: "⊙", title: "Punto" };
            let areaMeters = null;

            // --- Obtener venueSDK lo antes posible ---
            let venueSDK = null;
            if (wmeSDK && wmeSDK.DataModel && wmeSDK.DataModel.Venues && wmeSDK.DataModel.Venues.getById)
                try {
                    venueSDK = await wmeSDK.DataModel.Venues.getById({ venueId: currentVenueId });
                }
                catch (sdkError) {
                    console.error(`[WME_PLN] Error al obtener venueSDK para ID ${currentVenueId}:`, sdkError);
                }

            let originalNameRaw;
            if (venueSDK && venueSDK.name)
                originalNameRaw = venueSDK.name;
            else
                originalNameRaw = currentPlaceForLog && currentPlaceForLog.attributes ? (currentPlaceForLog.attributes.name?.value || currentPlaceForLog.attributes.name || '') : '';
            originalNameRaw = originalNameRaw.trim();
            const nameForProcessing = removeEmoticons(originalNameRaw);

            // Asegurarse de que typeInfo y areaMeters se obtengan si venueSDK está disponible
            if (venueSDK) {
                typeInfo = getPlaceTypeInfo(venueSDK);
                areaMeters = calculateAreaMeters(venueSDK);
            }

            //console.log(`[WME PLN - processNextPlace] Nombre Original Raw (de Waze/SDK): "${originalNameRaw}"`);
            //console.log(`[WME PLN - processNextPlace] Nombre para Procesamiento (sin emojis): "${nameForProcessing}"`);
            //console.log(`[WME PLN - DEBUG] Place ID: ${currentVenueId}, Name: "${originalNameRaw}"`);
            //console.log(`[WME PLN - DEBUG]   -> typeInfo:`, typeInfo);
            //console.log(`[WME PLN - DEBUG]   -> areaMeters:`, areaMeters);

            //Obtener el ID del usuario actual (si está disponible globalmente de forma confiable)
            const useFullPipeline = true; // Siempre usar el pipeline completo para este flujo
            const applyGeneralReplacements = useFullPipeline || (document.getElementById("chk-general-replacements")?.checked ?? true); // Aplicar reemplazos generales por defecto
            const checkExcludedWords = useFullPipeline || (document.getElementById("chk-check-excluded")?.checked ?? false); // Verificar palabras excluidas por defecto
            const checkDictionaryWords = true;// Siempre verificar palabras del diccionario para este flujo
            const restoreCommas = document.getElementById("chk-restore-commas")?.checked ?? false;// Restaurar comas por defecto
            const similarityThreshold = parseFloat(document.getElementById("similarityThreshold")?.value || "85") / 100;//  Umbral de similitud por defecto (convertido a porcentaje)

            // 2. Condición de salida principal (todos los lugares procesados)
            if (index >= places.length) {

                finalizeRender(inconsistents, places, sugerenciasPorPalabra);
                return;
            }
            // 1. Verificar si el lugar actual es válido y tiene un ID
            const venueFromOldModel = places[index];
            const currentVenueNameObj = venueFromOldModel?.attributes?.name;
            const nameValue = typeof currentVenueNameObj === 'object' && currentVenueNameObj !== null && typeof currentVenueNameObj.value === 'string' ? currentVenueNameObj.value.trim() !== ''
                ? currentVenueNameObj.value : undefined : typeof currentVenueNameObj === 'string' && currentVenueNameObj.trim() !== '' ? currentVenueNameObj : undefined;
            //
            // Validación mejorada: Detectar lugares sin nombre
            // En lugar de saltar estos lugares, los procesamos y marcamos especialmente
            const hasEmptyName = !nameForProcessing || nameForProcessing.trim() === '';

            if (hasEmptyName) {
                // Verificar si es RESIDENCE_HOME para saltarlo
                const categories = venueFromOldModel?.attributes?.categories || [];
                if (categories.includes('RESIDENCE_HOME')) {
                    // console.log(`[WME_PLN] Saltando lugar residencial sin nombre: ${currentVenueId}`);
                    index++;
                    setTimeout(() => processNextPlace(), 0);
                    return;
                }

                console.warn(`[WME_PLN] ⚠️ Lugar sin nombre detectado en el índice ${index}:`, {
                    venueId: currentVenueId,
                    categories: categories,
                    originalNameRaw: originalNameRaw
                });
            }

            // 3. Continuar procesando incluso si no tiene nombre (para mostrarlo en el panel)

            // --- NUEVO: Filtro por Nivel de Bloqueo ---
            const maxLockRankInput = document.getElementById("maxLockRankInput");
            if (maxLockRankInput && maxLockRankInput.value) {
                const maxRank = parseInt(maxLockRankInput.value, 10);
                const placeLockRank = (venueFromOldModel && venueFromOldModel.attributes && venueFromOldModel.attributes.lockRank)
                    ? venueFromOldModel.attributes.lockRank
                    : 0; // Asumir 0 si no tiene lockRank definido

                // El lockRank en WME suele ser 0-based (0=L1, 1=L2, etc) o 1-based dependiendo de la propiedad.
                // venue.attributes.lockRank suele ser 0-based (0 es Auto/L1).
                // El input del usuario es 1-based (1 a 6).
                // Ajustamos: si el usuario pone 1, quiere ver hasta lockRank 0.
                // Si pone 2, hasta lockRank 1.
                // Entonces: si (placeLockRank + 1) > maxRank -> Saltar.

                if (!isNaN(maxRank) && (placeLockRank + 1) > maxRank) {
                    // Saltar silenciosamente o con log debug
                    // console.log(`[WME_PLN] Saltando lugar ${currentVenueId} por nivel de bloqueo (${placeLockRank + 1} > ${maxRank})`);
                    index++;
                    setTimeout(() => processNextPlace(), 0);
                    return;
                }
            }

            // Se usa la variable limpia de emojis para generar el nombre normalizado.
            const originalName = nameForProcessing; // 'originalName' ahora es explícitamente para el pipeline de procesamiento.
            const normalizedName = processPlaceName(originalName); // Normalizar el nombre del lugar
            //console.log(`[WME PLN - processNextPlace] Nombre Original para pipeline (sin emojis): "${originalName}"`);
            //console.log(`[WME PLN - processNextPlace] Nombre Normalizado (después de processPlaceName): "${normalizedName}"`);

            // 4. Verificar si el nombre ya está normalizado (sin emojis) y no requiere cambios

            const { lat: placeLat, lon: placeLon } = getPlaceCoordinates(venueFromOldModel, venueSDK); // Obtener las coordenadas del lugar
            // `isNameEffectivelyNormalized` debe calcularse DESPUÉS de `processPlaceName` y `aplicarReemplazosGenerales`
            // y todas las transformaciones que definen el `suggestedName` final.
            // Por lo tanto, esta línea no debería estar aquí para definir la variable `isNameEffectivelyNormalized` globalmente
            // o para las condiciones de salto que se evalúan ANTES de que `suggestedName` sea final.
            // Asegúrate de que `isNameEffectivelyNormalized` se calcula una vez al final, como en código original.

            // Lógica unificada y robusta para obtener resolvedEditorName, lastEditorIdForComparison y calcular wasEditedByMe
            resolvedEditorName = "Desconocido"; // Reinicializar para cada place
            lastEditorIdForComparison = null; // Reinicializar para cada place
            // Obtener el ID del usuario actual de forma robusta
            if (venueSDK && venueSDK.modificationData) {
                const updatedByDataFromSDK = venueSDK.modificationData.updatedBy;
                if (typeof updatedByDataFromSDK === 'string' && updatedByDataFromSDK.trim() !== '') {
                    resolvedEditorName = updatedByDataFromSDK; // El nombre del editor es una cadena
                }
                else if (typeof updatedByDataFromSDK === 'number') {
                    lastEditorIdForComparison = updatedByDataFromSDK; // El ID numérico es la fuente principal
                    resolvedEditorName = `ID ${updatedByDataFromSDK}`; // Nombre temporal
                    if (W && W.model && W.model.users) {
                        const userObjectW = W.model.users.getObjectById(updatedByDataFromSDK);
                        if (userObjectW && userObjectW.userName) { // Si el usuario está en el modelo Waze
                            resolvedEditorName = userObjectW.userName; // Obtener nombre real del usuario si está en el modelo
                        }
                    }
                }
            }
            else if (venueFromOldModel && venueFromOldModel.attributes && (venueFromOldModel.attributes.updatedBy !== null && venueFromOldModel.attributes.updatedBy !== undefined)) {
                // Fallback al modelo antiguo si el SDK no dio datos de editor
                const oldModelUpdatedBy = venueFromOldModel.attributes.updatedBy;
                lastEditorIdForComparison = oldModelUpdatedBy; // El ID numérico es la fuente principal
                resolvedEditorName = `ID ${oldModelUpdatedBy}`; // Nombre temporal
                if (W && W.model && W.model.users) {
                    const userObjectW = W.model.users.getObjectById(oldModelUpdatedBy);
                    if (userObjectW && userObjectW.userName) {
                        resolvedEditorName = userObjectW.userName; // Obtener nombre real del usuario si está en el modelo
                    }
                }
            }
            else {
                resolvedEditorName = "N/D"; // No hay información de editor
            }
            // Calcular wasEditedByMe de forma robusta aquí mismo
            wasEditedByMe = false; // Resetear para este place
            if (currentLoggedInUserId !== null && currentLoggedInUserId !== undefined && resolvedEditorName !== "N/D") { // Solo si tenemos un nombre de usuario logueado y el resolvedEditorName no es N/D
                if (lastEditorIdForComparison !== null && lastEditorIdForComparison !== undefined && typeof lastEditorIdForComparison === 'number') {
                    // PRIORIDAD 1: Comparar IDs numéricos si ambos están disponibles y son válidos
                    if (typeof currentLoggedInUserId === 'number') { // Si el ID también es numérico
                        wasEditedByMe = (lastEditorIdForComparison === currentLoggedInUserId); //
                    }
                    else { // Si el ID es string (userName) y el del place es number
                        wasEditedByMe = (String(lastEditorIdForComparison) === currentLoggedInUserId); // Convertir solo el del place a string
                    }
                }
                else if (resolvedEditorName && typeof resolvedEditorName === 'string') {
                    // PRIORIDAD 2: Si no hay ID numérico del editor del place, pero sí su nombre, comparar por nombre
                    wasEditedByMe = (resolvedEditorName.toLowerCase() === String(currentLoggedInUserId).toLowerCase()); //
                }
            }

            //console.log(`[WME_PLN] Nombre sugerido final: "${suggestedName}"`); // Comentario opcional.

            // --- Lógica para generar sugerencias del diccionario ---
            const originalWords = originalName.split(/\s+/).filter(word => word.length > 0);
            let sugerenciasLugar = {};
            let suggestedName = normalizedName; // Usa el resultado correcto y final de processPlaceName
            const suggestedNameWords = suggestedName.split(/\s+/).filter(word => word.length > 0);

            const processedWordKeys = new Set();
            originalWords.forEach((originalWord, wordIndex) => {
                //console.log(`\n[WME PLN - processNextPlace] Procesando palabra original: "${originalWord}" (Índice: ${wordIndex})`);
                if (!originalWord) return;

                const normalizedKeyForDuplicates = removeDiacritics(originalWord.toLowerCase());
                if (processedWordKeys.has(normalizedKeyForDuplicates)) {
                    return;
                }
                processedWordKeys.add(normalizedKeyForDuplicates);

                const lowerOriginalWord = originalWord.toLowerCase();
                const cleanedLowerNoDiacritics = removeDiacritics(lowerOriginalWord);
                //console.log(`[WME PLN - processNextPlace] lowerOriginalWord: "${lowerOriginalWord}", cleanedLowerNoDiacritics: "${cleanedLowerNoDiacritics}"`);

                let tildeCorrectionSuggested = false; // Bandera para saber si ya sugerimos tilde para esta palabra

                const currentSuggestedWord = suggestedNameWords[wordIndex] || '';
                const lowerCurrentSuggestedWord = currentSuggestedWord.toLowerCase();
                const currentSuggestedWordHasDiacritics = /[áéíóúÁÉÍÓÚüÜñÑ]/.test(lowerCurrentSuggestedWord);
                //console.log(`[WME PLN - processNextPlace] currentSuggestedWord (del nombre sugerido): "${currentSuggestedWord}", lowerCurrentSuggestedWord: "${lowerCurrentSuggestedWord}", currentSuggestedWordHasDiacritics: ${currentSuggestedWordHasDiacritics}`);
                // *******************************************************************
                // PASO 1: PRIORIDAD - SUGERIR CORRECCIÓN DE TILDES
                // PASO 1: PRIORIDAD - SUGERIR CORRECCIÓN DE TILDES
                // ✨ OPTIMIZACIÓN: Búsqueda O(1) usando mapa inverso (28-nov-2024)
                if (window.normalizedToDictionaryMap && window.normalizedToDictionaryMap.has(cleanedLowerNoDiacritics)) {
                    const dictWord = window.normalizedToDictionaryMap.get(cleanedLowerNoDiacritics);
                    const lowerDictWord = dictWord.toLowerCase();
                    const originalHasDiacritics = /[áéíóúÁÉÍÓÚüÜñÑ]/.test(lowerOriginalWord);

                    // Replicar condiciones originales:
                    // - La palabra del diccionario debe ser diferente a la actual sugerida
                    // - La actual sugerida no debe tener tildes (para no sobreescribir si ya tiene)
                    // - La original no debe tener tildes (solo agregamos tildes faltantes)
                    if (lowerDictWord !== lowerCurrentSuggestedWord &&
                        !currentSuggestedWordHasDiacritics &&
                        !originalHasDiacritics) {

                        let suggestedTildeWord = normalizeWordInternal(dictWord, true, false);

                        if (!sugerenciasLugar[originalWord]) sugerenciasLugar[originalWord] = [];
                        sugerenciasLugar[originalWord].push({
                            word: suggestedTildeWord,
                            similarity: 0.999,
                            fuente: 'dictionary_tilde'
                        });
                        tildeCorrectionSuggested = true;
                    }
                }
                // *******************************************************************

                // *******************************************************************
                // PASO 2: OTRAS SUGERENCIAS DEL DICCIONARIO (SOLO SI NO SE SUGIRIÓ CORRECCIÓN DE TILDE)
                if (!tildeCorrectionSuggested && checkDictionaryWords && window.dictionaryWords) {
                    //console.log(`[WME PLN - processNextPlace] Buscando otras sugerencias del diccionario para: "${originalWord}" (No se sugirió tilde).`);
                    const similarDictionary = findSimilarWords(cleanedLowerNoDiacritics, window.dictionaryIndex, similarityThreshold);
                    if (similarDictionary.length > 0) {
                        const finalSuggestions = similarDictionary.filter(d =>
                            d.word.toLowerCase() !== lowerOriginalWord && // No es la misma palabra original
                            d.word.toLowerCase() !== lowerCurrentSuggestedWord && // No es la misma palabra que ya está en el sugerido
                            !sugerenciasLugar[originalWord]?.some(s => s.word === normalizeWordInternal(d.word, true, false)) // No duplica una sugerencia de tilde ya agregada
                        );
                        //console.log(`[WME PLN - processNextPlace]   Otras sugerencias filtradas:`, finalSuggestions);

                        if (finalSuggestions.length > 0) {
                            if (!sugerenciasLugar[originalWord]) sugerenciasLugar[originalWord] = [];
                            finalSuggestions.forEach(dictSuggestion => {
                                if (!sugerenciasLugar[originalWord].some(s => s.word === normalizeWordInternal(dictSuggestion.word, true, false))) {
                                    sugerenciasLugar[originalWord].push({ ...dictSuggestion, fuente: 'dictionary' });
                                }
                            });
                        }
                    }
                }
                // *******************************************************************
            });
            // console.log(`[WME_PLN] Nombre sugerido después de trim/espacios múltiples: "${suggestedName}"`);
            // 6.1 --- QUITAR PUNTO FINAL SI EXISTE ---
            if (suggestedName.endsWith('.')) {
                suggestedName = suggestedName.slice(0, -1);
                // console.log(`[WME_PLN] Nombre sugerido después de quitar punto final: "${suggestedName}"`);
            }
            // 6.2 --- QUITAR ESPACIOS MÚLTIPLES ---
            //console.log(`[WME_PLN] Evaluando lógica de salto...`);
            const tieneSugerencias = Object.keys(sugerenciasLugar).length > 0;
            // Comparación estricta: solo colapsa espacios. Respeta mayúsculas, tildes y orden
            const cleanedOriginalName = String(nameForProcessing || '').replace(/\s+/g, ' ').trim();
            const cleanedSuggestedName = String(suggestedName || '').replace(/\s+/g, ' ').trim();
            const originalHasEmoticon = originalNameRaw !== nameForProcessing;

            // Un lugar es inconsistente si el nombre cambia O si tenía un emoticón.
            const isTrulyInconsistent = (cleanedOriginalName !== cleanedSuggestedName) || originalHasEmoticon;

            // Esta variable ahora refleja la verdadera condición de normalización
            const isNameEffectivelyNormalized = !isTrulyInconsistent;

            if (isNameEffectivelyNormalized && !tieneSugerencias && !hasEmptyName) {
                shouldSkipThisPlace = true;
                skipReasonLog = `[SKIP NORMALIZED]`;
            }
            const equalExact = (cleanedOriginalName === cleanedSuggestedName);
            const equalCaseInsensitive = (cleanedOriginalName.toLowerCase() === cleanedSuggestedName.toLowerCase());
            const equalNoDiacritics = (removeDiacritics(cleanedOriginalName.toLowerCase()) === removeDiacritics(cleanedSuggestedName.toLowerCase()));
            //let isNameEffectivelyNormalized = equalExact; // solo si es EXACTO tras colapsar espacios


            // PASO 1: Comprobar si se debe excluir por ser una edición tuya DENTRO del rango de fecha.
            if (avoidMyEdits && wasEditedByMe) {
                // Es un lugar editado por mí y el filtro está activo.
                const dateFilterValue = document.getElementById("dateFilterSelect")?.value || "all";
                const placeEditDate = (venueSDK && venueSDK.modificationData && venueSDK.modificationData.updatedOn)
                    ? new Date(venueSDK.modificationData.updatedOn)
                    : null;
                // Comprobar si la fecha de edición del lugar está dentro del rango seleccionado
                if (placeEditDate && isDateWithinRange(placeEditDate, dateFilterValue)) {
                    // Está DENTRO del rango, por lo tanto, se omite. La decisión es final.
                    shouldSkipThisPlace = true;
                    skipReasonLog = `[SKIP MY OWN EDIT - In Range: ${dateFilterValue}]`;
                }
                // Si está FUERA del rango, no hacemos nada aquí. Dejamos que 'shouldSkipThisPlace' siga siendo 'false'
                // y pase al siguiente filtro de abajo.
            }
            // Condición de Salto 2: Lugar está en la lista de excluidos (por ID).
            //console.log(`[WME PLN - SKIP] Verificando exclusión para ID: "${currentVenueId}"`); // <--- USAR currentVenueId
            //console.log(`[WME PLN - SKIP] 'excludedPlaces' contiene ID: ${excludedPlaces.has(currentVenueId)}`); // true/false // <--- USAR currentVenueId
            //console.log(`[WME PLN - SKIP] Contenido de 'excludedPlaces' (primeros 5 entries):`, Array.from(excludedPlaces.entries()).slice(0, 5)); // Ver algunos IDs guardados
            if (!shouldSkipThisPlace && excludedPlaces.has(currentVenueId)) { // <--- USAR currentVenueId
                shouldSkipThisPlace = true;
                skipReasonLog = `[SKIP EXCLUDED PLACE]`;
            }
            // PASO 2: Comprobar si el lugar ya está normalizado.
            // Esta regla se aplica a TODOS los lugares que NO fueron omitidos en el PASO 1.
            // (Incluye los lugares de otros editores y  propias ediciones fuera del rango de fecha).
            if (!shouldSkipThisPlace && isNameEffectivelyNormalized && !hasEmptyName) {
                shouldSkipThisPlace = true;
                skipReasonLog = `[SKIP NORMALIZED]`;
            }

            // PASO ADICIONAL DE SALTO: Si es un área y no se pudo calcular su área
            if (!shouldSkipThisPlace && typeInfo.isArea && areaMeters === null) { // <-- typeInfo se usa aquí
                shouldSkipThisPlace = true;
                skipReasonLog = `[SKIP AREA_CALC_FAILED]`;
            }
            //console.log(`[WME PLN - processNextPlace] LUGAR NO SALTADO. Procediendo. ID: ${currentVenueId}. Nombre: "${originalNameRaw}"`);



            //console.log(`[WME PLN - SKIP] Verificando exclusión para ID: "${currentVenueId}"`);
            //console.log(`[WME PLN - SKIP] 'excludedPlaces' contiene ID: ${excludedPlaces.has(currentVenueId)}`); // true/false
            //console.log(`[WME PLN - SKIP] Contenido de 'excludedPlaces' (primeros 5 entries):`, Array.from(excludedPlaces.entries()).slice(0, 5)); // Ver algunos IDs guardados

            //  PASO 2.5: Comprobar si el lugar está en la lista de excluidos
            if (!shouldSkipThisPlace && excludedPlaces.has(currentVenueId)) {
                shouldSkipThisPlace = true;
                skipReasonLog = `[SKIP EXCLUDED PLACE]`;
            }

            // --- Salto temprano si se determinó omitir el lugar ---
            if (shouldSkipThisPlace) {
                //console.log(`[WME PLN - processNextPlace] LUGAR SALTADO. Razón: ${skipReasonLog}. ID: ${currentVenueId}. Nombre: "${originalNameRaw}"`); // <--- USAR currentVenueId

                //  if (skipReasonLog) console.log(`[WME_PLN] ${skipReasonLog} Descartado "${originalName}" (ID: ${currentVenueId})`); //Añadir ID al log
                const updateFrequency = 3; // Actualiza cada 3 lugares la barra de progreso
                if ((index + 1) % updateFrequency === 0 || (index + 1) === places.length) {
                    updateScanProgressBar(index, places.length);
                }
                index++;
                setTimeout(() => processNextPlace(), 0); // Continúa con el siguiente lugar
                return;
            }
            else {
                //console.log(`[WME PLN - processNextPlace] LUGAR NO SALTADO. Procediendo. ID: ${currentVenueId}. Nombre: "${originalNameRaw}"`); // <--- USAR currentVenueId
            }
            //console.log(`[WME_PLN] Decisión de salto: ${shouldSkipThisPlace} (${skipReasonLog})`);
            // 8. Registrar o no en la lista de inconsistentes
            //console.log(`[WME_PLN] Registrando lugar con inconsistencias...`);
            // *** Si Llegamos Aquí, El Lugar No Se Salta Y Necesitamos Su Info Completa Para La Tabla ***
            if (processingStepLabel) {
                processingStepLabel.textContent = "Registrando lugar(es) con inconsistencias...";
            }
            // Lógica de Categorías (solo para lugares no saltados)
            const shouldRecommendCategories = document.getElementById("chk-recommend-categories")?.checked ?? true;
            let currentCategoryKey;
            let currentCategoryIcon;
            let currentCategoryTitle;
            let currentCategoryName;
            let dynamicSuggestions;
            try {
                const lang = getWazeLanguage();
                currentCategoryKey = getPlaceCategoryName(venueFromOldModel, venueSDK);
                const categoryDetails = getCategoryDetails(currentCategoryKey);
                currentCategoryIcon = categoryDetails.icon;
                currentCategoryTitle = categoryDetails.description;
                currentCategoryName = categoryDetails.description;

                if (shouldRecommendCategories)
                    dynamicSuggestions = findCategoryForPlace(originalName);
                else
                    dynamicSuggestions = [];
            }
            catch (e) {
                console.error("[WME PLN] Error procesando las categorías:", e);
                currentCategoryName = "Error";
                currentCategoryIcon = "❓";
                currentCategoryTitle = "Error al obtener categoría";
                dynamicSuggestions = [];
                currentCategoryKey = "UNKNOWN";
            }
            // Lógica unificada y robusta para obtener resolvedEditorName, lastEditorIdForComparison y calcular wasEditedByMe
            resolvedEditorName = "Desconocido"; // Reinicializar para cada place
            lastEditorIdForComparison = null; // Reinicializar para cada place
            if (venueSDK && venueSDK.modificationData) {
                const updatedByDataFromSDK = venueSDK.modificationData.updatedBy;
                if (typeof updatedByDataFromSDK === 'string' && updatedByDataFromSDK.trim() !== '') {
                    resolvedEditorName = updatedByDataFromSDK; // El nombre del editor es una cadena
                }
                else if (typeof updatedByDataFromSDK === 'number') {
                    lastEditorIdForComparison = updatedByDataFromSDK; // El ID numérico es la fuente principal
                    resolvedEditorName = `ID ${updatedByDataFromSDK}`; // Nombre temporal
                    if (W && W.model && W.model.users) {
                        const userObjectW = W.model.users.getObjectById(updatedByDataFromSDK);
                        if (userObjectW && userObjectW.userName) {
                            resolvedEditorName = userObjectW.userName; // Obtener nombre real del usuario si está en el modelo
                        }
                    }
                }
            }
            else if (venueFromOldModel && venueFromOldModel.attributes && (venueFromOldModel.attributes.updatedBy !== null && venueFromOldModel.attributes.updatedBy !== undefined)) {
                // Fallback al modelo antiguo si el SDK no dio datos de editor
                const oldModelUpdatedBy = venueFromOldModel.attributes.updatedBy;
                lastEditorIdForComparison = oldModelUpdatedBy; // El ID numérico es la fuente principal
                resolvedEditorName = `ID ${oldModelUpdatedBy}`; // Nombre temporal
                if (W && W.model && W.model.users) {
                    const userObjectW = W.model.users.getObjectById(oldModelUpdatedBy);
                    if (userObjectW && userObjectW.userName) {
                        resolvedEditorName = userObjectW.userName; // Obtener nombre real del usuario si está en el modelo
                    }
                }
            }
            else {
                resolvedEditorName = "N/D"; // No hay información de editor
            }
            wasEditedByMe = false; // Resetear para este place
            // Calcular wasEditedByMe de forma robusta aquí mismo
            if (currentLoggedInUserId !== null && currentLoggedInUserId !== undefined && resolvedEditorName !== "N/D") { // Solo si tenemos un nombre de usuario logueado y el resolvedEditorName no es N/D
                if (lastEditorIdForComparison !== null && lastEditorIdForComparison !== undefined && typeof lastEditorIdForComparison === 'number') {
                    // PRIORIDAD 1: Comparar IDs numéricos si ambos están disponibles y son válidos
                    if (typeof currentLoggedInUserId === 'number') { // Si el ID también es numérico
                        wasEditedByMe = (lastEditorIdForComparison === currentLoggedInUserId);
                    }
                    else { // Si el ID es string (userName) y el del place es number
                        wasEditedByMe = (String(lastEditorIdForComparison) === currentLoggedInUserId); // Convertir solo el del place a string
                    }
                }
                else if (resolvedEditorName && typeof resolvedEditorName === 'string') {
                    // PRIORIDAD 2: Si no hay ID numérico del editor del place, pero sí su nombre, comparar por nombre
                    wasEditedByMe = (resolvedEditorName.toLowerCase() === String(currentLoggedInUserId).toLowerCase());
                }
            }
            // Obtener información de la ciudad (esto ya estaba bien, solo reubicado)
            try {
                cityInfo = await getPlaceCityInfo(venueFromOldModel, venueSDK, currentCategoryKey);
            }
            catch (e) {
                console.error(`[WME_PLN] Error al obtener información de la ciudad para el venue ID ${currentVenueId}:`, e);
            }
            //Determinar nivel de bloqueo correspondiente
            let lockRank = 0; // Valor por defecto
            if (venueSDK && venueSDK.lockRank !== undefined && venueSDK.lockRank !== null)
                lockRank = venueSDK.lockRank;
            else if (venueFromOldModel && venueFromOldModel.attributes && venueFromOldModel.attributes.lockRank !== undefined && venueFromOldModel.attributes.lockRank !== null)
                lockRank = venueFromOldModel.attributes.lockRank;
            // Asignar el emoji correspondiente al nivel de bloqueo
            let lockRankEmoji = (lockRank >= 0 && lockRank <= 5) ? lockRankEmojis[lockRank + 1] : lockRankEmojis[0];
            // Calcular si el lugar tiene horarios superpuestos (solo si venueSDK está disponible)
            const hasOverlappingHours = checkForOverlappingHours(venueSDK); // Llama a la nueva función.
            // Calcular si el lugar tiene ediciones pendientes (solo si venueSDK está disponible)
            const hasPendingEdits = checkForPendingEdits(venueSDK); // Llama a la nueva función.




            //console.log(`[WME_PLN][DEBUG] Assigned LockRankEmoji: ${lockRankEmoji}`);

            // Limpiar guiones colgantes del suggestedName antes de mostrarlo
            // Esto evita que nombres como "Terpel ⛽" se conviertan en "Terpel -"
            if (typeof plnFixDanglingHyphen === 'function') {
                suggestedName = plnFixDanglingHyphen(suggestedName);
            }

            // Manejar lugares sin nombre: asignar nombre especial y marcador
            const displayOriginalName = hasEmptyName ? '⚠️ Sin nombre' : originalNameRaw;

            // Lógica especial para FOREST_GROVE: sugerencia vacía
            let displayNormalizedName = hasEmptyName ? 'Requiere corrección manual' : suggestedName;
            const categoriesForSuggestion = venueFromOldModel?.attributes?.categories || [];
            if (categoriesForSuggestion.includes('FOREST_GROVE')) {
                displayNormalizedName = "";
            }

            // ✨ REGLAS ESPECIALES PARA LUGARES SIN NOMBRE (28-nov-2024)
            if (hasEmptyName) {
                // REGLA 1: PARK tipo AREA sin nombre → Recomendar categoría FOREST
                if (categoriesForSuggestion.includes('PARK') && typeInfo.type === 'AREA') {
                    // Añadir FOREST a las sugerencias de categoría si no está ya
                    const forestSuggestion = {
                        categoryKey: 'FOREST_GROVE',
                        icon: '🌳',
                        desc_es: 'Bosque',
                        desc_en: 'Forest',
                        score: '⭐⭐⭐ Regla automática: Parques sin nombre tipo AREA son bosques'
                    };

                    if (!dynamicSuggestions.some(s => s.categoryKey === 'FOREST_GROVE')) {
                        dynamicSuggestions.unshift(forestSuggestion); // Añadir al inicio
                    }
                }

                if (categoriesForSuggestion.includes('PARKING_LOT')) {
                    displayNormalizedName = 'Parqueadero';
                }
            }

            // REGLA 3: FOREST_GROVE tipo PUNTO (no AREA) -> Requiere corrección
            // Si es un bosque y es un punto, estructuralmente está mal en Waze.
            if (categoriesForSuggestion && categoriesForSuggestion.includes('FOREST_GROVE') && typeInfo && !typeInfo.isArea) {
                displayNormalizedName = 'Requiere corrección manual';
            }

            // REGLA 4: PARKING_LOT tipo PUNTO -> Error, debe ser AREA
            let isParkingPointError = false;
            if (categoriesForSuggestion && categoriesForSuggestion.includes('PARKING_LOT') && typeInfo && !typeInfo.isArea) {
                isParkingPointError = true;
            }

            // Agregar a la lista de inconsistencias
            inconsistents.push({
                lockRankEmoji: lockRankEmoji,
                id: currentVenueId,
                original: displayOriginalName,
                normalized: displayNormalizedName,
                editor: resolvedEditorName, // Usamos el nombre del editor resuelto
                cityIcon: cityInfo.icon,
                cityTitle: cityInfo.title,
                hasCity: cityInfo.hasCity,
                venueSDKForRender: venueSDK,
                currentCategoryName: currentCategoryName,
                currentCategoryIcon: currentCategoryIcon,
                currentCategoryTitle: currentCategoryTitle,
                currentCategoryKey: currentCategoryKey,
                dynamicCategorySuggestions: dynamicSuggestions,
                // Asegurarse de incluir lat y lon obtenidos de getPlaceCoordinates
                lat: placeLat,
                lon: placeLon,
                typeInfo: typeInfo, // Guardar el objeto completo para su uso en el render
                areaMeters: areaMeters, // Ya se calcula con venueSDK
                hasOverlappingHours: hasOverlappingHours,
                hasPendingEdits: hasPendingEdits,
                hasEmptyName: hasEmptyName,  // Marcador especial para lugares sin nombre
                isParkingPointError: isParkingPointError
            });
            // 9. Agregar datos del lugar para la verificación de duplicados
            if (!hasEmptyName) {
                sugerenciasPorPalabra[currentVenueId] = sugerenciasLugar;// Guardar sugerencias por palabra para este lugar
            }
            // 10. Finalizar procesamiento del 'place' actual y pasar al siguiente
            const updateFrequency = 5;
            if ((index + 1) % updateFrequency === 0 || (index + 1) === places.length)
                updateScanProgressBar(index, places.length);
            index++;
            setTimeout(() => processNextPlace(), 0);
        }
        // console.log("[WME_PLN] Iniciando primer processNextPlace...");
        try {
            setTimeout(() => { processNextPlace(); }, 10);
        }
        catch (error) {
            console.error("[WME_PLN][ERROR_CRITICAL] Fallo al iniciar processNextPlace:", error, error.stack);
            enableScanControls();
            const outputFallback = document.querySelector("#wme-place-inspector-output");
            if (outputFallback) {
                outputFallback.innerHTML = `<div style='color:red; padding:10px;'><b>Error Crítico:</b> El script de normalización encontró un problema grave y no pudo continuar. Revise la consola para más detalles (F12).<br>Detalles: ${error.message}</div>`;
            }
            const scanBtn = document.querySelector("button[type='button']"); // Asumiendo que es el botón de Start Scan
            if (scanBtn) {
                scanBtn.disabled = false;
                scanBtn.textContent = "Start Scan... (Error Previo)";
            }
            if (window.processingDotsInterval) {
                clearInterval(window.processingDotsInterval);
            }
        }// processNextPlace

        // Función para re-aplicar la lógica de palabras excluidas al texto normalizado
        function reapplyExcludedWordsLogic(text, excludedWordsSet) {
            if (typeof text !== 'string' || !excludedWordsSet || excludedWordsSet.size === 0) {
                return text;
            }
            const wordsInText = text.split(/\s+/);
            const processedWordsArray = wordsInText.map(word => {
                if (word === "") return "";
                const wordWithoutDiacriticsLower = removeDiacritics(word.toLowerCase());
                // Encontrar la palabra excluida que coincida (insensible a may/min y diacríticos)
                const matchingExcludedWord = Array.from(excludedWordsSet).find(
                    w_excluded => removeDiacritics(w_excluded.toLowerCase()) === wordWithoutDiacriticsLower);
                if (matchingExcludedWord) {
                    // Si coincide, DEVOLVER LA FORMA EXACTA DE LA LISTA DE EXCLUIDAS
                    return matchingExcludedWord;
                }
                // Si no, devolver la palabra como estaba (ya normalizada por pasos previos)
                return word;
            });
            return processedWordsArray.join(' ');
        }// reapplyExcludedWordsLogic


        //Función para finalizar renderizado una vez completado el análisis
        function finalizeRender(inconsistents, placesArr, allSuggestions) {   // Limpiar el mensaje de procesamiento y spinner al finalizar el análisis
            try {
                // Pre-declare para evitar TDZ cuando se referencian antes de asignar
                // Evitar TDZ y nulls intermitentes
                //const typeInfo = venueSDK?.typeInfo || {};
                enableScanControls();
                // Detener animación de puntos suspensivos si existe
                if (window.processingDotsInterval) {
                    clearInterval(window.processingDotsInterval);
                    window.processingDotsInterval = null;
                }

                // Refuerza el restablecimiento del botón de escaneo al entrar
                const scanBtn = document.querySelector("button[type='button']");
                if (scanBtn) {
                    scanBtn.textContent = "Start Scan...";
                    scanBtn.disabled = false;
                    scanBtn.style.opacity = "1";
                    scanBtn.style.cursor = "pointer";
                }
                // Verificar si el botón de escaneo existe
                const output = document.querySelector("#wme-place-inspector-output");
                if (!output) {
                    //  console.error("[WME_PLN]❌ No se pudo montar el panel flotante. Revisar estructura del DOM.");
                    alert("Hubo un problema al mostrar los resultados. Intenta recargar la página.");
                    return;
                }
                // Limpiar el mensaje de procesamiento y spinner
                const undoRedoHandler = function () {// Maneja el evento de deshacer/rehacer
                    if (floatingPanelElement && floatingPanelElement.style.display !== 'none') {
                        waitForWazeAPI(() => {
                            const places = getVisiblePlaces();
                            renderPlacesInFloatingPanel(places); // Esto mostrará el panel de "procesando" y luego resultados
                            reactivateAllActionButtons(); // No necesitamos setTimeout aquí si renderPlacesInFloatingPanel es síncrono.
                        });
                    }
                    else {
                        //console.log("[WME PLN] Undo/Redo: Panel de resultados no visible, no se re-escanea.");
                    }
                };
                // Objeto para almacenar referencias de listeners para desregistro
                if (!window._wmePlnUndoRedoListeners) {
                    window._wmePlnUndoRedoListeners = {};
                }
                // Desregistrar listeners previos si existen
                if (window._wmePlnUndoRedoListeners.undo) {
                    W.model.actionManager.events.unregister("afterundoaction", null, window._wmePlnUndoRedoListeners.undo);
                }
                if (window._wmePlnUndoRedoListeners.redo) {
                    W.model.actionManager.events.unregister("afterredoaction", null, window._wmePlnUndoRedoListeners.redo);
                }
                // Registrar nuevos listeners
                W.model.actionManager.events.register("afterundoaction", null, undoRedoHandler);
                W.model.actionManager.events.register("afterredoaction", null, undoRedoHandler);
                // Almacenar referencias para poder desregistrar en el futuro
                window._wmePlnUndoRedoListeners.undo = undoRedoHandler;
                window._wmePlnUndoRedoListeners.redo = undoRedoHandler;
                // Esta llamada se hace ANTES de limpiar el output. El primer argumento es el estado, el segundo es el número de inconsistencias.
                createFloatingPanel("results", inconsistents.length);
                // Limpiar el mensaje de procesamiento y spinner
                if (output) {
                    // Mostrar el panel flotante al terminar el procesamiento se usa para mostrar los resultados y llamados al console.log
                }
                // Limitar a 30 resultados y mostrar advertencia si excede
                const maxRenderLimit = 30;
                const totalInconsistentsOriginal = inconsistents.length; // Guardar el total original
                let isLimited = false; // Declarar e inicializar isLimited
                // Si hay más de 30 resultados, limitar a 30 y mostrar mensaje
                // Limit logic removed to support batching
                /*
                if (totalInconsistentsOriginal > maxRenderLimit) {
                    // ... (removed slicing and modal)
                }
                */
                // Llamar a la función para detectar y alertar nombres duplicados
                detectAndAlertDuplicateNames(inconsistents);

                // ✨ OPTIMIZACIÓN: Reagrupar lugares duplicados juntos al inicio (28-nov-2024)
                // Los duplicados se colocan al principio, agrupados por nombre
                const duplicates = [];
                const nonDuplicates = [];

                // Separar duplicados de no duplicados
                inconsistents.forEach(place => {
                    if (place.isDuplicate) {
                        duplicates.push(place);
                    } else {
                        nonDuplicates.push(place);
                    }
                });

                // Ordenar duplicados por nombre normalizado para agruparlos
                duplicates.sort((a, b) => a.normalized.localeCompare(b.normalized));

                // Reconstruir array: duplicados primero, luego no duplicados
                inconsistents = [...duplicates, ...nonDuplicates];

                // Crear un contenedor para los elementos fijos de la cabecera del panel de resultados
                const fixedHeaderContainer = document.createElement("div");

                fixedHeaderContainer.style.background = "#fff"; // Fondo para que no se vea el scroll debajo
                fixedHeaderContainer.style.padding = "0 10px 8px 10px"; // Padding para espacio y que no esté pegado
                fixedHeaderContainer.style.borderBottom = "1px solid #ccc"; // Un borde para separarlo de la tabla
                fixedHeaderContainer.style.zIndex = "11"; // Asegurarse de que esté por encima de la tabla
                // Añadir Estas Dos Líneas Clave Al FixedHeaderContainer
                fixedHeaderContainer.style.position = "sticky"; // Hacer Que Este Contenedor Sea Sticky
                fixedHeaderContainer.style.top = "0";            // Pegado A La Parte Superior Del Contenedor De Scroll

                // =======================================================
                // INICIO DEL BLOQUE CORREGIDO
                // =======================================================
                // 1. Contenedor Flex para el texto y el botón
                const headerControlsContainer = document.createElement("div");
                headerControlsContainer.style.display = "flex";
                // LA SIGUIENTE LÍNEA ES EL CAMBIO PRINCIPAL:
                headerControlsContainer.style.justifyContent = "flex-start"; // Alinea los elementos al inicio
                headerControlsContainer.style.alignItems = "center";
                headerControlsContainer.style.gap = "15px"; // Mantiene el espacio entre texto y botón

                const resultsCounter = document.createElement("div");
                resultsCounter.className = "results-counter-display";
                resultsCounter.style.fontSize = "13px";
                resultsCounter.style.color = "#555";
                resultsCounter.style.textAlign = "left";

                resultsCounter.dataset.currentCount = inconsistents.length;
                resultsCounter.dataset.totalOriginal = totalInconsistentsOriginal;
                resultsCounter.dataset.maxRenderLimit = maxRenderLimit;


                // Mostrar el contador y el botón solo si hay inconsistencias
                if (totalInconsistentsOriginal > 0) {
                    // 1. Configura el mensaje del contador
                    if (isLimited) {
                        resultsCounter.innerHTML = `<span style="color: #ff0000;">Inconsistencias encontradas: <b>${totalInconsistentsOriginal}</b></span>. Mostrando las primeras <span style="color: #ff0000;"><b>${inconsistents.length}</b></span>.`;
                    }
                    else {
                        resultsCounter.innerHTML = `Inconsistencias encontradas: <b style="color: #ff0000;">${inconsistents.length}</b> de <b style="color: #ff0000;">${totalInconsistentsOriginal}</b>. Mostrando <b style="color: #ff0000;">${inconsistents.length}</b>.`;
                    }
                    headerControlsContainer.appendChild(resultsCounter);

                    // 2. Muestra el botón "Mostrar/Ocultar procesados"
                    const toggleBtn = document.getElementById('pln-toggle-hidden-btn');
                    if (toggleBtn) {
                        headerControlsContainer.appendChild(toggleBtn);
                        toggleBtn.style.display = 'inline-block'; // Asegurarse de que sea visible
                    }

                }
                else {
                    // Si no hay inconsistencias, muestra el mensaje de éxito
                    const outputDiv = document.querySelector("#wme-place-inspector-output");
                    if (outputDiv) {
                        outputDiv.innerHTML = `<div style='color:green; padding:10px;'>✔ Todos los lugares visibles están correctamente normalizados o excluidos.</div>`;
                    }
                    // Y nos aseguramos de que el botón esté oculto
                    const toggleBtn = document.getElementById('pln-toggle-hidden-btn');
                    if (toggleBtn) {
                        toggleBtn.style.display = 'none';
                    }
                }

                fixedHeaderContainer.appendChild(headerControlsContainer);


                // 2. Lógica del botón (sin cambios respecto a la corrección anterior)
                // En la función donde creas el botón toggle
                // Lógica unificada para el botón "Mostrar/Ocultar Normalizados"
                if (totalInconsistentsOriginal > 0) {
                    let toggleBtn = document.getElementById('pln-toggle-hidden-btn');
                    if (!toggleBtn) {
                        toggleBtn = document.createElement("button");
                        toggleBtn.id = 'pln-toggle-hidden-btn';
                        toggleBtn.style.padding = "5px 10px";
                        toggleBtn.style.marginLeft = "15px";
                        toggleBtn.style.fontSize = "12px"; // Mantenemos el tamaño de letra

                        // Aseguramos que el estado inicial sea ocultar las filas
                        document.body.classList.add('pln-hide-normalized-rows');

                        toggleBtn.addEventListener('click', function () {
                            // Simplemente alterna la clase en el body
                            document.body.classList.toggle('pln-hide-normalized-rows');

                            // Y actualiza el texto del botón según el estado
                            this.textContent = document.body.classList.contains('pln-hide-normalized-rows')
                                ? "Mostrar Normalizados"
                                : "Ocultar Normalizados";
                        });
                    }

                    // Sincroniza siempre el texto del botón con el estado actual
                    toggleBtn.textContent = document.body.classList.contains('pln-hide-normalized-rows')
                        ? "Mostrar Normalizados"
                        : "Ocultar Normalizados";

                    headerControlsContainer.appendChild(toggleBtn);
                }

                fixedHeaderContainer.appendChild(headerControlsContainer);

                if (inconsistents.length === 0) {
                    if (totalInconsistentsOriginal === 0) {
                        const checkIcon = document.createElement("div");
                        checkIcon.innerHTML = "";
                        checkIcon.style.marginTop = "10px";
                        checkIcon.style.fontSize = "14px";
                        checkIcon.style.color = "green";
                        output.appendChild(checkIcon);
                        const successMsg = document.createElement("div");
                        successMsg.textContent = "";
                        successMsg.style.marginTop = "10px";
                        successMsg.style.fontSize = "14px";
                        successMsg.style.color = "green";
                        successMsg.style.fontWeight = "bold";
                        output.appendChild(successMsg);
                    }
                    const existingOverlay = document.getElementById("scanSpinnerOverlay");
                    if (existingOverlay) existingOverlay.remove();
                    const progressBarInnerTab = document.getElementById("progressBarInnerTab");
                    const progressBarTextTab = document.getElementById("progressBarTextTab");
                    if (progressBarInnerTab && progressBarTextTab) {
                        progressBarInnerTab.style.width = "100%";
                        progressBarTextTab.textContent = `Progreso: 100% (${placesArr.length}/${placesArr.length})`;
                    }
                    const outputTab = document.getElementById("wme-normalization-tab-output");
                    if (outputTab) {
                        outputTab.innerHTML = `✔ Todos los nombres están normalizados. Se analizaron ${placesArr.length} lugares.`;
                        outputTab.style.color = "green";
                        outputTab.style.fontWeight = "bold";
                    }
                    const scanBtn = document.querySelector("button[type='button']");
                    if (scanBtn) {
                        scanBtn.textContent = "Start Scan...";
                        scanBtn.disabled = false;
                        scanBtn.style.opacity = "1";
                        scanBtn.style.cursor = "pointer";
                        const iconCheck = document.createElement("span");
                        iconCheck.textContent = " ✔";
                        iconCheck.style.marginLeft = "8px";
                        iconCheck.style.color = "green";
                        scanBtn.appendChild(iconCheck);
                    }
                    return;
                }
                if (output) {
                    output.style.display = 'flex';
                    output.style.flexDirection = 'column';
                    output.style.position = 'relative';
                    output.appendChild(fixedHeaderContainer);
                }
                const table = document.createElement("table");
                table.style.width = "100%";
                table.style.borderCollapse = "collapse";
                table.style.fontSize = "12px";
                const thead = document.createElement("thead");
                const headerRow = document.createElement("tr");
                [
                    "N°",
                    "Perma",
                    "Tipo/Ciudad",
                    "LL",
                    "Editor",
                    "Nombre Actual",
                    "⚠️",
                    "Nombre Sugerido",
                    "Sugerencias<br>de reemplazo",
                    "Categoría",
                    "Categoría<br>Recomendada",
                    "Acción"
                ].forEach(header => {
                    const th = document.createElement("th");
                    th.innerHTML = header;
                    th.style.borderBottom = "1px solid #ccc";
                    th.style.padding = "4px";
                    th.style.textAlign = "center";
                    th.style.fontSize = "14px";
                    if (header === "N°") {
                        th.style.width = "30px";
                    }
                    else if (header === "LL") {
                        th.title = "Nivel de Bloqueo (Lock Level)";
                        th.style.width = "40px";
                    }
                    else if (header === "Perma" || header === "Tipo/Ciudad") {
                        th.style.width = "65px";
                    }
                    else if (header === "⚠️") {
                        th.title = "Alertas y advertencias";
                        th.style.width = "30px";
                    }
                    else if (header === "Categoría") {
                        th.style.width = "110px";
                    }
                    else if (header === "Categoría<br>Recomendada" || header === "Sugerencias<br>de reemplazo") {
                        th.style.width = "150px";
                    }
                    else if (header === "Editor") {
                        th.style.width = "100px";
                    }
                    else if (header === "Acción") {
                        th.style.width = "100px";
                    }
                    else if (header === "Nombre Actual" || header === "Nombre Sugerido") {
                        th.style.width = "250px";
                    }
                    headerRow.appendChild(th);
                });
                thead.appendChild(headerRow);
                table.appendChild(thead);
                thead.style.position = "sticky";
                thead.style.top = "0";
                thead.style.background = "#f1f1f1";
                thead.style.zIndex = "10";
                headerRow.style.backgroundColor = "#003366";
                headerRow.style.color = "#ffffff";
                const tbody = document.createElement("tbody");

                const buildPlacePermalink = (placeId) => {
                    try {
                        const { protocol, host, pathname, search } = window.location;
                        const origin = `${protocol}//${host}`;
                        const editorKeyword = '/editor';
                        let editorPath = editorKeyword;
                        const editorIndex = pathname.toLowerCase().indexOf(editorKeyword);
                        if (editorIndex !== -1) {
                            editorPath = pathname.slice(0, editorIndex + editorKeyword.length);
                        }
                        let envParam = 'row';
                        if (typeof URLSearchParams === "function") {
                            const params = new URLSearchParams(search);
                            envParam = params.get('env') || envParam;
                        }
                        else {
                            const envMatch = /env=([^&]+)/i.exec(search);
                            if (envMatch) {
                                envParam = envMatch[1];
                            }
                        }
                        return `${origin}${editorPath}?env=${envParam}&venueId=${placeId}`;
                    }
                    catch (_) {
                        return `https://www.waze.com/editor?env=row&venueId=${placeId}`;
                    }
                };

                function handlePlaceLinkNavigation(event, placeId, placeName, placePermalink, venueSDKForRender) {
                    if (event) {
                        if (event.ctrlKey || event.metaKey || event.shiftKey || event.button === 1) {
                            return; // allow default behavior to open in new tab/window
                        }
                        event.preventDefault();
                        event.stopPropagation();
                    }

                    const venueObj = W.model.venues.getObjectById(placeId);
                    const venueSDKForUse = venueSDKForRender;

                    let targetLat = null;
                    let targetLon = null;

                    if (venueSDKForUse && venueSDKForUse.geometry && Array.isArray(venueSDKForUse.geometry.coordinates) && venueSDKForUse.geometry.coordinates.length >= 2) {
                        targetLon = venueSDKForUse.geometry.coordinates[0];
                        targetLat = venueSDKForUse.geometry.coordinates[1];
                    }

                    if ((targetLat === null || targetLon === null) && venueObj && typeof venueObj.getOLGeometry === 'function') {
                        try {
                            const geometryOL = venueObj.getOLGeometry();
                            if (geometryOL && typeof geometryOL.getCentroid === 'function') {
                                const centroidOL = geometryOL.getCentroid();
                                if (typeof OpenLayers !== 'undefined' && OpenLayers.Projection) {
                                    const transformedPoint = new OpenLayers.Geometry.Point(centroidOL.x, centroidOL.y).transform(
                                        new OpenLayers.Projection("EPSG:3857"),
                                        new OpenLayers.Projection("EPSG:4326")
                                    );
                                    targetLat = transformedPoint.y;
                                    targetLon = transformedPoint.x;
                                } else {
                                    targetLat = centroidOL.y;
                                    targetLon = centroidOL.x;
                                }
                            }
                        } catch (e) {
                            console.error("[WME PLN] Error al obtener/transformar geometría OL para navegación:", e);
                        }
                    }

                    let navigated = false;

                    if (venueObj && W.selectionManager && typeof W.selectionManager.select === "function") {
                        W.selectionManager.select(venueObj);
                        navigated = true;
                    }
                    else if (venueObj && W.selectionManager && typeof W.selectionManager.setSelectedModels === "function") {
                        W.selectionManager.setSelectedModels([venueObj]);
                        navigated = true;
                    }

                    if (!navigated) {
                        const confirmOpen = confirm(`El lugar "${placeName}" (ID: ${placeId}) no se pudo seleccionar o centrar directamente. ¿Deseas abrirlo en una nueva pestaña del editor?`);
                        if (confirmOpen) {
                            window.open(placePermalink, '_blank');
                        }
                        else {
                            showTemporaryMessage("El lugar podría estar fuera de vista o no cargado.", 4000, 'warning');
                        }
                    }
                    else {
                        showTemporaryMessage("Presentando detalles del lugar...", 2000, 'info');
                    }
                }

                const renderRow = (item, index) => {
                    const { lockRankEmoji, id, original, normalized, editor, cityIcon, cityTitle, hasCity, currentCategoryName, currentCategoryIcon, currentCategoryTitle, currentCategoryKey, dynamicCategorySuggestions, venueSDKForRender, isDuplicate = false, duplicatePartners = [], typeInfo, areaMeters, hasOverlappingHours, hasPendingEdits, hasEmptyName } = item;
                    const progressPercent = Math.floor(((index + 1) / inconsistents.length) * 100);
                    const progressBarInnerTab = document.getElementById("progressBarInnerTab");
                    const progressBarTextTab = document.getElementById("progressBarTextTab");
                    if (progressBarInnerTab && progressBarTextTab) {
                        progressBarInnerTab.style.width = `${progressPercent}%`;
                        progressBarTextTab.textContent = `Progreso: ${progressPercent}% (${index + 1}/${inconsistents.length})`;
                    }
                    const row = document.createElement("tr");

                    // Aplicar estilo para lugares sin nombre
                    if (hasEmptyName) {
                        row.classList.add("pln-empty-name-row");
                        row.title = "⚠️ Este lugar no tiene nombre y requiere corrección manual";
                    }
                    const placePermalink = buildPlacePermalink(id);
                    row.querySelectorAll("td").forEach(td => td.style.verticalAlign = "top");
                    row.dataset.placeId = id;
                    const numberCell = document.createElement("td");
                    numberCell.textContent = index + 1;
                    numberCell.style.textAlign = "center";
                    numberCell.style.padding = "4px";
                    row.appendChild(numberCell);
                    const permalinkCell = document.createElement("td");
                    const link = document.createElement("a");
                    link.href = placePermalink;
                    link.target = "_blank";
                    link.rel = "noopener noreferrer";
                    link.addEventListener("click", (e) => {
                        handlePlaceLinkNavigation(e, id, original, placePermalink, venueSDKForRender);
                    });
                    link.title = "Seleccionar lugar en el mapa";
                    link.textContent = "🔗";
                    permalinkCell.appendChild(link);
                    permalinkCell.style.padding = "4px";
                    permalinkCell.style.fontSize = "18px";
                    permalinkCell.style.textAlign = "center";
                    permalinkCell.style.width = "65px";
                    row.appendChild(permalinkCell);

                    const typeCityCell = document.createElement("td");
                    typeCityCell.style.padding = "4px";
                    typeCityCell.style.width = "65px";
                    typeCityCell.style.verticalAlign = "middle";

                    const cellContentWrapper = document.createElement("div");
                    cellContentWrapper.style.display = "flex";
                    cellContentWrapper.style.justifyContent = "space-around";
                    cellContentWrapper.style.alignItems = "center";

                    const typeContainer = document.createElement("div");
                    typeContainer.style.display = "flex";
                    typeContainer.style.flexDirection = "column";
                    typeContainer.style.alignItems = "center";
                    typeContainer.style.justifyContent = "center";
                    typeContainer.style.gap = "2px";

                    const typeIconSpan = document.createElement("span");
                    typeIconSpan.textContent = typeInfo.icon;
                    typeIconSpan.style.fontSize = "20px";

                    let tooltipText = `Tipo: ${typeInfo.title}`;

                    // Lógica para convertir Parking Punto a Área
                    if (item.isParkingPointError) {
                        typeIconSpan.style.cursor = "pointer";
                        typeIconSpan.style.border = "1px solid red";
                        typeIconSpan.style.borderRadius = "50%";
                        typeIconSpan.style.backgroundColor = "#ffe6e6";
                        tooltipText += "\n⚠️ Parqueadero tipo PUNTO. Click para convertir a ÁREA.";

                        typeIconSpan.addEventListener("click", async (e) => {
                            e.stopPropagation();
                            if (confirm(`¿Convertir este parqueadero a tipo ÁREA?\n\nSe creará un cuadrado de 30x30m alrededor del punto.`)) {
                                await convertPointToArea(id);
                            }
                        });
                    }

                    typeIconSpan.title = tooltipText;

                    typeContainer.appendChild(typeIconSpan);

                    if (typeInfo.isArea && areaMeters !== null && areaMeters !== undefined) {
                        const areaSpan = document.createElement("span");
                        const areaFormatted = areaMeters.toLocaleString('es-ES', { maximumFractionDigits: 0 });
                        areaSpan.textContent = `${areaFormatted} m²`;
                        areaSpan.style.fontSize = "10px";
                        areaSpan.style.fontWeight = "bold";
                        areaSpan.style.textAlign = "center";
                        areaSpan.style.lineHeight = "1";
                        areaSpan.style.whiteSpace = "nowrap";

                        if (areaMeters < 400) {
                            areaSpan.style.color = "red";
                            areaSpan.classList.add("area-blink");
                        }
                        else {
                            areaSpan.style.color = "blue";
                        }
                        areaSpan.title = `Área: ${areaFormatted} m²`;
                        typeContainer.appendChild(areaSpan);
                    }
                    cellContentWrapper.appendChild(typeContainer);

                    const cityStatusIconSpan = document.createElement("span");
                    cityStatusIconSpan.className = 'city-status-icon';
                    cityStatusIconSpan.style.fontSize = "18px";
                    cityStatusIconSpan.style.cursor = "pointer";

                    // ✅ LÓGICA ESPECIAL PARA RÍOS: Si es RIVER_STREAM y tiene ciudad, mostrar bandera para eliminar
                    const isRiver = currentCategoryKey === 'RIVER_STREAM';

                    if (isRiver && hasCity) {
                        // ❗ Mostrar bandera ROJA para ríos con ciudad (incorrecto, se debe eliminar)
                        cityStatusIconSpan.innerHTML = '❗';
                        cityStatusIconSpan.style.color = 'red';
                        cityStatusIconSpan.title = `${cityTitle}\n\n⚠️ Los ríos no deben tener ciudad.\nClick para limpiar ciudad.`;

                        // Al hacer click, mostrar modal de confirmación para eliminar ciudad
                        cityStatusIconSpan.addEventListener("click", async () => {
                            const confirmModal = document.createElement("div");
                            confirmModal.style.position = "fixed";
                            confirmModal.style.top = "50%";
                            confirmModal.style.left = "50%";
                            confirmModal.style.transform = "translate(-50%, -50%)";
                            confirmModal.style.background = "#fff";
                            confirmModal.style.border = "1px solid #aad";
                            confirmModal.style.padding = "28px 32px 20px 32px";
                            confirmModal.style.zIndex = "20000";
                            confirmModal.style.boxShadow = "0 4px 24px rgba(0,0,0,0.18)";
                            confirmModal.style.fontFamily = "sans-serif";
                            confirmModal.style.borderRadius = "10px";
                            confirmModal.style.textAlign = "center";
                            confirmModal.style.minWidth = "340px";

                            const iconElement = document.createElement("div");
                            iconElement.innerHTML = "🌊";
                            iconElement.style.fontSize = "38px";
                            iconElement.style.marginBottom = "10px";
                            confirmModal.appendChild(iconElement);

                            const message = document.createElement("div");
                            const venue = W.model.venues.getObjectById(id);
                            const placeName = venue?.attributes?.name?.value || venue?.attributes?.name || "este río";
                            message.innerHTML = `<b>¿Eliminar ciudad de "${placeName}"?</b>`;
                            message.style.fontSize = "20px";
                            message.style.marginBottom = "8px";
                            confirmModal.appendChild(message);

                            const cityDiv = document.createElement("div");
                            cityDiv.textContent = `Ciudad actual: ${cityTitle.replace('Ciudad: ', '')}`;
                            cityDiv.style.fontSize = "15px";
                            cityDiv.style.color = "#007bff";
                            cityDiv.style.marginBottom = "18px";
                            confirmModal.appendChild(cityDiv);

                            const buttonWrapper = document.createElement("div");
                            buttonWrapper.style.display = "flex";
                            buttonWrapper.style.justifyContent = "center";
                            buttonWrapper.style.gap = "18px";

                            const cancelBtn = document.createElement("button");
                            cancelBtn.textContent = "Cancelar";
                            cancelBtn.style.padding = "7px 18px";
                            cancelBtn.style.background = "#eee";
                            cancelBtn.style.border = "none";
                            cancelBtn.style.borderRadius = "4px";
                            cancelBtn.style.cursor = "pointer";
                            cancelBtn.addEventListener("click", () => confirmModal.remove());

                            const confirmBtn = document.createElement("button");
                            confirmBtn.textContent = "Eliminar";
                            confirmBtn.style.padding = "7px 18px";
                            confirmBtn.style.background = "#d9534f";
                            confirmBtn.style.color = "#fff";
                            confirmBtn.style.border = "none";
                            confirmBtn.style.borderRadius = "4px";
                            confirmBtn.style.cursor = "pointer";
                            confirmBtn.style.fontWeight = "bold";

                            confirmBtn.addEventListener("click", async () => {
                                const venue = W.model.venues.getObjectById(id);
                                if (!venue) {
                                    console.error("[WME_PLN]El lugar no está disponible o ya fue eliminado.");
                                    confirmModal.remove();
                                    return;
                                }
                                try {
                                    const UpdateObject = require("Waze/Action/UpdateObject");
                                    const action = new UpdateObject(venue, {
                                        cityID: null,
                                        streetID: null
                                    });
                                    W.model.actionManager.add(action);

                                    recordNormalizationEvent();
                                    const row = document.querySelector(`tr[data-place-id="${id}"]`);
                                    if (row) {
                                        const iconToUpdate = row.querySelector('.city-status-icon');
                                        if (iconToUpdate) {
                                            iconToUpdate.innerHTML = '✅';
                                            iconToUpdate.style.color = 'green';
                                            iconToUpdate.title = 'Sin ciudad (correcto para ríos)';
                                            iconToUpdate.style.pointerEvents = 'none';
                                        }
                                    }
                                } catch (e) {
                                    console.error("[WME PLN] Error al eliminar ciudad del río: " + e.message, e);
                                }
                                confirmModal.remove();
                            });

                            buttonWrapper.appendChild(cancelBtn);
                            buttonWrapper.appendChild(confirmBtn);
                            confirmModal.appendChild(buttonWrapper);

                            document.body.appendChild(confirmModal);
                        });
                    }
                    else if (isRiver && !hasCity) {
                        // ✅ RÍOS SIN CIUDAD: ESTADO CORRECTO
                        cityStatusIconSpan.innerHTML = '✅';
                        cityStatusIconSpan.style.color = 'green';
                        cityStatusIconSpan.title = 'Río sin ciudad (Correcto)';
                        cityStatusIconSpan.style.cursor = 'default';
                    }
                    else if (hasCity) {
                        // ✅ Lugares normales (NO ríos) con ciudad: mostrar check verde
                        cityStatusIconSpan.innerHTML = '✅';
                        cityStatusIconSpan.style.color = 'green';
                        cityStatusIconSpan.title = cityTitle;
                    }
                    else {
                        // 🚩 Lugares sin ciudad: mostrar bandera para asignar ciudad
                        cityStatusIconSpan.innerHTML = '🚩';
                        cityStatusIconSpan.style.color = 'red';
                        cityStatusIconSpan.title = cityTitle;

                        cityStatusIconSpan.addEventListener("click", async () => {
                            const coords = getPlaceCoordinates(W.model.venues.getObjectById(id), venueSDKForRender);
                            const placeLat = coords.lat;
                            const placeLon = coords.lon;

                            if (placeLat === null || placeLon === null) {
                                alert("No se pudieron obtener las coordenadas del lugar.");
                                return;
                            }
                            const allCities = Object.values(W.model.cities.objects)
                                .filter(city =>
                                    city &&
                                    city.attributes &&
                                    typeof city.attributes.name === 'string' &&
                                    city.attributes.name.trim() !== ''
                                );
                            const citiesWithDistance = allCities.map(city => {
                                if (!city.attributes.geoJSONGeometry ||
                                    !Array.isArray(city.attributes.geoJSONGeometry.coordinates) ||
                                    city.attributes.geoJSONGeometry.coordinates.length < 2)
                                    return null;
                                const cityLon = city.attributes.geoJSONGeometry.coordinates[0];
                                const cityLat = city.attributes.geoJSONGeometry.coordinates[1];
                                const distanceInMeters = calculateDistance(placeLat, placeLon, cityLat, cityLon);
                                const distanceInKm = distanceInMeters / 1000;
                                return {
                                    name: city.attributes.name,
                                    distance: distanceInKm,
                                    cityId: city.getID()
                                };
                            }).filter(Boolean);
                            const closestCities = citiesWithDistance.sort((a, b) => a.distance - b.distance).slice(0, 5);
                            const modal = document.createElement("div");
                            modal.style.position = "fixed";
                            modal.style.top = "50%";
                            modal.style.left = "50%";
                            modal.style.transform = "translate(-50%, -50%)";
                            modal.style.background = "#fff";
                            modal.style.border = "1px solid #aad";
                            modal.style.padding = "28px 32px 20px 32px";
                            modal.style.zIndex = "20000";
                            modal.style.boxShadow = "0 4px 24px rgba(0,0,0,0.18)";
                            modal.style.fontFamily = "sans-serif";
                            modal.style.borderRadius = "10px";
                            modal.style.textAlign = "center";
                            modal.style.minWidth = "340px";

                            const iconElement = document.createElement("div");
                            iconElement.innerHTML = "🏙️";
                            iconElement.style.fontSize = "38px";
                            iconElement.style.marginBottom = "10px";
                            modal.appendChild(iconElement);

                            const messageTitle = document.createElement("div");
                            messageTitle.innerHTML = `<b>Asignar ciudad al lugar</b>`;
                            messageTitle.style.fontSize = "20px";
                            messageTitle.style.marginBottom = "8px";
                            modal.appendChild(messageTitle);

                            const listDiv = document.createElement("div");
                            listDiv.style.textAlign = "left";
                            listDiv.style.marginTop = "10px";

                            if (closestCities.length === 0) {
                                const noCityLine = document.createElement("div");
                                noCityLine.textContent = "No se encontraron ciudades cercanas para mostrar.";
                                noCityLine.style.color = "#888";
                                listDiv.appendChild(noCityLine);
                            }
                            else {
                                closestCities.forEach((city, idx) => {
                                    const cityLine = document.createElement("div");
                                    cityLine.style.marginBottom = "8px";
                                    cityLine.style.display = "flex";
                                    cityLine.style.alignItems = "center";

                                    const radioInput = document.createElement("input");
                                    radioInput.type = "radio";
                                    radioInput.name = `city-selection-${id}`;
                                    radioInput.value = city.cityId;
                                    radioInput.id = `city-radio-${city.cityId}`;
                                    radioInput.style.marginRight = "10px";
                                    radioInput.style.marginTop = "0";
                                    if (idx === 0) radioInput.checked = true;

                                    const radioLabel = document.createElement("label");
                                    radioLabel.htmlFor = `city-radio-${city.cityId}`;
                                    radioLabel.style.cursor = "pointer";
                                    radioLabel.innerHTML = `<b>${city.name}</b> <span style="color: #666; font-size: 11px;">(ID: ${city.cityId})</span> <span style="color: #007bff;">${city.distance.toFixed(1)} km</span>`;
                                    cityLine.appendChild(radioInput);
                                    cityLine.appendChild(radioLabel);
                                    listDiv.appendChild(cityLine);
                                });
                            }
                            modal.appendChild(listDiv);

                            const buttonWrapper = document.createElement("div");
                            buttonWrapper.style.display = "flex";
                            buttonWrapper.style.justifyContent = "flex-end";
                            buttonWrapper.style.gap = "12px";
                            buttonWrapper.style.marginTop = "20px";

                            const applyBtn = document.createElement("button");
                            applyBtn.textContent = "Aplicar Ciudad";
                            applyBtn.style.padding = "8px 16px";
                            applyBtn.style.background = "#28a745";
                            applyBtn.style.color = "#fff";
                            applyBtn.style.border = "none";
                            applyBtn.style.borderRadius = "4px";
                            applyBtn.style.cursor = "pointer";
                            applyBtn.style.fontWeight = "bold";

                            applyBtn.addEventListener('click', () => {
                                const selectedRadio = modal.querySelector(`input[name="city-selection-${id}"]:checked`);
                                if (!selectedRadio) {
                                    alert("Por favor, selecciona una ciudad de la lista.");
                                    return;
                                }
                                const selectedCityId = parseInt(selectedRadio.value, 10);
                                const selectedCityName = selectedRadio.parentElement.querySelector('label b').textContent;

                                const venueToUpdate = W.model.venues.getObjectById(id);
                                if (!venueToUpdate) {
                                    alert("Error: No se pudo encontrar el lugar para actualizar. Puede que ya no esté visible.");
                                    modal.remove();
                                    return;
                                }

                                try {
                                    const UpdateObject = require("Waze/Action/UpdateObject");
                                    const action = new UpdateObject(venueToUpdate, { cityID: selectedCityId });
                                    W.model.actionManager.add(action);

                                    const row = document.querySelector(`tr[data-place-id="${id}"]`);
                                    if (row) {
                                        row.dataset.addressChanged = 'true';

                                        const iconToUpdate = row.querySelector('.city-status-icon');
                                        if (iconToUpdate) {
                                            iconToUpdate.innerHTML = '✅';
                                            iconToUpdate.style.color = 'green';
                                            iconToUpdate.title = `Ciudad asignada: ${selectedCityName}`;
                                            iconToUpdate.style.pointerEvents = 'none';
                                        }

                                        updateApplyButtonState(row, original);
                                    }

                                    modal.remove();

                                    showTemporaryMessage("Ciudad asignada correctamente. No olvides Guardar los cambios.", 4000, 'success');

                                } catch (e) {
                                    console.error("[WME PLN] Error al crear o ejecutar la acción de actualizar ciudad:", e);
                                    alert("Ocurrió un error al intentar asignar la ciudad: " + e.message);
                                    modal.remove();
                                }
                            });


                            const closeBtn = document.createElement("button");
                            closeBtn.textContent = "Cerrar";
                            closeBtn.style.padding = "8px 16px";
                            closeBtn.style.background = "#888";
                            closeBtn.style.color = "#fff";
                            closeBtn.style.border = "none";
                            closeBtn.style.borderRadius = "4px";
                            closeBtn.style.cursor = "pointer";
                            closeBtn.style.fontWeight = "bold";
                            closeBtn.addEventListener("click", () => modal.remove());

                            buttonWrapper.appendChild(applyBtn);
                            buttonWrapper.appendChild(closeBtn);
                            modal.appendChild(buttonWrapper);

                            closeBtn.style.padding = "8px 16px";
                            closeBtn.style.background = "#888";
                            closeBtn.style.color = "#fff";
                            closeBtn.style.border = "none";
                            closeBtn.style.borderRadius = "4px";
                            closeBtn.style.cursor = "pointer";
                            closeBtn.style.fontWeight = "bold";
                            closeBtn.addEventListener("click", () => modal.remove());

                            buttonWrapper.appendChild(applyBtn);
                            buttonWrapper.appendChild(closeBtn);
                            modal.appendChild(buttonWrapper);

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

                    cellContentWrapper.appendChild(cityStatusIconSpan);
                    typeCityCell.appendChild(cellContentWrapper);
                    row.appendChild(typeCityCell);
                    const lockCell = document.createElement("td");
                    lockCell.textContent = lockRankEmoji;
                    lockCell.style.textAlign = "center";
                    lockCell.style.padding = "4px";
                    lockCell.style.width = "40px";
                    lockCell.style.fontSize = "18px";
                    row.appendChild(lockCell);
                    // Editor
                    const editorCell = document.createElement("td");
                    editorCell.textContent = editor || "Desconocido";
                    editorCell.title = "Último editor";
                    editorCell.style.padding = "4px";
                    editorCell.style.width = "140px";
                    editorCell.style.textAlign = "center";
                    row.appendChild(editorCell);
                    // Nombre Actual
                    const originalCell = document.createElement("td");
                    const inputOriginal = document.createElement("textarea");
                    inputOriginal.rows = 3; inputOriginal.readOnly = true;
                    inputOriginal.style.whiteSpace = "pre-wrap";
                    const venueLive = W.model.venues.getObjectById(id);
                    const currentLiveName = venueLive?.attributes?.name?.value || venueLive?.attributes?.name || "";
                    inputOriginal.value = currentLiveName || original;
                    if (currentLiveName.trim().toLowerCase() !== normalized.trim().toLowerCase()) {
                        inputOriginal.style.border = "1px solid red";
                        inputOriginal.title = "Este nombre es distinto del original mostrado en el panel";
                    }
                    inputOriginal.disabled = true;
                    inputOriginal.style.width = "270px";
                    inputOriginal.style.backgroundColor = "#eee";
                    originalCell.style.padding = "4px";
                    originalCell.style.width = "270px";
                    originalCell.style.display = "flex";
                    originalCell.style.alignItems = "flex-start";
                    originalCell.style.verticalAlign = "middle";
                    inputOriginal.style.flex = "1";
                    inputOriginal.style.height = "100%";
                    inputOriginal.style.boxSizing = "border-box";
                    originalCell.appendChild(inputOriginal);
                    row.appendChild(originalCell);
                    const alertCell = document.createElement("td");
                    alertCell.style.width = "30px";
                    alertCell.style.textAlign = "center";
                    alertCell.style.verticalAlign = "middle";
                    alertCell.style.padding = "4px";

                    // Alerta de Parqueadero Punto (debe ser Área)
                    if (item.isParkingPointError) {
                        const parkingAlert = document.createElement("div");
                        parkingAlert.textContent = "🅿️";
                        parkingAlert.title = "⚠️ Parqueadero tipo PUNTO. Debe ser ÁREA.\nHaz click en el icono de tipo (⊙) para convertirlo.";
                        parkingAlert.style.cursor = "help";
                        parkingAlert.style.fontSize = "16px";
                        alertCell.appendChild(parkingAlert);
                    }

                    if (isDuplicate) {
                        const warningIcon = document.createElement("span");
                        warningIcon.textContent = " ⚠️";
                        warningIcon.style.fontSize = "16px";
                        let tooltipText = `Nombre de lugar duplicado cercano.`;
                        if (duplicatePartners && duplicatePartners.length > 0) {
                            const partnerDetails = duplicatePartners.map(p => `Línea ${p.line}: "${p.originalName}"`).join(", ");
                            tooltipText += ` Duplicado(s) con: ${partnerDetails}.`;
                        }
                        else {
                            tooltipText += ` No se encontraron otros duplicados cercanos específicos.`;
                        }
                        warningIcon.title = tooltipText;
                        alertCell.appendChild(warningIcon);
                    }
                    // Lógica para el icono de advertencia por horarios que se cruzan
                    if (hasOverlappingHours) {
                        const clockIcon = document.createElement("span");
                        clockIcon.textContent = " ⏰";
                        clockIcon.style.fontSize = "16px";
                        clockIcon.style.color = "red";
                        clockIcon.title = "¡Alerta! Este lugar tiene horarios que se cruzan.";
                        alertCell.appendChild(clockIcon);
                    }
                    // Lógica para el icono de advertencia por ediciones pendientes
                    if (hasPendingEdits) {
                        const editLink = document.createElement("a");
                        editLink.href = placePermalink;
                        editLink.style.display = "inline-flex";
                        editLink.style.alignItems = "center";
                        editLink.style.justifyContent = "center";
                        editLink.style.textDecoration = "none";
                        editLink.style.marginLeft = "2px";
                        editLink.title = `¡Alerta! Este lugar tiene ediciones pendientes de aprobación.\nPerma: ${placePermalink}`;
                        editLink.dataset.permalink = placePermalink;
                        editLink.addEventListener("click", (event) => {
                            handlePlaceLinkNavigation(event, id, original, placePermalink, venueSDKForRender);
                        });

                        const editIcon = document.createElement("span");
                        editIcon.textContent = "📝";
                        editIcon.style.fontSize = "16px";
                        editIcon.style.color = "#ff9800"; // Un color naranja para destacar

                        editLink.appendChild(editIcon);
                        alertCell.appendChild(editLink);
                    }
                    row.appendChild(alertCell);
                    const suggestionCell = document.createElement("td");
                    suggestionCell.style.display = "flex";
                    suggestionCell.style.flexDirection = "column"; // Para que los botones queden debajo
                    suggestionCell.style.alignItems = "flex-start";
                    suggestionCell.style.justifyContent = "flex-start";
                    suggestionCell.style.padding = "4px";
                    suggestionCell.style.width = "270px";
                    const inputReplacement = document.createElement("textarea");
                    inputReplacement.className = 'replacement-input';

                    // Manejar "Requiere corrección manual" como placeholder
                    if (normalized === 'Requiere corrección manual') {
                        // NO asignar el texto como value, usar placeholder
                        inputReplacement.value = '';
                        inputReplacement.placeholder = 'Requiere corrección manual';
                        inputReplacement.style.border = "2px solid red";
                        // El color del placeholder se controla con CSS ::placeholder
                        inputReplacement.style.setProperty('--placeholder-color', 'red');
                    } else {
                        // Para nombres normales, asignar el valor
                        try {
                            inputReplacement.value = normalized;
                        }
                        catch (_) {
                            inputReplacement.value = normalized;
                        }
                    }

                    inputReplacement.style.width = "100%";
                    inputReplacement.style.height = "100%";
                    inputReplacement.style.boxSizing = "border-box";
                    inputReplacement.style.whiteSpace = "pre-wrap";
                    inputReplacement.rows = 3;

                    suggestionCell.appendChild(inputReplacement);

                    // ✨ BOTONES DE PALABRAS CLAVE POR CATEGORÍA (28-nov-2024)

                    // Crear contenedor para botones de prefijos rápidos
                    const quickPrefixContainer = document.createElement("div");
                    quickPrefixContainer.style.display = "flex";
                    quickPrefixContainer.style.flexWrap = "wrap";
                    quickPrefixContainer.style.gap = "2px";
                    quickPrefixContainer.style.marginTop = "2px";
                    quickPrefixContainer.style.width = "100%";
                    quickPrefixContainer.className = "quick-prefix-buttons"; // Identificador para encontrarlo

                    // Función para generar botones según categoría
                    const generatePrefixButtons = (categoryKey) => {
                        quickPrefixContainer.innerHTML = ""; // Limpiar botones existentes

                        if (!window.dynamicCategoryRules || !categoryKey) return;

                        const categoryRule = window.dynamicCategoryRules.find(r =>
                            r.categoryKey.toUpperCase() === categoryKey.toUpperCase()
                        );

                        // ✨ LÓGICA DINÁMICA (28-nov-2024): Usar palabras de columna F (prefixWords)
                        if (categoryRule && categoryRule.prefixWords) {
                            // Las palabras están separadas por ";"
                            const words = categoryRule.prefixWords.split(';')
                                .map(w => w.trim())
                                .filter(w => w.length > 0);


                            words.forEach(word => {
                                // ✨ VERIFICAR si la palabra ya está en el nombre actual
                                // Solo crear el botón si la palabra NO está presente
                                const currentName = inputReplacement.value.trim().toLowerCase();
                                const wordLower = word.toLowerCase();
                                const nameWords = currentName.split(/\s+/);

                                // Si alguna palabra del nombre contiene esta keyword, no crear el botón
                                if (nameWords.some(w => w.includes(wordLower))) {
                                    return; // Saltar este botón
                                }

                                // Usar la palabra EXACTA como pidió el usuario
                                const btn = document.createElement("button");
                                btn.textContent = word;
                                btn.style.padding = "1px 4px";
                                btn.style.fontSize = "9px";
                                btn.style.backgroundColor = "#007bff";
                                btn.style.color = "white";
                                btn.style.border = "none";
                                btn.style.borderRadius = "2px";
                                btn.style.cursor = "pointer";
                                btn.style.whiteSpace = "nowrap";
                                btn.style.transition = "background-color 0.2s";
                                btn.title = `Click para añadir "${word}" al inicio del nombre`;

                                btn.addEventListener("mouseenter", () => {
                                    btn.style.backgroundColor = "#0056b3";
                                });
                                btn.addEventListener("mouseleave", () => {
                                    btn.style.backgroundColor = "#007bff";
                                });

                                btn.addEventListener("click", (e) => {
                                    e.preventDefault();
                                    e.stopPropagation();

                                    // Desenfocar el input para evitar que se inserte en la posición del cursor
                                    inputReplacement.blur();

                                    let currentValue = inputReplacement.value.trim();

                                    // Buscar y reemplazar cualquier keyword existente en el nombre (no solo al inicio)
                                    let wasReplaced = false;
                                    words.forEach(kw => {
                                        // Saltar la palabra actual que estamos añadiendo
                                        if (kw === word) return;

                                        const kwCap = kw.charAt(0).toUpperCase() + kw.slice(1).toLowerCase();
                                        // ✨ Usar regex ESTRICTO con lookbehind/lookahead para evitar coincidencias parciales
                                        // Solo coincide si la palabra está rodeada de NO-word characters
                                        const kwPattern = new RegExp(`(?<![\\p{L}\\p{N}_-])${kwCap.replace(/[.*+?^${}()|[\]\\]/g, '\\$&')}(?![\\p{L}\\p{N}_-])`, 'giu');
                                        if (kwPattern.test(currentValue)) {
                                            currentValue = currentValue.replace(kwPattern, word);
                                            wasReplaced = true;
                                        }
                                    });

                                    // Si no se reemplazó ninguna, añadir al inicio
                                    let newValue;
                                    if (!wasReplaced) {
                                        // SIEMPRE al inicio, sin importar dónde esté el cursor
                                        newValue = currentValue.length > 0
                                            ? `${word} ${currentValue.trim()}`
                                            : word;
                                    } else {
                                        newValue = currentValue;
                                    }

                                    // Actualizar valor y mover cursor al final
                                    inputReplacement.value = newValue;
                                    inputReplacement.setSelectionRange(newValue.length, newValue.length);
                                    inputReplacement.dispatchEvent(new Event('input'));

                                    // Feedback visual
                                    btn.style.backgroundColor = "#28a745";
                                    setTimeout(() => {
                                        btn.style.backgroundColor = "#007bff";
                                    }, 300);
                                });

                                quickPrefixContainer.appendChild(btn);
                            });
                        }

                        // ✨ LÓGICA DE ARTÍCULOS (30-nov-2024): Botones morados para capitalizar artículos
                        const articles = ['el', 'la', 'los', 'las', 'un', 'una', 'unos', 'unas', 'del', 'al', 'de'];
                        const currentNameLower = inputReplacement.value.trim().toLowerCase();
                        const nameWords = currentNameLower.split(/\s+/);

                        // Encontrar artículos presentes en el nombre
                        const foundArticles = articles.filter(art => nameWords.includes(art));

                        if (foundArticles.length > 0) {
                            // Separador visual si hay botones previos
                            if (quickPrefixContainer.children.length > 0) {
                                const sep = document.createElement("span");
                                sep.style.borderLeft = "1px solid #ccc";
                                sep.style.margin = "0 4px";
                                quickPrefixContainer.appendChild(sep);
                            }

                            foundArticles.forEach(art => {
                                const artCap = art.charAt(0).toUpperCase() + art.slice(1);
                                const btn = document.createElement("button");
                                btn.textContent = artCap;
                                btn.style.padding = "1px 4px";
                                btn.style.fontSize = "9px";
                                btn.style.backgroundColor = "#6f42c1"; // Morado
                                btn.style.color = "white";
                                btn.style.border = "none";
                                btn.style.borderRadius = "2px";
                                btn.style.cursor = "pointer";
                                btn.style.whiteSpace = "nowrap";
                                btn.style.transition = "background-color 0.2s";
                                btn.title = `Click para cambiar "${art}" a "${artCap}"`;

                                btn.addEventListener("mouseenter", () => {
                                    btn.style.backgroundColor = "#5a32a3";
                                });
                                btn.addEventListener("mouseleave", () => {
                                    btn.style.backgroundColor = "#6f42c1";
                                });

                                btn.addEventListener("click", (e) => {
                                    e.preventDefault();
                                    e.stopPropagation();
                                    inputReplacement.blur();

                                    let currentValue = inputReplacement.value;
                                    // Reemplazar solo la instancia exacta de la palabra minúscula
                                    // Usamos regex con word boundaries para no reemplazar dentro de otras palabras
                                    const regex = new RegExp(`\\b${art}\\b`, 'g');
                                    const newValue = currentValue.replace(regex, artCap);

                                    inputReplacement.value = newValue;
                                    inputReplacement.dispatchEvent(new Event('input'));

                                    // Feedback visual
                                    btn.style.backgroundColor = "#28a745";
                                    setTimeout(() => {
                                        btn.style.backgroundColor = "#6f42c1";
                                    }, 300);
                                });

                                quickPrefixContainer.appendChild(btn);
                            });
                        }
                    };

                    // Generar botones iniciales
                    generatePrefixButtons(currentCategoryKey);

                    // Guardar referencia para poder regenerar después
                    suggestionCell.dataset.regenerateButtons = id; // Usar el ID del lugar
                    suggestionCell._generatePrefixButtons = generatePrefixButtons;

                    // Añadir contenedor (siempre, aunque esté vacío inicialmente)
                    suggestionCell.appendChild(quickPrefixContainer);

                    // Añade la celda de sugerencias al renglón


                    function debounce(func, delay) {
                        let timeout;
                        return function (...args) {
                            clearTimeout(timeout);
                            timeout = setTimeout(() => func.apply(this, args), delay);
                        };
                    }
                    const checkAndUpdateApplyButton = () => {
                        const nameIsDifferent = inputReplacement.value.trim() !== original.trim();
                        const categoryWasChanged = row.dataset.categoryChanged === 'true';
                        if (nameIsDifferent || categoryWasChanged) {
                            applyButton.disabled = false;
                            applyButton.style.opacity = "1";
                            const successIcon = applyButtonWrapper.querySelector('span');
                            if (successIcon) successIcon.remove();
                        }
                        else {
                            applyButton.disabled = true;
                            applyButton.style.opacity = "0.5";
                        }
                    };

                    inputReplacement.addEventListener('input', debounce(checkAndUpdateApplyButton, 300));
                    let autoApplied = false;
                    if (Object.values(allSuggestions).flat().some(s => s.fuente === 'excluded' && s.similarity === 1)) {
                        autoApplied = true;
                    }
                    if (autoApplied) {
                        inputReplacement.style.backgroundColor = "#c8e6c9";
                        inputReplacement.title = "Reemplazo automático aplicado (palabra especial con 100% similitud)";
                    }
                    else if (Object.values(allSuggestions).flat().some(s => s.fuente === 'excluded')) {
                        inputReplacement.style.backgroundColor = "#fff3cd";
                        inputReplacement.title = "Contiene palabra especial reemplazada";
                    }
                    function debounce(func, delay) {
                        let timeout;
                        return function (...args) {
                            clearTimeout(timeout);
                            timeout = setTimeout(() => func.apply(this, args), delay);
                        };
                    }
                    inputReplacement.addEventListener('input', debounce(() => {
                        if (inputReplacement.value.trim() !== original) {
                            applyButton.disabled = false;
                            applyButton.style.color = "";
                        }
                        else {
                            applyButton.disabled = true;
                            applyButton.style.color = "#bbb";
                        }
                    }, 300));
                    inputOriginal.addEventListener('input', debounce(() => {
                    }, 300));
                    const suggestionListCell = document.createElement("td");
                    suggestionListCell.style.padding = "4px";
                    suggestionListCell.style.width = "180px";
                    const suggestionContainer = document.createElement('div');
                    const palabrasYaProcesadas = new Set();
                    const currentPlaceSuggestions = allSuggestions[id];
                    if (currentPlaceSuggestions) {
                        Object.entries(currentPlaceSuggestions).forEach(([originalWordForThisPlace, suggestionsArray]) => {
                            if (Array.isArray(suggestionsArray)) {
                                suggestionsArray.forEach(s => {
                                    let icono = '';
                                    let textoSugerencia = '';
                                    let colorFondo = '#f9f9f9';
                                    let esSugerenciaValida = false;
                                    let palabraAReemplazar = originalWordForThisPlace;
                                    let palabraAInsertar = s.word;
                                    switch (s.fuente) {
                                        case 'original_preserved':
                                            esSugerenciaValida = true;
                                            icono = '⚙️';
                                            textoSugerencia = `¿"${originalWordForThisPlace}" x "${s.word}"?`;
                                            colorFondo = '#f0f0f0';
                                            palabraAReemplazar = originalWordForThisPlace;
                                            palabraAInsertar = s.word;
                                            break;
                                        case 'excluded':
                                            if (s.similarity < 1 || (s.similarity === 1 && originalWordForThisPlace.toLowerCase() !== s.word.toLowerCase())) {
                                                esSugerenciaValida = true;
                                                icono = '🏷️';
                                                textoSugerencia = `¿"${originalWordForThisPlace}" x "${s.word}"? (sim. ${(s.similarity * 100).toFixed(0)}%)`;
                                                colorFondo = '#f3f9ff';
                                                palabraAReemplazar = originalWordForThisPlace;
                                                palabraAInsertar = s.word;
                                                palabrasYaProcesadas.add(originalWordForThisPlace.toLowerCase());
                                            }
                                            break;
                                        case 'dictionary':
                                            esSugerenciaValida = true;
                                            icono = '📘';
                                            colorFondo = '#e6ffe6';
                                            const normalizedSuggestedWordForDisplay = normalizeWordInternal(s.word, true, false);

                                            textoSugerencia = `¿"${originalWordForThisPlace}" x "${normalizedSuggestedWordForDisplay}"? (sim. ${(s.similarity * 100).toFixed(0)}%)`;
                                            palabraAReemplazar = originalWordForThisPlace;
                                            palabraAInsertar = normalizedSuggestedWordForDisplay;
                                            break;

                                        case 'dictionary_tilde':
                                            esSugerenciaValida = true;
                                            icono = '✍️';
                                            colorFondo = '#ffe6e6';
                                            textoSugerencia = `¿"${originalWordForThisPlace}" x "${s.word}"? (Corregir Tilde)`;
                                            palabraAReemplazar = originalWordForThisPlace;
                                            palabraAInsertar = s.word;
                                            break;
                                    }
                                    if (esSugerenciaValida) {
                                        const suggestionDiv = document.createElement("div");
                                        suggestionDiv.innerHTML = `${icono} ${textoSugerencia}`;
                                        suggestionDiv.style.cursor = "pointer";
                                        suggestionDiv.style.padding = "2px 4px";
                                        suggestionDiv.style.margin = "2px 0";
                                        suggestionDiv.style.border = "1px solid #ddd";
                                        suggestionDiv.style.borderRadius = "3px";
                                        suggestionDiv.style.backgroundColor = colorFondo;

                                        suggestionDiv.addEventListener("click", () => {
                                            const currentSuggestedValue = inputReplacement.value;
                                            const searchRegex = new RegExp("\\b" + escapeRegExp(palabraAReemplazar) + "\\b", "gi");
                                            const newSuggestedValue = currentSuggestedValue.replace(searchRegex, palabraAInsertar);
                                            if (inputReplacement.value !== newSuggestedValue) {
                                                inputReplacement.value = newSuggestedValue;
                                            }
                                            checkAndUpdateApplyButton();
                                        });
                                        suggestionContainer.appendChild(suggestionDiv);
                                    }
                                });
                            }
                            else {
                                console.warn(`[WME_PLN][DEBUG] suggestionsArray para "${originalWordForThisPlace}" no es un array o es undefined:`, suggestionsArray);
                            }
                        });
                    }
                    suggestionListCell.appendChild(suggestionContainer);
                    row.appendChild(suggestionCell);
                    row.appendChild(suggestionListCell);
                    const categoryCell = document.createElement("td");
                    categoryCell.style.padding = "4px";
                    categoryCell.style.width = "130px";
                    categoryCell.style.textAlign = "center";
                    const currentCategoryDiv = document.createElement("div");
                    currentCategoryDiv.style.display = "flex";
                    currentCategoryDiv.style.flexDirection = "column";
                    currentCategoryDiv.style.alignItems = "center";
                    currentCategoryDiv.style.gap = "2px";
                    const currentCategoryText = document.createElement("span");
                    currentCategoryText.textContent = currentCategoryTitle;
                    currentCategoryText.title = `Categoría Actual: ${currentCategoryTitle}`;
                    currentCategoryDiv.appendChild(currentCategoryText);
                    const currentCategoryIconDisplay = document.createElement("span");
                    currentCategoryIconDisplay.textContent = currentCategoryIcon;
                    currentCategoryIconDisplay.style.fontSize = "20px";
                    currentCategoryDiv.appendChild(currentCategoryIconDisplay);
                    categoryCell.appendChild(currentCategoryDiv);
                    row.appendChild(categoryCell);
                    const recommendedCategoryCell = document.createElement("td");
                    recommendedCategoryCell.style.padding = "4px";
                    recommendedCategoryCell.style.width = "130px";
                    recommendedCategoryCell.style.textAlign = "left";
                    const categoryDropdown = createRecommendedCategoryDropdown(
                        id,
                        currentCategoryKey,
                        dynamicCategorySuggestions
                    );
                    recommendedCategoryCell.appendChild(categoryDropdown);
                    row.appendChild(recommendedCategoryCell);

                    const actionCell = document.createElement("td");
                    actionCell.style.padding = "4px";
                    actionCell.style.width = "120px";
                    const buttonGroup = document.createElement("div");
                    buttonGroup.style.display = "flex";
                    buttonGroup.style.flexDirection = "column";
                    buttonGroup.style.gap = "4px";
                    buttonGroup.style.alignItems = "flex-start";
                    const commonButtonStyle = {
                        width: "40px",
                        height: "30px",
                        minWidth: "40px",
                        minHeight: "30px",
                        padding: "4px",
                        border: "1px solid #ccc",
                        borderRadius: "4px",
                        backgroundColor: "#f0f0f0",
                        color: "#555",
                        cursor: "pointer",
                        fontSize: "18px",
                        display: "flex",
                        justifyContent: "center",
                        alignItems: "center",
                        boxSizing: "border-box"
                    };
                    const applyButton = document.createElement("button");
                    Object.assign(applyButton.style, commonButtonStyle);
                    applyButton.textContent = "✔";
                    applyButton.title = "Aplicar sugerencia";
                    applyButton.disabled = true;
                    applyButton.style.opacity = "0.5";
                    const applyButtonWrapper = document.createElement("div");
                    applyButtonWrapper.style.display = "flex";
                    applyButtonWrapper.style.alignItems = "center";
                    applyButtonWrapper.style.gap = "5px";
                    applyButtonWrapper.appendChild(applyButton);
                    buttonGroup.appendChild(applyButtonWrapper);
                    let deleteButton = document.createElement("button");
                    Object.assign(deleteButton.style, commonButtonStyle);
                    deleteButton.textContent = "🗑️";
                    deleteButton.title = "Eliminar lugar";
                    const deleteButtonWrapper = document.createElement("div");
                    Object.assign(deleteButtonWrapper.style, {
                        display: "flex",
                        alignItems: "center",
                        gap: "5px"
                    });
                    deleteButtonWrapper.appendChild(deleteButton);
                    buttonGroup.appendChild(deleteButtonWrapper);
                    const addToExclusionBtn = document.createElement("button");
                    Object.assign(addToExclusionBtn.style, commonButtonStyle);
                    addToExclusionBtn.textContent = "🏷️";
                    addToExclusionBtn.title = "Marcar palabra como especial (no se modifica)";
                    buttonGroup.appendChild(addToExclusionBtn);
                    actionCell.appendChild(buttonGroup);
                    row.appendChild(actionCell);

                    // Contenedor para alinear el botón de excluir y el nuevo checkbox
                    const excludeContainer = document.createElement("div");
                    excludeContainer.style.display = "flex";
                    excludeContainer.style.alignItems = "center";
                    excludeContainer.style.gap = "8px"; // Espacio entre el botón y el checkbox

                    const excludePlaceBtn = document.createElement("button");
                    Object.assign(excludePlaceBtn.style, commonButtonStyle);
                    excludePlaceBtn.textContent = "📵";
                    excludePlaceBtn.title = "Excluir este lugar inmediatamente"; // Tooltip actualizado

                    const excludeOnApplyCheckbox = document.createElement("input");
                    excludeOnApplyCheckbox.type = "checkbox";
                    excludeOnApplyCheckbox.id = `pln-exclude-checkbox-${id}`; // ID único
                    excludeOnApplyCheckbox.title = "Marcar para excluir al presionar 'Aplicar'";
                    excludeOnApplyCheckbox.style.width = "18px";
                    excludeOnApplyCheckbox.style.height = "18px";
                    excludeOnApplyCheckbox.style.cursor = "pointer";

                    // LEER LA PREFERENCIA LOCAL DEL ITEM O USAR EL DEFAULT GLOBAL
                    const defaultExcludeOnApply = localStorage.getItem('wme_pln_default_exclude_on_apply') !== 'false';

                    // Inicializar propiedad local si no existe
                    if (typeof item._excludeOnApply === 'undefined') {
                        item._excludeOnApply = defaultExcludeOnApply;
                    }

                    excludeOnApplyCheckbox.checked = item._excludeOnApply;

                    // ACTUALIZAR SOLO EL ESTADO LOCAL AL CAMBIAR
                    excludeOnApplyCheckbox.addEventListener('change', () => {
                        item._excludeOnApply = excludeOnApplyCheckbox.checked;
                    });

                    excludeContainer.appendChild(excludePlaceBtn);
                    excludeContainer.appendChild(excludeOnApplyCheckbox);
                    buttonGroup.appendChild(excludeContainer); // Añadimos el contenedor al grupo de botones

                    actionCell.appendChild(buttonGroup);
                    row.appendChild(actionCell);
                    // Eventos de los botones
                    applyButton.addEventListener("click", async () => {
                        // Validación de seguridad: No permitir guardar el nombre placeholder
                        if (inputReplacement.value.trim() === "Requiere corrección manual") {
                            alert("⚠️ Debes modificar el nombre antes de guardar.\n\n'Requiere corrección manual' es solo un indicador y no puede ser usado como nombre del lugar.");
                            return;
                        }

                        const row = applyButton.closest('tr');
                        const venueObj = W.model.venues.getObjectById(id);
                        let anActionWasPerformed = false; // Usaremos esta bandera para saber si se hizo algo

                        // 1. Manejar la exclusión si el checkbox está marcado
                        const excludeCheckbox = document.getElementById(`pln-exclude-checkbox-${id}`);
                        if (excludeCheckbox && excludeCheckbox.checked) {
                            const placeNameToExclude = row.querySelectorAll('textarea')[1].value || original || `ID: ${id}`;
                            excludedPlaces.set(id, placeNameToExclude);

                            saveExcludedPlacesToLocalStorage();
                            showTemporaryMessage(`'${placeNameToExclude}' añadido a excluidos`, 3000, 'warning');
                            //plnToast(`'${placeNameToExclude}' añadido a excluidos.`, 2500);
                            anActionWasPerformed = true; // Marcamos que se realizó una acción
                        }

                        // 2. Manejar cambios de nombre y categoría
                        const newName = inputReplacement.value.trim();
                        const nameWasChanged = venueObj && (newName !== (venueObj.attributes.name?.value || venueObj.attributes.name || ""));
                        const categoryWasChanged = row.dataset.categoryChanged === 'true';

                        if (nameWasChanged || categoryWasChanged) {
                            if (!venueObj) {
                                plnToast("Error: No se pudo aplicar el cambio porque el lugar no está disponible.", 4000);
                            }
                            else {
                                try {
                                    if (nameWasChanged) {
                                        const UpdateObject = require("Waze/Action/UpdateObject");
                                        const action = new UpdateObject(venueObj, { name: newName });
                                        W.model.actionManager.add(action);
                                    }
                                    showTemporaryMessage("Cambios aplicados. Presione 'Guardar' en WME.", 3000, 'success');
                                    recordNormalizationEvent();
                                    anActionWasPerformed = true; // Marcamos que se realizó una acción
                                }
                                catch (e) {
                                    alert("Error al actualizar: " + e.message);
                                    console.error("[WME_PLN] Error al actualizar lugar:", e);
                                }
                            }
                        }

                        // 3. Procesamiento visual final SOLO SI se realizó alguna acción
                        if (anActionWasPerformed) {
                            markRowAsProcessed(row, 'applied'); // Marcamos la fila como procesada
                            if (window.checkBatchCompletion) window.checkBatchCompletion(); // Actualizar contador
                        }
                        else {
                            // Este mensaje solo aparecerá si no se hizo NADA
                            showTemporaryMessage("No hay cambios para aplicar.", 3000, 'warning');
                        }
                    });
                    deleteButton.addEventListener("click", () => {
                        const confirmModal = document.createElement("div");
                        confirmModal.style.position = "fixed";
                        confirmModal.style.top = "50%";
                        confirmModal.style.left = "50%";
                        confirmModal.style.transform = "translate(-50%, -50%)";
                        confirmModal.style.background = "#fff";
                        confirmModal.style.border = "1px solid #aad";
                        confirmModal.style.padding = "28px 32px 20px 32px";
                        confirmModal.style.zIndex = "20000";
                        confirmModal.style.boxShadow = "0 4px 24px rgba(0,0,0,0.18)";
                        confirmModal.style.fontFamily = "sans-serif";
                        confirmModal.style.borderRadius = "10px";
                        confirmModal.style.textAlign = "center";
                        confirmModal.style.minWidth = "340px";
                        const iconElement = document.createElement("div");
                        iconElement.innerHTML = "⚠️";
                        iconElement.style.fontSize = "38px";
                        iconElement.style.marginBottom = "10px";
                        confirmModal.appendChild(iconElement);
                        const message = document.createElement("div");
                        const venue = W.model.venues.getObjectById(id);
                        const placeName = venue?.attributes?.name?.value || venue?.attributes?.name || "este lugar";
                        message.innerHTML = `<b>¿Eliminar "${placeName}"?</b>`;
                        message.style.fontSize = "20px";
                        message.style.marginBottom = "8px";
                        confirmModal.appendChild(message);
                        const nameDiv = document.createElement("div");
                        nameDiv.textContent = `"${placeName}"`;
                        nameDiv.style.fontSize = "15px";
                        nameDiv.style.color = "#007bff";
                        nameDiv.style.marginBottom = "18px";
                        confirmModal.appendChild(nameDiv);
                        const buttonWrapper = document.createElement("div");
                        buttonWrapper.style.display = "flex";
                        buttonWrapper.style.justifyContent = "center";
                        buttonWrapper.style.gap = "18px";
                        const cancelBtn = document.createElement("button");
                        cancelBtn.textContent = "Cancelar";
                        cancelBtn.style.padding = "7px 18px";
                        cancelBtn.style.background = "#eee";
                        cancelBtn.style.border = "none";
                        cancelBtn.style.borderRadius = "4px";
                        cancelBtn.style.cursor = "pointer";
                        cancelBtn.addEventListener("click", () => confirmModal.remove());
                        const confirmBtn = document.createElement("button");
                        confirmBtn.textContent = "Eliminar";
                        confirmBtn.style.padding = "7px 18px";
                        confirmBtn.style.background = "#d9534f";
                        confirmBtn.style.color = "#fff";
                        confirmBtn.style.border = "none";
                        confirmBtn.style.borderRadius = "4px";
                        confirmBtn.style.cursor = "pointer";
                        confirmBtn.style.fontWeight = "bold";
                        confirmBtn.addEventListener("click", () => {
                            const venue = W.model.venues.getObjectById(id);
                            if (!venue) {
                                console.error("[WME_PLN]El lugar no está disponible o ya fue eliminado.");
                                confirmModal.remove();
                                return;
                            }
                            try {
                                const DeleteObject = require("Waze/Action/DeleteObject");
                                const action = new DeleteObject(venue);
                                W.model.actionManager.add(action);
                                recordNormalizationEvent();
                                const row = deleteButton.closest('tr');
                                markRowAsProcessed(row, 'deleted');
                                if (window.checkBatchCompletion) window.checkBatchCompletion();

                                deleteButton.disabled = true;
                                deleteButton.style.color = "#bbb";
                                deleteButton.style.opacity = "0.5";
                                applyButton.disabled = true;
                                applyButton.style.color = "#bbb";
                                applyButton.style.opacity = "0.5";
                                const successIcon = document.createElement("span");
                                successIcon.textContent = " 🗑️";
                                successIcon.style.marginLeft = "0";
                                successIcon.style.fontSize = "20px";
                                deleteButtonWrapper.appendChild(successIcon);
                            }
                            catch (e) {
                                console.error("[WME_PLN] Error al eliminar lugar: " + e.message, e);
                            }
                            confirmModal.remove();
                        });
                        buttonWrapper.appendChild(cancelBtn);
                        buttonWrapper.appendChild(confirmBtn);
                        confirmModal.appendChild(buttonWrapper);
                        document.body.appendChild(confirmModal);
                    });
                    // Add to Exclusion Button
                    addToExclusionBtn.addEventListener("click", () => {
                        const words = original.split(/\s+/);
                        const modal = document.createElement("div");

                        modal.style.position = "fixed";
                        modal.style.top = "50%";
                        modal.style.left = "50%";
                        modal.style.transform = "translate(-50%, -50%)";
                        modal.style.background = "#fff";
                        modal.style.border = "1px solid #aad";
                        modal.style.padding = "28px 32px 20px 32px";
                        modal.style.zIndex = "20000";
                        modal.style.boxShadow = "0 4px 24px rgba(0,0,0,0.18)";
                        modal.style.fontFamily = "sans-serif";
                        modal.style.borderRadius = "10px";
                        modal.style.textAlign = "center";
                        modal.style.minWidth = "340px";
                        const title = document.createElement("h4");
                        title.textContent = "Agregar palabra a especiales";
                        modal.appendChild(title);
                        const instructions = document.createElement("p");
                        const list = document.createElement("ul");
                        list.style.listStyle = "none";
                        list.style.padding = "0";
                        words.forEach(w => {
                            const wordToTest = w.trim();
                            if (wordToTest === '') return;

                            const lowerW = w.trim().toLowerCase();
                            const isCommon = commonWords.has(lowerW);

                            // ✨ OPTIMIZACIÓN: Usar cache en lugar de Array.from + búsqueda directa
                            const lookup = isWordExcludedOrInDictionary(lowerW);
                            const isExcluded = lookup.isExcluded;
                            const isInDictionary = lookup.isInDictionary;

                            // Si la palabra es común, ya está excluida O está en el diccionario, no la mostramos.
                            if (isCommon || isExcluded || isInDictionary) return;


                            if (!/[a-zA-ZáéíóúÁÉÍÓÚñÑüÜ0-9]/.test(lowerW) || /^[^a-zA-Z0-9]+$/.test(lowerW)) return;
                            // Ya verificamos isExcluded arriba, no necesitamos hacerlo de nuevo
                            if (isCommon) return;
                            const li = document.createElement("li");
                            const checkbox = document.createElement("input");
                            checkbox.type = "checkbox";
                            checkbox.value = w;
                            checkbox.id = `cb-exc-${w.replace(/[^a-zA-Z0-9]/g, "")}`;
                            li.appendChild(checkbox);
                            const label = document.createElement("label");
                            label.htmlFor = checkbox.id;
                            label.appendChild(document.createTextNode(" " + w));
                            li.appendChild(label);
                            list.appendChild(li);
                        });
                        // Si después de filtrar no quedan palabras, se notifica al usuario.
                        if (list.childElementCount === 0) {
                            plnToast("No hay palabras nuevas o válidas para añadir a especiales.", 3000);
                            return;
                        }
                        modal.appendChild(list);
                        const confirmBtn = document.createElement("button");
                        confirmBtn.textContent = "Añadir Seleccionadas";
                        confirmBtn.addEventListener("click", () => {
                            const checked = modal.querySelectorAll("input[type=checkbox]:checked");
                            let wordsActuallyAdded = false;
                            checked.forEach(c => {
                                if (!excludedWords.has(c.value)) {
                                    excludedWords.add(c.value);
                                    wordsActuallyAdded = true;
                                }
                            });
                            if (wordsActuallyAdded) {
                                if (typeof renderExcludedWordsList === 'function') {
                                    const excludedListElement = document.getElementById("excludedWordsList");
                                    if (excludedListElement) {
                                        renderExcludedWordsList(excludedListElement);
                                    }
                                    else {
                                        renderExcludedWordsList();
                                    }
                                }
                            }
                            modal.remove();
                            if (wordsActuallyAdded) {
                                saveExcludedWordsToLocalStorage();
                                showTemporaryMessage("Palabra(s) añadida(s) a especiales y guardada(s).", 3000, 'success');
                            }
                            else {
                                showTemporaryMessage("No se seleccionaron palabras o ya estaban en la lista.", 3000, 'info');
                            }
                        });
                        modal.appendChild(confirmBtn);
                        const cancelBtn = document.createElement("button");
                        cancelBtn.textContent = "Cancelar";
                        cancelBtn.style.marginLeft = "8px";
                        cancelBtn.addEventListener("click", () => modal.remove());
                        modal.appendChild(cancelBtn);
                        document.body.appendChild(modal);
                        // --- Lógica de posicionamiento dinámico ---
                        const triggerButton = event.currentTarget;
                        if (triggerButton) {
                            const modalRect = modal.getBoundingClientRect();
                            const triggerRect = triggerButton.getBoundingClientRect();

                            // Abrimos el modal hacia la izquierda alineando su borde derecho con el botón
                            let newLeft = triggerRect.right - modalRect.width;
                            let newTop = triggerRect.bottom + 5; // 5px debajo del botón

                            if ((newTop + modalRect.height) > window.innerHeight) {
                                newTop = triggerRect.top - modalRect.height - 5; // Posicionar arriba si no cabe
                            }
                            if (newTop < 5) newTop = 5; // Ajuste final si se sale por arriba

                            // Mantener un margen mínimo dentro de la ventana
                            const horizontalPadding = 10;
                            if (newLeft < horizontalPadding) {
                                newLeft = horizontalPadding;
                            }
                            const maxLeft = window.innerWidth - modalRect.width - horizontalPadding;
                            if (newLeft > maxLeft) {
                                newLeft = maxLeft;
                            }

                            modal.style.left = `${newLeft}px`;
                            modal.style.top = `${newTop}px`;
                            modal.style.transform = 'none';
                        }
                    });
                    buttonGroup.appendChild(addToExclusionBtn);

                    // Exclude Place Button
                    excludePlaceBtn.addEventListener("click", (event) => { // <-- Añadimos "event"
                        const row = excludePlaceBtn.closest('tr');

                        const placeName = row.querySelector('textarea').value || original || `ID: ${id}`;

                        const confirmModal = document.createElement("div");
                        // Estilos básicos del modal
                        confirmModal.style.position = "fixed";
                        confirmModal.style.background = "#fff";
                        confirmModal.style.border = "1px solid #aad";
                        confirmModal.style.padding = "28px 32px 20px 32px";
                        confirmModal.style.zIndex = "20000";
                        confirmModal.style.boxShadow = "0 4px 24px rgba(0,0,0,0.18)";
                        confirmModal.style.fontFamily = "sans-serif";
                        confirmModal.style.borderRadius = "10px";
                        confirmModal.style.textAlign = "center";
                        confirmModal.style.minWidth = "340px";


                        const iconElement = document.createElement("div");
                        iconElement.innerHTML = "🚫";
                        iconElement.style.fontSize = "38px";
                        iconElement.style.marginBottom = "10px";
                        confirmModal.appendChild(iconElement);

                        const messageTitle = document.createElement("div");
                        messageTitle.innerHTML = `<b>¿Excluir "${placeName}"?</b>`;
                        messageTitle.style.fontSize = "20px";
                        messageTitle.style.marginBottom = "8px";
                        confirmModal.appendChild(messageTitle);

                        const explanationDiv = document.createElement("div");
                        explanationDiv.textContent = `Este lugar no aparecerá en futuras búsquedas del normalizador.`;
                        explanationDiv.style.fontSize = "15px";
                        explanationDiv.style.color = "#555";
                        explanationDiv.style.marginBottom = "18px";
                        confirmModal.appendChild(explanationDiv);

                        const buttonWrapper = document.createElement("div");
                        buttonWrapper.style.display = "flex";
                        buttonWrapper.style.justifyContent = "center";
                        buttonWrapper.style.gap = "18px";

                        const cancelBtn = document.createElement("button");
                        cancelBtn.textContent = "Cancelar";
                        cancelBtn.style.padding = "7px 18px";
                        cancelBtn.style.background = "#eee";
                        cancelBtn.style.border = "none";
                        cancelBtn.style.borderRadius = "4px";
                        cancelBtn.style.cursor = "pointer";
                        cancelBtn.addEventListener("click", () => confirmModal.remove());

                        const confirmExcludeBtn = document.createElement("button");
                        confirmExcludeBtn.textContent = "Excluir";
                        confirmExcludeBtn.style.padding = "7px 18px";
                        confirmExcludeBtn.style.background = "#d9534f";
                        confirmExcludeBtn.style.color = "#fff";
                        confirmExcludeBtn.style.border = "none";
                        confirmExcludeBtn.style.borderRadius = "4px";
                        confirmExcludeBtn.style.cursor = "pointer";
                        confirmExcludeBtn.style.fontWeight = "bold";

                        confirmExcludeBtn.addEventListener("click", () => {
                            excludedPlaces.set(id, placeName);
                            saveExcludedPlacesToLocalStorage();
                            showTemporaryMessage("Lugar excluido de futuras búsquedas.", 3000, 'success');
                            const row = excludePlaceBtn.closest('tr');
                            if (row) {
                                markRowAsProcessed(row, 'excluded');
                                if (window.checkBatchCompletion) window.checkBatchCompletion();
                            }
                            confirmModal.remove();
                        });

                        buttonWrapper.appendChild(cancelBtn);
                        buttonWrapper.appendChild(confirmExcludeBtn);
                        confirmModal.appendChild(buttonWrapper);

                        // Añadimos el modal al DOM para poder medirlo
                        document.body.appendChild(confirmModal);


                        const triggerButton = event.currentTarget;

                        // 1. Encontrar la columna de referencia ("Categoría Recomendada", la 11ª)
                        const headerCell = document.querySelector("#wme-place-inspector-panel thead th:nth-child(11)");

                        if (headerCell && triggerButton) {
                            const headerRect = headerCell.getBoundingClientRect();
                            const modalRect = confirmModal.getBoundingClientRect();
                            const triggerRect = triggerButton.getBoundingClientRect();

                            // Calcular 'left' para alinear el borde derecho del modal con el de la columna
                            let newLeft = headerRect.right - modalRect.width;
                            // Calcular 'top' para que aparezca cerca del botón
                            let newTop = triggerRect.top - modalRect.height - 10;

                            // Ajustar si se sale de la pantalla
                            if (newTop < 10) newTop = triggerRect.bottom + 10;
                            if (newLeft < 10) newLeft = 10;

                            // Aplicar los nuevos estilos
                            confirmModal.style.left = `${newLeft}px`;
                            confirmModal.style.top = `${newTop}px`;
                            confirmModal.style.transform = 'none'; // Importante: anular el centrado
                        } else {
                            // Fallback al centrado si algo falla
                            confirmModal.style.left = "50%";
                            confirmModal.style.top = "50%";
                            confirmModal.style.transform = "translate(-50%, -50%)";
                        }

                    });



                    actionCell.appendChild(buttonGroup);
                    row.appendChild(actionCell);
                    row.style.borderBottom = "1px solid #ddd";
                    row.style.backgroundColor = index % 2 === 0 ? "#f9f9f9" : "#ffffff";
                    row.querySelectorAll("td").forEach(td => {
                        td.style.verticalAlign = "top";
                    });
                    tbody.appendChild(row);
                    checkAndUpdateApplyButton();
                    setTimeout(() => {
                        const progress = Math.floor(((index + 1) / inconsistents.length) * 100);
                        const progressElem = document.getElementById("scanProgressText");
                        if (progressElem) {
                            progressElem.textContent = `Analizando lugares: ${progress}% (${index + 1}/${inconsistents.length})`;
                        }
                    }, 0);
                }; // End renderRow

                // Batching Logic
                window.plnAllInconsistents = inconsistents;
                const batchSize = 30;
                let totalProcessedPlaces = 0; // Contador de lugares ya procesados/eliminados

                function renderCurrentBatch() {
                    // Clear existing rows
                    tbody.innerHTML = "";

                    // Take top 30
                    const batch = window.plnAllInconsistents.slice(0, batchSize);

                    batch.forEach((item, i) => {
                        // Calcular el índice absoluto: total procesados + posición en batch actual
                        const absoluteIndex = totalProcessedPlaces + i;
                        renderRow(item, absoluteIndex);
                    });

                    updateBatchUI();
                    checkBatchCompletion(); // Check immediately
                }

                function loadNextBatch() {
                    // Remove hidden rows from master list
                    const rows = Array.from(tbody.querySelectorAll('tr'));
                    const hiddenIds = rows.map(r => r.dataset.placeId);

                    // Contar cuántos lugares se van a eliminar (ya procesados)
                    const placesToRemove = window.plnAllInconsistents.filter(item => hiddenIds.includes(item.id)).length;

                    // Remove items with these IDs from master list
                    window.plnAllInconsistents = window.plnAllInconsistents.filter(item => !hiddenIds.includes(item.id));

                    // Incrementar el contador de lugares procesados
                    totalProcessedPlaces += placesToRemove;

                    renderCurrentBatch();
                }

                function checkBatchCompletion() {
                    const rows = Array.from(tbody.querySelectorAll('tr'));
                    // Batch is complete when ALL rows in the current batch are normalized
                    // regardless of whether they are physically hidden or not (user might have "Show Normalized" ON)
                    const pendingInBatch = rows.filter(r => r.dataset.normalized !== 'true').length;

                    if (pendingInBatch === 0 && window.plnAllInconsistents.length > 0) {
                        // Automatically load next batch
                        setTimeout(() => {
                            loadNextBatch();
                        }, 500); // Small delay for better UX
                    }

                    // Update UI to reflect current counts
                    updateBatchUI();
                }

                function updateBatchUI() {
                    const resultsCounter = document.querySelector(".results-counter-display");
                    if (resultsCounter) {
                        // Calculate visible rows robustly
                        const rows = Array.from(tbody.querySelectorAll('tr'));
                        const isHidingNormalized = document.body.classList.contains('pln-hide-normalized-rows');

                        const visibleRows = rows.filter(row => {
                            if (row.style.display === 'none') return false;
                            if (isHidingNormalized && row.dataset.normalized === 'true') return false;
                            return true;
                        }).length;

                        const currentBatchTotal = Math.min(batchSize, window.plnAllInconsistents.length);
                        // Processed in batch = Total in batch - Visible (assuming visible means not processed yet)
                        // But wait, if "Mostrar Normalizados" is ON, normalized rows are visible.
                        // We want "Normalizados" to count items that ARE normalized, regardless of visibility.

                        const normalizedInBatch = rows.filter(r => r.dataset.normalized === 'true').length;
                        const realPending = window.plnAllInconsistents.length - normalizedInBatch;

                        // "En pantalla" should strictly mean "Not Normalized" for the purpose of the user workflow?
                        // Or literally visible? User said "en pantalla no hay 30 hay 30 menos 6".
                        // So "En pantalla" should probably mean "Pending in this batch".

                        const pendingInBatch = currentBatchTotal - normalizedInBatch;

                        // "Inconsistencias pendientes: 84. En pantalla: 29 (Normalizados: 1)"
                        resultsCounter.innerHTML = `Inconsistencias pendientes: <b style="color: #ff0000;">${realPending}</b>. En pantalla: <b style="color: #ff0000;">${pendingInBatch}</b> (Normalizados: <b style="color: green;">${normalizedInBatch}</b>).`;
                    }

                    // Remove button if it exists (cleanup)
                    const loadNextBtn = document.getElementById('pln-load-next-btn');
                    if (loadNextBtn) loadNextBtn.remove();

                    if (window.plnAllInconsistents.length === 0) {
                        // Optional: Show "All done" message
                    }
                }

                // Observer setup
                if (!window.plnPruneObserver) {
                    window.plnPruneObserver = new MutationObserver(() => {
                        checkBatchCompletion();
                    });
                }

                // Expose batch UI functions globally so markRowAsProcessed can call them
                window.checkBatchCompletion = checkBatchCompletion;
                window.updateBatchUI = updateBatchUI;

                // Initial Render
                renderCurrentBatch();

                table.appendChild(tbody);

                // Connect observer
                window.plnPruneObserver.disconnect();
                window.plnPruneObserver.observe(tbody, {
                    childList: true,
                    subtree: true,
                    attributes: true,
                    attributeFilter: ['style', 'class']
                });
                output.appendChild(table);
                const existingOverlay = document.getElementById("scanSpinnerOverlay");
                if (existingOverlay) {
                    existingOverlay.remove();
                }
                const progressBarInnerTab = document.getElementById("progressBarInnerTab");
                const progressBarTextTab = document.getElementById("progressBarTextTab");
                if (progressBarInnerTab && progressBarTextTab) {
                    progressBarInnerTab.style.width = "100%";
                    progressBarTextTab.textContent = `Progreso: 100% (${inconsistents.length}/${placesArr.length})`;
                }

                function reactivateAllActionButtons() {
                    document.querySelectorAll("#wme-place-inspector-output button")
                        .forEach(btn => {
                            btn.disabled = false;
                            btn.style.color = "";
                            btn.style.opacity = "";
                        });
                }

                W.model.actionManager.events.register("afterundoaction", null, () => {
                    if (floatingPanelElement && floatingPanelElement.style.display !== 'none') {
                        waitForWazeAPI(() => {
                            const places = getVisiblePlaces();
                            renderPlacesInFloatingPanel(places);
                            setTimeout(reactivateAllActionButtons, 250);
                        });
                    }
                    else {
                        //console.log("[WME PLN] Undo/Redo: Panel de resultados no visible, no se re-escanea.");
                    }
                });
                W.model.actionManager.events.register("afterredoaction", null, () => {
                    if (floatingPanelElement && floatingPanelElement.style.display !== 'none') {
                        waitForWazeAPI(() => {
                            const places = getVisiblePlaces();
                            renderPlacesInFloatingPanel(places);
                            setTimeout(reactivateAllActionButtons, 250);
                        });
                    }
                    else {
                        // console.log("[WME PLN] Undo/Redo: Panel de resultados no visible, no se re-escanea.");
                    }
                });
            } catch (e) {
                console.error("[WME PLN] Error in finalizeRender:", e);
            }
        }// finalizerender

    }// renderPlacesInFloatingPanel

    // Normaliza una palabra eliminando diacríticos (tildes) y caracteres especiales
    function getLevenshteinDistance(a, b) {
        const matrix = Array.from(
            { length: b.length + 1 },
            (_, i) => Array.from({ length: a.length + 1 }, (_, j) => (i === 0 ? j : (j === 0 ? i : 0))));
        for (let i = 1; i <= b.length; i++) {
            for (let j = 1; j <= a.length; j++) {
                if (b.charAt(i - 1) === a.charAt(j - 1)) {
                    matrix[i][j] = matrix[i - 1][j - 1];
                }
                else {
                    matrix[i][j] = Math.min(
                        matrix[i - 1][j] + 1,    // deletion
                        matrix[i][j - 1] + 1,    // insertion
                        matrix[i - 1][j - 1] + 1 // substitution
                    );
                }
            }
        }
        return matrix[b.length][a.length];
    }// getLevenshteinDistance

    // Normaliza una palabra eliminando caracteres especiales y convirtiéndola a minúsculas
    function calculateSimilarity(word1, word2) {
        const w1_lower = word1.toLowerCase();
        const w2_lower = word2.toLowerCase();

        // Optimización rápida para palabras idénticas
        if (w1_lower === w2_lower) return 1;

        // ✨ CACHE DE SIMILITUD: Verificar si ya calculamos esto
        // Ordenar para que calculateSimilarity(a,b) sea igual a calculateSimilarity(b,a)
        const cacheKey = w1_lower < w2_lower ? `${w1_lower}|${w2_lower}` : `${w2_lower}|${w1_lower}`;

        if (similarityCache.has(cacheKey)) {
            return similarityCache.get(cacheKey);
        }

        // Si las palabras son diferentes, pero al quitarles las tildes son idénticas,
        // dales una similitud muy alta (99%) para priorizarlas siempre.
        if (removeDiacritics(w1_lower) === removeDiacritics(w2_lower)) {
            similarityCache.set(cacheKey, 0.99);
            return 0.99; // Prioridad máxima para corrección de tildes
        }

        // Si no es un caso de tildes, procede con el cálculo normal de Levenshtein.
        const distance = getLevenshteinDistance(w1_lower, w2_lower);
        const maxLen = Math.max(w1_lower.length, w2_lower.length);

        let similarity;
        if (maxLen === 0) {
            similarity = 1;
        } else {
            similarity = 1 - distance / maxLen;
        }

        // Guardar en cache
        similarityCache.set(cacheKey, similarity);

        // Limpieza LRU si excede tamaño
        if (similarityCache.size > SIMILARITY_CACHE_MAX_SIZE) {
            const firstKey = similarityCache.keys().next().value;
            similarityCache.delete(firstKey);
        }

        return similarity;
    }// calculateSimilarity

    // Verifica si una fecha de edición está dentro del rango especificado
    function isDateWithinRange(editDate, filterRange) {
        if (!(editDate instanceof Date) || isNaN(editDate)) {
            console.warn("[WME PLN] Se proporcionó una fecha de edición inválida a isDateWithinRange.");
            return false; // No se puede comparar una fecha inválida.
        }
        const now = new Date();
        let cutoffDate = new Date();
        switch (filterRange) {
            case "all": // Si es "Elegir una opción", siempre se cumple la condición
                return true;
            case "6_months":
                cutoffDate.setMonth(now.getMonth() - 6);
                break;
            case "3_months":
                cutoffDate.setMonth(now.getMonth() - 3);
                break;
            case "1_month":
                cutoffDate.setMonth(now.getMonth() - 1);
                break;
            case "1_week":
                cutoffDate.setDate(now.getDate() - 7);
                break;
            case "1_day":
                cutoffDate.setDate(now.getDate() - 1);
                break;
            default:
                return true; // Si el filtro es desconocido, por seguridad no se filtra.
        }
        return editDate >= cutoffDate;
    }//isDateWithinRange

    // Encuentra palabras similares a una palabra dada en una lista o array indexado
    function findSimilarWords(word, indexedListOrArray, threshold) {
        const lowerWord = word.toLowerCase();
        const firstChar = lowerWord.charAt(0);
        let candidates = [];

        // Si el segundo argumento es un objeto literal (como window.dictionaryIndex)
        if (indexedListOrArray && typeof indexedListOrArray === 'object' && !Array.isArray(indexedListOrArray) && !(indexedListOrArray instanceof Map) && indexedListOrArray[firstChar]) {
            candidates = Array.from(indexedListOrArray[firstChar] || []);
        }
        // ✨ OPTIMIZACIÓN: Soporte para Map (como excludedWordsMap) (28-nov-2024)
        else if (indexedListOrArray instanceof Map) {
            if (indexedListOrArray.has(firstChar)) {
                candidates = Array.from(indexedListOrArray.get(firstChar));
            } else {
                candidates = []; // Si no hay bucket para esa letra, no hay candidatos
            }
        }
        // Si es un Set o Array (menos óptimo, pero fallback)
        else if (indexedListOrArray instanceof Set || Array.isArray(indexedListOrArray)) {
            candidates = Array.from(indexedListOrArray).filter(candidate => {
                // CORREGIDO: Extraer la palabra si es un objeto
                const candidateWord = typeof candidate === 'object' ? candidate.word : candidate;
                // CORREGIDO: Asegurar que es una string antes de llamar a charAt
                return typeof candidateWord === 'string' && candidateWord.charAt(0).toLowerCase() === firstChar;
            });
        }
        else {
            return [];
        }

        return candidates
            .map(candidate => {
                // CORREGIDO: Extraer la palabra si es un objeto
                const candidateWord = typeof candidate === 'object' ? candidate.word : candidate;
                // CORREGIDO: Asegurar que es una string antes de llamar a toLowerCase
                const candidateLower = typeof candidateWord === 'string' ? candidateWord.toLowerCase() : '';
                const similarity = calculateSimilarity(lowerWord, candidateLower);
                return { word: candidateWord, similarity };
            })
            .filter(item => item.similarity >= threshold)
            .sort((a, b) => b.similarity - a.similarity);
    }// findSimilarWords


    // Sugiere palabras excluidas basadas en el nombre actual y las palabras excluidas
    function suggestExcludedReplacements(currentName, excludedWords) {
        const words = currentName.split(/\s+/);
        const suggestions = {};
        const threshold = parseFloat(document.getElementById("similarityThreshold")?.value || "85") / 100;
        words.forEach(word => {
            // ✨ OPTIMIZACIÓN: Usar índice (Map) si está disponible (28-nov-2024)
            const sourceList = (window.excludedWordsMap && window.excludedWordsMap.size > 0)
                ? window.excludedWordsMap
                : Array.from(excludedWords);

            const similar = findSimilarWords(word, sourceList, threshold);
            if (similar.length > 0) {
                suggestions[word] = similar;
            }
        });
        return suggestions;
    }// suggestExcludedReplacements

    // Reset del inspector: progreso y texto de tab
    function resetInspectorState() {
        const inner = document.getElementById("progressBarInnerTab");
        const text = document.getElementById("progressBarTextTab");
        const outputTab = document.getElementById("wme-normalization-tab-output");
        if (inner)
            inner.style.width = "0%";
        if (text)
            text.textContent = `Progreso: 0% (0/0)`;
        if (outputTab)
            outputTab.textContent = "Presiona 'Start Scan...' para analizar los lugares visibles.";
    }// resetInspectorState

    // Función auxiliar para marcar una fila de la tabla como procesada/eliminada
    // Función auxiliar para marcar una fila de la tabla como procesada/eliminada
    function markRowAsProcessed(rowElement, actionType) {
        if (!rowElement) return;

        // CRITICAL: Set data-normalized to track processed state
        rowElement.dataset.normalized = 'true';
        rowElement.dataset.normalizedReason = actionType || 'manual-apply';

        // Estilos para atenuar y tachar la fila
        rowElement.style.opacity = '0.4';
        rowElement.style.textDecoration = 'line-through';
        rowElement.style.transition = 'opacity 0.5s ease'; // Transición suave

        // NUEVA LÓGICA: Verificar estado del botón toggle y ocultar si es necesario
        const toggleBtn = document.getElementById('pln-toggle-hidden-btn');
        if (toggleBtn && toggleBtn.dataset.state === 'hidden') {
            // Si el botón está en estado "hidden" (lo que significa que los elementos procesados deben ocultarse)
            rowElement.classList.add('pln-hidden-normalized');
        }

        // Deshabilitar todos los botones de acción en esta fila
        const buttons = rowElement.querySelectorAll('button');
        buttons.forEach(btn => {
            btn.disabled = true;
            btn.style.cursor = 'not-allowed';
            btn.style.opacity = '0.3';
        });

        // Opcional: Mostrar un pequeño icono de confirmación en la fila
        const numberCell = rowElement.querySelector('td:first-child');
        if (numberCell) {
            let icon = '';
            let tooltip = '';
            if (actionType === 'applied') {
                icon = '✓';
                tooltip = 'Cambios aplicados';
            } else if (actionType === 'deleted') {
                icon = '🗑️';
                tooltip = 'Lugar eliminado';
            } else if (actionType === 'excluded') {
                icon = '🚫';
                tooltip = 'Lugar excluido';
            }

            if (icon) {
                const iconSpan = document.createElement('span');
                iconSpan.textContent = ' ' + icon;
                iconSpan.title = tooltip;
                iconSpan.style.marginLeft = '3px';
                iconSpan.style.fontSize = '12px';
                numberCell.appendChild(iconSpan);
            }
        }
    }// markRowAsProcessed

    // Muestra un mensaje temporal en la parte superior de la pantalla


    function showTemporaryMessage(message, duration = 3000, type = 'info') {
        const CONTAINER_ID = 'pln-temp-message-container';

        // 1. Busca o crea el contenedor de notificaciones en la parte superior.
        let container = document.getElementById(CONTAINER_ID);
        if (!container) {
            container = document.createElement('div');
            container.id = CONTAINER_ID;
            container.style.cssText = `
            position: fixed;
            top: 90px;
            left: 50%;
            transform: translateX(-50%);
            z-index: 25000;
            display: flex;
            flex-direction: column;
            align-items: center;
            gap: 10px; /* Espacio entre notificaciones */
        `;
            document.body.appendChild(container);
        }

        // 2. Crea el popup individual para este mensaje.
        const popup = document.createElement('div');
        popup.textContent = message;
        // Estilos base (no cambian)
        popup.style.padding = '12px 25px';
        popup.style.borderRadius = '6px';
        popup.style.color = 'white';
        popup.style.fontWeight = 'bold';
        popup.style.boxShadow = '0 4px 10px rgba(0,0,0,0.15)';
        popup.style.opacity = '0';
        popup.style.transition = 'opacity 0.4s ease, transform 0.4s ease';
        popup.style.transform = 'translateY(-20px)'; // Animación de entrada

        // Estilos según el tipo de mensaje (no cambian)
        switch (type) {
            case 'success':
                popup.style.backgroundColor = '#28a745'; // Verde
                break;
            case 'warning':
                popup.style.backgroundColor = '#ffc107'; // Naranja
                popup.style.color = '#212529';
                break;
            case 'error':
                popup.style.backgroundColor = '#dc3545'; // Rojo
                break;
            default: // 'info'
                popup.style.backgroundColor = '#17a2b8'; // Azul
                break;
        }

        // 3. Añade el popup al contenedor y anímalo.
        container.appendChild(popup);
        setTimeout(() => {
            popup.style.opacity = '1';
            popup.style.transform = 'translateY(0)';
        }, 50);

        // 4. Configura su desaparición.
        setTimeout(() => {
            popup.style.opacity = '0';
            popup.style.transform = 'translateY(-20px)';
            popup.addEventListener('transitionend', () => popup.remove());
            // Fallback por si el evento no se dispara
            setTimeout(() => { if (popup.parentElement) popup.remove(); }, 500);
        }, duration);
    }// showTemporaryMessage

    //Permite crear un panel flotante para mostrar los resultados del escaneo
    function createFloatingPanel(status = "processing", numInconsistents = 0) {
        if (!floatingPanelElement) {
            floatingPanelElement = document.createElement("div");
            floatingPanelElement.id = "wme-place-inspector-panel";
            floatingPanelElement.style.position = "fixed";
            floatingPanelElement.style.zIndex = "10005"; // Z-INDEX DEL PANEL DE RESULTADOS
            floatingPanelElement.style.background = "#fff";
            floatingPanelElement.style.border = "1px solid #ccc";
            floatingPanelElement.style.borderRadius = "8px";
            floatingPanelElement.style.boxShadow = "0 5px 15px rgba(0,0,0,0.2)";
            floatingPanelElement.style.padding = "10px";
            floatingPanelElement.style.fontFamily = "'Helvetica Neue', Helvetica, Arial, sans-serif";
            floatingPanelElement.style.display = 'none';
            floatingPanelElement.style.transition = "width 0.25s, height 0.25s, left 0.25s, top 0.25s"; // Agregado left y top a la transición
            floatingPanelElement.style.overflow = "hidden";

            // Variables para almacenar el estado del panel
            floatingPanelElement._isMaximized = false;
            floatingPanelElement._isMinimized = false;
            floatingPanelElement._originalState = {};
            floatingPanelElement._isDragging = false;
            floatingPanelElement._currentStatus = status;

            // Crear barra de título con controles
            const titleBar = document.createElement("div");
            titleBar.style.display = "flex";
            titleBar.style.justifyContent = "space-between";
            titleBar.style.alignItems = "center";
            titleBar.style.marginBottom = "10px";
            titleBar.style.userSelect = "none";
            titleBar.style.cursor = "move";
            titleBar.style.padding = "5px 0";

            // Título del panel
            const titleElement = document.createElement("h4");
            titleElement.id = "wme-pln-panel-title";
            titleElement.style.margin = "0";
            titleElement.style.fontSize = "20px";
            titleElement.style.color = "#333";
            titleElement.style.fontWeight = "bold";
            titleElement.style.flex = "1";
            titleElement.style.textAlign = "center";

            // Contenedor de controles estilo macOS
            const controlsContainer = document.createElement("div");
            controlsContainer.style.display = "flex";
            controlsContainer.style.gap = "8px";
            controlsContainer.style.alignItems = "center";
            controlsContainer.style.position = "absolute";
            controlsContainer.style.left = "15px";
            controlsContainer.style.top = "15px";

            // Función para crear botones estilo macOS
            function createMacButton(color, action, tooltip) {
                const btn = document.createElement("div");
                btn.style.width = "12px";
                btn.style.height = "12px";
                btn.style.borderRadius = "50%";
                btn.style.backgroundColor = color;
                btn.style.cursor = "pointer";
                btn.style.border = "1px solid rgba(0,0,0,0.1)";
                btn.style.display = "flex";
                btn.style.alignItems = "center";
                btn.style.justifyContent = "center";
                btn.style.fontSize = "8px";
                btn.style.color = "rgba(0,0,0,0.6)";
                btn.style.transition = "all 0.2s";
                btn.title = tooltip;

                // Efectos hover
                btn.addEventListener("mouseenter", () => {
                    btn.style.transform = "scale(1.1)";
                    if (color === "#ff5f57") btn.textContent = "×";
                    else if (color === "#ffbd2e") btn.textContent = "−";
                    else if (color === "#28ca42") btn.textContent = action === "maximize" ? "⬜" : "🗗";
                });

                btn.addEventListener("mouseleave", () => {
                    btn.style.transform = "scale(1)";
                    btn.textContent = "";
                });

                btn.addEventListener("click", action);
                return btn;
            }

            // Botón cerrar (rojo)
            const closeBtn = createMacButton("#ff5f57", () => {
                if (floatingPanelElement._currentStatus === "processing") {
                    // Confirmar cancelación de búsqueda
                    const confirmCancel = confirm("¿Estás seguro de que quieres detener la búsqueda en progreso?");
                    if (!confirmCancel) return;

                    // Aquí puedes agregar lógica para cancelar la búsqueda actual
                    // Por ejemplo, detener cualquier proceso en curso
                    resetInspectorState();
                }

                if (floatingPanelElement) floatingPanelElement.style.display = 'none';
                resetInspectorState();
            }, "Cerrar panel");

            // Botón minimizar (amarillo)
            const minimizeBtn = createMacButton("#ffbd2e", () => {
                const outputDiv = floatingPanelElement.querySelector("#wme-place-inspector-output");

                if (!floatingPanelElement._isMinimized) {
                    // Guardar estado actual antes de minimizar
                    floatingPanelElement._originalState = {
                        width: floatingPanelElement.style.width,
                        height: floatingPanelElement.style.height,
                        top: floatingPanelElement.style.top,
                        left: floatingPanelElement.style.left,
                        transform: floatingPanelElement.style.transform,
                        outputHeight: outputDiv ? outputDiv.style.height : 'auto'
                    };

                    // Minimizar - mover a la parte superior
                    floatingPanelElement.style.top = "20px";
                    floatingPanelElement.style.left = "50%";
                    floatingPanelElement.style.transform = "translateX(-50%)";
                    floatingPanelElement.style.height = "50px";
                    floatingPanelElement.style.width = "300px";
                    if (outputDiv) outputDiv.style.display = "none";

                    floatingPanelElement._isMinimized = true;
                    updateButtonVisibility();
                } else {
                    // Restaurar desde minimizado
                    const originalState = floatingPanelElement._originalState;
                    floatingPanelElement.style.width = originalState.width;
                    floatingPanelElement.style.height = originalState.height;
                    floatingPanelElement.style.top = originalState.top;
                    floatingPanelElement.style.left = originalState.left;
                    floatingPanelElement.style.transform = originalState.transform;

                    if (outputDiv) {
                        outputDiv.style.display = "block";
                        outputDiv.style.height = originalState.outputHeight;
                    }

                    floatingPanelElement._isMinimized = false;
                    updateButtonVisibility();
                }
            }, "Minimizar panel");

            // Botón maximizar (verde)
            //  const maximizeBtn = createMacButton("#28ca42", () => {
            const outputDiv = floatingPanelElement.querySelector("#wme-place-inspector-output");

            // Función para actualizar visibilidad de botones
            // Replace the updateButtonVisibility function in createFloatingPanel
            function updateButtonVisibility() {
                const isProcessing = floatingPanelElement._currentStatus === "processing";

                // Limpiar contenedor
                controlsContainer.innerHTML = "";

                if (isProcessing) {
                    // Solo botón cerrar durante la búsqueda
                    controlsContainer.appendChild(closeBtn);
                } else if (floatingPanelElement._isMinimized) {
                    // Minimizado: cerrar y restaurar
                    controlsContainer.appendChild(closeBtn);

                    // Crear botón de restaurar si estamos minimizados
                    const restoreBtn = createMacButton("#28ca42", () => {
                        // Restaurar desde minimizado
                        const originalState = floatingPanelElement._originalState;
                        floatingPanelElement.style.width = originalState.width;
                        floatingPanelElement.style.height = originalState.height;
                        floatingPanelElement.style.top = originalState.top;
                        floatingPanelElement.style.left = originalState.left;
                        floatingPanelElement.style.transform = originalState.transform;

                        const outputDiv = floatingPanelElement.querySelector("#wme-place-inspector-output");
                        if (outputDiv) {
                            outputDiv.style.display = "block";
                            outputDiv.style.height = originalState.outputHeight;
                        }

                        floatingPanelElement._isMinimized = false;
                        updateButtonVisibility();
                    }, "Restaurar panel");

                    restoreBtn.textContent = "🗗";
                    controlsContainer.appendChild(restoreBtn);
                } else {
                    // Normal: cerrar y minimizar
                    controlsContainer.appendChild(closeBtn);
                    controlsContainer.appendChild(minimizeBtn);
                }
            }// updateButtonVisibility

            // Funcionalidad de arrastrar
            let isDragging = false;
            let dragOffset = { x: 0, y: 0 };

            titleBar.addEventListener("mousedown", (e) => {
                if (e.target === titleBar || e.target === titleElement) {
                    isDragging = true;
                    const rect = floatingPanelElement.getBoundingClientRect();
                    dragOffset.x = e.clientX - rect.left;
                    dragOffset.y = e.clientY - rect.top;
                    floatingPanelElement.style.transition = "none";
                    e.preventDefault();
                }
            });

            document.addEventListener("mousemove", (e) => {
                if (isDragging && !floatingPanelElement._isMaximized) {
                    const newLeft = e.clientX - dragOffset.x;
                    const newTop = e.clientY - dragOffset.y;

                    floatingPanelElement.style.left = `${newLeft}px`;
                    floatingPanelElement.style.top = `${newTop}px`;
                    floatingPanelElement.style.transform = "none";
                }
            });

            document.addEventListener("mouseup", () => {
                if (isDragging) {
                    isDragging = false;
                    floatingPanelElement.style.transition = "width 0.25s, height 0.25s, left 0.25s, top 0.25s";
                }
            });

            // Agregar controles y título a la barra
            titleBar.appendChild(controlsContainer);
            titleBar.appendChild(titleElement);

            // Agregar barra de título al panel
            floatingPanelElement.appendChild(titleBar);

            // Contenido del panel
            const outputDivLocal = document.createElement("div");
            outputDivLocal.id = "wme-place-inspector-output";
            outputDivLocal.style.fontSize = "18px";
            outputDivLocal.style.backgroundColor = "#fdfdfd";
            outputDivLocal.style.overflowY = "auto";
            outputDivLocal.style.flex = "1";
            floatingPanelElement.appendChild(outputDivLocal);

            // Función para actualizar botones (hacer accesible)
            floatingPanelElement._updateButtonVisibility = updateButtonVisibility;

            document.body.appendChild(floatingPanelElement);
        }

        // Actualizar estado actual
        floatingPanelElement._currentStatus = status;

        // Referencias a elementos existentes
        const titleElement = floatingPanelElement.querySelector("#wme-pln-panel-title");
        const outputDiv = floatingPanelElement.querySelector("#wme-place-inspector-output");

        // Limpiar contenido
        if (outputDiv) outputDiv.innerHTML = "";

        // Actualizar visibilidad de botones
        if (floatingPanelElement._updateButtonVisibility) {
            floatingPanelElement._updateButtonVisibility();
        }

        // Configurar según el estado
        if (status === "processing") {
            // Solo actualizar si no está maximizado o minimizado
            if (!floatingPanelElement._isMaximized && !floatingPanelElement._isMinimized) {
                floatingPanelElement.style.width = processingPanelDimensions.width;
                floatingPanelElement.style.height = processingPanelDimensions.height;
                floatingPanelElement.style.top = "50%";
                floatingPanelElement.style.left = "50%";
                floatingPanelElement.style.transform = "translate(-50%, -50%)";
            }

            if (outputDiv && !floatingPanelElement._isMinimized) {
                outputDiv.style.height = floatingPanelElement._isMaximized ? "calc(100vh - 100px)" : "150px";
                outputDiv.style.display = "block";
            }

            if (titleElement) titleElement.textContent = "Buscando...";

            // Oculta el botón si existe de una ejecución anterior.
            const toggleBtn = floatingPanelElement.querySelector('#pln-toggle-hidden-btn');
            if (toggleBtn) toggleBtn.style.display = 'none';

            if (outputDiv && !floatingPanelElement._isMinimized) {
                outputDiv.innerHTML = "<div style='display:flex; align-items:center; justify-content:center; height:100%;'><span class='loader-spinner' style='width:32px; height:32px; border:4px solid #ccc; border-top:4px solid #007bff; border-radius:50%; animation:spin 0.8s linear infinite;'></span></div>";
            }
        }
        else { // status === "results"
            // Solo actualizar si no está maximizado o minimizado
            if (!floatingPanelElement._isMaximized && !floatingPanelElement._isMinimized) {
                floatingPanelElement.style.width = resultsPanelDimensions.width;
                floatingPanelElement.style.height = resultsPanelDimensions.height;
                floatingPanelElement.style.top = "50%";
                floatingPanelElement.style.left = "60%";
                floatingPanelElement.style.transform = "translate(-50%, -50%)";
            }

            if (outputDiv && !floatingPanelElement._isMinimized) {
                outputDiv.style.height = floatingPanelElement._isMaximized ? "calc(100vh - 100px)" : "660px";
                outputDiv.style.display = "block";
            }
            if (titleElement) titleElement.textContent = "NrmliZer: Resultados";

            // --- BOTÓN MOSTRAR/OCULTAR NORMALIZADOS ---
            /* let showHidden = false;
             let toggleBtn = document.getElementById('pln-toggle-hidden-btn');
             if (!toggleBtn) {
                 toggleBtn = document.createElement('button');
                 toggleBtn.id = 'pln-toggle-hidden-btn';
                 toggleBtn.textContent = 'Mostrar normalizados';
                 toggleBtn.style.marginLeft = '12px';
                 toggleBtn.style.padding = '4px 10px';
                 toggleBtn.style.fontSize = '12px';
                 toggleBtn.style.border = '1px solid #bbb';
                 toggleBtn.style.borderRadius = '5px';
                 toggleBtn.style.background = '#f4f4f4';
                 toggleBtn.style.cursor = 'pointer';
                 toggleBtn.addEventListener('click', () => {
                     showHidden = !showHidden;
                     if (showHidden) {
                         // Mostrar lo oculto
                         const st = document.getElementById('pln-hide-style'); if (st) st.remove();
                         document.querySelectorAll('tr.pln-hidden-normalized')
                             .forEach(tr => tr.classList.remove('pln-hidden-normalized'));
                         toggleBtn.textContent = 'Ocultar normalizados';
                     } else {
                         // Volver a ocultar normalizados
                         if (!document.getElementById('pln-hide-style')) {
                             const st = document.createElement('style');
                             st.id = 'pln-hide-style';
                             st.textContent = `tr.pln-hidden-normalized{display:none !important;}`;
                             document.head.appendChild(st);
                         }
                         document.querySelectorAll('tr').forEach(tr => {
                             // Reaplicar la lógica de ocultar si corresponde
                             if (tr.dataset && tr.dataset.placeId) {
                                 // Si la fila ya estaba normalizada, volver a ocultarla
                                 // (esto depende de lógica de marcado, aquí solo se vuelve a aplicar la clase si no tiene cambios)
                                 // Si quieres forzar el ocultamiento, puedes volver a llamar a processAll() si la tienes global
                                 if (typeof window.__plnHideNormalizedRows === 'function') {
                                     window.__plnHideNormalizedRows();
                                 }
                             }
                         });
                         toggleBtn.textContent = 'Mostrar normalizados';
                     }
                 });
                 // Insertar el botón en la barra de título del panel
                 const titleBar = floatingPanelElement.querySelector('div');
                 if (titleBar) titleBar.appendChild(toggleBtn);
             }*/
        }

        floatingPanelElement.style.display = 'flex';
        floatingPanelElement.style.flexDirection = 'column';
    }
    // Escuchar el botón Guardar de WME para resetear el inspector
    const wmeSaveBtn = document.querySelector(
        "button.action.save, button[title='Guardar'], button[aria-label='Guardar']");
    if (wmeSaveBtn) {
        wmeSaveBtn.addEventListener("click", () => {
            // Ocultar el panel flotante de resultados si está visible
            if (floatingPanelElement) {
                floatingPanelElement.style.display = 'none';
            }
            // Resetear el estado del inspector en la pestaña lateral
            resetInspectorState();
        });
    }
    // Función para crear la pestaña lateral del script
    function createSidebarTab() {
        try {
            // 1. Verificar si WME y la función para registrar pestañas están listos
            if (!W || !W.userscripts || typeof W.userscripts.registerSidebarTab !== 'function') {
                console.error("[WME PLN] WME (userscripts o registerSidebarTab) no está listo para crear la pestaña lateral.");
                return;
            }
            // 2. Registrar la pestaña principal del script en WME y obtener tabPane
            let registration;
            try {
                registration = W.userscripts.registerSidebarTab("NrmliZer"); // Nombre del Tab que aparece en WME
            }
            catch (e) {
                if (e.message.includes("already been registered")) {
                    console.warn("[WME PLN] Tab 'NrmliZer' ya registrado. El script puede no funcionar como se espera si hay múltiples instancias.");
                    // Podrías intentar obtener el tabPane existente o simplemente
                    // retornar. Para evitar mayor complejidad, si ya está
                    // registrado, no continuaremos con la creación de la UI de la
                    // pestaña.
                    return;
                }

                throw e; // Relanzar otros errores para que se vean en consola
            }
            const { tabLabel, tabPane } = registration;
            if (!tabLabel || !tabPane) {

                return;
            }
            // Configurar el ícono y nombre de la pestaña principal del script
            // Corrección aquí: usar directamente MAIN_TAB_ICON_BASE64 en el src
            tabLabel.innerHTML = `
                <img src="${MAIN_TAB_ICON_BASE64}" style="height: 16px; vertical-align: middle; margin-right: 5px;">
                NrmliZer
            `;
            // 3. Inicializar las pestañas internas (General, Especiales,
            // Diccionario, Reemplazos)
            const tabsContainer = document.createElement("div");
            tabsContainer.style.display = "flex";
            tabsContainer.style.marginBottom = "8px";
            tabsContainer.style.gap = "8px";
            const tabButtons = {};
            const tabContents = {}; // Objeto para guardar los divs de contenido
            // Crear botones para cada pestaña
            tabNames.forEach(({ label, icon }) => {
                const btn = document.createElement("button");
                btn.innerHTML = icon
                    ? `<span style="display: inline-flex; align-items: center; font-size: 11px;">
                    <span style="font-size: 12px; margin-right: 4px;">${icon}</span>${label}
                </span>`
                    : `<span style="font-size: 11px;">${label}</span>`;
                btn.style.fontSize = "11px";
                btn.style.padding = "4px 8px";
                btn.style.marginRight = "4px";
                btn.style.minHeight = "28px";
                btn.style.border = "1px solid #ccc";
                btn.style.borderRadius = "4px 4px 0 0";
                btn.style.cursor = "pointer";
                btn.style.borderBottom = "none"; // Para que la pestaña activa se vea mejor integrada
                btn.className = "custom-tab-style";
                // Agrega el tooltip personalizado para cada tab
                if (label === "Gene") btn.title = "Configuración general";
                else if (label === "Espe") btn.title = "Palabras especiales (Excluidas)";
                else if (label === "Dicc") btn.title = "Diccionario de palabras válidas";
                else if (label === "Reemp") btn.title = "Gestión de reemplazos automáticos";
                // Estilo inicial: la primera pestaña es la activa
                if (label === tabNames[0].label) {
                    btn.style.backgroundColor = "#ffffff"; // Color de fondo activo (blanco)
                    btn.style.borderBottom = "2px solid #007bff"; // Borde inferior distintivo para la activa
                    btn.style.fontWeight = "bold";
                }
                else {
                    btn.style.backgroundColor = "#f0f0f0"; // Color de fondo inactivo (gris claro)
                    btn.style.fontWeight = "normal";
                }
                btn.addEventListener("click", () => {
                    tabNames.forEach(({ label: tabLabel_inner }) => {
                        const isActive = (tabLabel_inner === label);
                        const currentButton = tabButtons[tabLabel_inner];
                        if (tabContents[tabLabel_inner]) {
                            tabContents[tabLabel_inner].style.display = isActive ? "block" : "none";
                        }
                        if (currentButton) {
                            // Aplicar/Quitar estilos de pestaña activa directamente
                            if (isActive) {
                                currentButton.style.backgroundColor = "#ffffff"; // Activo
                                currentButton.style.borderBottom = "2px solid #007bff";
                                currentButton.style.fontWeight = "bold";
                            }
                            else {
                                currentButton.style.backgroundColor = "#f0f0f0"; // Inactivo
                                currentButton.style.borderBottom = "none";
                                currentButton.style.fontWeight = "normal";
                            }
                        }
                        // Llamar a la función de renderizado correspondiente
                        if (isActive) {
                            if (tabLabel_inner === "Espe") {
                                const ul = document.getElementById("excludedWordsList");
                                if (ul && typeof renderExcludedWordsList === 'function') renderExcludedWordsList(ul);
                            }
                            else if (tabLabel_inner === "Dicc") {
                                const ulDict = document.getElementById("dictionaryWordsList");
                                if (ulDict && typeof renderDictionaryList === 'function') renderDictionaryList(ulDict);
                            }
                            else if (tabLabel_inner === "Reemp") {
                                try {
                                    const ulReemplazos = document.getElementById("replacementsListElementID");
                                    if (ulReemplazos && typeof renderReplacementsList === 'function') renderReplacementsList(ulReemplazos);
                                } catch (e) {
                                    console.error("[WME PLN] Error rendering replacements list:", e);
                                }
                            }
                        }
                    });
                });
                tabButtons[label] = btn;
                tabsContainer.appendChild(btn);
            });
            tabPane.appendChild(tabsContainer);
            // Crear los divs contenedores para el contenido de cada pestaña
            tabNames.forEach(({ label }) => {
                const contentDiv = document.createElement("div");
                contentDiv.style.display = label === tabNames[0].label ? "block" : "none"; // Mostrar solo la primera
                contentDiv.style.padding = "10px";
                tabContents[label] = contentDiv; // Guardar referencia
                tabPane.appendChild(contentDiv);
            });
            // --- POBLAR EL CONTENIDO DE CADA PESTAÑA ---
            // 4. Poblar el contenido de la pestaña "General"
            const containerGeneral = tabContents["Gene"];
            if (containerGeneral) {
                // Crear el contenedor principal
                const mainTitle = document.createElement("h3");
                mainTitle.textContent = "NormliZer";
                mainTitle.style.textAlign = "center";
                mainTitle.style.fontSize = "20px";
                mainTitle.style.marginBottom = "2px";
                containerGeneral.appendChild(mainTitle);
                // Crear el subtítulo (información de la versión)
                const versionInfo = document.createElement("div");
                versionInfo.textContent = "V. " + VERSION; // VERSION global
                versionInfo.style.textAlign = "right";
                versionInfo.style.fontSize = "10px";
                versionInfo.style.color = "#777";
                versionInfo.style.marginBottom = "15px";
                containerGeneral.appendChild(versionInfo);
                //Crear un div para mostrar el ID del usuario
                const userIdInfo = document.createElement("div"); //
                userIdInfo.id = "wme-pln-user-id"; //
                userIdInfo.textContent = "Cargando usuario..."; //
                userIdInfo.style.textAlign = "right"; //
                userIdInfo.style.fontSize = "10px"; //
                userIdInfo.style.color = "#777"; //
                userIdInfo.style.marginBottom = "15px"; //
                containerGeneral.appendChild(userIdInfo); //
                // Esta función reemplaza la necesidad de las funciones getCurrentEditorViaSdk, etc.
                const pollAndDisplayUserInfo = () => {
                    let pollingAttempts = 0;
                    const maxPollingAttempts = 60;
                    const pollInterval = setInterval(async () => {
                        let currentUserInfoLocal = null; //: Usar una variable local temporal
                        // Primero intentar con wmeSDK.State.getUserInfo() ***
                        if (wmeSDK && wmeSDK.State && typeof wmeSDK.State.getUserInfo === 'function') {
                            try {
                                const sdkUserInfo = await wmeSDK.State.getUserInfo();
                                if (sdkUserInfo && sdkUserInfo.userName) {
                                    currentUserInfoLocal = {
                                        // Si sdkUserInfo.id NO existe, usar sdkUserInfo.userName DIRECTAMENTE (sin Number())
                                        id: sdkUserInfo.id !== undefined ? sdkUserInfo.id : sdkUserInfo.userName, //
                                        name: sdkUserInfo.userName,
                                        privilege: sdkUserInfo.privilege || 'N/A'
                                    };
                                    // Asegurarse de que el ID es válido para el log
                                    const displayId = typeof currentUserInfoLocal.id === 'number' ? currentUserInfoLocal.id : `"${currentUserInfoLocal.id}"`; //

                                }
                                else {

                                }
                            }
                            catch (e) {

                            }
                        }
                        else {
                            //console.warn(`[WME_PLN][DEBUG] SDK.State.getUserInfo no disponible. wmeSDK:`, wmeSDK);
                        }
                        // Fallback a W.loginManager (si SDK.State no funcionó)
                        if (!currentUserInfoLocal && typeof W !== 'undefined' && W.loginManager && W.loginManager.userName && W.loginManager.userId) { //: Usar currentUserInfoLocal
                            currentUserInfoLocal = {
                                id: Number(W.loginManager.userId), // Convertir a número
                                name: W.loginManager.userName,
                                privilege: W.loginManager.userPrivilege || 'N/A'
                            };
                            //console.log(`[WME PLN][DEBUG] W.loginManager SUCCESS: Usuario obtenido: ${currentUserInfoLocal.name} (ID: ${currentUserInfoLocal.id})`);
                        }
                        else if (!currentUserInfoLocal) { //: Solo logear si aún no se encontró en ningún método
                            // console.warn(`[WME_PLN][DEBUG] W.loginManager devolvió datos incompletos o null:`, W?.loginManager);
                        }
                        if (currentUserInfoLocal && currentUserInfoLocal.id && currentUserInfoLocal.name) {
                            clearInterval(pollInterval);
                            currentGlobalUserInfo = currentUserInfoLocal;
                            userIdInfo.textContent = `Editor Actual: ${currentGlobalUserInfo.name}`;
                            userIdInfo.title = `Privilegio: ${currentGlobalUserInfo.privilege}`;
                            updateStatsDisplay();//: Actualizar estadísticas con el nuevo usuario
                            //  console.log('[WME_PLN][DEBUG] USUARIO CARGADO EXITOSAMENTE mediante polling.');
                            const labelToUpdate = document.querySelector('label[for="chk-avoid-my-edits"]');
                            if (labelToUpdate) {
                                labelToUpdate.innerHTML = `Excluir lugares cuya última edición sea del Editor: <span style="color: #007bff; font-weight: normal;">${currentGlobalUserInfo.name}</span>`;
                            }
                            const avoidMyEditsCheckbox = document.getElementById("chk-avoid-my-edits");
                            if (avoidMyEditsCheckbox) {
                                avoidMyEditsCheckbox.disabled = false;
                                avoidMyEditsCheckbox.style.opacity = "1";
                                avoidMyEditsCheckbox.style.cursor = "pointer";
                            }
                        }
                        else if (pollingAttempts >= maxPollingAttempts - 1) {
                            clearInterval(pollInterval);
                            userIdInfo.textContent = "Usuario no detectado (agotados intentos)";
                            //console.log('[WME PLN][DEBUG]  Polling agotado. Usuario no detectado después de varios intentos.');
                            // Asignar el estado de fallo a currentGlobalUserInfo
                            currentGlobalUserInfo = { id: 0, name: 'No detectado', privilege: 'N/A' }; // Usar 0 o null como number
                            // Actualizar el texto del checkbox para evitar ediciones del usuario
                            const avoidTextSpanToUpdate = document.querySelector("#chk-avoid-my-edits + label span");
                            //: Actualizar el texto del checkbox para evitar ediciones del usuario
                            if (avoidTextSpanToUpdate) {
                                //: Usa innerHTML y estilo atenuado para el nombre "No detectado"
                                avoidTextSpanToUpdate.innerHTML = `Excluir lugares cuya última edición sea del Editor: <span style="color: #777; opacity: 0.5;">No detectado</span>`; //
                                avoidTextSpanToUpdate.style.opacity = "1"; //: Asegurar opacidad base para el span principal
                                // avoidTextSpanToUpdate.style.color = "#777"; //: Puedes quitar esta línea si el color del span es suficiente
                            }
                            const avoidMyEditsCheckbox = document.getElementById("chk-avoid-my-edits");
                            //: Deshabilitar el checkbox si no se detecta el usuario
                            if (avoidMyEditsCheckbox) {
                                avoidMyEditsCheckbox.disabled = true;
                                avoidMyEditsCheckbox.style.opacity = "0.5";
                                avoidMyEditsCheckbox.style.cursor = "not-allowed";
                            }
                        }
                        pollingAttempts++;
                    }, 200);

                };
                // Iniciar el polling para la información del usuario
                pollAndDisplayUserInfo(); //Llamada directa a la nueva función de polling
                // Título de la sección de normalización
                const normSectionTitle = document.createElement("h4");
                normSectionTitle.textContent = "Análisis de Nombres de Places";
                normSectionTitle.style.fontSize = "16px";
                normSectionTitle.style.marginTop = "10px";
                normSectionTitle.style.marginBottom = "5px";
                normSectionTitle.style.borderBottom = "1px solid #eee";
                normSectionTitle.style.paddingBottom = "3px";
                containerGeneral.appendChild(normSectionTitle);
                // Descripción de la sección
                const scanButton = document.createElement("button");
                scanButton.id = "pln-start-scan-btn";
                scanButton.textContent = "Start Scan...";
                scanButton.setAttribute("type", "button");
                scanButton.style.marginBottom = "10px";
                scanButton.style.fontSize = "14px";
                scanButton.style.width = "100%";
                scanButton.style.padding = "8px";
                scanButton.style.border = "none";
                scanButton.style.borderRadius = "4px";
                scanButton.style.backgroundColor = "#007bff";
                scanButton.style.color = "#fff";
                scanButton.style.cursor = "pointer";
                scanButton.addEventListener("click", () => {
                    disableScanControls(); // Deshabilitar controles durante el escaneo
                    scanButton.textContent = "Escaneando..."; // Cambia el texto del botón
                    const places = getVisiblePlaces();
                    const outputDiv = document.getElementById("wme-normalization-tab-output");
                    if (!outputDiv) { // Mover esta verificación antes
                        return;
                    }
                    if (places.length === 0) {
                        outputDiv.textContent = "No se encontraron lugares visibles para analizar.";
                        setTimeout(() => { renderPlacesInFloatingPanel([], { totalVisibleCount: 0, excludedCount: 0, skipExcludedFiltering: true }); }, 10);
                        return;
                    }


                    const { filtered: placesWithoutExcluded, excludedCount } = filterOutExcludedPlaces(places);
                    let workingPlaces = placesWithoutExcluded;
                    const totalVisibleCount = places.length;

                    // Aplicar filtro de update requests si está activo
                    let updateRequestsFilteredOutCount = 0;
                    if (filterUpdateRequestsOnly) {
                        const { filtered: placesWithUpdateRequests, filteredOutCount } = filterByUpdateRequests(workingPlaces);
                        workingPlaces = placesWithUpdateRequests;
                        updateRequestsFilteredOutCount = filteredOutCount;
                    }

                    const maxPlacesInput = document.getElementById("maxPlacesInput");
                    const maxPlacesToScan = parseInt(maxPlacesInput?.value || "100", 10);
                    const scannedCount = Math.min(workingPlaces.length, maxPlacesToScan);

                    if (workingPlaces.length === 0) {
                        if (filterUpdateRequestsOnly && updateRequestsFilteredOutCount > 0) {
                            outputDiv.textContent = `No se encontraron lugares con update requests pendientes.`;
                        } else if (excludedCount > 0) {
                            outputDiv.textContent = `Todos los ${totalVisibleCount} lugares visibles están en la lista de excluidos.`;
                        } else {
                            outputDiv.textContent = "No se encontraron lugares visibles para analizar.";
                        }
                    }
                    else {
                        let message = `Escaneando ${scannedCount} lugares...`;
                        if (filterUpdateRequestsOnly) {
                            message += ` (Filtrado: ${updateRequestsFilteredOutCount} sin update requests`;
                            if (excludedCount > 0) {
                                message += `, ${excludedCount} excluidos`;
                            }
                            message += `)`;
                        } else if (excludedCount > 0) {
                            message += ` (${excludedCount} excluidos que serán omitidos)`;
                        }
                        outputDiv.textContent = message;
                    }

                    setTimeout(() => {
                        renderPlacesInFloatingPanel(workingPlaces, {
                            totalVisibleCount,
                            excludedCount,
                            skipExcludedFiltering: true
                        });
                    }, 10);
                });
                containerGeneral.appendChild(scanButton);
                // Crear el contenedor para el checkbox de usuario
                const maxWrapper = document.createElement("div");
                maxWrapper.style.display = "flex";
                maxWrapper.style.alignItems = "center";
                maxWrapper.style.gap = "8px";
                maxWrapper.style.marginBottom = "8px";
                const maxLabel = document.createElement("label");
                maxLabel.textContent = "Máximo de places a revisar:";
                maxLabel.style.fontSize = "13px";
                maxWrapper.appendChild(maxLabel);
                const maxInput = document.createElement("input");
                maxInput.type = "number";
                maxInput.id = "maxPlacesInput";
                maxInput.min = "1";
                maxInput.value = "100";
                maxInput.style.width = "80px";
                maxWrapper.appendChild(maxInput);
                containerGeneral.appendChild(maxWrapper);
                const presets = [25, 50, 100, 250, 500];
                const presetContainer = document.createElement("div");
                presetContainer.style.textAlign = "center";
                presetContainer.style.marginBottom = "8px";
                presets.forEach(preset => {
                    const btn = document.createElement("button");
                    btn.className = "pln-preset-btn"; // Clase para aplicar estilos comunes
                    btn.textContent = preset.toString();
                    btn.style.margin = "2px";
                    btn.style.padding = "4px 6px";
                    btn.addEventListener("click", () => {
                        if (maxInput)
                            maxInput.value = preset.toString();
                    });
                    presetContainer.appendChild(btn);
                });
                containerGeneral.appendChild(presetContainer);

                // --- Input para Nivel Máximo de Bloqueo ---
                const lockRankWrapper = document.createElement("div");
                lockRankWrapper.style.marginBottom = "10px";
                lockRankWrapper.style.textAlign = "center";

                const lockRankLabel = document.createElement("label");
                lockRankLabel.textContent = "Nivel Máximo de Bloqueo (opcional): ";
                lockRankLabel.style.fontSize = "12px";
                lockRankWrapper.appendChild(lockRankLabel);

                const lockRankInput = document.createElement("input");
                lockRankInput.type = "number";
                lockRankInput.id = "maxLockRankInput";
                lockRankInput.min = "1";
                lockRankInput.max = "6";
                lockRankInput.placeholder = "1-6";
                lockRankInput.style.width = "50px";
                lockRankInput.style.marginLeft = "5px";

                // Cargar valor guardado
                const savedMaxLockRank = localStorage.getItem("wme_pln_max_lock_rank");
                if (savedMaxLockRank) {
                    lockRankInput.value = savedMaxLockRank;
                }

                // Guardar al cambiar
                lockRankInput.addEventListener("change", () => {
                    localStorage.setItem("wme_pln_max_lock_rank", lockRankInput.value);
                });

                lockRankWrapper.appendChild(lockRankInput);
                containerGeneral.appendChild(lockRankWrapper);
                // Checkbox para recomendar categorías
                const recommendCategoriesWrapper = document.createElement("div");
                recommendCategoriesWrapper.style.marginTop = "10px";
                recommendCategoriesWrapper.style.marginBottom = "5px";
                recommendCategoriesWrapper.style.display = "flex";
                recommendCategoriesWrapper.style.flexDirection = "column"; //Cambiar a columna para apilar checkboxes
                recommendCategoriesWrapper.style.alignItems = "flex-start"; //Alinear ítems al inicio
                recommendCategoriesWrapper.style.padding = "6px 8px"; // Añadir padding
                recommendCategoriesWrapper.style.backgroundColor = "#e0f7fa"; // Fondo claro para destacar
                recommendCategoriesWrapper.style.border = "1px solid #00bcd4"; // Borde azul
                recommendCategoriesWrapper.style.borderRadius = "4px"; // Bordes redondeados
                containerGeneral.appendChild(recommendCategoriesWrapper); //Añadir el wrapper aquí, antes de sus contenidos

                // Guardar referencia global para crear checkbox de URs dinámicamente
                recommendCategoriesWrapperGlobal = recommendCategoriesWrapper;

                // Contenedor para el checkbox "Recomendar categorías"
                const recommendCategoryCheckboxRow = document.createElement("div"); //
                recommendCategoryCheckboxRow.style.display = "flex"; //Fila para checkbox y etiqueta
                recommendCategoryCheckboxRow.style.alignItems = "center"; //
                recommendCategoryCheckboxRow.style.marginBottom = "5px"; //Margen inferior
                // Crear el checkbox y la etiqueta
                const recommendCategoriesCheckbox = document.createElement("input");
                recommendCategoriesCheckbox.type = "checkbox";
                recommendCategoriesCheckbox.id = "chk-recommend-categories";
                recommendCategoriesCheckbox.style.marginRight = "8px";
                const savedCategoryRecommendationState = localStorage.getItem("wme_pln_recommend_categories");
                recommendCategoriesCheckbox.checked = (savedCategoryRecommendationState === "true");
                const recommendCategoriesLabel = document.createElement("label");
                recommendCategoriesLabel.htmlFor = "chk-recommend-categories";
                recommendCategoriesLabel.style.fontSize = "14px";
                recommendCategoriesLabel.style.cursor = "pointer";
                recommendCategoriesLabel.style.fontWeight = "bold";
                recommendCategoriesLabel.style.color = "#00796b";
                recommendCategoriesLabel.style.display = "flex";
                recommendCategoriesLabel.style.alignItems = "center";
                const iconSpan = document.createElement("span");
                iconSpan.innerHTML = "✨ ";
                iconSpan.style.marginRight = "4px";
                iconSpan.style.fontSize = "16px";
                iconSpan.appendChild(document.createTextNode("Recomendar categorías"));
                recommendCategoriesLabel.appendChild(iconSpan);
                recommendCategoryCheckboxRow.appendChild(recommendCategoriesCheckbox); //
                recommendCategoryCheckboxRow.appendChild(recommendCategoriesLabel); //
                recommendCategoriesWrapper.appendChild(recommendCategoryCheckboxRow); //Añadir la fila al wrapper
                recommendCategoriesCheckbox.addEventListener("change", () => {
                    localStorage.setItem("wme_pln_recommend_categories", recommendCategoriesCheckbox.checked ? "true" : "false");
                });
                // --- Contenedor para AGRUPAR las opciones de exclusión ---
                const excludeContainer = document.createElement('div');
                excludeContainer.style.marginTop = '8px'; // Espacio que lo separa de la opción de arriba
                // --- Fila para el checkbox "Excluir lugares..." ---
                const avoidMyEditsCheckboxRow = document.createElement("div");
                avoidMyEditsCheckboxRow.style.display = "flex";
                avoidMyEditsCheckboxRow.style.alignItems = "center";
                //: Añadir un margen inferior para separar del checkbox de categorías
                const avoidMyEditsCheckbox = document.createElement("input");
                avoidMyEditsCheckbox.type = "checkbox";
                avoidMyEditsCheckbox.id = "chk-avoid-my-edits";
                avoidMyEditsCheckbox.style.marginRight = "8px";
                const savedAvoidMyEditsState = localStorage.getItem("wme_pln_avoid_my_edits");
                avoidMyEditsCheckbox.checked = (savedAvoidMyEditsState === "true");
                avoidMyEditsCheckboxRow.appendChild(avoidMyEditsCheckbox);
                //: Añadir un label con el texto de la opción
                const avoidMyEditsLabel = document.createElement("label");
                avoidMyEditsLabel.htmlFor = "chk-avoid-my-edits";
                avoidMyEditsLabel.style.fontSize = "16px"; // Tamaño de fuente consistente
                avoidMyEditsLabel.style.cursor = "pointer";
                avoidMyEditsLabel.style.fontWeight = "bold";
                avoidMyEditsLabel.style.color = "#00796b";
                avoidMyEditsLabel.innerHTML = `Excluir lugares cuya última edición sea del Editor: <span style="color: #007bff; font-weight: normal;">Cargando...</span>`;
                avoidMyEditsCheckboxRow.appendChild(avoidMyEditsLabel);
                // --- Fila para el dropdown de fecha (sub-menú) ---
                const dateFilterRow = document.createElement("div");
                dateFilterRow.style.display = "flex";
                dateFilterRow.style.alignItems = "center";
                dateFilterRow.style.marginTop = "8px"; // Espacio entre el checkbox y esta fila
                dateFilterRow.style.paddingLeft = "25px"; // Indentación para que parezca una sub-opción
                dateFilterRow.style.gap = "8px";
                //: Añadir un label para el dropdown
                const dateFilterLabel = document.createElement("label");
                dateFilterLabel.htmlFor = "dateFilterSelect";
                dateFilterLabel.textContent = "Excluir solo ediciones de:";
                dateFilterLabel.style.fontSize = "13px";
                dateFilterLabel.style.fontWeight = "500";
                dateFilterLabel.style.color = "#334";
                dateFilterRow.appendChild(dateFilterLabel);
                //: Crear el dropdown para seleccionar el filtro de fecha
                const dateFilterSelect = document.createElement("select");
                dateFilterSelect.id = "dateFilterSelect";
                dateFilterSelect.style.padding = "5px 8px";
                dateFilterSelect.style.border = "1px solid #b0c4de";
                dateFilterSelect.style.borderRadius = "4px";
                dateFilterSelect.style.backgroundColor = "#fff";
                dateFilterSelect.style.flexGrow = "1";
                dateFilterSelect.style.fontSize = "13px";
                dateFilterSelect.style.cursor = "pointer";
                // Añadir opciones al dropdown
                const dateOptions = {
                    "all": "Elegir una opción",
                    "6_months": "Últimos 6 meses",
                    "3_months": "Últimos 3 meses",
                    "1_month": "Último mes",
                    "1_week": "Última Semana",
                    "1_day": "Último día"
                };
                // Añadir las opciones al dropdown
                for (const [value, text] of Object.entries(dateOptions)) {
                    const option = document.createElement("option");
                    option.value = value;
                    option.textContent = text;
                    dateFilterSelect.appendChild(option);
                }
                // Cargar el valor guardado del localStorage
                const savedDateFilter = localStorage.getItem("wme_pln_date_filter");
                if (savedDateFilter) {
                    dateFilterSelect.value = savedDateFilter;
                }
                dateFilterSelect.addEventListener("change", () => {
                    localStorage.setItem("wme_pln_date_filter", dateFilterSelect.value);
                });
                dateFilterRow.appendChild(dateFilterSelect);
                // --- Añadir AMBAS filas al contenedor de exclusión ---
                excludeContainer.appendChild(avoidMyEditsCheckboxRow);
                excludeContainer.appendChild(dateFilterRow);
                // --- Añadir el contenedor AGRUPADO al wrapper principal (el cuadro azul) ---
                recommendCategoriesWrapper.appendChild(excludeContainer);
                // --- Lógica para habilitar/deshabilitar el dropdown ---
                const toggleDateFilterState = () => {
                    const isChecked = avoidMyEditsCheckbox.checked;
                    dateFilterSelect.disabled = !isChecked;
                    dateFilterRow.style.opacity = isChecked ? "1" : "0.5";
                    dateFilterRow.style.pointerEvents = isChecked ? "auto" : "none";
                };
                // --- Listener unificado para el checkbox ---
                avoidMyEditsCheckbox.addEventListener("change", () => {
                    toggleDateFilterState(); // Actualiza la UI del dropdown
                    localStorage.setItem("wme_pln_avoid_my_edits", avoidMyEditsCheckbox.checked ? "true" : "false"); // Guarda el estado
                });
                // Llamada inicial para establecer el estado correcto al cargar
                toggleDateFilterState();


                // --- Checkbox para filtrar solo lugares con Update Requests (Movido a función dinámica) ---
                // NOTA: Este checkbox ahora se crea dinámicamente después de detectar el rank del usuario
                // Ver función createUpdateRequestsCheckbox() que se llama después de getCurrentEditorViaSdk()
                /*
                const editorRank = getEditorRank();
                console.log('[WME PLN] Editor Rank detectado para update requests filter:', editorRank);

                // Acceso restringido: Solo editores nivel 4, 5 y 6
                if (editorRank === 4 || editorRank === 5 || editorRank === 6) {
                    console.log(`[WME PLN] Creando checkbox de URs pendientes (Rank ${editorRank} autorizado)`);
                    const updateRequestsContainer = document.createElement('div');
                    updateRequestsContainer.style.marginTop = '8px';

                    const updateRequestsCheckboxRow = document.createElement("div");
                    updateRequestsCheckboxRow.style.display = "flex";
                    updateRequestsCheckboxRow.style.alignItems = "center";

                    const updateRequestsCheckbox = document.createElement("input");
                    updateRequestsCheckbox.type = "checkbox";
                    updateRequestsCheckbox.id = "chk-update-requests-only";
                    updateRequestsCheckbox.style.marginRight = "8px";

                    // Restaurar estado guardado
                    const savedUpdateRequestsState = localStorage.getItem("wme_pln_update_requests_only");
                    updateRequestsCheckbox.checked = (savedUpdateRequestsState === "true");
                    filterUpdateRequestsOnly = updateRequestsCheckbox.checked; // Sincronizar con variable global

                    updateRequestsCheckboxRow.appendChild(updateRequestsCheckbox);

                    const updateRequestsLabel = document.createElement("label");
                    updateRequestsLabel.htmlFor = "chk-update-requests-only";
                    updateRequestsLabel.style.fontSize = "16px";
                    updateRequestsLabel.style.cursor = "pointer";
                    updateRequestsLabel.style.fontWeight = "bold";
                    updateRequestsLabel.style.color = "#d32f2f"; // Rojo para destacar
                    updateRequestsLabel.innerHTML = `🔔 Solo lugares con URs pendientes`;
                    updateRequestsLabel.title = `Solo disponible para editores nivel 4, 5 y 6. Filtra solo lugares con solicitudes de actualización pendientes de aprobación.`;

                    updateRequestsCheckboxRow.appendChild(updateRequestsLabel);
                    updateRequestsContainer.appendChild(updateRequestsCheckboxRow);
                    recommendCategoriesWrapper.appendChild(updateRequestsContainer);

                    // Listener para guardar estado y actualizar variable global
                    updateRequestsCheckbox.addEventListener("change", () => {
                        filterUpdateRequestsOnly = updateRequestsCheckbox.checked;
                        localStorage.setItem("wme_pln_update_requests_only", updateRequestsCheckbox.checked ? "true" : "false");
                        console.log('[WME PLN] Update requests filter:', updateRequestsCheckbox.checked);
                    });
                } else {
                    console.log(`[WME PLN] Checkbox de URs NO creado. Rank actual: ${editorRank} (Se requiere nivel 4, 5 o 6)`);
                    // Asegurar que el filtro esté desactivado si no tiene permisos
                    filterUpdateRequestsOnly = false;
                }
                */


                // --- Contenedor para el checkbox de estadísticas ---
                const statsContainer = document.createElement('div');
                statsContainer.style.marginTop = '8px';
                // Añadir un borde y fondo para destacar
                const statsCheckboxRow = document.createElement("div");
                statsCheckboxRow.style.display = "flex";
                statsCheckboxRow.style.alignItems = "center";
                // Añadir un margen inferior para separar del checkbox de exclusión
                const statsCheckbox = document.createElement("input");
                statsCheckbox.type = "checkbox";
                statsCheckbox.id = "chk-enable-stats";
                statsCheckbox.style.marginRight = "8px";
                statsCheckbox.checked = localStorage.getItem(STATS_ENABLED_KEY) === 'true';
                statsCheckboxRow.appendChild(statsCheckbox);
                // Crear la etiqueta para el checkbox de estadísticas
                const statsLabel = document.createElement("label");
                statsLabel.htmlFor = "chk-enable-stats";
                statsLabel.style.fontSize = "16px"; // Tamaño consistente
                statsLabel.style.cursor = "pointer";
                statsLabel.style.fontWeight = "bold";
                statsLabel.style.color = "#00796b";
                statsLabel.innerHTML = `📊 Habilitar panel de estadísticas`;
                statsCheckboxRow.appendChild(statsLabel);
                // Añadir un tooltip al checkbox de estadísticas
                statsContainer.appendChild(statsCheckboxRow);
                // Añadir el contenedor de estadísticas al wrapper principal (el cuadro azul)
                recommendCategoriesWrapper.appendChild(statsContainer);
                // Listener para el checkbox de estadísticas
                statsCheckbox.addEventListener("change", () => {
                    localStorage.setItem(STATS_ENABLED_KEY, statsCheckbox.checked ? "true" : "false");
                    toggleStatsPanelVisibility();
                });
                //===========================Finaliza bloque de estadísticas
                // Listener para guardar el estado del nuevo checkbox
                avoidMyEditsCheckbox.addEventListener("change", () => { //
                    localStorage.setItem("wme_pln_avoid_my_edits", avoidMyEditsCheckbox.checked ? "true" : "false"); //
                });
                // Barra de progreso y texto
                const tabProgressWrapper = document.createElement("div");
                tabProgressWrapper.style.margin = "10px 0";
                tabProgressWrapper.style.height = "18px";
                tabProgressWrapper.style.backgroundColor = "transparent";
                const tabProgressBar = document.createElement("div");
                tabProgressBar.style.height = "100%";
                tabProgressBar.style.width = "0%";
                tabProgressBar.style.backgroundColor = "#007bff";
                tabProgressBar.style.transition = "width 0.2s";
                tabProgressBar.id = "progressBarInnerTab";
                tabProgressWrapper.appendChild(tabProgressBar);
                containerGeneral.appendChild(tabProgressWrapper);
                // Texto de progreso
                const tabProgressText = document.createElement("div");
                tabProgressText.style.fontSize = "13px";
                tabProgressText.style.marginTop = "5px";
                tabProgressText.id = "progressBarTextTab";
                tabProgressText.textContent = "Progreso: 0% (0/0)";
                containerGeneral.appendChild(tabProgressText);
                // Div para mostrar el resultado del análisis
                const outputNormalizationInTab = document.createElement("div");
                outputNormalizationInTab.id = "wme-normalization-tab-output";
                outputNormalizationInTab.style.fontSize = "12px";
                outputNormalizationInTab.style.minHeight = "20px";
                outputNormalizationInTab.style.padding = "5px";
                outputNormalizationInTab.style.marginBottom = "15px";
                outputNormalizationInTab.textContent = "Presiona 'Start Scan...' para analizar los places visibles.";
                containerGeneral.appendChild(outputNormalizationInTab);
            }
            else {
                console.error("[WME PLN] No se pudo poblar la pestaña 'General' porque su contenedor no existe.");
            }
            // 5. Poblar las otras pestañas
            if (tabContents["Espe"])
                createSpecialItemsManager(tabContents["Espe"]);
            else {
                console.error("[WME PLN] No se pudo encontrar el contenedor para la pestaña 'Especiales'.");
            }
            // --- Llamada A La Función Para Poblar La Nueva Pestaña "Diccionario"
            if (tabContents["Dicc"]) {
                createDictionaryManager(tabContents["Dicc"]);
            }
            else {
                console.error("[WME PLN] No se pudo encontrar el contenedor para la pestaña 'Diccionario'.");
            }
            // --- Llamada A La Función Para Poblar La Nueva Pestaña "Reemplazos"
            if (tabContents["Reemp"]) {
                createReplacementsManager(tabContents["Reemp"]); // Esta es la llamada clave
            }
            else {
                console.error("[WME PLN] No se pudo encontrar el contenedor para la pestaña 'Reemplazos'.");
            }
        }
        catch (error) {
            console.error("[WME PLN] Error creando la pestaña lateral:", error, error.stack);
        }
    } // Fin de createSidebarTab

    // 2. Esperar a que Waze API esté disponible
    function waitForSidebarAPI() {
        // Comprobar si Waze API está disponible
        if (W && W.userscripts && W.userscripts.registerSidebarTab) {
            const savedExcluded = localStorage.getItem("excludedWordsList");
            if (savedExcluded) {
                try {
                    const parsed = JSON.parse(savedExcluded);
                    excludedWords = new Set(parsed);
                    // Reconstruir el mapa optimizado
                    excludedWordsMap.clear();
                    excludedWords.forEach(word => {
                        const firstChar = word.charAt(0).toLowerCase();
                        if (!excludedWordsMap.has(firstChar)) {
                            excludedWordsMap.set(firstChar, new Set());
                        }
                        excludedWordsMap.get(firstChar).add(word);
                    });
                }
                catch (e) {
                    console.error("[WME PLN] Error al cargar palabras excluidas:", e);
                    excludedWords = new Set();
                    excludedWordsMap.clear();
                }
            }
            else {
                excludedWords = new Set();
                excludedWordsMap.clear();
            }

            // --- Cargar Lugares Excluidos desde localStorage ---
            loadExcludedPlacesFromStorage(); // Llamar a la función directamente

            // --- Refrescar palabras especiales por defecto si está habilitado ---
            loadExcludedWordsFromSheet(FORCE_SHEET_RELOAD)
                .catch(err => console.warn('[WME PLN] No se pudo cargar palabras especiales desde Sheets:', err));



            // --- Cargar diccionario desde localStorage ---
            const savedDictionary = localStorage.getItem("dictionaryWordsList");
            if (savedDictionary) {
                try {
                    const parsed = JSON.parse(savedDictionary);
                    window.dictionaryWords = new Set(parsed);
                    // Reconstruir el índice del diccionario
                    window.dictionaryIndex = {};
                    window.dictionaryWords.forEach(word => {
                        const firstChar = word.charAt(0).toLowerCase();
                        if (!window.dictionaryIndex[firstChar]) {
                            window.dictionaryIndex[firstChar] = [];
                        }
                        window.dictionaryIndex[firstChar].push(word);
                    });
                }
                catch (e) {
                    console.error("[WME PLN] Error al cargar diccionario:", e);
                    window.dictionaryWords = new Set();
                    window.dictionaryIndex = {};
                }
            }
            else {
                window.dictionaryWords = new Set();
                window.dictionaryIndex = {};
            }

            updateDictionaryWordsCountLabel();
            // Esto añadirá nuevas palabras del Excel a window.dictionaryWords y se encarga de guardar en localStorage después.
            // Se hace de forma asíncrona pero no bloquea la UI.
            loadDictionaryWordsFromSheet(FORCE_SHEET_RELOAD).then(() => {
                console.log("[WME PLN] Diccionario por defecto cargado y combinado.");
            }).catch(err => {
                console.error("[WME PLN] No se pudo cargar el diccionario por defecto:", err);
            });
            // Cargar estadísticas del editor
            loadEditorStats();
            // --- Cargar palabras de reemplazo desde localStorage ---
            loadReplacementWordsFromStorage();
            // Cargar reemplazos por defecto y fusionar (bloqueando los de hoja)
            loadReplacementsFromSheet(FORCE_SHEET_RELOAD).then(() => {
                const ul = document.getElementById("replacementsListElementID");
                if (ul && typeof renderReplacementsList === 'function') renderReplacementsList(ul);
            }).catch(err => console.warn('[WME PLN] No se pudo cargar Replacements desde Sheets:', err));
            // Cargar Swap por defecto (bloqueado) y fusionar con los del usuario
            loadSwapWordsFromSheet(FORCE_SHEET_RELOAD).then(() => {
                try { if (typeof saveSwapWordsToStorage === 'function') saveSwapWordsToStorage(); } catch (_) { }
            }).catch(err => console.warn('[WME PLN] No se pudo cargar Swap desde Sheets:', err));

            // === [PLN] Verificación de acceso basada en nivel de editor ===
            let accessAttempts = 0;
            const maxAccessAttempts = 10;

            /**
             * Registra un intento de acceso en la whitelist del servidor
             * Se llama cuando un usuario con nivel insuficiente intenta usar el script
             * @param {string} userName - Nombre de usuario
             * @param {number} userLevel - Nivel actual del usuario
             */
            function registerAccessAttempt(userName, userLevel) {
                const accessCtrl = window.PLN_ACCESS_CONTROL || (typeof PLN_ACCESS_CONTROL !== 'undefined' ? PLN_ACCESS_CONTROL : {});
                if (!accessCtrl.enabled || !accessCtrl.endpoint) {
                    console.log('[WME PLN] Access Control no habilitado, no se registra intento.');
                    return;
                }

                const currentVersion = (typeof VERSION !== 'undefined') ? VERSION : '0.0.0';
                const userId = (currentGlobalUserInfo && currentGlobalUserInfo.id) ? currentGlobalUserInfo.id : '';
                
                const url = `${accessCtrl.endpoint}?action=registerAttempt&user=${encodeURIComponent(userName)}&userId=${encodeURIComponent(userId)}&level=${userLevel}&version=${encodeURIComponent(currentVersion)}`;

                makeRequest({
                    method: 'GET',
                    url: url,
                    timeout: accessCtrl.timeoutMs || 8000,
                    onload: function(response) {
                        try {
                            const txt = (response.responseText || '').trim();
                            if (txt.startsWith('{')) {
                                const data = JSON.parse(txt);
                                if (data.success) {
                                    if (data.action === 'registered') {
                                        console.log(`[WME PLN] 📝 Usuario ${userName} registrado en lista blanca (pendiente de activación).`);
                                        try { plnToast(`📝 Tu solicitud de acceso ha sido registrada. Contacta al administrador para activación.`, 7000); } catch (_) { }
                                    } else if (data.action === 'already_exists') {
                                        console.log(`[WME PLN] ⏳ Usuario ${userName} ya está registrado en lista blanca.`);
                                        try { plnToast(`⏳ Ya estás registrado. Si tu acceso no está activo, contacta al administrador.`, 6000); } catch (_) { }
                                    }
                                }
                            }
                        } catch (e) {
                            console.warn('[WME PLN] Error procesando respuesta de registerAttempt:', e);
                        }
                    },
                    onerror: function(err) {
                        console.warn('[WME PLN] Error registrando intento de acceso:', err);
                    }
                });
            }

            /**
             * Consulta al servidor si el usuario está en la lista blanca
             * @param {string} userName - Nombre de usuario
             * @param {string} module - 'main' o 'easyshortcuts'
             * @returns {Promise<Object>} Resultado de la verificación
             */
            async function checkWhitelistAccess(userName, module = 'main') {
                const accessCtrl = window.PLN_ACCESS_CONTROL || (typeof PLN_ACCESS_CONTROL !== 'undefined' ? PLN_ACCESS_CONTROL : {});
                if (!accessCtrl.enabled || !accessCtrl.endpoint) {
                    return { success: false, isWhitelisted: false, reason: 'Access Control no configurado' };
                }

                return new Promise((resolve) => {
                    // [PLN] Enviar versión para control de actualizaciones forzadas
                    const currentVersion = (typeof VERSION !== 'undefined') ? VERSION : '0.0.0';
                    const url = `${accessCtrl.endpoint}?action=checkAccess&user=${encodeURIComponent(userName)}&module=${module}&version=${encodeURIComponent(currentVersion)}`;

                    makeRequest({
                        method: 'GET',
                        url: url,
                        timeout: accessCtrl.timeoutMs,
                        onload: function(response) {
                            try {
                                // Verificar si parece JSON antes de parsear
                                const txt = (response.responseText || '').trim();
                                if (!txt.startsWith('{') && !txt.startsWith('[')) {
                                    // console.warn('[WME PLN] ⚠️ Access Control NO devolvió JSON. Probablemente el Apps Script no está actualizado.');
                                    // console.warn('[WME PLN] ⚠️ Respuesta (primeros 200 chars):', txt.slice(0, 200));
                                    resolve({ success: false, isWhitelisted: false, error: 'Respuesta no es JSON' });
                                    return;
                                }
                                const data = JSON.parse(txt);
                                // console.log('[WME PLN] 🔐 Respuesta de Access Control:', data);
                                resolve(data);
                            } catch (e) {
                                console.warn('[WME PLN] Error parseando respuesta de Access Control:', e);
                                console.warn('[WME PLN] Respuesta cruda:', response.responseText);
                                resolve({ success: false, isWhitelisted: false, error: e.message });
                            }
                        },
                        onerror: function(err) {
                            console.warn('[WME PLN] Error conectando a Access Control:', err);
                            resolve({ success: false, isWhitelisted: false, error: 'Error de conexión' });
                        },
                        ontimeout: function() {
                            console.warn('[WME PLN] Timeout en Access Control');
                            resolve({ success: false, isWhitelisted: false, error: 'Timeout' });
                        }
                    });
                });
            }

            /**
             * Obtiene la configuración global desde el servidor
             */
            async function fetchGlobalConfig() {
                const accessCtrl = window.PLN_ACCESS_CONTROL || (typeof PLN_ACCESS_CONTROL !== 'undefined' ? PLN_ACCESS_CONTROL : {});
                if (!accessCtrl.enabled || !accessCtrl.endpoint) return;

                return new Promise((resolve) => {
                    const url = `${accessCtrl.endpoint}?action=getConfig`;
                    makeRequest({
                        method: 'GET',
                        url: url,
                        timeout: accessCtrl.timeoutMs,
                        onload: function(response) {
                            try {
                                // Verificar si la respuesta parece JSON antes de parsear
                                const txt = (response.responseText || '').trim();
                                if (!txt.startsWith('{') && !txt.startsWith('[')) {
                                    // No es JSON, probablemente HTML de error o página de login
                                    // Silenciosamente ignorar - el servidor no tiene getConfig implementado
                                    resolve(false);
                                    return;
                                }
                                const data = JSON.parse(txt);
                                if (data.success && data.config) {
                                    // console.log('[WME PLN] ⚙️ Configuración recibida:', data.config);
                                    if (data.config.defaultMinLevel) accessCtrl.defaultMinLevel = Number(data.config.defaultMinLevel);
                                    if (data.config.easyShortcutsMinLevel) accessCtrl.easyShortcutsMinLevel = Number(data.config.easyShortcutsMinLevel);

                                    // Actualizar variables globales si existen
                                    if (typeof PLN_ACC_LVL !== 'undefined') PLN_ACC_LVL = accessCtrl.defaultMinLevel - 1; // rank is level - 1
                                    // EasyShortCuts usa accessCtrl directamente
                                }
                                resolve(true);
                            } catch (e) {
                                // Fallo silencioso - la configuración dinámica es opcional
                                // El script funcionará con los valores por defecto
                                resolve(false);
                            }
                        },
                        onerror: function() { resolve(false); },
                        ontimeout: function() { resolve(false); }
                    });
                });
            }

            async function plnValidateAccess(callback) {
                let userRank = -1;
                let userName = '';

                // Intentar obtener rank y nombre desde múltiples fuentes
                try {
                    if (typeof W !== 'undefined' && W.loginManager) {
                        // Obtener nombre de usuario desde múltiples fuentes
                        userName = W.loginManager.userName || '';
                        if (!userName && W.loginManager.user) {
                            userName = W.loginManager.user.userName || W.loginManager.user.name || '';
                        }
                        if (!userName && W.loginManager.user && W.loginManager.user.attributes) {
                            userName = W.loginManager.user.attributes.userName || '';
                        }

                        if (W.loginManager.user && W.loginManager.user.rank != null) {
                            userRank = Number(W.loginManager.user.rank);
                        } else if (W.loginManager.user && W.loginManager.user.attributes && W.loginManager.user.attributes.rank != null) {
                            userRank = Number(W.loginManager.user.attributes.rank);
                        } else if (W.loginManager.getUser) {
                            const u = W.loginManager.getUser();
                            if (u && u.rank != null) userRank = Number(u.rank);
                            else if (u && u.attributes && u.attributes.rank != null) userRank = Number(u.attributes.rank);
                        }
                    }

                    console.log(`[WME PLN] 🔍 DEBUG plnValidateAccess: userName="${userName}", userRank=${userRank}`);
                } catch (e) {
                    console.warn('[WME PLN] Error obteniendo rank del usuario:', e);
                }

                // Si aún no se puede determinar el rank, reintentar
                if (userRank < 0) {
                    accessAttempts++;
                    if (accessAttempts < maxAccessAttempts) {
                        console.log(`[WME PLN] Esperando información del usuario (intento ${accessAttempts}/${maxAccessAttempts})...`);
                        setTimeout(() => plnValidateAccess(callback), 1000);
                        return;
                    } else {
                        console.error('[WME PLN] ⛔ No se pudo determinar el nivel del usuario. Acceso denegado.');
                        try { plnToast('⛔ WME PLN: No se pudo verificar tu nivel de editor', 4000); } catch (_) { }
                        return;
                    }
                }

                const userLevel = userRank + 1; // rank 0 = nivel 1, rank 2 = nivel 3

            // 📌 Asegurar que currentGlobalUserInfo esté poblada ANTES del disclaimer
            if (!currentGlobalUserInfo || !currentGlobalUserInfo.id) {
                try {
                    let userId = null;
                    if (typeof W !== 'undefined' && W.loginManager) {
                        const u = W.loginManager.user || W.loginManager.getUser?.();
                        if (u) {
                            userId = u.id || u.attributes?.id || null;
                        }
                    }
                    currentGlobalUserInfo = {
                        id: userId,
                        name: userName || 'No detectado',
                        privilege: userRank >= 0 ? userRank : 'N/A'
                    };
                    console.log(`[WME PLN] 📌 currentGlobalUserInfo poblada:`, currentGlobalUserInfo);
                } catch (e) {
                    console.warn('[WME PLN] Error poblando currentGlobalUserInfo:', e);
                }
            }

            // 🔐 PASO 0: Obtener configuración global (si está habilitado)
            if (PLN_ACCESS_CONTROL.enabled) {
                await fetchGlobalConfig();
            }

            // 📋 PASO 0.5: Verificar disclaimer LOCAL para TODOS los usuarios (antes de cualquier otra validación)
            const DISCLAIMER_KEY = `wme_pln_disclaimer_v${VERSION}_accepted`;
            const localDisclaimerAccepted = localStorage.getItem(DISCLAIMER_KEY);
            
            // Si no ha respondido al disclaimer localmente (null), mostrar popup
            if (localDisclaimerAccepted === null) {
                console.log(`[WME PLN] 📋 Usuario ${userName} debe aceptar términos y condiciones (localStorage check).`);
                if (typeof showDisclaimerPopup === 'function') {
                    showDisclaimerPopup((accepted) => {
                        if (accepted) {
                            console.log(`[WME PLN] ✅ Términos aceptados localmente. Continuando validación...`);
                            // Reiniciar la validación ahora que aceptó
                            plnValidateAccess(callback);
                        } else {
                            console.log(`[WME PLN] ❌ Términos rechazados. Script no habilitado.`);
                            try { plnToast(`❌ Debes aceptar los términos para usar el script`, 5000); } catch (_) { }
                        }
                    });
                    return; // Esperar respuesta del popup
                }
            } else if (localDisclaimerAccepted === 'false') {
                // Si rechazó anteriormente, bloquear
                console.log(`[WME PLN] ❌ Usuario ${userName} rechazó términos anteriormente.`);
                try { plnToast(`❌ El script no se ha habilitado porque rechazaste los términos`, 5000); } catch (_) { }
                return;
            }
            // Si localDisclaimerAccepted === 'true', continuar con la validación normal

            // 🔐 PASO 1: Verificar lista blanca desde el servidor
                if (userName && PLN_ACCESS_CONTROL.enabled) {
                    // console.log(`[WME PLN] 🔐 Verificando lista blanca para usuario: ${userName}`);
                    const whitelistResult = await checkWhitelistAccess(userName, 'main');

                    // Si llegamos aquí, localStorage ya validó el disclaimer, así que solo verificamos whitelist
                    if (whitelistResult.success && whitelistResult.isWhitelisted && (whitelistResult.hasAccess || whitelistResult.disclaimerPending)) {
                        // ✅ Usuario en lista blanca - permitir uso (localStorage ya manejó disclaimer)
                        console.log(`[WME PLN] ✅ Usuario ${userName} en LISTA BLANCA. Habilitando script.`);
                        if (typeof callback === 'function') callback();
                        return;
                    } else if (whitelistResult.success && whitelistResult.isWhitelisted && !whitelistResult.hasAccess) {
                        // 🚫 Usuario en lista blanca pero sin acceso - verificar razón
                        // console.log(`[WME PLN] 🚫 Usuario ${userName} está en lista blanca pero sin acceso. Razón: ${whitelistResult.reason}`);
                        
                        // Mostrar mensaje específico según la razón
                        if (whitelistResult.disclaimerAccepted === false) {
                            try { plnToast(`❌ El script no se ha habilitado porque no aceptaste los términos y condiciones`, 8000); } catch (_) { }
                        } else {
                            try { plnToast(`🚫 Tu acceso a WME PLN ha sido desactivado por el administrador`, 8000); } catch (_) { }
                        }
                        return; // NO CONTINUAR - bloquear completamente
                    } else if (whitelistResult.success && !whitelistResult.isWhitelisted) {
                        // Usuario no está en lista blanca - usar validación de nivel normal
                        console.log(`[WME PLN] 📋 Usuario ${userName} no está en lista blanca. Verificando nivel...`);
                    } else {
                        // Error de servidor - usar validación de nivel como fallback
                        console.warn(`[WME PLN] ⚠️ No se pudo verificar lista blanca. Usando validación de nivel local.`);
                    }
                }

                // 🔐 PASO 2: Verificación de nivel local (fallback o no-whitelist)
                if (userRank >= PLN_ACC_LVL) {
                    console.log(`[WME PLN] ✅ Usuario Nivel ${userLevel} detectado. Habilitando script.`);
                    // Proceder con la inicialización
                    if (typeof callback === 'function') callback();
                } else {
                    console.log(`[WME PLN] ⛔ Usuario Nivel ${userLevel}. El script requiere Nivel ${PLN_ACC_LVL + 1}+.`);
                    try { plnToast(`⛔ WME PLN requiere nivel de editor ${PLN_ACC_LVL + 1} o superior`, 5000); } catch (_) { }
                    
                    // 🔐 PASO 2.1: Registrar intento de acceso en la whitelist para que el admin pueda activarlo
                    registerAccessAttempt(userName, userLevel);
                    return;
                }
            }

            // Validar acceso antes de inicializar
            plnValidateAccess(() => {
                // La llamada a waitForWazeAPI ya se encarga de la lógica de dynamicCategoriesLoaded.
                waitForWazeAPI(() => {
                if (!dynamicCategoriesLoaded) {
                    loadDynamicCategoriesFromSheet(FORCE_SHEET_RELOAD).then(() => {
                        dynamicCategoriesLoaded = true;
                        console.log("[WME PLN] Categorías dinámicas cargadas.");
                    }).catch(err => {
                        console.error("[WME PLN] Error al cargar categorías dinámicas:", err);
                    });
                }
                createSidebarTab();
                createStatsPanel();
                tryInitializeSDK(() => {
                    if (window.__PLN_DECISION_DEBUG_ON) console.log("[WME PLN] SDK inicializado, obteniendo usuario...");
                    getCurrentEditorViaSdk().then(userInfo => {
                        if (userInfo) {
                            currentGlobalUserInfo = userInfo;
                            console.log(`[WME PLN] Usuario SDK detectado: ${userInfo.name} (ID: ${userInfo.id})`);
                            console.log('[WME PLN] 🔍 DEBUG - userInfo completo:', userInfo);
                        } else {
                            console.warn("[WME PLN] No se pudo obtener usuario del SDK, intentando otros métodos.");
                            const wrapUser = getCurrentEditorViaWazeWrap();
                            if (wrapUser) {
                                currentGlobalUserInfo = wrapUser;
                                console.log(`[WME PLN] Usuario WazeWrap detectado: ${wrapUser.name}`);
                                console.log('[WME PLN] 🔍 DEBUG - wrapUser completo:', wrapUser);
                            } else {
                                const internalUser = getCurrentEditorViaWmeInternal();
                                if (internalUser) {
                                    currentGlobalUserInfo = internalUser;
                                    // console.log(`[WME PLN] Usuario WME interno detectado: ${internalUser.name}`);
                                    // console.log('[WME PLN] 🔍 DEBUG - internalUser completo:', internalUser);
                                } else {
                                    console.error("[WME PLN] No se pudo detectar el usuario.");
                                    currentGlobalUserInfo = { name: "No detectado", id: null, privilege: null };
                                }
                            }
                        }
                        updateStatsDisplay();

                        // 🆕 Track user activity in separate Google Sheet
                        // Asegurar que las estadísticas estén cargadas antes de enviar
                        loadEditorStats();
                        // console.log('[WME PLN] 📊 EditorStats cargado antes de tracking:', editorStats);

                        // This runs every time the script loads (unlike daily usage log)
                        plnTrackUserActivity();

                        //  Crear checkbox de URs después de detectar usuario y rank
                        createUpdateRequestsCheckbox();
                    });
                });
            });
            }); // Fin del callback de plnValidateAccess
        }
        else {
            // console.log("[WME PLN] Esperando W.userscripts API...");
            setTimeout(waitForSidebarAPI, 1000);
        }

        window.PLN_READY = { ts: Date.now(), version: VERSION };

        window.dispatchEvent(new CustomEvent('PLN_READY', { detail: { version: VERSION } }));
        console.log('[WME PLN] Señal PLN_READY emitida');
    }// Fin de waitForSidebarAPI

    // 1. normalizePlaceName
    // REEMPLAZA ESTA FUNCIÓN EN TU ARCHIVO wme_pln_8.2.0.js
    function normalizePlaceName(originalName) {
        if (isProcessingActive) {
            //plnToast('Ya hay un proceso de normalización en curso. Espera a que termine.', 3000);
            return;
        }
        isProcessingActive = true;


        // --- Helpers de capitalización (integrados para que la función sea autocontenida) ---
        function plnCapitalizeStart(str) {
            try { return String(str || '').replace(/^\s*([a-záéíóúñ])/iu, (m, c) => m.replace(c, c.toUpperCase())); } catch { return str; }
        }

        function plnCapitalizeAfterHyphen(str) {
            try {
                // Capitaliza PALABRAS COMPLETAS después de guion con espacios: " - la" → " - La"
                // Esto sobrescribe STOP words como "la", "el", "de", etc.
                let result = String(str || '').replace(/(\s-\s*)([a-záéíóúñ][\wáéíóúñ]*)/giu, (m, sep, word) => {
                    return sep + word.charAt(0).toUpperCase() + word.slice(1).toLowerCase();
                });
                // Capitaliza después de guion sin espacio antes: "word-a" → "word-A"
                result = result.replace(/([^\s])-([a-záéíóúñ][\wáéíóúñ]*)/giu, (m, before, word) => {
                    return before + '-' + word.charAt(0).toUpperCase() + word.slice(1).toLowerCase();
                });
                return result;
            } catch (_) { return String(str || ''); }
        }

        function plnTitleCaseEs(str) {
            try {
                const STOP = new Set(['de', 'la', 'las', 'el', 'los', 'y', 'e', 'o', 'u', 'un', 'una', 'unos', 'unas', 'a', 'en', 'con', 'tras', 'por', 'al', 'lo']);
                const isAllCaps = w => w.length > 1 && w === w.toUpperCase();
                const cap = w => w.charAt(0).toUpperCase() + w.slice(1).toLowerCase();
                let i = 0;
                return String(str || '').replace(/([\p{L}\p{M}][\p{L}\p{M}\.'']*)/gu, (m) => {
                    const w = m, lw = w.toLowerCase(), atStart = (i === 0); i += w.length;
                    const excl = (typeof isExcludedWord === 'function') ? isExcludedWord(w) : null;
                    if (excl) return excl;
                    if (isAllCaps(w)) return w;
                    if (STOP.has(lw) && !atStart) return lw;
                    return cap(w);
                });
            } catch { return str; }
        }

        function plnPostSwapCap(str) {
            let out = String(str || '');
            out = plnTitleCaseEs(out);
            out = plnCapitalizeStart(out);
            // Aplicar capitalización después de guion AL FINAL para sobrescribir STOP words
            out = plnCapitalizeAfterHyphen(out);
            return out.trim();
        }

        // --- Inicia el proceso de normalización ---
        const DBG = !!(window.__PLN_DECISION_DEBUG_ON || localStorage.getItem('wme_pln_debug_decision') === '1');
        if (DBG) console.group('[PLN Normalize] Proceso para:', originalName);

        // 1. Sanitiza la entrada (emoji-free) antes de aplicar reglas
        const sanitizedInput = (typeof removeEmoticons === 'function')
            ? removeEmoticons(originalName)
            : String(originalName || '');
        if (DBG && sanitizedInput !== originalName) console.log('0. Después de removeEmoticons:', sanitizedInput);

        // 2. Procesa el nombre base (capitalización inicial, reglas de palabras, etc.)
        let processedName = processPlaceName(sanitizedInput);
        if (DBG) console.log('1. Después de processPlaceName:', processedName);

        // 2. Aplica las reglas de SWAP para mover palabras
        let swappedName = applySwapRules(processedName);
        if (DBG) console.log('2. Después de applySwapRules:', swappedName);

        // 3. ✨ **CORRECCIÓN CLAVE**: Vuelve a aplicar la capitalización después del swap
        let finalName = plnPostSwapCap(swappedName);
        if (DBG) console.log('3. Después de plnPostSwapCap (capitalización final):', finalName);

        // 4. Limpieza final y restauración de palabras especiales
        finalName = plnApplyExclusions(finalName);
        if (typeof plnFixDanglingHyphen === 'function') {
            finalName = plnFixDanglingHyphen(finalName);
            if (DBG) console.log('4.1 Después de plnFixDanglingHyphen:', finalName);
        }

        // 5. PASO FINAL: Capitalización después de guiones (sobrescribe STOP words)
        // Esto se ejecuta después de todas las exclusiones y limpiezas para asegurar
        // que palabras como "la", "el", "de" después de guiones queden en mayúscula
        finalName = plnCapitalizeAfterHyphen(finalName);
        if (DBG) console.log('4.2 Después de capitalización final post-guion:', finalName);

        // 6. LIMPIEZA FINAL: Eliminar guiones colgantes que puedan quedar
        // Ejemplo: "Terpel -" (sin nada después del guion) → "Terpel"
        // Pero preserva: "Terpel - La Ceiba" (con palabras después del guion)
        if (typeof plnFixDanglingHyphen === 'function') {
            finalName = plnFixDanglingHyphen(finalName);
            if (DBG) console.log('4.3 Limpieza final de guiones colgantes:', finalName);
        }

        if (DBG) {
            console.log('4. Resultado final (tras exclusiones):', finalName);
            console.groupEnd();
        }

        return finalName.trim();
    }// Fin de normalizePlaceName

    // Función para aplicar el movimiento de palabras al inicio del nombre.
    function applyWordsToStartMovement(name, wordsArray = null) {
        let newName = name;
        const wordsToProcess = wordsArray || (window.swapWords || []);
        if (wordsToProcess.length === 0) {
            return newName;
        }
        const sortedWords = [...wordsToProcess].sort((a, b) => {
            const aWord = typeof a === 'object' ? a.word : a;
            const bWord = typeof b === 'object' ? b.word : b;
            return bWord.length - aWord.length;
        });

        for (const item of sortedWords) {
            const word = typeof item === 'object' ? item.word : item;
            const regex = new RegExp(`\\s*(${escapeRegExp(String(word))})\\s*$`, 'i');
            if (regex.test(newName)) {
                const match = newName.match(regex);
                const matchedWord = match[1];
                const remainingName = newName.replace(regex, '').trim();
                const capitalizedWord = capitalizeEachWord(matchedWord);

                // ✅ CORRECCIÓN: No se modifica la capitalización del resto del nombre.
                // Se asume que "La Calleja" ya viene correctamente capitalizado.
                newName = `${capitalizedWord} ${remainingName}`.trim();
                break;
            }
        }
        return newName;
    }//applyWordsToStartMovement


    // Esta función aplica el movimiento de palabras al final del nombre.
    function applyWordsToEndMovement(name, wordsArray = null) {
        let newName = name;
        const wordsToProcess = wordsArray || (window.swapWords || []);
        if (wordsToProcess.length === 0) {
            return newName;
        }
        const sortedWords = [...wordsToProcess].sort((a, b) => {
            const aWord = typeof a === 'object' ? a.word : a;
            const bWord = typeof b === 'object' ? b.word : b;
            return bWord.length - aWord.length;
        });

        for (const item of sortedWords) {
            const word = typeof item === 'object' ? item.word : item;
            const regex = new RegExp(`^\\s*(${escapeRegExp(String(word))})\\s+`, 'i');
            if (regex.test(newName)) {
                const match = newName.match(regex);
                const matchedWord = match[1];
                const remainingName = newName.replace(regex, '').trim();
                const capitalizedWord = capitalizeEachWord(matchedWord);

                // ✅ CORRECCIÓN: Se mantiene la capitalización del resto del nombre.
                newName = `${remainingName} ${capitalizedWord}`.trim();
                break;
            }
        }
        return newName;
    }//applyWordsToEndMovement

    // Función para aplicar el movimiento de palabras (unificada)
    function applySwapMovement(name) {
        let newName = name;

        if (!window.swapWords || window.swapWords.length === 0) {
            return newName;
        }

        // Separar palabras por dirección
        const startWords = window.swapWords
            .filter(item => item.direction === "start")
            .map(item => item.word);
        const endWords = window.swapWords
            .filter(item => item.direction === "end")
            .map(item => item.word);

        // Aplicar movimiento al inicio (mover del final al inicio)
        if (startWords.length > 0) {
            newName = applyWordsToStartMovement(newName, startWords);
        }

        // Aplicar movimiento al final (mover del inicio al final)
        if (endWords.length > 0) {
            newName = applyWordsToEndMovement(newName, endWords);
        }

        return newName;
    }//applySwapMovement

    //---------------------------------------------------------------------
    // Función para capitalizar la primera letra de cada palabra en una cadena
    function implementarCargaBajoDemanda(contenedor, datos, tamañoPagina = 50) {
        let indiceActual = 0;
        const contenedorElementos = document.createElement('div');
        contenedorElementos.className = 'diccionario-contenido';
        contenedor.appendChild(contenedorElementos);

        // Cargar batch inicial
        cargarSiguienteLote();

        // Agregar detector de scroll
        contenedor.addEventListener('scroll', manejarScroll);

        // Función para manejar el evento de scroll
        function manejarScroll() {
            if (contenedor.scrollTop + contenedor.clientHeight >= contenedorElementos.offsetHeight - 200) {
                cargarSiguienteLote();
            }
        }//manejarScroll
        // Función para cargar el siguiente lote de elementos
        function cargarSiguienteLote() {
            const fragmento = document.createDocumentFragment();
            const limite = Math.min(indiceActual + tamañoPagina, datos.length);

            for (let i = indiceActual; i < limite; i++) {
                const elemento = crearElementoDiccionario(datos[i]);
                fragmento.appendChild(elemento);
            }

            contenedorElementos.appendChild(fragmento);
            indiceActual = limite;

            // Eliminar detector de scroll si ya se cargaron todos los elementos
            if (indiceActual >= datos.length) {
                contenedor.removeEventListener('scroll', manejarScroll);
            }
        }//cargarSiguienteLote
        // Función para crear un elemento del diccionario
        function crearElementoDiccionario(datoElemento) {
            const elemento = document.createElement('div');
            elemento.className = 'diccionario-item';
            elemento.innerHTML = `
            <h3>${datoElemento.termino}</h3>
            <p>${datoElemento.definicion}</p>
        `;
            return elemento;
        }//crearElementoDiccionario
    }
    // Función para agregar un indicador de carga al contenedor
    function agregarIndicadorCarga(contenedor) {
        const indicador = document.createElement('div');
        indicador.className = 'indicador-carga';
        indicador.innerHTML = '<span>Cargando más términos...</span>';
        indicador.style.display = 'none';
        contenedor.appendChild(indicador);

        return {
            mostrar: () => { indicador.style.display = 'block'; },
            ocultar: () => { indicador.style.display = 'none'; }
        };
    }//agregarIndicadorCarga
    function agregarEstilosDiccionario() {
        // Creamos los elementos del diccionario dinámico con estilos aplicados directamente

        // Estilos para el contenedor de diccionario
        const dictionaryContainerStyle = {
            maxHeight: "250px",
            overflowY: "auto",
            border: "1px solid #ddd",
            padding: "5px",
            margin: "0",
            background: "#fff",
            borderRadius: "4px"
        };

        // Estilos para cada elemento de diccionario
        const dictionaryItemStyle = {
            padding: "8px 10px",
            borderBottom: "1px solid #eee",
            transition: "background-color 0.2s",
            cursor: "default",
            marginBottom: "4px",
            borderRadius: "3px"
        };

        // Estilos para el estado hover de los elementos
        const dictionaryItemHoverStyle = {
            backgroundColor: "#f5f5f5"
        };

        // Estilos para el título de cada elemento
        const dictionaryItemTitleStyle = {
            margin: "0 0 3px 0",
            fontSize: "14px",
            color: "#333",
            fontWeight: "500"
        };

        // Estilos para la descripción de cada elemento (si existe)
        const dictionaryItemDescriptionStyle = {
            margin: "0",
            fontSize: "12px",
            color: "#666",
            fontStyle: "italic"
        };

        // Estilos para el indicador de carga
        const indicadorCargaStyle = {
            textAlign: "center",
            padding: "10px",
            color: "#777",
            fontStyle: "italic",
            borderTop: "1px dashed #ddd",
            margin: "5px 0 0 0",
            fontSize: "12px"
        };

        // Aplicar estilos al contenedor principal cuando se cree
        const applyContainerStyles = (container) => {
            if (!container) return;

            Object.entries(dictionaryContainerStyle).forEach(([property, value]) => {
                container.style[property] = value;
            });
        };

        // Función para aplicar estilos a cada elemento del diccionario
        window.plnApplyDictionaryItemStyles = (element) => {
            if (!element) return;

            // Aplicar estilos base
            Object.entries(dictionaryItemStyle).forEach(([property, value]) => {
                element.style[property] = value;
            });

            // Añadir eventos para el hover
            element.addEventListener('mouseenter', () => {
                Object.entries(dictionaryItemHoverStyle).forEach(([property, value]) => {
                    element.style[property] = value;
                });
            });

            element.addEventListener('mouseleave', () => {
                element.style.backgroundColor = dictionaryItemStyle.backgroundColor || '';
            });

            // Aplicar estilos al título si existe
            const title = element.querySelector('h3');
            if (title) {
                Object.entries(dictionaryItemTitleStyle).forEach(([property, value]) => {
                    title.style[property] = value;
                });
            }

            // Aplicar estilos a la descripción si existe
            const description = element.querySelector('p');
            if (description) {
                Object.entries(dictionaryItemDescriptionStyle).forEach(([property, value]) => {
                    description.style[property] = value;
                });
            }
        };

        // Función para crear y aplicar estilos al indicador de carga
        window.plnCreateLoadingIndicator = (container) => {
            if (!container) return null;

            const indicador = document.createElement('div');
            indicador.className = 'indicador-carga';

            // Aplicar estilos al indicador
            Object.entries(indicadorCargaStyle).forEach(([property, value]) => {
                indicador.style[property] = value;
            });

            // Crear el contenido del indicador
            const spinner = document.createElement('div');
            spinner.style.width = "16px";
            spinner.style.height = "16px";
            spinner.style.border = "2px solid #ccc";
            spinner.style.borderTop = "2px solid #007bff";
            spinner.style.borderRadius = "50%";
            spinner.style.animation = "spin 0.8s linear infinite";
            spinner.style.display = "inline-block";
            spinner.style.marginRight = "8px";
            spinner.style.verticalAlign = "middle";

            const texto = document.createElement('span');
            texto.textContent = 'Cargando más términos...';
            texto.style.verticalAlign = "middle";

            indicador.appendChild(spinner);
            indicador.appendChild(texto);
            indicador.style.display = 'none';
            container.appendChild(indicador);

            return {
                mostrar: () => { indicador.style.display = 'block'; },
                ocultar: () => { indicador.style.display = 'none'; }
            };
        };

        // Inicializar aplicación de estilos al contenedor de diccionario
        const dictionaryContainer = document.getElementById('dictionaryContainer');
        if (dictionaryContainer) {
            applyContainerStyles(dictionaryContainer);
        } else {
            // Si el contenedor aún no existe, preparar un observador para detectarlo cuando se cree
            const observer = new MutationObserver((mutations) => {
                for (const mutation of mutations) {
                    if (mutation.type === 'childList') {
                        const dictionaryContainer = document.getElementById('dictionaryContainer');
                        if (dictionaryContainer) {
                            applyContainerStyles(dictionaryContainer);
                            observer.disconnect();
                            break;
                        }
                    }
                }
            });

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

        // Asegurarse de que existe la animación de spin en la página
        if (!document.getElementById('wme-pln-animations')) {
            const styleElement = document.createElement('style');
            styleElement.id = 'wme-pln-animations';
            styleElement.textContent = `
            @keyframes spin {
                0% { transform: rotate(0deg); }
                100% { transform: rotate(360deg); }
            }
        `;
            document.head.appendChild(styleElement);
        }
    }

    // Modificar la función inicializarDiccionarioDinamico para usar estos estilos
    function inicializarDiccionarioDinamico(contenedorId, datos) {
        const contenedor = document.getElementById(contenedorId);
        if (!contenedor) return;

        // Aplicar estilos al contenedor (por si se llamó antes de que existiera el observador)
        if (window.plnApplyDictionaryContainerStyles) {
            window.plnApplyDictionaryContainerStyles(contenedor);
        }

        // Usar el nuevo creador de indicador
        const indicadorCarga = window.plnCreateLoadingIndicator ?
            window.plnCreateLoadingIndicator(contenedor) :
            agregarIndicadorCarga(contenedor);

        let indiceActual = 0;
        const tamañoPagina = 50;
        const contenedorElementos = document.createElement('div');
        contenedorElementos.className = 'diccionario-contenido';
        contenedor.appendChild(contenedorElementos);

        // Cargar batch inicial
        cargarSiguienteLote();

        // Agregar detector de scroll
        contenedor.addEventListener('scroll', manejarScroll);

        // Función para manejar el evento de scroll
        function manejarScroll() {
            if (contenedor.scrollTop + contenedor.clientHeight >= contenedorElementos.offsetHeight - 200) {
                cargarSiguienteLote();
            }
        }

        // Función para cargar el siguiente lote de elementos
        function cargarSiguienteLote() {
            if (indiceActual >= datos.length) return;

            indicadorCarga.mostrar();

            // Simulamos un pequeño retraso para ver el indicador de carga
            setTimeout(() => {
                const fragmento = document.createDocumentFragment();
                const limite = Math.min(indiceActual + tamañoPagina, datos.length);

                for (let i = indiceActual; i < limite; i++) {
                    const elemento = crearElementoDiccionario(datos[i]);
                    fragmento.appendChild(elemento);
                }

                contenedorElementos.appendChild(fragmento);
                indiceActual = limite;

                indicadorCarga.ocultar();

                // Eliminar detector de scroll si ya se cargaron todos los elementos
                if (indiceActual >= datos.length) {
                    contenedor.removeEventListener('scroll', manejarScroll);
                }
            }, 300);
        }

        // Función para crear un elemento del diccionario
        function crearElementoDiccionario(datoElemento) {
            const elemento = document.createElement('div');
            elemento.className = 'diccionario-item';

            // Crear estructura interna
            const titulo = document.createElement('h3');
            titulo.textContent = datoElemento.termino;

            const descripcion = document.createElement('p');
            descripcion.textContent = datoElemento.definicion || '';

            elemento.appendChild(titulo);
            elemento.appendChild(descripcion);

            // Aplicar estilos si existe la función
            if (window.plnApplyDictionaryItemStyles) {
                window.plnApplyDictionaryItemStyles(elemento);
            }

            return elemento;
        }
    }

    // Llamar a la función para añadir estilos al inicio
    agregarEstilosDiccionario();

    //---------------------------------------------------------------------
    // Esta función normaliza una palabra individual, considerando palabras excluidas, tildes y capitalización
    function normalizeWordInternal(word, isFirstWordInSequence = false, isInsideQuotesOrParentheses = false) {
        //console.log(`[WME PLN - NWI] Inicia procesamiento de palabra: "${word}"`); // LOG INICIO
        if (!word || typeof word !== 'string') {
            return "";
        }
        // PRioridad 1: Palabras Especiales (Excluidas)
        if (excludedWords && excludedWordsMap) {
            //console.log(`[WME PLN - NWI] Intentando Prioridad 1 (Excluidas) para: "${word}"`); // LOG INICIO EXCLUIDAS

            // La limpieza para comparación ahora SÓLO quita tildes y convierte a minúsculas.
            // Ya no elimina símbolos como '&' o '.', haciendo la comparación más estricta.
            const cleanedInputWord = removeDiacritics(word.toLowerCase());
            const firstChar = word.charAt(0).toLowerCase();
            const excludedCandidates = excludedWordsMap.get(firstChar);
            //console.log(`[WME PLN - NWI]   cleanedInputWord: "${cleanedInputWord}", firstChar: "${firstChar}"`); // LOG CLEANED
            //console.log(`[WME PLN - NWI]   excludedCandidates para '${firstChar}':`, excludedCandidates ? Array.from(excludedCandidates) : 'Ninguno'); // LOG CANDIDATOS
            // Verifica si hay candidatos excluidos para la primera letra de la palabra.
            if (excludedCandidates) {
                for (const excludedWord of excludedCandidates) {
                    // Limpia la palabra de la lista de la misma manera estricta.
                    const cleanedExcludedWord = removeDiacritics(excludedWord.toLowerCase());
                    if (cleanedExcludedWord === cleanedInputWord) {
                        return excludedWord; // Si es una palabra excluida, devuelve su forma exacta y termina.
                    }
                }
                //console.log(`[WME PLN - NWI]   🚫 No se encontró coincidencia exacta para excluida: "${word}"`); // LOG NO COINCIDENCIA

            }
        }
        // FIN PRIORIDAD 1

        // Prioridad 2: Manejo De Guiones Dentro De Palabras (solo si no fue excluida completa)
        // La condición RE_HYPHEN_BETWEEN_LETTERS.test(word) es crucial: asegura que el guion esté entre letras
        if (word.includes('-') && RE_HYPHEN_BETWEEN_LETTERS.test(word)) {
            //console.log(`[WME PLN - NWI] Aplicando Prioridad 2: Manejo de guiones para: "${word}"`);
            const parts = word.split('-');
            const normalizedParts = parts.map((part, partIndex) => {
                let normalizedPart = part;
                const isAcronymLikePart = RE_ACRONYM_PART.test(part);
                if (isAcronymLikePart && part.length > 1) {
                    normalizedPart = part; // Mantener como acrónimo si lo es.
                } else {
                    normalizedPart = part.charAt(0).toUpperCase() + part.slice(1).toLowerCase();
                }
                return normalizedPart;
            });
            // ✨ Cambio: Unir con " - " en lugar de "-" para normalizar el formato de guiones
            return normalizedParts.join(' - ');
        }
        // Prioridad 3: Palabras Con Apóstrofe
        if (word.includes("'")) {
            //console.log(`[WME PLN - NWI] Aplicando Prioridad 3 (Apóstrofe): "${word}"`);
            return handleApostropheWord(word);
        }
        // Prioridad 4: Acrónimos Y Palabras Con Mayúsculas/Puntos/& (después de guiones y apóstrofes)
        const isAcronymLike = RE_ACRONYM_WORD.test(word);
        if (isAcronymLike && word.length > 1) {
            //console.log(`[WME PLN - NWI] Aplicando Prioridad 4 (Acrónimo): "${word}"`);
            const shouldKeepAllCaps =
                word.length <= 3 || RE_DIGIT.test(word) || RE_SYMBOL.test(word);
            if (shouldKeepAllCaps) {
                return word; // Mantener como está si es sigla corta o contiene números/símbolos
            }
            // De lo contrario, continuar con las reglas normales de capitalización
        }
        // Prioridad 5: Números Romanos
        const romanWrappedMatch = word.match(RE_ROMAN_WRAPPED);
        if (romanWrappedMatch && RE_ROMAN.test(romanWrappedMatch[2])) {
            const [, romanPrefix = '', romanCore = '', romanSuffix = ''] = romanWrappedMatch;
            //console.log(`[WME PLN - NWI] Aplicando Prioridad 5 (Números Romanos): "${word}"`);
            return `${romanPrefix}${romanCore.toUpperCase()}${romanSuffix}`;
        }
        // Prioridad 6: Palabras Comunes
        const lowerWord = word.toLowerCase().replace('.', '');
        // `commonWords` es una constante global.
        if (commonWords.has(lowerWord)) // Si es una palabra común
        {
            //console.log(`[WME PLN - NWI] Aplicando Prioridad 6 (Palabra Común): "${word}"`);
            // Artículos y preposiciones que siempre queremos capitalizar si están en commonWords
            const alwaysCapitalizeCommonWords = ["el", "la", "los", "las", "de", "y", "e", "o", "u", "al", "en", "con", "por"];

            // Solo capitalizar "y" si es la primera palabra o después de guion o punto
            if (lowerWord === "y") {
                if (isFirstWordInSequence) {
                    return "Y";
                }
                else {
                    return "y";
                }
            }
            // Solo capitalizar "e" si es la primera palabra o después de guion o punto
            if (lowerWord === "e") {
                if (isFirstWordInSequence) {
                    return "E";
                }
                else {
                    return "e";
                }
            }

            if (alwaysCapitalizeCommonWords.includes(lowerWord)) {
                // Si es un artículo/preposición de la lista, SIEMPRE capitalizar su primera letra.
                // Esto forzará "el" -> "El", "de" -> "De", incluso si no es la primera palabra.
                return lowerWord.charAt(0).toUpperCase() + lowerWord.slice(1);
            } else if (isFirstWordInSequence && !isInsideQuotesOrParentheses) {
                // Para otras palabras comunes (ej. "un", "una"), solo capitalizar si es la primera palabra
                return lowerWord.charAt(0).toUpperCase() + lowerWord.slice(1);
            } else {
                // Si es una palabra común que NO es un artículo/preposición de la lista,
                // y NO es la primera palabra, la minúsculas (comportamiento actual).
                return lowerWord;
            }
        }
        // Prioridad 7: Capitalización Estándar (Regla Por Defecto)
        let wordWithoutPunctuation = word.endsWith('.') ? word.slice(0, -1) : word;

        let result = wordWithoutPunctuation.charAt(0).toUpperCase() + wordWithoutPunctuation.slice(1).toLowerCase();

        //console.log(`[WME PLN - NWI] Aplicando Prioridad 7 (Capitalización Estándar). Resultado: "${result}"`); // LOG Capitalización estándar

        return result;
    }//normalizeWordInternal


    window.normalizeWordInternal = normalizeWordInternal;

    // Maneja la capitalización de palabras que contienen un apóstrofe.
    function handleApostropheWord(word) {
        const parts = word.split("'");
        // Solo aplica si hay exactamente un apóstrofe.
        if (parts.length === 2) {
            const before = parts[0];
            const after = parts[1];

            if (after.toLowerCase() === 's') {
                // Caso posesivo como McDonald's, la 's' va en minúscula.
                return before + "'s";
            }
            else {
                // Caso como Q'Menú, se capitaliza la parte después del apóstrofe.
                const capitalizedAfter = after.charAt(0).toUpperCase() + after.slice(1).toLowerCase();
                return before + "'" + capitalizedAfter;
            }
        }
        // Si no es un caso manejable, devuelve la palabra original para que la procesen otras reglas.
        return word;
    }// handleApostropheWord

    // Función para crear un dropdown de categorías
    function createCategoryDropdown(currentCategoryKey, rowIndex, venue) {
        const select = document.createElement("select");
        select.style.padding = "4px";
        select.style.borderRadius = "4px";
        select.style.fontSize = "12px";
        select.title = "Selecciona una categoría";
        select.id = `categoryDropdown-${rowIndex}`;

        Object.entries(categoryIcons).forEach(([key, value]) => {
            const option = document.createElement("option");
            option.value = key;
            option.textContent = `${value.icon} ${value.en}`;
            if (key === currentCategoryKey)
                option.selected = true;
            select.appendChild(option);
        });
        // Evento: al cambiar la categoría
        select.addEventListener("change", (e) => {
            const selectedCategory = e.target.value;
            if (!venue || !venue.model || !venue.model.attributes) {
                //console.error("[WME_PLN] Venue inválido al intentar actualizar la categoría");
                return;
            }
            // Actualizar la categoría en el modelo
            venue.model.attributes.categories = [selectedCategory];
            venue.model.save();
            // Mensaje opcional de confirmación
            WazeWrap.Alerts.success("Categoría actualizada", `Nueva categoría: ${categoryIcons[selectedCategory].en}`);
        });
        return select;
    }

    // 3. La función postProcessQuotesAndParentheses (CORREGIDA de la respuesta anterior)
    function postProcessQuotesAndParentheses(text) {
        if (typeof text !== 'string') return text;

        // Función auxiliar para capitalizar la primera letra de una cadena
        function capitalizeFirstLetter(string) {
            if (!string) return string;
            return string.charAt(0).toUpperCase() + string.slice(1);
        }

        // Normalizar contenido dentro de comillas dobles
        text = text.replace(/"([^"]*)"/g, (match, content) => {
            const trimmedContent = content.trim();
            if (trimmedContent === "") return '""';

            // Capitaliza la primera letra de todo el contenido interno
            const capitalizedContent = capitalizeFirstLetter(trimmedContent);

            return `"${capitalizedContent}"`; // Sin espacios extra
        });

        // Normalizar contenido dentro de paréntesis
        text = text.replace(/\(([^)]*)\)/g, (match, content) => {
            const trimmedContent = content.trim();
            if (trimmedContent === "") return '()';

            // Capitaliza la primera letra de todo el contenido interno
            const capitalizedContent = capitalizeFirstLetter(trimmedContent);

            return `(${capitalizedContent})`; // Sin espacios extra
        });

        return text.replace(/\s+/g, ' ').trim(); // Limpieza final general
    }// postProcessQuotesAndParentheses// postProcessQuotesAndParentheses

    // === Palabras especiales ===
    let excludedWords = new Set(); // Mantenemos el Set para facilitar el renderizado original
    let excludedWordsMap = new Map(); // Para la búsqueda optimizada
    let excludedPlaces = new Map(); // Nuevo Map para IDs de lugares excluidos
    let dictionaryWords = new Set(); // O window.dictionaryWords = un Set global
    let dictionaryWordsCountLabelElement = null;
    let specialWordsCountLabelElement = null;
    let excludedPlacesCountLabelElement = null; // Nuevo elemento para el conteo de lugares excluidos
    let replacementsCountLabelElement = null; // Nuevo elemento para el conteo de reemplazos
    let swapWordsCountLabelElement = null; // Nuevo elemento para el conteo de swap words
    // === Palabras especiales ===
    // --- ADICIÓN PARA DEPURACIÓN EN CONSOLA ---
    window.excludedWords = excludedWords;
    window.excludedWordsMap = excludedWordsMap;
    window.excludedPlaces = excludedPlaces;

    // Función para crear el gestor de palabras excluidas y lugares excluidos
    function createSpecialItemsManager(parentContainer) {
        const mainSection = document.createElement("div"); // <--- Nueva sección principal para la pestaña "Espe"
        mainSection.id = "specialItemsManagerSection";
        mainSection.style.marginTop = "20px";
        mainSection.style.borderTop = "1px solid #ccc";
        mainSection.style.paddingTop = "10px";

        // --- Dropdown para seleccionar el tipo de gestión ---
        const typeSelectorWrapper = document.createElement("div");
        typeSelectorWrapper.style.marginBottom = "15px";
        typeSelectorWrapper.style.textAlign = "center";

        const typeSelectorLabel = document.createElement("label");
        typeSelectorLabel.textContent = "Gestionar:";
        typeSelectorLabel.style.marginRight = "10px";
        typeSelectorLabel.style.fontWeight = "bold";
        typeSelectorWrapper.appendChild(typeSelectorLabel);

        const typeSelector = document.createElement("select");
        typeSelector.id = "specialTypeSelector";
        typeSelector.style.padding = "5px";
        typeSelector.style.borderRadius = "4px";
        typeSelector.style.fontSize = "13px";

        const optionWords = document.createElement("option");
        optionWords.value = "words";
        optionWords.textContent = "Palabras Especiales";
        typeSelector.appendChild(optionWords);

        const optionPlaces = document.createElement("option");
        optionPlaces.value = "places";
        optionPlaces.textContent = "Lugares Excluidos";
        typeSelector.appendChild(optionPlaces);

        typeSelectorWrapper.appendChild(typeSelector);
        mainSection.appendChild(typeSelectorWrapper); // Añadir a mainSection

        // --- Contenedores para las dos vistas ---
        const wordsView = document.createElement("div");
        wordsView.id = "specialWordsView";
        wordsView.style.display = "block"; // Visible por defecto

        const placesView = document.createElement("div");
        placesView.id = "excludedPlacesView";
        placesView.style.display = "none"; // Oculto por defecto

        mainSection.appendChild(wordsView); // Añadir a mainSection
        mainSection.appendChild(placesView); // Añadir a mainSection

        // ***********************************************************************************
        // INICIO DEL CONTENIDO DE LA VISTA DE PALABRAS ESPECIALES (Antigua createExcludedWordsManager)
        // ***********************************************************************************

        // Título de la sección
        const wordsTitle = document.createElement("h4");
        wordsTitle.textContent = "Gestión de Palabras Especiales";
        wordsTitle.style.fontSize = "15px";
        wordsTitle.style.marginBottom = "10px";
        wordsView.appendChild(wordsTitle); // AÑADIDO A wordsView

        // Contenedor para los controles de añadir palabra
        const addWordsControlsContainer = document.createElement("div"); // Renombrado para claridad
        addWordsControlsContainer.style.display = "flex";
        addWordsControlsContainer.style.gap = "8px";
        addWordsControlsContainer.style.marginBottom = "8px";
        addWordsControlsContainer.style.alignItems = "center"; // Alinear verticalmente
        // Input para añadir nueva palabra o frase
        const wordsInput = document.createElement("input"); // Renombrado para claridad
        wordsInput.type = "text";
        wordsInput.placeholder = "Nueva palabra o frase";
        wordsInput.style.flexGrow = "1";
        wordsInput.style.padding = "6px";
        wordsInput.style.border = "1px solid #ccc";
        wordsInput.style.borderRadius = "3px";
        addWordsControlsContainer.appendChild(wordsInput); // AÑADIDO A addWordsControlsContainer

        // Botón para añadir la palabra
        const addWordBtn = document.createElement("button"); // Renombrado para claridad
        addWordBtn.textContent = "Añadir";
        addWordBtn.style.padding = "6px 10px";
        addWordBtn.style.cursor = "pointer";
        // Añadir tooltip al botón
        addWordBtn.addEventListener("click", function () {
            const newWord = wordsInput.value.trim(); // Usa wordsInput
            const validation = isValidExcludedWord(newWord);
            if (!validation.valid) {
                alert(validation.msg);
                return;
            }
            excludedWords.add(newWord);
            const firstCharNew = newWord.charAt(0).toLowerCase();
            if (!excludedWordsMap.has(firstCharNew)) {
                excludedWordsMap.set(firstCharNew, new Set());
            }
            excludedWordsMap.get(firstCharNew).add(newWord);
            wordsInput.value = ""; // Limpia wordsInput

            // ✨ Limpiar cache porque las palabras excluidas cambiaron
            clearWordLookupCache();

            renderExcludedWordsList(document.getElementById("excludedWordsList"));
            saveExcludedWordsToLocalStorage();
            updateSpecialWordsCountLabel();
        });
        addWordsControlsContainer.appendChild(addWordBtn); // AÑADIDO A addWordsControlsContainer
        wordsView.appendChild(addWordsControlsContainer); // AÑADIDO A wordsView

        // Contenedor para los botones de acción (Exportar/Limpiar para Palabras)
        const wordsActionButtonsContainer = document.createElement("div"); // Renombrado
        wordsActionButtonsContainer.style.display = "flex";
        wordsActionButtonsContainer.style.gap = "8px";
        wordsActionButtonsContainer.style.alignItems = "center"; // Alineación vertical centrada
        wordsActionButtonsContainer.style.flexWrap = "wrap"; // Permitir que los elementos bajen si no caben
        wordsActionButtonsContainer.style.marginBottom = "10px";

        const exportWordsBtn = document.createElement("button"); // Renombrado
        exportWordsBtn.textContent = "Exportar";
        exportWordsBtn.title = "Exportar Lista a XML";
        exportWordsBtn.style.padding = "6px 10px";
        exportWordsBtn.style.cursor = "pointer";
        exportWordsBtn.addEventListener("click", () => exportSharedDataToXml("words")); // Pasa el tipo
        wordsActionButtonsContainer.appendChild(exportWordsBtn); // AÑADIDO A wordsActionButtonsContainer

        const clearWordsBtn = document.createElement("button"); // Renombrado
        clearWordsBtn.textContent = "Limpiar";
        clearWordsBtn.title = "Limpiar toda la lista";
        clearWordsBtn.style.padding = "6px 10px";
        clearWordsBtn.style.cursor = "pointer";
        clearWordsBtn.addEventListener("click", function () {
            if (confirm("¿Estás seguro de que deseas eliminar TODAS las palabras de la lista?")) {
                excludedWords.clear();
                excludedWordsMap.clear();
                renderExcludedWordsList(document.getElementById("excludedWordsList"));
                saveExcludedWordsToLocalStorage();
                updateSpecialWordsCountLabel();
            }
        });
        wordsActionButtonsContainer.appendChild(clearWordsBtn); // AÑADIDO A wordsActionButtonsContainer

        // Label para el conteo de palabras especiales
        const specialCountLabel = document.createElement("span");
        specialCountLabel.id = "specialWordsCountLabel";
        specialCountLabel.style.marginLeft = "5px"; // Margen fijo en lugar de auto para evitar empujarlo fuera
        specialCountLabel.style.fontSize = "12px";
        specialCountLabel.style.color = "#333";
        specialCountLabel.style.whiteSpace = "nowrap";
        specialWordsCountLabelElement = specialCountLabel;
        updateSpecialWordsCountLabel();
        wordsActionButtonsContainer.appendChild(specialCountLabel);

        wordsView.appendChild(wordsActionButtonsContainer); // AÑADIDO A wordsView

        // Contenedor para la lista de palabras excluidas (buscador y UL)
        const wordsSearchInput = document.createElement("input"); // Renombrado
        wordsSearchInput.type = "text";
        wordsSearchInput.placeholder = "Buscar en especiales...";
        wordsSearchInput.style.display = "block";
        wordsSearchInput.style.width = "calc(100% - 14px)";
        wordsSearchInput.style.padding = "6px";
        wordsSearchInput.style.border = "1px solid #ccc";
        wordsSearchInput.style.borderRadius = "3px";
        wordsSearchInput.style.marginBottom = "5px";
        wordsSearchInput.addEventListener("input", () => {
            renderExcludedWordsList(document.getElementById("excludedWordsList"), wordsSearchInput.value.trim()); // Usa wordsSearchInput
        });
        wordsView.appendChild(wordsSearchInput); // AÑADIDO A wordsView

        // UL para palabras excluidas
        const wordsListUL = document.createElement("ul"); // Renombrado
        wordsListUL.id = "excludedWordsList"; // Mantiene el ID original para compatibilidad con renderExcludedWordsList
        wordsListUL.style.maxHeight = "150px";
        wordsListUL.style.overflowY = "auto";
        wordsListUL.style.border = "1px solid #ddd";
        wordsListUL.style.padding = "5px";
        wordsListUL.style.margin = "0";
        wordsListUL.style.background = "#fff";
        wordsListUL.style.listStyle = "none";
        wordsView.appendChild(wordsListUL); // AÑADIDO A wordsView

        // Drop Area para XML de palabras
        const wordsDropArea = document.createElement("div"); // Renombrado
        wordsDropArea.textContent = "Arrastra aquí el archivo XML de palabras especiales";
        wordsDropArea.style.border = "2px dashed #ccc";
        wordsDropArea.style.borderRadius = "4px";
        wordsDropArea.style.padding = "15px";
        wordsDropArea.style.marginTop = "10px";
        wordsDropArea.style.textAlign = "center";
        wordsDropArea.style.background = "#f9f9f9";
        wordsDropArea.style.color = "#555";
        wordsDropArea.addEventListener("dragover", (e) => {
            e.preventDefault();
            wordsDropArea.style.background = "#e9e9e9";
            wordsDropArea.style.borderColor = "#aaa";
        });
        wordsDropArea.addEventListener("dragleave", () => {
            wordsDropArea.style.background = "#f9f9f9";
            wordsDropArea.style.borderColor = "#ccc";
        });
        wordsDropArea.addEventListener("drop", (e) => {
            e.preventDefault();
            wordsDropArea.style.background = "#f9f9f9";
            handleXmlFileDrop(e.dataTransfer.files[0], "words"); // Pasar el tipo de importación
        });
        wordsView.appendChild(wordsDropArea); // AÑADIDO A wordsView

        // ***********************************************************************************
        // FIN DEL CONTENIDO DE LA VISTA DE PALABRAS ESPECIALES
        // ***********************************************************************************

        // ***********************************************************************************
        // INICIO DEL CONTENIDO DE LA VISTA DE LUGARES EXCLUIDOS (Nueva lógica)
        // ***********************************************************************************

        // Título de la sección
        const placesTitle = document.createElement("h4");
        placesTitle.textContent = "Gestión de Lugares Excluidos";
        placesTitle.style.fontSize = "15px";
        placesTitle.style.marginBottom = "10px";
        placesView.appendChild(placesTitle);

        // Contador de lugares excluidos (debajo del título)
        const placesCountLabel = document.createElement("div");
        placesCountLabel.id = "excludedPlacesCountLabel";
        placesCountLabel.style.marginBottom = "10px";
        placesCountLabel.style.marginTop = "5px";
        placesCountLabel.style.fontSize = "13px";
        placesCountLabel.style.color = "#666";
        placesCountLabel.style.fontWeight = "500";
        excludedPlacesCountLabelElement = placesCountLabel;
        updateExcludedPlacesCountLabel();
        placesView.appendChild(placesCountLabel);

        // Controles de búsqueda y lista de lugares
        const placesSearchInput = document.createElement("input");
        placesSearchInput.type = "text";
        placesSearchInput.placeholder = "Buscar lugar excluido...";
        placesSearchInput.style.display = "block";
        placesSearchInput.style.width = "calc(100% - 14px)";
        placesSearchInput.style.padding = "6px";
        placesSearchInput.style.border = "1px solid #ccc";
        placesSearchInput.style.borderRadius = "3px";
        placesSearchInput.style.marginBottom = "5px";
        placesSearchInput.addEventListener("input", () => {
            renderExcludedPlacesList(document.getElementById("excludedPlacesListUL"), placesSearchInput.value.trim());
        });
        placesView.appendChild(placesSearchInput);

        const placesListUL = document.createElement("ul");
        placesListUL.id = "excludedPlacesListUL"; // Nuevo ID para la lista de Places
        placesListUL.style.maxHeight = "200px"; // Un poco más grande
        placesListUL.style.overflowY = "auto";
        placesListUL.style.border = "1px solid #ddd";
        placesListUL.style.padding = "5px";
        placesListUL.style.margin = "0";
        placesListUL.style.background = "#fff";
        placesListUL.style.listStyle = "none";
        placesView.appendChild(placesListUL);

        // Botones de acción para Lugares Excluidos
        const placesActionButtonsContainer = document.createElement("div");
        placesActionButtonsContainer.style.display = "flex";
        placesActionButtonsContainer.style.gap = "8px";
        placesActionButtonsContainer.style.alignItems = "center"; // Alineación vertical
        placesActionButtonsContainer.style.flexWrap = "wrap"; // Permitir wrap
        placesActionButtonsContainer.style.marginTop = "10px";

        const exportPlacesBtn = document.createElement("button");
        exportPlacesBtn.textContent = "Exportar";
        exportPlacesBtn.title = "Exportar Lugares Excluidos a XML";
        exportPlacesBtn.style.padding = "6px 10px";
        exportPlacesBtn.style.cursor = "pointer";
        exportPlacesBtn.addEventListener("click", () => exportSharedDataToXml("places")); // Pasa el tipo
        placesActionButtonsContainer.appendChild(exportPlacesBtn);

        const clearPlacesBtn = document.createElement("button");
        clearPlacesBtn.textContent = "Limpiar";
        clearPlacesBtn.title = "Limpiar lista de lugares excluidos";
        clearPlacesBtn.style.padding = "6px 10px";
        clearPlacesBtn.style.cursor = "pointer";
        clearPlacesBtn.addEventListener("click", () => {
            if (confirm("¿Estás seguro de que deseas eliminar TODOS los lugares de la lista?")) {
                excludedPlaces.clear();
                renderExcludedPlacesList(document.getElementById("excludedPlacesListUL"));
                saveExcludedPlacesToLocalStorage();
                updateExcludedPlacesCountLabel();
            }
        });
        placesActionButtonsContainer.appendChild(clearPlacesBtn);

        placesView.appendChild(placesActionButtonsContainer);

        // Drop Area para XML de Lugares Excluidos
        const placesDropArea = document.createElement("div");
        placesDropArea.textContent = "Arrastra aquí el archivo XML de lugares excluidos";
        placesDropArea.style.border = "2px dashed #ccc";
        placesDropArea.style.borderRadius = "4px";
        placesDropArea.style.padding = "15px";
        placesDropArea.style.marginTop = "10px";
        placesDropArea.style.textAlign = "center";
        placesDropArea.style.background = "#f9f9f9";
        placesDropArea.style.color = "#555";
        placesDropArea.addEventListener("dragover", (e) => {
            e.preventDefault();
            placesDropArea.style.background = "#e9e9e9";
            placesDropArea.style.borderColor = "#aaa";
        });
        placesDropArea.addEventListener("dragleave", () => {
            placesDropArea.style.background = "#f9f9f9";
            placesDropArea.style.borderColor = "#ccc";
        });
        placesDropArea.addEventListener("drop", (e) => {
            e.preventDefault();
            placesDropArea.style.background = "#f9f9f9";
            handleXmlFileDrop(e.dataTransfer.files[0], "places"); // Pasa el tipo de importación
        });
        placesView.appendChild(placesDropArea);

        // ***********************************************************************************
        // FIN DEL CONTENIDO DE LA VISTA DE LUGARES EXCLUIDOS
        // ***********************************************************************************

        // --- Lógica de alternancia del selector ---
        typeSelector.addEventListener("change", () => {
            if (typeSelector.value === "words") {
                wordsView.style.display = "block";
                placesView.style.display = "none";
                renderExcludedWordsList(document.getElementById("excludedWordsList"), wordsSearchInput.value.trim()); // Renderiza lista de palabras
            } else {
                wordsView.style.display = "none";
                placesView.style.display = "block";
                renderExcludedPlacesList(document.getElementById("excludedPlacesListUL"), placesSearchInput.value.trim()); // Renderiza lista de lugares
            }
        });

        // --- Renderizado inicial de las listas al cargar ---
        renderExcludedWordsList(wordsListUL, ""); // Usa la referencia directa a wordsListUL
        renderExcludedPlacesList(placesListUL, ""); // Usa la referencia directa a placesListUL

        parentContainer.appendChild(mainSection); // <--- AÑADE SOLO ESTA SECCIÓN PRINCIPAL AL PARENT CONTAINER
    }

    // Actualiza la etiqueta que muestra el conteo de palabras especiales
    function updateSpecialWordsCountLabel() {
        if (!specialWordsCountLabelElement) {
            specialWordsCountLabelElement = document.getElementById("specialWordsCountLabel");
        }
        if (!specialWordsCountLabelElement) {
            return;
        }

        const count = window.excludedWords ? window.excludedWords.size : 0;
        specialWordsCountLabelElement.innerHTML = `<b style="color: #333;">Palabras Cargadas:</b> <b style="color: #0066cc;">${count}</b>`;
    }

    // Actualiza la etiqueta que muestra el conteo de reemplazos
    function updateReplacementsCountLabel() {
        if (!replacementsCountLabelElement) {
            replacementsCountLabelElement = document.getElementById("replacementsCountLabel");
        }

        if (!replacementsCountLabelElement) {
            return;
        }

        const count = window.replacementWords ? Object.keys(window.replacementWords).length : 0;
        replacementsCountLabelElement.innerHTML = `<b style="color: #333;">Reemplazos Cargados:</b> <b style="color: #0066cc;">${count}</b>`;
    }
    window.updateReplacementsCountLabel = updateReplacementsCountLabel;

    // Actualiza la etiqueta que muestra el conteo de lugares excluidos
    function updateExcludedPlacesCountLabel() {
        if (!excludedPlacesCountLabelElement) {
            excludedPlacesCountLabelElement = document.getElementById("excludedPlacesCountLabel");
        }

        if (!excludedPlacesCountLabelElement) {
            return;
        }

        const count = window.excludedPlaces ? window.excludedPlaces.size : 0;
        excludedPlacesCountLabelElement.innerHTML = `<b style="color: #333;">Lugares Cargados:</b> <b style="color: #0066cc;">${count}</b>`;
    }
    window.updateExcludedPlacesCountLabel = updateExcludedPlacesCountLabel;

    // Actualiza la etiqueta que muestra el conteo de swap words
    function updateSwapWordsCountLabel() {
        if (!swapWordsCountLabelElement) {
            swapWordsCountLabelElement = document.getElementById("swapWordsCountLabel");
        }

        if (!swapWordsCountLabelElement) {
            return;
        }

        const count = window.swapWords ? window.swapWords.length : 0;
        swapWordsCountLabelElement.innerHTML = `<b style="color: #333;">Palabras de Intercambio (Swap):</b> <b style="color: #0066cc;">${count}</b>`;
    }
    window.updateSwapWordsCountLabel = updateSwapWordsCountLabel;


    // Función para validar una palabra o frase antes de añadirla a las palabras excluidas
    function prepararDatosDiccionario() {
        // Convertir el Set de palabras a un array de objetos con el formato requerido
        const datos = Array.from(window.dictionaryWords || []).map(palabra => {
            return {
                termino: palabra,
                definicion: "" // Si no tiene definición, dejar vacío
            };
        });
        // Ordenar alfabéticamente para mejor navegación
        return datos.sort((a, b) => a.termino.localeCompare(b.termino));
    }//prepararDatosDiccionario
    // Actualiza la etiqueta que muestra el conteo de palabras en el diccionario
    function updateDictionaryWordsCountLabel() {
        if (!dictionaryWordsCountLabelElement) {
            dictionaryWordsCountLabelElement = document.getElementById("dictionaryWordsCountLabel");
        }

        if (!dictionaryWordsCountLabelElement) {
            return;
        }

        const count = window.dictionaryWords ? window.dictionaryWords.size : 0;
        dictionaryWordsCountLabelElement.innerHTML = `<b style="color: #333;">Palabras Cargadas:</b> <b style="color: #0066cc;">${count}</b>`;
    }//updateDictionaryWordsCountLabel

    // Exponer globalmente para addWordToDictionary
    window.updateDictionaryWordsCountLabel = updateDictionaryWordsCountLabel;

    // === Diccionario ===
    function createDictionaryManager(parentContainer) {
        const section = document.createElement("div");
        section.id = "dictionaryManagerSection";
        section.style.marginTop = "20px";
        section.style.borderTop = "1px solid #ccc";
        section.style.paddingTop = "10px";

        // Título de la sección
        const title = document.createElement("h4");
        title.textContent = "Gestión del Diccionario";
        title.style.fontSize = "15px";
        title.style.marginBottom = "10px";
        section.appendChild(title);

        // Contenedor para los controles de añadir palabra
        const addControlsContainer = document.createElement("div");
        addControlsContainer.style.display = "flex";
        addControlsContainer.style.gap = "8px";
        addControlsContainer.style.marginBottom = "8px";
        addControlsContainer.style.alignItems = "center";

        // Input para añadir nueva palabra
        const input = document.createElement("input");
        input.type = "text";
        input.placeholder = "Nueva palabra";
        input.style.flexGrow = "1";
        input.style.padding = "6px";
        input.style.border = "1px solid #ccc";
        input.style.borderRadius = "3px";
        addControlsContainer.appendChild(input);

        // Botón para añadir la palabra
        const addBtn = document.createElement("button");
        addBtn.textContent = "Añadir";
        addBtn.style.padding = "6px 10px";
        addBtn.style.cursor = "pointer";
        addBtn.addEventListener("click", function () {
            // Usar la función centralizada que maneja correctamente userAddedDictionaryWords y persistencia
            addWordToDictionary(input);
        });
        addControlsContainer.appendChild(addBtn);
        section.appendChild(addControlsContainer);

        // Contenedor para los botones de acción
        const actionButtonsContainer = document.createElement("div");
        actionButtonsContainer.style.display = "flex";
        actionButtonsContainer.style.gap = "8px";
        actionButtonsContainer.style.alignItems = "center";
        actionButtonsContainer.style.marginBottom = "10px";

        /*
                // Botón para limpiar
                const clearBtn = document.createElement("button");
                clearBtn.textContent = "Limpiar";
                clearBtn.title = "Limpiar toda la lista";
                clearBtn.style.padding = "6px 10px";
                clearBtn.style.cursor = "pointer";
                clearBtn.addEventListener("click", function() {
                    if (confirm("¿Estás seguro de que deseas eliminar TODAS las palabras del diccionario?")) {
                        window.dictionaryWords.clear();

                        // Reinicializar el diccionario dinámico con los datos actualizados
                        const listContainer = document.getElementById("dictionaryContainer");
                        if (listContainer) {
                            listContainer.innerHTML = "";
                            inicializarDiccionarioDinamico("dictionaryContainer", prepararDatosDiccionario());
                        }

                        updateDictionaryWordsCountLabel();
                    }
                });
                actionButtonsContainer.appendChild(clearBtn);*/

        const dictionaryCountLabel = document.createElement("span");
        dictionaryCountLabel.id = "dictionaryWordsCountLabel";
        dictionaryCountLabel.style.marginLeft = "auto";
        dictionaryCountLabel.style.fontSize = "12px";
        dictionaryCountLabel.style.color = "#333";
        dictionaryCountLabel.style.whiteSpace = "nowrap";
        dictionaryWordsCountLabelElement = dictionaryCountLabel;
        updateDictionaryWordsCountLabel();
        actionButtonsContainer.appendChild(dictionaryCountLabel);

        section.appendChild(actionButtonsContainer);

        // Campo de búsqueda
        const search = document.createElement("input");
        search.type = "text";
        search.placeholder = "Buscar en diccionario...";
        search.style.display = "block";
        search.style.width = "calc(100% - 14px)";
        search.style.padding = "6px";
        search.style.border = "1px solid #ccc";
        search.style.borderRadius = "3px";
        search.style.marginTop = "5px";
        search.style.marginBottom = "10px";

        // Implementar búsqueda en tiempo real
        search.addEventListener("input", () => {
            const searchTerm = search.value.trim().toLowerCase();
            const filteredData = prepararDatosDiccionario().filter(item =>
                item.termino.toLowerCase().includes(searchTerm)
            );

            // Actualizar la visualización con datos filtrados
            const listContainer = document.getElementById("dictionaryContainer");
            if (listContainer) {
                listContainer.innerHTML = "";
                inicializarDiccionarioDinamico("dictionaryContainer", filteredData);
            }
        });
        section.appendChild(search);

        // Contenedor para el diccionario con carga bajo demanda
        const dictionaryContainer = document.createElement("div");
        dictionaryContainer.id = "dictionaryContainer";
        dictionaryContainer.style.maxHeight = "250px";
        dictionaryContainer.style.overflowY = "auto";
        dictionaryContainer.style.border = "1px solid #ddd";
        dictionaryContainer.style.padding = "5px";
        dictionaryContainer.style.margin = "0";
        dictionaryContainer.style.background = "#fff";
        section.appendChild(dictionaryContainer);

        // Función helper para crear modales de confirmación personalizadas
        function showCustomConfirm(message, title = "Confirmación", icon = "⚠️") {
            return new Promise((resolve) => {
                // Crear overlay
                const overlay = document.createElement("div");
                overlay.style.position = "fixed";
                overlay.style.top = "0";
                overlay.style.left = "0";
                overlay.style.width = "100%";
                overlay.style.height = "100%";
                overlay.style.backgroundColor = "rgba(0,0,0,0.5)";
                overlay.style.zIndex = "19999";

                // Crear modal
                const modal = document.createElement("div");
                modal.style.position = "fixed";
                modal.style.top = "50%";
                modal.style.left = "50%";
                modal.style.transform = "translate(-50%, -50%)";
                modal.style.background = "#fff";
                modal.style.border = "1px solid #aad";
                modal.style.padding = "28px 32px 20px 32px";
                modal.style.zIndex = "20000";
                modal.style.boxShadow = "0 4px 24px rgba(0,0,0,0.18)";
                modal.style.fontFamily = "sans-serif";
                modal.style.borderRadius = "10px";
                modal.style.textAlign = "center";
                modal.style.minWidth = "400px";
                modal.style.maxWidth = "500px";

                // Ícono
                const iconElement = document.createElement("div");
                iconElement.textContent = icon;
                iconElement.style.fontSize = "38px";
                iconElement.style.marginBottom = "10px";
                modal.appendChild(iconElement);

                // Título
                const titleElement = document.createElement("div");
                titleElement.innerHTML = `<b>${title}</b>`;
                titleElement.style.fontSize = "20px";
                titleElement.style.marginBottom = "15px";
                modal.appendChild(titleElement);

                // Mensaje
                const messageElement = document.createElement("div");
                messageElement.textContent = message;
                messageElement.style.fontSize = "15px";
                messageElement.style.color = "#555";
                messageElement.style.marginBottom = "20px";
                messageElement.style.lineHeight = "1.5";
                messageElement.style.whiteSpace = "pre-line";
                modal.appendChild(messageElement);

                // Botones
                const buttonWrapper = document.createElement("div");
                buttonWrapper.style.display = "flex";
                buttonWrapper.style.justifyContent = "center";
                buttonWrapper.style.gap = "12px";

                const cancelBtn = document.createElement("button");
                cancelBtn.textContent = "Cancelar";
                cancelBtn.style.padding = "7px 18px";
                cancelBtn.style.background = "#6c757d";
                cancelBtn.style.color = "#fff";
                cancelBtn.style.border = "none";
                cancelBtn.style.borderRadius = "4px";
                cancelBtn.style.cursor = "pointer";
                cancelBtn.style.fontWeight = "bold";
                cancelBtn.addEventListener("click", () => {
                    overlay.remove();
                    modal.remove();
                    resolve(false);
                });

                const confirmBtn = document.createElement("button");
                confirmBtn.textContent = "Confirmar";
                confirmBtn.style.padding = "7px 18px";
                confirmBtn.style.background = "#007bff";
                confirmBtn.style.color = "#fff";
                confirmBtn.style.border = "none";
                confirmBtn.style.borderRadius = "4px";
                confirmBtn.style.cursor = "pointer";
                confirmBtn.style.fontWeight = "bold";
                confirmBtn.addEventListener("click", () => {
                    overlay.remove();
                    modal.remove();
                    resolve(true);
                });

                buttonWrapper.appendChild(cancelBtn);
                buttonWrapper.appendChild(confirmBtn);
                modal.appendChild(buttonWrapper);

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

        // Función helper para mostrar alertas personalizadas
        function showCustomAlert(message, title = "Información", icon = "ℹ️", type = "info") {
            return new Promise((resolve) => {
                // Crear overlay
                const overlay = document.createElement("div");
                overlay.style.position = "fixed";
                overlay.style.top = "0";
                overlay.style.left = "0";
                overlay.style.width = "100%";
                overlay.style.height = "100%";
                overlay.style.backgroundColor = "rgba(0,0,0,0.5)";
                overlay.style.zIndex = "19999";

                // Crear modal
                const modal = document.createElement("div");
                modal.style.position = "fixed";
                modal.style.top = "50%";
                modal.style.left = "50%";
                modal.style.transform = "translate(-50%, -50%)";
                modal.style.background = "#fff";
                modal.style.border = "1px solid #aad";
                modal.style.padding = "28px 32px 20px 32px";
                modal.style.zIndex = "20000";
                modal.style.boxShadow = "0 4px 24px rgba(0,0,0,0.18)";
                modal.style.fontFamily = "sans-serif";
                modal.style.borderRadius = "10px";
                modal.style.textAlign = "center";
                modal.style.minWidth = "400px";
                modal.style.maxWidth = "500px";

                // Ícono
                const iconElement = document.createElement("div");
                iconElement.textContent = icon;
                iconElement.style.fontSize = "38px";
                iconElement.style.marginBottom = "10px";
                modal.appendChild(iconElement);

                // Título
                const titleElement = document.createElement("div");
                titleElement.innerHTML = `<b>${title}</b>`;
                titleElement.style.fontSize = "20px";
                titleElement.style.marginBottom = "15px";
                modal.appendChild(titleElement);

                // Mensaje
                const messageElement = document.createElement("div");
                messageElement.textContent = message;
                messageElement.style.fontSize = "15px";
                messageElement.style.color = "#555";
                messageElement.style.marginBottom = "20px";
                messageElement.style.lineHeight = "1.5";
                messageElement.style.whiteSpace = "pre-line";
                modal.appendChild(messageElement);

                // Botón OK
                const okBtn = document.createElement("button");
                okBtn.textContent = "OK";
                okBtn.style.padding = "7px 18px";
                okBtn.style.background = type === "error" ? "#dc3545" : (type === "success" ? "#28a745" : "#007bff");
                okBtn.style.color = "#fff";
                okBtn.style.border = "none";
                okBtn.style.borderRadius = "4px";
                okBtn.style.cursor = "pointer";
                okBtn.style.fontWeight = "bold";
                okBtn.addEventListener("click", () => {
                    overlay.remove();
                    modal.remove();
                    resolve(true);
                });

                modal.appendChild(okBtn);

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

        // ✨ Función helper para mostrar prompts personalizados con input (28-nov-2024)
        function showCustomPrompt(message, title = "Entrada de Datos", defaultValue = "", icon = "✏️") {
            return new Promise((resolve) => {
                const overlay = document.createElement("div");
                overlay.style.cssText = "position:fixed;top:0;left:0;width:100%;height:100%;background:rgba(0,0,0,0.5);z-index:19999";

                const modal = document.createElement("div");
                modal.style.cssText = "position:fixed;top:50%;left:50%;transform:translate(-50%,-50%);background:#fff;border:1px solid #aad;padding:28px 32px 20px 32px;z-index:20000;box-shadow:0 4px 24px rgba(0,0,0,0.18);font-family:sans-serif;border-radius:10px;text-align:center;min-width:400px;max-width:500px";

                const iconElement = document.createElement("div");
                iconElement.textContent = icon;
                iconElement.style.cssText = "font-size:38px;margin-bottom:10px";
                modal.appendChild(iconElement);

                const titleElement = document.createElement("div");
                titleElement.innerHTML = `<b>${title}</b>`;
                titleElement.style.cssText = "font-size:20px;margin-bottom:15px";
                modal.appendChild(titleElement);

                const messageElement = document.createElement("div");
                messageElement.textContent = message;
                messageElement.style.cssText = "font-size:15px;color:#555;margin-bottom:15px;line-height:1.5;white-space:pre-line";
                modal.appendChild(messageElement);

                const inputElement = document.createElement("input");
                inputElement.type = "text";
                inputElement.value = defaultValue;
                inputElement.style.cssText = "width:100%;padding:8px 12px;font-size:14px;border:1px solid #ccc;border-radius:4px;margin-bottom:20px;box-sizing:border-box";
                modal.appendChild(inputElement);

                const buttonWrapper = document.createElement("div");
                buttonWrapper.style.cssText = "display:flex;justify-content:center;gap:12px";

                const cancelBtn = document.createElement("button");
                cancelBtn.textContent = "Cancelar";
                cancelBtn.style.cssText = "padding:7px 18px;background:#6c757d;color:#fff;border:none;border-radius:4px;cursor:pointer;font-weight:bold";
                cancelBtn.addEventListener("click", () => {
                    overlay.remove();
                    modal.remove();
                    resolve(null);
                });

                const confirmBtn = document.createElement("button");
                confirmBtn.textContent = "Aceptar";
                confirmBtn.style.cssText = "padding:7px 18px;background:#007bff;color:#fff;border:none;border-radius:4px;cursor:pointer;font-weight:bold";
                confirmBtn.addEventListener("click", () => {
                    overlay.remove();
                    modal.remove();
                    resolve(inputElement.value);
                });

                buttonWrapper.appendChild(cancelBtn);
                buttonWrapper.appendChild(confirmBtn);
                modal.appendChild(buttonWrapper);

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

                setTimeout(() => {
                    inputElement.focus();
                    inputElement.select();
                }, 100);

                inputElement.addEventListener("keydown", (e) => {
                    if (e.key === "Enter") confirmBtn.click();
                    else if (e.key === "Escape") cancelBtn.click();
                });
            });
        }

        // Botón para recargar Diccionario Base (antes del área de arrastrar)
        const reloadFromSheetBtn = document.createElement("button");
        reloadFromSheetBtn.textContent = "Recargar Diccionario Base";
        reloadFromSheetBtn.title = "Limpiar diccionario local y recargar el Básico";
        reloadFromSheetBtn.style.padding = "8px 12px";
        reloadFromSheetBtn.style.cursor = "pointer";
        reloadFromSheetBtn.style.backgroundColor = "#ff9800";
        reloadFromSheetBtn.style.color = "white";
        reloadFromSheetBtn.style.border = "none";
        reloadFromSheetBtn.style.borderRadius = "4px";
        reloadFromSheetBtn.style.marginTop = "10px";
        reloadFromSheetBtn.style.width = "100%";
        reloadFromSheetBtn.style.fontSize = "13px";
        reloadFromSheetBtn.style.fontWeight = "bold";
        reloadFromSheetBtn.addEventListener("click", async function () {
            // Primera confirmación
            const firstConfirm = await showCustomConfirm(
                "Esta acción eliminará TODAS las palabras del diccionario local (incluyendo las que hayas agregado manualmente) y solo cargará las del Diccionario Base.\n\n¿Deseas continuar?",
                "⚠️ ADVERTENCIA",
                "⚠️"
            );

            if (!firstConfirm) return;

            // Segunda confirmación
            const secondConfirm = await showCustomConfirm(
                "¿Estás COMPLETAMENTE SEGURO de que deseas eliminar el diccionario local y recargar el Diccionario por Defecto?\n\nEsta acción NO se puede deshacer.",
                "⚠️ ÚLTIMA CONFIRMACIÓN",
                "⚠️"
            );

            if (!secondConfirm) return;

            try {
                // Deshabilitar el botón durante la operación
                reloadFromSheetBtn.disabled = true;
                reloadFromSheetBtn.textContent = "Recargando...";

                // 1. Limpiar el diccionario en memoria
                if (window.dictionaryWords) {
                    window.dictionaryWords.clear();
                }
                if (window.dictionaryIndex) {
                    window.dictionaryIndex = {};
                }

                // 2. Limpiar el localStorage
                localStorage.removeItem("dictionaryWordsList");
                localStorage.removeItem("wme_pln_dictionary_cache");
                localStorage.removeItem("wme_pln_dictionary_cache");
                // localStorage.removeItem("userAddedDictionaryWords"); // NO BORRAR palabras de usuario al recargar base

                // 3. Reinicializar las estructuras
                window.dictionaryWords = new Set();
                window.dictionaryIndex = {};
                window.dictionaryIndex = {};
                // window.userAddedDictionaryWords = new Set(); // NO REINICIALIZAR palabras de usuario

                // 4. Recargar Diccionario Base (forzando la recarga)
                await loadDictionaryWordsFromSheet(true);

                // ✨ Limpiar cache de lookups después de recargar diccionario
                clearWordLookupCache();

                // 5. Actualizar la UI
                const listContainer = document.getElementById("dictionaryContainer");
                if (listContainer) {
                    listContainer.innerHTML = "";
                    inicializarDiccionarioDinamico("dictionaryContainer", prepararDatosDiccionario());
                }

                updateDictionaryWordsCountLabel();

                // Mensaje de éxito
                await showCustomAlert(
                    `Diccionario por defecto Recargado con Éxito.\n\nTotal de palabras: ${window.dictionaryWords.size}`,
                    "✅ Éxito",
                    "✅",
                    "success"
                );

            } catch (error) {
                console.error("[WME PLN] Error recargando diccionario Base:", error);
                await showCustomAlert(
                    "Error al recargar el diccionario Base.\n\nRevisa la consola para más detalles.",
                    "❌ Error",
                    "❌",
                    "error"
                );
            } finally {
                // Rehabilitar el botón
                reloadFromSheetBtn.disabled = false;
                reloadFromSheetBtn.textContent = "Recargar Diccionario Base";
            }
        });
        section.appendChild(reloadFromSheetBtn);

        // ✨ Botón para exportar palabras propias para diccionario (28-nov-2024)
        const exportUserWordsBtn = document.createElement("button");
        exportUserWordsBtn.textContent = "📤 Exportar Mis Palabras";
        exportUserWordsBtn.title = "Exporta las palabras que has añadido manualmente al diccionario local en formato PALABRA;ES";
        exportUserWordsBtn.style.padding = "8px 12px";
        exportUserWordsBtn.style.cursor = "pointer";
        exportUserWordsBtn.style.backgroundColor = "#4CAF50";
        exportUserWordsBtn.style.color = "white";
        exportUserWordsBtn.style.border = "none";
        exportUserWordsBtn.style.borderRadius = "4px";
        exportUserWordsBtn.style.marginTop = "8px";
        exportUserWordsBtn.style.width = "100%";
        exportUserWordsBtn.style.fontSize = "12px";
        exportUserWordsBtn.style.fontWeight = "bold";

        exportUserWordsBtn.addEventListener("click", function () {
            if (!window.userAddedDictionaryWords || window.userAddedDictionaryWords.size === 0) {
                plnToast("⚠️ No hay palabras propias para exportar.", 3000);
                return;
            }

            // Generar el contenido en formato PALABRA;ES
            const lines = Array.from(window.userAddedDictionaryWords)
                .sort()
                .map(word => `${word};ES`);

            const content = lines.join('\n');

            // Crear el archivo para descargar
            const blob = new Blob([content], { type: 'text/plain;charset=utf-8' });
            const url = URL.createObjectURL(blob);
            const a = document.createElement('a');
            a.href = url;
            a.download = `diccionario_palabras_propias_${new Date().toISOString().slice(0, 10)}.txt`;
            document.body.appendChild(a);
            a.click();
            document.body.removeChild(a);
            URL.revokeObjectURL(url);

            // Mensaje de confirmación
            const count = window.userAddedDictionaryWords.size;
            plnToast(`✅ Exportadas ${count} palabras propias.`, 3000);
        });

        section.appendChild(exportUserWordsBtn);

        // Área para soltar archivos XML
        const dropArea = document.createElement("div");
        dropArea.textContent = "Arrastra aquí el archivo XML del diccionario";
        dropArea.style.border = "2px dashed #ccc";
        dropArea.style.borderRadius = "4px";
        dropArea.style.padding = "15px";
        dropArea.style.marginTop = "10px";
        dropArea.style.textAlign = "center";
        dropArea.style.background = "#f9f9f9";
        dropArea.style.color = "#555";

        // Eventos de arrastrar y soltar
        dropArea.addEventListener("dragover", (e) => {
            e.preventDefault();
            dropArea.style.background = "#e9e9e9";
            dropArea.style.borderColor = "#aaa";
        });

        dropArea.addEventListener("dragleave", () => {
            dropArea.style.background = "#f9f9f9";
            dropArea.style.borderColor = "#ccc";
        });

        dropArea.addEventListener("drop", (e) => {
            e.preventDefault();
            dropArea.style.background = "#f9f9f9";
            const file = e.dataTransfer.files[0];
            if (file && (file.type === "text/xml" || file.name.endsWith(".xml"))) {
                const reader = new FileReader();
                reader.onload = function (evt) {
                    try {
                        const parser = new DOMParser();
                        const xmlDoc = parser.parseFromString(evt.target.result, "application/xml");
                        const parserError = xmlDoc.querySelector("parsererror");
                        if (parserError) {
                            console.error("[WME PLN] Error parseando XML:", parserError.textContent);
                            alert("Error al parsear el archivo XML del diccionario.");
                            return;
                        }
                        const xmlWords = xmlDoc.querySelectorAll("word");
                        let newWordsAddedCount = 0;
                        for (let i = 0; i < xmlWords.length; i++) {
                            const val = xmlWords[i].textContent.trim();
                            if (val && !window.dictionaryWords.has(val)) {
                                window.dictionaryWords.add(val);
                                newWordsAddedCount++;
                            }
                        }

                        // Actualizar la visualización después de importar
                        const listContainer = document.getElementById("dictionaryContainer");
                        if (listContainer) {
                            listContainer.innerHTML = "";
                            inicializarDiccionarioDinamico("dictionaryContainer", prepararDatosDiccionario());
                        }

                        updateDictionaryWordsCountLabel();

                        if (newWordsAddedCount > 0) {
                            alert(`${newWordsAddedCount} nuevas palabras añadidas desde XML.`);
                        } else {
                            alert("No se encontraron palabras nuevas para añadir.");
                        }
                    } catch (err) {
                        alert("Error procesando el diccionario XML.");
                        console.error("[WME PLN] Error procesando XML:", err);
                    }
                };
                reader.readAsText(file);
            } else {
                alert("Por favor, arrastra un archivo XML válido.");
            }
        });
        section.appendChild(dropArea);

        // Añadir todo al contenedor principal
        parentContainer.appendChild(section);

        // Inicializar el diccionario con carga bajo demanda
        setTimeout(() => {
            inicializarDiccionarioDinamico("dictionaryContainer", prepararDatosDiccionario());
            updateDictionaryWordsCountLabel();
        }, 100);
    }//createDictionaryManager

    // Helper de normalización para comparación de reemplazos (insensible a mayúsculas, tildes y espacios extra)
    function plnNormalizeReplacementKey(s) {
        try {
            return String(s || '')
                .normalize('NFD')
                .replace(/[\u0300-\u036f]/g, '')  // quitar diacríticos
                .replace(/\s+/g, ' ')              // normalizar espacios consecutivos
                .trim()
                .toLowerCase();
        } catch (_) { return String(s || '').toLowerCase().trim(); }
    }

    // Carga las palabras excluidas desde localStorage
    function loadReplacementWordsFromStorage() {
        const savedReplacements = localStorage.getItem("replacementWordsList");
        if (savedReplacements) {
            try {
                replacementWords = JSON.parse(savedReplacements);
                window.replacementWords = replacementWords;
                if (typeof replacementWords !== 'object' || replacementWords === null) { // Asegurar que sea un objeto
                    replacementWords = {};
                    window.replacementWords = replacementWords;
                }
                // Cargar mapa de fuentes (usuario/hoja) y asegurar valores por defecto
                try {
                    const savedSources = localStorage.getItem("replacementSources");
                    window.replacementSources = savedSources ? JSON.parse(savedSources) : {};
                } catch (e) {
                    window.replacementSources = {};
                }
                // Para cualquier reemplazo existente sin fuente, asumir que es del usuario
                Object.keys(replacementWords).forEach(k => {
                    if (!window.replacementSources || typeof window.replacementSources !== 'object') {
                        window.replacementSources = {};
                    }
                    if (!window.replacementSources[k]) {
                        window.replacementSources[k] = 'user';
                    }
                });

                // --- NUEVO: Normalización y eliminación de conflictos con reglas de hoja ---
                try {
                    // Construir índice normalizado de claves para evitar duplicados por mayúsculas/puntos/tildes
                    const seen = new Map(); // norm(from) -> canonical from
                    Object.keys(replacementWords).forEach(from => {
                        const lc = plnNormalizeReplacementKey(from);
                        if (seen.has(lc)) {
                            // Preferir la regla de hoja si hay duplicado, si no mantener la primera
                            const keepKey = (window.replacementSources[seen.get(lc)] === 'sheet') ? seen.get(lc)
                                : (window.replacementSources[from] === 'sheet' ? from : seen.get(lc));
                            const dropKey = keepKey === from ? seen.get(lc) : from;
                            if (dropKey && dropKey !== keepKey) {
                                delete replacementWords[dropKey];
                                if (window.replacementSources) delete window.replacementSources[dropKey];
                            }
                            seen.set(lc, keepKey);
                        } else {
                            seen.set(lc, from);
                        }
                    });
                } catch (_) { }
                // Guardar por si hubo saneo
                saveReplacementWordsToStorage();
            }
            catch (e) {
                console.error("[WME PLN] Error cargando lista de reemplazos desde localStorage:", e);
                replacementWords = {};
                window.replacementWords = replacementWords;
                window.replacementSources = {};
            }
        }
        else {
            replacementWords = {}; // Inicializar si no hay nada guardado
            window.replacementWords = replacementWords;
            window.replacementSources = {};
        }
        if (typeof window.updateReplacementsCountLabel === 'function') {
            window.updateReplacementsCountLabel();
        }
        //console.log("[WME PLN] Reemplazos cargados:",    Object.keys(replacementWords).length, "reglas.");
    }// loadReplacementWordsFromStorage

    // Carga las palabras excluidas desde localStorage
    // Función para guardar las palabras swap en localStorage (formato nuevo)
    function saveSwapWordsToStorage() {
        try {
            localStorage.setItem("swapWords", JSON.stringify(window.swapWords || []));
            //console.log("[WME PLN] SwapWords guardadas en localStorage:", window.swapWords ? window.swapWords.length : 0, "palabras");
        }
        catch (e) {
            console.error("[WME PLN] Error guardando swapWords en localStorage:", e);
        }
    }// saveSwapWordsToStorage

    // Carga las palabras reemplazo
    function saveReplacementWordsToStorage() {
        try {
            localStorage.setItem("replacementWordsList",
                JSON.stringify(replacementWords));
            // Guardar también las fuentes de reemplazo
            try {
                localStorage.setItem("replacementSources", JSON.stringify(window.replacementSources || {}));
            } catch (e) {
                console.warn('[WME PLN] Error guardando replacementSources en localStorage:', e);
            }
            // console.log("[WME PLN] Lista de reemplazos guardada en localStorage.");
        }
        catch (e) {
            console.error("[WME PLN] Error guardando lista de reemplazos en localStorage:", e);
        }
    }// saveReplacementWordsToStorage

    // Carga reemplazos base y fusiona con los del usuario, bloqueando los de hoja
    async function loadReplacementsFromSheet(forceReload = false) {
        const SPREADSHEET_ID = "10HSyonrHd__sC0wO0hmOQyBbNRlWF8UFr6kkOuSru00";
        const API_KEY = "AIzaSyAQbvIQwSPNWfj6CcVEz5BmwfNkao533i8";
        const SHEET_CANDIDATES = [
            "Replace!A2:B",        // Solicitud del usuario: la hoja se llama "Replace"
        ]; // A=from, B=to

        // Asegurar estructuras
        if (typeof replacementWords !== 'object' || !replacementWords) replacementWords = {};

        if (typeof window.replacementSources !== 'object' || !window.replacementSources) window.replacementSources = {};
        // Evitar si no se configuró
        if (SPREADSHEET_ID === "TU_SPREADSHEET_ID" || API_KEY === "TU_API_KEY") {
            console.warn('[WME PLN] SPREADSHEET_ID o API_KEY no configurados para reemplazos.');
            return;
        }

        // Intentar usar caché (24h)
        const cacheKey = 'wme_pln_replacements_cache';
        const cached = localStorage.getItem(cacheKey);
        if (!forceReload && cached) {
            try {
                const { data, timestamp } = JSON.parse(cached);
                if (data && timestamp && (Date.now() - timestamp < 24 * 60 * 60 * 1000)) {
                    // Exponer Set con orígenes fijos de hoja (exactos) para validaciones UI
                    try {
                        window.fixedReplacementFroms = new Set(Array.isArray(data) ? data.map(d => d.from) : []);
                        // También mantener sets normalizados para comparaciones flexibles (tildes, puntos, espacios, mayúsculas)
                        const rows = Array.isArray(data) ? data : [];
                        window.fixedReplacementFromsNorm = new Set(rows.map(d => plnNormalizeReplacementKey(d.from)));
                        // NUEVO: proteger también los destinos (columna B) del Sheet
                        window.fixedReplacementTargets = new Set(rows.map(d => d.to));
                        window.fixedReplacementTargetsNorm = new Set(rows.map(d => plnNormalizeReplacementKey(d.to)));
                        // NUEVO: índice normalizado de hoja y reverso para detectar contradicciones de forma global
                        try {
                            window.fixedReplacementIndex = new Map(rows.map(d => [plnNormalizeReplacementKey(d.from), plnNormalizeReplacementKey(d.to)]));
                            window.fixedReplacementReverseIndex = new Map(rows.map(d => [plnNormalizeReplacementKey(d.to), plnNormalizeReplacementKey(d.from)]));
                            window._fixedReplacementPairsRaw = rows.slice();
                        } catch (_) { window.fixedReplacementIndex = null; window.fixedReplacementReverseIndex = null; }
                    } catch (_) {
                        window.fixedReplacementFroms = new Set();
                        window.fixedReplacementFromsNorm = new Set();
                        window.fixedReplacementTargets = new Set();
                        window.fixedReplacementTargetsNorm = new Set();
                    }

                    // Fusionar datos en memoria
                    mergeSheetReplacementsIntoLocal(data);
                    // NUEVO: poda adicional de reglas de usuario en contradicción
                    try { plnPruneContradictoryUserReplacements?.(); } catch (_) { }
                    // Re-render si existe la lista
                    {
                        const _el = document.getElementById("replacementsListUL") || document.querySelector("#replacementsContainer ul");
                        const _sel = document.getElementById("replacementModeSelector");
                        if (_el) {
                            if (_sel && _sel.value === "swapStart" && typeof renderSwapList === "function") {
                                renderSwapList(_el);
                            }
                            else {
                                renderReplacementsList(_el);
                            }
                        }
                    }
                    return;
                }
            }
            catch (e) {
                console.warn('[WME PLN] Error leyendo caché de reemplazos:', e);
            }
        }

        // Intentar con múltiples posibles rangos por si el nombre de la pestaña difiere
        return new Promise((resolve) => {
            const tryNext = (idx) => {
                if (idx >= SHEET_CANDIDATES.length) {
                    console.warn('[WME PLN] No se encontraron reemplazos en ninguna pestaña candidata.');
                    {
                        try {
                            if (typeof plnCanonicalizeReplacementsBySheet === 'function') plnCanonicalizeReplacementsBySheet();
                        }
                        catch (_) {
                        }
                        requestAnimationFrame(() => {
                            const _el = document.getElementById("replacementsListUL") || document.querySelector("#replacementsContainer ul");
                            const _sel = document.getElementById("replacementModeSelector");
                            if (_el) {
                                if (_sel && _sel.value === "swapStart" && typeof renderSwapList === "function") {
                                    renderSwapList(_el);
                                }
                                else {
                                    renderReplacementsList(_el);
                                }
                            }
                        });
                    }
                    resolve();
                    return;
                }
                const RANGE = SHEET_CANDIDATES[idx];
                const url = `https://sheets.googleapis.com/v4/spreadsheets/${SPREADSHEET_ID}/values/${RANGE}?key=${API_KEY}`;
                // console.log('[WME PLN] Cargando reemplazos desde:', RANGE);
                makeRequest({
                    method: 'GET',
                    url,
                    timeout: 10000,
                    onload: function (response) {
                        if (response.status >= 200 && response.status < 300) {
                            try {
                                const payload = JSON.parse(response.responseText);
                                const rows = Array.isArray(payload.values) ? payload.values : [];
                                const sheetData = rows
                                    .map(r => ({ from: (r[0] ?? '').toString().trim(), to: (r[1] ?? '').toString() }))
                                    .filter(item => item.from.length > 0);
                                if (sheetData.length === 0) {
                                    // Intentar con la siguiente candidata
                                    tryNext(idx + 1);
                                    return;
                                }
                                // Guardar caché y fusionar
                                try { localStorage.setItem(cacheKey, JSON.stringify({ data: sheetData, timestamp: Date.now() })); } catch { }

                                // Exponer Set con orígenes fijos de hoja (exactos) para validaciones UI
                                try {
                                    window.fixedReplacementFroms = new Set(sheetData.map(d => d.from));
                                    // Sets adicionales con claves normalizadas para detección/lock por equivalencia
                                    window.fixedReplacementFromsNorm = new Set(sheetData.map(d => plnNormalizeReplacementKey(d.from)));
                                    window.fixedReplacementTargets = new Set(sheetData.map(d => d.to));
                                    window.fixedReplacementTargetsNorm = new Set(sheetData.map(d => plnNormalizeReplacementKey(d.to)));
                                    // Índices globales normalizados para detección de contradicciones
                                    try {
                                        window.fixedReplacementIndex = new Map(sheetData.map(d => [plnNormalizeReplacementKey(d.from), plnNormalizeReplacementKey(d.to)]));
                                        window.fixedReplacementReverseIndex = new Map(sheetData.map(d => [plnNormalizeReplacementKey(d.to), plnNormalizeReplacementKey(d.from)]));
                                        window._fixedReplacementPairsRaw = sheetData.slice();
                                    } catch (_) { window.fixedReplacementIndex = null; window.fixedReplacementReverseIndex = null; }
                                } catch (_) {
                                    window.fixedReplacementFroms = new Set();
                                    window.fixedReplacementFromsNorm = new Set();
                                    window.fixedReplacementTargets = new Set();
                                    window.fixedReplacementTargetsNorm = new Set();
                                }
                                mergeSheetReplacementsIntoLocal(sheetData);

                                // Poda adicional (defensa en profundidad) de reglas de usuario contradictorias
                                try { plnPruneContradictoryUserReplacements?.(); } catch (_) { }

                                // NUEVO: eliminar reglas de usuario que contradigan reglas de hoja (reverse mapping)
                                try {
                                    const sheetIndex = new Map(); // norm(from) -> norm(to)
                                    sheetData.forEach(({ from, to }) => {
                                        sheetIndex.set(plnNormalizeReplacementKey(from), plnNormalizeReplacementKey(to));
                                    });
                                    let changed = false;
                                    Object.keys(replacementWords).forEach(userFrom => {
                                        if (window.replacementSources[userFrom] === 'sheet') return; // solo usuario
                                        const userTo = String(replacementWords[userFrom] || '');
                                        // Si existe una regla de hoja que sea el reverso (normalizada)
                                        const sheetTo = sheetIndex.get(plnNormalizeReplacementKey(userTo));
                                        if (sheetTo && sheetTo === plnNormalizeReplacementKey(userFrom)) {
                                            delete replacementWords[userFrom];
                                            if (window.replacementSources) delete window.replacementSources[userFrom];
                                            changed = true;
                                        }
                                    });
                                    if (changed) saveReplacementWordsToStorage();
                                }
                                catch (_) {
                                }
                            }
                            catch (e) {
                                console.error('[WME PLN] Error procesando reemplazos básicos:', e);
                            }
                        }
                        else {

                            // Intentar siguiente pestaña si 404/400 (rango inválido)
                            // Para otros códigos, solo avisar y continuar
                            // console.warn(`[WME PLN] Error HTTP ${response.status} al cargar rango ${RANGE}`);
                            tryNext(idx + 1);
                            return;
                        }
                        {
                            try {
                                if (typeof plnCanonicalizeReplacementsBySheet === 'function') plnCanonicalizeReplacementsBySheet();
                            }
                            catch (_) {
                            }
                            requestAnimationFrame(() => {
                                const _el = document.getElementById("replacementsListUL") || document.querySelector("#replacementsContainer ul");
                                const _sel = document.getElementById("replacementModeSelector");
                                if (_el) {
                                    if (_sel && _sel.value === "swapStart" && typeof renderSwapList === "function") {
                                        renderSwapList(_el);
                                    }
                                    else {
                                        renderReplacementsList(_el);
                                    }
                                }
                            });
                        }
                        resolve();
                    },
                    onerror: function () {
                        tryNext(idx + 1);
                    },
                    ontimeout: function () {
                        tryNext(idx + 1);
                    }
                });
            };
            tryNext(0);
        });

        // Fusión de reemplazos de hoja con los del usuario. Previene duplicados; respeta los del usuario.
        function mergeSheetReplacementsIntoLocal(sheetPairs) {
            // Construir un mapa temporal para eliminar duplicados (case-insensitive, último gana)
            const tempMap = new Map(); // key: norm(from), value: {from, to}
            sheetPairs.forEach(({ from, to }) => {
                const key = plnNormalizeReplacementKey(from || '');
                if (!key) return;
                tempMap.set(key, { from, to });
            });

            // NUEVO: construir set de destinos normalizados de la hoja (columna B)
            let sheetTargetsNorm = new Set();
            try {
                sheetTargetsNorm = new Set(sheetPairs.map(p => plnNormalizeReplacementKey(p.to || '')));
            } catch (_) { }

            // Limpiar del caché local (usuario) cualquier regla cuyo origen coincida
            // con una regla de hoja, usando comparación normalizada para evitar
            // variantes de mayúsculas, puntos o tildes.
            try {
                const sheetNormSet = new Set(Array.from(tempMap.keys())); // claves normalizadas
                const keys = Object.keys(replacementWords);
                keys.forEach(fromKey => {
                    const isFromSheet = window.replacementSources && window.replacementSources[fromKey] === 'sheet';
                    if (isFromSheet) return;
                    const normLocal = plnNormalizeReplacementKey(fromKey);
                    if (sheetNormSet.has(normLocal) || sheetTargetsNorm.has(normLocal)) {
                        delete replacementWords[fromKey];
                        if (window.replacementSources) delete window.replacementSources[fromKey];
                    }
                });
            } catch (_) { }

            // Fusionar en replacementWords: la lista fija de hoja SIEMPRE prevalece
            for (const { from, to } of tempMap.values()) {
                replacementWords[from] = to;
                window.replacementSources[from] = 'sheet';

                // ✨ NUEVO: Agregar versión sin tildes para robustez (Institución -> Institucion)
                // Esto permite que reglas con tilde funcionen en entradas sin tilde
                try {
                    const fromNoDiacritics = removeDiacritics(from);
                    // Solo agregar si es diferente y no existe ya una regla específica
                    if (fromNoDiacritics !== from && !replacementWords[fromNoDiacritics]) {
                        replacementWords[fromNoDiacritics] = to;
                        window.replacementSources[fromNoDiacritics] = 'sheet';
                    }
                } catch (_) { }
            }

            // Persistir cambios
            saveReplacementWordsToStorage();
        }
    }// loadReplacementsFromSheet

    // Carga las palabras excluidas desde localStorage
    function saveExcludedWordsToLocalStorage() {
        try {
            localStorage.setItem("excludedWordsList", JSON.stringify(Array.from(excludedWords)));
            // console.log("[WME PLN] Lista de palabras especiales guardada en localStorage.");
        }
        catch (e) {
            console.error("[WME PLN] Error guardando palabras especiales en localStorage:", e);
        }
    }// saveExcludedWordsToLocalStorage

    // Función para guardar los IDs de lugares excluidos en localStorage
    function saveExcludedPlacesToLocalStorage() {
        try {
            // Convertir el Map a un array de arrays antes de stringify
            localStorage.setItem("excludedPlacesList", JSON.stringify(Array.from(excludedPlaces.entries())));
            console.log('[WME PLN] Lugares excluidos guardados:', excludedPlaces.size);
        } catch (e) {
            console.error('[WME PLN] Error guardando lugares excluidos en localStorage:', e);
        }
    }// saveExcludedPlacesToLocalStorage



    // Renderiza la lista de reemplazos
    function renderReplacementsList(ulElement) {
        //console.log("[WME_PLN][DEBUG] renderReplacementsList llamada para:", ulElement ? ulElement.id : "Elemento UL nulo");
        if (!ulElement) {
            //console.error("[WME PLN] Elemento UL para reemplazos no proporcionado a renderReplacementsList.");
            return;
        }
        // Asegurar depuración de contradicciones antes de pintar
        try { plnPruneContradictoryUserReplacements?.(); } catch (_) { }
        ulElement.innerHTML = ""; // Limpiar lista actual
        const entries = Object.entries(replacementWords);
        // Si no hay reemplazos, mostrar mensaje
        if (entries.length === 0) {
            const li = document.createElement("li");
            li.textContent = "No hay reemplazos definidos.";
            li.style.textAlign = "center";
            li.style.color = "#777";
            li.style.padding = "5px";
            ulElement.appendChild(li);
            return;
        }
        // Ordenar alfabéticamente por la palabra original (from)
        entries.sort((a, b) => a[0].toLowerCase().localeCompare(b[0].toLowerCase()));
        entries.forEach(([from, to]) => {
            const li = document.createElement("li");
            li.style.display = "flex";
            li.style.justifyContent = "space-between";
            li.style.alignItems = "center";
            li.style.padding = "4px 2px";
            li.style.borderBottom = "1px solid #f0f0f0";
            // Añadir un tooltip al elemento li
            const textContainer = document.createElement("div");
            textContainer.style.flexGrow = "1";
            textContainer.style.overflow = "hidden";
            textContainer.style.textOverflow = "ellipsis";
            textContainer.style.whiteSpace = "nowrap";
            // Determinar si esta regla proviene de hoja. Además del mapa de fuentes,
            // considerar el set normalizado para bloquear equivalentes (p.ej. "I.E" vs "i.e.").
            const isSheetLocked = (
                (window.replacementSources && window.replacementSources[from] === 'sheet') ||
                (window.fixedReplacementFromsNorm && window.fixedReplacementFromsNorm.has(plnNormalizeReplacementKey(from)))
            );
            const source = isSheetLocked ? 'sheet' : 'user';
            textContainer.title = `Reemplazar "${from}" con "${to}"` + (source === 'sheet' ? ' [bloqueado de hoja]' : '');
            // Crear los spans para mostrar el texto
            const fromSpan = document.createElement("span");
            fromSpan.textContent = from;
            fromSpan.style.fontWeight = "bold";
            textContainer.appendChild(fromSpan);
            // Añadir un espacio entre el "from" y el "to"
            const arrowSpan = document.createElement("span");
            arrowSpan.textContent = " → ";
            arrowSpan.style.margin = "0 5px";
            textContainer.appendChild(arrowSpan);
            // Span para el texto de reemplazo
            const toSpan = document.createElement("span");
            toSpan.textContent = to;
            toSpan.style.color = "#007bff";
            textContainer.appendChild(toSpan);
            // Indicador de bloqueo si es de hoja
            if (source === 'sheet') {
                const lockSpan = document.createElement('span');
                lockSpan.textContent = ' 🔒';
                lockSpan.title = 'Reemplazo bloqueado por definición de la Wazeopedia Colombia';
                textContainer.appendChild(lockSpan);
            }
            // Añadir el contenedor de texto al li
            li.appendChild(textContainer);
            // Botón Editar
            const editBtn = document.createElement("button");
            editBtn.innerHTML = "✏️";
            editBtn.title = "Editar este reemplazo";
            editBtn.style.border = "none";
            editBtn.style.background = "transparent";
            editBtn.style.cursor = "pointer";
            editBtn.style.padding = "2px 4px";
            editBtn.style.fontSize = "14px";
            editBtn.style.marginLeft = "4px";
            if (source === 'sheet') {
                editBtn.disabled = true;
                editBtn.style.opacity = '0.4';
                editBtn.style.cursor = 'not-allowed';
                editBtn.title = 'Reemplazo bloqueado por definición de la Wazeopedia Colombia';
            }
            editBtn.addEventListener("click", async () => {
                if ((window.replacementSources && window.replacementSources[from] === 'sheet')) {
                    alert('Este reemplazo proviene de reglas de la Wazeopedia Colombia y no puede editarse.');
                    return;
                }
                const newFrom = await showCustomPrompt("Editar texto original:", "📝 Editar Reemplazo", from, "✏️");
                if (newFrom === null) return;
                const newTo = await showCustomPrompt("Editar texto de reemplazo:", "📝 Editar Reemplazo", to, "✏️");
                if (newTo === null) return;
                if (!newFrom.trim()) {
                    alert("El campo 'Texto Original' es requerido.");
                    return;
                }
                if (newFrom === newTo) {
                    alert("El texto original y el de reemplazo no pueden ser iguales.");
                    return;
                }
                // Permitir que el destino contenga al origen y viceversa (ej. "Av" → "Av.", "CED." → "CED").
                // La prevención de bucles y conflictos se maneja más adelante con reglas
                // de hoja (reversa) y con el motor de reemplazo.
                // Regla: impedir editar para usar un ORIGEN que está en la lista fija si esta regla no es de hoja
                try {
                    if (
                        (window.fixedReplacementFroms && window.fixedReplacementFroms.has(newFrom)) ||
                        (window.fixedReplacementFromsNorm && window.fixedReplacementFromsNorm.has(plnNormalizeReplacementKey(newFrom))) ||
                        (window.fixedReplacementTargets && window.fixedReplacementTargets.has(newFrom)) ||
                        (window.fixedReplacementTargetsNorm && window.fixedReplacementTargetsNorm.has(plnNormalizeReplacementKey(newFrom)))
                    ) {
                        alert("Este origen está reservado por la lista de Wazeopedia Colombia (como origen o destino) y no puede usarse como 'Texto Original'.");
                        return;
                    }
                }
                catch (_) { }
                // Bloqueo por conflicto con regla de hoja (reversa)
                try {
                    const nFromLC = plnNormalizeReplacementKey(newFrom);
                    const nToLC = plnNormalizeReplacementKey(newTo);
                    const keys = Object.keys(replacementWords || {});
                    for (const k of keys) {
                        if (window.replacementSources && window.replacementSources[k] === 'sheet') {
                            const sFromLC = plnNormalizeReplacementKey(k);
                            const sToLC = plnNormalizeReplacementKey(replacementWords[k] || '');
                            if (sFromLC === nToLC && sToLC === nFromLC) {
                                alert(`Esta edición contradice una regla bloqueada en Wazeopedia Colombia: "${k}" → "${replacementWords[k]}".`);
                                return;
                            }
                        }
                    }
                } catch (_) { }

                // Duplicado case-insensitive de la clave
                try {
                    const nLc = plnNormalizeReplacementKey(newFrom);
                    const existingCI = Object.keys(replacementWords || {}).find(k => plnNormalizeReplacementKey(k) === nLc && k !== from);
                    if (existingCI) {
                        if (window.replacementSources && window.replacementSources[existingCI] === 'sheet') {
                            alert(`Ya existe una regla en Wazeopedia Colombia para "${existingCI}". No se puede sobrescribir.`);
                            return;
                        }
                        if (!confirm(`Ya existe una regla para "${existingCI}". ¿Deseas sobrescribirla con "${newFrom}" → "${newTo}"?`))
                            return;
                        delete replacementWords[existingCI];
                        if (window.replacementSources) delete window.replacementSources[existingCI];
                    }
                } catch (_) { }
                // Si cambia la clave, elimina la anterior
                if (newFrom !== from) delete replacementWords[from];
                replacementWords[newFrom] = newTo;
                if (!window.replacementSources) window.replacementSources = {};
                window.replacementSources[newFrom] = 'user';
                delete window.replacementSources[from];
                renderReplacementsList(ulElement);
                saveReplacementWordsToStorage();
            });

            // Botón Eliminar
            const deleteBtn = document.createElement("button");
            deleteBtn.innerHTML = "🗑️";
            deleteBtn.title = `Eliminar este reemplazo`;
            deleteBtn.style.border = "none";
            deleteBtn.style.background = "transparent";
            deleteBtn.style.cursor = "pointer";
            deleteBtn.style.padding = "2px 4px";
            deleteBtn.style.fontSize = "14px";
            deleteBtn.style.marginLeft = "4px";
            if (source === 'sheet') {
                deleteBtn.disabled = true;
                deleteBtn.style.opacity = '0.4';
                deleteBtn.style.cursor = 'not-allowed';
                deleteBtn.title = 'Reemplazo bloqueado desde Wazeopedia Colombia';
            }
            deleteBtn.addEventListener("click", () => {
                if ((window.replacementSources && window.replacementSources[from] === 'sheet')) {
                    alert('Este reemplazo proviene de Wazeopedia Colombia y no puede eliminarse.');
                    return;
                }
                if (confirm(`¿Estás seguro de eliminar el reemplazo:\n"${from}" → "${to}"?`)) {
                    delete replacementWords[from];
                    if (window.replacementSources) delete window.replacementSources[from];
                    renderReplacementsList(ulElement);
                    saveReplacementWordsToStorage();
                }
            });
            // Contenedor para los botones de acción
            const btnContainer = document.createElement("span");
            btnContainer.style.display = "flex";
            btnContainer.style.gap = "4px";
            btnContainer.appendChild(editBtn);
            btnContainer.appendChild(deleteBtn);
            // Añadir el contenedor de botones al li
            li.appendChild(btnContainer);
            ulElement.appendChild(li);
        });
        if (typeof window.updateReplacementsCountLabel === 'function') window.updateReplacementsCountLabel();
    }


    // Exporta las palabras especiales y reemplazos a un archivo XML
    function exportSharedDataToXml() {
        let xmlParts = [];
        const rootTagName = "WME_PLN_Backup";
        const fileName = "wme_pln_backup.xml";

        if (excludedWords.size === 0 && excludedPlaces.size === 0 && Object.keys(replacementWords).length === 0 &&
            (!window.swapWords || window.swapWords.length === 0) && Object.keys(editorStats).length === 0) {
            alert("No hay datos (palabras especiales, lugares excluidos, reemplazos, swap, estadísticas) para exportar.");
            return;
        }

        // Exportar palabras especiales (excludedWords)
        if (excludedWords.size > 0) {
            xmlParts.push("    <words>");
            Array.from(excludedWords)
                .sort((a, b) => a.toLowerCase().localeCompare(b.toLowerCase()))
                .forEach(w => xmlParts.push(`        <word>${xmlEscape(w)}</word>`));
            xmlParts.push("    </words>");
        }

        // Exportar reemplazos (replacementWords)
        if (Object.keys(replacementWords).length > 0) {
            xmlParts.push("    <replacements>");
            Object.entries(replacementWords)
                .sort((a, b) => a[0].toLowerCase().localeCompare(b[0].toLowerCase()))
                .forEach(([from, to]) => {
                    xmlParts.push(`        <replacement from="${xmlEscape(from)}">${xmlEscape(to)}</replacement>`);
                });
            xmlParts.push("    </replacements>");
        }

        // Exportar palabras de intercambio (swapWords)
        if (window.swapWords && window.swapWords.length > 0) {
            xmlParts.push("    <swapWords>");
            window.swapWords.forEach(item => {
                if (typeof item === 'object' && item.word && item.direction) {
                    xmlParts.push(`        <swap word="${xmlEscape(item.word)}" direction="${xmlEscape(item.direction)}"></swap>`);
                }
            });
            xmlParts.push("    </swapWords>");
        }

        // Exportar estadísticas (editorStats)
        if (Object.keys(editorStats).length > 0) {
            xmlParts.push("    <statistics>");
            Object.entries(editorStats).forEach(([userId, data]) => {
                xmlParts.push(`        <editor id="${userId}"
                    name="${xmlEscape(data.userName || '')}"
                    total_count="${data.total_count || 0}"
                    monthly_count="${data.monthly_count || 0}"
                    monthly_period="${data.monthly_period || ''}"
                    weekly_count="${data.weekly_count || 0}"
                    weekly_period="${data.weekly_period || ''}"
                    daily_count="${data.daily_count || 0}"
                    daily_period="${data.daily_period || ''}"
                    last_update="${data.last_update || 0}" />`);
            });
            xmlParts.push("    </statistics>");
        }

        // Exportar lugares excluidos (excludedPlaces)
        if (excludedPlaces.size > 0) {
            xmlParts.push("    <placeIds>");
            Array.from(excludedPlaces.entries())
                .sort((a, b) => (a[1] || '').toLowerCase().localeCompare(b[1] || ''))
                .forEach(([id, name]) => {
                    xmlParts.push(`        <placeId id="${xmlEscape(id)}" name="${xmlEscape(name || '')}"></placeId>`);
                });
            xmlParts.push("    </placeIds>");
        }

        // Construir el contenido XML completo
        const xmlContent = `<?xml version="1.0" encoding="UTF-8"?>\n<${rootTagName}>\n${xmlParts.join("\n")}\n</${rootTagName}>`;

        const blob = new Blob([xmlContent], { type: "application/xml;charset=utf-8" });
        const url = URL.createObjectURL(blob);
        const a = document.createElement("a");
        a.href = url;
        a.download = fileName;
        document.body.appendChild(a);
        a.click();
        document.body.removeChild(a);
        URL.revokeObjectURL(url);
    }

    // [PLN] Canonicalizar por hoja: drop si 'to' ∈ sheetTo; reencaminar si 'to' ∈ sheetFrom
    function plnCanonicalizeReplacementsBySheet() {
        try {
            const map = window.replacementWords || {};
            const src = window.replacementSources || {};

            // Map y conjuntos de hoja (from -> to canónico)
            const sheetFrom = [];
            const sheetMap = {};
            for (const f in map) {
                if (src[f] === 'sheet') {
                    sheetFrom.push(f);
                    sheetMap[f] = String(map[f] || '').trim();
                }
            }
            const sheetFromSet = new Set(sheetFrom);
            const sheetToSet = new Set(sheetFrom.map(f => sheetMap[f]));
            const newMap = {};
            const newSrc = {};
            for (const from in map) {
                const to = String(map[from] || '').trim();
                const isSheet = src[from] === 'sheet';
                // (A) Si el FROM local existe en hoja, imponer el B canónico de hoja
                if (!isSheet && sheetFromSet.has(from)) {
                    const canonicalTo = sheetMap[from];
                    if (canonicalTo) {
                        newMap[from] = canonicalTo;
                        if (src[from]) newSrc[from] = src[from];
                    }
                    continue;
                }
                // (B) Si el FROM local es de hoja, conservar tal cual (ya es canónico)
                if (isSheet) {
                    newMap[from] = to;
                    newSrc[from] = 'sheet';
                    continue;
                }
                // (C) Si el FROM local no es de hoja, aplicar reglas:
                if (sheetFromSet.has(to)) {
                    const canonical = sheetMap[to]; // B de la hoja para ese A
                    if (canonical) {
                        newMap[from] = canonical;
                        if (src[from]) newSrc[from] = src[from];
                    }
                    continue;
                }

                // 2) Si el 'to' local coincide con un 'from' de hoja → reencaminar al B canónico
                if (sheetFromSet.has(to)) {
                    const canonical = sheetMap[to]; // B de la hoja para ese A
                    if (canonical) {
                        newMap[from] = canonical;
                        if (src[from]) newSrc[from] = src[from];
                    }
                    continue;
                }

                // 3) Caso normal: conservar
                newMap[from] = to;
                if (src[from]) newSrc[from] = src[from];
            }
            window.replacementWords = newMap;
            window.replacementSources = newSrc;
        }
        catch (e) {
            try {
                console.error('[WME PLN] plnCanonicalizeReplacementsBySheet error', e);
            }
            catch (_) { }
        }
    }//plnCanonicalizeReplacementsBySheet

    // Función para manejar el archivo XML arrastrado
    function handleXmlFileDrop(file, type = "words") {
        if (file && (file.type === "text/xml" || file.name.endsWith(".xml"))) {
            const reader = new FileReader();
            reader.onload = function (evt) {
                try {
                    let newExcludedAdded = 0;
                    let newReplacementsAdded = 0;
                    let replacementsOverwritten = 0;
                    let newPlacesAdded = 0;

                    const parser = new DOMParser();
                    const xmlDoc = parser.parseFromString(evt.target.result, "application/xml");
                    const parserError = xmlDoc.querySelector("parsererror");
                    if (parserError) {
                        alert("Error al parsear el archivo XML: " + parserError.textContent);
                        return;
                    }
                    const rootTag = xmlDoc.documentElement.tagName.toLowerCase();

                    if (type === "words") {
                        if (rootTag !== "excludedwords") {
                            alert("El archivo XML no es válido para Palabras Especiales. Debe tener <ExcludedWords> como raíz.");
                            return;
                        }

                        const words = xmlDoc.getElementsByTagName("word");
                        for (let i = 0; i < words.length; i++) {
                            const val = words[i].textContent.trim();

                            // --- CORRECCIÓN CLAVE: Omitimos la validación al importar desde XML ---
                            // La validación es útil para la entrada manual, pero demasiado restrictiva para la importación,
                            // ya que rechaza palabras que ya existen en el diccionario.
                            if (val && !excludedWords.has(val)) {
                                excludedWords.add(val);
                                const firstChar = val.charAt(0).toLowerCase();
                                if (!excludedWordsMap.has(firstChar)) excludedWordsMap.set(firstChar, new Set());
                                excludedWordsMap.get(firstChar).add(val);
                                newExcludedAdded++;
                            }
                        }

                        const replacementNodes = xmlDoc.getElementsByTagName("replacement");
                        for (let i = 0; i < replacementNodes.length; i++) {
                            const from = replacementNodes[i].getAttribute("from")?.trim();
                            const to = replacementNodes[i].textContent.trim();
                            if (from && to) {
                                // Saltar si el origen pertenece a la lista fija de Sheets
                                try {
                                    if ((window.fixedReplacementFroms && window.fixedReplacementFroms.has(from)) ||
                                        (window.fixedReplacementFromsNorm && window.fixedReplacementFromsNorm.has(plnNormalizeReplacementKey(from))) ||
                                        (window.fixedReplacementTargets && window.fixedReplacementTargets.has(from)) ||
                                        (window.fixedReplacementTargetsNorm && window.fixedReplacementTargetsNorm.has(plnNormalizeReplacementKey(from)))) {
                                        continue;
                                    }
                                }
                                catch (_) { }
                                // Permitir que el destino contenga al origen y viceversa (ej. "Av" → "Av.", "CED." → "CED").
                                // Pero evitar introducir pares cuyo DESTINO coincide con un ORIGEN de la hoja (normalizado),
                                // para no colisionar con reglas canónicas: en ese caso, saltar la importación.
                                try {
                                    if (window.fixedReplacementFromsNorm && window.fixedReplacementFromsNorm.has(plnNormalizeReplacementKey(to))) {
                                        continue;
                                    }
                                } catch (_) { }
                                if (window.replacementSources && window.replacementSources[from] === 'sheet') {
                                    // No permitir sobreescribir los que vienen de hoja
                                    continue;
                                }
                                if (replacementWords.hasOwnProperty(from) && replacementWords[from] !== to) {
                                    replacementsOverwritten++;
                                }
                                else if (!replacementWords.hasOwnProperty(from)) {
                                    newReplacementsAdded++;
                                }
                                replacementWords[from] = to;
                                if (!window.replacementSources) window.replacementSources = {};
                                window.replacementSources[from] = 'user';
                            }
                        }
                    }
                    else if (type === "places") {
                        if (rootTag !== "placeIds" && rootTag !== "excludedwords") {
                            alert("El archivo XML no es válido para Lugares Excluidos. Debe tener <placeIds> o <ExcludedWords> como raíz.");
                            return;
                        }

                        const placesNodes = xmlDoc.getElementsByTagName("placeId");
                        let placesUpdated = 0;
                        for (let i = 0; i < placesNodes.length; i++) {
                            const placeId = placesNodes[i].getAttribute("id")?.trim();
                            const placeName = placesNodes[i].getAttribute("name")?.trim() || `ID: ${placeId}`;

                            if (placeId) {
                                if (!excludedPlaces.has(placeId)) {
                                    newPlacesAdded++;
                                }
                                else {
                                    placesUpdated++;
                                }
                                // Guardar solo el nombre
                                excludedPlaces.set(placeId, placeName);
                            }
                        }
                        alert(`Importación completada.\n- Lugares nuevos añadidos: ${newPlacesAdded}\n- Lugares existentes actualizados: ${placesUpdated}`);
                        saveExcludedPlacesToLocalStorage();
                    }
                    else {
                        alert("Tipo de importación XML desconocido.");
                        return;
                    }

                    // En handleXmlFileDrop():
                    const swapWordsNode = xmlDoc.querySelector("swapWords");
                    if (swapWordsNode) {
                        if (!window.swapWords) window.swapWords = [];

                        swapWordsNode.querySelectorAll("swap").forEach(swapNode => {
                            const word = swapNode.getAttribute("value");
                            const direction = swapNode.getAttribute("direction") || "start"; // Default a "start"

                            // Saltar si está protegido por hoja (Swap)
                            try {
                                const norm = (word || '').normalize('NFD').replace(/[\u0300-\u036f]/g, '').replace(/\s+/g, ' ').trim().toLowerCase();
                                if (window.fixedSwapNormSet && window.fixedSwapNormSet.has(norm)) return;
                            } catch (_) { }
                            if (word && !window.swapWords.some(item => (typeof item === 'object' ? item.word : item) === word)) {
                                window.swapWords.push({ word: word, direction: direction });
                            }
                        });
                        saveSwapWordsToStorage();
                    }

                    const statsNode = xmlDoc.querySelector("statistics");
                    if (statsNode) {
                        const editorNode = statsNode.querySelector("editor");
                        if (editorNode) {
                            if (editorNode.hasAttribute("total_count")) {
                                if (!currentGlobalUserInfo.id || currentGlobalUserInfo.name === 'No detectado') {
                                    showTemporaryMessage("Espera a que tu usuario se cargue para importar estadísticas.", 5000, "error");
                                }
                                else if (editorNode.getAttribute("id") === String(currentGlobalUserInfo.id)) {
                                    editorStats[currentGlobalUserInfo.id] = {
                                        userName: editorNode.getAttribute("name") || currentGlobalUserInfo.name,
                                        total_count: parseInt(editorNode.getAttribute("total_count"), 10) || 0,
                                        monthly_count: parseInt(editorNode.getAttribute("monthly_count"), 10) || 0,
                                        monthly_period: editorNode.getAttribute("monthly_period") || '',
                                        weekly_count: parseInt(editorNode.getAttribute("weekly_count"), 10) || 0,
                                        weekly_period: editorNode.getAttribute("weekly_period") || '',
                                        daily_count: parseInt(editorNode.getAttribute("daily_count"), 10) || 0,
                                        daily_period: editorNode.getAttribute("daily_period") || '',
                                        last_update: parseInt(editorNode.getAttribute("last_update"), 10) || 0
                                    };
                                }
                            }
                            else {
                                showTemporaryMessage("XML con formato de estadísticas antiguo detectado. Se omitió la importación.", 6000, "warning");
                            }
                        }
                    }

                    saveExcludedWordsToLocalStorage();
                    saveReplacementWordsToStorage();
                    saveEditorStats();
                    plnCanonicalizeReplacementsBySheet();
                    renderExcludedWordsList(document.getElementById("excludedWordsList"));
                    renderExcludedPlacesList(document.getElementById("excludedPlacesListUL"));
                    // [PLN] Re-render según modo actual (replacements vs swap) tras canonicalizar
                    {
                        const _el = document.getElementById("replacementsListUL") || document.querySelector("#replacementsContainer ul");
                        const _sel = document.getElementById("replacementModeSelector");
                        if (_el) {
                            if (_sel && _sel.value === "swapStart" && typeof renderSwapList === "function") {
                                renderSwapList(_el);
                            }
                            else {
                                renderReplacementsList(_el);
                            }
                        }
                    }
                    updateStatsDisplay();

                    alert(`Importación completada. \n - Palabras Especiales nuevas: ${newExcludedAdded} \n - Reemplazos nuevos: ${newReplacementsAdded} \n - Reemplazos sobrescritos: ${replacementsOverwritten} \n - Nuevos lugares: ${newPlacesAdded}`);
                }
                catch (err) {
                    console.error("[WME PLN] Error procesando el archivo XML importado:", err);
                    alert("Ocurrió un error procesando el archivo XML.");
                }
            };
            reader.readAsText(file);
        }
        else {
            alert("Por favor, arrastra un archivo XML válido.");
        }
    }//handleXmlFileDrop

    // Bloquea todos los controles de la pestaña principal durante el escaneo
    function disableScanControls() {
        const idsToDisable = [
            'pln-start-scan-btn', 'maxPlacesInput', 'chk-recommend-categories',
            'chk-avoid-my-edits', 'dateFilterSelect', 'chk-enable-stats'
        ];
        // Deshabilitar los controles principales
        idsToDisable.forEach(id => {
            const el = document.getElementById(id);
            if (el) {
                el.disabled = true;
                el.style.opacity = '0.6';
                el.style.cursor = 'not-allowed';
            }
        });
        // Deshabilitar los botones de presets
        document.querySelectorAll('.pln-preset-btn').forEach(btn => {
            btn.disabled = true;
            btn.style.opacity = '0.6';
            btn.style.cursor = 'not-allowed';
        });
    }// disableScanControls

    // Reactiva todos los controles de la pestaña principal al finalizar el escaneo
    function enableScanControls() {
        const idsToEnable = [
            'pln-start-scan-btn', 'maxPlacesInput', 'chk-recommend-categories',
            'chk-avoid-my-edits', 'dateFilterSelect', 'chk-enable-stats'
        ];
        // Reactivar los controles principales
        idsToEnable.forEach(id => {
            const el = document.getElementById(id);
            if (el) {
                el.disabled = false;
                el.style.opacity = '1';
                el.style.cursor = 'pointer';
            }
        });
        
        // Resetear el texto del botón de escaneo
        const scanBtn = document.getElementById('pln-start-scan-btn');
        if (scanBtn) {
            scanBtn.textContent = "Start Scan...";
        }
        // Reactivar los botones de presets
        document.querySelectorAll('.pln-preset-btn').forEach(btn => {
            btn.disabled = false;
            btn.style.opacity = '1';
            btn.style.cursor = 'pointer';
        });
        // Restaurar el estado del dropdown de fecha según el checkbox
        const avoidMyEditsCheckbox = document.getElementById("chk-avoid-my-edits");
        if (avoidMyEditsCheckbox) {
            const dateFilterRow = document.getElementById("dateFilterSelect").parentElement;
            dateFilterRow.style.opacity = avoidMyEditsCheckbox.checked ? "1" : "0.5";
            dateFilterRow.style.pointerEvents = avoidMyEditsCheckbox.checked ? "auto" : "none";
        }
    }// enableScanControls

    // Carga las palabras swap desde localStorage
    // Función para cargar las palabras swap desde localStorage con migración automática
    function loadSwapWordsFromStorage() {
        const stored = localStorage.getItem("swapWords");

        // Si hay datos en localStorage, intentar parsearlos
        if (stored) {
            try {
                const parsed = JSON.parse(stored);

                // MIGRACIÓN AUTOMÁTICA: Verificar el formato de los datos
                if (Array.isArray(parsed) && parsed.length > 0) {
                    // Verificar si es formato antiguo (array de strings)
                    if (typeof parsed[0] === "string") {
                        //console.log("[WME PLN] Detectado formato antiguo de swapWords. Migrando automáticamente...");

                        // Migrar formato antiguo a nuevo formato
                        window.swapWords = parsed.map(word => ({
                            word: word,
                            direction: "start" // Todas las palabras existentes se configuran como "start" por defecto
                        }));

                        // Guardar el nuevo formato inmediatamente
                        saveSwapWordsToStorage();
                        //console.log(`[WME PLN] Migración completada: ${window.swapWords.length} palabras migradas a formato 'start'.`);
                    }
                    else if (typeof parsed[0] === "object" && parsed[0].hasOwnProperty('word')) {
                        // Ya está en formato nuevo
                        window.swapWords = parsed;
                        //console.log("[WME PLN] Formato nuevo de swapWords detectado. No se requiere migración.");
                    }
                    else {
                        // Formato desconocido, inicializar vacío
                        console.warn("[WME PLN] Formato desconocido en swapWords. Inicializando lista vacía.");
                        window.swapWords = [];
                    }
                }
                else {
                    // Array vacío o null
                    window.swapWords = [];
                }
            }
            catch (e) {
                console.error("[WME PLN] Error al parsear swapWords desde localStorage:", e);
                window.swapWords = [];
            }
        }
        else {
            // No hay datos guardados
            window.swapWords = [];
        }
        if (typeof window.updateSwapWordsCountLabel === 'function') {
            window.updateSwapWordsCountLabel();
        }
    }// loadSwapWordsFromStorage

    // === Cargar Swap por defecto (hoja "Swap"): protege las entradas de hoja ===
    async function loadSwapWordsFromSheet(forceReload = false) {
        const SPREADSHEET_ID = "10HSyonrHd__sC0wO0hmOQyBbNRlWF8UFr6kkOuSru00";
        const API_KEY = "AIzaSyAQbvIQwSPNWfj6CcVEz5BmwfNkao533i8";
        const RANGE = "Swap!A2:B"; // A=token, B=direction

        const normDir = v => {
            const s = String(v || '').trim().toLowerCase();
            if (s === 'start') return 'start';
            if (s === 'end') return 'end';
            return null;
        };
        const plnNormalizeSwapKey = s => {
            try {
                return String(s || '')
                    .normalize('NFD')
                    .replace(/[\u0300-\u036f]/g, '')
                    .replace(/\s+/g, ' ')
                    .trim()
                    .toLowerCase();
            } catch (_) { return String(s || '').toLowerCase().trim(); }
        };

        if (SPREADSHEET_ID === "TU_SPREADSHEET_ID" || API_KEY === "TU_API_KEY") return;

        // Cache 24h
        const cacheKey = 'wme_pln_swap_cache';
        if (!forceReload) {
            try {
                const cached = JSON.parse(localStorage.getItem(cacheKey) || 'null');
                if (cached && cached.timestamp && (Date.now() - cached.timestamp < 24 * 60 * 60 * 1000)) {
                    applySwapSheetRows(cached.data || []);
                    return;
                }
            } catch (_) { }
        }

        return new Promise(resolve => {
            const url = `https://sheets.googleapis.com/v4/spreadsheets/${SPREADSHEET_ID}/values/${RANGE}?key=${API_KEY}`;
            makeRequest({
                method: 'GET', url, timeout: 10000,
                onload: (res) => {
                    try {
                        const payload = JSON.parse(res.responseText || '{}');
                        const rows = Array.isArray(payload.values) ? payload.values : [];
                        const data = rows.map(r => ({ word: (r[0] || '').toString().trim(), direction: normDir(r[1]) }))
                            .filter(x => x.word && x.direction);
                        try { localStorage.setItem(cacheKey, JSON.stringify({ data, timestamp: Date.now() })); } catch (_) { }
                        applySwapSheetRows(data);
                    } catch (e) { /* noop */ }
                    resolve();
                },
                onerror: () => resolve(),
                ontimeout: () => resolve()
            });
        });

        function applySwapSheetRows(rows) {
            try {
                // Crear set de palabras protegidas por hoja (normalizadas)
                window.fixedSwapNormSet = new Set(rows.map(r => plnNormalizeSwapKey(r.word)));
                if (!Array.isArray(window.swapWords)) window.swapWords = [];
                // Quitar de memoria local cualquier palabra de usuario que choque con la hoja
                window.swapWords = window.swapWords.filter(item => !window.fixedSwapNormSet.has(plnNormalizeSwapKey(item.word || item)));
                // Añadir las de hoja (canónicas)
                rows.forEach(r => {
                    const direction = r.direction === 'end' ? 'end' : 'start';
                    window.swapWords.push({ word: r.word, direction });
                });
                try { saveSwapWordsToStorage(); } catch (_) { }
                if (typeof window.updateSwapWordsCountLabel === 'function') {
                    window.updateSwapWordsCountLabel();
                }
            } catch (_) { }
        }
    }

    // === Recolector unificado de reglas de swap ===
    function plnCollectSwapRules() {
        try {
            const normDir = v => {
                v = String(v || '').toLowerCase();
                if (v === 'start') return 'before';
                if (v === 'end') return 'after';
                return null;
            };
            const key = s => String(s || '')
                .normalize('NFD').replace(/[\u0300-\u036f]/g, '')
                .toLowerCase().trim();

            // Precedencia: 1) window.swapWords  2) UI lists  3) localStorage  -> luego OVERRIDES
            const map = new Map(); // k -> { word, position, _pri }
            const setRule = (w, d, pri) => {
                w = String(w || '').trim();
                d = normDir(d);
                if (!w || !d) return;
                const k = key(w);
                const prev = map.get(k);
                if (!prev || (prev._pri || 0) <= pri) {
                    map.set(k, { word: w, position: d, _pri: pri });
                }
            };

            // 1) Base: window.swapWords
            (Array.isArray(window.swapWords) ? window.swapWords : []).forEach(x => {
                if (!x) return;
                if (typeof x === 'string') { setRule(x, 'before', 1); return; }
                const w = x.word || x.text || x.token || x.value || x.name;
                const d = x.position || x.where || x.dir || x.direction;
                setRule(w, d, 1);
            });

            // 2) Listas de la UI si existen
            const toArr = v => Array.isArray(v) ? v : (typeof v === 'string' ? (() => { try { return JSON.parse(v); } catch { return [] } })() : []);
            const sStart = window.swapWordsStart || window.wordsStart || window.startWords || null;
            const sEnd = window.swapWordsEnd || window.wordsEnd || window.endWords || null;
            [
                { source: sStart, dir: 'before' },
                { source: sEnd, dir: 'after' }
            ].forEach(({ source, dir }) => {
                if (source) {
                    toArr(source).forEach(w => setRule(w, dir, 2));
                }
            });

            // 3) Escaneo de localStorage por claves relacionadas
            try {
                for (let i = 0; i < localStorage.length; i++) {
                    const k = localStorage.key(i);
                    const low = k.toLowerCase();
                    if (!/(^|:|_)swap(words)?|\bwords\b|palabra/i.test(low)) continue;
                    let val = null; try { val = JSON.parse(localStorage.getItem(k)); } catch { }
                    if (!val) continue;
                    if (Array.isArray(val)) {
                        const allStr = val.every(x => typeof x === 'string');
                        const allObj = val.every(x => x && typeof x === 'object');
                        if (allStr) {
                            if (/inicio|start|before/.test(low)) val.forEach(w => setRule(w, 'before', 3));
                            else if (/final|end|after/.test(low)) val.forEach(w => setRule(w, 'after', 3));
                        } else if (allObj) {
                            val.forEach(x => {
                                const w = x.word || x.text || x.token || x.value || x.name;
                                let d = normDir(x.position || x.where || x.dir || x.direction);
                                if (!d) {
                                    if (/inicio|start|before/.test(low)) d = 'before';
                                    else if (/final|end|after/.test(low)) d = 'after';
                                }
                                setRule(w, d, 3);
                            });
                        }
                    }
                }
            } catch { }


            return Array.from(map.values());
        } catch (e) {
            try { console.error('[PLN Swap] plnCollectSwapRules error', e); } catch { }
            // Fallback mínimo a window.swapWords normalizado + override
            const key = s => String(s || '').normalize('NFD').replace(/[\u0300-\u036f]/g, '').toLowerCase().trim();
            const base = [];
            (Array.isArray(window.swapWords) ? window.swapWords : []).forEach(x => {
                if (!x) return;
                if (typeof x === 'string') base.push({ word: String(x).trim(), position: 'before' });
                else {
                    const w = x.word || x.text || x.token || x.value || x.name;
                    const d = String(x.position || x.where || x.dir || x.direction || 'before').toLowerCase();
                    base.push({ word: String(w || '').trim(), position: (d === 'after' || d === 'end' || d === 'despues' || d === 'después' || d === 'post') ? 'after' : 'before' });
                }
            });

            base.sort((a, b) => b.word.length - a.word.length);
            return base;
        }
    }
    // Función para calcular el área en metros cuadrados de una geometría de polígono
    // NOTA: Turf.js ha sido deshabilitado temporalmente, usando función alternativa
    function calculateAreaMeters(shape) {
        if (!shape || !shape.geometry) {
            return null; // Return null instead of Infinity
        }

        try {
            // Check if we have a valid polygon geometry
            if (shape.geometry.type === 'Polygon') {
                // Extract coordinates from the polygon
                const coordinates = shape.geometry.coordinates[0]; // First ring of coordinates

                if (!coordinates || !Array.isArray(coordinates) || coordinates.length < 3) {
                    return null; // Not enough points for a valid polygon
                }

                // Use the Shoelace formula (Gauss's area formula) to calculate polygon area
                let area = 0;
                for (let i = 0; i < coordinates.length - 1; i++) {
                    if (!Array.isArray(coordinates[i]) || !Array.isArray(coordinates[i + 1]) ||
                        coordinates[i].length < 2 || coordinates[i + 1].length < 2) {
                        return null; // Invalid coordinate pair
                    }
                    area += coordinates[i][0] * coordinates[i + 1][1];
                    area -= coordinates[i][1] * coordinates[i + 1][0];
                }

                area = Math.abs(area) / 2;

                // Convert to square meters based on projection
                // This assumes the coordinates are in a geographic coordinate system
                const metersPerDegree = 111319.9; // Approximate meters per degree at the equator
                return area * Math.pow(metersPerDegree, 2);
            }
        } catch (error) {
            console.warn("[WME PLN] Error calculating area:", error);
            return null; // Return null on error
        }

        return null; // Default return for non-polygon shapes
    }// calculateAreaMeters

    // Crea el gestor de reemplazos
    function createReplacementsManager(parentContainer) {
        loadSwapWordsFromStorage();
        parentContainer.innerHTML = ''; // Limpiar por si acaso

        // ===================================================================
        // INICIO: NUEVA FUNCIÓN HELPER PARA EL MODAL DE EDICIÓN
        // ===================================================================
        function openSwapWordEditor(item, index) {
            // Crear el fondo del modal
            const modalOverlay = document.createElement("div");
            modalOverlay.style.position = "fixed";
            modalOverlay.style.top = "0";
            modalOverlay.style.left = "0";
            modalOverlay.style.width = "100%";
            modalOverlay.style.height = "100%";
            modalOverlay.style.background = "rgba(0,0,0,0.5)";
            modalOverlay.style.zIndex = "20000";
            modalOverlay.style.display = "flex";
            modalOverlay.style.justifyContent = "center";
            modalOverlay.style.alignItems = "center";

            // Crear el contenido del modal
            const modalContent = document.createElement("div");
            modalContent.style.background = "#fff";
            modalContent.style.padding = "25px";
            modalContent.style.borderRadius = "8px";
            modalContent.style.boxShadow = "0 5px 15px rgba(0,0,0,0.3)";
            modalContent.style.width = "400px";
            modalContent.style.fontFamily = "sans-serif";

            // Título del modal
            const title = document.createElement("h4");
            title.textContent = "Editar Palabra Swap";
            title.style.marginTop = "0";
            title.style.marginBottom = "20px";
            title.style.textAlign = "center";
            modalContent.appendChild(title);

            // Input para la palabra
            const wordLabel = document.createElement("label");
            wordLabel.textContent = "Palabra o Frase:";
            wordLabel.style.display = "block";
            wordLabel.style.marginBottom = "5px";
            modalContent.appendChild(wordLabel);

            const wordInput = document.createElement("input");
            wordInput.type = "text";
            wordInput.value = item.word;
            wordInput.style.width = "calc(100% - 12px)";
            wordInput.style.padding = "8px";
            wordInput.style.marginBottom = "15px";
            wordInput.setAttribute('spellcheck', 'false');
            modalContent.appendChild(wordInput);

            // Radio buttons para la dirección
            const directionFieldset = document.createElement("fieldset");
            directionFieldset.style.border = "1px solid #ccc";
            directionFieldset.style.borderRadius = "5px";
            directionFieldset.style.padding = "10px";
            const legend = document.createElement("legend");
            legend.textContent = "Mover a:";
            directionFieldset.appendChild(legend);

            ['start', 'end'].forEach(dir => {
                const container = document.createElement("div");
                container.style.marginBottom = "5px";
                const radio = document.createElement("input");
                radio.type = "radio";
                radio.name = "editSwapDirection";
                radio.value = dir;
                radio.id = `editSwap_${dir}`;
                if (item.direction === dir) radio.checked = true;

                const label = document.createElement("label");
                label.htmlFor = `editSwap_${dir}`;
                label.textContent = ` ${dir === 'start' ? 'Al Inicio (Start ←A)' : 'Al Final (A→End)'}`;
                label.style.cursor = "pointer";

                container.appendChild(radio);
                container.appendChild(label);
                directionFieldset.appendChild(container);
            });
            modalContent.appendChild(directionFieldset);

            // Contenedor para los botones de acción
            const buttonContainer = document.createElement("div");
            buttonContainer.style.display = "flex";
            buttonContainer.style.justifyContent = "flex-end";
            buttonContainer.style.gap = "10px";
            buttonContainer.style.marginTop = "20px";

            // Botón Cancelar
            const cancelBtn = document.createElement("button");
            cancelBtn.textContent = "Cancelar";
            cancelBtn.style.padding = "8px 15px";
            cancelBtn.addEventListener("click", () => modalOverlay.remove());
            buttonContainer.appendChild(cancelBtn);

            // Botón Guardar
            const saveBtn = document.createElement("button");
            saveBtn.textContent = "Guardar Cambios";
            saveBtn.style.padding = "8px 15px";
            saveBtn.style.background = "#007bff";
            saveBtn.style.color = "white";
            saveBtn.style.border = "none";
            saveBtn.style.borderRadius = "4px";
            saveBtn.addEventListener("click", () => {
                const newWord = wordInput.value.trim();
                const newDirection = document.querySelector('input[name="editSwapDirection"]:checked').value;

                if (!newWord) {
                    alert("La palabra no puede estar vacía.");
                    return;
                }

                // Verificar si el nuevo nombre ya existe (excluyendo el item actual)
                if (newWord !== item.word && window.swapWords.some((sw, i) => i !== index && sw.word === newWord)) {
                    alert("Esa palabra ya existe en la lista.");
                    return;
                }

                // Actualizar el item en el array global
                window.swapWords[index].word = newWord;
                window.swapWords[index].direction = newDirection;

                saveSwapWordsToStorage();
                renderSwapList(document.getElementById("swapListUL"));
                if (typeof window.updateSwapWordsCountLabel === 'function') window.updateSwapWordsCountLabel();
                modalOverlay.remove();
            });
            buttonContainer.appendChild(saveBtn);

            modalContent.appendChild(buttonContainer);
            modalOverlay.appendChild(modalContent);
            document.body.appendChild(modalOverlay);
        }
        // ===================================================================
        // FIN: NUEVA FUNCIÓN HELPER
        // ===================================================================


        // --- Contenedor principal ---
        const title = document.createElement("h4");
        title.textContent = "Gestión de Reemplazos";
        title.style.fontSize = "15px";
        title.style.marginBottom = "10px";
        parentContainer.appendChild(title);
        // --- Dropdown de modo de reemplazo ---
        const modeSelector = document.createElement("select");
        modeSelector.id = "replacementModeSelector";
        modeSelector.style.marginBottom = "10px";
        modeSelector.style.marginTop = "5px";
        // Añadir opciones al selector
        const optionWords = document.createElement("option");
        optionWords.value = "words";
        optionWords.textContent = "Reemplazos de palabras";
        modeSelector.appendChild(optionWords);
        // Añadir opción para swap
        const optionSwap = document.createElement("option");
        optionSwap.value = "swapStart";
        optionSwap.textContent = "Palabras al inicio/final (swap)"; // Texto actualizado
        modeSelector.appendChild(optionSwap);
        parentContainer.appendChild(modeSelector);
        //Contenedor para reemplazos y controles
        const replacementsContainer = document.createElement("div");
        replacementsContainer.id = "replacementsContainer";
        // Sección para añadir nuevos reemplazos
        const addSection = document.createElement("div");
        addSection.style.display = "flex";
        addSection.style.gap = "8px";
        addSection.style.marginBottom = "12px";
        addSection.style.alignItems = "flex-end"; // Alinear inputs y botón
        // Contenedores para inputs de texto
        const fromInputContainer = document.createElement("div");
        fromInputContainer.style.flexGrow = "1";
        const fromLabel = document.createElement("label");
        fromLabel.textContent = "Texto Original:";
        fromLabel.style.display = "block";
        fromLabel.style.fontSize = "12px";
        fromLabel.style.marginBottom = "2px";
        // Input para el texto original
        const fromInput = document.createElement("input");
        fromInput.type = "text";
        fromInput.placeholder = "Ej: Urb.";
        fromInput.style.width = "95%"; // Para que quepa bien
        fromInput.style.padding = "6px";
        fromInput.style.border = "1px solid #ccc";
        // Añadir label e input al contenedor
        fromInputContainer.appendChild(fromLabel);
        fromInputContainer.appendChild(fromInput);
        addSection.appendChild(fromInputContainer);
        // Contenedor para el texto de reemplazo
        const toInputContainer = document.createElement("div");
        toInputContainer.style.flexGrow = "1";
        const toLabel = document.createElement("label");
        toLabel.textContent = "Texto de Reemplazo:";
        toLabel.style.display = "block";
        toLabel.style.fontSize = "12px";
        toLabel.style.marginBottom = "2px";
        // Input para el texto de reemplazo
        const toInput = document.createElement("input");
        toInput.type = "text";
        toInput.placeholder = "Ej: Urbanización";
        toInput.style.width = "95%";
        toInput.style.padding = "6px";
        toInput.style.border = "1px solid #ccc";
        toInputContainer.appendChild(toLabel);
        toInputContainer.appendChild(toInput);
        addSection.appendChild(toInputContainer);
        // Atributos para evitar corrección ortográfica
        fromInput.setAttribute('spellcheck', 'false');
        toInput.setAttribute('spellcheck', 'false');
        // Botón para añadir el reemplazo
        const addReplacementBtn = document.createElement("button");
        addReplacementBtn.textContent = "Añadir";
        addReplacementBtn.style.padding = "6px 10px";
        addReplacementBtn.style.cursor = "pointer";
        addReplacementBtn.style.height = "30px"; // Para alinear con los inputs
        addSection.appendChild(addReplacementBtn);
        // Elemento UL para la lista de reemplazos
        const listElement = document.createElement("ul");
        listElement.id = "replacementsListElementID"; // ID ÚNICO para esta lista
        listElement.style.maxHeight = "150px";
        listElement.style.overflowY = "auto";
        listElement.style.border = "1px solid #ddd";
        listElement.style.padding = "8px";
        listElement.style.margin = "0 0 10px 0";
        listElement.style.background = "#fff";
        listElement.style.listStyle = "none";
        // Event listener para el botón "Añadir"
        addReplacementBtn.addEventListener("click", () => {
            const fromValue = fromInput.value.trim();
            const toValue = toInput.value.trim();
            if (!fromValue) {
                alert("El campo 'Texto Original' es requerido.");
                return;
            }
            if (fromValue === toValue) {
                alert("El texto original y el de reemplazo no pueden ser iguales.");
                return;
            }
            // Permitir que el destino contenga al origen y viceversa (ej. "Av" → "Av.", "CED." → "CED").
            // La prevención de bucles y conflictos se maneja con las reglas de hoja (reversa)
            // y el motor de reemplazo.
            // Regla: no permitir crear reglas de usuario con orígenes fijos por defecto
            try {
                if ((window.fixedReplacementFroms && window.fixedReplacementFroms.has(fromValue)) ||
                    (window.fixedReplacementFromsNorm && window.fixedReplacementFromsNorm.has(plnNormalizeReplacementKey(fromValue))) ||
                    (window.fixedReplacementTargets && window.fixedReplacementTargets.has(fromValue)) ||
                    (window.fixedReplacementTargetsNorm && window.fixedReplacementTargetsNorm.has(plnNormalizeReplacementKey(fromValue)))) {
                    alert("Este origen está reservado por la lista de Wazeopedia(como origen o destino) y no puede ser agregado por el usuario.");
                    return;
                }
            } catch (_) { }
            if ((window.replacementSources && window.replacementSources[fromValue] === 'sheet')) {
                alert("No puedes modificar un reemplazo definido por la comunidad.");
                return;
            }
            // Bloqueo: no permitir reglas que contradicen una regla de hoja (reversa)
            try {
                const userFromLC = plnNormalizeReplacementKey(fromValue);
                const userToLC = plnNormalizeReplacementKey(toValue);
                const keys = Object.keys(replacementWords || {});
                for (const k of keys) {
                    if (window.replacementSources && window.replacementSources[k] === 'sheet') {
                        const sheetFromLC = plnNormalizeReplacementKey(k);
                        const sheetToLC = plnNormalizeReplacementKey(replacementWords[k] || '');
                        if (sheetFromLC === userToLC && sheetToLC === userFromLC) {
                            alert(`Esta regla contradice una regla bloqueada por la Comundiad de Editores: "${k}" → "${replacementWords[k]}".\nNo se puede crear "${fromValue}" → "${toValue}".`);
                            return;
                        }
                    }
                }
            } catch (_) { }

            // Tratar duplicados case-insensitive de la clave 'from'
            let existingKeySameCI = null;
            try {
                const lc = plnNormalizeReplacementKey(fromValue);
                existingKeySameCI = Object.keys(replacementWords || {}).find(k => plnNormalizeReplacementKey(k) === lc) || null;
                if (existingKeySameCI && existingKeySameCI !== fromValue) {
                    if (window.replacementSources && window.replacementSources[existingKeySameCI] === 'sheet') {
                        alert(`Ya existe una regla definida por la comunidad para "${existingKeySameCI}". No se puede crear otra con diferente capitalización.`);
                        return;
                    }
                    if (!confirm(`Ya existe una regla para "${existingKeySameCI}". ¿Deseas sobrescribirla con "${fromValue}" → "${toValue}"?`))
                        return;
                }
            } catch (_) { }

            if (replacementWords.hasOwnProperty(fromValue) && replacementWords[fromValue] !== toValue) {
                if (!confirm(`El reemplazo para "${fromValue}" ya existe ('${replacementWords[fromValue]}'). ¿Deseas sobrescribirlo con '${toValue}'?`))
                    return;
            }

            const keyToStore = existingKeySameCI || fromValue;
            if (existingKeySameCI && existingKeySameCI !== fromValue) {
                if (existingKeySameCI in replacementWords && existingKeySameCI !== keyToStore) {
                    delete replacementWords[existingKeySameCI];
                    if (window.replacementSources) delete window.replacementSources[existingKeySameCI];
                }
            }
            replacementWords[keyToStore] = toValue;
            if (!window.replacementSources) window.replacementSources = {};
            window.replacementSources[keyToStore] = 'user';
            fromInput.value = "";
            toInput.value = "";
            renderReplacementsList(listElement);
            saveReplacementWordsToStorage();
        });

        // Contador de Reemplazos (ARRIBA de los botones)
        const replacementsCountLabel = document.createElement("div");
        replacementsCountLabel.id = "replacementsCountLabel";
        replacementsCountLabel.style.marginBottom = "8px";
        replacementsCountLabel.style.marginTop = "10px";
        replacementsCountLabel.style.fontSize = "13px";
        replacementsCountLabel.style.color = "#666";
        replacementsCountLabel.style.fontWeight = "500";
        replacementsCountLabelElement = replacementsCountLabel;
        window.updateReplacementsCountLabel?.();
        replacementsContainer.appendChild(replacementsCountLabel);

        // Botones de Acción y Drop Area (usarán la lógica compartida)
        const actionButtonsContainer = document.createElement("div");
        actionButtonsContainer.style.display = "flex";
        actionButtonsContainer.style.gap = "8px";
        actionButtonsContainer.style.alignItems = "center"; // Alineación vertical
        actionButtonsContainer.style.flexWrap = "wrap"; // Permitir wrap
        actionButtonsContainer.style.marginBottom = "10px";

        // Botón para exportar solo reemplazos
        const clearButton = document.createElement("button");
        clearButton.textContent = "Limpiar Reemplazos";
        clearButton.title = "Limpiar solo la lista de reemplazos";
        clearButton.style.padding = "6px 10px";
        clearButton.addEventListener("click", async () => {
            if (!confirm("¿Estás seguro de que deseas eliminar TODOS los reemplazos definidos?")) {
                return;
            }

            const originalText = clearButton.textContent;
            clearButton.disabled = true;
            clearButton.textContent = "Limpiando...";

            try {
                replacementWords = {};
                window.replacementSources = {};
                saveReplacementWordsToStorage();
                renderReplacementsList(listElement);

                await loadReplacementsFromSheet(true);
                renderReplacementsList(listElement);
                showTemporaryMessage("Reemplazos por defecto recargados.", 3500, 'info');
            }
            catch (e) {
                console.error('[WME PLN] Error recargando reemplazos por defecto tras limpiar:', e);
                showTemporaryMessage("No se pudieron recargar los reemplazos por defecto.", 4000, 'error');
            }
            finally {
                clearButton.disabled = false;
                clearButton.textContent = originalText;
            }
        });
        actionButtonsContainer.appendChild(clearButton);
        // Botón para importar desde XML
        const dropArea = document.createElement("div");
        dropArea.textContent = "Arrastra aquí el archivo XML (contiene Excluidas y Reemplazos)";
        dropArea.style.border = "2px dashed #ccc";
        dropArea.style.borderRadius = "4px";
        dropArea.style.padding = "15px";
        dropArea.style.marginTop = "10px";
        dropArea.style.textAlign = "center";
        dropArea.style.background = "#f9f9f9";
        dropArea.style.color = "#555";
        // Añadir estilos para el drop area
        dropArea.addEventListener("dragover", (e) => {
            e.preventDefault();
            dropArea.style.background = "#e9e9e9";
        });
        // Cambiar el fondo al salir del área de arrastre
        dropArea.addEventListener("dragleave", () => { dropArea.style.background = "#f9f9f9"; });
        // Manejar el evento de drop
        dropArea.addEventListener("drop", (e) => {
            e.preventDefault();
            dropArea.style.background = "#f9f9f9";
            handleXmlFileDrop(e.dataTransfer.files[0]);
        });
        // --- Ensamblar en replacementsContainer ---
        replacementsContainer.appendChild(addSection);
        replacementsContainer.appendChild(listElement);
        replacementsContainer.appendChild(actionButtonsContainer);
        replacementsContainer.appendChild(dropArea);
        parentContainer.appendChild(replacementsContainer);


        // --- Contenedor para swapStart/frases al inicio ---
        const swapContainer = document.createElement("div");
        swapContainer.id = "swapContainer";
        swapContainer.style.display = "none";

        // === TÍTULO DINÁMICO CON CONTADOR ===
        const swapTitle = document.createElement("h4");
        swapTitle.id = "swapWordsCountLabel";
        swapTitle.style.fontSize = "14px";
        swapTitle.style.marginBottom = "8px";
        swapWordsCountLabelElement = swapTitle;
        if (typeof window.updateSwapWordsCountLabel === 'function') window.updateSwapWordsCountLabel();
        swapContainer.appendChild(swapTitle);

        const swapExplanationBox = document.createElement("div");
        swapExplanationBox.style.background = "#f4f8ff";
        swapExplanationBox.style.borderLeft = "4px solid #2d6df6";
        swapExplanationBox.style.padding = "10px";
        swapExplanationBox.style.margin = "10px 0";
        swapExplanationBox.style.fontSize = "13px";
        swapExplanationBox.style.lineHeight = "1.4";
        swapExplanationBox.innerHTML =
            "<strong>🔄 ¿Qué hace esta lista?</strong><br>" +
            "Las palabras aquí se moverán al inicio o al final del nombre.<br>" +
            "<em>Ej:</em> \"Las Palmas <b>Urbanización</b>\" → \"<b>Urbanización</b> Las Palmas\" (si se configura 'Al Inicio').";
        swapContainer.appendChild(swapExplanationBox);

        // =======================================================
        // INICIO DE LA MODIFICACIÓN DEL LAYOUT
        // =======================================================

        // Contenedor principal para los controles, ahora apilado verticalmente
        const swapInputContainer = document.createElement("div");
        swapInputContainer.style.display = "flex";
        swapInputContainer.style.flexDirection = "column"; // Apilado vertical
        swapInputContainer.style.gap = "8px";
        swapInputContainer.style.marginBottom = "8px";

        // Fila 1: Input de la palabra
        const swapInputDiv = document.createElement("div");
        const swapInputLabel = document.createElement("label");
        swapInputLabel.textContent = "Palabra a agregar:";
        swapInputLabel.style.fontSize = "12px";
        swapInputLabel.style.display = "block";
        swapInputLabel.style.marginBottom = "2px";
        const swapInput = document.createElement("input");
        swapInput.type = "text";
        swapInput.placeholder = "Ej: Urbanización";
        swapInput.style.width = "calc(100% - 12px)"; // Ancho completo
        swapInput.style.padding = "6px";
        swapInput.setAttribute('spellcheck', 'false');
        swapInputDiv.appendChild(swapInputLabel);
        swapInputDiv.appendChild(swapInput);

        // Fila 2: Controles de dirección y botón de añadir
        const controlsRow = document.createElement("div");
        controlsRow.style.display = "flex";
        controlsRow.style.alignItems = "center";
        controlsRow.style.gap = "10px";

        // Contenedor para los radio buttons
        const directionContainer = document.createElement("div");
        directionContainer.style.display = "flex";
        directionContainer.style.gap = "15px";
        directionContainer.style.padding = "5px 10px";
        directionContainer.style.border = "1px solid #ccc";
        directionContainer.style.borderRadius = "4px";

        ['start', 'end'].forEach(dir => {
            const optionContainer = document.createElement('div');
            optionContainer.style.display = 'flex';
            optionContainer.style.alignItems = 'center';

            const radio = document.createElement("input");
            radio.type = "radio";
            radio.name = "swapDirection";
            radio.value = dir;
            radio.id = `swap_${dir}`;
            if (dir === 'start') radio.checked = true;
            radio.style.marginRight = "4px";

            const label = document.createElement("label");
            label.htmlFor = `swap_${dir}`;
            label.textContent = dir === 'start' ? 'Mover a Inicio' : 'Mover al Final'; // ETIQUETAS ACTUALIZADAS
            label.style.fontSize = "13px";
            label.style.cursor = "pointer";

            optionContainer.appendChild(radio);
            optionContainer.appendChild(label);
            directionContainer.appendChild(optionContainer);
        });

        // Botón para añadir
        const swapBtn = document.createElement("button");
        swapBtn.textContent = "Añadir";
        swapBtn.style.padding = "6px 12px";
        swapBtn.style.height = "32px";

        // Ensamblar la fila 2
        controlsRow.appendChild(directionContainer);
        controlsRow.appendChild(swapBtn);

        // Ensamblar el contenedor principal
        swapInputContainer.appendChild(swapInputDiv);
        swapInputContainer.appendChild(controlsRow);

        swapContainer.appendChild(swapInputContainer); // Añadir el contenedor principal al panel

        // =======================================================
        // FIN DE LA MODIFICACIÓN DEL LAYOUT
        // =======================================================

        // === EVENT LISTENER PARA EL BOTÓN AÑADIR (con protección de hoja) ===
        swapBtn.addEventListener("click", () => {
            const val = swapInput.value.trim();
            const direction = document.querySelector('input[name="swapDirection"]:checked').value;

            if (!val || /^[^a-zA-Z0-9]+$/.test(val)) {
                alert("No se permiten caracteres especiales solos o palabras vacías.");
                return;
            }
            // Bloquear si la palabra está protegida por la hoja Swap
            try {
                const norm = (val || '').normalize('NFD').replace(/[\u0300-\u036f]/g, '').replace(/\s+/g, ' ').trim().toLowerCase();
                if (window.fixedSwapNormSet && window.fixedSwapNormSet.has(norm)) {
                    alert('Esta palabra está protegida por la lista de la Comundiad de Editores (Swap) y no puede agregarse.');
                    return;
                }
            } catch (_) { }

            if (window.swapWords.some(item => item.word === val)) {
                alert("Esa palabra ya existe en la lista.");
                return;
            }

            window.swapWords.push({ word: val, direction: direction });
            saveSwapWordsToStorage();
            if (typeof window.updateSwapWordsCountLabel === 'function') window.updateSwapWordsCountLabel();
            swapInput.value = "";
            renderSwapList(document.getElementById("swapListUL"));
        });

        // === CAMPO DE BÚSQUEDA ===
        const searchSwapInput = document.createElement("input");
        searchSwapInput.type = "text";
        searchSwapInput.placeholder = "Buscar palabra...";
        searchSwapInput.id = "searchSwapInput";
        searchSwapInput.style.width = "calc(100% - 12px)";
        searchSwapInput.style.padding = "6px";
        searchSwapInput.style.marginBottom = "8px";
        searchSwapInput.style.border = "1px solid #ccc";

        searchSwapInput.addEventListener("input", () => renderSwapList());
        swapContainer.appendChild(searchSwapInput);

        // === LISTA UL ===
        const swapList = document.createElement("ul");
        swapList.id = "swapListUL";
        swapList.style.maxHeight = "150px";
        swapList.style.overflowY = "auto";
        swapList.style.border = "1px solid #ddd";
        swapList.style.padding = "5px";
        swapList.style.margin = "0";
        swapList.style.background = "#fff";
        swapList.style.listStyle = "none";
        swapContainer.appendChild(swapList);

        // === BOTONES DE ACCIÓN ===
        const swapActionButtonsContainer = document.createElement("div");
        swapActionButtonsContainer.style.display = "flex";
        swapActionButtonsContainer.style.gap = "8px";
        swapActionButtonsContainer.style.alignItems = "center";
        swapActionButtonsContainer.style.flexWrap = "wrap";
        swapActionButtonsContainer.style.marginTop = "10px";

        const clearSwapBtn = document.createElement("button");
        clearSwapBtn.textContent = "Limpiar Swap";
        clearSwapBtn.title = "Limpiar lista de palabras de intercambio";
        clearSwapBtn.style.padding = "6px 10px";
        clearSwapBtn.style.cursor = "pointer";
        clearSwapBtn.addEventListener("click", () => {
            if (confirm("¿Estás seguro de que deseas eliminar TODAS las palabras de intercambio?")) {
                window.swapWords = [];
                renderSwapList();
                saveSwapWordsToStorage();
                if (typeof window.updateSwapWordsCountLabel === 'function') window.updateSwapWordsCountLabel();
            }
        });
        swapActionButtonsContainer.appendChild(clearSwapBtn);

        swapContainer.appendChild(swapActionButtonsContainer);

        // Render inicial
        renderSwapList();

        parentContainer.appendChild(swapContainer);

        // === LÓGICA DE RENDERIZADO DE LA LISTA (ACTUALIZADA) ===
        function renderSwapList() {
            const searchInput = document.getElementById("searchSwapInput");
            const swapList = swapContainer.querySelector("ul") || (() => {
                const ul = document.createElement("ul");
                ul.id = "swapList";
                ul.style.maxHeight = "120px";
                ul.style.overflowY = "auto";
                ul.style.border = "1px solid #ddd";
                ul.style.padding = "8px";
                ul.style.margin = "0";
                ul.style.background = "#fff";
                ul.style.listStyle = "none";
                swapContainer.appendChild(ul);
                return ul;
            })();

            swapList.innerHTML = "";

            if (!window.swapWords || window.swapWords.length === 0) {
                const li = document.createElement("li");
                li.textContent = "No hay palabras de intercambio definidas.";
                li.style.textAlign = "center";
                li.style.color = "#777";
                swapList.appendChild(li);
                return;
            }

            const searchTerm = searchInput ? searchInput.value.trim().toLowerCase() : "";
            const filteredSwapWords = window.swapWords.filter(item => typeof item.word === 'string' && item.word.toLowerCase().includes(searchTerm));

            filteredSwapWords.forEach((item, index) => {
                const li = document.createElement("li");
                li.style.display = "flex";
                li.style.justifyContent = "space-between";
                li.style.alignItems = "center";
                li.style.padding = "4px 2px";
                li.style.borderBottom = "1px solid #f0f0f0";

                const wordSpan = document.createElement("span");
                const directionIcon = item.direction === "start" ? "←" : "→";
                const directionText = item.direction === "start" ? "Al Inicio" : "Al Final";
                let isLocked = false;
                try {
                    const norm = (item.word || '').normalize('NFD').replace(/[\u0300-\u036f]/g, '').replace(/\s+/g, ' ').trim().toLowerCase();
                    isLocked = !!(window.fixedSwapNormSet && window.fixedSwapNormSet.has(norm));
                } catch (_) { isLocked = false; }
                wordSpan.innerHTML = `<b>${item.word}</b> <small style="color: #666;">(${directionIcon} ${directionText})</small>${isLocked ? ' 🔒' : ''}`;

                const btnContainer = document.createElement("span");
                btnContainer.style.display = "flex";
                btnContainer.style.gap = "4px";

                // Botón Editar
                const editBtn = document.createElement("button");
                editBtn.innerHTML = "✏️";
                editBtn.title = "Editar";
                editBtn.style.border = "none";
                editBtn.style.background = "transparent";
                editBtn.style.cursor = "pointer";
                editBtn.addEventListener("click", () => {
                    const originalIndex = window.swapWords.findIndex(sw => sw.word === item.word);
                    if (originalIndex > -1) {
                        openSwapWordEditor(window.swapWords[originalIndex], originalIndex);
                    }
                });
                if (isLocked) { editBtn.disabled = true; editBtn.style.opacity = '0.4'; editBtn.style.cursor = 'not-allowed'; }

                // Botón Eliminar
                const deleteBtn = document.createElement("button");
                deleteBtn.innerHTML = "🗑️";
                deleteBtn.title = "Eliminar";
                deleteBtn.style.border = "none";
                deleteBtn.style.background = "transparent";
                deleteBtn.style.cursor = "pointer";
                deleteBtn.addEventListener("click", () => {
                    if (confirm(`¿Eliminar la palabra swap '${item.word}'?`)) {
                        const indexToDelete = window.swapWords.findIndex(sw => sw.word === item.word);
                        if (indexToDelete > -1) {
                            window.swapWords.splice(indexToDelete, 1);
                            saveSwapWordsToStorage();
                            renderSwapList();
                        }
                    }
                });
                if (isLocked) { deleteBtn.disabled = true; deleteBtn.style.opacity = '0.4'; deleteBtn.style.cursor = 'not-allowed'; }

                btnContainer.appendChild(editBtn);
                btnContainer.appendChild(deleteBtn);

                li.appendChild(wordSpan);
                li.appendChild(btnContainer);
                swapList.appendChild(li);
            });
        }
        // Render inicial y listener del selector
        plnCanonicalizeReplacementsBySheet();
        renderReplacementsList(listElement);
        renderSwapList();
        modeSelector.addEventListener("change", () => {
            replacementsContainer.style.display = modeSelector.value === "words" ? "block" : "none";
            swapContainer.style.display = modeSelector.value === "swapStart" ? "block" : "none";
        });
    }
    // Crea el gestor de reemplazos


    // Renderiza la lista de palabras excluidas
    function renderExcludedWordsList(ulElement, filter = "") {
        if (!ulElement) {
            return;
        }
        // Asegurarse de que ulElement es válido
        const currentFilter = filter.toLowerCase();
        ulElement.innerHTML = "";
        // Asegurarse de que excludedWords es un Set
        const wordsToRender = Array.from(excludedWords).filter(word => word.toLowerCase().includes(currentFilter))
            .sort((a, b) => a.toLowerCase().localeCompare(b.toLowerCase()));
        // Si no hay palabras para renderizar, mostrar mensaje
        if (wordsToRender.length === 0) {
            const li = document.createElement("li");
            li.textContent = "No hay palabras excluidas.";
            li.style.textAlign = "center";
            li.style.color = "#777";
            ulElement.appendChild(li);
            return;
        }

        // ✨ PAGINACIÓN: Calcular índices de página
        const totalWords = wordsToRender.length;
        const totalPages = Math.ceil(totalWords / WORDS_PER_PAGE);

        // Asegurar que la página actual está en rango válido
        if (excludedWordsCurrentPage > totalPages) {
            excludedWordsCurrentPage = totalPages || 1;
        }

        const startIdx = (excludedWordsCurrentPage - 1) * WORDS_PER_PAGE;
        const endIdx = Math.min(startIdx + WORDS_PER_PAGE, totalWords);
        const pageWords = wordsToRender.slice(startIdx, endIdx);

        // Agregar indicador de resultados
        const infoLi = document.createElement("li");
        infoLi.style.textAlign = "center";
        infoLi.style.padding = "8px";
        infoLi.style.backgroundColor = "#f5f5f5";
        infoLi.style.fontWeight = "bold";
        infoLi.style.borderBottom = "2px solid #007bff";
        infoLi.textContent = `Mostrando ${startIdx + 1}-${endIdx} de ${totalWords} palabra${totalWords !== 1 ? 's' : ''}`;
        ulElement.appendChild(infoLi);

        // Renderizar solo palabras de la página actual
        pageWords.forEach(word => {
            const li = document.createElement("li");
            li.style.display = "flex"; // Agregado para alinear texto y botones
            li.style.justifyContent = "space-between"; // Agregado para espacio entre texto y botones
            li.style.alignItems = "center"; // Agregado para centrado vertical
            li.style.padding = "5px";
            li.style.borderBottom = "1px solid #ddd";
            // Span para el texto de la palabra
            const wordSpan = document.createElement("span");
            wordSpan.textContent = word;
            wordSpan.style.flexGrow = "1"; // Permite que el texto ocupe el espacio disponible
            wordSpan.style.marginRight = "10px"; // Espacio entre el texto y los botones
            li.appendChild(wordSpan);
            //Bloque para los botones de edición y eliminación ---
            const btnContainer = document.createElement("span");
            btnContainer.style.display = "flex";
            btnContainer.style.gap = "8px"; // Espacio entre los botones

            // Botón de edición
            const editBtn = document.createElement("button");
            editBtn.innerHTML = "✏️"; // Icono de lápiz
            editBtn.title = "Editar";
            editBtn.style.border = "none";
            editBtn.style.background = "transparent";
            editBtn.style.cursor = "pointer";
            editBtn.style.padding = "2px";
            editBtn.style.fontSize = "14px";
            editBtn.addEventListener("click", async () => {
                const newWord = await showCustomPrompt("Editar palabra:", "✏️ Editar Palabra Excluida", word, "📝");
                if (newWord !== null && newWord.trim() !== word) {
                    const validation = isValidExcludedWord(newWord.trim());
                    if (!validation.valid) {
                        alert(validation.msg);
                        return;
                    }

                    // Eliminar la palabra antigua del Set y Map
                    excludedWords.delete(word);
                    const oldFirstChar = word.charAt(0).toLowerCase();
                    if (excludedWordsMap.has(oldFirstChar)) {
                        excludedWordsMap.get(oldFirstChar).delete(word);
                        if (excludedWordsMap.get(oldFirstChar).size === 0) {
                            excludedWordsMap.delete(oldFirstChar);
                        }
                    }

                    // Añadir la nueva palabra al Set y Map
                    const trimmedNewWord = newWord.trim();
                    excludedWords.add(trimmedNewWord);
                    const newFirstChar = trimmedNewWord.charAt(0).toLowerCase();
                    if (!excludedWordsMap.has(newFirstChar)) {
                        excludedWordsMap.set(newFirstChar, new Set());
                    }
                    excludedWordsMap.get(newFirstChar).add(trimmedNewWord);

                    // ✨ Limpiar cache porque las palabras excluidas cambiaron
                    clearWordLookupCache();

                    renderExcludedWordsList(ulElement, currentFilter);
                    saveExcludedWordsToLocalStorage();
                }
            });
            btnContainer.appendChild(editBtn);

            // Botón de eliminación
            const deleteBtn = document.createElement("button");
            deleteBtn.innerHTML = "🗑️"; // Icono de bote de basura
            deleteBtn.title = "Eliminar";
            deleteBtn.style.border = "none";
            deleteBtn.style.background = "transparent";
            deleteBtn.style.cursor = "pointer";
            deleteBtn.style.padding = "2px";
            deleteBtn.style.fontSize = "14px";
            deleteBtn.addEventListener("click", () => {
                if (confirm(`¿Eliminar la palabra '${word}' de la lista de especiales?`)) {
                    excludedWords.delete(word);
                    const firstChar = word.charAt(0).toLowerCase();
                    if (excludedWordsMap.has(firstChar)) {
                        excludedWordsMap.get(firstChar).delete(word);
                        if (excludedWordsMap.get(firstChar).size === 0) {
                            excludedWordsMap.delete(firstChar);
                        }
                    }

                    // ✨ Limpiar cache porque las palabras excluidas cambiaron
                    clearWordLookupCache();

                    renderExcludedWordsList(ulElement, currentFilter);
                    saveExcludedWordsToLocalStorage();
                }
            });
            btnContainer.appendChild(deleteBtn);
            //--- Fin del contenedor de botones

            li.appendChild(btnContainer);
            ulElement.appendChild(li);
        });

        // ✨ PAGINACIÓN: Agregar controles de paginación
        if (totalPages > 1) {
            const paginationLi = document.createElement("li");
            paginationLi.style.textAlign = "center";
            paginationLi.style.padding = "10px";
            paginationLi.style.backgroundColor = "#f9f9f9";
            paginationLi.style.borderTop = "2px solid #007bff";
            paginationLi.style.display = "flex";
            paginationLi.style.justifyContent = "center";
            paginationLi.style.alignItems = "center";
            paginationLi.style.gap = "10px";

            // Botón Anterior
            const prevBtn = document.createElement("button");
            prevBtn.textContent = "◀ Anterior";
            prevBtn.disabled = excludedWordsCurrentPage === 1;
            prevBtn.style.padding = "6px 12px";
            prevBtn.style.border = "1px solid #007bff";
            prevBtn.style.borderRadius = "4px";
            prevBtn.style.backgroundColor = prevBtn.disabled ? "#e0e0e0" : "#007bff";
            prevBtn.style.color = prevBtn.disabled ? "#999" : "white";
            prevBtn.style.cursor = prevBtn.disabled ? "not-allowed" : "pointer";
            prevBtn.style.fontWeight = "bold";
            prevBtn.addEventListener("click", () => {
                if (excludedWordsCurrentPage > 1) {
                    excludedWordsCurrentPage--;
                    renderExcludedWordsList(ulElement, currentFilter);
                }
            });

            // Indicador de página
            const pageInfo = document.createElement("span");
            pageInfo.textContent = `Página ${excludedWordsCurrentPage} de ${totalPages}`;
            pageInfo.style.fontWeight = "bold";
            pageInfo.style.padding = "0 10px";

            // Botón Siguiente
            const nextBtn = document.createElement("button");
            nextBtn.textContent = "Siguiente ▶";
            nextBtn.disabled = excludedWordsCurrentPage === totalPages;
            nextBtn.style.padding = "6px 12px";
            nextBtn.style.border = "1px solid #007bff";
            nextBtn.style.borderRadius = "4px";
            nextBtn.style.backgroundColor = nextBtn.disabled ? "#e0e0e0" : "#007bff";
            nextBtn.style.color = nextBtn.disabled ? "#999" : "white";
            nextBtn.style.cursor = nextBtn.disabled ? "not-allowed" : "pointer";
            nextBtn.style.fontWeight = "bold";
            nextBtn.addEventListener("click", () => {
                if (excludedWordsCurrentPage < totalPages) {
                    excludedWordsCurrentPage++;
                    renderExcludedWordsList(ulElement, currentFilter);
                }
            });

            paginationLi.appendChild(prevBtn);
            paginationLi.appendChild(pageInfo);
            paginationLi.appendChild(nextBtn);
            ulElement.appendChild(paginationLi);
        }
    }// renderExcludedWordsList



    // Función para renderizar la lista de lugares excluidos
    function renderExcludedPlacesList(ulElement, filter = "") {
        if (!ulElement) return;
        ulElement.innerHTML = "";
        const lowerFilter = filter.toLowerCase();

        // Ahora excludedPlaces es un Map<ID, Name>.
        const placesToRender = Array.from(excludedPlaces.entries()).filter(([placeId, placeName]) =>
            // Filtra por ID o por el nombre guardado
            placeId.toLowerCase().includes(lowerFilter) ||
            (placeName && placeName.toLowerCase().includes(lowerFilter))
        ).sort(([idA, nameA], [idB, nameB]) => {
            // Ordena alfabéticamente por el nombre guardado
            const safeNameA = nameA || '';
            const safeNameB = nameB || '';
            return safeNameA.toLowerCase().localeCompare(safeNameB.toLowerCase());
        });

        if (placesToRender.length === 0) {
            const li = document.createElement("li");
            li.textContent = "No hay lugares excluidos.";
            li.style.textAlign = "center";
            li.style.color = "#777";
            li.style.padding = "5px";
            ulElement.appendChild(li);
            return;
        }

        placesToRender.forEach(([placeId, placeNameSaved]) => { // Ahora recibimos [ID, NombreGuardado]
            const li = document.createElement("li");
            li.style.display = "flex";
            li.style.justifyContent = "space-between";
            li.style.alignItems = "center";
            li.style.padding = "4px 2px";
            li.style.borderBottom = "1px solid #f0f0f0";

            // Muestra el nombre guardado, con un fallback si el nombre guardado está vacío.
            const displayName = placeNameSaved || `ID: ${placeId}`;
            const linkSpan = document.createElement("span");
            linkSpan.style.flexGrow = "1";
            linkSpan.style.marginRight = "10px";
            const link = document.createElement("a");
            link.href = "#";
            link.textContent = displayName; // Muestra el nombre guardado
            link.title = `Abrir lugar en WME (ID: ${placeId})`; // El tooltip sigue mostrando el ID
            link.addEventListener("click", (e) => {
                e.preventDefault();
                // Intenta obtener el lugar del modelo para seleccionarlo y centrarlo
                // Usamos W.model como fallback si wmeSDK.DataModel.Venues.getById no es eficiente aquí o no está diseñado para esta interacción
                const venueObj = W.model.venues.getObjectById(placeId); // <---
                const venueSDKForUse = venueSDKForRender; // Objeto del SDK que pasamos desde processNextPlace

                if (venueObj) {
                    if (W.map && typeof W.map.setCenter === 'function' && venueObj.getOLGeometry && venueObj.getOLGeometry().getCentroid) {
                        W.map.setCenter(venueObj.getOLGeometry().getCentroid(), null, false, 0); // <--- REINTRODUCIMOS W.map.setCenter
                    }
                    if (W.selectionManager && typeof W.selectionManager.select === 'function') {
                        W.selectionManager.select(venueObj); // <--- REINTRODUCIMOS W.selectionManager.select
                    } else if (W.selectionManager && typeof W.selectionManager.setSelectedModels === 'function') {
                        W.selectionManager.setSelectedModels([venueObj]); // Fallback para versiones antiguas
                    }
                }
                else {
                    // Si el lugar no está en el modelo (fuera de vista), avisa y ofrece abrir en nueva pestaña.
                    const confirmOpen = confirm(`Lugar '${displayName}' (ID: ${placeId}) no encontrado en el modelo actual. ¿Deseas abrirlo en una nueva pestaña del editor?`);
                    if (confirmOpen) {
                        const wmeUrl = `https://www.waze.com/editor?env=row&venueId=${placeId}`;
                        window.open(wmeUrl, '_blank');
                    }
                }
            });
            linkSpan.appendChild(link);
            li.appendChild(linkSpan);

            // Botón para eliminar el lugar de la lista de excluidos.
            const deleteBtn = document.createElement("button");
            deleteBtn.innerHTML = "🗑️";
            deleteBtn.title = "Eliminar lugar de la lista de excluidos";
            deleteBtn.style.border = "none";
            deleteBtn.style.background = "transparent";
            deleteBtn.style.cursor = "pointer";
            deleteBtn.style.padding = "2px";
            deleteBtn.style.fontSize = "14px";
            deleteBtn.addEventListener("click", () => {
                // ************************************************************
                // INICIO DE LA MODIFICACIÓN: Modal de confirmación "bonito"
                // ************************************************************
                const confirmModal = document.createElement("div");
                confirmModal.style.position = "fixed";
                confirmModal.style.top = "50%";
                confirmModal.style.left = "50%";
                confirmModal.style.transform = "translate(-50%, -50%)";
                confirmModal.style.background = "#fff";
                confirmModal.style.border = "1px solid #aad";
                confirmModal.style.padding = "28px 32px 20px 32px";
                confirmModal.style.zIndex = "20000"; // Z-INDEX ALTO
                confirmModal.style.boxShadow = "0 4px 24px rgba(0,0,0,0.18)";
                confirmModal.style.fontFamily = "sans-serif";
                confirmModal.style.borderRadius = "10px";
                confirmModal.style.textAlign = "center";
                confirmModal.style.minWidth = "340px";

                // Ícono visual
                const iconElement = document.createElement("div");
                iconElement.innerHTML = "⚠️"; // Ícono de advertencia
                iconElement.style.fontSize = "38px";
                iconElement.style.marginBottom = "10px";
                confirmModal.appendChild(iconElement);

                // Mensaje principal
                const messageTitle = document.createElement("div");
                messageTitle.innerHTML = `<b>¿Eliminar de excluidos "${placeNameSaved}"?</b>`;
                messageTitle.style.fontSize = "20px";
                messageTitle.style.marginBottom = "8px";
                confirmModal.appendChild(messageTitle);

                // Mensaje explicativo
                const explanationDiv = document.createElement("div");
                explanationDiv.textContent = `Este lugar volverá a aparecer en futuras búsquedas del normalizador.`;
                explanationDiv.style.fontSize = "15px";
                explanationDiv.style.color = "#555";
                explanationDiv.style.marginBottom = "18px";
                confirmModal.appendChild(explanationDiv);

                // Botones de confirmación
                const buttonWrapper = document.createElement("div");
                buttonWrapper.style.display = "flex";
                buttonWrapper.style.justifyContent = "center";
                buttonWrapper.style.gap = "18px";

                // Botón Cancelar
                const cancelBtn = document.createElement("button");
                cancelBtn.textContent = "Cancelar";
                cancelBtn.style.padding = "7px 18px";
                cancelBtn.style.background = "#eee";
                cancelBtn.style.border = "none";
                cancelBtn.style.borderRadius = "4px";
                cancelBtn.style.cursor = "pointer";
                cancelBtn.addEventListener("click", () => confirmModal.remove());

                // Botón Confirmar Eliminación
                const confirmDeleteBtn = document.createElement("button");
                confirmDeleteBtn.textContent = "Eliminar";
                confirmDeleteBtn.style.padding = "7px 18px";
                confirmDeleteBtn.style.background = "#d9534f"; // Rojo
                confirmDeleteBtn.style.color = "#fff";
                confirmDeleteBtn.style.border = "none";
                confirmDeleteBtn.style.borderRadius = "4px";
                confirmDeleteBtn.style.cursor = "pointer";
                confirmDeleteBtn.style.fontWeight = "bold";

                confirmDeleteBtn.addEventListener("click", () => {
                    // Aquí va la lógica que antes estaba directamente en el if(confirm)
                    excludedPlaces.delete(placeId); // Sigue eliminando por ID
                    renderExcludedPlacesList(ulElement, filter); // Vuelve a renderizar la lista después de eliminar.
                    saveExcludedPlacesToLocalStorage(); // Guarda los cambios en localStorage.
                    showTemporaryMessage("Lugar eliminado de la lista de excluidos.", 3000, 'success');
                    confirmModal.remove(); // Cerrar el modal después de la acción
                });

                buttonWrapper.appendChild(cancelBtn);
                buttonWrapper.appendChild(confirmDeleteBtn);
                confirmModal.appendChild(buttonWrapper);
                document.body.appendChild(confirmModal); // Añadir el modal al body
                // --- ✅ Lógica de posicionamiento dinámico para el popup de eliminar ✅ ---
                const triggerButton = event.currentTarget;
                if (triggerButton) {
                    const modalRect = confirmModal.getBoundingClientRect();
                    const triggerRect = triggerButton.getBoundingClientRect();
                    let newLeft = triggerRect.right + 5; // 5px a la derecha del botón
                    let newTop = triggerRect.top;

                    // Ajustar si se sale de la pantalla
                    if ((newLeft + modalRect.width) > window.innerWidth) {
                        newLeft = triggerRect.left - modalRect.width - 5;
                    }
                    if (newTop < 0) newTop = 5;

                    confirmModal.style.left = `${newLeft}px`;
                    confirmModal.style.top = `${newTop}px`;
                    confirmModal.style.transform = 'none'; // Anular el centrado
                }
                else {
                    // Fallback al centrado original
                    confirmModal.style.left = "50%";
                    confirmModal.style.top = "50%";
                    confirmModal.style.transform = "translate(-50%, -50%)";
                }
            });
            li.appendChild(deleteBtn);
            ulElement.appendChild(li);
        });
    }// renderExcludedPlacesList

    // Crea un dropdown para seleccionar categorías recomendadas
    function createRecommendedCategoryDropdown(placeId, currentCategoryKey, dynamicCategorySuggestions) {
        const wrapperDiv = document.createElement("div");
        wrapperDiv.style.position = "relative";
        wrapperDiv.style.width = "100%";
        wrapperDiv.style.minWidth = "150px";
        wrapperDiv.style.display = "flex";
        wrapperDiv.style.flexDirection = "column";
        // Parte de sugerencias dinámicas existentes
        const suggestionsWrapper = document.createElement("div"); // Contenedor para sugerencias
        suggestionsWrapper.style.display = "flex";
        suggestionsWrapper.style.flexDirection = "column";
        suggestionsWrapper.style.alignItems = "flex-start";
        suggestionsWrapper.style.gap = "4px";
        // Filtrar y ordenar las sugerencias dinámicas para la presentación
        const filteredSuggestions = dynamicCategorySuggestions.filter(suggestion => suggestion.categoryKey.toUpperCase() !== currentCategoryKey.toUpperCase());
        if (filteredSuggestions.length > 0) { // Solo si hay sugerencias diferentes a la actual
            filteredSuggestions.forEach(suggestion => {
                const suggestionEntry = document.createElement("div");
                suggestionEntry.style.display = "flex";
                suggestionEntry.style.alignItems = "center";
                suggestionEntry.style.gap = "4px";
                suggestionEntry.style.padding = "2px 4px";
                suggestionEntry.style.border = "1px solid #dcdcdc";
                suggestionEntry.style.borderRadius = "3px";
                suggestionEntry.style.backgroundColor = "#eaf7ff"; // Un color distinto para sugerencias
                suggestionEntry.style.cursor = "pointer";
                suggestionEntry.title = `Sugerencia: ${getCategoryDetails(suggestion.categoryKey).description}`;
                //Añadir icono y descripción de la categoría
                const suggestedIconSpan = document.createElement("span");// Icono de la sugerencia
                suggestedIconSpan.textContent = suggestion.icon;
                suggestedIconSpan.style.fontSize = "16px";
                suggestionEntry.appendChild(suggestedIconSpan);
                // Añadir descripción de la categoría
                const suggestedDescSpan = document.createElement("span");
                suggestedDescSpan.textContent = getCategoryDetails(suggestion.categoryKey).description;
                suggestionEntry.appendChild(suggestedDescSpan);
                suggestionEntry.addEventListener("click", async function handler() { // Cambiado a función con nombre 'handler'
                    const placeToUpdate = W.model.venues.getObjectById(placeId);
                    if (!placeToUpdate) {
                        console.error("[WME_PLN] Lugar no encontrado para actualizar categoría.");
                        return;
                    }
                    try {
                        const UpdateObject = require("Waze/Action/UpdateObject");
                        const action = new UpdateObject(placeToUpdate, { categories: [suggestion.categoryKey] });
                        W.model.actionManager.add(action);
                        // Obtener la celda de la categoría original y aplicar un estilo de opacidad
                        const row = document.querySelector(`tr[data-place-id="${placeId}"]`); // Obtener la fila
                        row.dataset.categoryChanged = 'true'; // Marcar fila como modificada
                        // Habilitar el botón de aplicar sugerencia
                        const applyButton = row.querySelector('button[title="Aplicar sugerencia"]');
                        if (applyButton) {
                            applyButton.disabled = false;
                            applyButton.style.opacity = "1";
                        }
                        //Actualizar visualmente la celda de Categoría Actual en la tabla
                        updateCategoryDisplayInTable(placeId, suggestion.categoryKey);

                        // ✨ REGENERAR BOTONES DE PREFIJOS (28-nov-2024)
                        // Cuando cambia la categoría, actualizar los botones de palabras clave
                        if (row) {
                            // Buscar la celda de sugerencias que tiene la función guardada
                            const suggestionCells = row.querySelectorAll('td');
                            for (const cell of suggestionCells) {
                                if (cell._generatePrefixButtons && typeof cell._generatePrefixButtons === 'function') {
                                    cell._generatePrefixButtons(suggestion.categoryKey);
                                    break;
                                }
                            }
                        }

                        // Asegurarse de que la fila existe antes de intentar acceder a sus celdas
                        if (row) {
                            const originalCategoryCell = row.querySelector('td:nth-child(10)'); // La décima columna es "Categoría"
                            if (originalCategoryCell) {
                                originalCategoryCell.style.opacity = '0.5'; // Atenuar la celda completa
                                originalCategoryCell.title += ' (Modificada)'; // Opcional, añadir un tooltip
                            }
                        }
                        // : Mostrar chulito verde en la sugerencia misma
                        const successIcon = document.createElement("span");
                        successIcon.textContent = " ✅";
                        successIcon.style.marginLeft = "5px";
                        suggestionEntry.appendChild(successIcon); // Añadir el chulito a la entrada de la sugerencia
                        suggestionEntry.style.cursor = "default"; // Deshabilitar clic posterior

                        suggestionEntry.removeEventListener("click", handler); // Deshabilita el listener una vez que se ha hecho clic
                        suggestionEntry.style.opacity = "0.7"; // Opcional: Atenúa la sugerencia para indicar que ya se usó

                        optionsListDiv.style.display = "none"; // Ocultar lista
                        searchInput.blur(); // Quitar el foco
                        // : Eliminar la selección temporal para la categoría, ya se guardó
                        tempSelectedCategories.delete(placeId); // Si esta categoría se guardó directamente
                    }
                    catch (e) {
                        //console.error("[WME_PLN] Error al actualizar la categoría desde sugerencia:", e);
                        alert("Error al actualizar la categoría: " + e.message); // Mantener alerta para errores
                    }
                });
                suggestionsWrapper.appendChild(suggestionEntry);
            });
            wrapperDiv.appendChild(suggestionsWrapper); // Añadir contenedor de sugerencias
        }// createRecommendedCategoryDropdown
        //Fin de parte de sugerencias dinámicas
        // Input para buscar
        const searchInput = document.createElement("input");
        searchInput.type = "text";
        searchInput.placeholder = "Buscar o Seleccionar Categoría";// Placeholder más descriptivo
        searchInput.style.width = "calc(100% - 10px)";
        searchInput.style.padding = "5px";
        searchInput.style.marginTop = "5px"; //  Espacio después de sugerencias
        searchInput.style.marginBottom = "5px";
        searchInput.style.border = "1px solid #ccc";
        searchInput.style.borderRadius = "3px";
        searchInput.setAttribute('spellcheck', 'false');// Evitar corrección ortográfica
        searchInput.readOnly = false;// Permitir escribir pero no editar directamente
        searchInput.style.cursor = 'auto';// Permitir escribir pero no editar directamente
        searchInput.style.opacity = '1.0'; // Opacidad normal para el input
        wrapperDiv.appendChild(searchInput); // Añadir el input al wrapper
        // Div que actuará como la lista desplegable de opciones
        const optionsListDiv = document.createElement("div");
        optionsListDiv.style.position = "absolute";
        // Ajuste de top para que aparezca debajo del input, incluso con sugerencias
        optionsListDiv.style.top = "calc(100% + 5px)"; // Se ajusta dinámicamente o se puede hacer con position: relative dentro de un contenedor fijo.
        optionsListDiv.style.left = "0";
        optionsListDiv.style.width = "calc(100% - 2px)";
        optionsListDiv.style.maxHeight = "200px";
        optionsListDiv.style.overflowY = "auto";
        optionsListDiv.style.border = "1px solid #ddd";
        optionsListDiv.style.backgroundColor = "#fff";
        optionsListDiv.style.zIndex = "1001";
        optionsListDiv.style.display = "none";
        optionsListDiv.style.borderRadius = "3px";
        optionsListDiv.style.boxShadow = "0 2px 5px rgba(0,0,0,0.2)";
        wrapperDiv.appendChild(optionsListDiv);

        // --- Populate options list ---
        function populateOptions(filterText = "") {
            optionsListDiv.innerHTML = ""; // Clear existing options
            const lowerFilterText = filterText.toLowerCase(); // Normalize filter text for case-insensitive search
            // Sort rules alphabetically by their Spanish description for display
            const sortedRules = [...window.dynamicCategoryRules].sort((a, b) => {
                const descA = (getWazeLanguage() === 'es' && a.desc_es) ? a.desc_es : a.desc_en;
                const descB = (getWazeLanguage() === 'es' && b.desc_es) ? b.desc_es : b.desc_en;
                return descA.localeCompare(descB);
            });
            sortedRules.forEach(rule => {// Iterate through each rule
                const displayDesc = (getWazeLanguage() === 'es' && rule.desc_es) ? rule.desc_es : rule.desc_en;
                if (filterText === "" || displayDesc.toLowerCase().includes(lowerFilterText) ||
                    rule.categoryKey.toLowerCase().includes(lowerFilterText)) {// Check if displayDesc or categoryKey contains the filter text
                    const optionDiv = document.createElement("div");
                    optionDiv.style.padding = "5px";
                    optionDiv.style.cursor = "pointer";
                    optionDiv.style.borderBottom = "1px solid #eee";
                    optionDiv.style.display = "flex";
                    optionDiv.style.alignItems = "center";
                    optionDiv.style.gap = "5px";
                    optionDiv.title = `Seleccionar: ${displayDesc} (${rule.categoryKey})`;
                    // Resaltar si es la categoría actual o la temporalmente seleccionada
                    const tempSelectedKey = tempSelectedCategories.get(placeId); // Obtener selección temporal
                    if (rule.categoryKey.toUpperCase() === currentCategoryKey.toUpperCase()) {// Resaltar la categoría actual
                        optionDiv.style.backgroundColor = "#e0f7fa"; // Azul claro para la actual
                        optionDiv.style.fontWeight = "bold";
                    }
                    else if (tempSelectedKey && rule.categoryKey.toUpperCase() === tempSelectedKey.toUpperCase())  // Resaltar selección temporal
                        optionDiv.style.backgroundColor = "#fffacd"; // Amarillo claro para la seleccionada temporalmente
                    else if (dynamicCategorySuggestions.some(s => s.categoryKey.toUpperCase() === rule.categoryKey.toUpperCase()))
                        optionDiv.style.backgroundColor = "#e6ffe6"; // Verde claro para sugerida por el sistema
                    const iconSpan = document.createElement("span");// Icono de la categoría
                    iconSpan.textContent = rule.icon;
                    iconSpan.style.fontSize = "16px";
                    optionDiv.appendChild(iconSpan);
                    const textSpan = document.createElement("span");// Descripción de la categoría
                    textSpan.textContent = displayDesc;
                    optionDiv.appendChild(textSpan);// Añadir descripción de la categoría
                    optionDiv.addEventListener("mouseenter", () => optionDiv.style.backgroundColor = "#f0f0f0");
                    optionDiv.addEventListener("mouseleave", () => {
                        if (tempSelectedKey && rule.categoryKey.toUpperCase() === tempSelectedKey.toUpperCase()) {
                            optionDiv.style.backgroundColor = "#fffacd";
                        }
                        else if (rule.categoryKey.toUpperCase() === currentCategoryKey.toUpperCase()) {
                            optionDiv.style.backgroundColor = "#e0f7fa";
                        }
                        else if (dynamicCategorySuggestions.some(s => s.categoryKey.toUpperCase() === rule.categoryKey.toUpperCase())) {
                            optionDiv.style.backgroundColor = "#e6ffe6";
                        }
                        else {
                            optionDiv.style.backgroundColor = "#fff";
                        }
                    });
                    // Añadir evento click para seleccionar la categoría
                    optionDiv.addEventListener("click", async () => {
                        const placeToUpdate = W.model.venues.getObjectById(placeId);
                        if (!placeToUpdate) {
                            //console.error("[WME_PLN] Lugar no encontrado para actualizar categoría.");
                            return;
                        }

                        try {
                            const UpdateObject = require("Waze/Action/UpdateObject");
                            const action = new UpdateObject(placeToUpdate, { categories: [rule.categoryKey] });
                            W.model.actionManager.add(action);

                            // ✅ CORRECCIÓN: Se declara 'row' aquí, ANTES de su primer uso.
                            const row = document.querySelector(`tr[data-place-id="${placeId}"]`);

                            // Ahora es seguro usar la variable 'row'.
                            if (row) {
                                row.dataset.categoryChanged = 'true'; // Marcar fila como modificada
                                const applyButton = row.querySelector('button[title="Aplicar sugerencia"]');
                                // Habilitar el botón de aplicar sugerencia
                                if (applyButton) {
                                    applyButton.disabled = false;
                                    applyButton.style.opacity = "1";
                                }
                            }

                            // Actualizar visualmente la celda de Categoría Actual en la tabla
                            updateCategoryDisplayInTable(placeId, rule.categoryKey);

                            // ✨ REGENERAR BOTONES DE PREFIJOS (28-nov-2024)
                            // Cuando cambia la categoría, actualizar los botones de palabras clave
                            if (row) {
                                const suggestionCells = row.querySelectorAll('td');
                                for (const cell of suggestionCells) {
                                    if (cell._generatePrefixButtons && typeof cell._generatePrefixButtons === 'function') {
                                        cell._generatePrefixButtons(rule.categoryKey);
                                        break;
                                    }
                                }
                            }

                            // Atenuar la celda de la categoría original
                            if (row) {
                                const categoryCell = row.querySelector('td:nth-child(10)');
                                if (categoryCell) {
                                    const currentCategoryDiv = categoryCell.querySelector('div');
                                    if (currentCategoryDiv) {
                                        currentCategoryDiv.style.opacity = '0.5';
                                        currentCategoryDiv.title += ' (Modificada)';
                                    }
                                }
                            }

                            // Actualizar el valor del input con icono y descripción de la selección
                            searchInput.value = `${rule.icon} ${displayDesc}`;
                            searchInput.style.setProperty('opacity', '1.0', 'important'); // Usar setProperty para asegurar visibilidad

                            // Ocultar la lista de opciones
                            optionsListDiv.style.display = "none";
                            searchInput.blur();

                        } catch (e) {
                            console.error("[WME_PLN] Error al actualizar la categoría desde dropdown:", e);
                            alert("Error al actualizar la categoría: " + e.message);
                        }
                    });
                    optionsListDiv.appendChild(optionDiv);
                }
            });
            if (optionsListDiv.childElementCount === 0) {// Si no hay opciones que coincidan con el filtro, mostrar mensaje
                const noResults = document.createElement("div");
                noResults.style.padding = "5px";
                noResults.style.color = "#777";
                noResults.textContent = "No hay resultados.";
                optionsListDiv.appendChild(noResults);
            }
        }// populateOptions

        // Limpiamos los listeners anteriores y los reescribimos de forma más robusta.

        let debounceTimer;
        searchInput.addEventListener("input", () => {
            clearTimeout(debounceTimer);
            // Muestra la lista y filtra mientras el usuario escribe.
            debounceTimer = setTimeout(() => {
                populateOptions(searchInput.value);
                optionsListDiv.style.display = "block";
            }, 200);
        });

        searchInput.addEventListener("focus", () => {
            // Al hacer foco, muestra la lista completa.
            populateOptions(searchInput.value);
            optionsListDiv.style.display = "block";
        });

        // Usamos 'mousedown' en lugar de 'click' para cerrar el menú.
        // Esto evita conflictos con el evento 'click' de las opciones.
        document.addEventListener("mousedown", (e) => {
            if (!wrapperDiv.contains(e.target)) {
                optionsListDiv.style.display = "none";
            }
        });
        populateOptions(""); // Cargar las opciones inicialmente (sin filtro)
        return wrapperDiv;
    }// createRecommendedCategoryDropdown

    // Función auxiliar para actualizar el display de la categoría actual en la tabla
    function updateCategoryDisplayInTable(placeId, newCategoryKey) {
        const row = document.querySelector(`tr[data-place-id="${placeId}"]`); // Asume que cada fila tiene un data-place-id
        if (!row) return;
        const categoryCell = row.querySelector('td:nth-child(8)'); // La décima columna es "Categoría"

        if (!categoryCell) return;// Asegurarse de que la celda existe
        const categoryDetails = getCategoryDetails(newCategoryKey); // Obtener detalles de la categoría
        const currentCategoryDiv = categoryCell.querySelector('div'); // Contenedor del texto y el ícono
        if (currentCategoryDiv) {// Actualizar el contenido del div existente
            const firstSpan = currentCategoryDiv.querySelector('span:first-child');
            const lastSpan = currentCategoryDiv.querySelector('span:last-child');

            // Verificar que los spans existan antes de actualizar
            if (firstSpan) {
                firstSpan.textContent = categoryDetails.description; // Actualiza el texto
                firstSpan.title = `Categoría Actual: ${categoryDetails.description}`; // Actualiza el título
            }
            if (lastSpan) {
                lastSpan.textContent = categoryDetails.icon; // Actualiza el ícono
            }
        }
    }

    // Renderizar lista de palabras del diccionario
    function renderDictionaryList(ulElement, filter = "") {
        updateDictionaryWordsCountLabel();

        // Asegurarse de que ulElement es válido
        if (!ulElement || !window.dictionaryWords)
            return;
        // Asegurarse de que ulElement es válido
        const currentFilter = filter.toLowerCase();
        ulElement.innerHTML = "";
        // Asegurarse de que dictionaryWords es un Set
        const wordsToRender =
            Array.from(window.dictionaryWords)
                .filter(word => word.toLowerCase().startsWith(currentFilter))
                .sort((a, b) => a.toLowerCase().localeCompare(b.toLowerCase()));
        // Si no hay palabras que renderizar, mostrar mensaje
        if (wordsToRender.length === 0) {
            const li = document.createElement("li");
            li.textContent = window.dictionaryWords.size === 0
                ? "El diccionario está vacío."
                : "No hay coincidencias.";
            li.style.textAlign = "center";
            li.style.color = "#777";
            ulElement.appendChild(li);
            // Guardar diccionario también cuando está vacío
            try {
                localStorage.setItem(
                    "dictionaryWordsList",
                    JSON.stringify(Array.from(window.dictionaryWords)));
            }
            catch (e) {
                console.error("[WME PLN] Error guardando el diccionario en localStorage:", e);
            }
            return;
        }

        // ✨ PAGINACIÓN: Calcular índices de página
        const totalWords = wordsToRender.length;
        const totalPages = Math.ceil(totalWords / WORDS_PER_PAGE);

        // Asegurar que la página actual está en rango válido
        if (dictionaryCurrentPage > totalPages) {
            dictionaryCurrentPage = totalPages || 1;
        }

        const startIdx = (dictionaryCurrentPage - 1) * WORDS_PER_PAGE;
        const endIdx = Math.min(startIdx + WORDS_PER_PAGE, totalWords);
        const pageWords = wordsToRender.slice(startIdx, endIdx);

        // Agregar indicador de resultados
        const infoLi = document.createElement("li");
        infoLi.style.textAlign = "center";
        infoLi.style.padding = "8px";
        infoLi.style.backgroundColor = "#f5f5f5";
        infoLi.style.fontWeight = "bold";
        infoLi.style.borderBottom = "2px solid #007bff";
        infoLi.textContent = `Mostrando ${startIdx + 1}-${endIdx} de ${totalWords} palabra${totalWords !== 1 ? 's' : ''}`;
        ulElement.appendChild(infoLi);

        // Renderizar solo palabras de la página actual
        pageWords.forEach(word => {
            const li = document.createElement("li");
            li.style.display = "flex";
            li.style.justifyContent = "space-between";
            li.style.alignItems = "center";
            li.style.padding = "4px 2px";
            li.style.borderBottom = "1px solid #f0f0f0";
            // Span para la palabra
            const wordSpan = document.createElement("span");
            wordSpan.textContent = word;
            wordSpan.style.maxWidth = "calc(100% - 60px)";
            wordSpan.style.overflow = "hidden";
            wordSpan.style.textOverflow = "ellipsis";
            wordSpan.style.whiteSpace = "nowrap";
            wordSpan.title = word;
            li.appendChild(wordSpan);
            // Contenedor para los iconos de acción
            const iconContainer = document.createElement("span");
            iconContainer.style.display = "flex";
            iconContainer.style.gap = "8px";
            //Botón de edición y eliminación
            const editBtn = document.createElement("button");
            editBtn.innerHTML = "✏️";
            editBtn.title = "Editar";
            editBtn.style.border = "none";
            editBtn.style.background = "transparent";
            editBtn.style.cursor = "pointer";
            editBtn.style.padding = "2px";
            editBtn.style.fontSize = "14px";
            editBtn.addEventListener("click", async () => {
                const newWord = await showCustomPrompt("Editar palabra:", "✏️ Editar Palabra del Diccionario", word, "📝");
                if (newWord !== null && newWord.trim() !== word) {
                    window.dictionaryWords.delete(word);
                    window.dictionaryWords.add(newWord.trim());

                    // ✨ MARCAR la nueva palabra como añadida por el usuario (28-nov-2024)
                    if (!window.userAddedDictionaryWords) window.userAddedDictionaryWords = new Set();
                    window.userAddedDictionaryWords.add(newWord.trim());

                    // Guardar en localStorage
                    try {
                        localStorage.setItem("dictionaryWordsList", JSON.stringify(Array.from(window.dictionaryWords)));
                        localStorage.setItem('userAddedDictionaryWords', JSON.stringify(Array.from(window.userAddedDictionaryWords)));
                    } catch (e) {
                        console.error("[WME PLN] Error guardando diccionario después de editar:", e);
                    }

                    renderDictionaryList(ulElement, currentFilter);
                }
            });
            // Botón de eliminación
            const deleteBtn = document.createElement("button");
            deleteBtn.innerHTML = "🗑️";
            deleteBtn.title = "Eliminar";
            deleteBtn.style.border = "none";
            deleteBtn.style.background = "transparent";
            deleteBtn.style.cursor = "pointer";
            deleteBtn.style.padding = "2px";
            deleteBtn.style.fontSize = "14px";
            deleteBtn.addEventListener("click", () => {
                // Confirmación antes de eliminar
                if (confirm(`¿Eliminar la palabra '${word}' del diccionario?`)) {
                    window.dictionaryWords.delete(word);
                    renderDictionaryList(ulElement, currentFilter);
                }
            });
            iconContainer.appendChild(editBtn);
            iconContainer.appendChild(deleteBtn);
            li.appendChild(iconContainer);
            ulElement.appendChild(li);
        });

        // ✨ PAGINACIÓN: Agregar controles de paginación
        if (totalPages > 1) {
            const paginationLi = document.createElement("li");
            paginationLi.style.textAlign = "center";
            paginationLi.style.padding = "10px";
            paginationLi.style.backgroundColor = "#f9f9f9";
            paginationLi.style.borderTop = "2px solid #007bff";
            paginationLi.style.display = "flex";
            paginationLi.style.justifyContent = "center";
            paginationLi.style.alignItems = "center";
            paginationLi.style.gap = "10px";

            // Botón Anterior
            const prevBtn = document.createElement("button");
            prevBtn.textContent = "◀ Anterior";
            prevBtn.disabled = dictionaryCurrentPage === 1;
            prevBtn.style.padding = "6px 12px";
            prevBtn.style.border = "1px solid #007bff";
            prevBtn.style.borderRadius = "4px";
            prevBtn.style.backgroundColor = prevBtn.disabled ? "#e0e0e0" : "#007bff";
            prevBtn.style.color = prevBtn.disabled ? "#999" : "white";
            prevBtn.style.cursor = prevBtn.disabled ? "not-allowed" : "pointer";
            prevBtn.style.fontWeight = "bold";
            prevBtn.addEventListener("click", () => {
                if (dictionaryCurrentPage > 1) {
                    dictionaryCurrentPage--;
                    renderDictionaryList(ulElement, currentFilter);
                }
            });

            // Indicador de página
            const pageInfo = document.createElement("span");
            pageInfo.textContent = `Página ${dictionaryCurrentPage} de ${totalPages}`;
            pageInfo.style.fontWeight = "bold";
            pageInfo.style.padding = "0 10px";

            // Botón Siguiente
            const nextBtn = document.createElement("button");
            nextBtn.textContent = "Siguiente ▶";
            nextBtn.disabled = dictionaryCurrentPage === totalPages;
            nextBtn.style.padding = "6px 12px";
            nextBtn.style.border = "1px solid #007bff";
            nextBtn.style.borderRadius = "4px";
            nextBtn.style.backgroundColor = nextBtn.disabled ? "#e0e0e0" : "#007bff";
            nextBtn.style.color = nextBtn.disabled ? "#999" : "white";
            nextBtn.style.cursor = nextBtn.disabled ? "not-allowed" : "pointer";
            nextBtn.style.fontWeight = "bold";
            nextBtn.addEventListener("click", () => {
                if (dictionaryCurrentPage < totalPages) {
                    dictionaryCurrentPage++;
                    renderDictionaryList(ulElement, currentFilter);
                }
            });

            paginationLi.appendChild(prevBtn);
            paginationLi.appendChild(pageInfo);
            paginationLi.appendChild(nextBtn);
            ulElement.appendChild(paginationLi);
        }
        // Guardar el diccionario actualizado en localStorage después de cada render
        try {
            localStorage.setItem("dictionaryWordsList", JSON.stringify(Array.from(window.dictionaryWords)));
        }
        catch (e) {
            console.error("[WME PLN] Error guardando el diccionario en localStorage:", e);
        }
    }// renderDictionaryList


    // Función para obtener el ícono de categoría (acepta nombre o clave)
    function getCategoryIcon(categoryName) {


        // Mapa de categorías a íconos con soporte bilingüe
        const categoryIcons = {
            // Comida y Restaurantes / Food & Restaurants
            "FOOD_AND_DRINK": { icon: "🦞🍷", es: "Comida y Bebidas", en: "Food and Drinks" },
            "RESTAURANT": { icon: "🍽️", es: "Restaurante", en: "Restaurant" },
            "FAST_FOOD": { icon: "🍔", es: "Comida rápida", en: "Fast Food" },
            "CAFE": { icon: "☕", es: "Cafetería", en: "Cafe" },
            "BAR": { icon: "🍺", es: "Bar", en: "Bar" },
            "BAKERY": { icon: "🥖", es: "Panadería", en: "Bakery" },
            "ICE_CREAM": { icon: "🍦", es: "Heladería", en: "Ice Cream Shop" },
            "DEPARTMENT_STORE": { icon: "🏬", es: "Tienda por departamentos", en: "Department Store" },
            "PARK": { icon: "🌳", es: "Parque", en: "Park" },
            // Compras y Servicios / Shopping & Services
            "FASHION_AND_CLOTHING": { icon: "👗", es: "Moda y Ropa", en: "Fashion and Clothing" },
            "SHOPPING_AND_SERVICES": { icon: "👜👝", es: "Mercado o Tienda", en: "Shopping and Services" },
            "SHOPPING_CENTER": { icon: "🛍️", es: "Centro comercial", en: "Shopping Center" },
            "SUPERMARKET_GROCERY": { icon: "🛒", es: "Supermercado", en: "Supermarket" },
            "MARKET": { icon: "🛒", es: "Mercado", en: "Market" },
            "CONVENIENCE_STORE": { icon: "🏪", es: "Tienda", en: "Convenience Store" },
            "PHARMACY": { icon: "💊", es: "Farmacia", en: "Pharmacy" },
            "BANK": { icon: "🏦", es: "Banco", en: "Bank" },
            "ATM": { icon: "💳", es: "Cajero automático", en: "ATM" },
            "HARDWARE_STORE": { icon: "🔧", es: "Ferretería", en: "Hardware Store" },
            "COURTHOUSE": { icon: "⚖️", es: "Corte", en: "Courthouse" },
            "FURNITURE_HOME_STORE": { icon: "🛋️", es: "Tienda de muebles", en: "Furniture Store" },
            "TOURIST_ATTRACTION_HISTORIC_SITE": { icon: "🗿", es: "Atracción turística o Sitio histórico", en: "Tourist Attraction or Historic Site" },
            "PET_STORE_VETERINARIAN_SERVICES": { icon: "🦮🐈", es: "Tienda de mascotas o Veterinaria", en: "Pet Store or Veterinary Services" },
            "CEMETERY": { icon: "🪦", es: "Cementerio", en: "Cemetery" },
            "KINDERGARDEN": { icon: "🍼", es: "Jardín Infantil", en: "Kindergarten" },
            "JUNCTION_INTERCHANGE": { icon: "🔀", es: "Cruce o Intercambio", en: "Junction or Interchange" },
            "OUTDOORS": { icon: "🏞️", es: "Aire libre", en: "Outdoors" },
            "ORGANIZATION_OR_ASSOCIATION": { icon: "👔", es: "Organización o Asociación", en: "Organization or Association" },
            "TRAVEL_AGENCY": { icon: "🧳", es: "Agencia de viajes", en: "Travel Agency" },
            "BANK_FINANCIAL": { icon: "💰", es: "Banco o Financiera", en: "Bank or Financial Institution" },
            "SPORTING_GOODS": { icon: "🛼🏀🏐", es: "Artículos deportivos", en: "Sporting Goods" },
            "TOY_STORE": { icon: "🧸", es: "Tienda de juguetes", en: "Toy Store" },
            "CURRENCY_EXCHANGE": { icon: "💶💱", es: "Casa de cambio", en: "Currency Exchange" },
            "PHOTOGRAPHY": { icon: "📸", es: "Fotografía", en: "Photography" },
            "DESSERT": { icon: "🍰", es: "Postre", en: "Dessert" },
            "FOOD_COURT": { icon: "🥗", es: "Comedor o Patio de comidas", en: "Food Court" },
            "CANAL": { icon: "〰", es: "Canal", en: "Canal" },
            "JEWELRY": { icon: "💍", es: "Joyería", en: "Jewelry" },
            // Transporte / Transportation
            "TRAIN_STATION": { icon: "🚂", es: "Estación de tren", en: "Train Station" },
            "GAS_STATION": { icon: "⛽", es: "Estación de servicio", en: "Gas Station" },
            "PARKING_LOT": { icon: "🅿️", es: "Estacionamiento", en: "Parking Lot" },
            "BUS_STATION": { icon: "🚍", es: "Terminal de bus", en: "Bus Station" },
            "AIRPORT": { icon: "✈️", es: "Aeropuerto", en: "Airport" },
            "CAR_WASH": { icon: "🚗💦", es: "Lavado de autos", en: "Car Wash" },
            "CAR_RENTAL": { icon: "🚘🛺🛻🚙", es: "Alquiler de Vehículos", en: "Car Rental" },
            "TAXI_STATION": { icon: "🚕", es: "Estación de taxis", en: "Taxi Station" },
            "FOREST_GROVE": { icon: "🌳", es: "Bosque", en: "Forest Grove" },
            "GARAGE_AUTOMOTIVE_SHOP": { icon: "🔧🚗", es: "Taller mecánico", en: "Automotive Garage" },
            "GIFTS": { icon: "🎁", es: "Tienda de regalos", en: "Gift Shop" },
            "TOLL_BOOTH": { icon: "🚧", es: "Peaje", en: "Toll Booth" },
            "CHARGING_STATION": { icon: "🔋", es: "Estación de carga", en: "Charging Station" },
            "CAR_SERVICES": { icon: "🚗🔧", es: "Servicios de automóviles", en: "Car Services" },
            "STADIUM_ARENA": { icon: "🏟️", es: "Estadio o Arena", en: "Stadium or Arena" },
            "CAR_DEALERSHIP": { icon: "🚘🏢", es: "Concesionario de autos", en: "Car Dealership" },
            "FERRY_PIER": { icon: "⛴️", es: "Muelle de ferry", en: "Ferry Pier" },
            "INFORMATION_POINT": { icon: "ℹ️", es: "Punto de información", en: "Information Point" },
            "REST_AREAS": { icon: "🏜", es: "Áreas de descanso", en: "Rest Areas" },
            "MUSIC_VENUE": { icon: "🎶", es: "Lugar de música", en: "Music Venue" },
            "CASINO": { icon: "🎰", es: "Casino", en: "Casino" },
            "CITY_HALL": { icon: "🎩", es: "Ayuntamiento", en: "City Hall" },
            "PERFORMING_ARTS_VENUE": { icon: "🎭", es: "Lugar de artes escénicas", en: "Performing Arts Venue" },
            "TUNNEL": { icon: "🔳", es: "Túnel", en: "Tunnel" },
            "SEAPORT_MARINA_HARBOR": { icon: "⚓", es: "Puerto o Marina", en: "Seaport or Marina" },
            // Alojamiento / Lodging
            "HOTEL": { icon: "🏨", es: "Hotel", en: "Hotel" },
            "HOSTEL": { icon: "🛏️", es: "Hostal", en: "Hostel" },
            "LODGING": { icon: "⛺", es: "Alojamiento", en: "Lodging" },
            "MOTEL": { icon: "🛕", es: "Motel", en: "Motel" },
            "SWIMMING_POOL": { icon: "🏊", es: "Piscina", en: "Swimming Pool" },
            "RIVER_STREAM": { icon: "🌊", es: "Río o Arroyo", en: "River or Stream" },
            "CAMPING_TRAILER_PARK": { icon: "🏕️", es: "Camping o Parque de Trailers", en: "Camping or Trailer Park" },
            "SEA_LAKE_POOL": { icon: "🏖️", es: "Mar, Lago o Piscina", en: "Sea, Lake or Pool" },
            "FARM": { icon: "🚜", es: "Granja", en: "Farm" },
            "NATURAL_FEATURES": { icon: "🌲", es: "Características naturales", en: "Natural Features" },
            // Salud / Healthcare
            "HOSPITAL": { icon: "🏥", es: "Hospital", en: "Hospital" },
            "HOSPITAL_URGENT_CARE": { icon: "🏥🚑", es: "Urgencias", en: "Urgent Care" },
            "DOCTOR_CLINIC": { icon: "🏥⚕️", es: "Clínica", en: "Clinic" },
            "DOCTOR": { icon: "👨‍⚕️", es: "Consultorio médico", en: "Doctor's Office" },
            "VETERINARY": { icon: "🐾", es: "Veterinaria", en: "Veterinary" },
            "PERSONAL_CARE": { icon: "💅💇🦷", es: "Cuidado personal", en: "Personal Care" },
            "FACTORY_INDUSTRIAL": { icon: "🏭", es: "Fábrica o Industrial", en: "Factory or Industrial" },
            "MILITARY": { icon: "🪖", es: "Militar", en: "Military" },
            "LAUNDRY_DRY_CLEAN": { icon: "🧺", es: "Lavandería o Tintorería", en: "Laundry or Dry Clean" },
            "PLAYGROUND": { icon: "🛝", es: "Parque infantil", en: "Playground" },
            "TRASH_AND_RECYCLING_FACILITIES": { icon: "🗑️♻️", es: "Instalaciones de basura y reciclaje", en: "Trash and Recycling Facilities" },
            // Educación / Education
            "UNIVERSITY": { icon: "🎓", es: "Universidad", en: "University" },
            "COLLEGE_UNIVERSITY": { icon: "🏫", es: "Colegio", en: "College" },
            "SCHOOL": { icon: "🎒", es: "Escuela", en: "School" },
            "LIBRARY": { icon: "📖", es: "Biblioteca", en: "Library" },
            "FLOWERS": { icon: "💐", es: "Floristería", en: "Flower Shop" },
            "CONVENTIONS_EVENT_CENTER": { icon: "🎤🥂", es: "Centro de convenciones o eventos", en: "Convention or Event Center" },
            "CLUB": { icon: "♣", es: "Club", en: "Club" },
            "ART_GALLERY": { icon: "🖼️", es: "Galería de arte", en: "Art Gallery" },
            "NATURAL_FEATURES": { icon: "🌄", es: "Características naturales", en: "Natural Features" },
            // Entretenimiento / Entertainment
            "CINEMA": { icon: "🎬", es: "Cine", en: "Cinema" },
            "THEATER": { icon: "🎭", es: "Teatro", en: "Theater" },
            "MUSEUM": { icon: "🖼", es: "Museo", en: "Museum" },
            "CULTURE_AND_ENTERTAINEMENT": { icon: "🎨", es: "Cultura y Entretenimiento", en: "Culture and Entertainment" },
            "STADIUM": { icon: "🏟️", es: "Estadio", en: "Stadium" },
            "GYM": { icon: "💪", es: "Gimnasio", en: "Gym" },
            "GYM_FITNESS": { icon: "🏋️", es: "Gimnasio o Fitness", en: "Gym or Fitness" },
            "GAME_CLUB": { icon: "⚽🏓", es: "Club de juegos", en: "Game Club" },
            "BOOKSTORE": { icon: "📖📚", es: "Librería", en: "Bookstore" },
            "ELECTRONICS": { icon: "📱💻", es: "Electrónica", en: "Electronics" },
            "SPORTS_COURT": { icon: "⚽🏀", es: "Cancha deportiva", en: "Sports Court" },
            "GOLF_COURSE": { icon: "⛳", es: "Campo de golf", en: "Golf Course" },
            "SKI_AREA": { icon: "⛷️", es: "Área de esquí", en: "Ski Area" },
            "RACING_TRACK": { icon: "🛷⛸🏎️", es: "Pista de carreras", en: "Racing Track" },
            // Gobierno y Servicios Públicos / Government & Public Services
            "GOVERNMENT": { icon: "🏛️", es: "Oficina gubernamental", en: "Government Office" },
            "POLICE_STATION": { icon: "👮", es: "Estación de policía", en: "Police Station" },
            "FIRE_STATION": { icon: "🚒", es: "Estación de bomberos", en: "Fire Station" },
            "FIRE_DEPARTMENT": { icon: "🚒", es: "Departamento de bomberos", en: "Fire Department" },
            "POST_OFFICE": { icon: "📫", es: "Correo", en: "Post Office" },
            "TRANSPORTATION": { icon: "🚌", es: "Transporte", en: "Transportation" },
            "PRISON_CORRECTIONAL_FACILITY": { icon: "👁️‍🗨️", es: "Prisión o Centro Correccional", en: "Prison or Correctional Facility" },
            // Religión / Religion
            "RELIGIOUS_CENTER": { icon: "⛪", es: "Iglesia", en: "Church" },
            // Otros / Others
            "RESIDENTIAL": { icon: "🏘️", es: "Residencial", en: "Residential" },
            "RESIDENCE_HOME": { icon: "🏠", es: "Residencia o Hogar", en: "Residence or Home" },
            "OFFICES": { icon: "🏢", es: "Oficina", en: "Office" },
            "FACTORY": { icon: "🏭", es: "Fábrica", en: "Factory" },
            "CONSTRUCTION_SITE": { icon: "🏗️", es: "Construcción", en: "Construction" },
            "MONUMENT": { icon: "🗽", es: "Monumento", en: "Monument" },
            "BRIDGE": { icon: "🌉", es: "Puente", en: "Bridge" },
            "PROFESSIONAL_AND_PUBLIC": { icon: "🗄💼", es: "Profesional y Público", en: "Professional and Public" },
            "OTHER": { icon: "🚪", es: "Otro", en: "Other" },
            "ARTS_AND_CRAFTS": { icon: "🎨", es: "Artes y Manualidades", en: "Arts and Crafts" },
            "COTTAGE_CABIN": { icon: "🏡", es: "Cabaña", en: "Cottage Cabin" },
            "TELECOM": { icon: "📡", es: "Telecomunicaciones", en: "Telecommunications" }
        };
        // Si no hay categoría, devolver ícono por defecto
        if (!categoryName) {
            return { icon: "❓", title: "Sin categoría / No category" };
        }
        // Normalizar el nombre de la categoría
        const normalizedInput = String(categoryName).toLowerCase()
            .normalize("NFD")
            .replace(/[\u0300-\u036f]/g, "")
            .trim();
        //console.log("[WME_PLN][DEBUG] Buscando ícono para categoría:", categoryName);
        //console.log("[WME_PLN][DEBUG] Nombre normalizado:", normalizedInput);
        // 1. Buscar coincidencia exacta por clave interna (ej: "PARK")
        for (const [key, data] of Object.entries(categoryIcons)) {
            if (key.toLowerCase() === normalizedInput) {
                return { icon: data.icon, title: `${data.es} / ${data.en}` };
            }
        }
        // Buscar coincidencia en el mapa de categorías
        for (const [key, data] of Object.entries(categoryIcons)) {
            // Normalizar los nombres en español e inglés para la comparación
            const normalizedES = data.es.toLowerCase()
                .normalize("NFD")
                .replace(/[\u0300-\u036f]/g, "")
                .trim();
            const normalizedEN = data.en.toLowerCase()
                .normalize("NFD")
                .replace(/[\u0300-\u036f]/g, "")
                .trim();
            if (normalizedInput === normalizedES || normalizedInput === normalizedEN) {
                return { icon: data.icon, title: `${data.es} / ${data.en}` };
            }
        }
        // Si no se encuentra coincidencia, devolver ícono por defecto
        //console.log("[WME_PLN][DEBUG] No se encontró coincidencia, usando ícono por defecto");
        return {
            icon: "⚪",
            title: `${categoryName} (Sin coincidencia / No match)`
        };
    }// getCategoryIcon

    // Función para manejar el archivo XML arrastrado
    function exportExcludedWordsList() {
        // Verificar si hay palabras excluidas
        if (excludedWords.size === 0 && Object.keys(replacementWords).length === 0) {
            alert("No hay palabras especiales ni reemplazos para exportar.");
            return;
        }
        // Crear el contenido XML
        let xmlContent = `<?xml version="1.0" encoding="UTF-8"?>\n<ExcludedWords>\n`;
        xmlContent +=
            Array.from(excludedWords)
                .sort((a, b) => a.toLowerCase().localeCompare(b.toLowerCase()))
                .map(w => `    <word>${xmlEscape(w)}</word>`)
                .join("\n");
        // Añadir reemplazos si existen
        if (Object.keys(replacementWords).length > 0) {
            xmlContent += "\n";
            xmlContent +=
                Object.entries(replacementWords)
                    .map(([from, to]) => `    <replacement from="${xmlEscape(from)}">${xmlEscape(to)}</replacement>`)
                    .join("\n");
        }
        xmlContent += "\n</ExcludedWords>";
        // Crear el Blob y descargarlo
        const blob = new Blob([xmlContent], { type: "application/xml;charset=utf-8" });
        // Crear un enlace temporal para descargar el archivo
        const url = URL.createObjectURL(blob);
        // Crear un elemento <a> para descargar el archivo
        const a = document.createElement("a");
        a.href = url;
        a.download = "wme_excluded_words_export.xml";
        document.body.appendChild(a);
        a.click();
        document.body.removeChild(a);
        URL.revokeObjectURL(url);
    }// exportExcludedWordsList

    // Función para exportar palabras del diccionario a XML
    function exportDictionaryWordsList() {
        // Verificar si hay palabras en el diccionario
        if (window.dictionaryWords.size === 0) {
            alert(
                "La lista de palabras del diccionario está vacía. Nada que exportar.");
            return;
        }
        // Crear el contenido XML
        const xmlContent = `<?xml version="1.0" encoding="UTF-8"?>\n<diccionario>\n${Array.from(window.dictionaryWords)
            .sort((a, b) => a.toLowerCase().localeCompare(
                b.toLowerCase()))                     // Exportar ordenado
            .map(w => `    <word>${xmlEscape(w)}</word>`) // Indentación y escape
            .join("\n")}\n</diccionario>`;
        // Crear el Blob y descargarlo
        const blob = new Blob([xmlContent], { type: "application/xml;charset=utf-8" }); // Añadir charset
        // Crear un enlace temporal para descargar el archivo
        const url = URL.createObjectURL(blob);
        // Crear un elemento <a> para descargar el archivo
        const a = document.createElement("a");
        a.href = url;
        a.download = "wme_dictionary_words_export.xml"; // Nombre más descriptivo
        document.body.appendChild(a);
        a.click();
        document.body.removeChild(a);
        URL.revokeObjectURL(url);
    }// exportDictionaryWordsList

    // Función para exportar datos compartidos a XML
    function xmlEscape(str) {
        return str.replace(/[<>&"']/g, function (match) {
            switch (match) {
                case '<':
                    return '&lt;';
                case '>':
                    return '&gt;';
                case '&':
                    return '&amp;';
                case '"':
                    return '&quot;';
                case "'":
                    return '&apos;';
                default:
                    return match;
            }
        });
    }// xmlEscape
    // Add this near the end of your script init
    window.addEventListener('beforeunload', function () {
        // Cancel any pending requests or cleanup tasks
        pendingRequests = [];

        // Save any unsaved data to localStorage
        if (window.dynamicCategoryRules && window.dynamicCategoryRules.length) {
            try {
                localStorage.setItem("wme_pln_categories_cache", JSON.stringify({
                    data: window.dynamicCategoryRules,
                    timestamp: Date.now()
                }));
            } catch (e) {
                console.warn('[WME PLN] Error saving categories on unload:', e);
            }
        }

        if (window.dictionaryWords && window.dictionaryWords.size) {
            try {
                localStorage.setItem("dictionaryWordsList", JSON.stringify(Array.from(window.dictionaryWords)));
            } catch (e) {
                console.warn('[WME PLN] Error saving dictionary on unload:', e);
            }
        }
    });
    // Función para manejar el archivo XML arrastrado
    waitForSidebarAPI();
    // Iniciar el bucle de procesamiento para el efecto de titilado
    requestAnimationFrame(processingLoop);
    //Llamar a la función para mostrar el changelog
    showChangelogOnUpdate();
    
    // NOTE: showDisclaimerPopup ya se maneja dentro de plnValidateAccess (línea ~9999)
    // No duplicar la llamada aquí para evitar múltiples popups

    // Agregar un observador para detectar cuándo se cierra el panel de resultados
    // Esto ayudará a restablecer los estados correctamente cuando el usuario cierra el panel manualmente
    function setupResultsPanelObserver() {
        const observer = new MutationObserver((mutations) => {
            for (const mutation of mutations) {
                if (mutation.type === 'attributes' && mutation.attributeName === 'style') {
                    const panel = document.getElementById('wme-place-inspector-panel');
                    if (panel && panel.style.display === 'none') {
                        // El panel se ha cerrado, restablecer los estados
                        isResultsPanelOpen = false;
                        isProcessingActive = false;
                        isNormalizationActive = true;
                        console.log("[WME PLN] Panel cerrado, estados restablecidos");
                    }
                }
            }
        });

        // Observar el panel principal para detectar cuando se cierra
        const panel = document.getElementById('wme-place-inspector-panel');
        if (panel) {
            observer.observe(panel, { attributes: true });
        } else {
            // Si el panel aún no existe, configurar un temporizador para intentar nuevamente
            setTimeout(setupResultsPanelObserver, 1000);
        }
    }

    // Llamar a esta función al inicio
    setTimeout(setupResultsPanelObserver, 1000);

})();



// Función reutilizable para mostrar el spinner de carga
function showLoadingSpinner() {
    const scanSpinner = document.createElement("div");
    scanSpinner.id = "scanSpinnerOverlay";
    scanSpinner.style.position = "fixed";
    scanSpinner.style.top = "0";
    scanSpinner.style.left = "0";
    scanSpinner.style.width = "100%";
    scanSpinner.style.height = "100%";
    scanSpinner.style.background = "rgba(0, 0, 0, 0.5)";
    scanSpinner.style.zIndex = "10000";
    scanSpinner.style.display = "flex";
    scanSpinner.style.justifyContent = "center";
    scanSpinner.style.alignItems = "center";
    // Estilos para centrar el contenido
    const scanContent = document.createElement("div");
    scanContent.style.background = "#fff";
    scanContent.style.padding = "20px";
    scanContent.style.borderRadius = "8px";
    scanContent.style.textAlign = "center";
    // Spinner de carga
    const spinner = document.createElement("div");
    spinner.classList.add("spinner");
    spinner.style.width = "40px";
    spinner.style.height = "40px";
    spinner.style.margin = "0 auto 10px auto";
    // Texto de progreso
    const progressText = document.createElement("div");
    progressText.id = "scanProgressText";
    progressText.textContent = "Analizando lugares: 0%";
    progressText.style.fontSize = "14px";
    progressText.style.color = "#333";
    // Añadir spinner y texto al contenido
    scanContent.appendChild(spinner);
    scanContent.appendChild(progressText);
    scanSpinner.appendChild(scanContent);
    document.body.appendChild(scanSpinner);
    // Añadir estilos de animación al documento si no existen
    if (!document.getElementById('wme-pln-animations')) {
        const style = document.createElement("style");
        style.id = 'wme-pln-animations';
        style.textContent = `
                @keyframes spin {
                    0% { transform: rotate(0deg); }
                    100% { transform: rotate(360deg); }
                }
                @keyframes area-blink {
                    0% { opacity: 1; }
                    50% { opacity: 0.3; }
                    100% { opacity: 1; }
                }
                .area-blink {
                    animation: area-blink 1s infinite;
                }
                .spinner {
                    border: 6px solid #f3f3f3;
                    border-top: 6px solid #3498db;
                    border-radius: 50%;
                    animation: spin 1s linear infinite;
                }
            `;
        document.head.appendChild(style);
    }

    // Añadir estilos para lugares sin nombre
    const emptyNameStyles = document.createElement('style');
    emptyNameStyles.textContent = `
            .pln-empty-name-row {
                background-color: #fff3e0;
            }
            .pln-empty-name-row:hover {
                background-color: #ffe0b2;
            }

            /* Estilos para el placeholder de "Requiere corrección manual" */
            .replacement-input::placeholder {
                color: var(--placeholder-color, #999);
                font-weight: bold;
            }
            .replacement-input::-webkit-input-placeholder {
                color: var(--placeholder-color, #999);
                font-weight: bold;
            }
            .replacement-input::-moz-placeholder {
                color: var(--placeholder-color, #999);
                font-weight: bold;
            }
            .replacement-input:-ms-input-placeholder {
                color: var(--placeholder-color, #999);
                font-weight: bold;
            }
        `;
    document.head.appendChild(emptyNameStyles);
}// showLoadingSpinner



// Función para agregar una palabra al diccionario
function addWordToDictionary(input) {
    const newWord = input.value.trim().toLowerCase();

    if (!newWord) {
        if (window.plnToast) window.plnToast("⚠️ La palabra no puede estar vacía.", 3000);
        return;
    }
    // Validaciones básicas antes de añadir
    if (newWord.length === 1 && !newWord.match(/[a-zA-Z0-9]/)) {
        if (window.plnToast) window.plnToast("⚠️ No se permite agregar un solo carácter que no sea alfanumérico.", 3000);
        return;
    }
    if (window.commonWords && window.commonWords.has(newWord)) {
        if (window.plnToast) window.plnToast("⚠️ Esa palabra es muy común y no debe agregarse al diccionario.", 3000);
        return;
    }
    if (window.excludedWords && window.excludedWords.has(newWord)) {
        if (window.plnToast) window.plnToast("⚠️ Esa palabra ya existe en la lista de especiales (excluidas).", 3000);
        return;
    }
    if (window.dictionaryWords.has(newWord)) {
        if (window.plnToast) window.plnToast("⚠️ La palabra ya existe en el diccionario.", 3000);
        return;
    }
    if (!window.dictionaryWords) window.dictionaryWords = new Set();
    if (!window.dictionaryIndex) window.dictionaryIndex = {};
    window.dictionaryWords.add(newWord); // Añadir al Set

    // ✨ MARCAR como palabra añadida por el usuario (28-nov-2024)
    if (!window.userAddedDictionaryWords) window.userAddedDictionaryWords = new Set();
    window.userAddedDictionaryWords.add(newWord);

    // === AÑADIR AL ÍNDICE ===
    const firstChar = newWord.charAt(0).toLowerCase();
    if (!window.dictionaryIndex[firstChar]) {
        window.dictionaryIndex[firstChar] = [];
    }
    window.dictionaryIndex[firstChar].push(newWord); // Añadir al índice
    input.value = ""; // Limpiar el input

    // ✨ Limpiar cache porque el diccionario cambió
    if (window.clearWordLookupCache) {
        window.clearWordLookupCache();
    }

    // Actualizar UI antigua (si existe)
    const oldList = document.getElementById("dictionaryWordsList");
    if (oldList) renderDictionaryList(oldList);

    // ✨ Actualizar UI Nueva (Diccionario Dinámico)
    const listContainer = document.getElementById("dictionaryContainer");
    if (listContainer && typeof inicializarDiccionarioDinamico === 'function' && typeof prepararDatosDiccionario === 'function') {
        listContainer.innerHTML = "";
        inicializarDiccionarioDinamico("dictionaryContainer", prepararDatosDiccionario());
    }

    if (window.updateDictionaryWordsCountLabel) {
        window.updateDictionaryWordsCountLabel();
    }
    // Guardar en localStorage después de añadir
    try {
        localStorage.setItem("dictionaryWordsList", JSON.stringify(Array.from(window.dictionaryWords)));
        // Guardar también las palabras de usuario
        localStorage.setItem('userAddedDictionaryWords', JSON.stringify(Array.from(window.userAddedDictionaryWords)));
        if (window.plnToast) window.plnToast(`✅ Palabra "${newWord}" agregada al diccionario.`, 2500);
    } catch (e) {
        console.error("Error guardando en localStorage:", e);
        if (window.plnToast) window.plnToast("❌ Error al guardar la palabra.", 3000);
    }

}// addWordToDictionary

// === [PLN] Export city register helper globally ===
try {
    window.plnRegisterWorkCity = plnAddWorkCity;
    if (typeof unsafeWindow !== 'undefined') unsafeWindow.plnRegisterWorkCity = plnAddWorkCity;
} catch (_) { }


// =============================================================================
// IIFE for EasyShortCuts Module
// =============================================================================
(function () {

    // Lógica de restricción de rango e inicialización
    async function checkRankAndInitEasyShortCuts() {
        console.log('[NrmliZer-EasyShortCuts] 🕵️ checkRankAndInitEasyShortCuts iniciado');
        
        // Intentar obtener el rango desde varias fuentes posibles
        let rank = -1;
        let userName = '';

        if (typeof W !== 'undefined' && W.loginManager) {
            userName = W.loginManager.userName || '';
            // Intento 1: W.loginManager.user.rank
            if (W.loginManager.user && W.loginManager.user.rank != null) {
                rank = Number(W.loginManager.user.rank);
            }
            // Intento 2: W.loginManager.user.attributes.rank
            else if (W.loginManager.user && W.loginManager.user.attributes && W.loginManager.user.attributes.rank != null) {
                rank = Number(W.loginManager.user.attributes.rank);
            }
            // Intento 3: W.loginManager.getUser()
            else if (W.loginManager.getUser) {
                const u = W.loginManager.getUser();
                if (u && u.rank != null) rank = Number(u.rank);
                else if (u && u.attributes && u.attributes.rank != null) rank = Number(u.attributes.rank);
            }
        }

        if (rank >= 0) {
            // 🔐 PASO 1: Verificar lista blanca para EasyShortCuts
            let whitelisted = false;
            const accessCtrl = window.PLN_ACCESS_CONTROL || (typeof PLN_ACCESS_CONTROL !== 'undefined' ? PLN_ACCESS_CONTROL : {});
            
            // 🔧 DEBUG: Mostrar config actual
            console.log('[NrmliZer-EasyShortCuts] 📋 Config actual:', {
                easyShortcutsMinLevel: accessCtrl.easyShortcutsMinLevel,
                defaultMinLevel: accessCtrl.defaultMinLevel,
                userRank: rank,
                userLevel: rank + 1,
                userName: userName
            });

            if (userName && accessCtrl.enabled) {
                try {
                    const whitelistResult = await checkWhitelistAccess(userName, 'easyshortcuts');
                    if (whitelistResult.success && whitelistResult.hasAccess && whitelistResult.isWhitelisted) {
                        console.log(`[NrmliZer-EasyShortCuts] ✅ Usuario ${userName} en LISTA BLANCA para EasyShortCuts. Acceso permitido.`);
                        whitelisted = true;
                    }
                } catch (e) {
                    console.warn('[NrmliZer-EasyShortCuts] Error verificando whitelist:', e);
                }
            }

            // 🔐 PASO 2: Verificación de nivel local (si no está en whitelist)
            // Nivel 4 corresponde a rank index 3 (0-based)
            const minRank = accessCtrl.easyShortcutsMinLevel ? (accessCtrl.easyShortcutsMinLevel - 1) : 3;
            
            // 🔧 DEBUG: Mostrar comparación
            console.log(`[NrmliZer-EasyShortCuts] 🔍 Comparación: userRank=${rank} (Nivel ${rank+1}) >= minRank=${minRank} (Nivel ${minRank+1})? ${rank >= minRank}`);

            if (whitelisted || rank >= minRank) {
                if (!whitelisted) console.log(`[NrmliZer-EasyShortCuts] ✅ Usuario Nivel ${rank + 1} detectado. Habilitando EasyShortCuts.`);
                initEasyShortCutsModule();
            } else {
                console.log(`[NrmliZer-EasyShortCuts] ⛔ Usuario Nivel ${rank + 1}. EasyShortCuts requiere Nivel ${minRank + 1}+. Módulo desactivado.`);
            }
        } else {
            // Reintentar si no está listo
            console.log('[NrmliZer-EasyShortCuts] ⏳ Esperando a Waze/User...');
            setTimeout(checkRankAndInitEasyShortCuts, 1000);
        }
    }

    // =====================================================================
    //  MÓDULO EASY SHORTCUTS (Integrado)
    //  Restringido a Nivel 4+
    // =====================================================================

    function initEasyShortCutsModule() {
        console.log('[NrmliZer] Inicializando módulo EasyShortCuts...');
        '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
            if (roadType === 18) continue; // tipo 18: vía privada (ignorar giros peligrosos)
            if (roadType === 20) continue; // tipo 20: parking (ignorar giros peligrosos)

            // 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 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_GREEN = 'ESC_School_WithZone';
            const LAYER_RED = 'ESC_School_NoZone';
            let layerGreenExists = false;
            let layerRedExists = false;

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

            // Función para verificar si hay una zona escolar cerca (permanentHazards)
            function hasSchoolZoneNearby(venueGeometry) {
                try {
                    if (!venueGeometry || typeof W === 'undefined' || !W.model) {
                        return false;
                    }

                    let centerX, centerY;
                    if (venueGeometry.getCentroid) {
                        const centroid = venueGeometry.getCentroid();
                        // Las coordenadas del venue están en EPSG:4326 (grados), necesitamos convertir a EPSG:3857 (metros)
                        const transformed = new OpenLayers.LonLat(centroid.x, centroid.y).transform('EPSG:4326', 'EPSG:3857');
                        centerX = transformed.lon;
                        centerY = transformed.lat;
                    } else if (venueGeometry.coordinates) {
                        if (venueGeometry.type === 'Point') {
                            const transformed = new OpenLayers.LonLat(venueGeometry.coordinates[0], venueGeometry.coordinates[1]).transform('EPSG:4326', 'EPSG:3857');
                            centerX = transformed.lon;
                            centerY = transformed.lat;
                        } else if (venueGeometry.type === 'Polygon') {
                            const coords = venueGeometry.coordinates[0];
                            let sumX = 0, sumY = 0;
                            for (const c of coords) {
                                sumX += c[0];
                                sumY += c[1];
                            }
                            const avgX = sumX / coords.length;
                            const avgY = sumY / coords.length;
                            const transformed = new OpenLayers.LonLat(avgX, avgY).transform('EPSG:4326', 'EPSG:3857');
                            centerX = transformed.lon;
                            centerY = transformed.lat;
                        }
                    }

                    if (!centerX || !centerY) return false;

                    const searchRadius = 100; // 100 metros
                    
                    // DEBUG: Mostrar que hay en W.model una vez
                    if (window.__PLN_DECISION_DEBUG_ON && !window._schoolZoneModelDebug) {
                        window._schoolZoneModelDebug = true;
                        console.log('[SchoolZone DEBUG] Keys de W.model:', Object.keys(W.model));
                        if (W.model.permanentHazards) {
                            console.log('[SchoolZone DEBUG] permanentHazards existe:', W.model.permanentHazards);
                            const hazards = Object.values(W.model.permanentHazards.objects || {});
                            console.log('[SchoolZone DEBUG] Total permanentHazards:', hazards.length);
                            
                            // Agrupar por tipo para ver qué tipos existen
                            const typeGroups = {};
                            for (const h of hazards) {
                                const t = h.attributes?.type;
                                if (!typeGroups[t]) {
                                    typeGroups[t] = [];
                                }
                                typeGroups[t].push(h);
                            }
                            
                            console.log('[SchoolZone DEBUG] Tipos de hazards encontrados:');
                            for (const [type, items] of Object.entries(typeGroups)) {
                                console.log(`  Type ${type}: ${items.length} hazards`);
                                // Mostrar ejemplo del primero
                                if (items[0]) {
                                    console.log(`    Ejemplo:`, items[0].attributes);
                                }
                            }
                        }
                    }

                    // Intentar buscar en permanentHazards
                    if (W.model.permanentHazards && W.model.permanentHazards.objects) {
                        const hazards = Object.values(W.model.permanentHazards.objects);
                        const schoolZoneHazards = hazards.filter(h => h.attributes?.type === 9);
                        
                        if (window.__PLN_DECISION_DEBUG_ON) {
                            console.log(`[SchoolZone DEBUG] Total hazards: ${hazards.length}, Type 9: ${schoolZoneHazards.length}`);
                            
                            if (schoolZoneHazards.length > 0) {
                                console.log('[SchoolZone DEBUG] Primer hazard type 9:', schoolZoneHazards[0]);
                            }
                        }
                        
                        for (const hazard of hazards) {
                            const attrs = hazard.attributes;
                            if (!attrs) continue;
                            
                            // Verificar si es una zona escolar (type = 9 en WME)
                            const hazardType = attrs.type;
                            const isSchoolZone = hazardType === 9;
                            
                            if (!isSchoolZone) continue;

                            const hazGeometry = hazard.getOLGeometry ? hazard.getOLGeometry() : null;
                            if (!hazGeometry) {
                                if (window.__PLN_DECISION_DEBUG_ON) console.log('[SchoolZone DEBUG] Hazard sin geometría:', hazard);
                                continue;
                            }

                            let hazCenterX, hazCenterY;
                            if (hazGeometry.getCentroid) {
                                const hazCenter = hazGeometry.getCentroid();
                                hazCenterX = hazCenter.x;
                                hazCenterY = hazCenter.y;
                                if (window.__PLN_DECISION_DEBUG_ON) console.log('[SchoolZone DEBUG] Hazard centroid:', hazCenterX, hazCenterY);
                            } else {
                                if (window.__PLN_DECISION_DEBUG_ON) console.log('[SchoolZone DEBUG] Hazard geometry sin getCentroid:', hazGeometry);
                                continue;
                            }

                            const distance = Math.sqrt(
                                Math.pow(centerX - hazCenterX, 2) + 
                                Math.pow(centerY - hazCenterY, 2)
                            );
                            
                            if (window.__PLN_DECISION_DEBUG_ON) console.log(`[SchoolZone DEBUG] Distancia al hazard: ${Math.round(distance)}m (limit: ${searchRadius}m)`);

                            if (distance <= searchRadius) {
                                if (window.__PLN_DECISION_DEBUG_ON) console.log('[SchoolZone] ✅ Encontre permanentHazard escolar a', Math.round(distance), 'm');
                                return true;
                            }
                        }
                    }
                    
                    // FALLBACK: Buscar en segmentos con flags
                    if (W.model.segments && W.model.segments.objects) {
                        const segments = Object.values(W.model.segments.objects);
                        
                        for (const segment of segments) {
                            const attrs = segment.attributes;
                            if (!attrs) continue;

                            let hasSchoolZoneFlag = false;
                            
                            // Verificar flags como objetos
                            if (attrs.flags && typeof attrs.flags === 'object') {
                                hasSchoolZoneFlag = attrs.flags.nearSchool === true || attrs.flags.schoolZone === true;
                            }
                            
                            // Verificar fwdFlags/revFlags como objetos
                            if (!hasSchoolZoneFlag && attrs.fwdFlags && typeof attrs.fwdFlags === 'object') {
                                hasSchoolZoneFlag = attrs.fwdFlags.nearSchool === true;
                            }
                            if (!hasSchoolZoneFlag && attrs.revFlags && typeof attrs.revFlags === 'object') {
                                hasSchoolZoneFlag = attrs.revFlags.nearSchool === true;
                            }
                            
                            // Verificar fwdFlags/revFlags como bitmask (número)
                            // Bit 5 (valor 32) = nearSchool
                            if (!hasSchoolZoneFlag && typeof attrs.fwdFlags === 'number') {
                                hasSchoolZoneFlag = (attrs.fwdFlags & 32) !== 0;
                            }
                            if (!hasSchoolZoneFlag && typeof attrs.revFlags === 'number') {
                                hasSchoolZoneFlag = (attrs.revFlags & 32) !== 0;
                            }
                            
                            if (!hasSchoolZoneFlag) continue;

                            const segGeometry = segment.getOLGeometry ? segment.getOLGeometry() : null;
                            if (!segGeometry) continue;

                            let segCenterX, segCenterY;
                            if (segGeometry.getCentroid) {
                                const segCenter = segGeometry.getCentroid();
                                segCenterX = segCenter.x;
                                segCenterY = segCenter.y;
                            } else {
                                continue;
                            }

                            const distance = Math.sqrt(
                                Math.pow(centerX - segCenterX, 2) + 
                                Math.pow(centerY - segCenterY, 2)
                            );

                            if (distance <= searchRadius) {
                                if (window.__PLN_DECISION_DEBUG_ON) console.log('[SchoolZone] ✅ Encontre segmento escolar a', Math.round(distance), 'm');
                                return true;
                            }
                        }
                    }

                    return false;
                } catch (e) {
                    console.warn('[SchoolZone] Error checking nearby school zone:', e);
                    return false;
                }
            }

            async function ensureLayers() {
                if (!wmeSDK) return;
                
                // Habilitar la capa de zonas escolares de WME (para que se vea el highlight naranja)
                try {
                    if (typeof W !== 'undefined' && W.map && W.map.getLayerByName) {
                        const schoolZoneLayer = W.map.getLayerByName('school_zone');
                        if (schoolZoneLayer) {
                            schoolZoneLayer.setVisibility(true);
                            if (window.__PLN_DECISION_DEBUG_ON) console.log('[SchoolZone] Capa WME school_zone habilitada');
                        }
                    }
                } catch (e) {
                    console.warn('[SchoolZone] No se pudo habilitar capa WME:', e.message);
                }
                
                // Capa VERDE para escuelas CON zona escolar
                if (!layerGreenExists) {
                    try {
                        await wmeSDK.Map.addLayer({
                            layerName: LAYER_GREEN,
                            title: 'Escuelas CON Zona',
                            type: 'feature',
                            visible: isActive,
                            zIndex: 321,
                            styleRules: [{
                                style: {
                                    strokeColor: '#00FF00',  // Verde brillante (lime)
                                    strokeOpacity: 1.0,
                                    strokeWidth: 6,
                                    fill: true,
                                    fillColor: '#4CAF50',
                                    fillOpacity: 0.5,
                                    pointRadius: 15,
                                    label: '✅ CON Zona Escolar',
                                    fontColor: '#1B5E20',
                                    labelOutlineColor: '#FFFFFF',
                                    labelOutlineWidth: 3,
                                    fontSize: '12px',
                                    fontWeight: 'bold',
                                    labelYOffset: -22
                                }
                            }]
                        });
                        layerGreenExists = true;
                        if (window.__PLN_DECISION_DEBUG_ON) console.log('[SchoolZone] Capa VERDE creada');
                    } catch (e) {
                        console.warn('[SchoolZone] Error creando capa verde:', e.message);
                    }
                }

                // Capa ROJA para escuelas SIN zona escolar
                if (!layerRedExists) {
                    try {
                        await wmeSDK.Map.addLayer({
                            layerName: LAYER_RED,
                            title: 'Escuelas SIN Zona',
                            type: 'feature',
                            visible: isActive,
                            zIndex: 322,
                            styleRules: [{
                                style: {
                                    strokeColor: '#D32F2F',
                                    strokeOpacity: 1.0,
                                    strokeWidth: 8,
                                    fill: true,
                                    fillColor: '#FFCDD2',
                                    fillOpacity: 0.6,
                                    pointRadius: 15,
                                    label: '⚠️ SIN Zona Escolar',
                                    fontColor: '#B71C1C',
                                    labelOutlineColor: '#FFFFFF',
                                    labelOutlineWidth: 4,
                                    fontSize: '13px',
                                    fontWeight: 'bold',
                                    labelYOffset: -22
                                }
                            }]
                        });
                        layerRedExists = true;
                        if (window.__PLN_DECISION_DEBUG_ON) console.log('[SchoolZone] Capa ROJA creada');
                    } catch (e) {
                        console.warn('[SchoolZone] Error creando capa roja:', e.message);
                    }
                }
            }

            async function highlightSchools() {
                console.log('[DEBUG SchoolZone] 1. Iniciando highlightSchools');
                if (!wmeSDK) {
                    console.error('[DEBUG SchoolZone] ERROR: wmeSDK no esta disponible');
                    return;
                }
                
                await ensureLayers();
                
                // Limpiar marcas anteriores
                try { await wmeSDK.Map.removeAllFeaturesFromLayer({ layerName: LAYER_GREEN }); } catch (e) {}
                try { await wmeSDK.Map.removeAllFeaturesFromLayer({ layerName: LAYER_RED }); } catch (e) {}

                logInfo('--- BUSCANDO INSTITUCIONES EDUCATIVAS ---');

                let venues = [];
                if (wmeSDK.DataModel.Venues && typeof wmeSDK.DataModel.Venues.getAll === 'function') {
                    venues = await wmeSDK.DataModel.Venues.getAll();
                } else if (typeof W !== 'undefined' && W.model && W.model.venues) {
                    venues = W.model.venues.getObjectArray();
                }
                console.log('[DEBUG SchoolZone] Total venues:', venues.length);

                const featuresGreen = [];
                const featuresRed = [];
                let schoolsWithZone = 0;
                let schoolsWithoutZone = 0;

                for (const venue of venues) {
                    const cats = venue.attributes ? venue.attributes.categories : venue.categories;
                    if (!cats || cats.length === 0) continue;

                    const isSchool = cats.some(c => SCHOOL_CATEGORIES.includes((c || '').toUpperCase()));
                    if (!isSchool) continue;

                    const venueName = venue.attributes?.name || venue.name || 'sin nombre';

                    let geometry = null;
                    if (typeof venue.getOLGeometry === 'function') {
                        geometry = venue.getOLGeometry();
                    } else if (venue.geometry) {
                        geometry = venue.geometry;
                    }

                    if (!geometry) continue;

                    const hasZone = hasSchoolZoneNearby(geometry);
                    
                    let featureGeo = null;
                    if (geometry.getType && typeof geometry.getType === 'function') {
                        const type = geometry.getType();
                        const coords = geometry.getCoordinates();
                        if (type === 'Point') {
                            // Para puntos, crear un cuadro de 100x100 metros (50m en cada dirección)
                            const [lon, lat] = coords;
                            // Convertir a Web Mercator para trabajar en metros
                            const transformed = new OpenLayers.LonLat(lon, lat).transform('EPSG:4326', 'EPSG:3857');
                            const centerX = transformed.lon;
                            const centerY = transformed.lat;
                            
                            // Crear cuadro de 100x100m (50m de radio)
                            const radius = 50; // 50 metros
                            const box3857 = [
                                [centerX - radius, centerY - radius],
                                [centerX + radius, centerY - radius],
                                [centerX + radius, centerY + radius],
                                [centerX - radius, centerY + radius],
                                [centerX - radius, centerY - radius] // Cerrar el polígono
                            ];
                            
                            // Convertir de vuelta a EPSG:4326 para la capa
                            const box4326 = box3857.map(coord => {
                                const pt = new OpenLayers.LonLat(coord[0], coord[1]).transform('EPSG:3857', 'EPSG:4326');
                                return [pt.lon, pt.lat];
                            });
                            
                            featureGeo = { type: 'Polygon', coordinates: [box4326] };
                        } else if (type === 'Polygon') {
                            featureGeo = { type: 'Polygon', coordinates: coords };
                        } else if (type === 'MultiPolygon') {
                            featureGeo = { type: 'MultiPolygon', coordinates: coords };
                        }
                    } else {
                        featureGeo = geometry;
                    }

                    if (!featureGeo) continue;

                    const feature = {
                        type: 'Feature',
                        id: 'school_' + (venue.attributes ? venue.attributes.id : venue.id),
                        geometry: featureGeo,
                        properties: { name: venueName }
                    };

                    if (hasZone) {
                        schoolsWithZone++;
                        featuresGreen.push(feature);
                        if (window.__PLN_DECISION_DEBUG_ON) console.log('[SchoolZone] ✅ CON zona:', venueName);
                    } else {
                        schoolsWithoutZone++;
                        featuresRed.push(feature);
                        if (window.__PLN_DECISION_DEBUG_ON) console.log('[SchoolZone] ⚠️ SIN zona:', venueName);
                    }
                }

                console.log(`[SchoolZone] Resumen: ✅ ${schoolsWithZone} con zona | ⚠️ ${schoolsWithoutZone} sin zona`);

                if (featuresGreen.length > 0) {
                    await wmeSDK.Map.addFeaturesToLayer({
                        layerName: LAYER_GREEN,
                        features: featuresGreen
                    });
                }

                if (featuresRed.length > 0) {
                    await wmeSDK.Map.addFeaturesToLayer({
                        layerName: LAYER_RED,
                        features: featuresRed
                    });
                }

                logInfo(`Marcadas ${featuresGreen.length + featuresRed.length} instituciones: ✅ ${schoolsWithZone} con zona | ⚠️ ${schoolsWithoutZone} sin zona`);
            }

            async function clearHighlights() {
                if (wmeSDK) {
                    try { await wmeSDK.Map.removeAllFeaturesFromLayer({ layerName: LAYER_GREEN }); } catch (e) {}
                    try { await wmeSDK.Map.removeAllFeaturesFromLayer({ layerName: LAYER_RED }); } catch (e) {}
                }
            }

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

            function setSDK(sdk) { wmeSDK = sdk; }

            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();

    };


    console.log('[WME PLN] 🏁 Script loaded. Starting initialization check...');

    // Iniciar verificación de EasyShortCuts con espera inteligente
    // Espera hasta que la configuración del servidor esté lista o pase un tiempo máximo (15s)
    let configAttempts = 0;
    const maxConfigAttempts = 15;
    
    const waitForConfigAndInit = function() {
        configAttempts++;
        const accessCtrl = window.PLN_ACCESS_CONTROL || (typeof PLN_ACCESS_CONTROL !== 'undefined' ? PLN_ACCESS_CONTROL : {});
        
        // Criterio de éxito: easyShortcutsMinLevel está definido en la config
        const configLoaded = accessCtrl && typeof accessCtrl.easyShortcutsMinLevel !== 'undefined';
        
        if (configLoaded || configAttempts >= maxConfigAttempts) {
            
            if (configLoaded) {
                 console.log(`[NrmliZer-EasyShortCuts] ✅ Configuración de servidor detectada en intento ${configAttempts}. loading...`);
            } else {
                 console.log('[NrmliZer-EasyShortCuts] ⚠️ Timeout esperando config del servidor (15s). Usando valores por defecto/locales.');
            }
            
            checkRankAndInitEasyShortCuts();
        } else {
            // Aún no está la config, esperar 1s más
            // console.log(`[NrmliZer-EasyShortCuts] ⏳ Esperando config del servidor (Intento ${configAttempts}/${maxConfigAttempts})...`);
            setTimeout(waitForConfigAndInit, 1000);
        }
    };
    
    // Iniciar el ciclo de espera (primer chequeo en 1s)
    setTimeout(waitForConfigAndInit, 1000);

    // [PLN] Force user tracking execution
    setTimeout(function() {
        if (typeof plnForceTrackUser === 'function') {
            plnForceTrackUser();
        }
    }, 2000);

})();