WME Places Name Normalizer

Normaliza nombres de lugares en Waze Map Editor (WME) por ahora solo con reglas de Colombia

// ==UserScript==
// @name         WME Places Name Normalizer
// @namespace    https://gf.qytechs.cn/en/users/mincho77
// @version      3.1
// @description  Normaliza nombres de lugares en Waze Map Editor (WME) por ahora solo con reglas de Colombia
// @author       mincho77
// @match        https://www.waze.com/*editor*
// @match      https://beta.waze.com/*user/editor*
// @grant        GM_xmlhttpRequest
// @connect      api.languagetool.org
// @connect      *
// @grant        unsafeWindow
// @license      MIT
// @run-at       document-end
// ==/UserScript==
/*global W*/

(() => {
    "use strict";
    try {


   // Insertar estilos globales en el <head>
const styles = `
<style>
    /* Estilos para los botones */
    .apply-suggestion-btn {
        background-color: #4CAF50;
        color: white;
        border: none;
        padding: 8px 12px;
        border-radius: 4px;
        cursor: pointer;
        font-size: 14px;
        font-weight: bold;
        transition: background-color 0.3s ease;
    }
    .apply-suggestion-btn:hover {
        background-color: #45a049;
    }

    #apply-changes-btn {
        background-color: #28a745;
        color: white;
        border: none;
        padding: 10px 15px;
        border-radius: 4px;
        cursor: pointer;
        font-size: 16px;
        font-weight: bold;
        display: flex;
        align-items: center;
        gap: 8px;
    }
    #apply-changes-btn:hover {
        background-color: #218838;
    }

    #cancel-btn {
        background-color: #dc3545;
        color: white;
        border: none;
        padding: 10px 15px;
        border-radius: 4px;
        cursor: pointer;
        font-size: 16px;
        font-weight: bold;
        display: flex;
        align-items: center;
        gap: 8px;
    }
    #cancel-btn:hover {
        background-color: #c82333;
    }
</style>
`;
document.head.insertAdjacentHTML('beforeend', styles);

    // Capturar todos los eventos de drag & drop a nivel de <body>
   // Agregar al inicio del script, justo después del "use strict"
    // Prevenir comportamiento por defecto de drag & drop a nivel global
    document.addEventListener("dragover", function(e) {
        const dropZone = document.getElementById("drop-zone");
        if (e.target === dropZone || dropZone?.contains(e.target)) {
            return; // Permitir el drop en la zona designada
        }
        e.preventDefault();
        e.stopPropagation();
    }, { passive: false }); // Agrega { passive: false }

    document.addEventListener("drop", function(e) {
        const dropZone = document.getElementById("drop-zone");
        if (e.target === dropZone || dropZone?.contains(e.target)) {
            return; // Permitir el drop en la zona designada
        }
        e.preventDefault();
        e.stopPropagation();
    }, { passive: false }); // Agrega { passive: false }

//**************************************************************************
//Nombre: evaluarOrtografiaConTildes
//Fecha modificación: 2025-04-02
//Autor: mincho77
//Entradas: name (string) - Nombre del lugar
//Salidas: objeto con errores detectados
//Descripción:
// Evalúa palabra por palabra si falta una tilde en las palabras que lo requieren,
// según las reglas del español. Primero normaliza el nombre y luego verifica
// si las palabras necesitan una tilde.
//**************************************************************************
function evaluarOrtografiaConTildes(name) {
  // Si el nombre está vacío, retornar inmediatamente una promesa resuelta
  if (!name) {
    return Promise.resolve({
        hasSpellingWarning: false,
        spellingWarnings: []
    });
}


    const palabras = name.trim().split(/\s+/);
    const spellingWarnings = [];

    console.log(`[evaluarOrtografiaConTildes] Verificando ortografía de: ${name}`);

    palabras.forEach((palabra, index) => {
        // Normalizar la palabra antes de cualquier verificación
        let normalizada = normalizePlaceNameOnly(palabra);

        // Ignorar palabras con "&" o que sean emoticonos
        if (/^[A-Za-z]&[A-Za-z]$/.test(normalizada) || /^[\u263a-\u263c\u2764\u1f600-\u1f64f\u1f680-\u1f6ff]+$/.test(normalizada)) {
            return; // No verificar ortografía
        }

        // Excluir palabras específicas como "y" o "Y"
        if (normalizada.toLowerCase() === "y" || /^\d+$/.test(normalizada) || normalizada === "-") {
            return; // Ignorar
        }

        // Verificar si la palabra está en la lista de excluidas
        if (excludeWords.some(w => w.toLowerCase() === normalizada.toLowerCase())) {
            return; // Ignorar palabra excluida
        }

        // Validar que no tenga más de una tilde
        const cantidadTildes = (normalizada.match(/[áéíóú]/g) || []).length;
        if (cantidadTildes > 1) {
            spellingWarnings.push({
                original: palabra,
                sugerida: null, // No hay sugerencia válida
                tipo: "Error de tildes",
                posicion: index
            });
            return;
        }

        // Verificar ortografía usando la API de LanguageTool
        checkSpellingWithAPI(normalizada).then(errores => {
            errores.forEach(error => {
                spellingWarnings.push({
                    original: error.palabra,
                    sugerida: error.sugerencia,
                    tipo: "LanguageTool",
                    posicion: index
                });
            });
        }).catch(err => {
            console.error("Error al verificar ortografía con LanguageTool:", err);
        });
    });

    return {
        hasSpellingWarning: spellingWarnings.length > 0,
        spellingWarnings
    };
}

// Función auxiliar para corregir tildes
function corregirTilde(palabra) {
    // Implementar lógica para corregir tildes según las reglas del español
    // Por ejemplo: "medellin" → "Medellín"
    return palabra; // Retornar la palabra corregida
}

    //**************************************************************************
    //Nombre: toggleSpinner
    //Fecha modificación: 2025-03-31
    //Autor: mincho77
    //Entradas:
    //   show (boolean) - true para mostrar el spinner, false para ocultarlo
    //   message (string, opcional) - mensaje personalizado a mostrar junto al spinner
    //Salidas: ninguna (modifica el DOM)
    //Prerrequisitos: debe existir el estilo CSS del spinner en el documento
    //Descripción:
    //   Muestra u oculta un indicador visual de carga con un mensaje opcional.
    //   El spinner usa un emoji de reloj de arena (⏳) con animación de rotación
    //   para indicar que el proceso está en curso.
    //**************************************************************************
    function toggleSpinner(show, message = 'Revisando ortografía...', progress = null) {
        let existingSpinner = document.querySelector('.spinner-overlay');
        if (existingSpinner) {
            if (show) {
                // Actualizar el mensaje y el progreso si el spinner ya existe
                const spinnerMessage = existingSpinner.querySelector('.spinner-message');
                spinnerMessage.innerHTML = `
                    ${message}
                    ${progress !== null ? `<br><strong>${progress}% completado</strong>` : ''}
                `;
            } else {
                existingSpinner.remove(); // Ocultar el spinner
            }
            return;
        }
    
        if (show) {
            const spinner = document.createElement('div');
            spinner.className = 'spinner-overlay';
            spinner.innerHTML = `
                <div class="spinner-content">
                    <div class="spinner-icon">⏳</div>
                    <div class="spinner-message">
                        ${message}
                        ${progress !== null ? `<br><strong>${progress}% completado</strong>` : ''}
                    </div>
                </div>
            `;
            document.body.appendChild(spinner);
        }
    }

    // Agregar los estilos CSS necesarios
    const spinnerStyles = `
    <style>
    .spinner-overlay {
        position: fixed;
        top: 0;
        left: 0;
        width: 100%;
        height: 100%;
        background: rgba(0, 0, 0, 0.5);
        display: flex;
        justify-content: center;
        align-items: center;
        z-index: 10000;
    }

    .spinner-content {
        background: white;
        padding: 20px;
        border-radius: 8px;
        text-align: center;
        box-shadow: 0 2px 10px rgba(0,0,0,0.3);
    }

    .spinner-icon {
        font-size: 24px;
        margin-bottom: 10px;
        animation: spin 1s linear infinite; /* Aseguramos que la animación esté activa */
        display: inline-block;
    }

    .spinner-message {
        color: #333;
        font-size: 14px;
    }

    @keyframes spin {
        0% { transform: rotate(0deg); }
        100% { transform: rotate(360deg); }
    }
    </style>`;

    // Insertar los estilos al inicio del documento
    document.head.insertAdjacentHTML('beforeend', spinnerStyles);

    if (!Array.prototype.flat)
    {
        Array.prototype.flat = function(depth = 1)
        {
            return this.reduce(function (flat, toFlatten)
            {
                return flat.concat(Array.isArray(toFlatten) ? toFlatten.flat(depth - 1) : toFlatten);
            }, []);
        };
    }
    const SCRIPT_NAME = "PlacesNameNormalizer";
    const VERSION = "3.1";
    let placesToNormalize = [];
    let excludeWords = [];
    let maxPlaces = 50;
    let normalizeArticles = true;

    // Expresión regular para detectar siglas (por ejemplo, "S.a.s", "L.T.D.A")
    const siglaRegex = /^[A-Za-z](\.[A-Za-z])+\.?$/;



    
    //**************************************************************************
    //Nombre: waitForSidebar
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas:
    //   retries (número, opcional) – Número máximo de intentos de verificación del sidebar (default: 20).
    //   delay (número, opcional) – Tiempo en milisegundos entre intentos (default: 1000ms).
    //Salidas:
    //   Promesa que se resuelve con el elemento del sidebar si se encuentra, o se rechaza si no se encuentra después de los intentos.
    //Descripción:
    //   Esta función espera a que el DOM cargue completamente el elemento con ID "sidebar".
    //   Realiza múltiples intentos con intervalos definidos, y resuelve la promesa cuando el sidebar esté disponible.
    //   Es útil para asegurarse de que el entorno de WME esté completamente cargado antes de continuar.
    //**************************************************************************
    function waitForSidebar(retries = 20, delay = 1000) {
        return new Promise((resolve, reject) => {
            const check = (attempt = 1) => {
                const sidebar = document.querySelector("#sidebar");
                if (sidebar) {
                    console.log("✅ Sidebar disponible.");
                    resolve(sidebar);
                } else if (attempt <= retries) {
                    console.warn(`⚠️ Sidebar no disponible aún. Reintentando... (${attempt})`);
                    setTimeout(() => check(attempt + 1), delay);
                } else {
                    reject("❌ Sidebar no disponible después de múltiples intentos.");
                }
            };
            check();
        });
    }
    //**************************************************************************
    //Nombre: initializeExcludeWords
    //Fecha modificación: 2025-03-31
    //Autor: mincho77
    //Entradas: Ninguna
    //Salidas: Ninguna
    //Prerrequisitos: excludeWords debe estar declarado globalmente.
    //Descripción: Inicializa la lista de palabras excluidas desde localStorage sin borrar entradas ya existentes.
    //**************************************************************************
    function initializeExcludeWords() {
        const saved = JSON.parse(localStorage.getItem("excludeWords")) || [];

        // Combinar con las actuales sin duplicar
        const merged = [...new Set([...saved, ...excludeWords])].sort((a, b) => a.localeCompare(b));

        // Solo guardar si hay diferencias
        const originalString = JSON.stringify(saved.sort());
        const newString = JSON.stringify(merged);

        if (originalString !== newString) {
            localStorage.setItem("excludeWords", newString);
            console.log(`[initializeExcludeWords] 💾 excludeWords actualizado con ${merged.length} palabras.`);
        } else {
            console.log(`[initializeExcludeWords] 🟢 Sin cambios en excludeWords.`);
        }

        // Actualizar variable global
        excludeWords = merged;
    }

    //Modulo de Ortografía
   



   

    // Insertar los estilos al inicio del documento
    document.head.insertAdjacentHTML('beforeend', spinnerStyles);

    if (!Array.prototype.flat)
    {
        Array.prototype.flat = function(depth = 1)
        {
            return this.reduce(function (flat, toFlatten)
            {
                return flat.concat(Array.isArray(toFlatten) ? toFlatten.flat(depth - 1) : toFlatten);
            }, []);
        };
    }
   

    
    //Modulo de Ortografía
    //**************************************************************************
    //Nombre: getUbicacionAcento
    //Fecha modificación: 2025-03-27
    //Autor: mincho77
    //Entradas: palabra (string) - Palabra en español a evaluar
    //Salidas: Posición del acento (última, penúltima, antepenúltima) o null si no aplica
    //Prerrequisitos si existen: Ninguno
    //Descripción: Determina la posición silábica del acento en una palabra según la ortografía del español.
    //**************************************************************************
    function getUbicacionAcento(palabra) {
        const original = palabra;
        const normalized = palabra.normalize("NFD").replace(/[\u0300-\u036f]/g, ""); // Remove diacritics
        const silabas = normalized
            .replace(/[^aeiou]/gi, "")
            .match(/[aeiou]+/gi);

        if (!silabas || silabas.length === 0) return null;

        const conTilde = silabas.findIndex((s, i) => {
            const originalSyllable = original.slice(
                normalized.indexOf(s),
                normalized.indexOf(s) + s.length
            );
            return /[áéíóú]/.test(originalSyllable);
        });

        if (conTilde !== -1) {
            const posicion = silabas.length - 1 - conTilde;
            if (posicion === 0) return "última";
            if (posicion === 1) return "penúltima";
            if (posicion === 2) return "antepenúltima";
            return "otras";
        }

        return "sin tilde";
    }

    //**************************************************************************
    //Nombre: checkSpellingWithAPI
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas: text (string) – Texto a evaluar ortográficamente.
    //Salidas: Promise – Resuelve con lista de errores ortográficos detectados.
    //Prerrequisitos: Requiere permisos GM_xmlhttpRequest y @connect a api.languagetool.org
    //Descripción:
    // Consulta la API de LanguageTool para verificar ortografía del texto.
    //**************************************************************************
    function checkSpellingWithAPI(text) {
        return new Promise((resolve, reject) => {
            GM_xmlhttpRequest({
                method: "POST",
                url: "https://api.languagetool.org/v2/check",
                headers: {
                    "Content-Type": "application/x-www-form-urlencoded"
                },
                data: `language=es&text=${encodeURIComponent(text)}`,
                onload: function(response) {
                    if (response.status === 200) {
                        const result = JSON.parse(response.responseText);
                        const errores = result.matches
                        .filter(match => match.rule.issueType === "misspelling")
                        .map(match => ({
                            palabra: match.context.text.substring(match.context.offset, match.context.offset + match.context.length),
                            sugerencia: match.replacements.length > 0 ? match.replacements[0].value : "(sin sugerencia)"
                        }));
                        resolve(errores);
                    } else {
                        reject("❌ Error en respuesta de LanguageTool");
                    }
                },
                onerror: function(err) {
                    reject("❌ Error de red al contactar LanguageTool");
                }
            });
        });
    }

    window.checkSpellingWithAPI = checkSpellingWithAPI;

// Combinar la lógica de checkSpelling
function checkSpelling(text) {
    return new Promise((resolve, reject) => {
        GM_xmlhttpRequest({
            method: "POST",
            url: "https://api.languagetool.org/v2/check",
            data: `text=${encodeURIComponent(text)}&language=es`,
            headers: {
                "Content-Type": "application/x-www-form-urlencoded"
            },
            onload: function (response) {
                if (response.status === 200) {
                    try {
                        const data = JSON.parse(response.responseText);
                        const errores = data.matches
                            .filter(match => match.rule.issueType === "misspelling")
                            .map(match => ({
                                palabra: match.context.text.substring(match.context.offset, match.context.offset + match.context.length),
                                sugerencia: match.replacements.length > 0 ? match.replacements[0].value : "(sin sugerencia)"
                            }));
                        resolve(errores);
                    } catch (err) {
                        reject(err);
                    }
                } else {
                    reject(`Error HTTP: ${response.status}`);
                }
            },
            onerror: function (err) {
                reject(err);
            }
        });
    });
}   

    //**************************************************************************
    //Nombre: validarWordSpelling
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas: palabra (string) - Palabra en español a validar ortográficamente
    //Salidas: true si cumple reglas ortográficas básicas, false si no
    //Descripción:
    // Evalúa si una palabra tiene el uso correcto de tilde o si le falta una tilde
    // según las reglas del español: esdrújulas siempre con tilde, agudas con tilde
    // si terminan en n, s o vocal, y llanas con tildse si NO terminan en n, s o vocal.
    // Se asegura que solo haya una tilde por palabra.
    //**************************************************************************
    function validarWordSpelling(palabra) {
        if (!palabra) return false;

        // Ignorar siglas con formato X&X
        if (/^[A-Za-z]&[A-Za-z]$/.test(palabra)) return true;

        // Si la palabra es un número, no necesita validación
        if (/^\d+$/.test(palabra)) return true;

        const tieneTilde = /[áéíóú]/.test(palabra);
        const cantidadTildes = (palabra.match(/[áéíóú]/g) || []).length;
        if (cantidadTildes > 1) return false; // Solo se permite una tilde

        const silabas = palabra.normalize("NFD").replace(/[^aeiouAEIOU\u0300-\u036f]/g, "").match(/[aeiouáéíóú]+/gi);
        if (!silabas || silabas.length === 0) return false;

        const totalSilabas = silabas.length;
        const ultimaLetra = palabra.slice(-1).toLowerCase();

        let tipo = "";
        if (totalSilabas >= 3 && /[áéíóú]/.test(palabra)) {
            tipo = "esdrújula";
        } else if (totalSilabas >= 2) {
            const penultimaSilaba = silabas[totalSilabas - 2];
            if (/[áéíóú]/.test(penultimaSilaba)) tipo = "grave";
        }

        if (!tipo) tipo = /[áéíóú]/.test(silabas[totalSilabas - 1]) ? "aguda" : "sin tilde";

        if (tipo === "esdrújula") return tieneTilde;
        if (tipo === "aguda") {
            return (/[nsáéíóúaeiou]$/.test(ultimaLetra) && tieneTilde) ||
                (!/[nsáéíóúaeiou]$/.test(ultimaLetra) && !tieneTilde);
        }
        if (tipo === "grave") {
            return (!/[nsáéíóúaeiou]$/.test(ultimaLetra) && tieneTilde) ||
                (/[nsáéíóúaeiou]$/.test(ultimaLetra) && !tieneTilde);
        }

        return true;
    }



    ///**************************************************************************
    //Nombre: separarSilabas
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas: palabra (string) – Palabra en español.
    //Salidas: array de sílabas (aproximado).
    //Descripción:
    //   Separa una palabra en sílabas usando reglas heurísticas.
    //   Esta versión simplificada considera diptongos y combinaciones comunes.
    //**************************************************************************
    function separarSilabas(palabra) {
        // Normaliza y quita acentos para facilitar la segmentación
        const limpia = palabra.normalize("NFD").replace(/[\u0300-\u036f]/g, "").toLowerCase();

        // Divide por vocales agrupadas como aproximación de sílabas
        const silabas = limpia.match(/[bcdfghjklmnñpqrstvwxyz]*[aeiou]{1,2}[bcdfghjklmnñpqrstvwxyz]*/g);

        return silabas || [palabra]; // fallback si no separa nada
    }

    //**************************************************************************
    //Nombre: clasificarPalabra
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas: silabas (array) – Arreglo con las sílabas de la palabra.
    //Salidas: string – 'aguda', 'grave' (llana), 'esdrújula'.
    //Descripción:
    //   Determina el tipo de palabra según el número de sílabas y la posición
    //   de la tilde (si existe).
    //**************************************************************************
    function clasificarPalabra(silabas) {
        const palabra = silabas.join("");
        const tieneTilde = /[áéíóúÁÉÍÓÚ]/.test(palabra);

        if (tieneTilde) {
            const posicionTilde = silabas.findIndex(s => /[áéíóúÁÉÍÓÚ]/.test(s));
            const posicionDesdeFinal = silabas.length - 1 - posicionTilde;

            if (posicionDesdeFinal === 0) return "aguda";
            if (posicionDesdeFinal === 1) return "grave";
            if (posicionDesdeFinal >= 2) return "esdrújula";
        }

        // Si no tiene tilde, asumimos que es:
        if (silabas.length === 1) return "aguda";
        if (silabas.length === 2) return "grave";
        return "grave"; // por convención para 3+ sin tilde
    }

    //**************************************************************************
    //Nombre: evaluarOrtografiaNombre
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas: name (string) - Nombre del lugar
    //Salidas: objeto con errores detectados
    //Descripción: Evalúa palabra por palabra si hay errores ortográficos o falta de tildes.
    // Ya no utiliza sugerencias automáticas para correcciones.
    //**************************************************************************
    function evaluarOrtografiaNombre(name) {
        if (!name) return {
            hasSpellingWarning: false,
            spellingWarnings: []
        };

        const checkOnlyTildes = document.getElementById("checkOnlyTildes")?.checked;
        const palabras = name.trim().split(/\s+/);
        const spellingWarnings = [];

        console.log(`[evaluarOrtografiaNombre] Verificando ortografía de: ${name}`);

        if (checkOnlyTildes) {
            palabras.forEach((palabra, index) => {
                // Verificar si la palabra está en la lista de exclusiones
                if (excludeWords.some(w => w.toLowerCase() === palabra.toLowerCase()) || /^\d+$/.test(palabra)) {
                    return; // Ignorar palabra excluida
                }

                if (!validarWordSpelling(palabra)) {
                    spellingWarnings.push({
                        original: palabra,
                        sugerida: null,
                        tipo: "Tilde",
                        posicion: index // Guardar la posición en la frase
                    });
                   
                }
            });

            return Promise.resolve({
                hasSpellingWarning: spellingWarnings.length > 0,
                spellingWarnings
            });
        } else {
            return checkSpellingWithAPI(name)
                .then(errores => {
                    errores.forEach(error => {
                        // Verificar si la palabra está en exclusiones
                        if (!excludeWords.some(w => w.toLowerCase() === error.palabra.toLowerCase())) {
                            spellingWarnings.push({
                                original: error.palabra,
                                sugerida: error.sugerencia,
                                tipo: "LanguageTool",
                                posicion: name.indexOf(error.palabra)
                            });
                        }
                    });
                    return {
                        hasSpellingWarning: spellingWarnings.length > 0,
                        spellingWarnings
                    };
                });
        }
    }
    
    //**************************************************************************
    //Nombre: handleImportList
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas: ninguna directa, depende del input file y checkbox en el DOM
    //Salidas: actualiza excludeWords en localStorage y visualmente
    //Prerrequisitos: existir un input file con id="importListInput" y checkbox con id="replaceExcludeListCheckbox"
    //Descripción: Importa una lista de palabras para excluir, opcionalmente reemplazando la lista actual
    //**************************************************************************
    function handleImportList() {
        const fileInput = document.getElementById("importListInput");
        const replaceCheckbox = document.getElementById("replaceExcludeListCheckbox");

        if (!fileInput || !fileInput.files || fileInput.files.length === 0) {
            alert("No se seleccionó ningún archivo.");
            return;
        }

        const reader = new FileReader();
        reader.onload = function (event)
        {
            const rawLines = event.target.result.split(/\r?\n/);

            const lines = rawLines
            .map(line => line.replace(/[^\p{L}\p{N}().\s-]/gu, "").trim())
            .filter(line => line.length > 0);

            const eliminadas = rawLines.length - lines.length;
            if (eliminadas > 0) {
                console.warn(`[Importar Lista] ⚠️ ${eliminadas} líneas inválidas fueron ignoradas (vacías, caracteres no permitidos o basura).`);
            }

            if (replaceCheckbox && replaceCheckbox.checked) {
                // Si se marcó reemplazar, limpiar todo
                excludeWords = [];
            } else {
                // Si no, recuperar la actual del localStorage
                excludeWords = JSON.parse(localStorage.getItem("excludeWords")) || excludeWords || [];
            }

            // Unificar, eliminar duplicados y ordenar
            excludeWords = [...new Set([...excludeWords, ...lines])]
                .filter(w => w.trim().length > 0)
                .sort((a, b) => a.localeCompare(b));

            localStorage.setItem("excludeWords", JSON.stringify(excludeWords));

            console.log("[handleImportList] Palabras actuales excluidas:", excludeWords);

            renderExcludedWordsPanel(); // O renderExcludeWordList() si usas otro nombre
            setupDragAndDropImport(); // Activa drag and drop

            alert(`✅ Palabras excluidas importadas correctamente: ${excludeWords.length}`);

            //Limpia el input para permitir recarga posterior
            fileInput.value = "";
        };

        reader.readAsText(fileInput.files[0]);
    }

    //**************************************************************************
    //Nombre: setupDragAndDropImport
    //Fecha modificación: 2025-03-31
    //Autor: mincho77
    //Entradas: ninguna
    //Salidas: habilita funcionalidad de arrastrar y soltar archivos al panel
    //Descripción:
    // Permite arrastrar y soltar un archivo .txt sobre el panel lateral (#normalizer-sidebar).
    // Extrae las palabras del archivo y las agrega a excludeWords sin duplicados.
    //**************************************************************************
    function setupDragAndDropImport() {
        const dropArea = document.getElementById("drop-zone");
        if (!dropArea) {
            console.warn("[setupDragAndDropImport] No se encontró la zona de drop");
            return;
        }

        // Highlight drop zone when dragging over it
        const highlight = (e) => {
            e.preventDefault();
            e.stopPropagation();
            dropArea.classList.add("drag-over");
            dropArea.style.backgroundColor = "#e8f5e9";
            dropArea.style.borderColor = "#4CAF50";
        };

        // Remove highlighting
        const unhighlight = (e) => {
            e.preventDefault();
            e.stopPropagation();
            dropArea.classList.remove("drag-over");
            dropArea.style.backgroundColor = "";
            dropArea.style.borderColor = "#ccc";
        };

        // Handle the dropped files
        const handleDrop = (e) => {
            console.log("[handleDrop] Evento drop detectado");
            e.preventDefault();
            e.stopPropagation();
            unhighlight(e);

            const file = e.dataTransfer.files[0];
            if (!file) {
                alert("❌ No se detectó ningún archivo");
                return;
            }

            // Validar extensión del archivo
            if (!file.name.match(/\.(txt|xml)$/i)) {
                alert("❌ Por favor, arrastra un archivo .txt o .xml");
                return;
            }

            console.log(`[handleDrop] Procesando archivo: ${file.name}`);

            const reader = new FileReader();
            reader.onload = (event) => {
                try {
                    const content = event.target.result;
                    const isXML = file.name.toLowerCase().endsWith('.xml');
                    let words = [];

                    if (isXML) {
                        const parser = new DOMParser();
                        const xmlDoc = parser.parseFromString(content, "text/xml");
                        if (xmlDoc.getElementsByTagName("parsererror").length > 0) {
                            throw new Error("XML inválido");
                        }
                        words = Array.from(xmlDoc.getElementsByTagName("word"))
                            .map(node => node.textContent.trim())
                            .filter(w => w.length > 0);
                    } else {
                        words = content.split(/\r?\n/)
                            .map(line => line.trim())
                            .filter(line => line.length > 0);
                    }

                    if (words.length === 0) {
                        alert("⚠️ No se encontraron palabras válidas en el archivo");
                        return;
                    }

                    // Actualizar lista de palabras excluidas
                    excludeWords = [...new Set([...excludeWords, ...words])].sort();
                    localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
                    renderExcludedWordsPanel();

                    alert(`✅ Se importaron ${words.length} palabras exitosamente`);
                    console.log(`[handleDrop] Importadas ${words.length} palabras`);

                } catch (error) {
                    console.error("[handleDrop] Error procesando archivo:", error);
                    alert("❌ Error procesando el archivo");
                }
            };

            reader.onerror = () => {
                console.error("[handleDrop] Error leyendo archivo");
                alert("❌ Error leyendo el archivo");
            };

            reader.readAsText(file);
        };

        // Attach the event listeners
        dropArea.addEventListener("dragenter", highlight, false);
        dropArea.addEventListener("dragover", highlight, false);
        dropArea.addEventListener("dragleave", unhighlight, false);
        dropArea.addEventListener("drop", handleDrop, false);

        console.log("[setupDragAndDropImport] Eventos de drag & drop configurados");
    }

    // Función auxiliar para procesar el archivo
    function handleImportedFile(content, isXML) {
        let words = [];

        if (isXML) {
            const parser = new DOMParser();
            const xmlDoc = parser.parseFromString(content, "text/xml");
            const nodes = xmlDoc.getElementsByTagName("word");
            words = Array.from(nodes).map(node => node.textContent.trim());
        } else {
            words = content.split(/\r?\n/).map(line => line.trim()).filter(line => line);
        }

        if (words.length === 0) {
            alert("No se encontraron palabras válidas en el archivo");
            return;
        }

        // Actualizar lista de palabras excluidas
        excludeWords = [...new Set([...excludeWords, ...words])].sort();
        localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
        renderExcludedWordsPanel();

        alert(`✅ Se importaron ${words.length} palabras exitosamente`);
    }

    //**************************************************************************
    //Nombre: renderExcludedWordsPanel
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas: ninguna (usa la variable global excludeWords)
    //Salidas: ninguna (actualiza el DOM y el localStorage)
    //Prerrequisitos si existen: Debe existir un contenedor con id="normalizer-sidebar"
    //Descripción:
    // Esta función limpia y vuelve a renderizar la lista de palabras excluidas
    // dentro del panel lateral del normalizador. Muestra una lista ordenada
    // alfabéticamente, evitando palabras vacías. Además, asegura que el localStorage
    // se actualice correctamente con la lista limpia y depurada.
    //**************************************************************************
    function renderExcludedWordsPanel() {
        const container = document.getElementById("normalizer-sidebar");
        if (!container) {
            console.warn("[renderExcludedWordsPanel] ❌ No se encontró el contenedor 'normalizer-sidebar'");
            return;
        }

        //Limpiar el contenedor visual
        container.innerHTML = "";
        console.log("[renderExcludedWordsPanel] ✅ Contenedor limpiado.");

        //Limpiar palabras vacías y ordenar
        const sortedWords = excludeWords.filter(w => !!w).sort((a, b) => a.localeCompare(b));
        console.log(`[renderExcludedWordsPanel] 📋 Lista excluida depurada: ${sortedWords.length} palabras`, sortedWords);

        const excludeListSection = document.createElement("div");
        excludeListSection.style.marginTop = "20px";
        excludeListSection.innerHTML = `
        <h4 style="margin-bottom: 5px;">Palabras Excluidas</h4>
        <div style="max-height: 150px; overflow-y: auto; border: 1px solid #ccc; padding: 8px; font-size: 13px; border-radius: 4px;">
            <ul style="margin: 0; padding-left: 18px;" id="excludeWordsList">
                ${sortedWords.map(w => `<li>${w}</li>`).join("")}
            </ul>
        </div>
    `;
        container.appendChild(excludeListSection);
        console.log("[renderExcludedWordsPanel] ✅ Lista renderizada en el DOM.");

        //Guardar en localStorage
        localStorage.setItem("excludeWords", JSON.stringify(sortedWords));
        console.log("[renderExcludedWordsPanel] 💾 excludeWords actualizado en localStorage.");
    }


    //**************************************************************************
    //Nombre: normalizePlaceName (unsafeWindow)
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas:
    //  - name (string): el nombre original del lugar.
    //Salidas:
    //  - string: nombre normalizado, respetando exclusiones y opciones del usuario.
    //Prerrequisitos si existen:
    //  - Debe estar cargada la lista global excludeWords.
    //  - Debe existir un checkbox con id “normalizeArticles” para definir si se normalizan artículos.
    //Descripción:
    //  Esta versión expuesta globalmente permite acceder a la normalización básica del nombre de un lugar
    //  desde otros contextos como el navegador o Tampermonkey. Capitaliza cada palabra, respeta las excluidas
    //  y no aplica normalización a artículos si el checkbox lo indica.
    //  Realiza limpieza básica: reemplazo de pipes, eliminación de espacios dobles y trim final.
    //**************************************************************************
    unsafeWindow.normalizePlaceName = function(name)
    {
        if (!name) return "";

        const normalizeArticles = !document.getElementById("normalizeArticles")?.checked;
        const articles = ["el", "la", "los", "las", "de", "del", "al", "y", "o"];

        const words = name.trim().split(/\s+/);
        const normalizedWords = words.map((word, index) => {
            const lowerWord = word.toLowerCase();

            // Saltar palabras excluidas
            if (excludeWords.includes(word)) return word;

            // Saltar artículos si el checkbox está activo y no es la primera palabra
            if (!normalizeArticles && articles.includes(lowerWord) && index !== 0) {
                return lowerWord;
            }
            //Mayúsculas
            return word.charAt(0).toUpperCase() + word.slice(1).toLowerCase();
        });

        name = normalizedWords.join(" ");
        name = name.replace(/\s*\|\s*/g, " - ");
        name = name.replace(/\s{2,}/g, " ").trim();

        return name;
    };

    //**************************************************************************
    //Nombre: normalizePlaceNameOnly
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas: name (string) – Nombre del lugar a normalizar.
    //Salidas: texto normalizado sin validación ortográfica.
    //Descripción:
    //   Realiza normalización visual del nombre: capitaliza, ajusta espacios,
    //   formatea guiones, paréntesis, y símbolos. No evalúa ortografía ni acentos.
    //**************************************************************************
    function normalizePlaceNameOnly(name) {
        if (!name) return "";
    
        const normalizeArticles = !document.getElementById("normalizeArticles")?.checked;
        const articles = ["el", "la", "los", "las", "de", "del", "al", "y"];
        const words = name.trim().split(/\s+/);
        const isRoman = word => /^(i{1,3}|iv|v|vi{0,3}|ix|x|xi{0,3}|xiv|xv|xvi{0,3}|xix|xx|xxi{0,3}|xxiv|xxv)$/i.test(word);
    
        const normalizedWords = words.map((word, index) => {
            const lowerWord = word.toLowerCase();
    
            // Si contiene "&", convertir a mayúsculas
            if (/^[A-Za-z]&[A-Za-z]$/.test(word)) return word.toUpperCase();
    
            // Verificar si está en la lista de excluidas
            const matchExcluded = excludeWords.find(w => w.toLowerCase() === lowerWord);
            if (matchExcluded) return matchExcluded;
    
            // Si es un número romano, convertir a mayúsculas
            if (isRoman(word)) return word.toUpperCase();
    
            // Si no se deben normalizar artículos y es un artículo, mantener en minúsculas
            if (!normalizeArticles && articles.includes(lowerWord) && index !== 0) return lowerWord;
    
            // Si es un número o un símbolo como "-", no modificar
            if (/^\d+$/.test(word) || word === "-") return word;
    
            // Verificar ortografía usando la API de LanguageTool
            return checkSpellingWithAPI(word)
                .then(errors => {
                    if (errors.length > 0) {
                        const suggestion = errors[0].sugerencia || word;
                        return suggestion.charAt(0).toUpperCase() + suggestion.slice(1).toLowerCase();
                    }
                    return word.charAt(0).toUpperCase() + word.slice(1).toLowerCase();
                })
                .catch(() => word.charAt(0).toUpperCase() + word.slice(1).toLowerCase());
        });
    
        let newName = normalizedWords.join(" ")
            .replace(/\s*\|\s*/g, " - ")
            .replace(/([(["'])\s*([\p{L}])/gu, (match, p1, p2) => p1 + p2.toUpperCase())
            .replace(/\s*-\s*/g, " - ")
            .replace(/\b(\d+)([A-Z])\b/g, (match, num, letter) => num + letter.toUpperCase())
            .replace(/\.$/, "")
            .replace(/&(\s*)([A-Z])/g, (match, space, letter) => "&" + space + letter.toUpperCase());
    
        return newName.replace(/\s{2,}/g, " ").trim();
    }

    //**************************************************************************
    //Nombre: validarOrtografiaConAPI
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas: texto (string) - palabra o frase a evaluar
    //Salidas: Promesa con resultado de corrección ortográfica
    //Descripción: Consulta la API pública de LanguageTool para identificar errores ortográficos
    //**************************************************************************
    function validarOrtografiaConAPI2(texto) {
        return new Promise((resolve, reject) => {
            GM_xmlhttpRequest({
                method: "POST",
                url: "https://api.languagetool.org/v2/check",
                headers: {
                    "Content-Type": "application/x-www-form-urlencoded"
                },
                data: `text=${encodeURIComponent(texto)}&language=es`,
                onload: function (response) {
                    if (response.status === 200) {
                        const result = JSON.parse(response.responseText);
                        resolve(result.matches || []);
                    } else {
                        reject("Error en la API de LanguageTool");
                    }
                },
                onerror: function () {
                    reject("Fallo la solicitud a la API de LanguageTool");
                }
            });
        });
    }

//**************************************************************************
//Nombre: importExcludeList
//Fecha modificación: 2025-03-31
//Autor: mincho77
//Entradas:
//   file (File) – Archivo cargado por el usuario, que contiene una lista de palabras excluidas.
//Salidas:
//   Ninguna (actualiza el array global excludeWords y el localStorage).
//Prerrequisitos:
//   Debe existir un panel visual para mostrar la lista y una función renderExcludeList().
//Descripción:
//   Lee un archivo .txt línea por línea, limpia espacios, elimina duplicados y vacíos,
//   ordena alfabéticamente la lista resultante y actualiza la lista global de palabras
//   excluidas (excludeWords). Guarda la lista en localStorage y actualiza el panel visual.
//**************************************************************************
    function importExcludeList(file) {
        const reader = new FileReader();
        reader.onload = function (e) {
            const newWords = e.target.result
            .split(/\r?\n/)
            .map(w => w.trim())
            .filter(w => w.length > 0); // eliminar vacíos

            excludeWords = [...new Set(newWords)].sort((a, b) => a.localeCompare(b));
            localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
            renderExcludeList(); // actualiza el panel visual
        };
        reader.readAsText(file);
    }

//**************************************************************************
//Nombre: applySpellCorrection
//Fecha modificación: 2025-03-31
//Autor: mincho77
//Entradas:
//   text (string) – Texto que se desea corregir automáticamente con sugerencias ortográficas.
//Salidas:
//   Promise<string> – Texto corregido con las mejores sugerencias aplicadas.
//Prerrequisitos:
//   Debe existir la función checkSpelling que retorna los errores detectados por LanguageTool.
//Descripción:
//   Llama a checkSpelling y aplica la primera sugerencia de reemplazo para cada error ortográfico,
//   procesando los reemplazos de atrás hacia adelante (para evitar desajustes de índice).
//   Devuelve el texto corregido como resultado final.
//**************************************************************************

    function applySpellCorrection(text) {
        return checkSpelling(text).then(data => {
            let corrected = text;
            // Ordenar los matches de mayor a menor offset
            const matches = data.matches.sort((a, b) => b.offset - a.offset);
            matches.forEach(match => {
                if (match.replacements && match.replacements.length > 0) {
                    const replacement = match.replacements[0].value;
                    corrected = corrected.substring(0, match.offset) + replacement + corrected.substring(match.offset + match.length);
                }
            });
            return corrected;
        });
    }



//**************************************************************************
//Nombre: createSidebarTab
//Fecha modificación: 2025-03-31
//Autor: mincho77
//Entradas: Ninguna directa. Usa `W.userscripts` para registrar el tab.
//Salidas: Renderiza un nuevo tab lateral personalizado con interfaz del normalizador.
//Prerrequisitos:
//  - `W.userscripts.registerSidebarTab` debe estar disponible.
//  - `getSidebarHTML()` debe retornar el HTML necesario.
//Descripción:
//  Crea y registra una nueva pestaña en el sidebar de WME con el título
//  "Places Name Normalizer". Al cargar correctamente, inserta el HTML
//  generado por `getSidebarHTML()` y espera a que se renderice completamente
//  para ejecutar los eventos mediante `waitForDOM`.
//**************************************************************************
function createSidebarTab() {
    try {
        // Check if the sidebar system is ready
        if (!W || !W.userscripts) {
            console.error(`[${SCRIPT_NAME}] WME not ready for sidebar creation`);
            return;
        }

        // Check for existing tab and clean up if needed
        const existingTab = document.getElementById("normalizer-tab");
        if (existingTab) {
            console.log(`[${SCRIPT_NAME}] Removing existing tab...`);
            existingTab.remove();
        }

        // Register new tab with error handling
        let registration;
        try {
            registration = W.userscripts.registerSidebarTab("PlacesNormalizer");
        } catch (e) {
            if (e.message.includes("already been registered")) {
                console.warn(`[${SCRIPT_NAME}] Tab registration conflict, attempting cleanup...`);
                // Additional cleanup could go here
                return;
            }
            throw e;
        }

        const { tabLabel, tabPane } = registration;
        
        if (!tabLabel || !tabPane) {
            throw new Error("Tab registration failed to return required elements");
        }

        // Configure tab
        tabLabel.innerHTML = `
        <img src=""
        style="height: 16px; vertical-align: middle; margin-right: 5px;">
        NrmliZer
        `;
        tabLabel.title = "Places Name Normalizer";
        
        // Set content and attach events
        tabPane.innerHTML = getSidebarHTML();
        waitForDOM("#normalizer-tab", (element) => {
            attachEvents();
            console.log(`[${SCRIPT_NAME}] Tab created and events attached`);
        }, 500, 10);

    } catch (error) {
        console.error(`[${SCRIPT_NAME}] Error in createSidebarTab:`, error);
    }
}

   

//**************************************************************************
//Nombre: attachEvents
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: Ninguna
//Salidas: Ninguna
//Prerrequisitos si existen:
// - Deben existir en el DOM los elementos con los siguientes IDs:
//   "normalizeArticles", "maxPlacesInput", "addExcludeWord", "scanPlaces",
//   "hiddenImportInput", "importExcludeWordsUnifiedBtn" y "exportExcludeWords".
// - Debe existir la función handleImportList y la función scanPlaces.
// - Debe estar definida la variable global excludeWords y la función renderExcludedWordsPanel.
//Descripción:
// Esta función adjunta los event listeners necesarios para gestionar la interacción del usuario
// con el panel del normalizador de nombres. Se encargan de:
//   - Actualizar la opción de normalizar artículos al cambiar el estado del checkbox.
//   - Modificar el número máximo de lugares a procesar a través de un input.
//   - Exportar la lista de palabras excluidas a un archivo XML.
//   - Añadir nuevas palabras a la lista de palabras excluidas, evitando duplicados, y actualizar el panel.
//   - Activar el botón unificado para la importación de palabras excluidas mediante un input oculto.
//   - Ejecutar la función de escaneo de lugares al hacer clic en el botón correspondiente.
//**************************************************************************
    function attachEvents()
    {
        console.log(`[${SCRIPT_NAME}] Adjuntando eventos...`);

        const normalizeArticlesCheckbox = document.getElementById("normalizeArticles");
        const maxPlacesInput = document.getElementById("maxPlacesInput");
        const addExcludeWordButton = document.getElementById("addExcludeWord");
        const scanPlacesButton = document.getElementById("scanPlaces");
        const hiddenInput = document.getElementById("hiddenImportInput");
        const importButtonUnified = document.getElementById("importExcludeWordsUnifiedBtn");

        // Validación de elementos necesarios
        if (!normalizeArticlesCheckbox || !maxPlacesInput || !addExcludeWordButton || !scanPlacesButton) {
            console.error(`[${SCRIPT_NAME}] Error: No se encontraron elementos en el DOM.`);
            return;
        }

        // ✅ Evento: cambiar estado de "no normalizar artículos"
        normalizeArticlesCheckbox.addEventListener("change", (e) => {
            normalizeArticles = e.target.checked;
        });

        // ✅ Evento: cambiar número máximo de places
        maxPlacesInput.addEventListener("input", (e) => {
            maxPlaces = parseInt(e.target.value, 10);
        });

        // ✅ Evento: exportar palabras excluidas a XML
        document.getElementById("exportExcludeWords").addEventListener("click", () => {
            const savedWords = JSON.parse(localStorage.getItem("excludeWords")) || [];
            if (savedWords.length === 0) {
                alert("No hay palabras excluidas para exportar.");
                return;
            }
            const sortedWords = [...savedWords].sort((a, b) => a.localeCompare(b));

            const xmlContent = `<?xml version="1.0" encoding="UTF-8"?>
<ExcludedWords>
  ${sortedWords.map(word => `  <word>${word}</word>`).join("\n  ")}
</ExcludedWords>`;

            const blob = new Blob([xmlContent], { type: "application/xml" });
            const url = URL.createObjectURL(blob);
            const link = document.createElement("a");

            link.href = url;
            link.download = "excluded_words.xml";
            document.body.appendChild(link); // Correctly appends the link
            link.click();
            document.body.removeChild(link); // Correctly removes the link
        });

        // ✅ Evento: añadir palabra excluida sin duplicados
        addExcludeWordButton.addEventListener("click", () => {
            const wordInput = document.getElementById("excludeWord") || document.getElementById("excludedWord");
            const word = wordInput?.value.trim();

            if (!word) return;

            const lowerWord = word.toLowerCase();
            const alreadyExists = excludeWords.some(w => w.toLowerCase() === lowerWord);
            if (!alreadyExists) {
                excludeWords.push(word);
                localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
                renderExcludedWordsPanel(); // Usar renderExcludedWordsPanel en lugar de updateExcludeList
            }

            wordInput.value = "";
        });

        // ✅ Evento: nuevo botón unificado de importación
        importButtonUnified.addEventListener("click", () => {
            hiddenInput.click(); // abre el file input oculto
        });

        hiddenInput.addEventListener("change", () => {
            handleImportList(); // ✅ Llama a la función handleImportList al importar
        });

        // ✅ Evento: escanear lugares
        scanPlacesButton.addEventListener("click", scanPlaces);
    }


    function updateExcludeList() {
        const list = document.getElementById("excludedWordsList");
        if (!list) return;

        // Ordena una copia del array para no alterar el original
        const sortedWords = [...excludeWords].sort((a, b) => a.toLowerCase().localeCompare(b.toLowerCase()));

        list.innerHTML = sortedWords.map(word => `<li>${word}</li>`).join("");
    }

//**************************************************************************
//Nombre: scanPlaces
//Fecha modificación: 2025-04-02
//Autor: mincho77
//Entradas: ninguna directamente (usa datos del modelo WME y del input #maxPlacesInput)
//Salidas: abre panel flotante con lugares que deben ser normalizados
//Descripción:
// Escanea los lugares visibles en el WME, normaliza los nombres y verifica
// si les falta una tilde en las palabras que lo requieren. Si se selecciona
// "Revisar solo tildes", utiliza la función evaluarOrtografiaConTildes.
//**************************************************************************
// ==UserScript==
// @name         WME Places Name Normalizer
// @namespace    https://gf.qytechs.cn/en/users/mincho77
// @version      2.1
// @description  Normaliza nombres de lugares en Waze Map Editor (WME) por ahora solo con reglas de Colombia
// @author       mincho77
// @match        https://www.waze.com/*editor*
// @match      https://beta.waze.com/*user/editor*
// @grant        GM_xmlhttpRequest
// @connect      api.languagetool.org
// @connect      *
// @grant        unsafeWindow
// @license      MIT
// @run-at       document-end
// @downloadURL https://update.gf.qytechs.cn/scripts/530268/WME%20Places%20Name%20Normalizer.user.js
// @updateURL https://update.gf.qytechs.cn/scripts/530268/WME%20Places%20Name%20Normalizer.meta.js
// ==/UserScript==
/*global W*/

(() => {
    "use strict";
    try {


   // Insertar estilos globales en el <head>
const styles = `
<style>
    /* Estilos para los botones */
    .apply-suggestion-btn {
        background-color: #4CAF50;
        color: white;
        border: none;
        padding: 8px 12px;
        border-radius: 4px;
        cursor: pointer;
        font-size: 14px;
        font-weight: bold;
        transition: background-color 0.3s ease;
    }
    .apply-suggestion-btn:hover {
        background-color: #45a049;
    }

    #apply-changes-btn {
        background-color: #28a745;
        color: white;
        border: none;
        padding: 10px 15px;
        border-radius: 4px;
        cursor: pointer;
        font-size: 16px;
        font-weight: bold;
        display: flex;
        align-items: center;
        gap: 8px;
    }
    #apply-changes-btn:hover {
        background-color: #218838;
    }

    #cancel-btn {
        background-color: #dc3545;
        color: white;
        border: none;
        padding: 10px 15px;
        border-radius: 4px;
        cursor: pointer;
        font-size: 16px;
        font-weight: bold;
        display: flex;
        align-items: center;
        gap: 8px;
    }
    #cancel-btn:hover {
        background-color: #c82333;
    }
</style>
`;
document.head.insertAdjacentHTML('beforeend', styles);

    // Capturar todos los eventos de drag & drop a nivel de <body>
   // Agregar al inicio del script, justo después del "use strict"
    // Prevenir comportamiento por defecto de drag & drop a nivel global
    document.addEventListener("dragover", function(e) {
        const dropZone = document.getElementById("drop-zone");
        if (e.target === dropZone || dropZone?.contains(e.target)) {
            return; // Permitir el drop en la zona designada
        }
        e.preventDefault();
        e.stopPropagation();
    }, { passive: false }); // Agrega { passive: false }

    document.addEventListener("drop", function(e) {
        const dropZone = document.getElementById("drop-zone");
        if (e.target === dropZone || dropZone?.contains(e.target)) {
            return; // Permitir el drop en la zona designada
        }
        e.preventDefault();
        e.stopPropagation();
    }, { passive: false }); // Agrega { passive: false }

//**************************************************************************
//Nombre: evaluarOrtografiaConTildes
//Fecha modificación: 2025-04-02
//Autor: mincho77
//Entradas: name (string) - Nombre del lugar
//Salidas: objeto con errores detectados
//Descripción:
// Evalúa palabra por palabra si falta una tilde en las palabras que lo requieren,
// según las reglas del español. Primero normaliza el nombre y luego verifica
// si las palabras necesitan una tilde.
//**************************************************************************
function evaluarOrtografiaConTildes(name) {
    return new Promise((resolve) => {
        if (!name) {
            return resolve({
                hasSpellingWarning: false,
                spellingWarnings: []
            });
        }

        const palabras = name.trim().split(/\s+/);
        const spellingWarnings = [];
        const totalPalabras = palabras.length;
        let procesadas = 0;

        // Función para procesar cada palabra secuencialmente
        function procesarPalabra(index) {
            if (index >= palabras.length) {
                // Todas las palabras procesadas
                toggleSpinner(false);
                return resolve({
                    hasSpellingWarning: spellingWarnings.length > 0,
                    spellingWarnings
                });
            }

            const palabra = palabras[index];
            let normalizada = normalizePlaceNameOnly(palabra);

            // Actualizar progreso
            procesadas++;
            const progress = Math.round((procesadas / totalPalabras) * 100);
            toggleSpinner(true, 'Revisando tildes...', progress);

            // Lógica de verificación existente
            if (/^[A-Za-z]&[A-Za-z]$/.test(normalizada) || 
                /^[\u263a-\u263c\u2764\u1f600-\u1f64f\u1f680-\u1f6ff]+$/.test(normalizada)) {
                return procesarPalabra(index + 1);
            }

            if (normalizada.toLowerCase() === "y" || /^\d+$/.test(normalizada) || normalizada === "-") {
                return procesarPalabra(index + 1);
            }

            if (excludeWords.some(w => w.toLowerCase() === normalizada.toLowerCase())) {
                return procesarPalabra(index + 1);
            }

            const cantidadTildes = (normalizada.match(/[áéíóú]/g) || []).length;
            if (cantidadTildes > 1) {
                spellingWarnings.push({
                    original: palabra,
                    sugerida: null,
                    tipo: "Error de tildes",
                    posicion: index
                });
            }

            // Procesar siguiente palabra
            procesarPalabra(index + 1);
        }

        // Iniciar el procesamiento
        toggleSpinner(true, 'Revisando tildes...', 0);
        procesarPalabra(0);
    });
}

// Función auxiliar para corregir tildes
function corregirTilde(palabra) {
    // Implementar lógica para corregir tildes según las reglas del español
    // Por ejemplo: "medellin" → "Medellín"
    return palabra; // Retornar la palabra corregida
}

    //**************************************************************************
    //Nombre: toggleSpinner
    //Fecha modificación: 2025-03-31
    //Autor: mincho77
    //Entradas:
    //   show (boolean) - true para mostrar el spinner, false para ocultarlo
    //   message (string, opcional) - mensaje personalizado a mostrar junto al spinner
    //Salidas: ninguna (modifica el DOM)
    //Prerrequisitos: debe existir el estilo CSS del spinner en el documento
    //Descripción:
    //   Muestra u oculta un indicador visual de carga con un mensaje opcional.
    //   El spinner usa un emoji de reloj de arena (⏳) con animación de rotación
    //   para indicar que el proceso está en curso.
    //**************************************************************************
    function toggleSpinner(show, message = 'Revisando ortografía...', progress = null) {
        let existingSpinner = document.querySelector('.spinner-overlay');
        if (existingSpinner) {
            if (show) {
                // Actualizar el mensaje y el progreso si el spinner ya existe
                const spinnerMessage = existingSpinner.querySelector('.spinner-message');
                spinnerMessage.innerHTML = `
                    ${message}
                    ${progress !== null ? `<br><strong>${progress}% completado</strong>` : ''}
                `;
            } else {
                existingSpinner.remove(); // Ocultar el spinner
            }
            return;
        }
    
        if (show) {
            const spinner = document.createElement('div');
            spinner.className = 'spinner-overlay';
            spinner.innerHTML = `
                <div class="spinner-content">
                    <div class="spinner-icon">⏳</div>
                    <div class="spinner-message">
                        ${message}
                        ${progress !== null ? `<br><strong>${progress}% completado</strong>` : ''}
                    </div>
                </div>
            `;
            document.body.appendChild(spinner);
        }
    }

    // Agregar los estilos CSS necesarios
    const spinnerStyles = `
    <style>
    .spinner-overlay {
        position: fixed;
        top: 0;
        left: 0;
        width: 100%;
        height: 100%;
        background: rgba(0, 0, 0, 0.5);
        display: flex;
        justify-content: center;
        align-items: center;
        z-index: 10000;
    }

    .spinner-content {
        background: white;
        padding: 20px;
        border-radius: 8px;
        text-align: center;
        box-shadow: 0 2px 10px rgba(0,0,0,0.3);
    }

    .spinner-icon {
        font-size: 24px;
        margin-bottom: 10px;
        animation: spin 1s linear infinite; /* Aseguramos que la animación esté activa */
        display: inline-block;
    }

    .spinner-message {
        color: #333;
        font-size: 14px;
    }

    @keyframes spin {
        0% { transform: rotate(0deg); }
        100% { transform: rotate(360deg); }
    }
    </style>`;

    // Insertar los estilos al inicio del documento
    document.head.insertAdjacentHTML('beforeend', spinnerStyles);

    if (!Array.prototype.flat)
    {
        Array.prototype.flat = function(depth = 1)
        {
            return this.reduce(function (flat, toFlatten)
            {
                return flat.concat(Array.isArray(toFlatten) ? toFlatten.flat(depth - 1) : toFlatten);
            }, []);
        };
    }
    const SCRIPT_NAME = "PlacesNameNormalizer";
    const VERSION = "2.1";
    let placesToNormalize = [];
    let excludeWords = [];
    let maxPlaces = 100;
    let normalizeArticles = true;

    // Expresión regular para detectar siglas (por ejemplo, "S.a.s", "L.T.D.A")
    const siglaRegex = /^[A-Za-z](\.[A-Za-z])+\.?$/;



    
    //**************************************************************************
    //Nombre: waitForSidebar
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas:
    //   retries (número, opcional) – Número máximo de intentos de verificación del sidebar (default: 20).
    //   delay (número, opcional) – Tiempo en milisegundos entre intentos (default: 1000ms).
    //Salidas:
    //   Promesa que se resuelve con el elemento del sidebar si se encuentra, o se rechaza si no se encuentra después de los intentos.
    //Descripción:
    //   Esta función espera a que el DOM cargue completamente el elemento con ID "sidebar".
    //   Realiza múltiples intentos con intervalos definidos, y resuelve la promesa cuando el sidebar esté disponible.
    //   Es útil para asegurarse de que el entorno de WME esté completamente cargado antes de continuar.
    //**************************************************************************
    function waitForSidebar(retries = 20, delay = 1000) {
        return new Promise((resolve, reject) => {
            const check = (attempt = 1) => {
                const sidebar = document.querySelector("#sidebar");
                if (sidebar) {
                    console.log("✅ Sidebar disponible.");
                    resolve(sidebar);
                } else if (attempt <= retries) {
                    console.warn(`⚠️ Sidebar no disponible aún. Reintentando... (${attempt})`);
                    setTimeout(() => check(attempt + 1), delay);
                } else {
                    reject("❌ Sidebar no disponible después de múltiples intentos.");
                }
            };
            check();
        });
    }
    //**************************************************************************
    //Nombre: initializeExcludeWords
    //Fecha modificación: 2025-03-31
    //Autor: mincho77
    //Entradas: Ninguna
    //Salidas: Ninguna
    //Prerrequisitos: excludeWords debe estar declarado globalmente.
    //Descripción: Inicializa la lista de palabras excluidas desde localStorage sin borrar entradas ya existentes.
    //**************************************************************************
    function initializeExcludeWords() {
        const saved = JSON.parse(localStorage.getItem("excludeWords")) || [];

        // Combinar con las actuales sin duplicar
        const merged = [...new Set([...saved, ...excludeWords])].sort((a, b) => a.localeCompare(b));

        // Solo guardar si hay diferencias
        const originalString = JSON.stringify(saved.sort());
        const newString = JSON.stringify(merged);

        if (originalString !== newString) {
            localStorage.setItem("excludeWords", newString);
            console.log(`[initializeExcludeWords] 💾 excludeWords actualizado con ${merged.length} palabras.`);
        } else {
            console.log(`[initializeExcludeWords] 🟢 Sin cambios en excludeWords.`);
        }

        // Actualizar variable global
        excludeWords = merged;
    }

    //Modulo de Ortografía
   



   

    // Insertar los estilos al inicio del documento
    document.head.insertAdjacentHTML('beforeend', spinnerStyles);

    if (!Array.prototype.flat)
    {
        Array.prototype.flat = function(depth = 1)
        {
            return this.reduce(function (flat, toFlatten)
            {
                return flat.concat(Array.isArray(toFlatten) ? toFlatten.flat(depth - 1) : toFlatten);
            }, []);
        };
    }
   

    
    //Modulo de Ortografía
    //**************************************************************************
    //Nombre: getUbicacionAcento
    //Fecha modificación: 2025-03-27
    //Autor: mincho77
    //Entradas: palabra (string) - Palabra en español a evaluar
    //Salidas: Posición del acento (última, penúltima, antepenúltima) o null si no aplica
    //Prerrequisitos si existen: Ninguno
    //Descripción: Determina la posición silábica del acento en una palabra según la ortografía del español.
    //**************************************************************************
    function getUbicacionAcento(palabra) {
        const original = palabra;
        const normalized = palabra.normalize("NFD").replace(/[\u0300-\u036f]/g, ""); // Remove diacritics
        const silabas = normalized
            .replace(/[^aeiou]/gi, "")
            .match(/[aeiou]+/gi);

        if (!silabas || silabas.length === 0) return null;

        const conTilde = silabas.findIndex((s, i) => {
            const originalSyllable = original.slice(
                normalized.indexOf(s),
                normalized.indexOf(s) + s.length
            );
            return /[áéíóú]/.test(originalSyllable);
        });

        if (conTilde !== -1) {
            const posicion = silabas.length - 1 - conTilde;
            if (posicion === 0) return "última";
            if (posicion === 1) return "penúltima";
            if (posicion === 2) return "antepenúltima";
            return "otras";
        }

        return "sin tilde";
    }

    //**************************************************************************
    //Nombre: checkSpellingWithAPI
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas: text (string) – Texto a evaluar ortográficamente.
    //Salidas: Promise – Resuelve con lista de errores ortográficos detectados.
    //Prerrequisitos: Requiere permisos GM_xmlhttpRequest y @connect a api.languagetool.org
    //Descripción:
    // Consulta la API de LanguageTool para verificar ortografía del texto.
    //**************************************************************************
    function checkSpellingWithAPI(text) {
        return new Promise((resolve, reject) => {
            GM_xmlhttpRequest({
                method: "POST",
                url: "https://api.languagetool.org/v2/check",
                headers: {
                    "Content-Type": "application/x-www-form-urlencoded"
                },
                data: `language=es&text=${encodeURIComponent(text)}`,
                onload: function(response) {
                    if (response.status === 200) {
                        const result = JSON.parse(response.responseText);
                        const errores = result.matches
                        .filter(match => match.rule.issueType === "misspelling")
                        .map(match => ({
                            palabra: match.context.text.substring(match.context.offset, match.context.offset + match.context.length),
                            sugerencia: match.replacements.length > 0 ? match.replacements[0].value : "(sin sugerencia)"
                        }));
                        resolve(errores);
                    } else {
                        reject("❌ Error en respuesta de LanguageTool");
                    }
                },
                onerror: function(err) {
                    reject("❌ Error de red al contactar LanguageTool");
                }
            });
        });
    }

    window.checkSpellingWithAPI = checkSpellingWithAPI;

// Combinar la lógica de checkSpelling
function checkSpelling(text) {
    return new Promise((resolve, reject) => {
        GM_xmlhttpRequest({
            method: "POST",
            url: "https://api.languagetool.org/v2/check",
            data: `text=${encodeURIComponent(text)}&language=es`,
            headers: {
                "Content-Type": "application/x-www-form-urlencoded"
            },
            onload: function (response) {
                if (response.status === 200) {
                    try {
                        const data = JSON.parse(response.responseText);
                        const errores = data.matches
                            .filter(match => match.rule.issueType === "misspelling")
                            .map(match => ({
                                palabra: match.context.text.substring(match.context.offset, match.context.offset + match.context.length),
                                sugerencia: match.replacements.length > 0 ? match.replacements[0].value : "(sin sugerencia)"
                            }));
                        resolve(errores);
                    } catch (err) {
                        reject(err);
                    }
                } else {
                    reject(`Error HTTP: ${response.status}`);
                }
            },
            onerror: function (err) {
                reject(err);
            }
        });
    });
}   

    //**************************************************************************
    //Nombre: validarWordSpelling
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas: palabra (string) - Palabra en español a validar ortográficamente
    //Salidas: true si cumple reglas ortográficas básicas, false si no
    //Descripción:
    // Evalúa si una palabra tiene el uso correcto de tilde o si le falta una tilde
    // según las reglas del español: esdrújulas siempre con tilde, agudas con tilde
    // si terminan en n, s o vocal, y llanas con tildse si NO terminan en n, s o vocal.
    // Se asegura que solo haya una tilde por palabra.
    //**************************************************************************
    function validarWordSpelling(palabra) {
        if (!palabra) return false;

        // Ignorar siglas con formato X&X
        if (/^[A-Za-z]&[A-Za-z]$/.test(palabra)) return true;

        // Si la palabra es un número, no necesita validación
        if (/^\d+$/.test(palabra)) return true;

        const tieneTilde = /[áéíóú]/.test(palabra);
        const cantidadTildes = (palabra.match(/[áéíóú]/g) || []).length;
        if (cantidadTildes > 1) return false; // Solo se permite una tilde

        const silabas = palabra.normalize("NFD").replace(/[^aeiouAEIOU\u0300-\u036f]/g, "").match(/[aeiouáéíóú]+/gi);
        if (!silabas || silabas.length === 0) return false;

        const totalSilabas = silabas.length;
        const ultimaLetra = palabra.slice(-1).toLowerCase();

        let tipo = "";
        if (totalSilabas >= 3 && /[áéíóú]/.test(palabra)) {
            tipo = "esdrújula";
        } else if (totalSilabas >= 2) {
            const penultimaSilaba = silabas[totalSilabas - 2];
            if (/[áéíóú]/.test(penultimaSilaba)) tipo = "grave";
        }

        if (!tipo) tipo = /[áéíóú]/.test(silabas[totalSilabas - 1]) ? "aguda" : "sin tilde";

        if (tipo === "esdrújula") return tieneTilde;
        if (tipo === "aguda") {
            return (/[nsáéíóúaeiou]$/.test(ultimaLetra) && tieneTilde) ||
                (!/[nsáéíóúaeiou]$/.test(ultimaLetra) && !tieneTilde);
        }
        if (tipo === "grave") {
            return (!/[nsáéíóúaeiou]$/.test(ultimaLetra) && tieneTilde) ||
                (/[nsáéíóúaeiou]$/.test(ultimaLetra) && !tieneTilde);
        }

        return true;
    }



    ///**************************************************************************
    //Nombre: separarSilabas
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas: palabra (string) – Palabra en español.
    //Salidas: array de sílabas (aproximado).
    //Descripción:
    //   Separa una palabra en sílabas usando reglas heurísticas.
    //   Esta versión simplificada considera diptongos y combinaciones comunes.
    //**************************************************************************
    function separarSilabas(palabra) {
        // Normaliza y quita acentos para facilitar la segmentación
        const limpia = palabra.normalize("NFD").replace(/[\u0300-\u036f]/g, "").toLowerCase();

        // Divide por vocales agrupadas como aproximación de sílabas
        const silabas = limpia.match(/[bcdfghjklmnñpqrstvwxyz]*[aeiou]{1,2}[bcdfghjklmnñpqrstvwxyz]*/g);

        return silabas || [palabra]; // fallback si no separa nada
    }

    //**************************************************************************
    //Nombre: clasificarPalabra
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas: silabas (array) – Arreglo con las sílabas de la palabra.
    //Salidas: string – 'aguda', 'grave' (llana), 'esdrújula'.
    //Descripción:
    //   Determina el tipo de palabra según el número de sílabas y la posición
    //   de la tilde (si existe).
    //**************************************************************************
    function clasificarPalabra(silabas) {
        const palabra = silabas.join("");
        const tieneTilde = /[áéíóúÁÉÍÓÚ]/.test(palabra);

        if (tieneTilde) {
            const posicionTilde = silabas.findIndex(s => /[áéíóúÁÉÍÓÚ]/.test(s));
            const posicionDesdeFinal = silabas.length - 1 - posicionTilde;

            if (posicionDesdeFinal === 0) return "aguda";
            if (posicionDesdeFinal === 1) return "grave";
            if (posicionDesdeFinal >= 2) return "esdrújula";
        }

        // Si no tiene tilde, asumimos que es:
        if (silabas.length === 1) return "aguda";
        if (silabas.length === 2) return "grave";
        return "grave"; // por convención para 3+ sin tilde
    }

    //**************************************************************************
    //Nombre: evaluarOrtografiaNombre
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas: name (string) - Nombre del lugar
    //Salidas: objeto con errores detectados
    //Descripción: Evalúa palabra por palabra si hay errores ortográficos o falta de tildes.
    // Ya no utiliza sugerencias automáticas para correcciones.
    //**************************************************************************
    function evaluarOrtografiaNombre(name) {
        if (!name) return {
            hasSpellingWarning: false,
            spellingWarnings: []
        };

        const checkOnlyTildes = document.getElementById("checkOnlyTildes")?.checked;
        const palabras = name.trim().split(/\s+/);
        const spellingWarnings = [];

        console.log(`[evaluarOrtografiaNombre] Verificando ortografía de: ${name}`);

        if (checkOnlyTildes) {
            palabras.forEach((palabra, index) => {
                // Verificar si la palabra está en la lista de exclusiones
                if (excludeWords.some(w => w.toLowerCase() === palabra.toLowerCase()) || /^\d+$/.test(palabra)) {
                    return; // Ignorar palabra excluida
                }

                if (!validarWordSpelling(palabra)) {
                    spellingWarnings.push({
                        original: palabra,
                        sugerida: null,
                        tipo: "Tilde",
                        posicion: index // Guardar la posición en la frase
                    });
                   
                }
            });

            return Promise.resolve({
                hasSpellingWarning: spellingWarnings.length > 0,
                spellingWarnings
            });
        } else {
            return checkSpellingWithAPI(name)
                .then(errores => {
                    errores.forEach(error => {
                        // Verificar si la palabra está en exclusiones
                        if (!excludeWords.some(w => w.toLowerCase() === error.palabra.toLowerCase())) {
                            spellingWarnings.push({
                                original: error.palabra,
                                sugerida: error.sugerencia,
                                tipo: "LanguageTool",
                                posicion: name.indexOf(error.palabra)
                            });
                        }
                    });
                    return {
                        hasSpellingWarning: spellingWarnings.length > 0,
                        spellingWarnings
                    };
                });
        }
    }
    
    //**************************************************************************
    //Nombre: handleImportList
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas: ninguna directa, depende del input file y checkbox en el DOM
    //Salidas: actualiza excludeWords en localStorage y visualmente
    //Prerrequisitos: existir un input file con id="importListInput" y checkbox con id="replaceExcludeListCheckbox"
    //Descripción: Importa una lista de palabras para excluir, opcionalmente reemplazando la lista actual
    //**************************************************************************
    function handleImportList() {
        const fileInput = document.getElementById("importListInput");
        const replaceCheckbox = document.getElementById("replaceExcludeListCheckbox");

        if (!fileInput || !fileInput.files || fileInput.files.length === 0) {
            alert("No se seleccionó ningún archivo.");
            return;
        }

        const reader = new FileReader();
        reader.onload = function (event)
        {
            const rawLines = event.target.result.split(/\r?\n/);

            const lines = rawLines
            .map(line => line.replace(/[^\p{L}\p{N}().\s-]/gu, "").trim())
            .filter(line => line.length > 0);

            const eliminadas = rawLines.length - lines.length;
            if (eliminadas > 0) {
                console.warn(`[Importar Lista] ⚠️ ${eliminadas} líneas inválidas fueron ignoradas (vacías, caracteres no permitidos o basura).`);
            }

            if (replaceCheckbox && replaceCheckbox.checked) {
                // Si se marcó reemplazar, limpiar todo
                excludeWords = [];
            } else {
                // Si no, recuperar la actual del localStorage
                excludeWords = JSON.parse(localStorage.getItem("excludeWords")) || excludeWords || [];
            }

            // Unificar, eliminar duplicados y ordenar
            excludeWords = [...new Set([...excludeWords, ...lines])]
                .filter(w => w.trim().length > 0)
                .sort((a, b) => a.localeCompare(b));

            localStorage.setItem("excludeWords", JSON.stringify(excludeWords));

            console.log("[handleImportList] Palabras actuales excluidas:", excludeWords);

            renderExcludedWordsPanel(); // O renderExcludeWordList() si usas otro nombre
            setupDragAndDropImport(); // Activa drag and drop

            alert(`✅ Palabras excluidas importadas correctamente: ${excludeWords.length}`);

            //Limpia el input para permitir recarga posterior
            fileInput.value = "";
        };

        reader.readAsText(fileInput.files[0]);
    }

    //**************************************************************************
    //Nombre: setupDragAndDropImport
    //Fecha modificación: 2025-03-31
    //Autor: mincho77
    //Entradas: ninguna
    //Salidas: habilita funcionalidad de arrastrar y soltar archivos al panel
    //Descripción:
    // Permite arrastrar y soltar un archivo .txt sobre el panel lateral (#normalizer-sidebar).
    // Extrae las palabras del archivo y las agrega a excludeWords sin duplicados.
    //**************************************************************************
    function setupDragAndDropImport() {
        const dropArea = document.getElementById("drop-zone");
        if (!dropArea) {
            console.warn("[setupDragAndDropImport] No se encontró la zona de drop");
            return;
        }

        // Highlight drop zone when dragging over it
        const highlight = (e) => {
            e.preventDefault();
            e.stopPropagation();
            dropArea.classList.add("drag-over");
            dropArea.style.backgroundColor = "#e8f5e9";
            dropArea.style.borderColor = "#4CAF50";
        };

        // Remove highlighting
        const unhighlight = (e) => {
            e.preventDefault();
            e.stopPropagation();
            dropArea.classList.remove("drag-over");
            dropArea.style.backgroundColor = "";
            dropArea.style.borderColor = "#ccc";
        };

        // Handle the dropped files
        const handleDrop = (e) => {
            console.log("[handleDrop] Evento drop detectado");
            e.preventDefault();
            e.stopPropagation();
            unhighlight(e);

            const file = e.dataTransfer.files[0];
            if (!file) {
                alert("❌ No se detectó ningún archivo");
                return;
            }

            // Validar extensión del archivo
            if (!file.name.match(/\.(txt|xml)$/i)) {
                alert("❌ Por favor, arrastra un archivo .txt o .xml");
                return;
            }

            console.log(`[handleDrop] Procesando archivo: ${file.name}`);

            const reader = new FileReader();
            reader.onload = (event) => {
                try {
                    const content = event.target.result;
                    const isXML = file.name.toLowerCase().endsWith('.xml');
                    let words = [];

                    if (isXML) {
                        const parser = new DOMParser();
                        const xmlDoc = parser.parseFromString(content, "text/xml");
                        if (xmlDoc.getElementsByTagName("parsererror").length > 0) {
                            throw new Error("XML inválido");
                        }
                        words = Array.from(xmlDoc.getElementsByTagName("word"))
                            .map(node => node.textContent.trim())
                            .filter(w => w.length > 0);
                    } else {
                        words = content.split(/\r?\n/)
                            .map(line => line.trim())
                            .filter(line => line.length > 0);
                    }

                    if (words.length === 0) {
                        alert("⚠️ No se encontraron palabras válidas en el archivo");
                        return;
                    }

                    // Actualizar lista de palabras excluidas
                    excludeWords = [...new Set([...excludeWords, ...words])].sort();
                    localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
                    renderExcludedWordsPanel();

                    alert(`✅ Se importaron ${words.length} palabras exitosamente`);
                    console.log(`[handleDrop] Importadas ${words.length} palabras`);

                } catch (error) {
                    console.error("[handleDrop] Error procesando archivo:", error);
                    alert("❌ Error procesando el archivo");
                }
            };

            reader.onerror = () => {
                console.error("[handleDrop] Error leyendo archivo");
                alert("❌ Error leyendo el archivo");
            };

            reader.readAsText(file);
        };

        // Attach the event listeners
        dropArea.addEventListener("dragenter", highlight, false);
        dropArea.addEventListener("dragover", highlight, false);
        dropArea.addEventListener("dragleave", unhighlight, false);
        dropArea.addEventListener("drop", handleDrop, false);

        console.log("[setupDragAndDropImport] Eventos de drag & drop configurados");
    }

    // Función auxiliar para procesar el archivo
    function handleImportedFile(content, isXML) {
        let words = [];

        if (isXML) {
            const parser = new DOMParser();
            const xmlDoc = parser.parseFromString(content, "text/xml");
            const nodes = xmlDoc.getElementsByTagName("word");
            words = Array.from(nodes).map(node => node.textContent.trim());
        } else {
            words = content.split(/\r?\n/).map(line => line.trim()).filter(line => line);
        }

        if (words.length === 0) {
            alert("No se encontraron palabras válidas en el archivo");
            return;
        }

        // Actualizar lista de palabras excluidas
        excludeWords = [...new Set([...excludeWords, ...words])].sort();
        localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
        renderExcludedWordsPanel();

        alert(`✅ Se importaron ${words.length} palabras exitosamente`);
    }

    //**************************************************************************
    //Nombre: renderExcludedWordsPanel
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas: ninguna (usa la variable global excludeWords)
    //Salidas: ninguna (actualiza el DOM y el localStorage)
    //Prerrequisitos si existen: Debe existir un contenedor con id="normalizer-sidebar"
    //Descripción:
    // Esta función limpia y vuelve a renderizar la lista de palabras excluidas
    // dentro del panel lateral del normalizador. Muestra una lista ordenada
    // alfabéticamente, evitando palabras vacías. Además, asegura que el localStorage
    // se actualice correctamente con la lista limpia y depurada.
    //**************************************************************************
    function renderExcludedWordsPanel() {
        const container = document.getElementById("normalizer-sidebar");
        if (!container) {
            console.warn("[renderExcludedWordsPanel] ❌ No se encontró el contenedor 'normalizer-sidebar'");
            return;
        }

        //Limpiar el contenedor visual
        container.innerHTML = "";
        console.log("[renderExcludedWordsPanel] ✅ Contenedor limpiado.");

        //Limpiar palabras vacías y ordenar
        const sortedWords = excludeWords.filter(w => !!w).sort((a, b) => a.localeCompare(b));
        console.log(`[renderExcludedWordsPanel] 📋 Lista excluida depurada: ${sortedWords.length} palabras`, sortedWords);

        const excludeListSection = document.createElement("div");
        excludeListSection.style.marginTop = "20px";
        excludeListSection.innerHTML = `
        <h4 style="margin-bottom: 5px;">Palabras Excluidas</h4>
        <div style="max-height: 150px; overflow-y: auto; border: 1px solid #ccc; padding: 8px; font-size: 13px; border-radius: 4px;">
            <ul style="margin: 0; padding-left: 18px;" id="excludeWordsList">
                ${sortedWords.map(w => `<li>${w}</li>`).join("")}
            </ul>
        </div>
    `;
        container.appendChild(excludeListSection);
        console.log("[renderExcludedWordsPanel] ✅ Lista renderizada en el DOM.");

        //Guardar en localStorage
        localStorage.setItem("excludeWords", JSON.stringify(sortedWords));
        console.log("[renderExcludedWordsPanel] 💾 excludeWords actualizado en localStorage.");
    }


    //**************************************************************************
    //Nombre: normalizePlaceName (unsafeWindow)
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas:
    //  - name (string): el nombre original del lugar.
    //Salidas:
    //  - string: nombre normalizado, respetando exclusiones y opciones del usuario.
    //Prerrequisitos si existen:
    //  - Debe estar cargada la lista global excludeWords.
    //  - Debe existir un checkbox con id “normalizeArticles” para definir si se normalizan artículos.
    //Descripción:
    //  Esta versión expuesta globalmente permite acceder a la normalización básica del nombre de un lugar
    //  desde otros contextos como el navegador o Tampermonkey. Capitaliza cada palabra, respeta las excluidas
    //  y no aplica normalización a artículos si el checkbox lo indica.
    //  Realiza limpieza básica: reemplazo de pipes, eliminación de espacios dobles y trim final.
    //**************************************************************************
    unsafeWindow.normalizePlaceName = function(name)
    {
        if (!name) return "";

        const normalizeArticles = !document.getElementById("normalizeArticles")?.checked;
        const articles = ["el", "la", "los", "las", "de", "del", "al", "y", "o"];

        const words = name.trim().split(/\s+/);
        const normalizedWords = words.map((word, index) => {
            const lowerWord = word.toLowerCase();

            // Saltar palabras excluidas
            if (excludeWords.includes(word)) return word;

            // Saltar artículos si el checkbox está activo y no es la primera palabra
            if (!normalizeArticles && articles.includes(lowerWord) && index !== 0) {
                return lowerWord;
            }
            //Mayúsculas
            return word.charAt(0).toUpperCase() + word.slice(1).toLowerCase();
        });

        name = normalizedWords.join(" ");
        name = name.replace(/\s*\|\s*/g, " - ");
        name = name.replace(/\s{2,}/g, " ").trim();

        return name;
    };

    //**************************************************************************
    //Nombre: normalizePlaceNameOnly
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas: name (string) – Nombre del lugar a normalizar.
    //Salidas: texto normalizado sin validación ortográfica.
    //Descripción:
    //   Realiza normalización visual del nombre: capitaliza, ajusta espacios,
    //   formatea guiones, paréntesis, y símbolos. No evalúa ortografía ni acentos.
    //**************************************************************************
    function normalizePlaceNameOnly(name) {
        if (!name) return "";
    
        const normalizeArticles = !document.getElementById("normalizeArticles")?.checked;
        const articles = ["el", "la", "los", "las", "de", "del", "al", "y"];
        const words = name.trim().split(/\s+/);
        const isRoman = word => /^(i{1,3}|iv|v|vi{0,3}|ix|x|xi{0,3}|xiv|xv|xvi{0,3}|xix|xx|xxi{0,3}|xxiv|xxv)$/i.test(word);
    
        const normalizedWords = words.map((word, index) => {
            const lowerWord = word.toLowerCase();
    
            // Si contiene "&", convertir a mayúsculas
            if (/^[A-Za-z]&[A-Za-z]$/.test(word)) return word.toUpperCase();
    
            // Verificar si está en la lista de excluidas
            const matchExcluded = excludeWords.find(w => w.toLowerCase() === lowerWord);
            if (matchExcluded) return matchExcluded;
    
            // Si es un número romano, convertir a mayúsculas
            if (isRoman(word)) return word.toUpperCase();
    
            // Si no se deben normalizar artículos y es un artículo, mantener en minúsculas
            if (!normalizeArticles && articles.includes(lowerWord) && index !== 0) return lowerWord;
    
            // Si es un número o un símbolo como "-", no modificar
            if (/^\d+$/.test(word) || word === "-") return word;
    
            // Verificar ortografía usando la API de LanguageTool
            return checkSpellingWithAPI(word)
                .then(errors => {
                    if (errors.length > 0) {
                        const suggestion = errors[0].sugerencia || word;
                        return suggestion.charAt(0).toUpperCase() + suggestion.slice(1).toLowerCase();
                    }
                    return word.charAt(0).toUpperCase() + word.slice(1).toLowerCase();
                })
                .catch(() => word.charAt(0).toUpperCase() + word.slice(1).toLowerCase());
        });
    
        let newName = normalizedWords.join(" ")
            .replace(/\s*\|\s*/g, " - ")
            .replace(/([(["'])\s*([\p{L}])/gu, (match, p1, p2) => p1 + p2.toUpperCase())
            .replace(/\s*-\s*/g, " - ")
            .replace(/\b(\d+)([A-Z])\b/g, (match, num, letter) => num + letter.toUpperCase())
            .replace(/\.$/, "")
            .replace(/&(\s*)([A-Z])/g, (match, space, letter) => "&" + space + letter.toUpperCase());
    
        return newName.replace(/\s{2,}/g, " ").trim();
    }

    //**************************************************************************
    //Nombre: validarOrtografiaConAPI
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas: texto (string) - palabra o frase a evaluar
    //Salidas: Promesa con resultado de corrección ortográfica
    //Descripción: Consulta la API pública de LanguageTool para identificar errores ortográficos
    //**************************************************************************
    function validarOrtografiaConAPI2(texto) {
        return new Promise((resolve, reject) => {
            GM_xmlhttpRequest({
                method: "POST",
                url: "https://api.languagetool.org/v2/check",
                headers: {
                    "Content-Type": "application/x-www-form-urlencoded"
                },
                data: `text=${encodeURIComponent(texto)}&language=es`,
                onload: function (response) {
                    if (response.status === 200) {
                        const result = JSON.parse(response.responseText);
                        resolve(result.matches || []);
                    } else {
                        reject("Error en la API de LanguageTool");
                    }
                },
                onerror: function () {
                    reject("Fallo la solicitud a la API de LanguageTool");
                }
            });
        });
    }

//**************************************************************************
//Nombre: importExcludeList
//Fecha modificación: 2025-03-31
//Autor: mincho77
//Entradas:
//   file (File) – Archivo cargado por el usuario, que contiene una lista de palabras excluidas.
//Salidas:
//   Ninguna (actualiza el array global excludeWords y el localStorage).
//Prerrequisitos:
//   Debe existir un panel visual para mostrar la lista y una función renderExcludeList().
//Descripción:
//   Lee un archivo .txt línea por línea, limpia espacios, elimina duplicados y vacíos,
//   ordena alfabéticamente la lista resultante y actualiza la lista global de palabras
//   excluidas (excludeWords). Guarda la lista en localStorage y actualiza el panel visual.
//**************************************************************************
    function importExcludeList(file) {
        const reader = new FileReader();
        reader.onload = function (e) {
            const newWords = e.target.result
            .split(/\r?\n/)
            .map(w => w.trim())
            .filter(w => w.length > 0); // eliminar vacíos

            excludeWords = [...new Set(newWords)].sort((a, b) => a.localeCompare(b));
            localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
            renderExcludeList(); // actualiza el panel visual
        };
        reader.readAsText(file);
    }

//**************************************************************************
//Nombre: applySpellCorrection
//Fecha modificación: 2025-03-31
//Autor: mincho77
//Entradas:
//   text (string) – Texto que se desea corregir automáticamente con sugerencias ortográficas.
//Salidas:
//   Promise<string> – Texto corregido con las mejores sugerencias aplicadas.
//Prerrequisitos:
//   Debe existir la función checkSpelling que retorna los errores detectados por LanguageTool.
//Descripción:
//   Llama a checkSpelling y aplica la primera sugerencia de reemplazo para cada error ortográfico,
//   procesando los reemplazos de atrás hacia adelante (para evitar desajustes de índice).
//   Devuelve el texto corregido como resultado final.
//**************************************************************************

    function applySpellCorrection(text) {
        return checkSpelling(text).then(data => {
            let corrected = text;
            // Ordenar los matches de mayor a menor offset
            const matches = data.matches.sort((a, b) => b.offset - a.offset);
            matches.forEach(match => {
                if (match.replacements && match.replacements.length > 0) {
                    const replacement = match.replacements[0].value;
                    corrected = corrected.substring(0, match.offset) + replacement + corrected.substring(match.offset + match.length);
                }
            });
            return corrected;
        });
    }



//**************************************************************************
//Nombre: createSidebarTab
//Fecha modificación: 2025-03-31
//Autor: mincho77
//Entradas: Ninguna directa. Usa `W.userscripts` para registrar el tab.
//Salidas: Renderiza un nuevo tab lateral personalizado con interfaz del normalizador.
//Prerrequisitos:
//  - `W.userscripts.registerSidebarTab` debe estar disponible.
//  - `getSidebarHTML()` debe retornar el HTML necesario.
//Descripción:
//  Crea y registra una nueva pestaña en el sidebar de WME con el título
//  "Places Name Normalizer". Al cargar correctamente, inserta el HTML
//  generado por `getSidebarHTML()` y espera a que se renderice completamente
//  para ejecutar los eventos mediante `waitForDOM`.
//**************************************************************************
function createSidebarTab() {
    try {
        // Check if the sidebar system is ready
        if (!W || !W.userscripts) {
            console.error(`[${SCRIPT_NAME}] WME not ready for sidebar creation`);
            return;
        }

        // Check for existing tab and clean up if needed
        const existingTab = document.getElementById("normalizer-tab");
        if (existingTab) {
            console.log(`[${SCRIPT_NAME}] Removing existing tab...`);
            existingTab.remove();
        }

        // Register new tab with error handling
        let registration;
        try {
            registration = W.userscripts.registerSidebarTab("PlacesNormalizer");
        } catch (e) {
            if (e.message.includes("already been registered")) {
                console.warn(`[${SCRIPT_NAME}] Tab registration conflict, attempting cleanup...`);
                // Additional cleanup could go here
                return;
            }
            throw e;
        }

        const { tabLabel, tabPane } = registration;
        
        if (!tabLabel || !tabPane) {
            throw new Error("Tab registration failed to return required elements");
        }

        // Configure tab
        tabLabel.innerHTML = `
        <img src=""
        style="height: 16px; vertical-align: middle; margin-right: 5px;">
        NrmliZer
        `;
        tabLabel.title = "Places Name Normalizer";
        
        // Set content and attach events
        tabPane.innerHTML = getSidebarHTML();
        waitForDOM("#normalizer-tab", (element) => {
            attachEvents();
            console.log(`[${SCRIPT_NAME}] Tab created and events attached`);
        }, 500, 10);

    } catch (error) {
        console.error(`[${SCRIPT_NAME}] Error in createSidebarTab:`, error);
    }
}

   

//**************************************************************************
//Nombre: attachEvents
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: Ninguna
//Salidas: Ninguna
//Prerrequisitos si existen:
// - Deben existir en el DOM los elementos con los siguientes IDs:
//   "normalizeArticles", "maxPlacesInput", "addExcludeWord", "scanPlaces",
//   "hiddenImportInput", "importExcludeWordsUnifiedBtn" y "exportExcludeWords".
// - Debe existir la función handleImportList y la función scanPlaces.
// - Debe estar definida la variable global excludeWords y la función renderExcludedWordsPanel.
//Descripción:
// Esta función adjunta los event listeners necesarios para gestionar la interacción del usuario
// con el panel del normalizador de nombres. Se encargan de:
//   - Actualizar la opción de normalizar artículos al cambiar el estado del checkbox.
//   - Modificar el número máximo de lugares a procesar a través de un input.
//   - Exportar la lista de palabras excluidas a un archivo XML.
//   - Añadir nuevas palabras a la lista de palabras excluidas, evitando duplicados, y actualizar el panel.
//   - Activar el botón unificado para la importación de palabras excluidas mediante un input oculto.
//   - Ejecutar la función de escaneo de lugares al hacer clic en el botón correspondiente.
//**************************************************************************
    function attachEvents()
    {
        console.log(`[${SCRIPT_NAME}] Adjuntando eventos...`);

        const normalizeArticlesCheckbox = document.getElementById("normalizeArticles");
        const maxPlacesInput = document.getElementById("maxPlacesInput");
        const addExcludeWordButton = document.getElementById("addExcludeWord");
        const scanPlacesButton = document.getElementById("scanPlaces");
        const hiddenInput = document.getElementById("hiddenImportInput");
        const importButtonUnified = document.getElementById("importExcludeWordsUnifiedBtn");

        // Validación de elementos necesarios
        if (!normalizeArticlesCheckbox || !maxPlacesInput || !addExcludeWordButton || !scanPlacesButton) {
            console.error(`[${SCRIPT_NAME}] Error: No se encontraron elementos en el DOM.`);
            return;
        }

        // ✅ Evento: cambiar estado de "no normalizar artículos"
        normalizeArticlesCheckbox.addEventListener("change", (e) => {
            normalizeArticles = e.target.checked;
        });

        // ✅ Evento: cambiar número máximo de places
        maxPlacesInput.addEventListener("input", (e) => {
            maxPlaces = parseInt(e.target.value, 10);
        });

        // ✅ Evento: exportar palabras excluidas a XML
        document.getElementById("exportExcludeWords").addEventListener("click", () => {
            const savedWords = JSON.parse(localStorage.getItem("excludeWords")) || [];
            if (savedWords.length === 0) {
                alert("No hay palabras excluidas para exportar.");
                return;
            }
            const sortedWords = [...savedWords].sort((a, b) => a.localeCompare(b));

            const xmlContent = `<?xml version="1.0" encoding="UTF-8"?>
<ExcludedWords>
  ${sortedWords.map(word => `  <word>${word}</word>`).join("\n  ")}
</ExcludedWords>`;

            const blob = new Blob([xmlContent], { type: "application/xml" });
            const url = URL.createObjectURL(blob);
            const link = document.createElement("a");

            link.href = url;
            link.download = "excluded_words.xml";
            document.body.appendChild(link); // Correctly appends the link
            link.click();
            document.body.removeChild(link); // Correctly removes the link
        });

        // ✅ Evento: añadir palabra excluida sin duplicados
        addExcludeWordButton.addEventListener("click", () => {
            const wordInput = document.getElementById("excludeWord") || document.getElementById("excludedWord");
            const word = wordInput?.value.trim();

            if (!word) return;

            const lowerWord = word.toLowerCase();
            const alreadyExists = excludeWords.some(w => w.toLowerCase() === lowerWord);
            if (!alreadyExists) {
                excludeWords.push(word);
                localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
                renderExcludedWordsPanel(); // Usar renderExcludedWordsPanel en lugar de updateExcludeList
            }

            wordInput.value = "";
        });

        // ✅ Evento: nuevo botón unificado de importación
        importButtonUnified.addEventListener("click", () => {
            hiddenInput.click(); // abre el file input oculto
        });

        hiddenInput.addEventListener("change", () => {
            handleImportList(); // ✅ Llama a la función handleImportList al importar
        });

        // ✅ Evento: escanear lugares
        scanPlacesButton.addEventListener("click", scanPlaces);
    }


    function updateExcludeList() {
        const list = document.getElementById("excludedWordsList");
        if (!list) return;

        // Ordena una copia del array para no alterar el original
        const sortedWords = [...excludeWords].sort((a, b) => a.toLowerCase().localeCompare(b.toLowerCase()));

        list.innerHTML = sortedWords.map(word => `<li>${word}</li>`).join("");
    }

//**************************************************************************
//Nombre: scanPlaces
//Fecha modificación: 2025-04-02
//Autor: mincho77
//Entradas: ninguna directamente (usa datos del modelo WME y del input #maxPlacesInput)
//Salidas: abre panel flotante con lugares que deben ser normalizados
//Descripción:
// Escanea los lugares visibles en el WME, normaliza los nombres y verifica
// si les falta una tilde en las palabras que lo requieren. Si se selecciona
// "Revisar solo tildes", utiliza la función evaluarOrtografiaConTildes.
//**************************************************************************
function scanPlaces() {
    console.log("scanPlaces ejecutado");
    if (!W || !W.model || !W.model.venues || !W.model.venues.objects) {
        console.error(`[${SCRIPT_NAME}] WME no está listo.`);
        return;
    }

    const allPlaces = Object.values(W.model.venues.objects);
    console.log(`[${SCRIPT_NAME}] Total de lugares encontrados: ${allPlaces.length}`);

    const maxPlaces = parseInt(document.getElementById("maxPlacesInput").value, 10) || 20;
    const checkOnlyTildes = document.getElementById("checkOnlyTildes")?.checked; // Verificar el estado del checkbox
    const placesToNormalize = allPlaces
        .filter(p => {
            let isValid = true;

            if (!p) {
                console.log("Lugar descartado: p es null o undefined", p);
                isValid = false;
            } else if (typeof p.getID !== "function") {
                console.log("Lugar descartado: p.getID no es una función", p);
                isValid = false;
            } else if (!p.attributes) {
                console.log("Lugar descartado: p.attributes es null o undefined", p);
                isValid = false;
            } else if (typeof p.attributes.name !== "string") {
                console.log("Lugar descartado: p.attributes.name no es una cadena", p);
                isValid = false;
            } else if (!p.attributes.name.trim()) {
                console.log("Lugar descartado: p.attributes.name está vacío después de trim", p);
                isValid = false;
            }

            return isValid;
        })
        .slice(0, maxPlaces)
        .map(place => ({
            id: place.getID(),
            name: place.attributes.name,
            attributes: place.attributes,
            place
        }));

    console.log("placesToNormalize:", placesToNormalize);

    toggleSpinner(true, 'Revisando ortografía...', 0); // Mostrar spinner al inicio con 0%

    const totalPlaces = placesToNormalize.length;
    let processedPlaces = 0;
    let results = []; // Almacenar resultados parciales

    function processPlace(index) {
        if (index >= totalPlaces) {
            toggleSpinner(false); // Ocultar spinner al terminar
            if (results.length === 0) {
                alert("No se encontraron Places que requieran cambio.");
            } else {
                openFloatingPanel(results); // Mostrar resultados parciales
            }
            return;
        }

        const place = placesToNormalize[index];
        const originalName = place.name;
        const normalized = normalizePlaceName(originalName);

        const ortografia = checkOnlyTildes
            ? evaluarOrtografiaConTildes(normalized) // Usar la función para revisar solo tildes
            : evaluarOrtografiaNombre(normalized);

        Promise.resolve(ortografia)
            .then(ortografiaResult => {
                processedPlaces++;
                const progress = Math.round((processedPlaces / totalPlaces) * 100);
                toggleSpinner(true, 'Revisando ortografía...', progress); // Actualizar progreso

                results.push({
                    id: place.id,
                    originalName,
                    newName: normalized,
                    hasSpellingWarning: ortografiaResult.hasSpellingWarning,
                    spellingWarnings: ortografiaResult.spellingWarnings
                });

                processPlace(index + 1); // Procesar el siguiente lugar
            })
            .catch(error => {
                console.error("Error durante el escaneo de lugares:", error);
                alert(`Error durante el escaneo de lugares: ${error}`);
                toggleSpinner(false); // Ocultar spinner en caso de error
                openFloatingPanel(results); // Mostrar resultados parciales
            });
    }

    processPlace(0); // Iniciar el procesamiento desde el primer lugar
}
    //**************************************************************************
    //Nombre: applyNormalization
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas: Ninguna
    //Salidas: Ninguna
    //Prerrequisitos si existen:
    // - Debe existir un elemento en el DOM con las clases .normalize-checkbox, .delete-checkbox y .new-name-input.
    // - El objeto global W debe estar disponible, incluyendo W.model.venues, W.model.actionManager y W.controller.
    // - Deben estar definidos los módulos "Waze/Action/UpdateObject" y "Waze/Action/DeleteObject" (accesibles mediante require()).
    // - Debe existir la variable placesToNormalize, que contiene datos de los lugares a normalizar, incluyendo sugerencias ortográficas.
    //Descripción:
    // Esta función aplica la normalización y/o eliminación de nombres de lugares en el Waze Map Editor
    // según las selecciones realizadas en el panel flotante. Primero, obtiene los checkboxes seleccionados
    // para normalización y eliminación. Si no hay ningún elemento seleccionado, se informa y se cancela la operación.
    // Si se han seleccionado TODOS los checkboxes de eliminación, se solicita una confirmación adicional.
    // Para cada checkbox de normalización seleccionado, se verifica si se debe aplicar la sugerencia ortográfica
    // (cuando se ha hecho clic en el botón correspondiente) o el nombre completo modificado, y se actualiza el lugar
    // mediante la acción de actualización. Posteriormente, se procesan los checkboxes de eliminación aplicando la
    // acción de eliminación a los lugares correspondientes. Si se realizaron cambios, se marca el modelo como modificado.
    // Finalmente, se cierra el panel flotante.
    //**************************************************************************
    function applyNormalization() {
        const normalizeCheckboxes = document.querySelectorAll(".normalize-checkbox:checked");
        const deleteCheckboxes = document.querySelectorAll(".delete-checkbox:checked");
        let changesMade = false;
    
        if (normalizeCheckboxes.length === 0 && deleteCheckboxes.length === 0) {
            alert("No hay lugares seleccionados para normalizar o eliminar.");
            return;
        }
    
        // Procesar normalización
        normalizeCheckboxes.forEach(cb => {
            const index = cb.dataset.index;
            const input = document.querySelector(`.new-name-input[data-index="${index}"]`);
            const newName = input?.value?.trim();
            const placeId = input?.getAttribute("data-place-id");
            const place = W.model.venues.getObjectById(placeId);
    
            if (!place || !place.attributes?.name) {
                console.warn(`No se encontró el lugar con ID: ${placeId}`);
                return;
            }
    
            const currentName = place.attributes.name.trim();
            if (currentName !== newName) {
                try {
                    const UpdateObject = require("Waze/Action/UpdateObject");
                    const action = new UpdateObject(place, { name: newName });
                    W.model.actionManager.add(action);
                    console.log(`Nombre actualizado: "${currentName}" → "${newName}"`);
                    changesMade = true;
                } catch (error) {
                    console.error("Error aplicando la acción de actualización:", error);
                }
            }
        });
    
        // Procesar eliminación
        deleteCheckboxes.forEach(cb => {
            const index = cb.dataset.index;
            const placeId = document.querySelector(`.new-name-input[data-index="${index}"]`)?.getAttribute("data-place-id");
            const place = W.model.venues.getObjectById(placeId);
    
            if (!place) {
                console.warn(`No se encontró el lugar con ID para eliminar: ${placeId}`);
                return;
            }
    
            try {
                const DeleteObject = require("Waze/Action/DeleteObject");
                const deleteAction = new DeleteObject(place);
                W.model.actionManager.add(deleteAction);
                console.log(`Lugar eliminado: ${placeId}`);
                changesMade = true;
            } catch (error) {
                console.error("Error eliminando el lugar:", error);
            }
        });
    
        if (changesMade) {
            alert("Cambios aplicados correctamente.");
        } else {
            alert("No se realizaron cambios.");
        }
    
        // Cerrar el panel flotante
        const panel = document.getElementById("normalizer-floating-panel");
        if (panel) panel.remove();
    }

    //**************************************************************************
    //Nombre: isSimilar
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas:
    //  - a (string): Primera palabra a comparar.
    //  - b (string): Segunda palabra a comparar.
    //Salidas:
    //  - boolean: Retorna true si las palabras son consideradas similares de forma leve; de lo contrario, retorna false.
    //Prerrequisitos si existen: Ninguno.
    //Descripción:
    // Esta función evalúa la similitud leve entre dos palabras. Primero, verifica si ambas palabras son
    // idénticas, en cuyo caso retorna true. Luego, comprueba si la diferencia en la cantidad de caracteres
    // entre ambas es mayor a 2; si es así, retorna false. Posteriormente, compara carácter por carácter
    // hasta el largo mínimo de las palabras, contando las diferencias. Si el número de discrepancias
    // excede 2, se considera que las palabras no son similares y retorna false; en caso contrario, retorna true.
    //**************************************************************************
    function isSimilar(a, b)
    {
        if (a === b) return true;
        if (Math.abs(a.length - b.length) > 2) return false;

        let mismatches = 0;
        for (let i = 0; i < Math.min(a.length, b.length); i++) {
            if (a[i] !== b[i]) mismatches++;
            if (mismatches > 2) return false;
        }

        return true;
    }


    //**************************************************************************
    //Nombre: normalizePlaceName
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas: name (string) - Nombre del lugar
    //Salidas: texto normalizado (string)
    //Descripción: Normaliza un nombre aplicando capitalización, manejo de artículos, números y paréntesis.
    //**************************************************************************
    function normalizePlaceName(name) {
        if (!name) return "";

        const normalizeArticles = !document.getElementById("normalizeArticles")?.checked;
        const articles = ["el", "la", "los", "las", "de", "del", "al", "y"];
        const words = name.trim().split(/\s+/);
        const isRoman = word => /^(i{1,3}|iv|v|vi{0,3}|ix|x|xi{0,3}|xiv|xv|xvi{0,3}|xix|xx|xxi{0,3}|xxiv|xxv)$/i.test(word);

        const normalizedWords = words.map((word, index) => {
            const lowerWord = word.toLowerCase();

            // Si la palabra es un número, no la analizamos
            if (/^\d+$/.test(word)) return word;

            // Si la palabra está en la lista de exclusión, no la modificamos
            if (excludeWords.some(w => w.toLowerCase() === lowerWord)) return word;

            // Si es un número romano, lo dejamos en mayúsculas
            if (isRoman(word)) return word.toUpperCase();

            // Si es una sigla con estructura T&T o a&A, convertirla a mayúsculas
            if (/^[A-Za-z]&[A-Za-z]$/.test(word)) return word.toUpperCase();

            // Si es una sigla con apóstrofe como "E's", también la dejamos igual
            if (/^[A-Z]'[A-Z][a-z]+$/.test(word)) return word;

            // Si no se deben normalizar artículos y es un artículo, lo dejamos en minúsculas (excepto la primera palabra)
            if (!normalizeArticles && articles.includes(lowerWord) && index !== 0) return lowerWord;

            // Si es un número seguido de letras, lo dejamos igual
            if (/^\d+[A-Z][a-zA-Z]*$/.test(word)) return word;

            // Si está entre paréntesis y es todo mayúsculas o minúsculas, lo dejamos igual
            if (/^\(.*\)$/.test(word)) {
                const inner = word.slice(1, -1);
                if (inner === inner.toUpperCase() || inner === inner.toLowerCase()) return word;
            }

            // Capitalizamos la palabra (primera letra en mayúscula, el resto en minúscula)
            return word.charAt(0).toUpperCase() + word.slice(1).toLowerCase();
        });

        let newName = normalizedWords.join(" ")
            .replace(/\s*\|\s*/g, " - ")
            .replace(/([(["'])\s*([\p{L}])/gu, (match, p1, p2) => p1 + p2.toUpperCase())
            .replace(/\s*-\s*/g, " - ")
            .replace(/\b(\d+)([A-Z])\b/g, (match, num, letter) => num + letter.toUpperCase())
            .replace(/\.$/, "")
            .replace(/&(\s*)([A-Z])/g, (match, space, letter) => "&" + space + letter.toUpperCase());

        return newName.replace(/\s{2,}/g, " ").trim();
    }

    // Para exponer al contexto global real desde Tampermonkey
    unsafeWindow.normalizePlaceName = normalizePlaceName;



    //**************************************************************************
    //Nombre: openFloatingPanel
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas: placesToNormalize (array de lugares con nombre original y sugerencias)
    //Salidas: Panel flotante con opciones de normalización y eliminación
    //Prerrequisitos si existen: Debe haberse definido normalizePlaceName y cargado excludeWords correctamente
    //Descripción: Crea un panel interactivo donde se presentan los lugares que requieren cambios,
    //             permitiendo su corrección o eliminación. Solo muestra lugares que requieren cambio
    //             y errores ortográficos verdaderos.
    //**************************************************************************
    function openFloatingPanel(placesToNormalize) {
        const panel = document.createElement("div");
        panel.id = "normalizer-floating-panel";
        panel.style.cssText = `
            position: fixed;
            top: 50%;
            left: 50%;
            transform: translate(-50%, -50%);
            background: white;
            padding: 20px;
            border: 1px solid #ccc;
            border-radius: 8px;
            box-shadow: 0 2px 10px rgba(0,0,0,0.1);
            z-index: 1000;
            max-height: 80vh;
            overflow-y: auto;
            min-width: 800px;
        `;

        let panelContent = `
            <style>
                #normalizer-table { width: 100%; border-collapse: collapse; }
                #normalizer-table th, #normalizer-table td { padding: 8px; border: 1px solid #ddd; }
                #normalizer-table tr:nth-child(even) { background-color: #f9f9f9; }
                .warning-row { background-color: #fff3cd !important; }
                .close-btn {
                    position: absolute;
                    top: 10px;
                    right: 10px;
                    background: #ccc;
                    color: black;
                    border: none;
                    border-radius: 4px;
                    width: 25px;
                    height: 25px;
                    font-size: 16px;
                    font-weight: bold;
                    cursor: pointer;
                    text-align: center;
                    line-height: 25px;
                    transition: background-color 0.3s ease;
                }
                .close-btn:hover {
                    background: #bbb;
                }
                .footer-buttons {
                    display: flex;
                    justify-content: center;
                    gap: 10px;
                    margin-top: 20px;
                }
            </style>
            <button class="close-btn" id="close-panel-btn">X</button>
            <h3>Places to Normalize</h3>
            <table id="normalizer-table">
                <thead>
                    <tr>
                        <th>Normalizar</th>
                        <th>Eliminar</th>
                        <th>Estado</th>
                        <th>Nombre Original</th>
                        <th>Nombre Sugerido</th>
                        <th>Corrección</th>
                        <th>Acción</th>
                    </tr>
                </thead>
                <tbody>
        `;

        // Procesar cada lugar y sus advertencias
        placesToNormalize.forEach((place, index) => {
            const { originalName, newName, spellingWarnings = [] } = place;

            // Crear una fila por cada advertencia ortográfica
            spellingWarnings.forEach((warning, warningIndex) => {
                const suggestionId = `suggestion-${index}-${warningIndex}`;

                panelContent += `
                <tr class="warning-row">
                    <td style="text-align: center;">
                        <input type="checkbox" class="normalize-checkbox"
                            data-index="${index}"
                            data-warning-index="${warningIndex}"
                            data-suggestion-id="${suggestionId}">
                    </td>
                    <td style="text-align: center;">
                        <input type="checkbox" class="delete-checkbox"
                            data-index="${index}"
                            data-warning-index="${warningIndex}">
                    </td>
                    <td style="text-align: center;">⚠️</td>
                    <td id="name-cell-${index}-${warningIndex}">${originalName}</td>
                    <td>
                        <input type="text" class="new-name-input"
                            data-index="${index}"
                            data-warning-index="${warningIndex}"
                            data-place-id="${place.id}"
                            data-suggestion-id="${suggestionId}"
                            value="${newName}">
                    </td>
                    <td>${warning.original} → ${warning.sugerida} (${warning.tipo})</td>
                    <td style="text-align: center;">
                        <button class="apply-suggestion-btn"
                                data-index="${index}"
                                data-warning-index="${warningIndex}"
                                title="Corregir ortografía de la palabra"
                                style="background-color: #4CAF50; color: white; border: none; padding: 5px 10px; border-radius: 4px; cursor: pointer;">
                            Fix
                        </button>
                        <button class="add-exclude-btn"
                                data-word="${warning.original}"
                                title="Adicionar palabra excluida nueva"
                                style="background-color: #007bff; color: white; border: none; padding: 5px 10px; border-radius: 4px; cursor: pointer;">
                            Add
                        </button>
                    </td>
                </tr>`;
            });
        });

        panelContent += `
                    </tbody>
                </table>
                <div class="footer-buttons">
                    <button class="apply-changes-btn" id="apply-changes-btn">✔️ Apply Changes</button>
                    <button class="cancel-btn" id="cancel-btn">❌ Cancel</button>
                </div>
            `;

        panel.innerHTML = panelContent;
        document.body.appendChild(panel);

        // Eventos para los botones
        document.getElementById('close-panel-btn').addEventListener('click', () => {
            panel.remove(); // Cerrar el panel flotante
        });

        document.getElementById('apply-changes-btn').addEventListener('click', () => {
            window.applyNormalization(); // Llamar a la función global para aplicar cambios
        });

        document.getElementById('cancel-btn').addEventListener('click', () => {
            panel.remove(); // Cerrar el panel flotante
        });

        // Asignar eventos a los botones "Fix"
        document.querySelectorAll('.apply-suggestion-btn').forEach(button => {
            button.addEventListener('click', function () {
                const index = this.dataset.index;
                const warningIndex = this.dataset.warningIndex;
                const input = document.querySelector(
                    `.new-name-input[data-index="${index}"][data-warning-index="${warningIndex}"]`
                );
                const checkbox = document.querySelector(
                    `.normalize-checkbox[data-index="${index}"][data-warning-index="${warningIndex}"]`
                );

                if (input && checkbox) {
                    // Aplicar la corrección ortográfica al campo de entrada
                    const warning = placesToNormalize[index].spellingWarnings[warningIndex];
                    input.value = input.value.replace(warning.original, warning.sugerida);

                    checkbox.checked = true; // Marcar el checkbox
                }
            });
        });

        // Asignar eventos a los botones "Add Excld Word"
        document.querySelectorAll('.add-exclude-btn').forEach(button => {
            button.addEventListener('click', function () {
                const word = this.dataset.word;

                // Verificar si es una sigla con formato X&X
                if (/^[A-Za-z]&[A-Za-z]$/.test(word)) {
                    alert("⚠️ No es necesario adicionar palabras excluidas que tengan '&'.");
                    return;
                }

                if (!excludeWords.includes(word)) {
                    excludeWords.push(word);
                    excludeWords.sort((a, b) => a.localeCompare(b)); // Ordenar alfabéticamente
                    localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
                    renderExcludedWordsPanel(); // Actualizar el panel lateral

                    // Mostrar popup en el centro del panel flotante
                    const panel = document.getElementById("normalizer-floating-panel");
                    const popup = document.createElement('div');
                    popup.textContent = `✅ The Word "${word}" has been added to the exclusion list.`;
                    popup.style.position = 'absolute';
                    popup.style.top = '50%';
                    popup.style.left = '50%';
                    popup.style.transform = 'translate(-50%, -50%)';
                    popup.style.backgroundColor = '#4CAF50';
                    popup.style.color = 'white';
                    popup.style.padding = '10px 20px';
                    popup.style.borderRadius = '5px';
                    popup.style.zIndex = '10000';
                    popup.style.opacity = '1';
                    popup.style.transition = 'opacity 1s ease-in-out';

                    panel.appendChild(popup);

                    setTimeout(() => {
                        popup.style.opacity = '0';
                        setTimeout(() => panel.removeChild(popup), 1000);
                    }, 2000);

                    // Bloquear el botón y cambiar su texto
                    this.textContent = 'Added';
                    this.disabled = true;
                    this.style.backgroundColor = '#6c757d'; // Cambiar color a gris
                    this.style.cursor = 'not-allowed';
                } else {
                    alert(`⚠️ The word "${word}" is already on the exclusion list.`);
                }
            });
        });
    }
    //**************************************************************************
    //Nombre: loadExcludeWordsFromXML
    //Fecha modificación:
    //Autor: mincho77
    //Entradas:
    //  - callback: función opcional que se ejecuta una vez se cargan y procesan las palabras excluidas.
    //Salidas: ninguna directa. Actualiza la variable global `excludeWords`.
    //Prerrequisitos:
    //  - Debe existir un archivo llamado 'excludeWords.xml' accesible por fetch.
    //  - Debe estar definida la variable global `excludeWords`.
    //Descripción:
    //  Carga un archivo XML que contiene una lista de palabras excluidas.
    //  Combina las palabras nuevas con las que ya están guardadas en localStorage
    //  y actualiza la lista global `excludeWords`. Si el XML no puede ser cargado,
    //  se usa únicamente el contenido almacenado localmente como respaldo.
    //**************************************************************************
    function loadExcludeWordsFromXML(callback) {
        fetch("excludeWords.xml")
            .then(response => response.text()) // Corrected the syntax here
            .then(xmlText => {
            const parser = new DOMParser();
            const xmlDoc = parser.parseFromString(xmlText, "text/xml");
            const wordNodes = xmlDoc.getElementsByTagName("word");
            const wordsFromXML = Array.from(wordNodes).map(node => node.textContent.trim());

            const existing = JSON.parse(localStorage.getItem("excludeWords")) || [];
            excludeWords = [...new Set([...existing, ...wordsFromXML])].sort((a, b) => a.localeCompare(b));
            localStorage.setItem("excludeWords", JSON.stringify(excludeWords));

            if (callback) callback();
        })
            .catch(() => {
            console.warn("⚠️ No se pudo cargar excludeWords.xml. Solo se usará localStorage.");
            excludeWords = JSON.parse(localStorage.getItem("excludeWords")) || ["EDS", "IPS", "McDonald's", "EPS"];
            localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
            if (callback) callback();
        });
    }

    function exportExcludeWordsToXML() {
        const xmlContent = `<?xml version="1.0" encoding="UTF-8"?>
<ExcludedWords>
${excludeWords.map(word => `  <word>${word}</word>`).join("\n")}
</ExcludedWords>`;

        const blob = new Blob([xmlContent], { type: "application/xml" });
        const url = URL.createObjectURL(blob); // ✅ ESTA LÍNEA ESTABA FALTANDO
        const link = document.createElement("a");
        link.href = url;
        link.download = "excludeWords.xml";
        document.body.appendChild(link);
        link.click();
        document.body.removeChild(link);
}


    function showFloatingMessage(message) {
        const msg = document.createElement("div");
        msg.textContent = message;
        msg.style.position = "fixed";
        msg.style.bottom = "30px";
        msg.style.left = "50%";
        msg.style.transform = "translateX(-50%)";
        msg.style.backgroundColor = "#333";
        msg.style.color = "#fff";
        msg.style.padding = "10px 20px";
        msg.style.borderRadius = "5px";
        msg.style.zIndex = 9999;
        msg.style.opacity = "0.95";
        msg.style.transition = "opacity 1s ease-in-out";

        document.body.appendChild(msg);

        setTimeout(() => {
            msg.style.opacity = "0";
            setTimeout(() => document.body.removeChild(msg), 1000);
        }, 3000);
    }
    //**************************************************************************
    //Nombre: waitForWME
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas: Ninguna
    //Salidas: Ninguna
    //Prerrequisitos si existen:
    // - El objeto global W debe estar disponible.
    // - Deben estar definidas las funciones: initializeExcludeWords, createSidebarTab y renderExcludedWordsPanel.
    //Descripción:
    // Esta función espera a que el entorno de edición de Waze (WME) esté completamente cargado,
    // verificando que existan los objetos necesarios para iniciar el script.
    // Una vez disponible, inicializa la lista de palabras excluidas desde localStorage,
    // crea el tab lateral personalizado y renderiza la lista visual de palabras excluidas.
    // Si WME aún no está listo, vuelve a intentar cada 1000 ms.
    //**************************************************************************
    function waitForWME() {
        if (W && W.userscripts && W.model && W.model.venues) {
            console.log(`[${SCRIPT_NAME}] Inicializando v${VERSION}`);

            initializeExcludeWords(); // ⚠️ Usa solo localStorage
            createSidebarTab();

            waitForDOM("#normalizer-tab", () => {
                console.log("[waitForWME] 🧩 Sidebar listo, renderizando palabras excluidas");

                renderExcludedWordsPanel();       // Muestra las palabras
                setupDragAndDropImport();         // Activa drag & drop

                //Evita que se abra el archivo si cae fuera del área
              //  window.addEventListener("dragover", e => e.preventDefault(), false);
             //   window.addEventListener("drop", e => e.preventDefault(), false);
            });

        } else {
            console.log(`[${SCRIPT_NAME}] Esperando que WME esté listo...`);
            setTimeout(waitForWME, 1000);
        }
    }


    //**************************************************************************
    //Nombre: cleanupEventListeners
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas: Ninguna
    //Salidas: Ninguna
    //Prerrequisitos si existen:
    // - Debe existir un elemento en el DOM con el id "normalizer-floating-panel".
    //Descripción:
    // Esta función limpia los event listeners asociados al panel flotante de normalización.
    // Lo hace clonando el nodo del panel y reemplazándolo en el DOM, lo cual elimina todos
    // los listeners previamente asignados a ese nodo, evitando posibles fugas de memoria
    // o comportamientos inesperados.
    //**************************************************************************
    function cleanupEventListeners() {
        const panel = document.getElementById("normalizer-floating-panel");
        if (panel) {
            const clone = panel.cloneNode(true);
            panel.parentNode.replaceChild(clone, panel);
        }
    }  

    //**************************************************************************
    //Nombre: normalizePlaceName (unsafeWindow)
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas:
    //  - name (string): el nombre original del lugar.
    //Salidas:
    //  - string: nombre normalizado, respetando exclusiones y opciones del usuario.
    //Prerrequisitos si existen:
    //  - Debe estar cargada la lista global excludeWords.
    //  - Debe existir un checkbox con id “normalizeArticles” para definir si se normalizan artículos.
    //Descripción:
    //  Esta versión expuesta globalmente permite acceder a la normalización básica del nombre de un lugar
    //  desde otros contextos como el navegador o Tampermonkey. Capitaliza cada palabra, respeta las excluidas
    //  y no aplica normalización a artículos si el checkbox lo indica.
    //  Realiza limpieza básica: reemplazo de pipes, eliminación de espacios dobles y trim final.
    //**************************************************************************
    unsafeWindow.normalizePlaceName = function(name)
    {
        if (!name) return "";

        const normalizeArticles = !document.getElementById("normalizeArticles")?.checked;
        const articles = ["el", "la", "los", "las", "de", "del", "al", "y", "o"];

        const words = name.trim().split(/\s+/);
        const normalizedWords = words.map((word, index) => {
            const lowerWord = word.toLowerCase();

            // Saltar palabras excluidas
            if (excludeWords.includes(word)) return word;

            // Saltar artículos si el checkbox está activo y no es la primera palabra
            if (!normalizeArticles && articles.includes(lowerWord) && index !== 0) {
                return lowerWord;
            }
            //Mayúsculas
            return word.charAt(0).toUpperCase() + word.slice(1).toLowerCase();
        });

        name = normalizedWords.join(" ");
        name = name.replace(/\s*\|\s*/g, " - ");
        name = name.replace(/\s{2,}/g, " ").trim();

        return name;
    };



//**************************************************************************
//Nombre: waitForDOM
//Fecha modificación: 2025-03-31
//Autor: mincho77
//Entradas:
//   selector (string): Selector CSS del nodo a esperar.
//   callback (function): Función a ejecutar cuando el nodo esté disponible.
//   interval (int, opcional): Tiempo entre intentos en milisegundos. Default: 500ms.
//   maxAttempts (int, opcional): Máximo de intentos antes de abortar. Default: 10.
//Salidas: Ejecuta el callback si encuentra el selector dentro del tiempo.
//Descripción:
//   Esta función monitorea el DOM en intervalos constantes hasta que
//   encuentra un nodo que coincida con el selector. Si lo encuentra,
//   ejecuta el callback con ese nodo. Si no, muestra un error por consola.
//**************************************************************************
    function waitForDOM(selector, callback, interval = 500, maxAttempts = 10)
    {
        let attempts = 0;
        const checkExist = setInterval(() => {
            const element = document.querySelector(selector);
            if (element) {
                clearInterval(checkExist);
                callback(element);
            } else if (attempts >= maxAttempts) {
                clearInterval(checkExist);
                console.error(`[${SCRIPT_NAME}] Error: No se encontraron elementos en el DOM después de ${maxAttempts} intentos.`);
            }
            attempts++;
        }, interval);
    }

    //**************************************************************************
    //Nombre: getSidebarHTML
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas: Ninguna.
    //Salidas: Retorna un string con HTML que define el contenido del panel lateral del script.
    //Prerrequisitos si existen: Debe estar disponible el valor de las variables globales:
    // - normalizeArticles (boolean): Define si los artículos deben ser normalizados o no.
    // - maxPlaces (number): Número máximo de lugares a escanear.
    //Descripción:
    // Esta función construye el HTML que se inyecta en el panel lateral (sidebar) de WME.
    // Incluye controles para:
    // - Activar o desactivar normalización de artículos ("el", "la", etc).
    // - Definir la cantidad máxima de lugares a procesar.
    // - Agregar palabras excluidas manualmente.
    // - Exportar palabras excluidas a un archivo XML.
    // - Importar una lista de palabras excluidas desde archivo XML.
    // - Disparar el escaneo de lugares para normalización.
    // La lista de palabras excluidas **no se renderiza aquí directamente**, sino en el div
    // con id "normalizer-sidebar" para permitir que sea manejada dinámicamente.
    //**************************************************************************
    function getSidebarHTML() {
        return `
        <div id="normalizer-tab">
          <h4>Places Name Normalizer <span style="font-size:11px;">v${VERSION}</span></h4>
    
          <div>
            <input type="checkbox" id="normalizeArticles" ${normalizeArticles ? "checked" : ""}>
            <label for="normalizeArticles">No Normalizar artículos (el, la, los, ...)</label>
          </div>
    
          <div>
            <label>Máximo de Places a buscar: </label>
            <input type='number' id='maxPlacesInput' value='${maxPlaces}' min='1' max='800' style='width: 60px;'>
          </div>
    
          <div>
            <label>Palabras Excluidas:</label>
            <input type='text' id='excludeWord' style='width: 120px;'>
            <button id='addExcludeWord'>Add Word</button>
            <div id="normalizer-sidebar" style="margin-top: 20px;"></div>
            <button id="exportExcludeWords" style="margin-top: 5px;">Export Words</button>
            <br>
            <button id="importExcludeWordsUnifiedBtn" style="margin-top: 5px;">Import List...</button>
            <input type="file" id="hiddenImportInput" accept=".xml" style="display: none;">
          </div>
    
          <div>
            <input type="checkbox" id="checkOnlyTildes" checked>
            <label for="checkOnlyTildes">Revisar solo tildes</label>
          </div>
    
          <div id="drop-zone" style="border: 2px dashed #ccc; border-radius: 6px; padding: 15px; margin: 15px 0; text-align: center; font-style: italic; color: #555; background-color: #f8f9fa; transition: all 0.3s ease;">
            📂 Arrastra aquí tu archivo .txt o .xml para importar palabras excluidas
          </div>
    
          <hr>
          <button id="scanPlaces">Scan...</button>
        </div>
      `;
    }

//**************************************************************************
//Nombre: attachEvents
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: Ninguna
//Salidas: Ninguna
//Prerrequisitos si existen:
// - Deben existir en el DOM los elementos con los siguientes IDs:
//   "normalizeArticles", "maxPlacesInput", "addExcludeWord", "scanPlaces",
//   "hiddenImportInput", "importExcludeWordsUnifiedBtn" y "exportExcludeWords".
// - Debe existir la función handleImportList y la función scanPlaces.
// - Debe estar definida la variable global excludeWords y la función renderExcludedWordsPanel.
//Descripción:
// Esta función adjunta los event listeners necesarios para gestionar la interacción del usuario
// con el panel del normalizador de nombres. Se encargan de:
//   - Actualizar la opción de normalizar artículos al cambiar el estado del checkbox.
//   - Modificar el número máximo de lugares a procesar a través de un input.
//   - Exportar la lista de palabras excluidas a un archivo XML.
//   - Añadir nuevas palabras a la lista de palabras excluidas, evitando duplicados, y actualizar el panel.
//   - Activar el botón unificado para la importación de palabras excluidas mediante un input oculto.
//   - Ejecutar la función de escaneo de lugares al hacer clic en el botón correspondiente.
//**************************************************************************
    function attachEvents2()
    {
        console.log(`[${SCRIPT_NAME}] Adjuntando eventos...`);

        const normalizeArticlesCheckbox = document.getElementById("normalizeArticles");
        const maxPlacesInput = document.getElementById("maxPlacesInput");
        const addExcludeWordButton = document.getElementById("addExcludeWord");
        const scanPlacesButton = document.getElementById("scanPlaces");
        const hiddenInput = document.getElementById("hiddenImportInput");
        const importButtonUnified = document.getElementById("importExcludeWordsUnifiedBtn");

        // Validación de elementos necesarios
        if (!normalizeArticlesCheckbox || !maxPlacesInput || !addExcludeWordButton || !scanPlacesButton) {
            console.error(`[${SCRIPT_NAME}] Error: No se encontraron elementos en el DOM.`);
            return;
        }

        // ✅ Evento: cambiar estado de "no normalizar artículos"
        normalizeArticlesCheckbox.addEventListener("change", (e) => {
            normalizeArticles = e.target.checked;
        });

        // ✅ Evento: cambiar número máximo de places
        maxPlacesInput.addEventListener("input", (e) => {
            maxPlaces = parseInt(e.target.value, 10);
        });

        // ✅ Evento: exportar palabras excluidas a XML
        document.getElementById("exportExcludeWords").addEventListener("click", () => {
            const savedWords = JSON.parse(localStorage.getItem("excludeWords")) || [];
            if (savedWords.length === 0) {
                alert("No hay palabras excluidas para exportar.");
                return;
            }
            const sortedWords = [...savedWords].sort((a, b) => a.localeCompare(b));

            const xmlContent = `<?xml version="1.0" encoding="UTF-8"?>
<ExcludedWords>
  ${sortedWords.map(word => `  <word>${word}</word>`).join("\n  ")}
</ExcludedWords>`;

            const blob = new Blob([xmlContent], { type: "application/xml" });
            const url = URL.createObjectURL(blob);
            const link = document.createElement("a");

            link.href = url;
            link.download = "excluded_words.xml";
            document.body.appendChild(link); // Correctly appends the link
            link.click();
            document.body.removeChild(link); // Correctly removes the link
        });

        // ✅ Evento: añadir palabra excluida sin duplicados
        addExcludeWordButton.addEventListener("click", () => {
            const wordInput = document.getElementById("excludeWord") || document.getElementById("excludedWord");
            const word = wordInput?.value.trim();

            if (!word) return;

            const lowerWord = word.toLowerCase();
            const alreadyExists = excludeWords.some(w => w.toLowerCase() === lowerWord);
            if (!alreadyExists) {
                excludeWords.push(word);
                localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
                renderExcludedWordsPanel(); // Usar renderExcludedWordsPanel en lugar de updateExcludeList
            }

            wordInput.value = "";
        });

        // ✅ Evento: nuevo botón unificado de importación
        importButtonUnified.addEventListener("click", () => {
            hiddenInput.click(); // abre el file input oculto
        });

        hiddenInput.addEventListener("change", () => {
            handleImportList(); // ✅ Llama a la función handleImportList al importar
        });

        // ✅ Evento: escanear lugares
        scanPlacesButton.addEventListener("click", scanPlaces);
    }

  

    //**************************************************************************
    //Nombre: NameChangeAction
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas:
    //   - venue (object): Objeto Place que contiene la información del lugar a modificar.
    //   - oldName (string): Nombre actual del lugar.
    //   - newName (string): Nuevo nombre sugerido para el lugar.
    //Salidas: Ninguna (función constructora que crea un objeto de acción).
    //Prerrequisitos si existen:
    //   - El objeto venue debe contar con la propiedad attributes, y dentro de ésta, con el campo id.
    //Descripción:
    // Esta función constructora crea un objeto que representa la acción de cambio de nombre
    // de un Place en el Waze Map Editor (WME). Asigna las propiedades correspondientes:
    //   - Guarda el objeto venue, el nombre original (oldName) y el nuevo nombre (newName).
    //   - Extrae y guarda el ID único del lugar desde venue.attributes.id.
    //   - Establece metadatos para identificar la acción, asignando el tipo "NameChangeAction"
    //     y marcando isGeometryEdit como false para indicar que no se trata de una edición de geometría.
    // Estos metadatos pueden ser utilizados por WME y otros plugins para gestionar y mostrar la acción.
    //**************************************************************************
    function NameChangeAction(venue, oldName, newName)
    {
        // Referencia al Place y los nombres
        this.venue = venue;
        this.oldName = oldName;
        this.newName = newName;

        // ID único del Place
        this.venueId = venue.attributes.id;

        // Metadatos que WME/Plugins pueden usar
        this.type = "NameChangeAction";
        this.isGeometryEdit = false; // no es una edición de geometría
    }

    /**
 * 1) getActionName: nombre de la acción en el historial.
 */
    NameChangeAction.prototype.getActionName = function() {
        return "Update place name";
    };

    /** 2) getActionText: texto corto que WME a veces muestra. */
    NameChangeAction.prototype.getActionText = function() {
        return "Update place name";
    };

    /** 3) getName: algunas versiones llaman a getName(). */
    NameChangeAction.prototype.getName = function() {
        return "Update place name";
    };

    /** 4) getDescription: descripción detallada de la acción. */
    NameChangeAction.prototype.getDescription = function() {
        return `Place name changed from "${this.oldName}" to "${this.newName}".`;
    };

    /** 5) getT: título (a veces requerido por plugins). */
    NameChangeAction.prototype.getT = function() {
        return "Update place name";
    };

    /** 6) getID: si un plugin llama a e.getID(). */
    NameChangeAction.prototype.getID = function() {
        return `NameChangeAction-${this.venueId}`;
    };

    /** 7) doAction: asigna el nuevo nombre (WME llama a esto al crear la acción). */
    NameChangeAction.prototype.doAction = function() {
        this.venue.attributes.name = this.newName;
        this.venue.isDirty = true;
        if (typeof W.model.venues.markObjectEdited === "function") {
            W.model.venues.markObjectEdited(this.venue);
        }
    };

    /** 8) undoAction: revertir al nombre anterior (Ctrl+Z). */
    NameChangeAction.prototype.undoAction = function() {
        this.venue.attributes.name = this.oldName;
        this.venue.isDirty = true;
        if (typeof W.model.venues.markObjectEdited === "function") {
            W.model.venues.markObjectEdited(this.venue);
        }
    };

    /** 9) redoAction: rehacer (Ctrl+Shift+Z), vuelve a doAction. */
    NameChangeAction.prototype.redoAction = function() {
        return this.doAction();
    };

    /** 10) undoSupported / redoSupported: indica si se puede des/rehacer. */
    NameChangeAction.prototype.undoSupported = function() {
        return true;
    };
    NameChangeAction.prototype.redoSupported = function() {
        return true;
    };

    /** 11) accept / supersede: evita fusionar con otras acciones. */
    NameChangeAction.prototype.accept = function() {
        return false;
    };
    NameChangeAction.prototype.supersede = function() {
        return false;
    };

    /** 12) isEditAction: true => habilita "Guardar". */
    NameChangeAction.prototype.isEditAction = function() {
        return true;
    };

    /** 13) getAffectedUniqueIds: objetos que se alteran. */
    NameChangeAction.prototype.getAffectedUniqueIds = function() {
        return [this.venueId];
    };

    /** 14) isSerializable: si no implementas serialize(), pon false. */
    NameChangeAction.prototype.isSerializable = function() {
        return false;
    };

    /** 15) isActionStackable: false => no combina con otras ediciones. */
    NameChangeAction.prototype.isActionStackable = function() {
        return false;
    };

    /** 16) getFocusFeatures: WME/Plugins pueden usarlo para "enfocar" el objeto. */
    NameChangeAction.prototype.getFocusFeatures = function() {
        // Devolvemos el venue para indicar que ese es el foco (o un array vacío si prefieres).
        return [this.venue];
    };

    /** 17) Métodos vacíos para evitar futuros "no es una función" si WME pide estos. */
    NameChangeAction.prototype.getFocusSegments = function() {
        return [];
    };
    NameChangeAction.prototype.getFocusNodes = function() {
        return [];
    };
    NameChangeAction.prototype.getFocusClosures = function() {
        return [];

    };

    /** 18) getTimestamp: método nuevo que WME/Plugins están llamando. */
    NameChangeAction.prototype.getTimestamp = function() {
        // Devolvemos un timestamp numérico (ms desde época UNIX).
        return Date.now();
    };

 
    //**************************************************************************
    //Nombre: openFloatingPanel
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas: placesToNormalize (array de lugares con nombre original y sugerencias)
    //Salidas: Panel flotante con opciones de normalización y eliminación
    //Prerrequisitos si existen: Debe haberse definido normalizePlaceName y cargado excludeWords correctamente
    //Descripción: Crea un panel interactivo donde se presentan los lugares que requieren cambios,
    //             permitiendo su corrección o eliminación. Solo muestra lugares que requieren cambio
    //             y errores ortográficos verdaderos.
    //**************************************************************************
    function openFloatingPanel2(placesToNormalize) {
        const panel = document.createElement("div");
        panel.id = "normalizer-floating-panel";
        panel.style.cssText = `
            position: fixed;
            top: 50%;
            left: 50%;
            transform: translate(-50%, -50%);
            background: white;
            padding: 20px;
            border: 1px solid #ccc;
            border-radius: 8px;
            box-shadow: 0 2px 10px rgba(0,0,0,0.1);
            z-index: 1000;
            max-height: 80vh;
            overflow-y: auto;
            min-width: 800px;
        `;

        let panelContent = `
            <style>
                #normalizer-table { width: 100%; border-collapse: collapse; }
                #normalizer-table th, #normalizer-table td { padding: 8px; border: 1px solid #ddd; }
                #normalizer-table tr:nth-child(even) { background-color: #f9f9f9; }
                .warning-row { background-color: #fff3cd !important; }
                .close-btn {
                    position: absolute;
                    top: 10px;
                    right: 10px;
                    background: #ccc;
                    color: black;
                    border: none;
                    border-radius: 4px;
                    width: 25px;
                    height: 25px;
                    font-size: 16px;
                    font-weight: bold;
                    cursor: pointer;
                    text-align: center;
                    line-height: 25px;
                    transition: background-color 0.3s ease;
                }
                .close-btn:hover {
                    background: #bbb;
                }
                .footer-buttons {
                    display: flex;
                    justify-content: center;
                    gap: 10px;
                    margin-top: 20px;
                }
            </style>
            <button class="close-btn" id="close-panel-btn">X</button>
            <h3>Places to Normalize</h3>
            <table id="normalizer-table">
                <thead>
                    <tr>
                        <th>Normalizar</th>
                        <th>Eliminar</th>
                        <th>Estado</th>
                        <th>Nombre Original</th>
                        <th>Nombre Sugerido</th>
                        <th>Corrección</th>
                        <th>Acción</th>
                    </tr>
                </thead>
                <tbody>
        `;

        // Procesar cada lugar y sus advertencias
        placesToNormalize.forEach((place, index) => {
            const { originalName, newName, spellingWarnings = [] } = place;

            // Crear una fila por cada advertencia ortográfica
            spellingWarnings.forEach((warning, warningIndex) => {
                const suggestionId = `suggestion-${index}-${warningIndex}`;

                panelContent += `
                <tr class="warning-row">
                    <td style="text-align: center;">
                        <input type="checkbox" class="normalize-checkbox"
                            data-index="${index}"
                            data-warning-index="${warningIndex}"
                            data-suggestion-id="${suggestionId}">
                    </td>
                    <td style="text-align: center;">
                        <input type="checkbox" class="delete-checkbox"
                            data-index="${index}"
                            data-warning-index="${warningIndex}">
                    </td>
                    <td style="text-align: center;">⚠️</td>
                    <td id="name-cell-${index}-${warningIndex}">${originalName}</td>
                    <td>
                        <input type="text" class="new-name-input"
                            data-index="${index}"
                            data-warning-index="${warningIndex}"
                            data-place-id="${place.id}"
                            data-suggestion-id="${suggestionId}"
                            value="${newName}">
                    </td>
                    <td>${warning.original} → ${warning.sugerida} (${warning.tipo})</td>
                    <td style="text-align: center;">
                        <button class="apply-suggestion-btn"
                                data-index="${index}"
                                data-warning-index="${warningIndex}"
                                title="Corregir ortografía de la palabra"
                                style="background-color: #4CAF50; color: white; border: none; padding: 5px 10px; border-radius: 4px; cursor: pointer;">
                            Fix
                        </button>
                        <button class="add-exclude-btn"
                                data-word="${warning.original}"
                                title="Adicionar palabra excluida nueva"
                                style="background-color: #007bff; color: white; border: none; padding: 5px 10px; border-radius: 4px; cursor: pointer;">
                            Add Excld
                        </button>
                    </td>
                </tr>`;
            });
        });

        panelContent += `
                    </tbody>
                </table>
                <div class="footer-buttons">
                    <button class="apply-changes-btn" id="apply-changes-btn">✔️ Apply Changes</button>
                    <button class="cancel-btn" id="cancel-btn">❌ Cancel</button>
                </div>
            `;

        panel.innerHTML = panelContent;
        document.body.appendChild(panel);

        // Eventos para los botones
        document.getElementById('close-panel-btn').addEventListener('click', () => {
            panel.remove(); // Cerrar el panel flotante
        });

        document.getElementById('apply-changes-btn').addEventListener('click', () => {
            window.applyNormalization(); // Llamar a la función global para aplicar cambios
        });

        document.getElementById('cancel-btn').addEventListener('click', () => {
            panel.remove(); // Cerrar el panel flotante
        });

        // Asignar eventos a los botones "Fix"
        document.querySelectorAll('.apply-suggestion-btn').forEach(button => {
            button.addEventListener('click', function () {
                const index = this.dataset.index;
                const warningIndex = this.dataset.warningIndex;
                const input = document.querySelector(
                    `.new-name-input[data-index="${index}"][data-warning-index="${warningIndex}"]`
                );
                const checkbox = document.querySelector(
                    `.normalize-checkbox[data-index="${index}"][data-warning-index="${warningIndex}"]`
                );
        
                if (input && checkbox) {
                    const warning = placesToNormalize[index].spellingWarnings[warningIndex];
                    input.value = input.value.replace(warning.original, warning.sugerida || warning.original);
        
                    checkbox.checked = true; // Marcar el checkbox
                }
            });
        });

        // Asignar eventos a los botones "Add Excld Word"
        document.querySelectorAll('.add-exclude-btn').forEach(button => {
            button.addEventListener('click', function () {
                const word = this.dataset.word;

                if (!excludeWords.includes(word)) {
                    excludeWords.push(word);
                    excludeWords.sort((a, b) => a.localeCompare(b)); // Ordenar alfabéticamente
                    localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
                    renderExcludedWordsPanel(); // Actualizar el panel lateral

                    // Mostrar popup en el centro del panel flotante
                    const panel = document.getElementById("normalizer-floating-panel");
                    const popup = document.createElement('div');
                    popup.textContent = `✅ The Word "${word}" has been added to the exclusion list.`;
                    popup.style.position = 'absolute';
                    popup.style.top = '50%';
                    popup.style.left = '50%';
                    popup.style.transform = 'translate(-50%, -50%)';
                    popup.style.backgroundColor = '#4CAF50';
                    popup.style.color = 'white';
                    popup.style.padding = '10px 20px';
                    popup.style.borderRadius = '5px';
                    popup.style.zIndex = '10000';
                    popup.style.opacity = '1';
                    popup.style.transition = 'opacity 1s ease-in-out';

                    panel.appendChild(popup);

                    setTimeout(() => {
                        popup.style.opacity = '0';
                        setTimeout(() => panel.removeChild(popup), 1000);
                    }, 2000);

                    // Bloquear el botón y cambiar su texto
                    this.textContent = 'Excluded Word Added';
                    this.disabled = true;
                    this.style.backgroundColor = '#6c757d'; // Cambiar color a gris
                    this.style.cursor = 'not-allowed';
                } else {
                    alert(`⚠️ The word "${word}" is already on the exclusion list.`);
                }
            });
        });
    }
    
    //**************************************************************************
    //Nombre: waitForWME
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas: Ninguna
    //Salidas: Ninguna
    //Prerrequisitos si existen:
    // - El objeto global W debe estar disponible.
    // - Deben estar definidas las funciones: initializeExcludeWords, createSidebarTab y renderExcludedWordsPanel.
    //Descripción:
    // Esta función espera a que el entorno de edición de Waze (WME) esté completamente cargado,
    // verificando que existan los objetos necesarios para iniciar el script.
    // Una vez disponible, inicializa la lista de palabras excluidas desde localStorage,
    // crea el tab lateral personalizado y renderiza la lista visual de palabras excluidas.
    // Si WME aún no está listo, vuelve a intentar cada 1000 ms.
    //**************************************************************************
    function waitForWME2() {
        if (W && W.userscripts && W.model && W.model.venues) {
            console.log(`[${SCRIPT_NAME}] Inicializando v${VERSION}`);

            initializeExcludeWords(); // ⚠️ Usa solo localStorage
            createSidebarTab();

            waitForDOM("#normalizer-tab", () => {
                console.log("[waitForWME] 🧩 Sidebar listo, renderizando palabras excluidas");

                renderExcludedWordsPanel();       // Muestra las palabras
                setupDragAndDropImport();         // Activa drag & drop

                //Evita que se abra el archivo si cae fuera del área
              //  window.addEventListener("dragover", e => e.preventDefault(), false);
             //   window.addEventListener("drop", e => e.preventDefault(), false);
            });

        } else {
            console.log(`[${SCRIPT_NAME}] Esperando que WME esté listo...`);
            setTimeout(waitForWME, 1000);
        }
    }
//**************************************************************************
//Nombre: init
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: Ninguna
//Salidas: Ninguna
//Prerrequisitos si existen:
// - El objeto global W debe estar disponible.
// - Deben estar definidas las funciones: initializeExcludeWords, createSidebarTab, waitForDOM, renderExcludedWordsPanel y setupDragAndDropImport.
//Descripción:
// Esta función espera a que el entorno de edición de Waze (WME) esté completamente cargado,
// verificando que existan los objetos necesarios para iniciar el script. Si WME aún no está listo,
// reintenta la inicialización cada 1000 ms. Una vez disponible, inicializa la lista de palabras
// excluidas, crea el tab lateral personalizado, espera a que el DOM del tab esté listo para
// renderizar el contenido (palabras excluidas y funcionalidad drag & drop), y expone globalmente
// las funciones applyNormalization y normalizePlaceName para uso externo.
//**************************************************************************
function init() {
    if (!W || !W.userscripts || !W.model || !W.model.venues) {
        console.log(`[${SCRIPT_NAME}] Esperando que WME esté listo...`);
        setTimeout(init, 1000);
        return;
    }

    console.log(`[${SCRIPT_NAME}] Inicializando v${VERSION}`);
    initializeExcludeWords();
    createSidebarTab();
    waitForDOM("#normalizer-tab", () => {
        console.log("[init] 🧩 Sidebar listo, renderizando palabras excluidas");
        renderExcludedWordsPanel();
        setupDragAndDropImport();
    });

    window.applyNormalization = applyNormalization;
    window.normalizePlaceName = normalizePlaceName;
}
init();
    
    } catch (e) {
        console.error('[PlacesNameNormalizer] Fatal initialization error:', e);
    }
})();

    //**************************************************************************
    //Nombre: applyNormalization
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas: Ninguna
    //Salidas: Ninguna
    //Prerrequisitos si existen:
    // - Debe existir un elemento en el DOM con las clases .normalize-checkbox, .delete-checkbox y .new-name-input.
    // - El objeto global W debe estar disponible, incluyendo W.model.venues, W.model.actionManager y W.controller.
    // - Deben estar definidos los módulos "Waze/Action/UpdateObject" y "Waze/Action/DeleteObject" (accesibles mediante require()).
    // - Debe existir la variable placesToNormalize, que contiene datos de los lugares a normalizar, incluyendo sugerencias ortográficas.
    //Descripción:
    // Esta función aplica la normalización y/o eliminación de nombres de lugares en el Waze Map Editor
    // según las selecciones realizadas en el panel flotante. Primero, obtiene los checkboxes seleccionados
    // para normalización y eliminación. Si no hay ningún elemento seleccionado, se informa y se cancela la operación.
    // Si se han seleccionado TODOS los checkboxes de eliminación, se solicita una confirmación adicional.
    // Para cada checkbox de normalización seleccionado, se verifica si se debe aplicar la sugerencia ortográfica
    // (cuando se ha hecho clic en el botón correspondiente) o el nombre completo modificado, y se actualiza el lugar
    // mediante la acción de actualización. Posteriormente, se procesan los checkboxes de eliminación aplicando la
    // acción de eliminación a los lugares correspondientes. Si se realizaron cambios, se marca el modelo como modificado.
    // Finalmente, se cierra el panel flotante.
    //**************************************************************************
    function applyNormalization() {
        const normalizeCheckboxes = document.querySelectorAll(".normalize-checkbox:checked");
        const deleteCheckboxes = document.querySelectorAll(".delete-checkbox:checked");
        let changesMade = false;
    
        if (normalizeCheckboxes.length === 0 && deleteCheckboxes.length === 0) {
            alert("No hay lugares seleccionados para normalizar o eliminar.");
            return;
        }
    
        // Procesar normalización
        normalizeCheckboxes.forEach(cb => {
            const index = cb.dataset.index;
            const input = document.querySelector(`.new-name-input[data-index="${index}"]`);
            const newName = input?.value?.trim();
            const placeId = input?.getAttribute("data-place-id");
            const place = W.model.venues.getObjectById(placeId);
    
            if (!place || !place.attributes?.name) {
                console.warn(`No se encontró el lugar con ID: ${placeId}`);
                return;
            }
    
            const currentName = place.attributes.name.trim();
            if (currentName !== newName) {
                try {
                    const UpdateObject = require("Waze/Action/UpdateObject");
                    const action = new UpdateObject(place, { name: newName });
                    W.model.actionManager.add(action);
                    console.log(`Nombre actualizado: "${currentName}" → "${newName}"`);
                    changesMade = true;
                } catch (error) {
                    console.error("Error aplicando la acción de actualización:", error);
                }
            }
        });
    
        // Procesar eliminación
        deleteCheckboxes.forEach(cb => {
            const index = cb.dataset.index;
            const placeId = document.querySelector(`.new-name-input[data-index="${index}"]`)?.getAttribute("data-place-id");
            const place = W.model.venues.getObjectById(placeId);
    
            if (!place) {
                console.warn(`No se encontró el lugar con ID para eliminar: ${placeId}`);
                return;
            }
    
            try {
                const DeleteObject = require("Waze/Action/DeleteObject");
                const deleteAction = new DeleteObject(place);
                W.model.actionManager.add(deleteAction);
                console.log(`Lugar eliminado: ${placeId}`);
                changesMade = true;
            } catch (error) {
                console.error("Error eliminando el lugar:", error);
            }
        });
    
        if (changesMade) {
            alert("Cambios aplicados correctamente.");
        } else {
            alert("No se realizaron cambios.");
        }
    
        // Cerrar el panel flotante
        const panel = document.getElementById("normalizer-floating-panel");
        if (panel) panel.remove();
    }

    //**************************************************************************
    //Nombre: isSimilar
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas:
    //  - a (string): Primera palabra a comparar.
    //  - b (string): Segunda palabra a comparar.
    //Salidas:
    //  - boolean: Retorna true si las palabras son consideradas similares de forma leve; de lo contrario, retorna false.
    //Prerrequisitos si existen: Ninguno.
    //Descripción:
    // Esta función evalúa la similitud leve entre dos palabras. Primero, verifica si ambas palabras son
    // idénticas, en cuyo caso retorna true. Luego, comprueba si la diferencia en la cantidad de caracteres
    // entre ambas es mayor a 2; si es así, retorna false. Posteriormente, compara carácter por carácter
    // hasta el largo mínimo de las palabras, contando las diferencias. Si el número de discrepancias
    // excede 2, se considera que las palabras no son similares y retorna false; en caso contrario, retorna true.
    //**************************************************************************
    function isSimilar(a, b)
    {
        if (a === b) return true;
        if (Math.abs(a.length - b.length) > 2) return false;

        let mismatches = 0;
        for (let i = 0; i < Math.min(a.length, b.length); i++) {
            if (a[i] !== b[i]) mismatches++;
            if (mismatches > 2) return false;
        }

        return true;
    }


    //**************************************************************************
    //Nombre: normalizePlaceName
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas: name (string) - Nombre del lugar
    //Salidas: texto normalizado (string)
    //Descripción: Normaliza un nombre aplicando capitalización, manejo de artículos, números y paréntesis.
    //**************************************************************************
    function normalizePlaceName(name) {
        if (!name) return "";

        const normalizeArticles = !document.getElementById("normalizeArticles")?.checked;
        const articles = ["el", "la", "los", "las", "de", "del", "al", "y"];
        const words = name.trim().split(/\s+/);
        const isRoman = word => /^(i{1,3}|iv|v|vi{0,3}|ix|x|xi{0,3}|xiv|xv|xvi{0,3}|xix|xx|xxi{0,3}|xxiv|xxv)$/i.test(word);

        const normalizedWords = words.map((word, index) => {
            const lowerWord = word.toLowerCase();

            // Si la palabra es un número, no la analizamos
            if (/^\d+$/.test(word)) return word;

            // Si la palabra está en la lista de exclusión, no la modificamos
            if (excludeWords.some(w => w.toLowerCase() === lowerWord)) return word;

            // Si es un número romano, lo dejamos en mayúsculas
            if (isRoman(word)) return word.toUpperCase();

            // Si es una sigla con estructura T&T o a&A, convertirla a mayúsculas
            if (/^[A-Za-z]&[A-Za-z]$/.test(word)) return word.toUpperCase();

            // Si es una sigla con apóstrofe como "E's", también la dejamos igual
            if (/^[A-Z]'[A-Z][a-z]+$/.test(word)) return word;

            // Si no se deben normalizar artículos y es un artículo, lo dejamos en minúsculas (excepto la primera palabra)
            if (!normalizeArticles && articles.includes(lowerWord) && index !== 0) return lowerWord;

            // Si es un número seguido de letras, lo dejamos igual
            if (/^\d+[A-Z][a-zA-Z]*$/.test(word)) return word;

            // Si está entre paréntesis y es todo mayúsculas o minúsculas, lo dejamos igual
            if (/^\(.*\)$/.test(word)) {
                const inner = word.slice(1, -1);
                if (inner === inner.toUpperCase() || inner === inner.toLowerCase()) return word;
            }

            // Capitalizamos la palabra (primera letra en mayúscula, el resto en minúscula)
            return word.charAt(0).toUpperCase() + word.slice(1).toLowerCase();
        });

        let newName = normalizedWords.join(" ")
            .replace(/\s*\|\s*/g, " - ")
            .replace(/([(["'])\s*([\p{L}])/gu, (match, p1, p2) => p1 + p2.toUpperCase())
            .replace(/\s*-\s*/g, " - ")
            .replace(/\b(\d+)([A-Z])\b/g, (match, num, letter) => num + letter.toUpperCase())
            .replace(/\.$/, "")
            .replace(/&(\s*)([A-Z])/g, (match, space, letter) => "&" + space + letter.toUpperCase());

        return newName.replace(/\s{2,}/g, " ").trim();
    }

    // Para exponer al contexto global real desde Tampermonkey
    unsafeWindow.normalizePlaceName = normalizePlaceName;



    //**************************************************************************
    //Nombre: openFloatingPanel
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas: placesToNormalize (array de lugares con nombre original y sugerencias)
    //Salidas: Panel flotante con opciones de normalización y eliminación
    //Prerrequisitos si existen: Debe haberse definido normalizePlaceName y cargado excludeWords correctamente
    //Descripción: Crea un panel interactivo donde se presentan los lugares que requieren cambios,
    //             permitiendo su corrección o eliminación. Solo muestra lugares que requieren cambio
    //             y errores ortográficos verdaderos.
    //**************************************************************************
    function openFloatingPanel(placesToNormalize) {
        const panel = document.createElement("div");
        panel.id = "normalizer-floating-panel";
        panel.style.cssText = `
            position: fixed;
            top: 50%;
            left: 50%;
            transform: translate(-50%, -50%);
            background: white;
            padding: 20px;
            border: 1px solid #ccc;
            border-radius: 8px;
            box-shadow: 0 2px 10px rgba(0,0,0,0.1);
            z-index: 1000;
            max-height: 80vh;
            overflow-y: auto;
            min-width: 800px;
        `;

        let panelContent = `
            <style>
                #normalizer-table { width: 100%; border-collapse: collapse; }
                #normalizer-table th, #normalizer-table td { padding: 8px; border: 1px solid #ddd; }
                #normalizer-table tr:nth-child(even) { background-color: #f9f9f9; }
                .warning-row { background-color: #fff3cd !important; }
                .close-btn {
                    position: absolute;
                    top: 10px;
                    right: 10px;
                    background: #ccc;
                    color: black;
                    border: none;
                    border-radius: 4px;
                    width: 25px;
                    height: 25px;
                    font-size: 16px;
                    font-weight: bold;
                    cursor: pointer;
                    text-align: center;
                    line-height: 25px;
                    transition: background-color 0.3s ease;
                }
                .close-btn:hover {
                    background: #bbb;
                }
                .footer-buttons {
                    display: flex;
                    justify-content: center;
                    gap: 10px;
                    margin-top: 20px;
                }
            </style>
            <button class="close-btn" id="close-panel-btn">X</button>
            <h3>Places to Normalize</h3>
            <table id="normalizer-table">
                <thead>
                    <tr>
                        <th>Normalizar</th>
                        <th>Eliminar</th>
                        <th>Estado</th>
                        <th>Nombre Original</th>
                        <th>Nombre Sugerido</th>
                        <th>Corrección</th>
                        <th>Acción</th>
                    </tr>
                </thead>
                <tbody>
        `;

        // Procesar cada lugar y sus advertencias
        placesToNormalize.forEach((place, index) => {
            const { originalName, newName, spellingWarnings = [] } = place;

            // Crear una fila por cada advertencia ortográfica
            spellingWarnings.forEach((warning, warningIndex) => {
                const suggestionId = `suggestion-${index}-${warningIndex}`;

                panelContent += `
                <tr class="warning-row">
                    <td style="text-align: center;">
                        <input type="checkbox" class="normalize-checkbox"
                            data-index="${index}"
                            data-warning-index="${warningIndex}"
                            data-suggestion-id="${suggestionId}">
                    </td>
                    <td style="text-align: center;">
                        <input type="checkbox" class="delete-checkbox"
                            data-index="${index}"
                            data-warning-index="${warningIndex}">
                    </td>
                    <td style="text-align: center;">⚠️</td>
                    <td id="name-cell-${index}-${warningIndex}">${originalName}</td>
                    <td>
                        <input type="text" class="new-name-input"
                            data-index="${index}"
                            data-warning-index="${warningIndex}"
                            data-place-id="${place.id}"
                            data-suggestion-id="${suggestionId}"
                            value="${newName}">
                    </td>
                    <td>${warning.original} → ${warning.sugerida} (${warning.tipo})</td>
                    <td style="text-align: center;">
                        <button class="apply-suggestion-btn"
                                data-index="${index}"
                                data-warning-index="${warningIndex}"
                                title="Corregir ortografía de la palabra"
                                style="background-color: #4CAF50; color: white; border: none; padding: 5px 10px; border-radius: 4px; cursor: pointer;">
                            Fix
                        </button>
                        <button class="add-exclude-btn"
                                data-word="${warning.original}"
                                title="Adicionar palabra excluida nueva"
                                style="background-color: #007bff; color: white; border: none; padding: 5px 10px; border-radius: 4px; cursor: pointer;">
                            Add
                        </button>
                    </td>
                </tr>`;
            });
        });

        panelContent += `
                    </tbody>
                </table>
                <div class="footer-buttons">
                    <button class="apply-changes-btn" id="apply-changes-btn">✔️ Apply Changes</button>
                    <button class="cancel-btn" id="cancel-btn">❌ Cancel</button>
                </div>
            `;

        panel.innerHTML = panelContent;
        document.body.appendChild(panel);

        // Eventos para los botones
        document.getElementById('close-panel-btn').addEventListener('click', () => {
            panel.remove(); // Cerrar el panel flotante
        });

        document.getElementById('apply-changes-btn').addEventListener('click', () => {
            window.applyNormalization(); // Llamar a la función global para aplicar cambios
        });

        document.getElementById('cancel-btn').addEventListener('click', () => {
            panel.remove(); // Cerrar el panel flotante
        });

        // Asignar eventos a los botones "Fix"
        document.querySelectorAll('.apply-suggestion-btn').forEach(button => {
            button.addEventListener('click', function () {
                const index = this.dataset.index;
                const warningIndex = this.dataset.warningIndex;
                const input = document.querySelector(
                    `.new-name-input[data-index="${index}"][data-warning-index="${warningIndex}"]`
                );
                const checkbox = document.querySelector(
                    `.normalize-checkbox[data-index="${index}"][data-warning-index="${warningIndex}"]`
                );

                if (input && checkbox) {
                    // Aplicar la corrección ortográfica al campo de entrada
                    const warning = placesToNormalize[index].spellingWarnings[warningIndex];
                    input.value = input.value.replace(warning.original, warning.sugerida);

                    checkbox.checked = true; // Marcar el checkbox
                }
            });
        });

        // Asignar eventos a los botones "Add Excld Word"
        document.querySelectorAll('.add-exclude-btn').forEach(button => {
            button.addEventListener('click', function () {
                const word = this.dataset.word;

                // Verificar si es una sigla con formato X&X
                if (/^[A-Za-z]&[A-Za-z]$/.test(word)) {
                    alert("⚠️ No es necesario adicionar palabras excluidas que tengan '&'.");
                    return;
                }

                if (!excludeWords.includes(word)) {
                    excludeWords.push(word);
                    excludeWords.sort((a, b) => a.localeCompare(b)); // Ordenar alfabéticamente
                    localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
                    renderExcludedWordsPanel(); // Actualizar el panel lateral

                    // Mostrar popup en el centro del panel flotante
                    const panel = document.getElementById("normalizer-floating-panel");
                    const popup = document.createElement('div');
                    popup.textContent = `✅ The Word "${word}" has been added to the exclusion list.`;
                    popup.style.position = 'absolute';
                    popup.style.top = '50%';
                    popup.style.left = '50%';
                    popup.style.transform = 'translate(-50%, -50%)';
                    popup.style.backgroundColor = '#4CAF50';
                    popup.style.color = 'white';
                    popup.style.padding = '10px 20px';
                    popup.style.borderRadius = '5px';
                    popup.style.zIndex = '10000';
                    popup.style.opacity = '1';
                    popup.style.transition = 'opacity 1s ease-in-out';

                    panel.appendChild(popup);

                    setTimeout(() => {
                        popup.style.opacity = '0';
                        setTimeout(() => panel.removeChild(popup), 1000);
                    }, 2000);

                    // Bloquear el botón y cambiar su texto
                    this.textContent = 'Added';
                    this.disabled = true;
                    this.style.backgroundColor = '#6c757d'; // Cambiar color a gris
                    this.style.cursor = 'not-allowed';
                } else {
                    alert(`⚠️ The word "${word}" is already on the exclusion list.`);
                }
            });
        });
    }
    //**************************************************************************
    //Nombre: loadExcludeWordsFromXML
    //Fecha modificación:
    //Autor: mincho77
    //Entradas:
    //  - callback: función opcional que se ejecuta una vez se cargan y procesan las palabras excluidas.
    //Salidas: ninguna directa. Actualiza la variable global `excludeWords`.
    //Prerrequisitos:
    //  - Debe existir un archivo llamado 'excludeWords.xml' accesible por fetch.
    //  - Debe estar definida la variable global `excludeWords`.
    //Descripción:
    //  Carga un archivo XML que contiene una lista de palabras excluidas.
    //  Combina las palabras nuevas con las que ya están guardadas en localStorage
    //  y actualiza la lista global `excludeWords`. Si el XML no puede ser cargado,
    //  se usa únicamente el contenido almacenado localmente como respaldo.
    //**************************************************************************
    function loadExcludeWordsFromXML(callback) {
        fetch("excludeWords.xml")
            .then(response => response.text()) // Corrected the syntax here
            .then(xmlText => {
            const parser = new DOMParser();
            const xmlDoc = parser.parseFromString(xmlText, "text/xml");
            const wordNodes = xmlDoc.getElementsByTagName("word");
            const wordsFromXML = Array.from(wordNodes).map(node => node.textContent.trim());

            const existing = JSON.parse(localStorage.getItem("excludeWords")) || [];
            excludeWords = [...new Set([...existing, ...wordsFromXML])].sort((a, b) => a.localeCompare(b));
            localStorage.setItem("excludeWords", JSON.stringify(excludeWords));

            if (callback) callback();
        })
            .catch(() => {
            console.warn("⚠️ No se pudo cargar excludeWords.xml. Solo se usará localStorage.");
            excludeWords = JSON.parse(localStorage.getItem("excludeWords")) || ["EDS", "IPS", "McDonald's", "EPS"];
            localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
            if (callback) callback();
        });
    }

    function exportExcludeWordsToXML() {
        const xmlContent = `<?xml version="1.0" encoding="UTF-8"?>
<ExcludedWords>
${excludeWords.map(word => `  <word>${word}</word>`).join("\n")}
</ExcludedWords>`;

        const blob = new Blob([xmlContent], { type: "application/xml" });
        const url = URL.createObjectURL(blob); // ✅ ESTA LÍNEA ESTABA FALTANDO
        const link = document.createElement("a");
        link.href = url;
        link.download = "excludeWords.xml";
        document.body.appendChild(link);
        link.click();
        document.body.removeChild(link);
}


    function showFloatingMessage(message) {
        const msg = document.createElement("div");
        msg.textContent = message;
        msg.style.position = "fixed";
        msg.style.bottom = "30px";
        msg.style.left = "50%";
        msg.style.transform = "translateX(-50%)";
        msg.style.backgroundColor = "#333";
        msg.style.color = "#fff";
        msg.style.padding = "10px 20px";
        msg.style.borderRadius = "5px";
        msg.style.zIndex = 9999;
        msg.style.opacity = "0.95";
        msg.style.transition = "opacity 1s ease-in-out";

        document.body.appendChild(msg);

        setTimeout(() => {
            msg.style.opacity = "0";
            setTimeout(() => document.body.removeChild(msg), 1000);
        }, 3000);
    }
    //**************************************************************************
    //Nombre: waitForWME
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas: Ninguna
    //Salidas: Ninguna
    //Prerrequisitos si existen:
    // - El objeto global W debe estar disponible.
    // - Deben estar definidas las funciones: initializeExcludeWords, createSidebarTab y renderExcludedWordsPanel.
    //Descripción:
    // Esta función espera a que el entorno de edición de Waze (WME) esté completamente cargado,
    // verificando que existan los objetos necesarios para iniciar el script.
    // Una vez disponible, inicializa la lista de palabras excluidas desde localStorage,
    // crea el tab lateral personalizado y renderiza la lista visual de palabras excluidas.
    // Si WME aún no está listo, vuelve a intentar cada 1000 ms.
    //**************************************************************************
    function waitForWME() {
        if (W && W.userscripts && W.model && W.model.venues) {
            console.log(`[${SCRIPT_NAME}] Inicializando v${VERSION}`);

            initializeExcludeWords(); // ⚠️ Usa solo localStorage
            createSidebarTab();

            waitForDOM("#normalizer-tab", () => {
                console.log("[waitForWME] 🧩 Sidebar listo, renderizando palabras excluidas");

                renderExcludedWordsPanel();       // Muestra las palabras
                setupDragAndDropImport();         // Activa drag & drop

                //Evita que se abra el archivo si cae fuera del área
              //  window.addEventListener("dragover", e => e.preventDefault(), false);
             //   window.addEventListener("drop", e => e.preventDefault(), false);
            });

        } else {
            console.log(`[${SCRIPT_NAME}] Esperando que WME esté listo...`);
            setTimeout(waitForWME, 1000);
        }
    }


    //**************************************************************************
    //Nombre: cleanupEventListeners
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas: Ninguna
    //Salidas: Ninguna
    //Prerrequisitos si existen:
    // - Debe existir un elemento en el DOM con el id "normalizer-floating-panel".
    //Descripción:
    // Esta función limpia los event listeners asociados al panel flotante de normalización.
    // Lo hace clonando el nodo del panel y reemplazándolo en el DOM, lo cual elimina todos
    // los listeners previamente asignados a ese nodo, evitando posibles fugas de memoria
    // o comportamientos inesperados.
    //**************************************************************************
    function cleanupEventListeners() {
        const panel = document.getElementById("normalizer-floating-panel");
        if (panel) {
            const clone = panel.cloneNode(true);
            panel.parentNode.replaceChild(clone, panel);
        }
    }  

    //**************************************************************************
    //Nombre: normalizePlaceName (unsafeWindow)
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas:
    //  - name (string): el nombre original del lugar.
    //Salidas:
    //  - string: nombre normalizado, respetando exclusiones y opciones del usuario.
    //Prerrequisitos si existen:
    //  - Debe estar cargada la lista global excludeWords.
    //  - Debe existir un checkbox con id “normalizeArticles” para definir si se normalizan artículos.
    //Descripción:
    //  Esta versión expuesta globalmente permite acceder a la normalización básica del nombre de un lugar
    //  desde otros contextos como el navegador o Tampermonkey. Capitaliza cada palabra, respeta las excluidas
    //  y no aplica normalización a artículos si el checkbox lo indica.
    //  Realiza limpieza básica: reemplazo de pipes, eliminación de espacios dobles y trim final.
    //**************************************************************************
    unsafeWindow.normalizePlaceName = function(name)
    {
        if (!name) return "";

        const normalizeArticles = !document.getElementById("normalizeArticles")?.checked;
        const articles = ["el", "la", "los", "las", "de", "del", "al", "y", "o"];

        const words = name.trim().split(/\s+/);
        const normalizedWords = words.map((word, index) => {
            const lowerWord = word.toLowerCase();

            // Saltar palabras excluidas
            if (excludeWords.includes(word)) return word;

            // Saltar artículos si el checkbox está activo y no es la primera palabra
            if (!normalizeArticles && articles.includes(lowerWord) && index !== 0) {
                return lowerWord;
            }
            //Mayúsculas
            return word.charAt(0).toUpperCase() + word.slice(1).toLowerCase();
        });

        name = normalizedWords.join(" ");
        name = name.replace(/\s*\|\s*/g, " - ");
        name = name.replace(/\s{2,}/g, " ").trim();

        return name;
    };



//**************************************************************************
//Nombre: waitForDOM
//Fecha modificación: 2025-03-31
//Autor: mincho77
//Entradas:
//   selector (string): Selector CSS del nodo a esperar.
//   callback (function): Función a ejecutar cuando el nodo esté disponible.
//   interval (int, opcional): Tiempo entre intentos en milisegundos. Default: 500ms.
//   maxAttempts (int, opcional): Máximo de intentos antes de abortar. Default: 10.
//Salidas: Ejecuta el callback si encuentra el selector dentro del tiempo.
//Descripción:
//   Esta función monitorea el DOM en intervalos constantes hasta que
//   encuentra un nodo que coincida con el selector. Si lo encuentra,
//   ejecuta el callback con ese nodo. Si no, muestra un error por consola.
//**************************************************************************
    function waitForDOM(selector, callback, interval = 500, maxAttempts = 10)
    {
        let attempts = 0;
        const checkExist = setInterval(() => {
            const element = document.querySelector(selector);
            if (element) {
                clearInterval(checkExist);
                callback(element);
            } else if (attempts >= maxAttempts) {
                clearInterval(checkExist);
                console.error(`[${SCRIPT_NAME}] Error: No se encontraron elementos en el DOM después de ${maxAttempts} intentos.`);
            }
            attempts++;
        }, interval);
    }

    //**************************************************************************
    //Nombre: getSidebarHTML
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas: Ninguna.
    //Salidas: Retorna un string con HTML que define el contenido del panel lateral del script.
    //Prerrequisitos si existen: Debe estar disponible el valor de las variables globales:
    // - normalizeArticles (boolean): Define si los artículos deben ser normalizados o no.
    // - maxPlaces (number): Número máximo de lugares a escanear.
    //Descripción:
    // Esta función construye el HTML que se inyecta en el panel lateral (sidebar) de WME.
    // Incluye controles para:
    // - Activar o desactivar normalización de artículos ("el", "la", etc).
    // - Definir la cantidad máxima de lugares a procesar.
    // - Agregar palabras excluidas manualmente.
    // - Exportar palabras excluidas a un archivo XML.
    // - Importar una lista de palabras excluidas desde archivo XML.
    // - Disparar el escaneo de lugares para normalización.
    // La lista de palabras excluidas **no se renderiza aquí directamente**, sino en el div
    // con id "normalizer-sidebar" para permitir que sea manejada dinámicamente.
    //**************************************************************************
    function getSidebarHTML() {
        return `
        <div id="normalizer-tab">
          <h4>Places Name Normalizer <span style="font-size:11px;">v${VERSION}</span></h4>
    
          <div>
            <input type="checkbox" id="normalizeArticles" ${normalizeArticles ? "checked" : ""}>
            <label for="normalizeArticles">No Normalizar artículos (el, la, los, ...)</label>
          </div>
    
          <div>
            <label>Máximo de Places a buscar: </label>
            <input type='number' id='maxPlacesInput' value='${maxPlaces}' min='1' max='800' style='width: 60px;'>
          </div>
    
          <div>
            <label>Palabras Excluidas:</label>
            <input type='text' id='excludeWord' style='width: 120px;'>
            <button id='addExcludeWord'>Add Word</button>
            <div id="normalizer-sidebar" style="margin-top: 20px;"></div>
            <button id="exportExcludeWords" style="margin-top: 5px;">Export Words</button>
            <br>
            <button id="importExcludeWordsUnifiedBtn" style="margin-top: 5px;">Import List...</button>
            <input type="file" id="hiddenImportInput" accept=".xml" style="display: none;">
          </div>
    
          <div>
            <input type="checkbox" id="checkOnlyTildes" checked>
            <label for="checkOnlyTildes">Revisar solo tildes</label>
          </div>
    
          <div id="drop-zone" style="border: 2px dashed #ccc; border-radius: 6px; padding: 15px; margin: 15px 0; text-align: center; font-style: italic; color: #555; background-color: #f8f9fa; transition: all 0.3s ease;">
            📂 Arrastra aquí tu archivo .txt o .xml para importar palabras excluidas
          </div>
    
          <hr>
          <button id="scanPlaces">Scan...</button>
        </div>
      `;
    }

//**************************************************************************
//Nombre: attachEvents
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: Ninguna
//Salidas: Ninguna
//Prerrequisitos si existen:
// - Deben existir en el DOM los elementos con los siguientes IDs:
//   "normalizeArticles", "maxPlacesInput", "addExcludeWord", "scanPlaces",
//   "hiddenImportInput", "importExcludeWordsUnifiedBtn" y "exportExcludeWords".
// - Debe existir la función handleImportList y la función scanPlaces.
// - Debe estar definida la variable global excludeWords y la función renderExcludedWordsPanel.
//Descripción:
// Esta función adjunta los event listeners necesarios para gestionar la interacción del usuario
// con el panel del normalizador de nombres. Se encargan de:
//   - Actualizar la opción de normalizar artículos al cambiar el estado del checkbox.
//   - Modificar el número máximo de lugares a procesar a través de un input.
//   - Exportar la lista de palabras excluidas a un archivo XML.
//   - Añadir nuevas palabras a la lista de palabras excluidas, evitando duplicados, y actualizar el panel.
//   - Activar el botón unificado para la importación de palabras excluidas mediante un input oculto.
//   - Ejecutar la función de escaneo de lugares al hacer clic en el botón correspondiente.
//**************************************************************************
    function attachEvents2()
    {
        console.log(`[${SCRIPT_NAME}] Adjuntando eventos...`);

        const normalizeArticlesCheckbox = document.getElementById("normalizeArticles");
        const maxPlacesInput = document.getElementById("maxPlacesInput");
        const addExcludeWordButton = document.getElementById("addExcludeWord");
        const scanPlacesButton = document.getElementById("scanPlaces");
        const hiddenInput = document.getElementById("hiddenImportInput");
        const importButtonUnified = document.getElementById("importExcludeWordsUnifiedBtn");

        // Validación de elementos necesarios
        if (!normalizeArticlesCheckbox || !maxPlacesInput || !addExcludeWordButton || !scanPlacesButton) {
            console.error(`[${SCRIPT_NAME}] Error: No se encontraron elementos en el DOM.`);
            return;
        }

        // ✅ Evento: cambiar estado de "no normalizar artículos"
        normalizeArticlesCheckbox.addEventListener("change", (e) => {
            normalizeArticles = e.target.checked;
        });

        // ✅ Evento: cambiar número máximo de places
        maxPlacesInput.addEventListener("input", (e) => {
            maxPlaces = parseInt(e.target.value, 10);
        });

        // ✅ Evento: exportar palabras excluidas a XML
        document.getElementById("exportExcludeWords").addEventListener("click", () => {
            const savedWords = JSON.parse(localStorage.getItem("excludeWords")) || [];
            if (savedWords.length === 0) {
                alert("No hay palabras excluidas para exportar.");
                return;
            }
            const sortedWords = [...savedWords].sort((a, b) => a.localeCompare(b));

            const xmlContent = `<?xml version="1.0" encoding="UTF-8"?>
<ExcludedWords>
  ${sortedWords.map(word => `  <word>${word}</word>`).join("\n  ")}
</ExcludedWords>`;

            const blob = new Blob([xmlContent], { type: "application/xml" });
            const url = URL.createObjectURL(blob);
            const link = document.createElement("a");

            link.href = url;
            link.download = "excluded_words.xml";
            document.body.appendChild(link); // Correctly appends the link
            link.click();
            document.body.removeChild(link); // Correctly removes the link
        });

        // ✅ Evento: añadir palabra excluida sin duplicados
        addExcludeWordButton.addEventListener("click", () => {
            const wordInput = document.getElementById("excludeWord") || document.getElementById("excludedWord");
            const word = wordInput?.value.trim();

            if (!word) return;

            const lowerWord = word.toLowerCase();
            const alreadyExists = excludeWords.some(w => w.toLowerCase() === lowerWord);
            if (!alreadyExists) {
                excludeWords.push(word);
                localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
                renderExcludedWordsPanel(); // Usar renderExcludedWordsPanel en lugar de updateExcludeList
            }

            wordInput.value = "";
        });

        // ✅ Evento: nuevo botón unificado de importación
        importButtonUnified.addEventListener("click", () => {
            hiddenInput.click(); // abre el file input oculto
        });

        hiddenInput.addEventListener("change", () => {
            handleImportList(); // ✅ Llama a la función handleImportList al importar
        });

        // ✅ Evento: escanear lugares
        scanPlacesButton.addEventListener("click", scanPlaces);
    }

  

    //**************************************************************************
    //Nombre: NameChangeAction
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas:
    //   - venue (object): Objeto Place que contiene la información del lugar a modificar.
    //   - oldName (string): Nombre actual del lugar.
    //   - newName (string): Nuevo nombre sugerido para el lugar.
    //Salidas: Ninguna (función constructora que crea un objeto de acción).
    //Prerrequisitos si existen:
    //   - El objeto venue debe contar con la propiedad attributes, y dentro de ésta, con el campo id.
    //Descripción:
    // Esta función constructora crea un objeto que representa la acción de cambio de nombre
    // de un Place en el Waze Map Editor (WME). Asigna las propiedades correspondientes:
    //   - Guarda el objeto venue, el nombre original (oldName) y el nuevo nombre (newName).
    //   - Extrae y guarda el ID único del lugar desde venue.attributes.id.
    //   - Establece metadatos para identificar la acción, asignando el tipo "NameChangeAction"
    //     y marcando isGeometryEdit como false para indicar que no se trata de una edición de geometría.
    // Estos metadatos pueden ser utilizados por WME y otros plugins para gestionar y mostrar la acción.
    //**************************************************************************
    function NameChangeAction(venue, oldName, newName)
    {
        // Referencia al Place y los nombres
        this.venue = venue;
        this.oldName = oldName;
        this.newName = newName;

        // ID único del Place
        this.venueId = venue.attributes.id;

        // Metadatos que WME/Plugins pueden usar
        this.type = "NameChangeAction";
        this.isGeometryEdit = false; // no es una edición de geometría
    }

    /**
 * 1) getActionName: nombre de la acción en el historial.
 */
    NameChangeAction.prototype.getActionName = function() {
        return "Update place name";
    };

    /** 2) getActionText: texto corto que WME a veces muestra. */
    NameChangeAction.prototype.getActionText = function() {
        return "Update place name";
    };

    /** 3) getName: algunas versiones llaman a getName(). */
    NameChangeAction.prototype.getName = function() {
        return "Update place name";
    };

    /** 4) getDescription: descripción detallada de la acción. */
    NameChangeAction.prototype.getDescription = function() {
        return `Place name changed from "${this.oldName}" to "${this.newName}".`;
    };

    /** 5) getT: título (a veces requerido por plugins). */
    NameChangeAction.prototype.getT = function() {
        return "Update place name";
    };

    /** 6) getID: si un plugin llama a e.getID(). */
    NameChangeAction.prototype.getID = function() {
        return `NameChangeAction-${this.venueId}`;
    };

    /** 7) doAction: asigna el nuevo nombre (WME llama a esto al crear la acción). */
    NameChangeAction.prototype.doAction = function() {
        this.venue.attributes.name = this.newName;
        this.venue.isDirty = true;
        if (typeof W.model.venues.markObjectEdited === "function") {
            W.model.venues.markObjectEdited(this.venue);
        }
    };

    /** 8) undoAction: revertir al nombre anterior (Ctrl+Z). */
    NameChangeAction.prototype.undoAction = function() {
        this.venue.attributes.name = this.oldName;
        this.venue.isDirty = true;
        if (typeof W.model.venues.markObjectEdited === "function") {
            W.model.venues.markObjectEdited(this.venue);
        }
    };

    /** 9) redoAction: rehacer (Ctrl+Shift+Z), vuelve a doAction. */
    NameChangeAction.prototype.redoAction = function() {
        return this.doAction();
    };

    /** 10) undoSupported / redoSupported: indica si se puede des/rehacer. */
    NameChangeAction.prototype.undoSupported = function() {
        return true;
    };
    NameChangeAction.prototype.redoSupported = function() {
        return true;
    };

    /** 11) accept / supersede: evita fusionar con otras acciones. */
    NameChangeAction.prototype.accept = function() {
        return false;
    };
    NameChangeAction.prototype.supersede = function() {
        return false;
    };

    /** 12) isEditAction: true => habilita "Guardar". */
    NameChangeAction.prototype.isEditAction = function() {
        return true;
    };

    /** 13) getAffectedUniqueIds: objetos que se alteran. */
    NameChangeAction.prototype.getAffectedUniqueIds = function() {
        return [this.venueId];
    };

    /** 14) isSerializable: si no implementas serialize(), pon false. */
    NameChangeAction.prototype.isSerializable = function() {
        return false;
    };

    /** 15) isActionStackable: false => no combina con otras ediciones. */
    NameChangeAction.prototype.isActionStackable = function() {
        return false;
    };

    /** 16) getFocusFeatures: WME/Plugins pueden usarlo para "enfocar" el objeto. */
    NameChangeAction.prototype.getFocusFeatures = function() {
        // Devolvemos el venue para indicar que ese es el foco (o un array vacío si prefieres).
        return [this.venue];
    };

    /** 17) Métodos vacíos para evitar futuros "no es una función" si WME pide estos. */
    NameChangeAction.prototype.getFocusSegments = function() {
        return [];
    };
    NameChangeAction.prototype.getFocusNodes = function() {
        return [];
    };
    NameChangeAction.prototype.getFocusClosures = function() {
        return [];

    };

    /** 18) getTimestamp: método nuevo que WME/Plugins están llamando. */
    NameChangeAction.prototype.getTimestamp = function() {
        // Devolvemos un timestamp numérico (ms desde época UNIX).
        return Date.now();
    };

 
    //**************************************************************************
    //Nombre: openFloatingPanel
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas: placesToNormalize (array de lugares con nombre original y sugerencias)
    //Salidas: Panel flotante con opciones de normalización y eliminación
    //Prerrequisitos si existen: Debe haberse definido normalizePlaceName y cargado excludeWords correctamente
    //Descripción: Crea un panel interactivo donde se presentan los lugares que requieren cambios,
    //             permitiendo su corrección o eliminación. Solo muestra lugares que requieren cambio
    //             y errores ortográficos verdaderos.
    //**************************************************************************
    function openFloatingPanel2(placesToNormalize) {
        const panel = document.createElement("div");
        panel.id = "normalizer-floating-panel";
        panel.style.cssText = `
            position: fixed;
            top: 50%;
            left: 50%;
            transform: translate(-50%, -50%);
            background: white;
            padding: 20px;
            border: 1px solid #ccc;
            border-radius: 8px;
            box-shadow: 0 2px 10px rgba(0,0,0,0.1);
            z-index: 1000;
            max-height: 80vh;
            overflow-y: auto;
            min-width: 800px;
        `;

        let panelContent = `
            <style>
                #normalizer-table { width: 100%; border-collapse: collapse; }
                #normalizer-table th, #normalizer-table td { padding: 8px; border: 1px solid #ddd; }
                #normalizer-table tr:nth-child(even) { background-color: #f9f9f9; }
                .warning-row { background-color: #fff3cd !important; }
                .close-btn {
                    position: absolute;
                    top: 10px;
                    right: 10px;
                    background: #ccc;
                    color: black;
                    border: none;
                    border-radius: 4px;
                    width: 25px;
                    height: 25px;
                    font-size: 16px;
                    font-weight: bold;
                    cursor: pointer;
                    text-align: center;
                    line-height: 25px;
                    transition: background-color 0.3s ease;
                }
                .close-btn:hover {
                    background: #bbb;
                }
                .footer-buttons {
                    display: flex;
                    justify-content: center;
                    gap: 10px;
                    margin-top: 20px;
                }
            </style>
            <button class="close-btn" id="close-panel-btn">X</button>
            <h3>Places to Normalize</h3>
            <table id="normalizer-table">
                <thead>
                    <tr>
                        <th>Normalizar</th>
                        <th>Eliminar</th>
                        <th>Estado</th>
                        <th>Nombre Original</th>
                        <th>Nombre Sugerido</th>
                        <th>Corrección</th>
                        <th>Acción</th>
                    </tr>
                </thead>
                <tbody>
        `;

        // Procesar cada lugar y sus advertencias
        placesToNormalize.forEach((place, index) => {
            const { originalName, newName, spellingWarnings = [] } = place;

            // Crear una fila por cada advertencia ortográfica
            spellingWarnings.forEach((warning, warningIndex) => {
                const suggestionId = `suggestion-${index}-${warningIndex}`;

                panelContent += `
                <tr class="warning-row">
                    <td style="text-align: center;">
                        <input type="checkbox" class="normalize-checkbox"
                            data-index="${index}"
                            data-warning-index="${warningIndex}"
                            data-suggestion-id="${suggestionId}">
                    </td>
                    <td style="text-align: center;">
                        <input type="checkbox" class="delete-checkbox"
                            data-index="${index}"
                            data-warning-index="${warningIndex}">
                    </td>
                    <td style="text-align: center;">⚠️</td>
                    <td id="name-cell-${index}-${warningIndex}">${originalName}</td>
                    <td>
                        <input type="text" class="new-name-input"
                            data-index="${index}"
                            data-warning-index="${warningIndex}"
                            data-place-id="${place.id}"
                            data-suggestion-id="${suggestionId}"
                            value="${newName}">
                    </td>
                    <td>${warning.original} → ${warning.sugerida} (${warning.tipo})</td>
                    <td style="text-align: center;">
                        <button class="apply-suggestion-btn"
                                data-index="${index}"
                                data-warning-index="${warningIndex}"
                                title="Corregir ortografía de la palabra"
                                style="background-color: #4CAF50; color: white; border: none; padding: 5px 10px; border-radius: 4px; cursor: pointer;">
                            Fix
                        </button>
                        <button class="add-exclude-btn"
                                data-word="${warning.original}"
                                title="Adicionar palabra excluida nueva"
                                style="background-color: #007bff; color: white; border: none; padding: 5px 10px; border-radius: 4px; cursor: pointer;">
                            Add Excld
                        </button>
                    </td>
                </tr>`;
            });
        });

        panelContent += `
                    </tbody>
                </table>
                <div class="footer-buttons">
                    <button class="apply-changes-btn" id="apply-changes-btn">✔️ Apply Changes</button>
                    <button class="cancel-btn" id="cancel-btn">❌ Cancel</button>
                </div>
            `;

        panel.innerHTML = panelContent;
        document.body.appendChild(panel);

        // Eventos para los botones
        document.getElementById('close-panel-btn').addEventListener('click', () => {
            panel.remove(); // Cerrar el panel flotante
        });

        document.getElementById('apply-changes-btn').addEventListener('click', () => {
            window.applyNormalization(); // Llamar a la función global para aplicar cambios
        });

        document.getElementById('cancel-btn').addEventListener('click', () => {
            panel.remove(); // Cerrar el panel flotante
        });

        // Asignar eventos a los botones "Fix"
        document.querySelectorAll('.apply-suggestion-btn').forEach(button => {
            button.addEventListener('click', function () {
                const index = this.dataset.index;
                const warningIndex = this.dataset.warningIndex;
                const input = document.querySelector(
                    `.new-name-input[data-index="${index}"][data-warning-index="${warningIndex}"]`
                );
                const checkbox = document.querySelector(
                    `.normalize-checkbox[data-index="${index}"][data-warning-index="${warningIndex}"]`
                );
        
                if (input && checkbox) {
                    const warning = placesToNormalize[index].spellingWarnings[warningIndex];
                    input.value = input.value.replace(warning.original, warning.sugerida || warning.original);
        
                    checkbox.checked = true; // Marcar el checkbox
                }
            });
        });

        // Asignar eventos a los botones "Add Excld Word"
        document.querySelectorAll('.add-exclude-btn').forEach(button => {
            button.addEventListener('click', function () {
                const word = this.dataset.word;

                if (!excludeWords.includes(word)) {
                    excludeWords.push(word);
                    excludeWords.sort((a, b) => a.localeCompare(b)); // Ordenar alfabéticamente
                    localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
                    renderExcludedWordsPanel(); // Actualizar el panel lateral

                    // Mostrar popup en el centro del panel flotante
                    const panel = document.getElementById("normalizer-floating-panel");
                    const popup = document.createElement('div');
                    popup.textContent = `✅ The Word "${word}" has been added to the exclusion list.`;
                    popup.style.position = 'absolute';
                    popup.style.top = '50%';
                    popup.style.left = '50%';
                    popup.style.transform = 'translate(-50%, -50%)';
                    popup.style.backgroundColor = '#4CAF50';
                    popup.style.color = 'white';
                    popup.style.padding = '10px 20px';
                    popup.style.borderRadius = '5px';
                    popup.style.zIndex = '10000';
                    popup.style.opacity = '1';
                    popup.style.transition = 'opacity 1s ease-in-out';

                    panel.appendChild(popup);

                    setTimeout(() => {
                        popup.style.opacity = '0';
                        setTimeout(() => panel.removeChild(popup), 1000);
                    }, 2000);

                    // Bloquear el botón y cambiar su texto
                    this.textContent = 'Excluded Word Added';
                    this.disabled = true;
                    this.style.backgroundColor = '#6c757d'; // Cambiar color a gris
                    this.style.cursor = 'not-allowed';
                } else {
                    alert(`⚠️ The word "${word}" is already on the exclusion list.`);
                }
            });
        });
    }
    
    //**************************************************************************
    //Nombre: waitForWME
    //Fecha modificación: 2025-03-30
    //Autor: mincho77
    //Entradas: Ninguna
    //Salidas: Ninguna
    //Prerrequisitos si existen:
    // - El objeto global W debe estar disponible.
    // - Deben estar definidas las funciones: initializeExcludeWords, createSidebarTab y renderExcludedWordsPanel.
    //Descripción:
    // Esta función espera a que el entorno de edición de Waze (WME) esté completamente cargado,
    // verificando que existan los objetos necesarios para iniciar el script.
    // Una vez disponible, inicializa la lista de palabras excluidas desde localStorage,
    // crea el tab lateral personalizado y renderiza la lista visual de palabras excluidas.
    // Si WME aún no está listo, vuelve a intentar cada 1000 ms.
    //**************************************************************************
    function waitForWME2() {
        if (W && W.userscripts && W.model && W.model.venues) {
            console.log(`[${SCRIPT_NAME}] Inicializando v${VERSION}`);

            initializeExcludeWords(); // ⚠️ Usa solo localStorage
            createSidebarTab();

            waitForDOM("#normalizer-tab", () => {
                console.log("[waitForWME] 🧩 Sidebar listo, renderizando palabras excluidas");

                renderExcludedWordsPanel();       // Muestra las palabras
                setupDragAndDropImport();         // Activa drag & drop

                //Evita que se abra el archivo si cae fuera del área
              //  window.addEventListener("dragover", e => e.preventDefault(), false);
             //   window.addEventListener("drop", e => e.preventDefault(), false);
            });

        } else {
            console.log(`[${SCRIPT_NAME}] Esperando que WME esté listo...`);
            setTimeout(waitForWME, 1000);
        }
    }
//**************************************************************************
//Nombre: init
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: Ninguna
//Salidas: Ninguna
//Prerrequisitos si existen:
// - El objeto global W debe estar disponible.
// - Deben estar definidas las funciones: initializeExcludeWords, createSidebarTab, waitForDOM, renderExcludedWordsPanel y setupDragAndDropImport.
//Descripción:
// Esta función espera a que el entorno de edición de Waze (WME) esté completamente cargado,
// verificando que existan los objetos necesarios para iniciar el script. Si WME aún no está listo,
// reintenta la inicialización cada 1000 ms. Una vez disponible, inicializa la lista de palabras
// excluidas, crea el tab lateral personalizado, espera a que el DOM del tab esté listo para
// renderizar el contenido (palabras excluidas y funcionalidad drag & drop), y expone globalmente
// las funciones applyNormalization y normalizePlaceName para uso externo.
//**************************************************************************
function init() {
    if (!W || !W.userscripts || !W.model || !W.model.venues) {
        console.log(`[${SCRIPT_NAME}] Esperando que WME esté listo...`);
        setTimeout(init, 1000);
        return;
    }

    console.log(`[${SCRIPT_NAME}] Inicializando v${VERSION}`);
    initializeExcludeWords();
    createSidebarTab();
    waitForDOM("#normalizer-tab", () => {
        console.log("[init] 🧩 Sidebar listo, renderizando palabras excluidas");
        renderExcludedWordsPanel();
        setupDragAndDropImport();
    });

    window.applyNormalization = applyNormalization;
    window.normalizePlaceName = normalizePlaceName;
}
init();
    
    } catch (e) {
        console.error('[PlacesNameNormalizer] Fatal initialization error:', e);
    }
})();

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址