// ==UserScript== // @name ChatGPT Model Switcher // @name:zh-CN ChatGPT Model Switcher // @name:zh-TW ChatGPT Model Switcher // @namespace https://github.com/hydrotho/ChatGPT_Model_Switcher // @copyright 2023, Hydrotho (https://github.com/hydrotho) // @version 1.0.1 // @description Override GPT-4 usage limits in the ChatGPT web interface by enabling the GPT-4 Mobile model. Additional models can also be enabled for switching, providing more flexibility. // @description:zh-CN 通过启用 GPT-4 Mobile 模型,解除 ChatGPT 网页端对 GPT-4 模型使用次数的限制。同时还可启用其他模型进行切换,提供更多的灵活性。 // @description:zh-TW 通过启用 GPT-4 Mobile 模型,解除 ChatGPT 网页端对 GPT-4 模型使用次数的限制。同时还可启用其他模型进行切换,提供更多的灵活性。 // @icon  // @grant none // @author Hydrotho // @match http*://chat.openai.com/* // @supportURL https://github.com/hydrotho/ChatGPT_Model_Switcher/issues // @license MIT // @downloadURL none // ==/UserScript== (function () { "use strict"; let useModelSwitcher = localStorage.getItem("useModelSwitcher") !== "false"; let selectedModel = localStorage.getItem("selectedModel") || "GPT-4 (Mobile)"; const modelMapping = { "GPT-3.5": "text-davinci-002-render-sha", "GPT-4": "gpt-4", "GPT-4 Web Browsing": "gpt-4-browsing", "GPT-4 Plugins": "gpt-4-plugins", "GPT-3.5 (Mobile)": "text-davinci-002-render-sha-mobile", "GPT-4 (Mobile)": "gpt-4-mobile", }; window.fetch = new Proxy(window.fetch, { apply: function (target, that, args) { let resource = args[0]; let options = args[1]; if ( useModelSwitcher && resource === "https://chat.openai.com/backend-api/conversation" ) { const requestBody = JSON.parse(options.body); requestBody.model = modelMapping[selectedModel]; options = { ...options, body: JSON.stringify(requestBody) }; args[0] = resource; args[1] = options; } const fetchPromise = Reflect.apply(target, that, args); if ( resource.includes( "https://chat.openai.com/backend-api/models?history_and_training_disabled=false" ) ) { return fetchPromise.then((response) => { if (response.ok) { response .clone() .json() .then((data) => { const accessibleModels = data.models.map((model) => model.slug); Object.keys(modelMapping).forEach((model) => { const mappedSlug = modelMapping[model]; const selectOption = document.querySelector( `#modelSelect option[value="${model}"]` ); if (selectOption && !accessibleModels.includes(mappedSlug)) { selectOption.disabled = true; if (selectedModel === model) { selectedModel = "GPT-3.5"; localStorage.setItem("selectedModel", selectedModel); document.querySelector("#modelSelect").value = selectedModel; } } }); }); } return response; }); } return fetchPromise; }, }); function createSwitchElement() { const switchLabel = document.createElement("label"); switchLabel.className = "switch"; switchLabel.title = "Check to enable the model switcher"; const switchCheckbox = document.createElement("input"); switchCheckbox.type = "checkbox"; switchCheckbox.id = "useModelSwitcherCheckbox"; switchCheckbox.checked = useModelSwitcher; switchCheckbox.addEventListener("change", (event) => { useModelSwitcher = event.target.checked; localStorage.setItem("useModelSwitcher", useModelSwitcher); }); const switchSlider = document.createElement("span"); switchSlider.className = "slider round"; switchLabel.appendChild(switchCheckbox); switchLabel.appendChild(switchSlider); return switchLabel; } function createModelSelectElement() { const selectContainer = document.createElement("div"); selectContainer.style.position = "relative"; const select = document.createElement("select"); select.id = "modelSelect"; select.addEventListener("change", (event) => { selectedModel = event.target.value; localStorage.setItem("selectedModel", selectedModel); }); for (const model in modelMapping) { const option = document.createElement("option"); option.text = model; option.value = model; select.appendChild(option); } select.value = selectedModel; const selectArrow = document.createElement("div"); selectArrow.style.cssText = ` position: absolute; top: 50%; right: 8px; transform: translateY(-50%); width: 12px; height: 12px; background-image: url('data:image/svg+xml,%3Csvg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" fill="%23333" width="18px" height="18px"%3E%3Cpath d="M7 10l5 5 5-5z"/%3E%3Cpath d="M0 0h24v24H0z" fill="none"/%3E%3C/svg%3E'); background-repeat: no-repeat; background-position: center; pointer-events: none; `; selectContainer.appendChild(select); selectContainer.appendChild(selectArrow); return selectContainer; } function createModelSwitcherContainer() { const container = document.createElement("div"); container.style.cssText = ` position: fixed; top: 10px; right: 18px; background-color: rgb(32, 33, 35); border: 1px solid #ddd; padding: 10px; border-radius: 5px; z-index: 9999; transition: 0.3s; box-shadow: 0 0 10px rgba(0, 0, 0, 0.2); display: flex; align-items: center; opacity: 0.5; `; container.addEventListener("mouseenter", () => { container.style.opacity = "1"; }); container.addEventListener("mouseleave", () => { container.style.opacity = "0.5"; }); const switchElement = createSwitchElement(); const modelSelectElement = createModelSelectElement(); container.appendChild(switchElement); container.appendChild(modelSelectElement); return container; } const container = createModelSwitcherContainer(); document.body.appendChild(container); const style = document.createElement("style"); style.textContent = ` .switch { position: relative; display: inline-block; width: 40px; height: 20px; margin-right: 10px; } .switch input { opacity: 0; width: 0; height: 0; } .slider { position: absolute; cursor: pointer; top: 0; left: 0; right: 0; bottom: 0; background-color: #ccc; transition: .5s; border-radius: 35px; } .slider:before { position: absolute; content: ""; height: 16px; width: 16px; left: 2px; bottom: 2px; background-color: white; transition: .5s; border-radius: 50%; } input:checked + .slider { background-color: #2196F3; } input:focus + .slider { box-shadow: 0 0 1px #2196F3; } input:checked + .slider:before { transform: translateX(20px); } .slider.round { border-radius: 35px; } .slider.round:before { border-radius: 50%; } select { color: #000000; background-color: #ffffff; padding: 5px; border: none; border-radius: 5px; appearance: none; -webkit-appearance: none; -moz-appearance: none; background-image: url('data:image/svg+xml,%3Csvg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" fill="%23333" width="18px" height="18px"%3E%3Cpath d="M7 10l5 5 5-5z"/%3E%3Cpath d="M0 0h24v24H0z" fill="none"/%3E%3C/svg%3E'); background-repeat: no-repeat; background-position: right center; padding-right: 20px; text-overflow: ellipsis; white-space: nowrap; overflow: hidden; } `; document.head.appendChild(style); })();