WME Places Name Normalizer

Normaliza nombres de lugares en Waze Map Editor (WME)

目前为 2025-04-15 提交的版本。查看 最新版本

// ==UserScript==
// @name         WME Places Name Normalizer
// @namespace    https://gf.qytechs.cn/en/users/mincho77
// @version      4.0
// @description  Normaliza nombres de lugares en Waze Map Editor (WME)
// @author       mincho77
// @match        https://www.waze.com/*editor*
// @match        https://beta.waze.com/*user/editor*
// @grant        GM_xmlhttpRequest
// @connect      api.languagetool.org
// @grant        unsafeWindow
// @license      MIT
// @run-at       document-end
// ==/UserScript==
/*global W*/
(() => {
    "use strict";
    // Variables globales básicas
    const SCRIPT_NAME = "PlacesNameNormalizer";
    const VERSION = "4.0";
    let excludeWords = [];
    let maxPlaces = 100;
    let normalizeArticles = true;
    let placesToNormalize = [];
    // Declaración global de placesToNormalize
    // --------------------------------------------------------------------
    // Prevención global del comportamiento por defecto en drag & drop
    // (Evita que se abra el archivo en otra ventana)
    // Se aplican los eventos de arrastre y suelta a todo el documento.
    // Se previene el comportamiento por defecto para todos los eventos
    // de arrastre y suelta, excepto en el drop-zone.
    // Se establece el efecto de arrastre como "none" para evitar
    // cualquier efecto visual no deseado.
    // --------------------------------------------------------------------
    ["dragenter", "dragover", "dragleave", "drop"].forEach((evt) => {
        document.body.addEventListener(evt, (e) => {
            // Si el evento se origina en (o es descendiente de) #drop-zone, no
            // se bloquea.
            if (e.target && e.target.closest && e.target.closest("#drop-zone"))
            {
                return; // Permite que el drop-zone maneje el evento.
            }
            e.preventDefault();
            e.stopPropagation();
            if (e.dataTransfer)
            {
                e.dataTransfer.dropEffect = "none";
                e.dataTransfer.effectAllowed = "none";
            }
            return false;
        }, { capture : true });
    });
    // *****************************************************************************************************
    // Nombre: showNoPlacesFoundMessage
    // Fecha modificación: 2025-04-10
    // Autor: mincho77
    // Entradas: Ninguna
    // Salidas: Ninguna. Crea un modal que informa al usuario que no se
    // encontraron lugares que cumplan con los criterios actuales. Descripción:
    // Muestra un mensaje modal cuando no se encuentran lugares. Este mensaje
    // incluye un botón para cerrar el modal. Se utiliza para mostrar
    // información al usuario sobre la falta de lugares encontrados.
    // Ejemplo de uso: showNoPlacesFoundMessage();
    // *****************************************************************************************************
    function showNoPlacesFoundMessage()
    { // Crear el modal
        const modal = document.createElement("div");
        modal.className = "no-places-modal-overlay";
        modal.innerHTML = `
          <div class="no-places-modal">
              <div class="no-places-header">
                  <h3>⚠️ No se encontraron lugares</h3>
              </div>
              <div class="no-places-body">
                  <p>No se encontraron lugares que cumplan con los criterios actuales.</p>
                  <p>Intenta ajustar los filtros o ampliar el área de búsqueda.</p>
              </div>
              <div class="no-places-footer">
                  <button id="close-no-places-btn" class="no-places-btn">Aceptar</button>
              </div>
          </div>
      `;
        // Agregar el modal al documento
        document.body.appendChild(modal);
        // Manejar el evento de cierre
        document.getElementById("close-no-places-btn")
          .addEventListener("click", () => { modal.remove(); });
    }
    // Estilos CSS para el mensaje
    const noPlacesStyles = `
  <style>
  .no-places-modal-overlay
  {
      position: fixed;
      top: 0;
      left: 0;
      width: 100%;
      height: 100%;
      background: rgba(0, 0, 0, 0.6);
      display: flex;
      justify-content: center;
      align-items: center;
      z-index: 10000;
      animation: fadeIn 0.3s ease-in-out;
  }

  .no-places-modal {
      background: #fff;
      border-radius: 10px;
      box-shadow: 0 4px 15px rgba(0, 0, 0, 0.3);
      width: 90%;
      max-width: 400px;
      overflow: hidden;
      animation: slideIn 0.3s ease-in-out;
      text-align: center;
      padding: 20px;
  }

  .no-places-header {
      background: #f39c12;
      color: white;
      padding: 15px;
      font-size: 18px;
      font-weight: bold;
      border-radius: 10px 10px 0 0;
  }

  .no-places-body {
      padding: 20px;
      font-size: 14px;
      color: #333;
  }

  .no-places-footer {
      padding: 15px;
      background: #f4f4f4;
      text-align: center;
  }

  .no-places-btn {
      padding: 10px 20px;
      border: none;
      border-radius: 5px;
      cursor: pointer;
      font-size: 14px;
      font-weight: bold;
      background: #3498db;
      color: white;
      transition: background 0.3s, transform 0.2s;
  }

  .no-places-btn:hover {
      background: #2980b9;
      transform: scale(1.05);
  }

  /* Animaciones */
  @keyframes fadeIn {
      from {
          opacity: 0;
      }
      to {
          opacity: 1;
      }
  }

  @keyframes slideIn {
      from {
          transform: translateY(-20px);
      }
      to {
          transform: translateY(0);
      }
  }
  </style>
  `;
    // Insertar los estilos en el documento
    document.head.insertAdjacentHTML("beforeend", noPlacesStyles);
    // *****************************************************************************************************
    // Nombre: showModal
    // Fecha modificación: 2025-04-10
    // Autor: mincho77
    // Entradas: title (string): Título del modal.
    //          message (string): Mensaje a mostrar en el modal.
    //          confirmText (string): Texto del botón de confirmación.
    //          cancelText (string): Texto del botón de cancelación.
    //          onConfirm (function): Función a ejecutar al confirmar.
    //          onCancel (function): Función a ejecutar al cancelar.
    //          type (string): Tipo de modal (info, error, warning, question).
    //          autoClose (number): Tiempo en milisegundos para cerrar
    //          automáticamente. prependText (string): Texto a mostrar antes del
    //          mensaje.
    // Salidas: Ninguna. Crea un modal en el DOM.
    // Descripción:
    // Esta función crea un modal personalizado en el DOM. Permite mostrar
    // mensajes de información, advertencia, error o pregunta. Incluye botones
    // de confirmación y cancelación, así como la opción de cerrar
    // automáticamente después de un tiempo. Se pueden agregar estilos
    // personalizados y un texto opcional al principio del mensaje. ejemplos:
    // modal con texto adicional al principio showModal({ title: "Advertencia",
    // message: "Esta acción podría tener consecuencias.", prependText: "⚠️
    // Atención: Esto es importante.", confirmText: "Entendido", cancelText:
    // "Cancelar", type: "warning"
    // });
    // Modal de confirmación con signo de interrogación
    // showModal({
    // title: "¿Estás seguro?",
    // message: "Esta acción no se puede deshacer.",
    // confirmText: "Sí, estoy seguro",
    // cancelText: "No, cancelar",
    // type: "question",
    // isQuestion: true,
    // onConfirm: () => { console.log("Acción confirmada"); },
    // onCancel: () => { console.log("Acción cancelada"); }
    // });
    // Modal de exito que desaparece automaticamente
    // showModal({
    // title: "Éxito",
    // message: "La operación se completó con éxito.",
    // confirmText: "Aceptar",
    // type: "info",
    // autoClose: 3000, // Cierra automáticamente después de 3 segundos
    // onConfirm: () => { console.log("Modal cerrado automáticamente"); }
    // });
    // Modal con prependtext
    // showModal({
    // title: "Información",
    // message:  "El proceso {prependText} se completó correctamente.",
    // prependText: "de importación",
    // confirmText: "Aceptar",
    // type: "info",
    // });
    // *****************************************************************************************************
    function showModal({
        title,
        message,
        confirmText,
        cancelText,
        onConfirm,
        onCancel,
        type = "info",
        autoClose = null,
        prependText = "",
    })
    {
        // Determinar el ícono según el tipo
        let icon;
        switch (type)
        {
            case "error":
                icon = "⛔"; // Ícono para error
                break;
            case "warning":
                icon = "⚠️"; // Ícono para advertencia
                break;
            case "info":
                icon = "ℹ️"; // Ícono para información
                break;
            case "question":
                icon = "❓"; // Ícono para preguntas
                break;
            case "success":
                icon = "✅"; // Ícono para éxito
                break;
            default:
                icon = "ℹ️"; // Ícono por defecto
                break;
        }
        // Reemplazar el marcador de posición `{prependText}` en el mensaje
        const fullMessage = message.replace("{prependText}", prependText);
        // Crear el modal
        const modal = document.createElement("div");
        modal.className = "custom-modal-overlay";
        modal.innerHTML = `
          <div class="custom-modal">
              <div class="custom-modal-header">
                  <h3>${icon} ${title}</h3>
                  <button class="close-modal-btn" title="Cerrar">×</button>
              </div>
              <div class="custom-modal-body">
                  <p>${fullMessage}</p>
              </div>
              <div class="custom-modal-footer">
                  ${
          cancelText
            ? `<button id="modal-cancel-btn" class="modal-btn cancel-btn">${
                cancelText}</button>`
            : ""}
                  ${
          confirmText
            ? `<button id="modal-confirm-btn" class="modal-btn confirm-btn">${
                confirmText}</button>`
            : ""}
              </div>
          </div>
      `;
        // Agregar el modal al documento
        document.body.appendChild(modal);
        // Manejar eventos de los botones
        if (confirmText)
        {
            document.getElementById("modal-confirm-btn")
              .addEventListener("click", () => {
                  if (onConfirm)
                      onConfirm();

                  modal.remove();
              });
        }
        if (cancelText)
        {
            document.getElementById("modal-cancel-btn")
              .addEventListener("click", () => {
                  if (onCancel)
                      onCancel();

                  modal.remove();
              });
        }
        // Cerrar modal al hacer clic en el botón de cerrar
        modal.querySelector(".close-modal-btn")
          .addEventListener("click", () => { modal.remove(); });
        // Cerrar automáticamente si se especifica autoClose
        if (autoClose)
        {
            setTimeout(() => { modal.remove(); }, autoClose);
        }
    }
    // Estilos CSS mejorados para el modal
    const modalStyles = `
<style>
.custom-modal-overlay {
  position: fixed;
  top: 0;
  left: 0;
  width: 100%;
  height: 100%;
  background: rgba(0, 0, 0, 0.6);
  display: flex;
  justify-content: center;
  align-items: center;
  z-index: 10000;
  animation: fadeIn 0.3s ease-in-out;
}

.custom-modal {
  background: #fff;
  border-radius: 10px;
  box-shadow: 0 4px 15px rgba(0, 0, 0, 0.3);
  width: 90%;
  max-width: 400px;
  overflow: hidden;
  animation: slideIn 0.3s ease-in-out;
}

.custom-modal-header {
  background: #3498db;
  color: white;
  padding: 15px;
  display: flex;
  justify-content: space-between;
  align-items: center;
}

.custom-modal-header h3 {
  margin: 0;
  font-size: 18px;
}

.close-modal-btn {
  background: none;
  border: none;
  color: white;
  font-size: 20px;
  cursor: pointer;
  transition: color 0.3s;
}

.close-modal-btn:hover {
  color: #e74c3c;
}

.custom-modal-body {
  padding: 20px;
  font-size: 14px;
  color: #333;
  text-align: center;
}

.custom-modal-footer {
  display: flex;
  justify-content: space-between;
  padding: 15px;
  background: #f4f4f4;
}

.modal-btn {
  padding: 10px 20px;
  border: none;
  border-radius: 5px;
  cursor: pointer;
  font-size: 14px;
  font-weight: bold;
  transition: background 0.3s, transform 0.2s;
}

.confirm-btn {
  background: #27ae60;
  color: white;
}

.confirm-btn:hover {
  background: #2ecc71;
  transform: scale(1.05);
}

.cancel-btn {
  background: #e74c3c;
  color: white;
}

.cancel-btn:hover {
  background: #c0392b;
  transform: scale(1.05);
}

/* Animaciones */
@keyframes fadeIn {
  from {
      opacity: 0;
  }
  to {
      opacity: 1;
  }
}

@keyframes slideIn {
  from {
      transform: translateY(-20px);
  }
  to {
      transform: translateY(0);
  }
}
</style>
`;
    // Insertar los estilos en el documento
    document.head.insertAdjacentHTML("beforeend", modalStyles);
    // *****************************************************************************************************
    // Nombre: openEditPopup
    // Fecha modificación: 2025-04-15 04:55
    // Autor: mincho77
    // Entradas: index (number): Índice de la palabra a editar en la lista
    // excludeWords. Salidas: Ninguna. Crea un modal para editar la palabra
    // seleccionada. Descripción: Muestra un modal para permitir al usuario
    // editar una palabra en la lista excludeWords. Se valida que la palabra no
    // esté vacía y que no exista otra igual en la lista. Si la palabra es
    // válida, se actualiza la lista y se guarda en localStorage.
    // *****************************************************************************************************
    function openEditPopup(index)
    {
        const currentWord = excludeWords[index];
        if (!currentWord)
            return;

        showModal({
            title : "Editar palabra",
            message : `<input type="text" id="editWordInput" value="${
              currentWord}" style="width: 95%; padding: 5px; border-radius: 4px; border: 1px solid #ccc;">`,
            confirmText : "Guardar",
            cancelText : "Cancelar",
            type : "question",
            onConfirm : () => {
                const newWord =
                  document.getElementById('editWordInput').value.trim();
                if (!newWord)
                {
                    showModal({
                        title : "Error",
                        message : "La palabra no puede estar vacía.",
                        confirmText : "Aceptar",
                        type : "error"
                    });
                    return;
                }

                if (excludeWords.includes(newWord) &&
                    excludeWords[index] !== newWord)
                {
                    showModal({
                        title : "Duplicada",
                        message : "Esa palabra ya está en la lista.",
                        confirmText : "Aceptar",
                        type : "warning"
                    });
                    return;
                }

                excludeWords[index] = newWord;
                localStorage.setItem("excludeWords",
                                     JSON.stringify(excludeWords));
                renderExcludedWordsPanel();
                showModal({
                    title : "Actualizada",
                    message : "La palabra fue modificada correctamente.",
                    confirmText : "Aceptar",
                    type : "success",
                    autoClose : 3000
                });
            }
        });
    }

    // ********************************************************************************************************************************
    // Nombre: waitForElement
    // Fecha modificación: 2025-04-10
    // Autor: mincho77
    // Entradas:
    // - selector (string): El selector CSS del elemento que se desea esperar en
    // el DOM.
    // - callback (function): Función que se ejecutará una vez que el elemento
    // se encuentre en el DOM.
    // - interval (number, opcional): Tiempo en milisegundos entre cada intento
    // de búsqueda (por defecto: 300ms).
    // - maxAttempts (number, opcional): Número máximo de intentos antes de
    // abandonar la búsqueda (por defecto: 20). Salidas: Ninguna. Ejecuta el
    // callback pasando el elemento encontrado o muestra una advertencia en la
    // consola si no se encuentra. Prerrequisitos:
    // - El DOM debe estar cargado.
    // Descripción:
    // Esta función espera a que un elemento definido por un selector CSS
    // aparezca en el DOM. Utiliza un intervalo de tiempo (interval) para
    // realizar múltiples comprobaciones, hasta un máximo definido
    // (maxAttempts). Si el elemento se encuentra dentro de esos intentos, se
    // ejecuta la función callback con el elemento como argumento. Si no se
    // encuentra después de los intentos máximos, se detiene y se muestra una
    // advertencia en la consola. Esto es útil para asegurarse de que elementos
    // dinámicos estén disponibles antes de asignarles event listeners o
    // manipularlos.
    // ********************************************************************************************************************************
    function waitForElement(
      selector, callback, interval = 300, maxAttempts = 20)
    {
        let attempts = 0;
        const checkExist = setInterval(() => {
            const element = document.querySelector(selector);
            attempts++;
            if (element)
            {
                clearInterval(checkExist);
                callback(element);
            }
            else if (attempts >= maxAttempts)
            {
                clearInterval(checkExist);
                console.warn(`No se encontró el elemento ${
                  selector} después de ${maxAttempts} intentos.`);
            }
        }, interval);
    }
    // ********************************************************************************************************************************
    // Nombre: safeRedirect
    // Fecha modificación: 2025-03-31
    // Autor: mincho77
    // Entradas:
    // - url (string): La URL a la que se desea redirigir.
    // Salidas: Ninguna.
    // Descripción:
    // Esta función redirige a una URL solo si el dominio es uno de los
    // permitidos.
    // ********************************************************************************************************************************
    function safeRedirect(url)
    {
        const allowedDomains =
          [ "example.com", "mywebsite.com" ]; // Dominios permitidos
        try
        {
            const parsedUrl = new URL(url);
            if (allowedDomains.includes(parsedUrl.hostname))
            {
                window.location.href =
                  url; // Redirige solo si el dominio es válido
            }
            else
            {
                console.error("Redirección no permitida a:", url);
            }
        }
        catch (e)
        {
            console.error("URL inválida:", url);
        }
    }
    // ********************************************************************************************************************************
    // Nombre: redirectTo
    // Fecha modificación: 2025-03-31
    // Autor: mincho77
    // Entradas: path (string): Ruta a la que se desea redirigir.
    // ********************************************************************************************************************************
    function redirectTo(path)
    {
        const allowedPaths =
          [ "/home", "/profile", "/settings" ]; // Rutas permitidas
        if (allowedPaths.includes(path))
        {
            window.location.pathname = path;
        }
        else
        {
            console.error("Ruta no permitida:", path);
        }
    }
    // ********************************************************************************************************************************
    // Nombre: sanitizeUrl
    // Fecha modificación: 2025-03-31
    // Autor: mincho77
    // Entradas: url (string): La URL que se desea sanitizar.
    // Salidas: string: La URL sanitizada.
    // ********************************************************************************************************************************
    function sanitizeUrl(url)
    {
        const div = document.createElement("div");
        div.innerText = url;
        return div.innerHTML; // Escapa caracteres peligrosos
    }
    // ********************************************************************************************************************************
    // Nombre: waitForDOM
    // Fecha modificación: 2025-03-31
    // Autor: mincho77
    // Entradas:
    // - selector (string): Selector del elemento a esperar.
    // - callback (function): Función a ejecutar cuando se encuentre el
    // elemento.
    // - interval (number, opcional): Intervalo de tiempo entre intentos
    // (default: 500ms).
    // - maxAttempts (number, opcional): Número máximo de intentos (default:
    // 10). Salidas: Ninguna. Descripción: Espera a que un elemento identificado
    // por el selector exista en el DOM. Si se encuentra antes de llegar al
    // número máximo de intentos, se ejecuta el callback.
    // ********************************************************************************************************************************
    function waitForDOM(selector, callback, interval = 500, maxAttempts = 10)
    {
        let attempts = 0;
        const checkExist = setInterval(() => {
            const element = document.querySelector(selector);
            if (element)
            {
                clearInterval(checkExist);
                callback(element);
            }
            else if (attempts >= maxAttempts)
            {
                clearInterval(checkExist);
                console.error(`[${
                  SCRIPT_NAME}] Error: No se encontraron elementos con selector "${
                  selector}" después de ${maxAttempts} intentos.`);
            }
            attempts++;
        }, interval);
    }
    // ********************************************************************************************************************************
    // Nombre: initializeExcludeWords
    // Fecha modificación: 2025-03-30
    // Autor: mincho77
    // Entradas: Ninguna
    // Salidas: Ninguna
    // Prerrequisitos si existen:
    // - localStorage debe estar disponible.
    // Descripción:
    // Inicializa la lista de palabras excluidas a partir del localStorage,
    // combinando con las palabras ya cargadas en la variable global
    // excludeWords y actualizando el almacenamiento local.
    // ********************************************************************************************************************************
    function initializeExcludeWords()
    {
        const saved = JSON.parse(localStorage.getItem("excludeWords")) || [];
        // Se combinan sin duplicados y se ordena
        const merged = [...new Set([...saved, ...excludeWords ]) ].sort(
          (a, b) => a.localeCompare(b));
        if (JSON.stringify(saved.sort()) !== JSON.stringify(merged))
        {
            localStorage.setItem("excludeWords", JSON.stringify(merged));
            console.log(`[initializeExcludeWords] Actualizado: ${
              merged.length} palabras.`);
        }
        else
        {
            console.log("[initializeExcludeWords] Sin cambios en la lista.");
        }
        excludeWords = merged;
    }
    // ********************************************************************************************************************************
    // Nombre: getSidebarHTML
    // Fecha modificación: 2025-04-09
    // Autor: mincho77
    // Entradas: Ninguna
    // Salidas: Retorna un string que contiene el HTML para el panel lateral del
    // normalizador. Descripción: Esta función construye el HTML que se
    // inyectará en el panel lateral del script. Se agregó un nuevo bloque para
    // incluir un dropdown (select) con id "categoryDropdown" que permite
    // filtrar por categoría de los places. La opción por defecto es
    // "Categorías". Además, se incluyen controles para manejar el número máximo
    // de places, palabras excluidas, exportación/importación de la lista de
    // palabras excluidas, y un área de drag & drop para importar archivos.
    // Finalmente, se incluye un botón "Scan..." para iniciar el escaneo de
    // lugares.
    // ********************************************************************************************************************************
    function getSidebarHTML()
    {
        return `
      <div id="normalizer-tab">
          <h4>Places Name Normalizer <span style="font-size:11px;">${
          VERSION}</span></h4>

          <!-- No Normalizar artículos -->
          <div style="margin-top: 15px;">
              <input type="checkbox" id="normalizeArticles" ${
          normalizeArticles ? "checked" : ""}>
              <label for="normalizeArticles">No Normalizar artículos (el, la, los, ...)</label>
          </div>

          <!-- Máximo de Places a buscar -->
          <div style="margin-top: 15px;">
              <label>Máximo de Places a buscar: </label>
              <input type="number" id="maxPlacesInput" value="${
          maxPlaces}" min="1" max="800" style="width: 60px;">
          </div>

          <!-- Palabras Especiales con flecha -->
          <details id="details-special-words" style="margin-top: 15px;">
            <summary style="cursor: pointer; font-weight: bold; list-style: none;">
              <span id="arrow" style="display: inline-block; transition: transform 0.2s;">▶</span> Palabras Especiales
            </summary>

            <div style="margin-top: 10px; display: flex; gap: 5px;">
              <input type="text" id="excludeWord" placeholder="Agregar palabra..." style="flex: 1; padding: 5px; border: 1px solid #ccc; border-radius: 4px;">
              <button id="addExcludeWord" style="background: #3498db; color: white; border: none; padding: 5px 10px; border-radius: 4px; cursor: pointer;">
                Agregar
              </button>
            </div>

            <!-- Campo para buscar y filtrar palabras -->
            <div style="margin-top: 10px; display: flex; gap: 5px;">
              <input type="text" id="searchWord" placeholder="Buscar palabra..." style="flex: 1; padding: 5px; border: 1px solid #ccc; border-radius: 4px;">
              <button id="searchExcludeWord" style="background: #27ae60; color: white; border: none; padding: 5px 10px; border-radius: 4px; cursor: pointer;">
                Buscar
              </button>
            </div>

            <div id="normalizer-sidebar" style="margin-top: 10px; max-height: 200px; overflow-y: auto;"></div>

            <button id="exportExcludeWords" style="margin-top: 10px;">Exportar Palabras</button>
            <button id="importExcludeWordsUnifiedBtn" style="margin-top: 5px;">Importar Lista</button>
            <input type="file" id="hiddenImportInput" accept=".xml,.txt" style="display: none;">

            <div style="margin-top: 5px;">
              <input type="checkbox" id="replaceExcludeListCheckbox">
              <label for="replaceExcludeListCheckbox">Reemplazar lista actual</label>
            </div>

            <!-- Drag & Drop -->
            <div id="drop-zone" style="border: 2px dashed #ccc; border-radius: 6px; padding: 15px; margin: 15px 0; text-align: center; font-style: italic; color: #555; background-color: #f8f9fa;">
              📂 Arrastra aquí tu archivo .txt o .xml para importar palabras especiales
            </div>
          </details>

          <hr>

          <!-- Botón Scan -->
          <button id="scanPlaces">Scan...</button>

          <script>
          (function waitForSearchElements() {
              const searchInput = document.getElementById('searchWord');
              const searchButton = document.getElementById('searchExcludeWord');

              if (searchInput && searchButton) {
                  // Rotación de flecha
                  const detailsElem = document.getElementById('details-special-words');
                  const arrow = document.getElementById('arrow');
                  if (detailsElem && arrow) {
                      detailsElem.addEventListener('toggle', function() {
                          arrow.style.transform = detailsElem.open ? 'rotate(90deg)' : 'rotate(0deg)';
                      });
                  }

                  // Evento de búsqueda
                  searchButton.addEventListener('click', function () {
                      const query = searchInput?.value?.toLowerCase()?.trim() || "";
                      const items = document.querySelectorAll('#normalizer-sidebar div');
                      items.forEach(item => {
                          const text = item.textContent.toLowerCase();
                          item.style.display = text.includes(query) ? 'flex' : 'none';
                      });
                  });
              } else {
                  // Reintenta en 200ms
                  setTimeout(waitForSearchElements, 200);
              }
          })();
          </script>
      </div>
      `;
    }
    // ********************************************************************************************************************************
    // Nombre: attachEvents
    // Fecha modificación: 2025-03-30
    // Autor: mincho77
    // Entradas: Ninguna
    // Salidas: Ninguna
    // Prerrequisitos si existen:
    // - Deben existir en el DOM los elementos con los siguientes IDs:
    // "normalizeArticles", "maxPlacesInput", "addExcludeWord", "scanPlaces",
    // "hiddenImportInput", "importExcludeWordsUnifiedBtn" y
    // "exportExcludeWords".
    // - Debe existir la función handleImportList y la función scanPlaces.
    // - Debe estar definida la variable global excludeWords y la función
    // renderExcludedWordsPanel. Descripción: Esta función adjunta los event
    // listeners necesarios para gestionar la interacción del usuario con el
    // panel del normalizador de nombres. Se encargan de:
    // - Actualizar la opción de normalizar artículos al cambiar el estado del
    // checkbox.
    // - Modificar el número máximo de lugares a procesar a través de un input.
    // - Exportar la lista de palabras excluidas a un archivo XML.
    // - Añadir nuevas palabras a la lista de palabras excluidas, evitando
    // duplicados, y actualizar el panel.
    // - Activar el botón unificado para la importación de palabras excluidas
    // mediante un input oculto.
    // - Ejecutar la función de escaneo de lugares al hacer clic en el botón
    // correspondiente.
    // ********************************************************************************************************************************
    function attachEvents()
    {
        console.log(`[${SCRIPT_NAME}] Adjuntando eventos...`);
        const normalizeArticlesCheckbox =
          document.getElementById("normalizeArticles");
        const maxPlacesInput = document.getElementById("maxPlacesInput");
        const addExcludeWordButton = document.getElementById("addExcludeWord");
        const scanPlacesButton = document.getElementById("scanPlaces");
        const hiddenInput = document.getElementById("hiddenImportInput");
        const importButtonUnified =
          document.getElementById("importExcludeWordsUnifiedBtn");
        // Validación de elementos necesarios
        if (!normalizeArticlesCheckbox || !maxPlacesInput ||
            !addExcludeWordButton || !scanPlacesButton)
        {
            console.error(
              `[${SCRIPT_NAME}] Error: No se encontraron elementos en el DOM.`);
            return;
        }
        // Evento: cambiar estado de "no normalizar artículos"
        normalizeArticlesCheckbox.addEventListener(
          "change", (e) => { normalizeArticles = e.target.checked; });
        // Evento: cambiar número máximo de places
        maxPlacesInput.addEventListener(
          "input", (e) => { maxPlaces = parseInt(e.target.value, 10); });
        // Evento: exportar palabras excluidas a XML
        document.getElementById("exportExcludeWords")
          .addEventListener("click", () => {
              const savedWords =
                JSON.parse(localStorage.getItem("excludeWords")) || [];
              if (savedWords.length === 0)
              {
                  showModal({
                      title : "Error",
                      message : "No hay palabras excluidas para exportar.",
                      confirmText : "Aceptar",
                      onConfirm :
                        () => { console.log("El usuario cerró el modal."); }
                  });
                  return;
              }
              const sortedWords =
                [...savedWords ].sort((a, b) => a.localeCompare(b));
              const xmlContent = `<?xml version="1.0" encoding="UTF-8"?>
          <ExcludedWords>
             ${sortedWords.map((word) => `  <word>${word}</word>`).join("\n  ")}
          </ExcludedWords>`;
              const blob =
                new Blob([ xmlContent ], { type : "application/xml" });
              const url = URL.createObjectURL(blob);
              const link = document.createElement("a");
              link.href = url;
              link.download = "excluded_words.xml";
              document.body.appendChild(link);
              link.click();
              document.body.removeChild(link);
          });
        // Evento: añadir palabra excluida sin duplicados
        addExcludeWordButton.addEventListener("click", () => {
            const wordInput = document.getElementById("excludeWord") ||
                              document.getElementById("excludedWord");
            const word = wordInput?.value.trim();
            if (!word)
                return;

            const lowerWord = word.toLowerCase();
            const alreadyExists =
              excludeWords.some((w) => w.toLowerCase() === lowerWord);
            if (!alreadyExists)
            {
                excludeWords.push(word);
                localStorage.setItem("excludeWords",
                                     JSON.stringify(excludeWords));
                renderExcludedWordsPanel();
            }
            wordInput.value = "";
        });
        // Evento: nuevo botón unificado de importación
        importButtonUnified.addEventListener("click",
                                             () => { hiddenInput.click(); });
        hiddenInput.addEventListener("change", () => { handleImportList(); });
        // Evento: escanear lugares
        scanPlacesButton.addEventListener("click", scanPlaces);
    }
    // ********************************************************************************************************************************
    // Nombre: populateCategoryDropdownStatic
    // Fecha modificación: 2025-04-10
    // Autor: mincho77
    // Entradas: Ninguna.
    // Salidas: Ninguna. Actualiza el contenido del dropdown con id
    // "categoryDropdown" usando una
    //          lista de categorías estáticas.
    // Prerrequisitos:
    // - El DOM debe contener un <select id="categoryDropdown">.
    // Descripción:
    // Rellena el dropdown con una lista estática de categorías. Primero limpia
    // el contenido actual y añade la opción "Todos los places", luego recorre
    // un arreglo de categorías definidas de forma estática y agrega una opción
    // (<option>) para cada categoría.
    // ********************************************************************************************************************************
    function populateCategoryDropdownStatic()
    {
        waitForElement("#categoryDropdown", function(dropdown) {
            console.log(
              ">> populateCategoryDropdownStatic: Elemento encontrado:",
              dropdown);
            const categoriesStatic = [
                { id : "FOOD", name : "Food & Drink" },
                { id : "SHOP", name : "Shopping" },
                { id : "TRANSPORT", name : "Transportation" },
                { id : "LODGING", name : "Lodging" },
                { id : "ENTERTAINMENT", name : "Entertainment" },
                { id : "HEALTH", name : "Health" },
                { id : "EDUCATION", name : "Education" },
                { id : "GOVERNMENT", name : "Government" },
                { id : "FINANCE", name : "Finance" },
                { id : "RESIDENTIAL", name : "Residential" },
                { id : "COMMERCIAL", name : "Commercial" }
            ];
            // Limpia el contenido actual
            dropdown.innerHTML = "";
            // Agrega la opción por defecto: "Todas las Categorías"
            const optAll = document.createElement("option");
            optAll.value = "all";
            optAll.textContent = "All";
            dropdown.appendChild(optAll);
            // Agrega cada categoría estática
            categoriesStatic.forEach((cat) => {
                const option = document.createElement("option");
                option.value = cat.id;
                option.textContent = cat.name;
                dropdown.appendChild(option);
            });
            console.log(">> Dropdown poblado:", dropdown.innerHTML);
            for (let prop in W.model)
            {
                console.log("W.model prop:", prop, W.model[prop]);
            }
        });
    }
    // ********************************************************************************************************************************
    // Nombre: populateCategoryDropdown
    // Fecha modificación: 2025-06-20 14:30 GMT-5
    // Autor: mincho77
    // Entradas:
    // - Ninguna (usa el elemento DOM con id "categoryDropdown")
    // Salidas: Ninguna. Modifica el DOM al poblar el dropdown de categorías.
    // Prerrequisitos:
    // - El DOM debe contener un elemento <select> con id "categoryDropdown"
    // - El objeto global W debe estar disponible (entorno WME)
    // Descripción:
    // Esta función llena un dropdown con todas las categorías de lugares
    // disponibles en WME. Primero intenta obtener las categorías dinámicamente
    // del modelo WME (W.model.venues.getCategories()). Si falla la carga
    // dinámica, utiliza una lista estática completa de categorías como
    // respaldo. Las categorías se ordenan alfabéticamente y se añade siempre
    // una opción "Todas las categorías" al inicio. La función incluye manejo
    // robusto de errores y registra mensajes en la consola para diagnóstico.
    // ********************************************************************************************************************************
    function populateCategoryDropdown()
    {
        waitForElement("#categoryDropdown", (dropdown) => {
            if (!dropdown)
            {
                console.error("No se encontró el dropdown de categorías");
                return;
            }
            // Limpiar dropdown existente
            dropdown.innerHTML = "";
            // 1. Añadir opción "Todas" por defecto
            const defaultOption = document.createElement("option");
            defaultOption.value = "all";
            defaultOption.textContent = "Todas";
            dropdown.appendChild(defaultOption);
            // 2. Intentar cargar categorías dinámicas de WME
            try
            {
                const categories = W.model.venues.getCategories?.() || [];
                if (categories.length > 0)
                { // Ordenar alfabéticamente
                    categories.sort((a, b) => a.name.localeCompare(b.name));
                    // Agregar al dropdown
                    categories.forEach((cat) => {
                        const option = document.createElement("option");
                        option.value = cat.id;
                        option.textContent = cat.name;
                        dropdown.appendChild(option);
                    });
                    console.log(
                      `Categorías cargadas desde WME: ${categories.length}`);
                    return;
                }
            }
            catch (e)
            {
                console.error("Error al cargar categorías de WME:", e);
            }
            // 3. Respaldo: Categorías estáticas completas
            const staticCategories = [
                { id : "AIRPORT", name : "Aeropuerto" },
                { id : "ATM", name : "Cajero automático" },
                { id : "BANK", name : "Banco" },
                { id : "BAR", name : "Bar" },
                { id : "CAFE", name : "Cafetería" },
                { id : "CAR_RENTAL", name : "Renta de autos" },
                { id : "CAR_REPAIR", name : "Taller mecánico" },
                { id : "DELIVERY", name : "Entrega a domicilio" },
                { id : "EDUCATION", name : "Educación" },
                { id : "EMBASSY", name : "Embajada" },
                { id : "EMERGENCY", name : "Emergencia" },
                { id : "ENTERTAINMENT", name : "Entretenimiento" },
                { id : "FERRY", name : "Ferry" },
                { id : "FOOD", name : "Comida" },
                { id : "GAS_STATION", name : "Gasolinera" },
                { id : "GOVERNMENT", name : "Gobierno" },
                { id : "HOSPITAL", name : "Hospital" },
                { id : "HOTEL", name : "Hotel" },
                { id : "PARKING", name : "Estacionamiento" },
                { id : "PHARMACY", name : "Farmacia" },
                { id : "POLICE", name : "Policía" },
                { id : "POST_OFFICE", name : "Correo" },
                { id : "PUBLIC_TRANSPORT", name : "Transporte público" },
                { id : "RELIGIOUS", name : "Lugar religioso" },
                { id : "RESTAURANT", name : "Restaurante" },
                { id : "SHOPPING", name : "Compras" },
                { id : "TAXI", name : "Taxi" },
                { id : "TOURISM", name : "Turismo" },
                { id : "TRAIN_STATION", name : "Estación de tren" },
                { id : "UNIVERSITY", name : "Universidad" },
                { id : "ZOO", name : "Zoológico" }
            ];
            staticCategories.forEach((cat) => {
                const option = document.createElement("option");
                option.value = cat.id;
                option.textContent = cat.name;
                dropdown.appendChild(option);
            });
            console.log("Categorías cargadas desde lista estática");
        });
    }
    // ********************************************************************************************************************************
    // Nombre: getCategoriaTextoDesdeIDs
    // Fecha modificación: 2025-04-10
    // Autor: mincho77
    // Entradas:
    // - venue (object): Objeto de lugar de WME.
    // Salidas: string: Texto de la categoría o "Sin categoría" si no se
    // encuentra.
    // Descripción:
    // Esta función toma un objeto de lugar de WME y devuelve un string que
    // representa la categoría del lugar. Si el lugar no tiene categorías
    // asignadas, devuelve "Sin categoría". Si se producen errores durante el
    // proceso, se captura la excepción y se devuelve "Sin categoría".
    // ********************************************************************************************************************************
    function getCategoriaTextoDesdeIDs(ids = [])
    {
        try {
            const allCats = W.model?.categories?.objects;
            if (!allCats) throw new Error("No se pudo acceder a las categorías");

            return ids
                .map(id => allCats[id]?.name || `ID ${id}`)
                .join(", ");
        } catch (err) {
            console.error("Error al mapear IDs de categoría:", err);
            return "Sin categoría";
        }
    }




    // ********************************************************************************************************************************
    // Nombre: createSidebarTab
    // Fecha modificación: 2025-04-09
    // Autor: mincho77
    // Entradas: Ninguna.
    // Salidas: Ninguna.
    // Prerrequisitos si existen:
    // - La función W.userscripts.registerSidebarTab debe estar disponible en el
    // entorno WME. Descripción: Crea y registra una nueva pestaña lateral en el
    // WME para el normalizador. Inyecta el HTML generado por getSidebarHTML() y
    // espera a que se renderice el DOM para adjuntar los eventos.
    // ********************************************************************************************************************************
    function createSidebarTab()
    {
        try
        { // Check if the sidebar system is ready
            if (!W || !W.userscripts)
            {
                console.error(
                  `[${SCRIPT_NAME}] WME not ready for sidebar creation`);
                return;
            }
            // Check for existing tab and clean up if needed
            const existingTab = document.getElementById("normalizer-tab");
            if (existingTab)
            {
                console.log(`[${SCRIPT_NAME}] Removing existing tab...`);
                existingTab.remove();
            }
            // Register new tab with error handling
            let registration;
            try
            {
                registration =
                  W.userscripts.registerSidebarTab("PlacesNormalizer");
            }
            catch (e)
            {
                if (e.message.includes("already been registered"))
                {
                    console.warn(`[${
                      SCRIPT_NAME}] Tab registration conflict, attempting cleanup...`);
                    // Additional cleanup could go here
                    return;
                }
                throw e;
            }
            const { tabLabel, tabPane } = registration;
            if (!tabLabel || !tabPane)
            {
                throw new Error(
                  "Tab registration failed to return required elements");
            }
            // Configure tab
            tabLabel.innerHTML = `
          <img src=""
          style="height: 16px; vertical-align: middle; margin-right: 5px;">
          NrmliZer
          `;
            tabLabel.title = "Places Name Normalizer";
            // Set content and attach events
            tabPane.innerHTML = getSidebarHTML();
            waitForDOM("#normalizer-tab", (element) => {
                attachEvents();
                console.log(`[${SCRIPT_NAME}] Tab created and events attached`);
            }, 500, 10);
        }
        catch (error)
        {
            console.error(`[${SCRIPT_NAME}] Error in createSidebarTab:`, error);
        }
    }
    // *****************************************************************************************************
    // Nombre: checkSpellingWithAPI
    // Fecha modificación: 2025-03-30
    // Autor: mincho77
    // Entradas: text (string) – Texto a evaluar ortográficamente.
    // Salidas: Promise – Resuelve con lista de errores ortográficos detectados.
    // Prerrequisitos: Requiere permisos GM_xmlhttpRequest y @connect a
    // api.languagetool.org Descripción: Consulta la API de LanguageTool para
    // verificar ortografía del texto.
    // *****************************************************************************************************
    function checkSpellingWithAPI(text)
    {
        return new Promise((resolve, reject) => {
            GM_xmlhttpRequest({
                method : "POST",
                url : "https://api.languagetool.org/v2/check",
                headers :
                  { "Content-Type" : "application/x-www-form-urlencoded" },
                data : `language=es&text=${encodeURIComponent(text)}`,
                onload : function(response) {
                    if (response.status === 200)
                    {
                        const result = JSON.parse(response.responseText);
                        const errores = result.matches.map(
                          (match) => ({
                              palabra : match.context.text.substring(
                                match.context.offset,
                                match.context.offset + match.context.length),
                              sugerencia : match.replacements.length > 0
                                             ? match.replacements[0].value
                                             : "(sin sugerencia)"
                          }));
                        resolve(errores);
                    }
                    else
                    {
                        reject("❌ Error en respuesta de LanguageTool");
                    }
                },
                onerror : function(
                  err) { reject("❌ Error de red al contactar LanguageTool"); }
            });
        });
    }
    window.checkSpellingWithAPI = checkSpellingWithAPI;
    // *****************************************************************************************************
    // Nombre: evaluarOrtografiaCompleta
    // Fecha modificación: 2025-04-10 22:15 GMT-5
    // Autor: mincho77
    // Entradas:
    // - texto (string): Texto a evaluar
    // - config (opcional): {
    //       usarAPI: true,       // Usar LanguageTool
    //       reglasLocales: true, // Aplicar reglas de tildes
    //       timeout: 5000        // Tiempo máximo para API
    //     }
    // Salidas:
    // Promise<{
    //     original: string,
    //     normalizado: string,
    //     errores: Array<{
    //       palabra: string,
    //       sugerencia: string,
    //       tipo: 'ortografia'|'tilde'|'gramatica',
    //       severidad: 'alta'|'media'|'baja'
    //     }>,
    //     metadata: {
    //       totalErrores: number,
    //       apiUsada: boolean,
    //       tiempoProcesamiento: number
    //     }
    // }>
    // Descripción:
    // Sistema completo que combina normalización y revisión ortográfica real
    // *****************************************************************************************************
    async function evaluarOrtografiaCompleta(texto, config = {})
    {
        const inicio = Date.now();
        const resultadoBase = {
            original : texto,
            normalizado : "",
            errores : [],
            metadata :
              { totalErrores : 0, apiUsada : false, tiempoProcesamiento : 0 }
        };
        // 1. Normalización básica inicial
        const normalizado = normalizePlaceName(texto);
        resultadoBase.normalizado = normalizado;
        // 2. Detección de errores locales (síncrono)
        if (config.reglasLocales !== false)
        {
            const erroresLocales = detectarErroresLocales(texto, normalizado);
            resultadoBase.errores.push(...erroresLocales);
        }
        // 3. Revisión con API LanguageTool (asíncrono)
        if (config.usarAPI !== false && texto.length > 1)
        {
            try
            {
                const resultadoAPI =
                  await revisarConLanguageTool(texto, config.timeout);
                resultadoBase.errores.push(...resultadoAPI.errores);
                resultadoBase.metadata.apiUsada = true;
            }
            catch (error)
            {
                console.error("Error en API LanguageTool:", error);
            }
        }
        // 4. Filtrar y clasificar errores
        resultadoBase.errores = filtrarErrores(resultadoBase.errores);
        resultadoBase.metadata.totalErrores = resultadoBase.errores.length;
        resultadoBase.metadata.tiempoProcesamiento = Date.now() - inicio;
        return resultadoBase;
    }
    // ==================== FUNCIONES DE SOPORTE ====================
    // *****************************************************************************************************
    // Nombre: detectarErroresLocales
    // Descripción: Detecta errores de tildes y mayúsculas
    // *****************************************************************************************************
    function detectarErroresLocales(original, normalizado)
    {
        const errores = [];
        const palabrasOriginal = original.split(/\s+/);
        const palabrasNormalizado = normalizado.split(/\s+/);
        palabrasOriginal.forEach((palabra, i) => {
            const palabraNormalizada = palabrasNormalizado[i] || palabra;
            // 1. Comparación directa para detectar cambios
            if (palabra !== palabraNormalizada)
            {
                errores.push({
                    palabra,
                    sugerencia : palabraNormalizada,
                    tipo : "ortografia",
                    severidad : "media"
                });
            }
            // 2. Detección específica de tildes
            if (tieneTildesIncorrectas(palabra))
            {
                errores.push({
                    palabra,
                    sugerencia : corregirTildeLocal(palabra),
                    tipo : "tilde",
                    severidad : "alta"
                });
            }
        });
        return errores;
    }
    // *****************************************************************************************************
    // Nombre: revisarConLanguageTool
    // Fecha modificación: 2025-04-10 22:15 GMT-5
    // Autor: mincho77
    // Entradas:
    // - texto (string): Texto a evaluar
    // - timeout (opcional): Tiempo máximo para la API (en milisegundos)
    // Salidas:
    // Promise<{
    //     errores: Array<{
    //       palabra: string,
    //       sugerencia: string,
    //       tipo: 'ortografia'|'gramatica',
    //       severidad: 'alta'|'media'
    //     }>,
    //     apiStatus:
    //     'success'|'timeout'|'parse_error'|'api_error'|'network_error'
    // }>
    // Prerrequisitos: Requiere permisos GM_xmlhttpRequest y @connect a
    // api.languagetool.org Descripción: Consulta la API para errores
    // ortográficos y gramaticales
    // *****************************************************************************************************
    // Descripción: Consulta la API para errores avanzados
    // *****************************************************************************************************
    function revisarConLanguageTool(texto, timeout = 5000)
    {
        return new Promise((resolve) => {
            const timer = setTimeout(
              () => { resolve({ errores : [], apiStatus : "timeout" }); },
              timeout);
            GM_xmlhttpRequest({
                method : "POST",
                url : "https://api.languagetool.org/v2/check",
                headers :
                  { "Content-Type" : "application/x-www-form-urlencoded" },
                data : `language=es&text=${encodeURIComponent(texto)}`,
                onload : function(response) {
                    clearTimeout(timer);
                    if (response.status === 200)
                    {
                        try
                        {
                            const data = JSON.parse(response.responseText);
                            const errores = data.matches.map(
                              (match) => ({
                                  palabra : match.context.text.substring(
                                    match.context.offset,
                                    match.context.offset +
                                      match.context.length),
                                  sugerencia : match.replacements[0]?.value ||
                                                 match.context.text,
                                  tipo : match.rule.category.id === "TYPOS"
                                           ? "ortografia"
                                           : "gramatica",
                                  severidad :
                                    match.rule.issueType === "misspelling"
                                      ? "alta"
                                      : "media"
                              }));
                            resolve({ errores, apiStatus : "success" });
                        }
                        catch (e)
                        {
                            resolve(
                              { errores : [], apiStatus : "parse_error" });
                        }
                    }
                    else
                    {
                        resolve({ errores : [], apiStatus : "api_error" });
                    }
                },
                onerror : function() {
                    clearTimeout(timer);
                    resolve({ errores : [], apiStatus : "network_error" });
                }
            });
        });
    }
    // *****************************************************************************************************
    // Nombre: filtrarErrores
    // Descripción: Elimina duplicados y errores menores
    // *****************************************************************************************************
    function filtrarErrores(errores)
    {
        const unicos = [];
        const vistas = new Set();
        errores.forEach((error) => {
            const clave = `${error.palabra}-${error.sugerencia}-${error.tipo}`;
            if (!vistas.has(clave))
            {
                vistas.add(clave);
                unicos.push(error);
            }
        });
        return unicos.sort((a, b) => {
            if (a.severidad === b.severidad)
                return 0;

            return a.severidad === "alta" ? -1 : 1;
        });
    }
    // *****************************************************************************************************
    // Nombre: tieneTildesIncorrectas
    // Fecha modificación: 2025-04-10 21:30 GMT-5
    // Autor: mincho77
    // Entradas:
    // - palabra (string): Palabra a evaluar
    // - config (opcional): {
    //       ignorarMayusculas: true,
    //       considerarAdverbios: true,
    //       considerarMonosílabos: false
    //     }
    // Salidas: boolean - true si la palabra requiere corrección de tilde
    // Descripción:
    // Evalúa si una palabra en español tiene tildes incorrectas según las
    // reglas RAE. Incluye casos especiales para adverbios, hiatos, diptongos y
    // monosílabos.
    // *****************************************************************************************************
    function tieneTildesIncorrectas(palabra, config = {})
    {
        if (typeof palabra !== "string" || palabra.length === 0)
            return false;

        const settings = {
            ignorarMayusculas : config.ignorarMayusculas !==
                                  false, // No marcar errores en MAYÚSCULAS
            considerarAdverbios :
              config.considerarAdverbios !==
                false, // Evaluar adverbios terminados en -mente
            considerarMonosílabos :
              config.considerarMonosílabos || false, // Seguir reglas pre-2010
        };
        // Normalizar palabra (quitar tildes existentes para evaluación)
        const palabraNormalizada = palabra.normalize("NFD")
                                     .replace(/[\u0300-\u036f]/g, "")
                                     .toLowerCase();
        const tieneTildeActual = /[áéíóú]/.test(palabra);
        // 1. Reglas para palabras específicas (excepciones)
        const reglasEspecificas = {
            // Adverbios terminados en -mente
            mente :
              settings.considerarAdverbios && /mente$/i.test(palabra)
                ? tieneTildesIncorrectas(palabra.replace(/mente$/i, ""), config)
                : false,
            // Monosílabos
            monosilabos : settings.considerarMonosílabos &&
                            [
                                "fe",
                                "fue",
                                "fui",
                                "vio",
                                "dio",
                                "lia",
                                "lie",
                                "lio",
                                "rion",
                                "ries",
                                "se",
                                "te",
                                "de",
                                "si",
                                "ti"
                            ].includes(palabraNormalizada),
            // Casos especiales
            solo : palabraNormalizada === "solo" && !tieneTildeActual,
            este : /^este(s)?$/i.test(palabraNormalizada) && !tieneTildeActual,
            aun : palabraNormalizada === "aun" && !tieneTildeActual,
            guion : palabraNormalizada === "guion" && !tieneTildeActual,
            hui : palabraNormalizada === "hui" && !tieneTildeActual
        };
        if (Object.values(reglasEspecificas).some((v) => v))
            return true;

        // 2. Reglas generales de acentuación
        const silabas = separarSilabas(palabraNormalizada);
        const numSilabas = silabas.length;
        const ultimaLetra = palabraNormalizada.slice(-1);
        // Palabras agudas (tildan en última sílaba)
        if (numSilabas === 1)
            return false;

        // Monosílabos ya evaluados
        const esAguda = numSilabas === 1 ||
                        (numSilabas > 1 && silabas[numSilabas - 1].acento);
        const debeTildarAguda =
          esAguda && /[nsaeiouáéíóú]$/i.test(palabraNormalizada);
        const palabraLower = palabra.toLowerCase();
        if (correccionesEspecificas[palabraLower])
        {
            return aplicarCapitalizacion(palabra,
                                         correccionesEspecificas[palabraLower]);
        }
        // Determinar sílaba a tildar
        if (numSilabas > 2 && esEsdrujula(palabra))
        {
            silabaTildada = numSilabas - 3;
        }
        else if (numSilabas > 1 && esGrave(palabra))
        {
            silabaTildada = numSilabas - 2;
        }
        else if (esAguda(palabra))
        {
            silabaTildada = numSilabas - 1;
        }
        if (silabaTildada >= 0)
        {
            return aplicarTildeSilaba(palabra, silabas, silabaTildada);
        }
        return palabra;
    }
    // ==================== FUNCIONES AUXILIARES ====================
    // *****************************************************************************************************
    // Nombre: separarSilabas
    // Fecha modificación: 2025-04-10 22:15 GMT-5
    // Autor: mincho77
    // Entradas: palabra (string) – Palabra a separar en sílabas.
    // Salidas: Array<{ texto: string, acento: boolean }> – Lista de sílabas
    // Descripción: Separa la palabra en sílabas y determina si cada sílaba
    // tiene acento. Implementación simplificada para propósitos de
    // normalización visual.
    // *****************************************************************************************************
    function separarSilabas(palabra)
    { // Implementación simplificada (usar librería completa en producción)
        const vocalesFuertes = /[aeoáéó]/;
        const vocalesDebiles = /[iuü]/;
        const silabas = [];
        let silabaActual = "";
        let tieneVocalFuerte = false;
        for (let i = 0; i < palabra.length; i++)
        {
            const c = palabra[i];
            silabaActual += c;
            if (vocalesFuertes.test(c))
            {
                tieneVocalFuerte = true;
            }
            // Lógica simplificada de separación
            if (i < palabra.length - 1 &&
                ((vocalesFuertes.test(c) &&
                  vocalesFuertes.test(palabra[i + 1])) ||
                 (vocalesDebiles.test(c) &&
                  vocalesFuertes.test(palabra[i + 1]) && !tieneVocalFuerte)))
            {
                silabas.push(
                  { texto : silabaActual, acento : tieneVocalFuerte });
                silabaActual = "";
                tieneVocalFuerte = false;
            }
        }
        if (silabaActual)
        {
            silabas.push({ texto : silabaActual, acento : tieneVocalFuerte });
        }
        return silabas;
    }
    // *****************************************************************************************************
    // Nombre: aplicarCapitalizacion
    // Fecha modificación: 2025-04-10 22:15 GMT-5
    // Autor: mincho77
    // Entradas: original (string) – Palabra original
    //           corregida (string) – Palabra corregida
    // Salidas: string – Palabra corregida con mayúsculas/minúsculas
    // Descripción: Aplica mayúsculas/minúsculas a la palabra corregida
    // según la original. Mantiene mayúsculas y minúsculas en la primera letra
    // y el resto de la palabra.
    // *****************************************************************************************************
    function aplicarCapitalizacion(original, corregida)
    {
        if (original === original.toUpperCase())
        {
            return corregida.toUpperCase();
        }
        else if (original[0] === original[0].toUpperCase())
        {
            return corregida[0].toUpperCase() + corregida.slice(1);
        }
        return corregida;
    }
    // *****************************************************************************************************
    // Nombre: aplicarTildeSilaba
    // Descripción: Aplica tilde a la sílaba especificada
    // *****************************************************************************************************
    function aplicarTildeSilaba(palabra, silabas, indiceSilaba)
    {
        let resultado = "";
        let posActual = 0;
        silabas.forEach((silaba, i) => {
            if (i === indiceSilaba)
            {
                const conTilde = silaba.texto.replace(
                  /([aeiou])([^aeiou]*)$/, (match, vocal, resto) => {
                      return (
                        vocal.normalize("NFD").replace(/[\u0300-\u036f]/g, "") +
                        "́" + resto);
                  });
                resultado += conTilde;
            }
            else
            {
                resultado += silaba.texto;
            }
        });
        return resultado;
    }
    // *****************************************************************************************************
    // Nombre: normalizePlaceNameOnly
    // Fecha modificación: 2025-03-30
    // Autor: mincho77
    // Entradas: name (string) – Nombre del lugar a normalizar.
    // Salidas: texto normalizado sin validación ortográfica.
    // Descripción:
    // Realiza normalización visual del nombre: capitaliza, ajusta espacios,
    // formatea guiones, paréntesis, y símbolos. No evalúa ortografía ni
    // acentos.
    // *****************************************************************************************************
    function normalizePlaceNameOnly(name)
    {
        if (!name)
            return "";

        const normalizeArticles =
          !document.getElementById("normalizeArticles")?.checked;
        const articles = [ "el", "la", "los", "las", "de", "del", "al", "y" ];
        const words = name.trim().split(/\s+/);
        const isRoman = (word) =>
          /^(i{1,3}|iv|v|vi{0,3}|ix|x|xi{0,3}|xiv|xv|xvi{0,3}|xix|xx|xxi{0,3}|xxiv|xxv)$/i
            .test(word);
        const normalizedWords = words.map((word, index) => {
            const lowerWord = word.toLowerCase();
            // Si contiene "&", convertir a mayúsculas
            if (/^[A-Za-z]&[A-Za-z]$/.test(word))
                return word.toUpperCase();

            // Verificar si está en la lista de excluidas
            const matchExcluded =
              excludeWords.find((w) => w.toLowerCase() === lowerWord);
            if (matchExcluded)
                return matchExcluded;

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

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

            // Si es un número o un símbolo como "-", no modificar
            if (/^\d+$/.test(word) || word === "-")
                return word;

            // Verificar ortografía usando la API de LanguageTool
            return checkSpellingWithAPI(word)
              .then((errors) => {
                  if (errors.length > 0)
                  {
                      const suggestion = errors[0].sugerencia || word;
                      return (suggestion.charAt(0).toUpperCase() +
                              suggestion.slice(1).toLowerCase());
                  }
                  return word.charAt(0).toUpperCase() +
                         word.slice(1).toLowerCase();
              })
              .catch(() => word.charAt(0).toUpperCase() +
                           word.slice(1).toLowerCase());
        });
        let newName =
          normalizedWords.join(" ")
            .replace(/\s*\|\s*/g, " - ")
            .replace(/([(["'])\s*([\p{L}])/gu,
                     (match, p1, p2) => p1 + p2.toUpperCase())
            .replace(/\s*-\s*/g, " - ")
            .replace(/\b(\d+)([A-Z])\b/g,
                     (match, num, letter) => num + letter.toUpperCase())
            .replace(/\.$/, "")
            .replace(/&(\s*)([A-Z])/g,
                     (match, space, letter) =>
                       "&" + space + letter.toUpperCase());
        return newName.replace(/\s{2,}/g, " ").trim();
    }

    //**************************************************************************
    // Nombre: applyNormalization
    // Fecha modificación: 2025-04-15
    // Hora: 13:30:00
    // Autor: mincho77
    // Entradas: Ninguna directamente (usa el arreglo `changes` ya cargado)
    // Salidas: Aplica acciones en WME y muestra resultados
    // Prerrequisitos: `changes` debe contener objetos válidos con `place`, `newName`, y opcionalmente `delete`
    //**************************************************************************
    function applyNormalization(changes) {
        if (!Array.isArray(changes) || changes.length === 0) {
            showModal({
                title: "Información",
                message: "No hay cambios seleccionados para aplicar",
                confirmText: "Aceptar",
                type: "info"
            });
            return;
        }

        let lastAttemptedPlace = null;
        let cambiosRechazados = 0;

        try {
            changes.forEach((change) => {
                lastAttemptedPlace = {
                    name: change.originalName || change.place.attributes?.name || "Sin nombre",
                    id: change.place.getID?.() || "ID no disponible"
                };

                if (change.delete) {
                    const DeleteObject = require("Waze/Action/DeleteObject");
                    const action = new DeleteObject(change.place);
                    W.model.actionManager.add(action);
                } else {
                    const UpdateObject = require("Waze/Action/UpdateObject");
                    const action = new UpdateObject(change.place, { name: change.newName });
                    W.model.actionManager.add(action);
                }
            });

            observarErroresDeWME(changes.length, lastAttemptedPlace);

            W.controller?.setModified?.(true);
            showModal({
                title: "Éxito",
                message: `${changes.length} cambio(s) enviados. Clic en Guardar para aplicar en WME.`,
                type: "success",
                autoClose: 2000
            });
        } catch (error) {
            console.error("Error aplicando cambios:", error);
            showModal({
                title: "Error",
                message: "Error al aplicar cambios. Ver consola para detalles.",
                confirmText: "Aceptar",
                type: "error"
            });
        }
    }
    // *****************************************************************************************************
    // Nombre: evaluarOrtografiaConTildes
    // Fecha modificación: 2025-04-02
    // Autor: mincho77
    // Entradas: name (string) - Nombre del lugar
    // Salidas: objeto con errores detectados
    // Descripción:
    // Evalúa palabra por palabra si falta una tilde en las palabras que lo
    // requieren, según las reglas del español. Primero normaliza el nombre y
    // luego verifica si las palabras necesitan una tilde.
    // *****************************************************************************************************
    function evaluarOrtografiaConTildes(name)
    { // Si el nombre está vacío, retornar inmediatamente una promesa resuelta
        if (!name)
        {
            return Promise.resolve(
              { hasSpellingWarning : false, spellingWarnings : [] });
        }
        const palabras = name.trim().split(/\s+/);
        const spellingWarnings = [];
        console.log(
          `[evaluarOrtografiaConTildes] Verificando ortografía de: ${name}`);
        palabras.forEach(
          (palabra,
           index) => { // Normalizar la palabra antes de cualquier verificación
              let normalizada = normalizePlaceNameOnly(palabra);
              // Ignorar palabras con "&" o que sean emoticonos
              if (/^[A-Za-z]&[A-Za-z]$/.test(normalizada) ||
                  /^[\u263a-\u263c\u2764\u1f600-\u1f64f\u1f680-\u1f6ff]+$/.test(
                    normalizada))
              {
                  return; // No verificar ortografía
              }
              // Excluir palabras específicas como "y" o "Y"
              if (normalizada.toLowerCase() === "y" ||
                  /^\d+$/.test(normalizada) || normalizada === "-")
              {
                  return; // Ignorar
              }
              // Excluir palabras específicas como "e" o "E"
              if (normalizada.toLowerCase() === "e" ||
                  /^\d+$/.test(normalizada) || normalizada === "-")
              {
                  return; // Ignorar
              }
              // Verificar si la palabra está en la lista de excluidas
              if (excludeWords.some((w) => w.toLowerCase() ===
                                           normalizada.toLowerCase()))
              {
                  return; // Ignorar palabra excluida
              }
              // Validar que no tenga más de una tilde
              const cantidadTildes =
                (normalizada.match(/[áéíóú]/g) || []).length;
              if (cantidadTildes > 1)
              {
                  spellingWarnings.push({
                      original : palabra,
                      sugerida : null, // No hay sugerencia válida
                      tipo : "Error de tildes",
                      posicion : index
                  });
                  return;
              }
              // Verificar ortografía usando la API de LanguageTool
              checkSpellingWithAPI(normalizada)
                .then((errores) => {
                    errores.forEach((error) => {
                        spellingWarnings.push({
                            original : error.palabra,
                            sugerida : error.sugerencia,
                            tipo : "LanguageTool",
                            posicion : index
                        });
                    });
                })
                .catch((err) => {
                    console.error(
                      "Error al verificar ortografía con LanguageTool:", err);
                });
          });
        return {
            hasSpellingWarning : spellingWarnings.length > 0,
            spellingWarnings
        };
    }
    // *****************************************************************************************************
    // Nombre: toggleSpinner
    // Fecha modificación: 2025-03-31
    // Autor: mincho77
    // Entradas:
    // show (boolean) - true para mostrar el spinner, false para ocultarlo
    // message (string, opcional) - mensaje personalizado a mostrar junto al
    // spinner Salidas: ninguna (modifica el DOM) Prerrequisitos: debe existir
    // el estilo CSS del spinner en el documento Descripción: Muestra u oculta
    // un indicador visual de carga con un mensaje opcional. El spinner usa un
    // emoji de reloj de arena (⏳) con animación de rotación para indicar que
    // el proceso está en curso.
    // *****************************************************************************************************
    function toggleSpinner(
      show, message = "Revisando ortografía...", progress = null)
    {
        let existingSpinner = document.querySelector(".spinner-overlay");
        if (existingSpinner)
        {
            if (show)
            { // Actualizar el mensaje y el progreso si el spinner ya existe
                const spinnerMessage =
                  existingSpinner.querySelector(".spinner-message");
                spinnerMessage.innerHTML = `
                  ${message}
                  ${
                  progress !== null
                    ? `<br><strong>${progress}% completado</strong>`
                    : ""}
              `;
            }
            else
            {
                existingSpinner.remove(); // Ocultar el spinner
            }
            return;
        }
        if (show)
        {
            const spinner = document.createElement("div");
            spinner.className = "spinner-overlay";
            spinner.innerHTML = `
              <div class="spinner-content">
                  <div class="spinner-icon">⏳</div>
                  <div class="spinner-message">
                      ${message}
                      ${
              progress !== null ? `<br><strong>${progress}% completado</strong>`
                                : ""}
                  </div>
              </div>
          `;
            document.body.appendChild(spinner);
        }
    }
    // Agregar los estilos CSS necesarios
    const spinnerStyles = `
  <style>
  .spinner-overlay {
      position: fixed;
      top: 0;
      left: 0;
      width: 100%;
      height: 100%;
      background: rgba(0, 0, 0, 0.5);
      display: flex;
      justify-content: center;
      align-items: center;
      z-index: 10000;
  }

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

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

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

  @keyframes spin {
      0% { transform: rotate(0deg); }
      100% { transform: rotate(360deg); }
  }
  </style>`;
    // Insertar los estilos al inicio del documento
    document.head.insertAdjacentHTML("beforeend", spinnerStyles);
    if (!Array.prototype.flat)
    {
        Array.prototype.flat = function(depth = 1) {
            return this.reduce(function(flat, toFlatten) {
                return flat.concat(Array.isArray(toFlatten)
                                     ? toFlatten.flat(depth - 1)
                                     : toFlatten);
            }, []);
        };
    }
    // *****************************************************************************************************
    // Nombre: validarWordSpelling
    // Fecha modificación: 2025-03-30
    // Autor: mincho77
    // Entradas: palabra (string) - Palabra en español a validar
    // ortográficamente Salidas: true si cumple reglas ortográficas básicas,
    // false si no Descripción: Evalúa si una palabra tiene el uso correcto de
    // tilde o si le falta una tilde según las reglas del español: esdrújulas
    // siempre con tilde, agudas con tilde si terminan en n, s o vocal, y llanas
    // con tildse si NO terminan en n, s o vocal. Se asegura que solo haya una
    // tilde por palabra.
    // *****************************************************************************************************
    function validarWordSpelling(palabra)
    {
        if (!palabra)
            return false;

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

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

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

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

        const totalSilabas = silabas.length;
        const ultimaLetra = palabra.slice(-1).toLowerCase();
        let tipo = "";
        if (totalSilabas >= 3 && /[áéíóú]/.test(palabra))
        {
            tipo = "esdrújula";
        }
        else if (totalSilabas >= 2)
        {
            const penultimaSilaba = silabas[totalSilabas - 2];
            if (/[áéíóú]/.test(penultimaSilaba))
                tipo = "grave";
        }
        if (!tipo)
            tipo = /[áéíóú]/.test(silabas[totalSilabas - 1]) ? "aguda"
                                                                  : "sin tilde";

        if (tipo === "esdrújula")
            return tieneTilde;

        if (tipo === "aguda")
        {
            return ((/[nsáéíóúaeiou]$/.test(ultimaLetra) && tieneTilde) ||
                    (!/[nsáéíóúaeiou]$/.test(ultimaLetra) && !tieneTilde));
        }
        if (tipo === "grave")
        {
            return ((!/[nsáéíóúaeiou]$/.test(ultimaLetra) && tieneTilde) ||
                    (/[nsáéíóúaeiou]$/.test(ultimaLetra) && !tieneTilde));
        }
        return true;
    }
    // *****************************************************************************************************
    // Nombre: getCategoriaTexto
    // Fecha modificación: 2025-04-15
    // Autor: mincho77
    // Entradas: venue (object) - Objeto de lugar de Waze
    // Salidas: string - Texto de categoría
    // Descripción: Obtiene el texto de la categoría del lugar. Si no hay
    // categoría, devuelve "Sin categoría". Maneja errores y excepciones.
    // *****************************************************************************************************
    function getCategoriaTexto(venue)
    {

        try
        {
            if (!venue)
                return "Sin categoría";

            const allCategories = venue.getCategories?.();
            if (Array.isArray(allCategories) && allCategories.length > 0)
            {
                return allCategories.map(cat => cat?.name || "").join(", ");
            }

            const mainCategory = venue.getMainCategory?.();
            if (mainCategory?.name)
            {
                return mainCategory.name;
            }

            return "Sin categoría";
        }
        catch (e)
        {
            console.warn("Error al obtener categorías:", e);
            return "Sin categoría";
        }
    }

    // *****************************************************************************************************
    // Nombre: escapeHtml
    // Fecha modificación: 2025-06-20 18:30 GMT-5
    // Autor: mincho77
    // Entradas:
    // - unsafe (string|any): Valor a escapar
    // Salidas:
    // - string: Texto escapado seguro para usar en HTML
    // Prerrequisitos:
    // - Ninguno
    // Descripción:
    // Convierte caracteres especiales en entidades HTML para prevenir XSS.
    // Escapa los siguientes caracteres:
    // & → &amp;
    // < → &lt;
    // > → &gt;
    // " → &quot;
    // ' → &#039;
    // Si el input no es string, lo convierte a string.
    // Devuelve string vacío si el input es null/undefined.
    // *****************************************************************************************************
    function escapeHtml(unsafe)
    {
        if (unsafe === null || unsafe === undefined)
            return "";

        return String(unsafe)
          .replace(/&/g, "&amp;")
          .replace(/</g, "&lt;")
          .replace(/>/g, "&gt;")
          .replace(/"/g, "&quot;")
          .replace(/'/g, "&#039;");
    }

    let cambiosRechazados = 0;
    //**********************************************************************
    // Nombre: observarErroresDeWME
    // Fecha modificación: 2025-04-15
    // Hora: 13:01:25
    // Autor: mincho77
    // Entradas: Ninguna
    // Salidas: Ninguna
    // Descripción: Observa errores de WME y muestra un modal si se detecta
    // un mensaje de error relacionado con restricciones de edición.
    // Prerrequisitos: Ninguno
    //**********************************************************************
    function observarErroresDeWME(totalEsperado, lastAttemptedPlace) {
        const observer = new MutationObserver((mutationsList) => {
            for (const mutation of mutationsList) {
                for (const node of mutation.addedNodes) {
                    if (
                        node.nodeType === 1 &&
                        node.innerText?.includes("That change isn't allowed at this time")
                    ) {
                        observer.disconnect();

                        const ahora = new Date().toLocaleString("es-CO");
                        const historico = JSON.parse(localStorage.getItem("rechazosWME") || "[]");

                        historico.push({
                            timestamp: ahora,
                            motivo: "Cambio no permitido por WME",
                            lugar: lastAttemptedPlace?.name || "Desconocido",
                            id: lastAttemptedPlace?.id || "N/A"
                        });

                        localStorage.setItem("rechazosWME", JSON.stringify(historico));

                        showModal({
                            title: "Resultado parcial",
                            message:
                                `⚠️ Algunos lugares no pudieron ser modificados por restricciones de WME.\n` +
                                `Verifica el historial o vuelve a intentarlo.`,
                            confirmText: "Aceptar",
                            type: "warning"
                        });

                        break;
                    }
                }
            }
        });

        observer.observe(document.body, { childList: true, subtree: true });
    }

    //**************************************************************************
    // Nombre: getCategoriaTexto
    // Fecha modificación: 2025-04-15
    // Hora: 13:01:25
    // Autor: mincho77
    // Entradas: venue (object) - Objeto de lugar de Waze
    // Salidas: string - Texto de categoría
    // Descripción: Obtiene el texto de la categoría del lugar. Si no hay
    // categoría, devuelve "Sin categoría". Maneja errores y excepciones.
    //**************************************************************************
    function getCategoriaTexto(venue)
    {
        try
        {
            if (!venue)
                return "Sin categoría";

            const allCategories = venue.getCategories?.();
            if (Array.isArray(allCategories) && allCategories.length > 0)
            {
                return allCategories.map(cat => cat?.name || "").join(", ");
            }

            const mainCategory = venue.getMainCategory?.();
            if (mainCategory?.name)
            {
                return mainCategory.name;
            }

            return "Sin categoría";
        }
        catch (e)
        {
            console.warn("Error al obtener categorías:", e);
            return "Sin categoría";
        }
    }
    //**************************************************************************
    // Nombre: openFloatingPanel
    // Fecha modificación: 2025-04-15
    // Hora: 13:01:25
    // Autor: mincho77
    // Entradas: Arreglo placesToNormalize
    // Salidas: Panel flotante HTML para normalización
    // Prerrequisitos: Los objetos place deben incluir un .place con métodos
    // getCategories y getMainCategory Descripción: Construye un panel
    // interactivo para visualizar y aplicar normalizaciones y correcciones
    // ortográficas
    //**************************************************************************
    function openFloatingPanel(placesToNormalize)
    {
        const panel = document.createElement("div");
        panel.id = "normalizer-floating-panel";
        panel.style.cssText = `
            position: fixed; top: 50%; left: 50%; transform: translate(-50%, -50%);
            width: 90%; max-width: 1200px; max-height: 80vh; background: white;
            padding: 20px; border-radius: 8px; box-shadow: 0 0 25px rgba(0,0,0,0.4);
            z-index: 10000; overflow-y: auto; font-family: Arial, sans-serif;
        `;

        let html = `
            <style>
                #normalizer-table { width: 100%; border-collapse: collapse; margin: 15px 0; }
                #normalizer-table th { background: #2c3e50; color: white; padding: 10px; text-align: left; }
                #normalizer-table td { padding: 8px 10px; border-bottom: 1px solid #eee; }
                .warning-row { background: #fff8e1; }
                .normalize-btn, .apply-btn, .add-exclude-btn {
                    padding: 6px 12px; margin: 2px; border: none; border-radius: 4px;
                    cursor: pointer; font-weight: bold; transition: all 0.3s;
                }
                .normalize-btn { background: #3498db; color: white; }
                .apply-btn { background: #2ecc71; color: white; }
                .add-exclude-btn { background: #e67e22; color: white; }
                .close-btn {
                    position: absolute; top: 10px; right: 10px;
                    background: #e74c3c; color: white; border: none;
                    width: 30px; height: 30px; border-radius: 50%; font-weight: bold;
                }
                .tool-source {
                    font-size: 0.8em; color: #7f8c8d; margin-top: 3px;
                    font-style: italic;
                }
                input[type="checkbox"] { transform: scale(1.3); margin: 0 5px; }
                input[type="text"] { width: 100%; padding: 5px; box-sizing: border-box; }
            </style>

            <button class="close-btn" id="close-panel-btn">×</button>
            <h2 style="color: #2c3e50; margin-top: 5px;">Normalizador de Nombres</h2>
            <div style="margin: 10px 0; color: #7f8c8d;">
                <span id="places-count">${
          placesToNormalize.length} lugares para revisar</span>
            </div>

            <table id="normalizer-table">
                <thead>
                    <tr>
                        <th width="5%">Aplicar</th>
                        <th width="5%">Eliminar</th>
                        <th width="15%">Categoría</th>
                        <th width="25%">Nombre Actual</th>
                        <th width="25%">Nombre Normalizado</th>
                        <th width="15%">Problema Detectado</th>
                        <th width="10%">Acciones</th>
                    </tr>
                </thead>
            <tbody>`;

        placesToNormalize.forEach((place, index) => {
            const {
                originalName,
                newName,
                hasSpellingWarning,
                spellingWarnings,
                place : venue
            } = place;

            const category = getCategoriaTextoDesdeIDs(venue);
            const placeId = venue.getID();

            html += `
                <tr>
                    <td><input type="checkbox" class="normalize-checkbox" data-index="${
              index}" data-type="full"></td>
                    <td><input type="checkbox" class="delete-checkbox" data-index="${
              index}"></td>
                    <td title="${escapeHtml(category)}">${
              escapeHtml(category)}</td>
                    <td>${escapeHtml(originalName)}</td>
                    <td>
                        <input type="text" class="new-name-input" value="${
              escapeHtml(newName)}"
                        data-index="${index}" data-place-id="${
              placeId}" data-type="full"
                        data-original="${escapeHtml(originalName)}">
                    </td>
                    <td>${
              originalName !== newName ? "Normalización necesaria" : "-"}</td>
                    <td>
                        <button class="normalize-btn" data-index="${
              index}">NrmliZer</button>
                    </td>
                </tr>`;

            spellingWarnings.forEach((warning, warningIndex) => {
                html += `
                <tr class="warning-row">
                    <td><input type="checkbox" class="normalize-checkbox" data-index="${
                  index}" data-warning-index="${
                  warningIndex}" data-type="warning"></td>
                    <td></td>
                    <td title="${escapeHtml(category)}">${
                  escapeHtml(category)}</td>
                    <td>${escapeHtml(warning.original)}</td>
                    <td><input type="text" class="new-name-input" value="${
                  escapeHtml(warning.sugerida || newName)}"
                        data-index="${index}" data-place-id="${
                  placeId}" data-warning-index="${
                  warningIndex}" data-type="warning"></td>
                    <td>${escapeHtml(warning.tipo || "Error ortográfico")}
                        <div class="tool-source">${
                  warning.origen || "Reglas locales"}</div>
                    </td>
                    <td>
                        <button class="apply-btn" data-index="${
                  index}" data-warning-index="${warningIndex}">Aplicar</button>
                        <button class="add-exclude-btn" data-word="${
                  escapeHtml(warning.original)}">Excluir</button>
                    </td>
                </tr>`;
            });
        });

        html += `</tbody></table>
            <div style="margin-top: 20px; text-align: right;">
                <button id="apply-all-btn" style="background: #27ae60; color: white; padding: 10px 20px;
                    border: none; border-radius: 4px; font-weight: bold;">Aplicar Cambios Seleccionados</button>
                <button id="cancel-btn" style="background: #e74c3c; color: white; padding: 10px 20px;
                    border: none; border-radius: 4px; margin-left: 10px; font-weight: bold;">Cancelar</button>
            </div>`;

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

        document.getElementById("close-panel-btn")
          .addEventListener("click", () => panel.remove());
        document.getElementById("cancel-btn")
          .addEventListener("click", () => panel.remove());
          document.getElementById("apply-all-btn")
          .addEventListener("click", () => {
              const selectedPlaces = placesToNormalize.filter((place, index) => {
                  const checkbox = panel.querySelector(
                      `.normalize-checkbox[data-index="${index}"]`
                  );
                  return checkbox && checkbox.checked; // Solo incluir lugares seleccionados
              });

              if (selectedPlaces.length === 0)
            {
                  showModal({
                      title: "Advertencia",
                      message: "No se seleccionaron lugares para aplicar cambios.",
                      confirmText: "Aceptar",
                      type: "warning"
                  });
                  return;
              }

              applyNormalization(selectedPlaces);
              panel.remove();
          });

        panel.querySelectorAll(".normalize-btn").forEach((btn) => {
            btn.addEventListener("click", function() {
                const index = this.dataset.index;
                const input = panel.querySelector(
                  `.new-name-input[data-index="${index}"][data-type="full"]`);
                if (input)
                {
                    input.value = normalizePlaceName(input.value);
                    panel
                      .querySelector(`.normalize-checkbox[data-index="${
                        index}"][data-type="full"]`)
                      .checked = true;
                    this.textContent = "✓ Listo";
                    this.style.backgroundColor = "#95a5a6";
                    this.disabled = true;
                }
            });
        });

        panel.querySelectorAll(".apply-btn").forEach((btn) => {
            btn.addEventListener("click", function() {
                const index = this.dataset.index;
                const warningIndex = this.dataset.warningIndex;
                const checkbox =
                  panel.querySelector(`.normalize-checkbox[data-index="${
                    index}"][data-warning-index="${warningIndex}"]`);
                if (checkbox)
                {
                    checkbox.checked = true;
                    this.textContent = "✓ Aplicado";
                    this.style.backgroundColor = "#95a5a6";
                    this.disabled = true;
                }
            });
        });

        panel.querySelectorAll(".new-name-input").forEach((input) => {
            input.addEventListener("input", function() {
                const index = parseInt(this.dataset.index, 10);
                const original = this.dataset.original?.trim() || "";
                const currentValue = this.value.trim();
                const checkbox =
                  panel.querySelector(`.normalize-checkbox[data-index="${
                    index}"][data-type="full"]`);
                const normalizeButton =
                  panel.querySelector(`.normalize-btn[data-index="${index}"]`);

                if (!checkbox || !normalizeButton)
                    return;

                if (currentValue !== original)
                {
                    checkbox.checked = true;
                    normalizeButton.textContent = "Listo";
                    normalizeButton.style.backgroundColor = "#95a5a6";
                    normalizeButton.disabled = true;
                }
                else
                {
                    checkbox.checked = false;
                    normalizeButton.textContent = "NrmliZer";
                    normalizeButton.style.backgroundColor = "#3498db";
                    normalizeButton.disabled = false;
                }
            });
        });

        panel.querySelectorAll(".add-exclude-btn").forEach((btn) => {
            btn.addEventListener("click", function() {
                const word = this.dataset.word;
                if (word && !excludeWords.includes(word))
                {
                    excludeWords.push(word);
                    localStorage.setItem("excludeWords",
                                         JSON.stringify(excludeWords));
                    renderExcludedWordsPanel();
                    this.textContent = "✓ Excluida";
                    this.style.backgroundColor = "#95a5a6";
                    this.disabled = true;
                }
            });
        });
    }
    // *****************************************************************************************************
    // Nombre: checkOnlyTildes (4)
    // Fecha modificación: 2025-06-21
    // Autor: mincho77
    // Entradas:
    // - original (string): Palabra original a comparar.
    // - sugerida (string): Palabra sugerida a comparar.
    // Salidas:
    // - boolean:
    //     - true si las palabras son iguales excepto por tildes.
    //     - false si difieren en otros caracteres o si alguna es
    //     undefined/null.
    // Descripción:
    // Compara dos palabras ignorando tildes/diacríticos para determinar si la
    // única diferencia entre ellas es la acentuación. Utiliza normalización
    // Unicode para una comparación precisa. Optimizada para reducir operaciones
    // innecesarias.
    // *****************************************************************************************************
    function checkOnlyTildes(original, sugerida)
    {
        if (typeof original !== "string" || typeof sugerida !== "string")
        {
            return false;
        }
        if (original === sugerida)
        {
            return false;
        }
        const normalize = (str) =>
          str.normalize("NFD").replace(/[\u0300-\u036f]/g, "").toLowerCase();
        return normalize(original) === normalize(sugerida);
    }
    // *****************************************************************************************************
    // Nombre: deleteWord
    // Fecha modificación: 2025-04-14
    // Autor: mincho77
    // Entradas:
    // - index (number): Índice de la palabra a eliminar.
    // Salidas: Ninguna. Muestra un modal de confirmación.
    // Descripción:
    // Muestra un modal de confirmación para eliminar una palabra de la lista de
    // exclusiones. Si el usuario confirma, elimina la palabra de la lista y
    // actualiza el almacenamiento local.
    // *****************************************************************************************************
    function deleteWord(index)
    {
        const wordToDelete = excludeWords[index];
        if (!wordToDelete)
            return;

        showModal({
            title : "Eliminar palabra",
            message :
              `¿Estás seguro de que deseas eliminar la palabra <strong>${
                wordToDelete}</strong>?`,
            confirmText : "Eliminar",
            cancelText : "Cancelar",
            type : "question",
            onConfirm : () => {
                excludeWords.splice(index, 1);
                localStorage.setItem("excludeWords",
                                     JSON.stringify(excludeWords));
                renderExcludedWordsPanel();
                showModal({
                    title : "Eliminada",
                    message : "La palabra fue eliminada correctamente.",
                    confirmText : "Aceptar",
                    type : "success",
                    autoClose : 3000
                });
            }
        });
    }

    // *****************************************************************************************************
    // Nombre: openDeletePopup
    // Fecha modificación: 2025-04-14
    // Autor: mincho77
    // Entradas:
    // - index (number): Índice de la palabra a eliminar.
    // Salidas: Ninguna. Muestra un modal de confirmación.
    // Descripción:
    // Muestra un modal de confirmación para eliminar una palabra de la lista de
    // exclusiones. Si el usuario confirma, elimina la palabra de la lista y
    // actualiza el almacenamiento local.
    // *****************************************************************************************************
    function openDeletePopup(index)
    {
        const wordToDelete = excludeWords[index];
        if (!wordToDelete)
        {
            console.error(`No se encontró la palabra en el índice ${index}`);
            return;
        }

        showModal({
            title : "Eliminar palabra",
            message :
              `¿Estás seguro de que deseas eliminar la palabra <strong>${
                wordToDelete}</strong>?`,
            confirmText : "Eliminar",
            cancelText : "Cancelar",
            type : "warning",
            onConfirm : () => {
                excludeWords.splice(index, 1);
                localStorage.setItem("excludeWords",
                                     JSON.stringify(excludeWords));
                renderExcludedWordsPanel();
                showModal({
                    title : "Éxito",
                    message : "Palabra eliminada correctamente.",
                    confirmText : "Aceptar",
                    type : "success",
                    autoClose : 3000
                });
            }
        });
    }
    // *****************************************************************************************************
    // Nombre: evaluarOrtografiaNombre
    // Fecha modificación: 2025-04-10 20:45 GMT-5
    // Autor: mincho77
    // Entradas:
    // - name (string): Texto a evaluar
    // - opciones (object opcional): {
    //       timeout: 5000,       // Tiempo máximo en ms
    //       usarCache: true,     // Almacenar resultados temporalmente
    //       modoEstricto: false  // Verificar mayúsculas y puntuación
    //     }
    // Salidas:
    // Promise que resuelve a {
    //     hasSpellingWarning: boolean,
    //     spellingWarnings: Array<{
    //       original: string,
    //       sugerida: string,
    //       tipo: string,
    //       origen: 'API'|'Reglas locales',
    //       regla?: string,
    //       contexto?: string
    //     }>,
    //     metadata: {
    //       apiStatus: string,
    //       tiempoRespuesta?: number
    //     }
    // }
    // Descripción:
    // Evalúa ortografía usando API LanguageTool + reglas locales con:
    // - Validación robusta de entrada
    // - Timeout configurable
    // - Cache local de resultados
    // - Detección de tildes incorrectas
    // - Manejo completo de errores
    // Prerrequisitos:
    // - GM_xmlhttpRequest disponible
    // - Función tieneTildesIncorrectas() definida
    // *****************************************************************************************************
    function evaluarOrtografiaNombre(name, opciones = {})
    {
        const config = {
            timeout : opciones.timeout || 5000,
            usarCache : opciones.usarCache !== false,
            modoEstricto : opciones.modoEstricto || false
        };
        // Cache simple (evita llamadas duplicadas durante la sesión)
        const cache = evaluarOrtografiaNombre.cache ||
                      (evaluarOrtografiaNombre.cache = new Map());
        const cacheKey = `${config.modoEstricto}-${name}`;
        if (config.usarCache && cache.has(cacheKey))
        {
            return Promise.resolve(cache.get(cacheKey));
        }
        return new Promise((resolve) => { // 1. Validación de entrada
            if (typeof name !== "string" || name.trim().length === 0)
            {
                const resultado = {
                    hasSpellingWarning : false,
                    spellingWarnings : [],
                    metadata : { apiStatus : "invalid_input" }
                };
                cache.set(cacheKey, resultado);
                return resolve(resultado);
            }
            const inicio = Date.now();
            let timeoutExcedido = false;
            // 2. Timeout de seguridad
            const timeoutId = setTimeout(() => {
                timeoutExcedido = true;
                const resultado = {
                    hasSpellingWarning : false,
                    spellingWarnings : [],
                    metadata : {
                        apiStatus : "timeout",
                        tiempoRespuesta : Date.now() - inicio
                    }
                };
                cache.set(cacheKey, resultado);
                resolve(resultado);
            }, config.timeout);
            // 3. Primero verificar reglas locales (sincrónicas)
            const problemasLocales = [];
            const palabras = name.split(/\s+/);
            palabras.forEach((palabra) => {
                if (tieneTildesIncorrectas(palabra))
                {
                    problemasLocales.push({
                        original : palabra,
                        sugerida : corregirTildeLocal(palabra),
                        tipo : "Tilde incorrecta",
                        origen : "Reglas locales"
                    });
                }
            });
            // 4. Si hay problemas locales y no es modo estricto, devolver
            // inmediato
            if (problemasLocales.length > 0 && !config.modoEstricto)
            {
                clearTimeout(timeoutId);
                const resultado = {
                    hasSpellingWarning : true,
                    spellingWarnings : problemasLocales,
                    metadata : { apiStatus : "local_rules_applied" }
                };
                cache.set(cacheKey, resultado);
                return resolve(resultado);
            }
            // 5. Consultar API LanguageTool
            GM_xmlhttpRequest({
                method : "POST",
                url : "https://api.languagetool.org/v2/check",
                headers : {
                    "Content-Type" : "application/x-www-form-urlencoded",
                    Accept : "application/json"
                },
                data : `language=es&text=${encodeURIComponent(name)}`,
                onload : (response) => {
                    if (timeoutExcedido)
                        return;

                    clearTimeout(timeoutId);
                    const tiempoRespuesta = Date.now() - inicio;
                    let resultado;
                    try
                    {
                        if (response.status === 200)
                        {
                            const data = JSON.parse(response.responseText);
                            const problemasAPI = data.matches.map(
                              (match) => ({
                                  original : match.context.text.substring(
                                    match.context.offset,
                                    match.context.offset +
                                      match.context.length),
                                  sugerida : match.replacements[0]?.value ||
                                               match.context.text,
                                  tipo :
                                    match.rule.category.name || "Ortografía",
                                  origen : "API",
                                  regla : match.rule.id,
                                  contexto : match.context.text
                              }));
                            // Combinar resultados locales y de API
                            const todosProblemas =
                              [...problemasLocales, ...problemasAPI ];
                            resultado = {
                                hasSpellingWarning : todosProblemas.length > 0,
                                spellingWarnings : todosProblemas,
                                metadata : {
                                    apiStatus : "success",
                                    tiempoRespuesta,
                                    totalErrores : todosProblemas.length
                                }
                            };
                        }
                        else
                        {
                            resultado = {
                                hasSpellingWarning :
                                  problemasLocales.length > 0,
                                spellingWarnings : problemasLocales,
                                metadata : {
                                    apiStatus : `api_error_${response.status}`,
                                    tiempoRespuesta
                                }
                            };
                        }
                    }
                    catch (error)
                    {
                        resultado = {
                            hasSpellingWarning : problemasLocales.length > 0,
                            spellingWarnings : problemasLocales,
                            metadata :
                              { apiStatus : "parse_error", tiempoRespuesta }
                        };
                    }
                    cache.set(cacheKey, resultado);
                    resolve(resultado);
                },
                onerror : () => {
                    if (timeoutExcedido)
                        return;

                    clearTimeout(timeoutId);
                    const resultado = {
                        hasSpellingWarning : problemasLocales.length > 0,
                        spellingWarnings : problemasLocales,
                        metadata : {
                            apiStatus : "network_error",
                            tiempoRespuesta : Date.now() - inicio
                        }
                    };
                    cache.set(cacheKey, resultado);
                    resolve(resultado);
                }
            });
        });
    }
    // Funciones auxiliares requeridas
    // *****************************************************************************************************
    // Nombre: corregirTildeLocal
    // Fecha modificación: 2025-04-10 20:45 GMT-5
    // Autor: mincho77
    // Entradas:
    // - palabra (string): Palabra a corregir
    // Salidas:  (string): Palabra corregida o la original si no hay corrección.
    // Descripción: Esta función corrige las tildes de palabras específicas en
    // español. Se basa en un objeto de correcciones predefinido. Si la palabra
    // no está en el objeto, se devuelve la palabra original.
    // *****************************************************************************************************
    function corregirTildeLocal(palabra)
    {
        const correcciones = {
            solo : "sólo",
            aun : "aún",
            // ... otras correcciones
        };
        return correcciones[palabra.toLowerCase()] || palabra;
    }
    // *****************************************************************************************************
    // Nombre: populateCategoryDropdownFromWaze
    // Fecha modificación: 2025-04-09
    // Autor: mincho77
    // Entradas: Ninguna. Se asume que W.model.venues.getCategories() está
    // disponible y devuelve un arreglo
    //          de objetos de categoría (cada objeto debe tener al menos las
    //          propiedades "id" y "name").
    // Salidas: Ninguna. Actualiza el contenido del dropdown con id
    // "categoryDropdown" agregando las opciones correspondientes a
    //         cada categoría encontrada.
    // Prerrequisitos si existen:
    // - W.model.venues.getCategories() debe estar definido y devolver un
    // arreglo válido.
    // - El DOM debe contener un elemento <select> con id "categoryDropdown".
    // Descripción:
    // La función limpia el contenido actual del elemento dropdown y crea una
    // opción por defecto ("Categorías"). Luego, llama a
    // W.model.venues.getCategories() para obtener las categorías disponibles en
    // el modelo de WME. Por cada categoría en el arreglo, crea una opción
    // (<option>) asignándole como valor (value) la propiedad "id" y como texto
    // (textContent) la propiedad "name". Finalmente, agrega cada opción al
    // dropdown. Esto permite que el usuario seleccione la categoría por la que
    // desea filtrar los places.
    // *****************************************************************************************************
    function populateCategoryDropdownFromWaze()
    {
        const dropdown = document.getElementById("categoryDropdown");
        if (!dropdown)
            return;

        dropdown.innerHTML = "";
        const optAll = document.createElement("option");
        optAll.value = "all";
        optAll.textContent = "All";
        dropdown.appendChild(optAll);
        // Obtener categorías mediante getCategories()
        const catData =
          W.model.venues.getCategories && W.model.venues.getCategories();
        console.log("getCategories():", catData);
        if (Array.isArray(catData))
        { // catData debe ser un arreglo de objetos {id, name, ...}
            catData.forEach((catObj) => {
                const option = document.createElement("option");
                option.value = catObj.id;
                option.textContent = catObj.name;
                dropdown.appendChild(option);
            });
        }
        else
        {
            console.warn("No se encontraron categorías usando getCategories()");
        }
    }
    // *****************************************************************************************************
    // Nombre: populateCategoryDropdownWithSubcategories
    // Fecha modificación: 2025-04-08
    // Autor: mincho77
    // Entradas: Ninguna. Se asume que existe el objeto global
    // W.model.categories con las propiedades
    //          "groups" y "items", y que en el DOM hay un elemento <select> con
    //          id "categoryDropdown".
    // Salidas: Ninguna. La función actualiza el contenido del dropdown con id
    // "categoryDropdown",
    //         agregando una opción por defecto ("Categorías"), las categorías
    //         principales y sus subcategorías de forma jerárquica.
    // Prerrequisitos si existen:
    // - W.model.categories debe estar definido y contener "groups" (categorías
    // principales) y "items" (subcategorías).
    // - El DOM debe incluir un elemento <select> con id "categoryDropdown".
    // Descripción:
    // Esta función rellena el dropdown con las categorías y subcategorías
    // disponibles en la aplicación WME. Primero, limpia el contenido actual del
    // dropdown y agrega la opción por defecto "Todos los places". Luego,
    // recorre cada categoría principal en W.model.categories.groups y agrega
    // una opción para dicha categoría, marcándola visualmente como grupo (por
    // ejemplo, con un prefijo "[+ Grupo]"). A continuación, para cada grupo, si
    // existen subcategorías (la propiedad subCategories es un arreglo), recorre
    // cada subcategoría y crea una opción adicional para cada una, con un
    // indentado visual (por ejemplo, "   └ ") que indique su jerarquía. Esto
    // permite al usuario seleccionar la categoría específica por la que desea
    // filtrar los places.
    // *****************************************************************************************************
    function populateCategoryDropdownWithSubcategories()
    {
        const dropdown = document.getElementById("categoryDropdown");
        if (!dropdown)
            return;

        // Limpiar el contenido actual del dropdown.
        dropdown.innerHTML = "";
        // Agregar la opción por defecto: "Categorías".
        const optAll = document.createElement("option");
        optAll.value = "all";
        optAll.textContent = "All";
        dropdown.appendChild(optAll);
        // Obtener las categorías principales del modelo de WME.
        const groups = W.model.categories.groups;
        for (const groupKey in groups)
        {
            const groupData = groups[groupKey];
            // Agregar la opción de la categoría principal.
            const optionGroup = document.createElement("option");
            optionGroup.value =
              groupKey; // Puedes usar groupData.name si se requiere
            optionGroup.textContent = `[+ Grupo] ${groupData.name}`;
            dropdown.appendChild(optionGroup);
            // Si existen subcategorías para este grupo, agregarlas con un
            // indentado.
            if (groupData.subCategories && groupData.subCategories.length > 0)
            {
                groupData.subCategories.forEach((subCatKey) => {
                    const subCatData = W.model.categories.items[subCatKey];
                    if (!subCatData)
                        return;

                    // Seguridad en caso de datos faltantes.
                    const optionSub = document.createElement("option");
                    optionSub.value = subCatKey;
                    // Se agrega un prefijo de indentado para indicar la
                    // jerarquía.
                    optionSub.textContent = `   └ ${subCatData.name}`;
                    dropdown.appendChild(optionSub);
                });
            }
        }
    }
    // *****************************************************************************************************
    // Nombre: scanPlaces
    // Fecha modificación: 2025-04-10 18:30 GMT-5
    // Autor: mincho77
    // Entradas: Ninguna (usa elementos del DOM y el modelo WME)
    // Salidas: Ninguna. Ejecuta el escaneo de lugares y muestra resultados.
    // Prerrequisitos:
    // - El dropdown de categorías debe estar inicializado
    // - El modelo WME debe estar cargado (W.model.venues)
    // - Deben existir las funciones: normalizePlaceName,
    // evaluarOrtografiaNombre,
    //     openFloatingPanel, toggleSpinner
    // Descripción:
    // Escanea los lugares visibles en el WME, filtrando por categoría
    // seleccionada. Normaliza nombres, revisa ortografía usando API
    // LanguageTool y reglas locales, y muestra resultados en panel flotante.
    // Incluye:
    // - Spinner de carga con progreso
    // - Manejo robusto de errores
    // - Validación de tildes según reglas del español
    // - Procesamiento por lotes asíncrono
    // *****************************************************************************************************
    function scanPlaces()
    {
        const selectedCategory =
          document.getElementById("categoryDropdown")?.value || "all";
        const maxPlaces =
          parseInt(document.getElementById("maxPlacesInput")?.value || 100, 10);
        if (!W?.model?.venues?.objects)
        {
            console.error("Modelo WME no disponible");
            return;
        }
        // Convertir a string para comparación consistente
        const selectedCategoryStr = String(selectedCategory);
        const allPlaces =
          Object.values(W.model.venues.objects)
            .filter((place) => {
                if (!place?.attributes?.name)
                    return false;

                // Comparación de categoría mejorada
                if (selectedCategoryStr !== "all")
                {
                    const placeCategory =
                      String(place.attributes.category || "");
                    return placeCategory === selectedCategoryStr;
                }
                return true;
            })
            .slice(0, maxPlaces);
        if (allPlaces.length === 0)
        {
            toggleSpinner(false);
            showNoPlacesFoundMessage(); // Mostrar el mensaje mejorado
            return;
        }
        // 6. Procesamiento asíncrono con progreso
        let processedCount = 0;
        const placesToNormalize = [];
        const processBatch = async (index) => {
            if (index >= allPlaces.length)
            {
                toggleSpinner(false);
                if (placesToNormalize.length > 0)
                {
                    openFloatingPanel(placesToNormalize);
                }
                else
                {
                    showModal({
                        title : "Advertencia",
                        message :
                          "No se encontraron lugares que requieran normalización.",
                        confirmText : "Entendido",
                        type : "warning"
                    });
                }
                return;
            }
            const place = allPlaces[index];
            try
            {
                const originalName = place.attributes.name;
                const normalizedName = normalizePlaceName(originalName);
                // Actualizar progreso
                processedCount++;
                toggleSpinner(
                  true,
                  `Procesando lugares... (${processedCount}/${
                    allPlaces.length})`,
                  Math.round((processedCount / allPlaces.length) * 100));
                // Evaluar ortografía (usando el modo seleccionado)
                const ortografia =
                  checkOnlyTildes
                    ? await evaluarOrtografiaConTildes(normalizedName)
                    : await evaluarOrtografiaNombre(normalizedName);
                if (ortografia.hasSpellingWarning ||
                    originalName !== normalizedName)
                {
                    placesToNormalize.push({
                        id : place.getID(),
                        originalName,
                        newName : normalizedName,
                        category : place.attributes.category || "Sin categoría",
                        hasSpellingWarning : ortografia.hasSpellingWarning,
                        spellingWarnings : ortografia.spellingWarnings,
                        place
                    });
                }
                // Procesar siguiente lugar con pequeño retardo para no bloquear
                // UI
                setTimeout(() => processBatch(index + 1), 50);
            }
            catch (error)
            {
                console.error(`Error procesando lugar ${place.getID()}:`,
                              error);
                // Continuar con el siguiente lugar a pesar del error
                setTimeout(() => processBatch(index + 1), 50);
            }
        };
        // Iniciar procesamiento por lotes
        processBatch(0);
    }
    // *****************************************************************************************************
    // Nombre: renderExcludedWordsPanel
    // Fecha modificación: 2025-03-30
    // Autor: mincho77
    // Entradas: Ninguna (usa la variable global excludeWords).
    // Salidas: Ninguna.
    // Descripción:
    // Limpia y renderiza la lista de palabras excluidas en el panel lateral.
    // Ordena las palabras alfabéticamente y actualiza el localStorage.
    // *****************************************************************************************************
    function renderExcludedWordsPanel()
    { // 1. Obtener el contenedor del panel
        const container = document.getElementById("normalizer-sidebar");
        if (!container)
        {
            console.warn(`[${
              SCRIPT_NAME}] No se encontró el contenedor "normalizer-sidebar".`);
            return;
        }
        // 2. Limpiar el contenido del contenedor
        container.innerHTML = "";
        // 3. Crear el título "Palabras Especiales"
        // const title = document.createElement("h4");
        // title.textContent = "Palabras Especiales";
        // title.style.marginBottom = "10px";
        // 4. Crear la sección de palabras excluidas
        const excludeListSection = document.createElement("div");
        excludeListSection.style.marginTop = "10px";
        excludeListSection.innerHTML = `
          <div style="max-height: 150px; overflow-y: auto; border: 1px solid #ccc; padding: 8px; font-size: 13px; border-radius: 4px;">
              <ul style="margin: 0; padding-left: 0; list-style: none;" id="excludeWordsList"></ul>
          </div>
      `;
        // 5. Agregar los elementos al contenedor en el orden deseado
        // container.appendChild(title); // "Palabras Especiales"
        container.appendChild(excludeListSection);
        // Lista de palabras excluidas
        // 6. Función para renderizar la lista de palabras
        const renderList = () => {
            const list = document.getElementById("excludeWordsList");
            if (!list)
                return;

            // Ordenar las palabras alfabéticamente
            const sortedWords = excludeWords.sort((a, b) => a.localeCompare(b));
            // Actualizar el contenido de la lista
            list.innerHTML =
              sortedWords
                .map(
                  (word, index) => `
          <li style="display: flex; justify-content: space-between; align-items: center; padding: 5px 0;">
              <span>${word}</span>
              <div style="display: flex; gap: 10px;">
                  <span class="edit-word-icon" data-index="${
                    index}" style="cursor: pointer; color: #3498db;" title="Editar">✏️</span>
                  <span class="delete-word-icon" data-index="${
                    index}" style="cursor: pointer; color: #e74c3c;" title="Eliminar">🗑️</span>
              </div>
          </li>
          `).join("");
            // Agregar eventos para los íconos de edición y eliminación
            list.querySelectorAll(".edit-word-icon").forEach((icon) => {
                icon.addEventListener("click", (e) => {
                    const index = parseInt(
                      e.target.closest(".edit-word-icon").dataset.index, 10);
                    openEditPopup(index);
                });
            });
            list.querySelectorAll(".delete-word-icon").forEach(icon => {
                icon.addEventListener("click", (e) => {
                    const index = parseInt(
                      e.target.closest(".delete-word-icon").dataset.index, 10);
                    openDeletePopup(index);
                });
            });
        };
        // Renderizar la lista inicial
        renderList();
        // 7. Agregar funcionalidad de búsqueda en tiempo real
        const searchInput = document.getElementById("searchWord");
        if (searchInput)
        {
            searchInput.addEventListener("input", () => {
                const query = searchInput.value.toLowerCase().trim();
                const items = document.querySelectorAll("#excludeWordsList li");
                if (!items.length)
                {
                    console.warn(
                      "No se encontraron elementos en la lista de palabras excluidas.");
                    return;
                }
                items.forEach((item) => {
                    const text =
                      item.querySelector("span")?.textContent.toLowerCase() ||
                      "";
                    item.style.display = text.includes(query) ? "flex" : "none";
                });
            });
        }
    }
    // *****************************************************************************************************
    // Nombre: setupDragAndDropImport
    // Fecha modificación: 2025-03-31
    // Autor: mincho77
    // Entradas: Ninguna.
    // Salidas: Ninguna.
    // Descripción:
    // Activa la funcionalidad de drag & drop sobre el elemento con id
    // "drop-zone" para importar un archivo con palabras excluidas. Procesa
    // archivos .xml y .txt.
    // *****************************************************************************************************
    function setupDragAndDropImport()
    {
        const dropZone = document.getElementById("drop-zone");
        if (!dropZone)
        {
            console.warn(
              "setupDragAndDropImport: No se encontró el elemento #drop-zone");
            return;
        }
        dropZone.addEventListener("dragover", (e) => {
            e.preventDefault();
            dropZone.style.borderColor = "#4CAF50";
            dropZone.style.backgroundColor = "#f0fff0";
            console.log("dragover detectado");
        });
        dropZone.addEventListener("dragleave", (e) => {
            dropZone.style.borderColor = "#ccc";
            dropZone.style.backgroundColor = "";
            console.log("dragleave detectado");
        });
        dropZone.addEventListener("drop", (e) => {
            e.preventDefault();
            dropZone.style.borderColor = "#ccc";
            dropZone.style.backgroundColor = "";
            console.log("drop detectado");
            const file = e.dataTransfer.files[0];
            if (!file)
            {
                console.log("No se detectó ningún archivo");
                return;
            }
            console.log("Archivo soltado:", file.name);
            const reader = new FileReader();
            reader.onload = function(event) {
                console.log("Contenido del archivo:", event.target.result);
                let palabras = [];
                if (file.name.endsWith(".xml"))
                {
                    const parser = new DOMParser();
                    const xml =
                      parser.parseFromString(event.target.result, "text/xml");
                    const nodes = xml.querySelectorAll(
                      "word, palabra, item, excluded, exclude");
                    palabras = Array.from(nodes)
                                 .map((n) => n.textContent.trim())
                                 .filter((p) => p.length > 0);
                }
                else
                {
                    palabras = event.target.result.split(/\r?\n/)
                                 .map((line) => line.trim())
                                 .filter((line) => line.length > 0);
                }
                if (palabras.length === 0)
                { // alert("⚠️ No se encontraron palabras válidas.");
                    showModal({
                        title : "Advertencia",
                        message : "No se encontraron palabras válidas.",
                        type : "warning",
                        autoClose :
                          3000, // El modal desaparecerá después de 3 segundos
                    });
                    return;
                }
                const replace =
                  document.getElementById("replaceExcludeListCheckbox");
                if (replace && replace.checked)
                {
                    excludeWords = [];
                    localStorage.removeItem("excludeWords");
                }
                else
                {
                    excludeWords =
                      JSON.parse(localStorage.getItem("excludeWords")) || [];
                }
                excludeWords = [...new Set([...excludeWords, ...palabras ]) ]
                                 .filter((w) => w.trim().length > 0)
                                 .sort((a, b) => a.localeCompare(b));
                localStorage.setItem("excludeWords",
                                     JSON.stringify(excludeWords));
                renderExcludedWordsPanel();
                showModal({
                    title : "Información",
                    message :
                      "Se importaron {prependText} palabras desde el archivo.",
                    prependText : palabras.length,
                    confirmText : "Aceptar",
                    type : "info"
                });
                // alert(`✅ Se importaron ${palabras.length} palabras desde el
                // archivo.`);
            };
            reader.readAsText(file);
        });
    }
    // *****************************************************************************************************
    // Nombre: handleImportList
    // Fecha modificación: 2025-03-30
    // Autor: mincho77
    // Entradas: Ninguna (depende del input file "importListInput" y checkbox
    // "replaceExcludeListCheckbox"). Salidas: Ninguna. Descripción: Lee un
    // archivo seleccionado por el usuario, procesa sus líneas para extraer
    // palabras válidas, y actualiza la lista de palabras excluidas
    // (localStorage y panel).
    // *****************************************************************************************************
    function handleImportList()
    {
        const fileInput = document.getElementById("importListInput");
        const replaceCheckbox =
          document.getElementById("replaceExcludeListCheckbox");
        if (!fileInput || !fileInput.files || fileInput.files.length === 0)
        {
            // alert("No se seleccionó ningún archivo.");
            showModal({
                title : "Inoformación",
                message : "No se seleccionó ningun archivo.",
                confirmText : "Aceptar",
                type : "info"
            });
            return;
        }
        const reader = new FileReader();
        reader.onload = function(event) {
            const rawLines = event.target.result.split(/\r?\n/);
            const lines =
              rawLines
                .map((line) => line.replace(/[^\p{L}\p{N}().\s-]/gu, "").trim())
                .filter((line) => line.length > 0);
            const eliminadas = rawLines.length - lines.length;
            if (eliminadas > 0)
            {
                console.warn(`[handleImportList] Se ignoraron ${
                  eliminadas} líneas inválidas.`);
            }
            if (replaceCheckbox && replaceCheckbox.checked)
            {
                excludeWords = [];
            }
            else
            {
                excludeWords =
                  JSON.parse(localStorage.getItem("excludeWords")) ||
                  excludeWords || [];
            }
            excludeWords = [...new Set([...excludeWords, ...lines ]) ]
                             .filter((w) => w.trim().length > 0)
                             .sort((a, b) => a.localeCompare(b));
            localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
            renderExcludedWordsPanel();
            setupDragAndDropImport();
            showModal({
                title : "Éxito",
                message :
                  "Palabras excluidas importadas correctamente: {prependText}.",
                prependText : excludeWords.length,
                type : "info",
                autoClose : 2000, // El modal desaparecerá después de 3 segundos
            });
            // alert(`✅ Palabras excluidas importadas correctamente: ${
            //   excludeWords.length}`);
            fileInput.value = "";
        };
        reader.readAsText(fileInput.files[0]);
    }
    // *****************************************************************************************************
    // Nombre: isSimilar
    // Fecha modificación: 2025-03-30
    // Autor: mincho77
    // Entradas:
    // - a (string): Primera palabra a comparar.
    // - b (string): Segunda palabra a comparar.
    // Salidas:
    // - boolean: Retorna true si las palabras son consideradas similares de
    // forma leve; de lo contrario, retorna false. Prerrequisitos si existen:
    // Ninguno. Descripción: Esta función evalúa la similitud leve entre dos
    // palabras. Primero, verifica si ambas palabras son idénticas, en cuyo caso
    // retorna true. Luego, comprueba si la diferencia en la cantidad de
    // caracteres entre ambas es mayor a 2; si es así, retorna false.
    // Posteriormente, compara carácter por carácter hasta el largo mínimo de
    // las palabras, contando las diferencias. Si el número de discrepancias
    // excede 2, se considera que las palabras no son similares y retorna false;
    // en caso contrario, retorna true.
    // *****************************************************************************************************
    function isSimilar(a, b)
    {
        if (a === b)
            return true;

        if (Math.abs(a.length - b.length) > 2)
            return false;

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

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


    // *****************************************************************************************************
    // Nombre: cleanupEventListeners
    // Fecha modificación: 2025-03-30
    // Autor: mincho77
    // Entradas: Ninguna
    // Salidas: Ninguna
    // Prerrequisitos si existen:
    // - Debe existir un elemento en el DOM con el id
    // "normalizer-floating-panel". Descripción: Esta función elimina los event
    // listeners asociados al panel flotante de normalización. Lo hace clonando
    // el nodo del panel y reemplazándolo en el DOM, lo que remueve todos los
    // event listeners asignados a ese nodo, evitando posibles fugas de memoria
    // o comportamientos inesperados.
    // *****************************************************************************************************
    function cleanupEventListeners()
    {
        const panel = document.getElementById("normalizer-floating-panel");
        if (panel)
        {
            const clone = panel.cloneNode(true);
            panel.parentNode.replaceChild(clone, panel);
        }
    }
    // *****************************************************************************************************
    // Nombre: normalizePlaceName
    // Fecha modificación: 2025-04-14 11:45 GMT-5
    // Autor: mincho77
    // Entradas:
    // - name (string): El nombre original del lugar.
    // Salidas:
    // - string: Nombre normalizado.
    // Descripción:
    // Normaliza el nombre del lugar aplicando capitalización, manejo de
    // artículos, y ajustes de espacios y símbolos. Respeta la configuración del
    // checkbox "normalizeArticles" y la lista de palabras excluidas. Además, no
    // capitaliza letras después de un apóstrofo.
    // *****************************************************************************************************
    function normalizePlaceName(name)
    {
        if (!name)
            return "";

        const normalizeArticles =
          !document.getElementById("normalizeArticles")?.checked;
        const articles =
          [ "el", "la", "los", "las", "de", "del", "al", "y", "e" ];
        const words = name.trim().split(/\s+/);
        const isRoman = (word) =>
          /^(i{1,3}|iv|v|vi{0,3}|ix|x|xi{0,3}|xiv|xv|xvi{0,3}|xix|xx|xxi{0,3}|xxiv|xxv|xl)$/i
            .test(word);
        const normalizedWords = words.map((word, index) => {
            const lowerWord = word.normalize("NFD").toLowerCase();
            // Si es un número, se mantiene igual
            if (/^\d+$/.test(word))
                return word;

            // Si la palabra está en la lista de excluidas, se devuelve
            // EXACTAMENTE tal cual está.
            const match = excludeWords.find(
              (w) => w.normalize("NFD").toLowerCase() === lowerWord);
            if (match)
                return match;

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

            // Si contiene un apóstrofo, no capitalizar la letra siguiente
            if (/^[A-Za-z]+'[A-Za-z]/.test(word))
            {
                return (word.charAt(0).toUpperCase() +
                        word.slice(1, word.indexOf("'") + 1) +
                        word.slice(word.indexOf("'") + 1).toLowerCase());
            }
            // Si no se deben normalizar artículos y es un artículo, mantener en
            // minúsculas
            if (!normalizeArticles && articles.includes(lowerWord) &&
                index !== 0)
                return lowerWord;

            // Capitalizar la primera letra de la palabra
            return word.charAt(0).toUpperCase() + word.slice(1).toLowerCase();
        });
        let newName =
          normalizedWords.join(" ")
            .replace(/\s*\|\s*/g, " - ")
            .replace(/([(["'])\s*([\p{L}])/gu,
                     (match, p1, p2) => p1 + p2.toUpperCase())
            .replace(/\s*-\s*/g, " - ")
            .replace(/\b(\d+)([A-Z])\b/g,
                     (match, num, letter) => num + letter.toUpperCase())
            .replace(/\.$/, "")
            .replace(/&(\s*)([A-Z])/g,
                     (match, space, letter) =>
                       "&" + space + letter.toUpperCase());
        // Asegurar que las letras después de un apóstrofo estén en minúscula
        newName = newName.replace(
          /([A-Za-z])'([A-Za-z])/g,
          (match,
           before,
           after) => { return `${before}'${after.toLowerCase()}`; });
        // Asegurar que la primera letra después de un guion esté en mayúscula
        newName = newName.replace(
          /-\s*([a-z])/g, (match, letter) => `- ${letter.toUpperCase()}`);
        return newName.replace(/\s{2,}/g, " ").trim();
    }
    // *****************************************************************************************************
    // Nombre: init
    // Fecha modificación: 2025-04-09
    // Autor: mincho77
    // Entradas: Ninguna
    // Salidas: Ninguna
    // Prerrequisitos si existen:
    // - El objeto global W debe estar disponible.
    // - Deben estar definidas las funciones: initializeExcludeWords,
    // createSidebarTab, waitForDOM, renderExcludedWordsPanel y
    // setupDragAndDropImport. Descripción: Esta función espera a que el entorno
    // de edición de Waze (WME) esté completamente cargado, verificando que
    // existan los objetos necesarios para iniciar el script. Una vez
    // disponible, inicializa la lista de palabras excluidas, crea el tab
    // lateral personalizado, y espera a que el DOM del tab esté listo para
    // renderizar el panel de palabras excluidas y activar la funcionalidad de
    // arrastrar y soltar para importar palabras. Finalmente, expone globalmente
    // las funciones applyNormalization y normalizePlaceName.
    // *****************************************************************************************************
    function init()
    {
        if (!W || !W.userscripts || !W.model || !W.model.venues)
        {
            console.log(`[${SCRIPT_NAME}] Esperando que WME esté listo...`);
            setTimeout(init, 1000);
            return;
        }
        console.log(`[${SCRIPT_NAME}] Inicializando v${VERSION}`);
        initializeExcludeWords();
        createSidebarTab();
        waitForDOM("#normalizer-tab", () => {
            console.log("[init] Sidebar listo");
            renderExcludedWordsPanel();
            setupDragAndDropImport();
            // Cargar categorías con 2 intentos (inmediato y con retardo)
            populateCategoryDropdown();
            setTimeout(populateCategoryDropdown,
                       3000); // Respaldo por si carga async
        });
        // Agregar el evento para rotar la flecha en el elemento <details>
        waitForElement("#details-special-words", (detailsElem) => {
            const arrow = document.getElementById("arrow");
            if (detailsElem && arrow)
            {
                detailsElem.addEventListener("toggle", function() {
                    arrow.style.transform =
                      detailsElem.open ? "rotate(90deg)" : "rotate(0deg)";
                });
            }
            else
            {
                console.error(
                  "No se encontró el elemento #details-special-words o #arrow");
            }
        });
        window.applyNormalization = applyNormalization;
        window.normalizePlaceName = normalizePlaceName;
        // reinicializa la lista y cierra el panel cuando cambia el zoom
        if (W && W.model && W.model.venues)
        {
            // Suponiendo que W.model.venues emite el evento 'mapviewchanged' o
            // 'zoomchanged'.
            W.model.venues.on("zoomchanged", () => { // Reinicia la variable
                                                     // global
                placesToNormalize = [];
                // Cierra el panel flotante, si está presente
                const existingPanel =
                  document.getElementById("normalizer-floating-panel");
                if (existingPanel)
                {
                    existingPanel.remove();
                }
                console.log(
                  "Cambio de zoom detectado: Se ha reiniciado la búsqueda de lugares.");
            });
        }
        console.log("W.model.categories:", W.model.categories);
        console.log("W.model.venues.getCategories():",
                    W.model.venues.getCategories?.());
    }
    // Inicia el script
    init();
    // --------------------------------------------------------------------
    // Fin del script principal
    // Exponer algunas funciones clave globalmente (opcional)
    unsafeWindow.normalizePlaceName = normalizePlaceName;
    unsafeWindow.applyNormalization = applyNormalization;
})();

QingJ © 2025

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