🔗 链接助手

支持所有网站在新标签页中打开第三方网站链接(外链),在新标签页中打开符合指定规则的本站链接,解析文本链接为超链接,微信公众号文本转可点击的超链接,图片链接转图片标签,解析 Markdown 格式链接与图片标签,解析 BBCode 格式链接与图片标签

当前为 2023-06-30 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name 🔗 Links Helper
  3. // @name:zh-CN 🔗 链接助手
  4. // @namespace https://github.com/utags/links-helper
  5. // @homepageURL https://github.com/utags/links-helper#readme
  6. // @supportURL https://github.com/utags/links-helper/issues
  7. // @version 0.4.0
  8. // @description Open external links in a new tab, open internal links matching the specified rules in a new tab, convert text to hyperlinks, convert image links to image tags(<img>), parse Markdown style links and image tags, parse BBCode style links and image tags
  9. // @description:zh-CN 支持所有网站在新标签页中打开第三方网站链接(外链),在新标签页中打开符合指定规则的本站链接,解析文本链接为超链接,微信公众号文本转可点击的超链接,图片链接转图片标签,解析 Markdown 格式链接与图片标签,解析 BBCode 格式链接与图片标签
  10. // @icon data:image/svg+xml;base64,PHN2ZyB3aWR0aD0nMTUnIGhlaWdodD0nMTUnIHZpZXdCb3g9JzAgMCAxNSAxNScgZmlsbD0nbm9uZScgeG1sbnM9J2h0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnJz48cGF0aCBkPSdNMyAyQzIuNDQ3NzIgMiAyIDIuNDQ3NzIgMiAzVjEyQzIgMTIuNTUyMyAyLjQ0NzcyIDEzIDMgMTNIMTJDMTIuNTUyMyAxMyAxMyAxMi41NTIzIDEzIDEyVjguNUMxMyA4LjIyMzg2IDEyLjc3NjEgOCAxMi41IDhDMTIuMjIzOSA4IDEyIDguMjIzODYgMTIgOC41VjEySDNWM0w2LjUgM0M2Ljc3NjE0IDMgNyAyLjc3NjE0IDcgMi41QzcgMi4yMjM4NiA2Ljc3NjE0IDIgNi41IDJIM1pNMTIuODUzNiAyLjE0NjQ1QzEyLjkwMTUgMi4xOTQzOSAxMi45Mzc3IDIuMjQ5NjQgMTIuOTYyMSAyLjMwODYxQzEyLjk4NjEgMi4zNjY2OSAxMi45OTk2IDIuNDMwMyAxMyAyLjQ5N0wxMyAyLjVWMi41MDA0OVY1LjVDMTMgNS43NzYxNCAxMi43NzYxIDYgMTIuNSA2QzEyLjIyMzkgNiAxMiA1Ljc3NjE0IDEyIDUuNVYzLjcwNzExTDYuODUzNTUgOC44NTM1NUM2LjY1ODI5IDkuMDQ4ODIgNi4zNDE3MSA5LjA0ODgyIDYuMTQ2NDUgOC44NTM1NUM1Ljk1MTE4IDguNjU4MjkgNS45NTExOCA4LjM0MTcxIDYuMTQ2NDUgOC4xNDY0NUwxMS4yOTI5IDNIOS41QzkuMjIzODYgMyA5IDIuNzc2MTQgOSAyLjVDOSAyLjIyMzg2IDkuMjIzODYgMiA5LjUgMkgxMi40OTk5SDEyLjVDMTIuNTY3OCAyIDEyLjYzMjQgMi4wMTM0OSAxMi42OTE0IDIuMDM3OTRDMTIuNzUwNCAyLjA2MjM0IDEyLjgwNTYgMi4wOTg1MSAxMi44NTM2IDIuMTQ2NDVaJyBmaWxsPSdjdXJyZW50Q29sb3InIGZpbGwtcnVsZT0nZXZlbm9kZCcgY2xpcC1ydWxlPSdldmVub2RkJz48L3BhdGg+PC9zdmc+
  11. // @author Pipecraft
  12. // @license MIT
  13. // @match https://*/*
  14. // @match http://*/*
  15. // @run-at document-start
  16. // @grant GM.getValue
  17. // @grant GM.setValue
  18. // @grant GM_addValueChangeListener
  19. // @grant GM_removeValueChangeListener
  20. // @grant GM_addElement
  21. // @grant GM_addStyle
  22. // @grant GM.registerMenuCommand
  23. // ==/UserScript==
  24. //
  25. //// Recent Updates
  26. //// - 0.3.5 2023.05.17
  27. //// - Fix some edge cases
  28. //// - 0.3.4 2023.05.16
  29. //// - Parse BBCode style links and image tags
  30. //// - Update parsing links logic
  31. //// - 0.3.3 2023.05.11
  32. //// - Fix parse markdown style text
  33. //// - 0.3.2 2023.05.10
  34. //// - Parse Markdown style links and image tags
  35. //// - 0.3.0 2023.05.10
  36. //// - Convert image links to image tags
  37. //// - 0.2.0 2023.05.09
  38. //// - Convert text to hyperlinks
  39. //// - Fix opening internal links in a new tab in SPA apps
  40. //// - 0.1.3 2023.05.08
  41. //// - Fix compatibility issues on Violentmonkey, Greasemonkey(Firefox), Userscripts(Safari)
  42. //// - 0.1.1 2023.04.23
  43. //// - Change to run_at: document_start
  44. //// - 0.1.0 2023.04.23
  45. //// - Setting for url rules, open links matching the specified rules in a new tab
  46. //// - 0.0.2 2023.04.22
  47. //// - Add settings menu
  48. //// - Enable/Disable userscript
  49. //// - Enable/Disable current site
  50. ////
  51. ;(() => {
  52. "use strict"
  53. var listeners = {}
  54. var getValue = async (key) => {
  55. const value = await GM.getValue(key)
  56. return value && value !== "undefined" ? JSON.parse(value) : void 0
  57. }
  58. var setValue = async (key, value) => {
  59. if (value !== void 0) {
  60. const newValue = JSON.stringify(value)
  61. if (listeners[key]) {
  62. const oldValue = await GM.getValue(key)
  63. await GM.setValue(key, newValue)
  64. if (newValue !== oldValue) {
  65. for (const func of listeners[key]) {
  66. func(key, oldValue, newValue)
  67. }
  68. }
  69. } else {
  70. await GM.setValue(key, newValue)
  71. }
  72. }
  73. }
  74. var _addValueChangeListener = (key, func) => {
  75. listeners[key] = listeners[key] || []
  76. listeners[key].push(func)
  77. return () => {
  78. if (listeners[key] && listeners[key].length > 0) {
  79. for (let i = listeners[key].length - 1; i >= 0; i--) {
  80. if (listeners[key][i] === func) {
  81. listeners[key].splice(i, 1)
  82. }
  83. }
  84. }
  85. }
  86. }
  87. var addValueChangeListener = (key, func) => {
  88. if (typeof GM_addValueChangeListener !== "function") {
  89. console.warn("Do not support GM_addValueChangeListener!")
  90. return _addValueChangeListener(key, func)
  91. }
  92. const listenerId = GM_addValueChangeListener(key, func)
  93. return () => {
  94. GM_removeValueChangeListener(listenerId)
  95. }
  96. }
  97. var doc = document
  98. var $ = (selectors, element) => (element || doc).querySelector(selectors)
  99. var $$ = (selectors, element) => [
  100. ...(element || doc).querySelectorAll(selectors),
  101. ]
  102. var createElement = (tagName, attributes) =>
  103. setAttributes(doc.createElement(tagName), attributes)
  104. var addElement = (parentNode, tagName, attributes) => {
  105. if (!parentNode) {
  106. return
  107. }
  108. if (typeof parentNode === "string") {
  109. attributes = tagName
  110. tagName = parentNode
  111. parentNode = doc.head
  112. }
  113. if (typeof tagName === "string") {
  114. const element = createElement(tagName, attributes)
  115. parentNode.append(element)
  116. return element
  117. }
  118. setAttributes(tagName, attributes)
  119. parentNode.append(tagName)
  120. return tagName
  121. }
  122. var addStyle = (styleText) => {
  123. const element = createElement("style", { textContent: styleText })
  124. doc.head.append(element)
  125. return element
  126. }
  127. var addEventListener = (element, type, listener, options) => {
  128. if (!element) {
  129. return
  130. }
  131. if (typeof type === "object") {
  132. for (const type1 in type) {
  133. if (Object.hasOwn(type, type1)) {
  134. element.addEventListener(type1, type[type1])
  135. }
  136. }
  137. } else if (typeof type === "string" && typeof listener === "function") {
  138. element.addEventListener(type, listener, options)
  139. }
  140. }
  141. var removeEventListener = (element, type, listener, options) => {
  142. if (!element) {
  143. return
  144. }
  145. if (typeof type === "object") {
  146. for (const type1 in type) {
  147. if (Object.hasOwn(type, type1)) {
  148. element.removeEventListener(type1, type[type1])
  149. }
  150. }
  151. } else if (typeof type === "string" && typeof listener === "function") {
  152. element.removeEventListener(type, listener, options)
  153. }
  154. }
  155. var getAttribute = (element, name) =>
  156. element ? element.getAttribute(name) : null
  157. var setAttribute = (element, name, value) =>
  158. element ? element.setAttribute(name, value) : void 0
  159. var setAttributes = (element, attributes) => {
  160. if (element && attributes) {
  161. for (const name in attributes) {
  162. if (Object.hasOwn(attributes, name)) {
  163. const value = attributes[name]
  164. if (value === void 0) {
  165. continue
  166. }
  167. if (/^(value|textContent|innerText|innerHTML)$/.test(name)) {
  168. element[name] = value
  169. } else if (name === "style") {
  170. setStyle(element, value, true)
  171. } else if (/on\w+/.test(name)) {
  172. const type = name.slice(2)
  173. addEventListener(element, type, value)
  174. } else {
  175. setAttribute(element, name, value)
  176. }
  177. }
  178. }
  179. }
  180. return element
  181. }
  182. var addAttribute = (element, name, value) => {
  183. const orgValue = getAttribute(element, name)
  184. if (!orgValue) {
  185. setAttribute(element, name, value)
  186. } else if (!orgValue.includes(value)) {
  187. setAttribute(element, name, orgValue + " " + value)
  188. }
  189. }
  190. var addClass = (element, className) => {
  191. if (!element || !element.classList) {
  192. return
  193. }
  194. element.classList.add(className)
  195. }
  196. var removeClass = (element, className) => {
  197. if (!element || !element.classList) {
  198. return
  199. }
  200. element.classList.remove(className)
  201. }
  202. var setStyle = (element, values, overwrite) => {
  203. if (!element) {
  204. return
  205. }
  206. const style = element.style
  207. if (typeof values === "string") {
  208. style.cssText = overwrite ? values : style.cssText + ";" + values
  209. return
  210. }
  211. if (overwrite) {
  212. style.cssText = ""
  213. }
  214. for (const key in values) {
  215. if (Object.hasOwn(values, key)) {
  216. style[key] = values[key].replace("!important", "")
  217. }
  218. }
  219. }
  220. var throttle = (func, interval) => {
  221. let timeoutId = null
  222. let next = false
  223. const handler = (...args) => {
  224. if (timeoutId) {
  225. next = true
  226. } else {
  227. func.apply(void 0, args)
  228. timeoutId = setTimeout(() => {
  229. timeoutId = null
  230. if (next) {
  231. next = false
  232. handler()
  233. }
  234. }, interval)
  235. }
  236. }
  237. return handler
  238. }
  239. if (typeof Object.hasOwn !== "function") {
  240. Object.hasOwn = (instance, prop) =>
  241. Object.prototype.hasOwnProperty.call(instance, prop)
  242. }
  243. var addElement2 =
  244. typeof GM_addElement === "function"
  245. ? (parentNode, tagName, attributes) => {
  246. if (!parentNode) {
  247. return
  248. }
  249. if (typeof parentNode === "string") {
  250. attributes = tagName
  251. tagName = parentNode
  252. parentNode = doc.head
  253. }
  254. if (typeof tagName === "string") {
  255. const element = GM_addElement(tagName)
  256. setAttributes(element, attributes)
  257. parentNode.append(element)
  258. return element
  259. }
  260. setAttributes(tagName, attributes)
  261. parentNode.append(tagName)
  262. return tagName
  263. }
  264. : addElement
  265. var addStyle2 =
  266. typeof GM_addStyle === "function"
  267. ? (styleText) => GM_addStyle(styleText)
  268. : addStyle
  269. var registerMenuCommand = (name, callback, accessKey) => {
  270. if (window !== top) {
  271. return
  272. }
  273. if (typeof GM.registerMenuCommand !== "function") {
  274. console.warn("Do not support GM.registerMenuCommand!")
  275. return
  276. }
  277. GM.registerMenuCommand(name, callback, accessKey)
  278. }
  279. var style_default = `#browser_extension_settings_container{--browser-extension-settings-background-color: #f2f2f7;--browser-extension-settings-text-color: #444444;--browser-extension-settings-link-color: #217dfc;--sb-track-color: #00000000;--sb-thumb-color: #33334480;--sb-size: 2px;position:fixed;top:10px;right:30px;max-height:90%;height:600px;overflow:hidden;display:none;z-index:100000;border-radius:5px;-webkit-box-shadow:0px 10px 39px 10px rgba(62,66,66,.22);-moz-box-shadow:0px 10px 39px 10px rgba(62,66,66,.22);box-shadow:0px 10px 39px 10px rgba(62,66,66,.22) !important}#browser_extension_settings_container .browser_extension_settings_wrapper{display:flex;height:100%;overflow:hidden;background-color:var(--browser-extension-settings-background-color)}#browser_extension_settings_container .browser_extension_settings_wrapper h1{font-size:26px;font-weight:800;border:none}#browser_extension_settings_container .browser_extension_settings_wrapper h2{font-size:18px;font-weight:600;border:none}#browser_extension_settings_container .browser_extension_settings_wrapper .extension_list_container{overflow-x:auto;box-sizing:border-box;padding:10px 15px;background-color:var(--browser-extension-settings-background-color);color:var(--browser-extension-settings-text-color)}#browser_extension_settings_container .browser_extension_settings_wrapper .extension_list_container .installed_extension_list div,#browser_extension_settings_container .browser_extension_settings_wrapper .extension_list_container .related_extension_list div{background-color:#fff;font-size:14px;border-top:1px solid #ccc;padding:6px 15px 6px 15px}#browser_extension_settings_container .browser_extension_settings_wrapper .extension_list_container .installed_extension_list div a,#browser_extension_settings_container .browser_extension_settings_wrapper .extension_list_container .installed_extension_list div a:visited,#browser_extension_settings_container .browser_extension_settings_wrapper .extension_list_container .related_extension_list div a,#browser_extension_settings_container .browser_extension_settings_wrapper .extension_list_container .related_extension_list div a:visited{display:flex;justify-content:space-between;align-items:center;cursor:pointer;text-decoration:none;color:var(--browser-extension-settings-text-color)}#browser_extension_settings_container .browser_extension_settings_wrapper .extension_list_container .installed_extension_list div a:hover,#browser_extension_settings_container .browser_extension_settings_wrapper .extension_list_container .installed_extension_list div a:visited:hover,#browser_extension_settings_container .browser_extension_settings_wrapper .extension_list_container .related_extension_list div a:hover,#browser_extension_settings_container .browser_extension_settings_wrapper .extension_list_container .related_extension_list div a:visited:hover{text-decoration:none;color:var(--browser-extension-settings-text-color)}#browser_extension_settings_container .browser_extension_settings_wrapper .extension_list_container .installed_extension_list div a span,#browser_extension_settings_container .browser_extension_settings_wrapper .extension_list_container .installed_extension_list div a:visited span,#browser_extension_settings_container .browser_extension_settings_wrapper .extension_list_container .related_extension_list div a span,#browser_extension_settings_container .browser_extension_settings_wrapper .extension_list_container .related_extension_list div a:visited span{margin-right:10px;line-height:24px}#browser_extension_settings_container .browser_extension_settings_wrapper .extension_list_container .installed_extension_list div.active,#browser_extension_settings_container .browser_extension_settings_wrapper .extension_list_container .installed_extension_list div:hover,#browser_extension_settings_container .browser_extension_settings_wrapper .extension_list_container .related_extension_list div.active,#browser_extension_settings_container .browser_extension_settings_wrapper .extension_list_container .related_extension_list div:hover{background-color:#e4e4e6}#browser_extension_settings_container .browser_extension_settings_wrapper .extension_list_container .installed_extension_list div.active a,#browser_extension_settings_container .browser_extension_settings_wrapper .extension_list_container .related_extension_list div.active a{cursor:default}#browser_extension_settings_container .browser_extension_settings_wrapper .extension_list_container .installed_extension_list div:first-of-type,#browser_extension_settings_container .browser_extension_settings_wrapper .extension_list_container .related_extension_list div:first-of-type{border-top:none;border-top-right-radius:10px;border-top-left-radius:10px}#browser_extension_settings_container .browser_extension_settings_wrapper .extension_list_container .installed_extension_list div:last-of-type,#browser_extension_settings_container .browser_extension_settings_wrapper .extension_list_container .related_extension_list div:last-of-type{border-bottom-right-radius:10px;border-bottom-left-radius:10px}#browser_extension_settings_container .thin_scrollbar{scrollbar-color:var(--sb-thumb-color) var(--sb-track-color);scrollbar-width:thin}#browser_extension_settings_container .thin_scrollbar::-webkit-scrollbar{width:var(--sb-size)}#browser_extension_settings_container .thin_scrollbar::-webkit-scrollbar-track{background:var(--sb-track-color);border-radius:10px}#browser_extension_settings_container .thin_scrollbar::-webkit-scrollbar-thumb{background:var(--sb-thumb-color);border-radius:10px}#browser_extension_settings_main{min-width:250px;overflow-y:auto;overflow-x:hidden;box-sizing:border-box;padding:10px 15px;background-color:var(--browser-extension-settings-background-color);color:var(--browser-extension-settings-text-color)}#browser_extension_settings_main h2{text-align:center;margin:5px 0 0;font-size:18px;font-weight:600;border:none}#browser_extension_settings_main footer{display:flex;justify-content:center;flex-direction:column;font-size:11px;margin:10px auto 0px;background-color:var(--browser-extension-settings-background-color);color:var(--browser-extension-settings-text-color)}#browser_extension_settings_main footer a{color:var(--browser-extension-settings-link-color) !important;text-decoration:none;padding:0}#browser_extension_settings_main footer p{text-align:center;padding:0;margin:2px;line-height:13px}#browser_extension_settings_main a.navigation_go_previous{color:var(--browser-extension-settings-link-color);cursor:pointer;display:none}#browser_extension_settings_main a.navigation_go_previous::before{content:"< "}#browser_extension_settings_main .option_groups{background-color:#fff;padding:6px 15px 6px 15px;border-radius:10px;display:flex;flex-direction:column;margin:10px 0 0}#browser_extension_settings_main .option_groups .action{font-size:14px;border-top:1px solid #ccc;padding:6px 0 6px 0;color:var(--browser-extension-settings-link-color);cursor:pointer}#browser_extension_settings_main .option_groups textarea{margin:10px 0 10px 0;height:100px;width:100%;border:1px solid #a9a9a9;border-radius:4px;box-sizing:border-box}#browser_extension_settings_main .switch_option{display:flex;justify-content:space-between;align-items:center;border-top:1px solid #ccc;padding:6px 0 6px 0;font-size:14px}#browser_extension_settings_main .switch_option:first-of-type,#browser_extension_settings_main .option_groups .action:first-of-type{border-top:none}#browser_extension_settings_main .switch_option>span{margin-right:10px}#browser_extension_settings_main .option_groups .tip{position:relative;margin:0;padding:0 15px 0 0;border:none;max-width:none;font-size:14px}#browser_extension_settings_main .option_groups .tip .tip_anchor{cursor:help;text-decoration:underline}#browser_extension_settings_main .option_groups .tip .tip_content{position:absolute;bottom:15px;left:0;background-color:#fff;color:var(--browser-extension-settings-text-color);text-align:left;padding:10px;display:none;border-radius:5px;-webkit-box-shadow:0px 10px 39px 10px rgba(62,66,66,.22);-moz-box-shadow:0px 10px 39px 10px rgba(62,66,66,.22);box-shadow:0px 10px 39px 10px rgba(62,66,66,.22) !important}#browser_extension_settings_main .option_groups .tip .tip_anchor:hover+.tip_content,#browser_extension_settings_main .option_groups .tip .tip_content:hover{display:block}#browser_extension_settings_main .option_groups .tip p,#browser_extension_settings_main .option_groups .tip pre{margin:revert;padding:revert}#browser_extension_settings_main .option_groups .tip pre{font-family:Consolas,panic sans,bitstream vera sans mono,Menlo,microsoft yahei,monospace;font-size:13px;letter-spacing:.015em;line-height:120%;white-space:pre;overflow:auto;background-color:#f5f5f5;word-break:normal;overflow-wrap:normal;padding:.5em;border:none}#browser_extension_settings_main .container{--button-width: 51px;--button-height: 24px;--toggle-diameter: 20px;--color-off: #e9e9eb;--color-on: #34c759;width:var(--button-width);height:var(--button-height);position:relative;padding:0;margin:0;flex:none}#browser_extension_settings_main input[type=checkbox]{opacity:0;width:0;height:0;position:absolute}#browser_extension_settings_main .switch{width:100%;height:100%;display:block;background-color:var(--color-off);border-radius:calc(var(--button-height)/2);border:none;cursor:pointer;transition:all .2s ease-out}#browser_extension_settings_main .switch::before{display:none}#browser_extension_settings_main .slider{width:var(--toggle-diameter);height:var(--toggle-diameter);position:absolute;left:2px;top:calc(50% - var(--toggle-diameter)/2);border-radius:50%;background:#fff;box-shadow:0px 3px 8px rgba(0,0,0,.15),0px 3px 1px rgba(0,0,0,.06);transition:all .2s ease-out;cursor:pointer}#browser_extension_settings_main input[type=checkbox]:checked+.switch{background-color:var(--color-on)}#browser_extension_settings_main input[type=checkbox]:checked+.switch .slider{left:calc(var(--button-width) - var(--toggle-diameter) - 2px)}#browser_extension_side_menu{min-height:200px;width:40px;opacity:0;position:fixed;top:80px;right:0;padding-top:20px;z-index:10000}#browser_extension_side_menu:hover{opacity:1}#browser_extension_side_menu button{cursor:pointer;width:24px;height:24px;border:none;background-color:rgba(0,0,0,0);background-image:url("data:image/svg+xml,%3Csvg width='24' height='24' viewBox='0 0 24 24' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M12.0002 16C14.2094 16 16.0002 14.2091 16.0002 12C16.0002 9.79086 14.2094 8 12.0002 8C9.79109 8 8.00023 9.79086 8.00023 12C8.00023 14.2091 9.79109 16 12.0002 16ZM12.0002 14C13.1048 14 14.0002 13.1046 14.0002 12C14.0002 10.8954 13.1048 10 12.0002 10C10.8957 10 10.0002 10.8954 10.0002 12C10.0002 13.1046 10.8957 14 12.0002 14Z' fill='black'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M15.1182 1.86489L15.5203 4.81406C15.8475 4.97464 16.1621 5.1569 16.4623 5.35898L19.2185 4.23223C19.6814 4.043 20.2129 4.2248 20.463 4.65787L22.5901 8.34213C22.8401 8.77521 22.7318 9.3264 22.3365 9.63266L19.9821 11.4566C19.9941 11.6362 20.0002 11.8174 20.0002 12C20.0002 12.1826 19.9941 12.3638 19.9821 12.5434L22.3365 14.3673C22.7318 14.6736 22.8401 15.2248 22.5901 15.6579L20.463 19.3421C20.2129 19.7752 19.6814 19.957 19.2185 19.7678L16.4623 18.641C16.1621 18.8431 15.8475 19.0254 15.5203 19.1859L15.1182 22.1351C15.0506 22.6306 14.6274 23 14.1273 23H9.87313C9.37306 23 8.94987 22.6306 8.8823 22.1351L8.48014 19.1859C8.15296 19.0254 7.83835 18.8431 7.53818 18.641L4.78195 19.7678C4.31907 19.957 3.78756 19.7752 3.53752 19.3421L1.41042 15.6579C1.16038 15.2248 1.26869 14.6736 1.66401 14.3673L4.01841 12.5434C4.00636 12.3638 4.00025 12.1826 4.00025 12C4.00025 11.8174 4.00636 11.6362 4.01841 11.4566L1.66401 9.63266C1.26869 9.3264 1.16038 8.77521 1.41041 8.34213L3.53752 4.65787C3.78755 4.2248 4.31906 4.043 4.78195 4.23223L7.53818 5.35898C7.83835 5.1569 8.15296 4.97464 8.48014 4.81406L8.8823 1.86489C8.94987 1.3694 9.37306 1 9.87313 1H14.1273C14.6274 1 15.0506 1.3694 15.1182 1.86489ZM13.6826 6.14004L14.6392 6.60948C14.8842 6.72975 15.1201 6.86639 15.3454 7.01805L16.231 7.61423L19.1674 6.41382L20.4216 8.58619L17.9153 10.5278L17.9866 11.5905C17.9956 11.7255 18.0002 11.8621 18.0002 12C18.0002 12.1379 17.9956 12.2745 17.9866 12.4095L17.9153 13.4722L20.4216 15.4138L19.1674 17.5862L16.231 16.3858L15.3454 16.982C15.1201 17.1336 14.8842 17.2702 14.6392 17.3905L13.6826 17.86L13.2545 21H10.746L10.3178 17.86L9.36131 17.3905C9.11626 17.2702 8.88037 17.1336 8.6551 16.982L7.76954 16.3858L4.83313 17.5862L3.57891 15.4138L6.0852 13.4722L6.01392 12.4095C6.00487 12.2745 6.00024 12.1379 6.00024 12C6.00024 11.8621 6.00487 11.7255 6.01392 11.5905L6.0852 10.5278L3.57891 8.58619L4.83312 6.41382L7.76953 7.61423L8.6551 7.01805C8.88037 6.86639 9.11625 6.72976 9.36131 6.60949L10.3178 6.14004L10.746 3H13.2545L13.6826 6.14004Z' fill='black'/%3E%3C/svg%3E")}#browser_extension_side_menu button:hover{opacity:70%}#browser_extension_side_menu button:active{opacity:100%}`
  280. function createSwitch(options = {}) {
  281. const container = createElement("label", { class: "container" })
  282. const checkbox = createElement(
  283. "input",
  284. options.checked ? { type: "checkbox", checked: "" } : { type: "checkbox" }
  285. )
  286. addElement2(container, checkbox)
  287. const switchElm = createElement("span", { class: "switch" })
  288. addElement2(switchElm, "span", { class: "slider" })
  289. addElement2(container, switchElm)
  290. if (options.onchange) {
  291. addEventListener(checkbox, "change", options.onchange)
  292. }
  293. return container
  294. }
  295. function createSwitchOption(text, options) {
  296. const div = createElement("div", { class: "switch_option" })
  297. addElement2(div, "span", { textContent: text })
  298. div.append(createSwitch(options))
  299. return div
  300. }
  301. var version = 11
  302. var openButton = `<svg viewBox="0 0 60.2601318359375 84.8134765625" version="1.1" xmlns="http://www.w3.org/2000/svg" class=" glyph-box" style="height: 9.62969px; width: 6.84191px;"><g transform="matrix(1 0 0 1 -6.194965820312518 77.63671875)"><path d="M66.4551-35.2539C66.4551-36.4746 65.9668-37.5977 65.0391-38.4766L26.3672-76.3672C25.4883-77.1973 24.4141-77.6367 23.1445-77.6367C20.6543-77.6367 18.7012-75.7324 18.7012-73.1934C18.7012-71.9727 19.1895-70.8496 19.9707-70.0195L55.5176-35.2539L19.9707-0.488281C19.1895 0.341797 18.7012 1.41602 18.7012 2.68555C18.7012 5.22461 20.6543 7.12891 23.1445 7.12891C24.4141 7.12891 25.4883 6.68945 26.3672 5.81055L65.0391-32.0312C65.9668-32.959 66.4551-34.0332 66.4551-35.2539Z"></path></g></svg>`
  303. var openInNewTabButton = `<svg viewBox="0 0 72.127685546875 72.2177734375" version="1.1" xmlns="http://www.w3.org/2000/svg" class=" glyph-box" style="height: 8.19958px; width: 8.18935px;"><g transform="matrix(1 0 0 1 -12.451127929687573 71.3388671875)"><path d="M84.5703-17.334L84.5215-66.4551C84.5215-69.2383 82.7148-71.1914 79.7852-71.1914L30.6641-71.1914C27.9297-71.1914 26.0742-69.0918 26.0742-66.748C26.0742-64.4043 28.1738-62.4023 30.4688-62.4023L47.4609-62.4023L71.2891-63.1836L62.207-55.2246L13.8184-6.73828C12.9395-5.85938 12.4512-4.73633 12.4512-3.66211C12.4512-1.31836 14.5508 0.878906 16.9922 0.878906C18.1152 0.878906 19.1895 0.488281 20.0684-0.439453L68.5547-48.877L76.6113-58.0078L75.7324-35.2051L75.7324-17.1387C75.7324-14.8438 77.7344-12.6953 80.127-12.6953C82.4707-12.6953 84.5703-14.6973 84.5703-17.334Z"></path></g></svg>`
  304. var relatedExtensions = [
  305. {
  306. id: "utags",
  307. title: "\u{1F3F7}\uFE0F UTags - Add usertags to links",
  308. url: "https://greasyfork.org/scripts/460718",
  309. },
  310. {
  311. id: "links-helper",
  312. title: "\u{1F517} \u94FE\u63A5\u52A9\u624B",
  313. description:
  314. "\u5728\u65B0\u6807\u7B7E\u9875\u4E2D\u6253\u5F00\u7B2C\u4E09\u65B9\u7F51\u7AD9\u94FE\u63A5\uFF0C\u56FE\u7247\u94FE\u63A5\u8F6C\u56FE\u7247\u6807\u7B7E\u7B49",
  315. url: "https://greasyfork.org/scripts/464541",
  316. },
  317. {
  318. id: "v2ex.rep",
  319. title:
  320. "V2EX.REP - \u4E13\u6CE8\u63D0\u5347 V2EX \u4E3B\u9898\u56DE\u590D\u6D4F\u89C8\u4F53\u9A8C",
  321. url: "https://greasyfork.org/scripts/466589",
  322. },
  323. {
  324. id: "v2ex.min",
  325. title: "v2ex.min - V2EX \u6781\u7B80\u98CE\u683C",
  326. url: "https://greasyfork.org/scripts/463552",
  327. },
  328. ]
  329. var getInstalledExtesionList = () => {
  330. return $(".extension_list_container .installed_extension_list")
  331. }
  332. var getRelatedExtesionList = () => {
  333. return $(".extension_list_container .related_extension_list")
  334. }
  335. var isInstalledExtension = (id) => {
  336. const list = getInstalledExtesionList()
  337. if (!list) {
  338. return false
  339. }
  340. const installed = $(`[data-extension-id="${id}"]`, list)
  341. return Boolean(installed)
  342. }
  343. var addCurrentExtension = (extension) => {
  344. const list = getInstalledExtesionList()
  345. if (!list) {
  346. return
  347. }
  348. if (isInstalledExtension(extension.id)) {
  349. return
  350. }
  351. const element = createInstalledExtension(extension)
  352. list.append(element)
  353. const list2 = getRelatedExtesionList()
  354. if (list2) {
  355. updateRelatedExtensions(list2)
  356. }
  357. }
  358. var activeExtension = (id) => {
  359. const list = getInstalledExtesionList()
  360. if (!list) {
  361. return false
  362. }
  363. for (const element of $$(".active", list)) {
  364. removeClass(element, "active")
  365. }
  366. const installed = $(`[data-extension-id="${id}"]`, list)
  367. if (installed) {
  368. addClass(installed, "active")
  369. }
  370. }
  371. var createInstalledExtension = (installedExtension) => {
  372. const div = createElement("div", {
  373. class: "installed_extension",
  374. "data-extension-id": installedExtension.id,
  375. })
  376. const a = addElement2(div, "a", {
  377. onclick: installedExtension.onclick,
  378. })
  379. addElement2(a, "span", {
  380. textContent: installedExtension.title,
  381. })
  382. const svg = addElement2(a, "svg")
  383. svg.outerHTML = openButton
  384. return div
  385. }
  386. var updateRelatedExtensions = (container) => {
  387. container.innerHTML = ""
  388. for (const relatedExtension of relatedExtensions) {
  389. console.log(relatedExtension)
  390. if (isInstalledExtension(relatedExtension.id)) {
  391. continue
  392. }
  393. const div4 = addElement2(container, "div", {
  394. class: "related_extension",
  395. })
  396. const a = addElement2(div4, "a", {
  397. href: relatedExtension.url,
  398. target: "_blank",
  399. })
  400. addElement2(a, "span", {
  401. textContent: relatedExtension.title,
  402. })
  403. const svg = addElement2(a, "svg")
  404. svg.outerHTML = openInNewTabButton
  405. }
  406. }
  407. function createExtensionList(installedExtensions) {
  408. const div = createElement("div", {
  409. class: "extension_list_container thin_scrollbar",
  410. })
  411. addElement2(div, "h1", { textContent: "Settings" })
  412. const div2 = addElement2(div, "div", {
  413. class: "installed_extension_list",
  414. })
  415. for (const installedExtension of installedExtensions) {
  416. console.log(installedExtension)
  417. if (isInstalledExtension(installedExtension.id)) {
  418. continue
  419. }
  420. const element = createInstalledExtension(installedExtension)
  421. div2.append(element)
  422. }
  423. addElement2(div, "h2", { textContent: "Other Extensions" })
  424. const div3 = addElement2(div, "div", {
  425. class: "related_extension_list",
  426. })
  427. updateRelatedExtensions(div3)
  428. return div
  429. }
  430. var prefix = "browser_extension_settings_"
  431. var randomId = String(Math.round(Math.random() * 1e4))
  432. var settingsContainerId = prefix + "container_" + randomId
  433. var settingsElementId = prefix + "main_" + randomId
  434. var getSettingsElement = () => $("#" + settingsElementId)
  435. var getSettingsStyle = () =>
  436. style_default
  437. .replace(/browser_extension_settings_container/gm, settingsContainerId)
  438. .replace(/browser_extension_settings_main/gm, settingsElementId)
  439. var storageKey = "settings"
  440. var settingsOptions
  441. var settingsTable = {}
  442. var settings = {}
  443. async function getSettings() {
  444. var _a
  445. return (_a = await getValue(storageKey)) != null ? _a : {}
  446. }
  447. async function saveSattingsValue(key, value) {
  448. const settings2 = await getSettings()
  449. settings2[key] =
  450. settingsTable[key] && settingsTable[key].defaultValue === value
  451. ? void 0
  452. : value
  453. await setValue(storageKey, settings2)
  454. }
  455. function getSettingsValue(key) {
  456. var _a
  457. return Object.hasOwn(settings, key)
  458. ? settings[key]
  459. : (_a = settingsTable[key]) == null
  460. ? void 0
  461. : _a.defaultValue
  462. }
  463. var modalHandler = (event) => {
  464. let target = event.target
  465. const settingsContainer = getSettingsContainer()
  466. if (settingsContainer) {
  467. while (target !== settingsContainer && target) {
  468. target = target.parentNode
  469. }
  470. if (target === settingsContainer) {
  471. return
  472. }
  473. settingsContainer.style.display = "none"
  474. }
  475. removeEventListener(document, "click", modalHandler)
  476. }
  477. async function updateOptions() {
  478. if (!getSettingsElement()) {
  479. return
  480. }
  481. for (const key in settingsTable) {
  482. if (Object.hasOwn(settingsTable, key)) {
  483. const checkbox = $(
  484. `#${settingsElementId} .option_groups .switch_option[data-key="${key}"] input`
  485. )
  486. if (checkbox) {
  487. checkbox.checked = getSettingsValue(key)
  488. }
  489. }
  490. }
  491. const host2 = location.host
  492. const group2 = $(`#${settingsElementId} .option_groups:nth-of-type(2)`)
  493. if (group2) {
  494. group2.style.display = getSettingsValue(
  495. `enableCustomRulesForCurrentSite_${host2}`
  496. )
  497. ? "block"
  498. : "none"
  499. }
  500. const customStyleValue = $(`#${settingsElementId} .option_groups textarea`)
  501. if (customStyleValue) {
  502. customStyleValue.value =
  503. settings[`customRulesForCurrentSite_${host2}`] || ""
  504. }
  505. }
  506. function getSettingsContainer() {
  507. const container = $(`.${prefix}container`)
  508. if (container) {
  509. const theVersion = Number.parseInt(
  510. container.dataset.besVersion || "0",
  511. 10
  512. )
  513. if (theVersion < version) {
  514. container.id = settingsContainerId
  515. container.dataset.besVersion = String(version)
  516. }
  517. return container
  518. }
  519. return addElement2(doc.body, "div", {
  520. id: settingsContainerId,
  521. class: `${prefix}container`,
  522. "data-bes-version": version,
  523. })
  524. }
  525. function getSettingsWrapper() {
  526. return (
  527. $(`.${prefix}container .${prefix}wrapper`) ||
  528. addElement2(getSettingsContainer(), "div", {
  529. class: `${prefix}wrapper`,
  530. })
  531. )
  532. }
  533. function initExtensionList() {
  534. if (!doc.body) {
  535. setTimeout(initExtensionList, 100)
  536. return
  537. }
  538. const wrapper = getSettingsWrapper()
  539. if (!$(".extension_list_container", wrapper)) {
  540. const list = createExtensionList([])
  541. wrapper.append(list)
  542. }
  543. addCurrentExtension({
  544. id: settingsOptions.id,
  545. title: settingsOptions.title,
  546. onclick: showSettings,
  547. })
  548. }
  549. function createSettingsElement() {
  550. let settingsMain = getSettingsElement()
  551. if (!settingsMain) {
  552. const wrapper = getSettingsWrapper()
  553. for (const element of $$(`.${prefix}main`)) {
  554. element.remove()
  555. }
  556. settingsMain = addElement2(wrapper, "div", {
  557. id: settingsElementId,
  558. class: `${prefix}main thin_scrollbar`,
  559. })
  560. addElement2(settingsMain, "a", {
  561. textContent: "Settings",
  562. class: "navigation_go_previous",
  563. onclick() {
  564. console.log(1)
  565. },
  566. })
  567. if (settingsOptions.title) {
  568. addElement2(settingsMain, "h2", { textContent: settingsOptions.title })
  569. }
  570. const options = addElement2(settingsMain, "div", {
  571. class: "option_groups",
  572. })
  573. for (const key in settingsTable) {
  574. if (Object.hasOwn(settingsTable, key)) {
  575. const item = settingsTable[key]
  576. if (!item.type || item.type === "switch") {
  577. const switchOption = createSwitchOption(item.title, {
  578. async onchange(event) {
  579. await saveSattingsValue(key, event.target.checked)
  580. },
  581. })
  582. switchOption.dataset.key = key
  583. addElement2(options, switchOption)
  584. }
  585. }
  586. }
  587. const options2 = addElement2(settingsMain, "div", {
  588. class: "option_groups",
  589. })
  590. let timeoutId
  591. addElement2(options2, "textarea", {
  592. placeholder: `/* Custom rules for internal URLs, matching URLs will be opened in new tabs */`,
  593. onkeyup(event) {
  594. if (timeoutId) {
  595. clearTimeout(timeoutId)
  596. timeoutId = null
  597. }
  598. timeoutId = setTimeout(async () => {
  599. const host2 = location.host
  600. await saveSattingsValue(
  601. `customRulesForCurrentSite_${host2}`,
  602. event.target.value.trim()
  603. )
  604. }, 100)
  605. },
  606. })
  607. const tip = addElement2(options2, "div", {
  608. class: "tip",
  609. })
  610. addElement2(tip, "a", {
  611. class: "tip_anchor",
  612. textContent: "Examples",
  613. })
  614. const tipContent = addElement2(tip, "div", {
  615. class: "tip_content",
  616. innerHTML: `<p>Custom rules for internal URLs, matching URLs will be opened in new tabs</p>
  617. <p>
  618. - One line per url pattern<br>
  619. - All URLs contains '/posts' or '/users/'<br>
  620. <pre>/posts/
  621. /users/</pre>
  622. - Regex is supported<br>
  623. <pre>^/(posts|members)/d+</pre>
  624. - '*' for all URLs
  625. </p>`,
  626. })
  627. if (settingsOptions.footer) {
  628. const footer = addElement2(settingsMain, "footer")
  629. footer.innerHTML =
  630. typeof settingsOptions.footer === "string"
  631. ? settingsOptions.footer
  632. : `<p>Made with \u2764\uFE0F by
  633. <a href="https://www.pipecraft.net/" target="_blank">
  634. Pipecraft
  635. </a></p>`
  636. }
  637. }
  638. return settingsMain
  639. }
  640. function addSideMenu() {
  641. if (!doc.body) {
  642. setTimeout(addSideMenu, 100)
  643. return
  644. }
  645. const menu =
  646. $("#browser_extension_side_menu") ||
  647. addElement2(doc.body, "div", {
  648. id: "browser_extension_side_menu",
  649. "data-bes-version": version,
  650. })
  651. const button = $("button[data-bes-version]", menu)
  652. if (button) {
  653. const theVersion = Number.parseInt(button.dataset.besVersion || "0", 10)
  654. if (theVersion >= version) {
  655. return
  656. }
  657. button.remove()
  658. }
  659. addElement2(menu, "button", {
  660. type: "button",
  661. "data-bes-version": version,
  662. title: "\u8BBE\u7F6E",
  663. onclick() {
  664. setTimeout(showSettings, 1)
  665. },
  666. })
  667. }
  668. async function showSettings() {
  669. const settingsContainer = getSettingsContainer()
  670. const settingsMain = createSettingsElement()
  671. await updateOptions()
  672. settingsContainer.style.display = "block"
  673. addEventListener(document, "click", modalHandler)
  674. activeExtension(settingsOptions.id)
  675. }
  676. var initSettings = async (options) => {
  677. settingsOptions = options
  678. settingsTable = options.settingsTable || {}
  679. addValueChangeListener(storageKey, async () => {
  680. settings = await getSettings()
  681. await updateOptions()
  682. if (typeof options.onValueChange === "function") {
  683. options.onValueChange()
  684. }
  685. })
  686. settings = await getSettings()
  687. addStyle2(getSettingsStyle())
  688. initExtensionList()
  689. addSideMenu()
  690. }
  691. var image_url_default =
  692. '{\n "imgur.com": [\n "https?://imgur.com/(\\\\w+)($|\\\\?) -> https://i.imgur.com/$1.png # ex: https://imgur.com/gi2b1rj",\n "https?://imgur.com/(\\\\w+)\\\\.(\\\\w+) -> https://i.imgur.com/$1.$2 # ex: https://imgur.com/gi2b1rj.png"\n ],\n "imgur.io": [\n "https?://imgur.io/(\\\\w+)($|\\\\?) -> https://i.imgur.com/$1.png # ex: https://imgur.io/gi2b1rj",\n "https?://imgur.io/(\\\\w+)\\\\.(\\\\w+) -> https://i.imgur.com/$1.$2 # ex: https://imgur.io/gi2b1rj.png"\n ],\n "i.imgur.com": [\n "https?://i.imgur.com/(\\\\w+)($|\\\\?) -> https://i.imgur.com/$1.png"\n ],\n "camo.githubusercontent.com": [\n "https://camo.githubusercontent.com/.* # This is a img url, no need to replace value"\n ]\n}\n'
  693. var rules = JSON.parse(image_url_default)
  694. var cachedRules = {}
  695. var getHostname = (url) => (/https?:\/\/([^/]+)/.exec(url) || [])[1]
  696. var processRule = (rule, href) => {
  697. var _a
  698. let pattern
  699. let replacement
  700. const cachedRule = cachedRules[rule]
  701. try {
  702. if (cachedRule) {
  703. pattern = cachedRule.pattern
  704. replacement = cachedRule.replacement
  705. } else {
  706. const result = rule.replace(/ #.*/, "").split("->")
  707. const patternString = result[0].trim()
  708. pattern = new RegExp(
  709. patternString.startsWith("http")
  710. ? "^" + patternString
  711. : patternString,
  712. "i"
  713. )
  714. replacement = (_a = result[1]) == null ? void 0 : _a.trim()
  715. cachedRules[rule] = { pattern, replacement }
  716. }
  717. if (pattern.test(href)) {
  718. return replacement ? href.replace(pattern, replacement) : href
  719. }
  720. } catch (error) {
  721. console.error(error)
  722. }
  723. }
  724. var convertImgUrl = (href) => {
  725. if (!href) {
  726. return
  727. }
  728. const hostname = getHostname(href)
  729. if (Object.hasOwn(rules, hostname)) {
  730. for (const rule of rules[hostname]) {
  731. const newHref = processRule(rule, href)
  732. if (newHref) {
  733. return newHref
  734. }
  735. }
  736. }
  737. }
  738. var createImgTagString = (src, text) =>
  739. `<img src="${src}" title="${text || "image"}" alt="${
  740. // eslint-disable-next-line @typescript-eslint/prefer-nullish-coalescing
  741. text || "image"
  742. }" role="img" style="max-width: 100% !important; vertical-align: bottom;" loading="lazy" referrerpolicy="no-referrer" rel="noreferrer" data-lh-status="1"/>`
  743. var bindOnError = () => {
  744. for (const element of $$('img[data-lh-status="1"]')) {
  745. setAttribute(element, "data-lh-status", "2")
  746. addEventListener(element, "error", (event) => {
  747. const img = event.target
  748. const anchor = img.parentElement
  749. img.outerHTML = getAttribute(img, "src")
  750. if ((anchor == null ? void 0 : anchor.tagName) === "A") {
  751. setStyle(anchor, "opacity: 50%;")
  752. setAttribute(anchor, "data-message", "failed to load image")
  753. }
  754. })
  755. }
  756. }
  757. var anchorElementToImgElement = (anchor, href, text) => {
  758. anchor.innerHTML = createImgTagString(href, text)
  759. setAttribute(anchor, "target", "_blank")
  760. addAttribute(anchor, "rel", "noopener")
  761. addAttribute(anchor, "rel", "noreferrer")
  762. }
  763. var linkToImg = (anchor) => {
  764. if (
  765. !anchor ||
  766. anchor.childElementCount !== 0 ||
  767. (anchor.childNodes[0] && anchor.childNodes[0].nodeType !== 3)
  768. ) {
  769. return
  770. }
  771. const href = anchor.href
  772. const text = anchor.textContent || href
  773. const newHref = convertImgUrl(href)
  774. if (newHref) {
  775. anchorElementToImgElement(anchor, newHref, text)
  776. } else if (
  777. /^https:[^?]+\.(?:jpg|jpeg|jpe|bmp|png|gif|webp|ico|svg)/i.test(href)
  778. ) {
  779. anchorElementToImgElement(anchor, href, text)
  780. }
  781. }
  782. var ignoredTags = /* @__PURE__ */ new Set([
  783. "A",
  784. "BUTTON",
  785. "SVG",
  786. "PATH",
  787. "G",
  788. "SCRIPT",
  789. "STYLE",
  790. "TEXTAREA",
  791. "CODE",
  792. "PRE",
  793. "TEMPLATE",
  794. "NOSCRIPT",
  795. "TITLE",
  796. ])
  797. var urlPattern =
  798. "\\b((?:https?:\\/\\/(?:[\\w-.]+\\.[a-z]{2,15}|localhost|(?:\\d{1,3}\\.){3}\\d{1,3}))(?::\\d+)?(?:\\/[\\w-/%.~+:;!@=&?#]*)?)"
  799. var linkPattern1 = new RegExp(
  800. `!\\[([^\\[\\]]*)\\]\\((?:\\s|<br/?>)*${urlPattern}(?:\\s|<br/?>)*\\)`,
  801. "gim"
  802. )
  803. var linkPattern2 = new RegExp(
  804. `\\[([^\\[\\]]*)\\]\\((?:\\s|<br/?>)*${urlPattern}(?:\\s|<br/?>)*\\)`,
  805. "gim"
  806. )
  807. var linkPattern3 = new RegExp(urlPattern, "gim")
  808. var linkPattern4 = new RegExp(
  809. `\\[img\\](?:\\s|<br/?>)*${urlPattern}(?:\\s|<br/?>)*\\[/img\\]`,
  810. "gim"
  811. )
  812. var linkPattern5 = new RegExp(
  813. `\\[url\\](?:\\s|<br/?>)*${urlPattern}(?:\\s|<br/?>)*\\[/url\\]`,
  814. "gim"
  815. )
  816. var linkPattern6 = new RegExp(
  817. `\\[url=${urlPattern}\\]([^\\[\\]]+)\\[/url\\]`,
  818. "gim"
  819. )
  820. var replaceMarkdownImgLinks = (text) => {
  821. if (text.search(linkPattern1) >= 0) {
  822. text = text.replace(linkPattern1, (m, p1, p2) => {
  823. return createImgTagString(convertImgUrl(p2) || p2, p1)
  824. })
  825. }
  826. return text
  827. }
  828. var replaceMarkdownLinks = (text) => {
  829. if (text.search(linkPattern2) >= 0) {
  830. text = text.replace(linkPattern2, (m, p1, p2) => {
  831. return `<a href="${p2}">${p1.replace(/<br>$/gi, "")}</a>`
  832. })
  833. }
  834. return text
  835. }
  836. var replaceTextLinks = (text) => {
  837. if (text.search(linkPattern3) >= 0) {
  838. text = text.replace(linkPattern3, (m, p1) => {
  839. return `<a href="${p1}">${p1}</a>`
  840. })
  841. }
  842. return text
  843. }
  844. var replaceBBCodeImgLinks = (text) => {
  845. if (text.search(linkPattern4) >= 0) {
  846. text = text.replace(linkPattern4, (m, p1) => {
  847. return createImgTagString(convertImgUrl(p1) || p1, p1)
  848. })
  849. }
  850. return text
  851. }
  852. var replaceBBCodeLinks = (text) => {
  853. if (text.search(linkPattern5) >= 0) {
  854. text = text.replace(linkPattern5, (m, p1) => {
  855. return `<a href="${p1}">${p1}</a>`
  856. })
  857. }
  858. if (text.search(linkPattern6) >= 0) {
  859. text = text.replace(linkPattern6, (m, p1, p2) => {
  860. return `<a href="${p1}">${p2}</a>`
  861. })
  862. }
  863. return text
  864. }
  865. var textToLink = (textNode, previousText) => {
  866. var _a, _b
  867. const textContent = (_a = textNode.textContent) != null ? _a : ""
  868. const parentNode = textNode.parentNode
  869. const mergedText = previousText + textContent
  870. if (
  871. !parentNode ||
  872. textNode.nodeName !== "#text" ||
  873. textContent.trim().length === 0 ||
  874. mergedText.trim().length < 3
  875. ) {
  876. return
  877. }
  878. if (textContent.includes("://")) {
  879. const original = textContent
  880. let newContent = original
  881. if (new RegExp("\\[.*]\\(", "ms").test(original)) {
  882. newContent = replaceMarkdownImgLinks(newContent)
  883. newContent = replaceMarkdownLinks(newContent)
  884. }
  885. if (/\[(img|url)]|\[url=/.test(textContent)) {
  886. newContent = replaceBBCodeImgLinks(newContent)
  887. newContent = replaceBBCodeLinks(newContent)
  888. }
  889. if (newContent === original) {
  890. newContent = replaceTextLinks(original)
  891. } else {
  892. newContent = newContent.replace(
  893. new RegExp(
  894. "(<a(?:\\s[^<>]*)?>.*?<\\/a>)|(<img(?:\\s[^<>]*)?\\/?>)|(.+?(?=(?:<a|<img))|.+$)",
  895. "gims"
  896. ),
  897. (m, p1, p2) => (p1 || p2 ? m : replaceTextLinks(m))
  898. )
  899. }
  900. if (newContent !== original) {
  901. const span = createElement("span")
  902. span.innerHTML = newContent
  903. textNode.after(span)
  904. textNode.remove()
  905. return true
  906. }
  907. }
  908. const parentTextContent = (_b = parentNode.textContent) != null ? _b : ""
  909. if (
  910. new RegExp("\\[.*]\\(", "ms").test(mergedText) &&
  911. (parentTextContent.search(linkPattern2) >= 0 ||
  912. $$("img", parentNode).length > 0)
  913. ) {
  914. const original = parentNode.innerHTML
  915. const newContent = original
  916. .replace(new RegExp("\\[.*]\\([^[\\]()]+?\\)", "gims"), (m) =>
  917. m
  918. .replace(
  919. /<img[^<>]*\ssrc=['"]?(http[^'"]+)['"]?(\s[^<>]*)?>/gim,
  920. "$1"
  921. )
  922. .replace(
  923. /\((?:\s|<br\/?>)*<a[^<>]*\shref=['"]?(http[^'"]+)['"]?(\s[^<>]*)?>\1<\/a>(?:\s|<br\/?>)*\)/gim,
  924. "($1)"
  925. )
  926. )
  927. .replace(
  928. new RegExp("\\[!\\[.*]\\([^()]+\\)]\\([^[\\]()]+?\\)", "gims"),
  929. (m) =>
  930. m
  931. .replace(
  932. /<img[^<>]*\ssrc=['"]?(http[^'"]+)['"]?(\s[^<>]*)?>/gim,
  933. "$1"
  934. )
  935. .replace(
  936. /\((?:\s|<br\/?>)*<a[^<>]*\shref=['"]?(http[^'"]+)['"]?(\s[^<>]*)?>\1<\/a>(?:\s|<br\/?>)*\)/gim,
  937. "($1)"
  938. )
  939. )
  940. if (newContent !== original) {
  941. let newContent2 = replaceMarkdownImgLinks(newContent)
  942. newContent2 = replaceMarkdownLinks(newContent2)
  943. if (newContent2 !== newContent) {
  944. parentNode.innerHTML = newContent2
  945. return true
  946. }
  947. }
  948. }
  949. if (
  950. /\[(img|url)]|\[url=/.test(textContent) &&
  951. parentTextContent.search(/\[(img|url)[^\]]*]([^[\]]*?)\[\/\1]/) >= 0
  952. ) {
  953. const original = parentNode.innerHTML
  954. let before = ""
  955. let after = original
  956. let count = 0
  957. while (before !== after && count < 5) {
  958. count++
  959. before = after
  960. after = before.replace(
  961. /\[(img|url)[^\]]*]([^[\]]+?)\[\/\1]/gim,
  962. (m, p1) => {
  963. let tagsRemoved
  964. let converted
  965. if (p1 === "img") {
  966. tagsRemoved = m
  967. .replace(
  968. /<img[^<>]*\ssrc=['"]?(http[^'"]+)['"]?(\s[^<>]*)?>/gim,
  969. "$1"
  970. )
  971. .replace(
  972. /\[img](?:\s|<br\/?>)*<a[^<>]*\shref=['"]?(http[^'"]+)['"]?(\s[^<>]*)?>\1<\/a>(?:\s|<br\/?>)*\[\/img]/gim,
  973. "[img]$1[/img]"
  974. )
  975. converted = replaceBBCodeImgLinks(tagsRemoved)
  976. } else {
  977. tagsRemoved = m
  978. .replace(
  979. /\[url](?:\s|<br\/?>)*<a[^<>]*\shref=['"]?(http[^'"]+)['"]?(\s[^<>]*)?>\1<\/a>(?:\s|<br\/?>)*\[\/url]/gim,
  980. "[url]$1[/url]"
  981. )
  982. .replace(
  983. /\[url=<a[^<>]*\shref=['"]?(http[^'"]+)['"]?(\s[^<>]*)?>\1<\/a>]/gim,
  984. "[url=$1]"
  985. )
  986. converted = replaceBBCodeLinks(tagsRemoved)
  987. }
  988. return converted === tagsRemoved ? m : converted
  989. }
  990. )
  991. }
  992. const newContent = after
  993. if (newContent !== original) {
  994. parentNode.innerHTML = newContent
  995. return true
  996. }
  997. }
  998. }
  999. var fixAnchorTag = (anchorElement) => {
  1000. var _a
  1001. const href = anchorElement.href
  1002. const textContent = (_a = anchorElement.textContent) != null ? _a : ""
  1003. const nextSibling = anchorElement.nextSibling
  1004. if (
  1005. anchorElement.childElementCount === 0 &&
  1006. href.includes(")") &&
  1007. textContent.includes(")")
  1008. ) {
  1009. const index = textContent.indexOf(")")
  1010. const removed = textContent.slice(Math.max(0, index))
  1011. anchorElement.textContent = textContent.slice(0, Math.max(0, index))
  1012. anchorElement.href = anchorElement.href.slice(
  1013. 0,
  1014. Math.max(0, href.indexOf(")"))
  1015. )
  1016. if (nextSibling && nextSibling.nodeType === 3) {
  1017. nextSibling.textContent = removed + nextSibling.textContent
  1018. } else {
  1019. anchorElement.after(doc.createTextNode(removed))
  1020. }
  1021. }
  1022. }
  1023. var scanAndConvertChildNodes = (parentNode) => {
  1024. if (
  1025. !parentNode ||
  1026. parentNode.nodeType === 8 ||
  1027. !parentNode.tagName ||
  1028. ignoredTags.has(parentNode.tagName.toUpperCase())
  1029. ) {
  1030. if (parentNode.tagName === "A") {
  1031. fixAnchorTag(parentNode)
  1032. }
  1033. return
  1034. }
  1035. let previousText = ""
  1036. for (const child of parentNode.childNodes) {
  1037. try {
  1038. if (child.nodeName === "#text") {
  1039. if (textToLink(child, previousText)) {
  1040. scanAndConvertChildNodes(parentNode)
  1041. break
  1042. }
  1043. previousText += child.textContent
  1044. } else if (child.nodeName === "BR") {
  1045. previousText += "\n"
  1046. } else {
  1047. previousText = ""
  1048. scanAndConvertChildNodes(child)
  1049. }
  1050. } catch (error) {
  1051. console.error(error)
  1052. }
  1053. }
  1054. }
  1055. var origin = location.origin
  1056. var host = location.host
  1057. var config = {
  1058. run_at: "document_start",
  1059. }
  1060. var settingsTable2 = {
  1061. enable: {
  1062. title: "Enable",
  1063. defaultValue: true,
  1064. },
  1065. [`enableCurrentSite_${host}`]: {
  1066. title: "Enable current site",
  1067. defaultValue: true,
  1068. },
  1069. [`enableCustomRulesForCurrentSite_${host}`]: {
  1070. title: "Enable custom rules for current site",
  1071. defaultValue: false,
  1072. },
  1073. [`customRulesForCurrentSite_${host}`]: {
  1074. title: "Enable custom rules for current site",
  1075. defaultValue: "",
  1076. type: "textarea",
  1077. },
  1078. }
  1079. function registerMenuCommands() {
  1080. registerMenuCommand("\u2699\uFE0F \u8BBE\u7F6E", showSettings, "o")
  1081. }
  1082. var getWithoutOrigin = (url) => url.replace(/(^https?:\/\/[^/]+)/, "")
  1083. var shouldOpenInNewTab = (element) => {
  1084. var _a
  1085. const url = element.href
  1086. if (
  1087. !url ||
  1088. !/^https?:\/\//.test(url) ||
  1089. ((_a = element.getAttribute("href")) == null
  1090. ? void 0
  1091. : _a.startsWith("#"))
  1092. ) {
  1093. return false
  1094. }
  1095. if (element.origin !== origin) {
  1096. return true
  1097. }
  1098. if (getSettingsValue(`enableCustomRulesForCurrentSite_${host}`)) {
  1099. const rules2 = (
  1100. getSettingsValue(`customRulesForCurrentSite_${host}`) || ""
  1101. ).split("\n")
  1102. if (rules2.includes("*")) {
  1103. return true
  1104. }
  1105. const hrefWithoutOrigin = getWithoutOrigin(url)
  1106. for (let rule of rules2) {
  1107. rule = rule.trim()
  1108. if (rule.length === 0) {
  1109. continue
  1110. }
  1111. try {
  1112. const regexp = new RegExp(rule)
  1113. if (regexp.test(hrefWithoutOrigin)) {
  1114. return true
  1115. }
  1116. } catch (error) {
  1117. console.log(error.message)
  1118. if (hrefWithoutOrigin.includes(rule)) {
  1119. return true
  1120. }
  1121. }
  1122. }
  1123. }
  1124. }
  1125. var setAttributeAsOpenInNewTab = (element) => {
  1126. if (shouldOpenInNewTab(element)) {
  1127. setAttribute(element, "target", "_blank")
  1128. addAttribute(element, "rel", "noopener")
  1129. }
  1130. }
  1131. async function main() {
  1132. await initSettings({
  1133. id: "links-helper",
  1134. title: "\u{1F517} Links Helper",
  1135. footer: `
  1136. <p>After change settings, reload the page to take effect</p>
  1137. <p>
  1138. <a href="https://github.com/utags/links-helper/issues" target="_blank">
  1139. Report and Issue...
  1140. </a></p>
  1141. <p>Made with \u2764\uFE0F by
  1142. <a href="https://www.pipecraft.net/" target="_blank">
  1143. Pipecraft
  1144. </a></p>`,
  1145. settingsTable: settingsTable2,
  1146. })
  1147. registerMenuCommands()
  1148. if (
  1149. !getSettingsValue("enable") ||
  1150. !getSettingsValue(`enableCurrentSite_${host}`)
  1151. ) {
  1152. return
  1153. }
  1154. addEventListener(
  1155. document,
  1156. "click",
  1157. (event) => {
  1158. let anchorElement = event.target
  1159. while (anchorElement && anchorElement.tagName !== "A") {
  1160. anchorElement = anchorElement.parentNode
  1161. }
  1162. if (anchorElement) {
  1163. setAttributeAsOpenInNewTab(anchorElement)
  1164. if (getAttribute(anchorElement, "target") === "_blank") {
  1165. event.stopImmediatePropagation()
  1166. event.stopPropagation()
  1167. }
  1168. }
  1169. },
  1170. true
  1171. )
  1172. const scanAnchors = () => {
  1173. for (const element of $$("a")) {
  1174. if (element.__links_helper_scaned) {
  1175. continue
  1176. }
  1177. element.__links_helper_scaned = 1
  1178. try {
  1179. setAttributeAsOpenInNewTab(element)
  1180. } catch (error) {
  1181. console.error(error)
  1182. }
  1183. try {
  1184. linkToImg(element)
  1185. } catch (error) {
  1186. console.error(error)
  1187. }
  1188. }
  1189. }
  1190. const scanNodes = throttle(() => {
  1191. scanAndConvertChildNodes(doc.body)
  1192. scanAnchors()
  1193. bindOnError()
  1194. }, 500)
  1195. const observer = new MutationObserver(() => {
  1196. scanNodes()
  1197. })
  1198. const startObserver = () => {
  1199. observer.observe(doc.body, {
  1200. childList: true,
  1201. subtree: true,
  1202. characterData: true,
  1203. })
  1204. }
  1205. if (doc.body) {
  1206. startObserver()
  1207. scanAndConvertChildNodes(doc.body)
  1208. } else {
  1209. const intervalId = setInterval(() => {
  1210. if (doc.body) {
  1211. clearInterval(intervalId)
  1212. startObserver()
  1213. scanAndConvertChildNodes(doc.body)
  1214. }
  1215. }, 100)
  1216. }
  1217. scanAnchors()
  1218. }
  1219. main()
  1220. })()