Translator for Whatsapp

Translator for Whatsapp web

您需要先安裝使用者腳本管理器擴展,如 TampermonkeyGreasemonkeyViolentmonkey 之後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyUserscripts 後才能安裝該腳本。

你需要先安裝一款使用者腳本管理器擴展,比如 Tampermonkey,才能安裝此腳本

您需要先安裝使用者腳本管理器擴充功能後才能安裝該腳本。

(我已經安裝了使用者腳本管理器,讓我安裝!)

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

(我已經安裝了使用者樣式管理器,讓我安裝!)

// ==UserScript==
// @name         Translator for Whatsapp
// @namespace    http://tampermonkey.net/
// @homepage     https://greasyfork.org/zh-CN/scripts/28218-translator-for-whatsapp
// @version      2.6.6
// @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
// ==/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 = '<div class="tranlate-bottom">'
+'<div style="margin-top:10px;margin-right:30px;margin-left:20px;"><input type="text" id="originalTextInput" placeholder="Type your message and press Enter to send the translated message" style="width:100%;border:0px;height:30px;border-radius:8px;padding:6px;" autocomplete="off" selectable/></div>'
+ '<div style="margin:20px;min-height:20px" id="translatedMessage"></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',
    readyTranslation = '';


//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);
    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
        console.log('$(#main)', $('#main'));
        $('#main').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="wds-ic-send-filled"]').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();
    }
}
})();