Greasy Fork is available in English.

链接助手

支持全网主流网盘和小众网盘自动填写密码; 资源站点下载页网盘密码预处理; 文本转链接; 移除链接重定向; 重定向页面自动跳转; 维基百科及镜像、开发者文档、谷歌商店自动切换中文, 维基百科、谷歌开发者、谷歌商店、Github链接转为镜像链接; 新标签打开链接; (外部)链接净化直达

Εγκατάσταση αυτού του κώδικαΒοήθεια
Κώδικας προτεινόμενος από τον δημιιουργό

Μπορεί, επίσης, να σας αρέσει ο κώδικας 雷利子.

Εγκατάσταση αυτού του κώδικα
// ==UserScript==
// @name            链接助手
// @namespace       https://github.com/oneNorth7
// @include         *
// @version         2.0.2
// @author          一个北七
// @run-at          document-body
// @description     支持全网主流网盘和小众网盘自动填写密码; 资源站点下载页网盘密码预处理; 文本转链接; 移除链接重定向; 重定向页面自动跳转; 维基百科及镜像、开发者文档、谷歌商店自动切换中文, 维基百科、谷歌开发者、谷歌商店、Github链接转为镜像链接; 新标签打开链接; (外部)链接净化直达
// @icon            https://gitee.com/oneNorth7/pics/raw/master/picgo/link-helper.png
// @compatible      chrome 69+
// @compatible      firefox 78+
// @compatible      edge Latest
// @noframes
// @license           GPL-3.0 License
// @exclude         *://www.kdocs.cn/p/*
// @exclude         *://docs.google.com/document/d/*
// @exclude         *://www.notion.so/*
// @exclude         *://www.wolai.com/*
// @exclude         *://yiqixie.qingque.cn/d/home/*
// @exclude         *://www.yuque.com/*/edit
// @exclude         *://*.cqaso.com/*
// @exclude         *://xiezuocat.com/#/doc/*
// @exclude         *://mail.*
// @grant              GM_registerMenuCommand
// @grant              GM_unregisterMenuCommand
// @grant              GM_notification
// @grant              GM_info
// @grant              GM_setValue
// @grant              GM_getValue
// @grant              GM_deleteValue
// @grant              GM_openInTab
// @grant              GM_addStyle
// @require          https://cdn.bootcdn.net/ajax/libs/jquery/3.5.1/jquery.min.js
// @require          https://cdn.jsdelivr.net/npm/sweetalert2@10.15.5/dist/sweetalert2.all.min.js
// @created         2021年3月19日 09:48:14
// ==/UserScript==

