WME Places Name Normalizer

Normaliza nombres de lugares en Waze Map Editor (WME)

// ==UserScript==
// @name         WME Places Name Normalizer
// @namespace    https://gf.qytechs.cn/en/users/mincho77
// @version      6.2.2
// @author       Mincho77
// @description  Normaliza nombres de lugares en Waze Map Editor (WME)
// @license      MIT
// @include      https://beta.waze.com/*
// @include      https://www.waze.com/editor*
// @include      https://www.waze.com/*/editor*
// @exclude      https://www.waze.com/user/editor*
// @grant        none
// @run-at       document-end
// ==/UserScript==
(function() {
    // Variables globales básicas
        const SCRIPT_NAME = GM_info.script.name;
        const VERSION = GM_info.script.version.toString();

    // Variables globales para el panel flotante
    let floatingPanelElement = null;
    const processingPanelDimensions = { width: '400px', height: '200px' };  // Panel pequeño para procesamiento
    const resultsPanelDimensions = { width: '1300px', height: '700px' };    // Panel grande para resultados

        const commonWords = [
            'de',  'del', 'el',    'la',   'los',  'las',  'y', 'e',
            'o',   'u',   'un',    'una',  'unos', 'unas', 'a', 'en',
            'con', 'sin', 'sobre', 'tras', 'por',  'al',   'lo'
        ];

        // --- Definir nombres de pestañas cortos antes de la generación de botones ---
        const tabNames = [
          { label: "Gene", icon: "⚙️" },
          { label: "Espe", icon: "🏷️" },
          { label: "Dicc", icon: "📘" },
          { label: "Reemp", icon: "🔂" }
        ];


        let wmeSDK = null; // Almacena la instancia del SDK de WME.

        //****************************************************************************************************
        // Nombre: tryInitializeSDK
        // Autor: mincho77
        // Fecha: 2025-05-27
        // Descripción: Intenta inicializar el SDK de WME de forma asíncrona.
        // Parámetros:
        //   finalCallback (Function): Función a llamar después del intento de inicialización (exitoso o no).
        //****************************************************************************************************
        function tryInitializeSDK(finalCallback)
        {
            let attempts = 0;
            const maxAttempts = 60; // Intentos máximos (60 * 500ms = 30 segundos)
            const intervalTime = 500;
            let sdkAttemptInterval = null;

            function attempt()
            {
                if (typeof getWmeSdk === 'function')
                {
                    if (sdkAttemptInterval) clearInterval(sdkAttemptInterval);
                    try
                    {
                        wmeSDK = getWmeSdk({
                            scriptId : 'WMEPlacesNameInspector',
                            scriptName : SCRIPT_NAME,
                        });
                        if (wmeSDK) {
                            // console.log("[SDK INIT SUCCESS] SDK obtenido exitosamente:", wmeSDK);
                        } else {
                            // console.warn("[SDK INIT WARNING] getWmeSdk() fue llamada pero devolvió null/undefined.");
                        }
                    }
                    catch (e)
                    {
                        // console.error("[SDK INIT ERROR] Error al llamar a getWmeSdk():", e);
                        wmeSDK = null;
                    }
                    finalCallback();
                    return;
                }
                attempts++;
                if (attempts >= maxAttempts)
                {
                    if (sdkAttemptInterval) clearInterval(sdkAttemptInterval);
                    // console.error(`[SDK INIT FAILURE] No se pudo encontrar getWmeSdk() después de ${maxAttempts} intentos.`);
                    wmeSDK = null;
                    finalCallback();
                }
            }
            sdkAttemptInterval = setInterval(attempt, intervalTime);
            attempt();
        }
        //****************************************************************************************************

    function waitForWazeAPI(callbackPrincipalDelScript)
    {
        let wAttempts = 0;
        const wMaxAttempts = 40;
        const wInterval = setInterval(() => {
            wAttempts++;
            if (typeof W !== 'undefined' && W.map && W.loginManager && W.model &&
                W.model.venues && W.userscripts &&
                typeof W.userscripts.registerSidebarTab === 'function')
            {
                clearInterval(wInterval);
               // console.log("✅ Waze API (objeto W) cargado correctamente.");
                tryInitializeSDK(callbackPrincipalDelScript);
            }
            else if (wAttempts >= wMaxAttempts)
            {
                clearInterval(wInterval);
              //  console.error("Error: No se pudo cargar la API principal de Waze (objeto W) después de varios intentos.");
                callbackPrincipalDelScript();
            }
        }, 500);
    }


    function updateScanProgressBar(currentIndex, totalPlaces)
    {
        if (totalPlaces === 0)
        {
            return;
        }

        let progressPercent = Math.floor(((currentIndex + 1) / totalPlaces) * 100);
        progressPercent = Math.min(progressPercent, 100);

        const progressBarInnerTab = document.getElementById("progressBarInnerTab");
        const progressBarTextTab = document.getElementById("progressBarTextTab");

        if (progressBarInnerTab && progressBarTextTab)
        {
            progressBarInnerTab.style.width = `${progressPercent}%`;
            const currentDisplay = Math.min(currentIndex + 1, totalPlaces);
            progressBarTextTab.textContent =
              `Progreso: ${progressPercent}% (${currentDisplay}/${totalPlaces})`;
        }
    }

    // Función para escapar caracteres especiales en expresiones regulares
    function escapeRegExp(string)
    {
        return string.replace(/[.*+?^${}()|[\]\\]/g, '\\$&');
    }

    // Función para eliminar tildes/diacríticos de una cadena
    function removeDiacritics(str)
    {
        return str.normalize("NFD").replace(/[\u0300-\u036f]/g, "");
    }
    // Función para calcular la similitud entre dos cadenas
    function isValidExcludedWord(newWord)
    {
        if (!newWord)
            return { valid : false, msg : "La palabra no puede estar vacía." };

        const lowerNewWord = newWord.toLowerCase();

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

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

        // No permitir si ya está en el diccionario (ignorando mayúsculas)
        if (window.dictionaryWords &&
            Array.from(window.dictionaryWords)
              .some(w => w.toLowerCase() === lowerNewWord))
        {
            return {
                valid : false,
                msg :
                  "La palabra ya existe en el diccionario. No se puede agregar a especiales."
            };
        }

        // No permitir palabras comunes
        if (commonWords.includes(lowerNewWord))
        {
            return {
                valid : false,
                msg : "Esa palabra es muy común y no debe agregarse a la lista."
            };
        }

        // No permitir duplicados en excluidas
        if (excludedWords &&
            Array.from(excludedWords).some(w => w.toLowerCase() === lowerNewWord))
        {
            return {
                valid : false,
                msg : "La palabra ya está en la lista (ignorando mayúsculas)."
            };
        }

        return { valid : true };
    }

    // Función para calcular la similitud entre dos cadenas
    function aplicarReemplazosGenerales(name)
    {
        if (typeof window.skipGeneralReplacements === "boolean" &&
            window.skipGeneralReplacements)
        {
            return name;
        }
        const reglas = [
            // Nueva regla: reemplazar | por espacio, guion y espacio
            { buscar: /\|/g, reemplazar: " - " },
            // Nueva regla: reemplazar / por espacio, barra y espacio, eliminando espacios alrededor
            { buscar : /\s*\/\s*/g, reemplazar : " / " },
            { buscar : /\[P\]|\[p\]/g, reemplazar : "" },
            { buscar : /\s*-\s*/g, reemplazar : " - " },
        ];
        reglas.forEach(
          regla => { name = name.replace(regla.buscar, regla.reemplazar); });
        name = name.replace(/\s{2,}/g, ' ');
        return name;
    }

    function aplicarReglasEspecialesNombre(newName)
    {
        newName = newName.replace(/([A-Za-z])'([A-Za-z])/g,
                                  (match, before, after) =>
                                    `${before}'${after.toLowerCase()}`);
        newName = newName.replace(/-\s*([a-z])/g,
                                  (match, letter) => `- ${letter.toUpperCase()}`);
        newName = newName.replace(/\.\s+([a-z])/g,
                                  (match, letter) => `. ${letter.toUpperCase()}`);
        // --- INICIO: NUEVA REGLA PARA CAPITALIZAR DESPUÉS DE PARÉNTESIS DE APERTURA ---
        newName = newName.replace(/(\(\s*)([a-z])/g,
            (match, P1, P2) => {
                // P1 es el paréntesis de apertura y cualquier espacio después (ej. "( " o "(")
                // P2 es la primera letra minúscula después de eso.
                return P1 + P2.toUpperCase();
            }
        );
        // --- FIN: NUEVA REGLA ---
        const frasesAlFinal = [
            "Conjunto Residencial",
            "Urbanización",
            "Conjunto Cerrado",
            "Unidad Residencial",
            "Parcelación",
            "Condominio Campestre",
            "Condominio",
            "Edificio",
            "Conjunto Habitacional",
            "Apartamentos",
            "Club Campestre",
            "Club Residencial",
            "Motel",
            "Restaurante",
            "Eco Hotel",
            "Finca Hotel",
            "Hotel"
        ];

        for (const frase of frasesAlFinal)
        {
            const regex = new RegExp(`\\s+${escapeRegExp(frase)}$`, 'i');
            if (regex.test(newName))
            {
                const match = newName.match(regex);
                const fraseEncontrada = match[0].trim();
                const restoDelNombre = newName.replace(regex, '').trim();
                newName = `${fraseEncontrada} ${restoDelNombre}`;
                break;
            }
        }

        newName = newName.replace(/\s([a-zA-Z])$/,
                                  (match, letter) => ` ${letter.toUpperCase()}`);
        return newName;
    }



    function aplicarReemplazosDefinidos(text, replacementRules) {
        let newText = text;
        // Verificar si replacementRules es un objeto y tiene claves
        if (typeof replacementRules !== 'object' || replacementRules === null || Object.keys(replacementRules).length === 0) {
            return newText; // No hay reglas o no es un objeto, devolver el texto original
        }

        // Ordenar las claves de reemplazo por longitud descendente para manejar correctamente
        // los casos donde una clave puede ser subcadena de otra (ej. "D1 Super" y "D1").
        const sortedFromKeys = Object.keys(replacementRules).sort((a, b) => b.length - a.length);

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

            // Regex mejorada para definir "palabra completa" usando propiedades Unicode.
            // Grupo 1: Captura el delimitador previo (un no-palabra o inicio de cadena).
            // Grupo 2: Captura la clave de reemplazo que coincidió.
            // Grupo 3: Captura el delimitador posterior o fin de cadena.
            const regex = new RegExp(`(^|[^\\p{L}\\p{N}_])(${escapedFromKey})($|(?=[^\\p{L}\\p{N}_]))`, 'giu');

            newText = newText.replace(regex, (
                match,                          // La subcadena coincidente completa
                delimitadorPrevio,              // Contenido del grupo de captura 1
                matchedKey,                     // Contenido del grupo de captura 2
                _delimitadorPosteriorOculto,    // Contenido del grupo de captura 3 (no usado directamente en la lógica de abajo, pero necesario para alinear parámetros)
                offsetOfMatchInCurrentText,     // El desplazamiento de la subcadena coincidente
                stringBeingProcessed            // La cadena completa que se está examinando
            ) => {

                // Asegurarse de que stringBeingProcessed es una cadena
                if (typeof stringBeingProcessed !== 'string') {
                    console.error(
                        "[WME PLN Error] aplicarReemplazosDefinidos: el argumento 'stringBeingProcessed' (la cadena original en el reemplazo) no es una cadena.",
                        "Tipo:", typeof stringBeingProcessed, "Valor:", stringBeingProcessed,
                        "Regla actual (fromKey):", fromKey, "Texto parcial procesado:", newText
                    );
                    return match;
                }
                // Asegurarse de que offsetOfMatchInCurrentText es un número
                 if (typeof offsetOfMatchInCurrentText !== 'number') {
                    console.error(
                        "[WME PLN Error] aplicarReemplazosDefinidos: el argumento 'offsetOfMatchInCurrentText' no es un número.",
                        "Tipo:", typeof offsetOfMatchInCurrentText, "Valor:", offsetOfMatchInCurrentText
                    );
                    return match;
                }


                // Lógica existente para evitar el reemplazo si toValue ya está presente contextualizando fromKey.
                const fromKeyLower = fromKey.toLowerCase();
                const toValueLower = toValue.toLowerCase();
                const indexOfFromInTo = toValueLower.indexOf(fromKeyLower);

                if (indexOfFromInTo !== -1) {
                    // El offset real de matchedKey dentro de stringBeingProcessed
                    const actualMatchedKeyOffset = offsetOfMatchInCurrentText + (delimitadorPrevio ? delimitadorPrevio.length : 0);
                    const potentialExistingToStart = actualMatchedKeyOffset - indexOfFromInTo;

                    if (potentialExistingToStart >= 0 &&
                        (potentialExistingToStart + toValue.length) <= stringBeingProcessed.length) {
                        const substringInOriginal = stringBeingProcessed.substring(potentialExistingToStart, potentialExistingToStart + toValue.length);
                        if (substringInOriginal.toLowerCase() === toValueLower) {
                            return match;
                        }
                    }
                }

                // --- INICIO: NUEVA LÓGICA PARA EVITAR DUPLICACIÓN DE PALABRAS EN LOS BORDES ---
                const palabrasDelToValue = toValue.trim().split(/\s+/).filter(p => p.length >0); // Filtrar palabras vacías
                if (palabrasDelToValue.length > 0) {
                    const primeraPalabraToValueLimpia = removeDiacritics(palabrasDelToValue[0].toLowerCase());
                    const ultimaPalabraToValueLimpia = removeDiacritics(palabrasDelToValue[palabrasDelToValue.length - 1].toLowerCase());

                    // Palabra ANTES del inicio del 'match' (delimitadorPrevio + matchedKey)
                    // El offsetOfMatchInCurrentText es el inicio de 'match', no de 'delimitadorPrevio' si son diferentes.
                    // Necesitamos el texto ANTES de delimitadorPrevio. El offset real del inicio del match completo es offsetOfMatchInCurrentText.
                    const textoAntesDelMatch = stringBeingProcessed.substring(0, offsetOfMatchInCurrentText);
                    const palabrasEnTextoAntes = textoAntesDelMatch.trim().split(/\s+/).filter(p => p.length > 0);
                    const palabraAnteriorLimpia = palabrasEnTextoAntes.length > 0 ? removeDiacritics(palabrasEnTextoAntes[palabrasEnTextoAntes.length - 1].toLowerCase()) : "";

                    // Palabra DESPUÉS del final del 'match' (delimitadorPrevio + matchedKey)
                    const textoDespuesDelMatch = stringBeingProcessed.substring(offsetOfMatchInCurrentText + match.length);
                    const palabrasEnTextoDespues = textoDespuesDelMatch.trim().split(/\s+/).filter(p => p.length > 0);
                    const palabraSiguienteLimpia = palabrasEnTextoDespues.length > 0 ? removeDiacritics(palabrasEnTextoDespues[0].toLowerCase()) : "";


                    if (palabraAnteriorLimpia && primeraPalabraToValueLimpia && palabraAnteriorLimpia === primeraPalabraToValueLimpia) {
                         // Solo prevenir si el delimitador previo es solo espacio o vacío,
                         // indicando adyacencia real de palabras.
                        if (delimitadorPrevio.trim() === "" || delimitadorPrevio.match(/^\s+$/)) {
                           return match;
                        }
                    }

                    if (palabraSiguienteLimpia && ultimaPalabraToValueLimpia && ultimaPalabraToValueLimpia === palabraSiguienteLimpia) {
                        // El delimitadorPosteriorOculto nos dice qué sigue inmediatamente al matchedKey.
                        // Si este delimitador es solo espacio o vacío, y luego viene la palabra duplicada.
                        if (_delimitadorPosteriorOculto.trim() === "" || _delimitadorPosteriorOculto.match(/^\s+$/)) {
                           return match;
                        }
                    }
                }
                // --- FIN: NUEVA LÓGICA PARA EVITAR DUPLICACIÓN ---

                return delimitadorPrevio + toValue;
            });
        }
        return newText;
    }

    function getVisiblePlaces()
    {
        if (typeof W === 'undefined' || !W.map || !W.model || !W.model.venues)
        {
            console.warn('Waze Map Editor no está completamente cargado.');
            return [];
        }

        const venues = W.model.venues.objects;
        const visiblePlaces = Object.values(venues).filter(venue => {
            const olGeometry = venue.getOLGeometry?.();
            const bounds = olGeometry?.getBounds?.();
            return bounds && W.map.getExtent().intersectsBounds(bounds);
        });

        return visiblePlaces;
    }

    //**************************************************************************************************************************************
    // Nombre: renderPlacesInFloatingPanel
    // Autor: mincho77
    // Fecha: 2025-05-27
    // Descripción: Procesa y muestra los lugares con posibles inconsistencias en un panel flotante.
    //              Incluye la lógica para obtener detalles de cada lugar, aplicar normalizaciones,
    //              comparar con el nombre original y generar sugerencias.
    // Parámetros:
    //   places (Array): Un array de objetos 'venue' de WME para analizar.
    //**************************************************************************************************************************************
    function renderPlacesInFloatingPanel(places)
    {
        createFloatingPanel("processing"); // Mostrar panel en modo "procesando"
        const maxPlacesToScan = parseInt(document.getElementById("maxPlacesInput")?.value || "100", 10);
        if (places.length > maxPlacesToScan)
        {
            places = places.slice(0, maxPlacesToScan); // Limitar el número de places a escanear
        }

        //  console.log("[DEBUG] Preparando panel flotante. Total places a analizar:",places.length);

        // --- Funciones auxiliares para tipo y categoría ---
        function getPlaceCategoryName(venueFromOldModel, venueSDKObject)
        { // Acepta ambos tipos de venue
            let categoryId = null;
            let categoryName = null;
            // let source = ""; // Para saber de dónde vino la info (comentario de depuración eliminado)

            // Intento 1: Usar el venueSDKObject si está disponible y tiene la info
            if (venueSDKObject)
            {
                if (venueSDKObject.mainCategory && venueSDKObject.mainCategory.id)
                {
                    categoryId = venueSDKObject.mainCategory.id;
                    if (venueSDKObject.mainCategory.name)
                    {
                        categoryName = venueSDKObject.mainCategory.name;
                        // source = "SDK (mainCategory.name)";
                    }
                }
                else if (Array.isArray(venueSDKObject.categories) &&
                         venueSDKObject.categories.length > 0)
                {
                    const firstCategorySDK = venueSDKObject.categories[0];
                    if (typeof firstCategorySDK === 'object' && firstCategorySDK.id)
                    {
                        categoryId = firstCategorySDK.id;
                        if (firstCategorySDK.name)
                        {
                            categoryName = firstCategorySDK.name;
                            // source = "SDK (categories[0].name)";
                        }
                    }
                    else if (typeof firstCategorySDK === 'string')
                    {
                        categoryName = firstCategorySDK;
                        // source = "SDK (categories[0] as string)";
                    }
                }
                else if (venueSDKObject.primaryCategoryID)
                {
                    categoryId = venueSDKObject.primaryCategoryID;
                    // source = "SDK (primaryCategoryID)";
                }
            }

            if (categoryName)
            {
                // console.log(`[CATEGORÍA] Usando nombre de categoría de ${source}: ${categoryName} ${categoryId ? `(ID: ${categoryId})` : ''}`); // Comentario de depuración eliminado
                return categoryName;
            }

            // Intento 2: Usar W.model si no se obtuvo del SDK
            if (!categoryId && venueFromOldModel && venueFromOldModel.attributes &&
                Array.isArray(venueFromOldModel.attributes.categories) &&
                venueFromOldModel.attributes.categories.length > 0)
            {
                categoryId = venueFromOldModel.attributes.categories[0];
                // source = "W.model (attributes.categories[0])";
            }

            if (!categoryId)
            {
                return "Sin categoría";
            }

            let categoryObjWModel = null;
            if (typeof W !== 'undefined' && W.model)
            {
                if (W.model.venueCategories &&
                    typeof W.model.venueCategories.getObjectById === "function")
                {
                    categoryObjWModel =
                      W.model.venueCategories.getObjectById(categoryId);
                }
                if (!categoryObjWModel && W.model.categories &&
                    typeof W.model.categories.getObjectById === "function")
                {
                    categoryObjWModel =
                      W.model.categories.getObjectById(categoryId);
                }
            }

            if (categoryObjWModel && categoryObjWModel.attributes &&
                categoryObjWModel.attributes.name)
            {
                // console.log(`[CATEGORÍA] Usando nombre de categoría de W.model.categories (para ID ${categoryId} de ${source}): ${categoryObjWModel.attributes.name}`); // Comentario de depuración eliminado
                return categoryObjWModel.attributes.name;
            }

            if (typeof categoryId === 'number' ||
                (typeof categoryId === 'string' && categoryId.trim() !== ''))
            {
                // console.log(`[CATEGORÍA] No se pudo resolver el nombre para ID de categoría ${categoryId} (obtenido de ${source}). Devolviendo ID.`); // Comentario de depuración eliminado
                return `${categoryId}`; // Devuelve el ID si no se encuentra el nombre.
            }

            return "Sin categoría";
        }


        //****************************************************************************************************
        // Nombre: getPlaceTypeInfo
        // Autor: mincho77
        // Fecha: 2025-05-27
        // Descripción: Determina si un lugar es un área o un punto y devuelve información iconográfica relacionada.
        // Parámetros:
        //   venue (Object): El objeto 'venue' de WME.
        // Retorna:
        //   Object: Un objeto con las propiedades 'isArea' (boolean), 'icon' (String) y 'title' (String).
        //****************************************************************************************************
        function getPlaceTypeInfo(venue)
        {
            const geometry = venue?.getOLGeometry ? venue.getOLGeometry() : null;
            const isArea = geometry?.CLASS_NAME?.endsWith("Polygon");
            return {
                isArea,
                icon : isArea ? "⭔" : "⊙", // Icono para área o punto
                title : isArea ? "Área" : "Punto"
            };
        }
        //****************************************************************************************************
//****************************************************************************************************
// Nombre: shouldForceSuggestionForReview
// Autor: Gemini AI / mincho77
// Fecha: 2025-05-27 // Ajusta la fecha según sea necesario
// Descripción: Determina si una palabra del diccionario debe mostrarse siempre como sugerencia
//              para revisión manual debido a tildes y letras/combinaciones específicas.
// Parámetros:
//   word (String): La palabra del diccionario a evaluar.
// Retorna:
//   Boolean: true si la palabra cumple los criterios, false en caso contrario.
//****************************************************************************************************
function shouldForceSuggestionForReview(word) {
    if (typeof word !== 'string') {
        return false;
    }
    const lowerWord = word.toLowerCase();

    // Verificar si la palabra tiene alguna tilde (incluyendo mayúsculas acentuadas)
    const hasTilde = /[áéíóúÁÉÍÓÚ]/.test(word);

    if (!hasTilde) {
        return false; // Si no hay tilde, no forzar sugerencia por esta regla
    }

    // Lista de patrones de letras/combinaciones que, junto con una tilde, fuerzan la sugerencia
    // (insensible a mayúsculas debido a lowerWord)
    const problematicSubstrings = [
        'c', 's', 'x', 'cc', 'sc', 'cs', 'g', 'j', 'b', 'v','z','ll', 'y'
        // Podrías añadir más si es necesario, ej. 'z', 'b', 'v' si son problemáticas en tu contexto
    ];

    for (const sub of problematicSubstrings)
    {
        if (lowerWord.includes(sub)) {
            return true; // Tiene tilde y una de las letras/combinaciones problemáticas
        }
    }

    return false; // Tiene tilde, pero no una de las letras/combinaciones problemáticas
}
//****************************************************************************************************



        // --- Renderizar barra de progreso en el TAB PRINCIPAL justo después del
        // slice ---
        const tabOutput = document.querySelector("#wme-normalization-tab-output");
        if (tabOutput)
        {
            // Reiniciar el estilo del mensaje en el tab al valor predeterminado
            tabOutput.style.color = "#000";
            tabOutput.style.fontWeight = "normal";
            // Crear barra de progreso visual
            const progressBarWrapperTab = document.createElement("div");
            progressBarWrapperTab.style.margin = "10px 0";
            progressBarWrapperTab.style.marginTop = "10px";
            progressBarWrapperTab.style.height = "18px";
            progressBarWrapperTab.style.backgroundColor = "transparent";

            const progressBarTab = document.createElement("div");
            progressBarTab.style.height = "100%";
            progressBarTab.style.width = "0%";
            progressBarTab.style.backgroundColor = "#007bff";
            progressBarTab.style.transition = "width 0.2s";
            progressBarTab.id = "progressBarInnerTab";
            progressBarWrapperTab.appendChild(progressBarTab);

            const progressTextTab = document.createElement("div");
            progressTextTab.style.fontSize = "12px";
            progressTextTab.style.marginTop = "5px";
            progressTextTab.id = "progressBarTextTab";
            tabOutput.appendChild(progressBarWrapperTab);
            tabOutput.appendChild(progressTextTab);
        }

        // Asegurar que la barra de progreso en el tab se actualice desde el
        // principio
        const progressBarInnerTab = document.getElementById("progressBarInnerTab");
        const progressBarTextTab = document.getElementById("progressBarTextTab");
        if (progressBarInnerTab && progressBarTextTab)
        {
            progressBarInnerTab.style.width = "0%";
            progressBarTextTab.textContent = `Progreso: 0% (0/${places.length})`;
        }

        // Mostrar el panel flotante desde el inicio
        // createFloatingPanel(0); // Mostrar el panel flotante desde el inicio

        // --- PANEL FLOTANTE: limpiar y preparar salida ---
        const output = document.querySelector("#wme-place-inspector-output");
        if (!output) {
            console.error("❌ Panel flotante no está disponible");
            return;
        }
        output.innerHTML = ""; // Limpia completamente el contenido del panel flotante
        output.innerHTML = "<div style='display:flex; align-items:center; gap:10px;'><span class='loader-spinner' style='width:16px; height:16px; border:2px solid #ccc; border-top:2px solid #007bff; border-radius:50%; animation:spin 0.8s linear infinite;'></span><div><div id='processingText'>Procesando lugares visibles<span class='dots'>.</span></div><div id='processingStep' style='font-size:11px; color:#555;'>Inicializando escaneo...</div></div></div>";

        // Animación de puntos suspensivos
        const dotsSpan = output.querySelector(".dots");
        if (dotsSpan)
        {
            const dotStates = ["", ".", "..", "..."];
            let dotIndex = 0;
            window.processingDotsInterval = setInterval(() => {
                dotIndex = (dotIndex + 1) % dotStates.length;
                dotsSpan.textContent = dotStates[dotIndex];
            }, 500);
        }
        output.style.height = "calc(55vh - 40px)";

        // Si no hay places, mostrar mensaje y salir
        if (!places.length)
        {
            output.appendChild(
              document.createTextNode("No hay places visibles para analizar."));
            const existingOverlay = document.getElementById("scanSpinnerOverlay");
            if (existingOverlay)
                existingOverlay.remove();
            return;
        }

        // --- Procesamiento incremental para evitar congelamiento ---
        let inconsistents = [];
        let index = 0;

        // Remover ícono de ✔ previo si existe
        const scanBtn = document.querySelector("button[type='button']");
        if (scanBtn)
        {
            const existingCheck = scanBtn.querySelector("span");
            if (existingCheck)
            {
                existingCheck.remove();
            }
        }

        // --- Sugerencias por palabra global para toda la ejecución ---
        let sugerenciasPorPalabra = {};
        // Convertir excludedWords a array solo una vez al inicio del análisis,
        // seguro ante undefined
        const excludedArray =
          (typeof excludedWords !== "undefined" && Array.isArray(excludedWords))
            ? excludedWords
            : (typeof excludedWords !== "undefined" ? Array.from(excludedWords)
                                                    : []);


        async function processNextPlace()
        {
            // 1. Leer estados de checkboxes y configuraciones iniciales
            const chkHideMyEditsChecked = document.getElementById("chk-hide-my-edits")?.checked ?? false;

            const useFullPipeline =
              true; // Como definimos, siempre ejecutar el flujo completo
            const applyGeneralReplacements =
              useFullPipeline ||
              (document.getElementById("chk-general-replacements")?.checked ??
               true);
            const checkExcludedWords =
              useFullPipeline ||
              (document.getElementById("chk-check-excluded")?.checked ?? false);
            const checkDictionaryWords =
              true;
            const restoreCommas =
              document.getElementById("chk-restore-commas")?.checked ?? false;
            const similarityThreshold =
              parseFloat(document.getElementById("similarityThreshold")?.value ||
                         "85") /
              100;

            // 2. Condición de salida principal (todos los lugares procesados)
            if (index >= places.length)
            {
                //console.log("[DEBUG] Todos los lugares procesados. Finalizando render...");
                finalizeRender(inconsistents, places);
                return;
            }

            const venueFromOldModel =
              places[index]; // Usaremos este para datos base y fallback
            const currentVenueNameObj = venueFromOldModel?.attributes?.name;
            const nameValue = typeof currentVenueNameObj === 'object' && currentVenueNameObj !== null &&
                              typeof currentVenueNameObj.value === 'string' ? currentVenueNameObj.value.trim() !== ''
                                    ? currentVenueNameObj.value : undefined : typeof currentVenueNameObj === 'string' &&
                                  currentVenueNameObj.trim() !== '' ? currentVenueNameObj : undefined;

            // Protección extra si el place en el índice es inválido
            if (!places[index] || typeof places[index] !== 'object') {
               // console.warn(`[DEBUG] Lugar inválido o tipo inesperado en el índice ${index}:`, places[index]);
                updateScanProgressBar(index, places.length);
                index++;
                setTimeout(() => processNextPlace(), 0);
                return;
            }

            // 3. Salto temprano si el venue es inválido o no tiene nombre
            if (!venueFromOldModel || typeof venueFromOldModel !== 'object' ||
                !venueFromOldModel.attributes || typeof nameValue !== 'string' || nameValue.trim() === '')
            {
                //console.warn(`[DEBUG] Lugar inválido o sin nombre en el índice ${index}:`, venueFromOldModel);
                updateScanProgressBar(index, places.length);
                index++;
                setTimeout(() => processNextPlace(), 0);
                return;
            }

            const originalName = venueFromOldModel?.attributes?.name?.value ||
                                 venueFromOldModel?.attributes?.name || '';
            const currentVenueId = venueFromOldModel.getID();

            // 4. --- OBTENER INFO DEL EDITOR Y DEFINIR wasEditedByMe (PRIORIZANDO
            // SDK) ---
            let lastEditorInfoForLog = "Editor: Desconocido";
            let lastEditorIdForComparison = null;
            let wasEditedByMe = false;

            let currentLoggedInUserId = null;
            let currentLoggedInUserName =
              null; // Para comparar si el SDK devuelve nombre

            if (typeof W !== 'undefined' && W.loginManager && W.loginManager.user)
            {
                if (typeof W.loginManager.user.id === 'number')
                {
                    currentLoggedInUserId = W.loginManager.user.id;
                }
                if (typeof W.loginManager.user.userName === 'string')
                {
                    currentLoggedInUserName = W.loginManager.user.userName;
                }
            }

            if (wmeSDK && wmeSDK.DataModel && wmeSDK.DataModel.Venues &&
                wmeSDK.DataModel.Venues.getById)
            {
                // console.log(`[SDK] Intentando obtener venue ID: ${currentVenueId} con sdk.DataModel.Venues.getById`);
                try
                {
                    const venueSDK = await wmeSDK.DataModel.Venues.getById(
                      { venueId : currentVenueId });

                    if (venueSDK && venueSDK.modificationData)
                    {
                        const updatedByDataFromSDK =
                          venueSDK.modificationData.updatedBy;
                        /*  console.log(
                            `[SDK] Para ID ${
                              currentVenueId}, venue.modificationData.updatedBy: `,
                            updatedByDataFromSDK,
                            `(Tipo: ${typeof updatedByDataFromSDK})`);*/

                        if (typeof updatedByDataFromSDK === 'string' &&
                            updatedByDataFromSDK.trim() !== '')
                        {
                            lastEditorInfoForLog =
                              `Editor (SDK): ${updatedByDataFromSDK}`;
                            resolvedEditorName =
                              updatedByDataFromSDK; // SDK provided username
                                                    // directly
                            if (currentLoggedInUserName &&
                                currentLoggedInUserName === updatedByDataFromSDK)
                            {
                                wasEditedByMe = true;
                            }
                        }
                        else if (typeof updatedByDataFromSDK === 'number')
                        {
                            lastEditorInfoForLog =
                              `Editor (SDK): ID ${updatedByDataFromSDK}`;
                            resolvedEditorName =
                              `ID ${updatedByDataFromSDK}`; // Default to ID
                            lastEditorIdForComparison = updatedByDataFromSDK;
                            if (typeof W !== 'undefined' && W.model &&
                                W.model.users)
                            {
                                const userObjectW =
                                  W.model.users.getObjectById(updatedByDataFromSDK);
                                if (userObjectW && userObjectW.userName)
                                {
                                    lastEditorInfoForLog = `Editor (SDK ID ${
                                      updatedByDataFromSDK} -> W.model): ${
                                      userObjectW.userName}`;
                                    resolvedEditorName =
                                      userObjectW
                                        .userName; // Username found via W.model
                                }
                                else if (userObjectW)
                                {
                                    lastEditorInfoForLog = `Editor (SDK ID ${
                                      updatedByDataFromSDK} -> W.model): ID ${
                                      updatedByDataFromSDK} (sin userName en W.model)`;
                                    // resolvedEditorName remains `ID
                                    // ${updatedByDataFromSDK}`
                                }
                            }
                        }
                        else if (updatedByDataFromSDK === null)
                        {
                            lastEditorInfoForLog =
                              "Editor (SDK): N/D (updatedBy es null)";
                            resolvedEditorName = "N/D";
                        }
                        else
                        {
                            lastEditorInfoForLog =
                              `Editor (SDK): Valor inesperado para updatedBy ('${
                                updatedByDataFromSDK}')`;
                            resolvedEditorName = "Inesperado (SDK)";
                        }
                    }
                    else
                    { // venueSDK o venueSDK.modificationData no encontrados
                        lastEditorInfoForLog =
                          "Editor (SDK): No venue/modificationData. Usando fallback W.model.";
                        if (venueSDK)
                            console.log(
                              "[SDK] venueSDK.modificationData no encontrado. Venue SDK:",
                              venueSDK);
                        else
                            console.log("[SDK] venueSDK no fue obtenido para ID:", currentVenueId);

                        const oldModelUpdatedBy =
                          venueFromOldModel.attributes.updatedBy;
                        if (oldModelUpdatedBy !== null &&
                            oldModelUpdatedBy !== undefined)
                        {
                            lastEditorIdForComparison = oldModelUpdatedBy;
                            resolvedEditorName =
                              `ID ${oldModelUpdatedBy}`; // Default to ID
                            let usernameFromOldModel = `ID ${oldModelUpdatedBy}`;
                            if (typeof W !== 'undefined' && W.model &&
                                W.model.users)
                            {
                                const userObjectW =
                                  W.model.users.getObjectById(oldModelUpdatedBy);
                                if (userObjectW && userObjectW.userName)
                                {
                                    usernameFromOldModel = userObjectW.userName;
                                    resolvedEditorName =
                                      userObjectW.userName; // Username found
                                }
                                else if (userObjectW)
                                {
                                    usernameFromOldModel =
                                      `ID ${oldModelUpdatedBy} (sin userName)`;
                                }
                            }
                            lastEditorInfoForLog =
                              `Editor (W.model Fallback): ${usernameFromOldModel}`;
                        }
                        else
                        {
                            lastEditorInfoForLog = "Editor (W.model Fallback): N/D";
                            resolvedEditorName = "N/D";
                        }
                    }
                }
                catch (e)
                {
                  //  console.error(`[SDK] Error al obtener venue ${currentVenueId} con el SDK:`, e);
                    lastEditorInfoForLog = "Editor: Error con SDK (usando fallback W.model)";
                    resolvedEditorName = "Error SDK"; // Initial state for error
                    const oldModelUpdatedBy =
                      venueFromOldModel.attributes.updatedBy;
                    if (oldModelUpdatedBy !== null &&
                        oldModelUpdatedBy !== undefined)
                    {
                        lastEditorIdForComparison = oldModelUpdatedBy;
                        resolvedEditorName =
                          `ID ${oldModelUpdatedBy}`; // Default to ID on fallback
                        let usernameFromOldModel = `ID ${oldModelUpdatedBy}`;
                        if (typeof W !== 'undefined' && W.model && W.model.users)
                        {
                            const userObjectW =
                              W.model.users.getObjectById(oldModelUpdatedBy);
                            if (userObjectW && userObjectW.userName)
                            {
                                usernameFromOldModel = userObjectW.userName;
                                resolvedEditorName =
                                  userObjectW.userName; // Username found
                            }
                            else if (userObjectW)
                            {
                                usernameFromOldModel =
                                  `ID ${oldModelUpdatedBy} (sin userName)`;
                            }
                        }
                        lastEditorInfoForLog =
                          `Editor (W.model Fallback): ${usernameFromOldModel}`;
                    }
                    else
                    {
                        lastEditorInfoForLog = "Editor (W.model Fallback): N/D";
                        resolvedEditorName = "N/D";
                    }
                }
            }
            else
            {
                // Fallback completo a W.model si el SDK no está inicializado
                const oldModelUpdatedBy = venueFromOldModel.attributes.updatedBy;
                if (oldModelUpdatedBy !== null && oldModelUpdatedBy !== undefined)
                {
                    lastEditorIdForComparison = oldModelUpdatedBy;
                    resolvedEditorName = `ID ${oldModelUpdatedBy}`; // Default to ID
                    let usernameFromOldModel = `ID ${oldModelUpdatedBy}`;
                    if (typeof W !== 'undefined' && W.model && W.model.users)
                    {
                        const userObjectW =
                          W.model.users.getObjectById(oldModelUpdatedBy);
                        if (userObjectW && userObjectW.userName)
                        {
                            usernameFromOldModel = userObjectW.userName;
                            resolvedEditorName =
                              userObjectW.userName; // Username found
                        }
                        else if (userObjectW)
                        {
                            usernameFromOldModel =
                              `ID ${oldModelUpdatedBy} (sin userName)`;
                        }
                    }
                    lastEditorInfoForLog =
                      `Editor (W.model): ${usernameFromOldModel}`;
                }
                else
                {
                    lastEditorInfoForLog = "Editor (W.model): N/D";
                    resolvedEditorName = "N/D";
                }
            }

            // Actualizar wasEditedByMe basado en ID numérico si lo tenemos
            if (currentLoggedInUserId !== null &&
                typeof lastEditorIdForComparison === 'number' &&
                currentLoggedInUserId === lastEditorIdForComparison)
            {
                wasEditedByMe = true;
            }

            //console.log(`[DEBUG] Usuario logueado: ${currentLoggedInUserName} (ID: ${currentLoggedInUserId})`);

            const editedByMeText = wasEditedByMe ? ' (Editado por mí)' : '';
            /*  console.log(`[INFO EDITOR LUGAR] Nombre: "${originalName}" (ID: ${
                currentVenueId}) | ${lastEditorInfoForLog}${editedByMeText}`);*/
            // ---- FIN INFO DEL EDITOR ----

            // 5. --- PROCESAMIENTO DEL NOMBRE PALABRA POR PALABRA ---
            let nombreSugeridoParcial = [];
            let sugerenciasLugar = {};
            const originalWords = originalName.split(/\s+/);
            const processingStepLabel = document.getElementById("processingStep");

            if (index === 0)
            {
                sugerenciasPorPalabra = {};
            }

            // Definir expresiones regulares fuera del bucle para eficiencia
            const romanRegexStrict = /^M{0,3}(CM|CD|D?C{0,3})(XC|XL|L?X{0,3})(IX|IV|V?I{0,3})$/; // Sensible a mayúsculas
            const romanRegexStrictInsensitive = /^M{0,3}(CM|CD|D?C{0,3})(XC|XL|L?X{0,3})(IX|IV|V?I{0,3})$/i; // Insensible

            originalWords.forEach((P, idx) => {
                const endsWithComma = restoreCommas && P.endsWith(",");
                const baseWord = endsWithComma ? P.slice(0, -1) : P;
                const cleaned = baseWord.trim();

                if (cleaned === "") {
                    nombreSugeridoParcial.push(cleaned);
                    return;
                }

                let isExcluded = false;
                let matchingExcludedWord = null;
                if (checkExcludedWords) {
                    matchingExcludedWord = excludedArray.find(
                      w_excluded => removeDiacritics(w_excluded.toLowerCase()) ===
                                    removeDiacritics(cleaned.toLowerCase()));
                    isExcluded = !!matchingExcludedWord;
                }

                let tempReplaced;
                const isCommon = commonWords.includes(cleaned.toLowerCase());
                // Usar la versión insensible para la detección inicial flexible de si ES un romano
                const isPotentiallyRomanNumeral = romanRegexStrictInsensitive.test(cleaned);

                if (isExcluded) {
                    tempReplaced = matchingExcludedWord;
                     // Si la palabra excluida es un número romano, asegurarse de que esté en mayúsculas
                    if (romanRegexStrictInsensitive.test(tempReplaced)) {
                        tempReplaced = tempReplaced.toUpperCase();
                    }
                } else {
                    let dictionaryFormToUse = null;

                    // Paso 1: Consultar el diccionario
                    if (checkDictionaryWords && window.dictionaryWords && typeof window.dictionaryWords.forEach === "function") {
                        const cleanedLowerNoDiacritics = removeDiacritics(cleaned.toLowerCase());
                        for (const diccWord of window.dictionaryWords) {
                            if (removeDiacritics(diccWord.toLowerCase()) === cleanedLowerNoDiacritics) {
                                // Si la palabra original es potencialmente romana, solo aceptamos del diccionario si también está en MAYÚSCULAS.
                                if (isPotentiallyRomanNumeral) {
                                    if (diccWord === diccWord.toUpperCase() && romanRegexStrict.test(diccWord)) { // Verificar que la forma del dicc también sea un romano válido en mayus
                                        dictionaryFormToUse = diccWord;
                                    }
                                } else {
                                    dictionaryFormToUse = diccWord;
                                }
                                break;
                            }
                        }
                    }

                    // Paso 2: Decidir la forma base
                    if (dictionaryFormToUse !== null) {
                        tempReplaced = dictionaryFormToUse;
                    } else {
                        // normalizePlaceName ya maneja la capitalización de romanos a MAYUS si no viene del diccionario
                        tempReplaced = normalizePlaceName(cleaned);
                    }

                    // Paso 3: Ajustes finales de capitalización y forzar MAYUS para romanos
                    if (isPotentiallyRomanNumeral) {
                        // Si se detectó como romano (incluso si el diccionario dio algo mixto o normalizePlaceName falló),
                        // forzar a mayúsculas. Usamos la expresión regular estricta (sensible a mayúsculas)
                        // para validar que la forma final sea un romano correcto.
                        const upperVersion = tempReplaced.toUpperCase();
                        if (romanRegexStrict.test(upperVersion)) { // Validar que la versión en mayúsculas sea un romano
                           tempReplaced = upperVersion;
                        } else {
                           // Si al pasar a mayúsculas deja de ser un romano válido (raro, pero posible si cleaned era "mixToRoman"),
                           // entonces aplicar capitalización normal.
                           tempReplaced = tempReplaced.charAt(0).toUpperCase() + tempReplaced.slice(1).toLowerCase();
                        }
                    } else if (idx === 0) { // Primera palabra y no es romano
                        tempReplaced = tempReplaced.charAt(0).toUpperCase() + tempReplaced.slice(1);
                    } else { // No es la primera palabra y no es romano
                        if (isCommon) {
                            tempReplaced = tempReplaced.toLowerCase();
                        } else {
                            // No es común, no es primera, no es romano: Capitalizar.
                            tempReplaced = tempReplaced.charAt(0).toUpperCase() + tempReplaced.slice(1);
                        }
                    }
                }

                // <<< DEBUG LOG INICIAL PARA "Panaderia" / "Cafe" >>>
                if (cleaned.toLowerCase() === "panaderia" || cleaned.toLowerCase() === "cafe" || cleaned.toLowerCase() === "ii") {
                    console.log(`[DEBUG WORD FINAL] Original: "${cleaned}", isExcluded: ${isExcluded}, matchingExcluded: "${matchingExcludedWord}", tempReplaced: "${tempReplaced}"`);
                }

                // Ahora, la lógica para generar las SUGERENCIAS CLICKEABLES del diccionario.
                if (!isExcluded && checkDictionaryWords && window.dictionaryWords &&
                    typeof window.dictionaryWords.forEach === "function") {

                    const processedWordForCompare = removeDiacritics(tempReplaced.toLowerCase());

                    window.dictionaryWords.forEach(diccWord => {
                        const diccWordForCompare = removeDiacritics(diccWord.toLowerCase());

                        if (processedWordForCompare === diccWordForCompare) {
                            if (diccWord !== tempReplaced) {
                                if (!sugerenciasLugar[baseWord]) {
                                    sugerenciasLugar[baseWord] = [];
                                }
                                if (!sugerenciasLugar[baseWord].some(s => s.word === diccWord && s.fuente === 'dictionary')) {
                                    sugerenciasLugar[baseWord].push({
                                        word: diccWord,
                                        similarity: 1,
                                        fuente: 'dictionary'
                                    });
                                }
                            }
                        }
                    });
                }

                // Sugerencias de palabras excluidas (se mantiene igual)
                if (checkExcludedWords) {
                    const similarExcluded =
                      findSimilarWords(cleaned, excludedArray, similarityThreshold)
                        .filter(s => s.similarity < 1);
                    if (similarExcluded.length > 0) {
                        sugerenciasLugar[baseWord] =
                          (sugerenciasLugar[baseWord] || [])
                            .concat(similarExcluded.map(
                              s => ({...s, fuente: 'excluded' })));
                    }
                }

                if (endsWithComma && !tempReplaced.endsWith(",")) {
                    tempReplaced += ",";
                }
                nombreSugeridoParcial.push(tempReplaced);
            });
            // ---- FIN PROCESAMIENTO PALABRA POR PALABRA ----

            // 6. --- COMPILACIÓN DE suggestedName ---
            const joinedSuggested = nombreSugeridoParcial.join(' ');

            let processedName = joinedSuggested;
            if (applyGeneralReplacements) {
                processedName = aplicarReemplazosGenerales(processedName);
            }
            processedName = aplicarReglasEspecialesNombre(processedName);
            processedName = postProcessQuotesAndParentheses(processedName);

            if (typeof replacementWords === 'object' && Object.keys(replacementWords).length > 0) {
                processedName = aplicarReemplazosDefinidos(processedName, replacementWords);
            }

            let suggestedName = processedName.replace(/\s{2,}/g, ' ').trim(); // Nombre base sugerido

            // ***** NUEVO: Eliminar punto solo si está al FINAL del nombre completo sugerido *****
            if (suggestedName.endsWith('.')) {
                suggestedName = suggestedName.slice(0, -1);
            }
            // ***** FIN NUEVO *****

            // ---- FIN COMPILACIÓN DE suggestedName ----
            console.log(`[WME PLN DEBUG] Original: "${originalName}", Procesado palabra a palabra (joined): "${joinedSuggested}", Sugerido Final (antes de skip): "${suggestedName}"`);

            // 7. --- LÓGICA DE SALTO (SKIP) CONSOLIDADA ---
            const tieneSugerencias = Object.keys(sugerenciasLugar).length > 0;
            let shouldSkipThisPlace = false;
            let skipReasonLog = "";

            if (originalName.trim() === suggestedName.trim())
            {
                shouldSkipThisPlace = true;
                skipReasonLog = `[SKIP EXACT MATCH] Descartado "${
                  originalName}" porque es idéntico al nombre sugerido "${
                  suggestedName}".`;
            }
            // --- INICIO: LÓGICA DE SALTO POR "OMITIR MIS EDICIONES" ---
            else if (chkHideMyEditsChecked && wasEditedByMe)
            {
                shouldSkipThisPlace = true;
                skipReasonLog = `[SKIP MY EDIT] Descartado "${
                  originalName}" (ID: ${
                  currentVenueId}) porque fue editado por mí y la opción "Omitir mis ediciones" está activa.`;
            }
            // --- FIN: LÓGICA DE SALTO POR "OMITIR MIS EDICIONES" ---
            else
            {
                let tempOriginalNormalized = aplicarReemplazosGenerales(originalName.trim());
                tempOriginalNormalized = aplicarReglasEspecialesNombre(tempOriginalNormalized);
                tempOriginalNormalized = postProcessQuotesAndParentheses(tempOriginalNormalized);
                if (tempOriginalNormalized.endsWith('.')) {
                    tempOriginalNormalized = tempOriginalNormalized.slice(0, -1);
                }
                tempOriginalNormalized = tempOriginalNormalized.replace(/\s{2,}/g, ' ').trim();

                if (tempOriginalNormalized.toLowerCase() === suggestedName.toLowerCase() && !tieneSugerencias)
                {
                    shouldSkipThisPlace = true;
                    skipReasonLog = `[SKIP NORMALIZED] Se descartó "${
                      originalName}" (normalizado a "${tempOriginalNormalized}") porque ya coincide con el sugerido "${
                      suggestedName}" y no hay otras sugerencias.`;
                }
            }
            // ---- FIN LÓGICA DE SALTO ---

            // 8. Registrar o no en la lista de inconsistentes
            if (shouldSkipThisPlace)
            {
                if (skipReasonLog)
                    console.log(skipReasonLog);
            }
            else
            {
                if (processingStepLabel)
                {
                    processingStepLabel.textContent =
                      "Registrando lugar(es) con inconsistencias...";
                }

                let categoryNameToStore = "Sin categoría (Error)";
                if (venueFromOldModel)
                {
                    let venueSDKObjectForCategory =
                      null; // Necesitaríamos obtener el venueSDKObject de nuevo
                            // aquí si no lo guardamos o pasarlo si lo guardamos
                            // antes. Para esta iteración, si el SDK dio el venue
                            // antes, podríamos reusarlo, pero por simplicidad,
                            // getPlaceCategoryName puede intentar obtenerlo si le
                            // pasamos el ID, o solo usar W.model por ahora.
                            // simplificar y solo pasar venueFromOldModel,
                            // getPlaceCategoryName usará W.model.
                            // use el SDK para categoría, getPlaceCategoryName debe
                            // modificarse para aceptar ID y hacer la llamada async
                            // al SDK.
                    try
                    {
                        // Pasamos el venue del W.model; getPlaceCategoryName
                        // intentará usar SDK si wmeSDK está disponible y se le pasa
                        // el venueSDK Para esto, necesitaríamos haber guardado el
                        // venueSDK que obtuvimos arriba. Por ahora,
                        // getPlaceCategoryName ya intenta usar un
                        // venueSDKObject si se le pasa. Lo obtendremos de nuevo
                        // aquí para la categoría para mantenerlo encapsulado en
                        // esta prueba.
                        let venueSDKForCategory = null;
                        if (wmeSDK && wmeSDK.DataModel && wmeSDK.DataModel.Venues &&
                            wmeSDK.DataModel.Venues.getById)
                        {
                            try
                            {
                                venueSDKForCategory =
                                  await wmeSDK.DataModel.Venues.getById(
                                    { venueId : currentVenueId });
                            }
                            catch (catSDKError)
                            {
                                console.error("[SDK CATEGORY] Error obteniendo venueSDK para categoría:", catSDKError);
                            }
                        }
                        categoryNameToStore = getPlaceCategoryName(
                          venueFromOldModel, venueSDKForCategory);
                    }
                    catch (e)
                    {
                        console.error("Error llamando a getPlaceCategoryName desde processNextPlace:", e);
                    }
                }

                inconsistents.push({
                    id : currentVenueId,
                    original : originalName, // Mantener el original real
                    normalized : suggestedName, // El sugerido ya procesado y sin punto final
                    category : categoryNameToStore,
                    editor : resolvedEditorName
                });
                sugerenciasPorPalabra[currentVenueId] = sugerenciasLugar;
            }

            // 9. Finalizar procesamiento del 'place' actual y pasar al siguiente
            updateScanProgressBar(index, places.length);
            index++;
            setTimeout(() => processNextPlace(), 0);
        } // ---- FIN DE LA FUNCIÓN processNextPlace ----

        /*   console.log("[DEBUG] ¿excludedArray es array?:",
                       Array.isArray(excludedArray));
           console.log("[DEBUG] ¿processNextPlace está definido?:",
                       typeof processNextPlace === "function");
           console.log("[DEBUG] Justo antes de llamar a processNextPlace");*/
        // Inicializar procesamiento incremental
        try
        {
            // console.log("[DEBUG] Justo antes de llamar a processNextPlace");
            setTimeout(() => { processNextPlace(); }, 10);
        }
        catch (error)
        {
            console.error("[ERROR] Fallo en processNextPlace:", error);
        }
        // Fallback automático si finalizeRender no se ejecuta
        scanFallbackTimeoutId = setTimeout(() => {
            if (!document.querySelector("#wme-place-inspector-output")?.innerHTML)
            {
               /* console.warn(
                  "[WME PLN] Fallback activado: finalizeRender no se llamó en 30s. Mostrando panel flotante vacío.");
                console.error(
                  "[WME PLN] Error: finalizeRender no se completó. Se invocó fallback tras 30 segundos de espera.");*/
                createFloatingPanel("results", 0); // <--- CAMBIO AQUÍ: Llamar con "results" y 0 inconsistents si hay fallback
                const outputFallback =
                  document.querySelector("#wme-place-inspector-output");
                if (outputFallback) // Re-seleccionar por si acaso
                {
                    outputFallback.innerHTML =
                      "El proceso tomó demasiado tiempo o se interrumpió. No se completó el análisis.";
                }
            }
        }, 30000);


        function reapplyExcludedWordsLogic(text, excludedWordsSet)
        {
            if (typeof text !== 'string' || !excludedWordsSet || excludedWordsSet.size === 0)
            {
                return text;
            }

            const wordsInText = text.split(/\s+/);
            const processedWordsArray = wordsInText.map(word => {
                if (word === "") return "";

                const wordWithoutDiacriticsLower = removeDiacritics(word.toLowerCase());

                // Encontrar la palabra excluida que coincida (insensible a may/min y diacríticos)
                const matchingExcludedWord = Array.from(excludedWordsSet).find(
                    w_excluded => removeDiacritics(w_excluded.toLowerCase()) === wordWithoutDiacriticsLower
                );

                if (matchingExcludedWord)
                {
                    // Si coincide, DEVOLVER LA FORMA EXACTA DE LA LISTA DE EXCLUIDAS
                    return matchingExcludedWord;
                }
                // Si no, devolver la palabra como estaba (ya normalizada por pasos previos)
                return word;
            });
            return processedWordsArray.join(' ');
        }


        // Mostrar el panel flotante solo al terminar el procesamiento
        // (mover esta llamada al final del procesamiento)

        // --- Función para finalizar renderizado una vez completado el análisis ---
        function finalizeRender(inconsistents, placesArr)
        {
            // alert("🟢 Entrando a finalizeRender"); // ← punto 6
            //  Log de depuración al entrar a finalizeRender

            /*  console.log("[WME PLN] Finalizando render. Inconsistentes
               encontrados:", inconsistents.length);*/
            // Limpiar el mensaje de procesamiento y spinner al finalizar el
            // análisis
            // Detener animación de puntos suspensivos si existe
            if (window.processingDotsInterval)
            {
                clearInterval(window.processingDotsInterval);
                window.processingDotsInterval = null;
            }
            // Refuerza el restablecimiento del botón de escaneo al entrar
            const scanBtn = document.querySelector("button[type='button']");
            if (scanBtn)
            {
                scanBtn.textContent = "Start Scan...";
                scanBtn.disabled = false;
                scanBtn.style.opacity = "1";
                scanBtn.style.cursor = "pointer";
            }
            const output = document.querySelector("#wme-place-inspector-output");
            if (!output)
            {
                console.error("❌ No se pudo montar el panel flotante. Revisar estructura del DOM.");
                alert("Hubo un problema al mostrar los resultados. Intenta recargar la página.");
                return;
            }
            // Esta llamada se hace ANTES de limpiar el output.
            // El primer argumento es el estado, el segundo es el número de inconsistencias.
            createFloatingPanel("results", inconsistents.length); // <--- CAMBIO AQUÍ: Llamar con "results"

            if (output) // Ya no necesitamos output.innerHTML = ""; porque createFloatingPanel lo hace
            {
                // output.innerHTML = ""; // Limpiar el mensaje de procesamiento y spinner // ESTA LÍNEA SE ELIMINA O COMENTA
                // Mostrar el panel flotante al terminar el procesamiento
                // createFloatingPanel(inconsistents.length); // ESTA LÍNEA SE ELIMINA O COMENTA, YA SE HIZO ARRIBA
            }
            // Limitar a 30 resultados y mostrar advertencia si excede
            const maxRenderLimit = 30;
            const totalInconsistents = inconsistents.length;
            if (totalInconsistents > maxRenderLimit)
            {
                inconsistents = inconsistents.slice(0, maxRenderLimit);

                if (!sessionStorage.getItem("popupShown"))
                {
                    const modalLimit = document.createElement("div"); // Renombrado a modalLimit para claridad
                    modalLimit.style.position = "fixed";
                    modalLimit.style.top = "50%";
                    modalLimit.style.left = "50%";
                    modalLimit.style.transform = "translate(-50%, -50%)";
                    modalLimit.style.background = "#fff";
                    modalLimit.style.border = "1px solid #ccc";
                    modalLimit.style.padding = "20px";
                    modalLimit.style.zIndex = "10007"; // <<<<<<< Z-INDEX AUMENTADO
                    modalLimit.style.width = "400px";
                    modalLimit.style.boxShadow = "0 0 15px rgba(0,0,0,0.3)";
                    modalLimit.style.borderRadius = "8px";
                    modalLimit.style.fontFamily = "sans-serif";

                    // Fondo suave azul y mejor presentación
                    modalLimit.style.backgroundColor = "#f0f8ff";
                    modalLimit.style.border = "1px solid #aad";
                    modalLimit.style.boxShadow = "0 0 10px rgba(0, 123, 255, 0.2)";

                    // --- Insertar ícono visual de información arriba del mensaje ---
                    const iconInfo = document.createElement("div"); // Renombrado
                    iconInfo.innerHTML = "ℹ️";
                    iconInfo.style.fontSize = "24px";
                    iconInfo.style.marginBottom = "10px";
                    modalLimit.appendChild(iconInfo);

                    const message = document.createElement("p");
                    message.innerHTML = `Se encontraron <strong>${
                      totalInconsistents}</strong> lugares con nombres no normalizados.<br><br>Solo se mostrarán los primeros <strong>${
                      maxRenderLimit}</strong>.<br><br>Una vez corrijas estos, presiona nuevamente <strong>'Start Scan...'</strong> para continuar con el análisis del resto.`;
                    message.style.marginBottom = "20px";
                    modalLimit.appendChild(message);

                    const acceptBtn = document.createElement("button");
                    acceptBtn.textContent = "Aceptar";
                    acceptBtn.style.padding = "6px 12px";
                    acceptBtn.style.cursor = "pointer";
                    acceptBtn.style.backgroundColor = "#007bff";
                    acceptBtn.style.color = "#fff";
                    acceptBtn.style.border = "none";
                    acceptBtn.style.borderRadius = "4px";
                    acceptBtn.addEventListener("click", () => {
                        sessionStorage.setItem("popupShown", "true");
                        modalLimit.remove();
                    });

                    modalLimit.appendChild(acceptBtn);
                    document.body.appendChild(modalLimit); // Se añade al body, así que el z-index debería funcionar globalmente
                }
            }
            // Si no hay inconsistencias, mostrar mensaje y salir (progreso visible)
            if (inconsistents.length === 0)
            {
                output.appendChild(document.createTextNode(
                  "Todos los nombres de lugares visibles están correctamente normalizados."));
                // Mensaje visual de análisis finalizado sin inconsistencias
                const checkIcon = document.createElement("div");
                checkIcon.innerHTML = "✔ Análisis finalizado sin inconsistencias.";
                checkIcon.style.marginTop = "10px";
                checkIcon.style.fontSize = "14px";
                checkIcon.style.color = "green";
                output.appendChild(checkIcon);
                // Mensaje visual adicional solicitado
                const successMsg = document.createElement("div");
                successMsg.textContent =
                  "Todos los nombres están correctamente normalizados.";
                successMsg.style.marginTop = "10px";
                successMsg.style.fontSize = "14px";
                successMsg.style.color = "green";
                successMsg.style.fontWeight = "bold";
                output.appendChild(successMsg);
                const existingOverlay =
                  document.getElementById("scanSpinnerOverlay");
                if (existingOverlay)
                    existingOverlay.remove();
                // Actualizar barra de progreso 100%
                const progressBarInnerTab =
                  document.getElementById("progressBarInnerTab");
                const progressBarTextTab =
                  document.getElementById("progressBarTextTab");
                if (progressBarInnerTab && progressBarTextTab)
                {
                    progressBarInnerTab.style.width = "100%";
                    progressBarTextTab.textContent =
                      `Progreso: 100% (${placesArr.length}/${placesArr.length})`;
                }
                // Mensaje adicional en el tab principal (pestaña)
                const outputTab =
                  document.getElementById("wme-normalization-tab-output");
                if (outputTab)
                {
                    outputTab.innerHTML =
                      `✔ Todos los nombres están normalizados. Se analizaron ${
                        placesArr.length} lugares.`;
                    outputTab.style.color = "green";
                    outputTab.style.fontWeight = "bold";
                }
                // Restaurar el texto y estado del botón de escaneo
                const scanBtn = document.querySelector("button[type='button']");
                if (scanBtn)
                {
                    scanBtn.textContent = "Start Scan...";
                    scanBtn.disabled = false;
                    scanBtn.style.opacity = "1";
                    scanBtn.style.cursor = "pointer";
                    // Agregar check verde al lado del botón al finalizar sin
                    // errores
                    const iconCheck = document.createElement("span");
                    iconCheck.textContent = " ✔";
                    iconCheck.style.marginLeft = "8px";
                    iconCheck.style.color = "green";
                    scanBtn.appendChild(iconCheck);
                }
                return;
            }

            // Mostrar spinner solo si hay inconsistencias a procesar
            // showLoadingSpinner();

            const table = document.createElement("table");
            table.style.width = "100%";
            table.style.borderCollapse = "collapse";
            table.style.fontSize = "12px";

            const thead = document.createElement("thead");
            const headerRow = document.createElement("tr");
            [
                "Permalink",
                "Tipo",
                "Categoría",
                "Editor",
                "Nombre Actual",
                "Nombre Sugerido",
                "Sugerencias de reemplazo",
                "Acción"
            ].forEach(header => {
                const th = document.createElement("th");
                th.textContent = header;
                th.style.borderBottom = "1px solid #ccc";
                th.style.padding = "4px";
                th.style.textAlign = "left";
                headerRow.appendChild(th);
            });

            thead.style.position = "sticky";
            thead.style.top = "0";
            thead.style.background = "#f1f1f1";
            thead.style.zIndex = "10"; // z-index de la cabecera de la tabla
            headerRow.style.backgroundColor = "#003366";
            headerRow.style.color = "#ffffff";

            thead.appendChild(headerRow);
            table.appendChild(thead);

            const tbody = document.createElement("tbody");

            inconsistents.forEach(({ id, original, normalized, category, editor },
                                   index) => {
                // Actualizar barra de progreso visual EN EL TAB PRINCIPAL
                const progressPercent =
                  Math.floor(((index + 1) / inconsistents.length) * 100);
                // Actualiza barra de progreso en el tab principal
                const progressBarInnerTab =
                  document.getElementById("progressBarInnerTab");
                const progressBarTextTab =
                  document.getElementById("progressBarTextTab");
                if (progressBarInnerTab && progressBarTextTab)
                {
                    progressBarInnerTab.style.width = `${progressPercent}%`;
                    progressBarTextTab.textContent = `Progreso: ${
                      progressPercent}% (${index + 1}/${inconsistents.length})`;
                }
                const row = document.createElement("tr");

                const permalinkCell = document.createElement("td");
                const link = document.createElement("a");
                link.href = "#";
                // Reemplazado onclick por addEventListener para mejor
                // compatibilidad y centrado de mapa
                link.addEventListener("click", (e) => {
                    e.preventDefault();
                    const venue = W.model.venues.getObjectById(id);
                    if (!venue)
                        return;

                    // Centrar mapa y seleccionar el lugar
                    const geometry = venue.getGeometry();
                    if (geometry && geometry.getCentroid)
                    {
                        const center = geometry.getCentroid();
                        W.map.setCenter(center, null, false, 0);
                    }

                    if (W.selectionManager &&
                        typeof W.selectionManager.select === "function")
                    {
                        W.selectionManager.select(venue);
                    }
                    else if (W.selectionManager &&
                             typeof W.selectionManager.setSelectedModels ===
                               "function")
                    {
                        W.selectionManager.setSelectedModels([ venue ]);
                    }
                });
                link.title = "Abrir en panel lateral";
                link.textContent = "🔗";
                permalinkCell.appendChild(link);
                permalinkCell.style.padding = "4px";
                permalinkCell.style.width = "65px";
                row.appendChild(permalinkCell);

                // Columna Tipo de place
                const venue = W.model.venues.getObjectById(id);
                const { icon : typeIcon, title : typeTitle } =
                  getPlaceTypeInfo(venue);
                const typeCell = document.createElement("td");
                typeCell.textContent = typeIcon;
                typeCell.title = `Lugar tipo ${typeTitle}`;
                typeCell.style.padding = "4px";
                typeCell.style.width = "65px";
                row.appendChild(typeCell);

                // Columna Categoría del place
                const categoryCell = document.createElement("td");
                const categoryName = getPlaceCategoryName(venue);
                categoryCell.textContent = categoryName;
                categoryCell.title = `Categoría: ${categoryName}`;
                categoryCell.style.padding = "4px";
                categoryCell.style.width = "130px";
                row.appendChild(categoryCell);


                // Columna Editor (username)
                const editorCell = document.createElement("td");
                editorCell.textContent =
                  editor || "Desconocido"; // Use the stored editor name
                editorCell.title = "Último editor";
                editorCell.style.padding = "4px";
                editorCell.style.width = "140px";
                row.appendChild(editorCell);
                const originalCell = document.createElement("td");
                const inputOriginal = document.createElement("input");
                inputOriginal.type = "text";
                const venueLive = W.model.venues.getObjectById(id);
                const currentLiveName = venueLive?.attributes?.name?.value ||
                                        venueLive?.attributes?.name || "";
                inputOriginal.value = currentLiveName;
                // --- Resaltar en rojo si hay diferencia con el sugerido ---
                if (currentLiveName.trim().toLowerCase() !==
                    normalized.trim().toLowerCase())
                {
                    inputOriginal.style.border = "1px solid red";
                    inputOriginal.title =
                      "Este nombre difiere del original mostrado en el panel";
                }
                inputOriginal.disabled = true;
                inputOriginal.style.width = "270px";
                inputOriginal.style.backgroundColor = "#eee";
                originalCell.appendChild(inputOriginal);
                originalCell.style.padding = "4px";
                originalCell.style.width = "270px";
                row.appendChild(originalCell);

                const suggestionCell = document.createElement("td");
                // Nueva columna: sugerencia de reemplazo seleccionada
                const suggestionListCell = document.createElement("td");
                suggestionListCell.style.padding = "4px";
                suggestionListCell.style.fontSize = "11px";
                suggestionListCell.style.color = "#333";
                // Permitir múltiples líneas en la celda de sugerencias
                suggestionListCell.style.whiteSpace = "pre-wrap";
                suggestionListCell.style.wordBreak = "break-word";
                suggestionListCell.style.width = "270px";
                // Calcular sugerencias similares de especiales aquí
                // --- Nueva lógica para separar sugerencias por fuente ---
                const allSuggestions = sugerenciasPorPalabra?.[id] || {};
                const similarList = {};
                const similarDictList = {};
                Object.entries(allSuggestions)
                  .forEach(([ originalWord, suggestions ]) => {
                      suggestions.forEach(s => {
                          if (s.fuente === 'excluded')
                          {
                              if (!similarList[originalWord])
                                  similarList[originalWord] = [];
                              similarList[originalWord].push(s);
                          }
                          else if (s.fuente === 'dictionary')
                          {
                              if (!similarDictList[originalWord])
                                  similarDictList[originalWord] = [];
                              similarDictList[originalWord].push(s);
                          }
                      });
                  });

                // --- NUEVA LÓGICA: aplicar reemplazos automáticos y solo mostrar
                // sugerencias < 1 ---
                let autoApplied = false;
                let localNormalized = normalized;
                // 1. Procesar sugerencias de especiales (similarList)
                let hasExcludedSuggestionsToShow = false;
                if (similarList && Object.keys(similarList).length > 0)
                {
                    Object.entries(similarList)
                      .forEach(([ originalWord, suggestions ]) => {
                          suggestions.forEach(s => {
                              if (s.similarity === 1)
                              {
                                  // Si encontramos una sugerencia 100%, ya fue
                                  // aplicada en el pipeline principal.
                                  autoApplied = true;
                                  // NO mostrar sugerencia clickable ni texto en
                                  // columna de sugerencias.
                              }
                              else if (s.similarity < 1)
                              {
                                  hasExcludedSuggestionsToShow = true;
                              }
                          });
                      });
                }
                // 2. Procesar sugerencias de diccionario (similarDictList)
                let hasDictSuggestionsToShow = false;
                if (similarDictList && Object.keys(similarDictList).length > 0)
                {
                    Object.entries(similarDictList)
                      .forEach(([ originalWord, suggestions ]) => {
                          suggestions.forEach(s => {
                              if (s.similarity < 1)
                              {
                                  hasDictSuggestionsToShow = true;
                              }
                          });
                      });
                }

                // --- EVITAR DUPLICADOS DE SUGERENCIAS ENTRE "ESPECIALES" Y
                // "DICCIONARIO" --- Crear set de palabras ya procesadas en
                // sugerencias especiales
                const palabrasYaProcesadas = new Set();
                if (similarList && Object.keys(similarList).length > 0)
                {
                    Object.keys(similarList)
                      .forEach(palabra =>
                                 palabrasYaProcesadas.add(palabra.toLowerCase()));
                }

                // Render input de sugerencia
                const inputReplacement = document.createElement("input");
                inputReplacement.type = "text";
                let mainSuggestion = localNormalized;
                inputReplacement.value = mainSuggestion;
                inputReplacement.style.width = "270px";
                // Visual cue if change was due to excluded word
                if (localNormalized !== normalized)
                {
                    inputReplacement.style.backgroundColor =
                      "#fff3cd"; // color amarillo claro
                    inputReplacement.title = "Contiene palabra excluida reemplazada";
                }
                else if (autoApplied)
                {
                    inputReplacement.style.backgroundColor =
                      "#c8e6c9"; // verde claro
                    inputReplacement.title =
                      "Reemplazo automático aplicado (100% similitud)";
                }
                else if (normalized !== inputReplacement.value)
                {
                    inputReplacement.style.backgroundColor =
                      "#e6f7ff"; // Azul claro para cambios automáticos del
                                 // diccionario (≥ 90%)
                    inputReplacement.title =
                      "Cambio automático basado en diccionario (≥ 90%)";
                }
                else
                {
                    inputReplacement.title = "Nombre normalizado";
                }
                // NUEVA LÓGICA: marcar si contiene palabra sugerida por el
                // diccionario
                const palabrasDelDiccionario = new Set();
                Object.values(similarDictList).forEach(arr => {
                    arr.forEach(s => {
                        if (mainSuggestion.toLowerCase().includes(
                              s.word.toLowerCase()))
                        {
                            palabrasDelDiccionario.add(s.word.toLowerCase());
                        }
                    });
                });
                if (palabrasDelDiccionario.size > 0)
                {
                    inputReplacement.style.backgroundColor = "#cce5ff"; // Azul claro
                    inputReplacement.title =
                      "Contiene sugerencias del diccionario aplicadas manualmente";
                }
                suggestionCell.appendChild(inputReplacement);
                suggestionCell.style.padding = "4px";
                suggestionCell.style.width = "270px";

                // --- Función debounce ---
                function debounce(func, delay) {
                    let timeout;
                    return function (...args) {
                        clearTimeout(timeout);
                        timeout = setTimeout(() => func.apply(this, args), delay);
                    };
                }

                // --- Activar/desactivar el botón Aplicar según si hay cambios ---
                inputReplacement.addEventListener('input', debounce(() => {
                    if (inputReplacement.value.trim() !== original)
                    {
                        applyButton.disabled = false;
                        applyButton.style.color = "";
                    }
                    else
                    {
                        applyButton.disabled = true;
                        applyButton.style.color = "#bbb";
                    }
                }, 300));

                // --- Listener para inputOriginal con debounce (puede personalizarse la lógica) ---
                inputOriginal.addEventListener('input', debounce(() => {
                    // Opcional: alguna lógica si se desea manejar cambios en inputOriginal
                }, 300));

                // Renderizar solo sugerencias < 1 en sugerencias de reemplazo
                // especiales primero
                if (similarList && Object.keys(similarList).length > 0) {
                    Object.entries(similarList).forEach(([originalWord, suggestions]) => {
                        suggestions.forEach(s => { // 's' aquí es { word: candidate, similarity: sim, fuente: 'excluded' }
                            // Mostrar todas las sugerencias < 100% de similitud,
                            // o incluso las de 100% si la forma en la lista de excluidas es diferente (ej. mayúsculas/minúsculas)
                            // a la originalWord.
                            // La condición original s.similarity < 1 es buena para evitar sugerir lo mismo que ya está.
                            if (s.similarity < 1 || (s.similarity === 1 && originalWord !== s.word) ) {
                                const suggestionDiv = document.createElement("div");
                                const icono = "🏷️"; // Icono para palabras especiales/excluidas
                                suggestionDiv.textContent =
                                  `${icono} ¿"${originalWord}" por "${s.word}"? (simil. ${(s.similarity * 100).toFixed(0)}%)`;
                                suggestionDiv.style.cursor = "pointer";
                                suggestionDiv.style.padding = "2px 4px";
                                suggestionDiv.style.margin = "2px 0";
                                suggestionDiv.style.border = "1px solid #ddd";
                                suggestionDiv.style.backgroundColor = "#f3f9ff"; // Color distintivo si quieres

                                suggestionDiv.addEventListener("click", () => {
                                    const currentSuggestedValue = inputReplacement.value; // Valor actual del campo "Nombre Sugerido"

                                   // console.log("--- DEBUG: Clic en Sugerencia ESPECIAL ---");
                                   // console.log("Palabra Original (base de la sugerencia):", originalWord);
                                   // console.log("Sugerencia (palabra a usar de lista Especiales):", s.word);

                                    // 1. Normalizamos la 'originalWord' para saber qué forma buscar
                                    const normalizedOriginalWord = normalizePlaceName(originalWord);
                                    //console.log("Forma Normalizada de Palabra Original (para buscar):", normalizedOriginalWord);

                                    // 2. Normalizamos la palabra sugerida 's.word' de la lista de Especiales
                                    const wordToInsert = normalizePlaceName(s.word);
                                    //console.log("Palabra Normalizada para Insertar:", wordToInsert);

                                    //console.log("Valor ACTUAL del campo 'Nombre Sugerido':", currentSuggestedValue);

                                    // 3. Creamos la regex para buscar la 'normalizedOriginalWord'
                                    const searchRegex = new RegExp("\\b" + escapeRegExp(normalizedOriginalWord) + "\\b", "gi");
                                    //console.log("Regex para buscar en 'Nombre Sugerido':", searchRegex.toString());

                                    if (!searchRegex.test(currentSuggestedValue)) {
                                        console.warn("¡ADVERTENCIA ESPECIAL! La forma normalizada de la palabra original ('" + normalizedOriginalWord + "') no se encontró en el 'Nombre Sugerido' actual ('" + currentSuggestedValue + "'). No se hará reemplazo.");
                                    }

                                    const newSuggestedValue = currentSuggestedValue.replace(searchRegex, wordToInsert);
                                    //console.log("Valor DESPUÉS de .replace() (Especial):", newSuggestedValue);

                                    if (currentSuggestedValue !== newSuggestedValue)
                                    {
                                        inputReplacement.value = newSuggestedValue;
                                       // console.log("Campo 'Nombre Sugerido' ACTUALIZADO (Especial) a:", newSuggestedValue);
                                    }
                                    else
                                    {
                                        console.log("No hubo cambios en 'Nombre Sugerido' (Especial) (el nuevo valor es idéntico al anterior o la palabra a reemplazar no se encontró/ya era igual).");
                                    }
                                    inputReplacement.dispatchEvent(new Event("input"));
                                   // console.log("--- FIN DEBUG (Especial) ---");
                                });
                                suggestionListCell.appendChild(suggestionDiv);
                            }
                        });
                    });
                }
                // Diccionario después, evitando duplicados de palabras ya sugeridas
                // en especiales
                if (similarDictList && Object.keys(similarDictList).length > 0)
                {
                    Object.entries(similarDictList)
                      .forEach(([ originalWord, suggestions ]) => { // originalWord es la palabra del nombre del lugar ANTES de normalizePlaceName (la que fue clave en sugerenciasLugar)
                          if (palabrasYaProcesadas.has(originalWord.toLowerCase()))
                              return;
                          suggestions.forEach(s => { // s es { word: diccWordDelDiccionario, similarity: sim, fuente: 'dictionary' }

                              const normalizedOriginalWordForDisplay = normalizePlaceName(originalWord);
                              const normalizedDictionaryWordToApply = normalizePlaceName(s.word);

                              // Condición A: Si la aplicación de la sugerencia del diccionario
                              // resulta en el nombre COMPLETO que ya está sugerido (localNormalized), Y la palabra original
                              // (normalizada) también era igual a ese nombre completo, entonces la palabra es única y ya está perfecta.
                              // Principalmente para nombres de una sola palabra.
                              if (normalizedDictionaryWordToApply === localNormalized && normalizedOriginalWordForDisplay === localNormalized) {
                                  return;
                              }

                              // Condición B: Si la palabra original (normalizada para mostrar) es idéntica
                              // a la palabra del diccionario (normalizada para aplicar), entonces la sugerencia
                              // sería del tipo "¿X por X?", lo cual es inútil y no ofrece ningún cambio para esa palabra específica.
                              if (normalizedOriginalWordForDisplay === normalizedDictionaryWordToApply) {
                                  return;
                              }

                              // Si hemos pasado ambas condiciones, significa que la sugerencia ofrece un cambio útil.
                              const suggestionItem = document.createElement("div");
                              const icono = "📘";

                              suggestionItem.textContent = `${icono} ¿"${normalizedOriginalWordForDisplay}" por "${normalizedDictionaryWordToApply}"? (simil. ${(s.similarity * 100).toFixed(0)}%)`;

                              suggestionItem.style.cursor = "pointer";
                              suggestionItem.style.padding = "2px 4px";
                              suggestionItem.style.margin = "2px 0";
                              suggestionItem.style.border = "1px solid #ddd";
                              suggestionItem.style.backgroundColor = "#f9f9f9";

                              suggestionItem.addEventListener("click", () => {
                                  const currentSuggestedValue = inputReplacement.value;

                                  const wordToInsert = normalizedDictionaryWordToApply;
                                  const wordToSearchAndReplace = normalizedOriginalWordForDisplay; // Usar la forma normalizada de la palabra original para la búsqueda

                                  const searchRegex = new RegExp("\\b" + escapeRegExp(wordToSearchAndReplace) + "\\b", "gi");

                                  let newSuggestedValue = currentSuggestedValue;
                                  if (searchRegex.test(currentSuggestedValue)) {
                                      newSuggestedValue = currentSuggestedValue.replace(searchRegex, wordToInsert);
                                  } else {
                                       console.warn(`¡ADVERTENCIA! La palabra a reemplazar ('${wordToSearchAndReplace}') no se encontró en el 'Nombre Sugerido' actual ('${currentSuggestedValue}').`);
                                  }

                                  if (inputReplacement.value !== newSuggestedValue) {
                                      inputReplacement.value = newSuggestedValue;
                                  } else {
                                       console.log("No hubo cambios efectivos en 'Nombre Sugerido' o la palabra a reemplazar no se encontró.");
                                  }
                                  inputReplacement.dispatchEvent(new Event("input"));
                              });
                              suggestionListCell.appendChild(suggestionItem);
                          });
                      });
                }
                row.appendChild(suggestionCell);
                row.appendChild(suggestionListCell);

                const actionCell = document.createElement("td");
                actionCell.style.padding = "4px";
                actionCell.style.width = "120px";

                const buttonGroup = document.createElement("div");
                buttonGroup.style.display = "flex";
                buttonGroup.style.gap = "4px";

                const applyButton = document.createElement("button");
                applyButton.textContent = "✔";
                applyButton.title = "Aplicar sugerencia";
                applyButton.style.padding = "4px 8px";
                applyButton.style.cursor = "pointer";

                const deleteButton = document.createElement("button");
                deleteButton.textContent = "💣";
                deleteButton.title = "Eliminar lugar";
                deleteButton.style.padding = "4px 8px";
                deleteButton.style.cursor = "pointer";

                applyButton.relatedDelete = deleteButton;
                deleteButton.relatedApply = applyButton;

                applyButton.addEventListener("click", () => {
                    const venue = W.model.venues.getObjectById(id);
                    if (!venue)
                    {
                        alert(
                          "Error: El lugar no está disponible o ya fue eliminado.");
                        return;
                    }
                    const newName = inputReplacement.value.trim();
                    try
                    {
                        const UpdateObject = require("Waze/Action/UpdateObject");
                        const action = new UpdateObject(venue, { name : newName });
                        W.model.actionManager.add(action);
                        applyButton.disabled = true;
                        applyButton.style.color = "#bbb";
                        applyButton.style.opacity = "0.5";
                        if (applyButton.relatedDelete)
                        {
                            applyButton.relatedDelete.disabled = true;
                            applyButton.relatedDelete.style.color = "#bbb";
                            applyButton.relatedDelete.style.opacity = "0.5";
                        }
                        const successIcon = document.createElement("span");
                        successIcon.textContent = " ✅";
                        successIcon.style.marginLeft = "5px";
                        applyButton.parentElement.appendChild(successIcon);
                    }
                    catch (e)
                    {
                        alert("Error al actualizar: " + e.message);
                    }
                });
                // Listener para el botón de eliminar
                deleteButton.addEventListener("click", () => {
                // Modal bonito de confirmación
                const confirmModal = document.createElement("div");
                confirmModal.style.position = "fixed";
                confirmModal.style.top = "50%";
                confirmModal.style.left = "50%";
                confirmModal.style.transform = "translate(-50%, -50%)";
                confirmModal.style.background = "#fff";
                confirmModal.style.border = "1px solid #aad";
                confirmModal.style.padding = "28px 32px 20px 32px";
                confirmModal.style.zIndex = "20000"; // Z-INDEX AUMENTADO
                confirmModal.style.boxShadow = "0 4px 24px rgba(0,0,0,0.18)";
                confirmModal.style.fontFamily = "sans-serif";
                confirmModal.style.borderRadius = "10px";
                confirmModal.style.textAlign = "center";
                confirmModal.style.minWidth = "340px";

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

                // Mensaje principal
                const message = document.createElement("div");
                const venue = W.model.venues.getObjectById(id);
                const placeName = venue?.attributes?.name?.value || venue?.attributes?.name || "este lugar";
                message.innerHTML = `<b>¿Eliminar "${placeName}"?</b>`; // CORREGIDO para mostrar el nombre del lugar
                message.style.fontSize = "18px";
                message.style.marginBottom = "8px";
                confirmModal.appendChild(message);

                // Nombre del lugar (puede ser redundante si ya está en el mensaje, pero se mantiene por si acaso)
                const nameDiv = document.createElement("div");
                nameDiv.textContent = `"${placeName}"`;
                nameDiv.style.fontSize = "15px";
                nameDiv.style.color = "#007bff";
                nameDiv.style.marginBottom = "18px";
                confirmModal.appendChild(nameDiv);

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

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

                const confirmBtn = document.createElement("button");
                confirmBtn.textContent = "Eliminar";
                confirmBtn.style.padding = "7px 18px";
                confirmBtn.style.background = "#d9534f";
                confirmBtn.style.color = "#fff";
                confirmBtn.style.border = "none";
                confirmBtn.style.borderRadius = "4px";
                confirmBtn.style.cursor = "pointer";
                confirmBtn.style.fontWeight = "bold";
                confirmBtn.addEventListener("click", () => {
                    const venue = W.model.venues.getObjectById(id);
                    if (!venue) {
                        alert("El lugar no está disponible o ya fue eliminado.");
                        confirmModal.remove();
                        return;
                    }
                    try {
                        const DeleteObject = require("Waze/Action/DeleteObject");
                        const action = new DeleteObject(venue);
                        W.model.actionManager.add(action);
                        deleteButton.disabled = true;
                        deleteButton.style.color = "#bbb";
                        deleteButton.style.opacity = "0.5";
                        if (deleteButton.relatedApply) {
                            deleteButton.relatedApply.disabled = true;
                            deleteButton.relatedApply.style.color = "#bbb";
                            deleteButton.relatedApply.style.opacity = "0.5";
                        }
                        const successIcon = document.createElement("span");
                        successIcon.textContent = " 🗑️";
                        successIcon.style.marginLeft = "5px";
                        deleteButton.parentElement.appendChild(successIcon);
                    } catch (e) {
                        alert("Error al eliminar: " + e.message);
                    }
                    confirmModal.remove();
                });

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

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

                buttonGroup.appendChild(applyButton);
                buttonGroup.appendChild(deleteButton);

                const addToExclusionBtn = document.createElement("button");
                addToExclusionBtn.textContent = "🏷️";
                addToExclusionBtn.title =
                  "Marcar palabra como especial (no se modifica)";
                addToExclusionBtn.style.padding = "4px 6px";
                addToExclusionBtn.addEventListener("click", () => {
                    const words = original.split(/\s+/);
                    const modal = document.createElement("div");
                    modal.style.position = "fixed";
                    modal.style.top = "50%";
                    modal.style.left = "50%";
                    modal.style.transform = "translate(-50%, -50%)";
                    modal.style.background = "#fff";
                    modal.style.border = "1px solid #ccc";
                    modal.style.padding = "10px";
                    modal.style.zIndex = "20000"; // Z-INDEX AUMENTADO
                    modal.style.maxWidth = "300px";

                    const title = document.createElement("h4");
                    title.textContent = "Agregar palabra a especiales";
                    modal.appendChild(title);

                    const list = document.createElement("ul");
                    list.style.listStyle = "none";
                    list.style.padding = "0";
                    words.forEach(w => {
                        // --- Filtro: palabras vacías, comunes, o ya existentes
                        // (ignorar mayúsculas) ---
                        if (w.trim() === '')
                            return;
                        const lowerW = w.trim().toLowerCase();
                       /* const commonWords = [
                            'de',  'del',   'el',   'la', 'los', 'las',
                            'y',   'e',     'o',    'u',  'un',  'una',
                            'y',   'unos',  'unas', 'a',  'en',  'con',
                            'sin', 'sobre', 'tras', 'por'
                        ];*/
                        const alreadyExists =
                          Array.from(excludedWords)
                            .some(existing => existing.toLowerCase() === lowerW);
                        if (commonWords.includes(lowerW) || alreadyExists)
                            return;
                        const li = document.createElement("li");
                        const checkbox = document.createElement("input");
                        checkbox.type = "checkbox";
                        checkbox.value = w;
                        checkbox.id = `cb-exc-${w.replace(/[^a-zA-Z0-9]/g, "")}`;
                        li.appendChild(checkbox);
                        const label = document.createElement("label");
                        label.htmlFor = checkbox.id;
                        label.appendChild(document.createTextNode(" " + w));
                        li.appendChild(label);
                        list.appendChild(li);
                    });
                    modal.appendChild(list);

                    const confirmBtn = document.createElement("button");
                    confirmBtn.textContent = "Añadir Seleccionadas";
                    confirmBtn.addEventListener("click", () => {
                        const checked =  modal.querySelectorAll("input[type=checkbox]:checked");
                        let wordsActuallyAdded = false; // Para saber si se añadió algo nuevo

                        checked.forEach(c => {
                            // Antes de añadir, podrías verificar si ya existe para evitar
                            // trabajo innecesario, aunque un Set maneja duplicados.
                            // También, considera si quieres aplicar isValidExcludedWord aquí,
                            // aunque usualmente las palabras del nombre del lugar son candidatas directas.
                            if (!excludedWords.has(c.value)) {
                                excludedWords.add(c.value);
                                wordsActuallyAdded = true;
                            }
                        });

                        if (wordsActuallyAdded) {
                            // Llama a renderExcludedWordsList para actualizar la UI en la pestaña "Especiales"
                            // y para guardar en localStorage
                            if (typeof renderExcludedWordsList === 'function') {
                                // Es mejor pasar el elemento si se puede, o dejar que la función lo encuentre
                                const excludedListElement = document.getElementById("excludedWordsList");
                                if (excludedListElement) {
                                    renderExcludedWordsList(excludedListElement);
                                } else {
                                    renderExcludedWordsList(); // Fallback
                                }
                            }
                        }


                        modal.remove();
                    });
                    modal.appendChild(confirmBtn);

                    const cancelBtn = document.createElement("button");
                    cancelBtn.textContent = "Cancelar";
                    cancelBtn.style.marginLeft = "8px";
                    cancelBtn.addEventListener("click", () => modal.remove());
                    modal.appendChild(cancelBtn);

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



                buttonGroup.appendChild(addToExclusionBtn);
             //   buttonGroup.appendChild(addToDictionaryBtn);
                actionCell.appendChild(buttonGroup);
                row.appendChild(actionCell);
                // Añadir borde inferior visible entre cada lugar
                row.style.borderBottom = "1px solid #ddd";
                row.style.backgroundColor = index % 2 === 0 ? "#f9f9f9" : "#ffffff";

                tbody.appendChild(row);
                // Actualizar progreso al final del ciclo usando setTimeout para
                // liberar el hilo visual
                setTimeout(() => {
                    const progress =
                      Math.floor(((index + 1) / inconsistents.length) * 100);
                    const progressElem =
                      document.getElementById("scanProgressText");
                    if (progressElem)
                    {
                        progressElem.textContent = `Analizando lugares: ${
                          progress}% (${index + 1}/${inconsistents.length})`;
                    }
                }, 0);
            });

            table.appendChild(tbody);
            output.appendChild(table);

            // Log de cierre
            //   console.log("✔ Panel finalizado y tabla renderizada.");

            // Quitar overlay spinner justo antes de mostrar la tabla
            const existingOverlay = document.getElementById("scanSpinnerOverlay");
            if (existingOverlay)
            {
                existingOverlay.remove();
            }
            // Al finalizar, actualizar el texto final en el tab principal (progreso
            // 100%)
            const progressBarInnerTab =
              document.getElementById("progressBarInnerTab");
            const progressBarTextTab =
              document.getElementById("progressBarTextTab");
            if (progressBarInnerTab && progressBarTextTab)
            {
                progressBarInnerTab.style.width = "100%";
                progressBarTextTab.textContent =
                  `Progreso: 100% (${inconsistents.length}/${placesArr.length})`;
            }
            // Función para reactivar todos los botones de acción en el panel
            // flotante
            function reactivateAllActionButtons()
            {
                document.querySelectorAll("#wme-place-inspector-output button")
                  .forEach(btn => {
                      btn.disabled = false;
                      btn.style.color = "";
                      btn.style.opacity = "";
                  });
            }

            W.model.actionManager.events.register("afterundoaction", null, () => {
                    waitForWazeAPI(() => {
                        const places = getVisiblePlaces();
                    renderPlacesInFloatingPanel(places);
                    setTimeout(reactivateAllActionButtons,
                               250); // Esperar a que el panel se reconstruya
                    });
            });
            W.model.actionManager.events.register("afterredoaction", null, () => {
                    waitForWazeAPI(() => {
                        const places = getVisiblePlaces();
                        renderPlacesInFloatingPanel(places);
                    setTimeout(reactivateAllActionButtons, 250);
                });
            });
            // Mostrar el panel flotante al terminar el procesamiento
            // createFloatingPanel(inconsistents.length); // Ahora se invoca arriba
            // si output existe
        }
    }

    function getLevenshteinDistance(a, b)
    {
        const matrix = Array.from(
          { length : b.length + 1 },
          (_, i) => Array.from({ length : a.length + 1 },
                               (_, j) => (i === 0 ? j : (j === 0 ? i : 0))));

        for (let i = 1; i <= b.length; i++)
        {
            for (let j = 1; j <= a.length; j++)
            {
                if (b.charAt(i - 1) === a.charAt(j - 1))
                {
                    matrix[i][j] = matrix[i - 1][j - 1];
                }
                else
                {
                    matrix[i][j] = Math.min(
                      matrix[i - 1][j] + 1,    // deletion
                      matrix[i][j - 1] + 1,    // insertion
                      matrix[i - 1][j - 1] + 1 // substitution
                    );
                }
            }
        }

        return matrix[b.length][a.length];
    }

    function calculateSimilarity(word1, word2)
    {
        const distance =
          getLevenshteinDistance(word1.toLowerCase(), word2.toLowerCase());
        const maxLen = Math.max(word1.length, word2.length);
        return 1 - distance / maxLen;
    }

    function findSimilarWords(word, excludedWords, threshold)
    {
        const userThreshold =
          parseFloat(document.getElementById("similarityThreshold")?.value ||
                     "85") /
          100;
        const lowerWord = word.toLowerCase();
        // excludedWords is now always an array
        const firstChar = lowerWord.charAt(0);
        let candidates = excludedWords;

        if (typeof excludedWords === 'object' && !Array.isArray(excludedWords))
        {
            // Estamos usando el índice
            candidates = excludedWords[firstChar] || [];
        }

        return candidates
          .map(candidate => {
              const similarity =
                calculateSimilarity(lowerWord, candidate.toLowerCase());
              return { word : candidate, similarity };
          })
          .filter(item => item.similarity >= threshold)
          .sort((a, b) => b.similarity - a.similarity);
    }
    function suggestExcludedReplacements(currentName, excludedWords)
    {
        const words = currentName.split(/\s+/);
        const suggestions = {};
        const threshold =
          parseFloat(document.getElementById("similarityThreshold")?.value ||
                     "85") /
          100;
        words.forEach(word => {
            const similar =
              findSimilarWords(word, Array.from(excludedWords), threshold);
            if (similar.length > 0)
            {
                suggestions[word] = similar;
            }
        });
        return suggestions;
    }

    // Reset del inspector: progreso y texto de tab
        function resetInspectorState()
        {
        const inner = document.getElementById("progressBarInnerTab");
        const text = document.getElementById("progressBarTextTab");
        const outputTab = document.getElementById("wme-normalization-tab-output");
        if (inner)
            inner.style.width = "0%";
        if (text)
            text.textContent = `Progreso: 0% (0/0)`;
        if (outputTab)
            outputTab.textContent = "Presiona 'Start Scan...' para analizar los lugares visibles.";
    }
    ///***************************************************************************
    // Nombre: createFloatingPanel
    // Autor: mincho77
    // Fecha: 2025-05-26 // Actualiza la fecha si es necesario
    // Descripción: Crea el panel flotante con dimensiones y títulos correctos, y ajusta la posición del panel de resultados.
    //***************************************************************************
    function createFloatingPanel(status = "processing", numInconsistents = 0) {
        if (!floatingPanelElement) {
            floatingPanelElement = document.createElement("div");
            floatingPanelElement.id = "wme-place-inspector-panel";
            floatingPanelElement.style.position = "fixed";
            floatingPanelElement.style.zIndex = "10005"; // Z-INDEX DEL PANEL DE RESULTADOS
            floatingPanelElement.style.background = "#fff";
            floatingPanelElement.style.border = "1px solid #ccc";
            floatingPanelElement.style.borderRadius = "8px";
            floatingPanelElement.style.boxShadow = "0 5px 15px rgba(0,0,0,0.2)";
            floatingPanelElement.style.padding = "10px";
            floatingPanelElement.style.fontFamily = "'Helvetica Neue', Helvetica, Arial, sans-serif";
            floatingPanelElement.style.display = 'none';
            floatingPanelElement.style.transition = "width 0.25s, height 0.25s, left 0.25s, top 0.25s"; // Agregado left y top a la transición
            floatingPanelElement.style.overflow = "hidden"; // ESTA LÍNEA

            const closeBtn = document.createElement("span");
            closeBtn.textContent = "×";
            closeBtn.style.position = "absolute";
            closeBtn.style.top = "8px";
            closeBtn.style.right = "12px";
            closeBtn.style.cursor = "pointer";
            closeBtn.style.fontSize = "22px";
            closeBtn.style.color = "#555";
            closeBtn.title = "Cerrar panel";
            closeBtn.addEventListener("click", () => {
                if (floatingPanelElement) floatingPanelElement.style.display = 'none';
                     resetInspectorState();
            });
            floatingPanelElement.appendChild(closeBtn);

            const titleElement = document.createElement("h4");
            titleElement.id = "wme-pln-panel-title";
            titleElement.style.marginTop = "0";
            titleElement.style.marginBottom = "10px";
            titleElement.style.fontSize = "20px";
            titleElement.style.color = "#333";
            titleElement.style.textAlign = "center";
            titleElement.style.fontWeight = "bold";
            floatingPanelElement.appendChild(titleElement);

            const outputDivLocal = document.createElement("div");
            outputDivLocal.id = "wme-place-inspector-output";
            outputDivLocal.style.fontSize = "14px";
            outputDivLocal.style.backgroundColor = "#fdfdfd";
            outputDivLocal.style.overflowY = "auto"; // Y ESTA
            floatingPanelElement.appendChild(outputDivLocal);

            document.body.appendChild(floatingPanelElement);
        }

        const titleElement = floatingPanelElement.querySelector("#wme-pln-panel-title");
        const outputDiv = floatingPanelElement.querySelector("#wme-place-inspector-output");

        if(outputDiv) outputDiv.innerHTML = "";

        if (status === "processing") {
            floatingPanelElement.style.width = processingPanelDimensions.width;
            floatingPanelElement.style.height = processingPanelDimensions.height;
            if(outputDiv) outputDiv.style.height = "150px";
            if(titleElement) titleElement.textContent = "Buscando...";
            if(outputDiv) {
                 outputDiv.innerHTML = "<div style='display:flex; align-items:center; justify-content:center; height:100%;'><span class='loader-spinner' style='width:32px; height:32px; border:4px solid #ccc; border-top:4px solid #007bff; border-radius:50%; animation:spin 0.8s linear infinite;'></span></div>";
            }
            // Centrar el panel de procesamiento
            floatingPanelElement.style.top = "50%";
            floatingPanelElement.style.left = "50%";
            floatingPanelElement.style.transform = "translate(-50%, -50%)";

        } else { // status === "results"
            floatingPanelElement.style.width = resultsPanelDimensions.width;
            floatingPanelElement.style.height = resultsPanelDimensions.height;
            if(outputDiv) outputDiv.style.height = "660px";
            if(titleElement) titleElement.textContent = "Resultado de la búsqueda";

            // Mover el panel de resultados más a la derecha
            floatingPanelElement.style.top = "50%";
            floatingPanelElement.style.left = "60%";
            floatingPanelElement.style.transform = "translate(-50%, -50%)";
        }

        floatingPanelElement.style.display = 'flex';
        floatingPanelElement.style.flexDirection = 'column';
    }
    //***************************************************************************


        // Escuchar el botón Guardar de WME para resetear el inspector
        const wmeSaveBtn = document.querySelector(
        "button.action.save, button[title='Guardar'], button[aria-label='Guardar']");
        if (wmeSaveBtn)
        {
            wmeSaveBtn.addEventListener("click", () => resetInspectorState());
        }
        function createSidebarTab()
        {
            try
            {
                // 1. Verificar si WME y la función para registrar pestañas están listos
                if (!W || !W.userscripts ||
                    typeof W.userscripts.registerSidebarTab !== 'function')
                {
                    console.error("[WME PLN] WME (userscripts o registerSidebarTab) no está listo para crear la pestaña lateral.");
                    return;
                }

                // 2. Registrar la pestaña principal del script en WME y obtener tabPane
                let registration;
                try
                {
                    registration = W.userscripts.registerSidebarTab(
                    "NrmliZer"); // Nombre del Tab que aparece en WME
                }
                catch (e)
                {
                    if (e.message.includes("already been registered"))
                    {
                        console.warn(
                        "[WME PLN] Tab 'NrmliZer' ya registrado. El script puede no funcionar como se espera si hay múltiples instancias.");
                        // Podrías intentar obtener el tabPane existente o simplemente
                        // retornar. Para evitar mayor complejidad, si ya está
                        // registrado, no continuaremos con la creación de la UI de la
                        // pestaña.
                        return;
                    }
                    //console.error("[WME PLN] Error registrando el sidebar tab:", e);
                    throw e; // Relanzar otros errores para que se vean en consola
                }

                const { tabLabel, tabPane } = registration;
                if (!tabLabel || !tabPane)
                {
                    //console.error("[WME PLN] Falló el registro del Tab: 'tabLabel' o 'tabPane' no fueron retornados.");
                    return;
                }

                // Configurar el ícono y nombre de la pestaña principal del script
                tabLabel.innerHTML = `
                <img src=""
                style="height: 16px; vertical-align: middle; margin-right: 5px;">
                NrmliZer
            `;

                // 3. Inicializar las pestañas internas (General, Especiales,
                // Diccionario, Reemplazos)
                const tabsContainer = document.createElement("div");
                tabsContainer.style.display = "flex";
                tabsContainer.style.marginBottom = "8px";
                tabsContainer.style.gap = "8px";

                const tabButtons = {};
                const tabContents = {}; // Objeto para guardar los divs de contenido



                // Crear botones para cada pestaña
                tabNames.forEach(({ label, icon }) => {
                    const btn = document.createElement("button");
                    btn.innerHTML = icon
                    ? `<span style="display: inline-flex; align-items: center; font-size: 11px;">
                        <span style="font-size: 12px; margin-right: 4px;">${icon}</span>${label}
                    </span>`
                    : `<span style="font-size: 11px;">${label}</span>`;



                    btn.style.fontSize = "11px";
                    btn.style.padding = "4px 8px";
                    btn.style.marginRight = "4px";
                    btn.style.minHeight = "28px";
                    btn.style.border = "1px solid #ccc";
                    btn.style.borderRadius = "4px 4px 0 0";
                    btn.style.cursor = "pointer";
                    btn.style.borderBottom = "none"; // Para que la pestaña activa se vea mejor integrada

                    btn.className = "custom-tab-style";

                    // Agrega aquí el tooltip personalizado para cada tab
                    if (label === "Gene") btn.title = "Configuración general";
                    else if (label === "Espe") btn.title = "Palabras especiales (Excluidas)";
                    else if (label === "Dicc") btn.title = "Diccionario de palabras válidas";
                    else if (label === "Reemp") btn.title = "Gestión de reemplazos automáticos";

                    // Estilo inicial: la primera pestaña es la activa
                    if (label === tabNames[0].label) {
                        btn.style.backgroundColor = "#ffffff"; // Color de fondo activo (blanco)
                        btn.style.borderBottom = "2px solid #007bff"; // Borde inferior distintivo para la activa
                        btn.style.fontWeight = "bold";
                    } else {
                        btn.style.backgroundColor = "#f0f0f0"; // Color de fondo inactivo (gris claro)
                        btn.style.fontWeight = "normal";
                    }

                    btn.addEventListener("click", () => {
                        tabNames.forEach(({label : tabLabel_inner}) => {
                            const isActive = (tabLabel_inner === label);
                            const currentButton = tabButtons[tabLabel_inner];

                            if (tabContents[tabLabel_inner]) {
                                tabContents[tabLabel_inner].style.display = isActive ? "block" : "none";
                            }

                            if (currentButton) {
                                // Aplicar/Quitar estilos de pestaña activa directamente
                                if (isActive) {
                                    currentButton.style.backgroundColor = "#ffffff"; // Activo
                                    currentButton.style.borderBottom = "2px solid #007bff";
                                    currentButton.style.fontWeight = "bold";
                                } else {
                                    currentButton.style.backgroundColor = "#f0f0f0"; // Inactivo
                                    currentButton.style.borderBottom = "none";
                                    currentButton.style.fontWeight = "normal";
                                }
                            }
                            // Llamar a la función de renderizado correspondiente
                            if (isActive) {
                                if (tabLabel_inner === "Espe")
                                {
                                    const ul = document.getElementById("excludedWordsList");
                                    if (ul && typeof renderExcludedWordsList === 'function') renderExcludedWordsList(ul);
                                }
                                else if (tabLabel_inner === "Dicc")
                                {
                                    const ulDict = document.getElementById("dictionaryWordsList");
                                    if (ulDict && typeof renderDictionaryList === 'function') renderDictionaryList(ulDict);
                                }
                                else if (tabLabel_inner === "Reemp")
                                {
                                const ulReemplazos = document.getElementById("replacementsListElementID");
                                    if (ulReemplazos && typeof renderReplacementsList === 'function') renderReplacementsList(ulReemplazos);
                                }
                            }
                        });
                    });
                    tabButtons[label] = btn;
                    tabsContainer.appendChild(btn);
                });
                tabPane.appendChild(tabsContainer);

                // Crear los divs contenedores para el contenido de cada pestaña
                tabNames.forEach(({ label }) => {
                    const contentDiv = document.createElement("div");
                    contentDiv.style.display = label === tabNames[0].label ? "block" : "none"; // Mostrar solo la primera
                    contentDiv.style.padding = "10px";
                    tabContents[label] = contentDiv; // Guardar referencia
                    tabPane.appendChild(contentDiv);
                });
                // --- POBLAR EL CONTENIDO DE CADA PESTAÑA ---

                // 4. Poblar el contenido de la pestaña "General"
                const containerGeneral = tabContents["Gene"];
                if (containerGeneral)
                {
                    let initialUsernameAttempt =
                    "Pendiente"; // Para la etiqueta simplificada
                    // No es necesario el polling complejo si solo es para la lógica
                    // interna del checkbox
                    if (typeof W !== 'undefined' && W.loginManager &&
                        W.loginManager.user && W.loginManager.user.userName)
                    {
                        initialUsernameAttempt =
                        W.loginManager.user
                            .userName; // Se usará internamente en processNextPlace
                    }
                    const mainTitle = document.createElement("h3");
                    mainTitle.textContent = "NormliZer";
                    mainTitle.style.textAlign = "center";
                    mainTitle.style.fontSize = "18px";
                    mainTitle.style.marginBottom = "2px";
                    containerGeneral.appendChild(mainTitle);

                    const versionInfo = document.createElement("div");
                    versionInfo.textContent =
                    "V. " + VERSION; // VERSION global
                    versionInfo.style.textAlign = "right";
                    versionInfo.style.fontSize = "10px";
                    versionInfo.style.color = "#777";
                    versionInfo.style.marginBottom = "15px";
                    containerGeneral.appendChild(versionInfo);

                    const normSectionTitle = document.createElement("h4");
                    normSectionTitle.textContent = "Análisis de Nombres de Places";
                    normSectionTitle.style.fontSize = "15px";
                    normSectionTitle.style.marginTop = "10px";
                    normSectionTitle.style.marginBottom = "5px";
                    normSectionTitle.style.borderBottom = "1px solid #eee";
                    normSectionTitle.style.paddingBottom = "3px";
                    containerGeneral.appendChild(normSectionTitle);

                    const scanButton = document.createElement("button");
                    scanButton.textContent = "Start Scan...";
                    scanButton.setAttribute("type", "button");
                    scanButton.style.marginBottom = "10px";
                    scanButton.style.width = "100%";
                    scanButton.style.padding = "8px";
                    scanButton.style.border = "none";
                    scanButton.style.borderRadius = "4px";
                    scanButton.style.backgroundColor = "#007bff";
                    scanButton.style.color = "#fff";
                    scanButton.style.cursor = "pointer";
                    scanButton.addEventListener("click", () => {
                        const places = getVisiblePlaces();
                        const outputDiv =
                        document.getElementById("wme-normalization-tab-output");
                        if (!outputDiv)
                        { // Mover esta verificación antes
                        // console.error("Div de salida (wme-normalization-tab-output) no encontrado en el tab.");
                            return;
                        }
                        if (places.length === 0)
                        {
                            outputDiv.textContent =  "No se encontraron lugares visibles para analizar.";
                            return;
                        }
                        const maxPlacesInput =  document.getElementById("maxPlacesInput");
                        const maxPlacesToScan = parseInt(maxPlacesInput?.value || "100", 10);
                        const scannedCount = Math.min(places.length, maxPlacesToScan);
                        outputDiv.textContent = `Escaneando ${scannedCount} lugares...`;
                        setTimeout(() => {renderPlacesInFloatingPanel(places.slice(0, maxPlacesToScan));
                        }, 10);
                    });
                    containerGeneral.appendChild(scanButton);

                    const maxWrapper = document.createElement("div");
                    maxWrapper.style.display = "flex";
                    maxWrapper.style.alignItems = "center";
                    maxWrapper.style.gap = "8px";
                    maxWrapper.style.marginBottom = "8px";
                    const maxLabel = document.createElement("label");
                    maxLabel.textContent = "Máximo de places a revisar:";
                    maxLabel.style.fontSize = "13px";
                    maxWrapper.appendChild(maxLabel);
                    const maxInput = document.createElement("input");
                    maxInput.type = "number";
                    maxInput.id = "maxPlacesInput";
                    maxInput.min = "1";
                    maxInput.value = "100";
                    maxInput.style.width = "80px";
                    maxWrapper.appendChild(maxInput);
                    containerGeneral.appendChild(maxWrapper);

                    const presets = [ 25, 50, 100, 250, 500 ];
                    const presetContainer = document.createElement("div");
                    presetContainer.style.textAlign = "center";
                    presetContainer.style.marginBottom = "8px";
                    presets.forEach(preset => {
                        const btn = document.createElement("button");
                        btn.textContent = preset.toString();
                        btn.style.margin = "2px";
                        btn.style.padding = "4px 6px";
                        btn.addEventListener("click", () => {
                            if (maxInput)
                                maxInput.value = preset.toString();
                        });
                        presetContainer.appendChild(btn);
                    });
                    containerGeneral.appendChild(presetContainer);

                    const similarityLabel = document.createElement("label");
                    similarityLabel.textContent = "Similitud mínima para sugerencia de palabras:";
                    similarityLabel.title = "Ajusta el umbral mínimo de similitud (entre 80% y 95%) que se usará para sugerencias de reemplazo en nombres. El 100% es un reemplazo directo.";
                    similarityLabel.style.fontSize = "12px";
                    similarityLabel.style.display = "block";
                    similarityLabel.style.marginTop = "8px";
                    similarityLabel.style.marginBottom = "4px";
                    containerGeneral.appendChild(similarityLabel);

                    const similaritySlider = document.createElement("input");
                    similaritySlider.type = "range";
                    similaritySlider.min = "80";
                    similaritySlider.max = "95";
                    similaritySlider.value = "85";
                    similaritySlider.id = "similarityThreshold";
                    similaritySlider.style.width = "100%";
                    similaritySlider.title = "Desliza para ajustar la similitud mínima";
                    const similarityValueDisplay = document.createElement("span");
                    similarityValueDisplay.textContent = similaritySlider.value + "%";
                    similarityValueDisplay.style.marginLeft = "8px";
                    similaritySlider.addEventListener("input", () => {
                        similarityValueDisplay.textContent =
                        similaritySlider.value + "%";
                    });
                    containerGeneral.appendChild(similaritySlider);
                    containerGeneral.appendChild(similarityValueDisplay);

                    // ELIMINAR ESTO (SI SE APLICÓ EL CAMBIO ANTERIOR):
                    // const editorFilterWrapper = document.getElementById("editorFilterInput")?.closest("div");
                    // if (editorFilterWrapper) editorFilterWrapper.remove();

                    // --- NUEVO: Checkbox para omitir mis ediciones ---
                    const hideMyEditsWrapper = document.createElement("div");
                    hideMyEditsWrapper.style.marginTop = "10px";
                    hideMyEditsWrapper.style.marginBottom = "5px";
                    hideMyEditsWrapper.style.display = "flex";
                    hideMyEditsWrapper.style.alignItems = "center";

                    const hideMyEditsCheckbox = document.createElement("input");
                    hideMyEditsCheckbox.type = "checkbox";
                    hideMyEditsCheckbox.id = "chk-hide-my-edits";
                    hideMyEditsCheckbox.style.marginRight = "5px";
                    // Opcional: Cargar preferencia desde localStorage
                    // const savedHideMyEdits = localStorage.getItem('WMEPlnHOM');
                    // if (savedHideMyEdits !== null) {
                    //    hideMyEditsCheckbox.checked = savedHideMyEdits === 'true';
                    // }
                    // hideMyEditsCheckbox.addEventListener('change', (event) => {
                    //    localStorage.setItem('WMEPlnHOM', event.target.checked);
                    // });


                    const hideMyEditsLabel = document.createElement("label");
                    // --- MODIFICACIÓN AQUÍ ---
                    let labelText = "Omitir lugares editados por mí";
                    let currentUserName = null;
                    if (typeof W !== 'undefined' && W.loginManager && W.loginManager.user && W.loginManager.user.userName) {
                        currentUserName = W.loginManager.user.userName;
                        labelText += ` (${currentUserName})`;
                    } else {
                        labelText += " (Usuario no detectado)";
                    }
                    hideMyEditsLabel.textContent = labelText;
                    // --- FIN MODIFICACIÓN ---
                    hideMyEditsLabel.htmlFor = "chk-hide-my-edits";
                    hideMyEditsLabel.style.fontSize = "13px";
                    hideMyEditsLabel.style.cursor = "pointer";

                    hideMyEditsWrapper.appendChild(hideMyEditsCheckbox);
                    hideMyEditsWrapper.appendChild(hideMyEditsLabel);
                   // containerGeneral.appendChild(hideMyEditsWrapper);
                    // --- FIN: Checkbox para omitir mis ediciones ---

                    const tabProgressWrapper = document.createElement("div");
                    tabProgressWrapper.style.margin = "10px 0";
                    tabProgressWrapper.style.height = "18px";
                    tabProgressWrapper.style.backgroundColor = "transparent";
                    const tabProgressBar = document.createElement("div");
                    tabProgressBar.style.height = "100%";
                    tabProgressBar.style.width = "0%";
                    tabProgressBar.style.backgroundColor = "#007bff";
                    tabProgressBar.style.transition = "width 0.2s";
                    tabProgressBar.id = "progressBarInnerTab";
                    tabProgressWrapper.appendChild(tabProgressBar);
                    containerGeneral.appendChild(tabProgressWrapper);

                    const tabProgressText = document.createElement("div");
                    tabProgressText.style.fontSize = "12px";
                    tabProgressText.style.marginTop = "5px";
                    tabProgressText.id = "progressBarTextTab";
                    tabProgressText.textContent = "Progreso: 0% (0/0)";
                    containerGeneral.appendChild(tabProgressText);

                    const outputNormalizationInTab = document.createElement("div");
                    outputNormalizationInTab.id = "wme-normalization-tab-output";
                    outputNormalizationInTab.style.fontSize = "12px";
                    outputNormalizationInTab.style.minHeight = "20px";
                    outputNormalizationInTab.style.padding = "5px";
                    outputNormalizationInTab.style.marginBottom = "15px";
                    outputNormalizationInTab.textContent =
                    "Presiona 'Start Scan...' para analizar los places visibles.";
                    containerGeneral.appendChild(outputNormalizationInTab);
                }
                else
                {
                    console.error("[WME PLN] No se pudo poblar la pestaña 'General' porque su contenedor no existe.");
                }

                // 5. Poblar las otras pestañas
                if (tabContents["Espe"])
                {
                    createExcludedWordsManager(tabContents["Espe"]) ;
                }
                else
                {
                    console.error("[WME PLN] No se pudo encontrar el contenedor para la pestaña 'Especiales'.");
                }

                if (tabContents["Dicc"])
                {
                    createDictionaryManager(tabContents["Dicc"]);
                }
                else
                {
                    console.error(
                    "[WME PLN] No se pudo encontrar el contenedor para la pestaña 'Diccionario'.");
                }

                // --- LLAMADA A LA FUNCIÓN PARA POBLAR LA NUEVA PESTAÑA "Reemplazos"
                // ---
                if (tabContents["Reemp"])
                {
                    createReplacementsManager(
                    tabContents["Reemp"]); // Esta es la llamada clave
                }
                else
                {
                    console.error(
                    "[WME PLN] No se pudo encontrar el contenedor para la pestaña 'Reemplazos'.");
                }
            }
            catch (error)
            {
                console.error("[WME PLN] Error catastrófico creando la pestaña lateral:", error, error.stack);
            }
        } // Fin de createSidebarTab



    function waitForSidebarAPI()
    {
        if (W && W.userscripts && W.userscripts.registerSidebarTab)
        {
            const savedExcluded = localStorage.getItem("excludedWordsList");
            if (savedExcluded)
            {
                try
                {
                    const parsed = JSON.parse(savedExcluded);
                    excludedWords = new Set(parsed);
                    /*   console.log(
                         "[WME PLN] Palabras especiales restauradas desde
                       localStorage:", Array.from(excludedWords));*/
                }
                catch (e)
                {
                    /*console.error(
                      "[WME PLN] Error al cargar excludedWordsList del localStorage:",
                      e);*/
                    excludedWords = new Set();
                }
            }
            else
            {
                excludedWords = new Set();
                /*  console.log(
                    "[WME PLN] No se encontraron palabras especiales en
                   localStorage.");*/
            }
            // --- Cargar diccionario desde localStorage ---
            const savedDictionary = localStorage.getItem("dictionaryWordsList");
            if (savedDictionary)
            {
                try
                {
                    const parsed = JSON.parse(savedDictionary);
                    window.dictionaryWords = new Set(parsed);
                    window.dictionaryIndex = {};
                    parsed.forEach(word => {
                        const letter = word.charAt(0).toLowerCase();
                        if (!window.dictionaryIndex[letter])
                        {
                            window.dictionaryIndex[letter] = [];
                        }
                        window.dictionaryIndex[letter].push(word);
                    });
                    /*  console.log(
                        "[WME PLN] Diccionario restaurado desde localStorage:",
                        parsed);*/
                }
                catch (e)
                {
                    /* console.error(
                       "[WME PLN] Error al cargar dictionaryWordsList del
                       localStorage:", e);*/
                    window.dictionaryWords = new Set();
                }
            }
            else
            {
                window.dictionaryWords = new Set();
                //  console.log("[WME PLN] No se encontró diccionario en
                //  localStorage.");
            }
            loadReplacementWordsFromStorage();
            waitForWazeAPI(() => { createSidebarTab(); });
        }
        else
        {
            // console.log("[WME PLN] Esperando W.userscripts API...");
            setTimeout(waitForSidebarAPI, 1000);
        }
    }


    // 1. MODIFICAR normalizePlaceName
    function normalizePlaceName(word)
    {
        if (!word || typeof word !== "string")
        {
            return "";
        }

        // Manejar palabras con "/" recursivamente
        if (word.includes("/")) {
            if (word === "/") return "/";
            return word.split("/").map(part => normalizePlaceName(part.trim())).join("/");
        }

        // Casos especiales como "MI" y "DI"
        if (word.toUpperCase() === "MI" || word.toUpperCase() === "DI") {
            return word.charAt(0).toUpperCase() + word.slice(1).toLowerCase();
        }

        // Números romanos: todo en mayúsculas. No elimina puntos aquí.
        const romanRegexStrict = /^M{0,3}(CM|CD|D?C{0,3})(XC|XL|L?X{0,3})(IX|IV|V?I{0,3})$/i;
        if (romanRegexStrict.test(word)) {
            return word.toUpperCase();
        }

        // Si hay un número seguido de letra (sin espacio), convertir la letra en mayúscula
        word = word.replace(/(\d)([a-zÁÉÍÓÚÑáéíóúñ])/gi, (_, num, letter) => `${num}${letter.toUpperCase()}`);

        let normalizedWord;
        if (/^[0-9]+$/.test(word)) { // Solo números
            normalizedWord = word;
        } else if (/^[A-ZÁÉÍÓÚÑ0-9.]+$/.test(word) && word.length > 1 && word.includes('.'))
        {
            // Si es todo mayúsculas (o números) Y CONTIENE UN PUNTO (ej. "St."), mantener como está.
            // Podrías añadir más heurísticas aquí si es necesario para acrónimos con puntos.
            normalizedWord = word;
        }
        else
        { // Capitalización estándar
            normalizedWord = word.charAt(0).toUpperCase() + word.slice(1).toLowerCase();
        }

        // NO eliminamos el punto aquí. Eso se hará al final del nombre completo del place si es necesario.
        // Ejemplo: "St." se mantendrá como "St." después de esta función.

        return normalizedWord;
    }

    //***************************************************************************
    // Nombre: normalizeWordInternal
    // Autor: mincho77
    // Fecha: 2025-05-27
    // Descripción: Usada por postProcessQuotesAndParentheses. Similar a normalizePlaceName
    //              pero con contexto de si es la primera palabra o dentro de comillas/paréntesis.
    //***************************************************************************
    function normalizeWordInternal(word, isFirstWordInSequence = false, isInsideQuotesOrParentheses = false) {
        if (!word || typeof word !== "string") return "";

        // Replicar lógica esencial de normalizePlaceName para consistencia
        if (word.toUpperCase() === "MI" || word.toUpperCase() === "DI") {
            return word.charAt(0).toUpperCase() + word.slice(1).toLowerCase();
        }

        const romanRegexStrict = /^M{0,3}(CM|CD|D?C{0,3})(XC|XL|L?X{0,3})(IX|IV|V?I{0,3})$/i;
        if (romanRegexStrict.test(word)) {
            return word.toUpperCase();
        }

        word = word.replace(/(\d)([a-zÁÉÍÓÚÑáéíóúñ])/gi, (_, num, letter) => `${num}${letter.toUpperCase()}`);

        let resultWord;
        if (isInsideQuotesOrParentheses && !isFirstWordInSequence && commonWords.includes(word.toLowerCase())) {
            resultWord = word.toLowerCase();
        } else if (/^[0-9]+$/.test(word)) {
            resultWord = word;
        } else if (isInsideQuotesOrParentheses && /^[A-ZÁÉÍÓÚÑ0-9.]+$/.test(word) && word.length > 1 && word.includes('.')) {
            resultWord = word; // Mantener "St." dentro de comillas/paréntesis
        }
         else if (isInsideQuotesOrParentheses && /^[A-ZÁÉÍÓÚÑ0-9]+$/.test(word) && word.length > 1) {
            resultWord = word; // Mantener acrónimos sin puntos
        }
         else {
            resultWord = word.charAt(0).toUpperCase() + word.slice(1).toLowerCase();
        }

        // NO eliminamos el punto aquí tampoco.
        return resultWord;
    }

    // 3. La función postProcessQuotesAndParentheses (CORREGIDA de la respuesta anterior)
    function postProcessQuotesAndParentheses(text) {
        if (typeof text !== 'string') return text;

        // Normalizar contenido dentro de comillas dobles
        text = text.replace(/"([^"]*)"/g, (match, content) => {
            const trimmedContent = content.trim();
            if (trimmedContent === "") return '""';

            const wordsInside = trimmedContent.split(/\s+/).filter(w => w.length > 0);
            const normalizedWordsInside = wordsInside.map((singleWord, index) => {
                return normalizeWordInternal(singleWord, index === 0, true); // true para isInsideQuotesOrParentheses
            }).join(' ');
            return `"${normalizedWordsInside}"`; // Sin espacios extra
        });

        // Normalizar contenido dentro de paréntesis
        text = text.replace(/\(([^)]*)\)/g, (match, content) => {
            const trimmedContent = content.trim();
            if (trimmedContent === "") return '()';

            const wordsInside = trimmedContent.split(/\s+/).filter(w => w.length > 0);
            const normalizedWordsInside = wordsInside.map((singleWord, index) => {
                return normalizeWordInternal(singleWord, index === 0, true); // true para isInsideQuotesOrParentheses
            }).join(' ');
            return `(${normalizedWordsInside})`; // Sin espacios extra
        });

        return text.replace(/\s+/g, ' ').trim(); // Limpieza final general
    }
    // === Palabras especiales ===
        let excludedWords = new Set(); // Inicializada en waitForSidebarAPI
        let replacementWords = {};     // { "Urb.": "Urbanización", ... }
        let dictionaryWords = new Set(); // O window.dictionaryWords = new Set();

        function createExcludedWordsManager(parentContainer)
        {
            const section = document.createElement("div");
            section.id = "excludedWordsManagerSection"; // ID para la sección
            section.style.marginTop = "20px";
            section.style.borderTop = "1px solid #ccc";
            section.style.paddingTop = "10px";

            const title =
              document.createElement("h4"); // Cambiado a h4 para jerarquía
            title.textContent = "Gestión de Palabras Especiales";
            title.style.fontSize =
              "15px"; // Consistente con el otro título de sección
            title.style.marginBottom = "10px"; // Más espacio abajo
            section.appendChild(title);

            const addControlsContainer = document.createElement("div");
            addControlsContainer.style.display = "flex";
            addControlsContainer.style.gap = "8px";
            addControlsContainer.style.marginBottom = "8px";
            addControlsContainer.style.alignItems =
              "center"; // Alinear verticalmente

            const input = document.createElement("input");
            input.type = "text";
            input.placeholder = "Nueva palabra o frase";
            input.style.flexGrow = "1";
            input.style.padding = "6px"; // Mejor padding
            input.style.border = "1px solid #ccc";
            input.style.borderRadius = "3px";
            addControlsContainer.appendChild(input);

            const addBtn = document.createElement("button");
            addBtn.textContent = "Añadir";
            addBtn.style.padding = "6px 10px"; // Mejor padding
            addBtn.style.cursor = "pointer";

            addBtn.addEventListener("click", function() {
                const newWord = input.value.trim();
                const validation = isValidExcludedWord(newWord);
                if (!validation.valid)
                {
                    alert(validation.msg);
                    return;
                }
                // isValidExcludedWord ya comprueba duplicados en excludedWords y
                // commonWords y ahora también si existe en dictionaryWords.
                excludedWords.add(newWord);
                input.value = "";
                renderExcludedWordsList(
                  document.getElementById("excludedWordsList"));
                saveExcludedWordsToLocalStorage(); // Asumiendo que tienes esta
                                                   // función
            });
            addControlsContainer.appendChild(addBtn);
            section.appendChild(addControlsContainer);

            const actionButtonsContainer = document.createElement("div");
            actionButtonsContainer.style.display = "flex";
            actionButtonsContainer.style.gap = "8px";
            actionButtonsContainer.style.marginBottom = "10px"; // Más espacio

            const exportBtn = document.createElement("button");
            exportBtn.textContent = "Exportar"; // Más corto
            exportBtn.title = "Exportar Lista a XML";
            exportBtn.style.padding = "6px 10px";
            exportBtn.style.cursor = "pointer";

            exportBtn.addEventListener("click", exportSharedDataToXml);
            actionButtonsContainer.appendChild(exportBtn);

            const clearBtn = document.createElement("button");
            clearBtn.textContent = "Limpiar"; // Más corto
            clearBtn.title = "Limpiar toda la lista";
            clearBtn.style.padding = "6px 10px";
            clearBtn.style.cursor = "pointer";
            clearBtn.addEventListener("click", function() {
                if (
                  confirm(
                    "¿Estás seguro de que deseas eliminar TODAS las palabras de la lista?"))
                {
                    excludedWords.clear();
                    renderExcludedWordsList(document.getElementById(
                      "excludedWordsList")); // Pasar el elemento UL
                }
            });
            actionButtonsContainer.appendChild(clearBtn);
            section.appendChild(actionButtonsContainer);

            const search = document.createElement("input");
            search.type = "text";
            search.placeholder = "Buscar en especiales...";
            search.style.display = "block";
            search.style.width = "calc(100% - 14px)"; // Considerar padding y borde
            search.style.padding = "6px";
            search.style.border = "1px solid #ccc";
            search.style.borderRadius = "3px";
            search.style.marginBottom = "5px";

            search.addEventListener("input", () => {
                // Pasar el ulElement directamente
                renderExcludedWordsList(
                  document.getElementById("excludedWordsList"),
                  search.value.trim());
            });
            section.appendChild(search);

            const listContainerElement = document.createElement("ul");
            listContainerElement.id = "excludedWordsList"; // Este es el UL
            listContainerElement.style.maxHeight = "150px";
            listContainerElement.style.overflowY = "auto";
            listContainerElement.style.border = "1px solid #ddd";
            listContainerElement.style.padding = "5px"; // Padding interno
            listContainerElement.style.margin = "0";    // Resetear margen
            listContainerElement.style.background = "#fff";
            listContainerElement.style.listStyle = "none";
            section.appendChild(listContainerElement);

            const dropArea = document.createElement("div");
            dropArea.textContent =
              "Arrastra aquí el archivo XML de palabras especiales";
            dropArea.style.border = "2px dashed #ccc"; // Borde más visible
            dropArea.style.borderRadius = "4px";
            dropArea.style.padding = "15px"; // Más padding
            dropArea.style.marginTop = "10px";
            dropArea.style.textAlign = "center";
            dropArea.style.background = "#f9f9f9";
            dropArea.style.color = "#555";
            dropArea.addEventListener("dragover", (e) => {
                e.preventDefault();
                dropArea.style.background = "#e9e9e9";
                dropArea.style.borderColor = "#aaa";
            });
            dropArea.addEventListener("dragleave", () => {
                dropArea.style.background = "#f9f9f9";
                dropArea.style.borderColor = "#ccc";
            });
            dropArea.addEventListener("drop", (e) => {
                e.preventDefault();
                dropArea.style.background = "#f9f9f9";
                handleXmlFileDrop(e.dataTransfer.files[0]);

                if (file &&
                    (file.type === "text/xml" || file.name.endsWith(".xml")))
                {
                    const reader = new FileReader();
                    reader.onload = function(evt) {
                        try
                        {
                            const parser = new DOMParser();
                            const xmlDoc = parser.parseFromString(
                              evt.target.result, "application/xml");
                            const parserError = xmlDoc.querySelector("parsererror");
                            if (parserError)
                            {
                                alert("Error al parsear el archivo XML.");
                                return;
                            }
                            // Detectar raíz
                            const rootTag =
                              xmlDoc.documentElement.tagName.toLowerCase();
                            if (rootTag !== "excludedwords" &&
                                rootTag !== "diccionario")
                            {
                                alert(
                                  "El archivo XML no es válido. Debe tener <ExcludedWords> o <diccionario> como raíz.");
                                return;
                            }
                            // Importar palabras
                            const words = xmlDoc.getElementsByTagName("word");
                            let newWordsAddedCount = 0;
                            for (let i = 0; i < words.length; i++)
                            {
                                const val = words[i].textContent.trim();
                                if (val && !excludedWords.has(val))
                                {
                                    excludedWords.add(val);
                                    newWordsAddedCount++;
                                }
                            }
                            // Importar reemplazos si existen
                            const replacements =
                              xmlDoc.getElementsByTagName("replacement");
                            for (let i = 0; i < replacements.length; i++)
                            {
                                const from = replacements[i].getAttribute("from");
                                const to = replacements[i].textContent.trim();
                                if (from && to)
                                {
                                    replacementWords[from] = to;
                                }
                            }
                            renderExcludedWordsList(
                              document.getElementById("excludedWordsList"));
                            alert(`Importación completada. Palabras nuevas: ${
                              newWordsAddedCount}`);
                        }
                        catch (err)
                        {
                            alert("Error procesando el archivo XML.");
                        }
                    };
                    reader.readAsText(file);
                }
                else
                {
                    alert("Por favor, arrastra un archivo XML válido.");
                }
            });
            section.appendChild(dropArea);
            parentContainer.appendChild(section);
        }

    // === Diccionario ===
    function createDictionaryManager(parentContainer)
    {
        const section = document.createElement("div");
        section.id = "dictionaryManagerSection";
        section.style.marginTop = "20px";
        section.style.borderTop = "1px solid #ccc";
        section.style.paddingTop = "10px";

        const title = document.createElement("h4");
        title.textContent = "Gestión del Diccionario";
        title.style.fontSize = "15px";
        title.style.marginBottom = "10px";
        section.appendChild(title);

        const addControlsContainer = document.createElement("div");
        addControlsContainer.style.display = "flex";
        addControlsContainer.style.gap = "8px";
        addControlsContainer.style.marginBottom = "8px";
        addControlsContainer.style.alignItems = "center"; // Alinear verticalmente

        const input = document.createElement("input");
        input.type = "text";
        input.placeholder = "Nueva palabra";
        input.style.flexGrow = "1";
        input.style.padding = "6px"; // Mejor padding
        input.style.border = "1px solid #ccc";
        input.style.borderRadius = "3px";
        addControlsContainer.appendChild(input);

        const addBtn = document.createElement("button");
        addBtn.textContent = "Añadir";
        addBtn.style.padding = "6px 10px"; // Mejor padding
        addBtn.style.cursor = "pointer";
        addBtn.addEventListener("click", function() {
            const newWord = input.value.trim();
            if (newWord)
            {
                const lowerNewWord = newWord.toLowerCase();

                const alreadyExists =
                  Array.from(window.dictionaryWords)
                    .some(w => w.toLowerCase() === lowerNewWord);

                if (commonWords.includes(lowerNewWord))
                {
                    alert(
                      "La palabra es muy común y no debe agregarse a la lista.");
                    return;
                }

                if (alreadyExists)
                {
                    alert("La palabra ya está en la lista (ignorando mayúsculas).");
                    return;
                }

                window.dictionaryWords.add(newWord);
                input.value = "";
                renderDictionaryList(
                  document.getElementById("dictionaryWordsList"));

            }
        });

        addControlsContainer.appendChild(addBtn);
        section.appendChild(addControlsContainer);

        const actionButtonsContainer = document.createElement("div");
        actionButtonsContainer.style.display = "flex";
        actionButtonsContainer.style.gap = "8px";
        actionButtonsContainer.style.marginBottom = "10px"; // Más espacio

        const exportBtn = document.createElement("button");
        exportBtn.textContent = "Exportar"; // Más corto
        exportBtn.title = "Exportar Diccionario a XML";
        exportBtn.style.padding = "6px 10px";
        exportBtn.style.cursor = "pointer";
        exportBtn.addEventListener("click", exportDictionaryWordsList);
        actionButtonsContainer.appendChild(exportBtn);

        const clearBtn = document.createElement("button");
        clearBtn.textContent = "Limpiar"; // Más corto
        clearBtn.title = "Limpiar toda la lista";
        clearBtn.style.padding = "6px 10px";
        clearBtn.style.cursor = "pointer";
        clearBtn.addEventListener("click", function() {
            if (
              confirm(
                "¿Estás seguro de que deseas eliminar TODAS las palabras del diccionario?"))
            {
                window.dictionaryWords.clear();
                renderDictionaryList(document.getElementById(
                  "dictionaryWordsList")); // Pasar el elemento UL
            }
        });
        actionButtonsContainer.appendChild(clearBtn);
        section.appendChild(actionButtonsContainer);

        // Diccionario: búsqueda
        const search = document.createElement("input");
        search.type = "text";
        search.placeholder = "Buscar en diccionario...";
        search.style.display = "block";
        search.style.width = "calc(100% - 14px)";
        search.style.padding = "6px";
        search.style.border = "1px solid #ccc";
        search.style.borderRadius = "3px";
        search.style.marginTop = "5px";
        // On search input, render filtered list
        search.addEventListener("input", () => {
            renderDictionaryList(document.getElementById("dictionaryWordsList"),
                                 search.value.trim());
        });
        section.appendChild(search);

        // Lista UL para mostrar palabras del diccionario
        const listContainerElement = document.createElement("ul");
        listContainerElement.id = "dictionaryWordsList";
        listContainerElement.style.maxHeight = "150px";
        listContainerElement.style.overflowY = "auto";
        listContainerElement.style.border = "1px solid #ddd";
        listContainerElement.style.padding = "5px";
        listContainerElement.style.margin = "0";
        listContainerElement.style.background = "#fff";
        listContainerElement.style.listStyle = "none";
        section.appendChild(listContainerElement);

        const dropArea = document.createElement("div");
        dropArea.textContent = "Arrastra aquí el archivo XML del diccionario";
        dropArea.style.border = "2px dashed #ccc";
        dropArea.style.borderRadius = "4px";
        dropArea.style.padding = "15px";
        dropArea.style.marginTop = "10px";
        dropArea.style.textAlign = "center";
        dropArea.style.background = "#f9f9f9";
        dropArea.style.color = "#555";
        dropArea.addEventListener("dragover", (e) => {
            e.preventDefault();
            dropArea.style.background = "#e9e9e9";
            dropArea.style.borderColor = "#aaa";
        });
        dropArea.addEventListener("dragleave", () => {
            dropArea.style.background = "#f9f9f9";
            dropArea.style.borderColor = "#ccc";
        });

        dropArea.addEventListener("drop", (e) => {
            e.preventDefault();
            dropArea.style.background = "#f9f9f9";
            dropArea.style.borderColor = "#ccc";
            const file = e.dataTransfer.files[0];
            if (file && (file.type === "text/xml" || file.name.endsWith(".xml")))
            {
                const reader = new FileReader();
                reader.onload = function(evt) {
                    try
                    {
                        const parser = new DOMParser();
                        const xmlDoc = parser.parseFromString(evt.target.result,
                                                              "application/xml");
                        const parserError = xmlDoc.querySelector("parsererror");
                        if (parserError)
                        {
                            console.error("[WME PLN] Error parseando XML:",
                                          parserError.textContent);
                            alert(
                              "Error al parsear el archivo XML del diccionario.");
                            return;
                        }
                        const xmlWords = xmlDoc.querySelectorAll("word");
                        let newWordsAddedCount = 0;
                        for (let i = 0; i < xmlWords.length; i++)
                        {
                            const val = xmlWords[i].textContent.trim();
                            if (val && !window.dictionaryWords.has(val))
                            {
                                window.dictionaryWords.add(val);
                                newWordsAddedCount++;
                            }
                        }
                        if (newWordsAddedCount > 0)
                            console.log(`[WME PLN] ${
                              newWordsAddedCount} nuevas palabras añadidas desde XML.`);

                        // Renderizar la lista en el panel
                        renderDictionaryList(listContainerElement);
                    }
                    catch (err)
                    {
                        alert("Error procesando el diccionario XML.");
                    }
                };
                reader.readAsText(file);
            }
            else
            {
                alert("Por favor, arrastra un archivo XML válido.");
            }
        });
        section.appendChild(dropArea);

        parentContainer.appendChild(section);
        renderDictionaryList(listContainerElement);
    }

    function loadReplacementWordsFromStorage()
    {
        const savedReplacements = localStorage.getItem("replacementWordsList");
        if (savedReplacements)
        {
            try
            {
                replacementWords = JSON.parse(savedReplacements);
                if (typeof replacementWords !== 'object' ||
                    replacementWords === null)
                { // Asegurar que sea un objeto
                    replacementWords = {};
                }
            }
            catch (e)
            {
                console.error("[WME PLN] Error cargando lista de reemplazos desde localStorage:", e);
                replacementWords = {};
            }
        }
        else
        {
            replacementWords = {}; // Inicializar si no hay nada guardado
        }
        console.log("[WME PLN] Reemplazos cargados:",
                    Object.keys(replacementWords).length,
                    "reglas.");
    }

    function saveReplacementWordsToStorage()
    {
        try
        {
            localStorage.setItem("replacementWordsList",
                                 JSON.stringify(replacementWords));
            // console.log("[WME PLN] Lista de reemplazos guardada en
            // localStorage.");
        }
        catch (e)
        {
            console.error("[WME PLN] Error guardando lista de reemplazos en localStorage:", e);
        }
    }



    // Renderiza la lista de reemplazos
    function renderReplacementsList(ulElement)
    {
        //console.log("[WME PLN DEBUG] renderReplacementsList llamada para:", ulElement ? ulElement.id : "Elemento UL nulo");
        if (!ulElement)
        {
            //console.error("[WME PLN] Elemento UL para reemplazos no proporcionado a renderReplacementsList.");
            return;
        }
        ulElement.innerHTML = ""; // Limpiar lista actual
        const entries = Object.entries(replacementWords);

        if (entries.length === 0)
        {
            const li = document.createElement("li");
            li.textContent = "No hay reemplazos definidos.";
            li.style.textAlign = "center";
            li.style.color = "#777";
            li.style.padding = "5px";
            ulElement.appendChild(li);
            return;
        }

        // Ordenar alfabéticamente por la palabra original (from)
        entries.sort((a, b) =>
                       a[0].toLowerCase().localeCompare(b[0].toLowerCase()));

        entries.forEach(([ from, to ]) => {
            const li = document.createElement("li");
            li.style.display = "flex";
            li.style.justifyContent = "space-between";
            li.style.alignItems = "center";
            li.style.padding = "4px 2px";
            li.style.borderBottom = "1px solid #f0f0f0";

            const textContainer = document.createElement("div");
            textContainer.style.flexGrow = "1";
            textContainer.style.overflow = "hidden";
            textContainer.style.textOverflow = "ellipsis";
            textContainer.style.whiteSpace = "nowrap";
            textContainer.title = `Reemplazar "${from}" con "${to}"`;

            const fromSpan = document.createElement("span");
            fromSpan.textContent = from;
            fromSpan.style.fontWeight = "bold";
            textContainer.appendChild(fromSpan);

            const arrowSpan = document.createElement("span");
            arrowSpan.textContent = " → ";
            arrowSpan.style.margin = "0 5px";
            textContainer.appendChild(arrowSpan);

            const toSpan = document.createElement("span");
            toSpan.textContent = to;
            toSpan.style.color = "#007bff";
            textContainer.appendChild(toSpan);

            li.appendChild(textContainer);

            // Botón Editar
            const editBtn = document.createElement("button");
            editBtn.innerHTML = "✏️";
            editBtn.title = "Editar este reemplazo";
            editBtn.style.border = "none";
            editBtn.style.background = "transparent";
            editBtn.style.cursor = "pointer";
            editBtn.style.padding = "2px 4px";
            editBtn.style.fontSize = "14px";
            editBtn.style.marginLeft = "4px";
            editBtn.addEventListener("click", () => {
                const newFrom = prompt("Editar texto original:", from);
                if (newFrom === null) return;
                const newTo = prompt("Editar texto de reemplazo:", to);
                if (newTo === null) return;
                if (!newFrom.trim()) {
                    alert("El campo 'Texto Original' es requerido.");
                    return;
                }
                if (newFrom === newTo) {
                    alert("El texto original y el de reemplazo no pueden ser iguales.");
                    return;
                }
                // Si cambia la clave, elimina la anterior
                if (newFrom !== from) delete replacementWords[from];
                replacementWords[newFrom] = newTo;
                renderReplacementsList(ulElement);
                saveReplacementWordsToStorage();
            });

            // Botón Eliminar
            const deleteBtn = document.createElement("button");
            deleteBtn.innerHTML = "🗑️";
            deleteBtn.title = `Eliminar este reemplazo`;
            deleteBtn.style.border = "none";
            deleteBtn.style.background = "transparent";
            deleteBtn.style.cursor = "pointer";
            deleteBtn.style.padding = "2px 4px";
            deleteBtn.style.fontSize = "14px";
            deleteBtn.style.marginLeft = "4px";
            deleteBtn.addEventListener("click", () => {
                if (confirm(`¿Estás seguro de eliminar el reemplazo:\n"${from}" → "${to}"?`))
                {
                    delete replacementWords[from];
                    renderReplacementsList(ulElement);
                    saveReplacementWordsToStorage();
                }
            });

            const btnContainer = document.createElement("span");
            btnContainer.style.display = "flex";
            btnContainer.style.gap = "4px";
            btnContainer.appendChild(editBtn);
            btnContainer.appendChild(deleteBtn);

            li.appendChild(btnContainer);
            ulElement.appendChild(li);
        });
    }


    function exportSharedDataToXml()
    {
        if (excludedWords.size === 0 && Object.keys(replacementWords).length === 0)
        {
            alert(
              "No hay palabras especiales ni reemplazos definidos para exportar.");
            return;
        }

        let xmlParts = [];
        // Exportar palabras excluidas
        Array.from(excludedWords)
          .sort((a, b) => a.toLowerCase().localeCompare(b.toLowerCase()))
          .forEach(w => xmlParts.push(`    <word>${xmlEscape(w)}</word>`));

        // Exportar reemplazos
        Object.entries(replacementWords)
          .sort((a, b) => a[0].toLowerCase().localeCompare(
                  b[0].toLowerCase())) // Ordenar por 'from'
          .forEach(([ from, to ]) => {
              xmlParts.push(`    <replacement from="${xmlEscape(from)}">${
                xmlEscape(to)}</replacement>`);
          });

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

        const blob =
          new Blob([ xmlContent ], { type : "application/xml;charset=utf-8" });
        const url = URL.createObjectURL(blob);
        const a = document.createElement("a");
        a.href = url;
        // Puedes darle un nombre genérico ya que contiene ambos tipos de datos
        a.download = "wme_normalizer_data_export.xml";
        document.body.appendChild(a);
        a.click();
        document.body.removeChild(a);
        URL.revokeObjectURL(url);
    }

        function handleXmlFileDrop(file)
        {
            if (file && (file.type === "text/xml" || file.name.endsWith(".xml")))
            {
                const reader = new FileReader();
                reader.onload = function(evt) {
                    try
                    {
                        const parser = new DOMParser();
                        const xmlDoc =
                        parser.parseFromString(evt.target.result, "application/xml");
                        const parserError = xmlDoc.querySelector("parsererror");
                        if (parserError)
                        {
                            alert("Error al parsear el archivo XML: " +
                                parserError.textContent);
                            return;
                        }
                        const rootTag = xmlDoc.documentElement.tagName.toLowerCase();
                        if (rootTag !== "excludedwords")
                        { // Asumiendo que la raíz sigue siendo esta
                            alert(
                            "El archivo XML no es válido. Debe tener <ExcludedWords> como raíz.");
                            return;
                        }

                        let newExcludedAdded = 0;
                        let newReplacementsAdded = 0;
                        let replacementsOverwritten = 0;

                        // Importar palabras excluidas
                        const words = xmlDoc.getElementsByTagName("word");
                        for (let i = 0; i < words.length; i++)
                        {
                            const val = words[i].textContent.trim();
                            if (val && !excludedWords.has(val))
                            { // Asegúrate que isValidExcludedWord no lo bloquee si
                            // viene de XML
                                const validation =
                                isValidExcludedWord(val); // Revalidar o ajustar esta
                                                            // lógica para importación
                                if (validation.valid)
                                { // Omitir esta validación si el XML es la fuente de
                                // verdad
                                    excludedWords.add(val);
                                    newExcludedAdded++;
                                }
                                else
                                {
                                    console.warn(`Palabra excluida omitida desde XML "${
                                    val}": ${validation.msg}`);
                                }
                            }
                        }

                        // Importar reemplazos
                        const replacements = xmlDoc.getElementsByTagName("replacement");
                        for (let i = 0; i < replacements.length; i++)
                        {
                            const from = replacements[i].getAttribute("from")?.trim();
                            const to = replacements[i].textContent.trim();
                            if (from && to)
                            { // Permitir que 'to' esté vacío si se quiere "reemplazar
                            // con nada"
                                if (replacementWords.hasOwnProperty(from) &&
                                    replacementWords[from] !== to)
                                {
                                    replacementsOverwritten++;
                                }
                                else if (!replacementWords.hasOwnProperty(from))
                                {
                                    newReplacementsAdded++;
                                }
                                replacementWords[from] = to;
                            }
                        }

                        // Guardar y Re-renderizar AMBAS listas
                        saveExcludedWordsToLocalStorage(); // Asumo que tienes esta
                                                        // función o la adaptas
                        saveReplacementWordsToStorage();

                        // Re-renderizar las listas en sus respectivas pestañas si están
                        // visibles o al activarse
                        const excludedListElement =
                        document.getElementById("excludedWordsList");
                        if (excludedListElement)
                            renderExcludedWordsList(excludedListElement);

                        const replacementsListElement =
                        document.getElementById("replacementsListElementID");
                        if (replacementsListElement)
                            renderReplacementsList(replacementsListElement);

                        alert(`Importación completada.\nPalabras Especiales nuevas: ${
                        newExcludedAdded}\nReemplazos nuevos: ${
                        newReplacementsAdded}\nReemplazos sobrescritos: ${
                        replacementsOverwritten}`);
                    }
                    catch (err)
                    {
                        console.error(
                        "[WME PLN] Error procesando el archivo XML importado:", err);
                        alert("Ocurrió un error procesando el archivo XML.");
                    }
                };
                reader.readAsText(file);
            }
            else
            {
                alert("Por favor, arrastra un archivo XML válido.");
            }
        }



        function createReplacementsManager(parentContainer)
        {
            parentContainer.innerHTML = ''; // Limpiar por si acaso

            const title = document.createElement("h4");
            title.textContent = "Gestión de Reemplazos de Palabras/Frases";
            title.style.fontSize = "15px";
            title.style.marginBottom = "10px";
            parentContainer.appendChild(title);

            // Sección para añadir nuevos reemplazos
            const addSection = document.createElement("div");
            addSection.style.display = "flex";
            addSection.style.gap = "8px";
            addSection.style.marginBottom = "12px";
            addSection.style.alignItems = "flex-end"; // Alinear inputs y botón

            const fromInputContainer = document.createElement("div");
            fromInputContainer.style.flexGrow = "1";
            const fromLabel = document.createElement("label");
            fromLabel.textContent = "Texto Original:";
            fromLabel.style.display = "block";
            fromLabel.style.fontSize = "12px";
            fromLabel.style.marginBottom = "2px";
            const fromInput = document.createElement("input");
            fromInput.type = "text";
            fromInput.placeholder = "Ej: Urb.";
            fromInput.style.width = "95%"; // Para que quepa bien
            fromInput.style.padding = "6px";
            fromInput.style.border = "1px solid #ccc";

            fromInputContainer.appendChild(fromLabel);
            fromInputContainer.appendChild(fromInput);
            addSection.appendChild(fromInputContainer);


            const toInputContainer = document.createElement("div");
            toInputContainer.style.flexGrow = "1";
            const toLabel = document.createElement("label");
            toLabel.textContent = "Texto de Reemplazo:";
            toLabel.style.display = "block";
            toLabel.style.fontSize = "12px";
            toLabel.style.marginBottom = "2px";
            const toInput = document.createElement("input");
            toInput.type = "text";
            toInput.placeholder = "Ej: Urbanización";
            toInput.style.width = "95%";
            toInput.style.padding = "6px";
            toInput.style.border = "1px solid #ccc";
            toInputContainer.appendChild(toLabel);
            toInputContainer.appendChild(toInput);
            addSection.appendChild(toInputContainer);

            fromInput.setAttribute('spellcheck', 'false');
            toInput.setAttribute('spellcheck', 'false');


            const addReplacementBtn = document.createElement("button");
            addReplacementBtn.textContent = "Añadir";
            addReplacementBtn.style.padding = "6px 10px";
            addReplacementBtn.style.cursor = "pointer";
            addReplacementBtn.style.height = "30px"; // Para alinear con los inputs
            addSection.appendChild(addReplacementBtn);
            parentContainer.appendChild(addSection);

            // Elemento UL para la lista de reemplazos
            const listElement = document.createElement("ul");
            listElement.id = "replacementsListElementID"; // ID ÚNICO para esta lista
            listElement.style.maxHeight = "150px";
            listElement.style.overflowY = "auto";
            listElement.style.border = "1px solid #ddd";
            listElement.style.padding = "8px";
            listElement.style.margin = "0 0 10px 0";
            listElement.style.background = "#fff";
            listElement.style.listStyle = "none";
            parentContainer.appendChild(listElement);

            // Event listener para el botón "Añadir"
           addReplacementBtn.addEventListener("click", () => {
            const fromValue = fromInput.value.trim();
            const toValue = toInput.value.trim();

            if (!fromValue)
            {
                alert("El campo 'Texto Original' es requerido.");
                return;
            }
            if (fromValue === toValue)
            {
                alert("El texto original y el de reemplazo no pueden ser iguales.");
                return;
            }
            if (replacementWords.hasOwnProperty(fromValue) && replacementWords[fromValue] !== toValue)
            {
                if (!confirm(`El reemplazo para "${fromValue}" ya existe ('${replacementWords[fromValue]}'). ¿Deseas sobrescribirlo con '${toValue}'?`))
                    return;

            }
            replacementWords[fromValue] = toValue;
            fromInput.value = "";
            toInput.value = "";

            // Renderiza toda la lista (más seguro y rápido en la práctica)
            renderReplacementsList(listElement);

            saveReplacementWordsToStorage();
            });




       // Coloca esta función junto a aplicarReemplazosGenerales y aplicarReglasEspecialesNombre
        /*function aplicarReemplazosDefinidos(text, replacementRules)
        {
            let newText = text;
            if (typeof replacementRules !== 'object' || Object.keys(replacementRules).length === 0) {
                return newText; // No hay reglas o no es un objeto, devolver el texto original
            }

            // Ordenar las claves de reemplazo por longitud descendente para manejar casos como
            // "D1 Super" y "D1" donde "D1 Super" debe procesarse primero.
            const sortedFromKeys = Object.keys(replacementRules).sort((a, b) => b.length - a.length);

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

                // Crear una expresión regular para encontrar 'fromKey' como palabra completa,
                // insensible a mayúsculas/minúsculas (flag 'gi').
                // escapeRegExp es importante si 'fromKey' puede contener caracteres especiales de regex.
                const regex = new RegExp('\\b' + escapeRegExp(fromKey) + '\\b', 'giu'); // <--- MODIFICADO AQUÍ: Añadido flag 'u'

                newText = newText.replace(regex, (matchedInText, offset, originalString) => {
                    // matchedInText: La cadena que coincidió en el texto (ej. "d1", "D1").
                    // fromKey: La clave canónica del objeto replacementWords (ej. "D1").
                    // toValue: El valor de reemplazo canónico (ej. "Tiendas D1").

                    // Condición para evitar reemplazo redundante:
                    // Verificar si 'matchedInText' ya forma parte de 'toValue' en el contexto correcto.
                    // Ej: fromKey="D1", toValue="Tiendas D1".
                    // Si encontramos "D1" en "Mis Tiendas D1", queremos evitar cambiarlo a "Mis Tiendas Tiendas D1".

                    // Encontrar dónde estaría 'fromKey' (canónico) dentro de 'toValue' (canónico)
                    // para alinear la comprobación. Hacemos esta parte insensible a mayúsculas para la búsqueda de índice.
                    const fromKeyLower = fromKey.toLowerCase();
                    const toValueLower = toValue.toLowerCase();
                    const indexOfFromInTo = toValueLower.indexOf(fromKeyLower);

                    if (indexOfFromInTo !== -1) {
                        // Calculamos el inicio de donde podría estar el 'toValue' completo en el 'originalString'
                        const potentialExistingToStart = offset - indexOfFromInTo;

                        if (potentialExistingToStart >= 0 &&
                            (potentialExistingToStart + toValue.length) <= originalString.length) {

                            const substringInOriginal = originalString.substring(potentialExistingToStart, potentialExistingToStart + toValue.length);

                            // Comparamos (insensible a mayúsculas) si el 'toValue' ya existe en esa posición
                            if (substringInOriginal.toLowerCase() === toValueLower) {
                                // Ya existe el 'toValue' completo en el lugar correcto, así que no reemplazamos.
                                // Devolvemos el texto que coincidió originalmente.
                                return matchedInText;
                            }
                        }
                    }
                    // Si no se cumple la condición anterior, realizamos el reemplazo con 'toValue'.
                    return toValue;
                });
            }
            return newText;
        }*/


        // Botones de Acción y Drop Area (usarán la lógica compartida)
        const actionButtonsContainer = document.createElement("div");
        actionButtonsContainer.style.display = "flex";
        actionButtonsContainer.style.gap = "8px";
        actionButtonsContainer.style.marginBottom = "10px";

        const exportButton = document.createElement("button");
        exportButton.textContent = "Exportar Todo";
        exportButton.title = "Exportar Excluidas y Reemplazos a XML";
        exportButton.style.padding = "6px 10px";
        exportButton.addEventListener(
          "click", exportSharedDataToXml); // Llamar a la función compartida
        actionButtonsContainer.appendChild(exportButton);

        const clearButton = document.createElement("button");
        clearButton.textContent = "Limpiar Reemplazos";
        clearButton.title = "Limpiar solo la lista de reemplazos";
        clearButton.style.padding = "6px 10px";
        clearButton.addEventListener("click", () => {
            if (
              confirm(
                "¿Estás seguro de que deseas eliminar TODOS los reemplazos definidos?"))
            {
                replacementWords = {};
                saveReplacementWordsToStorage();
                renderReplacementsList(listElement);
            }
        });
        actionButtonsContainer.appendChild(clearButton);
        parentContainer.appendChild(actionButtonsContainer);

        const dropArea = document.createElement("div");
        dropArea.textContent =
          "Arrastra aquí el archivo XML (contiene Excluidas y Reemplazos)";
        // ... (estilos para dropArea como en createExcludedWordsManager) ...
        dropArea.style.border = "2px dashed #ccc";
        dropArea.style.borderRadius = "4px";
        dropArea.style.padding = "15px";
        dropArea.style.marginTop = "10px";
        dropArea.style.textAlign = "center";
        dropArea.style.background = "#f9f9f9";
        dropArea.style.color = "#555";

        dropArea.addEventListener("dragover", (e) => {
            e.preventDefault();
            dropArea.style.background = "#e9e9e9";
        });
        dropArea.addEventListener("dragleave",
                                  () => { dropArea.style.background = "#f9f9f9"; });
        dropArea.addEventListener("drop", (e) => {
            e.preventDefault();
            dropArea.style.background = "#f9f9f9";
            handleXmlFileDrop(
              e.dataTransfer.files[0]); // Usar una función manejadora compartida
        });
        parentContainer.appendChild(dropArea);

        // Cargar y renderizar la lista inicial
        renderReplacementsList(listElement);
    }

    // === Renderizar lista de palabras excluidas ===
    function renderExcludedWordsList(ulElement, filter = "")
    { // AHORA RECIBE ulElement
        if (!ulElement)
        {
            // Intentar obtenerlo por ID como último recurso si no se pasó,
            // pero idealmente siempre se pasa desde el llamador.
            ulElement = document.getElementById("excludedWordsList");
            if (!ulElement)
            {
                //console.error("[WME PLN] Contenedor 'excludedWordsList' no encontrado para renderizar.");
                return;
            }
        }

        const currentFilter = filter.toLowerCase();
        /*   console.log("[WME PLN] Renderizando lista. Filtro:",
                       currentFilter,
                       "Total palabras:",
                       excludedWords.size);*/
        ulElement.innerHTML = ""; // Limpiar lista anterior

        const wordsToRender =
          Array.from(excludedWords)
            .filter(word => word.toLowerCase().includes(currentFilter))
            .sort((a, b) => a.toLowerCase().localeCompare(
                    b.toLowerCase())); // Ordenar alfabéticamente

        if (wordsToRender.length === 0)
        {
            const li = document.createElement("li");
            li.style.padding = "5px";
            li.style.textAlign = "center";
            li.style.color = "#777";
            if (excludedWords.size === 0)
            {
                li.textContent = "La lista está vacía.";
            }
            else if (currentFilter !== "")
            {
                li.textContent = "No hay coincidencias para el filtro.";
            }
            else
            {
                li.textContent =
                  "La lista está vacía (o error inesperado)."; // Fallback
            }
            ulElement.appendChild(li);
        }
        else
        {
            wordsToRender.forEach(word => {
                const li = document.createElement("li");
                li.style.display = "flex";
                li.style.justifyContent = "space-between";
                li.style.alignItems = "center";
                li.style.padding = "4px 2px"; // Ajuste
                li.style.borderBottom = "1px solid #f0f0f0";

                const wordSpan = document.createElement("span");
                wordSpan.textContent = word;
                wordSpan.style.maxWidth =
                  "calc(100% - 60px)"; // Dejar espacio para botones
                wordSpan.style.overflow = "hidden";
                wordSpan.style.textOverflow = "ellipsis";
                wordSpan.style.whiteSpace = "nowrap";
                wordSpan.title = word;
                li.appendChild(wordSpan);

                const iconContainer = document.createElement("span");
                iconContainer.style.display = "flex";
                iconContainer.style.gap = "8px"; // Más espacio entre iconos

                const editBtn = document.createElement("button");
                editBtn.innerHTML = "✏️";
                editBtn.title = "Editar";
                editBtn.style.border = "none";
                editBtn.style.background = "transparent";
                editBtn.style.cursor = "pointer";
                editBtn.style.padding = "2px";
                editBtn.style.fontSize = "14px"; // Iconos un poco más grandes
                editBtn.addEventListener("click", () => {
                    const newWord = prompt("Editar palabra:", word);
                    if (newWord !== null && newWord.trim() !== word)
                    { // Permitir string vacío para borrar si se quisiera, pero
                        // trim() lo evita
                        const trimmedNewWord = newWord.trim();
                        if (trimmedNewWord === "")
                        {
                            alert("La palabra no puede estar vacía.");
                            return;
                        }
                        if (excludedWords.has(trimmedNewWord) &&
                            trimmedNewWord !== word)
                        {
                            alert("Esa palabra ya existe en la lista.");
                            return;
                        }
                        excludedWords.delete(word);
                        excludedWords.add(trimmedNewWord);
                        renderExcludedWordsList(ulElement, currentFilter);
                    }
                });

                const deleteBtn = document.createElement("button");
                deleteBtn.innerHTML = "🗑️";
                deleteBtn.title = "Eliminar";
                deleteBtn.style.border = "none";
                deleteBtn.style.background = "transparent";
                deleteBtn.style.cursor = "pointer";
                deleteBtn.style.padding = "2px";
                deleteBtn.style.fontSize = "14px";
                deleteBtn.addEventListener("click", () => {
                    if (confirm(`¿Estás seguro de que deseas eliminar la palabra '${
                          word}'?`))
                    {
                        excludedWords.delete(word);
                        renderExcludedWordsList(ulElement, currentFilter);
                    }
                });

                iconContainer.appendChild(editBtn);
                iconContainer.appendChild(deleteBtn);
                li.appendChild(iconContainer);
                ulElement.appendChild(li);
            });
        }

        try
        {
            localStorage.setItem("excludedWordsList",
                                 JSON.stringify(Array.from(excludedWords)));
            // console.log("[WME PLN] Lista guardada en localStorage:",
            // Array.from(excludedWords));
        }
        catch (e)
        {
            console.error("[WME PLN] Error guardando en localStorage:", e);
            // Considerar no alertar cada vez para no ser molesto si el localStorage
            // está lleno. Podría ser un mensaje en consola o una notificación sutil
            // en la UI.
        }
    }

    // Nueva función: renderDictionaryList
    function renderDictionaryList(ulElement, filter = "")
    {
        if (!ulElement || !window.dictionaryWords)
            return;
        const currentFilter = filter.toLowerCase();
        ulElement.innerHTML = "";

        const wordsToRender =
          Array.from(window.dictionaryWords)
            .filter(word => word.toLowerCase().startsWith(currentFilter))
            .sort((a, b) => a.toLowerCase().localeCompare(b.toLowerCase()));

        if (wordsToRender.length === 0)
        {
            const li = document.createElement("li");
            li.textContent = window.dictionaryWords.size === 0
                               ? "El diccionario está vacío."
                               : "No hay coincidencias.";
            li.style.textAlign = "center";
            li.style.color = "#777";
            ulElement.appendChild(li);
            // Guardar diccionario también cuando está vacío
            try
            {
                localStorage.setItem(
                  "dictionaryWordsList",
                  JSON.stringify(Array.from(window.dictionaryWords)));
            }
            catch (e)
            {
                console.error( "[WME PLN] Error guardando el diccionario en localStorage:", e);
            }
            return;
        }

        wordsToRender.forEach(word => {
            const li = document.createElement("li");
            li.style.display = "flex";
            li.style.justifyContent = "space-between";
            li.style.alignItems = "center";
            li.style.padding = "4px 2px";
            li.style.borderBottom = "1px solid #f0f0f0";

            const wordSpan = document.createElement("span");
            wordSpan.textContent = word;
            wordSpan.style.maxWidth = "calc(100% - 60px)";
            wordSpan.style.overflow = "hidden";
            wordSpan.style.textOverflow = "ellipsis";
            wordSpan.style.whiteSpace = "nowrap";
            wordSpan.title = word;
            li.appendChild(wordSpan);

            const iconContainer = document.createElement("span");
            iconContainer.style.display = "flex";
            iconContainer.style.gap = "8px";

            const editBtn = document.createElement("button");
            editBtn.innerHTML = "✏️";
            editBtn.title = "Editar";
            editBtn.style.border = "none";
            editBtn.style.background = "transparent";
            editBtn.style.cursor = "pointer";
            editBtn.style.padding = "2px";
            editBtn.style.fontSize = "14px";
            editBtn.addEventListener("click", () => {
                const newWord = prompt("Editar palabra:", word);
                if (newWord !== null && newWord.trim() !== word)
                {
                    window.dictionaryWords.delete(word);
                    window.dictionaryWords.add(newWord.trim());
                    renderDictionaryList(ulElement, currentFilter);
                }
            });

            const deleteBtn = document.createElement("button");
            deleteBtn.innerHTML = "🗑️";
            deleteBtn.title = "Eliminar";
            deleteBtn.style.border = "none";
            deleteBtn.style.background = "transparent";
            deleteBtn.style.cursor = "pointer";
            deleteBtn.style.padding = "2px";
            deleteBtn.style.fontSize = "14px";
            deleteBtn.addEventListener("click", () => {
                if (confirm(`¿Eliminar la palabra '${word}' del diccionario?`))
                {
                    window.dictionaryWords.delete(word);
                    renderDictionaryList(ulElement, currentFilter);
                }
            });

            iconContainer.appendChild(editBtn);
            iconContainer.appendChild(deleteBtn);
            li.appendChild(iconContainer);
            ulElement.appendChild(li);
        });
        // Guardar el diccionario actualizado en localStorage después de cada
        // render
        try
        {
            localStorage.setItem(
              "dictionaryWordsList",
              JSON.stringify(Array.from(window.dictionaryWords)));
        }
        catch (e)
        {
            console.error("[WME PLN] Error guardando el diccionario en localStorage:", e);
        }
    }

    function exportExcludedWordsList()
    {
        if (excludedWords.size === 0 && Object.keys(replacementWords).length === 0)
        {
            alert("No hay palabras especiales ni reemplazos para exportar.");
            return;
        }
        let xmlContent =
          `<?xml version="1.0" encoding="UTF-8"?>\n<ExcludedWords>\n`;
        xmlContent +=
          Array.from(excludedWords)
            .sort((a, b) => a.toLowerCase().localeCompare(b.toLowerCase()))
            .map(w => `    <word>${xmlEscape(w)}</word>`)
            .join("\n");
        if (Object.keys(replacementWords).length > 0)
        {
            xmlContent += "\n";
            xmlContent +=
              Object.entries(replacementWords)
                .map(([ from, to ]) => `    <replacement from="${
                       xmlEscape(from)}">${xmlEscape(to)}</replacement>`)
                .join("\n");
        }
        xmlContent += "\n</ExcludedWords>";
        const blob =
          new Blob([ xmlContent ], { type : "application/xml;charset=utf-8" });
        const url = URL.createObjectURL(blob);
        const a = document.createElement("a");
        a.href = url;
        a.download = "wme_excluded_words_export.xml";
        document.body.appendChild(a);
        a.click();
        document.body.removeChild(a);
        URL.revokeObjectURL(url);
    }

    function exportDictionaryWordsList()
    {
        if (window.dictionaryWords.size === 0)
        {
            alert(
              "La lista de palabras del diccionario está vacía. Nada que exportar.");
            return;
        }
        const xmlContent =
          `<?xml version="1.0" encoding="UTF-8"?>\n<diccionario>\n${
            Array.from(window.dictionaryWords)
              .sort((a, b) => a.toLowerCase().localeCompare(
                      b.toLowerCase()))                     // Exportar ordenado
              .map(w => `    <word>${xmlEscape(w)}</word>`) // Indentación y escape
              .join("\n")}\n</diccionario>`;
        const blob =
          new Blob([ xmlContent ],
                   { type : "application/xml;charset=utf-8" }); // Añadir charset
        const url = URL.createObjectURL(blob);
        const a = document.createElement("a");
        a.href = url;
        a.download = "wme_dictionary_words_export.xml"; // Nombre más descriptivo
        document.body.appendChild(a);
        a.click();
        document.body.removeChild(a);
        URL.revokeObjectURL(url);
    }

    function xmlEscape(str)
    {
        return str.replace(/[<>&"']/g, function(match) {
            switch (match)
            {
                case '<':
                    return '&lt;';
                case '>':
                    return '&gt;';
                case '&':
                    return '&amp;';
                case '"':
                    return '&quot;';
                case "'":
                    return '&apos;';
                default:
                    return match;
            }
        });
    }

    waitForSidebarAPI();
    })();

    // Función reutilizable para mostrar el spinner de carga
    function showLoadingSpinner()
    {
        const scanSpinner = document.createElement("div");
        scanSpinner.id = "scanSpinnerOverlay";
        scanSpinner.style.position = "fixed";
        scanSpinner.style.top = "0";
        scanSpinner.style.left = "0";
        scanSpinner.style.width = "100%";
        scanSpinner.style.height = "100%";
        scanSpinner.style.background = "rgba(0, 0, 0, 0.5)";
        scanSpinner.style.zIndex = "10000";
        scanSpinner.style.display = "flex";
        scanSpinner.style.justifyContent = "center";
        scanSpinner.style.alignItems = "center";

        const scanContent = document.createElement("div");
        scanContent.style.background = "#fff";
        scanContent.style.padding = "20px";
        scanContent.style.borderRadius = "8px";
        scanContent.style.textAlign = "center";

        const spinner = document.createElement("div");
        spinner.classList.add("spinner");
        spinner.style.border = "6px solid #f3f3f3";
        spinner.style.borderTop = "6px solid #3498db";
        spinner.style.borderRadius = "50%";
        spinner.style.width = "40px";
        spinner.style.height = "40px";
        spinner.style.animation = "spin 1s linear infinite";
        spinner.style.margin = "0 auto 10px auto";

        const progressText = document.createElement("div");
        progressText.id = "scanProgressText";
        progressText.textContent = "Analizando lugares: 0%";
        progressText.style.fontSize = "14px";
        progressText.style.color = "#333";

        scanContent.appendChild(spinner);
        scanContent.appendChild(progressText);
        scanSpinner.appendChild(scanContent);
        document.body.appendChild(scanSpinner);

        const style = document.createElement("style");
        style.textContent = `
                        @keyframes spin {
                        0% { transform: rotate(0deg); }
                        100% { transform: rotate(360deg); }
                        }
                    `;
        document.head.appendChild(style);
    }

QingJ © 2025

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