更好看的PoW检测(解除降智)脚本

感觉之前的PoW检测脚本不太美观 根据F-Droid的源码改了一版,并且加入了伪装UA的功能。

v1.6 修复在GPTs页面指示器依旧存在的问题 [已知问题]套餐购买界面依旧未修复 暂时没有好办法
v1.5 将指示器移动到输入框的右下角,防止和最大化输出按钮冲突
v1.4 修正指示器触发范围过大的问题 修正指示器缩放动画溢出的问题 增加更优雅的动画
v1.3 更新深色模式支持

一键导入

// ==UserScript==
// @name         ChatGPT PoW 显示助手
// @namespace    https://linux.do/u/zgccrui
// @version      1.6
// @description  实时显示PoW难度值及模拟手机UA。
// @license      GNU Affero General Public License v3.0 or later
// @author       zgccrui
// @match        https://chatgpt.com/*
// @icon         https://registry.npmmirror.com/@lobehub/icons-static-svg/latest/files/icons/openai.svg
// @grant        none
// @run-at       document-start
// ==/UserScript==

// 本脚本包含以下开源脚本的内容,这些脚本同样遵循GNU Affero General Public License v3.0 or later:
// - 脚本名称:ChatGPT Helper,作者:F-Droid

(function () {
    'use strict';
    /*** 模拟手机UA部分 ***/
    // 函数:应用模拟手机UA设置
    function applyMobileUASimulation() {
        // 定义iPhone 14 Pro Max的User-Agent字符串
        const mobileUA = "Mozilla/5.0 (iPhone; CPU iPhone OS 16_0 like Mac OS X) " +
            "AppleWebKit/605.1.15 (KHTML, like Gecko) Version/16.0 Mobile/15E148 Safari/604.1";
        // 覆盖 navigator.userAgent
        Object.defineProperty(navigator, 'userAgent', {
            get: function () { return mobileUA; },
            configurable: false
        });
        // 覆盖 navigator.platform
        Object.defineProperty(navigator, 'platform', {
            get: function () { return 'iPhone'; },
            configurable: false
        });
        // 覆盖 devicePixelRatio
        Object.defineProperty(window, 'devicePixelRatio', {
            get: function () { return 3; }, // iPhone 14 Pro Max 的 devicePixelRatio
            configurable: false
        });
        // 覆盖 touch 事件支持
        window.ontouchstart = function () { };
        // 模拟触摸事件
        Object.defineProperty(navigator, 'maxTouchPoints', {
            get: function () { return 5; }, // iPhone 14 Pro Max 支持多点触控
            configurable: false
        });
        // 覆盖 window.matchMedia 以支持移动设备的媒体查询
        const originalMatchMedia = window.matchMedia;
        window.matchMedia = function (query) {
            const mql = originalMatchMedia.call(this, query);
            if (query.includes('(pointer: coarse)') || query.includes('(max-width: 428px)')) {
                return {
                    matches: true,
                    media: query,
                    onchange: null,
                    addListener: function () { },
                    removeListener: function () { },
                    addEventListener: function () { },
                    removeEventListener: function () { },
                    dispatchEvent: function () { return false; }
                };
            }
            return mql;
        };
        // 触发 resize 和 orientationchange 事件以确保页面适应新尺寸
        window.dispatchEvent(new Event('resize'));
        window.dispatchEvent(new Event('orientationchange'));
    }
    // 检查本地存储中是否启用了模拟手机UA
    const isMobileUASimulated = localStorage.getItem('simulateMobileUA') === 'true';
    if (isMobileUASimulated) {
        applyMobileUASimulation();
    }
    /*** PoW Display Helper 部分 ***/
    // 等待 DOM 加载完成后执行
    window.addEventListener('DOMContentLoaded', () => {
        // 创建样式
        const style = document.createElement('style');
        style.textContent = `
            #pow-display {
                position: absolute;
                padding: 4px;
                border-radius: 12px;
                font-family: 'Inter', system-ui, -apple-system, sans-serif;
                font-size: 14px;
                font-weight: 500;
                z-index: 10000;
                display: flex;
                flex-direction: column;
                align-items: flex-end;
                cursor: pointer;
                transition: all 0.4s cubic-bezier(0.4, 0, 0.2, 1);
                background: none;
                border: 1px solid rgba(230, 230, 230, 0);
                right: 10px;
                color: #333; /* 默认文本颜色 */
            }
            #pow-display:hover {
                background: rgba(255, 255, 255, 0.95);
                border: 1px solid rgba(230, 230, 230, 0.8);
                box-shadow: 0 4px 12px rgba(0, 0, 0, 0.1);
                transform: translateY(-2px);
                padding: 10px 16px;
                backdrop-filter: blur(8px);
            }
            #pow-indicator-wrapper {
                display: flex;
                align-items: center;
                justify-content: flex-end;
                width: 100%;
                overflow: visible;
            }
            #pow-indicator {
                width: 10px;
                height: 10px;
                border-radius: 50%;
                transition: all 0.4s ease;
                box-shadow: 0 2px 4px rgba(0, 0, 0, 0.15);
                flex-shrink: 0;
                margin-left: 8px;
            }
            #pow-text {
                white-space: nowrap;
                transition: all 0.4s ease;
                opacity: 0;
                max-width: 0;
                overflow: hidden;
            }
            #pow-display:hover #pow-text,
            #pow-display.expanded #pow-text {
                opacity: 1;
                max-width: 300px;
            }
            #pow-display.updating #pow-indicator {
                animation: pulse 1.2s cubic-bezier(0.4, 0, 0.6, 1) infinite;
            }
            @keyframes pulse {
                0%, 100% {
                    transform: scale(1);
                    opacity: 1;
                }
                50% {
                    transform: scale(1.3);
                    opacity: 0.7;
                }
            }
            @keyframes fadeIn {
                from {
                    opacity: 0;
                    transform: translateY(10px);
                }
                to {
                    opacity: 1;
                    transform: translateY(0);
                }
            }
            #pow-display {
                animation: fadeIn 0.5s ease-out;
            }
            /* 修改 pow-ua-container 的样式 */
            #pow-ua-container {
                max-height: 0;
                opacity: 0;
                overflow: hidden;
                flex-direction: column;
                gap: 8px;
                margin-top: 0;
                padding-top: 0;
                border-top: 1px solid rgba(230, 230, 230, 0);
                font-size: 12px;
                width: 0; /* 初始宽度为0 */
                transition: all 0.4s ease-out;
                white-space: nowrap; /* 防止文字换行 */
                overflow-x: hidden; /* 隐藏超出部分 */
                color: #555;
            }

            /* 悬停时显示 UA 容器 */
            #pow-display:hover #pow-ua-container,
            #pow-display.expanded #pow-ua-container {
                max-height: 100px;
                opacity: 1;
                margin-top: 8px;
                padding-top: 8px;
                width: 100%; /* 悬停时宽度为100% */
                transition-delay: 0s; /* 悬停时立即开始过渡 */
                border-top: 1px solid rgba(230, 230, 230, 0.8);
            }

            /* 优化 pow-ua-wrapper 的过渡效果 */
            #pow-ua-wrapper {
                display: flex;
                align-items: center;
                gap: 8px;
                transform: translateY(0);
                transition: transform 0.4s ease-out;
            }

            /* UA提示文本的过渡效果 */
            #pow-ua-hint {
                color: #666;
                font-size: 11px;
                margin-top: 4px;
                opacity: 0;
                transform: translateY(-5px);
                transition: all 0.4s ease-out;
            }

            #pow-ua-hint.visible {
                opacity: 1;
                transform: translateY(0);
            }
            /* 删除当前UA显示样式
            #current-ua {
                font-size: 11px;
                color: #666;
                word-break: break-all;
                margin-top: 4px;
            }
            */
            #pow-display:hover #pow-ua-container,
            #pow-display.expanded #pow-ua-container {
                display: flex;
            }
            /* 美化复选框 */
            .custom-checkbox {
                display: inline-block;
                position: relative;
                padding-left: 25px;
                cursor: pointer;
                font-size: 12px;
                user-select: none;
                color: inherit; /* 继承父元素的文本颜色 */
            }
            .custom-checkbox input {
                position: absolute;
                opacity: 0;
                cursor: pointer;
                height: 0;
                width: 0;
            }
            .checkmark {
                position: absolute;
                top: 0;
                left: 0;
                height: 18px;
                width: 18px;
                background-color: #eee;
                border-radius: 4px;
                transition: all 0.4s ease;
            }
            .custom-checkbox:hover input ~ .checkmark {
                background-color: #ccc;
            }
            .custom-checkbox input:checked ~ .checkmark {
                background-color: #2196F3;
            }
            .checkmark:after {
                content: "";
                position: absolute;
                display: none;
            }
            .custom-checkbox input:checked ~ .checkmark:after {
                display: block;
            }
            .custom-checkbox .checkmark:after {
                left: 6px;
                top: 2px;
                width: 5px;
                height: 10px;
                border: solid white;
                border-width: 0 2px 2px 0;
                transform: rotate(45deg);
            }

            /* 深色模式样式 */
            @media (prefers-color-scheme: dark) {
                #pow-display {
                    background: none;
                    border: none;
                    color: #f0f0f0;
                    border: 1px solid rgba(70, 70, 70, 0);
                }
                #pow-display:hover {
                    background: rgba(40, 40, 40, 0.95); /* 深色背景 */
                    border: 1px solid rgba(70, 70, 70, 0.8); /* 深灰边框 */
                    box-shadow: 0 4px 12px rgba(0, 0, 0, 0.5); /* 更深的阴影以适应深色背景 */
                }
                #pow-ua-container {
                    border-top: none; /* 移除边框 */
                    color: #ddd;
                }
                .custom-checkbox {
                    color: #f0f0f0;
                }
                .checkmark {
                    background-color: #555;
                }
                .custom-checkbox:hover input ~ .checkmark {
                    background-color: #666;
                }
                .custom-checkbox input:checked ~ .checkmark {
                    background-color: #1e90ff;
                }
                #pow-ua-hint {
                    color: #bbb;
                }
            }
        `;
        // 添加样式到页面
        document.head.appendChild(style);
        // 创建显示元素
        const powDisplay = document.createElement('div');
        powDisplay.id = 'pow-display';
        const indicatorWrapper = document.createElement('div');
        indicatorWrapper.id = 'pow-indicator-wrapper';
        const textContent = document.createElement('span');
        textContent.id = 'pow-text';
        const indicator = document.createElement('div');
        indicator.id = 'pow-indicator';
        indicatorWrapper.appendChild(textContent);
        indicatorWrapper.appendChild(indicator);
        powDisplay.appendChild(indicatorWrapper);
        // 创建复选框容器
        const uaContainer = document.createElement('div');
        uaContainer.id = 'pow-ua-container';
        const uaWrapper = document.createElement('div');
        uaWrapper.id = 'pow-ua-wrapper';
        const uaLabel = document.createElement('label');
        uaLabel.className = 'custom-checkbox';
        uaLabel.innerHTML = `
            模拟手机UA
            <input type="checkbox" id="pow-ua-checkbox">
            <span class="checkmark"></span>
        `;
        const uaHint = document.createElement('div');
        uaHint.id = 'pow-ua-hint';
        uaHint.style.display = 'none';
        uaHint.innerText = '✻ 需要刷新页面生效';
        // 改为不添加 currentUA 元素
        uaWrapper.appendChild(uaLabel);
        uaContainer.appendChild(uaWrapper);
        uaContainer.appendChild(uaHint);
        powDisplay.appendChild(uaContainer);
        document.body.appendChild(powDisplay);
        // 更新显示函数
        const updateDifficultyIndicator = (difficulty) => {
            const powDisplay = document.getElementById('pow-display');
            const indicator = document.getElementById('pow-indicator');
            const textContent = document.getElementById('pow-text');
            powDisplay.classList.add('updating');
            setTimeout(() => powDisplay.classList.remove('updating'), 1200);
            if (difficulty === 'N/A') {
                textContent.innerText = 'PoW难度: 等待中...';
                indicator.style.backgroundColor = '#95a5a6';
                textContent.style.color = '#95a5a6';
                powDisplay.classList.remove('expanded'); // 重置展开状态
                return;
            }
            const cleanDifficulty = difficulty.replace('0x', '').replace(/^0+/, '');
            const hexLength = cleanDifficulty.length;
            let level, color;
            if (hexLength <= 2) {
                level = '困难';
                color = '#e74c3c';
            } else if (hexLength === 3) {
                level = '中等';
                color = '#f39c12';
            } else if (hexLength === 4) {
                level = '简单';
                color = '#3498db';
            } else {
                level = '极易';
                color = '#27ae60';
            }
            textContent.innerHTML = `PoW难度: ${difficulty} (${level})`;
            indicator.style.backgroundColor = color;
            textContent.style.color = color;
            // 根据难度决定是否默认展开
            if (level !== '极易') {
                powDisplay.classList.add('expanded');
            } else {
                powDisplay.classList.remove('expanded');
            }
        };
        // 拦截Fetch请求
        const originalFetch = window.fetch;
        window.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';
                    updateDifficultyIndicator(difficulty);
                }
                return response;
            } catch (e) {
                console.error('请求拦截时出错:', e);
                return originalFetch(resource, options);
            }
        };

        // 更新显示位置的函数
        const updatePosition = () => {
            const target = document.getElementById('composer-background');
            if (target) {
                const rect = target.getBoundingClientRect();
                const scrollX = window.scrollX || window.pageXOffset;
                const scrollY = window.scrollY || window.pageYOffset;
                // 设置内部右下角
                powDisplay.style.position = 'absolute';
                powDisplay.style.top = '';
                powDisplay.style.right = `${document.documentElement.clientWidth - (rect.right + scrollX) + 50}px`;
                powDisplay.style.bottom = `${document.documentElement.clientHeight - (rect.bottom + scrollY) + 8}px`;
                powDisplay.style.left = '';
            }
        };

        // 使用 requestAnimationFrame 来持续更新位置
        const raf = () => {
            updatePosition();
            requestAnimationFrame(raf);
        };
        // 初始化并监听位置变化
        const waitForElement = (selector, timeout = 10000) => {
            return new Promise((resolve, reject) => {
                const element = document.querySelector(selector);
                if (element) {
                    return resolve(element);
                }
                const observer = new MutationObserver((mutations, obs) => {
                    const el = document.querySelector(selector);
                    if (el) {
                        obs.disconnect();
                        resolve(el);
                    }
                });
                observer.observe(document.body, {
                    childList: true,
                    subtree: true
                });
                setTimeout(() => {
                    observer.disconnect();
                    reject(new Error(`Element ${selector} not found within ${timeout}ms`));
                }, timeout);
            });
        };

        waitForElement('#composer-background')
            .then(target => {
                // 开始使用 requestAnimationFrame 持续更新位置
                raf();
                // 监听窗口事件以更新位置
                window.addEventListener('resize', updatePosition);
                window.addEventListener('scroll', updatePosition);
            })
            .catch(error => {
                console.error(error);
            });

        // 添加鼠标事件以控制展开状态
        powDisplay.addEventListener('mouseenter', () => {
            if (powDisplay.classList.contains('expanded')) {
                powDisplay.classList.remove('expanded');
            }
        });

        // 复选框事件监听
        const uaCheckbox = document.getElementById('pow-ua-checkbox');
        uaCheckbox.checked = isMobileUASimulated;
        uaCheckbox.addEventListener('change', () => {
            const isChecked = uaCheckbox.checked;
            const savedState = localStorage.getItem('simulateMobileUA') === 'true';
            const uaHint = document.getElementById('pow-ua-hint');

            if (isChecked !== savedState) {
                uaHint.style.display = 'block';
                // 使用 setTimeout 来确保 display: block 生效后再添加 visible 类
                setTimeout(() => {
                    uaHint.classList.add('visible');
                }, 10);
            } else {
                uaHint.classList.remove('visible');
                // 等待过渡效果完成后再隐藏元素
                setTimeout(() => {
                    uaHint.style.display = 'none';
                }, 300);
            }

            localStorage.setItem('simulateMobileUA', isChecked);
        });

        // 初始化显示
        updateDifficultyIndicator('N/A');


        // 添加以下代码以实现元素可见性检测
        function checkElementVisibility() {
            const target = document.getElementById('composer-background');
            const powDisplay = document.getElementById('pow-display');
            if (!target || target.offsetParent === null) { // offsetParent 为 null 表示元素隐藏
                powDisplay.style.display = 'none';
            } else {
                powDisplay.style.display = 'flex'; // 或根据您的布局选择适当的显示方式
            }
        }

        // 初始检查
        checkElementVisibility();

        // 使用 MutationObserver 监听 DOM 变化
        const observer = new MutationObserver(() => {
            checkElementVisibility();
        });

        // 开始观察整个文档的变化
        observer.observe(document.body, {
            childList: true,
            subtree: true,
            attributes: true,
            attributeFilter: ['style', 'class'] // 监听可能影响可见性的属性变化
        });

        // 可选:在页面卸载时断开观察器
        window.addEventListener('beforeunload', () => {
            observer.disconnect();
        });
    });
})();

