Last active
November 1, 2021 18:23
-
-
Save neoOpus/24316c9c5cde5189e3b936ae11ca07c9 to your computer and use it in GitHub Desktop.
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
// ==UserScript== | |
// @name Translator for Whatsapp | |
// @namespace http://tampermonkey.net/ | |
// @homepage https://greasyfork.org/scripts/28218-translator-for-whatsapp | |
// @version 2.4.3 | |
// @description Translator for Whatsapp web | |
// @author JedLiu modded as suggested by neoOpus | |
// @match https://web.whatsapp.com/* | |
// @run-at document-start | |
// @grant GM_setValue | |
// @grant GM_getValue | |
// @grant GM_xmlhttpRequest | |
// @connect translate.googleapis.com | |
// @connect www.googleapis.com | |
// @require https://cdnjs.cloudflare.com/ajax/libs/jquery/3.5.1/jquery.min.js | |
// ==/UserScript== | |
var API_KEY = 'INSERT YOUR GOOGlE TRANSLATE API KEY HERE'; | |
(function() { | |
'use strict'; | |
/************************************************************* | |
ATTENTION: | |
All supported languages | |
Remove // if you want to include this for translation | |
*************************************************************/ | |
var all_languages = [ | |
{id:'zh-CN', name:'Chinese Simplified'}, | |
//{id:'zh-TW', name:'Chinese Traditional'}, | |
//{id:'af', name:'Afrikaans'}, | |
//{id:'sq', name:'Albanian'}, | |
{id:'ar', name:'Arabic'}, | |
//{id:'hy', name:'Armenian'}, | |
//{id:'az', name:'Azerbaijani'}, | |
//{id:'eu', name:'Basque'}, | |
//{id:'be', name:'Belarusian'}, | |
//{id:'bn', name:'Bengali'}, | |
//{id:'bs', name:'Bosnian'}, | |
//{id:'bg', name:'Bulgarian'}, | |
//{id:'ca', name:'Catalan'}, | |
//{id:'ceb', name:'Cebuano'}, | |
//{id:'ny', name:'Chichewa'}, | |
//{id:'co', name:'Corsican'}, | |
//{id:'hr', name:'Croatian'}, | |
//{id:'cs', name:'Czech'}, | |
//{id:'da', name:'Danish'}, | |
{id:'nl', name:'Dutch'}, | |
{id:'en', name:'English'}, | |
//{id:'eo', name:'Esperanto'}, | |
//{id:'et', name:'Estonian'}, | |
//{id:'tl', name:'Filipino'}, | |
//{id:'fi', name:'Finnish'}, | |
{id:'fr', name:'French'}, | |
//{id:'fy', name:'Frisian'}, | |
//{id:'gl', name:'Galician'}, | |
//{id:'ka', name:'Georgian'}, | |
{id:'de', name:'German'}, | |
//{id:'el', name:'Greek'}, | |
//{id:'gu', name:'Gujarati'}, | |
//{id:'ht', name:'Haitian Creole'}, | |
//{id:'ha', name:'Hausa'}, | |
//{id:'haw', name:'Hawaiian'}, | |
//{id:'iw', name:'Hebrew'}, | |
//{id:'hi', name:'Hindi'}, | |
//{id:'hmn', name:'Hmong'}, | |
//{id:'hu', name:'Hungarian'}, | |
//{id:'is', name:'Icelandic'}, | |
//{id:'ig', name:'Igbo'}, | |
//{id:'id', name:'Indonesian'}, | |
//{id:'ga', name:'Irish'}, | |
{id:'it', name:'Italian'}, | |
//{id:'ja', name:'Japanese'}, | |
//{id:'jw', name:'Javanese'}, | |
//{id:'kn', name:'Kannada'}, | |
//{id:'kk', name:'Kazakh'}, | |
//{id:'km', name:'Khmer'}, | |
//{id:'ko', name:'Korean'}, | |
//{id:'ku', name:'Kurdish (Kurmanji)'}, | |
//{id:'ky', name:'Kyrgyz'}, | |
//{id:'lo', name:'Lao'}, | |
//{id:'la', name:'Latin'}, | |
//{id:'lv', name:'Latvian'}, | |
//{id:'lt', name:'Lithuanian'}, | |
//{id:'lb', name:'Luxembourgish'}, | |
//{id:'mk', name:'Macedonian'}, | |
//{id:'mg', name:'Malagasy'}, | |
//{id:'ms', name:'Malay'}, | |
//{id:'ml', name:'Malayalam'}, | |
//{id:'mt', name:'Maltese'}, | |
//{id:'mi', name:'Maori'}, | |
//{id:'mr', name:'Marathi'}, | |
//{id:'mn', name:'Mongolian'}, | |
//{id:'my', name:'Myanmar (Burmese)'}, | |
//{id:'ne', name:'Nepali'}, | |
//{id:'no', name:'Norwegian'}, | |
//{id:'ps', name:'Pashto'}, | |
//{id:'fa', name:'Persian'}, | |
//{id:'pl', name:'Polish'}, | |
//{id:'pt', name:'Portuguese'}, | |
//{id:'ma', name:'Punjabi'}, | |
//{id:'ro', name:'Romanian'}, | |
//{id:'ru', name:'Russian'}, | |
//{id:'sm', name:'Samoan'}, | |
//{id:'gd', name:'Scots Gaelic'}, | |
//{id:'sr', name:'Serbian'}, | |
//{id:'st', name:'Sesotho'}, | |
//{id:'sn', name:'Shona'}, | |
//{id:'sd', name:'Sindhi'}, | |
//{id:'si', name:'Sinhala'}, | |
//{id:'sk', name:'Slovak'}, | |
//{id:'sl', name:'Slovenian'}, | |
//{id:'so', name:'Somali'}, | |
{id:'es', name:'Spanish'}, | |
//{id:'su', name:'Sudanese'}, | |
//{id:'sw', name:'Swahili'}, | |
//{id:'sv', name:'Swedish'}, | |
//{id:'tg', name:'Tajik'}, | |
//{id:'ta', name:'Tamil'}, | |
//{id:'te', name:'Telugu'}, | |
//{id:'th', name:'Thai'}, | |
//{id:'tr', name:'Turkish'}, | |
//{id:'uk', name:'Ukrainian'}, | |
//{id:'ur', name:'Urdu'}, | |
//{id:'uz', name:'Uzbek'}, | |
//{id:'vi', name:'Vietnamese'}, | |
//{id:'cy', name:'Welsh'}, | |
//{id:'xh', name:'Xhosa'}, | |
//{id:'yi', name:'Yiddish'}, | |
//{id:'yo', name:'Yoruba'}, | |
//{id:'zu', name:'Zulu'} | |
]; | |
var SOURCE_LANGUAGE = 'en', | |
TRANSLATED_LANGUAGE = 'es'; | |
var $ = $ || window.$, | |
addListenerInterval = null, | |
translateInterval = null, | |
translateTimeout = null, | |
translate_enabled = true, | |
translate_ready = false, | |
translate_string = '', | |
custom_style = '.language_selected{background-color: #00bfa5;}', | |
image_uri = '', | |
custom_html = '<div class="block-compose tranlate-bottom"><div tabindex="-1" class="input-container" style="padding-top:0px;padding-bottom:0px;padding-left:0px;"><button title="Click for translation help!" class="trans_help_btn" style="float:left"><img alt="Translator" draggable="false" src="' + image_uri + '" style="width:30px;height:30px;padding-left:15px;padding-right:30px;padding-bottom:8px;"></button><div class="input" dir="auto" style="padding-top:6px;"></div></div></div>', | |
html_language1 = '<div class="menu-item" style="display:table"><button title="Click for translation help!" class="trans_help_btn"><img alt="Translator" draggable="false" src="data:'+ image_uri +'" style="width:32px;height:32px;"/></button></div>', | |
username = '', | |
is_debug = true, | |
lan_select = '', | |
help_url = 'https://greasyfork.org/zh-CN/scripts/28218-translator-for-whatsapp'; | |
//For menu html | |
for(var i=0;i<all_languages.length;i++){ | |
lan_select = lan_select + '<option value="'+ all_languages[i].id +'">' + all_languages[i].name +'</option>'; | |
} | |
var lan_select_1 = '<span style="padding-left:5px;padding-right:5px;color:green;font-size:10pt;">From:</span><select class="languageSelect1" style="padding-right:5px;width: 126px; text-align-last:center;">' + lan_select + '</select>'; | |
var lan_select_2 = '<span style="padding-left:20px;padding-right:5px;color:green;font-size:10pt;">To:</span><select class="languageSelect" style="padding-right:5px;width: 126px; text-align-last:center;border-bottom-width: 0px !important;"><option value="off">OFF</option>' + lan_select + '</select>'; | |
html_language1 = html_language1 + '<div style="display:table;"><div style="display:table-row">'+ lan_select_1 +'</div><div style="display:table-row">'+ lan_select_2 +'</div></div>'; | |
//Add style | |
var customStyleNode = document.createElement('style'); | |
customStyleNode.textContent = custom_style; | |
document.querySelector('head').appendChild(customStyleNode); | |
//Replace all function | |
function replaceAll(str, find, replace) { | |
return str.replace(new RegExp(find, 'g'), replace); | |
} | |
//Show debug | |
var debugMessage = function(mes){ | |
if(is_debug){ | |
console.info(mes); | |
} | |
}; | |
//Show error message | |
var showError = function(err){ | |
alert(err); | |
console.error(err); | |
}; | |
//Translate | |
//sl - source language | |
//dl - target language | |
//txt - content to be translated | |
//cb - callback after translation | |
var translate = function(sl,dl,txt,cb){ | |
//debugMessage('Source='+ txt+",sl="+sl+",dl="+dl+",txt="+txt); | |
GM_xmlhttpRequest({ | |
method: "GET", | |
url: "https://www.googleapis.com/language/translate/v2?key=" + API_KEY + "&client=gtx&source="+ sl +"&target=" + dl +"&format=text&q=" + encodeURI(txt), | |
//url: "https://www.googleapis.com/language/translate/v2?key=AIzaSyBMk2j1E4aRDn9Rx9DhNYnDyOHnkMR62tY&client=gtx&target=" + dl +"&format=text&q=" + encodeURI(txt), | |
onload: function(response) { | |
//replace the \n | |
var _r_text = replaceAll(response.responseText, '\n"', '"'); | |
var _r = JSON.parse(_r_text); | |
translate_string = _r.data.translations[0].translatedText; | |
debugMessage('Translation:'+translate_string); | |
cb.apply({text: translate_string}); | |
} | |
}); | |
}; | |
//Bind to get user input | |
var onInput = function(){ | |
var $_translate_input_1 = $('.tranlate-bottom').find('.input'); | |
$_translate_input_1.html('Typing...'); | |
translate_ready = false; | |
var _this = $(this); | |
delay(function(){ | |
var _input = $.trim(_this.text()); | |
if(_input){ | |
translate(SOURCE_LANGUAGE, TRANSLATED_LANGUAGE, _input, function(){ | |
$_translate_input_1.html(this.text); | |
translate_ready = true; | |
}); | |
}else{ | |
$_translate_input_1.html(''); | |
} | |
}, 1000); | |
}; | |
//Bind to send the translated content | |
//Updated: 2018-07-30 | |
var onEnterKeyPressed = function( event ) { | |
if (event.which == 13 && translate_enabled) { | |
debugMessage('Waiting translation'); | |
event.preventDefault(); | |
var _this = $(this); | |
translateInterval = setInterval(function(){ | |
if(translate_ready){ | |
debugMessage('Now sending message:'+translate_string); | |
sendTranslatedMessage(_this, translate_string); | |
debugMessage('Message sent'); | |
clearInterval(translateInterval); | |
} | |
}, 100); | |
} | |
}; | |
//Send translated message | |
//Updated: 2018-07-31 | |
var sendTranslatedMessage = function(inputTarget, message){ | |
translate_string = ''; | |
inputTarget.focus(); | |
document.execCommand("selectAll"); | |
document.execCommand("insertText", false, message); | |
if($('footer button:has(span):last span').data('icon') == 'send'){ | |
$('footer button:has(span):last').click(); | |
}else{ | |
showError('Not able to send the translated message'); | |
} | |
translate_ready = false; | |
} | |
//Add translation bindings | |
var addTranslateFunc = function(selectChange){ | |
if(!username){ | |
showError('Can not get the username'); | |
return; | |
} | |
if(selectChange){ | |
GM_setValue(username, $('.languageSelect').val()); | |
GM_setValue(username+'_o', $('.languageSelect1').val()); | |
} | |
TRANSLATED_LANGUAGE = GM_getValue(username) ? GM_getValue(username) : TRANSLATED_LANGUAGE; | |
SOURCE_LANGUAGE = GM_getValue(username+'_o') ? GM_getValue(username+'_o') : SOURCE_LANGUAGE; | |
//Menu | |
debugMessage('Set original language to: ' + SOURCE_LANGUAGE + ', translate to: '+ TRANSLATED_LANGUAGE); | |
$('.languageSelect').val(TRANSLATED_LANGUAGE); | |
$('.languageSelect1').val(SOURCE_LANGUAGE); | |
//Add translation input | |
var $_input_body = $('footer div.copyable-text.selectable-text'); | |
if(TRANSLATED_LANGUAGE !== 'off' && $('.tranlate-bottom').length === 0){ | |
$('footer').append($(custom_html)); | |
if($_input_body === null || $_input_body.length !== 1){ | |
showError('Error binding for Whatsapp translator plugin!'); | |
}else{ | |
$_input_body.on('input', onInput) | |
.on('keydown', onEnterKeyPressed); | |
} | |
//translate sent or received messages | |
$('.copyable-area').on('click', '.selectable-text', function(){ | |
if(TRANSLATED_LANGUAGE!='off'){ | |
var $_t_this = $(this); | |
translate(TRANSLATED_LANGUAGE, SOURCE_LANGUAGE, $(this).text(), function(){ | |
$_t_this.html(this.text); | |
}); | |
} | |
}); | |
//visit the help page | |
$('.trans_help_btn').on('click', function(){ | |
window.open(help_url,'_blank'); | |
}); | |
}else if(TRANSLATED_LANGUAGE === 'off' && $('.tranlate-bottom').length !== 0){ | |
//remove bindings | |
$('.tranlate-bottom').remove(); | |
$_input_body.off('input', onInput) | |
.off('keydown', onEnterKeyPressed); | |
} | |
}; | |
//Add listener when user activates a new chat | |
addListenerInterval = setInterval(function(){ | |
var $_div_chat = $('#pane-side'); | |
//console.log('div_chat_length', $_div_chat.length); | |
if($_div_chat.length){ | |
//console.log('found #pane-side'); | |
var contacts = document.querySelector('div[role="grid"]').children; | |
if(!contacts || contacts.length === 0){ | |
showError('Not able to get the contacts sidebar'); | |
return; | |
} | |
var c_name = contacts[0].className; | |
console.log('One of the chat selector', contacts[0]); | |
//更新会经常导致这个地方需要修改 | |
$('#pane-side').on('click','div.'+c_name, function(){ | |
//Get the username | |
//username = escape($(this).find('.chat-title').text()); | |
console.info($(this)); | |
var _tusername = ''; | |
$(this).find('span').each(function(i,x){ | |
if(x.hasAttribute('title')) { | |
//console.info(x.title); | |
_tusername = x.title; | |
return false; | |
} | |
}); | |
if(_tusername !== ''){username = escape(_tusername);} | |
else{showError('Not able to get the user name');} | |
debugMessage('Chat menu clicked'); | |
//Return if the translation input is added | |
if($('.languageSelect').length>0){return;} | |
var $header = $('#main header div:first').next(); | |
if($header.length != 1){showError('Not able to insert translate menu');} | |
$header.after($(html_language1)); | |
//Bind lanaguage select change event | |
$('.languageSelect').on('change', function(){ | |
addTranslateFunc(true); | |
}); | |
$('.languageSelect1').on('change', function(){ | |
addTranslateFunc(true); | |
}); | |
//Apply the translate function | |
addTranslateFunc(); | |
}); | |
clearInterval(addListenerInterval); | |
} | |
}, 1000); | |
//Delay function | |
var delay = (function(){ | |
var timer = 0; | |
return function(callback, ms){ | |
clearTimeout (timer); | |
timer = setTimeout(callback, ms); | |
}; | |
})(); | |
})(); |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment