Merge pull request #544 from ueokande/refactor-background

Refactor background
jh-changes
Shin'ya Ueoka 6 years ago committed by GitHub
commit dfeb7e7549
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
  1. 1
      .eslintrc
  2. 11
      src/background/controllers/AddonEnabledController.js
  3. 22
      src/background/controllers/CommandController.js
  4. 15
      src/background/controllers/FindController.js
  5. 15
      src/background/controllers/LinkController.js
  6. 15
      src/background/controllers/MarkController.js
  7. 77
      src/background/controllers/OperationController.js
  8. 18
      src/background/controllers/SettingController.js
  9. 11
      src/background/controllers/VersionController.js
  10. 11
      src/background/controllers/addon-enabled.js
  11. 15
      src/background/controllers/find.js
  12. 15
      src/background/controllers/link.js
  13. 15
      src/background/controllers/mark.js
  14. 69
      src/background/controllers/operation.js
  15. 18
      src/background/controllers/setting.js
  16. 11
      src/background/controllers/version.js
  17. 0
      src/background/domains/CommandDocs.js
  18. 0
      src/background/domains/CompletionGroup.js
  19. 0
      src/background/domains/CompletionItem.js
  20. 0
      src/background/domains/Completions.js
  21. 0
      src/background/domains/GlobalMark.js
  22. 0
      src/background/domains/Setting.js
  23. 6
      src/background/index.js
  24. 3
      src/background/infrastructures/ConsoleClient.js
  25. 3
      src/background/infrastructures/ContentMessageClient.js
  26. 14
      src/background/infrastructures/ContentMessageListener.js
  27. 0
      src/background/infrastructures/MemoryStorage.js
  28. 0
      src/background/infrastructures/Notifier.js
  29. 0
      src/background/presenters/IndicatorPresenter.js
  30. 2
      src/background/presenters/TabPresenter.js
  31. 0
      src/background/presenters/WindowPresenter.js
  32. 0
      src/background/repositories/BookmarkRepository.js
  33. 8
      src/background/repositories/BrowserSettingRepository.js
  34. 0
      src/background/repositories/CompletionsRepository.js
  35. 2
      src/background/repositories/FindRepository.js
  36. 4
      src/background/repositories/MarkRepository.js
  37. 6
      src/background/repositories/PersistentSettingRepository.js
  38. 2
      src/background/repositories/SettingRepository.js
  39. 0
      src/background/repositories/VersionRepository.js
  40. 8
      src/background/usecases/AddonEnabledUseCase.js
  41. 16
      src/background/usecases/CommandUseCase.js
  42. 26
      src/background/usecases/CompletionsUseCase.js
  43. 61
      src/background/usecases/ConsoleUseCase.js
  44. 24
      src/background/usecases/FindUseCase.js
  45. 6
      src/background/usecases/LinkUseCase.js
  46. 16
      src/background/usecases/MarkUseCase.js
  47. 13
      src/background/usecases/SettingUseCase.js
  48. 51
      src/background/usecases/TabSelectUseCase.js
  49. 77
      src/background/usecases/TabUseCase.js
  50. 8
      src/background/usecases/VersionUseCase.js
  51. 35
      src/background/usecases/ZoomUseCase.js
  52. 15
      src/background/usecases/find.js
  53. 208
      src/background/usecases/operation.js
  54. 6
      src/shared/urls.js
  55. 2
      test/background/domains/GlobalMark.test.js
  56. 2
      test/background/infrastructures/MemoryStorage.test.js
  57. 4
      test/background/repositories/Mark.test.js
  58. 2
      test/background/repositories/Version.js
  59. 9
      test/shared/urls.test.js

@ -40,6 +40,7 @@
"no-alert": "off", "no-alert": "off",
"no-bitwise": "off", "no-bitwise": "off",
"no-console": ["error", { "allow": ["warn", "error"] }], "no-console": ["error", { "allow": ["warn", "error"] }],
"no-continue": "off",
"no-empty-function": "off", "no-empty-function": "off",
"no-magic-numbers": "off", "no-magic-numbers": "off",
"no-mixed-operators": "off", "no-mixed-operators": "off",

@ -0,0 +1,11 @@
import AddonEnabledUseCase from '../usecases/AddonEnabledUseCase';
export default class AddonEnabledController {
constructor() {
this.addonEnabledUseCase = new AddonEnabledUseCase();
}
indicate(enabled) {
return this.addonEnabledUseCase.indicate(enabled);
}
}

