Reddit Bypass Enhancer

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

当前为 2025-03-18 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name Reddit Bypass Enhancer
  3. // @version 2.4
  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.  
  15. 'use strict';
  16.  
  17. (function () {
  18. function queryElementsDeep(selector, root = document) {
  19. const elements = new Set();
  20. try {
  21. root.querySelectorAll(selector).forEach(el => elements.add(el));
  22. root.querySelectorAll('*').forEach(el => {
  23. if (el.shadowRoot) {
  24. el.shadowRoot.querySelectorAll(selector).forEach(shadowEl => elements.add(shadowEl));
  25. }
  26. });
  27. } catch (error) {
  28. console.error("Error in queryElementsDeep:", error);
  29. }
  30. return [...elements];
  31. }
  32.  
  33. const SELECTORS = {
  34. nsfwModal: 'shreddit-async-loader[bundlename*="nsfw_blocking_modal"]',
  35. promptContainer: 'xpromo-nsfw-blocking-container > *',
  36. prompt: '.prompt',
  37. blurredContainer: 'shreddit-blurred-container',
  38. thumbnailBlur: '.thumbnail-blur',
  39. communityHighlightCard: 'community-highlight-card',
  40. thumbnailShadow: '.thumbnail-shadow',
  41. mediaBackground: '.bg-media-background',
  42. blurredSpan: 'span.inner.blurred',
  43. scrim: '.absolute.top-0.left-0.w-full.h-full.bg-scrim',
  44. imageFilter: 'img.mb-0.h-full.w-full.object-cover',
  45. video: 'shreddit-player, video',
  46. mediaTelemetryObserver: 'media-telemetry-observer',
  47. spoilerOverlay: '.absolute.inset-0.overflow-visible.flex.items-center.justify-center'
  48. };
  49.  
  50. function removeNSFWBlock() {
  51. try {
  52. document.querySelector(SELECTORS.nsfwModal)?.remove();
  53. document.querySelector(SELECTORS.promptContainer)?.shadowRoot?.querySelector(SELECTORS.prompt)?.remove();
  54. document.getElementById("nsfw-qr-dialog")?.remove();
  55. document.querySelector("body > div:nth-child(7)")?.remove();
  56.  
  57. queryElementsDeep(SELECTORS.blurredContainer).forEach(container => {
  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. e.target.click();
  65. }, { once: true });
  66. container.firstElementChild.click();
  67. }
  68. });
  69.  
  70. queryElementsDeep(SELECTORS.thumbnailBlur).forEach(el => el.classList.remove('thumbnail-blur'));
  71. queryElementsDeep(SELECTORS.communityHighlightCard).forEach(card => card.removeAttribute('blurred'));
  72. queryElementsDeep(SELECTORS.imageFilter).forEach(img => img.style.removeProperty('filter'));
  73. queryElementsDeep(SELECTORS.video).forEach(video => video.classList.remove('blur'));
  74. queryElementsDeep(SELECTORS.blurredSpan).forEach(span => span.style.removeProperty('filter'));
  75. queryElementsDeep(SELECTORS.scrim).forEach(scrim => scrim.remove());
  76. queryElementsDeep(SELECTORS.spoilerOverlay).forEach(spoiler => spoiler.remove());
  77.  
  78. document.querySelector(SELECTORS.thumbnailShadow)?.remove();
  79. document.querySelector(SELECTORS.mediaBackground)?.style.setProperty('background-color', 'transparent');
  80.  
  81. } catch (error) {
  82. console.error("Error in removeNSFWBlock:", error);
  83. }
  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.  
  95. observer.observe(document.documentElement, { childList: true, subtree: true });
  96.  
  97. const shredditCheckInterval = setInterval(() => {
  98. if (!document.querySelector('shreddit-app')) {
  99. observer.disconnect();
  100. clearInterval(shredditCheckInterval);
  101. }
  102. }, 5000);
  103. })();