$(function () {
    "use strict";

    const scriptInfo = GM_info.script,
        locHost = location.host,
        locHref = location.href,
        locHash = location.hash,
        locPath = location.pathname;

    let t = {
        showNotice(msg) {
            GM_notification({
                text: msg,
                title: scriptInfo.name,
                image: scriptInfo.icon,
                highlight: false,
                silent: false,
                timeout: 1500,
            });
        },
        
        clog() {
            console.group("[链接助手]");
            if (locHost === "cloud.189.cn" || locHost === "pan.xunlei.com")
                console.log = console.dir;

            for (let m of arguments) {
                if (void 0 !== m) console.log(m);
            }
            console.groupEnd();
        },

        get(name, def) {
            return GM_getValue(name, def);
        },

        set(name, value) {
            GM_setValue(name, value);
        },

        delete(name) {
            GM_deleteValue(name);
        },

        registerMenu(title, func) {
            return GM_registerMenuCommand(title, func);
        },

        unregisterMenu(menuID) {
            GM_unregisterMenuCommand(menuID);
        },
        
        open(url, options = { active: true, insert: true, setParent :true }) {
            GM_openInTab(url, options);
        },

        http(link, s = false) {
            return link.startsWith("http")
                ? link
                : (s ? "https://" : "http://") + link;
        },
        
        title(a, mark="") {
            if (a.title)
                a.title += "\n" + mark + decodeURIComponent(a.href);
            else a.title = mark + decodeURIComponent(a.href);
        },
        
        hashcode(l=location) {
            return l.hash.slice(1);
        },
        
        search(l=location, p = "password") {
            let args = l.search.slice(1).split("&");
            for (let a of args) {
                if (a.includes(p + "="))
                    return a.replace(p + "=", "");
            }
            return "";
        },

        clean(src, str) {
            for (let s of str) {
                src = src.replace(s, "");
            }
            return src;
        },

        loop(func, times) {
            let tid = setInterval(() => {
                if (times <= 0) clearInterval(tid);
                func();
                this.clog(times);
                times--;
            }, 100);
        },
        
        confirm(title, yes, no = () => {}, deny = false) {
            let option = {
                        toast: true,
                        showCancelButton: true,
                        position: "center",
                        title,
                        confirmButtonText: "是",
                        cancelButtonText: "否",
                        showDenyButton: deny,
                        denyButtonText: "取消",
                        customClass: {
                            popup: "lh-popup",
                            content: "lh-content",
                            closeButton: "lh-close"
                        },
                    };
            return Swal.fire(option).then((res) => {
                if (res.isConfirmed) yes();
                else if (res.isDismissed) no();
                else if (res.isDenied) deny();
            });
        },
        
        increase() {
            success_times = +this.get("success_times") + 1;
            this.set("success_times", success_times);
        },

        subscribe() {
            let isFollowed = t.get("isFollowed", false), least_times = t.get("least_times", 30);
            success_times = +this.get("success_times");
            if (success_times > least_times && !isFollowed) {
                Swal.fire({
                          title: '\u5173\u6ce8\u516c\u4f17\u53f7\uff0c\u4e0d\u8ff7\u8def\uff01',
                          html: $(
                        `<div><img style="width: 300px;margin: 5px auto;" src="https://gitee.com/oneNorth7/pics/raw/master/picgo/oneNorth7.png"><p style="font-size: 16px;color: red;">\u7b2c\u4e00\u65f6\u95f4\u83b7\u53d6<span style="color: gray;font-weight: 700;">\u3010\u94fe\u63a5\u52a9\u624b\u3011</span>\u66f4\u65b0\u63a8\u9001\uff01</p></div>`
                    )[0],
                          showCancelButton: true,
                          allowOutsideClick: false,
                          confirmButtonColor: "#d33",
                          confirmButtonText: "\u5df2\u5173\u6ce8\uff0c\u4e0d\u518d\u63d0\u9192\uff01",
                          cancelButtonColor: "#3085d6",
                          cancelButtonText: "\u7a0d\u540e\u5173\u6ce8",
                        }).then((result) => {
                          if (result.isConfirmed) {
                            Swal.fire({
                              position: "center",
                              icon: "success",
                              title: "\u611f\u8c22\u5173\u6ce8\uff01\uff01\uff01",
                              text: "\u4e00\u4e2a\u5317\u4e03\u4f1a\u7ee7\u7eed\u4e0d\u9057\u4f59\u529b\u5730\u521b\u4f5c\u66f4\u591a\u5b9e\u7528\u5de5\u5177",
                              showConfirmButton: false,
                              timer: 2000
                            });
                            t.set("isFollowed", true);
                          } else t.set("least_times", least_times + 30);
                        });
            }
        },

        update(name, value) {
            if (this.get("updated_version", "") != scriptInfo.version) {
                let data = this.get(name, false);
                if (data) {
                    value.push("uniportal.huawei.com", "cn.bing.com");
                    let temp = data.filter(h => !value.includes(h));
                    if (temp.length) 
                        this.set(name, temp);
                }

                this.rename("excludeSites", "excludeHosts");
                this.rename("autoClickSites", "autoClickHosts");
                this.set("updated_version", scriptInfo.version);
            }
        },

        rename(name, newName) {
            if (this.get("updated_version", "") != scriptInfo.version) {
                let data = this.get(name, false);
                if (data) {
                    this.set(newName, data);
                    this.delete(name);
                }
            }
        },
        
        rand(min, max) {
            if (arguments.length == 1) max = min, min = 0;
            let random = Math.random(),
                randInt = Math.floor(random * (max + 1 - min)) + min;
            return randInt;
        },
    };

    let host_suffix = "(?:com|cn|org|net|info|tv|cc|gov|edu|nz|me|io|ke|im|top|xyz|app|moe|in|pw|one|co|ml|art|vip|cam|fun)\\b",
        http_re_str = "(?:https?:\\/\\/|www\\.)[-\\w_.~/=?&#%+:!*@]+|(?<!@)(?:\\w[-\\w._]*\\." + host_suffix + ")(?:\\/[-\\w_.~/=?&#%+:!*@\\u4e00-\\u9fa5]*)?",
        bdpan_re_str = "(?:\\/?s)?\\/[-\\w_]{23}|(?:\\/?s)?\\/\\w{6,8}",
        email_re_str = "(?<![.@])\\w(?:[-\\w._])+@\\w[-\\w._]+\\." + host_suffix,
        ed2k_re_str = "ed2k:\\/\\/\\|file\\|[^\\|]+\\|\\d+\\|\\w{32}\\|(?:h=\\w{32}\\|)?\\/",
        magnet_re_str = "(magnet:\\?xt=urn:btih:(?:[a-fA-F0-9]{40}|[a-zA-Z2-7]{32})|(?<![|/?#=])\\b(?:[a-f0-9]{40}|[A-F0-9]{40}|[a-z2-7]{32}|[A-Z2-7]{32})\\b)",
        magnet_suffix = "(?:&[\\S]+)?",
        base64_re_str = "(?:[A-Za-z0-9+/]{4})*(?:[A-Za-z0-9+/]{4}|[A-Za-z0-9+/]{3}=|[A-Za-z0-9+/]{2}==)",
        thunder_re_str = "thunder:\\/\\/" + base64_re_str,
        url_regexp = new RegExp("\\b" + ed2k_re_str +
                            "|" + email_re_str +
                            "|" + http_re_str +
                            "|" + thunder_re_str +
                            (locHost === "tieba.baidu.com" ? ("|" + bdpan_re_str ) : "") +
                            "|" + magnet_re_str + magnet_suffix
                            , "i");
    
    let Preprocess = {
        "www.mikuclub.cc": function () {
            if (/\/\d+/.test(locPath)) {
                let password = $(".password1"),
                    link = $("a.download");
                if (password.length && link.length)
                    link[0].hash = password[0].value;
            }
        },

        "www.olecn.com": function () {
            if (/http:\/\/www\.olecn\.com\/download\.php\?id=\d+/.test(locHref)) {
                let link = $("div.panel-body a"),
                    pass = $("div.plus_l li:eq(3)");
                if (link.length && pass.length)
                    link[0].hash = pass
                        .text()
                        .trim()
                        .replace("网盘提取码 :", "");
            }
        },
        
        "www.qiuziyuan.net": function () {
            if (/\/(?:pcrj\/|Android\/\d+\.html)/.test(locPath)) {
                let filetit = $("div.filetit:first");
                for (let child of filetit.children()) {
                    if (child.href) {
                        let result = child.innerHTML.match(http_re_str);
                        if (result) child.href = t.http(result[0]);
                    }

                    if (
                        child.innerHTML.startsWith("90网盘:") ||
                        child.innerHTML.includes("90pan")
                    ) {
                        let dom = filetit.next().next(),
                            result = /(?:90网盘:|\/\s*)(\d+)/.exec(dom.html());
                        if (result) child.href += "#" + result[1];
                    }
                }
            }
        },

        "www.appmiu.com": function () {
            if (/https:\/\/www\.appmiu\.com\/download\/\?post_id=/.test(locHref)) {
                setTimeout(() => {
                    let code = $("#tq").attr("data-clipboard-text");
                    if (code) {
                        let a = $("a.empty.button"), url = a.prop("href");
                        if (url) a.prop("href", url + "#" + code);
                    }
                }, 1000);
            }
        },

        "www.acgjc.com": function () {
            if (/http:\/\/www.acgjc.com\/storage-download\/\?code=/.test(locHref)) {
                let codeNode = $("#theme_custom_storage-0-download-pwd");
                if (codeNode.length) {
                    let code = codeNode.val(),
                        link = codeNode.parents("div.fieldset-content").find("a");
                    if (link) link.prop("href", link[0].href + "#" + code);
                }
            }
        },
        
        "www.bsh.me": function () {
            if (/https:\/\/www\.bsh\.me\/download\.php\?author=/.test(locHref)) {
                $('ul.list-group:last').find('a').each((i, a) => {
                    let text = $(a).text(),
                        codeNode = $(a).parents('ul.list-group').find(`span.item-title:contains("${text}"):last`),
                        result = /\w{2,10}/.exec(codeNode.text());

                    if (a.search === "?%3E") a.search = "";
                    
                    if (result) a.hash = result[0];
                    else a.hash = codeNode.parent().text().match("\\w{2,10}")[0];
                });
                
                $(".card div.text-center, footer.blockquote-footer").hide();
                $("div.card-signup").css("margin-bottom", "20px");
            }
        },

        "www.yxssp.com": function () {
            if (/https:\/\/www\.yxssp\.com\/download\.php\?author=/.test(locHref)) {
                $("ul.list-group:last").find("a").each((i, a) => {
                    let text = $(a).text(),
                        codeNode = $(a).parents("ul.list-group").find(`span.item-title:contains("${text}"):last`);

                    if (codeNode[0].nextSibling) codeNode = codeNode[0].nextSibling;
                    let result = /[.\w]{2,13}/.exec($(codeNode).text());

                    if (a.search === "?%3E") a.search = "";
                    
                    if (result) a.hash = result[0];
                });
                
                $(".card div.text-center, footer.blockquote-footer").hide();
                $("div.card-signup").css("margin-bottom", "20px");
            }
        },
        
        "www.zhiruanku.com": function () {
            if (/\/\d+/.test(locPath)) {
                $("div.wp-block-zibllblock-buttons a").each((i, a) => {
                    a.href = a.dataset.id
                    let result = a.textContent.match(":(\\w{2,10})");
                    if (result) a.hash = result[1];
                });
            }
        },
        
        "zhidao.baidu.com": function () {
            if (/\/question\/\d+\.html.*/.test(locPath)) {
                $("baiduyun.ikqb-yun-box").each((i, e) => {
                    let title = $(e).attr("data_title"),
                        url = $(e).attr("data_sharelink") + "#" + $(e).attr("data_code"),
                        p = $(e).parent("p");
                    p.before(`<p style="font-size:24px"><a href="${url}" title="点我可自动填写密码!" target="_blank"><span style="color:#36BE63">预处理链接:</span>${title}</a></p>`);
                });
            }
        },
        
        "yun.hei521.cn": function () {
            if (/\/index\.php\/archives\/\d+\.html/.test(locPath)) {
                let text = $("ol.comment-list .zface-box p:contains('码')").text();
                if (!addCode(text)) {
                    text = $("div.post-content p:contains('码')").text();
                    addCode(text);
                }
            }
            
            function addCode(text) {
                if (text) {
                    let codes = text.match(/码[::]\s*[a-z\d]{4}/g);
                    if (codes) {
                        $("div.post-content a:contains('http')").each((i, a) => {
                            let result = a.nextSibling && a.nextSibling.nodeValue && a.nextSibling.nodeValue.match(/[a-z\d]{4}/);
                            if (result) a.hash = "#" + result[0];
                            else if (codes[i]) a.hash = "#" + codes[i].replace(/码[::]\s*/, "");
                        });
                        return true;
                    } 
                }
            }
        },

        "www.iapps.me": function () {
            if (/\?download=\d+/.test(location.search)) {
                let code = $("div.alert").text().match(/[a-z\d]{4}/g);
                if (code)
                    $("button.download-btn").attr("data-href", $("button.download-btn").attr("data-href") + "#" + code[0]);
                
                $("#carouselAdIndicators").parent().remove();
            }
            
            if (/\/archives\/\d+/.test(locPath)) {
                let func = $("#directDownload_2 .btn-primary").prop("onclick");
                if (func) {
                    let funcStr = func.toString(),
                        result = funcStr.match(/open\("(.+)"\)/);
                    if (result) $("div.getit>a").prop("href", result[1]).click(o => o.stopImmediatePropagation());
                }
            }
        },

        "www.sxpdf.com": function() {
            if (locHref.startsWith("https://www.sxpdf.com/wp-content/themes/niao/down.php?id="))
                $('a[onclick="copyUrl2()"]').removeAttr("onclick");
        },
    };

    if (Preprocess[locHost]) Preprocess[locHost]();

    let YunDisk = {
        sites: {
            "pan.baidu.com": {
                // 百度云
                inputSelector: "#accessCode",
                buttonSelector: "#submitBtn",
                regStr: "[a-z\\d]{4}",
                redirect: { pathname: {"/wap/": "/share/"} },
            },

            "eyun.baidu.com": {
                // 百度企业网盘
                inputSelector: "input.share-access-code",
                buttonSelector: "a.g-button",
                regStr: "[a-z\\d]{4,6}",
            },

            "cloud.189.cn": {
                // 天翼云
                inputSelector: "#code_txt",
                buttonSelector: "a.btn-primary",
                regStr: "[a-z\\d]{4}",
                timeout: 1000,
                inputEvent: true,
                noNotice: true,
                checkError: true,
            },
            
            "h5.cloud.189.cn": {
                // 手机天翼云
                inputSelector: "input.access-code-input",
                buttonSelector: "div.button",
                regStr: "[a-z\\d]{4}",
                timeout: 100,
                password: true,
                inputEvent: true,
                redirect: { href: {"h5.cloud.189.cn/share.html#/t/": "cloud.189.cn/web/share?code="} },
            },

            "lanzou.com": {
                // 蓝奏云
                inputSelector: "#pwd",
                buttonSelector: "#sub, .passwddiv-btn",
                regStr: "[a-z\\d]{2,10}|-{4}",
                redirect: { host: { "lanzous": "lanzoux" } },
                noNotice: true,
            },

            "ctfile.com": {
                // 城通网盘
                inputSelector: "#passcode",
                buttonSelector: "button.btn-primary",
                regStr: "[a-z\\d]{4,6}",
                timeout: 500,
            },

            "vdisk.weibo.com": {
                // 微盘
                inputSelector: "#keypass",
                buttonSelector: "div.search_btn_wrap>a",
                regStr: "[a-z\\d]{4}",
            },

            "pan.xunlei.com": {
                // 迅雷云盘
                inputSelector: "#__nuxt input.td-input__inner",
                buttonSelector: "#__nuxt button.td-button",
                regStr: "[a-z\\d]{4}",
                timeout: 1200,
                password: true,
                searchPath: true,
                inputEvent: true,
            },

            "share.weiyun.com": {
                // 微云
                inputSelector: "input.input-txt",
                buttonSelector: "button.btn-main",
                regStr: "[a-z\\d]{4,6}",
                timeout: 500,
                inputEvent: true,
            },

            "115.com": {
                // 115网盘
                inputSelector: "input.text",
                buttonSelector: "a.btn-large",
                regStr: "[a-z\\d]{4}",
                timeout: 500,
                password: true,
            },

            "quqi.com": {
                // 曲奇云
                inputSelector: "div.webix_el_box>input",
                buttonSelector: "button.webixtype_base",
                regStr: "[a-z\\d]{6}",
                timeout: 800,
            },

            "caiyun.139.com": {
                // 和彩云
                inputSelector: "input",
                buttonSelector: "a.btn-token",
                regStr: "[a-z\\d]{4}",
                timeout: 100,
                clickTimeout: 10,
                inputEvent: true,
                store: true,
            },

            "mo.own-cloud.cn": {
                // 小麦魔方
                inputSelector: "#pwd",
                buttonSelector: "button.MuiButton-root",
                regStr: "[a-z\\d\\u4e00-\\u9fa5]{2,8}",
                timeout: 500,
                password: true,
                cleanHash: true,
            },

            "moecloud.cn": {
                // 萌云
                inputSelector: "#pwd",
                buttonSelector: "button.MuiButton-root",
                regStr: "\\w+\\.\\w+\\.\\w+|[a-z\\d]{3,8}",
                timeout: 500,
                password: true,
                cleanHash: true,
                redirect: { host: {"cloud.qingstore.cn": "moecloud.cn"} },
            },
            
            "www.wenshushu.cn": {
                // 文叔叔
                inputSelector: "input.ivu-input",
                buttonSelector: "button.m-mg_t40",
                regStr: "[a-z\\d]{4,8}",
                timeout: 1000,
                inputEvent: true,
            },
            
            "mega.nz": {
                // regStr: "[a-z\\d\\-_]{22}",
            },
            
            "www.jianguoyun.com": {
                // 坚果云
                inputSelector: "#access-pwd",
                buttonSelector: "button.action-button",
                regStr: "[a-z\\d]{4,16}",
            },
            
            "yunpan.360.cn": {
                // 360安全云盘
                inputSelector: "input.pwd-input",
                buttonSelector: "input.submit-btn",
                regStr: "[a-z\\d]{4}",
            },
            
            "pan-yz.chaoxing.com": {
                // 超星云盘
                inputSelector: "input.tqInp",
                buttonSelector: "a.blueBgBtn",
                regStr: "[a-z\\d]{6}",
            },
            
            "shandianpan.com": {
                // 闪电盘
                inputSelector: 'input[placeholder="请输入文件密码"]',
                buttonSelector: "div.btn",
                regStr: "[a-z\\d]{4}",
                timeout: 500,
                password: true,
                inputEvent: true,
                pathHash: true,
            },
            
            "my.sharepoint.com": {
                // SharePoint
                inputSelector: '#txtPassword',
                buttonSelector: "#btnSubmitPassword",
                regStr: "[a-z\\d]{3,5}",
                password: true,
            },

            "onedrive.live.com": {
                // OneDrive
                inputSelector: 'input[type="password"]',
                buttonSelector: "button.od-Button--primary",
                regStr: "[a-z\\d]{3,7}",
                store: true,
                inputEvent: true,
                timeout: 5000,
            },
            
            "u.163.com": {
                // 网易网盘
                inputSelector: "#pickupCode",
                buttonSelector: "#wpDownloadHref",
                regStr: "[a-z\\d]{8}",
            },
            
            "www.aliyundrive.com": {
                // 阿里云盘
                inputSelector: "input.ant-input",
                buttonSelector: "button.button--fep7l",
                regStr: "[a-z\\d]{4}",
                timeout: 1000,
                react: true,
            },

            "cowtransfer.com": {
                // 奶牛快传
                inputSelector: "div.receive-code-input input",
                buttonSelector: "div.button.open-buttom",
                regStr: "[a-z\\d]{6}",
                timeout: 500,
                inputEvent: true,
                reverse: true,
                hidden: true,
            },

            "www.123pan.com": {
                // 123云盘
                inputSelector: "input.ant-input",
                buttonSelector: "input.ant-input+button",
                regStr: "[a-z\\d]{4,8}",
                timeout: 1000,
                react: true,
            },
        },
        
        pans: [
            "disk.yandex.com", // YandexDisk
            "yadi.sk", // YandexDisk
            "www.mediafire.com", // MediaFire     
            "drive.google.com", // GoogleDrive
            "down.52pojie.cn", // 爱盘
            "www.yunzhongzhuan.com", // 云中转
            "yiqixie.qingque.cn", // 一起写
            "www.androiddownload.net",
            "www.dropbox.com", // Dropbox
            "www.kufile.net", // 库云
            "www.kdocs.cn", // 金山文档
            "pan.bitqiu.com", // 比特球云盘
            "www.feimaoyun.com", // 飞猫云
            "www.fmpan.com",  // 飞猫云
            "www.fangcloud.com", // 亿方云
            "gd.188988.xyz", // GD DISK
            "www.yun.cn", // UC网盘
            "www.yuque.com", // 语雀
            "shimo.im", // 石墨文档
            "www.showdoc.com.cn", // ShowDoc
            "zijieyunpan.com", // 字节网盘
        ],

        mapHost(host) {
            let dict = {
                "^yun\\.baidu\\.com": "pan.baidu.com",
                ".*lanzou[iswx]?\\.com": "lanzou.com",
                "^(?:[a-z]\\d{3}|\\d{3}[a-z]|t00y|\\w+\\.(?:ctfile|pipipan))\\.com$|^ctfile\\.\\w+\\.cn$": "ctfile.com",
                "^ct\\.\\w+\\.(?:com|me|org)$": "ctfile.com",
                "quqi\\.\\w+\\.com": "quqi.com",
                "\\w+\\-my\\.sharepoint\\.(?:com|cn)": "my.sharepoint.com",
                "\\w{6}\\.link\\.yunpan\\.360\\.cn|yunpan\\.cn": "yunpan.360.cn",
            };
            
            for (let key in dict)
                if (host.match(key))
                    return host.replace(host, dict[key]);

            if (t.get("ctpanHosts", []).concat(["dl.sixyin.com", "dw.yxssp.com", "down.sxpdf.com", "dl.ooopn.com", "pd.ggtrj.com", "72k.us", "u.yfxj91.top"]).includes(host))
                return host.replace(host, "ctfile.com");
            if (t.get("cloudreveHosts", []).concat(["cloud.qingstore.cn", "cncncloud.com", "ilolita945.softether.net:5212", "my-file.cn",
                      "pan.bilnn.com", "drive.dnxshare.cn", "pan.mba", "pan.oddba.cn", "pan.adycloud.com", "pan.yrxitong.com", "disk.onji.cn", "www.naikuai.cn"]).includes(host))
                return host.replace(host, "moecloud.cn");

            let mapped = {
                "feixin.10086.cn": "139.com",
                "ws28.cn": "www.wenshushu.cn",
                "wss1.cn": "www.wenshushu.cn",
                "zb.my.to:5000": "gofile.me",
                "cloud.dnxshare.cn": "drive.dnxshare.cn",
                "mofile.own-cloud.cn": "mo.own-cloud.cn",
                "nf.mail.163.com": "u.163.com",
                "1drv.ms": "onedrive.live.com",
                "alywp.net": "www.aliyundrive.com",
                "app.mediatrack.cn": "mdl.ink",
            }[host];
            if (mapped)
                return host.replace(host, mapped);
            
            return host;
        },

        redirect(a, d) {
            if (d) {
                for (let k in d) {
                    for (let v in d[k])
                        a[k] = a[k].replace(v, d[k][v]);
                }
            }
        },

        autoFill(host) {
            let site = this.sites[host];
            // 百度云文档
            if (host === "pan.baidu.com" && locPath.startsWith("/doc/share/"))
                site = {
                    inputSelector: "input.u-input__inner",
                    buttonSelector: "div.dialog-footer button.u-btn.u-btn--primary",
                    regStr: "[a-z\\d]{4}",
                    inputEvent: true,
                    timeout: 500,
                    clickTimeout: 10,
                };
            
            // 自动填写密码
            if (site.timeout) setTimeout(fillOnce, site.timeout);
            else fillOnce();
            function fillOnce() {
                if (site.checkError && $("div.error-content:visible").length)
                    return;
                if (site.inputSelector) {
                    let input = $(site.inputSelector + (site.hidden ? "" : ":visible")),
                        button = $(site.buttonSelector),
                        code = null;
                    function click() {
                        if (site.clickTimeout)
                            setTimeout(() => {
                                button = $(site.buttonSelector);
                                button[0].click();
                            }, site.clickTimeout);
                        else button[0].click();
                    }
                    
                    if (input.length) {
                        if (site.store) code = t.get(host, false);
                        else if (site.password) code = decodeURIComponent(t.search()) || t.hashcode();
                        else code = t.hashcode();
                        if (code) {
                            let codeRe = RegExp("^" + site.regStr + "$", "i");
                            if (codeRe.test(code)) {
                                if (site.inputEvent) {
                                    let tid = setInterval(() => {
                                        input.val(code);
                                        if (input.val() !== "") {
                                            if (InputEvent) {
                                                input[0].dispatchEvent(
                                                    new InputEvent("input")
                                                );
                                            } else if (KeyboardEvent) {
                                                input[0].dispatchEvent(
                                                    new KeyboardEvent("input")
                                                );
                                            }

                                            clearInterval(tid);
                                            click();
                                        }
                                    }, 1000);
                                } else if (site.react) {
                                    let lastValue = input.val();
                                    input.val(code);
                                    let tracker = input[0]._valueTracker;
                                    if (tracker) tracker.setValue(lastValue);
                                    input[0].dispatchEvent(new Event("input", {bubbles: true}));
                                    click();
                                } else if (site.reverse) {
                                    click();
                                    input.val(code);
                                } else {
                                    input.val(code);
                                    click();
                                }
                                t.increase();
                                if (!site.Notice) t.subscribe();
                            } else {
                                t.clog("未找到合适的提取码!");
                            }
                        } else {
                            t.clog("未找到提取码!");
                        }
                    } else {
                        t.clog("无需填写密码!");
                    }
                }
            }
         
        },

        addCode(a, isInput = false) {
            // 奶牛快传
            if (a.host === "cowtransfer.com" && a.pathname !== "/")
                return;
            
            let mapped = this.mapHost(a.host),
                site = this.sites[mapped];
            if (site.regStr) {
                let codeRe = new RegExp("^" + site.regStr + "$", "i"),
                    other = Object.keys(this.sites).filter(s => s !== mapped);
                if (mapped !== "lanzou.com")
                    other.push("lanzou[iswx]\.com");
                else if (mapped !== "ctfile.com")
                    other.push("^(?:[a-z]\d{3}|\d{3}[a-z]|t00y|\\w+\\.ctfile)\.com$");

                if (site.redirect) this.redirect(a, site.redirect);
                
                if (site.cleanHash) {
                    let result = a.hash && /#(\/s\/\w{6})/.exec(a.hash);
                    if (result) {
                        if (a.pathname == '/') {
                            a.pathname = result[1];
                            a.hash = '';
                        }
                    }
                } else if (site.pathHash) {
                    if (a.pathname.match(/\/f\/\w+/))
                        a.href = a.href.replace("f/", "#/share-detail?id=");
                } else if (site.searchPath) {
                    if (!a.search)
                        a.search = "?path=%2F";
                }

                if (site.password) {
                    let result = a.hash.match("#(" + site.regStr + ")");
                    if (result) {
                        if (!t.search(a))
                                a.search = a.search ? a.search + "&password=" + encodeURIComponent(result[1]) : "password=" + encodeURIComponent(result[1]);
                        a.hash = "";
                    }
                }
                
                if (!codeRe.test(t.hashcode(a)) && !codeRe.test(t.search(a))) {
                    let reg = new RegExp(
                            "\\s*(?:提[取示]|访问|查阅|取件|密\\s*|口令|艾|Extracted-code|key|password|pwd)" + (locHost.startsWith("www.meijumi.") ? "?" : "") + "[码碼]?(?:--)?[】\\]))]?\\s*[\\u4e00-\\u9fa5]?[:: ((是为]?\\s*(" +
                                site.regStr +
                                ")|^[码碼]?[】\\]))]?\\s*[::【\\[ ((]*\\s*(" +
                                site.regStr +
                                ")[】\\]))]?" + (isInput ? "\\b" : "$"),
                            "i"
                        ),
                        code = reg.exec($(a).text().trim());
                    if (code && (/^http/.test(code[1]) || /^http/.test(code[2])))
                        code = null;
                    for (
                        let i = 10, current = a;
                        current && current.localName != "body" && !code && i > 0;
                        i--, current = current.parentElement
                    ) {
                        if (locHost === "yun.hei521.cn" && current.id === "main")
                            break;
                        let next = current;
                        while (!code) {
                            if (!next) break;
                            else if (next.nodeValue) code = reg.exec(next.nodeValue.trim());
                            else if (!other.some(s => next.textContent.match(s)))
                                code = reg.exec(next.innerText.trim());

                            if (code && (/^http/.test(code[1]) || /^http/.test(code[2])))
                            code = null;
                            
                            next = next.nextSibling;
                        }
                    }

                    if (code) {
                        let c = code[1] || code[2];
                        a.href = a.href.replace(/%E6%8F%90%E5%8F%96%E7%A0%81$/, "");
                        if (site.store) t.set(mapped, c);
                        else if (site.password) {
                            if (!t.search(a))
                                a.search = a.search ? a.search + "&password=" + encodeURIComponent(c) : "password=" + encodeURIComponent(c);
                        } else {
                            a.href = a.href.replace(/%23.*$/, "");
                            a.hash = c;
                        }
                    } else {
                        if (site.store) t.delete(mapped);
                        t.clog("找不到code!");
                    }
                }
            }
        },
    };
    let success_times = t.get("success_times");
    if (!success_times) t.set("success_times", 0);
    
    let dealedHost = YunDisk.mapHost(locHost);
    if (YunDisk.sites[dealedHost]) YunDisk.autoFill(dealedHost);
    else {
        let RedirectPage = {
            sites: {
                "show.bookmarkearth.com": {
                    // 书签地球
                    include: "view/",
                    selector: "p.link",
                },

                "t.cn": {
                    // 新浪短链
                    include: "",
                    selector: "a.m-btn-orange",
                },

                "sunbox.cc": {
                    // 阳光盒子
                    include: "wp-content/themes/begin/go.php?url=",
                    selector: "a.alert-btn",
                },

                "www.itdaan.com": {
                    // 开发者知识库
                    include: "link/",
                    selector: "a.c-footer-a1",
                },

                "to.redircdn.com": {
                    include: "?",
                    selector: "a.bglink",
                },
                
                "link.csdn.net": {
                    // CSDN
                    include: "?target=",
                    selector: "a.loading-btn",
                    timeout: 100,
                },
                
                "support.qq.com": {
                    // 兔小巢
                    match: "products\\/\\d+\\/link-jump\\?jump=",
                    selector: "span.link_url",
                },
                
                "c.pc.qq.com": {
                    // QQ非官方页面
                    include: "middlem.html?pfurl=",
                    selector: "#url",
                },
                
                "docs.qq.com": {
                    // 腾讯文档
                    include: "scenario/link.html?url=",
                    selector: "span.url-src",
                    timeout: 500,
                },
                
                "www.tianyancha.com": {
                    // 天眼查
                    include: "security?target=",
                    selector: "div.security-link",
                },
                
                "www.yuque.com": {
                    // 语雀
                    include: "r/goto?url=",
                    selector: "button.ant-btn-primary>a",
                    timeout: 300,
                },
                
                "jump.bdimg.com": {
                    // 百度贴吧
                    include: "safecheck/index?url=",
                    selector: "div.warning_info.fl>a",
                },
                
                "jump2.bdimg.com": {
                    // 百度贴吧
                    include: "safecheck/index?url=",
                    selector: "div.warning_info.fl>a",
                },
                
                "iphone.myzaker.com" : {
                    // Zaker
                    include: "zaker/link.php?",
                    selector: "a.btn",
                },
                
                "game.bilibili.com": {
                    // 哔哩哔哩游戏
                    include: "linkfilter/?url=",
                    selector: "#copy-url",
                },
                
                "www.chinaz.com": {
                    // 站长之家
                    include: "go.shtml?url=",
                    selector: "div.link-bd__text",
                },

                "link.logonews.cn": {
                    // 标志情报局
                    include: "?url=",
                    selector: "a.button",
                },

                "www.douban.com": {
                    // 豆瓣
                    include: "link2/?url=",
                    selector: "a.btn-redir",
                },
                
                "link.zhihu.com": {
                    // 知乎
                    include: "?target=",
                    selector: "a.button",
                },
                
                "www.jianshu.com": {
                    // 简书
                    include: "go-wild?ac=2&url=",
                    selector: 'div[title^="http"], div[title^="www"]',
                },
                
                "link.juejin.cn": {
                    // 掘金
                    include: "?target=",
                    selector: 'p[style="margin: 0px;"]',
                },
                
                "www.oschina.net": {
                    // 开源中国
                    include: "action/GoToLink?url=",
                    selector: "a.link-button",
                },
                
                "www.youtube.com": {
                    // youtube
                    include: "redirect?q=",
                    selector: "#invalid-token-redirect-goto-site-button",
                },
            },

            redirect(host) {
                let site = this.sites[host];
                if (site) {
                    let include = host + "/" + site.include;
                    if (locHref.includes(include) || site.match && locHref.match(site.match)) {
                        setTimeout(redirect, site.timeout || 0);
                        return true;
                    }
                }

                function redirect() {
                    let target = $(site.selector);
                    if (target.length) location.replace(t.http(target[0].href || target[0].innerText));
                    else if (locHost == "t.cn" && $("div.text:contains('绿色上网')").length)
                        fetch(locHref).then(res => location.replace(res.headers.get("location")));
                    else t.clog('找不到跳转目标!');
                    t.increase();
                }
            },

            wiki() {
                let isZh = locHost.includes("zh"),
                    jumpToZh = t.get("jumpToZh", true),
                    a = $("a.interlanguage-link-target[lang='zh']");

                if (!isZh && jumpToZh) {
                    history.pushState(null, null, locHref);
                    if (a.length) location.replace(a[0].href);
                    else t.showNotice("没有找到中文页面!");
                }

                let menuID = t.registerMenu(
                    `${jumpToZh ? "[✔]" : "[✖]"}自动切换中文`,
                    autoJump
                );

                function autoJump() {
                    jumpToZh = !jumpToZh;
                    t.set("jumpToZh", jumpToZh);
                    t.unregisterMenu(menuID);
                    menuID = t.registerMenu(
                        `${jumpToZh ? "[✔]" : "[✖]"}自动切换中文`,
                        autoJump
                    );
                    if (!isZh && jumpToZh) {
                        history.pushState(null, null, locHref);
                        if (a.length) location.replace(a[0].href);
                        else t.showNotice("没有找到中文页面!");
                    }// else history.back();
                }
            },

            mozilla() {
                let isZh = locPath.includes("zh-"),
                    jumpToZh = t.get("jumpToZh", true);
                jump();
                function jump() {
                    if (!isZh && jumpToZh) {
                        let result = /developer\.mozilla\.org\/(.+?)\//.exec(
                            locHref
                        ), options = $("#language-selector").children(), flag = "";

                        if (result) {
                            for (let i = options.length; i > 0; i--) {
                                if (options[i - 1].value.startsWith("zh-")) {
                                    if (flag) {
                                        flag = options[i - 1].value;
                                        break;
                                    }
                                    flag = options[i - 1].value;
                                }
                            }
                            if (flag) {
                                let zh_url = locHref.replace(result[1], flag);
                                history.pushState(null, null, locHref);
                                location.replace(zh_url);
                            }
                            else t.showNotice("没有找到中文页面!");
                        }
                    }
                }
                let menuID = t.registerMenu(
                    `${jumpToZh ? "[✔]" : "[✖]"}自动切换中文`,
                    autoJump
                );

                function autoJump() {
                    jumpToZh = !jumpToZh;
                    t.set("jumpToZh", jumpToZh);
                    t.unregisterMenu(menuID);
                    menuID = t.registerMenu(
                        `${jumpToZh ? "[✔]" : "[✖]"}自动切换中文`,
                        autoJump
                    );
                    jump();
                }
            },
            
            MSDocs() {
                let isZh = locPath.includes("zh-cn"),
                    jumpToZh = t.get("jumpToZh", true);
                if (!isZh && jumpToZh) {
                    history.pushState(null, null, locHref);
                    location.replace(locHref.replace(/docs.microsoft.com\/[a-z\-]{5}\//i, 'docs.microsoft.com/zh-cn/'));
                }
                
                let menuID = t.registerMenu(
                    `${jumpToZh ? "[✔]" : "[✖]"}自动切换中文`,
                    autoJump
                );

                function autoJump() {
                    jumpToZh = !jumpToZh;
                    t.set("jumpToZh", jumpToZh);
                    t.unregisterMenu(menuID);
                    menuID = t.registerMenu(
                        `${jumpToZh ? "[✔]" : "[✖]"}自动切换中文`,
                        autoJump
                    );
                    if (!isZh && jumpToZh) {
                        history.pushState(null, null, locHref);
                        location.replace(locHref.replace(/docs.microsoft.com\/[a-z\-]{5}\//i, 'docs.microsoft.com/zh-cn/'));
                    }
                }
            },
            
            chrome() {
                if (location.search.includes('hl=')) {
                    let isZh = location.search.includes('hl=zh-CN'),
                        jumpToZh = t.get("jumpToZh", true);
                    if (!isZh && jumpToZh) {
                        history.pushState(null, null, locHref);
                        location.search = location.search.replace(/hl=[a-zA-Z]{2}-[a-zA-Z]{2}/, 'hl=zh-CN');
                    }

                    let menuID = t.registerMenu(
                        `${jumpToZh ? "[✔]" : "[✖]"}自动切换中文`,
                        autoJump
                    );
                } else location.search += location.search ? '&hl=zh-CN' : '?hl=zh-CN';
                

                function autoJump() {
                    jumpToZh = !jumpToZh;
                    t.set("jumpToZh", jumpToZh);
                    t.unregisterMenu(menuID);
                    menuID = t.registerMenu(
                        `${jumpToZh ? "[✔]" : "[✖]"}自动切换中文`,
                        autoJump
                    );
                    if (!isZh && jumpToZh) {
                        history.pushState(null, null, locHref);
                        location.search = location.search.replace(/hl=[a-zA-Z]{2}-[a-zA-Z]{2}/, 'hl=zh-CN');
                    }
                }
            },
        };
        
        if (RedirectPage.redirect(locHost)) return;
        else if (locHost.match(/.+wiki(?:\.hancel|pedia)\.org/))
            RedirectPage.wiki();
        else if (locHost == "chrome.google.com")
            RedirectPage.chrome();
        else {
            if (locHost === "developer.mozilla.org") RedirectPage.mozilla();
            else if (locHost === "docs.microsoft.com") RedirectPage.MSDocs();
            
            let isChromium = navigator.userAgent.includes("Chrome");
            
            if (isChromium)
                $(document).on("selectstart mousedown", (obj) => listener(obj));
            else
                $(document).on("mouseup", (obj) => listener(obj));

            if (!t.get("excludeAll", false)) {
                if (locHost.includes("blog.csdn.net"))
                    document.body.addEventListener("click", function (obj) {
                        let e = obj.target;
                        if (e.localName === "a" && e.href && e.href.match(http_re_str)) {
                            if (e.id !== "btn-readmore-zk" && !(e.attributes.href && e.attributes.href.nodeValue.startsWith("#"))) {
                                obj.stopImmediatePropagation();
                                if (e.host !== "github.com" && e.host !== "chrome.google.com") window.open(e.href.replace(/\?utm_source=csdn_blog$/, ""));
                                obj.preventDefault();
                            }
                        }
                    }, true);
            }

            // 移除登录和注册按钮
            if (["hub.fastgit.org", "github.com.cnpmjs.org", "github.rc1844.workers.dev"].some(h => locHost === h)) {
                $(".HeaderMenu div.position-relative.mr-3, div.position-relative.mr-3+a, div.d-flex.flex-items-center>a").remove();
                if (locPath === "/")
                    $("form div.d-flex, div.home-nav-hidden>a").remove();
            }

            if (locHost === "bbs.pcbeta.com")
                $(document.head).append(`<link rel="stylesheet" type="text/css" href="https://cdn.jsdelivr.net/npm/sweetalert2@10.15.5/dist/sweetalert2.min.css">`);

            GM_addStyle(`
                .lh-popup {
                    font-size: 1em;
                    font: 16px/1.5 'Microsoft Yahei',arial,helvetica,sans-serif;
                }

                .lh-content {
                    padding: 0;
                }
                
                .lh-close {
                    box-shadow: none;
                }

                .lh-close:focus {
                    box-shadow: none;
                }


                .lh-menu {
                    margin: 0;
                    padding: 0;
                    list-style: none;
                    font-size: 18px;
                }

                .lh-item {
                    padding-top: 20px;
                    margin-bottom: 0;
                }

                .lh-footer {
                    font-size: 16px;
                }

                .lh-footer a {
                    font-size: 18px;
                    font-weight: 700;
                }

                .lh-item label{
                    font-weight: normal;
                    display: inline-block;
                    font-size: 18px;
                    margin-bottom: 0;
                }

                .lh-item input {
                    -webkit-appearance: auto !important;
                    background: white;
                    width: auto;
                    height: auto;
                    float: none;
                    margin-bottom: 0;
                    border: 1px solid #e2e2e2;
                    font-size: 18px;
                    padding: 0;
                }

                .lh-item input[type="range"] {
                    display: inline-block;
                }

                .lh-item select {
                    width: auto;
                    border: 1px solid #e2e2e2;
                    font-size: 16px;
                    display: inline-block;
                    background-color: #f8f8f8;
                    color: #aaaaaa;
                    padding: 0 30px 0 2px;
                    border-radius: 3px;
                    height: 30px;
                    line-height: 28px;
                    outline: none;
                    appearance: none;
                    background-image: url();
                    background-position: center right;
                    background-repeat: no-repeat;
                }

                .lh-item button {
                    border: 1px solid #e2e2e2;
                    font-size: 16px;
                    padding: 2px 5px;
                    border-radius: 10px;
                    color: #000;
                    background-color: #f0f0f0;
                }

                .lh-item button:hover {
                    background-color: #3e97eb;
                    color: white;
                }

                @keyframes hover-color {
                    from {
                        border-color: #e2e2e2;
                    }
                    to {
                        border-color: #3e97eb;
                    } 
                }

                .lh-item input[type="checkbox"] {
                    display: none;
                    position: absolute;
                }

                .lh-item input[type="checkbox"]+span {
                    position: relative;
                    padding-left: 30px;
                    cursor: pointer;
                }

                .lh-item input[type="checkbox"]+span:hover:before {
                    animation-duration: 0.4s;
                    animation-fill-mode: both;
                    animation-name: hover-color;
                }

                .lh-item input[type="checkbox"]+span:before {
                    position: absolute;
                    top: 2px;
                    left: 5px;
                    display: inline-block;
                    width: 20px;
                    height: 20px;
                    content: "";
                    border: 1px solid #e2e2e2;
                    border-radius: 3px;
                }

                .lh-item input[type="checkbox"]+span:after {
                    position: absolute;
                    display: none;
                    content: "";
                    top: 4px;
                    left: 12px;
                    box-sizing: border-box;
                    width: 6px;
                    height: 12px;
                    transform: rotate(45deg);
                    border-width: 2px;
                    border-style: solid;
                    border-color: #fff;
                    border-top: 0;
                    border-left: 0
                }

                .lh-item input[type="checkbox"]:checked+span:before {
                    animation-name: none;
                    border: #3e97eb;
                    background: #3e97eb;
                }

                .lh-item input[type="checkbox"]:checked+span:after {
                    display: block;
                }
            `);
            
            function showSettings() {
                let html = `<ul class="lh-menu">
                    <li class="lh-item">
                        <label title="勾选此项可关闭移除链接重定向功能">
                            <input type="checkbox" id="excludeAll" />
                            <span style="color: red">*</span>
                        </label>
                        <label title="例外域名对应的链接不会被净化或替换, 误净化或误替换时可添加以不移除重定向">
                            <input list="excludeHosts" id="excludeHost" placeholder="例外域名" />
                        </label>
                        <button id="addExcludeHost" title="添加例外域名">添加</button>
                        <datalist id="excludeHosts"></datalist>
                    </li>
                    <li class="lh-item">
                        <label title="默认500,步进50">
                            文本转链接的字符数限制
                            <br />
                            <input type="range" id="limitText" min="200" max="800" step="50" />
                            <br />
                            <span>500</span>
                        </label>
                    </li>
                    <li class="lh-item">
                        <label title="勾选后所有文本转链接后自动打开">
                            <input type="checkbox" id="autoClickAll" />
                            <span style="color: red">*</span>
                        </label>
                        <label title="域名对应的文本转链接后自动打开">
                            <input list="autoClickHosts" id="autoClickHost" placeholder="自动打开域名" />
                        </label>
                        <button id="addAutoClickHost" title="添加自动打开域名">添加</button>
                        <datalist id="autoClickHosts"></datalist>
                    </li>
                    <li class="lh-item">
                        <button type="button" id="autoClick" title="当前站点的文本转链接后自动打开">自动打开链接文本</button>
                        <select id="autoClickSites" title="自动打开链接文本的站点" style="display: none"></select>
                        <button id="removeAutoClick" title="删除文本转链接后自动打开的站点" style="display: none">删除</button>
                    </li>
                    <li class="lh-item">
                        <label title="维基百科、谷歌开发者、谷歌商店、Github无法访问时请勾选此项">
                            <input type="checkbox" id="jumpToMirror"/>
                            <span>自动切换镜像</span>
                        </label>
                        <label title="Github镜像站选择“是”后替换原链接,需要刷新页面才能重新选择,不勾选自动切换镜像时无法更改此项" style="margin-left: 20px">
                            <input type="checkbox" id="replaceMirror" />
                            <span>替换为镜像链接</span>
                        </label>
                    </li>
                    <li class="lh-item lh-target-blank">
                        <label title="启用新标签打开链接功能">
                            <input id="addBlank" type="checkbox" />
                            <span>新标签打开链接</span>
                        </label>
                        <label title="新标签打开相对链接,不勾选新标签打开时无法更改此项" style="margin-left: 20px">
                            <input id="relative" type="checkbox" />
                            <span>相对链接</span>
                        </label>
                    </li>
                    <li class="lh-item">
                        <button type="button" id="defaultTarget" title="当前站点的链接保持默认打开方式,优先级最高,多用于文档教程类网站">保持链接默认打开方式</button>
                        <select id="defaultTargetSites" title="链接保持默认打开方式的站点" style="display: none"></select>
                        <button id="removeDefault" title="删除链接默认打开方式的站点" style="display: none">删除</button>
                    </li>
                </ul>`;

                Swal.fire({
                        title: '链接助手配置',
                        html,
                        footer: '<div class="lh-footer">前往<a href="https://github.com/oneNorth7/LinkHelper" target="_blank" style="color: #3e97eb">项目主页</a>查看<a href="https://mp.weixin.qq.com/s?__biz=Mzg2MjU4MDM2NQ==&mid=2247486650&idx=1&sn=d4373164fbc1df2d4399ebfb542a44fe&chksm=ce04f758f9737e4e296d394f71d59a15948bd150f1b2d6746658a085be7aa716b5d54f81a1c5&token=2052139541&lang=zh_CN#rd" target="_blank" style="color: #36be63">详细说明</a>,有问题或建议请<a href="https://greasyfork.org/zh-CN/scripts/422773-链接助手/feedback#post-discussion" target="_blank" style="color: red">反馈</a></div>',
                        showCloseButton: true,
                        showConfirmButton: false,
                        customClass: {
                            popup: "lh-popup",
                            content: "lh-content",
                            closeButton: "lh-close"
                        },
                        didRender: () => {
                            
                            checkbox("excludeAll");
                            
                            datainput("excludeHost", "#addExcludeHost", "例外");
                            
                            $("#addExcludeHost")
                            .on("mouseenter", o => $("#excludeHost").blur());
                            
                            let len = t.get("textLength", 500);
                            if(len != 500) $("#limitText").val(len).nextAll("span").text(len);
                            $("#limitText").on("input", o => $(o.target).nextAll("span").text($(o.target).val()));
                            
                            checkbox("autoClickAll");
                            
                            datainput("autoClickHost", "#addAutoClickHost", "自动打开");
                            
                            $("#addAutoClickHost")
                            .on("mouseenter", o => $("#autoClickHost").blur());
                            
                            select("autoClickSites", "#autoClick");
                            
                            $("#removeAutoClick").on("click", o => {
                                let autoClickSites = t.get("autoClickSites", []),
                                    select = $("#autoClickSites"),
                                    selected = select[0].selectedOptions[0],
                                    value = selected.value;
                                if (confirm(`是否删除自动打开站点[${value}]`)) {
                                    autoClickSites = autoClickSites.filter(s => s != value)
                                    selected.remove();
                                    if (autoClickSites.length)
                                        t.set("autoClickSites", autoClickSites);
                                    else {
                                        t.delete("autoClickSites");
                                        $("#autoClick").show();
                                        select.hide();
                                        $("#removeAutoClick").hide();
                                    }
                                    if (value === locHost)
                                        $("#autoClick").show();
                                }
                            });
                            
                            $("#autoClick").on("click", o => {
                                let autoClickSites = t.get("autoClickSites", []),
                                    select = $("#autoClickSites");
                                autoClickSites.push(locHost);
                                select.append(`<option value="${locHost}" selected>${locHost}</option>`);
                                t.set("autoClickSites", autoClickSites);
                                $("#autoClick").hide();
                                select.show();
                                $("#removeAutoClick").show();
                            });
                            
                            checkbox("jumpToMirror");

                            checkbox("replaceMirror", true);
                            
                            checkbox("isAddBlank", true, "#addBlank");

                            checkbox("relative");
                            
                            select("defaultTargetSites", "#defaultTarget, .lh-target-blank");
                            
                            $("#removeDefault").on("click", o => {
                                let defaultSites = t.get("defaultTargetSites", []),
                                    select = $("#defaultTargetSites"),
                                    selected = select[0].selectedOptions[0],
                                    value = selected.value;
                                if (confirm(`是否删除保持默认打开方法站点[${value}]`)) {
                                    defaultSites = defaultSites.filter(s => s != value)
                                    selected.remove();
                                    if (defaultSites.length)
                                        t.set("defaultTargetSites", defaultSites);
                                    else {
                                        t.delete("defaultTargetSites");
                                        $("#defaultTarget").show();
                                        select.hide();
                                        $("#removeDefault").hide();
                                    }
                                    if (value === locHost) {
                                        $("#addBlank").parents("li").fadeIn();
                                        $("#relative").parents("li").fadeIn();
                                        $("#defaultTarget").show();
                                    }
                                }
                            });
                            
                            $("#defaultTarget").on("click", o => {
                                let defaultSites = t.get("defaultTargetSites", []),
                                    select = $("#defaultTargetSites");
                                defaultSites.push(locHost);
                                select.append(`<option value="${locHost}" selected>${locHost}</option>`);
                                t.set("defaultTargetSites", defaultSites);
                                $("#defaultTarget").hide();
                                select.show();
                                $("#removeDefault").show();
                                $("#addBlank").parents("li").fadeOut();
                                $("#relative").parents("li").fadeOut();
                            });
                        },
                        willClose: () => {
                            let len = $("#limitText").val();
                            if (len != "500" || t.get("textLength", false)) t.set("textLength", len);
                        },
                    });
                
                function checkbox(name, value = false, selector = "#" + name, disableFunc = () => {}) {
                    let checked = t.get(name, value),
                        current = $(selector),
                        next = current.parent().next().children();
                    current.prop("checked", checked);
                    if (next.length) {
                        if (next.prop("type") === "checkbox") checked = !checked;
                        next.prop("disabled", checked) && disableFunc();
                    }
                    
                    current.on("change", o => {
                        let checked = o.target.checked;
                        t.set(name, checked);
                        if (next.length) {
                            if (next.prop("type") === "checkbox") checked = !checked;
                            next.prop("disabled", checked) && disableFunc();
                        }
                    });
                }
                
                function select(name, hide = "") {
                    let data = t.get(name, []),
                        select = $("#" + name);
                    if (data.length) {
                        for (let d of data) {
                            if (d === locHost) {
                                hide && $(hide).hide();
                                select.append(`<option value="${d}" selected>${d}</option>`);
                            } else
                                select.append(`<option value="${d}">${d}</option>`);
                        }
                        select.show().next("button").show();
                    }
                }
                
                function datalist(name) {
                    let data = t.get(name, []),
                        list = $("#" + name);
                    if (data.length) {
                        for (let d of data)
                        list.append(`<option value="${d}"></option>`);
                    }
                }
                
                function datainput(inputId, btnSlt, title) {
                    let name = inputId + "s",
                        inputSlt = "#" + inputId;
                    datalist(name);
                    
                    
                    $(inputSlt).on("change", o => {
                        let data = t.get(name, []),
                            host = o.target.value.trim();
                        
                        if (data.includes(host))
                            $(btnSlt).prop("title", `删除${title}域名`).text("删除").one("click", o => {
                                if (confirm(`是否删除${title}域名<${host}>?`))
                                    del(name, inputSlt);
                                else
                                    $(inputSlt).val("");
                            });
                        else
                            $(btnSlt).prop("title", `添加${title}域名`).text("添加").one("click", o => add(name, inputSlt));
                    });
                }
                
                function add(name, selector) {
                    let node = $(selector),
                        value = node.val();
                    if (value && !value.includes("example.com") && value.match(/^[-\w]+(?:\.[-\w]+)+$/)) {
                        let data = t.get(name, []),
                            list = $("#" + name);
                        data.push(value);
                        t.set(name, data);
                        node.val("");
                        list.append(`<option value="${value}"></option>`);
                        t.showNotice(`添加域名<${value}>成功!`);
                    } else t.showNotice("请输入有效域名!");
                }
                
                function del(name, selector) {
                    let node = $(selector),
                        value = node.val();
                    if (value && !value.includes("example.com") && value.match(/^[-\w]+(?:\.[-\w]+)+$/)) {
                        let data = t.get(name, []),
                            list = $("#" + name);
                        data = data.filter(d => d != value);
                        t.set(name, data);
                        node.val("");
                        $("#" + name + `>option[value="${value}"]`).remove();
                        t.showNotice(`删除域名<${value}>成功!`);
                    } else t.showNotice("请输入有效域名!");
                }
            }
            
            t.registerMenu('🔗配置', showSettings);
            
            async function listener(obj) {
                let e = obj.originalEvent.explicitOriginalTarget || obj.originalEvent.target,
                    isTextToLink = false, isInput = false;
                if (e && !e.href) {
                    let flag = true,
                        selectNode = null;
                    for (
                        let current = e, limit = 5;
                        current.localName !== "html" && current.localName !== "body" && limit > 0;
                        current = current.parentElement, limit--
                    ) {
                        if (current.localName === "a") {
                            e = current;
                            break;
                        } else if (
                            ["code", "pre"].some(
                                (tag) => tag === current.localName
                            )
                        ) {
                            let selection = getSelection(),
                                text = selection.toString();
                            if (url_regexp.test(text))
                                selectNode =
                                    selection.anchorNode || selection.focusNode;
                            else flag = false;
                            break;
                        } else if (['input', 'textarea'].some((tag) => tag === current.localName) && current.className == 'direct-input') {
                            let text = t.clean(current.value.replace(/点/g, '.').replace(/冒号/g, ":").replace(/再?斜杠/g, "/").replace("一八九", "189").replace("康姆", "com").replace(/[码碼]/, ":"), [/[\u4e00-\u9fa5\u201c\u201d\uff08\uff09\u3008-\u3011]+/g, /^[::]/]),
                                result = url_regexp.exec(text);
                            if (result) {
                                selectNode = document.createTextNode(text);
                                isInput = true;
                            }
                            else flag = false;
                            break;
                        }
                    }

                    if (e.localName !== "a" && flag) {
                        let node = selectNode || e;
                        if (node && node.nodeValue) e = text2Link(node, isInput);
                        if (e)
                            isTextToLink = true;
                    }
                }

                if (e && e.localName === "a" && e.href) {
                    let a = e, isPrevent = false, isCancel = false;
                    if (/^magnet:\?xt=urn:btih:|^ed2k:\/\/\|file\||^thunder:\/\//i.test(a.href)) {
                        $(a).removeAttr('target');
                        if (isTextToLink) a.click();
                        return;
                    }
                    
                    if (a.host === "pan.baidu.com" && a.hash.startsWith("#/transfer/send?surl="))
                        return;

                    let pan = YunDisk.sites[YunDisk.mapHost(a.host)];
                    if (!t.get("excludeAll", false)) {         
                        if (locHost == "bbs.nga.cn" || locHost == "nga.178.com" || locHost == "ngabbs.com") {
                            if (!(a.host == "bbs.nga.cn" || a.host == "nga.178.com" || a.host == "ngabbs.com"))
                                if (a.attributes.onclick && a.attributes.onclick.nodeValue.startsWith("ubbcode.showUrlAlert(event,this)"))
                                    a.onclick = null;
                        }
                        
                        if (locHost == "www.youtube.com" && a.href.includes("www.youtube.com/redirect?")) {
                            if (!a.style.padding) {
                                $("#secondary-links.ytd-c4-tabbed-header-renderer a.ytd-c4-tabbed-header-renderer").css({padding: "10px 10px 10px 2px", lineHeight: 0, display: "inline-block"});
                                $("#secondary-links.ytd-c4-tabbed-header-renderer a.ytd-c4-tabbed-header-renderer:first-child").css("padding-left", "10px");
                            }
                            a.classList.remove("yt-simple-endpoint");
                        }
                        
                        if (locHost == "www.facebook.com") {
                            a.onclick = function() { return false; };
                            if (a.host !== "github.com" || !t.get("jumpToMirror", false))
                                t.open(a.href);
                        }

                        if (!(pan || locHost === "blog.csdn.net"
                              || t.get("excludeHosts", []).some(s => a.host.includes(s))
                              || cleanRedirectLink(a))
                            ) {
                            let text = a.textContent.trim().replace(/…$/, "");
                            if (RegExp("^(" + http_re_str + ")$").test(text)) {
                                if (isLinkText(a)) {
                                    t.title(a, '【替换】');
                                    a.href = t.http(text, true);
                                    t.increase();
                                } else if (locHost == "twitter.com" && a.host == "t.co")
                                    a.href = t.http(text, true);
                                else if (!isTextToLink && !a.parentElement.className.includes("text2Link") && locHost !== "www.facebook.com" && a.host != "download.downsx.org" && isDifferent(a)) {
                                    a.onclick = function() { return false; };
                                    isPrevent = true;
                                    await t.confirm("是否使用链接文本替换目标链接后打开?",
                                            () => {
                                                // 是
                                                let linkTextPrefixes = t.get("linkTextPrefixes", []),
                                                    reg = /(?:http|https|\/|\%2F).*?\?.+?=|.*?\?/,
                                                    result = reg.exec(a.href);
                                                if (result) {
                                                    linkTextPrefixes.push(result[0]);
                                                    t.set("linkTextPrefixes", linkTextPrefixes);
                                                }
                                                t.title(a, '【替换】');
                                                a.href = t.http(text, true);
                                                t.increase();
                                            },
                                            () => {
                                                // 否
                                            },
                                            () => {
                                                // 取消
                                                isPrevent = false;
                                                isCancel = true;
                                                a.onclick = null;
                                    });
                                }
                            }
                        }
                    }
                    
                    if (!obj.originalEvent.button || isTextToLink) {
                        if (t.get("jumpToMirror", false)) {
                            if (a.host.includes("wikipedia.org")) {
                                // 维基百科
                                if (locHost !== "www.bing.com" && !locHost.includes("www.google."))
                                    a.host = a.host.replace(
                                        "wikipedia.org",
                                        "wiki.hancel.org"
                                    );
                            } else if (a.host.includes("developers.google.com")) {
                                // 谷歌开发者
                                if (!locHost == "developers.google.com")
                                    a.host = a.host.replace(
                                        "developers.google.com",
                                        "developers.google.cn"
                                    );
                            } else if (locHost !== "github.com" && a.host === "github.com" && !a.pathname.startsWith("/login") && !a.dataset.cancel) {
                                // Github
                                let mirrors = [
                                                ["fastgit", "hub.fastgit.org"],
                                                ["cnpmjs", "github.com.cnpmjs.org"],
                                                ["rc1844", "github.rc1844.workers.dev"]
                                              ],
                                    rand = t.rand(1, 9) % 3,
                                    mirror = mirrors[rand],
                                    next = mirrors[(rand + 1) % 3];
                                a.onclick = function() { return false; };
                                isPrevent = true;
                                await t.confirm(`是否跳转到【${mirror[0]}】镜像站?`,
                                                () => {
                                                    // 是
                                                    if (t.get("replaceMirror", true)) {
                                                        t.title(a, `已替换为【${mirror[0]}】镜像链接,请不要登录帐号!!!`);
                                                        a.host = a.host.replace("github.com", mirror[1]);
                                                    } else {
                                                        t.open(a.href.replace("github.com", mirror[1]));
                                                        isPrevent = false;
                                                    }
                                                    setTimeout(() => t.showNotice("镜像站请不要登录账号!!!\n镜像站请不要登录账号!!!\n镜像站请不要登录账号!!!"), 1000);
                                                },
                                                () => {
                                                    // 否
                                                    if (locHost === "blog.csdn.net") t.open(a.href);
                                                },
                                                () => {
                                                    // 取消
                                                    isPrevent = false;
                                                    isCancel = true;
                                                    a.onclick = null;
                                                    a.dataset.cancel = true;
                                                });
                            } else if (a.host.includes("chrome.google.com")) {
                                // 谷歌应用商店
                                if (isChromium) {
                                    a.onclick = function() { return false; };
                                    isPrevent = true;
                                    await t.confirm("是否跳转到【crx4chrome】镜像站?",
                                                    () => {
                                                        // 是
                                                        t.title(a);
                                                        a.href = a.href.replace(/chrome\.google\.com\/webstore\/detail[\/\w\-%]*(?=\w{32})/i, "www.crx4chrome.com/extensions/");
                                                    },
                                                    () => {
                                                        // 否
                                                        if (locHost === "blog.csdn.net") t.open(a.href);
                                                    },
                                                    () => {
                                                        // 取消
                                                        isPrevent = false;
                                                        isCancel = true;
                                                        a.onclick = null;
                                    });
                                }
                            }
                        }
                    }
                    
                    pan = YunDisk.sites[YunDisk.mapHost(a.host)];
                    if (pan) YunDisk.addCode(a, isInput);
                    
                    if (isTextToLink) {
                        let isClicked = false;
                        if (t.get("autoClickAll", false) || pan
                            || t.get("autoClickHosts", []).concat(YunDisk.pans).some(h => h === a.host)
                            || t.get("autoClickSites", []).some(h => h === locHost)) {
                            if (!isCancel) {
                                a.click();
                                isClicked = true;
                            }
                        }

                        if (isInput) {
                            if (!isClicked) a.click();
                            $('#L_DirectInput').val("");
                        }
                    }
                    
                    if (/^https?:\/\/www\.nruan\.com\/(page\/\d+)?/i.test(a.href))
                        $(a).removeAttr('target');
                    
                    add_blank(a);
                    
                    if (isPrevent) {
                        a.onclick = null;
                        a.click();
                    }
                }
            }
            
            let url_regexp_g = new RegExp(url_regexp, "ig");

            function text2Link(node, isInput) {
                let text = node.nodeValue;
                if (!["flashget://", "qqdl://", "tg://", "ss://", "ssr://", "vmess://", "trojan://", "115://", "aliyunpan://", "bdpan://", "BDLINK"].some(p => text.includes(p)) && ![/SHA-?(1|256)/i, /MD-?5/i].some(e => e.test(node.parentElement && node.parentElement.parentElement.textContent)) && (text.length < t.get("textLength", 500) || isInput)) {
                    let parent = null;
                    if (locHost === "tieba.baidu.com") {
                        if ((node.parentElement.localName === "div" && node.parentElement.id.match(/^post_content_\d+$/)) ||
                           (node.parentElement.localName === "span" && node.parentElement.className === "lzl_content_main")) {
                            text = node.parentElement.innerText.replace(/\n/g, "<br>");
                            parent = node.parentElement;
                        }
                    }

                    let result = url_regexp_g.test(text),
                        span = null,
                        count = 0,
                        isMail = false;
                    if (result) {
                        span = $("<span class='text2Link'></span>");
                        span.html(
                            text.replace(url_regexp_g, function ($1, $2) {
                                count++;
                                if ($1.includes("@") && !$1.match(/^https?:\/\/|\/@?|^magnet:/))
                                    return (isMail = true, `<a class="text2Link" href="mailto:${$1}">${$1}</a>`);
                                return $1.startsWith("http")
                                       ? `<a href="${$1}" target="_blank">${$1}</a>`
                                       : /^thunder:\/\//i.test($1)
                                       ? `<a href="${$1}" title="使用迅雷下载">${$1}</a>`
                                       : $1.includes("ed2k")
                                       ? `<a href="${$1}" title="使用BT软件下载">${$1}</a>`
                                       : $1.match(magnet_re_str)
                                       ? `<a href="magnet:?xt=urn:btih:${$1.includes("&tr=") ? $1.replace("magnet:?xt=urn:btih:", "") : $2.replace("magnet:?xt=urn:btih:", "")}" title="使用BT软件下载">${$1}</a>`
                                       : /^(?:\/?s)?\/[\w\-_]{23}$|^(?:\/?s)?\/\w{7,8}$/.test($1)
                                       ? `<a href="https://pan.baidu.com/s/${$1.replace(/^(?:\/?s)?\//, "")}" target="_blank">${$1}</a>`
                                       : `<a href="https://${$1}" target="_blank">${isInput ? "https://" + $1 : $1}</a>`;
                            })/*.replace(/点/g, '.')*/
                        );
                        if (parent) $(parent).html(span);
                        else if (isMail) $(node).replaceWith(span.html());
                        else $(node).replaceWith(span);
                    }
                    
                    if (count) t.increase();
                    
                    return !isMail && span && span.children("a")[0];
                }
            }

            function isLinkText(a) {
                let keywords = [
                    "jump.bdimg.com/safecheck/index?url=",
                    "jump2.bdimg.com/safecheck/index?url=",
                    "iphone.myzaker.com/zaker/link.php?pk=",
                    "www.coolapp.wang/goto/",
                ],
                    linkTextPrefixes = t.get("linkTextPrefixes", []);
                return keywords.some((k) => a.href.includes(k)) || linkTextPrefixes.some((k) => a.href.includes(k));
            }
            
            function isDifferent(a) {
                if (/(?:http|https|\/|\%2F).*?\?.+?=|.*?\?/.test(a.href)) {
                    let hash = a.hash, search = a.search, password = t.search(a);
                    a.hash = "";
                    if (password) a.search = "";
                    let text = decodeURIComponent(a.innerText.trim()).toLowerCase().replace(/^https?:\/\/|\/$/, '').replace(hash, ''),
                        href = decodeURIComponent(a.href).toLowerCase().replace(/^https?:\/\/|\/$/, '');
                    a.hash = hash;
                    if (password) a.search = search;
                    return !(text.includes('...') || !text.includes('/') || text == href);
                }
                return false;
            }

            let excludes = [
                "v.qq.com",
                "v.youku.com",
                "blog.csdn.net",
                "cloud.tencent.com",
                "translate.google.com",
                "domains.live.com",
                "www.iconfont.cn",
                "www.kdocs.cn",
                "help.aliyun.com",
                "service.weibo.com",
                "zhannei.baidu.com",
                "pc.woozooo.com",
                "play.google.com",
                "nimg.ws.126.net",
                "iapk.cc",
                "www.microsofttranslator.com",
                "whois.chinaz.com",
                "yandex.com",
                "lixian.vip.xunlei.com",
                "fanyi.baidu.com",
                "lanjing.jd.com",
                "image.baidu.com",
                "detail.1688.com",
                "api.",
                "passport.",
                "www.360kuai.com",
                "zhidao.baidu.com",
                "image.",
                "img.",
                "pic.",
            ];

            function cleanRedirectLink(a) {
                // 小众软件
                if (locHost == "www.appinn.com" && (a.search.includes("ref=appinn") || a.hash.includes("ref=appinn"))) {
                    t.title(a, "【净化】");
                    a.search = a.search.replace(/[?&]ref=appinn$/, '');
                    a.hash = a.hash.replace(/[#&]ref=appinn$/, '');
                    t.increase();
                    return true;
                }
                
                // 净化跳转链接
                let hosts = ["dalao.ru", "niao.su", "iao.su", "nicelinks.site", "www.appinn.com", "support.qq.com", locHost];
                for (let h of hosts) {
                    let reg = RegExp(`\\?(?:utm_source=)?${h}$`), result = reg.exec(a.href);
                    if (result) {
                        t.title(a, "【净化】");
                        a.href = a.href.replace(result[0], "");
                        t.increase();
                    }
                }
                
                // 语雀
                if (locHost === "www.yuque.com" && a.search.includes("fileGuid=")) {
                    t.title(a, "【净化】");
                    a.search = a.search.replace(/[?&]fileGuid=\w{16}$/, "");
                    t.increase();
                    return true;
                }
                
                // 那些免费的砖
                if (locHost === "www.thosefree.com") {
                    if (a.search.match("\\?from=thosefree\\.com")) {
                        t.title(a, "【净化】");
                        a.search = "";
                    }
                }
                if (!(["login", "logout", "signin", "signup", "signout", "auth", "oauth", "translate.google.com", "/images/"].some(k => a.href.includes(k))
                    || /登录|登入|登出|退出|注册|login|logout|signin|signup|signout/i.test(a.textContent)
                    || excludes.some((s) => a.host.includes(s)))
                ) {
                    let reg = new RegExp("^((?:http|https|\\/|\\%2F)(?:.*?[?&].+?=|.*?[?&]))(" + http_re_str + ")", "i"),
                        result = reg.exec(decodeURIComponent(a.href));
                    if (result) {
                        let temp = decodeURIComponent(
                            decodeURIComponent(result[2])
                        ).replace(/https?:\/\//, "");
                        if (!decodeURIComponent(locHref).replace(/https?:\/\//, "").includes(temp.split("&")[0])) {
                            if (!/t\d+\.html/i.test(temp)) {
                                let href = decodeURIComponent(
                                    decodeURIComponent(
                                        t.http(result[2])
                                    )
                                );

                                t.title(a, "【净化】");
                                if (!href.includes("?") && href.includes("&") || a.host.includes("google.com"))
                                    a.href = href.split("&")[0];
                                else a.href = href.replace(/______/g, ".");
                            }
                            t.increase();
                            return true;
                        }
                    } else {
                        reg = new RegExp("((?:http|https|\\/|\\%2F)(?:.*[?&].+?=|.*[?&]|.*\\/(?:go|goto|link)\\/))(" + base64_re_str + ")", "i");
                        result = reg.exec(decodeURIComponent(a.href));
                        if (result) {
                            try {
                                let temp = decodeURIComponent(escape(atob(result[2])));
                                if (temp.match("^" + http_re_str + "$")) {
                                    t.title(a, '【Base64】');
                                    a.href = temp;
                                    t.increase();
                                    return true;
                                }
                            } catch (err) {}
                        }
                    }
                }
            }

            // 给链接添加[target="_blank"]属性
            function add_blank(a) {
                let defaultTargetSites = t.get("defaultTargetSites", []),
                    isAddBlank = t.get("isAddBlank", true),
                    isDefault = defaultTargetSites.some((s) => s == location.host);
                if (isAddBlank && !isDefault) {
                    let result =
                            a.href == "" || a.target == "_blank" ||
                            /javascript[\w:;()]+/.test(a.href) ||
                            /\/\w+-\d+-\d+\.html|.+page\/\d+|category-\d+_?\d*/.test(
                                a.href
                            ) ||
                            /[前后後上下首末].+[页頁篇张張]|^\.*\s*\d+\s*\.*$|^next$|^previous$|^[<>]$/i.test(
                                a.innerText
                            ) ||
                            ["prev", "next"].some(
                                (r) =>
                                    r == a.attributes.rel &&
                                    a.attributes.rel.nodeValue
                            ) ||
                            ["prev", "next", "nxt"].some((r) =>
                                a.className.includes(r)
                            ) ||
                            a.href == location.origin + "/" ||
                            /^#.+/.test(a.attributes.href && a.attributes.href.nodeValue) ||
                            a.href.endsWith(".user.js"),
                        relative = t.get("relative", false);
                    if (!relative && locHost !== "www.amazon.com")
                        result =
                            result ||
                            !/^(?:https?:|\/\/).+/.test(
                                a.attributes.href && a.attributes.href.nodeValue
                            );
                    if (!result) a.target = "_blank";
                }
            }
            
            // 添加链接直达输入框
            let addDirectTo = t.get("addDirectTo", true);
            if (addDirectTo) add_direct();
            let menuID2 = t.registerMenu(`${addDirectTo ? "[✔]" : "[✖]"}显示链接直达输入框`, directToMenu);
            
            function directToMenu() {
                addDirectTo = !addDirectTo;
                t.set("addDirectTo", addDirectTo);
                t.unregisterMenu(menuID2);
                menuID2 = t.registerMenu(`${addDirectTo ? "[✔]" : "[✖]"}显示链接直达输入框`, directToMenu);
                if (addDirectTo) add_direct();
                else if ($("#L_DirectTo").length) $("#L_DirectTo").remove();
            }
            
            function add_direct() {
                $('body').append(`<div id="L_DirectTo" class="l-direct-to">
                                    <input type="image" src="" alt="直达" id="L_DirectButton" class="direct-button" />
                                    <span title="粘贴或输入包含单链接的文本后点击输入框直达链接">
                                        <input type="text" id="L_DirectInput" class="direct-input" placeholder="粘贴或输入链接" />
                                    </span>
                                </div>`);
                GM_addStyle(`#L_DirectTo input {
                                outline: none;
                            }
                            #L_DirectTo {
                                position: fixed;
                                left: 0;
                                top: 25%;
                                z-index: 9999;
                            }
                            #L_DirectButton {
                                width: 30px;
                                height: 30px;
                                position: absolute;
                                z-index: 1;
                                left: -16px;
                                user-select: none;
                                padding: 0;
                                border: none;
                                background: transparent;
                                box-shadow: none;
                            }
                            #L_DirectInput {
                                width: 300px;
                                height: 22px;
                                position: absolute;
                                left: -350px;
                                border: 3px solid #b52bff;
                                border-radius: 20px;
                                background-color: #99adf7;
                                padding-left: 10px;
                                box-sizing: content-box;
                                user-select: none;
                                padding: 0;
                                box-shadow: none;
                            }
                            #L_DirectInput::placeholder {
                                color: #333;
                            }
                            `);
                
                
                
                let direct = $('#L_DirectTo'),
                    input = $('#L_DirectInput'),
                    button = $('#L_DirectButton');
                
                input.on('click', obj => listener(obj));
                input.on('paste', () => {
                    setTimeout(() => input[0].click(), 500);
                });
                
                input.val('');

                button.on('click', () => {
                        if (button.hasClass('open')) {
                            input.animate({ left: '-350px' }).val('').blur();
                            button.removeClass('open');
                            button.prop('title', '点击展开输入框');
                        } else {
                            button.addClass('open');
                            input.focus().animate({ left: '28px' });
                            button.prop('title', '点击收起输入框');
                        }
                    })
                    .on('mouseover', () => {
                        button.animate({ left: 0 });
                        if (!button.hasClass('open')) button.click();
                    })
                    .on('mouseout', () => {
                        if (!button.hasClass('open'))
                            button.animate({ left: '-16px' });
                    });
                
                let timeId = null;
                direct.on('mouseout', () => {
                        timeId = setTimeout(() => {
                            if (button.hasClass('open')) {
                                button.click();
                                button.animate({ left: '-16px' });
                            }
                        }, 5000);
                    })
                    .on('mouseover', () => {
                        if (timeId) clearTimeout(timeId);
                    });
            }
        }
    }
});