AI Enter as Newline

Enable Enter key for newline in AI chat input, use Cmd+Enter (Mac) or Ctrl+Enter (Windows) to send message.

// ==UserScript==
// @name         AI Enter as Newline
// @namespace    http://tampermonkey.net/
// @version      1.0
// @description  Enable Enter key for newline in AI chat input, use Cmd+Enter (Mac) or Ctrl+Enter (Windows) to send message.
// @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        http://192.168.*
// @match        http://localhost*
// @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
  );
})();