WME Places Name Normalizer

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

当前为 2025-03-28 提交的版本,查看 最新版本

您需要先安装一个扩展,例如 篡改猴Greasemonkey暴力猴,之后才能安装此脚本。

You will need to install an extension such as Tampermonkey to install this script.

您需要先安装一个扩展,例如 篡改猴暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴Userscripts ,之后才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。

您需要先安装用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name         WME Places Name Normalizer
// @namespace    https://greasyfork.org/en/users/mincho77
// @version      2.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/*editor*
// @exclude     https://beta.waze.com/*user/*editor/*
// @exclude     https://www.waze.com/*user/*editor/*
// @exclude     https://www.waze.com/discuss/*
// @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.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 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 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;
            }

            // Si empieza con número seguido de letras, conservar como está (para casos como 2Go, 2Sur)
            if (/^\d+[A-Z][a-zA-Z]*$/.test(word))
            {
                return word;
            }

            // Conservar texto dentro de paréntesis si ya está todo en mayúsculas o minúsculas
            if (/^\(.*\)$/.test(word))
            {
                const inner = word.slice(1, -1); // quitar paréntesis
                if (inner === inner.toUpperCase() || inner === inner.toLowerCase()) {
                    return word;
                }
            }
            // 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());

        //Quita el punto final en el nombre
        name = name.replace(/\.$/, "");

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