链接管理

绕过搜索引擎(百度、搜狗、360、必应、谷歌)搜索结果中的重定向链接,直链访问原始网站,删除网站重定向到安全页面,自动跳转中文文档,减少操作步骤和响应时间;长期维护、PC+手机全平台支持:CSDN+掘金+简书+知乎+知乎专栏+百度贴吧+开源中国+码云gitee+扣丁leetcode+51CTO+百度搜索+360搜索+搜狗搜索+必应搜索+423down+酷安+eslint+微软文档+火狐MDN+tampermonkey文档;

您需要先安裝使用者腳本管理器擴展,如 TampermonkeyGreasemonkeyViolentmonkey 之後才能安裝該腳本。

You will need to install an extension such as Tampermonkey to install this script.

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyUserscripts 後才能安裝該腳本。

你需要先安裝一款使用者腳本管理器擴展,比如 Tampermonkey,才能安裝此腳本

您需要先安裝使用者腳本管理器擴充功能後才能安裝該腳本。

(我已經安裝了使用者腳本管理器,讓我安裝!)

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

(我已經安裝了使用者樣式管理器,讓我安裝!)

// ==UserScript==
// @name         链接管理
// @version      1.3.27
// @namespace    airbash/LinkManager
// @homepageURL  https://github.com/AirBashX/UserScript
// @author       airbash
// @description  绕过搜索引擎(百度、搜狗、360、必应、谷歌)搜索结果中的重定向链接,直链访问原始网站,删除网站重定向到安全页面,自动跳转中文文档,减少操作步骤和响应时间;长期维护、PC+手机全平台支持:CSDN+掘金+简书+知乎+知乎专栏+百度贴吧+开源中国+码云gitee+扣丁leetcode+51CTO+百度搜索+360搜索+搜狗搜索+必应搜索+423down+酷安+eslint+微软文档+火狐MDN+tampermonkey文档;
// @match        *://link.csdn.net/*
// @match        *://link.juejin.cn/*
// @match        *://juejin.cn/*
// @match        *://www.jianshu.com/p/*
// @match        *://www.jianshu.com/go-wild?*
// @match        *://*.zhihu.com/*
// @match        *://tieba.baidu.com/*
// @match        *://*.oschina.net/*
// @match        *://gitee.com/*
// @match        *://leetcode.cn/link/*
// @match        *://blog.51cto.com/*
// @match        *://*.baidu.com/*
// @match        *://m.so.com/s?*
// @match        *://www.so.com/s?*
// @match        *://www.sogou.com/web?*
// @match        *://*.bing.com/search*
// @match        *://www.423down.com/*
// @match        *://www.coolapk.com/link*
// @match        *://*.eslint.org/docs/latest/*
// @match        *://learn.microsoft.com/*
// @match        *://developer.mozilla.org/*
// @match        *://www.tampermonkey.net/*
// @icon         
// @grant        GM_registerMenuCommand
// @grant        GM_setValue
// @grant        GM_getValue
// @grant        GM_xmlhttpRequest
// @grant        GM_info
// @license      GPL-3.0
// @run-at       document-body
// ==/UserScript==
///<reference path="./tampermonkey-reference.d.ts" />

