use key-queue for input

jh-changes
Shin'ya Ueoka 7 years ago
parent e7342e9c23
commit 082450928a
  1. 56
      src/background/index.js
  2. 28
      src/background/key-queue.js
  3. 21
      src/background/keys.js
  4. 8
      src/background/tabs.js
  5. 23
      src/content/index.js
  6. 8
      src/content/scrolls.js
  7. 22
      src/shared/actions.js

@ -1,28 +1,46 @@
import * as actions from '../shared/actions';
import * as tabs from './tabs'; import * as tabs from './tabs';
import KeyQueue from './key-queue';
const KEY_MAP = { const queue = new KeyQueue();
'tabs.prev': KeyboardEvent.DOM_VK_H,
'tabs.next': KeyboardEvent.DOM_VK_L,
'scroll.up': KeyboardEvent.DOM_VK_K,
'scroll.down': KeyboardEvent.DOM_VK_J
};
browser.runtime.onMessage.addListener((request, sender, sendResponse) => { const keyDownHandle = (request) => {
let response = null; return queue.push({
code: request.code,
shift: request.shift,
ctrl: request.ctrl,
alt: request.alt,
meta: request.meta
})
}
switch (request.code) { const doBackgroundAction = (sender, action) => {
case KEY_MAP['tabs.prev']: switch(action[0]) {
tabs.selectPrevTab(sender.tab.index); case actions.TABS_PREV:
break; tabs.selectPrevTab(sender.tab.index, actions[1] || 1);
case KEY_MAP['tabs.next']:
tabs.selectNextTab(sender.tab.index);
break; break;
case KEY_MAP['scroll.up']: case actions.TABS_NEXT:
response = 'scroll.up' tabs.selectNextTab(sender.tab.index, actions[1] || 1);
break; break;
case KEY_MAP['scroll.down']: }
response = 'scroll.down' }
browser.runtime.onMessage.addListener((request, sender, sendResponse) => {
let action = null;
switch (request.type) {
case 'event.keydown':
action = keyDownHandle(request);
break; break;
} }
sendResponse(response);
if (action == null) {
return;
}
if (actions.isBackgroundAction(action[0])) {
doBackgroundAction(sender, action);
} else if (actions.isContentAction(action[0])) {
sendResponse(action);
}
}); });

@ -0,0 +1,28 @@
import * as keys from './keys';
import * as actions from '../shared/actions';
const DEFAULT_KEYMAP = [
{ keys: [{ code: KeyboardEvent.DOM_VK_K }], action: [ actions.SCROLL_UP, 1 ]},
{ keys: [{ code: KeyboardEvent.DOM_VK_J }], action: [ actions.SCROLL_DOWN, 1 ]},
{ keys: [{ code: KeyboardEvent.DOM_VK_H }], action: [ actions.TABS_PREV, 1 ]},
{ keys: [{ code: KeyboardEvent.DOM_VK_L }], action: [ actions.TABS_NEXT, 1 ]},
]
export default class KeyQueue {
constructor(keymap) {
this.data = [];
this.keymap = keymap;
}
push(key) {
this.data.push(key);
for (let map of DEFAULT_KEYMAP) {
if (keys.keysEquals(map.keys, this.data)) {
this.data = [];
return map.action
}
}
return null;
}
}

@ -0,0 +1,21 @@
const keyEquals = (key1, key2) => {
return (key1.code === key2.code) &&
((key1.shift || false) === (key2.shift || false)) &&
((key1.ctrl || false) === (key2.ctrl || false)) &&
((key1.alt || false) === (key2.alt || false)) &&
((key1.meta || false) === (key2.meta || false));
};
const keysEquals = (keys1, keys2) => {
if (keys1.length !== keys2.length) {
return false;
}
for (let i = 0; i < keys1.length; ++i) {
if (!keyEquals(keys1[i], keys2[i])) {
return false;
}
}
return true;
}
export { keyEquals, keysEquals };

