您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
让 AI 聊天输入区的 Enter 键可换行,使用 Cmd+Enter(Mac)或 Ctrl+Enter(Windows)发送消息。
- // ==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
- );
- })();