- // ==UserScript==
- // @name ChatGPT Auto-Continue 🔄
- // @description ⚡ Automatically click the 'Continue Generating' button in ChatGPT, handling errors!
- // @author mefengl
- // @version 1.1.18
- // @namespace https://github.com/mefengl
- // @icon https://www.google.com/s2/favicons?sz=64&domain=openai.com
- // @license MIT
- // @match https://chatgpt.com/*
- // @grant none
-
- // @name:en ChatGPT Auto-Continue 🔄
- // @description:en ⚡ Automatically click the 'Continue Generating' button in ChatGPT, handling errors!
- // @name:zh-CN ChatGPT 自动继续 🔄
- // @description:zh-CN ⚡ 自动点击ChatGPT中的“继续生成”按钮,处理错误!
- // @name:es ChatGPT Auto-Continuar 🔄
- // @description:es ⚡ ¡Haz clic automáticamente en el botón 'Continuar generando' en ChatGPT, manejando errores!
- // @name:hi ChatGPT स्वचालित जारी रखें 🔄
- // @description:hi ⚡ ChatGPT में 'जारी रखने' बटन पर स्वचालित रूप से क्लिक करें, त्रुटियों को संभालते हुए!
- // @name:ar ChatGPT التكميل الآلي 🔄
- // @description:ar ⚡ انقر تلقائيًا على زر 'متابعة التوليد' في ChatGPT ، معالجة الأخطاء!
- // @name:pt ChatGPT Auto-Continuar 🔄
- // @description:pt ⚡ Clique automaticamente no botão 'Continuar Gerando' no ChatGPT, tratando erros!
- // @name:ru ChatGPT Авто-Продолжение 🔄
- // @description:ru ⚡ Автоматически нажимайте на кнопку "Продолжить генерацию" в ChatGPT, обрабатывая ошибки!
- // @name:ja ChatGPT 自動続行 🔄
- // @description:ja ⚡ ChatGPTの「続けて生成」ボタンを自動的にクリックし、エラーを処理します!
- // @name:de ChatGPT Auto-Fortsetzen 🔄
- // @description:de ⚡ Klicken Sie automatisch auf die Schaltfläche "Generierung fortsetzen" in ChatGPT, Fehler behandeln!
- // @name:fr ChatGPT Auto-Continuer 🔄
- // @description:fr ⚡ Cliquez automatiquement sur le bouton 'Continuer à générer' dans ChatGPT, gérer les erreurs!
- // ==/UserScript==
- "use strict";
- (() => {
- var __create = Object.create;
- var __defProp = Object.defineProperty;
- var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
- var __getOwnPropNames = Object.getOwnPropertyNames;
- var __getProtoOf = Object.getPrototypeOf;
- var __hasOwnProp = Object.prototype.hasOwnProperty;
- var __commonJS = (cb, mod) => function __require() {
- return mod || (0, cb[__getOwnPropNames(cb)[0]])((mod = { exports: {} }).exports, mod), mod.exports;
- };
- var __copyProps = (to, from, except, desc) => {
- if (from && typeof from === "object" || typeof from === "function") {
- for (let key of __getOwnPropNames(from))
- if (!__hasOwnProp.call(to, key) && key !== except)
- __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
- }
- return to;
- };
- var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
- // If the importer is in node compatibility mode or this is not an ESM
- // file that has been converted to a CommonJS file using a Babel-
- // compatible transform (i.e. "__esModule" has not been set), then set
- // "default" to the CommonJS "module.exports" for node compatibility.
- isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
- mod
- ));
- var __async = (__this, __arguments, generator) => {
- return new Promise((resolve, reject) => {
- var fulfilled = (value) => {
- try {
- step(generator.next(value));
- } catch (e) {
- reject(e);
- }
- };
- var rejected = (value) => {
- try {
- step(generator.throw(value));
- } catch (e) {
- reject(e);
- }
- };
- var step = (x) => x.done ? resolve(x.value) : Promise.resolve(x.value).then(fulfilled, rejected);
- step((generator = generator.apply(__this, __arguments)).next());
- });
- };
-
- // ../../packages/chatkit/dist/chatgpt/index.js
- var require_chatgpt = __commonJS({
- "../../packages/chatkit/dist/chatgpt/index.js"(exports, module) {
- "use strict";
- var __defProp2 = Object.defineProperty;
- var __getOwnPropDesc2 = Object.getOwnPropertyDescriptor;
- var __getOwnPropNames2 = Object.getOwnPropertyNames;
- var __hasOwnProp2 = Object.prototype.hasOwnProperty;
- var __export = (target, all) => {
- for (var name in all)
- __defProp2(target, name, { get: all[name], enumerable: true });
- };
- var __copyProps2 = (to, from, except, desc) => {
- if (from && typeof from === "object" || typeof from === "function") {
- for (let key of __getOwnPropNames2(from))
- if (!__hasOwnProp2.call(to, key) && key !== except)
- __defProp2(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc2(from, key)) || desc.enumerable });
- }
- return to;
- };
- var __toCommonJS = (mod) => __copyProps2(__defProp2({}, "__esModule", { value: true }), mod);
- var chatgpt_exports = {};
- __export(chatgpt_exports, {
- clickFollowUpButton: () => clickFollowUpButton,
- getButton: () => getButton,
- getContinueGeneratingButton: () => getContinueGeneratingButton2,
- getConversation: () => getConversation,
- getCopyLinkButton: () => getCopyLinkButton,
- getFollowUpButtons: () => getFollowUpButtons,
- getHistoryBlockTitle: () => getHistoryBlockTitle,
- getHistoryBlocks: () => getHistoryBlocks,
- getHistoryBlocksWithTitle: () => getHistoryBlocksWithTitle,
- getInitialButtons: () => getInitialButtons,
- getLastResponse: () => getLastResponse,
- getLastResponseElement: () => getLastResponseElement,
- getModelSelectButton: () => getModelSelectButton,
- getNav: () => getNav,
- getNewModelSelectButtons: () => getNewModelSelectButtons,
- getRegenerateButton: () => getRegenerateButton2,
- getResponseElementHTMLs: () => getResponseElementHTMLs,
- getShareChatButton: () => getShareChatButton,
- getStopGeneratingButton: () => getStopGeneratingButton,
- getSubmitButton: () => getSubmitButton,
- getTextarea: () => getTextarea2,
- getTextareaValue: () => getTextareaValue,
- hasNewModelSelectButtons: () => hasNewModelSelectButtons,
- isConversationStarted: () => isConversationStarted,
- isGenerating: () => isGenerating2,
- isHorizontalConversation: () => isHorizontalConversation,
- onSend: () => onSend,
- regenerate: () => regenerate,
- send: () => send,
- sendArray: () => sendArray,
- setHorizontalConversation: () => setHorizontalConversation,
- setPromptListener: () => setPromptListener,
- setPureConversation: () => setPureConversation,
- setTextarea: () => setTextarea,
- waitForIdle: () => waitForIdle
- });
- module.exports = __toCommonJS(chatgpt_exports);
- function getNav() {
- return document.querySelector("nav");
- }
- function getHistoryBlocks() {
- const nav = getNav();
- if (!nav)
- return [];
- const result = Array.from(nav.querySelectorAll("ol")).map((ol) => ol.parentElement);
- return result;
- }
- function getHistoryBlockTitle(historyBlock) {
- var _a;
- return ((_a = historyBlock.querySelector("h3")) == null ? void 0 : _a.textContent) || "";
- }
- function getHistoryBlocksWithTitle() {
- const historyBlocks = getHistoryBlocks();
- const result = historyBlocks.map((historyBlock) => ({
- block: historyBlock,
- title: getHistoryBlockTitle(historyBlock)
- }));
- return result;
- }
- function getTextarea2() {
- const form = document.querySelector("form");
- if (!form)
- return;
- const textareas = form.querySelectorAll("textarea");
- const result = textareas[0];
- return result;
- }
- function getNewSubmitButton() {
- return document.querySelector('button[data-testid$="send-button"]');
- }
- function getSubmitButton() {
- if (getNewSubmitButton()) {
- return getNewSubmitButton();
- }
- const textarea = getTextarea2();
- if (!textarea)
- return;
- return textarea.nextElementSibling;
- }
- function getInitialButtons() {
- return Array.from(document.querySelectorAll('button[as="button"]')).filter((button) => button.querySelectorAll(".truncate").length === 2);
- }
- function getFollowUpButtons() {
- return Array.from(document.querySelectorAll('button[as="button"]')).filter((button) => {
- var _a;
- return (_a = button.textContent) == null ? void 0 : _a.trim().match(/[.!?]$/);
- });
- }
- function clickFollowUpButton(index) {
- const followUpButtons = getFollowUpButtons();
- if (followUpButtons.length === 0)
- return;
- if (index === void 0 || index < 0 || index >= followUpButtons.length) {
- index = Math.floor(Math.random() * followUpButtons.length);
- }
- followUpButtons[index].click();
- }
- function getButton(text) {
- const button = Array.from(document.querySelectorAll('button[data-testid$="button"]')).find((button2) => {
- var _a;
- return (_a = button2.getAttribute("data-testid")) == null ? void 0 : _a.includes(text);
- });
- if (button)
- return button;
- return Array.from(document.querySelectorAll('button[as="button"]')).find((button2) => {
- var _a;
- return (_a = button2.textContent) == null ? void 0 : _a.trim().toLowerCase().includes(text);
- });
- }
- function getRegenerateButton2() {
- return getButton("regenerate");
- }
- function getContinueGeneratingButton2() {
- const buttonInWideScreen = getButton("continue");
- if (buttonInWideScreen)
- return buttonInWideScreen;
- function getNthGenerationDescendants(element, generation) {
- const descendants = [];
- function findDescendants(node, currentDepth) {
- if (currentDepth === generation) {
- descendants.push(node);
- return;
- }
- node.childNodes.forEach((child) => findDescendants(child, currentDepth + 1));
- }
- findDescendants(element, 0);
- return descendants;
- }
- const form = document.querySelector("form");
- if (!form)
- return;
- const seventhGenerationDescendants = getNthGenerationDescendants(form, 7);
- if (seventhGenerationDescendants.length === 0 || seventhGenerationDescendants[0].nodeName !== "BUTTON")
- return;
- return seventhGenerationDescendants[0];
- }
- function getNewStopGeneratingButton() {
- const stopButtonNotLogin = document.querySelector('button[aria-label="Stop generating"]');
- const stopButton = document.querySelector('button[data-testid$="stop-button"]');
- return stopButtonNotLogin || stopButton;
- }
- function getStopGeneratingButton() {
- return getNewStopGeneratingButton() || getButton("stop");
- }
- function getResponseElementHTMLs() {
- return Array.from(document.querySelectorAll(".markdown")).map((m) => m.innerHTML);
- }
- function getLastResponseElement() {
- const responseElements = document.querySelectorAll(".group.w-full");
- return responseElements[responseElements.length - 1];
- }
- function getLastResponse() {
- const lastResponseElement = getLastResponseElement();
- if (!lastResponseElement)
- return;
- const lastResponse = lastResponseElement.textContent;
- return lastResponse;
- }
- function getTextareaValue() {
- var _a;
- return ((_a = getTextarea2()) == null ? void 0 : _a.value) || "";
- }
- function setTextarea(message) {
- const textarea = getTextarea2();
- if (!textarea)
- return;
- textarea.value = message;
- textarea.dispatchEvent(new Event("input", { bubbles: true }));
- }
- function send(message) {
- return __async(this, null, function* () {
- var _a;
- setTextarea(message);
- const textarea = getTextarea2();
- if (!textarea)
- return;
- (_a = getSubmitButton()) == null ? void 0 : _a.click();
- for (let i = 0; i < 10; i++) {
- if (isGenerating2()) {
- break;
- }
- yield new Promise((resolve) => setTimeout(resolve, 1e3));
- }
- });
- }
- function regenerate() {
- const regenerateButton = getRegenerateButton2();
- if (!regenerateButton)
- return;
- regenerateButton.click();
- }
- function onSend(callback) {
- const textarea = getTextarea2();
- if (!textarea)
- return;
- textarea.addEventListener("keydown", function(event) {
- if (event.key === "Enter" && !event.shiftKey) {
- callback();
- }
- });
- const sendButton = getSubmitButton();
- if (!sendButton)
- return;
- sendButton.addEventListener("mousedown", callback);
- }
- function isGenerating2() {
- var _a, _b;
- if (getNewStopGeneratingButton()) {
- return true;
- }
- return ((_b = (_a = getSubmitButton()) == null ? void 0 : _a.firstElementChild) == null ? void 0 : _b.childElementCount) === 3;
- }
- function waitForIdle() {
- return new Promise((resolve) => {
- const interval = setInterval(() => {
- if (!isGenerating2()) {
- clearInterval(interval);
- resolve();
- }
- }, 1e3);
- });
- }
- function sendArray(messages) {
- return __async(this, null, function* () {
- var _a, _b;
- let firstTime = true;
- const isLong = messages.length > 60;
- let stop = false;
- while (messages.length > 0 || stop) {
- stop = false;
- const waitTime = isLong && !document.hasFocus() ? 20 * 1e3 : 2e3;
- if (!firstTime) {
- yield new Promise((resolve) => setTimeout(resolve, waitTime));
- }
- firstTime = false;
- if (isGenerating2()) {
- continue;
- } else if (getContinueGeneratingButton2()) {
- (_a = getContinueGeneratingButton2()) == null ? void 0 : _a.click();
- stop = true;
- continue;
- } else if (getRegenerateButton2() && !getTextarea2()) {
- yield new Promise((resolve) => setTimeout(resolve, 10 * 1e3));
- (_b = getRegenerateButton2()) == null ? void 0 : _b.click();
- stop = true;
- continue;
- }
- if (messages.length === 0) {
- break;
- }
- yield send(messages.shift() || "");
- }
- });
- }
- function setPromptListener(key = "prompt_texts") {
- let last_trigger_time = +/* @__PURE__ */ new Date();
- if (location.href.includes("chatgpt.com")) {
- GM_addValueChangeListener(key, (name, old_value, new_value) => __async(this, null, function* () {
- if (+/* @__PURE__ */ new Date() - last_trigger_time < 500) {
- return;
- }
- last_trigger_time = +/* @__PURE__ */ new Date();
- setTimeout(() => __async(this, null, function* () {
- sendArray(new_value);
- GM_setValue(key, []);
- }), 0);
- }));
- }
- }
- function getConversation() {
- var _a, _b;
- return (_b = (_a = document.querySelector('div[class^="react-scroll-to-bottom"]')) == null ? void 0 : _a.firstChild) == null ? void 0 : _b.firstChild;
- }
- function getModelSelectButton() {
- const conversation = getConversation();
- if (!conversation)
- return;
- return Array.from(conversation.querySelectorAll("button")).find((button) => {
- var _a;
- return (_a = button.textContent) == null ? void 0 : _a.trim().toLowerCase().includes("model");
- });
- }
- function getNewModelSelectButtons() {
- return Array.from(document.querySelectorAll("[class^='group/button']"));
- }
- function hasNewModelSelectButtons() {
- return getNewModelSelectButtons().length > 0;
- }
- function isConversationStarted() {
- return !getModelSelectButton();
- }
- function setPureConversation() {
- const conversation = getConversation();
- if (!conversation)
- return;
- const firstChild = conversation.firstChild;
- if (!firstChild)
- return;
- const newDiv = document.createElement("div");
- conversation.insertBefore(newDiv, firstChild.nextSibling);
- }
- function isHorizontalConversation() {
- const conversation = getConversation();
- if (!conversation)
- return true;
- if (!isConversationStarted())
- return true;
- return conversation.classList.contains("grid");
- }
- function setHorizontalConversation() {
- if (isHorizontalConversation())
- return;
- setPureConversation();
- const conversation = getConversation();
- if (!conversation)
- return;
- conversation.classList.remove("flex", "flex-col", "items-center");
- conversation.classList.add("grid", "grid-cols-2", "place-items-center");
- }
- function getShareChatButton() {
- return document.querySelector('button[aria-label="Share chat"]');
- }
- function getCopyLinkButton() {
- return Array.from(document.querySelectorAll('button[as="button"]')).filter((button) => {
- var _a;
- return (_a = button.textContent) == null ? void 0 : _a.trim().toLowerCase().includes("copy link");
- })[0];
- }
- }
- });
-
- // ../../packages/chatkit/chatgpt.js
- var require_chatgpt2 = __commonJS({
- "../../packages/chatkit/chatgpt.js"(exports, module) {
- module.exports = require_chatgpt();
- }
- });
-
- // src/index.ts
- var import_chatgpt = __toESM(require_chatgpt2(), 1);
- var retryCount = 0;
- var maxRetries = 3;
- var lastRetryTime = null;
- function initialize() {
- return __async(this, null, function* () {
- yield new Promise((resolve) => window.addEventListener("load", resolve));
- yield new Promise((resolve) => setTimeout(resolve, 1e3));
- });
- }
- function main() {
- return __async(this, null, function* () {
- yield initialize();
- let firstTime = true;
- setInterval(() => __async(this, null, function* () {
- var _a, _b;
- const currentTime = (/* @__PURE__ */ new Date()).getTime();
- if (lastRetryTime && currentTime - lastRetryTime >= 5 * 60 * 1e3) {
- retryCount = 0;
- }
- while (true) {
- const waitTime = !document.hasFocus() ? 20 * 1e3 : 2e3;
- if (!firstTime) {
- yield new Promise((resolve) => setTimeout(resolve, waitTime));
- }
- if (!firstTime && (0, import_chatgpt.isGenerating)()) {
- continue;
- } else if ((0, import_chatgpt.getContinueGeneratingButton)()) {
- (_a = (0, import_chatgpt.getContinueGeneratingButton)()) == null ? void 0 : _a.click();
- continue;
- } else if ((0, import_chatgpt.getRegenerateButton)() && !(0, import_chatgpt.getTextarea)()) {
- if (retryCount < maxRetries) {
- yield new Promise((resolve) => setTimeout(resolve, 2 * 1e3));
- (_b = (0, import_chatgpt.getRegenerateButton)()) == null ? void 0 : _b.click();
- retryCount++;
- lastRetryTime = (/* @__PURE__ */ new Date()).getTime();
- continue;
- } else {
- console.error("Failed to regenerate after 3 attempts. Stopping retries.");
- break;
- }
- }
- firstTime = false;
- break;
- }
- }), 1e3);
- });
- }
- (function() {
- main();
- })();
- })();