@ -1,20 +1,20 @@
const selectPrevTab = (current) => { const selectPrevTab = (current, count) => {
chrome.tabs.query({ currentWindow: true }, (tabs) => { chrome.tabs.query({ currentWindow: true }, (tabs) => {
if (tabs.length < 2) { if (tabs.length < 2) {
return; return;
} }
let select = (current - 1) % tabs.length let select = (current - count) % tabs.length
let id = tabs[select].id; let id = tabs[select].id;
chrome.tabs.update(id, { active: true }) chrome.tabs.update(id, { active: true })
}); });
}; };
const selectNextTab = (current) => { const selectNextTab = (current, count) => {
chrome.tabs.query({ currentWindow: true }, (tabs) => { chrome.tabs.query({ currentWindow: true }, (tabs) => {
if (tabs.length < 2) { if (tabs.length < 2) {
return; return;
} }
let select = (current + 1 + tabs.length) % tabs.length let select = (current + count + tabs.length) % tabs.length
let id = tabs[select].id; let id = tabs[select].id;
chrome.tabs.update(id, { active: true }) chrome.tabs.update(id, { active: true })
}); });

@ -1,28 +1,33 @@
import * as scrolls from './scrolls'; import * as scrolls from './scrolls';
const invokeEvent = (type) => { const invokeEvent = (action) => {
switch (type) { if (typeof action === 'undefined' || action === null) {
return;
}
switch (action[0]) {
case 'scroll.up': case 'scroll.up':
scrolls.scrollUp(window); scrolls.scrollUp(window, action[1] || 1);
break; break;
case 'scroll.down': case 'scroll.down':
scrolls.scrollDown(window); scrolls.scrollDown(window, action[1] || 1);
break; break;
} }
} }
window.addEventListener("keydown", (e) => { window.addEventListener("keydown", (e) => {
browser.runtime.sendMessage({ let request = {
type: 'event.keydown',
code: e.keyCode, code: e.keyCode,
shift: e.shift, shift: e.shift,
alt: e.alt, alt: e.alt,
meta: e.meta, meta: e.meta,
ctrl: e.ctrl, ctrl: e.ctrl,
}).then((response) => {
if (response) {
invokeEvent(response);
} }
}, (err) => {
browser.runtime.sendMessage(request)
.then(invokeEvent,
(err) => {
console.log(`Vim Vixen: ${err}`); console.log(`Vim Vixen: ${err}`);
}); });
}); });

@ -1,14 +1,14 @@
const SCROLL_DELTA = 48; const SCROLL_DELTA = 48;
const scrollUp = (page) => { const scrollUp = (page, count) => {
let x = page.scrollX; let x = page.scrollX;
var y = page.scrollY - SCROLL_DELTA; var y = page.scrollY - SCROLL_DELTA * count;
page.scrollTo(x, y); page.scrollTo(x, y);
}; };
const scrollDown = (page) => { const scrollDown = (page, count) => {
let x = page.scrollX; let x = page.scrollX;
var y = page.scrollY + SCROLL_DELTA; var y = page.scrollY + SCROLL_DELTA * count;
page.scrollTo(x, y); page.scrollTo(x, y);
}; };

@ -0,0 +1,22 @@
export const TABS_PREV = 'tabs.prev';
export const TABS_NEXT = 'tabs.next';
export const SCROLL_UP = 'scroll.up';
export const SCROLL_DOWN = 'scroll.down';
const BACKGROUND_ACTION_SET = new Set([
TABS_PREV,
TABS_NEXT
]);
const CONTENT_ACTION_SET = new Set([
SCROLL_UP,
SCROLL_DOWN
]);
export const isBackgroundAction = (action) => {
return BACKGROUND_ACTION_SET.has(action);
};
export const isContentAction = (action) => {
return CONTENT_ACTION_SET.has(action);
};