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);
}
相关推荐
韩曙亮几秒前
【Web APIs】鼠标经过、离开事件 ( mouseover、mouseout 事件 | mouseenter、mouseleave 事件 )
前端·javascript·web apis·mouseover·mouseout·mouseenter·mouseleave
brzhang1 分钟前
MCP A2A Skills 这三个词搞懂了 再去写你的智能体
前端·后端·架构
灰灰勇闯IT2 分钟前
RN原生模块交互:打通JS与原生的桥梁
开发语言·javascript·交互
LYFlied4 分钟前
浅谈跨端开发:大前端时代的融合之道
前端·flutter·react native·webview·大前端·跨端开发·hybrid
LYFlied4 分钟前
浅谈前端构建工具核心理解&&主流工具对比
前端·webpack·软件构建·rollup·vite·开发工具·工程化
weixin_3077791318 分钟前
Jenkins jQuery3 API 插件详解:赋能插件前端开发的利器
运维·开发语言·前端·jenkins·jquery
LinDon_27 分钟前
【企业微信快速登录适配 Chrome/Edge 142+】
前端·chrome·企业微信
JosieBook34 分钟前
【Vue】google chrome中安装vue_dev_tools.crx的时候提示“无法安装扩展程序,因为它使用了不受支持的清单版本。”
前端·vue.js·chrome
前端不太难36 分钟前
RN 性能优化:列表滚动掉帧、卡顿怎么办?
前端·react native·性能优化
亿元程序员37 分钟前
祖传项目二开快上线了,却还有很多旧的资源,怎么办?
前端