AI Enter 換行

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

目前為 2025-06-02 提交的版本,檢視 最新版本

您需要先安裝使用者腳本管理器擴展,如 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.2.1
// @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/*
// @match        https://duckduckgo.com/*
// @include      http://192.168.*.*:*/*
// @icon         

// @grant        GM_getValue
// @grant        GM_setValue
// @grant        GM_registerMenuCommand
// ==/UserScript==

(() => {
  "use strict";

  // ----- 設定管理 -----

  // 預設設定
  const defaultConfig = {
    shortcuts: {
      send: {
        ctrl: true, // Ctrl + Enter
        alt: false, // Alt/Option + Enter
        meta: true, // Win/Cmd/Super + Enter
      },
    },
  };

  // 多語系翻譯字典
  const translations = {
    en: {
      settings: "Settings",
      close: "✕",
      sendShortcut: "Send Message Shortcut (+ Enter):",
      save: "Save",
      reset: "Reset",
      saveSuccess: "Settings saved!",
      saveFailed: "Failed to save settings!",
      resetConfirm: "Are you sure you want to reset to default settings?",
      resetSuccess: "Settings reset to default!",
      ctrlEnter: "Ctrl + Enter",
      altEnter: "Alt + Enter",
      cmdEnter: "Cmd + Enter",
      winEnter: "Win + Enter",
      superEnter: "Super + Enter",
    },

    "zh-tw": {
      settings: "設定",
      close: "✕",
      sendShortcut: "傳送訊息快捷鍵(+ Enter):",
      save: "儲存",
      reset: "重設",
      saveSuccess: "設定已儲存!",
      saveFailed: "儲存設定失敗!",
      resetConfirm: "確定要重設為預設設定嗎?",
      resetSuccess: "設定已重設為預設值!",
      ctrlEnter: "Ctrl + Enter",
      altEnter: "Alt + Enter",
      cmdEnter: "Cmd + Enter",
      winEnter: "Win + Enter",
      superEnter: "Super + Enter",
    },

    "zh-cn": {
      settings: "设置",
      close: "✕",
      sendShortcut: "发送消息快捷键(+ Enter):",
      save: "保存",
      reset: "重置",
      saveSuccess: "设置已保存!",
      saveFailed: "保存设置失败!",
      resetConfirm: "确定要重置为默认设置吗?",
      resetSuccess: "设置已重置为默认值!",
      ctrlEnter: "Ctrl + Enter",
      altEnter: "Alt + Enter",
      cmdEnter: "Cmd + Enter",
      winEnter: "Win + Enter",
      superEnter: "Super + Enter",
    },
  };

  // 偵測瀏覽器語言偏好
  function detectBrowserLanguage() {
    const lang = navigator.language || navigator.userLanguage;
    if (lang.startsWith("zh")) {
      if (lang.includes("TW") || lang.includes("HK") || lang.includes("MO")) {
        return "zh-tw";
      } else {
        return "zh-cn";
      }
    } else {
      return "en";
    }
  }

  // 取得目前使用的語言
  function getCurrentLanguage() {
    return detectBrowserLanguage();
  }

  // 取得翻譯文字
  function t(key) {
    const lang = getCurrentLanguage();
    return translations[lang]?.[key] || translations.en[key] || key;
  }

  // 載入使用者設定
  function loadConfig() {
    try {
      const savedConfig = GM_getValue("aiEnterConfig");
      if (savedConfig) {
        const config = JSON.parse(savedConfig);
        return {
          shortcuts: {
            send: {
              ctrl:
                config.shortcuts?.send?.ctrl !== undefined
                  ? config.shortcuts.send.ctrl
                  : defaultConfig.shortcuts.send.ctrl,
              alt:
                config.shortcuts?.send?.alt !== undefined
                  ? config.shortcuts.send.alt
                  : defaultConfig.shortcuts.send.alt,
              meta:
                config.shortcuts?.send?.meta !== undefined
                  ? config.shortcuts.send.meta
                  : defaultConfig.shortcuts.send.meta,
            },
          },
        };
      }
    } catch (error) {
      console.error("載入設定時發生錯誤:", error);
    }
    return defaultConfig;
  }

  // 儲存設定
  function saveConfig(config) {
    try {
      GM_setValue("aiEnterConfig", JSON.stringify(config));
      return true;
    } catch (error) {
      console.error("儲存設定時發生錯誤:", error);
      return false;
    }
  }

  // 建立設定介面
  function createConfigInterface() {
    // 如果已經有設定視窗開啟,則關閉它
    const existingDialog = document.getElementById("ai-enter-config");
    if (existingDialog) {
      existingDialog.remove();
      return;
    }

    // 偵測使用者的作業系統
    function detectOS() {
      const userAgent = navigator.userAgent.toLowerCase();
      const platform = navigator.platform.toLowerCase();

      if (platform.includes("mac") || userAgent.includes("mac")) {
        return "mac";
      } else if (platform.includes("win") || userAgent.includes("win")) {
        return "windows";
      } else if (platform.includes("linux") || userAgent.includes("linux")) {
        return "linux";
      } else {
        return "other";
      }
    }

    const currentOS = detectOS();

    // 載入目前設定
    const config = loadConfig();

    // 偵測深色模式
    const isDarkMode =
      window.matchMedia &&
      window.matchMedia("(prefers-color-scheme: dark)").matches;

    // 根據深色/淺色模式設定配色
    const colors = {
      background: isDarkMode ? "#2d2d2d" : "#ffffff",
      text: isDarkMode ? "#e0e0e0" : "#333333",
      border: isDarkMode ? "#555555" : "#dddddd",
      inputBg: isDarkMode ? "#3d3d3d" : "#ffffff",
      inputBorder: isDarkMode ? "#666666" : "#dddddd",
      buttonBg: isDarkMode ? "#3d3d3d" : "#f5f5f5",
      buttonText: isDarkMode ? "#e0e0e0" : "#333333",
      primary: "#4caf50", // 綠色按鈕,保持不變
      shadow: isDarkMode ? "rgba(0,0,0,0.3)" : "rgba(0,0,0,0.15)",
    };

    // 建立設定對話框
    const dialogDiv = document.createElement("div");
    dialogDiv.id = "ai-enter-config";
    dialogDiv.style.position = "fixed";
    dialogDiv.style.top = "50%";
    dialogDiv.style.left = "50%";
    dialogDiv.style.transform = "translate(-50%, -50%)";
    dialogDiv.style.backgroundColor = colors.background;
    dialogDiv.style.color = colors.text;
    dialogDiv.style.border = `1px solid ${colors.border}`;
    dialogDiv.style.borderRadius = "8px";
    dialogDiv.style.padding = "20px";
    dialogDiv.style.width = "350px";
    dialogDiv.style.maxWidth = "90vw";
    dialogDiv.style.maxHeight = "90vh";
    dialogDiv.style.overflowY = "auto";
    dialogDiv.style.zIndex = "10000";
    dialogDiv.style.boxShadow = `0 4px 12px ${colors.shadow}`;
    dialogDiv.style.fontFamily =
      "-apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Helvetica, Arial, sans-serif";

    // 設定標題
    const titleDiv = document.createElement("div");
    titleDiv.style.display = "flex";
    titleDiv.style.justifyContent = "space-between";
    titleDiv.style.alignItems = "center";
    titleDiv.style.marginBottom = "16px";

    const title = document.createElement("h2");
    title.textContent = t("settings");
    title.style.margin = "0";
    title.style.fontSize = "18px";
    title.style.color = colors.text;

    const closeButton = document.createElement("button");
    closeButton.textContent = t("close");
    closeButton.style.background = "none";
    closeButton.style.border = "none";
    closeButton.style.color = colors.text;
    closeButton.style.cursor = "pointer";
    closeButton.style.fontSize = "18px";
    closeButton.onclick = () => dialogDiv.remove();

    titleDiv.appendChild(title);
    titleDiv.appendChild(closeButton);
    dialogDiv.appendChild(titleDiv);

    // 快捷鍵設定
    const shortcutsLabel = document.createElement("label");
    shortcutsLabel.textContent = t("sendShortcut");
    shortcutsLabel.style.display = "block";
    shortcutsLabel.style.marginBottom = "12px";
    shortcutsLabel.style.color = colors.text;
    shortcutsLabel.style.fontWeight = "bold";
    dialogDiv.appendChild(shortcutsLabel);

    // 快捷鍵選項容器
    const shortcutsContainer = document.createElement("div");
    shortcutsContainer.style.marginBottom = "16px";
    shortcutsContainer.style.padding = "12px";
    shortcutsContainer.style.backgroundColor = isDarkMode
      ? "#3a3a3a"
      : "#f8f9fa";
    shortcutsContainer.style.border = `1px solid ${colors.border}`;
    shortcutsContainer.style.borderRadius = "6px";

    // 根據作業系統顯示適當的快捷鍵標籤
    const shortcuts = [
      {
        key: "ctrl",
        label: currentOS === "mac" ? `⌃ ${t("ctrlEnter")}` : t("ctrlEnter"),
      },
      {
        key: "alt",
        label: currentOS === "mac" ? `⌥ ${t("altEnter")}` : t("altEnter"),
      },
      {
        key: "meta",
        label:
          currentOS === "mac"
            ? `⌘ ${t("cmdEnter")}`
            : currentOS === "windows"
            ? `⊞ ${t("winEnter")}`
            : currentOS === "linux"
            ? t("superEnter")
            : t("winEnter"),
      },
    ];

    shortcuts.forEach((shortcut) => {
      const optionDiv = document.createElement("div");
      optionDiv.style.display = "flex";
      optionDiv.style.alignItems = "center";
      optionDiv.style.marginBottom = "8px";

      const checkbox = document.createElement("input");
      checkbox.type = "checkbox";
      checkbox.id = `shortcut-${shortcut.key}`;
      checkbox.checked =
        config.shortcuts?.send?.[shortcut.key] !== undefined
          ? config.shortcuts.send[shortcut.key]
          : defaultConfig.shortcuts.send[shortcut.key];
      if (isDarkMode) {
        checkbox.style.accentColor = colors.primary;
      }

      const labelElement = document.createElement("label");
      labelElement.htmlFor = `shortcut-${shortcut.key}`;
      labelElement.style.marginLeft = "8px";
      labelElement.style.color = colors.text;
      labelElement.style.cursor = "pointer";
      labelElement.style.flexGrow = "1";
      labelElement.textContent = shortcut.label;

      optionDiv.appendChild(checkbox);
      optionDiv.appendChild(labelElement);
      shortcutsContainer.appendChild(optionDiv);
    });

    dialogDiv.appendChild(shortcutsContainer);

    // 按鈕區域
    const buttonDiv = document.createElement("div");
    buttonDiv.style.display = "flex";
    buttonDiv.style.justifyContent = "flex-end";
    buttonDiv.style.marginTop = "16px";

    const saveButton = document.createElement("button");
    saveButton.textContent = t("save");
    saveButton.style.padding = "8px 16px";
    saveButton.style.backgroundColor = colors.primary;
    saveButton.style.color = "white";
    saveButton.style.border = "none";
    saveButton.style.borderRadius = "4px";
    saveButton.style.cursor = "pointer";
    saveButton.style.marginLeft = "8px";

    saveButton.onclick = () => {
      // 取得勾選的快捷鍵設定
      const sendShortcuts = {
        ctrl: document.getElementById("shortcut-ctrl").checked,
        alt: document.getElementById("shortcut-alt").checked,
        meta: document.getElementById("shortcut-meta").checked,
      };

      const newConfig = {
        shortcuts: {
          send: sendShortcuts,
        },
      };

      if (saveConfig(newConfig)) {
        alert(t("saveSuccess"));
        dialogDiv.remove();
        // 重新載入設定
        currentConfig = loadConfig();
      } else {
        alert(t("saveFailed"));
      }
    };

    const resetButton = document.createElement("button");
    resetButton.textContent = t("reset");
    resetButton.style.padding = "8px 16px";
    resetButton.style.backgroundColor = colors.buttonBg;
    resetButton.style.color = colors.buttonText;
    resetButton.style.border = `1px solid ${colors.border}`;
    resetButton.style.borderRadius = "4px";
    resetButton.style.cursor = "pointer";

    resetButton.onclick = () => {
      if (confirm(t("resetConfirm"))) {
        saveConfig(defaultConfig);
        alert(t("resetSuccess"));
        dialogDiv.remove();
        // 重新載入設定
        currentConfig = loadConfig();
        // 移除背景遮罩
        const overlay = document.querySelector('div[style*="z-index: 9999"]');
        if (overlay) overlay.remove();
        // 重新開啟設定介面以顯示重設後的設定
        createConfigInterface();
      }
    };

    buttonDiv.appendChild(resetButton);
    buttonDiv.appendChild(saveButton);
    dialogDiv.appendChild(buttonDiv);

    // 新增設定對話框到頁面
    document.body.appendChild(dialogDiv);

    // 新增背景遮罩
    const overlay = document.createElement("div");
    overlay.style.position = "fixed";
    overlay.style.top = "0";
    overlay.style.left = "0";
    overlay.style.width = "100%";
    overlay.style.height = "100%";
    overlay.style.backgroundColor = isDarkMode
      ? "rgba(0,0,0,0.7)"
      : "rgba(0,0,0,0.5)";
    overlay.style.zIndex = "9999";
    overlay.onclick = () => {
      overlay.remove();
      dialogDiv.remove();
    };

    document.body.insertBefore(overlay, dialogDiv);
  }

  // 載入設定
  let currentConfig = loadConfig();

  // 註冊設定選單
  GM_registerMenuCommand("⚙️ Settings", createConfigInterface);

  // 輸出啟動資訊至 console
  console.log(
    "AI Enter Newline UserScript loaded. Current config:",
    currentConfig
  );

  // 輔助函數:取得事件目標元素
  function getEventTarget(e) {
    return e.composedPath ? e.composedPath()[0] || e.target : e.target;
  }

  // 輔助函數:檢查是否正在進行中文輸入
  function isChineseInputMode(e) {
    return e.isComposing || e.keyCode === 229;
  }

  // 輔助函數:檢查是否在 ChatGPT 輸入框內
  function isInChatGPTTextarea(target) {
    return (
      target.id === "prompt-textarea" ||
      target.closest("#prompt-textarea") ||
      (target.getAttribute && target.getAttribute("contenteditable") === "true")
    );
  }

  /**
   * 檢查按鍵組合是否為任何可能的發送快捷鍵(不論是否啟用)
   * @param {KeyboardEvent} e - 鍵盤事件
   * @returns {boolean} 是否為潛在的發送快捷鍵組合
   */
  function isPotentialSendShortcut(e) {
    if (e.key !== "Enter") return false;

    // 檢查是否為任何可能的發送快捷鍵組合:Ctrl+Enter、Alt+Enter 或 Cmd+Enter
    const isCtrlOnly = e.ctrlKey && !e.altKey && !e.metaKey && !e.shiftKey;
    const isAltOnly = e.altKey && !e.ctrlKey && !e.metaKey && !e.shiftKey;
    const isMetaOnly = e.metaKey && !e.ctrlKey && !e.altKey && !e.shiftKey;

    return isCtrlOnly || isAltOnly || isMetaOnly;
  }

  // 檢查是否為發送快捷鍵
  function isSendShortcut(e) {
    // 必須按下 Enter 鍵
    if (e.key !== "Enter") return false;

    const shortcuts =
      currentConfig.shortcuts?.send || defaultConfig.shortcuts.send;

    // 檢查是否有任何一個勾選的快捷鍵符合目前按鍵組合
    return (
      (shortcuts.ctrl && e.ctrlKey && !e.altKey && !e.metaKey) ||
      (shortcuts.alt && e.altKey && !e.ctrlKey && !e.metaKey) ||
      (shortcuts.meta && e.metaKey && !e.ctrlKey && !e.altKey)
    );
  }

  // 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 (isChineseInputMode(e)) {
          return;
        }

        // 如果是 Enter 鍵且沒有按下其他修飾鍵
        if (
          e.key === "Enter" &&
          !e.ctrlKey &&
          !e.shiftKey &&
          !e.metaKey &&
          !e.altKey
        ) {
          const target = getEventTarget(e);
          // 檢查是否在 prompt-textarea 或其他輸入區域
          if (isInChatGPTTextarea(target)) {
            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;
          }
        }

        // 使用自訂快捷鍵觸發送出
        if (isSendShortcut(e)) {
          // 同樣,如果正在中文輸入,不處理
          if (isChineseInputMode(e)) {
            return;
          }

          const target = getEventTarget(e);
          if (isInChatGPTTextarea(target)) {
            const submitButton = findChatGPTSubmitButton();
            if (submitButton && !submitButton.disabled) {
              e.preventDefault();
              e.stopPropagation();
              submitButton.click();
            }
          }
        }

        // 智慧型事件冒泡防止:如果是潛在的快捷鍵但未被使用者啟用,
        // 阻止事件傳播,避免觸發 ChatGPT 的原生快捷鍵行為
        if (isPotentialSendShortcut(e)) {
          const target = getEventTarget(e);
          if (isInChatGPTTextarea(target)) {
            e.preventDefault();
            e.stopPropagation();
          }
        }
      } else {
        // 其他網站的處理邏輯
        // 如果正在進行中文輸入法選字,不干擾原生行為
        if (isChineseInputMode(e)) {
          return;
        }

        // 如果是 Enter 鍵且沒有按下其他修飾鍵(純 Enter)
        if (
          e.key === "Enter" &&
          !e.ctrlKey &&
          !e.shiftKey &&
          !e.metaKey &&
          !e.altKey
        ) {
          const target = getEventTarget(e);
          if (
            /INPUT|TEXTAREA|SELECT|LABEL/.test(target.tagName) ||
            (target.getAttribute &&
              target.getAttribute("contenteditable") === "true")
          ) {
            // 阻止事件向上冒泡,避免觸發不必要的送出行為
            e.stopPropagation();
          }
        }

        // 如果是自訂快捷鍵組合,讓原生行為執行(不阻止)
        // 這樣使用者可以在其他網站使用相同的快捷鍵設定
        if (isSendShortcut(e)) {
          // 不做任何處理,讓網站的原生快捷鍵邏輯執行
          return;
        }

        // 智慧型事件冒泡防止:如果是潛在的快捷鍵但未被使用者啟用,
        // 也要阻止冒泡,避免觸發網站的原生快捷鍵行為
        // 但對於 felo.ai,允許 ctrl+enter 正常冒泡, 因為 felo.ai 的 ctrl+enter 是用來搜尋網頁的
        if (isPotentialSendShortcut(e)) {
          // 如果是 felo.ai 且是 ctrl+enter,不阻止冒泡
          if (
            window.location.href.includes("felo.ai") &&
            e.ctrlKey &&
            e.key === "Enter" &&
            !e.altKey &&
            !e.metaKey
          ) {
            return;
          }

          const target = getEventTarget(e);
          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 (isChineseInputMode(e)) return; // 如果是 Enter 鍵且沒有按下其他修飾鍵(純 Enter)
      if (
        e.key === "Enter" &&
        !e.ctrlKey &&
        !e.shiftKey &&
        !e.metaKey &&
        !e.altKey
      ) {
        const target = getEventTarget(e);
        if (
          /INPUT|TEXTAREA|SELECT|LABEL/.test(target.tagName) ||
          (target.getAttribute &&
            target.getAttribute("contenteditable") === "true")
        ) {
          // 同樣阻止事件冒泡
          e.stopPropagation();
        }
      }

      // 如果是自訂快捷鍵組合,讓原生行為執行(不阻止)
      if (isSendShortcut(e)) {
        return;
      }

      // 智慧型事件冒泡防止:如果是潛在的快捷鍵但未被使用者啟用,
      // 也要阻止冒泡,避免觸發網站的原生快捷鍵行為
      // 但對於 felo.ai,允許 ctrl+enter 正常冒泡
      if (isPotentialSendShortcut(e)) {
        // 如果是 felo.ai 且是 ctrl+enter,不阻止冒泡
        if (
          window.location.href.includes("felo.ai") &&
          e.ctrlKey &&
          e.key === "Enter" &&
          !e.altKey &&
          !e.metaKey
        ) {
          return;
        }

        const target = getEventTarget(e);
        if (
          /INPUT|TEXTAREA|SELECT|LABEL/.test(target.tagName) ||
          (target.getAttribute &&
            target.getAttribute("contenteditable") === "true")
        ) {
          e.stopPropagation();
        }
      }
    },
    true
  );
})();