Imgur Redirect (Bypass UK Block)

Redirect all links from imgur to a version on a Rimgo instance

您需要先安裝使用者腳本管理器擴展,如 TampermonkeyGreasemonkeyViolentmonkey 之後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyUserscripts 後才能安裝該腳本。

你需要先安裝一款使用者腳本管理器擴展,比如 Tampermonkey,才能安裝此腳本

您需要先安裝使用者腳本管理器擴充功能後才能安裝該腳本。

(我已經安裝了使用者腳本管理器,讓我安裝!)

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

(我已經安裝了使用者樣式管理器,讓我安裝!)

// ==UserScript==
// @name         Imgur Redirect (Bypass UK Block)
// @namespace    https://www.tampermonkey.net/
// @version      1.2
// @description  Redirect all links from imgur to a version on a Rimgo instance
// @author       zy
// @icon         
// @match        *://*.imgur.com/*
// @match        https://imgur.com/*
// @match        https://i.imgur.com/*
// @match        https://imgur.io/*
// @match        https://*.imgur.io/*
// @match        *://*/*
// @exclude      https://imgur.com/user/*
// @exclude      https://imgur.com/signin
// @exclude      https://imgur.com/register
// @exclude      https://imgur.com/arcade
// @exclude      https://imgur.com/upload
// @exclude      https://imgur.com/meme-generator
// @exclude      https://imgur.com/privacy
// @exclude      https://imgur.com/rules
// @exclude      https://imgur.com/emerald
// @exclude      https://imgur.com/ccpa
// @exclude      https://imgur.com/tos
// @exclude      https://imgur.com/about
// @exclude      https://imgur.com/apps
// @grant        none
// @run-at       document-start
// @license      MIT
// ==/UserScript==

(function () {
  'use strict';

  const redirectHost = 'https://imgur.sudovanilla.org/';

  function rewriteImgurUrl(url) {
    if (!url || typeof url !== 'string') return null;
    if (url.startsWith(redirectHost)) return null;

    if (/^https?:\/\/i\.imgur\.com\//.test(url)) {
      return url.replace(/^https?:\/\/i\.imgur\.com\//, redirectHost);
    }
    if (/^https?:\/\/imgur\.com\//.test(url)) {
      return url.replace(/^https?:\/\/imgur\.com\//, redirectHost);
    }
    if (/^\/\/i\.imgur\.com\//.test(url)) {
      return url.replace(/^\/\//, 'https://').replace(/^https?:\/\/i\.imgur\.com\//, redirectHost);
    }
    if (/imgur\.com/.test(url)) {
      try {
        const u = new URL(url, location.href);
        if (u.hostname.endsWith('imgur.com') || u.hostname.endsWith('imgur.io')) {
          return redirectHost + u.pathname.replace(/^\//, '') + (u.search || '');
        }
      } catch (e) {
        // ignore invalid URL parsing
      }
    }
    return null;
  }

  function processImageElement(img) {
    try {
      if (!(img instanceof HTMLImageElement)) return;
      const newSrc = rewriteImgurUrl(img.src);
      if (newSrc) {
        img.setAttribute('src', newSrc);
      }
    } catch (err) {
      console.error('processImageElement error', err);
    }
  }

  function processNodeRecursively(node) {
    if (!node || node.nodeType !== Node.ELEMENT_NODE) return;
    if (node.tagName === 'IMG') {
      processImageElement(node);
      return;
    }
    const imgs = node.querySelectorAll ? node.querySelectorAll('img') : [];
    imgs.forEach(processImageElement);
  }

  function rewriteAllImages() {
    document.querySelectorAll('img').forEach(processImageElement);
  }

  function attachObserver() {
    const target = document.body || document.documentElement;
    if (!target) {
      setTimeout(attachObserver, 50);
      return;
    }

    const observer = new MutationObserver(mutations => {
      for (const m of mutations) {
        // handle added nodes (subtrees)
        if (m.addedNodes && m.addedNodes.length) {
          m.addedNodes.forEach(node => processNodeRecursively(node));
        }
        // handle attribute changes (e.g., src changed)
        if (m.type === 'attributes' && m.target) {
          const t = m.target;
          if (t.tagName === 'IMG' && m.attributeName === 'src') {
            processImageElement(t);
          }
        }
      }
    });

    observer.observe(target, {
      childList: true,
      subtree: true,
      attributes: true,
      attributeFilter: ['src']
    });
  }

  function redirectPageIfImgur() {
    const href = location.href;
    if (!/https?:\/\/(i\.)?imgur\.com/.test(href)) return;
    try {
      const newUrl = rewriteImgurUrl(href);
      if (newUrl && newUrl !== href) {
        location.replace(newUrl);
      }
    } catch (err) {
      console.error('Redirect failed:', err);
    }
  }

  redirectPageIfImgur();

  if (document.readyState === 'loading') {
    window.addEventListener('DOMContentLoaded', rewriteAllImages, { once: true });
  } else {
    rewriteAllImages();
  }

  attachObserver();
})();