// ==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();
});
}
}
}
})();