diff --git a/README.md b/README.md
index 4a26ed7..f368dab 100644
--- a/README.md
+++ b/README.md
@@ -1,8 +1,15 @@
## Tampermonkey油猴子脚本
**部分预览图:**
+
+### translate.js
+
![translate](translate/translate.png)
+### translate-dictionary.js
+
+![translate](translate/translate-dictionary.png)
+
**脚本下载地址:**
[https://greasyfork.org/](https://greasyfork.org/)
diff --git a/translate/translate-dictionary.js b/translate/translate-dictionary.js
new file mode 100644
index 0000000..e77bf9a
--- /dev/null
+++ b/translate/translate-dictionary.js
@@ -0,0 +1,298 @@
+// ==UserScript==
+// @name 划词翻译:有道词典,金山词霸
+// @namespace http://tampermonkey.net/
+// @version 0.1
+// @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 += '
' + reHtml;
+ }
+ showContent(html);
+ }, function (rst) {
+ showContent(html + '
' + '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 += '
' + reHtml;
+ }
+ showContent(html);
+ }, function (rst) {
+ showContent(html + '
' + '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 += '英[' + ukphone + '] ';
+ }
+ if (usphone && usphone.length != 0) {
+ html += '美[' + usphone + '] ';
+ }
+ if (html.length != 0) {
+ html += '
';
+ } else if (phone && phone.length != 0) {
+ html += '[' + phone + ']
';
+ }
+ trs.forEach(element => {
+ tr += element.tr[0].l.i[0] + '
';
+ });
+ 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(//g, '')
+ .replace(/(?:class|id|style|xml:lang|lang)=\"([^"]*)\"/g, '')
+ .replace(/(?:label>|strong>)/g, 'span>')
+ .replace(/(?: