Greasy Fork 支持简体中文。

Bypass It

Automatically Bypass Restrictions and Get Straight to Your Destination!

// ==UserScript==
// @name         Bypass It
// @namespace    http://tampermonkey.net/
// @version      2025-03-01 (1)
// @description  Automatically Bypass Restrictions and Get Straight to Your Destination!
// @supportURL   https://greasyfork.org/scripts/527564/feedback
// @author       You
// @match        *://aylink.co/*
// @match        *://v2links.me/*
// @match        *://upfion.com/*
// @match        *://cutyion.com/*
// @match        *://*.devnote.in/*
// @match        *://naamlist.com/*
// @match        *://modsfire.com/*
// @match        *://*.gmsrweb.org/*
// @match        *://modijiurl.com/*
// @match        *://*.techyuth.xyz/*
// @match        *://financewada.com/*
// @match        *://cryptowidgets.net/*
// @match        *://*.wikijankari.com/*
// @match        *://*.idblogmarket.com/*
// @match        *://*.phonesparrow.com/*
// @match        *://financenova.online/*
// @match        *://rajasthantopnews.com/*
// @match        *://utkarshonlinetest.com/*
// @match        *://www.youtube.com/redirect*
// @match        *://www.facebook.com/flx/warn/*
// @match        *://www.instagram.com/linkshim/*
// @icon         
// @grant        GM_getValue
// @grant        GM_setValue
// @grant        unsafeWindow
// @grant        GM_openInTab
// @grant        GM_xmlhttpRequest
// @grant        GM_registerMenuCommand
// @license      MIT
// @noframes
// ==/UserScript==

const noop = () => {};
const rawWindow = unsafeWindow;
const currentUrl = location.href;
const autoRedirectToSocial = GM_getValue("autoRedirectToSocial", false);

/**
 * Waits for an element matching the selector to appear in the DOM.
 * @param {string} selector - The CSS selector of the element.
 * @returns {Promise<Element>} - A promise that resolves with the found element.
 */
function waitForElement(selector) {
    const findElement = () => {
        if (selector.startsWith("//")) {
            return document.evaluate(selector, document, null, 9).singleNodeValue;
        }
        return document.querySelector(selector);

    };

    return new Promise((resolve) => {
        const element = findElement();
        if (document.contains(element)) {
            return resolve(element);
        }
        const observer = new MutationObserver((mutations, observerInstance) => {
            const node = findElement();
            if (document.contains(node)) {
                observerInstance.disconnect();
                resolve(node);
            }
        });
        observer.observe(document.documentElement, {
            attributes: true,
            childList: true,
            subtree: true,
        });
    });
}

/**
 * Navigates the browser to the specified URL.
 * @param {string} url - The destination URL.
 */
function navigateTo(url) {
    location = url;
}

/**
 * Returns a promise that resolves after a specified number of seconds.
 * @param {number} seconds - The delay in seconds.
 * @returns {Promise<void>}
 */
function waitSeconds(seconds) {
    return new Promise((resolve) => setTimeout(resolve, seconds * 1000));
}

/**
 * Waits for an element matching the selector and then redirects to its href.
 * @param {string} selector - The CSS selector of the link element.
 */
function autoRedirectToLink(selector) {
    waitForElement(selector).then((element) => navigateTo(element.href));
}

/**
 * Creates a menu command in the Tampermonkey UI that toggles a boolean state.
 * @param {string} key - The key used to store and retrieve the boolean state in GM storage.
 * @param {string} title - The title displayed in the menu, which is prefixed with a checkmark or cross.
 */
function createMenu(key, title) {
    const isActive = GM_getValue(key, false);
    const prefix = isActive ? "✔️ " : "❌ ";

    GM_registerMenuCommand(prefix + title, function() {
        GM_setValue(key, !isActive);
        createMenu(key, title);
        alert(`${title} has been ${!isActive ? "enabled" : "disabled"}.`);
    }, {
        id: key
    });
}

/**
 * Converts an image URL to a Base64 string.
 * This function fetches the image using an HTTP GET request and converts it into a Base64-encoded string.
 *
 * @param {string} imageUrl - The URL of the image to convert.
 * @returns {Promise<string|null>} A Promise that resolves with the Base64-encoded image string, or null if an error occurs.
 */
function imageUrlToBase64(imageUrl) {
    if (!imageUrl) {
        console.error("No imageUrl provided");
        return null;
    }
    try {
        return new Promise((resolve, reject) => {
            GM_xmlhttpRequest({
                method: "GET",
                url: imageUrl,
                responseType: "blob",
                onload: function(response) {
                    if (!response.response) {
                        reject("No response received");
                        return;
                    }

                    const blob = response.response;
                    const reader = new FileReader();
                    reader.onload = () => resolve(reader.result);
                    reader.onerror = (error) => {
                        console.error("FileReader error:", error);
                        reject(error);
                    };
                    reader.readAsDataURL(blob);
                },
                onerror: function(error) {
                    console.error("Error fetching image:", error);
                    reject(error);
                }
            });
        });
    } catch (error) {
        console.error("Error converting image to Base64:", error);
        return null;
    }
}

/**
 * Sends a Base64-encoded image to a language model and receives a descriptive response based on specific icon classes.
 * This function sends the image as Base64 and asks the model to describe it with one of the provided icon classes.
 *
 * @param {string} base64Image - The Base64-encoded image string.
 * @param {Array<string>} iconClasses - An array of strings representing the icon classes to use for the description.
 * @returns {Promise<string|null>} A Promise that resolves with the description of the image in English, or null if an error occurs.
 */
