ChatGPT Helper (面板+小球可见增强版)

支持Access Token获取、服务降级检测等功能,确保小球和面板稳定显示,面板在小球旁边且可拖拽

// ==UserScript==
// @name         ChatGPT Helper (面板+小球可见增强版)
// @version      1.9
// @description  支持Access Token获取、服务降级检测等功能,确保小球和面板稳定显示,面板在小球旁边且可拖拽
// @license      GNU Affero General Public License v3.0 or later
// @match        https://chatgpt.com/*
// @match        https://oai.kylelv.com/*
// @match        https://plus.aivvm.com/*
// @match        https://new.oaifree.com/*
// @match        https://*.new.oaifree.com/*
// @match        https://shared.oaifree.com/*
// @match        https://chat.rawchat.top/*
// @match        https://chat.sharedchat.cn/*
// @match        https://gpt.github.cn.com/*
// @match        https://free.xyhelper.cn/*
// @match        https://chatgpt.dairoot.cn/*
// @icon         https://linux.do/user_avatar/linux.do/f-droid/288/228666_2.png
// @grant        GM_xmlhttpRequest
// @grant        GM_setClipboard
// @run-at       document-end
// @namespace https://gf.qytechs.cn/users/1416670
// ==/UserScript==

(function () {
    'use strict';

    const currentUrl = window.location.href;
    const myWindow = (typeof unsafeWindow !== 'undefined') ? unsafeWindow : window;
    if (!(
        currentUrl.startsWith('https://chatgpt.com/') ||
        currentUrl.startsWith('https://oai.kylelv.com/') ||
        currentUrl.startsWith('https://plus.aivvm.com/') ||
        currentUrl.startsWith('https://new.oaifree.com/') ||
        currentUrl.startsWith('https://shared.oaifree.com/') ||
        currentUrl.startsWith('https://chat.rawchat.top/') ||
        currentUrl.startsWith('https://chat.sharedchat.cn/') ||
        currentUrl.startsWith('https://gpt.github.cn.com/') ||
        currentUrl.startsWith('https://free.xyhelper.cn/') ||
        currentUrl.startsWith('https://chatgpt.dairoot.cn/') ||
        currentUrl.match(/^https:\/\/(?:[^\/]+\.)?new\.oaifree\.com\//)
    )) {
        return;
    }

    // ========== 1. Hook Function.prototype.toString 和 fetch ==========
    (() => {
        const originalToString = myWindow.Function.prototype.toString;
        const myFunction_toString_symbol = myWindow.Symbol('custom_toString');
        const myToString = function () {
            return typeof this === 'function' && this[myFunction_toString_symbol] || originalToString.call(this);
        };

        function set_native(func, key, value) {
            Object.defineProperty(func, key, {
                enumerable: false,
                configurable: true,
                writable: true,
                value: value,
            });
        }

        delete myWindow.Function.prototype.toString;
        set_native(myWindow.Function.prototype, 'toString', myToString);
        set_native(myWindow.Function.prototype.toString, myFunction_toString_symbol, 'function toString() { [native code] }');
        globalThis.hookFix = (func, functionName) => {
            set_native(func, myFunction_toString_symbol, `function ${functionName || ''}() { [native code] }`);
        };
    })();

    // ========== 2. 创建悬浮面板 (带ID) ==========
    const panelId = 'chatgpt-helper-panel';
    let panel = document.createElement('div');
    panel.id = panelId;
    panel.style.cssText = `
        position: fixed; /* 使用 fixed 以便在页面滚动时保持位置 */
        width: 350px;
        background: linear-gradient(145deg, #f8f9fa, #e9ecef);
        border-radius: 15px;
        box-shadow: 0 15px 35px rgba(0,0,0,0.1), 0 5px 15px rgba(0,0,0,0.05);
        padding: 20px;
        font-family: 'Inter', 'Segoe UI', Arial, sans-serif;
        color: #2c3e50;
        transition: all 0.3s ease;
        display: none;
        z-index: 10001; /* 确保在最前 */
    `;
    // 初始位置设置为小球旁边
    // 位置将在面板显示时动态设置

    const titleBar = document.createElement('div');
    titleBar.innerHTML = `
        <div style="display: flex; justify-content: space-between; align-items: center; margin-bottom: 15px;">
            <h2 style="margin: 0; font-size: 18px; color: #2c3e50;">
                ChatGPT Helper
            </h2>
            <button id="chatgpt-helper-close-btn" style="background: none; border: none; color: #6c757d; font-size: 20px; cursor: pointer;">×</button>
        </div>
    `;
    panel.appendChild(titleBar);

    const statusSection = document.createElement('div');
    statusSection.innerHTML = `
        <div style="background-color: #f1f3f5; border-radius: 10px; padding: 15px; margin-bottom: 0;">
            <div style="font-weight: 600;">服务降级检测</div>
            <div>PoW难度: <span id="difficulty">N/A</span> <span id="difficulty-level"></span></div>
            <div style="font-size: 12px; color: #6c757d; margin-top: 5px;">
                <em>这个值越小,代表PoW难度越高,ChatGPT认为你的IP风险越高。</em>
            </div>
            <div>IP质量: <span id="ip-quality">N/A</span></div>
            <div id="persona-container" style="display: none;">用户类型: <span id="persona">N/A</span></div>
        </div>
    `;
    panel.appendChild(statusSection);

    const tokenSection = document.createElement('div');
    tokenSection.innerHTML = `
        <div style="background-color: #f1f3f5; border-radius: 10px; padding: 15px; margin-bottom: 15px;">
            <div style="font-weight: 600; font-size: 16px; text-align: center; margin-bottom: 10px;">Access Token Tool</div>
            <div style="margin-bottom: 15px;">
                <textarea id="chatgpt-helper-token-display" style="
                    width: 100%;
                    height: 100px;
                    border: 1px solid #ced4da;
                    border-radius: 8px;
                    padding: 10px;
                    resize: none;
                    font-family: monospace;
                    background-color: #f8f9fa;
                "></textarea>
            </div>
            <div style="display: flex; justify-content: center; gap: 10px;">
                <button id="chatgpt-helper-fetch-btn" style="
                    flex: 1;
                    padding: 10px;
                    background-color: #2ecc71;
                    color: white;
                    border: none;
                    border-radius: 8px;
                    cursor: pointer;
                    transition: background-color 0.3s;
                ">获取</button>
                <button id="chatgpt-helper-copy-btn" style="
                    flex: 1;
                    padding: 10px;
                    background-color: #3498db;
                    color: white;
                    border: none;
                    border-radius: 8px;
                    cursor: pointer;
                    transition: background-color 0.3s;
                ">复制</button>
            </div>
        </div>
    `;
    panel.appendChild(tokenSection);


    const createToast = (message) => {
        const toast = document.createElement('div');
        toast.innerText = message;
        toast.style.cssText = `
            position: fixed;
            bottom: 20px;
            right: 20px;
            background: rgba(76, 175, 80, 0.9);
            color: white;
            border-radius: 8px;
            padding: 10px 20px;
            box-shadow: 0 5px 15px rgba(0,0,0,0.3);
            font-family: 'Inter', 'Segoe UI', Arial, sans-serif;
            z-index: 10002;
            transition: opacity 0.5s ease;
        `;
        document.body.appendChild(toast);
        setTimeout(() => {
            toast.style.opacity = '0';
            setTimeout(() => {
                document.body.removeChild(toast);
            }, 500);
        }, 1000);
    };

    // ========== 3. 创建悬浮小球 (带ID) ==========
    const ballId = 'chatgpt-helper-ball';
    let ball = document.createElement('div');
    ball.id = ballId;

    // 创建一个小圆点元素用于显示风险等级
    const riskIndicator = document.createElement('div');
    riskIndicator.id = 'risk-indicator';
    ball.appendChild(riskIndicator);

    // 设置小球的样式 + !important
    const setBallStyle = () => {
        ball.style.position = 'fixed';
        ball.style.top = '25%';
        ball.style.right = '20px';
        ball.style.width = '40px';
        ball.style.height = '40px';
        ball.style.borderRadius = '50%';
        ball.style.cursor = 'pointer';
        ball.style.boxShadow = '0 4px 6px rgba(0,0,0,0.1)';
        ball.style.transition = 'transform 0.2s';
        // 强化 z-index 与 display
        ball.style.setProperty('z-index', '999999999', 'important');
        ball.style.setProperty('display', 'block', 'important');

        // 设置风险指示器的小圆点样式
        riskIndicator.style.position = 'absolute';
        riskIndicator.style.top = '-2px';
        riskIndicator.style.right = '-2px';
        riskIndicator.style.width = '8px';
        riskIndicator.style.height = '8px';
        riskIndicator.style.borderRadius = '50%';
        riskIndicator.style.backgroundColor = 'transparent'; // 初始无颜色
        riskIndicator.style.opacity = '0.3'; // 初始透明度为30%
    };
    setBallStyle();

    // 根据风险等级更新小圆点的颜色
    const updateRiskIndicator = (color) => {
        riskIndicator.style.backgroundColor = color;
    };

    const getFavicon = () => {
        const favicon = document.querySelector("link[rel~='apple-touch-icon']") ||
            document.querySelector("link[rel~='icon']") ||
            document.querySelector("link[rel='shortcut icon']") ||
            document.querySelector("link[rel='icon']");
        if (favicon) {
            const faviconUrl = favicon.href;
            ball.style.backgroundImage = `url(${faviconUrl})`;
            ball.style.backgroundSize = 'cover';
            ball.style.backgroundPosition = 'center';
        } else {
            ball.style.backgroundColor = 'rgba(0, 123, 255, 0.8)';
        }
    };
    getFavicon();

    ball.addEventListener('mouseenter', () => {
    });
    ball.addEventListener('mouseleave', () => {
    });

    // 点击小球时,面板显示/隐藏
    // 在显示面板时,调整面板的位置相对于小球
    ball.addEventListener('click', () => {
        if (panel.style.display === 'none' || panel.style.display === '') {
            // 显示面板
            userManuallyHidePanel = false; // 记录:用户现在让它显示
            panel.style.display = 'block';
            positionPanel();
        } else {
            // 隐藏面板
            userManuallyHidePanel = true;
            panel.style.display = 'none';
        }
    });

    // ========== 4. 把小球和面板放到 body 中 ==========
    document.body.appendChild(panel);
    document.body.appendChild(ball);

    // ========== 5. 面板的拖拽功能 ==========
    const titleBarDiv = titleBar.querySelector('div');
    let isDragging = false;
    let currentX, currentY, initialX, initialY, xOffset = 0, yOffset = 0;

    const dragStart = (e) => {
        // 只有在拖拽 titleBar 时有效
        if (e.target.closest('div') === titleBarDiv) {
            isDragging = true;
            initialX = e.type === 'mousedown' ? e.clientX - xOffset : e.touches[0].clientX - xOffset;
            initialY = e.type === 'mousedown' ? e.clientY - yOffset : e.touches[0].clientY - yOffset;
        }
    };

    const drag = (e) => {
        if (isDragging) {
            e.preventDefault();
            currentX = e.type === 'mousemove' ? e.clientX - initialX : e.touches[0].clientX - initialX;
            currentY = e.type === 'mousemove' ? e.clientY - initialY : e.touches[0].clientY - initialY;
            xOffset = currentX;
            yOffset = currentY;
            setTranslate(currentX, currentY, panel);
        }
    };

    const dragEnd = () => {
        isDragging = false;
        initialX = currentX;
        initialY = currentY;
    };

    const setTranslate = (xPos, yPos, el) => {
        el.style.transform = `translate(${xPos}px, ${yPos}px)`;
    };

    titleBar.addEventListener('mousedown', dragStart);
    titleBar.addEventListener('touchstart', dragStart);
    document.addEventListener('mousemove', drag);
    document.addEventListener('touchmove', drag);
    document.addEventListener('mouseup', dragEnd);
    document.addEventListener('touchend', dragEnd);

    // ========== 6. 面板上 “关闭按钮” 的逻辑 ==========
    // 也要区分是用户主动点击,还是外部脚本设置
    let userManuallyHidePanel = false;
    const closeBtn = document.getElementById('chatgpt-helper-close-btn');
    closeBtn.addEventListener('click', () => {
        userManuallyHidePanel = true;
        panel.style.display = 'none';
    });

    // ========== 7. Access Token 工具 ==========
    const tokenDisplay = document.getElementById('chatgpt-helper-token-display');
    const fetchBtn = document.getElementById('chatgpt-helper-fetch-btn');
    const copyBtn = document.getElementById('chatgpt-helper-copy-btn');

    fetchBtn.onclick = function () {
        GM_xmlhttpRequest({
            method: "GET",
            url: "/api/auth/session",
            onload: function (response) {
                try {
                    const data = JSON.parse(response.responseText);
                    tokenDisplay.value = data.accessToken || '获取失败';
                } catch (e) {
                    tokenDisplay.value = '获取失败:' + e.message;
                }
            },
            onerror: function () {
                tokenDisplay.value = '网络错误,请重试';
            }
        });
    };

    copyBtn.onclick = function () {
        if (tokenDisplay.value) {
            GM_setClipboard(tokenDisplay.value);
            createToast('Access Token已复制到剪贴板');
        }
    };

    // ========== 8. 服务降级检测 & fetch Hook  ==========
    const updateDifficultyIndicator = (difficulty) => {
        const difficultyLevel = document.getElementById('difficulty-level');
        const ipQuality = document.getElementById('ip-quality');
        if (difficulty === 'N/A') {
            difficultyLevel.innerText = '';
            ipQuality.innerHTML = 'N/A';
            return;
        }
        const cleanDifficulty = difficulty.replace('0x', '').replace(/^0+/, '');
        const hexLength = cleanDifficulty.length;
        let level, qualityText, color;
        if (hexLength <= 2) {
            level = '(困难)';
            qualityText = '高风险';
            color = '#FF0000'; // 红色
            updateRiskIndicator(color); // 更新小圆点颜色
        } else if (hexLength === 3) {
            level = '(中等)';
            qualityText = '中等';
            color = '#FFA500'; // 橙色
            updateRiskIndicator(color); // 更新小圆点颜色
        } else if (hexLength === 4) {
            level = '(简单)';
            qualityText = '良好';
            color = '#FFFF00'; // 黄色
            updateRiskIndicator(color); // 更新小圆点颜色
        } else {
            level = '(极易)';
            qualityText = '优秀';
            color = '#00FF00'; // 绿色
            updateRiskIndicator(color); // 更新小圆点颜色
        }
        difficultyLevel.innerHTML = `<span style="color: ${color}">${level}</span>`;
        ipQuality.innerHTML = `<span style="color: ${color}">${qualityText}</span>`;
    };

    function findChallengeElements() {
        const formFound = document.querySelector('form#challenge-form') !== null;
        const paragraphFound = document.querySelector('p#cf-spinner-please-wait, p#cf-spinner-redirecting') !== null;
        if (formFound && paragraphFound) {
            console.log('发现cf盾');
            return true;
        }
        return false;
    }

    if (!findChallengeElements()) {
        const originalFetch = myWindow.fetch;
        myWindow.fetch = async function (resource, options) {
            try {
                const response = await originalFetch(resource, options);
                const url = typeof resource === 'string' ? resource : resource.url;
                if (
                    url.includes('/backend-api/sentinel/chat-requirements') ||
                    url.includes('/backend-anon/sentinel/chat-requirements')
                ) {
                    const data = await response.clone().json();
                    const difficulty = data.proofofwork?.difficulty || 'N/A';
                    document.getElementById('difficulty').innerText = difficulty;
                    updateDifficultyIndicator(difficulty);
                }
                return response;
            } catch (e) {
                console.error('请求拦截时出错:', e);
                return originalFetch(resource, options);
            }
        };
        hookFix(originalFetch, 'fetch');
    }

    // =============== 9. MutationObserver:防止小球/面板被外部脚本移除 ===============
    const reAddBall = () => {
        if (!document.body.contains(ball)) {
            document.body.appendChild(ball);
            setBallStyle();
            console.log('[ChatGPT Helper] 小球被移除,已重新添加');
        }
    };
    const reAddPanel = () => {
        if (!document.body.contains(panel)) {
            document.body.appendChild(panel);
            console.log('[ChatGPT Helper] 面板被移除,已重新添加');
        }
    };

    const observer = new MutationObserver(() => {
        reAddBall();
        reAddPanel();
    });
    observer.observe(document.body, { childList: true });

    // =============== 10. MutationObserver:防止小球/面板被“强制隐藏” ===============
    // 仅在外部脚本偷偷改 style.display 时,才恢复。 若用户主动关闭面板,我们记录在 userManuallyHidePanel。
    const styleObserver = new MutationObserver((mutations) => {
        mutations.forEach(m => {
            if (m.target === panel && m.attributeName === 'style') {
                const disp = panel.style.display;
                if (disp === 'none' && !userManuallyHidePanel) {
                    // 说明这是外部脚本干的,我们强制恢复
                    panel.style.display = 'block';
                    positionPanel(); // 重新定位
                    console.log('[ChatGPT Helper] 外部脚本把面板设为none,已强制恢复显示');
                }
            }
            if (m.target === ball && m.attributeName === 'style') {
                // 确保小球可见
                setBallStyle();
            }
        });
    });
    styleObserver.observe(panel, { attributes: true, attributeFilter: ['style'] });
    styleObserver.observe(ball, { attributes: true, attributeFilter: ['style'] });

    // ========== 11. 定位面板在小球旁边 ==========
    const positionPanel = () => {
        const ballRect = ball.getBoundingClientRect();
        const panelRect = panel.getBoundingClientRect();
        // 将面板放在小球的左侧,稍微偏上
        const top = ballRect.top + window.scrollY;
        const left = ballRect.left - panelRect.width - 10 + window.scrollX; // 10px 间距
        panel.style.top = `${top}px`;
        panel.style.left = `${left}px`;
    };

    // 初次显示面板时定位
    // 以及在窗口大小变化或滚动时,重新定位面板
    window.addEventListener('resize', () => {
        if (panel.style.display === 'block') {
            positionPanel();
        }
    });

    window.addEventListener('scroll', () => {
        if (panel.style.display === 'block') {
            positionPanel();
        }
    });

})();

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址