Traductor Plus 3.0!

Traduce el texto seleccionado usando la API gratuita de Google Translate y muestra los resultados de manera mejorada con opciones de color personalizadas y configuración de idioma.

// ==UserScript==
// @name         Traductor Plus 3.0!
// @namespace    https://greasyfork.org/es/scripts/486611-traductor-plus-3-0
// @version      0.2
// @icon         
// @description  Traduce el texto seleccionado usando la API gratuita de Google Translate y muestra los resultados de manera mejorada con opciones de color personalizadas y configuración de idioma.
// @author       PutoElQueLoLea
// @match        *://*/*
// @grant        GM_addStyle
// @grant        GM_registerMenuCommand
// @grant        GM_setValue
// @grant        GM_getValue
// @license      GPL-3.0-or-later

// ==/UserScript==

(function () {
    'use strict';

    const languages = {
        "Afrikáans": "af",
        "Albanés": "sq",
        "Alemán": "de",
        "Amhárico": "am",
        "Árabe": "ar",
        "Armenio": "hy",
        "Azerbaiyano": "az",
        "Bengalí": "bn",
        "Bielorruso": "be",
        "Birmano": "my",
        "Bosnia": "bs",
        "Búlgaro": "bg",
        "Canarés": "kn",
        "Catalán": "ca",
        "Cebuano": "ceb",
        "Checo": "cs",
        "Chichewa": "ny",
        "Chino simplificado": "zh-CN",
        "Chino tradicional": "zh-TW",
        "Cingalés": "si",
        "Coreano": "ko",
        "Corso": "co",
        "Criollo haitiano": "ht",
        "Croata": "hr",
        "Danés": "da",
        "Eslovaco": "sk",
        "Esloveno": "sl",
        "Español": "es",
        "Esperanto": "eo",
        "Estonio": "et",
        "Euskera": "eu",
        "Finés": "fi",
        "Francés": "fr",
        "Frisón occidental": "fy",
        "Gaélico escocés": "gd",
        "Galés": "cy",
        "Gallego": "gl",
        "Georgiano": "ka",
        "Griego": "el",
        "Gujarati": "gu",
        "Hausa": "ha",
        "Hawaiano": "haw",
        "Hebreo": "iw",
        "Hindi": "hi",
        "Hmong": "hmn",
        "Holandés": "nl",
        "Húngaro": "hu",
        "Igbo": "ig",
        "Indonesio": "id",
        "Inglés": "en",
        "Irlandés": "ga",
        "Islandés": "is",
        "Italiano": "it",
        "Japonés": "ja",
        "Javanés": "jw",
        "Kazajo": "kk",
        "Kirguís": "ky",
        "Kurdo": "ku",
        "Laosiano": "lo",
        "Latín": "la",
        "Letón": "lv",
        "Lituano": "lt",
        "Luxemburgués": "lb",
        "Macedonio": "mk",
        "Malayo": "ms",
        "Malayalam": "ml",
        "Malgache": "mg",
        "Maltés": "mt",
        "Maorí": "mi",
        "Maratí": "mr",
        "Mongol": "mn",
        "Neerlandés": "nl",
        "Nepalí": "ne",
        "Noruego": "no",
        "Panyabí": "pa",
        "Pastún": "ps",
        "Persa": "fa",
        "Polaco": "pl",
        "Portugués": "pt",
        "Rumano": "ro",
        "Ruso": "ru",
        "Samoano": "sm",
        "Serbio": "sr",
        "Sesoto": "st",
        "Shona": "sn",
        "Sindi": "sd",
        "Somalí": "so",
        "Suajili": "sw",
        "Sueco": "sv",
        "Sundanés": "su",
        "Tagalo": "tl",
        "Tailandés": "th",
        "Tamil": "ta",
        "Tayiko": "tg",
        "Telugu": "te",
        "Turco": "tr",
        "Ucraniano": "uk",
        "Urdu": "ur",
        "Uzbeco": "uz",
        "Vietnamita": "vi",
        "Xhosa": "xh",
        "Yidis": "yi",
        "Yoruba": "yo",
        "Zulú": "zu"
    };

    let selectedTextCache = '';
    let translationBox = null;

    let selectedBackgroundColor = GM_getValue('selectedBackgroundColor', '#333');
    let selectedTextColor = GM_getValue('selectedTextColor', 'white');
    let targetLanguage = GM_getValue('config', { targetLanguage: 'es' }).targetLanguage;

    function translateText(text, targetLang) {
        return new Promise((resolve, reject) => {
            const url = `https://translate.googleapis.com/translate_a/single?client=gtx&sl=auto&tl=${targetLang}&dt=t&q=${encodeURI(text)}`;

            fetch(url)
                .then(response => response.json())
                .then(data => {
                    const translatedText = data[0].map(sentence => sentence[0]).join(' '); // Unimos las traducciones individuales
                    resolve(translatedText);
                })
                .catch(error => {
                    reject(error);
                });
        });
    }

    function createTranslationBox(translatedText) {
        translationBox = document.createElement('div');
        translationBox.style.position = 'fixed';
        translationBox.style.top = '50%';
        translationBox.style.left = '50%';
        translationBox.style.transform = 'translate(-50%, -50%)';
        translationBox.style.padding = '20px';
        translationBox.style.background = selectedBackgroundColor;
        translationBox.style.color = selectedTextColor;
        translationBox.style.borderRadius = '10px';
        translationBox.textContent = translatedText;

        document.body.appendChild(translationBox);
    }

    function clearSelection() {
        selectedTextCache = '';
        if (translationBox) {
            translationBox.remove();
            translationBox = null;
        }
    }

    document.addEventListener('mouseup', async function () {
        let selectedText = window.getSelection().toString().trim();
        if (selectedText !== '') {
            selectedTextCache += selectedText + ';';

            try {
                const translatedText = await translateText(selectedTextCache, targetLanguage);

                // Guardar colores seleccionados
                saveColors();

                createTranslationBox(translatedText);
            } catch (error) {
                console.error('Error al traducir:', error);
            }
        }
    });

    document.addEventListener('click', clearSelection);

    // Configurador de colores y idioma
    GM_registerMenuCommand('Configurar Colores e Idioma', openConfigDialog);

    function openConfigDialog() {
        // Crear la ventana emergente
        const configDialog = document.createElement('div');
        configDialog.id = 'config-dialog';
        configDialog.innerHTML = `
            <h3>Configuración del traductor</h3>
            <div>
                Seleccione color de Fondo:
                <input type="color" id="color-picker-background" value="${selectedBackgroundColor}">
            </div></p></p>
            <div>
                Seleccione color de Texto:
                <input type="color" id="color-picker-text" value="${selectedTextColor}">
            </div></p></p>
            <div>
                <label for="languageList">Seleccione el idioma de traducción:</label><br>
               <center> <select id="languageList"></select></p>
            </div>
           <center> <button id="btnAceptar">Aceptar</button>
            <button id="btnCancelar">Cancelar</button></center>
        `;

        // Agregar opciones de idiomas al listbox
        const languageList = configDialog.querySelector('#languageList');
        for (const language in languages) {
            const option = document.createElement('option');
            option.value = languages[language];
            option.text = language;
            languageList.add(option);
        }

        // Establecer el idioma seleccionado en la configuración actual
        languageList.value = targetLanguage;

        // Establecer estilos
        GM_addStyle(`
            #config-dialog {
                position: fixed;
                top: 50%;
                left: 50%;
                transform: translate(-50%, -50%);
                padding: 20px;
                background: #333;
                color: #FFF;
                border: 2px solid #000;
                border-radius: 25px;
                box-shadow: 0 0 10px rgba(0, 0, 0, 0.5);
                z-index: 9999;
            }

            #btnAceptar, #btnCancelar {
                margin-top: 10px;
                padding: 5px 10px;
                cursor: pointer;
                border-radius: 25px;
            }

            #btnAceptar {
                background: #005200;
                color: #fff;
                border: none;
                margin-right: 10px;
            }

            #btnCancelar {
                background: #99c599;
                color: #fff;
                border: none;
            }
        `);

        // Agregar la ventana emergente al cuerpo del documento
        document.body.appendChild(configDialog);

        // Añadir eventos a los botones
        const btnAceptar = document.getElementById('btnAceptar');
        btnAceptar.addEventListener('click', () => {
            // Guardar la configuración
            targetLanguage = languageList.value;
            GM_setValue('config', { targetLanguage: targetLanguage });
            closeConfigDialog(); // Cierra la ventana después de aceptar
            // Actualizar colores en el cuadro de traducción
            updateTranslationBoxColors();
        });

        const btnCancelar = document.getElementById('btnCancelar');
        btnCancelar.addEventListener('click', () => {
            closeConfigDialog(); // Cierra la ventana después de cancelar
        });

        // Añadir evento onchange al selector de color de fondo para actualizar el color del cuadro
        const colorPickerBackground = document.getElementById('color-picker-background');
        colorPickerBackground.addEventListener('input', updateColorBackground);

        // Añadir evento onchange al selector de color de texto para actualizar el color del cuadro
        const colorPickerText = document.getElementById('color-picker-text');
        colorPickerText.addEventListener('input', updateColorText);
    }

    function closeConfigDialog() {
        // Eliminar la ventana emergente del cuerpo del documento
        const configDialog = document.getElementById('config-dialog');
        if (configDialog) {
            configDialog.remove();
        }
    }

    function updateColorBackground() {
        const colorPickerBackground = document.getElementById('color-picker-background');
        selectedBackgroundColor = colorPickerBackground.value;
    }

    function updateColorText() {
        const colorPickerText = document.getElementById('color-picker-text');
        selectedTextColor = colorPickerText.value;
    }

    function updateTranslationBoxColors() {
        if (translationBox) {
            translationBox.style.background = selectedBackgroundColor;
            translationBox.style.color = selectedTextColor;
        }
    }

    function saveColors() {
        GM_setValue('selectedBackgroundColor', selectedBackgroundColor);
        GM_setValue('selectedTextColor', selectedTextColor);
    }

    // Actualizar colores en el cuadro de traducción al inicio
    updateTranslationBoxColors();
})();