AI Enter as Newline

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

  1. // ==UserScript==
  2. // @name AI Enter as Newline
  3. // @namespace http://tampermonkey.net/
  4. // @version 1.0
  5. // @description Enable Enter key for newline in AI chat input, use Cmd+Enter (Mac) or Ctrl+Enter (Windows) to send message.
  6. // @author windofage
  7. // @license MIT
  8. // @match https://chatgpt.com/*
  9. // @match https://claude.ai/*
  10. // @match https://gemini.google.com/*
  11. // @match https://www.perplexity.ai/*
  12. // @match https://felo.ai/*
  13. // @match https://chat.deepseek.com/*
  14. // @match http://192.168.*
  15. // @match http://localhost*
  16. // @icon 
  17.  
  18. // ==/UserScript==
  19.  
  20. (() => {
  21. "use strict";
  22.  
  23. // 輸出啟動資訊至 console
  24. console.log("Chat UI Ctrl+Enter Sender Enabled");
  25.  
  26. // ChatGPT 特殊處理:尋找送出按鈕
  27. let findChatGPTSubmitButton = () => {
  28. return document.querySelector('button[data-testid="send-button"]');
  29. };
  30.  
  31. // 監聽 keydown 事件,攔截非預期的 Enter 按下事件,避免在輸入元件內誤觸送出
  32. window.addEventListener(
  33. "keydown",
  34. (e) => {
  35. // ChatGPT 網站特殊處理
  36. if (window.location.href.includes("chatgpt.com")) {
  37. // 如果正在進行中文輸入法選字,不干擾原生行為
  38. if (e.isComposing || e.keyCode === 229) {
  39. return;
  40. }
  41.  
  42. // 如果是 Enter 鍵且沒有按下其他修飾鍵
  43. if (e.key === "Enter" && !e.ctrlKey && !e.shiftKey && !e.metaKey) {
  44. let target = e.composedPath
  45. ? e.composedPath()[0] || e.target
  46. : e.target;
  47. // 檢查是否在 prompt-textarea 或其他輸入區域
  48. if (
  49. target.id === "prompt-textarea" ||
  50. target.closest("#prompt-textarea") ||
  51. (target.getAttribute &&
  52. target.getAttribute("contenteditable") === "true")
  53. ) {
  54. e.stopPropagation();
  55. e.preventDefault();
  56.  
  57. // 更可靠的換行方法:模擬 Shift+Enter 按鍵事件
  58. const shiftEnterEvent = new KeyboardEvent("keydown", {
  59. key: "Enter",
  60. code: "Enter",
  61. shiftKey: true,
  62. bubbles: true,
  63. cancelable: true,
  64. });
  65. target.dispatchEvent(shiftEnterEvent);
  66.  
  67. // 如果上述方法無效,嘗試使用 insertParagraph 命令
  68. if (!shiftEnterEvent.defaultPrevented) {
  69. document.execCommand("insertParagraph");
  70. }
  71.  
  72. return;
  73. }
  74. }
  75.  
  76. // 使用 Ctrl+Enter 觸發送出
  77. if (e.key === "Enter" && (e.ctrlKey || e.metaKey)) {
  78. // 同樣,如果正在中文輸入,不處理
  79. if (e.isComposing || e.keyCode === 229) {
  80. return;
  81. }
  82.  
  83. let target = e.composedPath
  84. ? e.composedPath()[0] || e.target
  85. : e.target;
  86. if (
  87. target.id === "prompt-textarea" ||
  88. target.closest("#prompt-textarea")
  89. ) {
  90. const submitButton = findChatGPTSubmitButton();
  91. if (submitButton && !submitButton.disabled) {
  92. e.preventDefault();
  93. e.stopPropagation();
  94. submitButton.click();
  95. }
  96. }
  97. }
  98. } else {
  99. // 其他網站的原始處理邏輯
  100. if (e.key !== "Enter" || e.ctrlKey || e.shiftKey || e.metaKey) return;
  101. // 如果正在中文輸入,不處理
  102. if (e.isComposing || e.keyCode === 229) return;
  103.  
  104. let target = e.composedPath
  105. ? e.composedPath()[0] || e.target
  106. : e.target;
  107. if (
  108. /INPUT|TEXTAREA|SELECT|LABEL/.test(target.tagName) ||
  109. (target.getAttribute &&
  110. target.getAttribute("contenteditable") === "true")
  111. ) {
  112. // 阻止事件向上冒泡,避免觸發不必要的送出行為
  113. e.stopPropagation();
  114. }
  115. }
  116. },
  117. true
  118. );
  119.  
  120. // 監聽 keypress 事件,防止在輸入元件內誤觸送出
  121. window.addEventListener(
  122. "keypress",
  123. (e) => {
  124. // ChatGPT 網站使用 keydown 處理就足夠,這裡保持原樣
  125. if (window.location.href.includes("chatgpt.com")) return;
  126.  
  127. // 如果正在中文輸入,不處理
  128. if (e.isComposing || e.keyCode === 229) return;
  129.  
  130. if (e.key !== "Enter" || e.ctrlKey || e.shiftKey || e.metaKey) return;
  131. let target = e.composedPath ? e.composedPath()[0] || e.target : e.target;
  132. if (
  133. /INPUT|TEXTAREA|SELECT|LABEL/.test(target.tagName) ||
  134. (target.getAttribute &&
  135. target.getAttribute("contenteditable") === "true")
  136. ) {
  137. // 同樣阻止事件冒泡
  138. e.stopPropagation();
  139. }
  140. },
  141. true
  142. );
  143. })();

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址