// ==UserScript==
// @name 解除网页右键限制
// @author [email protected]
// @description 解除网页禁止复制、剪切、文本选择、右键菜单的限制,不一定支持所有网页,但如果你有能解决部分无法处理的网页方案可以发送邮件到:[email protected]
// @version 1.0.0.20250403
// @icon 
// @match *://*/*
// @grant GM_getValue
// @grant GM_setValue
// @grant GM_deleteValue
// @grant GM_registerMenuCommand
// @grant GM_addStyle
// @run-at document-start
// @namespace https://greasyfork.org/users/1453515
// @license MIT
// ==/UserScript==
(function() {
'use strict';
// 域名规则列表
var rules = {
default_rule: {
name: "default",
hook_eventNames: "contextmenu|select|selectstart|copy|cut|dragstart",
unhook_eventNames: "mousedown|mouseup|keydown|keyup",
dom0: true,
hook_addEventListener: true,
hook_preventDefault: true,
hook_set_returnValue: true,
add_css: true
}
};
// 要处理的 event 列表
var hook_eventNames, unhook_eventNames, eventNames;
// 储存名称
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 rwl_userData = null;
var hostname = window.location.hostname;
var btn_node = null;
var rule = null;
var list = [];
var hasFrame = false;
// 初始化设置
var settingData = {
"version": 0.1,
"positionTop": "0",
"positionLeft": "0",
"positionRight": "auto",
"addBtn": true,
"data": [
"example.com",
],
"enabledSites": [] // 存储已启用的网站
};
// 加载用户设置
rwl_userData = GM_getValue("rwl_userData");
if (!rwl_userData) {
rwl_userData = JSON.parse(JSON.stringify(settingData));
GM_setValue("rwl_userData", rwl_userData);
} else {
for (let key in settingData) {
if (!rwl_userData.hasOwnProperty(key)) {
rwl_userData[key] = settingData[key];
}
}
GM_setValue("rwl_userData", rwl_userData);
}
// 检查当前域名是否在黑名单中
function isInBlacklist() {
if (!rwl_userData || !rwl_userData.data) return false;
return rwl_userData.data.some(domain => {
// 支持通配符匹配,如 *.example.com
if (domain.startsWith('*.')) {
const baseDomain = domain.substring(2);
return hostname === baseDomain || hostname.endsWith('.' + baseDomain);
}
return hostname === domain;
});
}
// 检查当前域名是否在已启用列表中
function isSiteEnabled() {
if (!rwl_userData || !rwl_userData.enabledSites) return false;
return rwl_userData.enabledSites.some(domain => {
// 支持通配符匹配,如 *.example.com
if (domain.startsWith('*.')) {
const baseDomain = domain.substring(2);
return hostname === baseDomain || hostname.endsWith('.' + baseDomain);
}
return hostname === domain;
});
}
// 添加悬浮框
addBtn();
btn_node = document.getElementById("black_node");
var timer = setInterval(function() {
if (document.getElementById("black_node")) {
clearInterval(timer);
rwlStart();
} else {
addBtn();
}
}, 500);
// 注册菜单命令
GM_registerMenuCommand("复制限制解除 设置", setMenu);
function rwlStart() {
console.log("脚本: 复制限制解除 --- 开始执行");
// 初始化时根据当前网站状态设置复选框
if (btn_node) {
btn_node.checked = isSiteEnabled();
}
// 检查黑名单 - 不再阻止设置功能
if (isInBlacklist()) {
console.log("当前域名在黑名单中,脚本不生效");
if (btn_node) btn_node.checked = false;
} else if (isSiteEnabled()) {
// 如果在已启用列表中,自动勾选并初始化
init();
}
addDragEven();
setBtnClick();
}
function addBtn() {
var node = document.createElement("remove-web-limits");
node.id = "rwl-float";
node.className = "rwl-exempt";
var screenClientHeight = document.documentElement.clientHeight;
var tempHeight;
if (rwl_userData.positionTop > screenClientHeight) {
tempHeight = screenClientHeight - 40;
} else {
tempHeight = rwl_userData.positionTop;
}
window.onresize = function() {
var screenClientHeight = document.documentElement.clientHeight;
var tempHeight;
if (rwl_userData.positionTop > screenClientHeight) {
tempHeight = screenClientHeight - 40;
} else {
tempHeight = rwl_userData.positionTop;
}
node.style.top = tempHeight + "px";
};
tempHeight = tempHeight < 0 ? 0 : tempHeight;
node.style.cssText = "position:fixed;top:" + tempHeight + "px;left:" + rwl_userData.positionLeft + "px;right:" + rwl_userData.positionRight + "px;";
node.innerHTML = '<button type="button" id="rwl-setbtn">⚙️</button> <span style="cursor:move; font-size:12px;">限制解除</span> <input type="checkbox" name="" id="black_node" >';
if (window.self === window.top) {
if (document.querySelector("body")) {
document.body.appendChild(node);
} else {
document.documentElement.appendChild(node);
}
}
node.addEventListener("mouseover", function() {
node.classList.add("rwl-active");
});
node.addEventListener("mouseleave", function() {
node.classList.remove("rwl-active");
});
var checkbox = node.querySelector("#black_node");
checkbox.checked = isSiteEnabled();
checkbox.addEventListener("change", function() {
toggleScript(this.checked);
});
var style = document.createElement("style");
style.type = "text/css";
var styleInner = `
#rwl-float {
position: fixed;
transform: translate(-90%, 0);
width: none;
height: 36px;
font-size: 14px;
font-weight: 500;
font-family: 'Segoe UI', Roboto, Oxygen, Ubuntu, Cantarell, 'Open Sans', 'Helvetica Neue', sans-serif;
color: #333;
background: rgba(255, 255, 255, 0.5);
backdrop-filter: blur(12px) saturate(180%);
-webkit-backdrop-filter: blur(12px) saturate(180%);
z-index: 2147483647;
margin: 0;
opacity: 0.9;
transition: all 0.3s cubic-bezier(0.25, 0.8, 0.25, 1);
overflow: hidden;
user-select: none;
text-align: center;
white-space: nowrap;
line-height: 36px;
padding: 0 16px;
border: 1px solid rgba(255, 255, 255, 0.2);
border-radius: 0 18px 18px 0;
box-shadow: 0 4px 30px rgba(0, 0, 0, 0.1);
box-sizing: border-box;
display: flex;
align-items: center;
gap: 8px;
}
#rwl-float input[type='checkbox'] {
margin: 0;
padding: 0;
width: 16px;
height: 16px;
vertical-align: middle;
position: relative;
cursor: pointer;
-webkit-appearance: none;
-moz-appearance: none;
appearance: none;
border: 1px solid #ccc;
border-radius: 4px;
background: #fff;
transition: all 0.2s ease;
}
#rwl-float input[type='checkbox']:checked {
background: #4285f4;
border-color: #4285f4;
}
#rwl-float input[type='checkbox']:checked::after {
content: '';
position: absolute;
top: 2px;
left: 5px;
width: 4px;
height: 8px;
border: solid #fff;
border-width: 0 2px 2px 0;
transform: rotate(45deg);
}
#rwl-float.rwl-active {
transform: translate(0, 0);
opacity: 1;
background: rgba(255, 255, 255, 0.6);
backdrop-filter: blur(16px) saturate(200%);
-webkit-backdrop-filter: blur(16px) saturate(200%);
box-shadow: 0 8px 32px rgba(0, 0, 0, 0.15);
}
#rwl-float label {
margin: 0;
padding: 0;
font-weight: 500;
}
#rwl-float #rwl-setbtn {
margin: 0;
padding: 0;
width: 20px;
height: 20px;
border: none;
border-radius: 50%;
cursor: pointer;
background: transparent;
color: #5f6368;
transition: all 0.2s ease;
display: flex;
align-items: center;
justify-content: center;
font-size: 16px;
}
#rwl-float #rwl-setbtn:hover {
background: rgba(66, 133, 244, 0.1);
color: #4285f4;
}
#rwl-float span {
cursor: move;
font-weight: 500;
color: #5f6368;
}
#rwl-float:hover span {
color: #202124;
}
`;
if (!rwl_userData.addBtn) {
var styleTemp = "#rwl-float{display:none}";
style.innerHTML = styleInner + styleTemp;
} else {
style.innerHTML = styleInner;
}
if (document.querySelector("#rwl-float")) {
document.querySelector("#rwl-float").appendChild(style);
} else {
GM_addStyle(styleInner);
}
}
function setBtnClick() {
document.querySelector("#rwl-setbtn").addEventListener("click", setMenu);
}
function setMenu() {
var oldEditBox = document.querySelector("#rwl-setMenu");
if (oldEditBox) {
oldEditBox.parentNode.removeChild(oldEditBox);
return;
}
var userSetting = GM_getValue("rwl_userData");
var btnchecked = userSetting.addBtn ? 'checked' : '';
var odom = document.createElement("div");
odom.id = "rwl-setMenu";
odom.style.cssText = "position: fixed;" +
"top: 10px;" +
"left: 10px;" +
"padding: 20px;" +
"background: rgba(255,255,255,0.8);" +
"backdrop-filter: blur(10px);" +
"border-radius: 8px;" +
"box-shadow: 0 4px 12px rgba(0,0,0,0.15);" +
"z-index: 999999;" +
"border: 1px solid rgba(0,0,0,0.1);";
GM_addStyle(`
#rwl-setMenuSave,
#rwl-reset,
#rwl-setMenuClose {
margin: 0 5px;
padding: 5px 10px;
border: none;
border-radius: 4px;
cursor: pointer;
background: rgba(255,255,255,0.8);
color: #000;
transition:0.2s;
}
#rwl-reset {
border: 1px solid #666;
}
#rwl-reset:hover {
background: rgba(0,0,0,0.2);
}
#rwl-setMenuSave {
border: 1px solid green;
background: rgba(76,175,80,0.8);
color: white;
}
#rwl-setMenuSave:hover {
background: rgba(76,175,80,1);
}
#rwl-setMenuClose {
border: 1px solid red;
background: rgba(244,67,54,0.8);
color: white;
}
#rwl-setMenuClose:hover {
background: rgba(244,67,54,1);
}
#rwl-setMenu {
text-align:left;
font-size:14px;
z-index:999999;
top: 10px !important;
left: 10px !important;
margin: 10px;
}
#rwl-setMenu p {
margin:10px auto;
}
#rwl-setMenu input[type=text],
#rwl-setMenu textarea,
#rwl-setMenu select {
padding:5px;
border-radius:4px;
border:1px solid #ccc;
background:rgba(255,255,255,0.8);
}
#rwl-setMenu textarea {
width:100%;
min-height:200px;
margin-top:10px;
}
.rwl-switch {
position: relative;
display: inline-block;
width: 50px;
height: 24px;
vertical-align: middle;
margin-left: 10px;
}
.rwl-switch input {
opacity: 0;
width: 0;
height: 0;
}
.rwl-slider {
position: absolute;
cursor: pointer;
top: 0;
left: 0;
right: 0;
bottom: 0;
background-color: #ccc;
transition: .4s;
border-radius: 24px;
}
.rwl-slider:before {
position: absolute;
content: '';
height: 16px;
width: 16px;
left: 4px;
bottom: 4px;
background-color: white;
transition: .4s;
border-radius: 50%;
}
input:checked + .rwl-slider {
background-color: #4285f4;
}
input:checked + .rwl-slider:before {
transform: translateX(26px);
}
`);
var innerH = "" +
"<p>悬浮框垂直位置 <input id='positiontop' type='text' value=" + userSetting.positionTop + " style='width: 25px;'> 像素" +
"<label style='display:inline-flex;align-items:center;justify-content:flex-end;width:180px;'>悬浮框开关" +
"<label class='rwl-switch'>" +
"<input id='btnchecked' type='checkbox' " + btnchecked + ">" +
"<span class='rwl-slider'></span>" +
"</label>" +
"</label></p>" +
"<p>黑名单列表(每行一个域名,支持通配符如 *.example.com)</p>" +
"<textarea wrap='off' cols='45' rows='10' style='overflow:auto;border-radius:4px;'>" + userSetting.data.join('\n') + "</textarea>" +
"<p>已启用网站列表(自动记录)</p>" +
"<textarea wrap='off' cols='45' rows='5' style='overflow:auto;border-radius:4px;' readonly>" + userSetting.enabledSites.join('\n') + "</textarea>" +
"<br>" +
"<div style='text-align: right; margin-top: 10px;'>" +
"<button id='rwl-reset'>清空设置</button>" +
"<button id='rwl-setMenuSave'>保存</button>" +
"<button id='rwl-setMenuClose' title='若无法关闭 请刷新界面' >关闭</button>" +
"</div>" +
"";
odom.innerHTML = innerH;
document.body.appendChild(odom);
document.querySelector("#rwl-setMenuSave").addEventListener("click", saveSetting);
document.querySelector("#rwl-setMenuClose").addEventListener("click", closeMenu);
document.querySelector("#rwl-reset").addEventListener("click", rwlReset);
}
function saveSetting() {
var positionTop = document.querySelector("#rwl-setMenu #positiontop").value;
var addBtnChecked = document.querySelector("#rwl-setMenu #btnchecked").checked;
var codevalue = document.querySelector("#rwl-setMenu textarea").value;
if (codevalue) {
var userSetting = GM_getValue("rwl_userData");
userSetting.addBtn = addBtnChecked;
userSetting.data = codevalue.split('\n').filter(item => item.trim() !== '');
userSetting.positionTop = parseInt(positionTop);
GM_setValue("rwl_userData", userSetting);
setTimeout(function() {
window.location.reload();
}, 300);
} else {
alert("输入为空");
}
closeMenu();
}
function rwlReset() {
GM_deleteValue("rwl_userData");
window.location.reload();
}
function closeMenu() {
var oldEditBox = document.querySelector("#rwl-setMenu");
if (oldEditBox) {
oldEditBox.parentNode.removeChild(oldEditBox);
return;
}
}
function addDragEven() {
setTimeout(function() {
try {
dragBtn();
} catch (e) {
console.error("dragBtn函数 报错");
}
}, 1000);
}
function dragBtn() {
var rwl_node = document.querySelector("#rwl-float");
rwl_node.addEventListener("mousedown", function(event) {
rwl_node.style.transition = "null";
var disX = event.clientX - rwl_node.offsetLeft;
var disY = event.clientY - rwl_node.offsetTop;
var move = function(event) {
rwl_node.style.left = event.clientX - disX + "px";
rwl_node.style.top = event.clientY - disY + "px";
};
document.addEventListener("mousemove", move);
document.addEventListener("mouseup", function() {
rwl_node.style.transition = "0.3s";
document.removeEventListener("mousemove", move);
rwl_node.style.right = rwl_userData.positionRight = "auto";
rwl_node.style.left = rwl_userData.positionLeft = 0;
rwl_userData.positionTop = rwl_node.offsetTop;
GM_setValue("rwl_userData", rwl_userData);
});
});
}
function toggleScript(bool) {
var userData = GM_getValue("rwl_userData");
if (!userData) {
userData = JSON.parse(JSON.stringify(settingData));
GM_setValue("rwl_userData", userData);
}
// 添加或移除当前网站到启用列表
var currentSite = window.location.hostname;
if (bool) {
if (!userData.enabledSites.includes(currentSite)) {
userData.enabledSites.push(currentSite);
// 确保没有重复
userData.enabledSites = [...new Set(userData.enabledSites)];
}
} else {
userData.enabledSites = userData.enabledSites.filter(site => site !== currentSite);
}
GM_setValue("rwl_userData", userData);
if (bool) {
init();
} else {
setTimeout(() => location.reload(), 350);
}
}
function init() {
// 获取最新设置
rwl_userData = GM_getValue("rwl_userData");
// 检查是否应该启用
var shouldEnable = isSiteEnabled();
if (btn_node) {
btn_node.checked = shouldEnable;
}
// 如果不在黑名单且应该启用,则执行解除限制的逻辑
if (!isInBlacklist() && shouldEnable) {
rule = rules.default_rule;
hook_eventNames = rule.hook_eventNames.split("|");
unhook_eventNames = rule.unhook_eventNames.split("|");
eventNames = hook_eventNames.concat(unhook_eventNames);
if (rule.dom0) {
setInterval(clearLoop, 10 * 1000);
setTimeout(clearLoop, 1500);
window.addEventListener('load', clearLoop, true);
clearLoop();
}
if (rule.hook_addEventListener) {
EventTarget.prototype.addEventListener = addEventListener;
document.addEventListener = addEventListener;
if (hasFrame) {
for (let i = 0; i < hasFrame.length; i++) {
hasFrame[i].contentWindow.document.addEventListener = addEventListener;
}
}
}
if (rule.hook_preventDefault) {
Event.prototype.preventDefault = function() {
if (hook_eventNames.indexOf(this.type) < 0) {
Event_preventDefault.apply(this, arguments);
}
};
if (hasFrame) {
for (let i = 0; i < hasFrame.length; i++) {
hasFrame[i].contentWindow.Event.prototype.preventDefault = function() {
if (hook_eventNames.indexOf(this.type) < 0) {
Event_preventDefault.apply(this, arguments);
}
};
}
}
}
if (rule.hook_set_returnValue) {
Event.prototype.__defineSetter__('returnValue', function() {
if (this.returnValue !== true && hook_eventNames.indexOf(this.type) >= 0) {
this.returnValue = true;
}
});
}
if (rule.add_css) {
GM_addStyle('html, :not([class*="rwl-exempt"]) {-webkit-user-select:text!important; -moz-user-select:text!important;} :not([class*="rwl-exempt"]) ::selection {color:#fff; background:#3390FF!important;}');
}
}
}
// 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;
}
}
}
}
}
// 返回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);
var frames = document.querySelectorAll("frame");
if (frames) {
hasFrame = frames;
var frames_element;
for (let i = 0; i < frames.length; i++) {
frames_element = Array.prototype.slice.call(frames[i].contentWindow.document.querySelectorAll("*"));
elements.push(frames[i].contentWindow.document);
elements = elements.concat(frames_element);
}
}
return elements;
}
})();