惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

博客园 - Franky
N
Netflix TechBlog - Medium
Google Online Security Blog
Google Online Security Blog
月光博客
月光博客
量子位
酷 壳 – CoolShell
酷 壳 – CoolShell
V
V2EX
腾讯CDC
OSCHINA 社区最新新闻
OSCHINA 社区最新新闻
博客园 - 聂微东
让小产品的独立变现更简单 - ezindie.com
让小产品的独立变现更简单 - ezindie.com
M
MIT News - Artificial intelligence
Vercel News
Vercel News
The GitHub Blog
The GitHub Blog
Hugging Face - Blog
Hugging Face - Blog
博客园 - 【当耐特】
Apple Machine Learning Research
Apple Machine Learning Research
aimingoo的专栏
aimingoo的专栏
博客园 - 三生石上(FineUI控件)
CTFtime.org: upcoming CTF events
CTFtime.org: upcoming CTF events
MongoDB | Blog
MongoDB | Blog
H
Help Net Security
The Cloudflare Blog
Blog — PlanetScale
Blog — PlanetScale
F
Full Disclosure
G
Google Developers Blog
罗磊的独立博客
Jina AI
Jina AI
钛媒体:引领未来商业与生活新知
钛媒体:引领未来商业与生活新知
Y
Y Combinator Blog
H
Hackread – Cybersecurity News, Data Breaches, AI and More
J
Java Code Geeks
A
About on SuperTechFans
IT之家
IT之家
大猫的无限游戏
大猫的无限游戏
S
SegmentFault 最新的问题
有赞技术团队
有赞技术团队
GbyAI
GbyAI
雷峰网
雷峰网
T
The Blog of Author Tim Ferriss
The Register - Security
The Register - Security
U
Unit 42
D
Docker
Martin Fowler
Martin Fowler
L
LINUX DO - 热门话题
NISL@THU
NISL@THU
阮一峰的网络日志
阮一峰的网络日志
C
Cybersecurity and Infrastructure Security Agency CISA
博客园_首页
Google DeepMind News
Google DeepMind News

Fooleap's Blog

