Remove Restrictions and Restore Default Behavior

Allows you select, cut, copy, paste, save and open the DevTools on any website.

目前为 2024-05-03 提交的版本。查看 最新版本

  1. // ==UserScript==
  2. // @name Remove Restrictions and Restore Default Behavior
  3. // @name:zh-CN 解除网页限制,恢复默认行为
  4. // @name:en-US Remove Restrictions and Restore Default Behavior
  5. // @namespace http://hl-bo.github.io/namespaces/user-script/remove-limits
  6. // @version 2.0
  7. // @license AGPLv3
  8. // @description Allows you select, cut, copy, paste, save and open the DevTools on any website.
  9. // @description:zh-CN 恢复选择、剪切、复制、粘贴、保存、右键菜单和打开开发者工具的默认行为。
  10. // @description:en-US Allows you select, cut, copy, paste, save and open the DevTools on any website.
  11. // @author HL-Bo
  12. // @match *://*/**
  13. // @exclude *://vscode.dev/**
  14. // @exclude *://github.com/*/*/edit/**
  15. // @exclude *://gitee.com/*/*/edit/**
  16. // @exclude *://codeberg.org/*/*/_edit/**
  17. // @exclude *://www.figma.com/file/**
  18. // @exclude *://www.notion.so/**
  19. // @exclude *://outlook.live.com/**
  20. // @exclude *://mail.netease.com/**
  21. // @exclude *://mail.163.com/**
  22. // @exclude *://mail.126.com/**
  23. // @exclude *://www.yeah.net/**
  24. // @exclude *://mail.qq.com/**
  25. // @exclude *://uutool.cn/*
  26. // @exclude *://anytexteditor.com/*/online-notepad
  27. // @icon data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAACAAAAAgCAYAAABzenr0AAAAAXNSR0IArs4c6QAAAspJREFUWEfVl89PE1EQx7/TdpcabKFAoEKQH22ktEYSqRevXiRijEr8cSDe/DNM9D/xhPFn4oF4MeBZuJi0SiiQikTjD2gLCLS7jNmFLt2mu/uWYtB36G73zWQ+b2bevDeEijE5Od/Q3ut5CNA9gMOVc3bvG7+3MLeYRU9nhxqP9r2WURoPh8ObIvpUKTSbzkww6I6IYqWMAdAVxmCkF8z8zk/KFREIA2BmbilGu7tpZpigRGCqATQdAk3LKI46QRjGZtOZ6wx6KWKwWqYWgCbDzNN+UmwhDID3HxduEuP5UQKIQPx1gDJEnpTRoRqJaQnw/dcqVvPrANf2iST50N/dCSKCVQjMmjyVg3K1GsISILvyFT/X8pYRkXw+nD3TD4/HIwigp+ZUDkUThG0I9MVbeIAq9srW9g7SmSW0t4ZwPjHglEZvf3xZGkkmk6W93bI/qpNQUVVs7xRtPOBFgyzr88yM1PwidoolRE53oS3UbAuhMo8Mx/re2ALMZ5dRWLcuZlrsh2JReL1e3ZiWB5nsMlR118kDII/39v2xy09tAQobm8jrALVjIEsSOtpaTMZKioLVXAHaU/OK1fAAt65duvjMFsBxGXUIMGHswmDkhS2AtpK1wnodZg5UQ8EAWpqDxgchgIXPK8gdEUBzMKAnZ3kIAWgxLCmq9T4U9g1B8nn1gvV/ARx7CI49CYVDfAhBoSR0KkTidglNgUYETza6S0KtrO5VwvqHBhDt6XYH4HQYucHyN8jw7Z8Z+uElUgl1QePHjbkqWao4cvenhACcLiRukNpCTejpOuUuBE5XMmEAAlqaAmhvPTg5hTwgbOAQgv8mQD2NiVsnEPjGcDz6ynQfmPkwFyPJd6jWzA0AEZhLSjx5buCTCUD7M5temGDAdXPqCgB4MhyP3C3rmBrRVColb5H/kdv2XAyAvgH8+ARvP0gkEsZ1+w+ixcUwoQ+80AAAAABJRU5ErkJggg==
  28. // @grant none
  29. // @run-at document-start
  30. // ==/UserScript==
  31.  
  32. (function () {
  33. 'use strict';
  34. console.info('Install user-script/remove-limits');
  35.  
  36. // 尝试禁用 debugger
  37. // 仅在 eval('debugger') 或 setInterval('debugger', sec) 构造前执行才能阻止
  38. try {
  39. Function.prototype.__constructor_back = Function.prototype.constructor;
  40. Function.prototype.constructor = function () {
  41. if (arguments && typeof arguments[0] === 'string') {
  42. if ('debugger' === arguments[0]) {
  43. console.debug('Disable an function which may execute debugger');
  44. return;
  45. }
  46. }
  47. return Function.prototype.__constructor_back.apply(this, arguments);
  48. }
  49. } catch (error) { } finally { }
  50.  
  51. let setEventListener = function (element, event_name, listener) {
  52. if (element.rl_events) {
  53. if (element.rl_events.has(event_name)) {
  54. element.removeEventListener(event_name, element.rl_events.get(event_name));
  55. }
  56. } else {
  57. element.rl_events = new Map();
  58. }
  59. element.rl_events.set(event_name, listener);
  60. element.addEventListener(event_name, listener);
  61. }
  62. let returnEventAllowed = function (event, event_name) {
  63. event.returnValue = true;
  64. if (event_name !== null) {
  65. console.debug(`Allow ${event_name}`);
  66. }
  67. };
  68. let allowEvent = function (element, event_name) {
  69. setEventListener(element, event_name, function (event) { returnEventAllowed(event, event_name); });
  70. };
  71. let onKeyEvents = function (event) {
  72. let keyCode = event.keyCode || event.which || event.charCode;
  73. let ctrlKey = event.ctrlKey || event.metaKey;
  74. let shiftKey = event.shiftKey;
  75. if (ctrlKey && (keyCode == 65 || keyCode == 88 || keyCode == 67 || keyCode == 86 || keyCode == 83 || keyCode == 85)) {
  76. // Ctrl+A (select-all), Ctrl+X (cut), Ctrl+C (copy), Ctrl+V (paste), Ctrl+S (save), Ctrl+U (view-source)
  77. returnEventAllowed(event, 'hotkey');
  78. } else if (ctrlKey && shiftKey && (keyCode == 73 || keyCode == 74 || keyCode == 67)) {
  79. // Ctrl+Shift+I (devtools), Ctrl+Shift+J (console), Ctrl+Shift+C (elements)
  80. returnEventAllowed(event, 'hotkey (DevTools)');
  81. } else if (keyCode && keyCode == 123) { // F12
  82. returnEventAllowed(event, 'hotkey (F12)');
  83. }
  84. };
  85. let allowKeyEvents = function (element, event_name) {
  86. setEventListener(element, event_name, onKeyEvents);
  87. };
  88. let allowElement = function (element) {
  89. try { element.oncopy = null; } catch (error) { } finally { allowEvent(element, 'copy'); }
  90. // 取消通过 JavaScript 实现的禁止文字选择
  91. try { element.onselectstart = null; } catch (error) { } finally { allowEvent(element, 'selectstart'); }
  92. // 取消通过 JavaScript 实现的禁止右键菜单
  93. try { element.oncontextmenu = null; } catch (error) { } finally { allowEvent(element, 'contextmenu'); }
  94. // 取消通过 JavaScript 实现的禁止剪切实现的禁止复制
  95. try { element.oncut = null; } catch (error) { } finally { allowEvent(element, 'cut'); }
  96. // 取消通过 JavaScript 实现的禁止粘贴
  97. try { element.onpaste = null; } catch (error) { } finally { allowEvent(element, 'paste'); }
  98. // 取消通过 CSS 实现的禁止选中
  99. try { element.style.webkitUserSelect = 'auto'; } catch (error) { } finally { } // Firefox
  100. try { element.style.userSelect = 'auto'; } catch (error) { } finally { } // Chrome
  101. // 取消通过 JavaScript 实现的禁用快捷键
  102. try { element.onkeypress = null; } catch (error) { } finally { allowKeyEvents(element, 'keypress'); }
  103. try { element.onkeydown = null; } catch (error) { } finally { allowKeyEvents(element, 'keydown'); }
  104. try { element.onkeyup = null; } catch (error) { } finally { allowKeyEvents(element, 'keyup'); }
  105. // 取消通过 JavaScript 实现的页面离开检测
  106. try { element.onvisibilitychange = null; } catch (error) { } finally { allowEvent(element, 'visibilitychange'); }
  107. }
  108. // let removeAllListeners = function (old_element) {
  109. // let new_element = old_element.cloneNode(true);
  110. // old_element.parentNode.replaceChild(new_element, old_element);
  111. // };
  112. let removeHiddenElements = function (element, recursion) {
  113. if (element.style.display == 'none' || element.style.visibility == 'hidden') {
  114. console.info(`Remove <${element.tagName}> id='${element.id}' class='${element.className}' />`);
  115. element.remove();
  116. } else if (recursion) {
  117. for (let i = 0; i < element.children.length; i++) {
  118. removeHiddenElements(element.children.item(i), recursion);
  119. }
  120. }
  121. };
  122.  
  123. setInterval(
  124. // 对抗延迟运行(即在此脚本执行后运行)的禁用程序和循环执行的禁用程序,
  125. // 每 0.2 秒执行一次。
  126. (function () {
  127. 'use strict';
  128. if (document) {
  129. try { allowElement(document); } catch (error) { } finally { }
  130. }
  131. }), 200
  132. );
  133. setInterval(
  134. // 对抗延迟运行(即在此脚本执行后运行)的禁用程序和循环执行的禁用程序,
  135. // 每 0.3 秒执行一次。
  136. (function () {
  137. 'use strict';
  138. if (document) {
  139. try { allowElement(document.body); } catch (error) { } finally { }
  140. }
  141. }), 300
  142. );
  143.  
  144. setInterval(
  145. // 对抗延迟运行(即在此脚本执行后运行)的混淆程序和循环执行的混淆程序,
  146. // 每 2.1 秒执行一次。
  147. function () {
  148. 'use strict';
  149. if (document) {
  150. // 移除正文中的不可见元素
  151. try {
  152. // 获取正文节点
  153. let content_element = null;
  154. if (content_element === null) { // 检查 main 标签
  155. let content_elements = document.getElementsByTagName('main');
  156. if (content_elements.length > 0) {
  157. content_element = content_elements[0];
  158. }
  159. }
  160. if (content_element === null) { // 检查 id='content' 的标签
  161. content_element = document.getElementById('content');
  162. }
  163. if (content_element === null) { // 检查 id='contents' 的标签
  164. content_element = document.getElementById('contents');
  165. }
  166. if (content_element === null) { // 检查 class='content' 的标签
  167. let content_elements = document.getElementsByClassName('content');
  168. if (content_elements.length > 0) {
  169. content_element = content_elements[0];
  170. }
  171. }
  172. if (content_element === null) { // 检查 class='contents' 的标签
  173. let content_elements = document.getElementsByClassName('contents');
  174. if (content_elements.length > 0) {
  175. content_element = content_elements[0];
  176. }
  177. }
  178. // 移除不可见元素
  179. if (content_element) {
  180. removeHiddenElements(content_element, true);
  181. }
  182. } catch (error) { } finally { }
  183. }
  184. }, 2100
  185. );
  186.  
  187. console.debug('Finish user-script/remove-limits');
  188. })();