@ -1,6 +1,6 @@
import CompletionsInteractor from '../usecases/completions'; import CompletionsUseCase from '../usecases/CompletionsUseCase';
import CommandInteractor from '../usecases/command'; import CommandUseCase from '../usecases/CommandUseCase';
import Completions from '../domains/completions'; import Completions from '../domains/Completions';
const trimStart = (str) => { const trimStart = (str) => {
// NOTE String.trimStart is available on Firefox 61 // NOTE String.trimStart is available on Firefox 61
@ -9,8 +9,8 @@ const trimStart = (str) => {
export default class CommandController { export default class CommandController {
constructor() { constructor() {
this.completionsInteractor = new CompletionsInteractor(); this.completionsUseCase = new CompletionsUseCase();
this.commandIndicator = new CommandInteractor(); this.commandIndicator = new CommandUseCase();
} }
getCompletions(line) { getCompletions(line) {
@ -18,7 +18,7 @@ export default class CommandController {
let words = trimmed.split(/ +/); let words = trimmed.split(/ +/);
let name = words[0]; let name = words[0];
if (words.length === 1) { if (words.length === 1) {
return this.completionsInteractor.queryConsoleCommand(name); return this.completionsUseCase.queryConsoleCommand(name);
} }
let keywords = trimStart(trimmed.slice(name.length)); let keywords = trimStart(trimmed.slice(name.length));
switch (words[0]) { switch (words[0]) {
@ -28,22 +28,22 @@ export default class CommandController {
case 'tabopen': case 'tabopen':
case 'w': case 'w':
case 'winopen': case 'winopen':
return this.completionsInteractor.queryOpen(name, keywords); return this.completionsUseCase.queryOpen(name, keywords);
case 'b': case 'b':
case 'buffer': case 'buffer':
return this.completionsInteractor.queryBuffer(name, keywords); return this.completionsUseCase.queryBuffer(name, keywords);
case 'bd': case 'bd':
case 'bdel': case 'bdel':
case 'bdelete': case 'bdelete':
case 'bdeletes': case 'bdeletes':
return this.completionsInteractor.queryBdelete(name, keywords); return this.completionsUseCase.queryBdelete(name, keywords);
case 'bd!': case 'bd!':
case 'bdel!': case 'bdel!':
case 'bdelete!': case 'bdelete!':
case 'bdeletes!': case 'bdeletes!':
return this.completionsInteractor.queryBdeleteForce(name, keywords); return this.completionsUseCase.queryBdeleteForce(name, keywords);
case 'set': case 'set':
return this.completionsInteractor.querySet(name, keywords); return this.completionsUseCase.querySet(name, keywords);
} }
return Promise.resolve(Completions.empty()); return Promise.resolve(Completions.empty());
} }

@ -0,0 +1,15 @@
import FindUseCase from '../usecases/FindUseCase';
export default class FindController {
constructor() {
this.findUseCase = new FindUseCase();
}
getKeyword() {
return this.findUseCase.getKeyword();
}
setKeyword(keyword) {
return this.findUseCase.setKeyword(keyword);
}
}

@ -0,0 +1,15 @@
import LinkUseCase from '../usecases/LinkUseCase';
export default class LinkController {
constructor() {
this.linkUseCase = new LinkUseCase();
}
openToTab(url, tabId) {
this.linkUseCase.openToTab(url, tabId);
}
openNewTab(url, openerId, background) {
this.linkUseCase.openNewTab(url, openerId, background);
}
}

@ -0,0 +1,15 @@
import MarkUseCase from '../usecases/MarkUseCase';
export default class MarkController {
constructor() {
this.markUseCase = new MarkUseCase();
}
setGlobal(key, x, y) {
this.markUseCase.setGlobal(key, x, y);
}
jumpGlobal(key) {
this.markUseCase.jumpGlobal(key);
}
}

@ -0,0 +1,77 @@
import operations from '../../shared/operations';
import FindUseCase from '../usecases/FindUseCase';
import ConsoleUseCase from '../usecases/ConsoleUseCase';
import TabUseCase from '../usecases/TabUseCase';
import TabSelectUseCase from '../usecases/TabSelectUseCase';
import ZoomUseCase from '../usecases/ZoomUseCase';
export default class OperationController {
constructor() {
this.findUseCase = new FindUseCase();
this.consoleUseCase = new ConsoleUseCase();
this.tabUseCase = new TabUseCase();
this.tabSelectUseCase = new TabSelectUseCase();
this.zoomUseCase = new ZoomUseCase();
}
// eslint-disable-next-line complexity, max-lines-per-function
exec(operation) {
switch (operation.type) {
case operations.TAB_CLOSE:
return this.tabUseCase.close(false);
case operations.TAB_CLOSE_RIGHT:
return this.tabUseCase.closeRight();
case operations.TAB_CLOSE_FORCE:
return this.tabUseCase.close(true);
case operations.TAB_REOPEN:
return this.tabUseCase.reopen();
case operations.TAB_PREV:
return this.tabSelectUseCase.selectPrev(1);
case operations.TAB_NEXT:
return this.tabSelectUseCase.selectNext(1);
case operations.TAB_FIRST:
return this.tabSelectUseCase.selectFirst();
case operations.TAB_LAST:
return this.tabSelectUseCase.selectLast();
case operations.TAB_PREV_SEL:
return this.tabSelectUseCase.selectPrevSelected();
case operations.TAB_RELOAD:
return this.tabUseCase.reload(operation.cache);
case operations.TAB_PIN:
return this.tabUseCase.setPinned(true);
case operations.TAB_UNPIN:
return this.tabUseCase.setPinned(false);
case operations.TAB_TOGGLE_PINNED:
return this.tabUseCase.togglePinned();
case operations.TAB_DUPLICATE:
return this.tabUseCase.duplicate();
case operations.PAGE_SOURCE:
return this.tabUseCase.openPageSource();
case operations.PAGE_HOME:
return this.tabUseCase.openHome(operation.newTab);
case operations.ZOOM_IN:
return this.zoomUseCase.zoomIn();
case operations.ZOOM_OUT:
return this.zoomUseCase.zoomOut();
case operations.ZOOM_NEUTRAL:
return this.zoomUseCase.zoomNutoral();
case operations.COMMAND_SHOW:
return this.consoleUseCase.showCommand();
case operations.COMMAND_SHOW_OPEN:
return this.consoleUseCase.showOpenCommand(operation.alter);
case operations.COMMAND_SHOW_TABOPEN:
return this.consoleUseCase.showTabopenCommand(operation.alter);
case operations.COMMAND_SHOW_WINOPEN:
return this.consoleUseCase.showWinopenCommand(operation.alter);
case operations.COMMAND_SHOW_BUFFER:
return this.consoleUseCase.showBufferCommand();
case operations.COMMAND_SHOW_ADDBOOKMARK:
return this.consoleUseCase.showAddbookmarkCommand(operation.alter);
case operations.FIND_START:
return this.findUseCase.findStart();
case operations.CANCEL:
return this.consoleUseCase.hideConsole();
}
}
}

@ -0,0 +1,18 @@
import SettingUseCase from '../usecases/SettingUseCase';
import ContentMessageClient from '../infrastructures/ContentMessageClient';
export default class SettingController {
constructor() {
this.settingUseCase = new SettingUseCase();
this.contentMessageClient = new ContentMessageClient();
}
getSetting() {
return this.settingUseCase.get();
}
async reload() {
await this.settingUseCase.reload();
this.contentMessageClient.broadcastSettingsChanged();
}
}

@ -0,0 +1,11 @@
import VersionUseCase from '../usecases/VersionUseCase';
export default class VersionController {
constructor() {
this.versionUseCase = new VersionUseCase();
}
notifyIfUpdated() {
this.versionUseCase.notifyIfUpdated();
}
}

@ -1,11 +0,0 @@
import AddonEnabledInteractor from '../usecases/addon-enabled';
export default class AddonEnabledController {
constructor() {
this.addonEnabledInteractor = new AddonEnabledInteractor();
}
indicate(enabled) {
return this.addonEnabledInteractor.indicate(enabled);
}
}

@ -1,15 +0,0 @@
import FindInteractor from '../usecases/find';
export default class FindController {
constructor() {
this.findInteractor = new FindInteractor();
}
getKeyword() {
return this.findInteractor.getKeyword();
}
setKeyword(keyword) {
return this.findInteractor.setKeyword(keyword);
}
}

@ -1,15 +0,0 @@
import LinkInteractor from '../usecases/link';
export default class LinkController {
constructor() {
this.linkInteractor = new LinkInteractor();
}
openToTab(url, tabId) {
this.linkInteractor.openToTab(url, tabId);
}
openNewTab(url, openerId, background) {
this.linkInteractor.openNewTab(url, openerId, background);
}
}

@ -1,15 +0,0 @@
import MarkInteractor from '../usecases/mark';
export default class MarkController {
constructor() {
this.markInteractor = new MarkInteractor();
}
setGlobal(key, x, y) {
this.markInteractor.setGlobal(key, x, y);
}
jumpGlobal(key) {
this.markInteractor.jumpGlobal(key);
}
}

@ -1,69 +0,0 @@
import operations from '../../shared/operations';
import OperationInteractor from '../usecases/operation';
export default class OperationController {
constructor() {
this.operationInteractor = new OperationInteractor();
}
// eslint-disable-next-line complexity, max-lines-per-function
exec(operation) {
switch (operation.type) {
case operations.TAB_CLOSE:
return this.operationInteractor.close(false);
case operations.TAB_CLOSE_RIGHT:
return this.operationInteractor.closeRight();
case operations.TAB_CLOSE_FORCE:
return this.operationInteractor.close(true);
case operations.TAB_REOPEN:
return this.operationInteractor.reopen();
case operations.TAB_PREV:
return this.operationInteractor.selectPrev(1);
case operations.TAB_NEXT:
return this.operationInteractor.selectNext(1);
case operations.TAB_FIRST:
return this.operationInteractor.selectFirst();
case operations.TAB_LAST:
return this.operationInteractor.selectLast();
case operations.TAB_PREV_SEL:
return this.operationInteractor.selectPrevSelected();
case operations.TAB_RELOAD:
return this.operationInteractor.reload(operation.cache);
case operations.TAB_PIN:
return this.operationInteractor.setPinned(true);
case operations.TAB_UNPIN:
return this.operationInteractor.setPinned(false);
case operations.TAB_TOGGLE_PINNED:
return this.operationInteractor.togglePinned();
case operations.TAB_DUPLICATE:
return this.operationInteractor.duplicate();
case operations.PAGE_SOURCE:
return this.operationInteractor.openPageSource();
case operations.PAGE_HOME:
return this.operationInteractor.openHome(operation.newTab);
case operations.ZOOM_IN:
return this.operationInteractor.zoomIn();
case operations.ZOOM_OUT:
return this.operationInteractor.zoomOut();
case operations.ZOOM_NEUTRAL:
return this.operationInteractor.zoomNutoral();
case operations.COMMAND_SHOW:
return this.operationInteractor.showCommand();
case operations.COMMAND_SHOW_OPEN:
return this.operationInteractor.showOpenCommand(operation.alter);
case operations.COMMAND_SHOW_TABOPEN:
return this.operationInteractor.showTabopenCommand(operation.alter);
case operations.COMMAND_SHOW_WINOPEN:
return this.operationInteractor.showWinopenCommand(operation.alter);
case operations.COMMAND_SHOW_BUFFER:
return this.operationInteractor.showBufferCommand();
case operations.COMMAND_SHOW_ADDBOOKMARK:
return this.operationInteractor.showAddbookmarkCommand(operation.alter);
case operations.FIND_START:
return this.operationInteractor.findStart();
case operations.CANCEL:
return this.operationInteractor.hideConsole();
}
}
}

@ -1,18 +0,0 @@
import SettingInteractor from '../usecases/setting';
import ContentMessageClient from '../infrastructures/content-message-client';
export default class SettingController {
constructor() {
this.settingInteractor = new SettingInteractor();
this.contentMessageClient = new ContentMessageClient();
}
getSetting() {
return this.settingInteractor.get();
}
async reload() {
await this.settingInteractor.reload();
this.contentMessageClient.broadcastSettingsChanged();
}
}

@ -1,11 +0,0 @@
import VersionInteractor from '../usecases/version';
export default class VersionController {
constructor() {
this.versionInteractor = new VersionInteractor();
}
notifyIfUpdated() {
this.versionInteractor.notifyIfUpdated();
}
}

@ -1,6 +1,6 @@
import ContentMessageListener from './infrastructures/content-message-listener'; import ContentMessageListener from './infrastructures/ContentMessageListener';
import SettingController from './controllers/setting'; import SettingController from './controllers/SettingController';
import VersionController from './controllers/version'; import VersionController from './controllers/VersionController';
new SettingController().reload(); new SettingController().reload();
new VersionController().notifyIfUpdated(); new VersionController().notifyIfUpdated();

@ -1,6 +1,6 @@
import messages from '../../shared/messages'; import messages from '../../shared/messages';
export default class ConsolePresenter { export default class ConsoleClient {
showCommand(tabId, command) { showCommand(tabId, command) {
return browser.tabs.sendMessage(tabId, { return browser.tabs.sendMessage(tabId, {
type: messages.CONSOLE_SHOW_COMMAND, type: messages.CONSOLE_SHOW_COMMAND,
@ -34,3 +34,4 @@ export default class ConsolePresenter {
}); });
} }
} }

@ -4,6 +4,9 @@ export default class ContentMessageClient {
async broadcastSettingsChanged() { async broadcastSettingsChanged() {
let tabs = await browser.tabs.query({}); let tabs = await browser.tabs.query({});
for (let tab of tabs) { for (let tab of tabs) {
if (tab.url.startsWith('about:')) {
continue;
}
browser.tabs.sendMessage(tab.id, { browser.tabs.sendMessage(tab.id, {
type: messages.SETTINGS_CHANGED, type: messages.SETTINGS_CHANGED,
}); });

@ -1,11 +1,11 @@
import messages from '../../shared/messages'; import messages from '../../shared/messages';
import CommandController from '../controllers/command'; import CommandController from '../controllers/CommandController';
import SettingController from '../controllers/setting'; import SettingController from '../controllers/SettingController';
import FindController from '../controllers/find'; import FindController from '../controllers/FindController';
import AddonEnabledController from '../controllers/addon-enabled'; import AddonEnabledController from '../controllers/AddonEnabledController';
import LinkController from '../controllers/link'; import LinkController from '../controllers/LinkController';
import OperationController from '../controllers/operation'; import OperationController from '../controllers/OperationController';
import MarkController from '../controllers/mark'; import MarkController from '../controllers/MarkController';
export default class ContentMessageListener { export default class ContentMessageListener {
constructor() { constructor() {

@ -1,4 +1,4 @@
import MemoryStorage from '../infrastructures/memory-storage'; import MemoryStorage from '../infrastructures/MemoryStorage';
const CURRENT_SELECTED_KEY = 'tabs.current.selected'; const CURRENT_SELECTED_KEY = 'tabs.current.selected';
const LAST_SELECTED_KEY = 'tabs.last.selected'; const LAST_SELECTED_KEY = 'tabs.last.selected';

@ -0,0 +1,8 @@
import * as urls from '../../shared/urls';
export default class BrowserSettingRepository {
async getHomepageUrls() {
let { value } = await browser.browserSettings.homepageOverride.get({});
return value.split('|').map(urls.normalizeUrl);
}
}

@ -1,4 +1,4 @@
import MemoryStorage from '../infrastructures/memory-storage'; import MemoryStorage from '../infrastructures/MemoryStorage';
const FIND_KEYWORD_KEY = 'find-keyword'; const FIND_KEYWORD_KEY = 'find-keyword';

@ -1,5 +1,5 @@
import MemoryStorage from '../infrastructures/memory-storage'; import MemoryStorage from '../infrastructures/MemoryStorage';
import GlobalMark from 'background/domains/global-mark'; import GlobalMark from '../domains/GlobalMark';
const MARK_KEY = 'mark'; const MARK_KEY = 'mark';

@ -1,10 +1,6 @@
import Setting from '../domains/setting'; import Setting from '../domains/Setting';
export default class SettingRepository { export default class SettingRepository {
save(settings) {
return browser.storage.local.set({ settings: settings.serialize() });
}
async load() { async load() {
let { settings } = await browser.storage.local.get('settings'); let { settings } = await browser.storage.local.get('settings');
if (!settings) { if (!settings) {

@ -1,4 +1,4 @@
import MemoryStorage from '../infrastructures/memory-storage'; import MemoryStorage from '../infrastructures/MemoryStorage';
const CACHED_SETTING_KEY = 'setting'; const CACHED_SETTING_KEY = 'setting';

@ -1,8 +1,8 @@
import IndicatorPresenter from '../presenters/indicator'; import IndicatorPresenter from '../presenters/IndicatorPresenter';
import TabPresenter from '../presenters/tab'; import TabPresenter from '../presenters/TabPresenter';
import ContentMessageClient from '../infrastructures/content-message-client'; import ContentMessageClient from '../infrastructures/ContentMessageClient';
export default class AddonEnabledInteractor { export default class AddonEnabledUseCase {
constructor() { constructor() {
this.indicatorPresentor = new IndicatorPresenter(); this.indicatorPresentor = new IndicatorPresenter();

@ -1,11 +1,11 @@
import * as parsers from './parsers'; import * as parsers from './parsers';
import * as urls from '../../shared/urls'; import * as urls from '../../shared/urls';
import TabPresenter from '../presenters/tab'; import TabPresenter from '../presenters/TabPresenter';
import WindowPresenter from '../presenters/window'; import WindowPresenter from '../presenters/WindowPresenter';
import SettingRepository from '../repositories/setting'; import SettingRepository from '../repositories/SettingRepository';
import BookmarkRepository from '../repositories/bookmark'; import BookmarkRepository from '../repositories/BookmarkRepository';
import ConsolePresenter from '../presenters/console'; import ConsoleClient from '../infrastructures/ConsoleClient';
import ContentMessageClient from '../infrastructures/content-message-client'; import ContentMessageClient from '../infrastructures/ContentMessageClient';
import * as properties from 'shared/settings/properties'; import * as properties from 'shared/settings/properties';
export default class CommandIndicator { export default class CommandIndicator {
@ -14,7 +14,7 @@ export default class CommandIndicator {
this.windowPresenter = new WindowPresenter(); this.windowPresenter = new WindowPresenter();
this.settingRepository = new SettingRepository(); this.settingRepository = new SettingRepository();
this.bookmarkRepository = new BookmarkRepository(); this.bookmarkRepository = new BookmarkRepository();
this.consolePresenter = new ConsolePresenter(); this.consoleClient = new ConsoleClient();
this.contentMessageClient = new ContentMessageClient(); this.contentMessageClient = new ContentMessageClient();
} }
@ -105,7 +105,7 @@ export default class CommandIndicator {
let tab = await this.tabPresenter.getCurrent(); let tab = await this.tabPresenter.getCurrent();
let item = await this.bookmarkRepository.create(title, tab.url); let item = await this.bookmarkRepository.create(title, tab.url);
let message = 'Saved current page: ' + item.url; let message = 'Saved current page: ' + item.url;
return this.consolePresenter.showInfo(tab.id, message); return this.consoleClient.showInfo(tab.id, message);
} }
async set(keywords) { async set(keywords) {

@ -1,19 +1,19 @@
import CompletionItem from '../domains/completion-item'; import CompletionItem from '../domains/CompletionItem';
import CompletionGroup from '../domains/completion-group'; import CompletionGroup from '../domains/CompletionGroup';
import Completions from '../domains/completions'; import Completions from '../domains/Completions';
import CommandDocs from '../domains/command-docs'; import CommandDocs from '../domains/CommandDocs';
import CompletionRepository from '../repositories/completions'; import CompletionsRepository from '../repositories/CompletionsRepository';
import * as filters from './filters'; import * as filters from './filters';
import SettingRepository from '../repositories/setting'; import SettingRepository from '../repositories/SettingRepository';
import TabPresenter from '../presenters/tab'; import TabPresenter from '../presenters/TabPresenter';
import * as properties from '../../shared/settings/properties'; import * as properties from '../../shared/settings/properties';
const COMPLETION_ITEM_LIMIT = 10; const COMPLETION_ITEM_LIMIT = 10;
export default class CompletionsInteractor { export default class CompletionsUseCase {
constructor() { constructor() {
this.tabPresenter = new TabPresenter(); this.tabPresenter = new TabPresenter();
this.completionRepository = new CompletionRepository(); this.completionsRepository = new CompletionsRepository();
this.settingRepository = new SettingRepository(); this.settingRepository = new SettingRepository();
} }
@ -86,7 +86,7 @@ export default class CompletionsInteractor {
tabs = [tab]; tabs = [tab];
} }
} else { } else {
tabs = await this.completionRepository.queryTabs(keywords, false); tabs = await this.completionsRepository.queryTabs(keywords, false);
} }
const flag = (tab) => { const flag = (tab) => {
if (tab.active) { if (tab.active) {
@ -153,7 +153,7 @@ export default class CompletionsInteractor {
} }
async queryTabs(name, excludePinned, args) { async queryTabs(name, excludePinned, args) {
let tabs = await this.completionRepository.queryTabs(args, excludePinned); let tabs = await this.completionsRepository.queryTabs(args, excludePinned);
let items = tabs.map(tab => new CompletionItem({ let items = tabs.map(tab => new CompletionItem({
caption: tab.title, caption: tab.title,
content: name + ' ' + tab.title, content: name + ' ' + tab.title,
@ -177,7 +177,7 @@ export default class CompletionsInteractor {
} }
async queryHistoryItems(name, keywords) { async queryHistoryItems(name, keywords) {
let histories = await this.completionRepository.queryHistories(keywords); let histories = await this.completionsRepository.queryHistories(keywords);
histories = [histories] histories = [histories]
.map(filters.filterBlankTitle) .map(filters.filterBlankTitle)
.map(filters.filterHttp) .map(filters.filterHttp)
@ -194,7 +194,7 @@ export default class CompletionsInteractor {
} }
async queryBookmarkItems(name, keywords) { async queryBookmarkItems(name, keywords) {
let bookmarks = await this.completionRepository.queryBookmarks(keywords); let bookmarks = await this.completionsRepository.queryBookmarks(keywords);
return bookmarks.slice(0, COMPLETION_ITEM_LIMIT) return bookmarks.slice(0, COMPLETION_ITEM_LIMIT)
.map(page => new CompletionItem({ .map(page => new CompletionItem({
caption: page.title, caption: page.title,

@ -0,0 +1,61 @@
import TabPresenter from '../presenters/TabPresenter';
import ConsoleClient from '../infrastructures/ConsoleClient';
export default class ConsoleUseCase {
constructor() {
this.tabPresenter = new TabPresenter();
this.consoleClient = new ConsoleClient();
}
async showCommand() {
let tab = await this.tabPresenter.getCurrent();
return this.consoleClient.showCommand(tab.id, '');
}
async showOpenCommand(alter) {
let tab = await this.tabPresenter.getCurrent();
let command = 'open ';
if (alter) {
command += tab.url;
}
return this.consoleClient.showCommand(tab.id, command);
}
async showTabopenCommand(alter) {
let tab = await this.tabPresenter.getCurrent();
let command = 'tabopen ';
if (alter) {
command += tab.url;
}
return this.consoleClient.showCommand(tab.id, command);
}
async showWinopenCommand(alter) {
let tab = await this.tabPresenter.getCurrent();
let command = 'winopen ';
if (alter) {
command += tab.url;
}
return this.consoleClient.showCommand(tab.id, command);
}
async showBufferCommand() {
let tab = await this.tabPresenter.getCurrent();
let command = 'buffer ';
return this.consoleClient.showCommand(tab.id, command);
}
async showAddbookmarkCommand(alter) {
let tab = await this.tabPresenter.getCurrent();
let command = 'addbookmark ';
if (alter) {
command += tab.title;
}
return this.consoleClient.showCommand(tab.id, command);
}
async hideConsole() {
let tab = await this.tabPresenter.getCurrent();
return this.consoleClient.hide(tab.id);
}
}

@ -0,0 +1,24 @@
import FindRepository from '../repositories/FindRepository';
import TabPresenter from '../presenters/TabPresenter';
import ConsoleClient from '../infrastructures/ConsoleClient';
export default class FindUseCase {
constructor() {
this.tabPresenter = new TabPresenter();
this.findRepository = new FindRepository();
this.consoleClient = new ConsoleClient();
}
getKeyword() {
return this.findRepository.getKeyword();
}
setKeyword(keyword) {
return this.findRepository.setKeyword(keyword);
}
async findStart() {
let tab = await this.tabPresenter.getCurrent();
return this.consoleClient.showFind(tab.id);
}
}

@ -1,7 +1,7 @@
import SettingRepository from '../repositories/setting'; import SettingRepository from '../repositories/SettingRepository';
import TabPresenter from '../presenters/tab'; import TabPresenter from '../presenters/TabPresenter';
export default class LinkInteractor { export default class LinkUseCase {
constructor() { constructor() {
this.settingRepository = new SettingRepository(); this.settingRepository = new SettingRepository();
this.tabPresenter = new TabPresenter(); this.tabPresenter = new TabPresenter();

@ -1,14 +1,14 @@
import GlobalMark from '../domains/global-mark'; import GlobalMark from '../domains/GlobalMark';
import TabPresenter from '../presenters/tab'; import TabPresenter from '../presenters/TabPresenter';
import MarkRepository from '../repositories/mark'; import MarkRepository from '../repositories/MarkRepository';
import ConsolePresenter from '../presenters/console'; import ConsoleClient from '../infrastructures/ConsoleClient';
import ContentMessageClient from '../infrastructures/content-message-client'; import ContentMessageClient from '../infrastructures/ContentMessageClient';
export default class MarkInteractor { export default class MarkUseCase {
constructor() { constructor() {
this.tabPresenter = new TabPresenter(); this.tabPresenter = new TabPresenter();
this.markRepository = new MarkRepository(); this.markRepository = new MarkRepository();
this.consolePresenter = new ConsolePresenter(); this.consoleClient = new ConsoleClient();
this.contentMessageClient = new ContentMessageClient(); this.contentMessageClient = new ContentMessageClient();
} }
@ -23,7 +23,7 @@ export default class MarkInteractor {
let mark = await this.markRepository.getMark(key); let mark = await this.markRepository.getMark(key);
if (!mark) { if (!mark) {
return this.consolePresenter.showError(current.id, 'Mark is not set'); return this.consoleClient.showError(current.id, 'Mark is not set');
} }
return this.contentMessageClient.scrollTo( return this.contentMessageClient.scrollTo(

@ -1,17 +1,14 @@
import Setting from '../domains/setting'; import Setting from '../domains/Setting';
import PersistentSettingRepository from '../repositories/persistent-setting'; // eslint-disable-next-line max-len
import SettingRepository from '../repositories/setting'; import PersistentSettingRepository from '../repositories/PersistentSettingRepository';
import SettingRepository from '../repositories/SettingRepository';
export default class SettingInteractor { export default class SettingUseCase {
constructor() { constructor() {
this.persistentSettingRepository = new PersistentSettingRepository(); this.persistentSettingRepository = new PersistentSettingRepository();
this.settingRepository = new SettingRepository(); this.settingRepository = new SettingRepository();
} }
save(settings) {
this.persistentSettingRepository.save(settings);
}
get() { get() {
return this.settingRepository.get(); return this.settingRepository.get();
} }

@ -0,0 +1,51 @@
import TabPresenter from '../presenters/TabPresenter';
export default class TabSelectUseCase {
constructor() {
this.tabPresenter = new TabPresenter();
}
async selectPrev(count) {
let tabs = await this.tabPresenter.getAll();
if (tabs.length < 2) {
return;
}
let tab = tabs.find(t => t.active);
if (!tab) {
return;
}
let select = (tab.index - count + tabs.length) % tabs.length;
return this.tabPresenter.select(tabs[select].id);
}
async selectNext(count) {
let tabs = await this.tabPresenter.getAll();
if (tabs.length < 2) {
return;
}
let tab = tabs.find(t => t.active);
if (!tab) {
return;
}
let select = (tab.index + count) % tabs.length;
return this.tabPresenter.select(tabs[select].id);
}
async selectFirst() {
let tabs = await this.tabPresenter.getAll();
return this.tabPresenter.select(tabs[0].id);
}
async selectLast() {
let tabs = await this.tabPresenter.getAll();
return this.tabPresenter.select(tabs[tabs.length - 1].id);
}
async selectPrevSelected() {
let tabId = await this.tabPresenter.getLastSelectedId();
if (tabId === null || typeof tabId === 'undefined') {
return;
}
this.tabPresenter.select(tabId);
}
}

@ -0,0 +1,77 @@
import TabPresenter from '../presenters/TabPresenter';
import BrowserSettingRepository from '../repositories/BrowserSettingRepository';
export default class TabUseCase {
constructor() {
this.tabPresenter = new TabPresenter();
this.browserSettingRepository = new BrowserSettingRepository();
}
async close(force) {
let tab = await this.tabPresenter.getCurrent();
if (!force && tab.pinned) {
return;
}
return this.tabPresenter.remove([tab.id]);
}
async closeRight() {
let tabs = await this.tabPresenter.getAll();
tabs.sort((t1, t2) => t1.index - t2.index);
let index = tabs.findIndex(t => t.active);
if (index < 0) {
return;
}
for (let i = index + 1; i < tabs.length; ++i) {
let tab = tabs[i];
if (!tab.pinned) {
this.tabPresenter.remove(tab.id);
}
}
}
reopen() {
return this.tabPresenter.reopen();
}
async reload(cache) {
let tab = await this.tabPresenter.getCurrent();
return this.tabPresenter.reload(tab.id, cache);
}
async setPinned(pinned) {
let tab = await this.tabPresenter.getCurrent();
return this.tabPresenter.setPinned(tab.id, pinned);
}
async togglePinned() {
let tab = await this.tabPresenter.getCurrent();
return this.tabPresenter.setPinned(tab.id, !tab.pinned);
}
async duplicate() {
let tab = await this.tabPresenter.getCurrent();
return this.tabPresenter.duplicate(tab.id);
}
async openPageSource() {
let tab = await this.tabPresenter.getCurrent();
let url = 'view-source:' + tab.url;
return this.tabPresenter.create(url);
}
async openHome(newTab) {
let tab = await this.tabPresenter.getCurrent();
let urls = await this.browserSettingRepository.getHomepageUrls();
if (urls.length === 1 && urls[0] === 'about:home') {
// eslint-disable-next-line max-len
throw new Error('Cannot open Firefox Home (about:home) by WebExtensions, set your custom URLs');
}
if (urls.length === 1 && !newTab) {
return this.tabPresenter.open(urls[0], tab.id);
}
for (let url of urls) {
this.tabPresenter.create(url);
}
}
}

@ -1,9 +1,9 @@
import manifest from '../../../manifest.json'; import manifest from '../../../manifest.json';
import VersionRepository from '../repositories/version'; import VersionRepository from '../repositories/VersionRepository';
import TabPresenter from '../presenters/tab'; import TabPresenter from '../presenters/TabPresenter';
import Notifier from '../infrastructures/notifier'; import Notifier from '../infrastructures/Notifier';
export default class VersionInteractor { export default class VersionUseCase {
constructor() { constructor() {
this.versionRepository = new VersionRepository(); this.versionRepository = new VersionRepository();
this.tabPresenter = new TabPresenter(); this.tabPresenter = new TabPresenter();

@ -0,0 +1,35 @@
import TabPresenter from '../presenters/TabPresenter';
const ZOOM_SETTINGS = [
0.33, 0.50, 0.66, 0.75, 0.80, 0.90, 1.00,
1.10, 1.25, 1.50, 1.75, 2.00, 2.50, 3.00
];
export default class ZoomUseCase {
constructor() {
this.tabPresenter = new TabPresenter();
}
async zoomIn(tabId) {
let tab = await this.tabPresenter.getCurrent();
let current = await this.tabPresenter.getZoom(tab.id);
let factor = ZOOM_SETTINGS.find(f => f > current);
if (factor) {
return this.tabPresenter.setZoom(tabId, factor);
}
}
async zoomOut(tabId) {
let tab = await this.tabPresenter.getCurrent();
let current = await this.tabPresenter.getZoom(tab.id);
let factor = [].concat(ZOOM_SETTINGS).reverse().find(f => f < current);
if (factor) {
return this.tabPresenter.setZoom(tabId, factor);
}
}
zoomNutoral(tabId) {
return this.tabPresenter.setZoom(tabId, 1);
}
}

@ -1,15 +0,0 @@
import FindRepository from '../repositories/find';
export default class FindInteractor {
constructor() {
this.findRepository = new FindRepository();
}
getKeyword() {
return this.findRepository.getKeyword();
}
setKeyword(keyword) {
return this.findRepository.setKeyword(keyword);
}
}

@ -1,208 +0,0 @@
import TabPresenter from '../presenters/tab';
import ConsolePresenter from '../presenters/console';
import * as urls from '../../shared/urls';
const ZOOM_SETTINGS = [
0.33, 0.50, 0.66, 0.75, 0.80, 0.90, 1.00,
1.10, 1.25, 1.50, 1.75, 2.00, 2.50, 3.00
];
export default class OperationInteractor {
constructor() {
this.tabPresenter = new TabPresenter();
this.consolePresenter = new ConsolePresenter();
}
async close(force) {
let tab = await this.tabPresenter.getCurrent();
if (!force && tab.pinned) {
return;
}
return this.tabPresenter.remove([tab.id]);
}
async closeRight() {
let tabs = await this.tabPresenter.getAll();
tabs.sort((t1, t2) => t1.index - t2.index);
let index = tabs.findIndex(t => t.active);
if (index < 0) {
return;
}
for (let i = index + 1; i < tabs.length; ++i) {
let tab = tabs[i];
if (!tab.pinned) {
this.tabPresenter.remove(tab.id);
}
}
}
reopen() {
return this.tabPresenter.reopen();
}
async selectPrev(count) {
let tabs = await this.tabPresenter.getAll();
if (tabs.length < 2) {
return;
}
let tab = tabs.find(t => t.active);
if (!tab) {
return;
}
let select = (tab.index - count + tabs.length) % tabs.length;
return this.tabPresenter.select(tabs[select].id);
}
async selectNext(count) {
let tabs = await this.tabPresenter.getAll();
if (tabs.length < 2) {
return;
}
let tab = tabs.find(t => t.active);
if (!tab) {
return;
}
let select = (tab.index + count) % tabs.length;
return this.tabPresenter.select(tabs[select].id);
}
async selectFirst() {
let tabs = await this.tabPresenter.getAll();
return this.tabPresenter.select(tabs[0].id);
}
async selectLast() {
let tabs = await this.tabPresenter.getAll();
return this.tabPresenter.select(tabs[tabs.length - 1].id);
}
async selectPrevSelected() {
let tabId = await this.tabPresenter.getLastSelectedId();
if (tabId === null || typeof tabId === 'undefined') {
return;
}
this.tabPresenter.select(tabId);
}
async reload(cache) {
let tab = await this.tabPresenter.getCurrent();
return this.tabPresenter.reload(tab.id, cache);
}
async setPinned(pinned) {
let tab = await this.tabPresenter.getCurrent();
return this.tabPresenter.setPinned(tab.id, pinned);
}
async togglePinned() {
let tab = await this.tabPresenter.getCurrent();
return this.tabPresenter.setPinned(tab.id, !tab.pinned);
}
async duplicate() {
let tab = await this.tabPresenter.getCurrent();
return this.tabPresenter.duplicate(tab.id);
}
async openPageSource() {
let tab = await this.tabPresenter.getCurrent();
let url = 'view-source:' + tab.url;
return this.tabPresenter.create(url);
}
async zoomIn(tabId) {
let tab = await this.tabPresenter.getCurrent();
let current = await this.tabPresenter.getZoom(tab.id);
let factor = ZOOM_SETTINGS.find(f => f > current);
if (factor) {
return this.tabPresenter.setZoom(tabId, factor);
}
}
async zoomOut(tabId) {
let tab = await this.tabPresenter.getCurrent();
let current = await this.tabPresenter.getZoom(tab.id);
let factor = [].concat(ZOOM_SETTINGS).reverse().find(f => f < current);
if (factor) {
return this.tabPresenter.setZoom(tabId, factor);
}
}
zoomNutoral(tabId) {
return this.tabPresenter.setZoom(tabId, 1);
}
async showCommand() {
let tab = await this.tabPresenter.getCurrent();
return this.consolePresenter.showCommand(tab.id, '');
}
async showOpenCommand(alter) {
let tab = await this.tabPresenter.getCurrent();
let command = 'open ';
if (alter) {
command += tab.url;
}
return this.consolePresenter.showCommand(tab.id, command);
}
async showTabopenCommand(alter) {
let tab = await this.tabPresenter.getCurrent();
let command = 'tabopen ';
if (alter) {
command += tab.url;
}
return this.consolePresenter.showCommand(tab.id, command);
}
async showWinopenCommand(alter) {
let tab = await this.tabPresenter.getCurrent();
let command = 'winopen ';
if (alter) {
command += tab.url;
}
return this.consolePresenter.showCommand(tab.id, command);
}
async showBufferCommand() {
let tab = await this.tabPresenter.getCurrent();
let command = 'buffer ';
return this.consolePresenter.showCommand(tab.id, command);
}
async showAddbookmarkCommand(alter) {
let tab = await this.tabPresenter.getCurrent();
let command = 'addbookmark ';
if (alter) {
command += tab.title;
}
return this.consolePresenter.showCommand(tab.id, command);
}
async findStart() {
let tab = await this.tabPresenter.getCurrent();
return this.consolePresenter.showFind(tab.id);
}
async hideConsole() {
let tab = await this.tabPresenter.getCurrent();
return this.consolePresenter.hide(tab.id);
}
async openHome(newTab) {
let tab = await this.tabPresenter.getCurrent();
let result = await browser.browserSettings.homepageOverride.get({});
let us = urls.homepageUrls(result.value);
if (us.length === 1 && us[0] === 'about:home') {
// eslint-disable-next-line max-len
throw new Error('Cannot open Firefox Home (about:home) by WebExtensions, set your custom URLs');
}
if (us.length === 1 && !newTab) {
return this.tabPresenter.open(us[0], tab.id);
}
for (let u of us) {
this.tabPresenter.create(u);
}
}
}

@ -40,8 +40,4 @@ const normalizeUrl = (url) => {
return 'http://' + url; return 'http://' + url;
}; };
const homepageUrls = (value) => { export { searchUrl, normalizeUrl };
return value.split('|').map(normalizeUrl);
};
export { searchUrl, normalizeUrl, homepageUrls };

@ -1,4 +1,4 @@
import GlobalMark from 'background/domains/global-mark'; import GlobalMark from 'background/domains/GlobalMark';
describe('background/domains/global-mark', () => { describe('background/domains/global-mark', () => {
describe('constructor and getter', () => { describe('constructor and getter', () => {

@ -1,4 +1,4 @@
import MemoryStorage from 'background/infrastructures/memory-storage'; import MemoryStorage from 'background/infrastructures/MemoryStorage';
describe("background/infrastructures/memory-storage", () => { describe("background/infrastructures/memory-storage", () => {
it('stores values', () => { it('stores values', () => {

@ -1,5 +1,5 @@
import MarkRepository from 'background/repositories/mark'; import MarkRepository from 'background/repositories/MarkRepository';
import GlobalMark from 'background/domains/global-mark'; import GlobalMark from 'background/domains/GlobalMark';
describe('background/repositories/mark', () => { describe('background/repositories/mark', () => {
let repository; let repository;

@ -1,4 +1,4 @@
import VersionRepository from 'background/repositories/version'; import VersionRepository from 'background/repositories/Version';
describe("background/repositories/version", () => { describe("background/repositories/version", () => {
let versionRepository; let versionRepository;

@ -44,14 +44,5 @@ describe("shared/commands/parsers", () => {
.to.equal('http://google.com'); .to.equal('http://google.com');
}); });
}); });
describe('#homepageUrls', () => {
it('split urls', () => {
expect(parsers.homepageUrls('https://google.com/'))
.to.deep.equal(['https://google.com/']);
expect(parsers.homepageUrls('yahoo.com|https://i-beam.org/'))
.to.deep.equal(['http://yahoo.com', 'https://i-beam.org/']);
});
});
}); });