AI Enter 换行

让 AI 聊天输入区的 Enter 键可换行,使用 Cmd+Enter(Mac)或 Ctrl+Enter(Windows)发送消息。

当前为 2025-05-28 提交的版本,查看 最新版本

您需要先安装一个扩展,例如 篡改猴Greasemonkey暴力猴,之后才能安装此脚本。

You will need to install an extension such as Tampermonkey to install this script.

您需要先安装一个扩展,例如 篡改猴暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴Userscripts ,之后才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。

您需要先安装用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name         AI Enter as Newline
// @name:zh-TW   AI Enter 換行
// @name:zh-CN   AI Enter 换行
// @namespace    http://tampermonkey.net/
// @version      1.0.3
// @description  Enable Enter key for newline in AI chat input, use Cmd+Enter (Mac) or Ctrl+Enter (Windows) to send message.
// @description:zh-TW  讓 AI 聊天輸入區的 Enter 鍵可換行,使用 Cmd+Enter(Mac)或 Ctrl+Enter(Windows)送出訊息。
// @description:zh-CN  让 AI 聊天输入区的 Enter 键可换行,使用 Cmd+Enter(Mac)或 Ctrl+Enter(Windows)发送消息。
// @author       windofage
// @license      MIT
// @match        https://chatgpt.com/*
// @match        https://claude.ai/*
// @match        https://gemini.google.com/*
// @match        https://www.perplexity.ai/*
// @match        https://felo.ai/*
// @match        https://chat.deepseek.com/*
// @match        https://grok.com/*
// @include      http://192.168.*.*:*/*
// @icon         

// ==/UserScript==

(() => {
  "use strict";

  // 輸出啟動資訊至 console
  console.log("Chat UI Ctrl+Enter Sender Enabled");

  // ChatGPT 特殊處理:尋找送出按鈕
  let findChatGPTSubmitButton = () => {
    return document.querySelector('button[data-testid="send-button"]');
  };

  // 監聽 keydown 事件,攔截非預期的 Enter 按下事件,避免在輸入元件內誤觸送出
  window.addEventListener(
    "keydown",
    (e) => {
      // ChatGPT 網站特殊處理
      if (window.location.href.includes("chatgpt.com")) {
        // 如果正在進行中文輸入法選字,不干擾原生行為
        if (e.isComposing || e.keyCode === 229) {
          return;
        }

        // 如果是 Enter 鍵且沒有按下其他修飾鍵
        if (e.key === "Enter" && !e.ctrlKey && !e.shiftKey && !e.metaKey) {
          let target = e.composedPath
            ? e.composedPath()[0] || e.target
            : e.target;
          // 檢查是否在 prompt-textarea 或其他輸入區域
          if (
            target.id === "prompt-textarea" ||
            target.closest("#prompt-textarea") ||
            (target.getAttribute &&
              target.getAttribute("contenteditable") === "true")
          ) {
            e.stopPropagation();
            e.preventDefault();

            // 更可靠的換行方法:模擬 Shift+Enter 按鍵事件
            const shiftEnterEvent = new KeyboardEvent("keydown", {
              key: "Enter",
              code: "Enter",
              shiftKey: true,
              bubbles: true,
              cancelable: true,
            });
            target.dispatchEvent(shiftEnterEvent);

            // 如果上述方法無效,嘗試使用 insertParagraph 命令
            if (!shiftEnterEvent.defaultPrevented) {
              document.execCommand("insertParagraph");
            }

            return;
          }
        }

        // 使用 Ctrl+Enter 觸發送出
        if (e.key === "Enter" && (e.ctrlKey || e.metaKey)) {
          // 同樣,如果正在中文輸入,不處理
          if (e.isComposing || e.keyCode === 229) {
            return;
          }

          let target = e.composedPath
            ? e.composedPath()[0] || e.target
            : e.target;
          if (
            target.id === "prompt-textarea" ||
            target.closest("#prompt-textarea")
          ) {
            const submitButton = findChatGPTSubmitButton();
            if (submitButton && !submitButton.disabled) {
              e.preventDefault();
              e.stopPropagation();
              submitButton.click();
            }
          }
        }
      } else {
        // 其他網站的原始處理邏輯
        if (e.key !== "Enter" || e.ctrlKey || e.shiftKey || e.metaKey) return;
        // 如果正在中文輸入,不處理
        if (e.isComposing || e.keyCode === 229) return;

        let target = e.composedPath
          ? e.composedPath()[0] || e.target
          : e.target;
        if (
          /INPUT|TEXTAREA|SELECT|LABEL/.test(target.tagName) ||
          (target.getAttribute &&
            target.getAttribute("contenteditable") === "true")
        ) {
          // 阻止事件向上冒泡,避免觸發不必要的送出行為
          e.stopPropagation();
        }
      }
    },
    true
  );

  // 監聽 keypress 事件,防止在輸入元件內誤觸送出
  window.addEventListener(
    "keypress",
    (e) => {
      // ChatGPT 網站使用 keydown 處理就足夠,這裡保持原樣
      if (window.location.href.includes("chatgpt.com")) return;

      // 如果正在中文輸入,不處理
      if (e.isComposing || e.keyCode === 229) return;

      if (e.key !== "Enter" || e.ctrlKey || e.shiftKey || e.metaKey) return;
      let target = e.composedPath ? e.composedPath()[0] || e.target : e.target;
      if (
        /INPUT|TEXTAREA|SELECT|LABEL/.test(target.tagName) ||
        (target.getAttribute &&
          target.getAttribute("contenteditable") === "true")
      ) {
        // 同樣阻止事件冒泡
        e.stopPropagation();
      }
    },
    true
  );
})();