WME Places Name Normalizer

Normaliza nombres de lugares en Waze Map Editor (WME)

目前为 2025-05-21 提交的版本,查看 最新版本

// ==UserScript==
// @name         WME Places Name Normalizer
// @namespace    https://gf.qytechs.cn/en/users/mincho77
// @version      6.0.1
// @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();

    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'
    ];

    let wmeSDK = null; // Declarar wmeSDK global

    function tryInitializeSDK(finalCallback)
    {
        let attempts = 0;
        const maxAttempts = 60; // Aumentado a 30 segundos (60 * 500ms)
        const intervalTime = 500;
        let sdkAttemptInterval = null; // Guardar la referencia al intervalo

        // console.log(
        //   "[SDK INIT ATTEMPT] Iniciando intentos para obtener getWmeSdk().");

        function attempt()
        {
            //  //console.log(`[SDK INIT ATTEMPT] Intento #${attempts + 1}`);
            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(); // Llama al callback final
                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(); // Llama al callback final igualmente
            }
            // Si no se encontró y no hemos llegado al máximo de intentos, el
            // intervalo seguirá.
        }

        // Iniciar el intervalo para los reintentos.
        sdkAttemptInterval = setInterval(attempt, intervalTime);
        // Hacer una primera llamada inmediata por si ya está disponible.
        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.");

            // Ahora que W está listo, intentamos inicializar el nuevo SDK,
            // y LUEGO llamamos al callbackPrincipalDelScript (que es
            // createSidebarTab).
            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)
{
    // Asegurarse de que totalPlaces no sea 0 para evitar división por cero
    if (totalPlaces === 0)
    {
        return;
    }
    // El currentIndex es 0-basado, así que (currentIndex + 1) es el número deitems procesados
    const itemsProcessed = currentIndex + 1;
    let progressPercent = Math.floor((itemsProcessed / totalPlaces) * 100);
    progressPercent = Math.min(progressPercent, 100); // No exceder el 100%

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

    if (progressBarInnerTab && progressBarTextTab)
    {
        progressBarInnerTab.style.width = `${progressPercent}%`;
        progressBarTextTab.textContent =
          `Progreso: ${progressPercent}% (${itemsProcessed}/${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 aplicarReemplazosGenerales(name)
{
    if (typeof window.skipGeneralReplacements === "boolean" &&
        window.skipGeneralReplacements)
    {
        return name;
    }
    const reglas = [
        // 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()}`);

    const frasesAlFinal = [
        "Conjunto Residencial",
        "Urbanización",
        "Conjunto Cerrado",
        "Unidad Residencial",
        "Parcelación",
        "Condominio Campestre",
        "Condominio",
        "Edificio",
        "Conjunto Habitacional",
        "Apartamentos",
        "Club Campestre",
        "Club Residencial"
    ];

    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 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;
}

