Tento skript by neměl být instalován přímo. Jedná se o knihovnu, kterou by měly jiné skripty využívat pomocí meta příkazu // @require https://update.greatest.deepsurf.us/scripts/452362/1281581/Baidu%20Translate.js
- /* eslint-disable no-multi-spaces */
- /* eslint-disable no-return-assign */
-
- /* eslint-disable curly */
- /* eslint-disable no-sequences */
-
- // ==UserScript==
- // @name Baidu Translate
- // @name:zh-CN 百度翻译
- // @name:en Baidu Translate
- // @namespace PY-DNG APIS
- // @version 0.2.7
- // @description Baidu translate api support for userscripts. No need for baidu's token.
- // @description:zh-CN 用户脚本版的百度翻译API支持,无需申请token,@require即用
- // @description:en Baidu translate api support for userscripts. No need for baidu's token.
- // @author PY-DNG
- // @license MIT
- // @icon 
- // @grant GM_xmlhttpRequest
- // @connect fanyi.baidu.com
- // ==/UserScript==
-
- /* Usage:
- ** baidu_translate(details): details = {
- text: <string> Text you want to translate.
- Required argument, no default value.
- callback: <function> Callback function when translation succeed,
- e.g. function(result) {console.log(result);}.
- Argument `result` will be <string> translate result.
- Required argument, no default value.
- [src]: <string> Source language; e.g. 'zh' for Chinese, 'en' for English, 'ja' for Japanese, ....
- Default: Auto detect.
- [dst]: <string> Destination language,
- Default: 'zh'.
- [split]: <number> Text spliting max length, just leave blank if you don't know what this means.
- What it is: Baidu has limited that up to 5000 letters can be translated each single API
- request, so the translate function will split text by about every ${split} letters, and
- translates each of them with a single API request then concatenate the translate result
- together again. Don't worry if any sentence to be splited, actually the translate
- function splits text with '\n' first, then concatenate the '\n'-splited strings as long
- as possible while keeping its length <= ${split}.
- Default: 5000
- [onerror]: <function> Callback function when translation failed,
- e.g. function(reason) {console.log(reason);}.
- Argument: `reason` may be anything that causes its failure, just for debugging and do
- not use it in production.
- Default: function() {}.
- [retry]: <number> Times to retry before onerror function being called or an error being thrown,
- Default: 3.
- }
- ** Or:
- ** baidu_translate(text, src, dst, callback, onerror, split, retry)
- **
- ** Overloads:
- ** baidu_translate(text, src, dst, callback, onerror, split)
- ** baidu_translate(text, dst, callback, onerror, split)
- ** baidu_translate(text, callback, onerror, split)
- ** baidu_translate(text, callback, onerror)
- ** baidu_translate(text, callback)
- */
-
- let baidu_translate, bdTransReady;
- [baidu_translate, bdTransReady] = (function() {
- const BDT = new BaiduTranslateAPI();
- return [baidu_translate, bdTransReady];
-
- function baidu_translate() {
- // Check BDT ready
- if (!BDT.gtk || !BDT.token) {
- //throw new Error('BaiduTranslateAPI not ready');
- return false;
- }
-
- // Get argument
- const [text, src, dst, callback, onerror, split, retry] = parseArgs([...arguments], [
- function(args, defaultValues) {
- const details = args[0];
- const parsed = [...defaultValues];
- details.hasOwnProperty('text') && (parsed[0] = details.text);
- details.hasOwnProperty('src') && (parsed[1] = details.src);
- details.hasOwnProperty('dst') && (parsed[2] = details.dst);
- details.hasOwnProperty('callback') && (parsed[3] = details.callback);
- details.hasOwnProperty('onerror') && (parsed[4] = details.onerror);
- details.hasOwnProperty('split') && (parsed[5] = details.split);
- details.hasOwnProperty('retry') && (parsed[6] = details.retry);
- return parsed;
- },
- [1,4],
- [1,4,5],
- [1,4,5,6],
- [1,3,4,5,6],
- [1,2,3,4,5,6],
- [1,2,3,4,5,6,7]
- ], ['', undefined, undefined, function() {}, function() {}, 5000, 3]);
-
- // Split lines
- const textarr = text.replaceAll('\r\n', '\n').replaceAll('\n\r', '\n').replaceAll('\r', '\n').split('\n');
- if (textarr.some((t) => (t.length > split))) {
- throw new Error('Some paragraph is longer than given split length (' + split + ')');
- }
-
- // Prepare
- const AM = new AsyncManager();
- const result = [];
- AM.onfinish = function() {
- callback(result.map(re => '\n'.repeat(re.newline_begin) + re.dst + '\n'.repeat(re.newline_end)).join('\n'));
- }
-
- // Send requests
- let index = 0;
- while (textarr.length > 0) {
- // Join translate api paragraph
- let para = '';
- while (textarr.length > 0 && para.length + textarr[0].length < split) {
- para += '\n' + textarr.shift();
- }
- para = para.substr(1);
-
- // Whether this paragraph contains \n at beginning
- const countNewLine = str => str.split('').filter(t => t === '\n').length;
- const newline_begin = countNewLine(para.match(/^\s*/) ? para.match(/^\s*/)[0] : '');
- const newline_end = countNewLine(para.match(/^\s*/) ? para.match(/\s*$/)[0] : '');
- result[index] = {src: para, dst: null, newline_begin, newline_end};
-
- // API request
- AM.add();
- BDT.translate({
- text: para,
- args: [index],
- src: src,
- dst: dst,
- onerror: onerror,
- retry: retry,
- callback: function(json, i) {
- const temp_result = json.trans_result.data.reduce(function(pre, cur) {
- return (pre.push('\n'.repeat(cur.prefixWrap) + cur.dst), pre);
- }, []).join('\n');
- result[i].dst = temp_result;
- AM.finish();
- }
- });
- index++;
- }
- AM.finishEvent = true;
-
- return true;
- }
-
- function bdTransReady(callback) {
- bdTransReady.callbacks = bdTransReady.callbacks || [];
- bdTransReady.callbacks.push(callback);
- BDT.oninit = dispatchCallback;
-
- function dispatchCallback() {
- bdTransReady.callbacks.forEach(cb => cb());
- bdTransReady.callbacks.splice(0, Infinity);
- }
- }
-
- function BaiduTranslateAPI() {
- const BT = this;
- BT.gtk = BT.token = null;
- BT.inited = false;
-
- let oninit=function() {};
- Object.defineProperty(BT, 'oninit', {
- enumerable: true,
- set: (f) => (typeof f === 'function' && (oninit = f) && BT.inited && oninit()),
- get: () => (oninit)
- });
-
- init();
-
- BT.calcSign = calcSign;
- BT.translate = translate;
-
- async function translate(details) {
- const callback = details.callback;
- const text = details.text;
- const src = details.src || await langDetect(text);
- const dst = details.dst || 'zh';
- const onerror = details.onerror || function() {};
- const retry = details.retry || 0;
- const args = details.args || [];
-
- GM_xmlhttpRequest({
- method: 'POST',
- url: 'https://fanyi.baidu.com/v2transapi',
- headers: {
- 'User-Agent': 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_14_2) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/71.0.3578.98 Safari/537.36',
- 'Content-Type': 'application/x-www-form-urlencoded'
- },
- data: toQueryString({
- 'from': src,
- 'to': dst,
- 'query': text,
- 'simple_means_flag': 3,
- 'sign': calcSign(text),
- 'token': BT.token
- }),
- onload: function(response) {
- response.status !== 200 && Err(response);
- const json = JSON.parse(response.responseText);
- json.error && Err(json);
- callback.apply(null, [json].concat(args));
-
- function Err(e) {
- !_onerror(e) && console.log('Retrying...\nleft: ' + (retry-1).toString());
- console.log(e);
- throw new Error('Server returned with an error (logged above)');
- }
- },
- onerror: _onerror,
- ontimeout: _onerror,
- onabort: _onerror,
- });
-
- // Returns true for error, false for retry
- function _onerror(e) {
- console.log('sign = ' + calcSign(text));
- if (retry > 0) {
- setTimeout(retryRequest.bind(null, e), 500);
- return false;
- }
- onerror(e);
- return true;
- }
-
- function retryRequest(e) {
- translate({
- callback: callback,
- text: text,
- src: src,
- dst: dst,
- onerror: onerror,
- retry: retry - 1,
- args: args
- });
- }
- }
-
- function langDetect(text) {
- return new Promise((resolve, reject) => {
- GM_xmlhttpRequest({
- method: 'POST',
- url: 'https://fanyi.baidu.com/langdetect',
- headers: {
- 'User-Agent': 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_14_2) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/71.0.3578.98 Safari/537.36',
- 'Content-Type': 'application/x-www-form-urlencoded'
- },
- data: toQueryString({'query': text}),
- onload: function(response) {
- const json = JSON.parse(response.responseText);
- resolve(json.lan);
- },
- onerror: function(e) {reject(e)},
- ontimeout: function(e) {reject(e)},
- onabort: function(e) {reject(e)},
- });
- });
- }
-
- // Calc sign based on query-text and gtk
- function calcSign(query) {
- function e(t, e) {
- (null == e || e > t.length) && (e = t.length);
- for (var n = 0, r = new Array(e); n < e; n++)
- r[n] = t[n];
- return r
- }
- function n(t, e) {
- for (var n = 0; n < e.length - 2; n += 3) {
- var r = e.charAt(n + 2);
- r = "a" <= r ? r.charCodeAt(0) - 87 : Number(r),
- r = "+" === e.charAt(n + 1) ? t >>> r : t << r,
- t = "+" === e.charAt(n) ? t + r & 4294967295 : t ^ r
- }
- return t
- }
- var r = null;
- var token = function(t, _gtk) {
- var o, i = t.match(/[\uD800-\uDBFF][\uDC00-\uDFFF]/g);
- if (null === i) {
- var a = t.length;
- a > 30 && (t = "".concat(t.substr(0, 10)).concat(t.substr(Math.floor(a / 2) - 5, 10)).concat(t.substr(-10, 10)))
- } else {
- for (var s = t.split(/[\uD800-\uDBFF][\uDC00-\uDFFF]/), c = 0, u = s.length, l = []; c < u; c++)
- "" !== s[c] && l.push.apply(l, function(t) {
- if (Array.isArray(t))
- return e(t)
- }(o = s[c].split("")) || function(t) {
- if ("undefined" != typeof Symbol && null != t[Symbol.iterator] || null != t["@@iterator"])
- return Array.from(t)
- }(o) || function(t, n) {
- if (t) {
- if ("string" == typeof t)
- return e(t, n);
- var r = Object.prototype.toString.call(t).slice(8, -1);
- return "Object" === r && t.constructor && (r = t.constructor.name),
- "Map" === r || "Set" === r ? Array.from(t) : "Arguments" === r || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(r) ? e(t, n) : void 0
- }
- }(o) || function() {
- throw new TypeError("Invalid attempt to spread non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.")
- }()),
- c !== u - 1 && l.push(i[c]);
- var p = l.length;
- p > 30 && (t = l.slice(0, 10).join("") + l.slice(Math.floor(p / 2) - 5, Math.floor(p / 2) + 5).join("") + l.slice(-10).join(""))
- }
- for (var h = (_gtk || "").split("."), f = Number(h[0]) || 0, m = Number(h[1]) || 0, g = [], y = 0, v = 0; v < t.length; v++) {
- var _ = t.charCodeAt(v);
- _ < 128 ? g[y++] = _ : (_ < 2048 ? g[y++] = _ >> 6 | 192 : (55296 == (64512 & _) && v + 1 < t.length && 56320 == (64512 & t.charCodeAt(v + 1)) ? (_ = 65536 + ((1023 & _) << 10) + (1023 & t.charCodeAt(++v)),
- g[y++] = _ >> 18 | 240,
- g[y++] = _ >> 12 & 63 | 128) : g[y++] = _ >> 12 | 224,
- g[y++] = _ >> 6 & 63 | 128),
- g[y++] = 63 & _ | 128)
- }
- for (var b = f, w = '+-a^+6', k = '+-3^+b+-f', x = 0; x < g.length; x++)
- b = n(b += g[x], w);
- return b = n(b, k),
- (b ^= m) < 0 && (b = 2147483648 + (2147483647 & b)),
- "".concat((b %= 1e6).toString(), ".").concat(b ^ f)
- }
-
- return token(query, BT.gtk);
- }
-
- function toQueryString(obj) {
- const USP = new URLSearchParams();
- for (const [key, value] of Object.entries(obj)) {
- USP.append(key, value);
- }
- return USP.toString();
- }
-
- // Request token and gtk
- function init() {
- // Request twice, make sure gtk is latest,
- // or may get 998 error while requesting translate API
- requestIndex(requestIndex.bind(null, function(html) {
- BT.token = html.match(/token: ["'](.*?)["'],/)[1];
- BT.gtk = html.match(/window.gtk = ["'](.*?)["'];/)[1];
- BT.inited = true;
- oninit();
- }));
-
- function requestIndex(callback) {
- const url = 'https://fanyi.baidu.com';
- GM_xmlhttpRequest({
- method: 'GET',
- headers: {
- 'User-Agent': 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_14_2) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/71.0.3578.98 Safari/537.36'
- },
- url: url,
- onload: function(response) {
- callback(response.responseText);
- }
- });
- }
- }
- }
-
- function AsyncManager() {
- const AM = this;
-
- // Ongoing xhr count
- this.taskCount = 0;
-
- // Whether generate finish events
- let finishEvent = false;
- Object.defineProperty(this, 'finishEvent', {
- configurable: true,
- enumerable: true,
- get: () => (finishEvent),
- set: (b) => {
- finishEvent = b;
- b && AM.taskCount === 0 && AM.onfinish && AM.onfinish();
- }
- });
-
- // Add one task
- this.add = () => (++AM.taskCount);
-
- // Finish one task
- this.finish = () => ((--AM.taskCount === 0 && AM.finishEvent && AM.onfinish && AM.onfinish(), AM.taskCount));
- }
-
- function parseArgs(args, rules, defaultValues=[]) {
- // args and rules should be array, but not just iterable (string is also iterable)
- if (!Array.isArray(args) || !Array.isArray(rules)) {
- throw new TypeError('parseArgs: args and rules should be array')
- }
-
- // fill rules[0]
- (!Array.isArray(rules[0]) || rules[0].length === 1) && rules.splice(0, 0, []);
-
- // max arguments length
- const count = rules.length - 1;
-
- // args.length must <= count
- if (args.length > count) {
- throw new TypeError(`parseArgs: args has more elements(${args.length}) longer than ruless'(${count})`);
- }
-
- // rules[i].length should be === i if rules[i] is an array, otherwise it should be a function
- for (let i = 1; i <= count; i++) {
- const rule = rules[i];
- if (Array.isArray(rule)) {
- if (rule.length !== i) {
- throw new TypeError(`parseArgs: rules[${i}](${rule}) should have ${i} numbers, but given ${rules[i].length}`);
- }
- if (!rule.every((num) => (typeof num === 'number' && num <= count))) {
- throw new TypeError(`parseArgs: rules[${i}](${rule}) should contain numbers smaller than count(${count}) only`);
- }
- } else if (typeof rule !== 'function') {
- throw new TypeError(`parseArgs: rules[${i}](${rule}) should be an array or a function.`)
- }
- }
-
- // Parse
- const rule = rules[args.length];
- let parsed;
- if (Array.isArray(rule)) {
- parsed = [...defaultValues];
- for (let i = 0; i < rule.length; i++) {
- parsed[rule[i]-1] = args[i];
- }
- } else {
- parsed = rule(args, defaultValues);
- }
- return parsed;
- }
- }) ();