告别JS keyCode

这篇文章发布于 2021年01月4日,星期一,23:29,归类于 JS API。 阅读 32454 次, 今日 7 次 25 条评论

 

键盘素材占位图封面

一、为什么keyCode不推荐使用了?

对于这个问题的答案,说实话,我找了很多资料,并没有特别明确的回答。

早些年,我得到的说法是,用户可能会自定义键盘,导致keyCode不准。

但是,这次搜寻资料,没有见到类似的说法。

MDN上的解释是对打印字符不友好。

我琢磨着可能是这几个意思:

1. 不同字符共用keyCode

键盘上有很多按键是同时对应两个字符的,例如“<,”和“>.”就在一个键上,数字键那里(1!、2@、3#、4$、5%、……)都是一个键对应多个字符,如果想要输入两位字符,往往需要按住Shift键才行,例如字符“<>!@#$%^&*()……”等字符的输入。

keyCode值是跟着键盘走的,而不是字符内容,也就是,当我们输入“<,”和“>.”等字符的时候,返回的keyCode值是一样的,我们需要通过判断用户是否按下了Shift键才知道究竟输入的是哪个字符。

这就很啰嗦。

例如:

window.addEventListener('keydown', function (event) {
  console.log(event.keyCode);
});

然后在页面中分别输入字符“.>”,则会看到输出的keyCode值都是一样的190,如下截图所示:

keyCode都是190示意

明明是两个不同的字符,结果输出的keyCode值是一样的,对开发就不友好。

2. 相同按键不同keyCode

例如全键盘中的数字键按住Shift键可以输出其他内容,例如下图所示的数字键盘。

数字键盘

例如右下角的小数点字符'.'同时有Del键的能力。

此时,按下此键和Shift+按下此键的keyCode值是不一样的。

同一按键不同keyCode值

然后大家就会发现按下'.'返回的keyCode是110,加了个Shift键之后返回的就变成了46了。

这就蛋疼了,因为上面同样按键返回同样keyCode,这里居然摇身一变,相同按键返回了不同的keyCode。

3. 相同字符不同keyCode

还没完,Shift+'.'返回keyCode就是直接按下delete键的keyCode值,给人感觉只要最终行为或字符输出是一样的,keyCode值也应该是一样的,实际上,并不是这样的。

即使输入同一字符,也可能会是不同的keyCode值。

这里大家注意力放在下图中右上方的加号和减号上。

同一按键不同keyCode值

同时,上方的一排数字键那里也是有加号和减号的,如下图所示:

加号和减号code值对比示意

此时,分别按下这两个键,会发现输出的字符是一模一样的,但是keyCode值却是不同的,如下截图示意。

短横线和减号不同字符

上方数字键那里的短横线连字符的keyCode是189,而数字键盘那里的减号字符的keyCode是109,实际上这两个字符一模一样。

试想下,如果开发过程中判断用户是否按下的是连字符,使用keyCode判断,是不是很容易出bug?

其他的几个数字可有类似的问题,例如数字键盘输出的1-9的keyCode和键盘上面1-9数字的keyCode值是不一样的。

4. 中文输入法下标点符号keyCode都是一样的

例如输入框中是中文输入法,此时,“,。;‘【】-=”这些字符的keyCode全部都返回229,根本就没法继续玩了。

中文输入法均返回keyCode 229

补充于2023-02-08

最近发现,中文输入法开启时候keyCode值是229还是有用的。

当我们希望监听输入框是否按下Enter键触发submit行为的时候,使用event.keyCode 可以避免输入法回车(使用英文)不小心触发submit事件的问题。

如果使用event.code或event.key,无论输入法是否开启,返回的都是 ‘Enter’,就会导致中文输入场景下,内容还没写入就被提交的情况出现。


或许是因为上面指出的4个keyCode对打印字符不友好的地方,keyCode才不推荐使用,目前规范推荐使用event.code或event.key。

二、event.code和event.key的区别

event.code指明按下的是具体哪个物理键,键盘上每一个按键都对应一个唯一的event.code值,均使用大写英文单词表示。

event.key指明具体输入的字符内容,如果是非打印字符(例如Enter键、Esc键、Shift键、Alt键等),则返回具体的非打印字符的英文名称,如果输入内容与输入法有关则返回固定的Process名称。

为了方便大家快速了解差异,我选取了几个具有代表性的按键,整理了个表格,显示了不同按键下event.codeevent.key的值。

keyCode值 code值 key值 描述
49 ‘Digit1’ ‘1’ 上方数字键1按下
97 ‘Numpad1’ ‘1’ 小键盘数字键1按下
16 ‘ShiftLeft’ ‘Shift’ 左侧的Shift键
16 ‘ShiftRight’ ‘Shift’ 右侧的Shift键
190 ‘Period’ ‘.’ 主键盘中的点符号
110 ‘NumpadDecimal’ ‘.’ 数字键盘中的小数点符号
229 ‘Period’ ‘Process’ 中文输入法下主键盘中的点符号
229 ‘Minus’ ‘Process’ 中文输入法下主键盘中的’-‘符号
189 ‘Minus’ ‘-‘ 主键盘中的’-‘符号
109 ‘NumpadSubtract’ ‘-‘ 数字键盘中的’-‘符号

对于英文场景,只需要使用event.key就可以知道键盘输入的内容了。

但是如果是中文场景,情况就变得复杂的多。

在中文输入框开启的场景下,如果按键的内容和非中文输入法下的内容不一样,则event.key的返回值是固定的Process,表示输入的字符内容和键盘对应的原始内容进行了处理。

例如主键盘中的字符点默认情况下就是个.,但是如果是中文输入框,字符点就是句号,而键盘上完全就没有句号这个字符,说明字符点被输入法给处理了,因此返回的就是Process。

这就导致,在中文输入法场景下,用户或者开发者是无法知道按键应该输入的内容的。

即使配合event.code也不行。

有人可能会反驳,event.code返回’Period’,不就表明按下的是点号键,此时event.key是’Process’,不就可以判断输入的是句号了。

理论上可行,实操起来却比预想的麻烦,且不具有可复制性。

  • 麻烦在于,如果用户输入的是这个符号,也就是右书名号,event.code返回的也是’Period’,event.key也是’Process’。还需要开发者判断event.shiftKey是否为true,表示Shift键是否按下。

    以及,换个其他语言输入法,则按下’Period’键返回的就不一定是句号了。

  • 不具有可复制性在于,中文输入法经常会使用空格或者回车表示选中,此时,event.code是Space或者Enter,event.key的值是Process,根本无法判断到底输入的是什么。

所以,event.codeevent.key这两个不适合中文输入法下的输入判断。

好在,实际开发中,很少有场景需要提前知道用户是否开启了中文输入法。

更多的是一些功能键的判断。

例如空格、回车、删除、上下左右键、上一页下一页键,home/end键,ESC键等。

因此,接下来给大家罗列以下常见的功能键对应的event.keyevent.code值。

三、常见功能键key值

功能键key值更实用,因此放在前面展示。

详见下表:

按键名称 event.key keyCode值
回车 Enter 13
delete删除 Delete 46
backspace退格 Backspace 8
esc取消 Escape 27
tab索引 Tab 9
ArrowUp 38
ArrowDown 40
ArrowLeft 37
ArrowRight 39
pageDown下一页 PageDown 34
pageUp上一页 PageUp 33
home键 Home 36
end键 End 35
shift键 Shift 16
control键 Control 17
alt键 Alt 18

KeyboardEvent.key值兼容性要比KeyboardEvent.code值好一些,如下所示:

KeyboardEvent.key兼容性

IE浏览器勉强支持,返回的值可能和规范中定义的有出入。如果要兼容IE浏览器,兼容可以用keyCode属性撑一会儿。

四、常见功能键的code值

如下表所示,大部分返回值和key值是一样的,因为都是功能键,如果是可打印字符,则code值和key值那就完全是两码事了:

按键名称 event.code 说明
回车 Enter
delete删除 Delete Shift+NumpadDecimal也可能是删除
backspace退格 Backspace
esc取消 Escape
tab索引 Tab
ArrowUp 38
ArrowDown 40
ArrowLeft
ArrowRight
pageDown下一页 PageDown
pageUp上一页 PageUp
home键 Home
end键 End
shift键 ShiftLeft/ShiftRight
control键 ControlLeft/ControlRight
alt键 AltLeft/AltRight

兼容性如下截图所示:

KeyboardEvent.code兼容性

五、送你一朵小红花

除了event.keyCode不推荐使用,event.which也不推荐使用了,官方名称为KeyboardEvent.which。

虽然不推荐使用,但是按照我的理解,99%的概率浏览器还会一直保持支持的,因为要是去掉这几个API特性,这世界上至少几百万个网站开发者会炸开锅。

当然,如果条件允许(不用考虑IE浏览器),我们还是优先使用event.key或者event.code来识别按键。

另外,相比原来的event.keyCode或者eevent.whichevent.keyevent.code要更好上手,例如上下左右键的几个数字,我老是记不住,总有run一下看看值是多少,但是如果是event.key,则直接使用语义化的英文单词即可。

只需要记住单词规则,首字母大小,每个分词首字母大写就OK了。

OK,以上就是本文全部内容,加深了下我自己对键盘事件的一些了解,整理了几个常用功能键表格,这个回头开发我会用到。MDN上的表格实在是冗长,不适合实战中使用。

这篇文章估计分享的人不会很多。

不管怎样,依然送你一朵小红花,感谢您的阅读与支持!

(本篇完)

分享到:


发表评论(目前25 条评论)

  1. 依云说道:

    应该说的是不同键盘布局上同样的 keyCode 对应的字符是不一样的。不过这个对中文用户没啥影响(大家都是 qwerty 布局)。倒是很多网页应用会判断 Esc 的 keydown 事件,然后我输入法打错字了要撤销呢,结果网页应用把我正在进行的编辑也给撤销了……

  2. 爸爸说道:

    很蛋疼的是中文输入法在打字的时候,按下回车也会直接触发Enter,而不是确认

    这就导致用户可能会误操作

    还是得用keyCode来判断

    • 依云说道:

      恰恰相反,我是看到有人用 Google Chrome 结果输入法里回车网页也回车,才找到这篇文章的。输入法在打字的时候按回车,你网页抢个啥啊……

  3. renmu说道:

    我最近在做类似的开发,遇到了即使都是Chrome,win和macOs下输入中文句号的表现是不一样的。

    在win下会返回“Process”,而在mac下返回的就是原有的标点。想要靠这种方法实现中文标点监控几乎不可能。
    有兴趣的可以去看这篇文章,里面有讲到 vscode 是如何解决键盘快捷键问题的。

  4. yuwei说道:

    张大湿,能否把图中那个键盘洗干净了再拍照,?,强迫症了

  5. 刷新说道:

    前些天vscode报错提示说keyCode被放弃,还一头雾水,但看Chrome还能用,就没在意了~今天就看到这个了,学习了

    • 依云说道:

      刚刚有人遇到了个输入法按回车相关的 bug,就 Google Chrome(以及各种基于 Electron 的应用)里有 bug,火狐好好的,哈哈。

  6. zj9495说道:

    在使用Reeder订阅rss阅读会出现以下文字

    //zxx: 如果你看到这段文字,说明你现在访问是体验糟糕的垃圾盗版网站,你可以访问原文获得很好的体验:https://www.zhangxinxu.com/wordpress/?p=9755(作者张鑫旭)

  7. unicloud说道:

    最近正好做了一个交互式的键码对照表,按什么键,就显示对应的 event.code, event.key 等信息,请大佬检阅 https://www.dute.org/keycodes

  8. MO说道:

    其实一直都是用keyCode判断的,接触过的场景里大多是监听回车而已~~

  9. 猪爸爸说道:

    测试了一些,中文输入法下「,。()——」等字符的 event.key 和 event.code 跟英文输入法下没区别,效果如下。

    https://pic4.zhimg.com/v2-d3e2a1732c6e9b6b617f278a4fd06609.png

    复现环境:https://432bq.csb.app/

  10. Zp说道:

    清楚keyCode的问题,才能解决输入监听的问题。学到了,赞。

  11. 某熊猫桑说道:

    中文场景……如果用`compositionend`呢?

    • 某某某说道:

      conpositionend,好东西。However,有一些诡异浏览器与输入法的组合会干一些莫名其妙的事情。

  12. livetune说道:

    有什么场景是需要判断中文符号的吗。。

  13. ClearScenery说道:

    这键帽可以

  14. DeathGhost说道:

    哈哈,我来阅读一番。

  15. 小鸭腿儿炒饭说道:

    大早上吃着早餐看着文章, 别有一番风味~~?

  16. 一个大胖子说道:

    学到了