渴望理想 | Fooleap's Blog 19 年底一些事 | Fooleap's Blog 藉秋风,跑起来 | Fooleap's Blog 一个人去跑步 | Fooleap's Blog 8 月的跑量仿佛是那暑假的作业 | Fooleap's Blog 三伏天跑步那么难受,为何还要跑? | Fooleap's Blog 解决小程序开发“当前系统代理不是安全代理” | Fooleap's Blog 忘却配速的夏日跑步 | Fooleap's Blog 六月天时“带水”跑步更爽 | Fooleap's Blog 出来混迟早要还的 | Fooleap's Blog 跑步不能当饭吃 | Fooleap's Blog 将京东移动端详情页链接转为 PC 端 | Fooleap's Blog 不是热就是雨 | Fooleap's Blog 这半月,我跑了 11 个 520 | Fooleap's Blog 跑完流汗一时爽,一直流汗一直爽 | Fooleap's Blog 初夏夜跑 | Fooleap's Blog Electron 中打开 QQ 临时会话 | Fooleap's Blog 春节期间的培隆角 | Fooleap's Blog 从春天跑到夏天 | Fooleap's Blog 晨雾中奔跑 | Fooleap's Blog 漫步春雨中 | Fooleap's Blog 跑在木棉花下 | Fooleap's Blog 我在春节依然坚持跑步 | Fooleap's Blog 伴随着日出跑步 | Fooleap's Blog 没有最好,只有更好 | Fooleap's Blog 电子气温计 | Fooleap's Blog 渡亭小学的金凤花 | Fooleap's Blog 随心而跑 | Fooleap's Blog 2018 跑步小结 | Fooleap's Blog 在 gVim 中使用“非等宽字体” | Fooleap's Blog 动车进汕,喜大普奔 | Fooleap's Blog 雨战汕马,漫步鮀城 | Fooleap's Blog 准备出发汕马 | Fooleap's Blog 环苏溪跑个半马 | Fooleap's Blog 不义之财 | Fooleap's Blog 跑去培隆看日落 | Fooleap's Blog 雨后跑土路 | Fooleap's Blog 秋意渐浓 | Fooleap's Blog 在夕阳下奔跑 | Fooleap's Blog 准备参加 2018 汕马 | Fooleap's Blog 不可立见的 spoiler 标签 | Fooleap's Blog TomTom Spark 表带 | Fooleap's Blog Disqus 支持新浪微博图床 | Fooleap's Blog 暂存 Disqus 匿名评论者邮箱地址 | Fooleap's Blog 组一台迷你主机 DeskMini 310 | Fooleap's Blog 我的个人信息卖给了谁? | Fooleap's Blog 我发了违法短信? | Fooleap's Blog 使用 Python 合并地图瓦片 | Fooleap's Blog 使用 Python 合并瓦片图 | Fooleap's Blog 拆电热水壶 | Fooleap's Blog 使用树莓派做监控显示 | Fooleap's Blog 南方的冷 | Fooleap's Blog 蓝牙耳机 Avantree Jogger Plus | Fooleap's Blog 新厝布网 | Fooleap's Blog 报装移动宽带 | Fooleap's Blog 双十一战绩 | Fooleap's Blog 郁闷的心情 | Fooleap's Blog 像 Disqus 一样获取链接颜色 | Fooleap's Blog Disqus 的 URL 编码问题 | Fooleap's Blog 选择框的全选联动 | Fooleap's Blog 弹出层中的视频全屏问题 | Fooleap's Blog 2016 年台风海马 | Fooleap's Blog 纯 CSS 实现导航图标动画 | Fooleap's Blog 湾头晨跑路线推荐:南湾小学跑道 | Fooleap's Blog Jekyll 显示每一年的文章数 | Fooleap's Blog 湾头晨跑路线推荐:南湾堤顶 | Fooleap's Blog Disqus 的 @ 提及功能 | Fooleap's Blog 近日渡亭堤顶的夕阳 | Fooleap's Blog 使用 Disqus API 上传图片 | Fooleap's Blog Disqus API 评论嵌套问题 | Fooleap's Blog Disqus API 的权限问题 | Fooleap's Blog 湾头晨跑路线推荐:环三湾 | Fooleap's Blog 如何下载 Apple Emoji 的 PNG 图片 公众号文章二维码 | Fooleap's Blog Disqus 的评论预审核 | Fooleap's Blog 湾头最好的跑道 | Fooleap's Blog 结合七牛和高德地图 API 显示照片位置 | Fooleap's Blog Zip 压缩排除特定目录 | Fooleap's Blog 流水涸摸蚬热 | Fooleap's Blog 2017 跨年跑 | Fooleap's Blog Jekyll 的中文字数统计 | Fooleap's Blog 为 Jekyll 文章页添加相关文章 | Fooleap's Blog 为 Jekyll 添加一个标签页面 | Fooleap's Blog 干了这瓶蛇草水 | Fooleap's Blog 在 macOS 上使用 NTFS 差点丢数据 Disqus Moderator Badge Text 已支持中文 为 Jekyll 添加一个简单的 API | Fooleap's Blog 为 Jekyll 加上简单搜索功能 | Fooleap's Blog 解决 Jemoji 的出错 | Fooleap's Blog 检测网络是否能够访问 Disqus | Fooleap's Blog 解决 This socket is closed 问题 更好的 Markdown 插图方式 | Fooleap's Blog 转换 Nike+ 的坐标数据 | Fooleap's Blog 高德地图 API 显示跑步路线 | Fooleap's Blog 善用 Google 搜索工具 | Fooleap's Blog 利用 Nike+ API 获取跑步路线数据 | Fooleap's Blog 七牛 API 生成页面 URL 二维码 旧年 12 月跑步笔记 | Fooleap's Blog 科学使用 Disqus | Fooleap's Blog 培隆角的日出 | Fooleap's Blog
JavaScript 实现 Vim 键绑定 | Fooleap's Blog
fooleap · 2015-09-20 · via Fooleap's Blog

