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
// @name:zh-TW   AI Enter 換行
// @name:zh-CN   AI Enter 换行
// @namespace    http://tampermonkey.net/
// @version      1.2.0
// @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         

// @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
  );
})();