解除网页对复制、切屏检测、全屏等限制
// ==UserScript==
// @name 全能网页限制解除工具(Sky & 丁大大)
// @namespace https://greasyfork.org/zh-CN/users/240922-dingzz3
// @version 2.0.2
// @description 解除网页对复制、切屏检测、全屏等限制
// @author Sky & 丁大大
// @run-at document-start
// @match *://*/*
// @icon 
// @exclude *://*.youtube.com/*
// @exclude *://*.wikipedia.org/*
// @exclude *://mail.qq.com/*
// @exclude *://translate.google.*
// @exclude *://*.bing.*
// @grant none
// @license MIT
// ==/UserScript==
(function() {
'use strict';
const key = encodeURIComponent('全能网页限制解除工具:执行判断');
if (window[key]) return;
window[key] = true;
//My PN:18643100778
// ===== 复制限制解除部分 =====
const default_rule = {
name: "default",
hook_eventNames: "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
};
let hook_eventNames, unhook_eventNames, eventNames;
const storageName = getRandStr(
'qwertyuiopasdfghjklzxcvbnmQWERTYUIOPASDFGHJKLZXCVBNM',
parseInt(Math.random() * 12 + 8)
);
const EventTarget_addEventListener = EventTarget.prototype.addEventListener;
const document_addEventListener = document.addEventListener;
const Event_preventDefault = Event.prototype.preventDefault;
function addEventListener(type, func, useCapture) {
const _addEventListener = this === document
? document_addEventListener
: EventTarget_addEventListener;
if (hook_eventNames.includes(type)) {
_addEventListener.call(this, type, returnTrue, useCapture);
} else if (unhook_eventNames.includes(type)) {
const funcsName = storageName + type + (useCapture ? 't' : 'f');
if (!this[funcsName]) {
this[funcsName] = [];
_addEventListener.call(
this,
type,
useCapture ? unhook_t : unhook_f,
useCapture
);
}
this[funcsName].push(func);
} else {
_addEventListener.apply(this, arguments);
}
}
function clearLoop() {
const elements = getElements();
elements.forEach(element => {
eventNames.forEach(eventName => {
const prop = 'on' + eventName;
if (element[prop] && element[prop] !== onxxx) {
if (unhook_eventNames.includes(eventName)) {
element[storageName + prop] = element[prop];
element[prop] = onxxx;
} else {
element[prop] = null;
}
}
});
});
}
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) {
(self[funcsName] || []).forEach(fn => fn(e));
e.returnValue = true;
return true;
}
function onxxx(e) {
this[storageName + 'on' + e.type](e);
e.returnValue = true;
return true;
}
function getRandStr(chars, len) {
return Array.from({length: len}, () =>
chars[Math.floor(Math.random() * chars.length)]
).join('');
}
function getElements() {
return Array.from(document.getElementsByTagName('*')).concat(document);
}
function addStyle(css) {
const style = document.createElement('style');
style.textContent = css;
document.head.appendChild(style);
}
// ===== 切屏检测解除部分 =====
function disableBlurDetection() {
window.addEventListener('blur', function(e) {
e.stopImmediatePropagation();
}, true);
Object.defineProperty(document, 'hidden', {
get: function() { return false; }
});
Object.defineProperty(document, 'visibilityState', {
get: function() { return 'visible'; }
});
// 覆盖可能存在的检测函数
if (window.addEventListener) {
const originalAddEventListener = window.addEventListener;
window.addEventListener = function(type, listener, options) {
if (type === 'visibilitychange' || type === 'blur' || type === 'focus') {
return;
}
originalAddEventListener.call(window, type, listener, options);
};
}
}
// ===== 全屏限制解除部分 =====
function enableFullscreen() {
// 覆盖全屏API检查
if (Element.prototype.requestFullscreen) {
const originalRequestFullscreen = Element.prototype.requestFullscreen;
Element.prototype.requestFullscreen = function() {
return originalRequestFullscreen.call(this);
};
}
// 覆盖webkit全屏API
if (Element.prototype.webkitRequestFullscreen) {
const originalWebkitRequestFullscreen = Element.prototype.webkitRequestFullscreen;
Element.prototype.webkitRequestFullscreen = function() {
return originalWebkitRequestFullscreen.call(this);
};
}
// 移除全屏变化事件监听
document.addEventListener('fullscreenchange', function(e) {
e.stopImmediatePropagation();
}, true);
document.addEventListener('webkitfullscreenchange', function(e) {
e.stopImmediatePropagation();
}, true);
}
// ===== 主初始化函数 =====
function init() {
const rule = default_rule;
hook_eventNames = rule.hook_eventNames.split('|');
unhook_eventNames = rule.unhook_eventNames.split('|');
eventNames = [...hook_eventNames, ...unhook_eventNames];
// 复制限制解除初始化
if (rule.dom0) {
setInterval(clearLoop, 30000);
setTimeout(clearLoop, 2500);
window.addEventListener('load', clearLoop, true);
clearLoop();
}
if (rule.hook_addEventListener) {
EventTarget.prototype.addEventListener = addEventListener;
document.addEventListener = addEventListener;
}
if (rule.hook_preventDefault) {
Event.prototype.preventDefault = function() {
if (!eventNames.includes(this.type)) {
Event_preventDefault.apply(this, arguments);
}
};
}
if (rule.hook_set_returnValue) {
Object.defineProperty(Event.prototype, 'returnValue', {
set: function(value) {
if (value !== true && eventNames.includes(this.type)) {
return;
}
this._returnValue = value;
},
get: function() {
return this._returnValue;
}
});
}
if (rule.add_css) {
addStyle('html,*{user-select:text!important;-webkit-user-select:text!important;-moz-user-select:text!important;-ms-user-select:text!important;}');
}
// 切屏检测解除
disableBlurDetection();
// 全屏限制解除
enableFullscreen();
}
// 使用MutationObserver持续监测和解除限制
const observer = new MutationObserver(function(mutations) {
disableBlurDetection();
enableFullscreen();
});
// 初始化
init();
// 页面加载完成后再次确保所有限制被解除
window.addEventListener('load', function() {
disableBlurDetection();
enableFullscreen();
// 开始观察DOM变化
observer.observe(document, {
childList: true,
subtree: true,
attributes: true,
characterData: true
});
});
// 立即执行一次
disableBlurDetection();
enableFullscreen();
})();