重定向链接转直链

将页面内所有重定向式的链接替换为直链

  1. // ==UserScript==
  2. // @name Direct Link
  3. // @name:zh-CN 重定向链接转直链
  4. // @description Replace redirect links with direct links
  5. // @description:zh-CN 将页面内所有重定向式的链接替换为直链
  6. // @namespace https://github.com/cilxe/JavaScriptProjects
  7. // @version 0.2.4
  8. // @author cilxe
  9. // @match *://*.youtube.com/*
  10. // @match *://*.zhihu.com/*
  11. // @match *://*.steampowered.com/*
  12. // @match *://*.steamcommunity.com/*
  13. // @match *://*.pixiv.net/*
  14. // @match *://*.vk.com/*
  15. // @match *://*.hoyolab.com/*
  16. // @match *://*.jianshu.com/*
  17. // @match *://*.juejin.cn/*
  18. // @match *://*.epicgames.com/*
  19. // @match *://*.mozilla.org/*
  20. // @match *://*.firefox.com/*
  21. // @match *://*.leetcode.cn/*
  22. // @match *://*.oschina.net/*
  23. // @match *://*.gitee.com/*
  24. // @match *://*.xda-developers.com/*
  25. // @match *://*.sspai.com/*
  26. // @match *://*.gcores.com/*
  27. // @match *://*.deviantart.com/*
  28. // @match *://union-click.jd.com/*
  29. // @match *://*.tmall.com/*
  30. // @match *://s.click.taobao.com/*
  31. // @match *://s.click.tmall.com/*
  32. // @match *://wiki.biligame.com/*
  33. // @match *://*.linkstars.com/*
  34. // @match *://tieba.baidu.com/*
  35. // @match *://ala.baidu.com/*
  36. // @match *://*.linkedin.com/*
  37. // @match *://*.theverge.com/*
  38. // @match *://*.douban.com/*
  39. // @match *://sourceforge.net/*
  40. // @icon 
  41. // @run-at document-start
  42. // @grant GM_registerMenuCommand
  43. // @grant unsafeWindow
  44. // @sandbox JavaScript
  45. // @license MIT
  46. // ==/UserScript==
  47.  
  48. /*
  49. ## Main features
  50. - Replace redirect links with direct links
  51. - Clean SourceForge tracking links
  52. - Additional features via script menu
  53.  
  54. ## Currently supported sites
  55. - youtube.com
  56. - epicgames.com
  57. - mozilla.org / firefox.com (adjust.com)
  58. - hoyolab.com (adjust.com)
  59. - juejin.cn
  60. - leetcode.cn
  61. - oschina.net
  62. - gitee.com
  63. - xda-developers.com
  64. - sspai.com
  65. - gcores.com
  66. - zhihu.com
  67. - Steam (Store, Hub)
  68. - pixiv.net
  69. - vk.com
  70. - deviantart.com
  71. - tmall.com (goto)
  72. - linkstars.com (Prevent redirection)
  73. - union-click.jd.com (Prevent redirection)
  74. - s.click.(tmall|taobao).com (Prevent redirection)
  75. - wiki.biligame.com
  76. - tieba.baidu.com
  77. - linkedin.com
  78. - sourceforge.net
  79. */
  80.  
  81. (() => {
  82. const DELAY_TIME = { fast: 600, normal: 1000, slow: 2500 };
  83. let topScroll = 0;
  84. const INDEX_TARGET = ['target'];
  85. const INDEX_ADJUST = ['redirect', 'fallback'];
  86. const INDEX_URL = ['url'];
  87. const INDEX_TO = ['to'];
  88. const INDEX_Q = ['q'];
  89. const INDEX_GOTO = ['goto'];
  90. const regStr = '(youtube|steamcommunity|zhihu|jianshu|juejin|leetcode|'
  91. + 'oschina|gitee|sspai|gcores|alipay|epicgames|linkedin|vk|adjust|'
  92. + 'game.bilibili|douban|sourceforge).(com|net|cn|hk)$';
  93. let hostRegex = new RegExp(regStr);
  94. const pageHost = window.location.hostname;
  95. const pageParams = window.location.search;
  96. const doc = document;
  97.  
  98. // SourceForge link cleaning function
  99. function cleanSourceForgeLink(originalLink) {
  100. try {
  101. const url = new URL(originalLink);
  102. const oaparams = new URLSearchParams(url.search).get('oaparams');
  103. if (!oaparams) return originalLink;
  104. const oaComponents = oaparams.split('__');
  105. const oadestComponent = oaComponents.find(comp => comp.startsWith('oadest='));
  106. if (!oadestComponent) return originalLink;
  107. const destUrlEncoded = oadestComponent.split('oadest=')[1];
  108. const destUrl = decodeURIComponent(destUrlEncoded);
  109. const finalUrl = new URL(destUrl);
  110. return finalUrl.origin + finalUrl.pathname.replace(/\/$/, '');
  111. } catch (e) {
  112. console.error('Error cleaning SourceForge URL:', e);
  113. return originalLink;
  114. }
  115. }
  116.  
  117. let linkDirect;
  118. switch (true) {
  119. case /(pixiv.net|deviantart.com)$/.test(pageHost):
  120. hostRegex = /(pixiv.net|deviantart.com)$/;
  121. linkDirect = (directURLParams, delayTime) => {
  122. const timeoutID = setTimeout(() => {
  123. const links = doc.getElementsByTagName('a');
  124. for (let i = 0; i < links.length; i++) {
  125. if (hostRegex.test(links[i].hostname)) {
  126. const params = new URLSearchParams(links[i].search);
  127. directURLParams.forEach((k) => {
  128. if (params.has(k) && links[i].href !== decodeURIComponent(params.get(k))) {
  129. links[i].href = decodeURIComponent(params.get(k));
  130. }
  131. });
  132. if (/jump.php|outgoing/.test(links[i].pathname)) {
  133. if (links[i].href !== decodeURIComponent(links[i].search.substring(1, links[i].href.length))) {
  134. links[i].href = decodeURIComponent(links[i].search.substring(1, links[i].href.length));
  135. }
  136. }
  137. }
  138. }
  139. clearTimeout(timeoutID);
  140. }, delayTime);
  141. };
  142. break;
  143. case /xda-developers.com$/.test(pageHost):
  144. hostRegex = /(xda-developers.com|shop-links.co|anrdoezrs.net|a9yw.net|pxf.io|viglink.com|awin1.com)$/;
  145. linkDirect = (directURLParams, delayTime) => {
  146. const timeoutID = setTimeout(() => {
  147. const links = doc.getElementsByTagName('a');
  148. for (let i = 0; i < links.length; i++) {
  149. if (hostRegex.test(links[i].hostname)) {
  150. const params = new URLSearchParams(links[i].search);
  151. directURLParams.forEach((k) => {
  152. if (params.has(k) && links[i].href !== decodeURIComponent(params.get(k))) {
  153. links[i].href = decodeURIComponent(params.get(k));
  154. }
  155. });
  156. let realLink = links[i].href;
  157. if (/https?/.test(links[i].search)) {
  158. realLink = links[i].search.substring(1, links[i].href.length);
  159. } else if (/https?/.test(links[i].pathname)) {
  160. realLink = links[i].pathname.substring(links[i].pathname.lastIndexOf('http'), links[i].href.length);
  161. }
  162. if (links[i].href !== decodeURIComponent(realLink)) {
  163. links[i].href = decodeURIComponent(realLink);
  164. }
  165. }
  166. }
  167. clearTimeout(timeoutID);
  168. }, delayTime);
  169. };
  170. break;
  171. case /^(tieba|ala).baidu.com$/.test(pageHost):
  172. linkDirect = (directURLParams, delayTime) => {
  173. const timeoutID = setTimeout(() => {
  174. const links = doc.getElementsByClassName('j-no-opener-url');
  175. for (let i = 0; i < links.length; i++) {
  176. if (/^jump2?.bdimg.com$/.test(links[i].hostname) && links[i].innerText.startsWith('http')) {
  177. links[i].href = links[i].innerText;
  178. }
  179. }
  180. clearTimeout(timeoutID);
  181. }, delayTime);
  182. };
  183. break;
  184. case /sourceforge.net$/.test(pageHost):
  185. linkDirect = (directURLParams, delayTime) => {
  186. const timeoutID = setTimeout(() => {
  187. const links = doc.getElementsByTagName('a');
  188. for (let i = 0; i < links.length; i++) {
  189. if (/sourceforge.net$/.test(links[i].hostname)) {
  190. const cleanedLink = cleanSourceForgeLink(links[i].href);
  191. if (links[i].href !== cleanedLink) {
  192. links[i].href = cleanedLink;
  193. }
  194. }
  195. }
  196. clearTimeout(timeoutID);
  197. }, delayTime);
  198. };
  199. break;
  200. default:
  201. linkDirect = (directURLParams, delayTime) => {
  202. const timeoutID = setTimeout(() => {
  203. const links = doc.getElementsByTagName('a');
  204. for (let i = 0; i < links.length; i++) {
  205. if (hostRegex.test(links[i].hostname)) {
  206. const params = new URLSearchParams(links[i].search);
  207. directURLParams.forEach((k) => {
  208. if (params.has(k) && links[i].href !== params.get(k)) {
  209. links[i].href = params.get(k);
  210. }
  211. });
  212. }
  213. }
  214. clearTimeout(timeoutID);
  215. }, delayTime);
  216. };
  217. break;
  218. }
  219.  
  220. // Youtube additional steps
  221. function youtubeDirect() {
  222. function run(delayTime) {
  223. linkDirect(INDEX_Q, DELAY_TIME.fast);
  224. linkDirect(INDEX_Q, DELAY_TIME.normal * 2);
  225. const timeoutID = setTimeout(() => {
  226. linkDirect(INDEX_Q, 0);
  227. document.addEventListener('click', () => {
  228. linkDirect(INDEX_Q, DELAY_TIME.fast);
  229. });
  230. clearTimeout(timeoutID);
  231. }, delayTime);
  232. }
  233. run(2000);
  234. doc.addEventListener('DOMContentLoaded', () => {
  235. run(1000);
  236. });
  237. doc.onvisibilitychange = () => {
  238. run(1500);
  239. };
  240. }
  241.  
  242. // Main function
  243. (() => {
  244. let indexParam;
  245. let MenuTitle;
  246. switch (navigator.language) {
  247. case 'zh-CN' || 'zh-SG':
  248. MenuTitle = '手动重新替换';
  249. break;
  250. case 'zh-TW' || 'zh-HK':
  251. MenuTitle = '手動再次替換';
  252. break;
  253. default:
  254. MenuTitle = 'Retry link replacing.';
  255. break;
  256. }
  257.  
  258. const adjust = /(hoyolab|mozilla|firefox)\.(com|org)$/.test(pageHost);
  259. const usingTarget = /(juejin|leetcode|gitee|sspai|gcores|zhihu)\.(com|cn)$/.test(pageHost);
  260. const isSourceForge = /sourceforge.net$/.test(pageHost);
  261. const urlParam = new URLSearchParams(pageParams);
  262. switch (true) {
  263. case usingTarget:
  264. indexParam = INDEX_TARGET;
  265. break;
  266. case adjust:
  267. indexParam = INDEX_ADJUST;
  268. linkDirect(indexParam, DELAY_TIME.normal * 2);
  269. break;
  270. case pageHost.endsWith('youtube.com'):
  271. indexParam = INDEX_Q;
  272. youtubeDirect();
  273. break;
  274. case /(s.click.taobao.com|tmall.com)$/.test(pageHost):
  275. indexParam = INDEX_GOTO;
  276. if (/^s.click.(tmall|taobao).com$/.test(window.location.hostname)
  277. && new URLSearchParams(pageParams).has('tar')) {
  278. window.stop();
  279. const targetLink = decodeURIComponent(new URLSearchParams(window.location.search).get('tar'));
  280. if (/^https?:\/\//.test(targetLink)) {
  281. window.location.replace(targetLink);
  282. }
  283. }
  284. break;
  285. case isSourceForge:
  286. linkDirect([], DELAY_TIME.normal);
  287. break;
  288. case /(steampowered|steamcommunity|wiki.biligame|linkedin|douban).com$|pixiv.net$/.test(pageHost):
  289. indexParam = INDEX_URL;
  290. break;
  291. case /(vk|jianshu).com$/.test(pageHost):
  292. indexParam = INDEX_TO;
  293. break;
  294. case pageHost.endsWith('epicgames.com'):
  295. indexParam = ['redirectTo'];
  296. break;
  297. case pageHost.endsWith('oschina.net'):
  298. INDEX_URL.push('goto_page');
  299. indexParam = INDEX_URL;
  300. break;
  301. case pageHost.endsWith('xda-developers.com'):
  302. INDEX_URL.push('u', 'ued', 'referer');
  303. indexParam = INDEX_URL;
  304. break;
  305. case /(union-click.jd.com|www.linkstars.com)$/.test(pageHost):
  306. indexParam = INDEX_TO;
  307. if (urlParam.has(indexParam) && /^https?/.test(urlParam.get(indexParam))) {
  308. window.stop();
  309. window.location.href = decodeURIComponent(urlParam.get(indexParam));
  310. }
  311. break;
  312. case /(theverge.com|7tiv.net)$/.test(pageHost):
  313. hostRegex = /sjv.io$/;
  314. INDEX_URL.push('u');
  315. indexParam = INDEX_URL;
  316. if (pageHost.endsWith('7tiv.net')
  317. && new URLSearchParams(pageParams).has(indexParam)) {
  318. window.stop();
  319. window.location.href = decodeURIComponent(
  320. new URLSearchParams(pageParams).get(indexParam)
  321. );
  322. }
  323. break;
  324. default:
  325. indexParam = [''];
  326. break;
  327. }
  328.  
  329. doc.addEventListener('DOMContentLoaded', () => {
  330. linkDirect(indexParam, DELAY_TIME.normal);
  331. });
  332.  
  333. GM_registerMenuCommand(
  334. MenuTitle,
  335. () => { linkDirect(indexParam, 0); },
  336. 'D'
  337. );
  338.  
  339. window.onscroll = () => {
  340. const scrolls = doc.documentElement.scrollTop;
  341. if (scrolls <= 200) {
  342. linkDirect(indexParam, 0);
  343. topScroll = scrolls;
  344. }
  345. if (scrolls - topScroll > 100 && scrolls > 200) {
  346. linkDirect(indexParam, 0);
  347. topScroll = scrolls;
  348. }
  349. };
  350. })();
  351. })();
  352.  
  353. /*
  354. v0.2.5 2025.06.01
  355. - Added theverge.com|7tiv.net|douban.com|sourceForge.net.
  356. - Optimized performance and error handling.
  357. - Maintained all existing features with DS.
  358.  
  359. v0.2.3 2023.08.15
  360. - Add a alternate host `ala.baidu.com`, same as `tieba.baidu.com`.
  361. - Fix an error when decoding the URLs on the Chinese pages of youtube.com.
  362.  
  363. v0.2.2 2023.07.02
  364. - Performance improvements and issue fixes.
  365.  
  366. v0.2.1 2023.06.07
  367. - Fix an issue where the replacements weren't active on `deviantart.com`, which is missing on `siteRegex`.
  368. - Fix most timeout issues.
  369. - Minor issue fixes and optimisations .
  370.  
  371. v0.2.0 2023.06.02
  372. - Improve replacing efficiency on youtube.
  373. - Replacing more links on xda (a9yw.net|pxf.io), tieba.baidu.com (jump.baidu.com), linkedin.com.
  374. - Prevent redirection on `s.click.tmall.com` (beta).
  375. - Code reduction and other improvements.
  376.  
  377. v0.1.9 2023.05.24
  378. - Directing for wiki.biligame.com, www.linkstars.com.
  379. - Performance optimisation and bug fixes.
  380.  
  381. v0.1.8 2023.05.18
  382. - Directing for JD.com, Tmall.com.
  383. - Improve replacing efficiency on youtube.
  384.  
  385. v0.1.7 2023.05.15
  386. - Replace more redirecting links for xda-developers(vglink.com anrdoezrs.net).
  387. - Add a index param for shop-links.co.
  388. - Direc links for `Steam store and hub`, `Pixiv.net`.
  389. - Optimise regexps matching.
  390.  
  391. v0.1.6.1 2023.05.10
  392. - Fix an issue where has an undefined function, which may cause some functions to fail to execute.
  393.  
  394. v0.1.6 2023.05.10
  395. - Add support for sspai|gcores|zhihu.com (target).
  396. - Add another redirecting index param for oschina.net.
  397. - Remove landiannews.com for its low usage.
  398.  
  399. v0.1.5 2023.05.05
  400. - Errors fixes and code reduction.
  401. - Replacing the shop-links with direct links on **xda-developers.com**.
  402.  
  403. v0.1.4 2023.04.28
  404. - Expand effecting area.
  405. - Several optimisation.
  406. - More url index for adjust.
  407.  
  408. v0.1.3 2023.04.18
  409. - Improve effecting stability.
  410. - Apply direct link for mozilla.org, firefox.com (Adjust.com - redirect),
  411. - Apply direct link for leetcode.cn, oschina.net, gitee.com (target).
  412. - Add a script submenu to the tampermonky menu, which for the function of manually replacing with direct links.
  413.  
  414. v0.1.2 2023.04.15
  415. - Optimised link directing on youtube.com.
  416. - Performance optimisation.
  417.  
  418. v0.1.1 2023.04.06
  419. - Spelling correction.
  420.  
  421. v0.1.0 2023.04.06
  422. - Script optimisation.
  423. - Fix youtube matching.
  424.  
  425. v0.0.6 2023.03.27
  426. Replace direct url on Epicgames.com.
  427.  
  428. v0.0.5
  429. - Remove [Youtube.com] event redirection.
  430.  
  431. v0.0.4 2023.02.24
  432. - Added [Juejin.cn] redirecting.
  433.  
  434. v0.0.3 2023.01.25
  435. - Added [Jianshu.com] redirecting.
  436.  
  437. v0.0.2 2023.01.06
  438. - Clean Hoyolab [app.adjust.com] tracking urls.
  439.  
  440. v0.0.1 2022.12.27
  441. - Initial release, direct link for landiannews.com.
  442. */

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址