function renderPlacesInFloatingPanel(places)
{
    createFloatingPanel(
      0); // Asegurar que el panel flotante exista antes de renderizar
    const maxPlacesToScan =
      parseInt(document.getElementById("maxPlacesInput")?.value || "100", 10);
    if (places.length > maxPlacesToScan)
    {
        places = places.slice(0, maxPlacesToScan);
    }

    //  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

        // Intento 1: Usar el venueSDKObject si está disponible y tiene la info
        if (venueSDKObject)
        {
            // HIPÓTESIS: Basado en la imagen del SDK, 'updatedBy' podría ser un
            // string. Necesitamos ver cómo el SDK estructura las categorías.
            // Suposición 1: El SDK tiene una propiedad 'mainCategory' con 'id'
            // y 'name'
            if (venueSDKObject.mainCategory && venueSDKObject.mainCategory.id)
            {
                categoryId = venueSDKObject.mainCategory.id;
                if (venueSDKObject.mainCategory.name)
                {
                    categoryName = venueSDKObject.mainCategory.name;
                    source = "SDK (mainCategory.name)";
                }
            }
            // Suposición 2: El SDK tiene un array 'categories', y el primer
            // elemento tiene 'id' y '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)";
                    }
                }
                // Suposición 3: El primer elemento del array 'categories' es
                // directamente el nombre
                else if (typeof firstCategorySDK === 'string')
                {
                    categoryName = firstCategorySDK;
                    source = "SDK (categories[0] as string)";
                    // No tendríamos ID en este caso desde esta fuente directa.
                }
            }
            // Suposición 4: El SDK tiene solo el ID de la categoría principal
            // en una propiedad simple
            else if (venueSDKObject.primaryCategoryID)
            { // Nombre de propiedad hipotético
                categoryId = venueSDKObject.primaryCategoryID;
                source = "SDK (primaryCategoryID)";
            }
            // (Añade más suposiciones aquí basándote en lo que veas en
            // console.dir(venueSDKObject))
        }

        // Si tenemos un nombre de categoría del SDK, lo usamos.
        if (categoryName)
        {
            //  console.log(`[CATEGORÍA] Usando nombre de categoría de
            //  ${source}: ${
            //     categoryName} ${categoryId ? `(ID: ${categoryId})` : ''}`);
            return categoryName;
        }

        // Intento 2: Si no obtuvimos nombre del SDK pero sí un ID, o si no
        // usamos el SDK, usamos W.model Primero, obtener el categoryId si aún
        // no lo tenemos (del venueFromOldModel)
        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";
        }

        // Ahora, resolver el categoryId (sea del SDK o de W.model) usando
        // W.model.categories
        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}`);
            return categoryObjWModel.attributes.name;
        }

        // Fallback final: si solo tenemos el ID y no pudimos resolver el nombre
        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.`);*/
            return `${categoryId}`;
        }

        return "Sin categoría";
    }


    function getPlaceTypeInfo(venue)
    {
        const geometry = venue?.getOLGeometry ? venue.getOLGeometry() : null;
        const isArea = geometry?.CLASS_NAME?.endsWith("Polygon");
        return {
            isArea,
            icon : isArea ? "⭔" : "⊙",
            title : isArea ? "Área" : "Punto"
        };
    }

    // --- 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)"; // Altura para el panel flotante

    // 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 hideMyEdits = false;
        const useFullPipeline =
          true; // Como definimos, siempre ejecutar el flujo completo
        const applyGeneralReplacements =
          useFullPipeline ||
          (document.getElementById("chk-general-replacements")?.checked ??
           true); // Asume que este ID podría existir si reañades el check
        const checkExcludedWords =
          useFullPipeline ||
          (document.getElementById("chk-check-excluded")?.checked ?? false);
        const checkDictionaryWords =
          true; // O leer de checkbox si lo reintroduces
        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;

        // 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 = {};
        }

        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 = cleaned;
            const isArticle = commonWords.includes(cleaned.toLowerCase());

            if (isExcluded)
            {
                tempReplaced = matchingExcludedWord;
            }
            else
            {
                if (isArticle)
                {
                    tempReplaced = cleaned.toLowerCase();
                    if (idx === 0 && tempReplaced.length > 0)
                    {
                        tempReplaced = tempReplaced.charAt(0).toUpperCase() +
                                       tempReplaced.slice(1);
                    }
                }
                else
                {
                    tempReplaced = normalizePlaceName(cleaned);
                }
            }

            if (!isExcluded && checkDictionaryWords && window.dictionaryWords &&
                typeof window.dictionaryWords.forEach === "function")
            {
                window.dictionaryWords.forEach(diccWord => {
                    const cleanTemp =
                      removeDiacritics(tempReplaced.toLowerCase());
                    const cleanDicc = removeDiacritics(diccWord.toLowerCase());
                    if (cleanTemp.charAt(0) !== cleanDicc.charAt(0))
                        return;
                    const sim = calculateSimilarity(cleanTemp, cleanDicc);
                    if (sim >= similarityThreshold)
                    { // Usar el umbral del slider
                        if (!sugerenciasLugar[baseWord])
                            sugerenciasLugar[baseWord] = [];
                        if (!sugerenciasLugar[baseWord].some(
                              s => s.word.toLowerCase() ===
                                   diccWord.toLowerCase()))
                        {
                            sugerenciasLugar[baseWord].push({
                                word : diccWord,
                                similarity : sim,
                                fuente : 'dictionary'
                            });
                        }
                    }
                });
            }
            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(' ');
        const cleanedSuggestedForLog =
          joinedSuggested.replace(/\s{2,}/g, ' ').trim();

        let processedName = joinedSuggested;
        if (applyGeneralReplacements)
        {
            processedName = aplicarReemplazosGenerales(processedName);
        }
        processedName = aplicarReglasEspecialesNombre(processedName);
        const suggestedName = processedName.replace(/\s{2,}/g, ' ').trim();
        // ---- FIN COMPILACIÓN DE 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}".`;
        }
        else
        {
            const normalizedCleanCheck = suggestedName.toLowerCase();
            const originalCleanCheck =
              originalName.replace(/\s+/g, ' ').trim().toLowerCase();
            const isAlreadyNormalizedCheck =
              normalizedCleanCheck === originalCleanCheck;

            if (isAlreadyNormalizedCheck && !tieneSugerencias)
            {
                shouldSkipThisPlace = true;
                skipReasonLog = `[SKIP] Se descartó "${
                  originalName}" porque ya se considera normalizado (${
                  originalCleanCheck} vs ${
                  normalizedCleanCheck}) y sin sugerencias.`;
            }
            else if (hideMyEdits && wasEditedByMe)
            {
                shouldSkipThisPlace = true;
                skipReasonLog = `[SKIP MY EDIT] Descartado "${
                  originalName}" (ID: ${
                  currentVenueId}) porque fue editado por mí y la opción está activa.`;
            }
        }
        // ---- 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 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,
                normalized : suggestedName,
                category : categoryNameToStore,
                editor : resolvedEditorName // Add the resolved editor name here
            });
            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(0); // Mostrar panel aunque esté vacío
            const output =
              document.querySelector("#wme-place-inspector-output");
            if (output)
            {
                output.innerHTML =
                  "El proceso tomó demasiado tiempo o se interrumpió. No se completó el análisis.";
            }
        }
    }, 30000);
    // 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;
        }
        if (output)
        {
            output.innerHTML =
              ""; // Limpiar el mensaje de procesamiento y spinner
            // Mostrar el panel flotante al terminar el procesamiento
            createFloatingPanel(inconsistents.length);
        }
        // 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 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 = "20px";
                modal.style.zIndex = "10000";
                modal.style.width = "400px";
                modal.style.boxShadow = "0 0 15px rgba(0,0,0,0.3)";
                modal.style.borderRadius = "8px";
                modal.style.fontFamily = "sans-serif";

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

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

                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";
                modal.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");
                    modal.remove();
                });

                modal.appendChild(acceptBtn);
                document.body.appendChild(modal);
            }
        }
        // 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";
        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 originalInput = document.createElement("input");
            originalInput.type = "text";
            const venueLive = W.model.venues.getObjectById(id);
            const currentLiveName = venueLive?.attributes?.name?.value ||
                                    venueLive?.attributes?.name || "";
            originalInput.value = currentLiveName;
            // --- Resaltar en rojo si hay diferencia con el sugerido ---
            if (currentLiveName.trim().toLowerCase() !==
                normalized.trim().toLowerCase())
            {
                originalInput.style.border = "1px solid red";
                originalInput.title =
                  "Este nombre difiere del original mostrado en el panel";
            }
            originalInput.disabled = true;
            originalInput.style.width = "270px";
            originalInput.style.backgroundColor = "#eee";
            originalCell.appendChild(originalInput);
            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 suggestionInput = document.createElement("input");
            suggestionInput.type = "text";
            let mainSuggestion = localNormalized;
            suggestionInput.value = mainSuggestion;
            suggestionInput.style.width = "270px";
            // Visual cue if change was due to excluded word
            if (localNormalized !== normalized)
            {
                suggestionInput.style.backgroundColor =
                  "#fff3cd"; // color amarillo claro
                suggestionInput.title = "Contiene palabra excluida reemplazada";
            }
            else if (autoApplied)
            {
                suggestionInput.style.backgroundColor =
                  "#c8e6c9"; // verde claro
                suggestionInput.title =
                  "Reemplazo automático aplicado (100% similitud)";
            }
            else if (normalized !== suggestionInput.value)
            {
                suggestionInput.style.backgroundColor =
                  "#e6f7ff"; // Azul claro para cambios automáticos del
                             // diccionario (≥ 90%)
                suggestionInput.title =
                  "Cambio automático basado en diccionario (≥ 90%)";
            }
            else
            {
                suggestionInput.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)
            {
                suggestionInput.style.backgroundColor = "#cce5ff"; // Azul claro
                suggestionInput.title =
                  "Contiene sugerencias del diccionario aplicadas manualmente";
            }
            suggestionCell.appendChild(suggestionInput);
            suggestionCell.style.padding = "4px";
            suggestionCell.style.width = "270px";

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

            // 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 => {
                        if (s.similarity < 1)
                        {
                            const suggestionDiv = document.createElement("div");
                            suggestionDiv.textContent =
                              `🏷️ ¿"${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";
                            suggestionDiv.addEventListener("click", () => {
                                // Escapar originalWord para asegurar que la
                                // regex funcione correctamente
                                const escapedOriginalWord = escapeRegExp(originalWord);
                                const regex = new RegExp(
                                  "\\b" + escapedOriginalWord + "\\b", "gi");

                                const replacedText =
                                  suggestionInput.value.replace(
                                    regex,
                                    s.word); // s.word es de la lista de
                                             // excluidas, se usa tal cual
                                suggestionInput.value = replacedText;
                                suggestionInput.dispatchEvent(
                                  new Event("input"));
                            });
                            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 ]) => {
                      if (palabrasYaProcesadas.has(originalWord.toLowerCase()))
                          return;
                      suggestions.forEach(s => {
                          if (s.similarity < 1 ||
                              (s.similarity === 1 && originalWord !== s.word))
                          {
                              const suggestionItem =
                                document.createElement("div");
                              const icono = s.fuente === 'dictionary' ? "📘"
                                            : s.fuente === 'excluded' ? "🏷️"
                                                                      : "❓";
                              suggestionItem.textContent = `${icono} ¿"${
                                originalWord}" por "${s.word}"? (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", () => {
                                  // Escapar originalWord para asegurar que la
                                  // regex funcione correctamente
                                  const escapedOriginalWord =
                                    escapeRegExp(originalWord);
                                  const regex = new RegExp(
                                    "\\b" + escapedOriginalWord + "\\b", "gi");

                                  const normalizedWordFromDictionary =
                                    normalizePlaceName(s.word);

                                  const finalReplacedText =
                                    suggestionInput.value.replace(
                                      regex, normalizedWordFromDictionary);

                                  suggestionInput.value = finalReplacedText;
                                  suggestionInput.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 = suggestionInput.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);
                }
            });
            deleteButton.addEventListener("click", () => {
                const confirmModal = document.createElement("div");
                confirmModal.style.position = "fixed";
                confirmModal.style.top = "50%";
                confirmModal.style.left = "50%";
                confirmModal.style.transform = "translate(-50%, -50%)";
                confirmModal.style.background = "#fff";
                confirmModal.style.border = "1px solid #ccc";
                confirmModal.style.padding = "20px";
                confirmModal.style.zIndex = "10001";
                confirmModal.style.boxShadow = "0 2px 10px rgba(0,0,0,0.3)";
                confirmModal.style.fontFamily = "sans-serif";
                confirmModal.style.borderRadius = "6px";

                const message = document.createElement("p");
                const venue = W.model.venues.getObjectById(id);
                const placeName =
                  venue?.attributes?.name?.value || "este lugar";
                message.textContent =
                  `¿Estás seguro que deseas eliminar "${placeName}" del mapa?`;
                confirmModal.appendChild(message);

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

                const cancelBtn = document.createElement("button");
                cancelBtn.textContent = "Cancelar";
                cancelBtn.style.padding = "4px 8px";
                cancelBtn.addEventListener("click",
                                           () => { confirmModal.remove(); });

                const confirmBtn = document.createElement("button");
                confirmBtn.textContent = "Eliminar";
                confirmBtn.style.padding = "4px 8px";
                confirmBtn.style.backgroundColor = "#d9534f";
                confirmBtn.style.color = "#fff";
                confirmBtn.style.border = "none";
                confirmBtn.style.borderRadius = "4px";
                confirmBtn.addEventListener("click", () => {
                    confirmModal.remove();
                    const venue = W.model.venues.getObjectById(id);
                    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 statusIcon = document.createElement("span");
                        statusIcon.textContent = " ❌";
                        statusIcon.style.marginLeft = "5px";
                        statusIcon.style.color = "red";
                        deleteButton.parentElement.appendChild(statusIcon);
                        if (deleteButton.relatedApply)
                        {
                            deleteButton.relatedApply.textContent = "✔";
                        }
                    }
                    catch (e)
                    {
                        alert("Error al borrar: " + e.message);
                    }
                });

                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 = "10000";
                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");
                    const newWords = [];
                    checked.forEach(c => {
                        excludedWords.add(c.value);
                        newWords.push(c.value);
                        // Ya no se modifica automáticamente el campo de
                        // sugerencia aquí
                    });
                    renderExcludedWordsList(); // Actualiza la lista en el tab

                    // Actualizar estado de botón aplicar según diferencia
                    if (suggestionInput.value !== original)
                    {
                        applyButton.disabled = false;
                        applyButton.style.color = "";
                    }
                    else
                    {
                        applyButton.disabled = true;
                        applyButton.style.color = "#bbb";
                    }

                    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.";
}