一旦习惯 Vim 的热键操作,网页浏览也会想用熟悉的热键操作,有人为热门浏览器开发了扩展,例如 Firefox 有个人觉得近完美的扩展解决方案 Vimperator 或 Pentadactyl,Chrome 则有不大完美的 Vichrome、Vimium 等[1],还有些人直接写了类 vi 的浏览器,例如 dwb[2],不过我想没用过 vi 的人不会去尝试这样的扩展或软件。

dwb 界面 dwb 界面

那么,能不能直接使用 JavaScript,在网站中实现热键操作,让访问你网站的人都能享受,答案是可以的,知乎的快捷键就有些 vi 的感觉[3]

知乎的快捷键 知乎的快捷键

功能介绍

小博在右下角弄个回到顶部按钮,对长文而言,使用鼠标慢慢滚,拉到下面,一点就回到顶部,感觉方便的,但有时候还是感觉有些多余。键盘也可用空格或 Home End 等键,但我想说离得太远了,如若和鄙人一样使用 Poker II 等迷你键盘更是麻烦,何不采用 Vim 的热键来操作呢?使用 g g 来返回顶部不是更方便么?所以,鄙人就想使用 JavaScript 实现这么几个简单的键绑定:

热键 操作
j 下滚一行
k 上滚一行
2 j 下滚 2 行
3 k 上滚 3 行
g g 回到顶部
G (Shift + g) 滚到底部

注:2 j 中数字可以为任意正整数。

键盘事件

键盘操作自然离不开监听键盘事件,键盘事件有三个,分别是

事件 说明 属性
keydown 按下按键 keyCode
keypress 字符键按下 charCode
keyup 释放按键 keyCode

注:其中 keypress 事件是按下可输入字符的键就会触发,例如 abc,若按下 CtrlShift 等则不触发。

触发事件会返回对象,其中 keydownkeyup 事件会返回 keyCode 属性,有些浏览器比如 Firefox 还会返回 key 属性,但一般还是使用兼容性比较好的 keyCodekeypress 事件则会返回 charCode 属性。

其中 keyCode 的值可以在 MDN 找到[4]charCode 则区分大小写,属性值基本等同于 ASCII 码[5]

JavaScript Event KeyCodes[6]这个网站可以方便获取 keyCode 值,另外可以使用代码自己获取 keyCode

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

charCode 值则可用下面代码获得:

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

说了这么多,上面所说需要用到的几个键,其 keyCodecharCode 如下:

按键 keyCode charCode
Shift 16
0 - 9 48 - 57 48 - 57
G 71 71
g 71 103
j 74 106
k 75 107

注:均为数字,非字符。

实现过程

上下滚一行

看几个快捷键操作,jk 想必是最简单的,其余几个可能没有那么简单,所以先从简单的出发。

监听 keydown 事件,如果按下 j, k 键,使用 scrollBy() 方法实现上滚下滚一行[7],假定行高为 20 个像素,那么:

function keysDown(event) {
  if (event.keyCode == 74) {
    window.scrollBy(0, 20);
  }
  if (event.keyCode == 75) {
    window.scrollBy(0, -20);
  }
}
window.addEventListener('keydown', keysDown, false);

上下滚 n 行

数字的 keyCode 并不等于原数字,所以这里引入一个数组,当输入为数字时,将数组内对应值赋值为原数字。每次输入数字的时候,还要结合此前输入的数字进行计算,结合上面的上下滚一行,实现如下:

var lineHeight = 20; // 行高,可设置
var keys = [];
var row = 0;
function keysDown(event) {
  if (event.keyCode == 74) {
    if (row) {
      window.scrollBy(0, lineHeight * row );
      row = 0;
    } else {
      window.scrollBy(0, lineHeight);
    }
  }
  if (event.keyCode == 75) {
    if (row) {
      window.scrollBy(0, -lineHeight * row );
      row = 0;
    } else {
      window.scrollBy(0, -lineHeight);
    }
  }
  if (event.keyCode >= 48 && event.keyCode <= 57) {
    for (var i = 48; i <= 57; i ++) {
      keys[i] = i - 48;
    }
    row = parseInt(row.toString() + keys[event.keyCode].toString());
  }
}
window.addEventListener('keydown', keysDown, false);