20241212173834_rec_

:tada:一站式ChatGPT增强工具ChatGPT Helper来了!支持检测降智等功能 - 资源荟萃 - LINUX DO

【更新条件】解除OAI降智脚本 - 搞七捻三 / 搞七捻三, Lv1 - LINUX DO

89 个赞

好看,无感,支持

绿点要不要和对话框R角对齐一下

打开模拟手机UA这个开关后对话框的小绿点就不见了,是bug吗 :tieba_009:

1 个赞

深色模式背景寄了

啊 不会啊 我这边开模拟UA正常啊

3 个赞

深色模式我没测试 :melting_face:

要加个prefer

2 个赞

刷新几次只会偶尔出现一次,可能是我深色模式的原因?

localstorage 里面有个值 theme,为 dark 就是 dark,light 就是 light,system 是跟随系统,然后不存在则默认跟随系统

我这边改下看看

1 个赞

感谢分享!!

感谢分享

感谢大佬分享

更新了下 现在深色模式应该正常了

1 个赞

谢谢佬,太好看了

好活当赏,话说模拟手机 UA 是可以防止降智吗?
而且原版有两个参数吧

2 个赞

这俩参数是一样的 都是PoW判断的

1 个赞

好活当赏

4 个赞

记得更新一下一键导入链接tieba_036

更新了 应该是因为CDN的原因 导致有缓存