aboutsummaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
Diffstat (limited to 'src')
-rw-r--r--src/background/commands.js2
-rw-r--r--src/background/index.js62
-rw-r--r--src/background/key-queue.js5
-rw-r--r--src/background/tabs.js22
-rw-r--r--src/content/footer-line.css46
-rw-r--r--src/content/footer-line.js78
-rw-r--r--src/content/index.js33
-rw-r--r--src/shared/actions.js8
8 files changed, 239 insertions, 17 deletions
diff --git a/src/background/commands.js b/src/background/commands.js
new file mode 100644
index 0000000..8bd52e5
--- /dev/null
+++ b/src/background/commands.js
@@ -0,0 +1,2 @@
+export const OPEN = 'open';
+export const TABOPEN = 'tabopen';
diff --git a/src/background/index.js b/src/background/index.js
index 604ea92..f3bd65a 100644
--- a/src/background/index.js
+++ b/src/background/index.js
@@ -1,21 +1,37 @@
import * as actions from '../shared/actions';
import * as tabs from './tabs';
+import * as commands from './commands';
import KeyQueue from './key-queue';
const queue = new KeyQueue();
-const keyDownHandle = (request) => {
- return queue.push({
+const keyDownHandle = (request, sender, sendResponse) => {
+ let action = queue.push({
code: request.code,
shift: request.shift,
ctrl: request.ctrl,
alt: request.alt,
meta: request.meta
- })
-}
+ });
+ if (!action) {
+ return;
+ }
+
+ if (actions.isBackgroundAction(action[0])) {
+ doBackgroundAction(sender, action);
+ } else if (actions.isContentAction(action[0])) {
+ sendResponse(action);
+ }
+};
const doBackgroundAction = (sender, action) => {
switch(action[0]) {
+ case actions.TABS_CLOSE:
+ tabs.closeTab(sender.tab.id);
+ break;
+ case actions.TABS_REOPEN:
+ tabs.reopenTab();
+ break;
case actions.TABS_PREV:
tabs.selectPrevTab(sender.tab.index, actions[1] || 1);
break;
@@ -25,22 +41,36 @@ const doBackgroundAction = (sender, action) => {
}
}
-browser.runtime.onMessage.addListener((request, sender, sendResponse) => {
- let action = null;
-
- switch (request.type) {
- case 'event.keydown':
- action = keyDownHandle(request);
- break;
+const normalizeUrl = (string) => {
+ try {
+ return new URL(string).href
+ } catch (e) {
+ return 'http://' + string;
}
+}
- if (action == null) {
+const cmdEnterHandle = (request, sender) => {
+ let words = request.text.split(' ').filter((s) => s.length > 0);
+ switch (words[0]) {
+ case commands.OPEN:
+ browser.tabs.update(sender.tab.id, { url: normalizeUrl(words[1]) });
+ return;
+ case commands.TABOPEN:
+ browser.tabs.create({ url: normalizeUrl(words[1]) });
return;
}
+};
- if (actions.isBackgroundAction(action[0])) {
- doBackgroundAction(sender, action);
- } else if (actions.isContentAction(action[0])) {
- sendResponse(action);
+browser.runtime.onMessage.addListener((request, sender, sendResponse) => {
+ switch (request.type) {
+ case 'event.keydown':
+ keyDownHandle(request, sender, sendResponse);
+ break;
+ case 'event.cmd.enter':
+ cmdEnterHandle(request, sender, sendResponse);
+ break;
+ case 'event.cmd.suggest':
+ // TODO make suggestion and return via sendResponse
+ break;
}
});
diff --git a/src/background/key-queue.js b/src/background/key-queue.js
index 666eec3..d753bc1 100644
--- a/src/background/key-queue.js
+++ b/src/background/key-queue.js
@@ -2,10 +2,15 @@ import * as keys from './keys';
import * as actions from '../shared/actions';
const DEFAULT_KEYMAP = [
+ { keys: [{ code: KeyboardEvent.DOM_VK_SEMICOLON, shift: true }], action: [ actions.CMD_OPEN ]},
+ { keys: [{ code: KeyboardEvent.DOM_VK_O }], action: [ actions.CMD_TABS_OPEN, false ]},
+ { keys: [{ code: KeyboardEvent.DOM_VK_O, shift: true }], action: [ actions.CMD_TABS_OPEN, true ]},
{ 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_G }, { code: KeyboardEvent.DOM_VK_G }], action: [ actions.SCROLL_TOP ]},
{ keys: [{ code: KeyboardEvent.DOM_VK_G, shift: true }], action: [ actions.SCROLL_BOTTOM ]},
+ { keys: [{ code: KeyboardEvent.DOM_VK_D }], action: [ actions.TABS_CLOSE ]},
+ { keys: [{ code: KeyboardEvent.DOM_VK_U }], action: [ actions.TABS_REOPEN]},
{ keys: [{ code: KeyboardEvent.DOM_VK_H }], action: [ actions.TABS_PREV, 1 ]},
{ keys: [{ code: KeyboardEvent.DOM_VK_L }], action: [ actions.TABS_NEXT, 1 ]},
]
diff --git a/src/background/tabs.js b/src/background/tabs.js
index 000bd7d..899284d 100644
--- a/src/background/tabs.js
+++ b/src/background/tabs.js
@@ -1,3 +1,23 @@
+const closeTab = (id) => {
+ browser.tabs.remove(id);
+};
+
+const reopenTab = () => {
+ browser.sessions.getRecentlyClosed({
+ maxResults: 1
+ }).then((sessions) => {
+ if (sessions.length === 0) {
+ return;
+ }
+ let session = sessions[0];
+ if (session.tab) {
+ browser.sessions.restore(session.tab.sessionId);
+ } else {
+ browser.sessions.restore(session.window.sessionId);
+ }
+ });
+};
+
const selectPrevTab = (current, count) => {
chrome.tabs.query({ currentWindow: true }, (tabs) => {
if (tabs.length < 2) {
@@ -20,4 +40,4 @@ const selectNextTab = (current, count) => {
});
};
-export { selectNextTab, selectPrevTab };
+export { closeTab, reopenTab, selectNextTab, selectPrevTab };
diff --git a/src/content/footer-line.css b/src/content/footer-line.css
new file mode 100644
index 0000000..041776c
--- /dev/null
+++ b/src/content/footer-line.css
@@ -0,0 +1,46 @@
+.vimvixen-footerline {
+ border-top: 1px solid gray;
+ bottom: 0;
+ box-sizing: border-box;
+ font-family: monospace;
+ font-size: 12px;
+ left: 0;
+ margin: 0;
+ padding: 0;
+ position: fixed;
+ right: 0;
+ z-index: 10000;
+}
+
+.vimvixen-footerline-title {
+ background-color: lightgray;
+ font-weight: bold;
+ margin: 0;
+ padding: 0;
+}
+
+.vimvixen-footerline-container-outer {
+ background-color: white;
+ position: relative;
+}
+
+.vimvixen-footerline-container-outer:before {
+ content: ':';
+ background-color: white;
+ float: left;
+ text-align: right;
+ width: 12px;
+}
+
+.vimvixen-footerline-container-inner {
+ position: absolute;
+ left: 12px;
+ right: 0;
+}
+
+.vimvixen-footerline-input {
+ margin: 0;
+ padding: 0;
+ width: 100%;
+ border: none;
+}
diff --git a/src/content/footer-line.js b/src/content/footer-line.js
new file mode 100644
index 0000000..fc1dc7b
--- /dev/null
+++ b/src/content/footer-line.js
@@ -0,0 +1,78 @@
+import './footer-line.css';
+
+export default class FooterLine {
+ constructor(doc, initial = '') {
+ this.initUi(doc);
+
+ this.enteredCallback = () => {}
+ this.promptChangeCallback = () => {}
+
+ this.input.addEventListener('blur', this.handleBlur.bind(this));
+ this.input.addEventListener('keydown', this.handleKeydown.bind(this));
+ this.input.addEventListener('keyup', this.handleKeyup.bind(this));
+ this.input.value = initial;
+ }
+
+ initUi(doc) {
+ this.title = doc.createElement('p');
+ this.title.className = 'vimvixen-footerline-title';
+
+ let containerInner = doc.createElement('div');
+ containerInner.className = 'vimvixen-footerline-container-inner';
+
+ let containerOuter = doc.createElement('div');
+ containerOuter.className = 'vimvixen-footerline-container-outer';
+
+ this.input = doc.createElement('input');
+ this.input.className = 'vimvixen-footerline-input';
+
+ this.wrapper = doc.createElement('div');
+ this.wrapper.className = 'vimvixen-footerline';
+
+ containerOuter.append(containerInner);
+ containerInner.append(this.input);
+ this.wrapper.append(this.title);
+ this.wrapper.append(containerOuter);
+ doc.body.append(this.wrapper)
+ }
+
+ focus() {
+ this.input.focus();
+ }
+
+ remove() {
+ this.wrapper.remove();
+ }
+
+ onPromptChange(callback) {
+ this.promptChangeCallback = callback;
+ }
+
+ onEntered(callback) {
+ this.enteredCallback = callback;
+ }
+
+ handleBlur() {
+ this.remove();
+ }
+
+ handleKeydown(e) {
+ this.prevValue = e.target.value;
+ switch(e.keyCode) {
+ case KeyboardEvent.DOM_VK_ESCAPE:
+ this.remove();
+ break;
+ case KeyboardEvent.DOM_VK_RETURN:
+ this.enteredCallback(e);
+ break;
+ }
+ }
+
+ handleKeyup(e) {
+ if (e.target.value === this.prevValue) {
+ return;
+ }
+ this.promptChangeCallback(e);
+ this.prevValue = e.target.value;
+ }
+}
diff --git a/src/content/index.js b/src/content/index.js
index 03efc5e..17ab308 100644
--- a/src/content/index.js
+++ b/src/content/index.js
@@ -1,12 +1,45 @@
import * as scrolls from './scrolls';
+import FooterLine from './footer-line';
import * as actions from '../shared/actions';
+var footer = null;
+
+const createFooterLine = (initial = '') => {
+ footer = new FooterLine(document, initial);
+ footer.onPromptChange((e) => {
+ let request = {
+ type: 'event.cmd.suggest',
+ text: e.target.value
+ };
+ browser.runtime.sendMessage(request);
+ });
+ footer.onEntered((e) => {
+ let request = {
+ type: 'event.cmd.enter',
+ text: e.target.value
+ };
+ browser.runtime.sendMessage(request);
+ });
+ footer.focus();
+}
+
const invokeEvent = (action) => {
if (typeof action === 'undefined' || action === null) {
return;
}
switch (action[0]) {
+ case actions.CMD_OPEN:
+ createFooterLine();
+ break;
+ case actions.CMD_TABS_OPEN:
+ if (action[1] || false) {
+ // alter url
+ createFooterLine('open ' + window.location.href);
+ } else {
+ createFooterLine('open ');
+ }
+ break;
case actions.SCROLL_UP:
scrolls.scrollUp(window, action[1] || 1);
break;
diff --git a/src/shared/actions.js b/src/shared/actions.js
index 3e3cbd0..be25d72 100644
--- a/src/shared/actions.js
+++ b/src/shared/actions.js
@@ -1,3 +1,7 @@
+export const CMD_OPEN = 'cmd.open';
+export const CMD_TABS_OPEN = 'cmd.tabs.open';
+export const TABS_CLOSE = 'tabs.close';
+export const TABS_REOPEN = 'tabs.reopen';
export const TABS_PREV = 'tabs.prev';
export const TABS_NEXT = 'tabs.next';
export const SCROLL_UP = 'scroll.up';
@@ -6,11 +10,15 @@ export const SCROLL_TOP = 'scroll.top';
export const SCROLL_BOTTOM = 'scroll.bottom';
const BACKGROUND_ACTION_SET = new Set([
+ TABS_CLOSE,
+ TABS_REOPEN,
TABS_PREV,
TABS_NEXT
]);
const CONTENT_ACTION_SET = new Set([
+ CMD_OPEN,
+ CMD_TABS_OPEN,
SCROLL_UP,
SCROLL_DOWN,
SCROLL_TOP,