tampermonkey-script/translate/translate-dictionary.js
2019-01-04 19:11:22 +08:00

298 lines
17 KiB
JavaScript
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

// ==UserScript==
// @name 划词翻译:有道词典,金山词霸
// @namespace http://tampermonkey.net/
// @version 0.2
// @description 划词翻译调用“有道词典(有道翻译)、金山词霸”
// @author https://github.com/barrer
// @match http://*/*
// @include https://*/*
// @include file:///*
// @run-at document-end
// @connect dict.youdao.com
// @connect open.iciba.com
// @grant GM_xmlhttpRequest
// ==/UserScript==
(function () {
'use strict';
// Your code here...
var iconArray = [{
name: '有道词典',
image: '',
trigger: function (text) {
ajax('http://dict.youdao.com/jsonapi?xmlVersion=5.1&jsonversion=2&q=' + text, function (rst) {
var html = parseYoudao(rst);
if (text.toLowerCase() != text) { // 再次翻译一遍小写的
ajax('http://dict.youdao.com/jsonapi?xmlVersion=5.1&jsonversion=2&q=' + text.toLowerCase(), function (rst) {
var reHtml = parseYoudao(rst);
if (html !== reHtml) {
log(html, reHtml);
html += '<hr>' + reHtml;
}
showContent(html);
}, function (rst) {
showContent(html + '<hr>' + 'error: 无法连接翻译服务');
});
} else {
showContent(html);
}
}, function (rst) {
showContent('error: 无法连接翻译服务');
});
}
}, {
name: '金山词霸',
image: '',
host: ['www.iciba.com'],
trigger: function (text) {
ajax('http://open.iciba.com/huaci_v3/dict.php?word=' + text, function (rst) {
var html = parseIciba(rst);
if (text.toLowerCase() != text) { // 再次翻译一遍小写的
ajax('http://open.iciba.com/huaci_v3/dict.php?word=' + text.toLowerCase(), function (rst) {
var reHtml = parseIciba(rst);
if (html !== reHtml) {
log(html, reHtml);
html += '<hr>' + reHtml;
}
showContent(html);
}, function (rst) {
showContent(html + '<hr>' + 'error: 无法连接翻译服务');
});
} else {
showContent(html);
}
}, function (rst) {
showContent('error: 无法连接翻译服务');
});
}
}];
// 翻译图标、内容面板
var icon = document.createElement('div'),
content = document.createElement('div');
// 绑定图标拖动事件
var iconDrag = new Drag(icon);
iconArray.forEach(function (obj) {
var img = document.createElement('img');
img.setAttribute('src', obj.image);
img.setAttribute('alt', obj.name);
img.setAttribute('title', obj.name);
img.addEventListener('mouseup', function () {
if (iconDrag.elementOriginalLeft == parseInt(icon.style.left) &&
iconDrag.elementOriginalTop == parseInt(icon.style.top)) // 没有拖动鼠标抬起的时候触发点击事件
obj.trigger(window.getSelection().toString().trim()); // 启动翻译引擎
});
img.setAttribute('style', '' +
'cursor:pointer!important;' +
'display:inline-block!important;' +
'width:22px!important;' +
'height:22px!important;' +
'border:1px solid #FFCC66!important;' +
'border-radius:22px!important;' +
'background-color:rgba(255,255,255,1)!important;' +
'padding:2px!important;' +
'margin:0!important;' +
'margin-right:5px!important;' +
'box-sizing:content-box!important;' +
'vertical-align:middle!important;' +
'');
icon.appendChild(img);
});
icon.appendChild(content); // 内容放图标后面
icon.setAttribute('style', '' +
'display:none!important;' +
'position:absolute!important;' +
'padding:0!important;' +
'margin:0!important;' +
'box-sizing:content-box!important;' +
'font-size:13px!important;' +
'text-align:left!important;' +
'border:0!important;' +
'background:transparent!important;' +
'z-index:2147483647!important;' +
'');
content.setAttribute('style', '' +
'border: 1px solid #FFCC66!important;' +
'background: white!important;' +
'border-radius: 3px!important;' +
'padding: 2px 8px!important;' +
'margin-top:5px!important;' +
'box-sizing:content-box!important;' +
'font-family:"Helvetica Neue","Helvetica","Arial","sans-serif"!important;' +
'font-size:14px!important;' +
'line-height:18px!important;' +
'');
// 添加翻译图标到 DOM
document.documentElement.appendChild(icon);
// 鼠标事件:防止选中的文本消失
document.addEventListener('mousedown', function (e) {
if (e.target == icon || (e.target.parentNode && e.target.parentNode == icon)) { // 点击了翻译图标
e.preventDefault();
}
});
// 选中变化事件:当点击已经选中的文本的时候,隐藏翻译图标和翻译面板(此时浏览器动作是:选中的文本已经取消选中了)
document.addEventListener("selectionchange", function () {
log('selectionchange:' + window.getSelection().toString());
if (!window.getSelection().toString().trim()) {
icon.style.display = 'none';
}
});
// 鼠标事件:防止选中的文本消失;显示、隐藏翻译图标
document.addEventListener('mouseup', function (e) {
if (e.target == icon || (e.target.parentNode && e.target.parentNode == icon)) { // 点击了翻译图标
e.preventDefault();
return;
}
var text = window.getSelection().toString().trim();
log('click text:' + text);
log(e);
if (text && icon.style.display == 'none') {
log('show icon');
log(text + ' | ' + e.pageX + ' | ' + e.pageY);
icon.style.top = e.pageY + 10 + 'px';
icon.style.left = e.pageX + 10 + 'px';
content.style.display = 'none'; // 内容先隐藏,因为还没有点击翻译引擎
icon.style.display = 'block';
} else if (!text) {
log('hide icon');
icon.style.display = 'none';
// 强制设置鼠标拖动事件结束防止由于网页本身的其它鼠标事件冲突而导致没有侦测到mouseup
iconDrag.dragging = false;
iconDrag.unsetMouseMove();
}
});
/**日志输出*/
function log() {
var debug = false;
if (!debug)
return;
if (arguments) {
for (var i = 0; i < arguments.length; i++) {
console.log(arguments[i]);
}
}
}
/**鼠标拖动*/
function Drag(element) {
this.dragging = false;
this.mouseDownPositionX = 0;
this.mouseDownPositionY = 0;
this.elementOriginalLeft = 0;
this.elementOriginalTop = 0;
var ref = this;
this.startDrag = function (e) {
e.preventDefault();
ref.dragging = true;
ref.mouseDownPositionX = e.clientX;
ref.mouseDownPositionY = e.clientY;
ref.elementOriginalLeft = parseInt(element.style.left);
ref.elementOriginalTop = parseInt(element.style.top);
// set mousemove event
window.addEventListener('mousemove', ref.dragElement);
log('startDrag');
};
this.unsetMouseMove = function () {
// unset mousemove event
window.removeEventListener('mousemove', ref.dragElement);
};
this.stopDrag = function (e) {
e.preventDefault();
ref.dragging = false;
ref.unsetMouseMove();
log('stopDrag');
};
this.dragElement = function (e) {
log('dragging');
if (!ref.dragging)
return;
e.preventDefault();
// move element
element.style.left = ref.elementOriginalLeft + (e.clientX - ref.mouseDownPositionX) + 'px';
element.style.top = ref.elementOriginalTop + (e.clientY - ref.mouseDownPositionY) + 'px';
log('dragElement');
};
element.onmousedown = this.startDrag;
element.onmouseup = this.stopDrag;
}
/**ajax 跨域访问公共方法*/
function ajax(url, success, error, element, method, data, headers) {
if (!!!method)
method = 'GET';
// >>>因为Tampermonkey跨域访问(a.com)时会自动携带对应域名(a.com)的对应cookie
// 不会携带当前域名的cookie
// 所以GM_xmlhttpRequest【不存在】cookie跨域访问安全性问题
// 以下设置默认headers不起作用<<<
if (!!!headers)
headers = {
'cookie': ''
};
GM_xmlhttpRequest({
method: method,
url: url,
headers: headers,
data: data,
onload: function (res) {
success(res.responseText, element);
},
onerror: function (res) {
error(res.responseText, element);
}
});
}
/**显示内容面板*/
function showContent(html) {
content.innerHTML = html;
content.style.display = 'block';
}
/**有道词典排版*/
function parseYoudao(rst) {
try {
var rstJson = JSON.parse(rst),
html = '';
if (rstJson.ec) {
var word = rstJson.ec.word[0],
tr = '';
var trs = word.trs,
ukphone = word.ukphone,
usphone = word.usphone,
phone = word.phone;
var phoneStyle = 'color:#777!important;';
if (ukphone && ukphone.length != 0) {
html += '<span style="' + phoneStyle + '">英[' + ukphone + '] </span>';
}
if (usphone && usphone.length != 0) {
html += '<span style="' + phoneStyle + '">美[' + usphone + '] </span>';
}
if (html.length != 0) {
html += '<br>';
} else if (phone && phone.length != 0) {
html += '<span style="' + phoneStyle + '">[' + phone + '] </span><br>';
}
trs.forEach(element => {
tr += element.tr[0].l.i[0] + '<br>';
});
html += tr;
}
if (rstJson.fanyi && rstJson.fanyi.tran) {
html += rstJson.fanyi.tran;
}
return html;
} catch (error) {
log(error);
return error;
}
}
/**金山词霸排版*/
function parseIciba(rst) {
rst = rst.replace(/\\"/g, '"')
.replace(/<a.*?<\/a>/g, '')
.replace(/(?:class|id|style|xml:lang|lang)=\"([^"]*)\"/g, '')
.replace(/(?:label>|strong>)/g, 'span>')
.replace(/(?:<label|<strong)/g, '<span')
.replace(/(?:p>)/g, 'div>')
.replace(/[ ]+/g, ' ');
var match = /dict.innerHTML='(.*)?';/g.exec(rst);
return match[1];
}
})();