diff options
Diffstat (limited to 'src')
| -rw-r--r-- | src/background/commands.js | 2 | ||||
| -rw-r--r-- | src/background/index.js | 62 | ||||
| -rw-r--r-- | src/background/key-queue.js | 5 | ||||
| -rw-r--r-- | src/background/tabs.js | 22 | ||||
| -rw-r--r-- | src/content/footer-line.css | 46 | ||||
| -rw-r--r-- | src/content/footer-line.js | 78 | ||||
| -rw-r--r-- | src/content/index.js | 33 | ||||
| -rw-r--r-- | src/shared/actions.js | 8 | 
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, | 
