// ==UserScript== // @name 🫧404小站 — 🛠️网页限制解除 // @name:en 🫧404小站 — 🛠️Remove web limits // @name:zh 🫧404小站 — 🛠️网页限制解除 // @name:zh-CN 🫧404小站 — 🛠️网页限制解除 // @name:zh-TW 🫧404小站 — 🛠️網頁限制解除 // @name:ja 🫧404小站 — 🛠️ウェブの規制緩和 // @description 通杀大部分网站,可以解除禁止复制、剪切、选择文本、右键菜单的限制。 // @description:en Pass to kill most of the site, you can lift the restrictions prohibited to copy, cut, select the text, right-click menu. // @description:zh 通杀大部分网站,可以解除禁止复制、剪切、选择文本、右键菜单的限制。 // @description:zh-CN 通杀大部分网站,可以解除禁止复制、剪切、选择文本、右键菜单的限制。 // @description:zh-TW 通殺大部分網站,可以解除禁止復制、剪切、選擇文本、右鍵菜單的限制。 // @description:ja サイトのほとんどを殺すために渡し、あなたは、コピー切り取り、テキスト、右クリックメニューを選択することは禁止の制限を解除することができます // @author yyy. // @version 1.0.2 // @license LGPLv3 // @match *://*/* // @grant GM_registerMenuCommand // @grant GM_getValue // @grant GM_setValue // @require https://unpkg.com/sweetalert2@10.16.6/dist/sweetalert2.min.js // @run-at document-start // ==/UserScript== (function() { 'use strict'; // 域名规则列表 var rules = { black_rule: { name: "black", hook_eventNames: "", unhook_eventNames: "" }, default_rule: { name: "default", hook_eventNames: "contextmenu|select|selectstart|copy|cut|dragstart", unhook_eventNames: "keydown|keyup", dom0: true, hook_addEventListener: true, hook_preventDefault: true, hook_set_returnValue: true, add_css: true } }; // 站点模式与存储 var MODES = { standard: 'standard', light: 'light', friendly: 'friendly', disabled: 'disabled' }; var MODE_LABELS = { standard: '标准模式', light: '轻量模式', friendly: '友好模式', disabled: '禁用' }; var siteModes = GM_getValue('site_modes', {}); function getSiteMode(host) { return siteModes[host] || ''; } function setSiteMode(host, mode) { siteModes[host] = mode; GM_setValue('site_modes', siteModes); } // 预设友好模式的常见视频网站后缀 var FRIENDLY_SUFFIXES = [ 'bilibili.com', 'iqiyi.com', 'youku.com', 'v.qq.com', 'video.qq.com', 'mgtv.com', 'acfun.cn', 'sohu.com', 'tv.sohu.com', 'pptv.com', 'le.com', 'tudou.com', 'youtube.com' ]; function getDefaultModeForHost(host) { for (var i = 0; i < FRIENDLY_SUFFIXES.length; i++) { var suf = FRIENDLY_SUFFIXES[i]; if (host === suf || host.slice(-suf.length - 1) === '.' + suf || host.slice(-suf.length) === suf) { return MODES.friendly; } } return ''; } // 域名列表(增加用户排除列表存储) var lists = { // 用户自定义排除列表 // 基础黑名单(不可删除) + 用户自定义排除列表 base_blacklist: GM_getValue('base_blacklist', [ 'youtube.com', 'wikipedia.org', 'mail.qq.com', 'translate.google.com' ]), // 保留给规则匹配的黑名单(正则表达式),默认为空 black_list: [], // 合并后的排除列表(仅用于显示) exclude_list: function() { return this.base_blacklist.concat(GM_getValue('exclude_list', [])) .filter((v, i, a) => a.indexOf(v) === i); } }; // 要处理的 event 列表 var hook_eventNames, unhook_eventNames, eventNames; // 全局状态(供弹窗展示) var g_currentMode = 'standard'; var g_rule = null; // 储存名称 var storageName = getRandStr('qwertyuiopasdfghjklzxcvbnmQWERTYUIOPASDFGHJKLZXCVBNM', parseInt(Math.random() * 12 + 8)); // 储存被 Hook 的函数 var EventTarget_addEventListener = EventTarget.prototype.addEventListener; var document_addEventListener = document.addEventListener; var Event_preventDefault = Event.prototype.preventDefault; var originalReturnValueDescriptor = Object.getOwnPropertyDescriptor(Event.prototype, 'returnValue'); // Hook addEventListener proc function addEventListener(type, func, useCapture) { var _addEventListener = this === document ? document_addEventListener : EventTarget_addEventListener; if(hook_eventNames.indexOf(type) >= 0) { _addEventListener.apply(this, [type, returnTrue, useCapture]); } else if(this && unhook_eventNames.indexOf(type) >= 0) { var funcsName = storageName + type + (useCapture ? 't' : 'f'); if(this[funcsName] === undefined) { this[funcsName] = []; _addEventListener.apply(this, [type, useCapture ? unhook_t : unhook_f, useCapture]); } this[funcsName].push(func); } else { _addEventListener.apply(this, arguments); } } // 清理循环 function clearLoop() { var elements = getElements(); for(var i in elements) { for(var j in eventNames) { var name = 'on' + eventNames[j]; if(elements[i][name] !== null && elements[i][name] !== onxxx) { if(unhook_eventNames.indexOf(eventNames[j]) >= 0) { elements[i][storageName + name] = elements[i][name]; elements[i][name] = onxxx; } else { elements[i][name] = null; } } } } } // 使用 MutationObserver 增量清理 DOM0 事件 function cleanseElement(root) { if(!root) return; var nodes = [root]; if(root.querySelectorAll) { nodes = nodes.concat(Array.prototype.slice.call(root.querySelectorAll('*'))); } for(var i = 0; i < nodes.length; i++) { var el = nodes[i]; for(var j in eventNames) { var evt = eventNames[j]; var name = 'on' + evt; if(el[name] !== null && el[name] !== onxxx) { if(unhook_eventNames.indexOf(evt) >= 0) { el[storageName + name] = el[name]; el[name] = onxxx; } else { el[name] = null; } } } } } function setupDom0Cleaner() { cleanseElement(document); try { var observer = new MutationObserver(function(mutations) { for(var i = 0; i < mutations.length; i++) { var m = mutations[i]; for(var k = 0; k < m.addedNodes.length; k++) { var node = m.addedNodes[k]; if(node && node.nodeType === 1) { cleanseElement(node); } } } }); observer.observe(document.documentElement, { childList: true, subtree: true }); window.addEventListener('load', function() { cleanseElement(document); }, true); } catch(e) { // 降级到原循环 setInterval(clearLoop, 30 * 1000); setTimeout(clearLoop, 2500); window.addEventListener('load', clearLoop, true); clearLoop(); } } // 轻量模式:仅在捕获阶段阻断常见拦截事件的传播 function enableLightModeCapture() { var captureTypes = ['contextmenu', 'copy', 'cut', 'selectstart', 'dragstart']; function stopAll(e) { e.stopImmediatePropagation(); e.stopPropagation(); } for(var i = 0; i < captureTypes.length; i++) { var t = captureTypes[i]; window.addEventListener(t, stopAll, true); document.addEventListener(t, stopAll, true); } } // 返回true的函数 function returnTrue(e) { return true; } function unhook_t(e) { return unhook(e, this, storageName + e.type + 't'); } function unhook_f(e) { return unhook(e, this, storageName + e.type + 'f'); } function unhook(e, self, funcsName) { var list = self[funcsName]; for(var i in list) { list[i](e); } e.returnValue = true; return true; } function onxxx(e) { var name = storageName + 'on' + e.type; this[name](e); e.returnValue = true; return true; } // 获取随机字符串 function getRandStr(chs, len) { var str = ''; while(len--) { str += chs[parseInt(Math.random() * chs.length)]; } return str; } // 获取所有元素 包括document function getElements() { var elements = Array.prototype.slice.call(document.getElementsByTagName('*')); elements.push(document); return elements; } // 添加css function addStyle(css) { var style = document.createElement('style'); style.innerHTML = css; document.head.appendChild(style); } // 获取目标域名应该使用的规则 function getRule(url) { function testUrl(list, url) { for(var i in list) { if(list[i].test(url)) { return true; } } return false; } if(testUrl(lists.black_list, url)) { return rules.black_rule; } return rules.default_rule; } // 初始化(始终注册菜单) function init() { // 注册菜单项 const isExcluded = lists.exclude_list().includes(location.host); // 计算当前模式(即使被排除也计算,用于菜单显示) var host = location.hostname; var defaultMode = getDefaultModeForHost(host) || MODES.light; var currentModeForMenu = isExcluded ? MODES.disabled : (getSiteMode(host) || defaultMode); g_currentMode = currentModeForMenu; g_rule = rules.default_rule; GM_registerMenuCommand(`当前网站:${isExcluded ? '❌' : '✔️'}`, () => { const currentList = lists.exclude_list(); const newList = isExcluded ? currentList.filter(h => h !== location.host) : [...currentList, location.host]; GM_setValue('exclude_list', newList); window.location.reload(); }); // 状态显示弹窗 GM_registerMenuCommand('📜 当前状态', () => { createPopup(); }); // 菜单:显示/切换模式(打开面板),始终注册,并使用中文标签 GM_registerMenuCommand(`站点模式:${MODE_LABELS[currentModeForMenu] || currentModeForMenu}`, () => { createPopup(); }); // 如果当前网站在排除列表中则不执行后续逻辑 if (isExcluded) { g_currentMode = MODES.disabled; g_rule = rules.default_rule; return; } // 获取当前域名的规则 var url = window.location.host + window.location.pathname; var rule = getRule(url); // host/defaultMode 已在上面计算 var currentMode = getSiteMode(host) || defaultMode; g_currentMode = currentMode; g_rule = rule; // 按模式调整策略 if (currentMode === MODES.disabled) { return; // 完全不影响页面 } else if (currentMode === MODES.friendly) { rule.unhook_eventNames = ""; rule.hook_addEventListener = false; rule.hook_preventDefault = false; rule.hook_set_returnValue = false; rule.dom0 = false; rule.hook_eventNames = ""; // 仅保留 CSS 放行 } else if (currentMode === MODES.light) { rule.unhook_eventNames = ""; rule.hook_addEventListener = false; rule.hook_preventDefault = false; rule.hook_set_returnValue = false; rule.dom0 = false; enableLightModeCapture(); } // 设置 event 列表 hook_eventNames = rule.hook_eventNames.split("|").filter(Boolean); // TODO Allowed to return value unhook_eventNames = rule.unhook_eventNames.split("|").filter(Boolean); eventNames = hook_eventNames.concat(unhook_eventNames); // 调用清理 DOM0 event 方法的循环 if(rule.dom0) { setupDom0Cleaner(); } // hook addEventListener if(rule.hook_addEventListener) { EventTarget.prototype.addEventListener = addEventListener; document.addEventListener = addEventListener; } // hook preventDefault if(rule.hook_preventDefault) { Event.prototype.preventDefault = function() { if(eventNames.indexOf(this.type) < 0) { Event_preventDefault.apply(this, arguments); } }; } // Hook set returnValue if(rule.hook_set_returnValue) { try { Object.defineProperty(Event.prototype, 'returnValue', { configurable: true, enumerable: false, set: function(v) { if(eventNames.indexOf(this.type) >= 0 && v !== true) { return; // 忽略将其设为 false 的尝试 } if(originalReturnValueDescriptor && originalReturnValueDescriptor.set) { originalReturnValueDescriptor.set.call(this, v); } }, get: function() { if(originalReturnValueDescriptor && originalReturnValueDescriptor.get) { return originalReturnValueDescriptor.get.call(this); } return true; } }); } catch(e) { // 退回旧方案 Event.prototype.__defineSetter__('returnValue', function() { if(this.returnValue !== true && eventNames.indexOf(this.type) >= 0) { this.returnValue = true; } }); } } console.debug('url: ' + url, 'storageName:' + storageName, 'rule: ' + rule.name); // 添加CSS if(rule.add_css) { addStyle(` html, * { -webkit-user-select:text!important; -moz-user-select:text!important; user-select:text!important; -ms-user-select:text!important; -khtml-user-select:text!important; } /* 悬浮窗样式 - 紧凑版 + 赞赏码居中 */ #rml-popup { position: fixed; top: 50%; left: 50%; transform: translate(-50%, -50%); width: 500px; /* 缩紧宽度,减少留白 */ min-height: 400px; /* 降低最小高度,更紧凑 */ padding: 0; background-color: #000; border: 1px solid #0d1117; border-radius: 10px; box-shadow: 0 4px 8px rgba(0, 0, 0, 0.1), 0 8px 24px rgba(0, 0, 0, 0.3); z-index: 9999; font-family: system-ui, -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Oxygen, Ubuntu, Cantarell, "Open Sans", "Helvetica Neue", sans-serif; user-select: none; -webkit-user-select: none; -moz-user-select: none; -ms-user-select: none; backdrop-filter: blur(10px); touch-action: none; overflow: hidden; color: #e6e6ef; transition: box-shadow 0.2s, height 0.3s ease; } #rml-popup:hover { box-shadow: 0 6px 12px rgba(0, 0, 0, 0.15), 0 12px 32px rgba(0, 0, 0, 0.4); } #rml-popup .mac-header { cursor: grab; } #rml-popup .mac-header:active { cursor: grabbing; } #rml-popup * { user-select: none !important; -webkit-user-select: none !important; -moz-user-select: none !important; -ms-user-select: none !important; } #rml-popup:active { cursor: grabbing; } .mac-header { display: flex; align-items: center; gap: 8px; padding: 10px 16px; /* 缩紧头部内边距 */ border-bottom: 1px solid #0d1117; background: linear-gradient(180deg, #1a1a1a 0%, #000 100%); min-height: 44px; /* 降低头部高度 */ box-sizing: border-box; } .window-controls { position: absolute; top: 12px; left: 16px; display: flex; gap: 6px; /* 缩紧按钮间距 */ z-index: 10; } .control-btn { width: 12px; height: 12px; border-radius: 50%; border: none; cursor: pointer; display: flex; align-items: center; justify-content: center; font-size: 10px; color: transparent; transition: transform 0.2s, opacity 0.2s; padding: 0; line-height: 1; } .close-btn { background-color: #ff5f57; } .min-btn { background-color: #ffbd2e; } .max-btn { background-color: #28c941; } .control-btn:hover { transform: scale(1.15); opacity: 0.9; } #rml-popup .content-wrapper { padding: 12px 16px; /* 大幅缩紧内边距,减少留白 */ max-height: calc(100vh - 70px); overflow-y: auto; } #rml-popup .content-wrapper::-webkit-scrollbar { width: 6px; /* 缩紧滚动条宽度 */ } #rml-popup .content-wrapper::-webkit-scrollbar-thumb { background: #555; border-radius: 3px; } #rml-popup .content-wrapper::-webkit-scrollbar-track { background: #0d1117; } .card-title { font-size: 14px; /* 缩小标题字体 */ font-weight: bold; margin: 0 0 6px; /* 缩紧标题下边距 */ color: #e6e6ef; } .card-description { font-size: 12px; /* 缩小描述字体 */ color: #999; margin-bottom: 8px; /* 缩紧描述下边距 */ line-height: 1.3; /* 缩小行高 */ } .card-tag { display: inline-block; font-size: 10px; border-radius: 4px; background-color: #0d1117; padding: 2px 6px; /* 缩紧标签内边距 */ margin-bottom: 6px; /* 缩紧标签下边距 */ color: #dcdcdc; border: 1px solid #333; } .code-editor { background-color: #0d1117; color: #dcdcdc; font-family: system-ui, -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Oxygen, Ubuntu, Cantarell, "Open Sans", "Helvetica Neue", monospace; font-size: 11px; /* 缩小编辑器字体 */ line-height: 1.4; /* 缩小行高 */ border-radius: 4px; padding: 8px; /* 缩紧编辑器内边距 */ overflow: auto; border: 1px solid #333; margin-bottom: 8px; /* 缩紧编辑器下边距 */ } .collapsed { height: 44px !important; min-height: 44px !important; overflow: hidden; opacity: 0.95; } .collapsed .content-wrapper { display: none !important; } .collapsed .mac-header { border-bottom: none; padding: 10px 16px; } /* 赞赏码样式 - 141x140原尺寸 + 居中展示 */ .appreciation-area { width: 100%; text-align: center; margin-bottom: 10px; /* 缩紧赞赏码下边距 */ } .appreciation-code { width: 141px; height: 140px; background-image: url(""); background-size: contain; background-repeat: no-repeat; background-position: center; border: 1px solid #333; border-radius: 4px; margin: 0 auto 3px; /* 水平居中,缩紧与文字间距 */ display: block; } .appreciation-tip { font-size: 10px; color: #999; white-space: nowrap; } /* 状态面板 - 紧凑布局 */ .status-panel { flex: 1; min-height: 320px; /* 紧凑最小高度 */ } /* 设置面板 - 紧凑布局 */ .settings-panel { flex: 1; border-left: 1px solid #0d1117; padding-left: 16px; /* 缩紧左侧内边距 */ min-height: 320px; /* 紧凑最小高度 */ } `); } // 创建控制弹窗 function createPopup() { // 创建容器并添加控制按钮 const container = document.createElement('div'); container.id = 'rml-popup'; const modeLabelMap = { standard: '标准模式', light: '轻量模式', friendly: '友好模式', disabled: '禁用' }; const modeLabel = modeLabelMap[g_currentMode] || g_currentMode || '标准模式'; const captureList = ['contextmenu','copy','cut','selectstart','dragstart']; const EVENT_LABELS = { contextmenu: '右键菜单', copy: '复制', cut: '剪切', select: '文本选择', selectstart: '选择开始', dragstart: '拖拽开始', keydown: '按键按下', keyup: '按键抬起', mousedown: '鼠标按下', mouseup: '鼠标抬起' }; function labelEvent(name){ return EVENT_LABELS[name] || name; } function mapEvents(list){ return (list && list.length) ? list.map(labelEvent).join(',') : '无'; } let processedEventsText = ''; if (g_currentMode === MODES.light) { processedEventsText = mapEvents(captureList); } else if (g_currentMode === MODES.friendly) { processedEventsText = '无(仅 CSS 放行)'; } else if (g_currentMode === MODES.disabled) { processedEventsText = '无(已禁用)'; } else { processedEventsText = mapEvents(eventNames || []); } let featureList = []; if (g_currentMode === MODES.disabled) { featureList = ['已禁用']; } else if (g_currentMode === MODES.friendly) { if (g_rule && g_rule.add_css) featureList.push('CSS 放行选择'); featureList.push('最小侵入(不 Hook 事件)'); } else if (g_currentMode === MODES.light) { if (g_rule && g_rule.add_css) featureList.push('CSS 放行选择'); featureList.push('捕获期阻断常见拦截事件'); } else { // 标准模式 if (g_rule && g_rule.add_css) featureList.push('CSS 放行选择'); if (g_rule && g_rule.hook_addEventListener) featureList.push('Hook addEventListener'); if (g_rule && g_rule.hook_preventDefault) featureList.push('过滤 preventDefault'); if (g_rule && g_rule.hook_set_returnValue) featureList.push('保护 returnValue'); if (g_rule && g_rule.dom0) featureList.push('清理内联 on* 事件'); } const featuresText = featureList.join('、'); container.innerHTML = `
感谢赞赏
当前状态
运行中
📌 当前模式: ${modeLabel}
✅ 已处理事件:
${processedEventsText}
🧩 实现的功能
${featuresText}
站点设置
当前域名模式:
排除设置
已排除域名(每行一个):
`; // 添加容器到文档中 document.body.appendChild(container); // 初始化站点模式下拉框 (function initModeSelect(){ var modeSelect = container.querySelector('#siteModeSelect'); if(modeSelect) { try { var host = location.host; var currentMode = getSiteMode(host) || getDefaultModeForHost(host) || MODES.light; modeSelect.value = currentMode; } catch(e) {} } })(); // 保存站点模式 var saveModeBtn = container.querySelector('#saveModeBtn'); if (saveModeBtn) { saveModeBtn.addEventListener('click', function() { var modeSelect = container.querySelector('#siteModeSelect'); var val = modeSelect ? modeSelect.value : 'standard'; setSiteMode(location.host, val); window.location.reload(); }); } // 添加保存事件监听 document.getElementById('saveExcludeList').addEventListener('click', () => { const domains = document.getElementById('excludeDomains').value .split('\n') .map(d => d.trim()) .filter(d => d.length > 0); // 分离基础黑名单和用户自定义列表 const baseEntries = domains.filter(d => lists.base_blacklist.includes(d)); const customEntries = domains.filter(d => !lists.base_blacklist.includes(d)); GM_setValue('base_blacklist', baseEntries); GM_setValue('exclude_list', customEntries); window.location.reload(); }); // 窗口控制功能 let isMaximized = false; let originalSize = { width: '500px', height: 'auto' }; // 同步紧凑宽度 container.querySelector('.min-btn').addEventListener('click', (e) => { e.stopPropagation(); container.classList.toggle('collapsed'); }); container.querySelector('.max-btn').addEventListener('click', (e) => { e.stopPropagation(); if (!isMaximized) { originalSize = { width: container.style.width || '500px', height: container.style.height || 'auto' }; container.style.width = '95vw'; container.style.height = '95vh'; isMaximized = true; } else { container.style.width = originalSize.width; container.style.height = originalSize.height; isMaximized = false; } }); container.querySelector('.close-btn').addEventListener('click', (e) => { e.stopPropagation(); container.remove(); }); // 优化后的拖拽功能 let isDragging = false; let startX, startY, initialX, initialY; let hasBeenDragged = false; const handleMouseDown = (e) => { // 排除按钮、输入框、文本区域、赞赏码等元素 if (e.target.closest('button') || e.target.closest('select') || e.target.closest('textarea') || e.target.closest('input') || e.target.closest('.window-controls') || e.target.closest('.appreciation-area')) { return; } // 只允许通过mac-header区域拖拽 if (e.target.closest('.mac-header')) { isDragging = true; startX = e.clientX; startY = e.clientY; const rect = container.getBoundingClientRect(); initialX = rect.left; initialY = rect.top; container.style.transition = 'none'; // 如果还没有拖拽过,移除transform居中 if (!hasBeenDragged) { container.style.transform = 'none'; container.style.left = `${initialX}px`; container.style.top = `${initialY}px`; } } }; const handleMouseMove = (e) => { if (!isDragging) return; const dx = e.clientX - startX; const dy = e.clientY - startY; let newX = initialX + dx; let newY = initialY + dy; // 边界检测,保持在视口内 const maxX = window.innerWidth - container.offsetWidth; const maxY = window.innerHeight - container.offsetHeight; newX = Math.max(0, Math.min(newX, maxX)); newY = Math.max(0, Math.min(newY, maxY)); container.style.left = `${newX}px`; container.style.top = `${newY}px`; hasBeenDragged = true; }; const handleMouseUp = () => { if (!isDragging) return; isDragging = false; container.style.transition = ''; }; container.querySelector('.mac-header').addEventListener('mousedown', handleMouseDown); document.addEventListener('mousemove', handleMouseMove); document.addEventListener('mouseup', handleMouseUp); document.addEventListener('mouseleave', handleMouseUp); } } // 确保在DOM加载完成后初始化 if (document.readyState === 'loading') { document.addEventListener('DOMContentLoaded', init); } else { init(); } })();