Reddit Bypass Enhancer

Bypass "open in app" prompts, unblur NSFW content & thumbnails, remove spoiler overlays.

目前為 2025-03-24 提交的版本,檢視 最新版本

  1. // ==UserScript==
  2. // @name Reddit Bypass Enhancer
  3. // @version 2.3
  4. // @description Bypass "open in app" prompts, unblur NSFW content & thumbnails, remove spoiler overlays.
  5. // @author UniverseDev
  6. // @license GPL-3.0-or-later
  7. // @match https://www.reddit.com/*
  8. // @match https://sh.reddit.com/*
  9. // @grant none
  10. // @run-at document-start
  11. // @noframes
  12. // @namespace https://greasyfork.org/en/users/1030895-universedev
  13. // ==/UserScript==
  14. 'use strict';
  15. (function () {
  16. function queryElementsDeep(selector) {
  17. const elements = new Set();
  18. try {
  19. document.querySelectorAll(selector).forEach(el => elements.add(el));
  20. document.querySelectorAll('*').forEach(el => {
  21. if (el.shadowRoot) {
  22. el.shadowRoot.querySelectorAll(selector).forEach(shadowEl => elements.add(shadowEl));
  23. }
  24. });
  25. } catch (error) {
  26. console.error("Error in queryElementsDeep:", error);
  27. }
  28. return [...elements];
  29. }
  30. const SELECTORS = {
  31. nsfwModal: 'shreddit-async-loader[bundlename*="nsfw_blocking_modal"]',
  32. promptContainer: 'xpromo-nsfw-blocking-container > *',
  33. prompt: '.prompt',
  34. blurredContainer: 'shreddit-blurred-container',
  35. thumbnailBlur: '.thumbnail-blur',
  36. communityHighlightCard: 'community-highlight-card',
  37. thumbnailImage: 'img.mb-0.h-full.w-full.object-cover',
  38. thumbnailShadow: '.thumbnail-shadow',
  39. mediaBackground: '.bg-media-background',
  40. blurredSpan: 'span.inner.blurred',
  41. scrim: '.absolute.top-0.left-0.w-full.h-full.bg-scrim',
  42. imageFilter: 'img.mb-0.h-full.w-full.object-cover',
  43. video: 'shreddit-player, video',
  44. mediaTelemetryObserver: 'media-telemetry-observer',
  45. spoilerOverlay: '.absolute.inset-0.overflow-visible.flex.items-center.justify-center'
  46. };
  47. function removeNSFWBlock() {
  48. try {
  49. document.querySelector(SELECTORS.nsfwModal)?.remove();
  50. document.querySelector(SELECTORS.promptContainer)?.shadowRoot?.querySelector(SELECTORS.prompt)?.remove();
  51. document.getElementById("nsfw-qr-dialog")?.remove();
  52. document.querySelector("body > div:nth-child(7)")?.remove();
  53. document.querySelectorAll('body > *').forEach(el => {
  54. if (el.style.backdropFilter) el.style.backdropFilter = 'none';
  55. });
  56. document.querySelectorAll(SELECTORS.blurredContainer).forEach(container => {
  57. try {
  58. if (container.shadowRoot?.innerHTML && container.firstElementChild) {
  59. container.firstElementChild.addEventListener('click', e => {
  60. e.preventDefault();
  61. if (e.target.closest(SELECTORS.mediaTelemetryObserver)) {
  62. e.stopPropagation();
  63. }
  64. container.classList.add('clicked-container');
  65. e.target.click();
  66. }, { once: true });
  67. container.firstElementChild.click();
  68. }
  69. } catch (error) {
  70. console.error("Error processing blurred container:", error, container);
  71. }
  72. });
  73. document.querySelectorAll(SELECTORS.thumbnailBlur).forEach(el => el.classList.remove('thumbnail-blur'));
  74. document.querySelectorAll(SELECTORS.communityHighlightCard).forEach(card => card.removeAttribute('blurred'));
  75. document.querySelectorAll(SELECTORS.imageFilter).forEach(img => img.style.removeProperty('filter'));
  76. document.querySelectorAll(SELECTORS.video).forEach(video => video.classList.remove('blur'));
  77. document.querySelector(SELECTORS.thumbnailShadow)?.remove();
  78. document.querySelector(SELECTORS.mediaBackground)?.style.setProperty('background-color', 'transparent');
  79. queryElementsDeep(SELECTORS.blurredSpan).forEach(span => span.style.removeProperty('filter'));
  80. queryElementsDeep(SELECTORS.scrim).forEach(scrim => scrim.remove());
  81. queryElementsDeep(SELECTORS.spoilerOverlay).forEach(spoiler => spoiler.remove());
  82. } catch (error) {
  83. console.error("Error in removeNSFWBlock:", error);
  84. }
  85. }
  86. const observer = new MutationObserver(mutations => {
  87. for (const mutation of mutations) {
  88. if (mutation.addedNodes.length) {
  89. removeNSFWBlock();
  90. break;
  91. }
  92. }
  93. });
  94. observer.observe(document.documentElement, { childList: true, subtree: true });
  95. const shredditCheckInterval = setInterval(() => {
  96. if (!document.querySelector('shreddit-app')) {
  97. observer.disconnect();
  98. clearInterval(shredditCheckInterval);
  99. }
  100. }, 5000);
  101. })();