SimpCity Redirect Bypass

Decode base64 redirect links and go directly to target

이 스크립트를 설치하려면 Tampermonkey, Greasemonkey 또는 Violentmonkey와 같은 확장 프로그램이 필요합니다.

이 스크립트를 설치하려면 Tampermonkey와 같은 확장 프로그램을 설치해야 합니다.

이 스크립트를 설치하려면 Tampermonkey 또는 Violentmonkey와 같은 확장 프로그램이 필요합니다.

이 스크립트를 설치하려면 Tampermonkey 또는 Userscripts와 같은 확장 프로그램이 필요합니다.

이 스크립트를 설치하려면 Tampermonkey와 같은 확장 프로그램이 필요합니다.

이 스크립트를 설치하려면 유저 스크립트 관리자 확장 프로그램이 필요합니다.

(이미 유저 스크립트 관리자가 설치되어 있습니다. 설치를 진행합니다!)

이 스타일을 설치하려면 Stylus와 같은 확장 프로그램이 필요합니다.

이 스타일을 설치하려면 Stylus와 같은 확장 프로그램이 필요합니다.

이 스타일을 설치하려면 Stylus와 같은 확장 프로그램이 필요합니다.

이 스타일을 설치하려면 유저 스타일 관리자 확장 프로그램이 필요합니다.

이 스타일을 설치하려면 유저 스타일 관리자 확장 프로그램이 필요합니다.

이 스타일을 설치하려면 유저 스타일 관리자 확장 프로그램이 필요합니다.

(이미 유저 스타일 관리자가 설치되어 있습니다. 설치를 진행합니다!)

// ==UserScript==
// @name         SimpCity Redirect Bypass
// @namespace    sleazyfork.org
// @version      1.0
// @description  Decode base64 redirect links and go directly to target
// @match        *://simpcity.cr/*
// @grant        none
// @license MIT
// ==/UserScript==

(function () {
    'use strict';

    function decodeBase64Url(b64) {
        try {
            return atob(b64);
        } catch (e) {
            return null;
        }
    }

    function processLinks(root = document) {
        const links = root.querySelectorAll('a[href*="/redirect/?to="]');

        links.forEach(link => {
            try {
                const url = new URL(link.href);
                const encoded = url.searchParams.get('to');

                if (!encoded) return;

                const decoded = decodeBase64Url(encoded);
                if (!decoded || !decoded.startsWith('http')) return;

                // Replace the link target
                link.href = decoded;

                // Optional: remove tracking junk
                link.removeAttribute('rel');
                link.removeAttribute('target');

            } catch (e) {
                // ignore malformed URLs
            }
        });
    }

    // Initial run
    processLinks();

    // Watch for dynamically added content (AJAX, infinite scroll, etc.)
    const observer = new MutationObserver(mutations => {
        for (const m of mutations) {
            for (const node of m.addedNodes) {
                if (node.nodeType === 1) {
                    processLinks(node);
                }
            }
        }
    });

    observer.observe(document.body, {
        childList: true,
        subtree: true
    });

})();