function createFloatingPanel(numInconsistents = 0)
{
    const existing = document.querySelector("#wme-place-inspector-panel");
    if (existing)
    {

        existing.style.display = 'block';
        return;
    }

    const panel = document.createElement("div");
    const closeBtn = document.createElement("span");
    closeBtn.textContent = "×";
    closeBtn.style.position = "absolute";
    closeBtn.style.top = "5px";
    closeBtn.style.right = "8px";
    closeBtn.style.cursor = "pointer";
    closeBtn.style.fontSize = "16px";
    closeBtn.style.color = "#aaa";
    closeBtn.title = "Cerrar panel";
    closeBtn.addEventListener("click", () => {
        panel.style.display = 'none'; // Ocultar el panel
        // Restaurar estado del botón escanear
        const scanBtn = document.querySelector("button[type='button']");
        if (scanBtn)
        {
            scanBtn.textContent = "Start Scan...";
            scanBtn.disabled = false;
            scanBtn.style.opacity = "1";
            scanBtn.style.cursor = "pointer";
            const iconCheck = scanBtn.querySelector("span");
            if (iconCheck)
            {
                iconCheck.remove(); // Remover icono de check si quedó
            }
        }
        // Restaurar el mensaje en el tab
        const outputTab =
          document.getElementById("wme-normalization-tab-output");
        if (outputTab)
        {
            outputTab.textContent =
              "Presiona 'Start Scan...' para analizar los Lugares visibles.";
            outputTab.style.color = "#000";
            outputTab.style.fontWeight = "normal";
        }
        // Resetear barra de progreso/tab
        resetInspectorState();
    });
    panel.appendChild(closeBtn);
    panel.id = "wme-place-inspector-panel";
    panel.style.position = "fixed";
    panel.style.top = "55%";
    panel.style.left = "70%";
    panel.style.transform = "translate(-70%, -55%)";
    panel.style.width = "1500px";
    panel.style.height = "60vh";
    panel.style.overflow =
      "hidden"; // El scroll interno lo maneja #wme-place-inspector-output
    panel.style.zIndex = "9999";
    panel.style.background = "#fff";
    panel.style.border = "1px solid #ccc";
    panel.style.borderRadius = "6px";
    panel.style.boxShadow = "0 2px 6px rgba(0,0,0,0.2)";
    panel.style.padding = "10px";
    panel.style.fontFamily = "sans-serif";
    panel.style.display = 'block'; // Asegurar que esté visible al crear

    const title = document.createElement("h4");
    title.textContent =
      numInconsistents > 0
        ? `Lugares con Nombres No Normalizados (${numInconsistents})`
        : "Resultado del Análisis";
    title.style.marginTop = "0";
    title.style.marginBottom = "10px"; // Espacio antes de la tabla
    panel.appendChild(title);

    const output = document.createElement("div");
    output.id = "wme-place-inspector-output"; // Este es el output para la tabla
                                              // de lugares
    output.style.height = "calc(100% - 40px)";
    output.style.overflowY = "auto";
    output.style.border = "1px solid #eee";
    output.style.padding = "6px";
    output.style.fontSize = "12px";
    output.style.backgroundColor = "#f9f9f9";

    panel.appendChild(output);
    document.body.appendChild(panel);

    if (!document.getElementById("wme-pln-spinner-style"))
    {
        const styleTag = document.createElement("style");
        styleTag.id = "wme-pln-spinner-style";
        styleTag.textContent = `
                        @keyframes spin {
                          0% { transform: rotate(0deg); }
                          100% { transform: rotate(360deg); }
                        }
                    `;
        document.head.appendChild(styleTag);
    }
}

