离线场景下任意文档的在线预览及原样格式翻译,不依赖其他厂商接口非侵入式一行js代码实现网站的翻译及国际化,可配置使用多种翻译语言

本文主要是介绍离线场景下任意文档的在线预览及原样格式翻译,不依赖其他厂商接口非侵入式一行js代码实现网站的翻译及国际化,可配置使用多种翻译语言,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

离线场景下任意文档的在线预览及原样格式翻译,不依赖其他厂商接口非侵入式一行js代码实现网站的翻译及国际化,可配置使用多种翻译语言。

在这里插入图片描述

要实现翻译需要解决以下3个主要问题:
1)from:内容本身的语言类型是什么?
2)to:需要翻译为目标语言是什么?
3)text:需要翻译的文本内容是什么?

转化为:

1)首先,如何识别文档内容的语言?一篇文章中有多种语言混合的如何识别?
2)其次,用户使用的是什么语言?如何获取目标语言?
3)在文档或者网页中,所有内容都是带有格式的,如何翻译之后进行还原保证样式不丢失?

在这里插入图片描述


在网上查了很多资料,也下载了很多 收费 的资料,结果不尽人意。

获取用户的语言:

1)通过浏览器的默认语言判断用户使用的语言:

<script>// 获取浏览器默认语言const getBrowserLang = function () {let browserLang = navigator.language ? navigator.language : navigator.browserLanguage;let defaultBrowserLang = "";if (browserLang.toLowerCase() === "us" ||browserLang.toLowerCase() === "en" ||browserLang.toLowerCase() === "en_us") {defaultBrowserLang = "en_US";} else {defaultBrowserLang = "zh_CN";}return defaultBrowserLang;};console.log(getBrowserLang());</script>

或者:

<script>var type = navigator.appName; //BOM对象获取浏览器名称if (type == "Netscape") {var lang = navigator.language.toLowerCase(); //获取浏览器配置语言,支持非IE浏览器} else {var lang = navigator.browserLanguage.toLowerCase(); //获取浏览器配置语言,支持IE5+}console.log(lang);var lang = lang.substr(0, 5); //获取浏览器配置语言前4位console.log(lang);</script>

如何通过js在html中动态导入其他的js库:

