您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Injects a button allowing you to toggle on/off 4o-mini during the chat
// ==UserScript== // @name ChatGPT Model Switcher: Toggle on/off 4o-mini Improved Version // @namespace http://tampermonkey.net/ // @version 0.24.1 // @description Injects a button allowing you to toggle on/off 4o-mini during the chat // @match *://chatgpt.com/* // @author d0gkiller87, improved UI by Aoshi Xu // @license MIT // @grant unsafeWindow // @grant GM.setValue // @grant GM.getValue // @run-at document-idle // ==/UserScript== (async function() { 'use strict'; class ModelSwitcher { constructor( useMini = true ) { this.useMini = useMini; this.containerSelector = '#composer-background div:nth-of-type(2) div:first-child'; } hookFetch() { const originalFetch = unsafeWindow.fetch; unsafeWindow.fetch = async ( resource, config = {} ) => { if ( resource === 'https://chatgpt.com/backend-api/conversation' && config.method === 'POST' && config.headers && config.headers['Content-Type'] === 'application/json' && config.body ) { if ( this.useMini ) { const body = JSON.parse( config.body ); body.model = 'gpt-4o-mini'; config.body = JSON.stringify( body ); } } return originalFetch( resource, config ); }; } injectToggleButtonStyle() { // Credit: https://webdevworkshop.io/code/css-toggle-button/ if ( !document.getElementById( 'toggleCss' ) ) { const styleNode = document.createElement( 'style' ); styleNode.id = 'toggleCss'; styleNode.type = 'text/css'; styleNode.textContent = `.toggle { position: relative; display: inline-block; width: 2.5rem; height: 1.5rem; background-color: hsl(0deg 0% 40%); border-radius: 25px; cursor: pointer; transition: background-color 0.2s ease-in; } .toggle::after { content: ''; position: absolute; width: 1.4rem; left: 0.1rem; height: calc(1.5rem - 2px); top: 1px; background-color: white; border-radius: 50%; transition: all 0.2s ease-out; } .hide-me { opacity: 0; height: 0; width: 0; }`; document.head.appendChild( styleNode ); } } getContainer() { return document.querySelector( this.containerSelector ); } injectToggleButton( container = null ) { console.log( 'inject' ); if ( !container ) container = this.getContainer(); if ( !container ) { console.error( 'container not found!' ); return; } if ( container.querySelector( '#cb-toggle' ) ) { console.log( '#cb-toggle already exists' ); return; } container.classList.add('items-center'); const button = document.createElement('button'); button.id = 'cb-toggle'; button.className = 'flex h-8 min-w-8 items-center justify-center rounded-lg p-1 text-xs font-semibold hover:bg-black/10 focus-visible:outline-black dark:focus-visible:outline-white'; button.setAttribute('aria-pressed', 'false'); button.setAttribute('aria-label', 'Toggle GPT-4o Mini'); button.innerHTML = `<svg width='24' height='24' viewBox='0 0 320 320' xmlns='http://www.w3.org/2000/svg'><path d='m297.06 130.97c7.26-21.79 4.76-45.66-6.85-65.48-17.46-30.4-52.56-46.04-86.84-38.68-15.25-17.18-37.16-26.95-60.13-26.81-35.04-.08-66.13 22.48-76.91 55.82-22.51 4.61-41.94 18.7-53.31 38.67-17.59 30.32-13.58 68.54 9.92 94.54-7.26 21.79-4.76 45.66 6.85 65.48 17.46 30.4 52.56 46.04 86.84 38.68 15.24 17.18 37.16 26.95 60.13 26.8 35.06.09 66.16-22.49 76.94-55.86 22.51-4.61 41.94-18.7 53.31-38.67 17.57-30.32 13.55-68.51-9.94-94.51zm-120.28 168.11c-14.03.02-27.62-4.89-38.39-13.88.49-.26 1.34-.73 1.89-1.07l63.72-36.8c3.26-1.85 5.26-5.32 5.24-9.07v-89.83l26.93 15.55c.29.14.48.42.52.74v74.39c-.04 33.08-26.83 59.9-59.91 59.97zm-128.84-55.03c-7.03-12.14-9.56-26.37-7.15-40.18.47.28 1.3.79 1.89 1.13l63.72 36.8c3.23 1.89 7.23 1.89 10.47 0l77.79-44.92v31.1c.02.32-.13.63-.38.83l-64.41 37.19c-28.69 16.52-65.33 6.7-81.92-21.95zm-16.77-139.09c7-12.16 18.05-21.46 31.21-26.29 0 .55-.03 1.52-.03 2.2v73.61c-.02 3.74 1.98 7.21 5.23 9.06l77.79 44.91-26.93 15.55c-.27.18-.61.21-.91.08l-64.42-37.22c-28.63-16.58-38.45-53.21-21.95-81.89zm221.26 51.49-77.79-44.92 26.93-15.54c.27-.18.61-.21.91-.08l64.42 37.19c28.68 16.57 38.51 53.26 21.94 81.94-7.01 12.14-18.05 21.44-31.2 26.28v-75.81c.03-3.74-1.96-7.2-5.2-9.06zm26.8-40.34c-.47-.29-1.3-.79-1.89-1.13l-63.72-36.8c-3.23-1.89-7.23-1.89-10.47 0l-77.79 44.92v-31.1c-.02-.32.13-.63.38-.83l64.41-37.16c28.69-16.55 65.37-6.7 81.91 22 6.99 12.12 9.52 26.31 7.15 40.1zm-168.51 55.43-26.94-15.55c-.29-.14-.48-.42-.52-.74v-74.39c.02-33.12 26.89-59.96 60.01-59.94 14.01 0 27.57 4.92 38.34 13.88-.49.26-1.33.73-1.89 1.07l-63.72 36.8c-3.26 1.85-5.26 5.31-5.24 9.06l-.04 89.79zm14.63-31.54 34.65-20.01 34.65 20v40.01l-34.65 20-34.65-20z'/></svg>` button.style.backgroundColor = this.useMini ? 'hsl(102, 58%, 39%)' : 'transparent'; button.addEventListener('click', () => { this.useMini = !this.useMini; console.log( `useMini: ${this.useMini}` ); GM.setValue('useMini', this.useMini); button.setAttribute('aria-pressed', this.useMini); button.title = `Using model: ${this.useMini ? 'GPT-4o mini' : 'original'}`; button.style.backgroundColor = this.useMini ? 'hsl(102, 58%, 39%)' : 'transparent'; }); container.appendChild(button); } monitorChild( nodeSelector, callback ) { const node = document.querySelector( nodeSelector ); if ( !node ) { console.log( `${ nodeSelector } not found!` ) return; } const observer = new MutationObserver( mutationsList => { for ( const mutation of mutationsList ) { console.log( nodeSelector ); callback( observer, mutation ); break; } }); observer.observe( node, { childList: true } ); } __tagAttributeRecursively(selector) { // Select the node using the provided selector const rootNode = document.querySelector(selector); if (!rootNode) { console.warn(`No element found for selector: ${selector}`); return; } // Recursive function to add the "xx" attribute to the node and its children function addAttribute(node) { node.setAttribute("xxx", ""); // Add the attribute to the current node Array.from(node.children).forEach(addAttribute); // Recurse for all child nodes } addAttribute(rootNode); } monitorNodesAndInject() { this.monitorChild( 'body main', () => { this.injectToggleButton(); this.monitorChild( 'main div:first-child div:first-child', ( observer, mutation ) => { observer.disconnect(); this.injectToggleButton(); }); }); this.monitorChild( this.containerSelector, ( observer, mutation ) => { observer.disconnect(); setTimeout( () => this.injectToggleButton(), 500 ); }); this.monitorChild( 'main div:first-child div:first-child', ( observer, mutation ) => { observer.disconnect(); this.injectToggleButton(); }); } } const useMini = await GM.getValue( 'useMini', true ); const switcher = new ModelSwitcher( useMini ); switcher.hookFetch(); switcher.injectToggleButtonStyle(); switcher.monitorNodesAndInject(); })();