AI Enter 換行

讓 AI 聊天輸入區的 Enter 鍵可換行,使用 Cmd+Enter(Mac)或 Ctrl+Enter(Windows)送出訊息。

目前為 2025-05-28 提交的版本,檢視 最新版本

您需要先安裝使用者腳本管理器擴展,如 TampermonkeyGreasemonkeyViolentmonkey 之後才能安裝該腳本。

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

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyUserscripts 後才能安裝該腳本。

你需要先安裝一款使用者腳本管理器擴展,比如 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
  );
})();