// ==UserScript==
// @name WME Places Name Normalizer
// @namespace https://gf.qytechs.cn/en/users/mincho77
// @version 3.1
// @description Normaliza nombres de lugares en Waze Map Editor (WME) por ahora solo con reglas de Colombia
// @author mincho77
// @match https://www.waze.com/*editor*
// @match https://beta.waze.com/*user/editor*
// @grant GM_xmlhttpRequest
// @connect api.languagetool.org
// @connect *
// @grant unsafeWindow
// @license MIT
// @run-at document-end
// ==/UserScript==
/*global W*/
(() => {
"use strict";
try {
// Insertar estilos globales en el <head>
const styles = `
<style>
/* Estilos para los botones */
.apply-suggestion-btn {
background-color: #4CAF50;
color: white;
border: none;
padding: 8px 12px;
border-radius: 4px;
cursor: pointer;
font-size: 14px;
font-weight: bold;
transition: background-color 0.3s ease;
}
.apply-suggestion-btn:hover {
background-color: #45a049;
}
#apply-changes-btn {
background-color: #28a745;
color: white;
border: none;
padding: 10px 15px;
border-radius: 4px;
cursor: pointer;
font-size: 16px;
font-weight: bold;
display: flex;
align-items: center;
gap: 8px;
}
#apply-changes-btn:hover {
background-color: #218838;
}
#cancel-btn {
background-color: #dc3545;
color: white;
border: none;
padding: 10px 15px;
border-radius: 4px;
cursor: pointer;
font-size: 16px;
font-weight: bold;
display: flex;
align-items: center;
gap: 8px;
}
#cancel-btn:hover {
background-color: #c82333;
}
</style>
`;
document.head.insertAdjacentHTML('beforeend', styles);
// Capturar todos los eventos de drag & drop a nivel de <body>
// Agregar al inicio del script, justo después del "use strict"
// Prevenir comportamiento por defecto de drag & drop a nivel global
document.addEventListener("dragover", function(e) {
const dropZone = document.getElementById("drop-zone");
if (e.target === dropZone || dropZone?.contains(e.target)) {
return; // Permitir el drop en la zona designada
}
e.preventDefault();
e.stopPropagation();
}, { passive: false }); // Agrega { passive: false }
document.addEventListener("drop", function(e) {
const dropZone = document.getElementById("drop-zone");
if (e.target === dropZone || dropZone?.contains(e.target)) {
return; // Permitir el drop en la zona designada
}
e.preventDefault();
e.stopPropagation();
}, { passive: false }); // Agrega { passive: false }
//**************************************************************************
//Nombre: evaluarOrtografiaConTildes
//Fecha modificación: 2025-04-02
//Autor: mincho77
//Entradas: name (string) - Nombre del lugar
//Salidas: objeto con errores detectados
//Descripción:
// Evalúa palabra por palabra si falta una tilde en las palabras que lo requieren,
// según las reglas del español. Primero normaliza el nombre y luego verifica
// si las palabras necesitan una tilde.
//**************************************************************************
function evaluarOrtografiaConTildes(name) {
// Si el nombre está vacío, retornar inmediatamente una promesa resuelta
if (!name) {
return Promise.resolve({
hasSpellingWarning: false,
spellingWarnings: []
});
}
const palabras = name.trim().split(/\s+/);
const spellingWarnings = [];
console.log(`[evaluarOrtografiaConTildes] Verificando ortografía de: ${name}`);
palabras.forEach((palabra, index) => {
// Normalizar la palabra antes de cualquier verificación
let normalizada = normalizePlaceNameOnly(palabra);
// Ignorar palabras con "&" o que sean emoticonos
if (/^[A-Za-z]&[A-Za-z]$/.test(normalizada) || /^[\u263a-\u263c\u2764\u1f600-\u1f64f\u1f680-\u1f6ff]+$/.test(normalizada)) {
return; // No verificar ortografía
}
// Excluir palabras específicas como "y" o "Y"
if (normalizada.toLowerCase() === "y" || /^\d+$/.test(normalizada) || normalizada === "-") {
return; // Ignorar
}
// Verificar si la palabra está en la lista de excluidas
if (excludeWords.some(w => w.toLowerCase() === normalizada.toLowerCase())) {
return; // Ignorar palabra excluida
}
// Validar que no tenga más de una tilde
const cantidadTildes = (normalizada.match(/[áéíóú]/g) || []).length;
if (cantidadTildes > 1) {
spellingWarnings.push({
original: palabra,
sugerida: null, // No hay sugerencia válida
tipo: "Error de tildes",
posicion: index
});
return;
}
// Verificar ortografía usando la API de LanguageTool
checkSpellingWithAPI(normalizada).then(errores => {
errores.forEach(error => {
spellingWarnings.push({
original: error.palabra,
sugerida: error.sugerencia,
tipo: "LanguageTool",
posicion: index
});
});
}).catch(err => {
console.error("Error al verificar ortografía con LanguageTool:", err);
});
});
return {
hasSpellingWarning: spellingWarnings.length > 0,
spellingWarnings
};
}
// Función auxiliar para corregir tildes
function corregirTilde(palabra) {
// Implementar lógica para corregir tildes según las reglas del español
// Por ejemplo: "medellin" → "Medellín"
return palabra; // Retornar la palabra corregida
}
//**************************************************************************
//Nombre: toggleSpinner
//Fecha modificación: 2025-03-31
//Autor: mincho77
//Entradas:
// show (boolean) - true para mostrar el spinner, false para ocultarlo
// message (string, opcional) - mensaje personalizado a mostrar junto al spinner
//Salidas: ninguna (modifica el DOM)
//Prerrequisitos: debe existir el estilo CSS del spinner en el documento
//Descripción:
// Muestra u oculta un indicador visual de carga con un mensaje opcional.
// El spinner usa un emoji de reloj de arena (⏳) con animación de rotación
// para indicar que el proceso está en curso.
//**************************************************************************
function toggleSpinner(show, message = 'Revisando ortografía...', progress = null) {
let existingSpinner = document.querySelector('.spinner-overlay');
if (existingSpinner) {
if (show) {
// Actualizar el mensaje y el progreso si el spinner ya existe
const spinnerMessage = existingSpinner.querySelector('.spinner-message');
spinnerMessage.innerHTML = `
${message}
${progress !== null ? `<br><strong>${progress}% completado</strong>` : ''}
`;
} else {
existingSpinner.remove(); // Ocultar el spinner
}
return;
}
if (show) {
const spinner = document.createElement('div');
spinner.className = 'spinner-overlay';
spinner.innerHTML = `
<div class="spinner-content">
<div class="spinner-icon">⏳</div>
<div class="spinner-message">
${message}
${progress !== null ? `<br><strong>${progress}% completado</strong>` : ''}
</div>
</div>
`;
document.body.appendChild(spinner);
}
}
// Agregar los estilos CSS necesarios
const spinnerStyles = `
<style>
.spinner-overlay {
position: fixed;
top: 0;
left: 0;
width: 100%;
height: 100%;
background: rgba(0, 0, 0, 0.5);
display: flex;
justify-content: center;
align-items: center;
z-index: 10000;
}
.spinner-content {
background: white;
padding: 20px;
border-radius: 8px;
text-align: center;
box-shadow: 0 2px 10px rgba(0,0,0,0.3);
}
.spinner-icon {
font-size: 24px;
margin-bottom: 10px;
animation: spin 1s linear infinite; /* Aseguramos que la animación esté activa */
display: inline-block;
}
.spinner-message {
color: #333;
font-size: 14px;
}
@keyframes spin {
0% { transform: rotate(0deg); }
100% { transform: rotate(360deg); }
}
</style>`;
// Insertar los estilos al inicio del documento
document.head.insertAdjacentHTML('beforeend', spinnerStyles);
if (!Array.prototype.flat)
{
Array.prototype.flat = function(depth = 1)
{
return this.reduce(function (flat, toFlatten)
{
return flat.concat(Array.isArray(toFlatten) ? toFlatten.flat(depth - 1) : toFlatten);
}, []);
};
}
const SCRIPT_NAME = "PlacesNameNormalizer";
const VERSION = "3.1";
let placesToNormalize = [];
let excludeWords = [];
let maxPlaces = 50;
let normalizeArticles = true;
// Expresión regular para detectar siglas (por ejemplo, "S.a.s", "L.T.D.A")
const siglaRegex = /^[A-Za-z](\.[A-Za-z])+\.?$/;
//**************************************************************************
//Nombre: waitForSidebar
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas:
// retries (número, opcional) – Número máximo de intentos de verificación del sidebar (default: 20).
// delay (número, opcional) – Tiempo en milisegundos entre intentos (default: 1000ms).
//Salidas:
// Promesa que se resuelve con el elemento del sidebar si se encuentra, o se rechaza si no se encuentra después de los intentos.
//Descripción:
// Esta función espera a que el DOM cargue completamente el elemento con ID "sidebar".
// Realiza múltiples intentos con intervalos definidos, y resuelve la promesa cuando el sidebar esté disponible.
// Es útil para asegurarse de que el entorno de WME esté completamente cargado antes de continuar.
//**************************************************************************
function waitForSidebar(retries = 20, delay = 1000) {
return new Promise((resolve, reject) => {
const check = (attempt = 1) => {
const sidebar = document.querySelector("#sidebar");
if (sidebar) {
console.log("✅ Sidebar disponible.");
resolve(sidebar);
} else if (attempt <= retries) {
console.warn(`⚠️ Sidebar no disponible aún. Reintentando... (${attempt})`);
setTimeout(() => check(attempt + 1), delay);
} else {
reject("❌ Sidebar no disponible después de múltiples intentos.");
}
};
check();
});
}
//**************************************************************************
//Nombre: initializeExcludeWords
//Fecha modificación: 2025-03-31
//Autor: mincho77
//Entradas: Ninguna
//Salidas: Ninguna
//Prerrequisitos: excludeWords debe estar declarado globalmente.
//Descripción: Inicializa la lista de palabras excluidas desde localStorage sin borrar entradas ya existentes.
//**************************************************************************
function initializeExcludeWords() {
const saved = JSON.parse(localStorage.getItem("excludeWords")) || [];
// Combinar con las actuales sin duplicar
const merged = [...new Set([...saved, ...excludeWords])].sort((a, b) => a.localeCompare(b));
// Solo guardar si hay diferencias
const originalString = JSON.stringify(saved.sort());
const newString = JSON.stringify(merged);
if (originalString !== newString) {
localStorage.setItem("excludeWords", newString);
console.log(`[initializeExcludeWords] 💾 excludeWords actualizado con ${merged.length} palabras.`);
} else {
console.log(`[initializeExcludeWords] 🟢 Sin cambios en excludeWords.`);
}
// Actualizar variable global
excludeWords = merged;
}
//Modulo de Ortografía
// Insertar los estilos al inicio del documento
document.head.insertAdjacentHTML('beforeend', spinnerStyles);
if (!Array.prototype.flat)
{
Array.prototype.flat = function(depth = 1)
{
return this.reduce(function (flat, toFlatten)
{
return flat.concat(Array.isArray(toFlatten) ? toFlatten.flat(depth - 1) : toFlatten);
}, []);
};
}
//Modulo de Ortografía
//**************************************************************************
//Nombre: getUbicacionAcento
//Fecha modificación: 2025-03-27
//Autor: mincho77
//Entradas: palabra (string) - Palabra en español a evaluar
//Salidas: Posición del acento (última, penúltima, antepenúltima) o null si no aplica
//Prerrequisitos si existen: Ninguno
//Descripción: Determina la posición silábica del acento en una palabra según la ortografía del español.
//**************************************************************************
function getUbicacionAcento(palabra) {
const original = palabra;
const normalized = palabra.normalize("NFD").replace(/[\u0300-\u036f]/g, ""); // Remove diacritics
const silabas = normalized
.replace(/[^aeiou]/gi, "")
.match(/[aeiou]+/gi);
if (!silabas || silabas.length === 0) return null;
const conTilde = silabas.findIndex((s, i) => {
const originalSyllable = original.slice(
normalized.indexOf(s),
normalized.indexOf(s) + s.length
);
return /[áéíóú]/.test(originalSyllable);
});
if (conTilde !== -1) {
const posicion = silabas.length - 1 - conTilde;
if (posicion === 0) return "última";
if (posicion === 1) return "penúltima";
if (posicion === 2) return "antepenúltima";
return "otras";
}
return "sin tilde";
}
//**************************************************************************
//Nombre: checkSpellingWithAPI
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: text (string) – Texto a evaluar ortográficamente.
//Salidas: Promise – Resuelve con lista de errores ortográficos detectados.
//Prerrequisitos: Requiere permisos GM_xmlhttpRequest y @connect a api.languagetool.org
//Descripción:
// Consulta la API de LanguageTool para verificar ortografía del texto.
//**************************************************************************
function checkSpellingWithAPI(text) {
return new Promise((resolve, reject) => {
GM_xmlhttpRequest({
method: "POST",
url: "https://api.languagetool.org/v2/check",
headers: {
"Content-Type": "application/x-www-form-urlencoded"
},
data: `language=es&text=${encodeURIComponent(text)}`,
onload: function(response) {
if (response.status === 200) {
const result = JSON.parse(response.responseText);
const errores = result.matches
.filter(match => match.rule.issueType === "misspelling")
.map(match => ({
palabra: match.context.text.substring(match.context.offset, match.context.offset + match.context.length),
sugerencia: match.replacements.length > 0 ? match.replacements[0].value : "(sin sugerencia)"
}));
resolve(errores);
} else {
reject("❌ Error en respuesta de LanguageTool");
}
},
onerror: function(err) {
reject("❌ Error de red al contactar LanguageTool");
}
});
});
}
window.checkSpellingWithAPI = checkSpellingWithAPI;
// Combinar la lógica de checkSpelling
function checkSpelling(text) {
return new Promise((resolve, reject) => {
GM_xmlhttpRequest({
method: "POST",
url: "https://api.languagetool.org/v2/check",
data: `text=${encodeURIComponent(text)}&language=es`,
headers: {
"Content-Type": "application/x-www-form-urlencoded"
},
onload: function (response) {
if (response.status === 200) {
try {
const data = JSON.parse(response.responseText);
const errores = data.matches
.filter(match => match.rule.issueType === "misspelling")
.map(match => ({
palabra: match.context.text.substring(match.context.offset, match.context.offset + match.context.length),
sugerencia: match.replacements.length > 0 ? match.replacements[0].value : "(sin sugerencia)"
}));
resolve(errores);
} catch (err) {
reject(err);
}
} else {
reject(`Error HTTP: ${response.status}`);
}
},
onerror: function (err) {
reject(err);
}
});
});
}
//**************************************************************************
//Nombre: validarWordSpelling
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: palabra (string) - Palabra en español a validar ortográficamente
//Salidas: true si cumple reglas ortográficas básicas, false si no
//Descripción:
// Evalúa si una palabra tiene el uso correcto de tilde o si le falta una tilde
// según las reglas del español: esdrújulas siempre con tilde, agudas con tilde
// si terminan en n, s o vocal, y llanas con tildse si NO terminan en n, s o vocal.
// Se asegura que solo haya una tilde por palabra.
//**************************************************************************
function validarWordSpelling(palabra) {
if (!palabra) return false;
// Ignorar siglas con formato X&X
if (/^[A-Za-z]&[A-Za-z]$/.test(palabra)) return true;
// Si la palabra es un número, no necesita validación
if (/^\d+$/.test(palabra)) return true;
const tieneTilde = /[áéíóú]/.test(palabra);
const cantidadTildes = (palabra.match(/[áéíóú]/g) || []).length;
if (cantidadTildes > 1) return false; // Solo se permite una tilde
const silabas = palabra.normalize("NFD").replace(/[^aeiouAEIOU\u0300-\u036f]/g, "").match(/[aeiouáéíóú]+/gi);
if (!silabas || silabas.length === 0) return false;
const totalSilabas = silabas.length;
const ultimaLetra = palabra.slice(-1).toLowerCase();
let tipo = "";
if (totalSilabas >= 3 && /[áéíóú]/.test(palabra)) {
tipo = "esdrújula";
} else if (totalSilabas >= 2) {
const penultimaSilaba = silabas[totalSilabas - 2];
if (/[áéíóú]/.test(penultimaSilaba)) tipo = "grave";
}
if (!tipo) tipo = /[áéíóú]/.test(silabas[totalSilabas - 1]) ? "aguda" : "sin tilde";
if (tipo === "esdrújula") return tieneTilde;
if (tipo === "aguda") {
return (/[nsáéíóúaeiou]$/.test(ultimaLetra) && tieneTilde) ||
(!/[nsáéíóúaeiou]$/.test(ultimaLetra) && !tieneTilde);
}
if (tipo === "grave") {
return (!/[nsáéíóúaeiou]$/.test(ultimaLetra) && tieneTilde) ||
(/[nsáéíóúaeiou]$/.test(ultimaLetra) && !tieneTilde);
}
return true;
}
///**************************************************************************
//Nombre: separarSilabas
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: palabra (string) – Palabra en español.
//Salidas: array de sílabas (aproximado).
//Descripción:
// Separa una palabra en sílabas usando reglas heurísticas.
// Esta versión simplificada considera diptongos y combinaciones comunes.
//**************************************************************************
function separarSilabas(palabra) {
// Normaliza y quita acentos para facilitar la segmentación
const limpia = palabra.normalize("NFD").replace(/[\u0300-\u036f]/g, "").toLowerCase();
// Divide por vocales agrupadas como aproximación de sílabas
const silabas = limpia.match(/[bcdfghjklmnñpqrstvwxyz]*[aeiou]{1,2}[bcdfghjklmnñpqrstvwxyz]*/g);
return silabas || [palabra]; // fallback si no separa nada
}
//**************************************************************************
//Nombre: clasificarPalabra
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: silabas (array) – Arreglo con las sílabas de la palabra.
//Salidas: string – 'aguda', 'grave' (llana), 'esdrújula'.
//Descripción:
// Determina el tipo de palabra según el número de sílabas y la posición
// de la tilde (si existe).
//**************************************************************************
function clasificarPalabra(silabas) {
const palabra = silabas.join("");
const tieneTilde = /[áéíóúÁÉÍÓÚ]/.test(palabra);
if (tieneTilde) {
const posicionTilde = silabas.findIndex(s => /[áéíóúÁÉÍÓÚ]/.test(s));
const posicionDesdeFinal = silabas.length - 1 - posicionTilde;
if (posicionDesdeFinal === 0) return "aguda";
if (posicionDesdeFinal === 1) return "grave";
if (posicionDesdeFinal >= 2) return "esdrújula";
}
// Si no tiene tilde, asumimos que es:
if (silabas.length === 1) return "aguda";
if (silabas.length === 2) return "grave";
return "grave"; // por convención para 3+ sin tilde
}
//**************************************************************************
//Nombre: evaluarOrtografiaNombre
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: name (string) - Nombre del lugar
//Salidas: objeto con errores detectados
//Descripción: Evalúa palabra por palabra si hay errores ortográficos o falta de tildes.
// Ya no utiliza sugerencias automáticas para correcciones.
//**************************************************************************
function evaluarOrtografiaNombre(name) {
if (!name) return {
hasSpellingWarning: false,
spellingWarnings: []
};
const checkOnlyTildes = document.getElementById("checkOnlyTildes")?.checked;
const palabras = name.trim().split(/\s+/);
const spellingWarnings = [];
console.log(`[evaluarOrtografiaNombre] Verificando ortografía de: ${name}`);
if (checkOnlyTildes) {
palabras.forEach((palabra, index) => {
// Verificar si la palabra está en la lista de exclusiones
if (excludeWords.some(w => w.toLowerCase() === palabra.toLowerCase()) || /^\d+$/.test(palabra)) {
return; // Ignorar palabra excluida
}
if (!validarWordSpelling(palabra)) {
spellingWarnings.push({
original: palabra,
sugerida: null,
tipo: "Tilde",
posicion: index // Guardar la posición en la frase
});
}
});
return Promise.resolve({
hasSpellingWarning: spellingWarnings.length > 0,
spellingWarnings
});
} else {
return checkSpellingWithAPI(name)
.then(errores => {
errores.forEach(error => {
// Verificar si la palabra está en exclusiones
if (!excludeWords.some(w => w.toLowerCase() === error.palabra.toLowerCase())) {
spellingWarnings.push({
original: error.palabra,
sugerida: error.sugerencia,
tipo: "LanguageTool",
posicion: name.indexOf(error.palabra)
});
}
});
return {
hasSpellingWarning: spellingWarnings.length > 0,
spellingWarnings
};
});
}
}
//**************************************************************************
//Nombre: handleImportList
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: ninguna directa, depende del input file y checkbox en el DOM
//Salidas: actualiza excludeWords en localStorage y visualmente
//Prerrequisitos: existir un input file con id="importListInput" y checkbox con id="replaceExcludeListCheckbox"
//Descripción: Importa una lista de palabras para excluir, opcionalmente reemplazando la lista actual
//**************************************************************************
function handleImportList() {
const fileInput = document.getElementById("importListInput");
const replaceCheckbox = document.getElementById("replaceExcludeListCheckbox");
if (!fileInput || !fileInput.files || fileInput.files.length === 0) {
alert("No se seleccionó ningún archivo.");
return;
}
const reader = new FileReader();
reader.onload = function (event)
{
const rawLines = event.target.result.split(/\r?\n/);
const lines = rawLines
.map(line => line.replace(/[^\p{L}\p{N}().\s-]/gu, "").trim())
.filter(line => line.length > 0);
const eliminadas = rawLines.length - lines.length;
if (eliminadas > 0) {
console.warn(`[Importar Lista] ⚠️ ${eliminadas} líneas inválidas fueron ignoradas (vacías, caracteres no permitidos o basura).`);
}
if (replaceCheckbox && replaceCheckbox.checked) {
// Si se marcó reemplazar, limpiar todo
excludeWords = [];
} else {
// Si no, recuperar la actual del localStorage
excludeWords = JSON.parse(localStorage.getItem("excludeWords")) || excludeWords || [];
}
// Unificar, eliminar duplicados y ordenar
excludeWords = [...new Set([...excludeWords, ...lines])]
.filter(w => w.trim().length > 0)
.sort((a, b) => a.localeCompare(b));
localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
console.log("[handleImportList] Palabras actuales excluidas:", excludeWords);
renderExcludedWordsPanel(); // O renderExcludeWordList() si usas otro nombre
setupDragAndDropImport(); // Activa drag and drop
alert(`✅ Palabras excluidas importadas correctamente: ${excludeWords.length}`);
//Limpia el input para permitir recarga posterior
fileInput.value = "";
};
reader.readAsText(fileInput.files[0]);
}
//**************************************************************************
//Nombre: setupDragAndDropImport
//Fecha modificación: 2025-03-31
//Autor: mincho77
//Entradas: ninguna
//Salidas: habilita funcionalidad de arrastrar y soltar archivos al panel
//Descripción:
// Permite arrastrar y soltar un archivo .txt sobre el panel lateral (#normalizer-sidebar).
// Extrae las palabras del archivo y las agrega a excludeWords sin duplicados.
//**************************************************************************
function setupDragAndDropImport() {
const dropArea = document.getElementById("drop-zone");
if (!dropArea) {
console.warn("[setupDragAndDropImport] No se encontró la zona de drop");
return;
}
// Highlight drop zone when dragging over it
const highlight = (e) => {
e.preventDefault();
e.stopPropagation();
dropArea.classList.add("drag-over");
dropArea.style.backgroundColor = "#e8f5e9";
dropArea.style.borderColor = "#4CAF50";
};
// Remove highlighting
const unhighlight = (e) => {
e.preventDefault();
e.stopPropagation();
dropArea.classList.remove("drag-over");
dropArea.style.backgroundColor = "";
dropArea.style.borderColor = "#ccc";
};
// Handle the dropped files
const handleDrop = (e) => {
console.log("[handleDrop] Evento drop detectado");
e.preventDefault();
e.stopPropagation();
unhighlight(e);
const file = e.dataTransfer.files[0];
if (!file) {
alert("❌ No se detectó ningún archivo");
return;
}
// Validar extensión del archivo
if (!file.name.match(/\.(txt|xml)$/i)) {
alert("❌ Por favor, arrastra un archivo .txt o .xml");
return;
}
console.log(`[handleDrop] Procesando archivo: ${file.name}`);
const reader = new FileReader();
reader.onload = (event) => {
try {
const content = event.target.result;
const isXML = file.name.toLowerCase().endsWith('.xml');
let words = [];
if (isXML) {
const parser = new DOMParser();
const xmlDoc = parser.parseFromString(content, "text/xml");
if (xmlDoc.getElementsByTagName("parsererror").length > 0) {
throw new Error("XML inválido");
}
words = Array.from(xmlDoc.getElementsByTagName("word"))
.map(node => node.textContent.trim())
.filter(w => w.length > 0);
} else {
words = content.split(/\r?\n/)
.map(line => line.trim())
.filter(line => line.length > 0);
}
if (words.length === 0) {
alert("⚠️ No se encontraron palabras válidas en el archivo");
return;
}
// Actualizar lista de palabras excluidas
excludeWords = [...new Set([...excludeWords, ...words])].sort();
localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
renderExcludedWordsPanel();
alert(`✅ Se importaron ${words.length} palabras exitosamente`);
console.log(`[handleDrop] Importadas ${words.length} palabras`);
} catch (error) {
console.error("[handleDrop] Error procesando archivo:", error);
alert("❌ Error procesando el archivo");
}
};
reader.onerror = () => {
console.error("[handleDrop] Error leyendo archivo");
alert("❌ Error leyendo el archivo");
};
reader.readAsText(file);
};
// Attach the event listeners
dropArea.addEventListener("dragenter", highlight, false);
dropArea.addEventListener("dragover", highlight, false);
dropArea.addEventListener("dragleave", unhighlight, false);
dropArea.addEventListener("drop", handleDrop, false);
console.log("[setupDragAndDropImport] Eventos de drag & drop configurados");
}
// Función auxiliar para procesar el archivo
function handleImportedFile(content, isXML) {
let words = [];
if (isXML) {
const parser = new DOMParser();
const xmlDoc = parser.parseFromString(content, "text/xml");
const nodes = xmlDoc.getElementsByTagName("word");
words = Array.from(nodes).map(node => node.textContent.trim());
} else {
words = content.split(/\r?\n/).map(line => line.trim()).filter(line => line);
}
if (words.length === 0) {
alert("No se encontraron palabras válidas en el archivo");
return;
}
// Actualizar lista de palabras excluidas
excludeWords = [...new Set([...excludeWords, ...words])].sort();
localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
renderExcludedWordsPanel();
alert(`✅ Se importaron ${words.length} palabras exitosamente`);
}
//**************************************************************************
//Nombre: renderExcludedWordsPanel
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: ninguna (usa la variable global excludeWords)
//Salidas: ninguna (actualiza el DOM y el localStorage)
//Prerrequisitos si existen: Debe existir un contenedor con id="normalizer-sidebar"
//Descripción:
// Esta función limpia y vuelve a renderizar la lista de palabras excluidas
// dentro del panel lateral del normalizador. Muestra una lista ordenada
// alfabéticamente, evitando palabras vacías. Además, asegura que el localStorage
// se actualice correctamente con la lista limpia y depurada.
//**************************************************************************
function renderExcludedWordsPanel() {
const container = document.getElementById("normalizer-sidebar");
if (!container) {
console.warn("[renderExcludedWordsPanel] ❌ No se encontró el contenedor 'normalizer-sidebar'");
return;
}
//Limpiar el contenedor visual
container.innerHTML = "";
console.log("[renderExcludedWordsPanel] ✅ Contenedor limpiado.");
//Limpiar palabras vacías y ordenar
const sortedWords = excludeWords.filter(w => !!w).sort((a, b) => a.localeCompare(b));
console.log(`[renderExcludedWordsPanel] 📋 Lista excluida depurada: ${sortedWords.length} palabras`, sortedWords);
const excludeListSection = document.createElement("div");
excludeListSection.style.marginTop = "20px";
excludeListSection.innerHTML = `
<h4 style="margin-bottom: 5px;">Palabras Excluidas</h4>
<div style="max-height: 150px; overflow-y: auto; border: 1px solid #ccc; padding: 8px; font-size: 13px; border-radius: 4px;">
<ul style="margin: 0; padding-left: 18px;" id="excludeWordsList">
${sortedWords.map(w => `<li>${w}</li>`).join("")}
</ul>
</div>
`;
container.appendChild(excludeListSection);
console.log("[renderExcludedWordsPanel] ✅ Lista renderizada en el DOM.");
//Guardar en localStorage
localStorage.setItem("excludeWords", JSON.stringify(sortedWords));
console.log("[renderExcludedWordsPanel] 💾 excludeWords actualizado en localStorage.");
}
//**************************************************************************
//Nombre: normalizePlaceName (unsafeWindow)
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas:
// - name (string): el nombre original del lugar.
//Salidas:
// - string: nombre normalizado, respetando exclusiones y opciones del usuario.
//Prerrequisitos si existen:
// - Debe estar cargada la lista global excludeWords.
// - Debe existir un checkbox con id “normalizeArticles” para definir si se normalizan artículos.
//Descripción:
// Esta versión expuesta globalmente permite acceder a la normalización básica del nombre de un lugar
// desde otros contextos como el navegador o Tampermonkey. Capitaliza cada palabra, respeta las excluidas
// y no aplica normalización a artículos si el checkbox lo indica.
// Realiza limpieza básica: reemplazo de pipes, eliminación de espacios dobles y trim final.
//**************************************************************************
unsafeWindow.normalizePlaceName = function(name)
{
if (!name) return "";
const normalizeArticles = !document.getElementById("normalizeArticles")?.checked;
const articles = ["el", "la", "los", "las", "de", "del", "al", "y", "o"];
const words = name.trim().split(/\s+/);
const normalizedWords = words.map((word, index) => {
const lowerWord = word.toLowerCase();
// Saltar palabras excluidas
if (excludeWords.includes(word)) return word;
// Saltar artículos si el checkbox está activo y no es la primera palabra
if (!normalizeArticles && articles.includes(lowerWord) && index !== 0) {
return lowerWord;
}
//Mayúsculas
return word.charAt(0).toUpperCase() + word.slice(1).toLowerCase();
});
name = normalizedWords.join(" ");
name = name.replace(/\s*\|\s*/g, " - ");
name = name.replace(/\s{2,}/g, " ").trim();
return name;
};
//**************************************************************************
//Nombre: normalizePlaceNameOnly
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: name (string) – Nombre del lugar a normalizar.
//Salidas: texto normalizado sin validación ortográfica.
//Descripción:
// Realiza normalización visual del nombre: capitaliza, ajusta espacios,
// formatea guiones, paréntesis, y símbolos. No evalúa ortografía ni acentos.
//**************************************************************************
function normalizePlaceNameOnly(name) {
if (!name) return "";
const normalizeArticles = !document.getElementById("normalizeArticles")?.checked;
const articles = ["el", "la", "los", "las", "de", "del", "al", "y"];
const words = name.trim().split(/\s+/);
const isRoman = word => /^(i{1,3}|iv|v|vi{0,3}|ix|x|xi{0,3}|xiv|xv|xvi{0,3}|xix|xx|xxi{0,3}|xxiv|xxv)$/i.test(word);
const normalizedWords = words.map((word, index) => {
const lowerWord = word.toLowerCase();
// Si contiene "&", convertir a mayúsculas
if (/^[A-Za-z]&[A-Za-z]$/.test(word)) return word.toUpperCase();
// Verificar si está en la lista de excluidas
const matchExcluded = excludeWords.find(w => w.toLowerCase() === lowerWord);
if (matchExcluded) return matchExcluded;
// Si es un número romano, convertir a mayúsculas
if (isRoman(word)) return word.toUpperCase();
// Si no se deben normalizar artículos y es un artículo, mantener en minúsculas
if (!normalizeArticles && articles.includes(lowerWord) && index !== 0) return lowerWord;
// Si es un número o un símbolo como "-", no modificar
if (/^\d+$/.test(word) || word === "-") return word;
// Verificar ortografía usando la API de LanguageTool
return checkSpellingWithAPI(word)
.then(errors => {
if (errors.length > 0) {
const suggestion = errors[0].sugerencia || word;
return suggestion.charAt(0).toUpperCase() + suggestion.slice(1).toLowerCase();
}
return word.charAt(0).toUpperCase() + word.slice(1).toLowerCase();
})
.catch(() => word.charAt(0).toUpperCase() + word.slice(1).toLowerCase());
});
let newName = normalizedWords.join(" ")
.replace(/\s*\|\s*/g, " - ")
.replace(/([(["'])\s*([\p{L}])/gu, (match, p1, p2) => p1 + p2.toUpperCase())
.replace(/\s*-\s*/g, " - ")
.replace(/\b(\d+)([A-Z])\b/g, (match, num, letter) => num + letter.toUpperCase())
.replace(/\.$/, "")
.replace(/&(\s*)([A-Z])/g, (match, space, letter) => "&" + space + letter.toUpperCase());
return newName.replace(/\s{2,}/g, " ").trim();
}
//**************************************************************************
//Nombre: validarOrtografiaConAPI
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: texto (string) - palabra o frase a evaluar
//Salidas: Promesa con resultado de corrección ortográfica
//Descripción: Consulta la API pública de LanguageTool para identificar errores ortográficos
//**************************************************************************
function validarOrtografiaConAPI2(texto) {
return new Promise((resolve, reject) => {
GM_xmlhttpRequest({
method: "POST",
url: "https://api.languagetool.org/v2/check",
headers: {
"Content-Type": "application/x-www-form-urlencoded"
},
data: `text=${encodeURIComponent(texto)}&language=es`,
onload: function (response) {
if (response.status === 200) {
const result = JSON.parse(response.responseText);
resolve(result.matches || []);
} else {
reject("Error en la API de LanguageTool");
}
},
onerror: function () {
reject("Fallo la solicitud a la API de LanguageTool");
}
});
});
}
//**************************************************************************
//Nombre: importExcludeList
//Fecha modificación: 2025-03-31
//Autor: mincho77
//Entradas:
// file (File) – Archivo cargado por el usuario, que contiene una lista de palabras excluidas.
//Salidas:
// Ninguna (actualiza el array global excludeWords y el localStorage).
//Prerrequisitos:
// Debe existir un panel visual para mostrar la lista y una función renderExcludeList().
//Descripción:
// Lee un archivo .txt línea por línea, limpia espacios, elimina duplicados y vacíos,
// ordena alfabéticamente la lista resultante y actualiza la lista global de palabras
// excluidas (excludeWords). Guarda la lista en localStorage y actualiza el panel visual.
//**************************************************************************
function importExcludeList(file) {
const reader = new FileReader();
reader.onload = function (e) {
const newWords = e.target.result
.split(/\r?\n/)
.map(w => w.trim())
.filter(w => w.length > 0); // eliminar vacíos
excludeWords = [...new Set(newWords)].sort((a, b) => a.localeCompare(b));
localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
renderExcludeList(); // actualiza el panel visual
};
reader.readAsText(file);
}
//**************************************************************************
//Nombre: applySpellCorrection
//Fecha modificación: 2025-03-31
//Autor: mincho77
//Entradas:
// text (string) – Texto que se desea corregir automáticamente con sugerencias ortográficas.
//Salidas:
// Promise<string> – Texto corregido con las mejores sugerencias aplicadas.
//Prerrequisitos:
// Debe existir la función checkSpelling que retorna los errores detectados por LanguageTool.
//Descripción:
// Llama a checkSpelling y aplica la primera sugerencia de reemplazo para cada error ortográfico,
// procesando los reemplazos de atrás hacia adelante (para evitar desajustes de índice).
// Devuelve el texto corregido como resultado final.
//**************************************************************************
function applySpellCorrection(text) {
return checkSpelling(text).then(data => {
let corrected = text;
// Ordenar los matches de mayor a menor offset
const matches = data.matches.sort((a, b) => b.offset - a.offset);
matches.forEach(match => {
if (match.replacements && match.replacements.length > 0) {
const replacement = match.replacements[0].value;
corrected = corrected.substring(0, match.offset) + replacement + corrected.substring(match.offset + match.length);
}
});
return corrected;
});
}
//**************************************************************************
//Nombre: createSidebarTab
//Fecha modificación: 2025-03-31
//Autor: mincho77
//Entradas: Ninguna directa. Usa `W.userscripts` para registrar el tab.
//Salidas: Renderiza un nuevo tab lateral personalizado con interfaz del normalizador.
//Prerrequisitos:
// - `W.userscripts.registerSidebarTab` debe estar disponible.
// - `getSidebarHTML()` debe retornar el HTML necesario.
//Descripción:
// Crea y registra una nueva pestaña en el sidebar de WME con el título
// "Places Name Normalizer". Al cargar correctamente, inserta el HTML
// generado por `getSidebarHTML()` y espera a que se renderice completamente
// para ejecutar los eventos mediante `waitForDOM`.
//**************************************************************************
function createSidebarTab() {
try {
// Check if the sidebar system is ready
if (!W || !W.userscripts) {
console.error(`[${SCRIPT_NAME}] WME not ready for sidebar creation`);
return;
}
// Check for existing tab and clean up if needed
const existingTab = document.getElementById("normalizer-tab");
if (existingTab) {
console.log(`[${SCRIPT_NAME}] Removing existing tab...`);
existingTab.remove();
}
// Register new tab with error handling
let registration;
try {
registration = W.userscripts.registerSidebarTab("PlacesNormalizer");
} catch (e) {
if (e.message.includes("already been registered")) {
console.warn(`[${SCRIPT_NAME}] Tab registration conflict, attempting cleanup...`);
// Additional cleanup could go here
return;
}
throw e;
}
const { tabLabel, tabPane } = registration;
if (!tabLabel || !tabPane) {
throw new Error("Tab registration failed to return required elements");
}
// Configure tab
tabLabel.innerHTML = `
<img src=""
style="height: 16px; vertical-align: middle; margin-right: 5px;">
NrmliZer
`;
tabLabel.title = "Places Name Normalizer";
// Set content and attach events
tabPane.innerHTML = getSidebarHTML();
waitForDOM("#normalizer-tab", (element) => {
attachEvents();
console.log(`[${SCRIPT_NAME}] Tab created and events attached`);
}, 500, 10);
} catch (error) {
console.error(`[${SCRIPT_NAME}] Error in createSidebarTab:`, error);
}
}
//**************************************************************************
//Nombre: attachEvents
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: Ninguna
//Salidas: Ninguna
//Prerrequisitos si existen:
// - Deben existir en el DOM los elementos con los siguientes IDs:
// "normalizeArticles", "maxPlacesInput", "addExcludeWord", "scanPlaces",
// "hiddenImportInput", "importExcludeWordsUnifiedBtn" y "exportExcludeWords".
// - Debe existir la función handleImportList y la función scanPlaces.
// - Debe estar definida la variable global excludeWords y la función renderExcludedWordsPanel.
//Descripción:
// Esta función adjunta los event listeners necesarios para gestionar la interacción del usuario
// con el panel del normalizador de nombres. Se encargan de:
// - Actualizar la opción de normalizar artículos al cambiar el estado del checkbox.
// - Modificar el número máximo de lugares a procesar a través de un input.
// - Exportar la lista de palabras excluidas a un archivo XML.
// - Añadir nuevas palabras a la lista de palabras excluidas, evitando duplicados, y actualizar el panel.
// - Activar el botón unificado para la importación de palabras excluidas mediante un input oculto.
// - Ejecutar la función de escaneo de lugares al hacer clic en el botón correspondiente.
//**************************************************************************
function attachEvents()
{
console.log(`[${SCRIPT_NAME}] Adjuntando eventos...`);
const normalizeArticlesCheckbox = document.getElementById("normalizeArticles");
const maxPlacesInput = document.getElementById("maxPlacesInput");
const addExcludeWordButton = document.getElementById("addExcludeWord");
const scanPlacesButton = document.getElementById("scanPlaces");
const hiddenInput = document.getElementById("hiddenImportInput");
const importButtonUnified = document.getElementById("importExcludeWordsUnifiedBtn");
// Validación de elementos necesarios
if (!normalizeArticlesCheckbox || !maxPlacesInput || !addExcludeWordButton || !scanPlacesButton) {
console.error(`[${SCRIPT_NAME}] Error: No se encontraron elementos en el DOM.`);
return;
}
// ✅ Evento: cambiar estado de "no normalizar artículos"
normalizeArticlesCheckbox.addEventListener("change", (e) => {
normalizeArticles = e.target.checked;
});
// ✅ Evento: cambiar número máximo de places
maxPlacesInput.addEventListener("input", (e) => {
maxPlaces = parseInt(e.target.value, 10);
});
// ✅ Evento: exportar palabras excluidas a XML
document.getElementById("exportExcludeWords").addEventListener("click", () => {
const savedWords = JSON.parse(localStorage.getItem("excludeWords")) || [];
if (savedWords.length === 0) {
alert("No hay palabras excluidas para exportar.");
return;
}
const sortedWords = [...savedWords].sort((a, b) => a.localeCompare(b));
const xmlContent = `<?xml version="1.0" encoding="UTF-8"?>
<ExcludedWords>
${sortedWords.map(word => ` <word>${word}</word>`).join("\n ")}
</ExcludedWords>`;
const blob = new Blob([xmlContent], { type: "application/xml" });
const url = URL.createObjectURL(blob);
const link = document.createElement("a");
link.href = url;
link.download = "excluded_words.xml";
document.body.appendChild(link); // Correctly appends the link
link.click();
document.body.removeChild(link); // Correctly removes the link
});
// ✅ Evento: añadir palabra excluida sin duplicados
addExcludeWordButton.addEventListener("click", () => {
const wordInput = document.getElementById("excludeWord") || document.getElementById("excludedWord");
const word = wordInput?.value.trim();
if (!word) return;
const lowerWord = word.toLowerCase();
const alreadyExists = excludeWords.some(w => w.toLowerCase() === lowerWord);
if (!alreadyExists) {
excludeWords.push(word);
localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
renderExcludedWordsPanel(); // Usar renderExcludedWordsPanel en lugar de updateExcludeList
}
wordInput.value = "";
});
// ✅ Evento: nuevo botón unificado de importación
importButtonUnified.addEventListener("click", () => {
hiddenInput.click(); // abre el file input oculto
});
hiddenInput.addEventListener("change", () => {
handleImportList(); // ✅ Llama a la función handleImportList al importar
});
// ✅ Evento: escanear lugares
scanPlacesButton.addEventListener("click", scanPlaces);
}
function updateExcludeList() {
const list = document.getElementById("excludedWordsList");
if (!list) return;
// Ordena una copia del array para no alterar el original
const sortedWords = [...excludeWords].sort((a, b) => a.toLowerCase().localeCompare(b.toLowerCase()));
list.innerHTML = sortedWords.map(word => `<li>${word}</li>`).join("");
}
//**************************************************************************
//Nombre: scanPlaces
//Fecha modificación: 2025-04-02
//Autor: mincho77
//Entradas: ninguna directamente (usa datos del modelo WME y del input #maxPlacesInput)
//Salidas: abre panel flotante con lugares que deben ser normalizados
//Descripción:
// Escanea los lugares visibles en el WME, normaliza los nombres y verifica
// si les falta una tilde en las palabras que lo requieren. Si se selecciona
// "Revisar solo tildes", utiliza la función evaluarOrtografiaConTildes.
//**************************************************************************
// ==UserScript==
// @name WME Places Name Normalizer
// @namespace https://gf.qytechs.cn/en/users/mincho77
// @version 2.1
// @description Normaliza nombres de lugares en Waze Map Editor (WME) por ahora solo con reglas de Colombia
// @author mincho77
// @match https://www.waze.com/*editor*
// @match https://beta.waze.com/*user/editor*
// @grant GM_xmlhttpRequest
// @connect api.languagetool.org
// @connect *
// @grant unsafeWindow
// @license MIT
// @run-at document-end
// @downloadURL https://update.gf.qytechs.cn/scripts/530268/WME%20Places%20Name%20Normalizer.user.js
// @updateURL https://update.gf.qytechs.cn/scripts/530268/WME%20Places%20Name%20Normalizer.meta.js
// ==/UserScript==
/*global W*/
(() => {
"use strict";
try {
// Insertar estilos globales en el <head>
const styles = `
<style>
/* Estilos para los botones */
.apply-suggestion-btn {
background-color: #4CAF50;
color: white;
border: none;
padding: 8px 12px;
border-radius: 4px;
cursor: pointer;
font-size: 14px;
font-weight: bold;
transition: background-color 0.3s ease;
}
.apply-suggestion-btn:hover {
background-color: #45a049;
}
#apply-changes-btn {
background-color: #28a745;
color: white;
border: none;
padding: 10px 15px;
border-radius: 4px;
cursor: pointer;
font-size: 16px;
font-weight: bold;
display: flex;
align-items: center;
gap: 8px;
}
#apply-changes-btn:hover {
background-color: #218838;
}
#cancel-btn {
background-color: #dc3545;
color: white;
border: none;
padding: 10px 15px;
border-radius: 4px;
cursor: pointer;
font-size: 16px;
font-weight: bold;
display: flex;
align-items: center;
gap: 8px;
}
#cancel-btn:hover {
background-color: #c82333;
}
</style>
`;
document.head.insertAdjacentHTML('beforeend', styles);
// Capturar todos los eventos de drag & drop a nivel de <body>
// Agregar al inicio del script, justo después del "use strict"
// Prevenir comportamiento por defecto de drag & drop a nivel global
document.addEventListener("dragover", function(e) {
const dropZone = document.getElementById("drop-zone");
if (e.target === dropZone || dropZone?.contains(e.target)) {
return; // Permitir el drop en la zona designada
}
e.preventDefault();
e.stopPropagation();
}, { passive: false }); // Agrega { passive: false }
document.addEventListener("drop", function(e) {
const dropZone = document.getElementById("drop-zone");
if (e.target === dropZone || dropZone?.contains(e.target)) {
return; // Permitir el drop en la zona designada
}
e.preventDefault();
e.stopPropagation();
}, { passive: false }); // Agrega { passive: false }
//**************************************************************************
//Nombre: evaluarOrtografiaConTildes
//Fecha modificación: 2025-04-02
//Autor: mincho77
//Entradas: name (string) - Nombre del lugar
//Salidas: objeto con errores detectados
//Descripción:
// Evalúa palabra por palabra si falta una tilde en las palabras que lo requieren,
// según las reglas del español. Primero normaliza el nombre y luego verifica
// si las palabras necesitan una tilde.
//**************************************************************************
function evaluarOrtografiaConTildes(name) {
return new Promise((resolve) => {
if (!name) {
return resolve({
hasSpellingWarning: false,
spellingWarnings: []
});
}
const palabras = name.trim().split(/\s+/);
const spellingWarnings = [];
const totalPalabras = palabras.length;
let procesadas = 0;
// Función para procesar cada palabra secuencialmente
function procesarPalabra(index) {
if (index >= palabras.length) {
// Todas las palabras procesadas
toggleSpinner(false);
return resolve({
hasSpellingWarning: spellingWarnings.length > 0,
spellingWarnings
});
}
const palabra = palabras[index];
let normalizada = normalizePlaceNameOnly(palabra);
// Actualizar progreso
procesadas++;
const progress = Math.round((procesadas / totalPalabras) * 100);
toggleSpinner(true, 'Revisando tildes...', progress);
// Lógica de verificación existente
if (/^[A-Za-z]&[A-Za-z]$/.test(normalizada) ||
/^[\u263a-\u263c\u2764\u1f600-\u1f64f\u1f680-\u1f6ff]+$/.test(normalizada)) {
return procesarPalabra(index + 1);
}
if (normalizada.toLowerCase() === "y" || /^\d+$/.test(normalizada) || normalizada === "-") {
return procesarPalabra(index + 1);
}
if (excludeWords.some(w => w.toLowerCase() === normalizada.toLowerCase())) {
return procesarPalabra(index + 1);
}
const cantidadTildes = (normalizada.match(/[áéíóú]/g) || []).length;
if (cantidadTildes > 1) {
spellingWarnings.push({
original: palabra,
sugerida: null,
tipo: "Error de tildes",
posicion: index
});
}
// Procesar siguiente palabra
procesarPalabra(index + 1);
}
// Iniciar el procesamiento
toggleSpinner(true, 'Revisando tildes...', 0);
procesarPalabra(0);
});
}
// Función auxiliar para corregir tildes
function corregirTilde(palabra) {
// Implementar lógica para corregir tildes según las reglas del español
// Por ejemplo: "medellin" → "Medellín"
return palabra; // Retornar la palabra corregida
}
//**************************************************************************
//Nombre: toggleSpinner
//Fecha modificación: 2025-03-31
//Autor: mincho77
//Entradas:
// show (boolean) - true para mostrar el spinner, false para ocultarlo
// message (string, opcional) - mensaje personalizado a mostrar junto al spinner
//Salidas: ninguna (modifica el DOM)
//Prerrequisitos: debe existir el estilo CSS del spinner en el documento
//Descripción:
// Muestra u oculta un indicador visual de carga con un mensaje opcional.
// El spinner usa un emoji de reloj de arena (⏳) con animación de rotación
// para indicar que el proceso está en curso.
//**************************************************************************
function toggleSpinner(show, message = 'Revisando ortografía...', progress = null) {
let existingSpinner = document.querySelector('.spinner-overlay');
if (existingSpinner) {
if (show) {
// Actualizar el mensaje y el progreso si el spinner ya existe
const spinnerMessage = existingSpinner.querySelector('.spinner-message');
spinnerMessage.innerHTML = `
${message}
${progress !== null ? `<br><strong>${progress}% completado</strong>` : ''}
`;
} else {
existingSpinner.remove(); // Ocultar el spinner
}
return;
}
if (show) {
const spinner = document.createElement('div');
spinner.className = 'spinner-overlay';
spinner.innerHTML = `
<div class="spinner-content">
<div class="spinner-icon">⏳</div>
<div class="spinner-message">
${message}
${progress !== null ? `<br><strong>${progress}% completado</strong>` : ''}
</div>
</div>
`;
document.body.appendChild(spinner);
}
}
// Agregar los estilos CSS necesarios
const spinnerStyles = `
<style>
.spinner-overlay {
position: fixed;
top: 0;
left: 0;
width: 100%;
height: 100%;
background: rgba(0, 0, 0, 0.5);
display: flex;
justify-content: center;
align-items: center;
z-index: 10000;
}
.spinner-content {
background: white;
padding: 20px;
border-radius: 8px;
text-align: center;
box-shadow: 0 2px 10px rgba(0,0,0,0.3);
}
.spinner-icon {
font-size: 24px;
margin-bottom: 10px;
animation: spin 1s linear infinite; /* Aseguramos que la animación esté activa */
display: inline-block;
}
.spinner-message {
color: #333;
font-size: 14px;
}
@keyframes spin {
0% { transform: rotate(0deg); }
100% { transform: rotate(360deg); }
}
</style>`;
// Insertar los estilos al inicio del documento
document.head.insertAdjacentHTML('beforeend', spinnerStyles);
if (!Array.prototype.flat)
{
Array.prototype.flat = function(depth = 1)
{
return this.reduce(function (flat, toFlatten)
{
return flat.concat(Array.isArray(toFlatten) ? toFlatten.flat(depth - 1) : toFlatten);
}, []);
};
}
const SCRIPT_NAME = "PlacesNameNormalizer";
const VERSION = "2.1";
let placesToNormalize = [];
let excludeWords = [];
let maxPlaces = 100;
let normalizeArticles = true;
// Expresión regular para detectar siglas (por ejemplo, "S.a.s", "L.T.D.A")
const siglaRegex = /^[A-Za-z](\.[A-Za-z])+\.?$/;
//**************************************************************************
//Nombre: waitForSidebar
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas:
// retries (número, opcional) – Número máximo de intentos de verificación del sidebar (default: 20).
// delay (número, opcional) – Tiempo en milisegundos entre intentos (default: 1000ms).
//Salidas:
// Promesa que se resuelve con el elemento del sidebar si se encuentra, o se rechaza si no se encuentra después de los intentos.
//Descripción:
// Esta función espera a que el DOM cargue completamente el elemento con ID "sidebar".
// Realiza múltiples intentos con intervalos definidos, y resuelve la promesa cuando el sidebar esté disponible.
// Es útil para asegurarse de que el entorno de WME esté completamente cargado antes de continuar.
//**************************************************************************
function waitForSidebar(retries = 20, delay = 1000) {
return new Promise((resolve, reject) => {
const check = (attempt = 1) => {
const sidebar = document.querySelector("#sidebar");
if (sidebar) {
console.log("✅ Sidebar disponible.");
resolve(sidebar);
} else if (attempt <= retries) {
console.warn(`⚠️ Sidebar no disponible aún. Reintentando... (${attempt})`);
setTimeout(() => check(attempt + 1), delay);
} else {
reject("❌ Sidebar no disponible después de múltiples intentos.");
}
};
check();
});
}
//**************************************************************************
//Nombre: initializeExcludeWords
//Fecha modificación: 2025-03-31
//Autor: mincho77
//Entradas: Ninguna
//Salidas: Ninguna
//Prerrequisitos: excludeWords debe estar declarado globalmente.
//Descripción: Inicializa la lista de palabras excluidas desde localStorage sin borrar entradas ya existentes.
//**************************************************************************
function initializeExcludeWords() {
const saved = JSON.parse(localStorage.getItem("excludeWords")) || [];
// Combinar con las actuales sin duplicar
const merged = [...new Set([...saved, ...excludeWords])].sort((a, b) => a.localeCompare(b));
// Solo guardar si hay diferencias
const originalString = JSON.stringify(saved.sort());
const newString = JSON.stringify(merged);
if (originalString !== newString) {
localStorage.setItem("excludeWords", newString);
console.log(`[initializeExcludeWords] 💾 excludeWords actualizado con ${merged.length} palabras.`);
} else {
console.log(`[initializeExcludeWords] 🟢 Sin cambios en excludeWords.`);
}
// Actualizar variable global
excludeWords = merged;
}
//Modulo de Ortografía
// Insertar los estilos al inicio del documento
document.head.insertAdjacentHTML('beforeend', spinnerStyles);
if (!Array.prototype.flat)
{
Array.prototype.flat = function(depth = 1)
{
return this.reduce(function (flat, toFlatten)
{
return flat.concat(Array.isArray(toFlatten) ? toFlatten.flat(depth - 1) : toFlatten);
}, []);
};
}
//Modulo de Ortografía
//**************************************************************************
//Nombre: getUbicacionAcento
//Fecha modificación: 2025-03-27
//Autor: mincho77
//Entradas: palabra (string) - Palabra en español a evaluar
//Salidas: Posición del acento (última, penúltima, antepenúltima) o null si no aplica
//Prerrequisitos si existen: Ninguno
//Descripción: Determina la posición silábica del acento en una palabra según la ortografía del español.
//**************************************************************************
function getUbicacionAcento(palabra) {
const original = palabra;
const normalized = palabra.normalize("NFD").replace(/[\u0300-\u036f]/g, ""); // Remove diacritics
const silabas = normalized
.replace(/[^aeiou]/gi, "")
.match(/[aeiou]+/gi);
if (!silabas || silabas.length === 0) return null;
const conTilde = silabas.findIndex((s, i) => {
const originalSyllable = original.slice(
normalized.indexOf(s),
normalized.indexOf(s) + s.length
);
return /[áéíóú]/.test(originalSyllable);
});
if (conTilde !== -1) {
const posicion = silabas.length - 1 - conTilde;
if (posicion === 0) return "última";
if (posicion === 1) return "penúltima";
if (posicion === 2) return "antepenúltima";
return "otras";
}
return "sin tilde";
}
//**************************************************************************
//Nombre: checkSpellingWithAPI
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: text (string) – Texto a evaluar ortográficamente.
//Salidas: Promise – Resuelve con lista de errores ortográficos detectados.
//Prerrequisitos: Requiere permisos GM_xmlhttpRequest y @connect a api.languagetool.org
//Descripción:
// Consulta la API de LanguageTool para verificar ortografía del texto.
//**************************************************************************
function checkSpellingWithAPI(text) {
return new Promise((resolve, reject) => {
GM_xmlhttpRequest({
method: "POST",
url: "https://api.languagetool.org/v2/check",
headers: {
"Content-Type": "application/x-www-form-urlencoded"
},
data: `language=es&text=${encodeURIComponent(text)}`,
onload: function(response) {
if (response.status === 200) {
const result = JSON.parse(response.responseText);
const errores = result.matches
.filter(match => match.rule.issueType === "misspelling")
.map(match => ({
palabra: match.context.text.substring(match.context.offset, match.context.offset + match.context.length),
sugerencia: match.replacements.length > 0 ? match.replacements[0].value : "(sin sugerencia)"
}));
resolve(errores);
} else {
reject("❌ Error en respuesta de LanguageTool");
}
},
onerror: function(err) {
reject("❌ Error de red al contactar LanguageTool");
}
});
});
}
window.checkSpellingWithAPI = checkSpellingWithAPI;
// Combinar la lógica de checkSpelling
function checkSpelling(text) {
return new Promise((resolve, reject) => {
GM_xmlhttpRequest({
method: "POST",
url: "https://api.languagetool.org/v2/check",
data: `text=${encodeURIComponent(text)}&language=es`,
headers: {
"Content-Type": "application/x-www-form-urlencoded"
},
onload: function (response) {
if (response.status === 200) {
try {
const data = JSON.parse(response.responseText);
const errores = data.matches
.filter(match => match.rule.issueType === "misspelling")
.map(match => ({
palabra: match.context.text.substring(match.context.offset, match.context.offset + match.context.length),
sugerencia: match.replacements.length > 0 ? match.replacements[0].value : "(sin sugerencia)"
}));
resolve(errores);
} catch (err) {
reject(err);
}
} else {
reject(`Error HTTP: ${response.status}`);
}
},
onerror: function (err) {
reject(err);
}
});
});
}
//**************************************************************************
//Nombre: validarWordSpelling
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: palabra (string) - Palabra en español a validar ortográficamente
//Salidas: true si cumple reglas ortográficas básicas, false si no
//Descripción:
// Evalúa si una palabra tiene el uso correcto de tilde o si le falta una tilde
// según las reglas del español: esdrújulas siempre con tilde, agudas con tilde
// si terminan en n, s o vocal, y llanas con tildse si NO terminan en n, s o vocal.
// Se asegura que solo haya una tilde por palabra.
//**************************************************************************
function validarWordSpelling(palabra) {
if (!palabra) return false;
// Ignorar siglas con formato X&X
if (/^[A-Za-z]&[A-Za-z]$/.test(palabra)) return true;
// Si la palabra es un número, no necesita validación
if (/^\d+$/.test(palabra)) return true;
const tieneTilde = /[áéíóú]/.test(palabra);
const cantidadTildes = (palabra.match(/[áéíóú]/g) || []).length;
if (cantidadTildes > 1) return false; // Solo se permite una tilde
const silabas = palabra.normalize("NFD").replace(/[^aeiouAEIOU\u0300-\u036f]/g, "").match(/[aeiouáéíóú]+/gi);
if (!silabas || silabas.length === 0) return false;
const totalSilabas = silabas.length;
const ultimaLetra = palabra.slice(-1).toLowerCase();
let tipo = "";
if (totalSilabas >= 3 && /[áéíóú]/.test(palabra)) {
tipo = "esdrújula";
} else if (totalSilabas >= 2) {
const penultimaSilaba = silabas[totalSilabas - 2];
if (/[áéíóú]/.test(penultimaSilaba)) tipo = "grave";
}
if (!tipo) tipo = /[áéíóú]/.test(silabas[totalSilabas - 1]) ? "aguda" : "sin tilde";
if (tipo === "esdrújula") return tieneTilde;
if (tipo === "aguda") {
return (/[nsáéíóúaeiou]$/.test(ultimaLetra) && tieneTilde) ||
(!/[nsáéíóúaeiou]$/.test(ultimaLetra) && !tieneTilde);
}
if (tipo === "grave") {
return (!/[nsáéíóúaeiou]$/.test(ultimaLetra) && tieneTilde) ||
(/[nsáéíóúaeiou]$/.test(ultimaLetra) && !tieneTilde);
}
return true;
}
///**************************************************************************
//Nombre: separarSilabas
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: palabra (string) – Palabra en español.
//Salidas: array de sílabas (aproximado).
//Descripción:
// Separa una palabra en sílabas usando reglas heurísticas.
// Esta versión simplificada considera diptongos y combinaciones comunes.
//**************************************************************************
function separarSilabas(palabra) {
// Normaliza y quita acentos para facilitar la segmentación
const limpia = palabra.normalize("NFD").replace(/[\u0300-\u036f]/g, "").toLowerCase();
// Divide por vocales agrupadas como aproximación de sílabas
const silabas = limpia.match(/[bcdfghjklmnñpqrstvwxyz]*[aeiou]{1,2}[bcdfghjklmnñpqrstvwxyz]*/g);
return silabas || [palabra]; // fallback si no separa nada
}
//**************************************************************************
//Nombre: clasificarPalabra
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: silabas (array) – Arreglo con las sílabas de la palabra.
//Salidas: string – 'aguda', 'grave' (llana), 'esdrújula'.
//Descripción:
// Determina el tipo de palabra según el número de sílabas y la posición
// de la tilde (si existe).
//**************************************************************************
function clasificarPalabra(silabas) {
const palabra = silabas.join("");
const tieneTilde = /[áéíóúÁÉÍÓÚ]/.test(palabra);
if (tieneTilde) {
const posicionTilde = silabas.findIndex(s => /[áéíóúÁÉÍÓÚ]/.test(s));
const posicionDesdeFinal = silabas.length - 1 - posicionTilde;
if (posicionDesdeFinal === 0) return "aguda";
if (posicionDesdeFinal === 1) return "grave";
if (posicionDesdeFinal >= 2) return "esdrújula";
}
// Si no tiene tilde, asumimos que es:
if (silabas.length === 1) return "aguda";
if (silabas.length === 2) return "grave";
return "grave"; // por convención para 3+ sin tilde
}
//**************************************************************************
//Nombre: evaluarOrtografiaNombre
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: name (string) - Nombre del lugar
//Salidas: objeto con errores detectados
//Descripción: Evalúa palabra por palabra si hay errores ortográficos o falta de tildes.
// Ya no utiliza sugerencias automáticas para correcciones.
//**************************************************************************
function evaluarOrtografiaNombre(name) {
if (!name) return {
hasSpellingWarning: false,
spellingWarnings: []
};
const checkOnlyTildes = document.getElementById("checkOnlyTildes")?.checked;
const palabras = name.trim().split(/\s+/);
const spellingWarnings = [];
console.log(`[evaluarOrtografiaNombre] Verificando ortografía de: ${name}`);
if (checkOnlyTildes) {
palabras.forEach((palabra, index) => {
// Verificar si la palabra está en la lista de exclusiones
if (excludeWords.some(w => w.toLowerCase() === palabra.toLowerCase()) || /^\d+$/.test(palabra)) {
return; // Ignorar palabra excluida
}
if (!validarWordSpelling(palabra)) {
spellingWarnings.push({
original: palabra,
sugerida: null,
tipo: "Tilde",
posicion: index // Guardar la posición en la frase
});
}
});
return Promise.resolve({
hasSpellingWarning: spellingWarnings.length > 0,
spellingWarnings
});
} else {
return checkSpellingWithAPI(name)
.then(errores => {
errores.forEach(error => {
// Verificar si la palabra está en exclusiones
if (!excludeWords.some(w => w.toLowerCase() === error.palabra.toLowerCase())) {
spellingWarnings.push({
original: error.palabra,
sugerida: error.sugerencia,
tipo: "LanguageTool",
posicion: name.indexOf(error.palabra)
});
}
});
return {
hasSpellingWarning: spellingWarnings.length > 0,
spellingWarnings
};
});
}
}
//**************************************************************************
//Nombre: handleImportList
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: ninguna directa, depende del input file y checkbox en el DOM
//Salidas: actualiza excludeWords en localStorage y visualmente
//Prerrequisitos: existir un input file con id="importListInput" y checkbox con id="replaceExcludeListCheckbox"
//Descripción: Importa una lista de palabras para excluir, opcionalmente reemplazando la lista actual
//**************************************************************************
function handleImportList() {
const fileInput = document.getElementById("importListInput");
const replaceCheckbox = document.getElementById("replaceExcludeListCheckbox");
if (!fileInput || !fileInput.files || fileInput.files.length === 0) {
alert("No se seleccionó ningún archivo.");
return;
}
const reader = new FileReader();
reader.onload = function (event)
{
const rawLines = event.target.result.split(/\r?\n/);
const lines = rawLines
.map(line => line.replace(/[^\p{L}\p{N}().\s-]/gu, "").trim())
.filter(line => line.length > 0);
const eliminadas = rawLines.length - lines.length;
if (eliminadas > 0) {
console.warn(`[Importar Lista] ⚠️ ${eliminadas} líneas inválidas fueron ignoradas (vacías, caracteres no permitidos o basura).`);
}
if (replaceCheckbox && replaceCheckbox.checked) {
// Si se marcó reemplazar, limpiar todo
excludeWords = [];
} else {
// Si no, recuperar la actual del localStorage
excludeWords = JSON.parse(localStorage.getItem("excludeWords")) || excludeWords || [];
}
// Unificar, eliminar duplicados y ordenar
excludeWords = [...new Set([...excludeWords, ...lines])]
.filter(w => w.trim().length > 0)
.sort((a, b) => a.localeCompare(b));
localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
console.log("[handleImportList] Palabras actuales excluidas:", excludeWords);
renderExcludedWordsPanel(); // O renderExcludeWordList() si usas otro nombre
setupDragAndDropImport(); // Activa drag and drop
alert(`✅ Palabras excluidas importadas correctamente: ${excludeWords.length}`);
//Limpia el input para permitir recarga posterior
fileInput.value = "";
};
reader.readAsText(fileInput.files[0]);
}
//**************************************************************************
//Nombre: setupDragAndDropImport
//Fecha modificación: 2025-03-31
//Autor: mincho77
//Entradas: ninguna
//Salidas: habilita funcionalidad de arrastrar y soltar archivos al panel
//Descripción:
// Permite arrastrar y soltar un archivo .txt sobre el panel lateral (#normalizer-sidebar).
// Extrae las palabras del archivo y las agrega a excludeWords sin duplicados.
//**************************************************************************
function setupDragAndDropImport() {
const dropArea = document.getElementById("drop-zone");
if (!dropArea) {
console.warn("[setupDragAndDropImport] No se encontró la zona de drop");
return;
}
// Highlight drop zone when dragging over it
const highlight = (e) => {
e.preventDefault();
e.stopPropagation();
dropArea.classList.add("drag-over");
dropArea.style.backgroundColor = "#e8f5e9";
dropArea.style.borderColor = "#4CAF50";
};
// Remove highlighting
const unhighlight = (e) => {
e.preventDefault();
e.stopPropagation();
dropArea.classList.remove("drag-over");
dropArea.style.backgroundColor = "";
dropArea.style.borderColor = "#ccc";
};
// Handle the dropped files
const handleDrop = (e) => {
console.log("[handleDrop] Evento drop detectado");
e.preventDefault();
e.stopPropagation();
unhighlight(e);
const file = e.dataTransfer.files[0];
if (!file) {
alert("❌ No se detectó ningún archivo");
return;
}
// Validar extensión del archivo
if (!file.name.match(/\.(txt|xml)$/i)) {
alert("❌ Por favor, arrastra un archivo .txt o .xml");
return;
}
console.log(`[handleDrop] Procesando archivo: ${file.name}`);
const reader = new FileReader();
reader.onload = (event) => {
try {
const content = event.target.result;
const isXML = file.name.toLowerCase().endsWith('.xml');
let words = [];
if (isXML) {
const parser = new DOMParser();
const xmlDoc = parser.parseFromString(content, "text/xml");
if (xmlDoc.getElementsByTagName("parsererror").length > 0) {
throw new Error("XML inválido");
}
words = Array.from(xmlDoc.getElementsByTagName("word"))
.map(node => node.textContent.trim())
.filter(w => w.length > 0);
} else {
words = content.split(/\r?\n/)
.map(line => line.trim())
.filter(line => line.length > 0);
}
if (words.length === 0) {
alert("⚠️ No se encontraron palabras válidas en el archivo");
return;
}
// Actualizar lista de palabras excluidas
excludeWords = [...new Set([...excludeWords, ...words])].sort();
localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
renderExcludedWordsPanel();
alert(`✅ Se importaron ${words.length} palabras exitosamente`);
console.log(`[handleDrop] Importadas ${words.length} palabras`);
} catch (error) {
console.error("[handleDrop] Error procesando archivo:", error);
alert("❌ Error procesando el archivo");
}
};
reader.onerror = () => {
console.error("[handleDrop] Error leyendo archivo");
alert("❌ Error leyendo el archivo");
};
reader.readAsText(file);
};
// Attach the event listeners
dropArea.addEventListener("dragenter", highlight, false);
dropArea.addEventListener("dragover", highlight, false);
dropArea.addEventListener("dragleave", unhighlight, false);
dropArea.addEventListener("drop", handleDrop, false);
console.log("[setupDragAndDropImport] Eventos de drag & drop configurados");
}
// Función auxiliar para procesar el archivo
function handleImportedFile(content, isXML) {
let words = [];
if (isXML) {
const parser = new DOMParser();
const xmlDoc = parser.parseFromString(content, "text/xml");
const nodes = xmlDoc.getElementsByTagName("word");
words = Array.from(nodes).map(node => node.textContent.trim());
} else {
words = content.split(/\r?\n/).map(line => line.trim()).filter(line => line);
}
if (words.length === 0) {
alert("No se encontraron palabras válidas en el archivo");
return;
}
// Actualizar lista de palabras excluidas
excludeWords = [...new Set([...excludeWords, ...words])].sort();
localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
renderExcludedWordsPanel();
alert(`✅ Se importaron ${words.length} palabras exitosamente`);
}
//**************************************************************************
//Nombre: renderExcludedWordsPanel
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: ninguna (usa la variable global excludeWords)
//Salidas: ninguna (actualiza el DOM y el localStorage)
//Prerrequisitos si existen: Debe existir un contenedor con id="normalizer-sidebar"
//Descripción:
// Esta función limpia y vuelve a renderizar la lista de palabras excluidas
// dentro del panel lateral del normalizador. Muestra una lista ordenada
// alfabéticamente, evitando palabras vacías. Además, asegura que el localStorage
// se actualice correctamente con la lista limpia y depurada.
//**************************************************************************
function renderExcludedWordsPanel() {
const container = document.getElementById("normalizer-sidebar");
if (!container) {
console.warn("[renderExcludedWordsPanel] ❌ No se encontró el contenedor 'normalizer-sidebar'");
return;
}
//Limpiar el contenedor visual
container.innerHTML = "";
console.log("[renderExcludedWordsPanel] ✅ Contenedor limpiado.");
//Limpiar palabras vacías y ordenar
const sortedWords = excludeWords.filter(w => !!w).sort((a, b) => a.localeCompare(b));
console.log(`[renderExcludedWordsPanel] 📋 Lista excluida depurada: ${sortedWords.length} palabras`, sortedWords);
const excludeListSection = document.createElement("div");
excludeListSection.style.marginTop = "20px";
excludeListSection.innerHTML = `
<h4 style="margin-bottom: 5px;">Palabras Excluidas</h4>
<div style="max-height: 150px; overflow-y: auto; border: 1px solid #ccc; padding: 8px; font-size: 13px; border-radius: 4px;">
<ul style="margin: 0; padding-left: 18px;" id="excludeWordsList">
${sortedWords.map(w => `<li>${w}</li>`).join("")}
</ul>
</div>
`;
container.appendChild(excludeListSection);
console.log("[renderExcludedWordsPanel] ✅ Lista renderizada en el DOM.");
//Guardar en localStorage
localStorage.setItem("excludeWords", JSON.stringify(sortedWords));
console.log("[renderExcludedWordsPanel] 💾 excludeWords actualizado en localStorage.");
}
//**************************************************************************
//Nombre: normalizePlaceName (unsafeWindow)
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas:
// - name (string): el nombre original del lugar.
//Salidas:
// - string: nombre normalizado, respetando exclusiones y opciones del usuario.
//Prerrequisitos si existen:
// - Debe estar cargada la lista global excludeWords.
// - Debe existir un checkbox con id “normalizeArticles” para definir si se normalizan artículos.
//Descripción:
// Esta versión expuesta globalmente permite acceder a la normalización básica del nombre de un lugar
// desde otros contextos como el navegador o Tampermonkey. Capitaliza cada palabra, respeta las excluidas
// y no aplica normalización a artículos si el checkbox lo indica.
// Realiza limpieza básica: reemplazo de pipes, eliminación de espacios dobles y trim final.
//**************************************************************************
unsafeWindow.normalizePlaceName = function(name)
{
if (!name) return "";
const normalizeArticles = !document.getElementById("normalizeArticles")?.checked;
const articles = ["el", "la", "los", "las", "de", "del", "al", "y", "o"];
const words = name.trim().split(/\s+/);
const normalizedWords = words.map((word, index) => {
const lowerWord = word.toLowerCase();
// Saltar palabras excluidas
if (excludeWords.includes(word)) return word;
// Saltar artículos si el checkbox está activo y no es la primera palabra
if (!normalizeArticles && articles.includes(lowerWord) && index !== 0) {
return lowerWord;
}
//Mayúsculas
return word.charAt(0).toUpperCase() + word.slice(1).toLowerCase();
});
name = normalizedWords.join(" ");
name = name.replace(/\s*\|\s*/g, " - ");
name = name.replace(/\s{2,}/g, " ").trim();
return name;
};
//**************************************************************************
//Nombre: normalizePlaceNameOnly
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: name (string) – Nombre del lugar a normalizar.
//Salidas: texto normalizado sin validación ortográfica.
//Descripción:
// Realiza normalización visual del nombre: capitaliza, ajusta espacios,
// formatea guiones, paréntesis, y símbolos. No evalúa ortografía ni acentos.
//**************************************************************************
function normalizePlaceNameOnly(name) {
if (!name) return "";
const normalizeArticles = !document.getElementById("normalizeArticles")?.checked;
const articles = ["el", "la", "los", "las", "de", "del", "al", "y"];
const words = name.trim().split(/\s+/);
const isRoman = word => /^(i{1,3}|iv|v|vi{0,3}|ix|x|xi{0,3}|xiv|xv|xvi{0,3}|xix|xx|xxi{0,3}|xxiv|xxv)$/i.test(word);
const normalizedWords = words.map((word, index) => {
const lowerWord = word.toLowerCase();
// Si contiene "&", convertir a mayúsculas
if (/^[A-Za-z]&[A-Za-z]$/.test(word)) return word.toUpperCase();
// Verificar si está en la lista de excluidas
const matchExcluded = excludeWords.find(w => w.toLowerCase() === lowerWord);
if (matchExcluded) return matchExcluded;
// Si es un número romano, convertir a mayúsculas
if (isRoman(word)) return word.toUpperCase();
// Si no se deben normalizar artículos y es un artículo, mantener en minúsculas
if (!normalizeArticles && articles.includes(lowerWord) && index !== 0) return lowerWord;
// Si es un número o un símbolo como "-", no modificar
if (/^\d+$/.test(word) || word === "-") return word;
// Verificar ortografía usando la API de LanguageTool
return checkSpellingWithAPI(word)
.then(errors => {
if (errors.length > 0) {
const suggestion = errors[0].sugerencia || word;
return suggestion.charAt(0).toUpperCase() + suggestion.slice(1).toLowerCase();
}
return word.charAt(0).toUpperCase() + word.slice(1).toLowerCase();
})
.catch(() => word.charAt(0).toUpperCase() + word.slice(1).toLowerCase());
});
let newName = normalizedWords.join(" ")
.replace(/\s*\|\s*/g, " - ")
.replace(/([(["'])\s*([\p{L}])/gu, (match, p1, p2) => p1 + p2.toUpperCase())
.replace(/\s*-\s*/g, " - ")
.replace(/\b(\d+)([A-Z])\b/g, (match, num, letter) => num + letter.toUpperCase())
.replace(/\.$/, "")
.replace(/&(\s*)([A-Z])/g, (match, space, letter) => "&" + space + letter.toUpperCase());
return newName.replace(/\s{2,}/g, " ").trim();
}
//**************************************************************************
//Nombre: validarOrtografiaConAPI
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: texto (string) - palabra o frase a evaluar
//Salidas: Promesa con resultado de corrección ortográfica
//Descripción: Consulta la API pública de LanguageTool para identificar errores ortográficos
//**************************************************************************
function validarOrtografiaConAPI2(texto) {
return new Promise((resolve, reject) => {
GM_xmlhttpRequest({
method: "POST",
url: "https://api.languagetool.org/v2/check",
headers: {
"Content-Type": "application/x-www-form-urlencoded"
},
data: `text=${encodeURIComponent(texto)}&language=es`,
onload: function (response) {
if (response.status === 200) {
const result = JSON.parse(response.responseText);
resolve(result.matches || []);
} else {
reject("Error en la API de LanguageTool");
}
},
onerror: function () {
reject("Fallo la solicitud a la API de LanguageTool");
}
});
});
}
//**************************************************************************
//Nombre: importExcludeList
//Fecha modificación: 2025-03-31
//Autor: mincho77
//Entradas:
// file (File) – Archivo cargado por el usuario, que contiene una lista de palabras excluidas.
//Salidas:
// Ninguna (actualiza el array global excludeWords y el localStorage).
//Prerrequisitos:
// Debe existir un panel visual para mostrar la lista y una función renderExcludeList().
//Descripción:
// Lee un archivo .txt línea por línea, limpia espacios, elimina duplicados y vacíos,
// ordena alfabéticamente la lista resultante y actualiza la lista global de palabras
// excluidas (excludeWords). Guarda la lista en localStorage y actualiza el panel visual.
//**************************************************************************
function importExcludeList(file) {
const reader = new FileReader();
reader.onload = function (e) {
const newWords = e.target.result
.split(/\r?\n/)
.map(w => w.trim())
.filter(w => w.length > 0); // eliminar vacíos
excludeWords = [...new Set(newWords)].sort((a, b) => a.localeCompare(b));
localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
renderExcludeList(); // actualiza el panel visual
};
reader.readAsText(file);
}
//**************************************************************************
//Nombre: applySpellCorrection
//Fecha modificación: 2025-03-31
//Autor: mincho77
//Entradas:
// text (string) – Texto que se desea corregir automáticamente con sugerencias ortográficas.
//Salidas:
// Promise<string> – Texto corregido con las mejores sugerencias aplicadas.
//Prerrequisitos:
// Debe existir la función checkSpelling que retorna los errores detectados por LanguageTool.
//Descripción:
// Llama a checkSpelling y aplica la primera sugerencia de reemplazo para cada error ortográfico,
// procesando los reemplazos de atrás hacia adelante (para evitar desajustes de índice).
// Devuelve el texto corregido como resultado final.
//**************************************************************************
function applySpellCorrection(text) {
return checkSpelling(text).then(data => {
let corrected = text;
// Ordenar los matches de mayor a menor offset
const matches = data.matches.sort((a, b) => b.offset - a.offset);
matches.forEach(match => {
if (match.replacements && match.replacements.length > 0) {
const replacement = match.replacements[0].value;
corrected = corrected.substring(0, match.offset) + replacement + corrected.substring(match.offset + match.length);
}
});
return corrected;
});
}
//**************************************************************************
//Nombre: createSidebarTab
//Fecha modificación: 2025-03-31
//Autor: mincho77
//Entradas: Ninguna directa. Usa `W.userscripts` para registrar el tab.
//Salidas: Renderiza un nuevo tab lateral personalizado con interfaz del normalizador.
//Prerrequisitos:
// - `W.userscripts.registerSidebarTab` debe estar disponible.
// - `getSidebarHTML()` debe retornar el HTML necesario.
//Descripción:
// Crea y registra una nueva pestaña en el sidebar de WME con el título
// "Places Name Normalizer". Al cargar correctamente, inserta el HTML
// generado por `getSidebarHTML()` y espera a que se renderice completamente
// para ejecutar los eventos mediante `waitForDOM`.
//**************************************************************************
function createSidebarTab() {
try {
// Check if the sidebar system is ready
if (!W || !W.userscripts) {
console.error(`[${SCRIPT_NAME}] WME not ready for sidebar creation`);
return;
}
// Check for existing tab and clean up if needed
const existingTab = document.getElementById("normalizer-tab");
if (existingTab) {
console.log(`[${SCRIPT_NAME}] Removing existing tab...`);
existingTab.remove();
}
// Register new tab with error handling
let registration;
try {
registration = W.userscripts.registerSidebarTab("PlacesNormalizer");
} catch (e) {
if (e.message.includes("already been registered")) {
console.warn(`[${SCRIPT_NAME}] Tab registration conflict, attempting cleanup...`);
// Additional cleanup could go here
return;
}
throw e;
}
const { tabLabel, tabPane } = registration;
if (!tabLabel || !tabPane) {
throw new Error("Tab registration failed to return required elements");
}
// Configure tab
tabLabel.innerHTML = `
<img src=""
style="height: 16px; vertical-align: middle; margin-right: 5px;">
NrmliZer
`;
tabLabel.title = "Places Name Normalizer";
// Set content and attach events
tabPane.innerHTML = getSidebarHTML();
waitForDOM("#normalizer-tab", (element) => {
attachEvents();
console.log(`[${SCRIPT_NAME}] Tab created and events attached`);
}, 500, 10);
} catch (error) {
console.error(`[${SCRIPT_NAME}] Error in createSidebarTab:`, error);
}
}
//**************************************************************************
//Nombre: attachEvents
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: Ninguna
//Salidas: Ninguna
//Prerrequisitos si existen:
// - Deben existir en el DOM los elementos con los siguientes IDs:
// "normalizeArticles", "maxPlacesInput", "addExcludeWord", "scanPlaces",
// "hiddenImportInput", "importExcludeWordsUnifiedBtn" y "exportExcludeWords".
// - Debe existir la función handleImportList y la función scanPlaces.
// - Debe estar definida la variable global excludeWords y la función renderExcludedWordsPanel.
//Descripción:
// Esta función adjunta los event listeners necesarios para gestionar la interacción del usuario
// con el panel del normalizador de nombres. Se encargan de:
// - Actualizar la opción de normalizar artículos al cambiar el estado del checkbox.
// - Modificar el número máximo de lugares a procesar a través de un input.
// - Exportar la lista de palabras excluidas a un archivo XML.
// - Añadir nuevas palabras a la lista de palabras excluidas, evitando duplicados, y actualizar el panel.
// - Activar el botón unificado para la importación de palabras excluidas mediante un input oculto.
// - Ejecutar la función de escaneo de lugares al hacer clic en el botón correspondiente.
//**************************************************************************
function attachEvents()
{
console.log(`[${SCRIPT_NAME}] Adjuntando eventos...`);
const normalizeArticlesCheckbox = document.getElementById("normalizeArticles");
const maxPlacesInput = document.getElementById("maxPlacesInput");
const addExcludeWordButton = document.getElementById("addExcludeWord");
const scanPlacesButton = document.getElementById("scanPlaces");
const hiddenInput = document.getElementById("hiddenImportInput");
const importButtonUnified = document.getElementById("importExcludeWordsUnifiedBtn");
// Validación de elementos necesarios
if (!normalizeArticlesCheckbox || !maxPlacesInput || !addExcludeWordButton || !scanPlacesButton) {
console.error(`[${SCRIPT_NAME}] Error: No se encontraron elementos en el DOM.`);
return;
}
// ✅ Evento: cambiar estado de "no normalizar artículos"
normalizeArticlesCheckbox.addEventListener("change", (e) => {
normalizeArticles = e.target.checked;
});
// ✅ Evento: cambiar número máximo de places
maxPlacesInput.addEventListener("input", (e) => {
maxPlaces = parseInt(e.target.value, 10);
});
// ✅ Evento: exportar palabras excluidas a XML
document.getElementById("exportExcludeWords").addEventListener("click", () => {
const savedWords = JSON.parse(localStorage.getItem("excludeWords")) || [];
if (savedWords.length === 0) {
alert("No hay palabras excluidas para exportar.");
return;
}
const sortedWords = [...savedWords].sort((a, b) => a.localeCompare(b));
const xmlContent = `<?xml version="1.0" encoding="UTF-8"?>
<ExcludedWords>
${sortedWords.map(word => ` <word>${word}</word>`).join("\n ")}
</ExcludedWords>`;
const blob = new Blob([xmlContent], { type: "application/xml" });
const url = URL.createObjectURL(blob);
const link = document.createElement("a");
link.href = url;
link.download = "excluded_words.xml";
document.body.appendChild(link); // Correctly appends the link
link.click();
document.body.removeChild(link); // Correctly removes the link
});
// ✅ Evento: añadir palabra excluida sin duplicados
addExcludeWordButton.addEventListener("click", () => {
const wordInput = document.getElementById("excludeWord") || document.getElementById("excludedWord");
const word = wordInput?.value.trim();
if (!word) return;
const lowerWord = word.toLowerCase();
const alreadyExists = excludeWords.some(w => w.toLowerCase() === lowerWord);
if (!alreadyExists) {
excludeWords.push(word);
localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
renderExcludedWordsPanel(); // Usar renderExcludedWordsPanel en lugar de updateExcludeList
}
wordInput.value = "";
});
// ✅ Evento: nuevo botón unificado de importación
importButtonUnified.addEventListener("click", () => {
hiddenInput.click(); // abre el file input oculto
});
hiddenInput.addEventListener("change", () => {
handleImportList(); // ✅ Llama a la función handleImportList al importar
});
// ✅ Evento: escanear lugares
scanPlacesButton.addEventListener("click", scanPlaces);
}
function updateExcludeList() {
const list = document.getElementById("excludedWordsList");
if (!list) return;
// Ordena una copia del array para no alterar el original
const sortedWords = [...excludeWords].sort((a, b) => a.toLowerCase().localeCompare(b.toLowerCase()));
list.innerHTML = sortedWords.map(word => `<li>${word}</li>`).join("");
}
//**************************************************************************
//Nombre: scanPlaces
//Fecha modificación: 2025-04-02
//Autor: mincho77
//Entradas: ninguna directamente (usa datos del modelo WME y del input #maxPlacesInput)
//Salidas: abre panel flotante con lugares que deben ser normalizados
//Descripción:
// Escanea los lugares visibles en el WME, normaliza los nombres y verifica
// si les falta una tilde en las palabras que lo requieren. Si se selecciona
// "Revisar solo tildes", utiliza la función evaluarOrtografiaConTildes.
//**************************************************************************
function scanPlaces() {
console.log("scanPlaces ejecutado");
if (!W || !W.model || !W.model.venues || !W.model.venues.objects) {
console.error(`[${SCRIPT_NAME}] WME no está listo.`);
return;
}
const allPlaces = Object.values(W.model.venues.objects);
console.log(`[${SCRIPT_NAME}] Total de lugares encontrados: ${allPlaces.length}`);
const maxPlaces = parseInt(document.getElementById("maxPlacesInput").value, 10) || 20;
const checkOnlyTildes = document.getElementById("checkOnlyTildes")?.checked; // Verificar el estado del checkbox
const placesToNormalize = allPlaces
.filter(p => {
let isValid = true;
if (!p) {
console.log("Lugar descartado: p es null o undefined", p);
isValid = false;
} else if (typeof p.getID !== "function") {
console.log("Lugar descartado: p.getID no es una función", p);
isValid = false;
} else if (!p.attributes) {
console.log("Lugar descartado: p.attributes es null o undefined", p);
isValid = false;
} else if (typeof p.attributes.name !== "string") {
console.log("Lugar descartado: p.attributes.name no es una cadena", p);
isValid = false;
} else if (!p.attributes.name.trim()) {
console.log("Lugar descartado: p.attributes.name está vacío después de trim", p);
isValid = false;
}
return isValid;
})
.slice(0, maxPlaces)
.map(place => ({
id: place.getID(),
name: place.attributes.name,
attributes: place.attributes,
place
}));
console.log("placesToNormalize:", placesToNormalize);
toggleSpinner(true, 'Revisando ortografía...', 0); // Mostrar spinner al inicio con 0%
const totalPlaces = placesToNormalize.length;
let processedPlaces = 0;
let results = []; // Almacenar resultados parciales
function processPlace(index) {
if (index >= totalPlaces) {
toggleSpinner(false); // Ocultar spinner al terminar
if (results.length === 0) {
alert("No se encontraron Places que requieran cambio.");
} else {
openFloatingPanel(results); // Mostrar resultados parciales
}
return;
}
const place = placesToNormalize[index];
const originalName = place.name;
const normalized = normalizePlaceName(originalName);
const ortografia = checkOnlyTildes
? evaluarOrtografiaConTildes(normalized) // Usar la función para revisar solo tildes
: evaluarOrtografiaNombre(normalized);
Promise.resolve(ortografia)
.then(ortografiaResult => {
processedPlaces++;
const progress = Math.round((processedPlaces / totalPlaces) * 100);
toggleSpinner(true, 'Revisando ortografía...', progress); // Actualizar progreso
results.push({
id: place.id,
originalName,
newName: normalized,
hasSpellingWarning: ortografiaResult.hasSpellingWarning,
spellingWarnings: ortografiaResult.spellingWarnings
});
processPlace(index + 1); // Procesar el siguiente lugar
})
.catch(error => {
console.error("Error durante el escaneo de lugares:", error);
alert(`Error durante el escaneo de lugares: ${error}`);
toggleSpinner(false); // Ocultar spinner en caso de error
openFloatingPanel(results); // Mostrar resultados parciales
});
}
processPlace(0); // Iniciar el procesamiento desde el primer lugar
}
//**************************************************************************
//Nombre: applyNormalization
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: Ninguna
//Salidas: Ninguna
//Prerrequisitos si existen:
// - Debe existir un elemento en el DOM con las clases .normalize-checkbox, .delete-checkbox y .new-name-input.
// - El objeto global W debe estar disponible, incluyendo W.model.venues, W.model.actionManager y W.controller.
// - Deben estar definidos los módulos "Waze/Action/UpdateObject" y "Waze/Action/DeleteObject" (accesibles mediante require()).
// - Debe existir la variable placesToNormalize, que contiene datos de los lugares a normalizar, incluyendo sugerencias ortográficas.
//Descripción:
// Esta función aplica la normalización y/o eliminación de nombres de lugares en el Waze Map Editor
// según las selecciones realizadas en el panel flotante. Primero, obtiene los checkboxes seleccionados
// para normalización y eliminación. Si no hay ningún elemento seleccionado, se informa y se cancela la operación.
// Si se han seleccionado TODOS los checkboxes de eliminación, se solicita una confirmación adicional.
// Para cada checkbox de normalización seleccionado, se verifica si se debe aplicar la sugerencia ortográfica
// (cuando se ha hecho clic en el botón correspondiente) o el nombre completo modificado, y se actualiza el lugar
// mediante la acción de actualización. Posteriormente, se procesan los checkboxes de eliminación aplicando la
// acción de eliminación a los lugares correspondientes. Si se realizaron cambios, se marca el modelo como modificado.
// Finalmente, se cierra el panel flotante.
//**************************************************************************
function applyNormalization() {
const normalizeCheckboxes = document.querySelectorAll(".normalize-checkbox:checked");
const deleteCheckboxes = document.querySelectorAll(".delete-checkbox:checked");
let changesMade = false;
if (normalizeCheckboxes.length === 0 && deleteCheckboxes.length === 0) {
alert("No hay lugares seleccionados para normalizar o eliminar.");
return;
}
// Procesar normalización
normalizeCheckboxes.forEach(cb => {
const index = cb.dataset.index;
const input = document.querySelector(`.new-name-input[data-index="${index}"]`);
const newName = input?.value?.trim();
const placeId = input?.getAttribute("data-place-id");
const place = W.model.venues.getObjectById(placeId);
if (!place || !place.attributes?.name) {
console.warn(`No se encontró el lugar con ID: ${placeId}`);
return;
}
const currentName = place.attributes.name.trim();
if (currentName !== newName) {
try {
const UpdateObject = require("Waze/Action/UpdateObject");
const action = new UpdateObject(place, { name: newName });
W.model.actionManager.add(action);
console.log(`Nombre actualizado: "${currentName}" → "${newName}"`);
changesMade = true;
} catch (error) {
console.error("Error aplicando la acción de actualización:", error);
}
}
});
// Procesar eliminación
deleteCheckboxes.forEach(cb => {
const index = cb.dataset.index;
const placeId = document.querySelector(`.new-name-input[data-index="${index}"]`)?.getAttribute("data-place-id");
const place = W.model.venues.getObjectById(placeId);
if (!place) {
console.warn(`No se encontró el lugar con ID para eliminar: ${placeId}`);
return;
}
try {
const DeleteObject = require("Waze/Action/DeleteObject");
const deleteAction = new DeleteObject(place);
W.model.actionManager.add(deleteAction);
console.log(`Lugar eliminado: ${placeId}`);
changesMade = true;
} catch (error) {
console.error("Error eliminando el lugar:", error);
}
});
if (changesMade) {
alert("Cambios aplicados correctamente.");
} else {
alert("No se realizaron cambios.");
}
// Cerrar el panel flotante
const panel = document.getElementById("normalizer-floating-panel");
if (panel) panel.remove();
}
//**************************************************************************
//Nombre: isSimilar
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas:
// - a (string): Primera palabra a comparar.
// - b (string): Segunda palabra a comparar.
//Salidas:
// - boolean: Retorna true si las palabras son consideradas similares de forma leve; de lo contrario, retorna false.
//Prerrequisitos si existen: Ninguno.
//Descripción:
// Esta función evalúa la similitud leve entre dos palabras. Primero, verifica si ambas palabras son
// idénticas, en cuyo caso retorna true. Luego, comprueba si la diferencia en la cantidad de caracteres
// entre ambas es mayor a 2; si es así, retorna false. Posteriormente, compara carácter por carácter
// hasta el largo mínimo de las palabras, contando las diferencias. Si el número de discrepancias
// excede 2, se considera que las palabras no son similares y retorna false; en caso contrario, retorna true.
//**************************************************************************
function isSimilar(a, b)
{
if (a === b) return true;
if (Math.abs(a.length - b.length) > 2) return false;
let mismatches = 0;
for (let i = 0; i < Math.min(a.length, b.length); i++) {
if (a[i] !== b[i]) mismatches++;
if (mismatches > 2) return false;
}
return true;
}
//**************************************************************************
//Nombre: normalizePlaceName
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: name (string) - Nombre del lugar
//Salidas: texto normalizado (string)
//Descripción: Normaliza un nombre aplicando capitalización, manejo de artículos, números y paréntesis.
//**************************************************************************
function normalizePlaceName(name) {
if (!name) return "";
const normalizeArticles = !document.getElementById("normalizeArticles")?.checked;
const articles = ["el", "la", "los", "las", "de", "del", "al", "y"];
const words = name.trim().split(/\s+/);
const isRoman = word => /^(i{1,3}|iv|v|vi{0,3}|ix|x|xi{0,3}|xiv|xv|xvi{0,3}|xix|xx|xxi{0,3}|xxiv|xxv)$/i.test(word);
const normalizedWords = words.map((word, index) => {
const lowerWord = word.toLowerCase();
// Si la palabra es un número, no la analizamos
if (/^\d+$/.test(word)) return word;
// Si la palabra está en la lista de exclusión, no la modificamos
if (excludeWords.some(w => w.toLowerCase() === lowerWord)) return word;
// Si es un número romano, lo dejamos en mayúsculas
if (isRoman(word)) return word.toUpperCase();
// Si es una sigla con estructura T&T o a&A, convertirla a mayúsculas
if (/^[A-Za-z]&[A-Za-z]$/.test(word)) return word.toUpperCase();
// Si es una sigla con apóstrofe como "E's", también la dejamos igual
if (/^[A-Z]'[A-Z][a-z]+$/.test(word)) return word;
// Si no se deben normalizar artículos y es un artículo, lo dejamos en minúsculas (excepto la primera palabra)
if (!normalizeArticles && articles.includes(lowerWord) && index !== 0) return lowerWord;
// Si es un número seguido de letras, lo dejamos igual
if (/^\d+[A-Z][a-zA-Z]*$/.test(word)) return word;
// Si está entre paréntesis y es todo mayúsculas o minúsculas, lo dejamos igual
if (/^\(.*\)$/.test(word)) {
const inner = word.slice(1, -1);
if (inner === inner.toUpperCase() || inner === inner.toLowerCase()) return word;
}
// Capitalizamos la palabra (primera letra en mayúscula, el resto en minúscula)
return word.charAt(0).toUpperCase() + word.slice(1).toLowerCase();
});
let newName = normalizedWords.join(" ")
.replace(/\s*\|\s*/g, " - ")
.replace(/([(["'])\s*([\p{L}])/gu, (match, p1, p2) => p1 + p2.toUpperCase())
.replace(/\s*-\s*/g, " - ")
.replace(/\b(\d+)([A-Z])\b/g, (match, num, letter) => num + letter.toUpperCase())
.replace(/\.$/, "")
.replace(/&(\s*)([A-Z])/g, (match, space, letter) => "&" + space + letter.toUpperCase());
return newName.replace(/\s{2,}/g, " ").trim();
}
// Para exponer al contexto global real desde Tampermonkey
unsafeWindow.normalizePlaceName = normalizePlaceName;
//**************************************************************************
//Nombre: openFloatingPanel
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: placesToNormalize (array de lugares con nombre original y sugerencias)
//Salidas: Panel flotante con opciones de normalización y eliminación
//Prerrequisitos si existen: Debe haberse definido normalizePlaceName y cargado excludeWords correctamente
//Descripción: Crea un panel interactivo donde se presentan los lugares que requieren cambios,
// permitiendo su corrección o eliminación. Solo muestra lugares que requieren cambio
// y errores ortográficos verdaderos.
//**************************************************************************
function openFloatingPanel(placesToNormalize) {
const panel = document.createElement("div");
panel.id = "normalizer-floating-panel";
panel.style.cssText = `
position: fixed;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
background: white;
padding: 20px;
border: 1px solid #ccc;
border-radius: 8px;
box-shadow: 0 2px 10px rgba(0,0,0,0.1);
z-index: 1000;
max-height: 80vh;
overflow-y: auto;
min-width: 800px;
`;
let panelContent = `
<style>
#normalizer-table { width: 100%; border-collapse: collapse; }
#normalizer-table th, #normalizer-table td { padding: 8px; border: 1px solid #ddd; }
#normalizer-table tr:nth-child(even) { background-color: #f9f9f9; }
.warning-row { background-color: #fff3cd !important; }
.close-btn {
position: absolute;
top: 10px;
right: 10px;
background: #ccc;
color: black;
border: none;
border-radius: 4px;
width: 25px;
height: 25px;
font-size: 16px;
font-weight: bold;
cursor: pointer;
text-align: center;
line-height: 25px;
transition: background-color 0.3s ease;
}
.close-btn:hover {
background: #bbb;
}
.footer-buttons {
display: flex;
justify-content: center;
gap: 10px;
margin-top: 20px;
}
</style>
<button class="close-btn" id="close-panel-btn">X</button>
<h3>Places to Normalize</h3>
<table id="normalizer-table">
<thead>
<tr>
<th>Normalizar</th>
<th>Eliminar</th>
<th>Estado</th>
<th>Nombre Original</th>
<th>Nombre Sugerido</th>
<th>Corrección</th>
<th>Acción</th>
</tr>
</thead>
<tbody>
`;
// Procesar cada lugar y sus advertencias
placesToNormalize.forEach((place, index) => {
const { originalName, newName, spellingWarnings = [] } = place;
// Crear una fila por cada advertencia ortográfica
spellingWarnings.forEach((warning, warningIndex) => {
const suggestionId = `suggestion-${index}-${warningIndex}`;
panelContent += `
<tr class="warning-row">
<td style="text-align: center;">
<input type="checkbox" class="normalize-checkbox"
data-index="${index}"
data-warning-index="${warningIndex}"
data-suggestion-id="${suggestionId}">
</td>
<td style="text-align: center;">
<input type="checkbox" class="delete-checkbox"
data-index="${index}"
data-warning-index="${warningIndex}">
</td>
<td style="text-align: center;">⚠️</td>
<td id="name-cell-${index}-${warningIndex}">${originalName}</td>
<td>
<input type="text" class="new-name-input"
data-index="${index}"
data-warning-index="${warningIndex}"
data-place-id="${place.id}"
data-suggestion-id="${suggestionId}"
value="${newName}">
</td>
<td>${warning.original} → ${warning.sugerida} (${warning.tipo})</td>
<td style="text-align: center;">
<button class="apply-suggestion-btn"
data-index="${index}"
data-warning-index="${warningIndex}"
title="Corregir ortografía de la palabra"
style="background-color: #4CAF50; color: white; border: none; padding: 5px 10px; border-radius: 4px; cursor: pointer;">
Fix
</button>
<button class="add-exclude-btn"
data-word="${warning.original}"
title="Adicionar palabra excluida nueva"
style="background-color: #007bff; color: white; border: none; padding: 5px 10px; border-radius: 4px; cursor: pointer;">
Add
</button>
</td>
</tr>`;
});
});
panelContent += `
</tbody>
</table>
<div class="footer-buttons">
<button class="apply-changes-btn" id="apply-changes-btn">✔️ Apply Changes</button>
<button class="cancel-btn" id="cancel-btn">❌ Cancel</button>
</div>
`;
panel.innerHTML = panelContent;
document.body.appendChild(panel);
// Eventos para los botones
document.getElementById('close-panel-btn').addEventListener('click', () => {
panel.remove(); // Cerrar el panel flotante
});
document.getElementById('apply-changes-btn').addEventListener('click', () => {
window.applyNormalization(); // Llamar a la función global para aplicar cambios
});
document.getElementById('cancel-btn').addEventListener('click', () => {
panel.remove(); // Cerrar el panel flotante
});
// Asignar eventos a los botones "Fix"
document.querySelectorAll('.apply-suggestion-btn').forEach(button => {
button.addEventListener('click', function () {
const index = this.dataset.index;
const warningIndex = this.dataset.warningIndex;
const input = document.querySelector(
`.new-name-input[data-index="${index}"][data-warning-index="${warningIndex}"]`
);
const checkbox = document.querySelector(
`.normalize-checkbox[data-index="${index}"][data-warning-index="${warningIndex}"]`
);
if (input && checkbox) {
// Aplicar la corrección ortográfica al campo de entrada
const warning = placesToNormalize[index].spellingWarnings[warningIndex];
input.value = input.value.replace(warning.original, warning.sugerida);
checkbox.checked = true; // Marcar el checkbox
}
});
});
// Asignar eventos a los botones "Add Excld Word"
document.querySelectorAll('.add-exclude-btn').forEach(button => {
button.addEventListener('click', function () {
const word = this.dataset.word;
// Verificar si es una sigla con formato X&X
if (/^[A-Za-z]&[A-Za-z]$/.test(word)) {
alert("⚠️ No es necesario adicionar palabras excluidas que tengan '&'.");
return;
}
if (!excludeWords.includes(word)) {
excludeWords.push(word);
excludeWords.sort((a, b) => a.localeCompare(b)); // Ordenar alfabéticamente
localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
renderExcludedWordsPanel(); // Actualizar el panel lateral
// Mostrar popup en el centro del panel flotante
const panel = document.getElementById("normalizer-floating-panel");
const popup = document.createElement('div');
popup.textContent = `✅ The Word "${word}" has been added to the exclusion list.`;
popup.style.position = 'absolute';
popup.style.top = '50%';
popup.style.left = '50%';
popup.style.transform = 'translate(-50%, -50%)';
popup.style.backgroundColor = '#4CAF50';
popup.style.color = 'white';
popup.style.padding = '10px 20px';
popup.style.borderRadius = '5px';
popup.style.zIndex = '10000';
popup.style.opacity = '1';
popup.style.transition = 'opacity 1s ease-in-out';
panel.appendChild(popup);
setTimeout(() => {
popup.style.opacity = '0';
setTimeout(() => panel.removeChild(popup), 1000);
}, 2000);
// Bloquear el botón y cambiar su texto
this.textContent = 'Added';
this.disabled = true;
this.style.backgroundColor = '#6c757d'; // Cambiar color a gris
this.style.cursor = 'not-allowed';
} else {
alert(`⚠️ The word "${word}" is already on the exclusion list.`);
}
});
});
}
//**************************************************************************
//Nombre: loadExcludeWordsFromXML
//Fecha modificación:
//Autor: mincho77
//Entradas:
// - callback: función opcional que se ejecuta una vez se cargan y procesan las palabras excluidas.
//Salidas: ninguna directa. Actualiza la variable global `excludeWords`.
//Prerrequisitos:
// - Debe existir un archivo llamado 'excludeWords.xml' accesible por fetch.
// - Debe estar definida la variable global `excludeWords`.
//Descripción:
// Carga un archivo XML que contiene una lista de palabras excluidas.
// Combina las palabras nuevas con las que ya están guardadas en localStorage
// y actualiza la lista global `excludeWords`. Si el XML no puede ser cargado,
// se usa únicamente el contenido almacenado localmente como respaldo.
//**************************************************************************
function loadExcludeWordsFromXML(callback) {
fetch("excludeWords.xml")
.then(response => response.text()) // Corrected the syntax here
.then(xmlText => {
const parser = new DOMParser();
const xmlDoc = parser.parseFromString(xmlText, "text/xml");
const wordNodes = xmlDoc.getElementsByTagName("word");
const wordsFromXML = Array.from(wordNodes).map(node => node.textContent.trim());
const existing = JSON.parse(localStorage.getItem("excludeWords")) || [];
excludeWords = [...new Set([...existing, ...wordsFromXML])].sort((a, b) => a.localeCompare(b));
localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
if (callback) callback();
})
.catch(() => {
console.warn("⚠️ No se pudo cargar excludeWords.xml. Solo se usará localStorage.");
excludeWords = JSON.parse(localStorage.getItem("excludeWords")) || ["EDS", "IPS", "McDonald's", "EPS"];
localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
if (callback) callback();
});
}
function exportExcludeWordsToXML() {
const xmlContent = `<?xml version="1.0" encoding="UTF-8"?>
<ExcludedWords>
${excludeWords.map(word => ` <word>${word}</word>`).join("\n")}
</ExcludedWords>`;
const blob = new Blob([xmlContent], { type: "application/xml" });
const url = URL.createObjectURL(blob); // ✅ ESTA LÍNEA ESTABA FALTANDO
const link = document.createElement("a");
link.href = url;
link.download = "excludeWords.xml";
document.body.appendChild(link);
link.click();
document.body.removeChild(link);
}
function showFloatingMessage(message) {
const msg = document.createElement("div");
msg.textContent = message;
msg.style.position = "fixed";
msg.style.bottom = "30px";
msg.style.left = "50%";
msg.style.transform = "translateX(-50%)";
msg.style.backgroundColor = "#333";
msg.style.color = "#fff";
msg.style.padding = "10px 20px";
msg.style.borderRadius = "5px";
msg.style.zIndex = 9999;
msg.style.opacity = "0.95";
msg.style.transition = "opacity 1s ease-in-out";
document.body.appendChild(msg);
setTimeout(() => {
msg.style.opacity = "0";
setTimeout(() => document.body.removeChild(msg), 1000);
}, 3000);
}
//**************************************************************************
//Nombre: waitForWME
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: Ninguna
//Salidas: Ninguna
//Prerrequisitos si existen:
// - El objeto global W debe estar disponible.
// - Deben estar definidas las funciones: initializeExcludeWords, createSidebarTab y renderExcludedWordsPanel.
//Descripción:
// Esta función espera a que el entorno de edición de Waze (WME) esté completamente cargado,
// verificando que existan los objetos necesarios para iniciar el script.
// Una vez disponible, inicializa la lista de palabras excluidas desde localStorage,
// crea el tab lateral personalizado y renderiza la lista visual de palabras excluidas.
// Si WME aún no está listo, vuelve a intentar cada 1000 ms.
//**************************************************************************
function waitForWME() {
if (W && W.userscripts && W.model && W.model.venues) {
console.log(`[${SCRIPT_NAME}] Inicializando v${VERSION}`);
initializeExcludeWords(); // ⚠️ Usa solo localStorage
createSidebarTab();
waitForDOM("#normalizer-tab", () => {
console.log("[waitForWME] 🧩 Sidebar listo, renderizando palabras excluidas");
renderExcludedWordsPanel(); // Muestra las palabras
setupDragAndDropImport(); // Activa drag & drop
//Evita que se abra el archivo si cae fuera del área
// window.addEventListener("dragover", e => e.preventDefault(), false);
// window.addEventListener("drop", e => e.preventDefault(), false);
});
} else {
console.log(`[${SCRIPT_NAME}] Esperando que WME esté listo...`);
setTimeout(waitForWME, 1000);
}
}
//**************************************************************************
//Nombre: cleanupEventListeners
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: Ninguna
//Salidas: Ninguna
//Prerrequisitos si existen:
// - Debe existir un elemento en el DOM con el id "normalizer-floating-panel".
//Descripción:
// Esta función limpia los event listeners asociados al panel flotante de normalización.
// Lo hace clonando el nodo del panel y reemplazándolo en el DOM, lo cual elimina todos
// los listeners previamente asignados a ese nodo, evitando posibles fugas de memoria
// o comportamientos inesperados.
//**************************************************************************
function cleanupEventListeners() {
const panel = document.getElementById("normalizer-floating-panel");
if (panel) {
const clone = panel.cloneNode(true);
panel.parentNode.replaceChild(clone, panel);
}
}
//**************************************************************************
//Nombre: normalizePlaceName (unsafeWindow)
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas:
// - name (string): el nombre original del lugar.
//Salidas:
// - string: nombre normalizado, respetando exclusiones y opciones del usuario.
//Prerrequisitos si existen:
// - Debe estar cargada la lista global excludeWords.
// - Debe existir un checkbox con id “normalizeArticles” para definir si se normalizan artículos.
//Descripción:
// Esta versión expuesta globalmente permite acceder a la normalización básica del nombre de un lugar
// desde otros contextos como el navegador o Tampermonkey. Capitaliza cada palabra, respeta las excluidas
// y no aplica normalización a artículos si el checkbox lo indica.
// Realiza limpieza básica: reemplazo de pipes, eliminación de espacios dobles y trim final.
//**************************************************************************
unsafeWindow.normalizePlaceName = function(name)
{
if (!name) return "";
const normalizeArticles = !document.getElementById("normalizeArticles")?.checked;
const articles = ["el", "la", "los", "las", "de", "del", "al", "y", "o"];
const words = name.trim().split(/\s+/);
const normalizedWords = words.map((word, index) => {
const lowerWord = word.toLowerCase();
// Saltar palabras excluidas
if (excludeWords.includes(word)) return word;
// Saltar artículos si el checkbox está activo y no es la primera palabra
if (!normalizeArticles && articles.includes(lowerWord) && index !== 0) {
return lowerWord;
}
//Mayúsculas
return word.charAt(0).toUpperCase() + word.slice(1).toLowerCase();
});
name = normalizedWords.join(" ");
name = name.replace(/\s*\|\s*/g, " - ");
name = name.replace(/\s{2,}/g, " ").trim();
return name;
};
//**************************************************************************
//Nombre: waitForDOM
//Fecha modificación: 2025-03-31
//Autor: mincho77
//Entradas:
// selector (string): Selector CSS del nodo a esperar.
// callback (function): Función a ejecutar cuando el nodo esté disponible.
// interval (int, opcional): Tiempo entre intentos en milisegundos. Default: 500ms.
// maxAttempts (int, opcional): Máximo de intentos antes de abortar. Default: 10.
//Salidas: Ejecuta el callback si encuentra el selector dentro del tiempo.
//Descripción:
// Esta función monitorea el DOM en intervalos constantes hasta que
// encuentra un nodo que coincida con el selector. Si lo encuentra,
// ejecuta el callback con ese nodo. Si no, muestra un error por consola.
//**************************************************************************
function waitForDOM(selector, callback, interval = 500, maxAttempts = 10)
{
let attempts = 0;
const checkExist = setInterval(() => {
const element = document.querySelector(selector);
if (element) {
clearInterval(checkExist);
callback(element);
} else if (attempts >= maxAttempts) {
clearInterval(checkExist);
console.error(`[${SCRIPT_NAME}] Error: No se encontraron elementos en el DOM después de ${maxAttempts} intentos.`);
}
attempts++;
}, interval);
}
//**************************************************************************
//Nombre: getSidebarHTML
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: Ninguna.
//Salidas: Retorna un string con HTML que define el contenido del panel lateral del script.
//Prerrequisitos si existen: Debe estar disponible el valor de las variables globales:
// - normalizeArticles (boolean): Define si los artículos deben ser normalizados o no.
// - maxPlaces (number): Número máximo de lugares a escanear.
//Descripción:
// Esta función construye el HTML que se inyecta en el panel lateral (sidebar) de WME.
// Incluye controles para:
// - Activar o desactivar normalización de artículos ("el", "la", etc).
// - Definir la cantidad máxima de lugares a procesar.
// - Agregar palabras excluidas manualmente.
// - Exportar palabras excluidas a un archivo XML.
// - Importar una lista de palabras excluidas desde archivo XML.
// - Disparar el escaneo de lugares para normalización.
// La lista de palabras excluidas **no se renderiza aquí directamente**, sino en el div
// con id "normalizer-sidebar" para permitir que sea manejada dinámicamente.
//**************************************************************************
function getSidebarHTML() {
return `
<div id="normalizer-tab">
<h4>Places Name Normalizer <span style="font-size:11px;">v${VERSION}</span></h4>
<div>
<input type="checkbox" id="normalizeArticles" ${normalizeArticles ? "checked" : ""}>
<label for="normalizeArticles">No Normalizar artículos (el, la, los, ...)</label>
</div>
<div>
<label>Máximo de Places a buscar: </label>
<input type='number' id='maxPlacesInput' value='${maxPlaces}' min='1' max='800' style='width: 60px;'>
</div>
<div>
<label>Palabras Excluidas:</label>
<input type='text' id='excludeWord' style='width: 120px;'>
<button id='addExcludeWord'>Add Word</button>
<div id="normalizer-sidebar" style="margin-top: 20px;"></div>
<button id="exportExcludeWords" style="margin-top: 5px;">Export Words</button>
<br>
<button id="importExcludeWordsUnifiedBtn" style="margin-top: 5px;">Import List...</button>
<input type="file" id="hiddenImportInput" accept=".xml" style="display: none;">
</div>
<div>
<input type="checkbox" id="checkOnlyTildes" checked>
<label for="checkOnlyTildes">Revisar solo tildes</label>
</div>
<div id="drop-zone" style="border: 2px dashed #ccc; border-radius: 6px; padding: 15px; margin: 15px 0; text-align: center; font-style: italic; color: #555; background-color: #f8f9fa; transition: all 0.3s ease;">
📂 Arrastra aquí tu archivo .txt o .xml para importar palabras excluidas
</div>
<hr>
<button id="scanPlaces">Scan...</button>
</div>
`;
}
//**************************************************************************
//Nombre: attachEvents
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: Ninguna
//Salidas: Ninguna
//Prerrequisitos si existen:
// - Deben existir en el DOM los elementos con los siguientes IDs:
// "normalizeArticles", "maxPlacesInput", "addExcludeWord", "scanPlaces",
// "hiddenImportInput", "importExcludeWordsUnifiedBtn" y "exportExcludeWords".
// - Debe existir la función handleImportList y la función scanPlaces.
// - Debe estar definida la variable global excludeWords y la función renderExcludedWordsPanel.
//Descripción:
// Esta función adjunta los event listeners necesarios para gestionar la interacción del usuario
// con el panel del normalizador de nombres. Se encargan de:
// - Actualizar la opción de normalizar artículos al cambiar el estado del checkbox.
// - Modificar el número máximo de lugares a procesar a través de un input.
// - Exportar la lista de palabras excluidas a un archivo XML.
// - Añadir nuevas palabras a la lista de palabras excluidas, evitando duplicados, y actualizar el panel.
// - Activar el botón unificado para la importación de palabras excluidas mediante un input oculto.
// - Ejecutar la función de escaneo de lugares al hacer clic en el botón correspondiente.
//**************************************************************************
function attachEvents2()
{
console.log(`[${SCRIPT_NAME}] Adjuntando eventos...`);
const normalizeArticlesCheckbox = document.getElementById("normalizeArticles");
const maxPlacesInput = document.getElementById("maxPlacesInput");
const addExcludeWordButton = document.getElementById("addExcludeWord");
const scanPlacesButton = document.getElementById("scanPlaces");
const hiddenInput = document.getElementById("hiddenImportInput");
const importButtonUnified = document.getElementById("importExcludeWordsUnifiedBtn");
// Validación de elementos necesarios
if (!normalizeArticlesCheckbox || !maxPlacesInput || !addExcludeWordButton || !scanPlacesButton) {
console.error(`[${SCRIPT_NAME}] Error: No se encontraron elementos en el DOM.`);
return;
}
// ✅ Evento: cambiar estado de "no normalizar artículos"
normalizeArticlesCheckbox.addEventListener("change", (e) => {
normalizeArticles = e.target.checked;
});
// ✅ Evento: cambiar número máximo de places
maxPlacesInput.addEventListener("input", (e) => {
maxPlaces = parseInt(e.target.value, 10);
});
// ✅ Evento: exportar palabras excluidas a XML
document.getElementById("exportExcludeWords").addEventListener("click", () => {
const savedWords = JSON.parse(localStorage.getItem("excludeWords")) || [];
if (savedWords.length === 0) {
alert("No hay palabras excluidas para exportar.");
return;
}
const sortedWords = [...savedWords].sort((a, b) => a.localeCompare(b));
const xmlContent = `<?xml version="1.0" encoding="UTF-8"?>
<ExcludedWords>
${sortedWords.map(word => ` <word>${word}</word>`).join("\n ")}
</ExcludedWords>`;
const blob = new Blob([xmlContent], { type: "application/xml" });
const url = URL.createObjectURL(blob);
const link = document.createElement("a");
link.href = url;
link.download = "excluded_words.xml";
document.body.appendChild(link); // Correctly appends the link
link.click();
document.body.removeChild(link); // Correctly removes the link
});
// ✅ Evento: añadir palabra excluida sin duplicados
addExcludeWordButton.addEventListener("click", () => {
const wordInput = document.getElementById("excludeWord") || document.getElementById("excludedWord");
const word = wordInput?.value.trim();
if (!word) return;
const lowerWord = word.toLowerCase();
const alreadyExists = excludeWords.some(w => w.toLowerCase() === lowerWord);
if (!alreadyExists) {
excludeWords.push(word);
localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
renderExcludedWordsPanel(); // Usar renderExcludedWordsPanel en lugar de updateExcludeList
}
wordInput.value = "";
});
// ✅ Evento: nuevo botón unificado de importación
importButtonUnified.addEventListener("click", () => {
hiddenInput.click(); // abre el file input oculto
});
hiddenInput.addEventListener("change", () => {
handleImportList(); // ✅ Llama a la función handleImportList al importar
});
// ✅ Evento: escanear lugares
scanPlacesButton.addEventListener("click", scanPlaces);
}
//**************************************************************************
//Nombre: NameChangeAction
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas:
// - venue (object): Objeto Place que contiene la información del lugar a modificar.
// - oldName (string): Nombre actual del lugar.
// - newName (string): Nuevo nombre sugerido para el lugar.
//Salidas: Ninguna (función constructora que crea un objeto de acción).
//Prerrequisitos si existen:
// - El objeto venue debe contar con la propiedad attributes, y dentro de ésta, con el campo id.
//Descripción:
// Esta función constructora crea un objeto que representa la acción de cambio de nombre
// de un Place en el Waze Map Editor (WME). Asigna las propiedades correspondientes:
// - Guarda el objeto venue, el nombre original (oldName) y el nuevo nombre (newName).
// - Extrae y guarda el ID único del lugar desde venue.attributes.id.
// - Establece metadatos para identificar la acción, asignando el tipo "NameChangeAction"
// y marcando isGeometryEdit como false para indicar que no se trata de una edición de geometría.
// Estos metadatos pueden ser utilizados por WME y otros plugins para gestionar y mostrar la acción.
//**************************************************************************
function NameChangeAction(venue, oldName, newName)
{
// Referencia al Place y los nombres
this.venue = venue;
this.oldName = oldName;
this.newName = newName;
// ID único del Place
this.venueId = venue.attributes.id;
// Metadatos que WME/Plugins pueden usar
this.type = "NameChangeAction";
this.isGeometryEdit = false; // no es una edición de geometría
}
/**
* 1) getActionName: nombre de la acción en el historial.
*/
NameChangeAction.prototype.getActionName = function() {
return "Update place name";
};
/** 2) getActionText: texto corto que WME a veces muestra. */
NameChangeAction.prototype.getActionText = function() {
return "Update place name";
};
/** 3) getName: algunas versiones llaman a getName(). */
NameChangeAction.prototype.getName = function() {
return "Update place name";
};
/** 4) getDescription: descripción detallada de la acción. */
NameChangeAction.prototype.getDescription = function() {
return `Place name changed from "${this.oldName}" to "${this.newName}".`;
};
/** 5) getT: título (a veces requerido por plugins). */
NameChangeAction.prototype.getT = function() {
return "Update place name";
};
/** 6) getID: si un plugin llama a e.getID(). */
NameChangeAction.prototype.getID = function() {
return `NameChangeAction-${this.venueId}`;
};
/** 7) doAction: asigna el nuevo nombre (WME llama a esto al crear la acción). */
NameChangeAction.prototype.doAction = function() {
this.venue.attributes.name = this.newName;
this.venue.isDirty = true;
if (typeof W.model.venues.markObjectEdited === "function") {
W.model.venues.markObjectEdited(this.venue);
}
};
/** 8) undoAction: revertir al nombre anterior (Ctrl+Z). */
NameChangeAction.prototype.undoAction = function() {
this.venue.attributes.name = this.oldName;
this.venue.isDirty = true;
if (typeof W.model.venues.markObjectEdited === "function") {
W.model.venues.markObjectEdited(this.venue);
}
};
/** 9) redoAction: rehacer (Ctrl+Shift+Z), vuelve a doAction. */
NameChangeAction.prototype.redoAction = function() {
return this.doAction();
};
/** 10) undoSupported / redoSupported: indica si se puede des/rehacer. */
NameChangeAction.prototype.undoSupported = function() {
return true;
};
NameChangeAction.prototype.redoSupported = function() {
return true;
};
/** 11) accept / supersede: evita fusionar con otras acciones. */
NameChangeAction.prototype.accept = function() {
return false;
};
NameChangeAction.prototype.supersede = function() {
return false;
};
/** 12) isEditAction: true => habilita "Guardar". */
NameChangeAction.prototype.isEditAction = function() {
return true;
};
/** 13) getAffectedUniqueIds: objetos que se alteran. */
NameChangeAction.prototype.getAffectedUniqueIds = function() {
return [this.venueId];
};
/** 14) isSerializable: si no implementas serialize(), pon false. */
NameChangeAction.prototype.isSerializable = function() {
return false;
};
/** 15) isActionStackable: false => no combina con otras ediciones. */
NameChangeAction.prototype.isActionStackable = function() {
return false;
};
/** 16) getFocusFeatures: WME/Plugins pueden usarlo para "enfocar" el objeto. */
NameChangeAction.prototype.getFocusFeatures = function() {
// Devolvemos el venue para indicar que ese es el foco (o un array vacío si prefieres).
return [this.venue];
};
/** 17) Métodos vacíos para evitar futuros "no es una función" si WME pide estos. */
NameChangeAction.prototype.getFocusSegments = function() {
return [];
};
NameChangeAction.prototype.getFocusNodes = function() {
return [];
};
NameChangeAction.prototype.getFocusClosures = function() {
return [];
};
/** 18) getTimestamp: método nuevo que WME/Plugins están llamando. */
NameChangeAction.prototype.getTimestamp = function() {
// Devolvemos un timestamp numérico (ms desde época UNIX).
return Date.now();
};
//**************************************************************************
//Nombre: openFloatingPanel
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: placesToNormalize (array de lugares con nombre original y sugerencias)
//Salidas: Panel flotante con opciones de normalización y eliminación
//Prerrequisitos si existen: Debe haberse definido normalizePlaceName y cargado excludeWords correctamente
//Descripción: Crea un panel interactivo donde se presentan los lugares que requieren cambios,
// permitiendo su corrección o eliminación. Solo muestra lugares que requieren cambio
// y errores ortográficos verdaderos.
//**************************************************************************
function openFloatingPanel2(placesToNormalize) {
const panel = document.createElement("div");
panel.id = "normalizer-floating-panel";
panel.style.cssText = `
position: fixed;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
background: white;
padding: 20px;
border: 1px solid #ccc;
border-radius: 8px;
box-shadow: 0 2px 10px rgba(0,0,0,0.1);
z-index: 1000;
max-height: 80vh;
overflow-y: auto;
min-width: 800px;
`;
let panelContent = `
<style>
#normalizer-table { width: 100%; border-collapse: collapse; }
#normalizer-table th, #normalizer-table td { padding: 8px; border: 1px solid #ddd; }
#normalizer-table tr:nth-child(even) { background-color: #f9f9f9; }
.warning-row { background-color: #fff3cd !important; }
.close-btn {
position: absolute;
top: 10px;
right: 10px;
background: #ccc;
color: black;
border: none;
border-radius: 4px;
width: 25px;
height: 25px;
font-size: 16px;
font-weight: bold;
cursor: pointer;
text-align: center;
line-height: 25px;
transition: background-color 0.3s ease;
}
.close-btn:hover {
background: #bbb;
}
.footer-buttons {
display: flex;
justify-content: center;
gap: 10px;
margin-top: 20px;
}
</style>
<button class="close-btn" id="close-panel-btn">X</button>
<h3>Places to Normalize</h3>
<table id="normalizer-table">
<thead>
<tr>
<th>Normalizar</th>
<th>Eliminar</th>
<th>Estado</th>
<th>Nombre Original</th>
<th>Nombre Sugerido</th>
<th>Corrección</th>
<th>Acción</th>
</tr>
</thead>
<tbody>
`;
// Procesar cada lugar y sus advertencias
placesToNormalize.forEach((place, index) => {
const { originalName, newName, spellingWarnings = [] } = place;
// Crear una fila por cada advertencia ortográfica
spellingWarnings.forEach((warning, warningIndex) => {
const suggestionId = `suggestion-${index}-${warningIndex}`;
panelContent += `
<tr class="warning-row">
<td style="text-align: center;">
<input type="checkbox" class="normalize-checkbox"
data-index="${index}"
data-warning-index="${warningIndex}"
data-suggestion-id="${suggestionId}">
</td>
<td style="text-align: center;">
<input type="checkbox" class="delete-checkbox"
data-index="${index}"
data-warning-index="${warningIndex}">
</td>
<td style="text-align: center;">⚠️</td>
<td id="name-cell-${index}-${warningIndex}">${originalName}</td>
<td>
<input type="text" class="new-name-input"
data-index="${index}"
data-warning-index="${warningIndex}"
data-place-id="${place.id}"
data-suggestion-id="${suggestionId}"
value="${newName}">
</td>
<td>${warning.original} → ${warning.sugerida} (${warning.tipo})</td>
<td style="text-align: center;">
<button class="apply-suggestion-btn"
data-index="${index}"
data-warning-index="${warningIndex}"
title="Corregir ortografía de la palabra"
style="background-color: #4CAF50; color: white; border: none; padding: 5px 10px; border-radius: 4px; cursor: pointer;">
Fix
</button>
<button class="add-exclude-btn"
data-word="${warning.original}"
title="Adicionar palabra excluida nueva"
style="background-color: #007bff; color: white; border: none; padding: 5px 10px; border-radius: 4px; cursor: pointer;">
Add Excld
</button>
</td>
</tr>`;
});
});
panelContent += `
</tbody>
</table>
<div class="footer-buttons">
<button class="apply-changes-btn" id="apply-changes-btn">✔️ Apply Changes</button>
<button class="cancel-btn" id="cancel-btn">❌ Cancel</button>
</div>
`;
panel.innerHTML = panelContent;
document.body.appendChild(panel);
// Eventos para los botones
document.getElementById('close-panel-btn').addEventListener('click', () => {
panel.remove(); // Cerrar el panel flotante
});
document.getElementById('apply-changes-btn').addEventListener('click', () => {
window.applyNormalization(); // Llamar a la función global para aplicar cambios
});
document.getElementById('cancel-btn').addEventListener('click', () => {
panel.remove(); // Cerrar el panel flotante
});
// Asignar eventos a los botones "Fix"
document.querySelectorAll('.apply-suggestion-btn').forEach(button => {
button.addEventListener('click', function () {
const index = this.dataset.index;
const warningIndex = this.dataset.warningIndex;
const input = document.querySelector(
`.new-name-input[data-index="${index}"][data-warning-index="${warningIndex}"]`
);
const checkbox = document.querySelector(
`.normalize-checkbox[data-index="${index}"][data-warning-index="${warningIndex}"]`
);
if (input && checkbox) {
const warning = placesToNormalize[index].spellingWarnings[warningIndex];
input.value = input.value.replace(warning.original, warning.sugerida || warning.original);
checkbox.checked = true; // Marcar el checkbox
}
});
});
// Asignar eventos a los botones "Add Excld Word"
document.querySelectorAll('.add-exclude-btn').forEach(button => {
button.addEventListener('click', function () {
const word = this.dataset.word;
if (!excludeWords.includes(word)) {
excludeWords.push(word);
excludeWords.sort((a, b) => a.localeCompare(b)); // Ordenar alfabéticamente
localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
renderExcludedWordsPanel(); // Actualizar el panel lateral
// Mostrar popup en el centro del panel flotante
const panel = document.getElementById("normalizer-floating-panel");
const popup = document.createElement('div');
popup.textContent = `✅ The Word "${word}" has been added to the exclusion list.`;
popup.style.position = 'absolute';
popup.style.top = '50%';
popup.style.left = '50%';
popup.style.transform = 'translate(-50%, -50%)';
popup.style.backgroundColor = '#4CAF50';
popup.style.color = 'white';
popup.style.padding = '10px 20px';
popup.style.borderRadius = '5px';
popup.style.zIndex = '10000';
popup.style.opacity = '1';
popup.style.transition = 'opacity 1s ease-in-out';
panel.appendChild(popup);
setTimeout(() => {
popup.style.opacity = '0';
setTimeout(() => panel.removeChild(popup), 1000);
}, 2000);
// Bloquear el botón y cambiar su texto
this.textContent = 'Excluded Word Added';
this.disabled = true;
this.style.backgroundColor = '#6c757d'; // Cambiar color a gris
this.style.cursor = 'not-allowed';
} else {
alert(`⚠️ The word "${word}" is already on the exclusion list.`);
}
});
});
}
//**************************************************************************
//Nombre: waitForWME
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: Ninguna
//Salidas: Ninguna
//Prerrequisitos si existen:
// - El objeto global W debe estar disponible.
// - Deben estar definidas las funciones: initializeExcludeWords, createSidebarTab y renderExcludedWordsPanel.
//Descripción:
// Esta función espera a que el entorno de edición de Waze (WME) esté completamente cargado,
// verificando que existan los objetos necesarios para iniciar el script.
// Una vez disponible, inicializa la lista de palabras excluidas desde localStorage,
// crea el tab lateral personalizado y renderiza la lista visual de palabras excluidas.
// Si WME aún no está listo, vuelve a intentar cada 1000 ms.
//**************************************************************************
function waitForWME2() {
if (W && W.userscripts && W.model && W.model.venues) {
console.log(`[${SCRIPT_NAME}] Inicializando v${VERSION}`);
initializeExcludeWords(); // ⚠️ Usa solo localStorage
createSidebarTab();
waitForDOM("#normalizer-tab", () => {
console.log("[waitForWME] 🧩 Sidebar listo, renderizando palabras excluidas");
renderExcludedWordsPanel(); // Muestra las palabras
setupDragAndDropImport(); // Activa drag & drop
//Evita que se abra el archivo si cae fuera del área
// window.addEventListener("dragover", e => e.preventDefault(), false);
// window.addEventListener("drop", e => e.preventDefault(), false);
});
} else {
console.log(`[${SCRIPT_NAME}] Esperando que WME esté listo...`);
setTimeout(waitForWME, 1000);
}
}
//**************************************************************************
//Nombre: init
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: Ninguna
//Salidas: Ninguna
//Prerrequisitos si existen:
// - El objeto global W debe estar disponible.
// - Deben estar definidas las funciones: initializeExcludeWords, createSidebarTab, waitForDOM, renderExcludedWordsPanel y setupDragAndDropImport.
//Descripción:
// Esta función espera a que el entorno de edición de Waze (WME) esté completamente cargado,
// verificando que existan los objetos necesarios para iniciar el script. Si WME aún no está listo,
// reintenta la inicialización cada 1000 ms. Una vez disponible, inicializa la lista de palabras
// excluidas, crea el tab lateral personalizado, espera a que el DOM del tab esté listo para
// renderizar el contenido (palabras excluidas y funcionalidad drag & drop), y expone globalmente
// las funciones applyNormalization y normalizePlaceName para uso externo.
//**************************************************************************
function init() {
if (!W || !W.userscripts || !W.model || !W.model.venues) {
console.log(`[${SCRIPT_NAME}] Esperando que WME esté listo...`);
setTimeout(init, 1000);
return;
}
console.log(`[${SCRIPT_NAME}] Inicializando v${VERSION}`);
initializeExcludeWords();
createSidebarTab();
waitForDOM("#normalizer-tab", () => {
console.log("[init] 🧩 Sidebar listo, renderizando palabras excluidas");
renderExcludedWordsPanel();
setupDragAndDropImport();
});
window.applyNormalization = applyNormalization;
window.normalizePlaceName = normalizePlaceName;
}
init();
} catch (e) {
console.error('[PlacesNameNormalizer] Fatal initialization error:', e);
}
})();
//**************************************************************************
//Nombre: applyNormalization
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: Ninguna
//Salidas: Ninguna
//Prerrequisitos si existen:
// - Debe existir un elemento en el DOM con las clases .normalize-checkbox, .delete-checkbox y .new-name-input.
// - El objeto global W debe estar disponible, incluyendo W.model.venues, W.model.actionManager y W.controller.
// - Deben estar definidos los módulos "Waze/Action/UpdateObject" y "Waze/Action/DeleteObject" (accesibles mediante require()).
// - Debe existir la variable placesToNormalize, que contiene datos de los lugares a normalizar, incluyendo sugerencias ortográficas.
//Descripción:
// Esta función aplica la normalización y/o eliminación de nombres de lugares en el Waze Map Editor
// según las selecciones realizadas en el panel flotante. Primero, obtiene los checkboxes seleccionados
// para normalización y eliminación. Si no hay ningún elemento seleccionado, se informa y se cancela la operación.
// Si se han seleccionado TODOS los checkboxes de eliminación, se solicita una confirmación adicional.
// Para cada checkbox de normalización seleccionado, se verifica si se debe aplicar la sugerencia ortográfica
// (cuando se ha hecho clic en el botón correspondiente) o el nombre completo modificado, y se actualiza el lugar
// mediante la acción de actualización. Posteriormente, se procesan los checkboxes de eliminación aplicando la
// acción de eliminación a los lugares correspondientes. Si se realizaron cambios, se marca el modelo como modificado.
// Finalmente, se cierra el panel flotante.
//**************************************************************************
function applyNormalization() {
const normalizeCheckboxes = document.querySelectorAll(".normalize-checkbox:checked");
const deleteCheckboxes = document.querySelectorAll(".delete-checkbox:checked");
let changesMade = false;
if (normalizeCheckboxes.length === 0 && deleteCheckboxes.length === 0) {
alert("No hay lugares seleccionados para normalizar o eliminar.");
return;
}
// Procesar normalización
normalizeCheckboxes.forEach(cb => {
const index = cb.dataset.index;
const input = document.querySelector(`.new-name-input[data-index="${index}"]`);
const newName = input?.value?.trim();
const placeId = input?.getAttribute("data-place-id");
const place = W.model.venues.getObjectById(placeId);
if (!place || !place.attributes?.name) {
console.warn(`No se encontró el lugar con ID: ${placeId}`);
return;
}
const currentName = place.attributes.name.trim();
if (currentName !== newName) {
try {
const UpdateObject = require("Waze/Action/UpdateObject");
const action = new UpdateObject(place, { name: newName });
W.model.actionManager.add(action);
console.log(`Nombre actualizado: "${currentName}" → "${newName}"`);
changesMade = true;
} catch (error) {
console.error("Error aplicando la acción de actualización:", error);
}
}
});
// Procesar eliminación
deleteCheckboxes.forEach(cb => {
const index = cb.dataset.index;
const placeId = document.querySelector(`.new-name-input[data-index="${index}"]`)?.getAttribute("data-place-id");
const place = W.model.venues.getObjectById(placeId);
if (!place) {
console.warn(`No se encontró el lugar con ID para eliminar: ${placeId}`);
return;
}
try {
const DeleteObject = require("Waze/Action/DeleteObject");
const deleteAction = new DeleteObject(place);
W.model.actionManager.add(deleteAction);
console.log(`Lugar eliminado: ${placeId}`);
changesMade = true;
} catch (error) {
console.error("Error eliminando el lugar:", error);
}
});
if (changesMade) {
alert("Cambios aplicados correctamente.");
} else {
alert("No se realizaron cambios.");
}
// Cerrar el panel flotante
const panel = document.getElementById("normalizer-floating-panel");
if (panel) panel.remove();
}
//**************************************************************************
//Nombre: isSimilar
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas:
// - a (string): Primera palabra a comparar.
// - b (string): Segunda palabra a comparar.
//Salidas:
// - boolean: Retorna true si las palabras son consideradas similares de forma leve; de lo contrario, retorna false.
//Prerrequisitos si existen: Ninguno.
//Descripción:
// Esta función evalúa la similitud leve entre dos palabras. Primero, verifica si ambas palabras son
// idénticas, en cuyo caso retorna true. Luego, comprueba si la diferencia en la cantidad de caracteres
// entre ambas es mayor a 2; si es así, retorna false. Posteriormente, compara carácter por carácter
// hasta el largo mínimo de las palabras, contando las diferencias. Si el número de discrepancias
// excede 2, se considera que las palabras no son similares y retorna false; en caso contrario, retorna true.
//**************************************************************************
function isSimilar(a, b)
{
if (a === b) return true;
if (Math.abs(a.length - b.length) > 2) return false;
let mismatches = 0;
for (let i = 0; i < Math.min(a.length, b.length); i++) {
if (a[i] !== b[i]) mismatches++;
if (mismatches > 2) return false;
}
return true;
}
//**************************************************************************
//Nombre: normalizePlaceName
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: name (string) - Nombre del lugar
//Salidas: texto normalizado (string)
//Descripción: Normaliza un nombre aplicando capitalización, manejo de artículos, números y paréntesis.
//**************************************************************************
function normalizePlaceName(name) {
if (!name) return "";
const normalizeArticles = !document.getElementById("normalizeArticles")?.checked;
const articles = ["el", "la", "los", "las", "de", "del", "al", "y"];
const words = name.trim().split(/\s+/);
const isRoman = word => /^(i{1,3}|iv|v|vi{0,3}|ix|x|xi{0,3}|xiv|xv|xvi{0,3}|xix|xx|xxi{0,3}|xxiv|xxv)$/i.test(word);
const normalizedWords = words.map((word, index) => {
const lowerWord = word.toLowerCase();
// Si la palabra es un número, no la analizamos
if (/^\d+$/.test(word)) return word;
// Si la palabra está en la lista de exclusión, no la modificamos
if (excludeWords.some(w => w.toLowerCase() === lowerWord)) return word;
// Si es un número romano, lo dejamos en mayúsculas
if (isRoman(word)) return word.toUpperCase();
// Si es una sigla con estructura T&T o a&A, convertirla a mayúsculas
if (/^[A-Za-z]&[A-Za-z]$/.test(word)) return word.toUpperCase();
// Si es una sigla con apóstrofe como "E's", también la dejamos igual
if (/^[A-Z]'[A-Z][a-z]+$/.test(word)) return word;
// Si no se deben normalizar artículos y es un artículo, lo dejamos en minúsculas (excepto la primera palabra)
if (!normalizeArticles && articles.includes(lowerWord) && index !== 0) return lowerWord;
// Si es un número seguido de letras, lo dejamos igual
if (/^\d+[A-Z][a-zA-Z]*$/.test(word)) return word;
// Si está entre paréntesis y es todo mayúsculas o minúsculas, lo dejamos igual
if (/^\(.*\)$/.test(word)) {
const inner = word.slice(1, -1);
if (inner === inner.toUpperCase() || inner === inner.toLowerCase()) return word;
}
// Capitalizamos la palabra (primera letra en mayúscula, el resto en minúscula)
return word.charAt(0).toUpperCase() + word.slice(1).toLowerCase();
});
let newName = normalizedWords.join(" ")
.replace(/\s*\|\s*/g, " - ")
.replace(/([(["'])\s*([\p{L}])/gu, (match, p1, p2) => p1 + p2.toUpperCase())
.replace(/\s*-\s*/g, " - ")
.replace(/\b(\d+)([A-Z])\b/g, (match, num, letter) => num + letter.toUpperCase())
.replace(/\.$/, "")
.replace(/&(\s*)([A-Z])/g, (match, space, letter) => "&" + space + letter.toUpperCase());
return newName.replace(/\s{2,}/g, " ").trim();
}
// Para exponer al contexto global real desde Tampermonkey
unsafeWindow.normalizePlaceName = normalizePlaceName;
//**************************************************************************
//Nombre: openFloatingPanel
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: placesToNormalize (array de lugares con nombre original y sugerencias)
//Salidas: Panel flotante con opciones de normalización y eliminación
//Prerrequisitos si existen: Debe haberse definido normalizePlaceName y cargado excludeWords correctamente
//Descripción: Crea un panel interactivo donde se presentan los lugares que requieren cambios,
// permitiendo su corrección o eliminación. Solo muestra lugares que requieren cambio
// y errores ortográficos verdaderos.
//**************************************************************************
function openFloatingPanel(placesToNormalize) {
const panel = document.createElement("div");
panel.id = "normalizer-floating-panel";
panel.style.cssText = `
position: fixed;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
background: white;
padding: 20px;
border: 1px solid #ccc;
border-radius: 8px;
box-shadow: 0 2px 10px rgba(0,0,0,0.1);
z-index: 1000;
max-height: 80vh;
overflow-y: auto;
min-width: 800px;
`;
let panelContent = `
<style>
#normalizer-table { width: 100%; border-collapse: collapse; }
#normalizer-table th, #normalizer-table td { padding: 8px; border: 1px solid #ddd; }
#normalizer-table tr:nth-child(even) { background-color: #f9f9f9; }
.warning-row { background-color: #fff3cd !important; }
.close-btn {
position: absolute;
top: 10px;
right: 10px;
background: #ccc;
color: black;
border: none;
border-radius: 4px;
width: 25px;
height: 25px;
font-size: 16px;
font-weight: bold;
cursor: pointer;
text-align: center;
line-height: 25px;
transition: background-color 0.3s ease;
}
.close-btn:hover {
background: #bbb;
}
.footer-buttons {
display: flex;
justify-content: center;
gap: 10px;
margin-top: 20px;
}
</style>
<button class="close-btn" id="close-panel-btn">X</button>
<h3>Places to Normalize</h3>
<table id="normalizer-table">
<thead>
<tr>
<th>Normalizar</th>
<th>Eliminar</th>
<th>Estado</th>
<th>Nombre Original</th>
<th>Nombre Sugerido</th>
<th>Corrección</th>
<th>Acción</th>
</tr>
</thead>
<tbody>
`;
// Procesar cada lugar y sus advertencias
placesToNormalize.forEach((place, index) => {
const { originalName, newName, spellingWarnings = [] } = place;
// Crear una fila por cada advertencia ortográfica
spellingWarnings.forEach((warning, warningIndex) => {
const suggestionId = `suggestion-${index}-${warningIndex}`;
panelContent += `
<tr class="warning-row">
<td style="text-align: center;">
<input type="checkbox" class="normalize-checkbox"
data-index="${index}"
data-warning-index="${warningIndex}"
data-suggestion-id="${suggestionId}">
</td>
<td style="text-align: center;">
<input type="checkbox" class="delete-checkbox"
data-index="${index}"
data-warning-index="${warningIndex}">
</td>
<td style="text-align: center;">⚠️</td>
<td id="name-cell-${index}-${warningIndex}">${originalName}</td>
<td>
<input type="text" class="new-name-input"
data-index="${index}"
data-warning-index="${warningIndex}"
data-place-id="${place.id}"
data-suggestion-id="${suggestionId}"
value="${newName}">
</td>
<td>${warning.original} → ${warning.sugerida} (${warning.tipo})</td>
<td style="text-align: center;">
<button class="apply-suggestion-btn"
data-index="${index}"
data-warning-index="${warningIndex}"
title="Corregir ortografía de la palabra"
style="background-color: #4CAF50; color: white; border: none; padding: 5px 10px; border-radius: 4px; cursor: pointer;">
Fix
</button>
<button class="add-exclude-btn"
data-word="${warning.original}"
title="Adicionar palabra excluida nueva"
style="background-color: #007bff; color: white; border: none; padding: 5px 10px; border-radius: 4px; cursor: pointer;">
Add
</button>
</td>
</tr>`;
});
});
panelContent += `
</tbody>
</table>
<div class="footer-buttons">
<button class="apply-changes-btn" id="apply-changes-btn">✔️ Apply Changes</button>
<button class="cancel-btn" id="cancel-btn">❌ Cancel</button>
</div>
`;
panel.innerHTML = panelContent;
document.body.appendChild(panel);
// Eventos para los botones
document.getElementById('close-panel-btn').addEventListener('click', () => {
panel.remove(); // Cerrar el panel flotante
});
document.getElementById('apply-changes-btn').addEventListener('click', () => {
window.applyNormalization(); // Llamar a la función global para aplicar cambios
});
document.getElementById('cancel-btn').addEventListener('click', () => {
panel.remove(); // Cerrar el panel flotante
});
// Asignar eventos a los botones "Fix"
document.querySelectorAll('.apply-suggestion-btn').forEach(button => {
button.addEventListener('click', function () {
const index = this.dataset.index;
const warningIndex = this.dataset.warningIndex;
const input = document.querySelector(
`.new-name-input[data-index="${index}"][data-warning-index="${warningIndex}"]`
);
const checkbox = document.querySelector(
`.normalize-checkbox[data-index="${index}"][data-warning-index="${warningIndex}"]`
);
if (input && checkbox) {
// Aplicar la corrección ortográfica al campo de entrada
const warning = placesToNormalize[index].spellingWarnings[warningIndex];
input.value = input.value.replace(warning.original, warning.sugerida);
checkbox.checked = true; // Marcar el checkbox
}
});
});
// Asignar eventos a los botones "Add Excld Word"
document.querySelectorAll('.add-exclude-btn').forEach(button => {
button.addEventListener('click', function () {
const word = this.dataset.word;
// Verificar si es una sigla con formato X&X
if (/^[A-Za-z]&[A-Za-z]$/.test(word)) {
alert("⚠️ No es necesario adicionar palabras excluidas que tengan '&'.");
return;
}
if (!excludeWords.includes(word)) {
excludeWords.push(word);
excludeWords.sort((a, b) => a.localeCompare(b)); // Ordenar alfabéticamente
localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
renderExcludedWordsPanel(); // Actualizar el panel lateral
// Mostrar popup en el centro del panel flotante
const panel = document.getElementById("normalizer-floating-panel");
const popup = document.createElement('div');
popup.textContent = `✅ The Word "${word}" has been added to the exclusion list.`;
popup.style.position = 'absolute';
popup.style.top = '50%';
popup.style.left = '50%';
popup.style.transform = 'translate(-50%, -50%)';
popup.style.backgroundColor = '#4CAF50';
popup.style.color = 'white';
popup.style.padding = '10px 20px';
popup.style.borderRadius = '5px';
popup.style.zIndex = '10000';
popup.style.opacity = '1';
popup.style.transition = 'opacity 1s ease-in-out';
panel.appendChild(popup);
setTimeout(() => {
popup.style.opacity = '0';
setTimeout(() => panel.removeChild(popup), 1000);
}, 2000);
// Bloquear el botón y cambiar su texto
this.textContent = 'Added';
this.disabled = true;
this.style.backgroundColor = '#6c757d'; // Cambiar color a gris
this.style.cursor = 'not-allowed';
} else {
alert(`⚠️ The word "${word}" is already on the exclusion list.`);
}
});
});
}
//**************************************************************************
//Nombre: loadExcludeWordsFromXML
//Fecha modificación:
//Autor: mincho77
//Entradas:
// - callback: función opcional que se ejecuta una vez se cargan y procesan las palabras excluidas.
//Salidas: ninguna directa. Actualiza la variable global `excludeWords`.
//Prerrequisitos:
// - Debe existir un archivo llamado 'excludeWords.xml' accesible por fetch.
// - Debe estar definida la variable global `excludeWords`.
//Descripción:
// Carga un archivo XML que contiene una lista de palabras excluidas.
// Combina las palabras nuevas con las que ya están guardadas en localStorage
// y actualiza la lista global `excludeWords`. Si el XML no puede ser cargado,
// se usa únicamente el contenido almacenado localmente como respaldo.
//**************************************************************************
function loadExcludeWordsFromXML(callback) {
fetch("excludeWords.xml")
.then(response => response.text()) // Corrected the syntax here
.then(xmlText => {
const parser = new DOMParser();
const xmlDoc = parser.parseFromString(xmlText, "text/xml");
const wordNodes = xmlDoc.getElementsByTagName("word");
const wordsFromXML = Array.from(wordNodes).map(node => node.textContent.trim());
const existing = JSON.parse(localStorage.getItem("excludeWords")) || [];
excludeWords = [...new Set([...existing, ...wordsFromXML])].sort((a, b) => a.localeCompare(b));
localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
if (callback) callback();
})
.catch(() => {
console.warn("⚠️ No se pudo cargar excludeWords.xml. Solo se usará localStorage.");
excludeWords = JSON.parse(localStorage.getItem("excludeWords")) || ["EDS", "IPS", "McDonald's", "EPS"];
localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
if (callback) callback();
});
}
function exportExcludeWordsToXML() {
const xmlContent = `<?xml version="1.0" encoding="UTF-8"?>
<ExcludedWords>
${excludeWords.map(word => ` <word>${word}</word>`).join("\n")}
</ExcludedWords>`;
const blob = new Blob([xmlContent], { type: "application/xml" });
const url = URL.createObjectURL(blob); // ✅ ESTA LÍNEA ESTABA FALTANDO
const link = document.createElement("a");
link.href = url;
link.download = "excludeWords.xml";
document.body.appendChild(link);
link.click();
document.body.removeChild(link);
}
function showFloatingMessage(message) {
const msg = document.createElement("div");
msg.textContent = message;
msg.style.position = "fixed";
msg.style.bottom = "30px";
msg.style.left = "50%";
msg.style.transform = "translateX(-50%)";
msg.style.backgroundColor = "#333";
msg.style.color = "#fff";
msg.style.padding = "10px 20px";
msg.style.borderRadius = "5px";
msg.style.zIndex = 9999;
msg.style.opacity = "0.95";
msg.style.transition = "opacity 1s ease-in-out";
document.body.appendChild(msg);
setTimeout(() => {
msg.style.opacity = "0";
setTimeout(() => document.body.removeChild(msg), 1000);
}, 3000);
}
//**************************************************************************
//Nombre: waitForWME
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: Ninguna
//Salidas: Ninguna
//Prerrequisitos si existen:
// - El objeto global W debe estar disponible.
// - Deben estar definidas las funciones: initializeExcludeWords, createSidebarTab y renderExcludedWordsPanel.
//Descripción:
// Esta función espera a que el entorno de edición de Waze (WME) esté completamente cargado,
// verificando que existan los objetos necesarios para iniciar el script.
// Una vez disponible, inicializa la lista de palabras excluidas desde localStorage,
// crea el tab lateral personalizado y renderiza la lista visual de palabras excluidas.
// Si WME aún no está listo, vuelve a intentar cada 1000 ms.
//**************************************************************************
function waitForWME() {
if (W && W.userscripts && W.model && W.model.venues) {
console.log(`[${SCRIPT_NAME}] Inicializando v${VERSION}`);
initializeExcludeWords(); // ⚠️ Usa solo localStorage
createSidebarTab();
waitForDOM("#normalizer-tab", () => {
console.log("[waitForWME] 🧩 Sidebar listo, renderizando palabras excluidas");
renderExcludedWordsPanel(); // Muestra las palabras
setupDragAndDropImport(); // Activa drag & drop
//Evita que se abra el archivo si cae fuera del área
// window.addEventListener("dragover", e => e.preventDefault(), false);
// window.addEventListener("drop", e => e.preventDefault(), false);
});
} else {
console.log(`[${SCRIPT_NAME}] Esperando que WME esté listo...`);
setTimeout(waitForWME, 1000);
}
}
//**************************************************************************
//Nombre: cleanupEventListeners
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: Ninguna
//Salidas: Ninguna
//Prerrequisitos si existen:
// - Debe existir un elemento en el DOM con el id "normalizer-floating-panel".
//Descripción:
// Esta función limpia los event listeners asociados al panel flotante de normalización.
// Lo hace clonando el nodo del panel y reemplazándolo en el DOM, lo cual elimina todos
// los listeners previamente asignados a ese nodo, evitando posibles fugas de memoria
// o comportamientos inesperados.
//**************************************************************************
function cleanupEventListeners() {
const panel = document.getElementById("normalizer-floating-panel");
if (panel) {
const clone = panel.cloneNode(true);
panel.parentNode.replaceChild(clone, panel);
}
}
//**************************************************************************
//Nombre: normalizePlaceName (unsafeWindow)
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas:
// - name (string): el nombre original del lugar.
//Salidas:
// - string: nombre normalizado, respetando exclusiones y opciones del usuario.
//Prerrequisitos si existen:
// - Debe estar cargada la lista global excludeWords.
// - Debe existir un checkbox con id “normalizeArticles” para definir si se normalizan artículos.
//Descripción:
// Esta versión expuesta globalmente permite acceder a la normalización básica del nombre de un lugar
// desde otros contextos como el navegador o Tampermonkey. Capitaliza cada palabra, respeta las excluidas
// y no aplica normalización a artículos si el checkbox lo indica.
// Realiza limpieza básica: reemplazo de pipes, eliminación de espacios dobles y trim final.
//**************************************************************************
unsafeWindow.normalizePlaceName = function(name)
{
if (!name) return "";
const normalizeArticles = !document.getElementById("normalizeArticles")?.checked;
const articles = ["el", "la", "los", "las", "de", "del", "al", "y", "o"];
const words = name.trim().split(/\s+/);
const normalizedWords = words.map((word, index) => {
const lowerWord = word.toLowerCase();
// Saltar palabras excluidas
if (excludeWords.includes(word)) return word;
// Saltar artículos si el checkbox está activo y no es la primera palabra
if (!normalizeArticles && articles.includes(lowerWord) && index !== 0) {
return lowerWord;
}
//Mayúsculas
return word.charAt(0).toUpperCase() + word.slice(1).toLowerCase();
});
name = normalizedWords.join(" ");
name = name.replace(/\s*\|\s*/g, " - ");
name = name.replace(/\s{2,}/g, " ").trim();
return name;
};
//**************************************************************************
//Nombre: waitForDOM
//Fecha modificación: 2025-03-31
//Autor: mincho77
//Entradas:
// selector (string): Selector CSS del nodo a esperar.
// callback (function): Función a ejecutar cuando el nodo esté disponible.
// interval (int, opcional): Tiempo entre intentos en milisegundos. Default: 500ms.
// maxAttempts (int, opcional): Máximo de intentos antes de abortar. Default: 10.
//Salidas: Ejecuta el callback si encuentra el selector dentro del tiempo.
//Descripción:
// Esta función monitorea el DOM en intervalos constantes hasta que
// encuentra un nodo que coincida con el selector. Si lo encuentra,
// ejecuta el callback con ese nodo. Si no, muestra un error por consola.
//**************************************************************************
function waitForDOM(selector, callback, interval = 500, maxAttempts = 10)
{
let attempts = 0;
const checkExist = setInterval(() => {
const element = document.querySelector(selector);
if (element) {
clearInterval(checkExist);
callback(element);
} else if (attempts >= maxAttempts) {
clearInterval(checkExist);
console.error(`[${SCRIPT_NAME}] Error: No se encontraron elementos en el DOM después de ${maxAttempts} intentos.`);
}
attempts++;
}, interval);
}
//**************************************************************************
//Nombre: getSidebarHTML
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: Ninguna.
//Salidas: Retorna un string con HTML que define el contenido del panel lateral del script.
//Prerrequisitos si existen: Debe estar disponible el valor de las variables globales:
// - normalizeArticles (boolean): Define si los artículos deben ser normalizados o no.
// - maxPlaces (number): Número máximo de lugares a escanear.
//Descripción:
// Esta función construye el HTML que se inyecta en el panel lateral (sidebar) de WME.
// Incluye controles para:
// - Activar o desactivar normalización de artículos ("el", "la", etc).
// - Definir la cantidad máxima de lugares a procesar.
// - Agregar palabras excluidas manualmente.
// - Exportar palabras excluidas a un archivo XML.
// - Importar una lista de palabras excluidas desde archivo XML.
// - Disparar el escaneo de lugares para normalización.
// La lista de palabras excluidas **no se renderiza aquí directamente**, sino en el div
// con id "normalizer-sidebar" para permitir que sea manejada dinámicamente.
//**************************************************************************
function getSidebarHTML() {
return `
<div id="normalizer-tab">
<h4>Places Name Normalizer <span style="font-size:11px;">v${VERSION}</span></h4>
<div>
<input type="checkbox" id="normalizeArticles" ${normalizeArticles ? "checked" : ""}>
<label for="normalizeArticles">No Normalizar artículos (el, la, los, ...)</label>
</div>
<div>
<label>Máximo de Places a buscar: </label>
<input type='number' id='maxPlacesInput' value='${maxPlaces}' min='1' max='800' style='width: 60px;'>
</div>
<div>
<label>Palabras Excluidas:</label>
<input type='text' id='excludeWord' style='width: 120px;'>
<button id='addExcludeWord'>Add Word</button>
<div id="normalizer-sidebar" style="margin-top: 20px;"></div>
<button id="exportExcludeWords" style="margin-top: 5px;">Export Words</button>
<br>
<button id="importExcludeWordsUnifiedBtn" style="margin-top: 5px;">Import List...</button>
<input type="file" id="hiddenImportInput" accept=".xml" style="display: none;">
</div>
<div>
<input type="checkbox" id="checkOnlyTildes" checked>
<label for="checkOnlyTildes">Revisar solo tildes</label>
</div>
<div id="drop-zone" style="border: 2px dashed #ccc; border-radius: 6px; padding: 15px; margin: 15px 0; text-align: center; font-style: italic; color: #555; background-color: #f8f9fa; transition: all 0.3s ease;">
📂 Arrastra aquí tu archivo .txt o .xml para importar palabras excluidas
</div>
<hr>
<button id="scanPlaces">Scan...</button>
</div>
`;
}
//**************************************************************************
//Nombre: attachEvents
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: Ninguna
//Salidas: Ninguna
//Prerrequisitos si existen:
// - Deben existir en el DOM los elementos con los siguientes IDs:
// "normalizeArticles", "maxPlacesInput", "addExcludeWord", "scanPlaces",
// "hiddenImportInput", "importExcludeWordsUnifiedBtn" y "exportExcludeWords".
// - Debe existir la función handleImportList y la función scanPlaces.
// - Debe estar definida la variable global excludeWords y la función renderExcludedWordsPanel.
//Descripción:
// Esta función adjunta los event listeners necesarios para gestionar la interacción del usuario
// con el panel del normalizador de nombres. Se encargan de:
// - Actualizar la opción de normalizar artículos al cambiar el estado del checkbox.
// - Modificar el número máximo de lugares a procesar a través de un input.
// - Exportar la lista de palabras excluidas a un archivo XML.
// - Añadir nuevas palabras a la lista de palabras excluidas, evitando duplicados, y actualizar el panel.
// - Activar el botón unificado para la importación de palabras excluidas mediante un input oculto.
// - Ejecutar la función de escaneo de lugares al hacer clic en el botón correspondiente.
//**************************************************************************
function attachEvents2()
{
console.log(`[${SCRIPT_NAME}] Adjuntando eventos...`);
const normalizeArticlesCheckbox = document.getElementById("normalizeArticles");
const maxPlacesInput = document.getElementById("maxPlacesInput");
const addExcludeWordButton = document.getElementById("addExcludeWord");
const scanPlacesButton = document.getElementById("scanPlaces");
const hiddenInput = document.getElementById("hiddenImportInput");
const importButtonUnified = document.getElementById("importExcludeWordsUnifiedBtn");
// Validación de elementos necesarios
if (!normalizeArticlesCheckbox || !maxPlacesInput || !addExcludeWordButton || !scanPlacesButton) {
console.error(`[${SCRIPT_NAME}] Error: No se encontraron elementos en el DOM.`);
return;
}
// ✅ Evento: cambiar estado de "no normalizar artículos"
normalizeArticlesCheckbox.addEventListener("change", (e) => {
normalizeArticles = e.target.checked;
});
// ✅ Evento: cambiar número máximo de places
maxPlacesInput.addEventListener("input", (e) => {
maxPlaces = parseInt(e.target.value, 10);
});
// ✅ Evento: exportar palabras excluidas a XML
document.getElementById("exportExcludeWords").addEventListener("click", () => {
const savedWords = JSON.parse(localStorage.getItem("excludeWords")) || [];
if (savedWords.length === 0) {
alert("No hay palabras excluidas para exportar.");
return;
}
const sortedWords = [...savedWords].sort((a, b) => a.localeCompare(b));
const xmlContent = `<?xml version="1.0" encoding="UTF-8"?>
<ExcludedWords>
${sortedWords.map(word => ` <word>${word}</word>`).join("\n ")}
</ExcludedWords>`;
const blob = new Blob([xmlContent], { type: "application/xml" });
const url = URL.createObjectURL(blob);
const link = document.createElement("a");
link.href = url;
link.download = "excluded_words.xml";
document.body.appendChild(link); // Correctly appends the link
link.click();
document.body.removeChild(link); // Correctly removes the link
});
// ✅ Evento: añadir palabra excluida sin duplicados
addExcludeWordButton.addEventListener("click", () => {
const wordInput = document.getElementById("excludeWord") || document.getElementById("excludedWord");
const word = wordInput?.value.trim();
if (!word) return;
const lowerWord = word.toLowerCase();
const alreadyExists = excludeWords.some(w => w.toLowerCase() === lowerWord);
if (!alreadyExists) {
excludeWords.push(word);
localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
renderExcludedWordsPanel(); // Usar renderExcludedWordsPanel en lugar de updateExcludeList
}
wordInput.value = "";
});
// ✅ Evento: nuevo botón unificado de importación
importButtonUnified.addEventListener("click", () => {
hiddenInput.click(); // abre el file input oculto
});
hiddenInput.addEventListener("change", () => {
handleImportList(); // ✅ Llama a la función handleImportList al importar
});
// ✅ Evento: escanear lugares
scanPlacesButton.addEventListener("click", scanPlaces);
}
//**************************************************************************
//Nombre: NameChangeAction
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas:
// - venue (object): Objeto Place que contiene la información del lugar a modificar.
// - oldName (string): Nombre actual del lugar.
// - newName (string): Nuevo nombre sugerido para el lugar.
//Salidas: Ninguna (función constructora que crea un objeto de acción).
//Prerrequisitos si existen:
// - El objeto venue debe contar con la propiedad attributes, y dentro de ésta, con el campo id.
//Descripción:
// Esta función constructora crea un objeto que representa la acción de cambio de nombre
// de un Place en el Waze Map Editor (WME). Asigna las propiedades correspondientes:
// - Guarda el objeto venue, el nombre original (oldName) y el nuevo nombre (newName).
// - Extrae y guarda el ID único del lugar desde venue.attributes.id.
// - Establece metadatos para identificar la acción, asignando el tipo "NameChangeAction"
// y marcando isGeometryEdit como false para indicar que no se trata de una edición de geometría.
// Estos metadatos pueden ser utilizados por WME y otros plugins para gestionar y mostrar la acción.
//**************************************************************************
function NameChangeAction(venue, oldName, newName)
{
// Referencia al Place y los nombres
this.venue = venue;
this.oldName = oldName;
this.newName = newName;
// ID único del Place
this.venueId = venue.attributes.id;
// Metadatos que WME/Plugins pueden usar
this.type = "NameChangeAction";
this.isGeometryEdit = false; // no es una edición de geometría
}
/**
* 1) getActionName: nombre de la acción en el historial.
*/
NameChangeAction.prototype.getActionName = function() {
return "Update place name";
};
/** 2) getActionText: texto corto que WME a veces muestra. */
NameChangeAction.prototype.getActionText = function() {
return "Update place name";
};
/** 3) getName: algunas versiones llaman a getName(). */
NameChangeAction.prototype.getName = function() {
return "Update place name";
};
/** 4) getDescription: descripción detallada de la acción. */
NameChangeAction.prototype.getDescription = function() {
return `Place name changed from "${this.oldName}" to "${this.newName}".`;
};
/** 5) getT: título (a veces requerido por plugins). */
NameChangeAction.prototype.getT = function() {
return "Update place name";
};
/** 6) getID: si un plugin llama a e.getID(). */
NameChangeAction.prototype.getID = function() {
return `NameChangeAction-${this.venueId}`;
};
/** 7) doAction: asigna el nuevo nombre (WME llama a esto al crear la acción). */
NameChangeAction.prototype.doAction = function() {
this.venue.attributes.name = this.newName;
this.venue.isDirty = true;
if (typeof W.model.venues.markObjectEdited === "function") {
W.model.venues.markObjectEdited(this.venue);
}
};
/** 8) undoAction: revertir al nombre anterior (Ctrl+Z). */
NameChangeAction.prototype.undoAction = function() {
this.venue.attributes.name = this.oldName;
this.venue.isDirty = true;
if (typeof W.model.venues.markObjectEdited === "function") {
W.model.venues.markObjectEdited(this.venue);
}
};
/** 9) redoAction: rehacer (Ctrl+Shift+Z), vuelve a doAction. */
NameChangeAction.prototype.redoAction = function() {
return this.doAction();
};
/** 10) undoSupported / redoSupported: indica si se puede des/rehacer. */
NameChangeAction.prototype.undoSupported = function() {
return true;
};
NameChangeAction.prototype.redoSupported = function() {
return true;
};
/** 11) accept / supersede: evita fusionar con otras acciones. */
NameChangeAction.prototype.accept = function() {
return false;
};
NameChangeAction.prototype.supersede = function() {
return false;
};
/** 12) isEditAction: true => habilita "Guardar". */
NameChangeAction.prototype.isEditAction = function() {
return true;
};
/** 13) getAffectedUniqueIds: objetos que se alteran. */
NameChangeAction.prototype.getAffectedUniqueIds = function() {
return [this.venueId];
};
/** 14) isSerializable: si no implementas serialize(), pon false. */
NameChangeAction.prototype.isSerializable = function() {
return false;
};
/** 15) isActionStackable: false => no combina con otras ediciones. */
NameChangeAction.prototype.isActionStackable = function() {
return false;
};
/** 16) getFocusFeatures: WME/Plugins pueden usarlo para "enfocar" el objeto. */
NameChangeAction.prototype.getFocusFeatures = function() {
// Devolvemos el venue para indicar que ese es el foco (o un array vacío si prefieres).
return [this.venue];
};
/** 17) Métodos vacíos para evitar futuros "no es una función" si WME pide estos. */
NameChangeAction.prototype.getFocusSegments = function() {
return [];
};
NameChangeAction.prototype.getFocusNodes = function() {
return [];
};
NameChangeAction.prototype.getFocusClosures = function() {
return [];
};
/** 18) getTimestamp: método nuevo que WME/Plugins están llamando. */
NameChangeAction.prototype.getTimestamp = function() {
// Devolvemos un timestamp numérico (ms desde época UNIX).
return Date.now();
};
//**************************************************************************
//Nombre: openFloatingPanel
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: placesToNormalize (array de lugares con nombre original y sugerencias)
//Salidas: Panel flotante con opciones de normalización y eliminación
//Prerrequisitos si existen: Debe haberse definido normalizePlaceName y cargado excludeWords correctamente
//Descripción: Crea un panel interactivo donde se presentan los lugares que requieren cambios,
// permitiendo su corrección o eliminación. Solo muestra lugares que requieren cambio
// y errores ortográficos verdaderos.
//**************************************************************************
function openFloatingPanel2(placesToNormalize) {
const panel = document.createElement("div");
panel.id = "normalizer-floating-panel";
panel.style.cssText = `
position: fixed;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
background: white;
padding: 20px;
border: 1px solid #ccc;
border-radius: 8px;
box-shadow: 0 2px 10px rgba(0,0,0,0.1);
z-index: 1000;
max-height: 80vh;
overflow-y: auto;
min-width: 800px;
`;
let panelContent = `
<style>
#normalizer-table { width: 100%; border-collapse: collapse; }
#normalizer-table th, #normalizer-table td { padding: 8px; border: 1px solid #ddd; }
#normalizer-table tr:nth-child(even) { background-color: #f9f9f9; }
.warning-row { background-color: #fff3cd !important; }
.close-btn {
position: absolute;
top: 10px;
right: 10px;
background: #ccc;
color: black;
border: none;
border-radius: 4px;
width: 25px;
height: 25px;
font-size: 16px;
font-weight: bold;
cursor: pointer;
text-align: center;
line-height: 25px;
transition: background-color 0.3s ease;
}
.close-btn:hover {
background: #bbb;
}
.footer-buttons {
display: flex;
justify-content: center;
gap: 10px;
margin-top: 20px;
}
</style>
<button class="close-btn" id="close-panel-btn">X</button>
<h3>Places to Normalize</h3>
<table id="normalizer-table">
<thead>
<tr>
<th>Normalizar</th>
<th>Eliminar</th>
<th>Estado</th>
<th>Nombre Original</th>
<th>Nombre Sugerido</th>
<th>Corrección</th>
<th>Acción</th>
</tr>
</thead>
<tbody>
`;
// Procesar cada lugar y sus advertencias
placesToNormalize.forEach((place, index) => {
const { originalName, newName, spellingWarnings = [] } = place;
// Crear una fila por cada advertencia ortográfica
spellingWarnings.forEach((warning, warningIndex) => {
const suggestionId = `suggestion-${index}-${warningIndex}`;
panelContent += `
<tr class="warning-row">
<td style="text-align: center;">
<input type="checkbox" class="normalize-checkbox"
data-index="${index}"
data-warning-index="${warningIndex}"
data-suggestion-id="${suggestionId}">
</td>
<td style="text-align: center;">
<input type="checkbox" class="delete-checkbox"
data-index="${index}"
data-warning-index="${warningIndex}">
</td>
<td style="text-align: center;">⚠️</td>
<td id="name-cell-${index}-${warningIndex}">${originalName}</td>
<td>
<input type="text" class="new-name-input"
data-index="${index}"
data-warning-index="${warningIndex}"
data-place-id="${place.id}"
data-suggestion-id="${suggestionId}"
value="${newName}">
</td>
<td>${warning.original} → ${warning.sugerida} (${warning.tipo})</td>
<td style="text-align: center;">
<button class="apply-suggestion-btn"
data-index="${index}"
data-warning-index="${warningIndex}"
title="Corregir ortografía de la palabra"
style="background-color: #4CAF50; color: white; border: none; padding: 5px 10px; border-radius: 4px; cursor: pointer;">
Fix
</button>
<button class="add-exclude-btn"
data-word="${warning.original}"
title="Adicionar palabra excluida nueva"
style="background-color: #007bff; color: white; border: none; padding: 5px 10px; border-radius: 4px; cursor: pointer;">
Add Excld
</button>
</td>
</tr>`;
});
});
panelContent += `
</tbody>
</table>
<div class="footer-buttons">
<button class="apply-changes-btn" id="apply-changes-btn">✔️ Apply Changes</button>
<button class="cancel-btn" id="cancel-btn">❌ Cancel</button>
</div>
`;
panel.innerHTML = panelContent;
document.body.appendChild(panel);
// Eventos para los botones
document.getElementById('close-panel-btn').addEventListener('click', () => {
panel.remove(); // Cerrar el panel flotante
});
document.getElementById('apply-changes-btn').addEventListener('click', () => {
window.applyNormalization(); // Llamar a la función global para aplicar cambios
});
document.getElementById('cancel-btn').addEventListener('click', () => {
panel.remove(); // Cerrar el panel flotante
});
// Asignar eventos a los botones "Fix"
document.querySelectorAll('.apply-suggestion-btn').forEach(button => {
button.addEventListener('click', function () {
const index = this.dataset.index;
const warningIndex = this.dataset.warningIndex;
const input = document.querySelector(
`.new-name-input[data-index="${index}"][data-warning-index="${warningIndex}"]`
);
const checkbox = document.querySelector(
`.normalize-checkbox[data-index="${index}"][data-warning-index="${warningIndex}"]`
);
if (input && checkbox) {
const warning = placesToNormalize[index].spellingWarnings[warningIndex];
input.value = input.value.replace(warning.original, warning.sugerida || warning.original);
checkbox.checked = true; // Marcar el checkbox
}
});
});
// Asignar eventos a los botones "Add Excld Word"
document.querySelectorAll('.add-exclude-btn').forEach(button => {
button.addEventListener('click', function () {
const word = this.dataset.word;
if (!excludeWords.includes(word)) {
excludeWords.push(word);
excludeWords.sort((a, b) => a.localeCompare(b)); // Ordenar alfabéticamente
localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
renderExcludedWordsPanel(); // Actualizar el panel lateral
// Mostrar popup en el centro del panel flotante
const panel = document.getElementById("normalizer-floating-panel");
const popup = document.createElement('div');
popup.textContent = `✅ The Word "${word}" has been added to the exclusion list.`;
popup.style.position = 'absolute';
popup.style.top = '50%';
popup.style.left = '50%';
popup.style.transform = 'translate(-50%, -50%)';
popup.style.backgroundColor = '#4CAF50';
popup.style.color = 'white';
popup.style.padding = '10px 20px';
popup.style.borderRadius = '5px';
popup.style.zIndex = '10000';
popup.style.opacity = '1';
popup.style.transition = 'opacity 1s ease-in-out';
panel.appendChild(popup);
setTimeout(() => {
popup.style.opacity = '0';
setTimeout(() => panel.removeChild(popup), 1000);
}, 2000);
// Bloquear el botón y cambiar su texto
this.textContent = 'Excluded Word Added';
this.disabled = true;
this.style.backgroundColor = '#6c757d'; // Cambiar color a gris
this.style.cursor = 'not-allowed';
} else {
alert(`⚠️ The word "${word}" is already on the exclusion list.`);
}
});
});
}
//**************************************************************************
//Nombre: waitForWME
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: Ninguna
//Salidas: Ninguna
//Prerrequisitos si existen:
// - El objeto global W debe estar disponible.
// - Deben estar definidas las funciones: initializeExcludeWords, createSidebarTab y renderExcludedWordsPanel.
//Descripción:
// Esta función espera a que el entorno de edición de Waze (WME) esté completamente cargado,
// verificando que existan los objetos necesarios para iniciar el script.
// Una vez disponible, inicializa la lista de palabras excluidas desde localStorage,
// crea el tab lateral personalizado y renderiza la lista visual de palabras excluidas.
// Si WME aún no está listo, vuelve a intentar cada 1000 ms.
//**************************************************************************
function waitForWME2() {
if (W && W.userscripts && W.model && W.model.venues) {
console.log(`[${SCRIPT_NAME}] Inicializando v${VERSION}`);
initializeExcludeWords(); // ⚠️ Usa solo localStorage
createSidebarTab();
waitForDOM("#normalizer-tab", () => {
console.log("[waitForWME] 🧩 Sidebar listo, renderizando palabras excluidas");
renderExcludedWordsPanel(); // Muestra las palabras
setupDragAndDropImport(); // Activa drag & drop
//Evita que se abra el archivo si cae fuera del área
// window.addEventListener("dragover", e => e.preventDefault(), false);
// window.addEventListener("drop", e => e.preventDefault(), false);
});
} else {
console.log(`[${SCRIPT_NAME}] Esperando que WME esté listo...`);
setTimeout(waitForWME, 1000);
}
}
//**************************************************************************
//Nombre: init
//Fecha modificación: 2025-03-30
//Autor: mincho77
//Entradas: Ninguna
//Salidas: Ninguna
//Prerrequisitos si existen:
// - El objeto global W debe estar disponible.
// - Deben estar definidas las funciones: initializeExcludeWords, createSidebarTab, waitForDOM, renderExcludedWordsPanel y setupDragAndDropImport.
//Descripción:
// Esta función espera a que el entorno de edición de Waze (WME) esté completamente cargado,
// verificando que existan los objetos necesarios para iniciar el script. Si WME aún no está listo,
// reintenta la inicialización cada 1000 ms. Una vez disponible, inicializa la lista de palabras
// excluidas, crea el tab lateral personalizado, espera a que el DOM del tab esté listo para
// renderizar el contenido (palabras excluidas y funcionalidad drag & drop), y expone globalmente
// las funciones applyNormalization y normalizePlaceName para uso externo.
//**************************************************************************
function init() {
if (!W || !W.userscripts || !W.model || !W.model.venues) {
console.log(`[${SCRIPT_NAME}] Esperando que WME esté listo...`);
setTimeout(init, 1000);
return;
}
console.log(`[${SCRIPT_NAME}] Inicializando v${VERSION}`);
initializeExcludeWords();
createSidebarTab();
waitForDOM("#normalizer-tab", () => {
console.log("[init] 🧩 Sidebar listo, renderizando palabras excluidas");
renderExcludedWordsPanel();
setupDragAndDropImport();
});
window.applyNormalization = applyNormalization;
window.normalizePlaceName = normalizePlaceName;
}
init();
} catch (e) {
console.error('[PlacesNameNormalizer] Fatal initialization error:', e);
}
})();