(function () {
    "use strict";

    /**
     * 安全页面重定向列表
     * @type       {安全页面网站}
     */
    const safePages = [
        {
            //https://blog.csdn.net/weixin_50829653/article/details/118119039
            //https://link.csdn.net/?from_id=118119039&target=https://baidu.com
            name: "CSDN",
            url: "link.csdn.net",
            handlers: [
                {
                    type: "forward",
                    start: "&target=",
                },
            ],
        },
        {
            //https://link.zhihu.com/?target=https://www.apifox.cn/
            name: "知乎",
            url: "link.zhihu.com/?target=",
            handlers: [
                {
                    type: "forward",
                    start: "?target=",
                },
            ],
        },
        {
            //https://link.juejin.cn/?target=https://baidu.com
            name: "掘金",
            url: "link.juejin.cn/?target=",
            handlers: [
                {
                    type: "forward",
                    start: "?target=",
                },
            ],
        },
        {
            //https://www.jianshu.com/go-wild?ac=2&url=https://baidu.com
            name: "简书",
            url: "www.jianshu.com/go-wild?",
            handlers: [
                {
                    type: "forward",
                    start: "&url=",
                },
            ],
        },
        {
            //https://www.oschina.net/action/GoToLink?url=https://baidu.com
            name: "开源中国",
            url: "www.oschina.net/action/GoToLink?url=",
            handlers: [
                {
                    type: "forward",
                    start: "GoToLink?url=",
                },
            ],
        },
        {
            //https://gitee.com/link?target=https://baidu.com
            name: "码云",
            url: "gitee.com/link?target=",
            handlers: [
                {
                    type: "forward",
                    start: "?target=",
                },
            ],
        },
        {
            //https://leetcode.cn/link/?target=https://www.baidu.com
            name: "扣丁",
            url: "leetcode.cn",
            handlers: [
                {
                    type: "forward",
                    start: "link/?target=",
                },
            ],
        },
        {
            //https://blog.51cto.com/u_15127617/4063137
            //https://blog.51cto.com/transfer?https://baidu.com
            name: "51CTO",
            url: "blog.51cto.com/transfer?",
            handlers: [
                {
                    type: "forward",
                    start: "transfer?",
                },
            ],
        },
        {
            //https://www.coolapk.com/link?url=https%3A%2F%2Flanzoux.com
            name: "酷安",
            url: "www.coolapk.com/link?url=",
            handlers: [
                {
                    type: "forward",
                    start: "link?url=",
                },
            ],
        },
        {
            //https://mail.qq.com/cgi-bin/readtemplate?t=safety&check=false&gourl=https://www.baidu.com
            name: "酷安",
            url: "mail.qq.com/cgi-bin/readtemplate?",
            handlers: [
                {
                    type: "forward",
                    start: "gourl=",
                },
            ],
        },
    ];

    /**
     * 去除页面重定向列表
     * @type       {重定向网站}
     */
    const websites = [
        {
            //https://www.zhihu.com/question/465346075/answer/2048804228
            //https://zhuanlan.zhihu.com/p/95937067
            name: "知乎",
            url: "zhihu.com",
            handlers: [
                {
                    selector: ".external,.LinkCard",
                    start: "?target=",
                    type: "sub",
                },
            ],
        },
        {
            //https://juejin.cn/post/6844903688524267534
            name: "掘金",
            url: "juejin.cn",
            handlers: [
                {
                    selector: "a[href]",
                    start: "?target=",
                    type: "sub",
                },
            ],
        },
        {
            //https://www.jianshu.com/p/cf7dc734dd6d
            name: "简书",
            url: "www.jianshu.com/p/",
            handlers: [
                {
                    selector: "a[href]",
                    start: "go?to=",
                    type: "sub",
                },
            ],
        },
        {
            //https://tieba.baidu.com/p/6313991324
            name: "百度贴吧",
            url: "tieba.baidu.com/p/",
            handlers: [
                //PC版
                {
                    selector: ".j-no-opener-url",
                    start: "jump.bdimg.com",
                    type: "text",
                },
                {
                    selector: ".j-no-opener-url",
                    start: "jump2.bdimg.com",
                    type: "text",
                },
                //手机版
                {
                    selector: ".rich-link",
                    start: "checkurl?url=",
                    end: "&urlrefer=",
                    type: "sub",
                },
            ],
        },
        {
            //https://my.oschina.net/lorryluMyRest/blog/731722
            name: "开源中国",
            url: "oschina.net",
            handlers: [
                {
                    selector: "a[href]",
                    start: "GoToLink?url=",
                    type: "sub",
                },
            ],
        },
        {
            //https://gitee.com/iflytek/iflearner
            name: "gitee",
            url: "gitee.com",
            handlers: [
                {
                    selector: "a[href]",
                    start: "link?target=",
                    type: "sub",
                },
            ],
        },
        {
            //https://www.so.com/s?ie=UTF-8&q=123
            name: "360搜索PC版",
            url: "www.so.com/s",
            handlers: [
                {
                    selector: "a[data-mdurl]",
                    start: "link?m=",
                    attribute: "data-mdurl",
                    type: "attribute",
                },
            ],
        },
        {
            //https://m.so.com/s?q=%E4%BD%A0%E5%A5%BD
            name: "360搜索手机版",
            url: "m.so.com",
            handlers: [
                {
                    selector: "a[href]",
                    start: "jump?u=",
                    end: "&m=",
                    type: "sub",
                },
            ],
        },
        {
            //https://www.423down.com/10579.html
            name: "423down",
            url: "www.423down.com",
            handlers: [
                {
                    selector: "p>a[href]",
                    start: "/go.php?url=",
                    type: "text",
                },
            ],
        },
    ];

    /**
     * 百度单独规则
     */
    if (location.href.includes("m.baidu.com") || location.href.includes("www.baidu.com")) {
        let interval = setInterval(() => {
            let flag = baidu_static();
        }, 100);
    }
    /**
     * 百度静态规则
     */
    function baidu_static() {
        //pc端
        let item1s = document.querySelectorAll("#content_left>div");
        for (let item of item1s) {
            let a = item.querySelector("a");
            if (a.href.includes("www.baidu.com/link?url=")) {
                let url = item.getAttribute("mu");
                //https://www.baidu.com/s?wd=一夜醒来欠地铁600多万?官方回应
                //智能精选
                if (url && url != "null" && !url.includes("nourl.ubs.baidu.com") && !url.includes("nourl.baidu.com")) {
                    a.href = url;
                    console.log(url);
                }
            }
            //single-card-wrapper: https://www.baidu.com/s?ie=UTF-8&wd=es6                          xxx的最新相关信息
            //group-wrapper:       https://www.baidu.com/s?ie=UTF-8&wd=五一消费成绩单折射市场活力     资讯
            let item_news = item.querySelectorAll("[class^=single-card-wrapper] div,[class^=group-wrapper] div");
            for (let item_new of item_news) {
                let data_url;
                let divs = item_new.querySelectorAll("div");
                for (let div of divs) {
                    if ((data_url = div.getAttribute("data-url"))) {
                        let a = item_new.querySelector("a");
                        a.setAttribute("href", data_url);
                    }
                }
            }
        }

        //移动端
        let item2s = document.querySelectorAll("#results>div");
        for (let item of item2s) {
            let rl_link_href = item.querySelector("[rl-link-href]");
            if (rl_link_href && rl_link_href != null) {
                let str = item.getAttribute("data-log");
                let json = JSON.parse(str);
                let url = json.mu;
                if (!url.startsWith("https://ks.baidu.com") && url) {
                    //ks.baidu.com                  https://www.baidu.com/s?word=赵匡胤
                    rl_link_href.setAttribute("rl-link-href", url);
                    let as = item.querySelectorAll("a");
                    for (let a of as) {
                        a.href = url;
                    }
                }
            }
        }
        return false;
    }

    /**
     * 必应单独规则
     * https://www.bing.com/search?q=必应
     */
    if (location.href.includes("bing.com/search")) {
        function handlerAnchor(a) {
            let url_ = a.href;
            if (url_.includes(".bing.com/ck/a?")) {
                // 截取 u= 及其后面的字符
                let tmp = url_.slice(url_.lastIndexOf("u="));
                // 找到 u= 后的第一个 `&`, 精确提取出 u
                tmp = tmp.slice("u=a1".length, tmp.indexOf("&"));
                const paddingNeeded = 4 - tmp.length % 4;
                // 如果长度不是4的倍数,则需要补充'='
                if (paddingNeeded != 4) {
                    tmp += '='.repeat(paddingNeeded);
                }
                try {
                    let ori_url = atob(tmp.replace('-', '+').replace('_', '/'));
                    a.href = ori_url;
                } catch (e) {
                    console.log("Error parsing", tmp);
                }
            }
        }

        function afterLoaded(f_) {
            if (document.readyState === 'loading') {
                document.addEventListener('DOMContentLoaded', f_);
            } else {
                typeof f_ === 'function' && f_();
            }
        }

        afterLoaded(() => {
            console.log("Started");
            document.querySelectorAll('a').forEach(handlerAnchor);
            // 使用 MutationObserver 监控 DOM 变化
            const observer = new MutationObserver((mutationsList) => {
                for (const mutation of mutationsList) {
                    if (mutation.type === 'childList') {
                        mutation.addedNodes.forEach(node => {
                            if (node.nodeType === Node.ELEMENT_NODE) {
                                node.querySelectorAll('a').forEach(handlerAnchor);
                            }
                        });
                    }
                }
            });

            // 观察整个文档的 DOM 变化
            observer.observe(document.body, {
                childList: true,
                subtree: true
            });
        });
    }

    /**
     * 安全页面跳转:处理
     */
    for (let safePage of safePages) {
        if (location.href.includes(safePage.url)) {
            for (let handler of safePage.handlers) {
                let str = location.href.split(handler.start)[1];
                console.log(str);
                let url = decodeURIComponent(str);
                console.log(url);
                location.replace(url);
            }
        }
    }

    /**
     * 页面重定向跳转:处理
     */
    for (let website of websites) {
        if (location.href.includes(website.url)) {
            let time = 0;
            let interval = setInterval(() => {
                if (++time == 100) {
                    clearInterval(interval);
                }
                for (let handler of website.handlers) {
                    let items = document.querySelectorAll(handler.selector);
                    for (let item of items) {
                        //进一步校验需要修改的元素,防止修改错元素
                        if (item.getAttribute("href").includes(handler.start)) {
                            if (handler.type == "sub") {
                                //从属性中截取地址
                                let href = item.getAttribute("href");
                                let start_index = href.indexOf(handler.start) + handler.start.length;
                                let str;
                                if (handler.end != null) {
                                    let end_index = href.indexOf(handler.end);
                                    str = href.substring(start_index, end_index);
                                } else {
                                    str = href.substring(start_index);
                                }
                                let url = decodeURIComponent(str);
                                item.setAttribute("href", url);
                            } else if (handler.type == "attribute") {
                                //从属性中获取地址
                                item.setAttribute("href", item.getAttribute(handler.attribute));
                            } else {
                                //从文本中获取地址
                                item.setAttribute("href", item.innerText);
                            }
                        }
                    }
                }
            }, 100);
        }
    }

    /**
     * 跳转中文文档规则
     */
    const otherSites = [
        {
            //https://eslint.org/docs/latest/user-guide/configuring/configuration-files
            //https://zh-hans.eslint.org/docs/latest/user-guide/configuring/configuration-files
            name: "eslint",
            url: "https://eslint.org/docs/latest/",
            type: "host",
            zh_str: "zh-hans",
        },
        {
            //https://learn.microsoft.com/zh-cn/powershell/scripting/how-to-use-docs
            name: "microsoft",
            url: "https://learn.microsoft.com/",
            type: "pathName",
            zh_str: "zh-cn",
        },
        {
            //https://developer.mozilla.org/zh-CN/
            name: "MDN",
            url: "https://developer.mozilla.org/",
            type: "pathName",
            zh_str: "zh-CN",
        },
        {
            //https://www.tampermonkey.net/faq.php?locale=zh
            name: "tampermonkey",
            url: "https://www.tampermonkey.net",
            type: "search",
            zh_str: "locale=zh",
        },
    ];

    /**
     * GM相关APi的操作
     */
    if (GM_info) {
        const scriptHandler = GM_info.scriptHandler;
        /**
         * 仅在油侯插件上运行,避免无法点击注册开关
         */
        if (scriptHandler == "Tampermonkey" || scriptHandler == "Violentmonkey" || scriptHandler == "ScriptCat" || scriptHandler == "Via") {
            /**
             * 自动跳转中文文档
             */
            if (GM_getValue("forward_zh")) {
                GM_registerMenuCommand("[√]跳转中文文档", function () {
                    GM_setValue("forward_zh", false);
                    location.reload();
                });
                let web_url = location.href;
                for (let otherSite of otherSites) {
                    if (web_url.startsWith(otherSite.url)) {
                        let zh_index = location.href.search(otherSite.zh_str);
                        // 当前页面不是中文文档
                        if (zh_index == -1) {
                            if (otherSite.type == "pathName") {
                                let pathname = location.pathname;
                                let str = pathname.split("/")[1];
                                if (str != otherSite.zh_str) {
                                    let new_url = web_url.replace(str, otherSite.zh_str);
                                    location.replace(new_url);
                                }
                            } else if (otherSite.type == "host") {
                                let host = location.host;
                                let str = host.split(".")[0];
                                if (str != otherSite.zh_str) {
                                    let new_host = otherSite.zh_str + "." + host;
                                    let new_url = web_url.replace(host, new_host);
                                    location.replace(new_url);
                                }
                            } else if (otherSite.type == "search") {
                                let search = location.search;
                                if (search) {
                                    //有参数
                                    let key = otherSite.zh_str.split("=")[0];
                                    let start_index = search.search(key);
                                    let end_index = start_index + otherSite.zh_str.length;
                                    if (start_index != -1) {
                                        //有指定参数
                                        let str = search.substring(start_index, end_index);
                                        if (str != otherSite.zh_str) {
                                            //数值不对
                                            let url = web_url.replace(str, otherSite.zh_str);
                                            location.replace(url);
                                        }
                                    } else {
                                        //没有指定参数
                                        let url = web_url + "&" + otherSite.zh_str;
                                        location.replace(url);
                                    }
                                } else {
                                    //没有参数
                                    let url = web_url + "?" + otherSite.zh_str;
                                    location.replace(url);
                                }
                            }
                        }
                    }
                }
            } else {
                GM_registerMenuCommand("[x]跳转中文文档", function () {
                    GM_setValue("forward_zh", true);
                    location.reload();
                });
            }
        }
    }
})();