async function getGeminiResponse(base64Image, iconClasses) {
    if (!window.ai) {
        console.error("window.ai is not available. Ensure the Prompt API is enabled.");
        return;
    }

    try {
        // Validate Gemini Nano availability
        const capabilities = await window.ai.languageModel.capabilities();
        if (!capabilities || !capabilities.available) {
            console.error("Gemini Nano is not available. Ensure the model is downloaded.");
            return;
        }

        // Create session with system prompt to enforce English responses
        const session = await window.ai.languageModel.create({
            systemPrompt: "Analyze the given image and describe it using one of the provided categories. Be direct and concise.",
            expectedInputLanguages: ["en"]
        });

        // Construct the prompt using the base64Image and iconClasses
        let result = await session.prompt(
            `Analyze the image: ${base64Image} and describe this image using one of: ${iconClasses.join(", ")}. Be direct.`
        );

        console.log("Gemini Nano Response:", result);
        return result; // Ensure you're getting the correct response
    } catch (error) {
        console.error("Error with Gemini Nano:", error);
        throw error; // Better error propagation
    }
}

/**
 * Executes a callback when the current host matches a pattern.
 * @param {string|RegExp} hostPattern - The host pattern or RegExp to match against location.host.
 * @param {Function} callback - The function to execute when matched.
 * @param {...any} args - Arguments to pass to the callback.
 */
function runWhenHostMatches(hostPattern, callback, ...args) {
    const isMatch = new RegExp(hostPattern).test(location.host);

    if (isMatch) callback(...args);
}

/**
 * Simulates a click on an element (or multiple elements) specified by a selector.
 * Dispatches "mouseover", "mousedown", "mouseup", and "click" events.
 * @param {string} selector - The CSS selector of the element(s) to click.
 * @param {number} [delay=0] - Optional delay in seconds before clicking.
 */
function simulateClick(selector, delay = 0) {
    const events = ["mouseover", "mousedown", "mouseup", "click"];
    const selectors = selector.split(", ");
    // If multiple selectors are provided, iterate over each.
    if (selectors.length > 1) {
        selectors.forEach((sel) => simulateClick(sel, delay));
        return;
    }

    waitForElement(selector).then(async (element) => {
        if (delay > 0) {
            await waitSeconds(delay);
        }

        events.forEach((eventName) => {
            const event = new MouseEvent(eventName, {
                bubbles: true,
            });
            element.dispatchEvent(event);
        });
    });
}

/**
 * Calls the callback function once a captcha challenge is solved.
 * Checks for available captcha libraries (turnstile, hcaptcha, or grecaptcha).
 * @param {Function} callback - The function to execute when the captcha is solved.
 * @param {Function} [onWait=noop] - Optional function to call if waiting.
 */
function whenCaptchaSolved(callback, onWait = noop) {
    let intervalId;
    const stopChecking = () => clearInterval(intervalId);

    const checkCaptcha = () => {
        try {
            const captcha = rawWindow.turnstile || rawWindow.hcaptcha || rawWindow.grecaptcha;
            const response = captcha.getResponse();

            if (response) {
                stopChecking();
                callback();
            }
        } catch (error) {
            onWait(stopChecking);
        }
    };

    checkCaptcha();
    intervalId = setInterval(checkCaptcha, 1000);
}

// Main execution
(function () {
    "use strict";
    runWhenHostMatches("modsfire.com", simulateClick, ".download-button, .download-button[href]");
    runWhenHostMatches("(financewada|utkarshonlinetest).com|financenova.online|v2links.me", simulateClick, ".get_btn a[href], //div[contains(text(),'Continue')], .get-link:not(.disabled)");

    runWhenHostMatches("devnote.in|techyuth.xyz", simulateClick, "#scroll:not(.no), #getlinks[style*='display: block;'], .get-link:not(.disabled)");
    autoRedirectToSocial && runWhenHostMatches("(instagram|youtube|facebook).com", simulateClick, ".-cx-PRIVATE-Linkshim__followLink__, #invalid-token-redirect-goto-site-button, .selected");
    createMenu("autoRedirectToSocial", "Auto-Redirect to Social Media");

    runWhenHostMatches("(wikijankari|idblogmarket|modijiurl|phonesparrow|naamlist|rajasthantopnews).com|gmsrweb.org", simulateClick, "center a[style*='display: block;'], .get-link:not(.disabled)");

    runWhenHostMatches("aylink.co", simulateClick, ".btn-go, .complete[style*='display: block;'] a, a.btn:not(.btn-go)")

    /*
    Sometimes, the AI model may provide inaccurate responses. Unfortunately, I can't control this. Sorry!
    
    https://danduh.github.io/window-ai/
    */
    runWhenHostMatches("cryptowidgets.net", function () {
        waitForElement('.captcha-icon img').then(async function (element) {
            const base64 = await imageUrlToBase64(element.src);
            if (!base64) return console.log("Failed to convert image.");

            const icons = document.querySelectorAll('#icon-options i');
            const iconClasses = Array.from(icons).map(icon => icon.classList[1]);

            console.log("Base64 Image:", base64);

            const description = await getGeminiResponse(base64, iconClasses);
            console.log("Final Description:", description);
            simulateClick(`.${description}`);
        });
    });
    
    runWhenHostMatches("upfion.com", function() {
        simulateClick("#link-button:not([disabled])");
        waitForElement("a#link-button:not([disabled])").then((element) => navigateTo(element.href));
    });
    
    runWhenHostMatches("cutyion.com", simulateClick, "#submit-button:not([disabled])");
})();