// 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
    {
        if (!W || !W.userscripts)
        {
            console.error("[WME PLN] WME not ready for sidebar creation");
            return;
        }

        let currentWMEUsername = "Pendiente"; // Valor por defecto
        if (typeof W !== 'undefined' && W.loginManager && W.loginManager.user &&
            W.loginManager.user.userName)
        {
            currentWMEUsername = W.loginManager.user.userName;
        }
        else
        {
            console.warn(
              "[CURRENT USER] No se pudo obtener el userName del usuario actual desde W.loginManager.");
        }



        let registration;
        try
        {
            registration =
              W.userscripts.registerSidebarTab("NrmliZer"); // Nombre del Tab
        }
        catch (e)
        {
            if (e.message.includes("already been registered"))
            {
                console.warn(
                  "[WME PLN] Tab 'NrmliZer' ya registrado. Intentando reusar o puede haber conflicto.");
                // Intenta obtener el tabPane si ya existe
                const existingTab = document.querySelector(
                  '.user-tabs div[data-tab-id="WME_PLN_NrmliZer_TAB"]'); // Asume
                                                                         // un
                                                                         // data-tab-id
                if (existingTab && existingTab.tabPane)
                {
                    // Si el tabPane existe y está vacío
                    return;
                }
            }
            throw e;
        }

        const { tabLabel, tabPane } = registration;
        if (!tabLabel || !tabPane)
        {
            throw new Error(
              "[WME PLN] Falló el registro del Tab o no retornó los elementos esperados.");
        }

        tabLabel.innerHTML = `
                            <img src=""
                            style="height: 16px; vertical-align: middle; margin-right: 5px;">
                            NrmliZer
                        `;

        // --- SISTEMA DE PESTAÑAS ---
        const tabsContainer = document.createElement("div");
        tabsContainer.style.display = "flex";
        tabsContainer.style.marginBottom = "8px";
        tabsContainer.style.gap = "8px";

        const tabNames = [
            { label : "General", icon : "" },
            { label : "Especiales", icon : "🏷️" },
            { label : "Diccionario", icon : "📘" }
        ];
        const tabButtons = {};
        const tabContents = {};

        tabNames.forEach(({ label, icon }) => {
            const btn = document.createElement("button");
            btn.innerHTML =
              icon
                ? `<span style="display: inline-flex; align-items: center; font-size: 13px;"><span style="font-size: 12px; margin-right: 5px;">${
                    icon}</span>${label}</span>`
                : `<span style="font-size: 13px;">${label}</span>`;
            btn.style.padding = "6px 12px";
            btn.style.border = "1px solid #ccc";
            btn.style.borderRadius = "4px 4px 0 0";
            btn.style.cursor = "pointer";
            btn.style.backgroundColor = label === "General" ? "#fff" : "#eee";
            btn.addEventListener("click", () => {
                tabNames.forEach(({ label : tabLabel }) => {
                    const isActive = (tabLabel === label);
                    tabContents[tabLabel].style.display =
                      isActive ? "block" : "none";
                    tabButtons[tabLabel].style.backgroundColor =
                      isActive ? "#fff" : "#eee";

                    if (isActive && tabLabel === "Especiales")
                    {
                        const ul = document.getElementById("excludedWordsList");
                        if (ul)
                            renderExcludedWordsList(ul);
                    }
                });
            });
            tabButtons[label] = btn;
            tabsContainer.appendChild(btn);
        });
        tabPane.appendChild(tabsContainer);

        tabNames.forEach(({ label }) => {
            const container = document.createElement("div");
            container.style.display = label === "General" ? "block" : "none";
            container.style.padding = "10px";
            tabContents[label] = container;
            tabPane.appendChild(container);
        });
        const container = tabContents["General"];


        const mainTitle =
          document.createElement("h3"); // Título principal del script
        mainTitle.textContent = "NormliZer";
        mainTitle.style.textAlign = "center";
        mainTitle.style.fontSize = "18px";
        mainTitle.style.marginBottom = "2px";
        container.appendChild(mainTitle);

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

        // --- Sección de Normalización ---
        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";
        container.appendChild(normSectionTitle);

        // --- LEYENDA DE ICONOS ---
        // (Eliminada por ser redundante)

        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", () => {
            // alert("🟢 Click en Scan");
            const places = getVisiblePlaces();
            //  alert("🟢 Lugares detectados: " + places.length); // ← punto 2
            //console.log("[WME PLN] Places visibles detectados:", places.length);
            if (places.length === 0)
            {
                alert("No se encontraron lugares visibles para analizar.");
                return;
            }
            const outputNormalizationDiv = document.getElementById(
              "wme-normalization-tab-output"); // Output en el tab
        /*    console.log("¿Existe outputNormalizationDiv?",
                        !!outputNormalizationDiv);*/
            if (!outputNormalizationDiv)
            {
                console.error(
                  "Div de salida para normalización no encontrado en el tab.");
                return;
            }

            if (places.length === 0)
            {
                outputNormalizationDiv.textContent =
                  "No se encontraron lugares visibles para escanear.";
                return;
            }
            // Almacenar correctamente los valores seleccionados antes de
            // renderizar
            const maxPlacesInput = document.getElementById("maxPlacesInput");
            const similaritySlider =
              document.getElementById("similarityThreshold");
            const maxPlacesToScan =
              parseInt(maxPlacesInput?.value || "100", 10);
            const similarityThreshold =
              parseFloat(similaritySlider?.value || "85") / 100;
            const scannedCount = Math.min(places.length, maxPlacesToScan);
            outputNormalizationDiv.textContent =
              `Escaneando ${scannedCount} lugares...`;

            setTimeout(() => {
               /* console.log(
                  "Llamando a render con", places.length, "lugares visibles");*/
                renderPlacesInFloatingPanel(places.slice(0, maxPlacesToScan));
            }, 10); // Ligeramente asincrónico para no bloquear render
        });

        // NUEVO BLOQUE: campo máximo de places y presets con estilos
        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);

        container.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",
                                 () => { maxInput.value = preset.toString(); });
            presetContainer.appendChild(btn);
        });

        container.appendChild(presetContainer);

        // control de porcentaje de similitud
        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";
        container.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 similarityValue = document.createElement("span");
        similarityValue.textContent = "85%";
        similarityValue.style.marginLeft = "8px";
        similaritySlider.addEventListener("input", () => {
            similarityValue.textContent = similaritySlider.value + "%";
        });

        container.appendChild(similaritySlider);
        container.appendChild(similarityValue);
        //***********************************

        container.appendChild(scanButton);

        // Barra de progreso (ubicada en el tab, no en el flotante)
        const tabProgressWrapper = document.createElement("div");
        tabProgressWrapper.style.margin = "10px 0";
        tabProgressWrapper.style.height = "18px";
        // Eliminada la barra gris de fondo (no color de fondo)
        // tabProgressWrapper.style.backgroundColor = "#e0e0e0";
        // tabProgressWrapper.style.borderRadius = "10px";
        // tabProgressWrapper.style.overflow = "hidden";
        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"; // importante para ser encontrado
        tabProgressWrapper.appendChild(tabProgressBar);

        const tabProgressText = document.createElement("div");
        tabProgressText.style.fontSize = "12px";
        tabProgressText.style.marginTop = "5px";
        tabProgressText.id = "progressBarTextTab";

        container.appendChild(tabProgressWrapper);
        container.appendChild(tabProgressText);

        const outputNormalizationInTab = document.createElement("div");
        outputNormalizationInTab.id = "wme-normalization-tab-output";
        outputNormalizationInTab.style.fontSize = "12px";
        outputNormalizationInTab.style.minHeight =
          "20px"; // Para que ocupe espacio incluso si está vacío
        outputNormalizationInTab.style.padding = "5px";
        // outputNormalizationInTab.style.border = "1px dashed #eee";
        outputNormalizationInTab.style.marginBottom = "15px";
        outputNormalizationInTab.textContent =
          "Presiona 'Escanear' para analizar los places visibles.";
        container.appendChild(outputNormalizationInTab);

        // --- Sección de Palabras especiales ---
        // createExcludedWordsManager añadirá su contenido a la pestaña
        // "especiales"
        createExcludedWordsManager(tabContents["Especiales"]);

        // --- Gestión Diccionario ---
        createDictionaryManager(tabContents["Diccionario"]);
    }
    catch (error)
    {
        console.error("[WME PLN] Error creando el tab:", error);
    }
}

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.");
        }
        waitForWazeAPI(() => { createSidebarTab(); });
    }
    else
    {
        // console.log("[WME PLN] Esperando W.userscripts API...");
        setTimeout(waitForSidebarAPI, 1000);
    }
}

