// ==UserScript== // @name Convert Text to Hyperlink // @name:zh-CN 文本识别为超链接 // @namespace https://github.com/KPI0/tampermonkey // @version 1.5 // @description Convert URLs in text nodes to hyperlinks using regular expressions // @description:zh-cn 通过正则表达式将文本中的链接转换为超链接 // @author KPI0 // @match *://*/* // @icon  // @grant none // @license MIT // @downloadURL https://update.greasyfork.icu/scripts/511554/Convert%20Text%20to%20Hyperlink.user.js // @updateURL https://update.greasyfork.icu/scripts/511554/Convert%20Text%20to%20Hyperlink.meta.js // ==/UserScript== (function () { 'use strict'; // Regular expression to match URLs starting with http or https const urlRegex = /(http:\/\/[^\s]+|https:\/\/[^\s]+)/g; function convertTextLinksToHyperlinks(node) { if (node.nodeType === Node.TEXT_NODE) { const text = node.nodeValue; const matches = text.match(urlRegex); if (matches) { const span = document.createElement('span'); let lastIndex = 0; matches.forEach((match) => { const matchIndex = text.indexOf(match, lastIndex); // Append normal text if (matchIndex > lastIndex) { span.appendChild(document.createTextNode(text.substring(lastIndex, matchIndex))); } // Create hyperlink element const link = document.createElement('a'); link.href = match; link.target = '_blank'; // Open in a new tab link.textContent = match; span.appendChild(link); lastIndex = matchIndex + match.length; }); // Append remaining normal text if (lastIndex < text.length) { span.appendChild(document.createTextNode(text.substring(lastIndex))); } node.parentNode.replaceChild(span, node); } } else if (node.nodeType === Node.ELEMENT_NODE) { // Skip certain tags that should not contain links or be processed const skipTags = ['A', 'SCRIPT', 'STYLE', 'IMG', 'VIDEO', 'AUDIO', 'PICTURE', 'IFRAME', 'BUTTON', 'CANVAS', 'NAV', 'HEADER', 'FOOTER']; if (!skipTags.includes(node.tagName)) { // Recursively process child nodes for (let child of Array.from(node.childNodes)) { convertTextLinksToHyperlinks(child); } } } } // Execute after the page has fully loaded window.addEventListener('load', function () { const mainContentSelectors = ['#main', '.content', '.article', '#content', '.post', '.entry']; // Common main content containers mainContentSelectors.forEach(selector => { const container = document.querySelector(selector); if (container) { convertTextLinksToHyperlinks(container); } }); }); })();