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