// ---- COPIA DESDE AQUÍ ----
function normalizePlaceName(word)
{
    if (!word || typeof word !== "string")
    {
        return "";
    }

    // Manejar palabras con "/" primero, normalizando cada parte recursivamente
    if (word.includes("/"))
    {
        return word.split("/")
          .map(part => normalizePlaceName(
                 part)) // Llamada recursiva para normalizar cada parte
          .join("/");
    }

    // Excepción específica para "MI"
    if (word.toUpperCase() === "MI")
    {
        return word.charAt(0).toUpperCase() +
               word.slice(1).toLowerCase(); // Resulta en "Mi"
    }

    // Regex para OTROS números romanos (incluyendo los que usan M, C, D, etc.)
    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))
    {
        // Si la palabra es un número romano válido, convertirla TODA a
        // mayúsculas.
        return word.toUpperCase();
    }
    else
    {
        // Si hay un número seguido de letra (sin espacio), convertir la
        // letra en mayúscula
        word = word.replace(
          /(\d)([a-z])/g, (_, num, letter) => `${num}${letter.toUpperCase()}`);
        // Si NO es un número romano, aplicar la normalización estándar:
        // primera letra mayúscula, resto minúsculas.
        return word.charAt(0).toUpperCase() + word.slice(1).toLowerCase();
    }
}