<script>var head = document.getElementsByTagName("head")[0];var script = document.createElement("script");script.type = "text/javascript";script.src = "http://localhost:8060/static/translate.min.js";script.onload = script.onreadystatechange = function () {translate.storage.set("to", "");//设置使用v2.x 版本translate.setUseVersion2();//SELECT 修改 onchange 事件translate.selectLanguageTag.selectOnChange = function (event) {//判断是否是第一次翻译,如果是,那就不用刷新页面了。 true则是需要刷新,不是第一次翻译var isReload = translate.to != null && translate.to.length > 0;if (isReload) {//如果要刷新页面的话,弹出友好提示alert("您好,快速体验暂时只能切换其中一种语言进行体验,只是提供效果展示,您可参考接入文档来接入您的项目中进行完整体验及使用。",);} else {var language = event.target.value;console.log(language);// translate.changeLanguage(language);}};};</script>

如何获取文档或者网页的需要翻译的内容?找了一个网页翻译助手实现的js插件代码进行参考,完整代码如下:

// ==UserScript==
// @name         网页翻译助手
// @version      1.3.3
// @namespace    https://github.com/zyufstudio/TM/tree/master/webTranslate
// @description  支持划词翻译,输入文本翻译,谷歌整页翻译。可以自行选择谷歌翻译,有道字典翻译和百度翻译。
// @icon         
// @author       Johnny Li
// @license      MIT
// @match        *://*/*
// @grant        GM_info
// @grant        GM_xmlhttpRequest
// @grant        GM_addStyle
// @grant        GM_registerMenuCommand
// @grant        GM_setValue
// @grant        GM_getValue
// @connect      cdn.jsdelivr.net
// @connect      cdn.bootcss.com
// @connect      translate.google.com.hk
// @connect      fanyi.youdao.com
// @connect      dict.youdao.com
// @connect      fanyi.baidu.com
// @connect      shared.ydstatic.com
// @require      https://cdn.jsdelivr.net/npm/jquery@2.2.3/dist/jquery.min.js
// @require      https://cdn.jsdelivr.net/npm/jquery.md5@1.0.2/index.min.js
// @require      https://cdn.jsdelivr.net/gh/zyufstudio/jQuery@3a09ff54b33fc2ae489b5083174698b3fa83f4a7/jPopBox/dist/jPopBox.min.js
// ==/UserScript==//文件使用Rollup+Gulp编译而成,如需查看源码请转到GitHub项目。(function () {'use strict';/*** 字符串模板格式化* @param {string} formatStr - 字符串模板* @returns {string} 格式化后的字符串* @example* StringFormat("ab{0}c{1}ed",1,"q")  output "ab1cqed"*/function StringFormat(formatStr) {var args = arguments;return formatStr.replace(/\{(\d+)\}/g, function (m, i) {i = parseInt(i);return args[i + 1];});}/*** 日期格式化* @param {Date} date - 日期* @param {string} formatStr - 格式化模板* @returns {string} 格式化日期后的字符串* @example* DateFormat(new Date(),"yyyy-MM-dd")  output "2020-03-23"* @example* DateFormat(new Date(),"yyyy/MM/dd hh:mm:ss")  output "2020/03/23 10:30:05"*/function DateFormat(date, formatStr) {var o = {"M+": date.getMonth() + 1, //月份"d+": date.getDate(), //日"h+": date.getHours(), //小时"m+": date.getMinutes(), //分"s+": date.getSeconds(), //秒"q+": Math.floor((date.getMonth() + 3) / 3), //季度"S": date.getMilliseconds() //毫秒};if (/(y+)/.test(formatStr)) {formatStr = formatStr.replace(RegExp.$1, (date.getFullYear() + "").substr(4 - RegExp.$1.length));}for (var k in o) {if (new RegExp("(" + k + ")").test(formatStr)) {formatStr = formatStr.replace(RegExp.$1, (RegExp.$1.length == 1) ? (o[k]) : (("00" + o[k]).substr(("" + o[k]).length)));}}return formatStr;}/*** 生成Guid* @param {boolean} hasLine - guid字符串是否包含短横线* @returns {string} guid* @example * Guid(false)  output "b72f78a6cb88362c0784cb82afae450b"* @example* Guid(true) output "67b25d43-4cfa-3edb-40d7-89961ce7f388"*/function Guid(hasLine){var guid="";function S4() {return (((1+Math.random())*0x10000)|0).toString(16).substring(1);}if(hasLine){guid=(S4()+S4()+"-"+S4()+"-"+S4()+"-"+S4()+"-"+S4()+S4()+S4());}else {guid=(S4()+S4()+S4()+S4()+S4()+S4()+S4()+S4());}return guid;}/*** 清除dom元素默认事件* @param {object} e - dom元素*/function ClearBubble(e) {if (e.stopPropagation) {e.stopPropagation();} else {e.cancelBubble = true;}if (e.preventDefault) {e.preventDefault();} else {e.returnValue = false;}}function ObjectToQueryString(object){var querystring=Object.keys(object).map(function(key) { return encodeURIComponent(key) + '=' + encodeURIComponent(object[key]) }).join('&');return querystring;}/*** 配置参数*/var options={//默认翻译引擎defaulttransengine:"yd"};/*** 获取配置参数*/function GetSettingOptions(){var optionsJson=GM_getValue("webtranslate-options")||"";if(optionsJson!=""){var optionsData=JSON.parse(optionsJson);for (var key in options) {if (options.hasOwnProperty(key) && optionsData.hasOwnProperty(key)) {options[key]= optionsData[key];   }}}return options;}/*** 设置配置参数*/function SetSettingOptions(){var optionsJson=JSON.stringify(options);GM_setValue("webtranslate-options", optionsJson);}//谷歌翻译var googleTrans = {code: "ge",codeText: "谷歌",defaultOrigLang: "auto", //默认源语言defaultTargetLang: "zh-CN", //默认目标语言langList: {"auto": "自动检测","zh-CN": "中文简体","zh-TW": "中文繁体","en": "英文","ja": "日文","ko": "韩文","fr": "法文","es": "西班牙文","pt": "葡萄牙文","it": "意大利文","ru": "俄文","vi": "越南文","de": "德文","ar": "阿拉伯文","id": "印尼文"},Execute: function (h_onloadfn) {GM_xmlhttpRequest({method: "POST",url: "https://translate.google.com.hk/_/TranslateWebserverUi/data/batchexecute",headers: {"Referer": `https://translate.google.com.hk/`,"Cache-Control": "max-age=0","Content-Type": "application/x-www-form-urlencoded;charset=utf-8",},data: "f.req=" + encodeURIComponent(JSON.stringify([[["MkEWBc", JSON.stringify([[Trans.transText, Trans.transOrigLang, Trans.transTargetLang, true],[null]]), null, "generic"]]])),onload: function (r) {setTimeout(function () {var resData=r.responseText;var transData=JSON.parse(JSON.parse(resData.match(/\[{2}.*\]{2}/g)[0])[0][2]);var transList=transData[1][0][0][5];var transTexts=[];for (let index = 0; index < transList.length; index++) {var transItem = transList[index];transTexts.push(transItem[0]);}Trans.transResult.trans = transTexts;Trans.transResult.orig = transData[1][4][0].split("\n");Trans.transResult.origLang = transData[2];h_onloadfn();}, 300);},onerror: function (e) {console.error(e);}});},};//获取signfunction getSign() {GM_xmlhttpRequest({method: "GET",url: "http://fanyi.youdao.com/",timeout: 5000,onload: function (ydRes) {var fanyijsUrlMatch = /<script\s+type="text\/javascript"\s+src="([http|https]*?:\/\/shared.ydstatic.com\/fanyi\/newweb\/v[\d.]+\/scripts\/newweb\/fanyi.min.js)"><\/script>/g.exec(ydRes.responseText);if (!fanyijsUrlMatch) {console.log("获取fanyi.min.js失败!!!");} else {var fanyijsUrl = fanyijsUrlMatch[1];if (typeof fanyijsUrl !== 'undefined') {GM_xmlhttpRequest({method: "GET",url: fanyijsUrl,timeout: 5000,onload: function (r) {var signMatch = /sign:[a-z]{1}\.md5\("fanyideskweb"\+[a-z]{1}\+[a-z]{1}\+"(.*)"\)}};/g.exec(r.responseText);if (!signMatch) {console.log("获取sign失败!!!");} else {var newSign = signMatch[1];if (typeof newSign !== 'undefined') {youdaoTrans.sign = newSign;}}},onerror: function (e) {console.error(e);}});}}},onerror: function (e) {console.error(e);}});}/*** 获取有道翻译音标* @param {String} transText * @param {Function} callback */function getYDSymbol(transText, callback) {var url = StringFormat("http://dict.youdao.com/fsearch?client=fanyideskweb&keyfrom=fanyi.web&q={0}&doctype=xml&xmlVersion=3.2&dogVersion=1.0&appVer=3.1.17.4208", encodeURIComponent(transText));GM_xmlhttpRequest({method: "GET",url: url,timeout: 5000,onload: function (ydRes) {var xmlnode=ydRes.responseXML;var symbol = {uk:"",us: ""};var root = xmlnode.getElementsByTagName("yodaodict")[0];if ("" + root.getElementsByTagName("uk-phonetic-symbol")[0] != "undefined" && "" + root.getElementsByTagName("uk-phonetic-symbol")[0].childNodes[0] != "undefined") {symbol.uk = root.getElementsByTagName("uk-phonetic-symbol")[0].childNodes[0].nodeValue;}if ("" + root.getElementsByTagName("us-phonetic-symbol")[0] != "undefined" && "" + root.getElementsByTagName("us-phonetic-symbol")[0].childNodes[0] != "undefined") {symbol.us = root.getElementsByTagName("us-phonetic-symbol")[0].childNodes[0].nodeValue;}callback(symbol);},onerror: function (e) {console.error(e);}});}//有道翻译var youdaoTrans = {code: "yd",codeText: "有道",sign: "",defaultOrigLang: "AUTO", //默认源语言defaultTargetLang: "ZH-CHS", //默认目标语言langList: {"AUTO": "自动检测","zh-CHS": "中文","en": "英文","ja": "日文","ko": "韩文","fr": "法文","es": "西班牙文","pt": "葡萄牙文","it": "意大利文","ru": "俄文","vi": "越南文","de": "德文","ar": "阿拉伯文","id": "印尼文"},Execute: function (h_onloadfn) {var h_url = "",h_headers = {},h_data = "";var youdaoTransApi = "http://fanyi.youdao.com/translate_o?client=fanyideskweb&keyfrom=fanyi.web&smartresult=dict&version=2.1&doctype=json";var userAgent=$.md5(navigator.userAgent);var currentTs="" + (new Date).getTime();var salt=currentTs + parseInt(10 * Math.random(), 10);var sign = this.sign != "" ? this.sign : "]BjuETDhU)zqSxf-=B#7m";var signStr = $.md5("fanyideskweb" + Trans.transText + salt + sign);h_url = youdaoTransApi;h_headers = {"Content-Type": "application/x-www-form-urlencoded","Referer": "http://fanyi.youdao.com/"};h_data = StringFormat("from={0}&to={1}&salt={2}&sign={3}&i={4}&lts={5}&bv={6}", Trans.transOrigLang, Trans.transTargetLang, salt, signStr, encodeURIComponent(Trans.transText),currentTs,userAgent);GM_xmlhttpRequest({method: "POST",url: h_url,headers: h_headers,data: h_data,onload: function (r) {setTimeout(function () {var data = JSON.parse(r.responseText);var trans = [],origs = [],src = "";if (data.errorCode == 0) {for (var j = 0; j < data.translateResult.length; j++) {var ydTransCont = data.translateResult[j];var ydtgt = "";var ydsrc = "";for (var k = 0; k < ydTransCont.length; k++) {var ydcont = ydTransCont[k];ydtgt += ydcont.tgt;ydsrc += ydcont.src;}trans.push(ydtgt);origs.push(ydsrc);}src = data.type;Trans.transResult.trans = trans;Trans.transResult.orig = origs;Trans.transResult.origLang = src.split("2")[0];var smartResult = data.smartResult;if (smartResult && smartResult.entries.length > 0) {getYDSymbol(Trans.transText, function (symbol) {Trans.transResult.symbols.en = symbol.uk;Trans.transResult.symbols.am = symbol.us;h_onloadfn();});}else {h_onloadfn();}}}, 300);},onerror: function (e) {console.error(e);}});},init: function () {getSign();}};function a(r) {if (Array.isArray(r)) {for (var o = 0, t = Array(r.length); o < r.length; o++)t[o] = r[o];return t}return Array.from(r)}function n(r, o) {for (var t = 0; t < o.length - 2; t += 3) {var a = o.charAt(t + 2);a = a >= "a" ? a.charCodeAt(0) - 87 : Number(a),a = "+" === o.charAt(t + 1) ? r >>> a : r << a,r = "+" === o.charAt(t) ? r + a & 4294967295 : r ^ a;}return r}function e(r,gtk) {var i = null;var o = r.match(/[\uD800-\uDBFF][\uDC00-\uDFFF]/g);if (null === o) {var t = r.length;t > 30 && (r = "" + r.substr(0, 10) + r.substr(Math.floor(t / 2) - 5, 10) + r.substr(-10, 10));} else {for (var e = r.split(/[\uD800-\uDBFF][\uDC00-\uDFFF]/), C = 0, h = e.length, f = []; h > C; C++)"" !== e[C] && f.push.apply(f, a(e[C].split(""))),C !== h - 1 && f.push(o[C]);var g = f.length;g > 30 && (r = f.slice(0, 10).join("") + f.slice(Math.floor(g / 2) - 5, Math.floor(g / 2) + 5).join("") + f.slice(-10).join(""));}var u = void 0;u = null !== i ? i : (i = gtk || "") || "";for (var d = u.split("."), m = Number(d[0]) || 0, s = Number(d[1]) || 0, S = [], c = 0, v = 0; v < r.length; v++) {var A = r.charCodeAt(v);128 > A ? S[c++] = A : (2048 > A ? S[c++] = A >> 6 | 192 : (55296 === (64512 & A) && v + 1 < r.length && 56320 === (64512 & r.charCodeAt(v + 1)) ? (A = 65536 + ((1023 & A) << 10) + (1023 & r.charCodeAt(++v)),S[c++] = A >> 18 | 240,S[c++] = A >> 12 & 63 | 128) : S[c++] = A >> 12 | 224,S[c++] = A >> 6 & 63 | 128),S[c++] = 63 & A | 128);}for (var p = m, F = "" + String.fromCharCode(43) + String.fromCharCode(45) + String.fromCharCode(97) + ("" + String.fromCharCode(94) + String.fromCharCode(43) + String.fromCharCode(54)), D = "" + String.fromCharCode(43) + String.fromCharCode(45) + String.fromCharCode(51) + ("" + String.fromCharCode(94) + String.fromCharCode(43) + String.fromCharCode(98)) + ("" + String.fromCharCode(43) + String.fromCharCode(45) + String.fromCharCode(102)), b = 0; b < S.length; b++)p += S[b],p = n(p, F);return p = n(p, D),p ^= s,0 > p && (p = (2147483647 & p) + 2147483648),p %= 1e6,p.toString() + "." + (p ^ m)}/*** @param  {string} word* @param  {string} gtk* @return {string}*/var calcSign =function(word,gtk){return e(word,gtk);};//获取gtk和tokenfunction GetToken(){GM_xmlhttpRequest({method: "GET",url: "https://fanyi.baidu.com/",timeout:5000,onload: function (r) {var gtkMatch = /window\.gtk = '(.*?)'/.exec(r.responseText);var commonTokenMatch = /token: '(.*?)',/.exec(r.responseText);if (!gtkMatch) {console.log("获取gtk失败!!!");}if (!commonTokenMatch) {console.log("获取token失败!!!");}var newGtk = gtkMatch[1];var newCommonToken = commonTokenMatch[1];if (typeof newGtk !== 'undefined') {baiduTrans.gtk=newGtk;}if (typeof newCommonToken !== 'undefined') {baiduTrans.token=newCommonToken;}},onerror: function (e) {console.error(e);}});}//百度翻译var baiduTrans = {code:"bd",codeText:"百度",gtk:"",token:"",defaultOrigLang:"auto",         //默认源语言defaultTargetLang:"zh",         //默认目标语言langList: {"auto": "自动检测","zh": "中文","cht": "繁体中文","en": "英语","jp": "日语","kor": "韩语","fra": "法语","spa": "西班牙语","pt": "葡萄牙语","it": "意大利语","ru": "俄语","vie": "越南语","de": "德语","ara": "阿拉伯语"},Execute: function (h_onloadfn) {if(Trans.transOrigLang=="auto")this.AutoTrans(h_onloadfn);elsethis.ExecTrans(h_onloadfn);},AutoTrans:function(h_onloadfn){var self=this;var datas={query:Trans.transText};GM_xmlhttpRequest({method: "POST",headers:{"referer": 'https://fanyi.baidu.com',"Content-Type": 'application/x-www-form-urlencoded; charset=UTF-8',},url: "https://fanyi.baidu.com/langdetect",data: ObjectToQueryString(datas),onload: function (r) {var data = JSON.parse(r.responseText);if(data.error===0){Trans.transOrigLang=data.lan;self.ExecTrans(h_onloadfn);}},onerror: function (e) {console.error(e);}});},ExecTrans:function(h_onloadfn){var tempSign=calcSign(Trans.transText,this.gtk);var datas={from:Trans.transOrigLang,to:Trans.transTargetLang,query:Trans.transText,transtype:"translang",simple_means_flag:3,sign:tempSign,token:this.token};GM_xmlhttpRequest({method: "POST",headers:{"referer": 'https://fanyi.baidu.com',"Content-Type": 'application/x-www-form-urlencoded; charset=UTF-8',//"User-Agent": window.navigator.userAgent,},url: "https://fanyi.baidu.com/v2transapi",data: ObjectToQueryString(datas),onload: function (r) {setTimeout(function () {var result= JSON.parse(r.responseText);var trans_result=result.trans_result;var dict_result=result.dict_result || null;var transDatas = trans_result.data;var trans = [],origs = [],src = "";for (var i = 0; i < transDatas.length; i++) {var getransCont = transDatas[i];trans.push(getransCont.dst);origs.push(getransCont.src);}src = trans_result.from;Trans.transResult.trans = trans;Trans.transResult.orig = origs;Trans.transResult.origLang = src;if(dict_result){var symbols=dict_result.simple_means.symbols;Trans.transResult.symbols.en=symbols[0].ph_en || "";Trans.transResult.symbols.am=symbols[0].ph_am || "";}h_onloadfn();}, 300);},onerror: function (e) {console.error(e);}});},init:function(){GetToken();}};var Trans={transEngineList:{},         //翻译引擎实例列表transEngine:"",             //当前翻译引擎。ge(谷歌)/yd(有道)transEngineObj:{},          //当前翻译引擎实例transTargetLang:"",         //目标语言。transOrigLang:"",           //源语言transType:"word",           //翻译类型。word(划词翻译)/text(输入文本翻译)/page(整页翻译)transText:"",               //被翻译内容transResult:{               //当前翻译内容//译文trans:[],//原文orig:[],//原文语言origLang:"",//音标symbols:{//英标en:"",//美标am:"",}},Execute:function(h_onloadfn){resetTransResult(this);this.transEngineObj.Execute(h_onloadfn);},GetLangList:function(){var langList={};langList=this.transEngineObj.langList;return langList;},Update:function(){resetTransResult(this);this.transEngineObj=this.transEngineList[this.transEngine];this.transTargetLang=this.transEngineObj.defaultTargetLang;this.transOrigLang=this.transEngineObj.defaultOrigLang;},Clear:function(){this.transEngine="";                //当前翻译引擎。ge(谷歌)/yd(有道)this.transTargetLang="";            //目标语言。this.transOrigLang="";             //源语言this.transText="";                   //被翻译内容this.transResult.trans=[];this.transResult.orig=[];this.transResult.origLang="";},//注册翻译引擎接口并执行翻译引擎的初始化接口RegisterEngine:function(){/*** 翻译引擎必须提供以下接口code:"",                    //代号codeText:"",                //代号描述defaultOrigLang:"",         //默认源语言defaultTargetLang:"",       //默认目标语言langList: {},               //支持翻译语言列表Execute: function (h_onloadfn) {},     //执行翻译init:function(){},          //可选,初始化接口,在脚本创建时立即执行*/var transEngineListObj={};transEngineListObj[googleTrans.code]=googleTrans;transEngineListObj[youdaoTrans.code]=youdaoTrans;transEngineListObj[baiduTrans.code]=baiduTrans;this.transEngineList=transEngineListObj;for (var key in this.transEngineList) {if (this.transEngineList.hasOwnProperty(key) && this.transEngineList[key].hasOwnProperty("init")) {this.transEngineList[key].init();}}}};function resetTransResult(that){that.transResult.trans=[];that.transResult.orig=[];that.transResult.origLang="";that.transResult.symbols.en="";that.transResult.symbols.am="";}//面板var Panel={popBoxEl:{},randomCode:"",Create:function(title,placement,isShowArrow,content,shownFn){var self=this;$(self.popBoxEl).jPopBox({title: title,className: 'JPopBox-tip-white',placement: placement,trigger: 'none',isTipHover: true,isShowArrow: isShowArrow,content: function(){return StringFormat('<div id="panelBody{0}">{1}</div>',self.randomCode,content);}});$(self.popBoxEl).on("shown.jPopBox",function(){var $panel=$("div.JPopBox-tip-white");typeof shownFn === 'function' && shownFn($panel);});$(self.popBoxEl).jPopBox('show');},Update:function(Fn){var $panel=$("div.JPopBox-tip-white");Fn($panel);    },Destroy:function(){//$(this.popBoxEl).jPopBox("hideDelayed");$(this.popBoxEl).jPopBox("destroy");},CreateStyle:function(){var s="";s+=StringFormat("#panelBody{0}>div input,#panelBody{0}>div select{padding: 3px; margin: 0; background: #fff; font-size: 14px; border: 1px solid #a9a9a9; color:black;width: auto;min-height: auto; }",this.randomCode);s+=StringFormat("#panelBody{0}>div:first-child{padding-bottom: 5px;height:30px}",this.randomCode);s+=StringFormat("#panelBody{0}>div:last-child hr{border: 1px inset #eeeeee;background: none;height: 0px;margin: 0px;}",this.randomCode);return s;}};//文本翻译面板var TextTransPanel={Create:function(popBoxEl,randomCode){var self=this;var html=this.GetHtml();var transEngineOptionsHtml="";//翻译引擎for (var k in Trans.transEngineList) {if (Trans.transEngineList.hasOwnProperty(k)) {var v = Trans.transEngineList[k].codeText;var selectOption="";if(Trans.transEngine==k){selectOption='selected="selected"';}transEngineOptionsHtml+=StringFormat('<option value="{0}" {2}>{1}</option>',k,v,selectOption);}}var TextTransPanelHtml=StringFormat('<div style="padding-bottom: 5px;">'+'翻译引擎:<select>{2}</select>&nbsp;&nbsp;&nbsp;&nbsp;'+'翻译语言:<select>{4}</select> &#x21E8; '+'<select>{3}</select> '+'<button style="width:46px; height:26px; cursor: pointer;overflow: visible;color: inherit;margin: 0;padding: 1px 7px;background-color: #dddddd;border: 2px outset #dddddd;text-align: center;display: inline-block;font-size: 14px; font-weight: 400; ">翻译</button></div>'+'<div style="word-wrap:break-word">'+'<div style="padding-bottom: 5px;"><textarea placeholder="请输入你要翻译的文字" style="word-wrap: break-word;word-break: keep-all;overflow-y: auto;width:450px;height:85px;padding: 3px;line-height: 18px;font-size: 14px;font-family: arial,simsun;border: 1px solid #999;border-color: #999 #d8d8d8 #d8d8d8 #999;outline: 0;resize: none;">{5}</textarea></div><hr/>'+'<div style="padding-top: 5px;">{6}</div>'+'</div>',randomCode,"",transEngineOptionsHtml,html.targetLangListHtml,html.origLangListHtml,"","");Panel.popBoxEl=popBoxEl;Panel.randomCode=randomCode;Panel.Create("文本翻译","auto bottom",false,TextTransPanelHtml,function($panel){$panel.css({position: "fixed",top:"20px"});//翻译引擎$panel.find(StringFormat("#panelBody{0} div:eq(0) select:eq(0)",randomCode)).change(function(e){Trans.transEngine=$(this).find("option:selected").val();Trans.Update();Panel.Update(function($panel){var html=self.GetHtml();//翻译内容$panel.find(StringFormat("#panelBody{0} div:eq(1) div:eq(1)",randomCode)).html("");$panel.find(StringFormat("#panelBody{0} div:eq(0) select:eq(1)",randomCode)).html(html.origLangListHtml);$panel.find(StringFormat("#panelBody{0} div:eq(0) select:eq(2)",randomCode)).html(html.targetLangListHtml);});});//源语言$panel.find(StringFormat("#panelBody{0} div:eq(0) select:eq(1)",randomCode)).change(function(e){Trans.transOrigLang=$(this).find("option:selected").val();Panel.Update(function($panel){var html=self.GetHtml();$panel.find(StringFormat("#panelBody{0} div:eq(0) select:eq(2)",randomCode)).html(html.targetLangListHtml);});});//目标语言$panel.find(StringFormat("#panelBody{0} div:eq(0) select:eq(2)",randomCode)).change(function(e){Trans.transTargetLang=$(this).find("option:selected").val();});//翻译$panel.find(StringFormat("#panelBody{0} div:eq(0)  button:eq(0)",randomCode)).click(function(e){var refTransText=$.trim($panel.find(StringFormat("#panelBody{0} div:eq(1) div:eq(0) textarea:eq(0)",randomCode)).val());if(refTransText==""){alert("请输入翻译文字!");return;}Trans.transText=refTransText;Trans.Execute(function(){Panel.Update(function($panel){var html=self.GetHtml();//源语言$panel.find(StringFormat("#panelBody{0} div:eq(0) select:eq(1)",randomCode)).html(html.origLangListHtml);//翻译内容$panel.find(StringFormat("#panelBody{0} div:eq(1) div:eq(1)",randomCode)).html(html.transHtml);});});});});},GetHtml:function(){var origLangListHtml=[];var targetLangListHtml=[];var returnHtml={};var transHtml=[];var langList=Trans.GetLangList();var origLang=Trans.transResult.origLang;if(Trans.transResult.trans.length>0 && Trans.transResult.orig.length>0){transHtml.push('<span>');for (var i = 0; i < Trans.transResult.trans.length; i++) {var transtxt = Trans.transResult.trans[i];transHtml.push(transtxt);}transHtml.push("</span>");Trans.transOrigLang=origLang;}else {var txt="该翻译引擎不支持 "+langList[Trans.transOrigLang]+" 翻译成 "+langList[Trans.transTargetLang];transHtml.push(StringFormat("<span>{0}</span>",txt));}//源语言for (var origKey in langList) {if (langList.hasOwnProperty(origKey)) {var origVal = langList[origKey]; var origSelectOption="";if(Trans.transOrigLang.toUpperCase()==origKey.toUpperCase()){origSelectOption='selected="selected"';}origLangListHtml.push(StringFormat('<option value="{0}" {2}>{1}</option>',origKey,origVal,origSelectOption));}}//目标语言for (var targetKey in langList) {if (langList.hasOwnProperty(targetKey) && targetKey!=Trans.transOrigLang && targetKey.toUpperCase()!="AUTO") {var targetVal = langList[targetKey];var targetSelectOption="";targetLangListHtml.push(StringFormat('<option value="{0}" {2}>{1}</option>',targetKey,targetVal,targetSelectOption));}}returnHtml.origLangListHtml=origLangListHtml.join("");returnHtml.targetLangListHtml=targetLangListHtml.join("");returnHtml.transHtml=transHtml.join("");return returnHtml;}};//划词翻译面板var WordTransPanel = {Create: function (popBoxEl, randomCode) {var self = this;var html = this.GetTransContHtml();var transEngineOptionsHtml = "";for (var k in Trans.transEngineList) {if (Trans.transEngineList.hasOwnProperty(k)) {var v = Trans.transEngineList[k].codeText;var selectOption = "";if (Trans.transEngine == k) {selectOption = 'selected="selected"';}transEngineOptionsHtml += StringFormat('<option value="{0}" {2}>{1}</option>', k, v, selectOption);}}var wordTransPanelHtml = StringFormat('<div>翻译引擎:<select>{2}</select>    翻译语言:<input type="text" value="{4}" readonly style="width:80px"/> &#x21E8; <select>{3}</select></div>' +'<div style="word-wrap:break-word">{1}</div>', randomCode, html.transHtml, transEngineOptionsHtml, html.langListHtml, html.origLangName);Panel.popBoxEl = popBoxEl;Panel.randomCode = randomCode;Panel.Create("", "auto bottom", false, wordTransPanelHtml, function ($panel) {+//目标语言$panel.find(StringFormat("#panelBody{0} div:eq(0) select:eq(1)", randomCode)).change(function (e) {Trans.transTargetLang = $(this).find("option:selected").val();Trans.Execute(function () {self.Update(randomCode);});});//翻译引擎$panel.find(StringFormat("#panelBody{0} div:eq(0) select:eq(0)", randomCode)).change(function (e) {Trans.transEngine = $(this).find("option:selected").val();Trans.Update();Trans.Execute(function () {self.Update(randomCode);});});});},Update: function (randomCode) {var self = this;Panel.Update(function ($panel) {var html = self.GetTransContHtml();$panel.find(StringFormat("#panelBody{0} div:eq(0) input:eq(0)", randomCode)).val("").val(html.origLangName);$panel.find(StringFormat("#panelBody{0} div:eq(0) select:eq(1)", randomCode)).html("").html(html.langListHtml);$panel.find(StringFormat("#panelBody{0} div:eq(1)", randomCode)).html("").html(html.transHtml);});},GetTransContHtml: function () {var transObj = {};var langListHtml = [];var langList = Trans.GetLangList();var origLang = Trans.transResult.origLang;var transContHtml = "";if (Trans.transResult.trans.length > 0 && Trans.transResult.orig.length > 0) {//译文var transHtml = [];transHtml.push('<div style="padding-top: 5px;"><ul style="list-style: none;margin: 0;padding: 0;">');for (var i = 0; i < Trans.transResult.trans.length; i++) {var transtxt = Trans.transResult.trans[i];transHtml.push(StringFormat('<li style="list-style: none;"><span>{0}</span></li>', transtxt));}transHtml.push("</ul></div>");//原文var origHtml = [];//原文内容origHtml.push('<div style="padding-bottom: 5px;"><ul style="list-style: none;margin: 0;padding: 0;">');for (var j = 0; j < Trans.transResult.orig.length; j++) {var origtxt = Trans.transResult.orig[j];origHtml.push(StringFormat('<li style="list-style: none;"><span>{0}</span></li>', origtxt));}origHtml.push("</ul>");//原文音标if (Trans.transResult.symbols.en!="" || Trans.transResult.symbols.am!="") {origHtml.push('<div>');if(Trans.transResult.symbols.en!="")origHtml.push(StringFormat('<span style="padding-right: 10px;">英 [{0}]</span>',Trans.transResult.symbols.en));if(Trans.transResult.symbols.am!="")origHtml.push(StringFormat('<span>美 [{0}]</span>',Trans.transResult.symbols.am));origHtml.push('</div>');}origHtml.push("</div>");transContHtml = origHtml.join("") + "<hr/>" + transHtml.join("");Trans.transOrigLang = origLang;} else {var txt = "该翻译引擎不支持 " + langList[Trans.transOrigLang] + " 翻译成 " + langList[Trans.transTargetLang];transContHtml = StringFormat("<div><span>{0}</span></div>", txt);}for (var k in langList) {if (langList.hasOwnProperty(k) && k != Trans.transOrigLang && k.toUpperCase() != "AUTO") {var v = langList[k];var selectOption = "";if (Trans.transTargetLang == k) {selectOption = 'selected="selected"';}langListHtml.push(StringFormat('<option value="{0}" {2}>{1}</option>', k, v, selectOption));}}transObj.origLangName = langList[Trans.transOrigLang];transObj.transHtml = transContHtml;transObj.langListHtml = langListHtml.join("");return transObj;}};//设置面板var SettingPanel={config:[{title:"",item:[{code:"",text:""}]}],Create:function(popBoxEl,randomCode){var self=this;var settingHtml=[];this.InitConfig();settingHtml.push('<div style="padding-left: 15px;display: inline-block;">');/*settingHtml.push('<div style="padding-bottom: 30px; max-width: 600px;">');settingHtml.push('<div style="font-size: 14px; padding-bottom: 3px;">默认翻译引擎:</div>');settingHtml.push(StringFormat('<div style="padding-bottom: 3px; margin-left: 10px;"><label style="font-size: 14px; cursor: pointer;"><input type="radio" name="transEngine{0}" style="cursor: pointer;" value="yd">有道</label></div>',randomCode));settingHtml.push(StringFormat('<div style="padding-bottom: 0px; margin-left: 10px;"><label style="font-size: 14px; cursor: pointer;"><input type="radio" name="transEngine{0}" style="cursor: pointer;" value="ge">谷歌</label></div>',randomCode));settingHtml.push('</div>');*/for (var index = 0; index < this.config.length; index++) {var configItem = this.config[index];settingHtml.push('<div style="padding-bottom: 30px; max-width: 600px;">');settingHtml.push(StringFormat('<div style="font-size: 14px; padding-bottom: 3px;">{0}</div>',configItem.title));for (var itemIndex = 0; itemIndex < configItem.item.length; itemIndex++) {var itemObj = configItem.item[itemIndex];settingHtml.push(StringFormat('<div style="padding-bottom: 0px; margin-left: 10px;"><label style="font-size: 14px; cursor: pointer;"><input type="radio" name="transEngine{0}" style="cursor: pointer;" value="{1}">{2}</label></div>',randomCode,itemObj.code,itemObj.text));}   settingHtml.push('</div>');}settingHtml.push('<div>');settingHtml.push(StringFormat('<button id="saveBtn{0}">保存</button>',randomCode));settingHtml.push(StringFormat('<span id="saveStatus{0}" style="display:none;margin-left:10px;background-color: #fff1a8;padding: 3px;">设置已保存。</span>',randomCode));settingHtml.push('</div>');settingHtml.push('</div>');var settingHtmlStr=settingHtml.join("");Panel.popBoxEl=popBoxEl;Panel.randomCode=randomCode;Panel.Create("网页翻译助手设置","auto bottom",false,settingHtmlStr,function($panel){$panel.css({position: "fixed",top:"20px"});self.Update(randomCode);//保存设置$panel.find(StringFormat("#panelBody{0} #saveBtn{0}",randomCode)).click(function(e){var defaultTransEngine=$panel.find(StringFormat("#panelBody{0} input[name='transEngine{0}']:checked",randomCode)).val();options.defaulttransengine=defaultTransEngine;SetSettingOptions();$panel.find(StringFormat("#panelBody{0} #saveStatus{0}",randomCode)).fadeIn(function(){setTimeout(function(){$panel.find(StringFormat("#panelBody{0} #saveStatus{0}",randomCode)).fadeOut();},1500);});});});},Update:function(randomCode){GetSettingOptions();Panel.Update(function($panel){$panel.find(StringFormat("#panelBody{0} input[name='transEngine{0}'][value='{1}']",randomCode,options.defaulttransengine)).prop("checked",true);});},InitConfig:function(){this.config=[];var configObj={title:"",item:[{code:"",text:""}]};configObj.title="默认翻译引擎:";configObj.item=[];for (var k in Trans.transEngineList) {if (Trans.transEngineList.hasOwnProperty(k)) {var v = Trans.transEngineList[k].codeText;configObj.item.push({code:k,text:v});}}this.config.push(configObj);}};//主程序var WebTranslate=function(){var transIconBase64="";var $doc=$(document);var $body=$("html body");var $head=$("html head");var randomCode="yyMM000000";    //属性随机码,年月加六位随机码。用于元素属性后缀,以防止属性名称重复。var createHtml=function(){var wordTransIconHtml=StringFormat('<div id="wordTrans{0}" class="wordTrans{0}"><div class="wordTransIcon{0}"></div></div>',randomCode,transIconBase64);$body.append(StringFormat('<div id="webTrans{0}">',randomCode)+wordTransIconHtml+'</div>');};var createStyle=function(){//尽可能避开csp认证GM_xmlhttpRequest({method:"get",url:"https://cdn.jsdelivr.net/gh/zyufstudio/jQuery@master/jPopBox/dist/jPopBox.min.css",onload:function(r){GM_addStyle(r.responseText+".JPopBox-tip-white{width: 482px;max-width: 550px;min-width: 450px;}");}});var s="";s+=StringFormat(".wordTrans{0}{background-color: rgb(245, 245, 245);box-sizing: content-box;cursor: pointer;z-index: 2147483647;border-width: 1px;border-style: solid;border-color: rgb(220, 220, 220);border-image: initial;border-radius: 5px;padding: 0.5px;position: absolute;display: none}",randomCode);s+=StringFormat(".wordTransIcon{0}{background-image: url({1});background-size: 25px;height: 25px;width: 25px;}",randomCode,transIconBase64);s+=Panel.CreateStyle();GM_addStyle(s);};var ShowWordTransIcon=function(){var $wordTransIcon=$("div#wordTrans"+randomCode);var isSelect=false;var isPanel=false;var isWordTransIcon=false;$doc.on({"selectionchange":function(e){isSelect=true;},"mousedown":function(e){var $targetEl=$(e.target);isPanel=$targetEl.parents().is("div.JPopBox-tip-white");isWordTransIcon=$targetEl.parents().is(StringFormat("div#wordTrans{0}",randomCode));//点击翻译图标外域和翻译面板外域时,隐藏图标和翻译面板if(!isWordTransIcon && !isPanel){$wordTransIcon.hide();Trans.Clear();Panel.Destroy();}else {//点击翻译图标,取消鼠标默认事件,防止选中的文本消失if(isWordTransIcon){ClearBubble(e);}}},"mouseup":function(e){var selectText = window.getSelection().toString().trim();if(!isPanel&&isSelect&&selectText){$wordTransIcon.show().css({left: e.pageX + 'px',top : e.pageY + 12 + 'px'});isSelect=false;}}});$wordTransIcon.click(function(e){Trans.Clear();Panel.Destroy();var selecter=window.getSelection();var selectText = selecter.toString().trim();GetSettingOptions();Trans.transText=selectText;Trans.transType="word";Trans.transEngine=options.defaulttransengine;//defaultTransEngine;Trans.Update();Trans.Execute(function(){WordTransPanel.Create($wordTransIcon,randomCode);$wordTransIcon.hide();});});};var guid="";var RegMenu=function(){GM_registerMenuCommand("文本翻译",function(){var $body=$("html body");$("div#wordTrans"+randomCode).hide();Trans.Clear();Panel.Destroy();GetSettingOptions();Trans.transEngine=options.defaulttransengine;//defaultTransEngine;Trans.Update();TextTransPanel.Create($body,randomCode);});GM_registerMenuCommand("Google整页翻译",function(){if(guid=="") guid=Guid();var cbscript=StringFormat('!function(){!function(){function e(){window.setTimeout(function(){window[t].showBanner(!0)},10)}function n(){return new google.translate.TranslateElement({autoDisplay:!1,floatPosition:0,multilanguagePage:!0,includedLanguages:"zh-CN,zh-TW,en",pageLanguage:"auto"})}var t=(document.documentElement.lang,"TE_{0}"),o="TECB_{0}";if(window[t])e();else if(!window.google||!google.translate||!google.translate.TranslateElement){window[o]||(window[o]=function(){window[t]=n(),e()});var a=document.createElement("script");a.src="https://translate.google.com.hk/translate_a/element.js?cb="+encodeURIComponent(o)+"&client=tee",document.getElementsByTagName("head")[0].appendChild(a)}}()}();',guid);$head.append(StringFormat('<script>{0}</script>',cbscript));});GM_registerMenuCommand("设置",function(){$("div#wordTrans"+randomCode).hide();Trans.Clear();Panel.Destroy();SettingPanel.Create($body,randomCode);});};this.init=function(){randomCode=DateFormat(new Date(),"yyMM").toString()+(Math.floor(Math.random() * (999999 - 100000 + 1) ) + 100000).toString();Trans.RegisterEngine();createStyle();createHtml();ShowWordTransIcon();RegMenu();};};var webTrans=new WebTranslate();webTrans.init();})();

如何判断文本内容的语言语种?

遍历所有字符,统计频率最高的语种为文档或者网页的语种,核心代码如下:

function isChinese(temp) 
{ var re = /[^\u4e00-\u9fa5]/; if(re.test(temp)) return false; return true; 
}function isJapanese(temp) 
{ var re = /[^\u0800-\u4e00]/; if(re.test(temp)) return false; return true; 
}function isKoera(chr) {if(((chr > 0x3130 && chr < 0x318F) || (chr >= 0xAC00 && chr <= 0xD7A3))) {return true;}return false;
}function isContainKoera(temp)
{var cnt = 0;for(var i=0;i < temp.length ; i++){if(isKoera(temp.charAt(i)))cnt++;}if (cnt > 0) return true;return false;
}function isContainChinese(temp)
{var cnt = 0;for(var i=0;i < temp.length ; i++){if(isChinese(temp.charAt(i)))cnt++;}if (cnt > 5) return true;return false;
}function isContainChinese2(temp)
{var cnt = 0;for(var i=0;i < temp.length ; i++){if(isChinese(temp.charAt(i)))cnt++;}if (cnt > 0 && temp.length<=3) return true;return false;
}function isContainJapanese(temp)
{var cnt = 0;for(var i=0;i < temp.length ; i++){if(isJapanese(temp.charAt(i)))cnt++;}if (cnt > 2) return true;return false;
}

当然也可以 截取 一段文本内容发送给 语言 模型 进行识别,但是对于混合文档还是不准确,还不如放在前端做节省计算资源。

看下谷歌翻译之前的实现方式:

<script type="text/javascript" src="http://www.google.com/jsapi"></script ><script type="text/javascript">google.load("language", "1");function initialize(){var text = document.getElementById("text").innerHTML;google.language.detect(text, function(result){if(!result.error && result.language){google.language.translate(text, result.language, "en", function(result){var translated = document.getElementById("translation");if(result.translation){translated.innerHTML = result.translation;}});}});}google.setOnLoadCallback(initialize);</script> 

或者:

<script src="https://translate.google.cn/translate_a/element.js?cb=googleTranslateElementInit"></script>

还有网友改进后的一种方式:

<script src="./el_main.js"></script>
<script src="./el_main.css"></script>
<script>
function googleTranslateElementInit() {new google.translate.TranslateElement({//这个参数不起作用,看文章底部更新,翻译面板的语言//pageLanguage: 'zh-CN',//这个是你需要翻译的语言,比如你只需要翻译成越南和英语,这里就只写en,viincludedLanguages: 'en,zh-CN,hr,cs,da,nl,fr,de,el,iw,hu,ga,it,ja,ko,pt,ro,ru,sr,es,th,vi',//选择语言的样式,这个是面板,还有下拉框的样式,具体的记不到了,找不到api~~layout: google.translate.TranslateElement.InlineLayout.SIMPLE,//自动显示翻译横幅,就是翻译后顶部出现的那个,有点丑,这个属性没有用的话,请看文章底部的其他方法autoDisplay: true, //还有些其他参数,由于原插件不再维护,找不到详细api了,将就了,实在不行直接上dom操作}, 'google_translate_element'//触发按钮的id);}
</script> 

还用另一个 js 库 去实现的:

<script src="./franc.js">// import {franc, francAll} from './franc.js'var ll=franc('Alle menneske er fødde til fridom') //=> 'nno'console.log(ll);</script>

依赖的 js 库太大了,不贴出来了,有需要可以留言。

查资料看到还有这种用法的:

<script>
javascript: void((function () {var script = document.createElement('script');script.src = '//translate.google.cn/translate_a/element.js?cb=googleTranslateElementInit';document.getElementsByTagName('head')[0].appendChild(script);var google_translate_element = document.createElement('div');google_translate_element.id = 'google_translate_element';google_translate_element.style = 'position:fixed; bottom:10px; right:10px; cursor:pointer;';document.documentElement.appendChild(google_translate_element);script = document.createElement('script');script.innerHTML = "function googleTranslateElementInit() {" +"new google.translate.TranslateElement({" +"layout: google.translate.TranslateElement.InlineLayout.SIMPLE," +"multilanguagePage: true," +"pageLanguage: 'auto'," +"includedLanguages: 'zh-CN,zh-TW,en'" +"}, 'google_translate_element');}";document.getElementsByTagName('head')[0].appendChild(script);
})());</script>

获取网页内容的所有标签的文本内容,过滤非文本标签,实现如下:

<script>function listen(callback) {// 获取 HTML 文档中的所有元素,但不包括 下列 选择器的元素var exclude = ['head', 'pre', 'script', 'textarea']//排除名单var selectors = []exclude.forEach((item, index) => {selectors.push(item)//排除该元素selectors.push(item + ' *')//排除该元素后代})get(document.querySelectorAll('*:not(' + selectors.join(',') + ')'))//*:not(pre,pre *)// 创建 MutationObserver 对象let observer = new MutationObserver(function (mutations) {mutations.forEach(function (mutation) {// 遍历新添加的节点for (let i = 0; i < mutation.addedNodes.length; i++) {let node = mutation.addedNodes[i];// 如果节点是元素节点,就调用 get 函数if (node.nodeType === 1) {callMyFunction(node)function callMyFunction(param1) {setTimeout(function () {get([...param1.querySelectorAll('*'), param1])}, 300);}}}});});// 设置 MutationObserver 的参数,表示监听所有元素的变化let config = {childList: true,subtree: true};// 启动 MutationObserverobserver.observe(document, config);function get(elements) {// 遍历所有元素for (let i = 0; i < elements.length; i++) {let element = elements[i];// 遍历元素的 childNodesfor (let j = 0; j < element.childNodes.length; j++) {let node = element.childNodes[j];// 如果当前节点是一个文本节点(nodeType 为 3)且不包含子节点(nodeName 为 '#text'),就将文本添加到数组中if (node.nodeType === 3 && node.nodeName.toLowerCase() === '#text') {// 过滤掉文本中的换行符let text = node.nodeValuevar v = { a: false, b: false }text.slice(0, 1) == " " ? v.a = true : v.a = falsetext.slice(-1) == " " ? v.b = true : v.b = falsetext = text.replace(/[\n\t\r]/g, '').trim();// 如果文本不仅包含空白字符,就将它添加到数组中if (/\S/.test(text)) {//不处理只有数字和符号的文本if (!/^[0-9\+\-\*\/\=><&\!@#\$%\^\*\\(\)\[\]\{\}_,.;',。、;’、]{1,}$/.test(text)) {//---------------处理//翻译text//text = "$" + text//---------------处理结束--显示v.a == true ? text = " " + text : textv.b == true ? text = text + " " : textif (!element.matches('script,textarea')) {//单元素阻断,白名单node.nodeValue = textcallback.call({ text: text, node: node, element: element })} else {//console.log("位于排除标签列表", element);}} else {//console.log("只有数字和符号的文本", text);}}}}}}}let time = null;var data = []listen(function () {if (time !== null) {clearTimeout(time);}time = setTimeout(async () => {console.log(data);//抖动结束,开始翻译var sl = []data.forEach((item, index) => {//取textsl.push(item['text'])});// var tl = await translation_arr(sl) //返回一个数组[[翻译结果,源语言类型],...*]//使用的谷歌批量翻译API,这里就不提供了var tl = []sl.forEach((item, index) => {tl.push('[ 编辑:' + item + ',' + index + '] ')});tl.forEach((item, index) => {data[index]['node'].origText = data[index]['node'].nodeValuedata[index]['node'].nodeValue = item//更改文本});//这里的this指向的是input}, 500)data.push(this)})/* 监听文本节点被点击document.onselectstart = function (e) {console.log(e.target,e.target.origText);}*/</script> 

在网页中自动插入一个 下拉框 用于展示支持的翻译语种,并根据网页内容识别的语种自动选中语种类型:


<script>// 获取浏览器默认语言const getBrowserLang = function() {let browserLang = navigator.language? navigator.language: navigator.browserLanguage;let defaultBrowserLang = "";if (browserLang.toLowerCase() === "us" ||browserLang.toLowerCase() === "en" ||browserLang.toLowerCase() === "en_us") {defaultBrowserLang = "en_US";} else {defaultBrowserLang = "zh_CN";}return defaultBrowserLang;
};console.log(getBrowserLang());
</script><script>var type = navigator.appName; //BOM对象获取浏览器名称if (type == "Netscape") {var lang = navigator.language.toLowerCase(); //获取浏览器配置语言,支持非IE浏览器} else {var lang = navigator.browserLanguage.toLowerCase(); //获取浏览器配置语言,支持IE5+};console.log(lang);var lang = lang.substr(0, 5); //获取浏览器配置语言前4位console.log(lang);/*极速测试体验,用于审查元素时直接执行的1. 随便打开一个网页2. 右键-审查元素3. 粘贴入一下代码:var head= document.getElementsByTagName('head')[0];  var script= document.createElement('script');  script.type= 'text/javascript';  script.src= 'http://localhost:8060/static/inspector.js';  head.appendChild(script); 4. Enter 回车键 , 执行5. 在当前网页的左上角,就出现了一个大大的切换语言了	使用的是 v2.x 版本进行的翻译*/var head= document.getElementsByTagName('head')[0]; 
var script= document.createElement('script'); 
script.type= 'text/javascript'; 
script.src= 'http://localhost:8060/static/translate.js'; 
script.onload = script.onreadystatechange = function() {translate.storage.set('to','');//设置使用v2.x 版本translate.setUseVersion2(); //SELECT 修改 onchange 事件translate.selectLanguageTag.selectOnChange = function(event){//判断是否是第一次翻译,如果是,那就不用刷新页面了。 true则是需要刷新,不是第一次翻译var isReload = translate.to != null && translate.to.length > 0;if(isReload){//如果要刷新页面的话,弹出友好提示alert('您好,快速体验暂时只能切换其中一种语言进行体验,只是提供效果展示,您可参考接入文档来接入您的项目中进行完整体验及使用。');}else{var language = event.target.value;translate.changeLanguage(language);console.log("ttttt");console.log(language);}			}translate.listener.start();	//开启html页面变化的监控,对变化部分会进行自动翻译。注意,这里变化区域,是指使用 translate.setDocuments(...) 设置的区域。如果未设置,那么为监控整个网页的变化translate.execute();document.getElementById('translate').style.position = 'fixed';document.getElementById('translate').style.color = 'red';document.getElementById('translate').style.left = '10px';document.getElementById('translate').style.top = '10px';document.getElementById('translate').style.zIndex = '9999999999999';setInterval(function() {try{if(document.getElementById('translateSelectLanguage') == null){return;}document.getElementById('translateSelectLanguage').style.fontSize = '2rem';document.getElementById('translateSelectLanguage').style.borderWidth = '0.5rem';document.getElementById('translateSelectLanguage').style.borderColor = 'red';}catch(e){//select数据是通过接口返回的}},1000);}
head.appendChild(script); </script>

不同页面或者不同 js 之间传递数据,粗暴简单一点可以通过如下方式:

window.localStorage.setItem('local_language',translate.language.local);



通过对前端js和后端翻译服务进行了长时间的研究和测试,最终实现的效果可以达到:

1)如果是自己的网站
在网页最末尾, 之前,加入以下代码,一般在页面的最底部就出现了选择语言的 select 切换标签。 其实就这么简单:

<script src="http://localhost:8060/static/translate.min.js"></script>
<script>// translate.language.setLocal('chinese_simplified');translate.request.api.host='http://localhost:8060/';translate.execute();
</script>

或者:

<script src="http://localhost:8060/static/i18n.js"></script>

2)如果是第三方的网站
随便打开一个网页
右键 - 检查(审查元素)〉控制台:
粘贴入以下代码:

var head= document.getElementsByTagName('head')[0]; var script= document.createElement('script'); script.type= 'text/javascript'; script.src= 'http://localhost:8060/static/inspector.js'; head.appendChild(script);

或者:

var head= document.getElementsByTagName('head')[0]; var script= document.createElement('script'); script.type= 'text/javascript'; script.src= 'http://localhost:8060/static/inspector.js'; head.appendChild(script);

Enter 回车键 , 执行
在当前网页的左上角,就出现了一个大大的切换语言,切换试试看。


效果参考:

点击查看

https://blog.csdn.net/u014374009/article/details/135401003

源码及后端翻译服务支持docker一键部署:

点击查看

https://blog.csdn.net/u014374009/article/details/135370222


有任何定制化的需求可以联系作者完成开发。

这篇关于离线场景下任意文档的在线预览及原样格式翻译,不依赖其他厂商接口非侵入式一行js代码实现网站的翻译及国际化,可配置使用多种翻译语言的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/688351

相关文章

windos server2022的配置故障转移服务的图文教程

《windosserver2022的配置故障转移服务的图文教程》本文主要介绍了windosserver2022的配置故障转移服务的图文教程,以确保服务和应用程序的连续性和可用性,文中通过图文介绍的非... 目录准备环境:步骤故障转移群集是 Windows Server 2022 中提供的一种功能,用于在多个

windos server2022里的DFS配置的实现

《windosserver2022里的DFS配置的实现》DFS是WindowsServer操作系统提供的一种功能,用于在多台服务器上集中管理共享文件夹和文件的分布式存储解决方案,本文就来介绍一下wi... 目录什么是DFS?优势:应用场景:DFS配置步骤什么是DFS?DFS指的是分布式文件系统(Distr

使用SQL语言查询多个Excel表格的操作方法

《使用SQL语言查询多个Excel表格的操作方法》本文介绍了如何使用SQL语言查询多个Excel表格,通过将所有Excel表格放入一个.xlsx文件中,并使用pandas和pandasql库进行读取和... 目录如何用SQL语言查询多个Excel表格如何使用sql查询excel内容1. 简介2. 实现思路3

python实现pdf转word和excel的示例代码

《python实现pdf转word和excel的示例代码》本文主要介绍了python实现pdf转word和excel的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价... 目录一、引言二、python编程1,PDF转Word2,PDF转Excel三、前端页面效果展示总结一

在MyBatis的XML映射文件中<trim>元素所有场景下的完整使用示例代码

《在MyBatis的XML映射文件中<trim>元素所有场景下的完整使用示例代码》在MyBatis的XML映射文件中,trim元素用于动态添加SQL语句的一部分,处理前缀、后缀及多余的逗号或连接符,示... 在MyBATis的XML映射文件中,<trim>元素用于动态地添加SQL语句的一部分,例如SET或W

IDEA如何将String类型转json格式

《IDEA如何将String类型转json格式》在Java中,字符串字面量中的转义字符会被自动转换,但通过网络获取的字符串可能不会自动转换,为了解决IDEA无法识别JSON字符串的问题,可以在本地对字... 目录问题描述问题原因解决方案总结问题描述最近做项目需要使用Ai生成json,可生成String类型

Go语言实现将中文转化为拼音功能

《Go语言实现将中文转化为拼音功能》这篇文章主要为大家详细介绍了Go语言中如何实现将中文转化为拼音功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 有这么一个需求:新用户入职 创建一系列账号比较麻烦,打算通过接口传入姓名进行初始化。想把姓名转化成拼音。因为有些账号即需要中文也需要英

关于Maven中pom.xml文件配置详解

《关于Maven中pom.xml文件配置详解》pom.xml是Maven项目的核心配置文件,它描述了项目的结构、依赖关系、构建配置等信息,通过合理配置pom.xml,可以提高项目的可维护性和构建效率... 目录1. POM文件的基本结构1.1 项目基本信息2. 项目属性2.1 引用属性3. 项目依赖4. 构

Spring核心思想之浅谈IoC容器与依赖倒置(DI)

《Spring核心思想之浅谈IoC容器与依赖倒置(DI)》文章介绍了Spring的IoC和DI机制,以及MyBatis的动态代理,通过注解和反射,Spring能够自动管理对象的创建和依赖注入,而MyB... 目录一、控制反转 IoC二、依赖倒置 DI1. 详细概念2. Spring 中 DI 的实现原理三、

使用C#代码计算数学表达式实例

《使用C#代码计算数学表达式实例》这段文字主要讲述了如何使用C#语言来计算数学表达式,该程序通过使用Dictionary保存变量,定义了运算符优先级,并实现了EvaluateExpression方法来... 目录C#代码计算数学表达式该方法很长,因此我将分段描述下面的代码片段显示了下一步以下代码显示该方法如