// ==UserScript== // @name 超链接过滤器 // @description 鼠标悬停时自动过滤链接 // @namespace http://tampermonkey.net/ // @match *://*/* // @grant GM_registerMenuCommand // @grant GM_setValue // @grant GM_getValue // @grant GM_addStyle // @run-at document-start // @version 2.0 // @author wOxxOm & Gemini // @license GPLv3 // @icon  // @downloadURL https://update.greasyfork.icu/scripts/557916/%E8%B6%85%E9%93%BE%E6%8E%A5%E8%BF%87%E6%BB%A4%E5%99%A8.user.js // @updateURL https://update.greasyfork.icu/scripts/557916/%E8%B6%85%E9%93%BE%E6%8E%A5%E8%BF%87%E6%BB%A4%E5%99%A8.meta.js // ==/UserScript== 'use strict'; const ICON_BASE64 = ''; const POPUP = document.createElement('a'); POPUP.id = GM_info.script.name; POPUP.title = '原始链接'; let isPopupStyled; let lastLink; let hoverTimer; let hoverStopTimer; // --- Custom Rules Logic Start --- // 初始化时加载规则 let cachedRules = []; const urlCache = new Map(); loadRules(); // 注册菜单 GM_registerMenuCommand("设置面板", openSettings); function getRules() { return GM_getValue('custom_rules', []); } // 加载并预编译规则 (优化性能的关键) function loadRules() { const rawRules = getRules(); cachedRules = rawRules.map(rule => { // 确保 enabled 默认为 true if (rule.enabled === undefined) rule.enabled = true; // 预编译匹配正则 let matchRegex = null; if (rule.useRegexMatch && rule.match) { try { matchRegex = new RegExp(rule.match); } catch(e) {} } // 预编译查找正则 let findRegex = null; if (rule.find) { try { if (rule.useRegexFind) { findRegex = new RegExp(rule.find, 'g'); } else { const escapedFind = rule.find.replace(/[.*+?^${}()|[\]\\]/g, '\\$&'); findRegex = new RegExp(escapedFind, 'g'); } } catch(e) {} } return { ...rule, _matchRegex: matchRegex, _findRegex: findRegex }; }); } function saveRules(rules) { GM_setValue('custom_rules', rules); loadRules(); urlCache.clear(); } // 应用自定义规则 function applyCustomRules(a) { const originalUrl = a.href; // 1. 查缓存 if (urlCache.has(originalUrl)) { const cachedResult = urlCache.get(originalUrl); if (cachedResult === null) return false; if (cachedResult.url !== originalUrl) { a.href = cachedResult.url; a.rel = 'external noreferrer nofollow noopener'; if (cachedResult.replaced) { a.hrefUndecloaked = originalUrl; } } return true; } const hostname = a.hostname || ""; // 2. 筛选阶段 const applicableRules = []; for (const rule of cachedRules) { if (rule.enabled === false) continue; let isMatch = false; if (rule.useRegexMatch) { if (rule._matchRegex) isMatch = rule._matchRegex.test(originalUrl); } else { if (rule.match && hostname.includes(rule.match)) isMatch = true; } if (isMatch) { applicableRules.push(rule); } } if (applicableRules.length === 0) { urlCache.set(originalUrl, null); return false; } // 3. 执行阶段 let currentUrl = originalUrl; let hasRuleReplacement = false; for (const rule of applicableRules) { try { let tempUrl = currentUrl; let urlBeforeRule = tempUrl; if (rule._findRegex) { let replaceText = rule.replace || ""; const upperReplace = replaceText.toUpperCase(); // 通用处理函数 const processMatch = (processor) => { return (...args) => { const match = args[0]; const captures = args.slice(1, -2); const target = (captures.length > 0 && captures[0] !== undefined) ? captures[0] : match; try { return processor(target); } catch (e) { // 解码失败时返回原字符串 避免破坏链接 return match; } }; }; // --- 变量处理逻辑 --- if (upperReplace === '{BASE64}') { tempUrl = tempUrl.replace(rule._findRegex, processMatch((target) => { // 1. 清理非 Base64 字符 (容错) 并处理 URL 安全字符 let base64 = target.replace(/[^A-Za-z0-9+/=_-]/g, '') .replace(/-/g, '+') .replace(/_/g, '/'); // 2. 补全 Padding while (base64.length % 4) base64 += '='; // 3. 解码为二进制字符串 const binary = atob(base64); // 4. 转换为字节数组并用 UTF-8 解码 const bytes = new Uint8Array(binary.length); for (let i = 0; i < binary.length; i++) bytes[i] = binary.charCodeAt(i); return new TextDecoder('utf-8').decode(bytes); })); } else if (upperReplace === '{URL}') { tempUrl = tempUrl.replace(rule._findRegex, processMatch((target) => { return decodeURIComponent(target); })); } else if (upperReplace === '{HEX}') { // --- 修复:支持 UTF-8 的 Hex 解码 --- tempUrl = tempUrl.replace(rule._findRegex, processMatch((target) => { // 1. 清理非 Hex 字符 (如空格) const hex = target.replace(/[^0-9a-fA-F]/g, ''); if (hex.length % 2 !== 0) return target; // 长度不对则不处理 // 2. 转换为字节数组 const bytes = new Uint8Array(hex.length / 2); for (let i = 0; i < hex.length; i += 2) { bytes[i / 2] = parseInt(hex.substr(i, 2), 16); } // 3. UTF-8 解码 return new TextDecoder('utf-8').decode(bytes); })); } else if (upperReplace === '{REVERSE}') { // --- 修复:支持 Emoji 和代理对的反转 --- tempUrl = tempUrl.replace(rule._findRegex, processMatch((target) => { return [...target].reverse().join(''); })); } else if (upperReplace === '{ROT13}') { tempUrl = tempUrl.replace(rule._findRegex, processMatch((target) => { // ROT13 仅针对 ASCII 字母 无需 UTF-8 处理 return target.replace(/[a-zA-Z]/g, c => String.fromCharCode((c <= 'Z' ? 90 : 122) >= (c = c.charCodeAt(0) + 13) ? c : c - 26)); })); } else { // 默认替换 if (!rule.useRegexReplace) replaceText = replaceText.replace(/\$/g, '$$$$'); tempUrl = tempUrl.replace(rule._findRegex, replaceText); } if (tempUrl !== urlBeforeRule) { hasRuleReplacement = true; } } try { tempUrl = decodeURIComponent(tempUrl); } catch (e) {} if (tempUrl !== currentUrl) { currentUrl = tempUrl; } } catch (e) { } } // 4. 结果处理 urlCache.set(originalUrl, { url: currentUrl, replaced: hasRuleReplacement }); if (currentUrl !== originalUrl) { a.href = currentUrl; a.rel = 'external noreferrer nofollow noopener'; if (hasRuleReplacement) { a.hrefUndecloaked = originalUrl; } } return true; } // 设置界面 UI function openSettings() { const existing = document.getElementById('decloak-settings-modal'); if (existing) return; // 样式调整:所有属性强制 !important 以防止网页样式干扰 GM_addStyle(` #decloak-settings-modal { all: initial !important; /* 重置所有继承属性 */ position: fixed !important; top: 0 !important; left: 0 !important; width: 100% !important; height: 100% !important; background: transparent !important; z-index: 2147483647 !important; display: flex !important; justify-content: center !important; align-items: center !important; font-family: sans-serif !important; font-size: 13px !important; color: #eee !important; pointer-events: none !important; line-height: normal !important; text-align: left !important; } #decloak-settings-modal * { box-sizing: border-box !important; } #decloak-settings-modal * { box-sizing: border-box !important; } #decloak-settings-content { background: rgb(44, 44, 44) !important; padding: 15px !important; border: 1px solid rgb(80, 80, 80) !important; width: 850px !important; max-width: 95% !important; max-height: 90% !important; display: flex !important; flex-direction: column !important; box-shadow: 0 10px 30px rgba(0,0,0,0.5) !important; pointer-events: auto !important; border-radius: 0 !important; } /* 头部布局:Flex 模拟表格列 + 绝对定位标题 */ .decloak-header { display: flex !important; gap: 5px !important; align-items: center !important; position: relative !important; margin-bottom: 10px !important; height: 30px !important; padding: 0 14px 0 6px !important; flex-shrink: 0 !important; } /* 标题绝对居中 */ .decloak-header-title { position: absolute !important; left: 0 !important; width: 100% !important; text-align: center !important; font-size: 16px !important; color: #fff !important; pointer-events: none !important; /* 防止遮挡点击 */ z-index: 0 !important; } /* 关闭按钮绝对定位在右侧 */ #decloak-close { position: absolute !important; right: 0 !important; z-index: 10 !important; border: none !important; background: none !important; cursor: pointer !important; font-size: 20px !important; line-height: 1 !important; color: #ccc !important; padding: 0 !important; } /* 帮助按钮样式 */ #decloak-help { position: static !important; width: 26px !important; height: auto !important; border: none !important; background: none !important; cursor: pointer !important; font-size: 15px !important; font-weight: bold !important; line-height: 1 !important; color: #999 !important; padding: 0 !important; display: flex !important; justify-content: center !important; align-items: center !important; } #decloak-help:hover { color: #fff !important; background: none !important; } /* 修改:帮助窗口 (独立居中 无遮罩) */ #decloak-help-window { display: none !important; position: fixed !important; top: 50% !important; left: 50% !important; transform: translate(-50%, -50%) !important; width: 320px !important; /* 稍微加宽 */ background: rgb(55, 55, 55) !important; border: 1px solid rgb(100, 100, 100) !important; box-shadow: 0 15px 40px rgba(0,0,0,0.8) !important; z-index: 2147483647 !important; flex-direction: column !important; padding: 15px !important; pointer-events: auto !important; } .decloak-help-header { display: flex !important; justify-content: space-between !important; align-items: center !important; margin-bottom: 15px !important; /* 移除底边框 改由网格标题承担 */ /* border-bottom: 1px solid #666 !important; padding-bottom: 5px !important; */ } .decloak-help-title { color: #fff !important; font-size: 14px !important; } #decloak-help-window-close { border: none !important; background: none !important; cursor: pointer !important; font-size: 18px !important; line-height: 1 !important; color: #ccc !important; padding: 0 !important; } #decloak-help-window-close:hover { color: #fff !important; } /* 新增:网格布局容器 */ #decloak-help-grid { display: grid !important; grid-template-columns: 100px 1fr !important; /* 左列100px 右列自动填充 */ gap: 8px 15px !important; /* 行间距10px 列间距15px */ align-items: center !important; } /* 新增:列标题样式 */ .decloak-help-col-header { color: #999 !important; font-size: 12px !important; border-bottom: 1px solid #666 !important; padding-bottom: 8px !important; margin-bottom: 5px !important; } .decloak-help-key { color: rgb(178, 139, 247) !important; user-select: none !important; cursor: pointer !important; transition: color 0.2s !important; font-size: 13px !important; } .decloak-help-key:active { transform: scale(0.98) !important; } /* 修改:描述样式 */ .decloak-help-desc { color: #ccc !important; user-select: text !important; font-size: 13px !important; line-height: 1.4 !important; } /* 搜索框样式 */ #decloak-search-input { background: #222 !important; border: 1px solid #555 !important; color: #eee !important; padding: 2px 5px !important; font-size: 13px !important; border-radius: 0 !important; margin: 0 !important; flex: 1 !important; /* 填满分配的空间 */ height: 26px !important; z-index: 5 !important; /* 确保在标题之上可点击 */ } #decloak-search-input:focus { border-color: #888 !important; outline: none !important; background: #111 !important; } .decloak-table-header { display: flex !important; gap: 5px !important; padding: 0 16px 5px 5px !important; font-size: 12px !important; color: #ccc !important; border-bottom: 1px solid #555 !important; margin-bottom: 0 !important; flex-shrink: 0 !important; } .decloak-rules-container { flex: 1 !important; overflow-y: scroll !important; margin-bottom: 10px !important; border: 1px solid #555 !important; border-top: none !important; background: #2a2a2a !important; } /* 紧凑行样式 */ .decloak-rule-row { display: flex !important; gap: 5px !important; align-items: center !important; background: #333 !important; padding: 4px 5px !important; border-bottom: 1px solid #444 !important; border-radius: 0 !important; margin: 0 !important; } .decloak-rule-row:nth-child(even) { background: #2e2e2e !important; } .decloak-rule-row:hover { background: #3a3a3a !important; } .decloak-rule-row.disabled { opacity: 0.5 !important; } .decloak-rule-row.disabled input { color: #999 !important; } .decloak-input-group { display: flex !important; flex-direction: column !important; flex: 1 !important; margin: 0 !important; padding: 0 !important; } .decloak-input-wrapper { display: flex !important; height: 26px !important; width: 100% !important; } .decloak-input { padding: 2px 5px !important; border: 1px solid #555 !important; background: #222 !important; flex: 1 !important; border-radius: 0 !important; height: 100% !important; width: 100% !important; font-size: 13px !important; font-family: sans-serif !important; margin: 0 !important; box-shadow: none !important; } .decloak-input:focus { border-color: #888 !important; outline: none !important; background: #111 !important; } /* 特定输入框颜色 */ .rule-match { color: rgb(77, 171, 247) !important; } .rule-find { color: rgb(246, 182, 78) !important; } .rule-replace { color: rgb(178, 139, 247) !important; } /* 按钮通用样式 */ .decloak-btn { padding: 0 !important; cursor: pointer !important; border: 1px solid #555 !important; background: #444 !important; color: #ccc !important; border-radius: 0 !important; height: 26px !important; min-width: 26px !important; display: flex !important; align-items: center !important; justify-content: center !important; font-size: 11px !important; margin: 0 !important; line-height: 1 !important; } .decloak-btn:hover { background: #555 !important; color: #fff !important; } /* 正则按钮激活颜色 RGB 118,202,83 */ .decloak-btn.active { background: rgb(118, 202, 83) !important; color: white !important; border-color: rgb(118, 202, 83) !important; } .decloak-btn-toggle { margin-right: 0 !important; } /* 删除按钮 */ .decloak-btn-danger { background: #333 !important; color: #ff6b6b !important; border: 1px solid #555 !important; width: 26px !important; } .decloak-btn-danger:hover { background: #d32f2f !important; color: white !important; border-color: #d32f2f !important; } /* 主要按钮 */ .decloak-btn-primary { background: #1976D2 !important; color: white !important; border: none !important; padding: 0 15px !important; width: auto !important; height: 30px !important; } .decloak-btn-primary:hover { background: #1565C0 !important; } /* 添加按钮 */ #decloak-add-rule { background: #333 !important; color: #ccc !important; border: 1px solid #555 !important; flex-shrink: 0 !important; width: auto !important; flex: 1 !important; margin-bottom: 0 !important; height: 30px !important; border-right: none !important; } #decloak-add-rule:hover { background: #3a3a3a !important; color: #fff !important; border-color: #777 !important; } .decloak-footer { display: flex !important; justify-content: flex-end !important; gap: 0 !important; padding: 4px 0px !important; flex-shrink: 0 !important; } /* 保存按钮样式 */ #decloak-save { width: 73px !important; height: 30px !important; padding: 0 !important; font-size: 12px !important; border-left: 1px solid #555 !important; white-space: nowrap !important; border: 1px solid #555 !important; } /* 输入框组合样式 */ .decloak-input-wrapper .decloak-input { border-right: none !important; } .decloak-input-wrapper .decloak-btn { border-left: 1px solid #555 !important; } /* 滚动条样式 */ .decloak-rules-container::-webkit-scrollbar { width: 10px !important; } .decloak-rules-container::-webkit-scrollbar-track { background: #222 !important; border-left: 1px solid #444 !important; } .decloak-rules-container::-webkit-scrollbar-thumb { background: #555 !important; } .decloak-rules-container::-webkit-scrollbar-thumb:hover { background: #777 !important; } `); const modal = document.createElement('div'); modal.id = 'decloak-settings-modal'; modal.innerHTML = `