// === Palabras especiales ===
let excludedWords = new Set(); // Inicializada en waitForSidebarAPI

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();
        if (newWord)
        {
            const lowerNewWord = newWord.toLowerCase();

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

            if (commonWords.includes(lowerNewWord))
            {
                alert(
                  "Esa 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;
            }

            excludedWords.add(newWord);
            input.value = "";
            renderExcludedWordsList(
              document.getElementById("excludedWordsList"));
        }
    });
    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", exportExcludedWordsList);
    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";
        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. Verifique el formato.");
                        return;
                    }
                    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++;
                        }
                    }
                    if (newWordsAddedCount > 0)
                        console.log(`[WME PLN] ${
                          newWordsAddedCount} nuevas palabras añadidas desde XML.`);
                    renderExcludedWordsList(listContainerElement);
                }
                catch (err)
                {
                    console.error("[WME PLN] Excepción al procesar XML:", err);
                    alert("Ocurrió un 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 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)
    {
        alert("La lista de palabras especiales está vacía. Nada que exportar.");
        return;
    }
    const xmlContent =
      `<?xml version="1.0" encoding="UTF-8"?>\n<ExcludedWords>\n${
        Array.from(excludedWords)
          .sort((a, b) => a.toLowerCase().localeCompare(
                  b.toLowerCase()))                     // Exportar ordenado
          .map(w => `    <word>${xmlEscape(w)}</word>`) // Indentación y escape
          .join("\n")}\n</ExcludedWords>`;
    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_excluded_words_export.xml"; // Nombre más descriptivo
    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或关注我们的公众号极客氢云获取最新地址