WME Places Name Normalizer

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

目前為 2025-03-27 提交的版本,檢視 最新版本

// ==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*
// @grant        GM_xmlhttpRequest
// @grant        unsafeWindow
// @license      MIT
// @run-at       document-end
// ==/UserScript==
/*global W*/

(() => {
    "use strict";
    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 = 20;
    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])+\.?$/;

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

    function initializeExcludeWords()
    {
         const saved = JSON.parse(localStorage.getItem("excludeWords")) || [];
        excludeWords = [...new Set([...excludeWords, ...saved])].sort((a, b) => a.localeCompare(b));
        localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
    }

     unsafeWindow.normalizePlaceName = function(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 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;
            }

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

   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);
                            resolve(data);
                        } catch (err) {
                            reject(err);
                        }
                    } else {
                        reject(`Error HTTP: ${response.status}`);
                    }
                },
                onerror: function(err) {
                    reject(err);
                }
            });
        });
    }

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

    function renderExcludedWordsSidebar() {
        const container = document.getElementById("normalizer-sidebar");
        if (!container) return;

        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">
        ${excludeWords.sort((a, b) => a.localeCompare(b)).map(w => `<li>${w}</li>`).join("")}
      </ul>
    </div>
  `;
        container.appendChild(excludeListSection);

    }

    

    function createSidebarTab() {
        const { tabLabel, tabPane } = W.userscripts.registerSidebarTab("PlacesNormalizer");

        if (!tabPane) {
            console.error(`[${SCRIPT_NAME}] Error: No se pudo registrar el sidebar tab.`);
            return;
        }

        tabLabel.innerHTML = `
  <img src=""
  style="height: 16px; vertical-align: middle; margin-right: 5px;">
  NrmliZer
`;
        tabLabel.title = "Places Name Normalizer";
        tabPane.innerHTML = getSidebarHTML();


        setTimeout(() => {
            // Llamar a la función para esperar el DOM antes de ejecutar eventos
            waitForDOM("#normalizer-tab", attachEvents);
            //attachEvents();
        }, 500);


    }


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

    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='500' style='width: 60px;'>
      </div>

      <div>
        <label>Palabras Excluidas:</label>
        <input type='text' id='excludeWord' style='width: 120px;'>
        <button id='addExcludeWord'>Add Word</button>

        <div style="max-height: 100px; overflow-y: auto; border: 1px solid #ccc; padding: 5px; margin-top: 5px;">
          <ul id="excludedWordsList" style="padding-left: 20px; margin: 0;">
            ${excludeWords.sort((a, b) => a.localeCompare(b)).map(w => `<li>${w}</li>`).join("")}
          </ul>
        </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>

      <hr>
      <button id="scanPlaces">Scan...</button>
    </div>
  `;
    }


 function getSidebarHTML2() {
  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='500' style='width: 60px;'>
      </div>

     <div>
  <label>Palabras Excluidas:</label>
  <input type='text' id='excludeWord' style='width: 120px;'>
  <button id='addExcludeWord'>Add Word</button>

  <div style="max-height: 100px; overflow-y: auto; border: 1px solid #ccc; padding: 5px; margin-top: 5px;">
    <ul id="excludedWordsList" style="padding-left: 20px; margin: 0;">
      ${excludeWords.sort((a, b) => a.localeCompare(b)).map(w => `<li>${w}</li>`).join("")}
    </ul>
  </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>

      <hr>
      <button id="scanPlaces">Scan...</button>
    </div>
  `;
}

    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);
            link.click();
            document.body.removeChild(link);
        });

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

            if (!word) return;

            const alreadyExists = excludeWords.some(w => w.toLowerCase() === word.toLowerCase());
            if (!alreadyExists) {
                excludeWords.push(word);
                localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
                updateExcludeList();
            }

            wordInput.value = "";
        });

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

        hiddenInput.addEventListener("change", () => {
            const file = hiddenInput.files[0];
            if (!file) return;

            const reader = new FileReader();
            reader.onload = function (event) {
                const parser = new DOMParser();
                const xml = parser.parseFromString(event.target.result, "application/xml");
                const words = Array.from(xml.getElementsByTagName("word")).map(node => node.textContent.trim());

                if (words.length > 0) {
                    excludeWords = words;
                    localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
                    updateExcludeList();
                    alert(`Palabras excluidas importadas correctamente: ${words.length}`);
                } else {
                    alert("No se encontraron palabras en el archivo XML.");
                }
            };
            reader.readAsText(file);
        });

        // ✅ 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("");
    }
    

    function scanPlaces() {
        const allPlaces = W.model.venues.getObjectArray();
        console.log(`[${SCRIPT_NAME}] Iniciando escaneo de lugares...`);

        // const inputValue = document.getElementById("maxPlacesInput")?.value;
        maxPlaces = parseInt(document.getElementById("maxPlacesInput").value, 10) || 20;

        console.log("➡️ Usando maxPlaces =", maxPlaces);

        const venues = Object.values(W.model.venues.objects);
        const sliced = venues.slice(0, maxPlaces);

        if (!W || !W.model || !W.model.venues || !W.model.venues.objects) {
            console.error(`[${SCRIPT_NAME}] WME no está listo.`);
            return;
        }

        // Obtener el nivel del editor; si no existe, usamos Infinity para incluir todos.
        let editorLevel = (W.model.user && typeof W.model.user.level === "number")
        ? W.model.user.level
        : Infinity;

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

        if (places.length === 0) {
            alert("No se encontraron Places en WME.");
            return;
        }

        placesToNormalize = allPlaces
            .filter(place =>
                    place &&
                    typeof place.getID === "function" &&
                    place.attributes &&
                    typeof place.attributes.name === "string"
                   )
            .map(place => ({
            id: place.getID(),
            name: place.attributes.name,
            attributes: place.attributes,
            place: place
        }));


        // Luego se mapea y se sigue con el flujo habitual...
        let placesMapped = placesToNormalize.map(place => {
            let originalName = place.attributes.name;
            let newName = normalizePlaceName(originalName);
            return {
                id: place.attributes.id,
                originalName,
                newName
            };
        });

        let filteredPlaces = placesMapped.filter(p =>
                                                 p.newName.trim() !== p.originalName.trim()
                                                );

        console.log(`[${SCRIPT_NAME}] Lugares que cambiarán: ${filteredPlaces.length}`);

        if (filteredPlaces.length === 0) {
            alert("No se encontraron Places que requieran cambio.");
            return;
        }

        openFloatingPanel(filteredPlaces);
    }


    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() {
        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();
    };


    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) {
            console.log("ℹ️ No hay lugares seleccionados para normalizar o eliminar.");
            return;
        }

        // Confirmación antes de procesar todo si TODOS están seleccionados para eliminar
        const allDeleteBoxes = document.querySelectorAll(".delete-checkbox");
        if (deleteCheckboxes.length === allDeleteBoxes.length) {
            const confirmDeleteAll = confirm("⚠️ Has seleccionado TODOS los lugares para eliminar. ¿Estás seguro?");
            if (!confirmDeleteAll) {
                console.log("🚫 Eliminación masiva cancelada por el usuario.");
                return;
            }
        }

        // Normalizar nombres
        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(`✅ Acción aplicada: "${currentName}" → "${newName}"`);
                    changesMade = true;
                } catch (error) {
                    console.error("⛔ Error aplicando la acción de actualización:", error);
                }
            } else {
                console.log(`⏭ Sin cambios reales para ID ${placeId}`);
            }
        });

        // Eliminar lugares marcados
        deleteCheckboxes.forEach(cb => {
            const index = cb.dataset.index;
            const input = document.querySelector(`.new-name-input[data-index="${index}"]`);
            const placeId = input?.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 con ID:", placeId, error);
            }
        });

        if (changesMade) {
            console.log("💾 Cambios marcados. Recuerda presionar el botón de guardar en el editor.");
        } else {
            console.log("ℹ️ No hubo cambios para aplicar.");
        }

        // ✅ Cerrar panel flotante
        const panel = document.getElementById("normalizer-floating-panel");
        if (panel) panel.remove();
    }

    


    // Función de similitud leve entre palabras
    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;
    }

    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|ii|iii|iv|v|vi|vii|viii|ix|x|xi|xii|xiii|xiv|xv|xvi|xvii|xviii|xix|xx|xxi|xxii|xxiii|xxiv|xxv)$/i.test(word);

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

            // Si la palabra está en la lista de excluidas (ignorando mayúsculas)
            const matchExcluded = excludeWords.find(w => w.toLowerCase() === lowerWord);
            if (matchExcluded) return matchExcluded;

            // Si es número romano
            if (isRoman(word)) return word.toUpperCase();

            // Artículos (si opción marcada y no es la primera palabra)
            if (!normalizeArticles && articles.includes(lowerWord) && index !== 0) {
                return lowerWord;
            }

            // Capitalización normal
            return word.charAt(0).toUpperCase() + word.slice(1).toLowerCase();
        });

        // Reemplazar pipes con guiones
        name = normalizedWords.join(" ").replace(/\s*\|\s*/g, " - ");

        // Mayúscula después de (, ", [, '
        name = name.replace(/([(\["'])(\s*)(\p{L})/gu, (match, p1, p2, p3) => p1 + p2 + p3.toUpperCase());

        // Asegurar espacios alrededor del guion
        name = name.replace(/\s*-\s*/g, " - ");

        // Preservar mayúsculas en letras pegadas a números (ej: 45A)
        name = name.replace(/\b(\d+)([A-Z])\b/g, (match, num, letter) => num + letter.toUpperCase());

        // Limpiar espacios dobles
        return name.replace(/\s{2,}/g, " ").trim();
    }

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



    function openFloatingPanel(placesToNormalize) {
        console.log(`[${SCRIPT_NAME}] Creando panel flotante...`);

        if (!placesToNormalize || placesToNormalize.length === 0) {
            console.warn(`[${SCRIPT_NAME}] No hay lugares para normalizar.`);
            return;
        }

        // Elimina cualquier panel flotante previo
        let existingPanel = document.getElementById("normalizer-floating-panel");
        if (existingPanel) existingPanel.remove();

        // Crear el panel flotante
        let panel = document.createElement("div");
        panel.id = "normalizer-floating-panel";
        panel.style.position = "fixed";
        panel.style.top = "100px";
        panel.style.left = "300px"; // deja espacio para la barra lateral
        panel.style.width = "calc(100vw - 400px)"; // margen adicional para que no se desborde
        panel.style.maxWidth = "calc(100vw - 30px)";
        panel.style.zIndex = 10000;
        panel.style.backgroundColor = "white";
        panel.style.border = "1px solid #ccc";
        panel.style.padding = "15px";
        panel.style.boxShadow = "0 2px 10px rgba(0,0,0,0.3)";
        panel.style.borderRadius = "8px";

        // Contenido del panel
        let panelContent = `

  <h3 style="text-align: center;">Lugares para Normalizar</h3>
        <div style="max-height: 60vh; overflow-y: auto; margin-bottom: 10px;">
           <table style="width: 100%; border-collapse: collapse;">
    <thead>
  <tr style="border-bottom: 2px solid black;">
    <th style="width: 40px; text-align: center;">
      <input type="checkbox" id="selectAllCheckbox" title="Seleccionar todos para aplicar normalización">
    </th>
    <th style="width: 40px; text-align: center;">
      <input type="checkbox" id="selectAllDeleteCheckbox" title="Seleccionar todos para eliminar">
    </th>

  </tr>
  <tr style="border-bottom: 2px solid #ccc;">
    <th style="text-align: center;" title="Aplicar normalización">🛠️</th>
    <th style="text-align: center;" title="Marcar para eliminar">🗑️</th>
    <th style="text-align: left;">Nombre Actual</th>
    <th style="text-align: left;">Nuevo Nombre</th>
  </tr>
</thead>
            <tbody>
    `;

        maxPlaces = parseInt(document.getElementById("maxPlacesInput").value, 10) || 20;
        placesToNormalize.sort((a, b) => a.originalName.localeCompare(b.originalName));
        placesToNormalize.slice(0, maxPlaces).forEach((place, index) => {
            // placesToNormalize.forEach((place, index) => {
            if (place && place.originalName) {
                const originalName = place.originalName;
                let newName = normalizePlaceName(originalName);
                // Escapa comillas dobles para evitar romper el HTML
                newName = newName.replace(/"/g, '&quot;');
                const placeId = place.id;
                panelContent += `
  <tr>
    <td style="text-align: center;"><input type="checkbox" class="normalize-checkbox" data-index="${index}"></td>
    <td style="text-align: center;"><input type="checkbox" class="delete-checkbox" data-index="${index}"></td>
    <td id="name-cell-${index}">${originalName}</td>
    <td><input type="text" class="new-name-input" data-index="${index}" data-place-id="${place.id}" value="${newName}" style="width: 100%;"></td>
  </tr>
`;
            }
        });
        panelContent += `</tbody></table>`;

        // Agregar botones al panel sin eventos inline
        // Ejemplo de la sección de botones en panelContent:
        panelContent += `
    <button id="applyNormalizationBtn" style="margin-top: 10px; width: 100%; padding: 8px; background: #4CAF50; color: white; border: none; cursor: pointer;">
        Aplicar Normalización
    </button>
    <button id="closeFloatingPanelBtn" style="margin-top: 5px; width: 100%; padding: 8px; background: #d9534f; color: white; border: none; cursor: pointer;">
        Cerrar
    </button>
`;

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

        // Sincroniza comportamiento entre eliminar y normalizar
        document.querySelectorAll(".delete-checkbox").forEach(deleteCheckbox => {
            deleteCheckbox.addEventListener("change", function () {
                const index = this.dataset.index;
                const normalizeCheckbox = document.querySelector(`.normalize-checkbox[data-index="${index}"]`);
                const originalNameCell = this.closest("tr").querySelector("td:nth-child(3)");

                if (this.checked) {
                    normalizeCheckbox.checked = true;
                    originalNameCell.style.color = "red";
                    originalNameCell.style.fontWeight = "bold";
                } else {
                    normalizeCheckbox.checked = false; // ❗ Desmarcar también el de normalizar
                    originalNameCell.style.color = "";
                    originalNameCell.style.fontWeight = "";
                }
            });
     /*   document.querySelectorAll(".delete-checkbox").forEach(deleteCheckbox => {
            deleteCheckbox.addEventListener("change", function () {
                const index = this.dataset.index;
                const normalizeCheckbox = document.querySelector(`.normalize-checkbox[data-index="${index}"]`);
                const originalNameCell = this.closest("tr").querySelector("td:nth-child(3)");

                if (this.checked) {
                    normalizeCheckbox.checked = true;
                    originalNameCell.style.color = "red";
                    originalNameCell.style.fontWeight = "bold";
                } else {
                    originalNameCell.style.color = "";
                    originalNameCell.style.fontWeight = "";
                }
            });*/
        });

        // ✅ Seleccionar todos para normalizar
        document.getElementById("selectAllCheckbox").addEventListener("change", function () {
            const isChecked = this.checked;
            document.querySelectorAll(".normalize-checkbox").forEach(cb => {
                cb.checked = isChecked;
            });
        });

      /*  // ✅ Seleccionar todos para eliminar
        document.getElementById("selectAllDeleteCheckbox").addEventListener("change", function () {
            const isChecked = this.checked;
            document.querySelectorAll(".delete-checkbox").forEach(cb => {
                cb.checked = isChecked;
            });
        });*/
        // Evento para seleccionar todos los eliminar
document.getElementById("selectAllDeleteCheckbox").addEventListener("change", function () {
  const isChecked = this.checked;
  const deleteCheckboxes = document.querySelectorAll(".delete-checkbox");

  if (isChecked) {
    const confirmDeleteAll = confirm("⚠️ ¿Estás seguro de seleccionar TODOS los lugares para eliminar?");
    if (!confirmDeleteAll) {
      this.checked = false;
      return;
    }
  }

  deleteCheckboxes.forEach(cb => {
    const index = cb.dataset.index;
    const normalizeCheckbox = document.querySelector(`.normalize-checkbox[data-index="${index}"]`);
    const nameCell = document.getElementById(`name-cell-${index}`);

    cb.checked = isChecked;

    if (isChecked) {
      if (normalizeCheckbox) normalizeCheckbox.checked = true;
      if (nameCell) nameCell.style.color = "red";
    } else {
      if (normalizeCheckbox) normalizeCheckbox.checked = false;
      if (nameCell) nameCell.style.color = "";
    }
  });
});

        // ✅ Seleccionar para cerrar
        document.getElementById("closeFloatingPanelBtn").addEventListener("click", function() {
            let panel = document.getElementById("normalizer-floating-panel");
            if (panel) panel.remove();
        });
       /* // Evento para corregir ortografía en cada input del panel:
        document.getElementById("checkSpellingBtn").addEventListener("click", function() {
            const inputs = document.querySelectorAll(".new-name-input");
            inputs.forEach(input => {
                const text = input.value;
                applySpellCorrection(text).then(corrected => {
                    input.value = corrected;
                });
            });
        });*/

        // Evento para aplicar normalización
    document.getElementById("applyNormalizationBtn").addEventListener("click", function () {
  const confirmed = confirm("¿Estás seguro de que deseas aplicar los cambios?");
  if (!confirmed) return;

  let changesMade = false;

  document.querySelectorAll(".normalize-checkbox").forEach(cb => {
    const index = cb.dataset.index;

    const placeId = document.querySelector(`.new-name-input[data-index="${index}"]`)?.getAttribute("data-place-id");
    const newName = document.querySelector(`.new-name-input[data-index="${index}"]`)?.value?.trim();
    const deleteCb = document.querySelector(`.delete-checkbox[data-index="${index}"]`);
    const place = W.model.venues.getObjectById(placeId);

    if (!place || !place.attributes?.name) return;

    // ✅ Eliminar si está seleccionado para eliminar
    if (deleteCb?.checked) {
      try {
        const DeleteObject = require("Waze/Action/DeleteObject");
        W.model.actionManager.add(new DeleteObject(place));
        console.log(`🗑 Eliminado: ${place.attributes.name}`);
        changesMade = true;
        return; // Saltar normalización si se elimina
      } catch (error) {
        console.error("⛔ Error al eliminar:", error);
      }
    }

    // ✅ Normalizar si está seleccionado para ello y no fue eliminado
    if (cb.checked && place.attributes.name.trim() !== newName) {
      try {
        const UpdateObject = require("Waze/Action/UpdateObject");
        const action = new UpdateObject(place, { name: newName });
        W.model.actionManager.add(action);
        console.log(`✅ Normalizado: ${place.attributes.name} → ${newName}`);
        changesMade = true;
      } catch (error) {
        console.error("⛔ Error al actualizar:", error);
      }
    }
  });

  // ✅ Marcar cambios si hubo acciones
  if (changesMade) {
    if (W.controller && typeof W.controller.setModified === "function") {
      W.controller.setModified(true);
    }
    console.log("💾 Cambios marcados para guardar.");
  } else {
    console.log("ℹ️ No hubo cambios para aplicar.");
  }

  // ✅ Cerrar panel flotante
  const panel = document.getElementById("normalizer-floating-panel");
  if (panel) panel.remove();
});
    }

    function loadExcludeWordsFromXML(callback) {
  fetch("excludeWords.xml")
    .then(response => {
      if (!response.ok) throw new Error("No se encontró el archivo XML");
      return response.text();
    })
    .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 loadExcludeWordsFromXML2(callback) {
        fetch("excludeWords.xml")
            .then(response => {
            if (!response.ok) throw new Error("No se encontró el archivo XML");
            return response.text();
        })
           .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());

            // Fusionar palabras actuales con las del archivo, eliminar duplicados y ordenar
            const current = JSON.parse(localStorage.getItem("excludeWords")) || [];
            const merged = [...new Set([...current, ...wordsFromXML])].sort((a, b) => a.localeCompare(b));

            excludeWords = merged;
            localStorage.setItem("excludeWords", JSON.stringify(merged));
            updateExcludeList();

            exportExcludeWordsToXML(); // vuelve a exportar el XML actualizado

            if (callback) callback();
        })
            .catch(() => {
            console.warn("⚠️ No se pudo cargar excludeWords.xml. Usando lista por defecto.");
            excludeWords = ["EDS", "IPS", "McDonald's", "EPS"];
            localStorage.setItem("excludeWords", JSON.stringify(excludeWords));
            updateExcludeList();
            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);
    }

    function waitForWME() {
        if (W && W.userscripts && W.model && W.model.venues) {
            console.log(`[${SCRIPT_NAME}] Inicializando v${VERSION}`);
            loadExcludeWordsFromXML(() =>
            {
               // initializeExcludeWords();     // 1. Carga de localStorage o palabras por defecto
                createSidebarTab();           // 3. Ya tienes excludeWords listas para usar
                renderExcludedWordsSidebar(); // 4. Renderiza la lista actual (ordenada)
            });
        }
        else {
            console.log(`[${SCRIPT_NAME}] Esperando que WME esté listo...`);
            setTimeout(waitForWME, 1000);
        }
    }
    console.log(window.applyNormalization);
window.applyNormalization = applyNormalization;
    waitForWME();
})();

QingJ © 2025

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