Greasy Fork is available in English.

Translate

划词翻译调用“金山词霸、有道词典(有道翻译)、Google Translate(谷歌翻译)、沪江小D、搜狗翻译、必应词典(必应翻译)、Microsoft Translator(必应在线翻译)、DeepL翻译、海词词典、百度翻译、Oxford Learner's Dictionaries、Oxford Dictionaries、Merriam-Webster、PDF 划词翻译、Google Search、Bing Search(必应搜索)、百度搜索、Wikipedia Search(维基百科搜索)”网页翻译

// ==UserScript==
// @name         Translate
// @namespace    http://tampermonkey.net/
// @version      10.15
// @description  划词翻译调用“金山词霸、有道词典(有道翻译)、Google Translate(谷歌翻译)、沪江小D、搜狗翻译、必应词典(必应翻译)、Microsoft Translator(必应在线翻译)、DeepL翻译、海词词典、百度翻译、Oxford Learner's Dictionaries、Oxford Dictionaries、Merriam-Webster、PDF 划词翻译、Google Search、Bing Search(必应搜索)、百度搜索、Wikipedia Search(维基百科搜索)”网页翻译
// @author       https://github.com/barrer
// @license      https://www.apache.org/licenses/LICENSE-2.0
// @match        http://*/*
// @include      https://*/*
// @include      file:///*
// @grant        GM_setValue
// @grant        GM_getValue
// @grant        GM_deleteValue
// ==/UserScript==

