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