// ==UserScript== // @name 划词翻译 // @name:ja 翻訳 // @name:zh-CN 划词翻译 // @namespace http://www.icycat.com // @description 选中文字自动翻译 // @description:ja 選択した文字の自動翻訳 // @description:zh-CN 选中文字自动翻译 // @author 冻猫 // @include * // @version 3.3.1 // @grant GM_xmlhttpRequest // @grant GM_addStyle // @grant GM_getValue // @grant GM_setValue // @run-at document-end // @downloadURL none // ==/UserScript== (function() { 'use strict'; if (!GM_getValue('toLanguage')) { console.log('没有设置语言,自动翻译到浏览器语言'); if (/zh/i.test(navigator.language)) { GM_setValue('toLanguage', 'zh'); } else if (/ja/i.test(navigator.language)) { GM_setValue('toLanguage', 'jp'); } else if (/en/i.test(navigator.language)) { GM_setValue('toLanguage', 'en'); } } function init() { var timer, holdTime, fromLanguage, toLanguage, text, spd, soundUrl, postData; document.addEventListener('mousedown', mouseStart, true); document.addEventListener('mouseup', mouseEnd, true); function mouseStart(e) { if (!document.getElementById('catTipBox')) { createTipBox(); } if (document.getElementById('catTipBox').style.display == 'block' && !/catTranslate/i.test(e.target.className)) { document.getElementById('catTipBox').style.display = ''; document.querySelector('.catdropdown').style.display = ''; document.getElementById('catContentBox').innerHTML = ''; try { window.catSource.stop(); } catch (e) {}; } document.addEventListener('mousemove', moveCheck, true); if (e.target.id == 'catPlaySound') { document.getElementById('catPlaySound').classList.add('catPlaySoundClick'); getRequest(soundUrl); } else if (e.target.id == 'catSet') { document.querySelector('.catdropdown').style.display = 'block'; } else if (e.target.id == 'catzh') { GM_setValue('toLanguage', 'zh'); document.querySelector('.catdropdown').style.display = ''; document.getElementById('catContentBox').innerHTML = '设置成功'; } else if (e.target.id == 'catja') { GM_setValue('toLanguage', 'jp'); document.querySelector('.catdropdown').style.display = ''; document.getElementById('catContentBox').innerHTML = '設定成功'; } else if (e.target.id == 'caten') { GM_setValue('toLanguage', 'en'); document.querySelector('.catdropdown').style.display = ''; document.getElementById('catContentBox').innerHTML = 'Set up successfully'; } } function moveCheck() { clearTimeout(timer); holdTime = false; timer = setTimeout(function() { holdTime = true; }, 300); } function mouseEnd(e) { document.removeEventListener('mousemove', moveCheck, true); clearTimeout(timer); if (holdTime == true && window.getSelection().toString()) { e.preventDefault() e.stopPropagation(); holdTime = false; showTipBox(e.clientX, e.clientY); text = window.getSelection().toString(); var encodeText = encodeURIComponent(text.replace(/([a-z])([A-Z])/g, '$1 $2').replace(/([A-Z]+)([A-Z][a-z])/g, '$1 $2')); if (/[\u3040-\u30FF]+/.test(text)) { //平假名Hiragana片假名Katakana fromLanguage = 'jp'; } else if (/[\uAC00-\uD7AF]/.test(text)) { //谚文音节Hangul Syllables fromLanguage = 'kor'; } else if (/[\u4E00-\u9FFF]+/.test(text)) { //中日韩统一表意文字CJK Unified Ideographs fromLanguage = 'zh'; } else if (/[\u0400-\u04FF]+/.test(text)) { //西里尔字母Cyrillic fromLanguage = 'ru'; } else { //英语 fromLanguage = 'en' } toLanguage = GM_getValue('toLanguage'); if (fromLanguage == toLanguage) { toLanguage = 'en'; } postData = `from=${fromLanguage}&to=${toLanguage}&query=${encodeText}&source=txt`; if (fromLanguage == 'zh') { spd = '5'; } else { spd = '3'; } soundUrl = `https://fanyi.baidu.com/gettts?lan=${fromLanguage}&text=${encodeText}&spd=${spd}&source=web`; postRequest('https://fanyi.baidu.com/transapi', postData); } if (document.querySelector('.catPlaySoundClick')) { document.getElementById('catPlaySound').classList.remove('catPlaySoundClick'); } } } function createTipBox() { GM_addStyle([ '#catTipBox {min-height:24px;min-width:100px;max-width:360px;font:normal 12px/24px Helvetica, Tahoma, Arial, sans-serif;text-align: left;position: absolute;z-index: 2147483647;top: 22px;left: -35px;background: #fff;border: 1px solid #dcdcdc;border: 1px solid rgba(0,0,0,.2);-webkit-transition: opacity .218s;transition: opacity .218s;-webkit-box-shadow: 0 2px 4px rgba(0,0,0,.2);box-shadow: 0 2px 4px rgba(0,0,0,.2);padding: 5px 0;display: none;font-size: 12px;line-height: 20px;}', '#catContentBox {margin:0 8px;color:#333;}', '#catContentBox .catTextBox{line-height:16px;border-bottom: 1px solid #ccc;padding: 2px 18px 9px 0;height: 14px;}', '#catContentBox .catText{font-size:14px;font-weight: bold;color:#333;}', '#catTipBox .catTextBox span{vertical-align: middle;margin-right: 4px;color:#333;font-weight: normal;font-size:12px;}', '#catPlaySound {margin-left: 1px;cursor:pointer;display: inline-block;vertical-align: middle;width: 14px;height: 11px;overflow: hidden;background: url("") no-repeat;text-decoration: none;}', '#catPlaySound.catPlaySoundClick {background-position:0 -14px;}', '#catTipBox .catExplains{padding: 2px 0 0 0;font-weight: normal;font-size:12px;}', '.catTipArrow {width: 0;height: 0;font-size: 0;line-height: 0;display: block;position: absolute;top: -16px;left: 10px;}', '.catTipArrow em, .catTipArrow ins {width: 0;height: 0;font-size: 0;line-height: 0;display: block;position: absolute;border: 8px solid transparent;border-style: dashed dashed solid;}', '.catTipArrow em {border-bottom-color: #d8d8d8;font-style: normal;color: #c00;}', '.catTipArrow ins {border-bottom-color: #fff;top: 2px;text-decoration: underline;background:none !important}', '#catSet {position:absolute;top:9px;right:10px;cursor: pointer;width: 14px;height: 14px;background: url("") no-repeat;text-decoration: none;}', '#catTipBox #catSet .catdropdown {display:none;top:13px;right:-60px;position: absolute;background-color: #ffffff;width: 59px;overflow: auto;z-index: 1;border: 1px solid rgba(0,0,0,.2);box-shadow: 0 2px 4px rgba(0,0,0,.2);}', '#catTipBox #catSet .catdropdown span {color: black;padding: 6px 8px;margin:0px;text-decoration: none;display: block;text-align:center;}', '#catTipBox #catSet .catdropdown span:hover { background-color: #f1f1f1;}' ].join('\n')); var catTipBox = document.createElement('div'); catTipBox.id = 'catTipBox'; catTipBox.className = 'catTranslate'; var catContentBox = document.createElement('div'); catContentBox.id = 'catContentBox'; catContentBox.className = 'catTranslate'; catContentBox.innerHTML = ''; var catTipArrow = document.createElement('div'); catTipArrow.className = 'catTipArrow'; catTipArrow.appendChild(document.createElement('em')); catTipArrow.appendChild(document.createElement('ins')); catTipBox.appendChild(catContentBox); catTipBox.appendChild(catTipArrow); var catSet = document.createElement('div'); catSet.id = 'catSet'; catSet.className = 'catTranslate'; catSet.innerHTML = '
' + '中文' + '日本語' + 'English' + '
'; catTipBox.appendChild(catSet); document.body.appendChild(catTipBox); } function showTipBox(mouseX, mouseY) { var catTipBox = document.getElementById('catTipBox'); var selectedRect = window.getSelection().getRangeAt(0).getBoundingClientRect(); var scrollTop = document.documentElement.scrollTop || document.body.scrollTop; if (selectedRect.width) { if (/Firefox/i.test(navigator.userAgent)) { catTipBox.style.top = selectedRect.y - document.body.getBoundingClientRect().y + selectedRect.height + 8 + 'px'; catTipBox.style.left = selectedRect.x + selectedRect.width / 2 - 18 + 'px'; } else { catTipBox.style.top = selectedRect.top - document.body.getBoundingClientRect().y + selectedRect.height + 8 + 'px'; catTipBox.style.left = selectedRect.left + selectedRect.width / 2 - 18 + 'px'; } } else { catTipBox.style.top = mouseY - document.body.getBoundingClientRect().y + selectedRect.height + 8 + 'px'; catTipBox.style.left = mouseX + selectedRect.width / 2 - 18 + 'px'; } catTipBox.style.display = 'block'; } function parseRes(jsonRes) { var explains = ''; var obj = JSON.parse(jsonRes); var result; //清空盒子 var catContentBox = document.getElementById('catContentBox'); catContentBox.innerHTML = ''; //上盒子 var catTextBox = document.createElement('div'); catTextBox.className = 'catTextBox catTranslate'; //选择的文本 var catText = document.createElement('span'); catText.className = 'catText catTranslate'; //音标 var catPhonetic = document.createElement('span'); catPhonetic.className = 'catPhonetic catTranslate'; //图标 var catPlaySound = document.createElement('span'); catPlaySound.id = 'catPlaySound'; catPlaySound.className = 'catTranslate'; //下盒子翻译 var catExplains = document.createElement('div'); catExplains.className = 'catExplains catTranslate'; //词语 if (obj.type == 1) { result = JSON.parse(obj.result); catText.innerHTML = result.src; catTextBox.appendChild(catText); if (result.voice) { if (obj.from == 'zh') { catPhonetic.innerHTML = result.voice[0].phonic; } else { catPhonetic.innerHTML = result.voice[1].us_phonic; } catTextBox.appendChild(catPhonetic); } if (result.content) { for (var i = 0; i < result.content[0].mean.length; i++) { if (result.content[0].mean[i].pre) { explains = explains + result.content[0].mean[i].pre + ' '; } for (var mean in result.content[0].mean[i].cont) { explains = explains + mean + '; '; } if (i < result.content[0].mean.length - 1) { explains = explains + '
'; } } } //句子 } else if (obj.type == 2) { for (var i = 0; i < obj.data.length; i++) { //catText.innerHTML = catText.innerHTML + obj.data[i].src; explains = explains + obj.data[i].dst; if (i < obj.data.length - 1) { //catText.innerHTML = catText.innerHTML + '
'; explains = explains + '
'; } } catTextBox.appendChild(catText); } catExplains.innerHTML = explains; catTextBox.appendChild(catPlaySound); catContentBox.appendChild(catTextBox); catContentBox.appendChild(catExplains); } function playSound(arraybuffer) { if (!window.audioCtx) { window.audioCtx = new AudioContext(); } audioCtx.decodeAudioData(arraybuffer).then(function(buffer) { window.catSource = audioCtx.createBufferSource(); window.catSource.buffer = buffer; window.catSource.connect(audioCtx.destination); window.catSource.start(); }); } function postRequest(url, data) { GM_xmlhttpRequest({ method: 'POST', url: url, data: data, headers: { 'content-type': 'application/x-www-form-urlencoded; charset=UTF-8', 'origin': 'https://fanyi.baidu.com', 'referer': 'https://fanyi.baidu.com/', 'user-agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/74.0.3729.131 Safari/537.36', 'cookie': 'locale=zh; BAIDUID=40D4B467750C147DE81CE4AC40335080:FG=1; Hm_lvt_64ecd82404c51e03dc91cb9e8c025574=1557733597; to_lang_often=%5B%7B%22value%22%3A%22en%22%2C%22text%22%3A%22%u82F1%u8BED%22%7D%2C%7B%22value%22%3A%22zh%22%2C%22text%22%3A%22%u4E2D%u6587%22%7D%5D; REALTIME_TRANS_SWITCH=1; FANYI_WORD_SWITCH=1; HISTORY_SWITCH=1; SOUND_SPD_SWITCH=1; SOUND_PREFER_SWITCH=1; Hm_lpvt_64ecd82404c51e03dc91cb9e8c025574=1557733619; yjs_js_security_passport=3ec7f2f0afde03f7cefaa60897f36fe62b8024b8_1557733619_js; from_lang_often=%5B%7B%22value%22%3A%22zh%22%2C%22text%22%3A%22%u4E2D%u6587%22%7D%2C%7B%22value%22%3A%22en%22%2C%22text%22%3A%22%u82F1%u8BED%22%7D%5D' }, onload: function(res) { console.log(res); parseRes(res.responseText); }, }); } function getRequest(url) { GM_xmlhttpRequest({ method: 'GET', url: url, headers: { 'referer': 'https://fanyi.baidu.com/', 'range': 'bytes=0-', 'user-agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/74.0.3729.131 Safari/537.36', 'cookie': 'locale=zh; BAIDUID=40D4B467750C147DE81CE4AC40335080:FG=1; Hm_lvt_64ecd82404c51e03dc91cb9e8c025574=1557733597; to_lang_often=%5B%7B%22value%22%3A%22en%22%2C%22text%22%3A%22%u82F1%u8BED%22%7D%2C%7B%22value%22%3A%22zh%22%2C%22text%22%3A%22%u4E2D%u6587%22%7D%5D; REALTIME_TRANS_SWITCH=1; FANYI_WORD_SWITCH=1; HISTORY_SWITCH=1; SOUND_SPD_SWITCH=1; SOUND_PREFER_SWITCH=1; Hm_lpvt_64ecd82404c51e03dc91cb9e8c025574=1557733619; yjs_js_security_passport=3ec7f2f0afde03f7cefaa60897f36fe62b8024b8_1557733619_js; from_lang_often=%5B%7B%22value%22%3A%22zh%22%2C%22text%22%3A%22%u4E2D%u6587%22%7D%2C%7B%22value%22%3A%22en%22%2C%22text%22%3A%22%u82F1%u8BED%22%7D%5D' }, responseType: 'arraybuffer', onload: function(res) { playSound(res.response); }, }); } init(); })();