Greasy Fork

Greasy Fork is available in English.

划词翻译:有道词典,金山词霸,谷歌翻译

划词翻译调用“有道词典(有道翻译)、金山词霸、谷歌翻译”

当前为 2019-01-07 提交的版本,查看 最新版本

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Greasemonkey 油猴子Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Userscripts ,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name         划词翻译:有道词典,金山词霸,谷歌翻译
// @namespace    http://tampermonkey.net/
// @version      1.0
// @description  划词翻译调用“有道词典(有道翻译)、金山词霸、谷歌翻译”
// @author       https://github.com/barrer
// @match        http://*/*
// @include      https://*/*
// @include      file:///*
// @run-at document-end
// @connect      dict.youdao.com
// @connect      open.iciba.com
// @connect      translate.google.cn
// @grant        GM_xmlhttpRequest
// ==/UserScript==

(function () {
    'use strict';

    // Your code here...
    /**样式*/
    var style = document.createElement('style');
    style.textContent = `
    * {
        word-wrap: break-word !important
    }
    
    img {
        cursor: pointer;
        display: inline-block;
        width: 22px;
        height: 22px;
        border: 1px solid #dfe1e5;
        border-radius: 22px;
        background-color: rgba(255, 255, 255, 1);
        padding: 2px;
        margin: 0;
        margin-right: 5px;
        box-sizing: content-box;
        vertical-align: middle;
    }
    
    tr-icon {
        display: none;
        position: absolute;
        padding: 0;
        margin: 0;
        cursor: move;
        box-sizing: content-box;
        font-size: 13px;
        text-align: left;
        border: 0;
        background: transparent;
        color: black;
        z-index: 2147483647;
    }
    
    tr-audio {
        display: block;
    }
    
    tr-audio a {
        color: #36f;
        text-decoration: none;
        cursor: pointer;
        margin-right: 10px;
    }
    
    tr-audio a:last-of-type {
        margin-right: auto;
    }
    
    tr-content {
        display: block;
        max-width: 400px;
        max-height: 200px;
        overflow: auto;
        border: 1px solid #dfe1e5;
        background: white;
        border-radius: 3px;
        padding: 2px 8px;
        margin-top: 5px;
        box-sizing: content-box;
        font-family: "Helvetica Neue", "Helvetica", "Arial", "sans-serif";
        font-size: 14px;
        line-height: 18px;
    }
    
    #google .sentences,
    #google .trans,
    #google .orig,
    #google .dict,
    #google .pos,
    #none {
        display: block;
    }
    
    #google .backend,
    #google .entry,
    #google .base_form,
    #google .pos_enum,
    #google .src,
    #google .confidence,
    #google .ld_result,
    #google .encodeHTML,
    #none {
        display: none;
    }
    
    #google .orig {
        font-style: italic;
        color: #777;
    }
    
    #google .pos {
        margin-top: 1em;
    }
    
    #google .pos:before {
        content: "[";
    }
    
    #google .pos:after {
        content: "]";
    }

    #google .terms:before {
        content: "【";
    }

    #google .terms:after {
        content: "】";
    }
    
    #google .terms {
        margin-right: .2em;
    }
    `;
    var link = document.createElement('link');
    link.rel = 'stylesheet';
    link.type = 'text/css';
    link.href = URL.createObjectURL(new Blob([style.textContent], {
        type: 'text/css;charset=UTF-8'
    }));
    // 图标数组
    var iconArray = [{
        name: '金山词霸',
        id: 'iciba',
        image: '',
        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: 无法连接翻译服务');
            });
        }
    }, {
        name: '有道词典',
        id: 'youdao',
        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: '谷歌翻译',
        id: 'google',
        image: '',
        trigger: function (text) {
            var url = 'https://translate.google.cn/translate_a/single?client=gtx&dt=t&dt=bd&dj=1&source=input&hl=zh-CN&sl=auto&tl=';
            if (hasChineseByRange(text))
                url += 'en&q=' + text;
            else
                url += 'zh-CN&q=' + text;
            ajax(url, function (rst) {
                var html = parseGoogle(rst);
                showContent(html);
            }, function (rst) {
                showContent('error: 无法连接翻译服务');
            });
        }
    }];
    // 翻译图标、内容面板、当前选中文本、当前翻译引擎
    var icon = document.createElement('tr-icon'),
        content = document.createElement('tr-content'),
        selected,
        engineId;
    // 绑定图标拖动事件
    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)) { // 没有拖动鼠标抬起的时候触发点击事件
                engineId = obj.id; // 翻译引擎 ID
                obj.trigger(selected); // 启动翻译引擎
            }
        });
        icon.appendChild(img);
    });
    icon.appendChild(content); // 内容面板放图标后面
    // 添加翻译图标到 DOM
    var root = document.createElement('div');
    document.documentElement.appendChild(root);
    var shadow = root.attachShadow({
        mode: 'open'
    });
    // 多种方式最大化兼容:Content Security Policy
    // https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Content-Security-Policy
    shadow.appendChild(style); // 内部样式表
    shadow.appendChild(link); // 外部样式表
    // 翻译图标加入 Shadow
    shadow.appendChild(icon);
    // 鼠标事件:防止选中的文本消失
    document.addEventListener('mousedown', function (e) {
        log('mousedown event:', e);
        if (e.target == icon || (e.target.parentNode && e.target.parentNode == icon)) { // 点击了翻译图标
            e.preventDefault();
        }
    });
    // 鼠标事件:防止选中的文本消失;显示、隐藏翻译图标
    document.addEventListener('mouseup', function (e) {
        log('mouseup event:', e);
        if (e.target == icon || (e.target.parentNode && e.target.parentNode == icon)) { // 点击了翻译图标
            e.preventDefault();
            return;
        }
        selected = window.getSelection().toString().trim(); // 当前选中文本
        log('click text:' + selected);
        if (selected && icon.style.display == 'none') {
            log('show icon');
            log(selected + ' | ' + e.pageX + ' | ' + e.pageY);
            icon.style.top = e.pageY + 10 + 'px';
            icon.style.left = e.pageX + 10 + 'px';
            icon.style.display = 'block';
            // 兼容部分 Content Security Policy
            icon.style.position = 'absolute';
            icon.style.zIndex = '2147483647';
        } else if (!selected) {
            log('hide icon');
            icon.style.display = 'none';
            content.style.display = 'none';
            // 强制设置鼠标拖动事件结束,防止由于网页本身的其它鼠标事件冲突而导致没有侦测到:mouseup
            iconDrag.dragging = false;
            iconDrag.unsetMouseMove();
        }
    });
    // 选中变化事件:当点击已经选中的文本的时候,隐藏翻译图标和翻译面板(此时浏览器动作是:选中的文本已经取消选中了)
    document.addEventListener("selectionchange", function (e) {
        log('selectionchange event:', e);
        log('selectionchange:' + window.getSelection().toString());
        if (!window.getSelection().toString().trim()) {
            icon.style.display = 'none';
            content.style.display = 'none';
        }
    });
    /**日志输出*/
    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;
    }
    /**是否包含汉字*/
    function hasChineseByRange(str) {
        for (var i = 0; i < str.length; i++) {
            if (str.charCodeAt(i) >= 0x4E00 && str.charCodeAt(i) <= 0x9FBF) {
                return true;
            }
        }
        return false;
    }
    /**对象转 xml*/
    function objToXml(obj) {
        var xml = '';
        for (var prop in obj) {
            xml += obj[prop] instanceof Array ? '' : "<" + prop + ">";
            if (obj[prop] instanceof Array) {
                for (var array in obj[prop]) {
                    xml += "<" + prop + ">";
                    xml += objToXml(new Object(obj[prop][array]));
                    xml += "</" + prop + ">";
                }
            } else if (typeof obj[prop] == "object") {
                xml += objToXml(new Object(obj[prop]));
            } else {
                xml += obj[prop];
            }
            xml += obj[prop] instanceof Array ? '' : "</" + prop + ">";
        }
        var xml = xml.replace(/<\/?[0-9]{1,}>/g, '');
        return xml
    }
    /**xml 转 html*/
    function xmlToHtml(xml, tag) {
        return xml.replace(/<([^/].+?)>/g, '<' + tag + ' class="$1">')
            .replace(/<\/(.+?)>/g, '</' + tag + '>');
    }
    /**ajax 跨域访问公共方法*/
    function ajax(url, success, error, obj) {
        if (!!!obj)
            obj = {};
        if (!!!obj.method)
            obj.method = 'GET';
        // >>>因为Tampermonkey跨域访问(a.com)时会自动携带对应域名(a.com)的对应cookie
        // 不会携带当前域名的cookie
        // 所以,GM_xmlhttpRequest【不存在】cookie跨域访问安全性问题
        // 以下设置默认headers不起作用<<<
        if (!!!obj.headers)
            obj.headers = {
                'cookie': ''
            };
        GM_xmlhttpRequest({
            method: obj.method,
            url: url,
            headers: obj.headers,
            responseType: obj.responseType,
            data: obj.data,
            onload: function (res) {
                success(res.responseText, res, obj);
            },
            onerror: function (res) {
                error(res.responseText, res, obj);
            }
        });
    }
    /**显示内容面板*/
    function showContent(html) {
        // 发音
        var audio = document.createElement('tr-audio'),
            us = document.createElement('a'),
            uk = document.createElement('a');

        us.innerHTML = '♪US';
        us.setAttribute('href', 'javascript:void(0)');
        us.addEventListener('click', playUS);
        // us.addEventListener('mouseover', playUS);
        uk.innerHTML = '♪UK';
        uk.setAttribute('href', 'javascript:void(0)');
        uk.addEventListener('click', playUK);
        // uk.addEventListener('mouseover', playUK);
        audio.appendChild(us);
        audio.appendChild(uk);
        // 翻译内容
        content.innerHTML = '<div id="' + engineId + '">' + html + '</div>';
        if (engineId != 'google') { // 谷歌翻译不显示发音图标
            content.insertBefore(audio, content.childNodes[0]);
        }
        content.style.display = 'block';
    }
    /**美式发音*/
    function playUS() {
        var url = 'http://dict.youdao.com/dictvoice?audio=' + selected + '&type=2';
        var audio = new Audio();
        ajax(url, function (rst, res) {
            audio.src = URL.createObjectURL(res.response);
            audio.play();
        }, function (rst) {
            log(rst);
        }, {
            responseType: 'blob'
        });
    }
    /**英式发音*/
    function playUK() {
        var url = 'http://dict.youdao.com/dictvoice?audio=' + selected + '&type=1';
        var audio = new Audio();
        ajax(url, function (rst, res) {
            audio.src = URL.createObjectURL(res.response);
            audio.play();
        }, function (rst) {
            log(rst);
        }, {
            responseType: 'blob'
        });
    }
    /**有道词典排版*/
    function parseYoudao(rst) {
        try {
            // if (true) return xmlToHtml(objToXml(JSON.parse(rst)), 'span');
            var rstJson = JSON.parse(rst),
                html = '',
                phoneStyle = 'color:#777;';
            if (rstJson.ec) {
                var word = rstJson.ec.word[0],
                    tr = '';
                var trs = word.trs,
                    ukphone = word.ukphone,
                    usphone = word.usphone,
                    phone = word.phone;
                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.ce_new && rstJson.ce_new.word) {
                rstJson.ce_new.word.forEach(function (w) {
                    if (w.phone)
                        html += '<span style="' + phoneStyle + '">[' + w.phone + '] </span><br>';
                });
            }
            // 长句翻译
            if (rstJson.fanyi && rstJson.fanyi.tran) {
                html += rstJson.fanyi.tran;
            }
            return html;
        } catch (error) {
            log(error);
            return error;
        }
    }
    /**金山词霸排版*/
    function parseIciba(rst) {
        try {
            rst = rst.replace(/class=\\"icIBahyI-prons\\"/g, '__mystyle__') // 音标
                .replace(/\\"/g, '"') // 引号
                // A标签
                .replace(/<a([^>]*)?>详细释义<\/a([^>]*)?>/g, '')
                .replace(/<a([^>]*)?>/g, '')
                .replace(/<\/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, ' ')
                // 音标
                .replace(/__mystyle__/g, ' style="color:#777;"');
            var match = /dict.innerHTML='(.*)?';/g.exec(rst);
            return match[1];
        } catch (error) {
            log(error);
            return error;
        }
    }
    /**谷歌翻译排版*/
    function parseGoogle(rst) {
        try {
            return xmlToHtml(objToXml(JSON.parse(rst)), 'span');
        } catch (error) {
            log(error);
            return error;
        }
    }
})();