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);
}
相关推荐
weixin199701080167 分钟前
《施耐德商品详情页前端性能优化实战》
前端·性能优化
不想上班只想要钱18 分钟前
模板里 item.xxx 报错 ,报 item的类型为未知
前端·vue
妖萌妹儿1 小时前
postman怎么做参数化批量测试,测试不同输入组合
开发语言·javascript·postman
阿琳a_1 小时前
在github上部署个人的vitepress文档网站
前端·vue.js·github·网站搭建·cesium
酉鬼女又兒1 小时前
零基础快速入门前端ES6 核心特性详解与蓝桥杯 Web 考点实践(可用于备赛蓝桥杯Web应用开发)
开发语言·前端·职场和发展·蓝桥杯·es6·css3·html5
Zk.Sun1 小时前
【RK3588 Mali610 适配 Qt6 】
前端·javascript·vue.js
不想吃菠萝1 小时前
vue3+ts 使用postcss-pxtorem依赖进行rem适配
前端·javascript·vue.js·postcss
人民广场吃泡面1 小时前
React新手快速入门学习指南(2026最新版)
前端·react.js·前端框架
kyriewen111 小时前
本地存储全家桶:从localStorage到IndexedDB,把数据塞进用户浏览器
开发语言·前端·javascript·ecmascript·html5
油丶酸萝卜别吃2 小时前
本地调试跨域问题:关闭 Chrome 同源策略的技巧
前端·chrome