// ==UserScript== // @name Translator for Whatsapp // @namespace http://tampermonkey.net/ // @homepage https://greasyfork.org/zh-CN/scripts/28218-translator-for-whatsapp // @version 2.6.5 // @description Translator for Whatsapp web // @author JedLiu // @match https://web.whatsapp.com/* // @run-at document-start // @grant GM_setValue // @grant GM_getValue // @grant GM_xmlhttpRequest // @connect translate.googleapis.com // @require https://cdnjs.cloudflare.com/ajax/libs/jquery/3.5.1/jquery.min.js // @downloadURL https://update.greasyfork.icu/scripts/28218/Translator%20for%20Whatsapp.user.js // @updateURL https://update.greasyfork.icu/scripts/28218/Translator%20for%20Whatsapp.meta.js // ==/UserScript== (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 = '', send_translation_image = '', custom_html = '
' +'
' + '
' + '
', html_language1 = '', username = '', is_debug = true, lan_select = '', help_url = 'https://greasyfork.org/zh-CN/scripts/28218-translator-for-whatsapp', readyTranslation = ''; //For menu html for(var i=0;i' + all_languages[i].name +''; } var lan_select_1 = 'From:'; var lan_select_2 = 'To:'; html_language1 = html_language1 + '
'+ lan_select_1 +'
'+ lan_select_2 +'
'; //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); GM_xmlhttpRequest({ method: "GET", url: "https://translate.googleapis.com/translate_a/single?client=gtx&sl="+ sl + "&tl=" + dl +"&dt=t&q=" + encodeURI(txt), onload: function(response) { var _r = JSON.parse(response.responseText);; translate_string = ''; for(var i=0; i<_r[0].length;i++){ translate_string += _r[0][i][0]; } debugMessage('Translation:'+translate_string); cb.apply({text: translate_string}); } }); }; //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 span .selectable-text'); if(TRANSLATED_LANGUAGE !== 'off' && $('.tranlate-bottom').length === 0){ console.log($_input_body); $('footer').append($(custom_html)); // cant bind whatsapp translator plugin error if($_input_body === null || $_input_body.length !== 1){ showError('Error binding for Whatsapp translator plugin!'); } //translate sent or received messages $('[role="application"]').on('click', '.selectable-text', function(){ console.log('translate sent or received messages'); if(TRANSLATED_LANGUAGE!='off'){ var $_t_this = $(this); console.log('translate to language:'+ SOURCE_LANGUAGE); translate('auto', SOURCE_LANGUAGE, $(this).text(), function(){ $_t_this.html(this.text); }); } }); const sendMessage = function() { setTimeout(function(){ $('footer p.selectable-text').parent().focus(); setTimeout(function(){ document.execCommand("selectAll"); setTimeout(function(){ let content = $('#translatedMessage').html(); document.execCommand("insertText", false, content); setTimeout(function(){ $('[data-icon="send"]').click(); setTimeout(function(){ $('#translatedMessage').html(''); $('#originalTextInput').val('').focus(); },200); },200); }, 200); }, 200); }, 200); } $('#originalTextInput').keyup(function(event){ if (event.which == 13) { debugMessage('Waiting translation'); retry( function(){console.log('translate_ready',translate_ready); return translate_ready;}, sendMessage, function(){showError('Translation failed. Please retry!');} ); return false; }else{ var $_translate_input_1 = $('#translatedMessage'); $_translate_input_1.html('Typing...'); translate_ready = false; delay(function(){ var _input = $.trim(document.getElementById('originalTextInput').value); if(_input.length>2 && _input != readyTranslation){ translate(SOURCE_LANGUAGE, TRANSLATED_LANGUAGE, _input, function(){ $_translate_input_1.html(this.text); translate_ready = true; }); }else{ $_translate_input_1.html(''); readyTranslation = ''; } }, 1000); } }); //focus the input delay(function(){$('#originalTextInput').focus();}, 500); //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(); } }; //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; } const selector = contacts[0].className.split(' ').join('.'); console.log('The selector: div.'+selector); //更新会经常导致这个地方需要修改 $('#pane-side').on('click','div.'+selector, 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); }; })(); //Retry function function retry(checkCallback, successCallback, failCallback, delay=500, tries=30) { if(tries && checkCallback() !== true){ setTimeout(retry.bind(this, checkCallback, successCallback, failCallback, delay, tries-1), delay); }else if(tries<=0){ failCallback(); }else{ successCallback(); } } })();