回到顶部

Vim 的普通模式下按下 g g,即双击 g,就回到顶部,双击一般有一个延时的处理,虽然看起来挺简单,但凭空想象还是需要时间的,二话不说先 Google 一番,很快便在 Stack Overflow 找到了解决方法[8],不得不说这方法有些小聪明。

返回顶部用到了 scrollTo() 方法[9],具体代码可以这样写:

var inCombo = false;
function keysDown(event) {
  if (event.keyCode == 71) {
    if (!inCombo) {
      inCombo = true;
      // 五百毫秒的延时,即需在半秒内完成双击
      setTimeout('inCombo = false;', 500);
    } else {
      window.scrollTo(0, 0);
    }
  }
}
window.addEventListener('keydown', keysDown, false);

滚到底部

滚到底部是 Shift + g,也即是组合键操作,不得不说 Google 才是人类的未来,一搜便可[10]

这里滚到底部需要获取整个网页的高度,其实这是个大坑,offsetHeightscrollHeight 傻傻分不清,各浏览器的标准也有区别,这里就随便丢了算是整个网页高度的 document.body.scrollHeight,其实真的很想简单粗暴丢个 99999

var keys = [];
function keysDown(event) {
  keys[event.keyCode] = true;
  if (keys[16] && keys[71]) {
    window.scrollTo(0, document.body.scrollHeight);
  }
}
function keysUp(event) {
  keys[event.keyCode] = false;
}
window.addEventListener('keydown', keysDown, false);
window.addEventListener('keyup', keysUp, false);

完整代码

整合以上功能,且为一些浏览器做兼容之后[11],完整代码如下:

var inCombo = false;
var lineHeight = 20;
var keys = [];
var row = 0;
function keysDown(event) {
  keys[event.keyCode] = true;
  if (keys[16] && keys[71]) {
    window.scrollTo(0, document.body.scrollHeight );
  }
  if (keys[71]) {
    if (!inCombo) {
      inCombo = true;
      setTimeout('inCombo = false;', 500);
    } else {
      window.scrollTo(0, 0);
    }
  }
  if (keys[74]) {
    if (row) {
      window.scrollBy(0, lineHeight * row );
      row = 0;
    } else {
      window.scrollBy(0, lineHeight);
    }
  }
  if (keys[75]) {
    if (row) {
      window.scrollBy(0, -lineHeight * row );
      row = 0;
    } else {
      window.scrollBy(0, -lineHeight);
    }
  }
  if (event.keyCode >= 48 && event.keyCode <= 57) {
    for (var i = 48; i <= 57; i ++) {
      keys[i] = i - 48;
    }
    row = parseInt(row.toString() + keys[event.keyCode].toString());
  }
}
function keysUp(event) {
  keys[event.keyCode] = false;
}
if(window.addEventListener){
  window.addEventListener('keydown', keysDown, false);
  window.addEventListener("keyup", keysUp, false);
} else if (document.attachEvent){
  document.attachEvent('onkeydown', keysDown);
  document.attachEvent('onkeyup', keysUp);
} else {
  document.addEventListener('keydown', keysDown, false);
  document.addEventListener("keyup", keysUp, false);
}

上面说了那么多,原来也不需要用到 keypress,如果做到严格一点,区分大小写,或许 keypress 就派上用场。不过我想现在大多数人的大写锁定键基本不用到,所以也没必要较劲。为啥不实现 Ctrl + fCtrl + b 等?因为 Ctrl + * 本身就是浏览器的热键,还需考虑如何屏蔽。

另外,可以用 jQuery 简单实现平滑滚动而非直接定位,或者也可自己写,在此鄙人不想折腾了。

哦,对了,鄙人已将代码部署到本博,顺便学一下知乎,加按 c 查看评论,赶紧使用 g g 试试吧~

参考资料

本文历史

  • 2015 年 09 月 20 日 完成初稿

最近更新

猜你喜欢