open the link directly

点击链接直接跳转

当前为 2022-03-27 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name open the link directly
  3. // @namespace http://tampermonkey.net/
  4. // @version 0.1.1
  5. // @description 点击链接直接跳转
  6. // @author nediiii
  7. // @match *://*.csdn.net/*
  8. // @match *://*.gitee.com/*
  9. // @match *://*.uisdc.com/*
  10. // @match *://*.logonews.cn/*
  11. // @match *://*.afdian.net/*
  12. // @match *://*.tianyancha.com/*
  13. // @match *://*.oschina.net/*
  14. // @match *://*.pixiv.net/*
  15. // @match *://*.jianshu.com/*
  16. // @match *://*.juejin.cn/*
  17. // @match *://*.weibo.cn/*
  18. // @match *://*.weibo.com/*
  19. // @match *://*.yuque.com/*
  20. // @match *://*.segmentfault.com/*
  21. // @match *://*.zhihu.com/*
  22. // @license GPLv3 License
  23. // @icon https://www.google.com/s2/favicons?sz=64&domain=greasyfork.org
  24. // @grant GM_xmlhttpRequest
  25. // ==/UserScript==
  26.  
  27. (function () {
  28. 'use strict';
  29.  
  30. const isValidURL = (url) => {
  31. try {
  32. new URL(url);
  33. return true;
  34. } catch (error) {
  35. return isValidURLWithBase(url);
  36. }
  37. }
  38.  
  39. const isValidURLWithBase = (url) => {
  40. try {
  41. new URL(url, getCurrentURLBase());
  42. return true;
  43. } catch (error) {
  44. return false;
  45. }
  46. }
  47.  
  48. const getCurrentURLBase = () => {
  49. return window.location.origin;
  50. }
  51.  
  52. const urlReg = /\bhttps?:\/\/\S+/gi;
  53.  
  54. const weiboResolver = async (href) => {
  55. return new Promise((resolve, reject) => {
  56. GM_xmlhttpRequest({
  57. method: "GET",
  58. url: href,
  59. onload: function (response) {
  60. console.log({ response });
  61. if (response.status != 200) {
  62. reject(href);
  63. return;
  64. }
  65.  
  66. // weibo跳转会区分目标网址是否备案
  67. // 未备案的, 中转
  68. // 已备案的, 直跳
  69. let realURI = href;
  70. if (response.finalUrl === href) {
  71. // 未备案, 中转网址
  72. // 网址在html里
  73. let doc = new DOMParser().parseFromString(response.responseText, "text/html");
  74. let node = doc.querySelector('body > div > div:nth-child(2)');
  75. let str = node.innerText;
  76. console.log({ doc });
  77. console.log({ node });
  78. console.log({ str });
  79.  
  80. var extractURL = str.match(urlReg);
  81. console.log({ extractURL });
  82. if (extractURL) {
  83. realURI = extractURL[0];
  84. }
  85. }
  86. else {
  87. // 已备案
  88. // 网址在finalUrl里
  89. var extractURL = response.finalUrl.match(urlReg);
  90. if (extractURL) {
  91. realURI = extractURL[0];
  92. }
  93. }
  94.  
  95. resolve(realURI)
  96. },
  97. onerror: function (error) {
  98. console.log({ error });
  99. reject(href)
  100. }
  101. });
  102. });
  103. }
  104.  
  105. const segfaultResolver = async (href) => {
  106. return new Promise((resolve, reject) => {
  107. GM_xmlhttpRequest({
  108. method: "GET",
  109. url: href,
  110. onload: function (response) {
  111. console.log({ response });
  112. if (response.status == 200) {
  113. let doc = new DOMParser().parseFromString(response.responseText, "text/html");
  114. let node = doc.querySelector('body > p')
  115. let str = node.innerText;
  116. console.log({ doc });
  117. console.log({ node });
  118. console.log({ str });
  119.  
  120. let realURI = href;
  121. var extractURL = str.match(urlReg);
  122. console.log({ extractURL });
  123. if (extractURL) {
  124. realURI = extractURL[0];
  125. }
  126. resolve(realURI)
  127. } else {
  128. reject(href)
  129. }
  130. },
  131. onerror: function (error) {
  132. console.log({ error });
  133. reject(href)
  134. }
  135. });
  136. });
  137. }
  138.  
  139. const patter_match = {
  140.  
  141. // 注意这里的pattern需要去看对应网站dom里的a标签的实际herf值, console也会打印日志, 可以自己添加正则来增加网站支持
  142. // https://link.zhihu.com/?target=https%3A//greasyfork.org/en/scripts/5029-yet-another-%25E8%2587%25AA%25E5%258F%25A4cb%25E5%2587%25BA%25E8%25AF%2584%25E8%25AE%25BA-sharing-plugin
  143. zhihu: { pattern: /https?:\/\/link\.zhihu\.com\/?\?target=(.+)$/ },
  144.  
  145. // https://www.jianshu.com/p/a6a63a0c6e53
  146. // https://links.jianshu.com/go?to=https%3A%2F%2Fnpm.taobao.org%2Fmirrors%2Felectron
  147. jianshu2: { pattern: /https?:\/\/links\.jianshu\.com\/go\?to=(.+)$/ },
  148.  
  149. // href="https://link.juejin.cn?target=https%3A%2F%2Fdeveloper.aliyun.com%2Fgroup%2Falisoftwaretech%2F"
  150. juejin: { pattern: /https?:\/\/link\.juejin\.cn\/?\?target=(.+)$/ },
  151.  
  152. // https://gitee.com/meetqy/acss-dnd
  153. // href="https://gitee.com/link?target=https%3A%2F%2Fcuyang.me%2Facss-dnd%2F"
  154. gitee: { pattern: /https?:\/\/gitee\.com\/link\?target=(.+)$/ },
  155.  
  156. // https://www.uisdc.com/build-b-end-grid-system
  157. // https://link.uisdc.com/?redirect=https%3A%2F%2Fuxdesign.cc%2Fresponsive-grids-and-how-to-actually-use-them-970de4c16e01
  158. uisdc: { pattern: /https?:\/\/link\.uisdc\.com\/?\?redirect=(.+)$/ },
  159.  
  160. // https://www.logonews.cn/apple-sues-sex-topic-blog-for-logo-infringement.html
  161. // https://link.logonews.cn/?url=http://aasd.k12.wi.us/district
  162. logonews: { pattern: /https?:\/\/link\.logonews\.cn\/?\?url=(.+)$/ },
  163.  
  164. // https://afdian.net/@AdventCirno
  165. // https://afdian.net/link?target=https%3A%2F%2Fwww.patreon.com%2Fuser%3Fu%3D6139561
  166. afdian: { pattern: /https?:\/\/afdian\.net\/link\?target=(.+)$/ },
  167.  
  168. // https://www.tianyancha.com/company/28723141
  169. // https://www.tianyancha.com/security?target=https%3A%2F%2Fss.knet.cn%2Fverifyseal.dll%3Fsn%3De18042711010873571xsuv000000%26pa%3D111332
  170. tianyancha: { pattern: /https?:\/\/www\.tianyancha\.com\/security\?target=(.+)$/ },
  171.  
  172. // https://www.pixiv.net/users/25237
  173. // href="/jump.php?url=https%3A%2F%2Ftwitter.com%2Fomiya_io"
  174. // href="/jump.php?https%3A%2F%2Finstagram.com%2Fsnatti89%2F"
  175. pixiv: { pattern: /\/jump\.php\?(?:url=)?(.+)$/ },
  176.  
  177. // https://my.oschina.net/androiddevs/blog/5496556
  178. // https://www.oschina.net/action/GoToLink?url=https%3A%2F%2Fmp.weixin.qq.com%2Fmp%2Fappmsgalbum%3F__biz%3DMzk0NDIwMTExNw%3D%3D%26action%3Dgetalbum%26album_id%3D1879128471667326981%23wechat_redirect
  179. // href="https://www.oschina.net/action/GoToLink?url=https%3A%2F%2Fmp.weixin.qq.com%2Fmp%2Fappmsgalbum%3F__biz%3DMzk0NDIwMTExNw%3D%3D%26action%3Dgetalbum%26album_id%3D1879128471667326981%23wechat_redirect"
  180. oschina: { pattern: /https?:\/\/www\.oschina\.net\/action\/GoToLink\?url=(.+)$/ },
  181.  
  182. // https://weibo.cn/sinaurl?u=https%3A%2F%2Fwww.freebsd.org%2F
  183. // https://weibo.cn/sinaurl?toasturl=https%3A%2F%2Ftime.geekbang.org%2F
  184. // https://weibo.cn/sinaurl?luicode=10000011&lfid=230259&u=http%3A%2F%2Ft.cn%2FA6qHeVlf
  185. // https://weibo.cn/sinaurl?f=w&u=http%3A%2F%2Ft.cn%2FA66XY2gI&ep=LlAsNz3HD%2C1683963007%2CLlpkandl6%2C7276218544
  186. // href="https://weibo.cn/sinaurl?f=w&u=http%3A%2F%2Ft.cn%2FA66XY2gI&ep=LlAsNz3HD%2C1683963007%2CLlpkandl6%2C7276218544"
  187. weibo: { pattern: /https?:\/\/weibo\.cn\/sinaurl\?f=w&u=(.+)$/, resolver: weiboResolver },
  188.  
  189. // http://t.cn/A66926Pm 未备案的, 跳转到中转网址, response.finalUrl仍然还是http://t.cn/A66926Pm 目标网址出现在response.responseText里
  190. // http://t.cn/A669K964 已备案的, 直接跳转到目标网址, 出现在response.finalUrl里
  191. weibo2: { pattern: /(https?:\/\/t\.cn\/.*)$/, resolver: weiboResolver },
  192.  
  193.  
  194. // segmentfault对链接进行加密处理, 不知道如何decode, 所以只能写一个函数去单独处理
  195. // https://link.segmentfault.com/?enc=LZyRulLABKpXOHl2vbA%2F4w%3D%3D.MWhFMvjhyBk1ReIRoGxyxa0VxGtg%2Foyk0DMtfzZTJoKbsgoJFtGCPHe8%2BZ1HbRdcvNsGaVfll9oGQXLsZCHK7w%3D%3D
  196. segfault: { pattern: /https?:\/\/link\.segmentfault\.com\/?\?enc=(.+)$/, resolver: segfaultResolver }
  197.  
  198. // 以下网站a标签的herf未修改, 推测是js做的弹窗, 所以不需要匹配, 也匹配不出来
  199. // csdn https://link.csdn.net/?target=https%3A%2F%2Fdeveloper.mozilla.org%2Fzh-CN%2Fdocs%2FWeb%2FJavaScript%2FReference%2FGlobal_Objects%2FRegExp
  200. // 语雀 https://www.yuque.com/r/goto?url=https%3A%2F%2Fwww.canva.cn%2F
  201. }
  202.  
  203. const resolveRealURI = async (href) => {
  204. const fallbackURI = href;
  205.  
  206. for (let i in patter_match) {
  207. const matcher = href.match(patter_match[i].pattern);
  208. if (!matcher) {
  209. continue;
  210. }
  211. console.log({ matcher });
  212.  
  213. if (patter_match[i].hasOwnProperty('resolver')) {
  214. // complex customize resolver
  215. console.log(patter_match[i].resolver)
  216. let realURI = await patter_match[i].resolver(href);
  217. return realURI;
  218. }
  219.  
  220. const encodeURI = matcher[1];
  221. // simple reg resolver
  222. return decodeURIComponent(encodeURI);
  223. }
  224. return fallbackURI;
  225. }
  226.  
  227. const getHref = (e) => {
  228. let target = e.target;
  229. while (target) {
  230. if (target.tagName.toLowerCase() === 'a' && target.hasAttribute('href')) {
  231. console.log("find element", { target })
  232. return target.getAttribute('href');
  233. }
  234. target = target.parentElement;
  235. }
  236. return null;
  237. }
  238.  
  239. document.addEventListener('click', (e) => {
  240. console.log({ e })
  241. let href = getHref(e);
  242. if (href) {
  243.  
  244. // 不是url, 则不做处理
  245. // 兼容如 href设置为 'javascript:void(0);' 等的情况
  246. if (!isValidURL(href)) {
  247. return;
  248. }
  249.  
  250. e.stopPropagation();
  251. e.preventDefault();
  252.  
  253. resolveRealURI(href).then((realURI) => {
  254. console.log({ realURI })
  255. window.open(realURI);
  256. }).catch(() => { window.open(href); })
  257. }
  258. }, { capture: true })
  259.  
  260. })();