Fix: 修复AI聊天输入框Safari回车发送bug

背景

AI聊天输入框场景下,我们通常需要支持用户按下enter回车的快捷发送,通过键盘事件实现

html 复制代码
<textarea
    @keydown="onInputKeydown"
    placeholder="请输入内容,按 Enter 发送,Shift + Enter 换行"
></textarea>
js 复制代码
// 回车发送
const onInputKeydown = (event) => {
  if(event.key === "Enter" || event.keyCode === '13') {
    if(event.shiftKey) return; // shift + enter 仅换行
    event.preventDefault();
    sendMessage(); // 发送处理
  }
}

问题

但实际上enter作为确认操作,除了确认内容发送动作,还有确认输入法输入的动作;

如用户在输入法输入状态下,按下enter期望确认输入内容到文本框,此时也会触发上面的发送逻辑导致内容内发送出去。

我们可以 onInputKeydown 加一个合成状态(isComposing)判断

js 复制代码
const onInputKeydown = (event) => {
  if(event.key === "Enter" || event.keyCode === '13') {
    if(event.shiftKey) return;
    if(event.isComposing) return; // 此次为输入法状态下的enter(即输入法合成状态),不发送内容
    event.preventDefault();
    sendMessage();
  }
}

这确实有效,但是在Safari下键盘输入文字(如输入英文)且在输入法合成状态,按下enter内容依旧被发送了!原因在于 event.isComposing 在Safari并不可信,导致不可用;

延展

那么解决Safari上这个问题,我们必须Safari上实现可信的isComposing;

尝试通过监听 composotionstart/compositionend 实现自己的 Composing 状态,但事实是不可行(Safari的composition事件与keydown、input等事件执行顺序问题);

但是input事件有 inputType 可判断到输入法状态 MDN inputType

最终实现:额外监听input事件(先于keydown事件),针对Safari使用自己实现的Composing状态判断是否为输入法合成下enter

html 复制代码
<textarea
    @input="onInput"
    @keydown="onInputKeydown"
    placeholder="请输入内容,按 Enter 发送,Shift + Enter 换行"
></textarea>
js 复制代码
// 引入浏览器检测库
import { UAParser } from 'ua-parser-js';
const parser = new UAParser();
const result = parser.getResult();

// 回车发送
const onInputKeydown = (event) => {
  if(event.key === "Enter" || event.keyCode === '13') {
    if(event.shiftKey) return;
    // 针对safari浏览器兼容
    if(result.browser.name === "safari") {
      if(isCompositionInputType) { // 使用自己实现的Composing状态
        isCompositionInputType = false;
        return;
      }
    }else {
      if(event.isComposing) return;
    }
    event.preventDefault();
    sendMessage(); // 发送
  }
}

// 手动实现判断是否为输入法合成状态
let isCompositionInputType = false;
const onInput = (event) => {
  isCompositionInputType = ["insertCompositionText", "insertFromComposition"].includes(event.inputType);
}
相关推荐
子琦啊2 分钟前
构造函数、this指向和原型链机制
javascript·算法·贴图
Maimai1080826 分钟前
React 多步骤表单工程化落地:从 Zod Schema、React Hook Form 到 Zustand 持久化
前端·javascript·react.js·前端框架·状态模式
程序员码歌27 分钟前
我是怎么部署开源 AI 编程助手 OpenCode,并在两个真实场景使用起来的
前端·人工智能·后端
Maimai1080829 分钟前
React Query + Zustand 正确结合方式:不要把接口数据复制进 Store
前端·javascript·react.js·前端框架·web3·状态模式
天才熊猫君31 分钟前
层叠上下文 z-index 的简单理解
前端
i220818 Faiz Ul32 分钟前
智慧养老平台|基于SprinBoot+vue的智慧养老平台系统(源码+数据库+文档)
java·前端·数据库·vue.js·spring boot·毕设·智慧养老平台
AI砖家32 分钟前
每日一个skill:web-artifacts-builder,构建复杂 Claude.ai HTML Artifact 的生产力工具包
java·前端·人工智能·python
icc_tips35 分钟前
Flutter runAppAsync() 详解:干净的异步应用启动
前端·flutter
转转技术团队36 分钟前
AI新名词比我头发掉得还快
前端
Lkstar37 分钟前
Pinia 进阶:Setup Store、插件系统与状态持久化,一篇全搞懂
前端·vue.js