/*
 * Copyright 2017-2023 https://github.com/barrer.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      https://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

(() => {
    'use strict';

    // Your code here...
    // 注意:自定义修改后把 “@version” 版本号改为 “10000” 防止自动更新
    /**样式*/
    const style = document.createElement('style');
    const zIndex = '2147473647'; // 渲染图层
    style.textContent = `
    :host{all:unset!important}
    :host{all:initial!important}
    *{word-wrap:break-word!important}
    img{cursor:pointer;display:inline-block;width:20px;height:20px;border:1px solid #dfe1e5;border-radius:4px;background-color:rgba(255,255,255,1);padding:2px;margin:0;margin-right:5px;box-sizing:content-box;vertical-align:middle}
    img:last-of-type{margin-right:auto}
    img:hover{border:1px solid #ff9900}
    img[is-more]{display:none}
    tr-icon{display:none;position:absolute;padding:0;margin:0;cursor:move;background:transparent;box-sizing:content-box;font-size:13px;text-align:left;border:0;z-index:${zIndex}}
    `;
    // iframe 工具库
    const iframe = document.createElement('iframe');
    let iframeWin = null;
    let iframeDoc = null;
    iframe.style.display = 'none';
    const gm = {
        TEXT: 'barrer.translate.data.transfer.text',
        REDIRECT_URL: 'barrer.translate.data.transfer.redirect_url',
        HIDE: 'barrer.translate.data.config.hide',
        SORT: 'barrer.translate.data.config.sort',
        reset() {
            GM_deleteValue(this.TEXT);
            GM_deleteValue(this.REDIRECT_URL);
            GM_deleteValue(this.HIDE);
            GM_deleteValue(this.SORT);
        },
        set(key, value) {
            GM_setValue(key, value);
        },
        get(key, myDefault) {
            const value = GM_getValue(key);
            return isNotNull(value) || !isNotNull(myDefault) ? value : myDefault;
        }
    };
    const dataTransfer = {
        beforePopup(popup, id) {
            const text = window.getSelection().toString().trim();
            if (!(id in beforePopupDoNotSetValueMaps)) {
                gm.set(gm.TEXT, text);
            }
            popup(text);
        },
        beforeCustom(custom) {
            const text = gm.get(gm.TEXT, '');
            gm.set(gm.TEXT, '');
            custom.forEach(cus => {
                cus(text);
            });
        }
    };
    const iconArray = [{
        name: 'Bing 词典',
        id: 'bingDict',
        image: '',
        host: ['www.bing.com', 'cn.bing.com'],
        popup(text) {
            popupCenter(`https://www.bing.com/dict/search?q=${encodeURIComponent(text)}`, null, 800, screen.height);
        },
        custom(text) { }
    },
    {
        name: '有道词典',
        id: 'youdao',
        image: '',
        host: ['dict.youdao.com'],
        popup(text) {
            popupCenter(`https://dict.youdao.com/w/eng/${encodeURIComponent(text)}`, null, 800, screen.height);
        },
        custom(text) { }
    },
    {
        name: 'More...',
        id: 'more',
        image: '',
        host: ['more.example.com'],
        popup(text) {
            icon.querySelectorAll('img[is-more]').forEach(ele => {
                if (ele.style.display == 'inline-block') {
                    ele.style.display = 'none';
                } else {
                    ele.style.display = 'inline-block';
                }
            });
        },
        custom(text) { }
    },
    {
        name: 'Vocabulary.com',
        id: 'vocabulary',
        image: '',
        host: ['www.vocabulary.com'],
        popup(text) {
            popupCenter(`https://www.vocabulary.com/dictionary/${encodeURIComponent(text)}`, null, 800, screen.height);
        },
        custom(text) { }
    },
    {
        name: 'Google Translate',
        id: 'googleCom',
        image: '',
        host: ['translate.google.com'],
        popup(text) {
            popupCenter('https://translate.google.com', null, 800, screen.height);
        },
        custom(text) {
            const source = document.querySelectorAll('textarea')[0];
            source.value = text;
            triggerEvent(source, 'input');
            triggerEvent(source, 'keyup');
        }
    },
    {
        name: 'DeepL翻译',
        id: 'deepl',
        image: '',
        host: ['www.deepl.com'],
        popup(text) {
            if (hasChineseByRange(text)) {
                popupCenter(`https://www.deepl.com/translator#zh/en/${encodeURIComponent(text)}`, null, 800, screen.height);
            } else {
                popupCenter(`https://www.deepl.com/translator#en/zh/${encodeURIComponent(text)}`, null, 800, screen.height);
            }
        },
        custom(text) {
            const source = document.querySelector('textarea');
            source.value = text;
            triggerEvent(source, 'change');
        }
    },
    {
        name: 'Microsoft Translator',
        id: 'bingTrans',
        image: '',
        host: ['www.bing.com', 'cn.bing.com'],
        popup(text) {
            popupCenter('https://www.bing.com/translator', null, 1024, screen.height);
        },
        custom(text) {
            if (window.location.href.includes('bing.com/translator')) {
                if (hasChineseByRange(text)) {
                    document.querySelector('#tta_tgtsl').value = 'en';
                } else {
                    document.querySelector('#tta_tgtsl').value = 'zh-Hans';
                }
                const source = document.querySelector('textarea');
                source.value = `${text}←`;
            }
        }
    },
    {
        name: '搜狗翻译',
        id: 'sougou',
        image: '',
        host: ['fanyi.sogou.com'],
        popup(text) {
            popupCenter(`https://fanyi.sogou.com/?keyword=${encodeURIComponent(text)}`, null, 1024, screen.height);
        },
        custom(text) { }
    },
    {
        name: '百度翻译',
        id: 'baidu',
        image: '',
        host: ['fanyi.baidu.com'],
        popup(text) {
            popupCenter(`https://fanyi.baidu.com/#en/zh/${encodeURIComponent(text)}`, null, 1024, screen.height);
        },
        custom(text) { }
    },
    {
        name: '金山词霸',
        id: 'iciba',
        image: '',
        host: ['www.iciba.com'],
        popup(text) {
            popupCenter(`https://www.iciba.com/word?w=${encodeURIComponent(text)}`, null, 800, screen.height);
        },
        custom(text) { }
    },
    {
        name: '海词词典',
        id: 'haici',
        image: '',
        host: ['dict.cn', 'hanyu.dict.cn', 'abbr.dict.cn', 'ename.dict.cn'],
        popup(text) {
            popupCenter(`https://dict.cn/${encodeURIComponent(text)}`, null, 800, screen.height);
        },
        custom(text) { }
    },
    {
        name: '沪江小D',
        id: 'hjenglish',
        image: '',
        host: ['dict.hjenglish.com'],
        popup(text) {
            popupCenter(`https://dict.hjenglish.com/w/${encodeURIComponent(text)}`, null, 1024, screen.height);
        },
        custom(text) {
            document.querySelectorAll('.word-details-button-expand').forEach(ele => {
                ele.click();
            });
        }
    },
    {
        name: 'Oxford Learner\'s Dictionaries',
        id: 'oald',
        image: '',
        host: ['www.oxfordlearnersdictionaries.com'],
        popup(text) {
            popupCenter(`https://www.oxfordlearnersdictionaries.com/search/english/?q=${encodeURIComponent(text)}`, null, 800, screen.height);
        },
        custom(text) { }
    },
    {
        name: 'Oxford Dictionaries',
        id: 'ode',
        image: '',
        host: ['ode.example.com'],
        popup(text) {
            popupCenter(`https://www.google.com/search?q=${encodeURIComponent(text)} meaning`, null, 800, screen.height);
        },
        custom(text) { }
    },
    {
        name: 'Merriam-Webster',
        id: 'mwcd',
        image: '',
        host: ['www.merriam-webster.com'],
        popup(text) {
            popupCenter(`https://www.merriam-webster.com/dictionary/${encodeURIComponent(text)}`, null, 800, screen.height);
        },
        custom(text) { }
    },
    {
        name: 'PDF 划词翻译',
        id: 'pdf',
        image: '',
        host: ['pdf.example.com'],
        popup(text) {
            openInNewTab('https://barrer.github.io/tools/pdfjs/web/viewer.html');
        },
        custom(text) { }
    },
    {
        name: 'Google',
        id: 'googleSearch',
        image: '',
        host: ['www.google.com'],
        popup(text) {
            popupCenter(`https://www.google.com/search?q=${encodeURIComponent(text)}`, null, 800, screen.height);
        },
        custom(text) { }
    },
    {
        name: 'Bing',
        id: 'bingSearch',
        image: '',
        host: ['www.bing.com', 'cn.bing.com'],
        popup(text) {
            popupCenter(`https://www.bing.com/search?q=${encodeURIComponent(text)}`, null, 800, screen.height);
        },
        custom(text) { }
    },
    {
        name: '百度',
        id: 'baiduSearch',
        image: '',
        host: ['www.baidu.com'],
        popup(text) {
            popupCenter(`https://www.baidu.com/s?wd=${encodeURIComponent(text)}`, null, 800, screen.height);
        },
        custom(text) { }
    },
    {
        name: 'Wikipedia',
        id: 'wikipediaSearch',
        image: '',
        host: ['en.wikipedia.org'],
        popup(text) {
            popupCenter(`https://en.wikipedia.org/wiki/${encodeURIComponent(text)}`, null, 800, screen.height);
        },
        custom(text) { }
    },
    {
        name: 'Settings',
        id: 'settings',
        image: '',
        host: ['example.com'],
        popup(text) {
            popupCenter('https://example.com', null, 800, screen.height);
        },
        custom(text) {
            settings();
        }
    }
    ];
    const hostCustomMap = {};// {host: [method, ...]}
    const customMadeIconArray = getCustomMadeIconArray(true);
    // id、host 唯一性校验
    const idMaps = {};
    const hostMaps = {};
    // 指定 id 在弹窗方法调用时不存储选中文字
    const beforePopupDoNotSetValueMaps = { more: 'more' };
    customMadeIconArray.forEach(({ id, host }) => {
        if (id in idMaps) {
            alert(`Duplicate Id: ${id}`);
        } else {
            idMaps[id] = id;
        }
        host.forEach(host => {
            if (host in hostMaps) {
                log(`Duplicate Host: ${host}`);
            } else {
                hostMaps[host] = host;
            }
        });
    });
    log('idMaps:', idMaps, 'hostMaps:', hostMaps);
    // 初始化 hostCustomMap
    customMadeIconArray.forEach(({ host, custom }) => {
        host.forEach(host => { // 赋值DOM加载后的自定义方法Map
            if (host in hostCustomMap) {
                hostCustomMap[host].push(custom);
            } else {
                hostCustomMap[host] = [custom];
            }
        });
    });
    log('hostCustomMap:', hostCustomMap);
    const icon = document.createElement('tr-icon');// 翻译图标
    let selected;// 当前选中文本
    let pageX;// 图标显示的 X 坐标
    let pageY;// 图标显示的 Y 坐标
    // 绑定图标拖动事件
    const iconDrag = new Drag(icon);
    const dragFluctuation = 16;// 当拖动多少像素以上时不触发查询
    // 翻译引擎添加到图标
    let isIconImgMore = false;
    customMadeIconArray.forEach(({ image, name, popup, id }) => {
        const img = document.createElement('img');
        img.setAttribute('src', image);
        img.setAttribute('alt', name);
        img.setAttribute('title', name);
        img.addEventListener('mouseup', () => {
            if (!isDrag(dragFluctuation)) {
                dataTransfer.beforePopup(popup, id);
            }
        });
        if (isIconImgMore) {
            img.setAttribute('is-more', 'true');
        }
        if (id == 'more') {
            isIconImgMore = true;
        }
        icon.appendChild(img);
    });
    // 翻译图标添加到 DOM
    const root = document.createElement('div');
    document.documentElement.appendChild(root);
    const shadow = root.attachShadow({
        mode: 'closed'
    });
    // iframe 工具库加入 Shadow
    shadow.appendChild(iframe);
    iframeWin = iframe.contentWindow;
    iframeDoc = iframe.contentDocument;
    // 外部样式表
    const link = document.createElement('link');
    link.rel = 'stylesheet';
    link.type = 'text/css';
    link.href = createObjectURLWithTry(new Blob(['\ufeff', style.textContent], {
        type: 'text/css;charset=UTF-8'
    }));
    shadow.appendChild(style); // 内部样式表
    shadow.appendChild(link); // 外部样式表
    adoptedStyleSheets(shadow, style.textContent); // CSSStyleSheet 样式
    shadow.appendChild(icon); // 翻译图标加入 Shadow
    // 重定向前隐藏页面主体
    if (gm.get(gm.REDIRECT_URL, '') && window.location.host == 'example.com') {
        document.documentElement.style.display = 'none';
    }
    // 重定向
    const redirect_url = gm.get(gm.REDIRECT_URL, '');
    log(`redirect_url:${redirect_url}`);
    if (redirect_url && window.location.host == 'example.com') {
        document.documentElement.style.display = 'none';
        let a = document.createElement('a');
        a.setAttribute('id', 'redirect_url');
        a.setAttribute('rel', 'noreferrer noopener');
        a.setAttribute('href', redirect_url);
        a.appendChild(document.createTextNode(redirect_url));
        document.body.appendChild(a);
        gm.set(gm.REDIRECT_URL, '');
        a.click();
        return;
    }
    // 弹出后的新页面判断是否进行自动化处理
    const text = gm.get(gm.TEXT, '');
    log(`${gm.TEXT}: ${text}`);
    log(`url: ${window.location.href}`);
    log(`host: ${window.location.host}`);
    if (text && window.location.host in hostCustomMap) {
        dataTransfer.beforeCustom(hostCustomMap[window.location.host]);
    }
    // 鼠标事件:防止选中的文本消失
    document.addEventListener('mousedown', e => {
        if (e.target == icon || (e.target.parentNode && e.target.parentNode == icon)) { // 点击了翻译图标
            e.preventDefault();
        }
    });
    // 鼠标事件:防止选中的文本消失;显示、隐藏翻译图标
    document.addEventListener('mouseup', showIcon);
    // 选中变化事件
    document.addEventListener('selectionchange', showIcon);
    document.addEventListener('touchend', showIcon);
    /**日志输出*/
    function log(...args) {
        const debug = false;
        if (!debug) {
            return;
        }
        if (args) {
            for (let i = 0; i < args.length; i++) {
                console.log(args[i]);
            }
        }
    }
    /**是否非空*/
    function isNotNull(obj) {
        return (obj != undefined && obj != null) || false;
    }
    /**转 int*/
    function myParseInt(str, myDefault) {
        const rst = parseInt(str);
        return isNaN(rst) ? (isNotNull(myDefault) ? myDefault : 0) : rst;
    }
    /**数组移动*/
    function arrayMove(arr, oldIndex, newIndex) {
        if (oldIndex < 0 || oldIndex >= arr.length || newIndex < 0 || newIndex >= arr.length) {
            return arr;
        }
        if (newIndex >= arr.length) {
            let k = newIndex - arr.length + 1;
            while (k--) {
                arr.push(undefined);
            }
        }
        arr.splice(newIndex, 0, arr.splice(oldIndex, 1)[0]);
        return arr;
    }
    /**带异常处理的 createObjectURL*/
    function createObjectURLWithTry(blob) {
        try {
            return iframeWin.URL.createObjectURL(blob);
        } catch (error) {
            log(error);
        }
        return '';
    }
    /**触发事件*/
    function triggerEvent(el, type) {
        if ('createEvent' in document) { // modern browsers, IE9+
            let e = document.createEvent('HTMLEvents');
            e.initEvent(type, true, true); // event.initEvent(type, bubbles, cancelable);
            el.dispatchEvent(e);
        } else { // IE 8
            let e = document.createEventObject();
            e.eventType = type;
            el.fireEvent(`on${e.eventType}`, e);
        }
    }
    /**弹出居中窗口*/
    function popupCenter(url, title, w, h) {
        const transfer = 'https://example.com';
        title = isNotNull(title) ? title : '_blank';// 留空默认“_self”,会复用当前弹出窗口
        w = w > screen.availWidth ? screen.availWidth : w;
        h = h > screen.availHeight ? screen.availHeight : h;
        let x = screen.availWidth / 2 - w / 2;
        let y = screen.availHeight / 2 - h / 2;
        x = x < 0 ? 0 : x;
        y = y < 0 ? 0 : y;
        let win;
        try {
            win = window.open('', title, `scrollbars=yes, width=${w}, height=${h}, top=${y}, left=${x}`);
            win.opener = null;
            let a = win.document.createElement('a');
            a.setAttribute('id', 'redirect_url');
            a.setAttribute('rel', 'noreferrer noopener');
            a.setAttribute('href', url);
            win.document.body.appendChild(a);
            a.click();
        } catch (e) {
            try {
                win.close();
            } catch (ce) {
                log(ce);
            }
            gm.set(gm.REDIRECT_URL, url);
            win = window.open(transfer, title, `scrollbars=yes, width=${w}, height=${h}, top=${y}, left=${x}, noopener, noreferrer`);
            log(e);
        }
        if (window.focus) {
            win.focus();
        }
        return win;
    }
    /**打开新的标签页*/
    function openInNewTab(url) {
        const a = document.createElement('a');
        a.setAttribute('target', '_blank');
        a.setAttribute('rel', 'noreferrer noopener'); // document.referrer, window.opener
        a.setAttribute('href', url);
        a.style.display = 'none';
        shadow.appendChild(a);
        a.click();
        a.remove();
    }
    /**是否包含汉字*/
    function hasChineseByRange(str) {
        return /[\u4e00-\u9fa5]/ig.test(str);
    }
    /**解决 Content-Security-Policy 样式文件加载问题(Chrome 实验功能)*/
    function adoptedStyleSheets(bindDocumentOrShadowRoot, cssText) {
        try {
            if (bindDocumentOrShadowRoot.adoptedStyleSheets) {
                cssText = cssText.replace(/\/\*.*?\*\//ig, ''); // remove CSS comments
                const cssSheet = new CSSStyleSheet();
                const styleArray = cssText.split('\n');
                for (let i = 0; i < styleArray.length; i++) {
                    const line = styleArray[i].trim();
                    if (line.length > 0) {
                        cssSheet.insertRule(line);
                    }
                }
                bindDocumentOrShadowRoot.adoptedStyleSheets = [cssSheet];
            }
        } catch (error) {
            log(error);
        }
    }
    /**鼠标拖动*/
    function Drag(element) {
        this.dragging = false;
        this.startDragTime = 0;
        this.stopDragTime = 0;
        this.mouseDownPositionX = 0;
        this.mouseDownPositionY = 0;
        this.elementOriginalLeft = parseInt(element.style.left);
        this.elementOriginalTop = parseInt(element.style.top);
        this.backAndForthLeftMax = 0;
        this.backAndForthTopMax = 0;
        const ref = this;
        this.startDrag = e => {
            e.preventDefault();
            ref.dragging = true;
            ref.startDragTime = new Date().getTime();
            ref.mouseDownPositionX = e.clientX;
            ref.mouseDownPositionY = e.clientY;
            ref.elementOriginalLeft = parseInt(element.style.left);
            ref.elementOriginalTop = parseInt(element.style.top);
            ref.backAndForthLeftMax = 0;
            ref.backAndForthTopMax = 0;
            // set global mouse events
            window.addEventListener('mousemove', ref.dragElement);
            window.addEventListener('mouseup', ref.stopDrag);
            log('startDrag');
        };
        this.unsetMouseMove = () => {
            // unset global mouse events
            window.removeEventListener('mousemove', ref.dragElement);
            window.removeEventListener('mouseup', ref.stopDrag);
        };
        this.stopDrag = e => {
            e.preventDefault();
            ref.dragging = false;
            ref.stopDragTime = new Date().getTime();
            ref.unsetMouseMove();
            log('stopDrag');
        };
        this.dragElement = 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`;
            // get max move
            let left = Math.abs(ref.elementOriginalLeft - parseInt(element.style.left));
            let top = Math.abs(ref.elementOriginalTop - parseInt(element.style.top));
            if (left > ref.backAndForthLeftMax) ref.backAndForthLeftMax = left;
            if (top > ref.backAndForthTopMax) ref.backAndForthTopMax = top;
            log('dragElement');
        };
        element.onmousedown = this.startDrag;
        element.onmouseup = this.stopDrag;
    }
    /**
     * 是否拖动图标
     * @param fluctuate 位移波动允许范围
    */
    function isDrag(fluctuate) {
        return (iconDrag.elementOriginalLeft != parseInt(icon.style.left)
            && Math.abs(iconDrag.elementOriginalLeft - parseInt(icon.style.left)) >= fluctuate) ||
            (iconDrag.elementOriginalTop != parseInt(icon.style.top)
                && Math.abs(iconDrag.elementOriginalTop - parseInt(icon.style.top)) >= fluctuate) ||
            iconDrag.backAndForthLeftMax >= fluctuate ||
            iconDrag.backAndForthTopMax >= fluctuate;
    }
    /**强制结束拖动*/
    function forceStopDrag() {
        // 强制设置鼠标拖动事件结束,防止由于网页本身的其它鼠标事件冲突而导致没有侦测到:mouseup
        if (iconDrag) {
            iconDrag.dragging = false;
            iconDrag.unsetMouseMove();
        }
    }
    /**显示 icon*/
    function showIcon(e) {
        log('showIcon event:', e);
        let offsetX = 4; // 横坐标翻译图标偏移
        let offsetY = 8; // 纵坐标翻译图标偏移
        // 更新翻译图标 X、Y 坐标
        if (e.pageX && e.pageY) { // 鼠标
            log('mouse pageX/Y');
            pageX = e.pageX;
            pageY = e.pageY;
        }
        if (e.changedTouches) { // 触屏
            if (e.changedTouches.length > 0) { // 多点触控选取第 1 个
                log('touch pageX/Y');
                pageX = e.changedTouches[0].pageX;
                pageY = e.changedTouches[0].pageY;
                // 触屏修改翻译图标偏移(Android、iOS 选中后的动作菜单一般在当前文字顶部,翻译图标则放到底部)
                offsetX = -26; // 单个翻译图标块宽度
                offsetY = 16 * 3; // 一般字体高度的 3 倍,距离系统自带动作菜单、选择光标太近会导致无法点按
            }
        }
        log(`selected:${selected}, pageX:${pageX}, pageY:${pageY}`)
        if (e.target == icon || (e.target.parentNode && e.target.parentNode == icon)) { // 点击了翻译图标
            e.preventDefault();
            return;
        }
        selected = window.getSelection().toString().trim(); // 当前选中文本
        log(`selected:${selected}, icon display:${icon.style.display}`);
        if (selected && icon.style.display != 'block' && pageX && pageY) { // 显示翻译图标
            log('show icon');
            icon.style.top = `${pageY + offsetY}px`;
            icon.style.left = `${pageX + offsetX}px`;
            icon.style.display = 'block';
            // 兼容部分 Content Security Policy
            icon.style.position = 'absolute';
            icon.style.zIndex = zIndex;
        } else if (!selected) { // 隐藏翻译图标
            log('hide icon');
            hideIcon();
        }
    }
    /**隐藏 icon*/
    function hideIcon() {
        icon.style.display = 'none';
        pageX = 0;
        pageY = 0;
        icon.querySelectorAll('img[is-more]').forEach(ele => {
            ele.style.display = 'none';
        });
        forceStopDrag();
    }
    /**设置*/
    function settings() {
        const hideConfig = gm.get(gm.HIDE, {});
        const sortConfig = gm.get(gm.SORT, []);
        log('hideConfig: ', hideConfig);
        log('sortConfig: ', sortConfig);
        const allSortedIconArray = getCustomMadeIconArray(false);
        document.querySelectorAll('style,link,script').forEach(ele => {
            ele.remove();
        });
        document.querySelectorAll('title').forEach(ele => {
            ele.innerHTML = 'configuration page';
        });
        document.title = 'configuration page';
        document.body.innerHTML = '';
        document.body.style.padding = '20px';
        const desc = document.createElement('div');
        desc.innerHTML = '<h3>After the change, close the configuration page and refresh the current page, the new configuration will take effect.</h3>';
        const reset = document.createElement('button'); // 重置配置
        reset.innerHTML = 'reset settings';
        reset.addEventListener('click', () => {
            const r = confirm("Do you want to reset user settings?");
            if (r == true) {
                gm.reset();
                settings();
            }
        });
        document.body.appendChild(desc);
        document.body.appendChild(reset);
        document.body.appendChild(document.createElement('hr'));
        allSortedIconArray.forEach((obj, i) => {
            const item = document.createElement('div');
            const name = document.createElement('span');
            const up = document.createElement('a');
            const down = document.createElement('a');
            const show = document.createElement('a');
            const span = document.createElement('span');
            name.innerHTML = obj.name;
            span.innerHTML = '&nbsp;&nbsp;';
            up.innerHTML = 'up';
            up.setAttribute('href', 'javascript:void(0)');
            up.setAttribute('index', i);
            up.addEventListener('click', function () {
                const index = myParseInt(this.getAttribute('index'));
                const newIconArray = arrayMove(allSortedIconArray, index, index - 1);
                const idArray = [];
                newIconArray.forEach(({ id }) => {
                    idArray.push(id);
                });
                gm.set(gm.SORT, idArray);
                settings();
            });
            down.innerHTML = 'down';
            down.setAttribute('href', 'javascript:void(0)');
            down.setAttribute('index', i);
            down.addEventListener('click', function () {
                const index = myParseInt(this.getAttribute('index'));
                const newIconArray = arrayMove(allSortedIconArray, index, index + 1);
                const idArray = [];
                newIconArray.forEach(({ id }) => {
                    idArray.push(id);
                });
                gm.set(gm.SORT, idArray);
                settings();
            });
            show.innerHTML = 'show';
            show.setAttribute('show-id', obj.id);
            if (isNotNull(hideConfig[obj.id])) {
                show.innerHTML = 'hide';
            }
            show.setAttribute('href', 'javascript:void(0)');
            show.addEventListener('click', function () {
                if (this.innerHTML == 'show') { // 隐藏
                    if (this.getAttribute('show-id') != 'settings') {
                        hideConfig[this.getAttribute('show-id')] = true;
                    }
                } else { // 显示
                    delete hideConfig[this.getAttribute('show-id')];
                }
                gm.set(gm.HIDE, hideConfig);
                settings();
            });
            item.appendChild(up);
            item.appendChild(span.cloneNode(true));
            item.appendChild(down);
            item.appendChild(span.cloneNode(true));
            item.appendChild(show);
            item.appendChild(span.cloneNode(true));
            item.appendChild(name);
            document.body.appendChild(item);
            document.body.appendChild(document.createElement('hr'));
        });
    }
    /**得到定制化的图标顺序*/
    function getCustomMadeIconArray(hide) {
        const hideConfig = gm.get(gm.HIDE, {});
        const sortConfig = gm.get(gm.SORT, []);
        log('hideConfig: ', hideConfig);
        log('sortConfig: ', sortConfig);
        const customMadeIconArray = [];
        const tempArray = [];
        // hide
        iconArray.forEach(obj => {
            if (hide && !isNotNull(hideConfig[obj.id])) {
                tempArray.push(obj);
            } else if (!hide) {
                tempArray.push(obj);
            }
        });
        // sort
        const sorted = {};
        sortConfig.forEach(id => {
            tempArray.forEach(tObj => {
                if (id == tObj.id) {
                    customMadeIconArray.push(tObj);
                    sorted[id] = true;
                }
            });
        });
        tempArray.forEach(tObj => {
            if (!isNotNull(sorted[tObj.id])) {
                customMadeIconArray.push(tObj);
            }
        });
        log('customMadeIconArray: ', customMadeIconArray);
        return customMadeIconArray;
    }
})();