// ==UserScript== // @name chat-octopus // @namespace https://github.com/mefengl // @version 0.2.6 // @description let octopus send message for you // @icon https://www.google.com/s2/favicons?sz=64&domain=openai.com // @author mefengl // @match https://chat.openai.com/* // @match https://bard.google.com/* // @match https://www.bing.com/search?q=Bing+AI* // @require https://cdn.staticfile.org/jquery/3.6.1/jquery.min.js // @grant GM_openInTab // @grant GM_registerMenuCommand // @grant GM_unregisterMenuCommand // @grant GM_getValue // @grant GM_setValue // @grant GM_addValueChangeListener // @license MIT // @downloadURL none // ==/UserScript== (function () { 'use strict'; const default_menu_all = { }; const menu_all = GM_getValue("menu_all", default_menu_all); // 菜单更新的逻辑 const menus = [ { checker: () => location.href.includes("chat.openai"), name: "openai", value: true }, { checker: () => location.href.includes("bard.google"), name: "bard", value: true }, { checker: () => location.href.includes("Bing+AI"), name: "bing", value: true }, ]; menus.forEach(menu => { $(() => menu.checker() && GM_setValue(menu.name, true)); if (GM_getValue(menu.name) == true) { default_menu_all[menu.name] = menu.value; } }); // 检查是否有新增菜单 for (let name in default_menu_all) { if (!(name in menu_all)) { menu_all[name] = default_menu_all[name]; } } const menu_id = GM_getValue("menu_id", {}); function registerMenuCommand(name, value) { const menuText = ` ${name}:${value ? '✅' : '❌'}`; const commandCallback = () => { menu_all[name] = !menu_all[name]; GM_setValue('menu_all', menu_all); update_menu(); location.reload(); }; return GM_registerMenuCommand(menuText, commandCallback); } function update_menu() { for (let name in menu_all) { const value = menu_all[name]; if (menu_id[name]) { GM_unregisterMenuCommand(menu_id[name]); } menu_id[name] = registerMenuCommand(name, value); } GM_setValue('menu_id', menu_id); } update_menu(); /* ************************************************************************* */ const chatgpt = { getSubmitButton: function () { const form = document.querySelector('form'); if (!form) return; const buttons = form.querySelectorAll('button'); const result = buttons[buttons.length - 1]; return result; }, getTextarea: function () { const form = document.querySelector('form'); if (!form) return; const textareas = form.querySelectorAll('textarea'); const result = textareas[0]; return result; }, getRegenerateButton: function () { const form = document.querySelector('form'); if (!form) return; const buttons = form.querySelectorAll('button'); for (let i = 0; i < buttons.length; i++) { const buttonText = buttons[i]?.textContent?.trim().toLowerCase(); if (buttonText?.includes('regenerate')) { return buttons[i]; } } }, getStopGeneratingButton: function () { const form = document.querySelector('form'); if (!form) return; const buttons = form.querySelectorAll('button'); if (buttons.length === 0) return; for (let i = 0; i < buttons.length; i++) { const buttonText = buttons[i]?.textContent?.trim().toLowerCase(); if (buttonText?.includes('stop')) { return buttons[i]; } } }, send: function (text) { const textarea = this.getTextarea(); if (!textarea) return; textarea.value = text; textarea.dispatchEvent(new Event('input', { bubbles: true })); textarea.dispatchEvent(new KeyboardEvent('keydown', { key: 'Enter', bubbles: true })); }, onSend: function (callback) { const textarea = this.getTextarea(); if (!textarea) return; textarea.addEventListener("keydown", function (event) { if (event.key === "Enter" && !event.shiftKey) { callback(); } }); const sendButton = this.getSubmitButton(); if (!sendButton) return; sendButton.addEventListener('mousedown', callback); }, }; // ChatGPT send prompt to other ai let chatgpt_last_prompt = ''; $(() => { if (menu_all.openai && location.href.includes("chat.openai")) { chatgpt.onSend(() => { const textarea = chatgpt.getTextarea(); const prompt = textarea.value; chatgpt_last_prompt = prompt; GM_setValue('bard_prompt_texts', [prompt]); GM_setValue('bing_prompt_texts', [prompt]); }); } }); // ChatGPT response to prompt comes from other ai let last_trigger_time = +new Date(); $(() => { if (location.href.includes("chat.openai")) { console.log("chatgpt add value change listener"); GM_addValueChangeListener("chatgpt_prompt_texts", (name, old_value, new_value) => { console.log("prompt_texts changed in chatgpt"); console.log(new_value); if (+new Date() - last_trigger_time < 500) { return; } last_trigger_time = + new Date(); setTimeout(async () => { const prompt_texts = new_value; if (prompt_texts.length > 0) { // get prompt_texts from local let firstTime = true; while (prompt_texts.length > 0) { if (!firstTime) { await new Promise(resolve => setTimeout(resolve, 2000)); } if (!firstTime && chatgpt.getRegenerateButton() == undefined) { continue; } firstTime = false; const prompt_text = prompt_texts.shift(); if (prompt_text === chatgpt_last_prompt) { continue; } console.log("chatgpt send prompt_text", prompt_text); chatgpt.send(prompt_text); } } }, 0); GM_setValue("chatgpt_prompt_texts", []); }); } }); /* ************************************************************************* */ const bard = { getSubmitButton: function () { return document.querySelector('button[aria-label="Send message"]'); }, getInputArea: function () { return document.querySelector(".input-area"); }, getTextarea: function () { const inputArea = this.getInputArea(); return inputArea.querySelector('textarea'); }, getRegenerateButton: function () { return document.querySelector('button[aria-label="Retry"]'); }, getLastPrompt: function () { const promptElements = document.querySelectorAll('.query-text'); const lastPrompt = promptElements[promptElements.length - 1]; return lastPrompt; }, getLatestPromptText: function () { const lastPrompt = this.getLastPrompt(); if (!lastPrompt) return ""; const lastPromptText = lastPrompt.textContent; return lastPromptText; }, send: function (text) { const textarea = this.getTextarea(); textarea.value = text; textarea.dispatchEvent(new Event('input')); const submitButton = this.getSubmitButton(); submitButton.click(); }, onSend: function (callback) { const textarea = this.getTextarea(); if (!textarea) return; textarea.addEventListener("keydown", function (event) { if (event.key === "Enter" && !event.shiftKey) { callback(); } }); const sendButton = this.getSubmitButton(); if (!sendButton) return; sendButton.addEventListener('mousedown', callback); }, }; // Bard send prompt to other ai let bard_last_prompt = ""; $(async () => { if (menu_all.bard && location.href.includes("bard.google")) { while (!bard.getSubmitButton()) { await new Promise(resolve => setTimeout(resolve, 500)); } bard.onSend(() => { console.log("bard send"); const textarea = bard.getTextarea(); let prompt = textarea.value; if (!prompt) { prompt = bard.getLatestPromptText(); } console.log(prompt); bard_last_prompt = prompt; GM_setValue('chatgpt_prompt_texts', [prompt]); GM_setValue('bing_prompt_texts', [prompt]); }); } }); // Bard response to prompt_texts let lastTriggerTime = +new Date(); if (location.href.includes("bard.google")) { GM_addValueChangeListener("bard_prompt_texts", (name, old_value, new_value) => { if (+new Date() - lastTriggerTime < 500) { return; } lastTriggerTime = + new Date(); setTimeout(async () => { const promptTexts = new_value; if (promptTexts.length > 0) { // get promptTexts from local let firstTime = true; while (promptTexts.length > 0) { if (!firstTime) { await new Promise(resolve => setTimeout(resolve, 2000)); } if (!firstTime && bard.getRegenerateButton() == undefined) { continue; } firstTime = false; const promptText = promptTexts.shift(); if (promptText === bard_last_prompt) { continue; } bard.send(promptText); } } }, 0); GM_setValue("bard_prompt_texts", []); }); } /* ************************************************************************* */ const bing = { getActionBar: function () { return document.querySelector("cib-serp")?.shadowRoot?.querySelector("cib-action-bar")?.shadowRoot; }, getSubmitButton: function () { const actionBar = this.getActionBar(); if (!actionBar) { return null; } return actionBar.querySelector('button[aria-label="Submit"]'); }, getTextarea: function () { const actionBar = this.getActionBar(); if (!actionBar) { return null; } return actionBar.querySelector('textarea'); }, getStopGeneratingButton: function () { const actionBar = this.getActionBar(); if (!actionBar) { return null; } const stopGeneratingButton = actionBar.querySelector('cib-typing-indicator')?.shadowRoot?.querySelector('button[aria-label="Stop Responding"]'); if (!stopGeneratingButton) { return null; } if (stopGeneratingButton.disabled) { return null; } return stopGeneratingButton; }, getNewChatButton: function () { const actionBar = this.getActionBar(); if (!actionBar) { return null; } return actionBar.querySelector('button[aria-label="New topic"]'); }, getConversation: function () { return document.querySelector("cib-serp")?.shadowRoot?.querySelector("cib-conversation")?.shadowRoot; }, getChatTurns: function () { const conversation = this.getConversation(); if (!conversation) { return null; } return Array.from(conversation.querySelectorAll('cib-chat-turn')).map(t => t.shadowRoot); }, getLastChatTurn: function () { const chatTurns = this.getChatTurns(); if (!chatTurns) { return null; } return chatTurns[chatTurns.length - 1]; }, getLastResponse: function () { const lastChatTurn = this.getLastChatTurn(); if (!lastChatTurn) { return null; } return lastChatTurn.querySelectorAll('cib-message-group')[1]?.shadowRoot; }, getLastResponseText: function () { const lastResponse = this.getLastResponse(); if (!lastResponse) { return null; } return Array.from(lastResponse.querySelectorAll('cib-message')) .map(m => m.shadowRoot) .find(m => m.querySelector('cib-shared')) .textContent.trim(); }, send: function (text) { const textarea = this.getTextarea(); if (!textarea) { return null; } textarea.value = text; textarea.dispatchEvent(new Event('input')); const submitButton = this.getSubmitButton(); if (!submitButton) { return null; } submitButton.click(); }, onSend: function (callback) { const textarea = this.getTextarea(); if (!textarea) return; textarea.addEventListener("keydown", function (event) { if (event.key === "Enter" && !event.shiftKey) { callback(); } }); const sendButton = this.getSubmitButton(); if (!sendButton) return; sendButton.addEventListener('mousedown', callback); } }; // bing send prompt to other ai let bing_last_prompt = ""; $(async () => { if (menu_all.bing && location.href.includes("Bing+AI")) { console.log("bing"); while (!bing.getSubmitButton()) { await new Promise(resolve => setTimeout(resolve, 500)); } console.log("get bing submit button"); bing.onSend(() => { console.log("bing send"); const textarea = bing.getTextarea(); const prompt = textarea.value; console.log(prompt); bing_last_prompt = prompt; GM_setValue('chatgpt_prompt_texts', [prompt]); GM_setValue('bard_prompt_texts', [prompt]); }); } }); // bing response to prompt_texts let last_trigger_time_bing = +new Date(); if (location.href.includes("Bing+AI")) { GM_addValueChangeListener("bing_prompt_texts", (name, old_value, new_value) => { if (+new Date() - last_trigger_time_bing < 500) { return; } last_trigger_time_bing = new Date(); setTimeout(async () => { const prompt_texts = new_value; if (prompt_texts.length > 0) { // get prompt_texts from local let firstTime = true; while (prompt_texts.length > 0) { if (!firstTime) { await new Promise(resolve => setTimeout(resolve, 2000)); } if (!firstTime && bing.getStopGeneratingButton() != undefined) { continue; } firstTime = false; const prompt_text = prompt_texts.shift(); if (prompt_text === bing_last_prompt) { continue; } console.log("bing send prompt_text", prompt_text); bing.send(prompt_text); } } }, 0); GM_setValue("bing_prompt_texts", []); }); } })();