Greasy Fork

Greasy Fork is available in English.

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

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

目前为 2019-01-06 提交的版本,查看 最新版本

// ==UserScript==
// @name         划词翻译:有道词典,金山词霸
// @namespace    http://tampermonkey.net/
// @version      0.6
// @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: '',
        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: 无法连接翻译服务');
            });
        }
    }, {
        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: 无法连接翻译服务');
            });
        }
    }];
    // 翻译图标、内容面板、当前选中
    var icon = document.createElement('div'),
        content = document.createElement('div'),
        text;
    // 绑定图标拖动事件
    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)) { // 没有拖动鼠标抬起的时候触发点击事件
                text = window.getSelection().toString().trim();
                obj.trigger(text); // 启动翻译引擎
            }
        });
        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;' +
        'cursor:move;!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, 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('div'),
            us = document.createElement('a'),
            uk = document.createElement('a'),
            space = document.createElement('span'),
            style = '';
        style += 'color:#36f!important;';
        style += 'text-decoration:none!important;';
        style += 'cursor:pointer!important;';
        us.innerHTML = '♪US';
        us.setAttribute('href', 'javascript:void(0)');
        us.addEventListener('click', playUS);
        // us.addEventListener('mouseover', playUS);
        us.setAttribute('style', style);
        uk.innerHTML = '♪UK';
        uk.setAttribute('href', 'javascript:void(0)');
        uk.addEventListener('click', playUK);
        // uk.addEventListener('mouseover', playUK);
        uk.setAttribute('style', style);
        space.innerHTML = '&nbsp;&nbsp;';
        audio.appendChild(us);
        audio.appendChild(space);
        audio.appendChild(uk);
        // 翻译内容
        content.innerHTML = html;
        content.insertBefore(audio, content.childNodes[0]);
        content.style.display = 'block';
    }
    /**美式发音*/
    function playUS() {
        var url = 'http://dict.youdao.com/dictvoice?audio=' + text + '&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=' + text + '&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 {
            var rstJson = JSON.parse(rst),
                html = '',
                phoneStyle = 'color:#777!important;';
            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) {
        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!important;"');
        var match = /dict.innerHTML='(.*)?';/g.exec(rst);
        return match[1];
    }
})();