Grok File & Post Switcher

Switch between your Grok imagine posts and direct file links.

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

You will need to install an extension such as Tampermonkey to install this script.

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

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

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

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

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

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

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

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

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

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

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

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

// ==UserScript==
// @name            Grok File & Post Switcher
// @name:de         Grok Datei & Post Wechsler
// @description     Switch between your Grok imagine posts and direct file links.
// @description:de  Wechselt zwischen deinen Grok Imagine-Beiträgen und direkten Dateilinks.
// @version         2.2.0
// @author          JAS1998 (https://greasyfork.org/users/4792)
// @copyright       2026+, JAS1998
// @namespace       https://greasyfork.org/users/4792
// @license         CC BY-NC-ND 4.0; http://creativecommons.org/licenses/by-nc-nd/4.0/
// @icon            https://grok.com/images/favicon.ico
//
// @match           https://grok.com/*
// @noframes
// @run-at          document-start
//
// @grant           GM_registerMenuCommand
// @grant           GM_info
// @grant           GM_setValue
// @grant           GM_getValue
// @grant           GM_openInTab
// @grant           GM_deleteValue
//
// @compatible      Chrome tested with Tampermonkey
// @supportURL      https://greasyfork.org/scripts/574064/feedback
// @contributionURL https://www.paypal.com/donate?hosted_button_id=9JEGCDFJJHWU8
// @contributionAmount €1.00
// ==/UserScript==

/* jshint esversion: 9 */

(function() {
    'use strict';

    const _vault = "4792";
    const _isOriginal = GM_info.script.namespace.includes(_vault);
    const _originalURL = GM_info.script.supportURL.replace("feedback", "");
    const _authorName = GM_info.script.author.split('(')[0].trim();

    let isRegistered = false;
    let hasExecuted = false;

    const postPath = "https://grok.com/imagine/post/";
    const filePath = "https://grok.com/files?file=";

    function switchToOriginalFile() {
        const url = window.location.href;
        if (url.includes(postPath)) {
            GM_setValue("pendingRename", true);
            const target = url.split("&")[0].replace(postPath, filePath);
            GM_openInTab(target, { active: false, insert: true, setParent: true });
        }
    }

    function switchToPost() {
        const url = window.location.href;
        if (url.includes(filePath)) {
            const target = url.split("&")[0].replace(filePath, postPath);
            window.open(target, "_blank");
        }
    }

    function autoConfirmDelete() {
        if (!GM_getValue("autoConfirmDelete", false)) return;

        const buttons = Array.from(document.querySelectorAll('button'));
        
        const deleteConfirmBtn = buttons.find(btn =>
            btn.textContent.trim() === 'Delete' &&
            !btn.textContent.toLowerCase().includes('cancel') &&
            window.getComputedStyle(btn).display !== 'none'
        );

        const checkmarkPath = "M19.7998 6.59961L10.1221 19.5039L4.30762 13.9219L5.69238 12.4785L9.87793 16.4961L18.2002 5.40039L19.7998 6.59961Z";
        const svgConfirmBtn = buttons.find(btn => {
            const svg = btn.querySelector('svg');
            return svg && svg.innerHTML.includes(checkmarkPath) && window.getComputedStyle(btn).display !== 'none';
        });

        const targetBtn = deleteConfirmBtn || svgConfirmBtn;

        if (targetBtn) {
            setTimeout(() => {
                targetBtn.click();
            }, 100);
        }
    }

    async function applyGrokRename() {
        if (!GM_getValue("autoRenameEnabled", false)) return;
        if (!GM_getValue("pendingRename", false)) return;
        if (hasExecuted) return;

        const titleSpan = document.querySelector('span.truncate.px-1\\.5.w-fit.max-w-\\[50\\%\\].h-8.cursor-text');

        if (titleSpan) {
            const currentText = titleSpan.innerText.toLowerCase();
            const isMedia = currentText.includes("jpg") || currentText.includes("jpeg") || currentText.includes("mp4");

            if (isMedia && !currentText.includes("dont delete")) {
                hasExecuted = true;

                let extension = ".jpg";
                if (currentText.includes("mp4")) extension = ".mp4";
                else if (currentText.includes("jpeg")) extension = ".jpeg";

                const newName = "DONT DELETE" + extension;
                titleSpan.click();

                setTimeout(() => {
                    const form = document.querySelector('form.w-full');
                    const input = form ? form.querySelector('input') : null;

                    if (input) {
                        input.value = newName;
                        input.dispatchEvent(new Event('input', { bubbles: true }));
                        input.dispatchEvent(new Event('change', { bubbles: true }));
                        form.dispatchEvent(new Event('submit', { bubbles: true, cancelable: true }));
                        GM_deleteValue("pendingRename");
                    } else {
                        hasExecuted = false;
                    }
                }, 400);
            }
        }
    }

    window.addEventListener('keydown', function(e) {
        if (e.target.tagName === 'INPUT' || e.target.tagName === 'TEXTAREA' || e.target.isContentEditable) {
            return;
        }

        if (e.key === '^' || e.key === 'Dead' || e.code === 'Backquote') {
            const url = window.location.href;
            if (url.includes(postPath)) {
                e.preventDefault();
                e.stopImmediatePropagation();
                switchToOriginalFile();
            }
            else if (url.includes(filePath)) {
                e.preventDefault();
                e.stopImmediatePropagation();
                switchToPost();
            }
        }
    }, true);

    function initMenuOnce() {
        if (isRegistered) return;
        isRegistered = true;

        const url = window.location.href;
        const isOnFilePage = url.includes("grok.com/files");

        const checkProtection = () => {
            if (!_isOriginal) {
                alert("Please install the Original Version");
                window.location.href = _originalURL;
                return false;
            }
            return true;
        };

        if (isOnFilePage) {
            const observer = new MutationObserver(() => {
                if (!hasExecuted) applyGrokRename();
                autoConfirmDelete();
            });
            observer.observe(document.documentElement, { childList: true, subtree: true });
            setTimeout(applyGrokRename, 1500);
        }

        GM_registerMenuCommand("📁 View Original File", function () {
            if (!checkProtection()) return;
            switchToOriginalFile();
        });

        GM_registerMenuCommand("🔗 View Post", function () {
            if (!checkProtection()) return;
            switchToPost();
        });

        const isRenameEnabled = GM_getValue("autoRenameEnabled", false);
        GM_registerMenuCommand(isRenameEnabled ? "🔒 Mark File as 'DONT DELETE' (ON)" : "🔓 Mark File as 'DONT DELETE' (OFF)", function() {
            GM_setValue("autoRenameEnabled", !isRenameEnabled);
            location.reload();
        });

        if (isOnFilePage) {
            const isDeleteEnabled = GM_getValue("autoConfirmDelete", false);
            GM_registerMenuCommand(isDeleteEnabled ? "🚮 Auto-Confirm Delete (ON)" : "🗑️ Auto-Confirm Delete (OFF)", function() {
                GM_setValue("autoConfirmDelete", !isDeleteEnabled);
                location.reload();
            });
        }

        GM_registerMenuCommand("🎁 Donate", function () {
            alert(`Hello, I'm ${_authorName}\nand I wrote this script as a hobby.\nIf you find it useful, I would appreciate a small donation! =)`);
            window.open("https://www.paypal.com/donate?hosted_button_id=9JEGCDFJJHWU8", "_blank");
        });
    }

    initMenuOnce();

})();