2016-03-26 21:16:21 +00:00
|
|
|
/*
|
2020-04-10 18:06:55 +00:00
|
|
|
* Copyright (C) 2016-2020 Yomichan Authors
|
2016-03-26 21:16:21 +00:00
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
2020-01-01 17:00:31 +00:00
|
|
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
2016-03-26 21:16:21 +00:00
|
|
|
*/
|
|
|
|
|
2020-03-11 02:30:36 +00:00
|
|
|
/* global
|
|
|
|
* AnkiConnect
|
|
|
|
* AnkiNoteBuilder
|
|
|
|
* AudioSystem
|
|
|
|
* AudioUriBuilder
|
|
|
|
* ClipboardMonitor
|
2020-06-28 21:24:06 +00:00
|
|
|
* DictionaryDatabase
|
2020-03-31 00:51:20 +00:00
|
|
|
* DictionaryImporter
|
2020-05-09 15:36:00 +00:00
|
|
|
* Environment
|
2020-08-15 21:23:09 +00:00
|
|
|
* JsonSchemaValidator
|
2020-03-11 02:30:36 +00:00
|
|
|
* Mecab
|
2020-05-06 23:32:28 +00:00
|
|
|
* ObjectPropertyAccessor
|
2020-08-01 15:46:35 +00:00
|
|
|
* OptionsUtil
|
2020-09-04 21:44:00 +00:00
|
|
|
* ProfileConditions
|
2020-08-02 22:58:19 +00:00
|
|
|
* RequestBuilder
|
2020-06-16 00:11:54 +00:00
|
|
|
* TemplateRenderer
|
2020-03-11 02:30:36 +00:00
|
|
|
* Translator
|
2020-03-12 00:33:01 +00:00
|
|
|
* jp
|
2020-03-11 02:30:36 +00:00
|
|
|
*/
|
2016-03-26 21:16:21 +00:00
|
|
|
|
2017-08-14 04:11:10 +00:00
|
|
|
class Backend {
|
2016-03-26 21:16:21 +00:00
|
|
|
constructor() {
|
2020-06-28 18:59:01 +00:00
|
|
|
this._environment = new Environment();
|
2020-06-28 21:24:06 +00:00
|
|
|
this._dictionaryDatabase = new DictionaryDatabase();
|
2020-06-28 18:59:01 +00:00
|
|
|
this._dictionaryImporter = new DictionaryImporter();
|
2020-06-28 21:24:06 +00:00
|
|
|
this._translator = new Translator(this._dictionaryDatabase);
|
2020-06-28 18:59:01 +00:00
|
|
|
this._anki = new AnkiConnect();
|
|
|
|
this._mecab = new Mecab();
|
|
|
|
this._clipboardMonitor = new ClipboardMonitor({getClipboard: this._onApiClipboardGet.bind(this)});
|
|
|
|
this._options = null;
|
|
|
|
this._optionsSchema = null;
|
2020-08-15 21:23:09 +00:00
|
|
|
this._optionsSchemaValidator = new JsonSchemaValidator();
|
2020-09-04 21:44:00 +00:00
|
|
|
this._profileConditionsSchemaCache = [];
|
|
|
|
this._profileConditionsUtil = new ProfileConditions();
|
2020-06-28 18:59:01 +00:00
|
|
|
this._defaultAnkiFieldTemplates = null;
|
2020-08-02 22:58:19 +00:00
|
|
|
this._requestBuilder = new RequestBuilder();
|
|
|
|
this._audioUriBuilder = new AudioUriBuilder({
|
|
|
|
requestBuilder: this._requestBuilder
|
|
|
|
});
|
2020-06-28 18:59:01 +00:00
|
|
|
this._audioSystem = new AudioSystem({
|
|
|
|
audioUriBuilder: this._audioUriBuilder,
|
2020-08-02 22:58:19 +00:00
|
|
|
requestBuilder: this._requestBuilder,
|
2020-04-10 20:12:55 +00:00
|
|
|
useCache: false
|
2020-04-10 17:44:31 +00:00
|
|
|
});
|
2020-06-28 18:59:01 +00:00
|
|
|
this._ankiNoteBuilder = new AnkiNoteBuilder({
|
|
|
|
anki: this._anki,
|
|
|
|
audioSystem: this._audioSystem,
|
2020-04-05 23:34:31 +00:00
|
|
|
renderTemplate: this._renderTemplate.bind(this)
|
|
|
|
});
|
2020-06-16 00:11:54 +00:00
|
|
|
this._templateRenderer = new TemplateRenderer();
|
2020-04-05 23:34:31 +00:00
|
|
|
|
2020-06-28 18:59:01 +00:00
|
|
|
this._clipboardPasteTarget = (
|
2020-05-24 18:01:21 +00:00
|
|
|
typeof document === 'object' && document !== null ?
|
|
|
|
document.querySelector('#clipboard-paste-target') :
|
|
|
|
null
|
|
|
|
);
|
2020-01-26 01:31:31 +00:00
|
|
|
|
2020-07-19 00:30:10 +00:00
|
|
|
this._searchPopupTabId = null;
|
|
|
|
this._searchPopupTabCreatePromise = null;
|
2019-10-27 13:46:27 +00:00
|
|
|
|
2020-04-12 00:45:23 +00:00
|
|
|
this._isPrepared = false;
|
2020-04-12 00:58:52 +00:00
|
|
|
this._prepareError = false;
|
2020-06-08 01:50:14 +00:00
|
|
|
this._preparePromise = null;
|
2020-06-28 18:39:43 +00:00
|
|
|
const {promise, resolve, reject} = deferPromise();
|
|
|
|
this._prepareCompletePromise = promise;
|
|
|
|
this._prepareCompleteResolve = resolve;
|
|
|
|
this._prepareCompleteReject = reject;
|
2020-06-08 01:50:14 +00:00
|
|
|
|
|
|
|
this._defaultBrowserActionTitle = null;
|
2020-04-12 00:53:18 +00:00
|
|
|
this._badgePrepareDelayTimer = null;
|
2020-04-26 20:55:25 +00:00
|
|
|
this._logErrorLevel = null;
|
2020-04-12 00:45:23 +00:00
|
|
|
|
2020-02-27 00:22:32 +00:00
|
|
|
this._messageHandlers = new Map([
|
2020-07-18 21:11:38 +00:00
|
|
|
['requestBackendReadySignal', {async: false, contentScript: true, handler: this._onApiRequestBackendReadySignal.bind(this)}],
|
2020-05-07 23:37:25 +00:00
|
|
|
['optionsSchemaGet', {async: false, contentScript: true, handler: this._onApiOptionsSchemaGet.bind(this)}],
|
|
|
|
['optionsGet', {async: false, contentScript: true, handler: this._onApiOptionsGet.bind(this)}],
|
|
|
|
['optionsGetFull', {async: false, contentScript: true, handler: this._onApiOptionsGetFull.bind(this)}],
|
|
|
|
['optionsSave', {async: true, contentScript: true, handler: this._onApiOptionsSave.bind(this)}],
|
|
|
|
['kanjiFind', {async: true, contentScript: true, handler: this._onApiKanjiFind.bind(this)}],
|
|
|
|
['termsFind', {async: true, contentScript: true, handler: this._onApiTermsFind.bind(this)}],
|
|
|
|
['textParse', {async: true, contentScript: true, handler: this._onApiTextParse.bind(this)}],
|
|
|
|
['definitionAdd', {async: true, contentScript: true, handler: this._onApiDefinitionAdd.bind(this)}],
|
|
|
|
['definitionsAddable', {async: true, contentScript: true, handler: this._onApiDefinitionsAddable.bind(this)}],
|
|
|
|
['noteView', {async: true, contentScript: true, handler: this._onApiNoteView.bind(this)}],
|
|
|
|
['templateRender', {async: true, contentScript: true, handler: this._onApiTemplateRender.bind(this)}],
|
|
|
|
['commandExec', {async: false, contentScript: true, handler: this._onApiCommandExec.bind(this)}],
|
|
|
|
['audioGetUri', {async: true, contentScript: true, handler: this._onApiAudioGetUri.bind(this)}],
|
|
|
|
['screenshotGet', {async: true, contentScript: true, handler: this._onApiScreenshotGet.bind(this)}],
|
|
|
|
['sendMessageToFrame', {async: false, contentScript: true, handler: this._onApiSendMessageToFrame.bind(this)}],
|
|
|
|
['broadcastTab', {async: false, contentScript: true, handler: this._onApiBroadcastTab.bind(this)}],
|
|
|
|
['frameInformationGet', {async: true, contentScript: true, handler: this._onApiFrameInformationGet.bind(this)}],
|
|
|
|
['injectStylesheet', {async: true, contentScript: true, handler: this._onApiInjectStylesheet.bind(this)}],
|
2020-06-25 01:46:13 +00:00
|
|
|
['getStylesheetContent', {async: true, contentScript: true, handler: this._onApiGetStylesheetContent.bind(this)}],
|
2020-05-09 15:36:00 +00:00
|
|
|
['getEnvironmentInfo', {async: false, contentScript: true, handler: this._onApiGetEnvironmentInfo.bind(this)}],
|
2020-05-07 23:37:25 +00:00
|
|
|
['clipboardGet', {async: true, contentScript: true, handler: this._onApiClipboardGet.bind(this)}],
|
|
|
|
['getDisplayTemplatesHtml', {async: true, contentScript: true, handler: this._onApiGetDisplayTemplatesHtml.bind(this)}],
|
|
|
|
['getQueryParserTemplatesHtml', {async: true, contentScript: true, handler: this._onApiGetQueryParserTemplatesHtml.bind(this)}],
|
|
|
|
['getZoom', {async: true, contentScript: true, handler: this._onApiGetZoom.bind(this)}],
|
|
|
|
['getDefaultAnkiFieldTemplates', {async: false, contentScript: true, handler: this._onApiGetDefaultAnkiFieldTemplates.bind(this)}],
|
|
|
|
['getAnkiDeckNames', {async: true, contentScript: false, handler: this._onApiGetAnkiDeckNames.bind(this)}],
|
|
|
|
['getAnkiModelNames', {async: true, contentScript: false, handler: this._onApiGetAnkiModelNames.bind(this)}],
|
|
|
|
['getAnkiModelFieldNames', {async: true, contentScript: false, handler: this._onApiGetAnkiModelFieldNames.bind(this)}],
|
|
|
|
['getDictionaryInfo', {async: true, contentScript: false, handler: this._onApiGetDictionaryInfo.bind(this)}],
|
|
|
|
['getDictionaryCounts', {async: true, contentScript: false, handler: this._onApiGetDictionaryCounts.bind(this)}],
|
|
|
|
['purgeDatabase', {async: true, contentScript: false, handler: this._onApiPurgeDatabase.bind(this)}],
|
|
|
|
['getMedia', {async: true, contentScript: true, handler: this._onApiGetMedia.bind(this)}],
|
|
|
|
['log', {async: false, contentScript: true, handler: this._onApiLog.bind(this)}],
|
|
|
|
['logIndicatorClear', {async: false, contentScript: true, handler: this._onApiLogIndicatorClear.bind(this)}],
|
|
|
|
['createActionPort', {async: false, contentScript: true, handler: this._onApiCreateActionPort.bind(this)}],
|
2020-05-24 17:50:34 +00:00
|
|
|
['modifySettings', {async: true, contentScript: true, handler: this._onApiModifySettings.bind(this)}],
|
2020-05-30 20:23:56 +00:00
|
|
|
['getSettings', {async: false, contentScript: true, handler: this._onApiGetSettings.bind(this)}],
|
2020-07-19 00:30:10 +00:00
|
|
|
['setAllSettings', {async: true, contentScript: false, handler: this._onApiSetAllSettings.bind(this)}],
|
|
|
|
['getOrCreateSearchPopup', {async: true, contentScript: true, handler: this._onApiGetOrCreateSearchPopup.bind(this)}]
|
2020-02-27 00:22:32 +00:00
|
|
|
]);
|
2020-05-06 23:28:26 +00:00
|
|
|
this._messageHandlersWithProgress = new Map([
|
2020-05-07 23:37:25 +00:00
|
|
|
['importDictionaryArchive', {async: true, contentScript: false, handler: this._onApiImportDictionaryArchive.bind(this)}],
|
|
|
|
['deleteDictionary', {async: true, contentScript: false, handler: this._onApiDeleteDictionary.bind(this)}]
|
2020-05-06 23:28:26 +00:00
|
|
|
]);
|
2020-02-27 00:22:32 +00:00
|
|
|
|
|
|
|
this._commandHandlers = new Map([
|
2020-05-07 23:37:25 +00:00
|
|
|
['search', this._onCommandSearch.bind(this)],
|
|
|
|
['help', this._onCommandHelp.bind(this)],
|
2020-02-27 00:22:32 +00:00
|
|
|
['options', this._onCommandOptions.bind(this)],
|
2020-05-07 23:37:25 +00:00
|
|
|
['toggle', this._onCommandToggle.bind(this)]
|
2020-02-27 00:22:32 +00:00
|
|
|
]);
|
2017-08-06 02:02:03 +00:00
|
|
|
}
|
2016-03-26 21:16:21 +00:00
|
|
|
|
2020-06-08 01:50:14 +00:00
|
|
|
prepare() {
|
|
|
|
if (this._preparePromise === null) {
|
|
|
|
const promise = this._prepareInternal();
|
|
|
|
promise.then(
|
|
|
|
(value) => {
|
|
|
|
this._isPrepared = true;
|
|
|
|
this._prepareCompleteResolve(value);
|
|
|
|
},
|
|
|
|
(error) => {
|
|
|
|
this._prepareError = true;
|
|
|
|
this._prepareCompleteReject(error);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
promise.finally(() => this._updateBadge());
|
|
|
|
this._preparePromise = promise;
|
|
|
|
}
|
|
|
|
return this._prepareCompletePromise;
|
|
|
|
}
|
|
|
|
|
2020-06-28 21:22:44 +00:00
|
|
|
_prepareInternalSync() {
|
|
|
|
if (isObject(chrome.commands) && isObject(chrome.commands.onCommand)) {
|
|
|
|
const onCommand = this._onWebExtensionEventWrapper(this._onCommand.bind(this));
|
|
|
|
chrome.commands.onCommand.addListener(onCommand);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isObject(chrome.tabs) && isObject(chrome.tabs.onZoomChange)) {
|
|
|
|
const onZoomChange = this._onWebExtensionEventWrapper(this._onZoomChange.bind(this));
|
|
|
|
chrome.tabs.onZoomChange.addListener(onZoomChange);
|
|
|
|
}
|
|
|
|
|
|
|
|
const onConnect = this._onWebExtensionEventWrapper(this._onConnect.bind(this));
|
|
|
|
chrome.runtime.onConnect.addListener(onConnect);
|
|
|
|
|
|
|
|
const onMessage = this._onMessageWrapper.bind(this);
|
|
|
|
chrome.runtime.onMessage.addListener(onMessage);
|
|
|
|
}
|
|
|
|
|
2020-06-08 01:50:14 +00:00
|
|
|
async _prepareInternal() {
|
2020-04-12 00:58:52 +00:00
|
|
|
try {
|
2020-06-28 21:22:44 +00:00
|
|
|
this._prepareInternalSync();
|
|
|
|
|
2020-04-12 00:58:52 +00:00
|
|
|
this._defaultBrowserActionTitle = await this._getBrowserIconTitle();
|
|
|
|
this._badgePrepareDelayTimer = setTimeout(() => {
|
|
|
|
this._badgePrepareDelayTimer = null;
|
|
|
|
this._updateBadge();
|
|
|
|
}, 1000);
|
2020-04-12 00:53:18 +00:00
|
|
|
this._updateBadge();
|
2019-11-28 20:18:27 +00:00
|
|
|
|
2020-06-13 14:20:12 +00:00
|
|
|
yomichan.on('log', this._onLog.bind(this));
|
|
|
|
|
2020-06-28 18:59:01 +00:00
|
|
|
await this._environment.prepare();
|
2020-06-21 20:12:56 +00:00
|
|
|
try {
|
2020-06-28 21:24:06 +00:00
|
|
|
await this._dictionaryDatabase.prepare();
|
2020-06-21 20:12:56 +00:00
|
|
|
} catch (e) {
|
|
|
|
yomichan.logError(e);
|
|
|
|
}
|
2020-06-28 18:59:01 +00:00
|
|
|
await this._translator.prepare();
|
2020-04-12 00:58:52 +00:00
|
|
|
|
2020-08-02 17:30:55 +00:00
|
|
|
this._optionsSchema = await this._fetchAsset('/bg/data/options-schema.json', true);
|
|
|
|
this._defaultAnkiFieldTemplates = (await this._fetchAsset('/bg/data/default-anki-field-templates.handlebars')).trim();
|
2020-08-01 15:46:35 +00:00
|
|
|
this._options = await OptionsUtil.load();
|
2020-08-15 21:23:09 +00:00
|
|
|
this._options = this._optionsSchemaValidator.getValidValueOrDefault(this._optionsSchema, this._options);
|
2019-11-28 20:18:27 +00:00
|
|
|
|
2020-06-28 18:59:01 +00:00
|
|
|
this._applyOptions('background');
|
2017-08-06 02:02:03 +00:00
|
|
|
|
2020-07-11 19:20:51 +00:00
|
|
|
const options = this.getOptions({current: true});
|
2020-04-12 00:58:52 +00:00
|
|
|
if (options.general.showGuide) {
|
|
|
|
chrome.tabs.create({url: chrome.runtime.getURL('/bg/guide.html')});
|
|
|
|
}
|
2019-09-07 19:06:15 +00:00
|
|
|
|
2020-06-28 18:59:01 +00:00
|
|
|
this._clipboardMonitor.on('change', this._onClipboardTextChange.bind(this));
|
2017-08-06 02:02:03 +00:00
|
|
|
|
2020-07-18 21:11:38 +00:00
|
|
|
this._sendMessageAllTabs('backendReady');
|
2020-06-28 18:59:01 +00:00
|
|
|
const callback = () => this._checkLastError(chrome.runtime.lastError);
|
2020-07-18 21:11:38 +00:00
|
|
|
chrome.runtime.sendMessage({action: 'backendReady'}, callback);
|
2020-04-12 00:58:52 +00:00
|
|
|
} catch (e) {
|
2020-04-26 20:55:25 +00:00
|
|
|
yomichan.logError(e);
|
2020-04-12 00:58:52 +00:00
|
|
|
throw e;
|
|
|
|
} finally {
|
|
|
|
if (this._badgePrepareDelayTimer !== null) {
|
|
|
|
clearTimeout(this._badgePrepareDelayTimer);
|
|
|
|
this._badgePrepareDelayTimer = null;
|
|
|
|
}
|
|
|
|
}
|
2020-04-12 00:45:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
isPrepared() {
|
|
|
|
return this._isPrepared;
|
2020-03-02 02:51:45 +00:00
|
|
|
}
|
2017-08-06 02:02:03 +00:00
|
|
|
|
2020-06-28 18:59:01 +00:00
|
|
|
getFullOptions(useSchema=false) {
|
|
|
|
const options = this._options;
|
2020-08-15 21:23:09 +00:00
|
|
|
return useSchema ? this._optionsSchemaValidator.createProxy(options, this._optionsSchema) : options;
|
2016-03-26 21:16:21 +00:00
|
|
|
}
|
2017-08-06 02:02:03 +00:00
|
|
|
|
2020-06-28 18:59:01 +00:00
|
|
|
getOptions(optionsContext, useSchema=false) {
|
|
|
|
return this._getProfile(optionsContext, useSchema).options;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Event handlers
|
|
|
|
|
2020-07-19 00:30:10 +00:00
|
|
|
async _onClipboardTextChange({text}) {
|
|
|
|
try {
|
|
|
|
const {tab, created} = await this._getOrCreateSearchPopup();
|
|
|
|
await this._focusTab(tab);
|
|
|
|
await this._updateSearchQuery(tab.id, text, !created);
|
|
|
|
} catch (e) {
|
|
|
|
// NOP
|
|
|
|
}
|
2020-06-28 18:59:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
_onLog({level}) {
|
|
|
|
const levelValue = this._getErrorLevelValue(level);
|
|
|
|
if (levelValue <= this._getErrorLevelValue(this._logErrorLevel)) { return; }
|
|
|
|
|
|
|
|
this._logErrorLevel = level;
|
|
|
|
this._updateBadge();
|
2020-03-02 02:51:45 +00:00
|
|
|
}
|
|
|
|
|
2020-06-28 21:22:44 +00:00
|
|
|
// WebExtension event handlers (with prepared checks)
|
|
|
|
|
|
|
|
_onWebExtensionEventWrapper(handler) {
|
|
|
|
return (...args) => {
|
|
|
|
if (this._isPrepared) {
|
|
|
|
handler(...args);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
this._prepareCompletePromise.then(
|
|
|
|
() => { handler(...args); },
|
|
|
|
() => {} // NOP
|
|
|
|
);
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
_onMessageWrapper(message, sender, sendResponse) {
|
|
|
|
if (this._isPrepared) {
|
|
|
|
return this._onMessage(message, sender, sendResponse);
|
|
|
|
}
|
|
|
|
|
|
|
|
this._prepareCompletePromise.then(
|
|
|
|
() => { this._onMessage(message, sender, sendResponse); },
|
|
|
|
() => { sendResponse(); }
|
|
|
|
);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2020-06-28 18:59:01 +00:00
|
|
|
// WebExtension event handlers
|
|
|
|
|
|
|
|
_onCommand(command) {
|
|
|
|
this._runCommand(command);
|
|
|
|
}
|
|
|
|
|
|
|
|
_onMessage({action, params}, sender, callback) {
|
2020-04-07 23:41:02 +00:00
|
|
|
const messageHandler = this._messageHandlers.get(action);
|
|
|
|
if (typeof messageHandler === 'undefined') { return false; }
|
|
|
|
|
2020-07-11 19:20:00 +00:00
|
|
|
if (!messageHandler.contentScript) {
|
|
|
|
try {
|
2020-05-07 23:37:25 +00:00
|
|
|
this._validatePrivilegedMessageSender(sender);
|
2020-07-11 19:20:00 +00:00
|
|
|
} catch (error) {
|
|
|
|
callback({error: errorToJson(error)});
|
2020-04-07 23:41:02 +00:00
|
|
|
return false;
|
|
|
|
}
|
2017-08-06 02:02:03 +00:00
|
|
|
}
|
2020-07-11 19:20:00 +00:00
|
|
|
|
|
|
|
return yomichan.invokeMessageHandler(messageHandler, params, callback, sender);
|
2017-08-06 02:02:03 +00:00
|
|
|
}
|
2019-02-20 03:49:25 +00:00
|
|
|
|
2020-05-23 17:34:55 +00:00
|
|
|
_onConnect(port) {
|
|
|
|
try {
|
2020-07-18 18:16:35 +00:00
|
|
|
let details;
|
|
|
|
try {
|
|
|
|
details = JSON.parse(port.name);
|
|
|
|
} catch (e) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (details.name !== 'background-cross-frame-communication-port') { return; }
|
2020-05-23 17:34:55 +00:00
|
|
|
|
|
|
|
const tabId = (port.sender && port.sender.tab ? port.sender.tab.id : null);
|
|
|
|
if (typeof tabId !== 'number') {
|
|
|
|
throw new Error('Port does not have an associated tab ID');
|
|
|
|
}
|
|
|
|
const senderFrameId = port.sender.frameId;
|
2020-07-18 18:16:35 +00:00
|
|
|
if (typeof senderFrameId !== 'number') {
|
2020-05-23 17:34:55 +00:00
|
|
|
throw new Error('Port does not have an associated frame ID');
|
|
|
|
}
|
2020-07-18 18:16:35 +00:00
|
|
|
let {targetTabId, targetFrameId} = details;
|
|
|
|
if (typeof targetTabId !== 'number') {
|
|
|
|
targetTabId = tabId;
|
|
|
|
}
|
2020-05-23 17:34:55 +00:00
|
|
|
|
2020-07-18 18:16:35 +00:00
|
|
|
const details2 = {
|
|
|
|
name: 'cross-frame-communication-port',
|
|
|
|
sourceFrameId: senderFrameId
|
|
|
|
};
|
|
|
|
let forwardPort = chrome.tabs.connect(targetTabId, {frameId: targetFrameId, name: JSON.stringify(details2)});
|
2020-05-23 17:34:55 +00:00
|
|
|
|
|
|
|
const cleanup = () => {
|
2020-06-28 18:59:01 +00:00
|
|
|
this._checkLastError(chrome.runtime.lastError);
|
2020-05-23 17:34:55 +00:00
|
|
|
if (forwardPort !== null) {
|
|
|
|
forwardPort.disconnect();
|
|
|
|
forwardPort = null;
|
|
|
|
}
|
|
|
|
if (port !== null) {
|
|
|
|
port.disconnect();
|
|
|
|
port = null;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
port.onMessage.addListener((message) => { forwardPort.postMessage(message); });
|
|
|
|
forwardPort.onMessage.addListener((message) => { port.postMessage(message); });
|
|
|
|
port.onDisconnect.addListener(cleanup);
|
|
|
|
forwardPort.onDisconnect.addListener(cleanup);
|
|
|
|
} catch (e) {
|
|
|
|
port.disconnect();
|
|
|
|
yomichan.logError(e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-23 16:59:47 +00:00
|
|
|
_onZoomChange({tabId, oldZoomFactor, newZoomFactor}) {
|
2020-06-28 18:59:01 +00:00
|
|
|
const callback = () => this._checkLastError(chrome.runtime.lastError);
|
2019-12-23 16:59:47 +00:00
|
|
|
chrome.tabs.sendMessage(tabId, {action: 'zoomChanged', params: {oldZoomFactor, newZoomFactor}}, callback);
|
|
|
|
}
|
|
|
|
|
2019-12-09 01:53:42 +00:00
|
|
|
// Message handlers
|
|
|
|
|
2020-07-18 21:11:38 +00:00
|
|
|
_onApiRequestBackendReadySignal(_params, sender) {
|
2020-03-02 21:26:55 +00:00
|
|
|
// tab ID isn't set in background (e.g. browser_action)
|
2020-06-28 18:59:01 +00:00
|
|
|
const callback = () => this._checkLastError(chrome.runtime.lastError);
|
2020-07-18 21:11:38 +00:00
|
|
|
const data = {action: 'backendReady'};
|
2020-03-02 21:26:55 +00:00
|
|
|
if (typeof sender.tab === 'undefined') {
|
2020-04-07 23:59:10 +00:00
|
|
|
chrome.runtime.sendMessage(data, callback);
|
|
|
|
return false;
|
|
|
|
} else {
|
|
|
|
chrome.tabs.sendMessage(sender.tab.id, data, callback);
|
|
|
|
return true;
|
2020-03-02 21:26:55 +00:00
|
|
|
}
|
2020-03-01 22:39:15 +00:00
|
|
|
}
|
|
|
|
|
2020-04-07 23:47:46 +00:00
|
|
|
_onApiOptionsSchemaGet() {
|
2020-06-28 18:59:01 +00:00
|
|
|
return this._optionsSchema;
|
2019-12-14 21:40:05 +00:00
|
|
|
}
|
|
|
|
|
2020-04-07 23:47:46 +00:00
|
|
|
_onApiOptionsGet({optionsContext}) {
|
2019-12-10 02:05:15 +00:00
|
|
|
return this.getOptions(optionsContext);
|
2019-12-09 01:53:42 +00:00
|
|
|
}
|
|
|
|
|
2020-04-07 23:47:46 +00:00
|
|
|
_onApiOptionsGetFull() {
|
2019-12-10 02:08:11 +00:00
|
|
|
return this.getFullOptions();
|
2019-12-10 02:00:49 +00:00
|
|
|
}
|
|
|
|
|
2019-12-10 02:11:10 +00:00
|
|
|
async _onApiOptionsSave({source}) {
|
2020-09-04 21:44:00 +00:00
|
|
|
this._clearProfileConditionsSchemaCache();
|
2020-03-01 21:06:37 +00:00
|
|
|
const options = this.getFullOptions();
|
2020-08-01 15:46:35 +00:00
|
|
|
await OptionsUtil.save(options);
|
2020-06-28 18:59:01 +00:00
|
|
|
this._applyOptions(source);
|
2019-12-10 02:00:49 +00:00
|
|
|
}
|
|
|
|
|
2019-12-10 02:13:03 +00:00
|
|
|
async _onApiKanjiFind({text, optionsContext}) {
|
2020-03-01 21:06:37 +00:00
|
|
|
const options = this.getOptions(optionsContext);
|
2020-06-28 18:59:01 +00:00
|
|
|
const definitions = await this._translator.findKanji(text, options);
|
2019-12-10 02:13:03 +00:00
|
|
|
definitions.splice(options.general.maxResults);
|
|
|
|
return definitions;
|
2019-12-09 01:53:42 +00:00
|
|
|
}
|
|
|
|
|
2019-12-10 02:15:37 +00:00
|
|
|
async _onApiTermsFind({text, details, optionsContext}) {
|
2020-03-01 21:06:37 +00:00
|
|
|
const options = this.getOptions(optionsContext);
|
2020-02-15 20:09:59 +00:00
|
|
|
const mode = options.general.resultOutputMode;
|
2020-06-28 18:59:01 +00:00
|
|
|
const [definitions, length] = await this._translator.findTerms(mode, text, details, options);
|
2019-12-10 02:15:37 +00:00
|
|
|
definitions.splice(options.general.maxResults);
|
|
|
|
return {length, definitions};
|
2019-12-09 01:53:42 +00:00
|
|
|
}
|
|
|
|
|
2019-12-10 02:18:23 +00:00
|
|
|
async _onApiTextParse({text, optionsContext}) {
|
2020-03-01 21:06:37 +00:00
|
|
|
const options = this.getOptions(optionsContext);
|
2019-12-10 02:18:23 +00:00
|
|
|
const results = [];
|
2020-04-12 00:53:24 +00:00
|
|
|
|
|
|
|
if (options.parsing.enableScanningParser) {
|
|
|
|
results.push({
|
|
|
|
source: 'scanning-parser',
|
|
|
|
id: 'scan',
|
|
|
|
content: await this._textParseScanning(text, options)
|
|
|
|
});
|
2019-12-10 02:18:23 +00:00
|
|
|
}
|
2019-12-09 01:53:42 +00:00
|
|
|
|
2020-04-12 00:53:24 +00:00
|
|
|
if (options.parsing.enableMecabParser) {
|
|
|
|
const mecabResults = await this._textParseMecab(text, options);
|
|
|
|
for (const [mecabDictName, mecabDictResults] of mecabResults) {
|
|
|
|
results.push({
|
|
|
|
source: 'mecab',
|
|
|
|
dictionary: mecabDictName,
|
|
|
|
id: `mecab-${mecabDictName}`,
|
|
|
|
content: mecabDictResults
|
|
|
|
});
|
2019-12-10 02:21:17 +00:00
|
|
|
}
|
|
|
|
}
|
2020-04-12 00:53:24 +00:00
|
|
|
|
2019-12-10 02:21:17 +00:00
|
|
|
return results;
|
2019-12-09 01:53:42 +00:00
|
|
|
}
|
|
|
|
|
2020-03-15 21:13:00 +00:00
|
|
|
async _onApiDefinitionAdd({definition, mode, context, details, optionsContext}) {
|
2020-03-01 21:06:37 +00:00
|
|
|
const options = this.getOptions(optionsContext);
|
2020-04-23 16:58:31 +00:00
|
|
|
const templates = this._getTemplates(options);
|
2019-12-10 02:25:34 +00:00
|
|
|
|
|
|
|
if (mode !== 'kanji') {
|
2020-04-10 17:44:31 +00:00
|
|
|
const {customSourceUrl} = options.audio;
|
2020-06-28 18:59:01 +00:00
|
|
|
await this._ankiNoteBuilder.injectAudio(
|
2019-12-10 02:25:34 +00:00
|
|
|
definition,
|
|
|
|
options.anki.terms.fields,
|
|
|
|
options.audio.sources,
|
2020-05-02 16:50:16 +00:00
|
|
|
customSourceUrl
|
2019-12-10 02:25:34 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-03-15 21:02:34 +00:00
|
|
|
if (details && details.screenshot) {
|
2020-06-28 18:59:01 +00:00
|
|
|
await this._ankiNoteBuilder.injectScreenshot(
|
2019-12-10 02:25:34 +00:00
|
|
|
definition,
|
|
|
|
options.anki.terms.fields,
|
2020-04-12 16:46:32 +00:00
|
|
|
details.screenshot
|
2019-12-10 02:25:34 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-06-28 18:59:01 +00:00
|
|
|
const note = await this._ankiNoteBuilder.createNote(definition, mode, context, options, templates);
|
|
|
|
return this._anki.addNote(note);
|
2019-12-09 01:53:42 +00:00
|
|
|
}
|
|
|
|
|
2020-03-15 21:13:00 +00:00
|
|
|
async _onApiDefinitionsAddable({definitions, modes, context, optionsContext}) {
|
2020-03-01 21:06:37 +00:00
|
|
|
const options = this.getOptions(optionsContext);
|
2020-04-23 16:58:31 +00:00
|
|
|
const templates = this._getTemplates(options);
|
2019-12-10 02:28:27 +00:00
|
|
|
const states = [];
|
|
|
|
|
|
|
|
try {
|
2020-06-21 19:54:34 +00:00
|
|
|
const notePromises = [];
|
2019-12-10 02:28:27 +00:00
|
|
|
for (const definition of definitions) {
|
|
|
|
for (const mode of modes) {
|
2020-06-28 18:59:01 +00:00
|
|
|
const notePromise = this._ankiNoteBuilder.createNote(definition, mode, context, options, templates);
|
2020-06-21 19:54:34 +00:00
|
|
|
notePromises.push(notePromise);
|
2019-12-10 02:28:27 +00:00
|
|
|
}
|
|
|
|
}
|
2020-06-21 19:54:34 +00:00
|
|
|
const notes = await Promise.all(notePromises);
|
2019-12-10 02:28:27 +00:00
|
|
|
|
|
|
|
const cannotAdd = [];
|
2020-06-28 18:59:01 +00:00
|
|
|
const results = await this._anki.canAddNotes(notes);
|
2019-12-10 02:28:27 +00:00
|
|
|
for (let resultBase = 0; resultBase < results.length; resultBase += modes.length) {
|
|
|
|
const state = {};
|
|
|
|
for (let modeOffset = 0; modeOffset < modes.length; ++modeOffset) {
|
|
|
|
const index = resultBase + modeOffset;
|
|
|
|
const result = results[index];
|
|
|
|
const info = {canAdd: result};
|
|
|
|
state[modes[modeOffset]] = info;
|
|
|
|
if (!result) {
|
|
|
|
cannotAdd.push([notes[index], info]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
states.push(state);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cannotAdd.length > 0) {
|
2020-06-28 18:59:01 +00:00
|
|
|
const noteIdsArray = await this._anki.findNoteIds(cannotAdd.map((e) => e[0]), options.anki.duplicateScope);
|
2019-12-10 02:28:27 +00:00
|
|
|
for (let i = 0, ii = Math.min(cannotAdd.length, noteIdsArray.length); i < ii; ++i) {
|
|
|
|
const noteIds = noteIdsArray[i];
|
|
|
|
if (noteIds.length > 0) {
|
|
|
|
cannotAdd[i][1].noteId = noteIds[0];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} catch (e) {
|
|
|
|
// NOP
|
|
|
|
}
|
|
|
|
|
|
|
|
return states;
|
2019-12-09 01:53:42 +00:00
|
|
|
}
|
|
|
|
|
2019-12-10 02:29:48 +00:00
|
|
|
async _onApiNoteView({noteId}) {
|
2020-06-28 18:59:01 +00:00
|
|
|
return await this._anki.guiBrowse(`nid:${noteId}`);
|
2019-12-09 01:53:42 +00:00
|
|
|
}
|
|
|
|
|
2020-02-14 01:18:15 +00:00
|
|
|
async _onApiTemplateRender({template, data}) {
|
2020-03-07 20:20:45 +00:00
|
|
|
return this._renderTemplate(template, data);
|
2019-12-09 01:53:42 +00:00
|
|
|
}
|
|
|
|
|
2020-04-07 23:47:46 +00:00
|
|
|
_onApiCommandExec({command, params}) {
|
2019-12-13 00:59:43 +00:00
|
|
|
return this._runCommand(command, params);
|
2019-12-09 01:53:42 +00:00
|
|
|
}
|
|
|
|
|
2020-04-10 17:44:31 +00:00
|
|
|
async _onApiAudioGetUri({definition, source, details}) {
|
2020-06-28 18:59:01 +00:00
|
|
|
return await this._audioUriBuilder.getUri(definition, source, details);
|
2019-12-09 01:53:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
_onApiScreenshotGet({options}, sender) {
|
2019-12-10 02:54:03 +00:00
|
|
|
if (!(sender && sender.tab)) {
|
|
|
|
return Promise.resolve();
|
|
|
|
}
|
|
|
|
|
|
|
|
const windowId = sender.tab.windowId;
|
|
|
|
return new Promise((resolve) => {
|
|
|
|
chrome.tabs.captureVisibleTab(windowId, options, (dataUrl) => resolve(dataUrl));
|
|
|
|
});
|
2019-12-09 01:53:42 +00:00
|
|
|
}
|
|
|
|
|
2020-07-08 23:58:06 +00:00
|
|
|
_onApiSendMessageToFrame({frameId: targetFrameId, action, params}, sender) {
|
2020-05-06 23:27:21 +00:00
|
|
|
if (!(sender && sender.tab)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
const tabId = sender.tab.id;
|
2020-07-08 23:58:06 +00:00
|
|
|
const frameId = sender.frameId;
|
2020-06-28 18:59:01 +00:00
|
|
|
const callback = () => this._checkLastError(chrome.runtime.lastError);
|
2020-07-08 23:58:06 +00:00
|
|
|
chrome.tabs.sendMessage(tabId, {action, params, frameId}, {frameId: targetFrameId}, callback);
|
2020-05-06 23:27:21 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2020-04-11 00:00:18 +00:00
|
|
|
_onApiBroadcastTab({action, params}, sender) {
|
2019-12-10 02:54:45 +00:00
|
|
|
if (!(sender && sender.tab)) {
|
2020-04-07 23:49:54 +00:00
|
|
|
return false;
|
2019-12-10 02:54:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const tabId = sender.tab.id;
|
2020-07-08 23:58:06 +00:00
|
|
|
const frameId = sender.frameId;
|
2020-06-28 18:59:01 +00:00
|
|
|
const callback = () => this._checkLastError(chrome.runtime.lastError);
|
2020-07-08 23:58:06 +00:00
|
|
|
chrome.tabs.sendMessage(tabId, {action, params, frameId}, callback);
|
2020-04-07 23:49:54 +00:00
|
|
|
return true;
|
2019-12-09 01:53:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
_onApiFrameInformationGet(params, sender) {
|
2019-12-10 02:55:45 +00:00
|
|
|
const frameId = sender.frameId;
|
|
|
|
return Promise.resolve({frameId});
|
2019-12-09 01:53:42 +00:00
|
|
|
}
|
|
|
|
|
2020-02-16 18:13:04 +00:00
|
|
|
_onApiInjectStylesheet({type, value}, sender) {
|
2019-12-10 02:56:47 +00:00
|
|
|
if (!sender.tab) {
|
|
|
|
return Promise.reject(new Error('Invalid tab'));
|
|
|
|
}
|
|
|
|
|
|
|
|
const tabId = sender.tab.id;
|
|
|
|
const frameId = sender.frameId;
|
2020-02-16 18:13:04 +00:00
|
|
|
const details = (
|
|
|
|
type === 'file' ?
|
|
|
|
{
|
|
|
|
file: value,
|
|
|
|
runAt: 'document_start',
|
|
|
|
cssOrigin: 'author',
|
2020-02-16 19:34:49 +00:00
|
|
|
allFrames: false,
|
|
|
|
matchAboutBlank: true
|
2020-02-16 18:13:04 +00:00
|
|
|
} :
|
|
|
|
{
|
|
|
|
code: value,
|
|
|
|
runAt: 'document_start',
|
|
|
|
cssOrigin: 'user',
|
2020-02-16 19:34:49 +00:00
|
|
|
allFrames: false,
|
|
|
|
matchAboutBlank: true
|
2020-02-16 18:13:04 +00:00
|
|
|
}
|
|
|
|
);
|
2019-12-10 02:56:47 +00:00
|
|
|
if (typeof frameId === 'number') {
|
|
|
|
details.frameId = frameId;
|
|
|
|
}
|
|
|
|
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
chrome.tabs.insertCSS(tabId, details, () => {
|
|
|
|
const e = chrome.runtime.lastError;
|
|
|
|
if (e) {
|
|
|
|
reject(new Error(e.message));
|
|
|
|
} else {
|
|
|
|
resolve();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
2019-12-09 01:53:42 +00:00
|
|
|
}
|
|
|
|
|
2020-06-25 01:46:13 +00:00
|
|
|
async _onApiGetStylesheetContent({url}) {
|
|
|
|
if (!url.startsWith('/') || url.startsWith('//') || !url.endsWith('.css')) {
|
|
|
|
throw new Error('Invalid URL');
|
|
|
|
}
|
2020-08-02 17:30:55 +00:00
|
|
|
return await this._fetchAsset(url);
|
2020-06-25 01:46:13 +00:00
|
|
|
}
|
|
|
|
|
2020-05-09 15:36:00 +00:00
|
|
|
_onApiGetEnvironmentInfo() {
|
2020-06-28 18:59:01 +00:00
|
|
|
return this._environment.getInfo();
|
2019-12-09 01:53:42 +00:00
|
|
|
}
|
|
|
|
|
2019-12-10 02:59:18 +00:00
|
|
|
async _onApiClipboardGet() {
|
2020-01-25 16:11:19 +00:00
|
|
|
/*
|
|
|
|
Notes:
|
|
|
|
document.execCommand('paste') doesn't work on Firefox.
|
|
|
|
This may be a bug: https://bugzilla.mozilla.org/show_bug.cgi?id=1603985
|
|
|
|
Therefore, navigator.clipboard.readText() is used on Firefox.
|
|
|
|
|
|
|
|
navigator.clipboard.readText() can't be used in Chrome for two reasons:
|
|
|
|
* Requires page to be focused, else it rejects with an exception.
|
|
|
|
* When the page is focused, Chrome will request clipboard permission, despite already
|
|
|
|
being an extension with clipboard permissions. It effectively asks for the
|
|
|
|
non-extension permission for clipboard access.
|
|
|
|
*/
|
2020-06-28 18:59:01 +00:00
|
|
|
const {browser} = this._environment.getInfo();
|
2020-01-25 16:11:19 +00:00
|
|
|
if (browser === 'firefox' || browser === 'firefox-mobile') {
|
|
|
|
return await navigator.clipboard.readText();
|
|
|
|
} else {
|
2020-06-28 18:59:01 +00:00
|
|
|
const clipboardPasteTarget = this._clipboardPasteTarget;
|
2020-05-24 18:01:21 +00:00
|
|
|
if (clipboardPasteTarget === null) {
|
|
|
|
throw new Error('Reading the clipboard is not supported in this context');
|
|
|
|
}
|
2020-01-25 16:11:19 +00:00
|
|
|
clipboardPasteTarget.value = '';
|
|
|
|
clipboardPasteTarget.focus();
|
|
|
|
document.execCommand('paste');
|
|
|
|
const result = clipboardPasteTarget.value;
|
|
|
|
clipboardPasteTarget.value = '';
|
|
|
|
return result;
|
|
|
|
}
|
2019-12-09 01:53:42 +00:00
|
|
|
}
|
2019-12-10 02:25:34 +00:00
|
|
|
|
2019-12-27 23:58:11 +00:00
|
|
|
async _onApiGetDisplayTemplatesHtml() {
|
2020-08-02 17:30:55 +00:00
|
|
|
return await this._fetchAsset('/mixed/display-templates.html');
|
2019-12-27 23:58:11 +00:00
|
|
|
}
|
|
|
|
|
2020-02-06 02:00:02 +00:00
|
|
|
async _onApiGetQueryParserTemplatesHtml() {
|
2020-08-02 17:30:55 +00:00
|
|
|
return await this._fetchAsset('/bg/query-parser-templates.html');
|
2020-02-06 02:00:02 +00:00
|
|
|
}
|
|
|
|
|
2019-12-23 16:59:47 +00:00
|
|
|
_onApiGetZoom(params, sender) {
|
|
|
|
if (!sender || !sender.tab) {
|
|
|
|
return Promise.reject(new Error('Invalid tab'));
|
|
|
|
}
|
|
|
|
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
const tabId = sender.tab.id;
|
2020-01-11 20:34:12 +00:00
|
|
|
if (!(
|
|
|
|
chrome.tabs !== null &&
|
|
|
|
typeof chrome.tabs === 'object' &&
|
|
|
|
typeof chrome.tabs.getZoom === 'function'
|
|
|
|
)) {
|
|
|
|
// Not supported
|
|
|
|
resolve({zoomFactor: 1.0});
|
|
|
|
return;
|
|
|
|
}
|
2019-12-23 16:59:47 +00:00
|
|
|
chrome.tabs.getZoom(tabId, (zoomFactor) => {
|
|
|
|
const e = chrome.runtime.lastError;
|
|
|
|
if (e) {
|
|
|
|
reject(new Error(e.message));
|
|
|
|
} else {
|
|
|
|
resolve({zoomFactor});
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2020-04-07 23:47:46 +00:00
|
|
|
_onApiGetDefaultAnkiFieldTemplates() {
|
2020-06-28 18:59:01 +00:00
|
|
|
return this._defaultAnkiFieldTemplates;
|
2020-02-28 01:33:13 +00:00
|
|
|
}
|
|
|
|
|
2020-05-07 23:37:25 +00:00
|
|
|
async _onApiGetAnkiDeckNames() {
|
2020-06-28 18:59:01 +00:00
|
|
|
return await this._anki.getDeckNames();
|
2020-04-11 19:21:43 +00:00
|
|
|
}
|
|
|
|
|
2020-05-07 23:37:25 +00:00
|
|
|
async _onApiGetAnkiModelNames() {
|
2020-06-28 18:59:01 +00:00
|
|
|
return await this._anki.getModelNames();
|
2020-04-11 19:21:43 +00:00
|
|
|
}
|
|
|
|
|
2020-05-07 23:37:25 +00:00
|
|
|
async _onApiGetAnkiModelFieldNames({modelName}) {
|
2020-06-28 18:59:01 +00:00
|
|
|
return await this._anki.getModelFieldNames(modelName);
|
2020-04-11 19:21:43 +00:00
|
|
|
}
|
|
|
|
|
2020-05-07 23:37:25 +00:00
|
|
|
async _onApiGetDictionaryInfo() {
|
2020-08-09 17:21:14 +00:00
|
|
|
return await this._dictionaryDatabase.getDictionaryInfo();
|
2020-04-11 19:21:43 +00:00
|
|
|
}
|
|
|
|
|
2020-05-07 23:37:25 +00:00
|
|
|
async _onApiGetDictionaryCounts({dictionaryNames, getTotal}) {
|
2020-08-09 17:21:14 +00:00
|
|
|
return await this._dictionaryDatabase.getDictionaryCounts(dictionaryNames, getTotal);
|
2020-04-11 19:21:43 +00:00
|
|
|
}
|
|
|
|
|
2020-05-07 23:37:25 +00:00
|
|
|
async _onApiPurgeDatabase() {
|
2020-06-28 18:59:01 +00:00
|
|
|
this._translator.clearDatabaseCaches();
|
2020-06-28 21:24:06 +00:00
|
|
|
await this._dictionaryDatabase.purge();
|
2020-04-11 19:21:43 +00:00
|
|
|
}
|
|
|
|
|
2020-04-11 18:23:02 +00:00
|
|
|
async _onApiGetMedia({targets}) {
|
2020-06-28 21:24:06 +00:00
|
|
|
return await this._dictionaryDatabase.getMedia(targets);
|
2020-04-11 18:23:02 +00:00
|
|
|
}
|
|
|
|
|
2020-04-26 20:55:25 +00:00
|
|
|
_onApiLog({error, level, context}) {
|
|
|
|
yomichan.log(jsonToError(error), level, context);
|
|
|
|
}
|
|
|
|
|
|
|
|
_onApiLogIndicatorClear() {
|
|
|
|
if (this._logErrorLevel === null) { return; }
|
|
|
|
this._logErrorLevel = null;
|
|
|
|
this._updateBadge();
|
|
|
|
}
|
|
|
|
|
2020-05-02 16:57:13 +00:00
|
|
|
_onApiCreateActionPort(params, sender) {
|
|
|
|
if (!sender || !sender.tab) { throw new Error('Invalid sender'); }
|
|
|
|
const tabId = sender.tab.id;
|
|
|
|
if (typeof tabId !== 'number') { throw new Error('Sender has invalid tab ID'); }
|
|
|
|
|
|
|
|
const frameId = sender.frameId;
|
2020-08-22 19:49:24 +00:00
|
|
|
const id = generateId(16);
|
2020-07-18 18:16:35 +00:00
|
|
|
const details = {
|
|
|
|
name: 'action-port',
|
|
|
|
id
|
|
|
|
};
|
2020-05-02 16:57:13 +00:00
|
|
|
|
2020-07-18 18:16:35 +00:00
|
|
|
const port = chrome.tabs.connect(tabId, {name: JSON.stringify(details), frameId});
|
2020-05-02 16:57:13 +00:00
|
|
|
try {
|
|
|
|
this._createActionListenerPort(port, sender, this._messageHandlersWithProgress);
|
|
|
|
} catch (e) {
|
|
|
|
port.disconnect();
|
|
|
|
throw e;
|
|
|
|
}
|
|
|
|
|
2020-07-18 18:16:35 +00:00
|
|
|
return details;
|
2020-05-02 16:57:13 +00:00
|
|
|
}
|
|
|
|
|
2020-05-06 23:28:26 +00:00
|
|
|
async _onApiImportDictionaryArchive({archiveContent, details}, sender, onProgress) {
|
2020-07-20 02:05:37 +00:00
|
|
|
return await this._dictionaryImporter.importDictionary(this._dictionaryDatabase, archiveContent, details, onProgress);
|
2020-05-06 23:28:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
async _onApiDeleteDictionary({dictionaryName}, sender, onProgress) {
|
2020-06-28 18:59:01 +00:00
|
|
|
this._translator.clearDatabaseCaches();
|
2020-06-28 21:24:06 +00:00
|
|
|
await this._dictionaryDatabase.deleteDictionary(dictionaryName, {rate: 1000}, onProgress);
|
2020-05-06 23:28:26 +00:00
|
|
|
}
|
|
|
|
|
2020-05-06 23:32:28 +00:00
|
|
|
async _onApiModifySettings({targets, source}) {
|
|
|
|
const results = [];
|
|
|
|
for (const target of targets) {
|
|
|
|
try {
|
2020-05-24 17:50:34 +00:00
|
|
|
const result = this._modifySetting(target);
|
2020-06-28 16:38:34 +00:00
|
|
|
results.push({result: clone(result)});
|
2020-05-06 23:32:28 +00:00
|
|
|
} catch (e) {
|
|
|
|
results.push({error: errorToJson(e)});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
await this._onApiOptionsSave({source});
|
|
|
|
return results;
|
|
|
|
}
|
|
|
|
|
2020-05-24 17:50:34 +00:00
|
|
|
_onApiGetSettings({targets}) {
|
|
|
|
const results = [];
|
|
|
|
for (const target of targets) {
|
|
|
|
try {
|
|
|
|
const result = this._getSetting(target);
|
2020-06-28 16:38:34 +00:00
|
|
|
results.push({result: clone(result)});
|
2020-05-24 17:50:34 +00:00
|
|
|
} catch (e) {
|
|
|
|
results.push({error: errorToJson(e)});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return results;
|
|
|
|
}
|
|
|
|
|
2020-05-30 20:23:56 +00:00
|
|
|
async _onApiSetAllSettings({value, source}) {
|
2020-08-15 21:23:09 +00:00
|
|
|
this._options = this._optionsSchemaValidator.getValidValueOrDefault(this._optionsSchema, value);
|
2020-05-30 20:23:56 +00:00
|
|
|
await this._onApiOptionsSave({source});
|
|
|
|
}
|
|
|
|
|
2020-07-19 00:30:10 +00:00
|
|
|
async _onApiGetOrCreateSearchPopup({focus=false, text=null}) {
|
|
|
|
const {tab, created} = await this._getOrCreateSearchPopup();
|
|
|
|
if (focus === true || (focus === 'ifCreated' && created)) {
|
|
|
|
await this._focusTab(tab);
|
|
|
|
}
|
|
|
|
if (typeof text === 'string') {
|
|
|
|
await this._updateSearchQuery(tab.id, text, !created);
|
|
|
|
}
|
|
|
|
return {tabId: tab.id, windowId: tab.windowId};
|
|
|
|
}
|
|
|
|
|
2019-12-10 02:41:24 +00:00
|
|
|
// Command handlers
|
|
|
|
|
2020-06-28 18:59:01 +00:00
|
|
|
async _onCommandSearch(params) {
|
|
|
|
const {mode='existingOrNewTab', query} = params || {};
|
|
|
|
|
|
|
|
const baseUrl = chrome.runtime.getURL('/bg/search.html');
|
2020-07-26 20:52:45 +00:00
|
|
|
const queryParams = {};
|
2020-06-28 18:59:01 +00:00
|
|
|
if (query && query.length > 0) { queryParams.query = query; }
|
|
|
|
const queryString = new URLSearchParams(queryParams).toString();
|
2020-07-26 20:52:45 +00:00
|
|
|
let url = baseUrl;
|
|
|
|
if (queryString.length > 0) {
|
|
|
|
url += `?${queryString}`;
|
|
|
|
}
|
2020-06-28 18:59:01 +00:00
|
|
|
|
|
|
|
const isTabMatch = (url2) => {
|
|
|
|
if (url2 === null || !url2.startsWith(baseUrl)) { return false; }
|
|
|
|
const {baseUrl: baseUrl2, queryParams: queryParams2} = parseUrl(url2);
|
|
|
|
return baseUrl2 === baseUrl && (queryParams2.mode === mode || (!queryParams2.mode && mode === 'existingOrNewTab'));
|
|
|
|
};
|
|
|
|
|
|
|
|
const openInTab = async () => {
|
|
|
|
const tab = await this._findTab(1000, isTabMatch);
|
|
|
|
if (tab !== null) {
|
|
|
|
await this._focusTab(tab);
|
|
|
|
if (queryParams.query) {
|
2020-07-19 00:30:10 +00:00
|
|
|
await this._updateSearchQuery(tab.id, queryParams.query, true);
|
2020-06-28 18:59:01 +00:00
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
switch (mode) {
|
|
|
|
case 'existingOrNewTab':
|
|
|
|
try {
|
|
|
|
if (await openInTab()) { return; }
|
|
|
|
} catch (e) {
|
|
|
|
// NOP
|
|
|
|
}
|
|
|
|
chrome.tabs.create({url});
|
|
|
|
return;
|
|
|
|
case 'newTab':
|
|
|
|
chrome.tabs.create({url});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
_onCommandHelp() {
|
|
|
|
chrome.tabs.create({url: 'https://foosoft.net/projects/yomichan/'});
|
|
|
|
}
|
|
|
|
|
|
|
|
_onCommandOptions(params) {
|
|
|
|
const {mode='existingOrNewTab'} = params || {};
|
|
|
|
if (mode === 'existingOrNewTab') {
|
|
|
|
chrome.runtime.openOptionsPage();
|
|
|
|
} else if (mode === 'newTab') {
|
|
|
|
const manifest = chrome.runtime.getManifest();
|
|
|
|
const url = chrome.runtime.getURL(manifest.options_ui.page);
|
|
|
|
chrome.tabs.create({url});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async _onCommandToggle() {
|
|
|
|
const source = 'popup';
|
2020-07-11 19:20:51 +00:00
|
|
|
const options = this.getOptions({current: true});
|
2020-06-28 18:59:01 +00:00
|
|
|
options.general.enable = !options.general.enable;
|
|
|
|
await this._onApiOptionsSave({source});
|
|
|
|
}
|
|
|
|
|
|
|
|
// Utilities
|
|
|
|
|
2020-07-19 00:30:10 +00:00
|
|
|
_getOrCreateSearchPopup() {
|
|
|
|
if (this._searchPopupTabCreatePromise === null) {
|
|
|
|
const promise = this._getOrCreateSearchPopup2();
|
|
|
|
this._searchPopupTabCreatePromise = promise;
|
|
|
|
promise.then(() => { this._searchPopupTabCreatePromise = null; });
|
|
|
|
}
|
|
|
|
return this._searchPopupTabCreatePromise;
|
|
|
|
}
|
|
|
|
|
|
|
|
async _getOrCreateSearchPopup2() {
|
|
|
|
// Reuse same tab
|
|
|
|
const baseUrl = chrome.runtime.getURL('/bg/search.html');
|
|
|
|
if (this._searchPopupTabId !== null) {
|
|
|
|
const tabId = this._searchPopupTabId;
|
|
|
|
const tab = await new Promise((resolve) => {
|
|
|
|
chrome.tabs.get(
|
|
|
|
tabId,
|
|
|
|
(result) => { resolve(chrome.runtime.lastError ? null : result); }
|
|
|
|
);
|
|
|
|
});
|
|
|
|
if (tab !== null) {
|
2020-08-09 17:11:41 +00:00
|
|
|
const url = await this._getTabUrl(tabId);
|
|
|
|
const isValidTab = (url !== null && url.startsWith(baseUrl));
|
2020-07-19 00:30:10 +00:00
|
|
|
if (isValidTab) {
|
|
|
|
return {tab, created: false};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
this._searchPopupTabId = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
// chrome.windows not supported (e.g. on Firefox mobile)
|
|
|
|
if (!isObject(chrome.windows)) {
|
|
|
|
throw new Error('Window creation not supported');
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create a new window
|
|
|
|
const options = this.getOptions({current: true});
|
|
|
|
const {popupWidth, popupHeight} = options.general;
|
|
|
|
const popupWindow = await new Promise((resolve, reject) => {
|
|
|
|
chrome.windows.create(
|
|
|
|
{
|
2020-08-09 17:11:41 +00:00
|
|
|
url: baseUrl,
|
2020-07-19 00:30:10 +00:00
|
|
|
width: popupWidth,
|
|
|
|
height: popupHeight,
|
|
|
|
type: 'popup'
|
|
|
|
},
|
|
|
|
(result) => {
|
|
|
|
const error = chrome.runtime.lastError;
|
|
|
|
if (error) {
|
|
|
|
reject(new Error(error.message));
|
|
|
|
} else {
|
|
|
|
resolve(result);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
const {tabs} = popupWindow;
|
|
|
|
if (tabs.length === 0) {
|
|
|
|
throw new Error('Created window did not contain a tab');
|
|
|
|
}
|
|
|
|
|
|
|
|
const tab = tabs[0];
|
|
|
|
await this._waitUntilTabFrameIsReady(tab.id, 0, 2000);
|
|
|
|
|
2020-08-09 17:11:41 +00:00
|
|
|
await this._sendMessageTab(
|
|
|
|
tab.id,
|
|
|
|
{action: 'setMode', params: {mode: 'popup'}},
|
|
|
|
{frameId: 0}
|
|
|
|
);
|
|
|
|
|
2020-07-19 00:30:10 +00:00
|
|
|
this._searchPopupTabId = tab.id;
|
|
|
|
return {tab, created: true};
|
|
|
|
}
|
|
|
|
|
|
|
|
_updateSearchQuery(tabId, text, animate) {
|
2020-08-09 17:11:41 +00:00
|
|
|
return this._sendMessageTab(
|
|
|
|
tabId,
|
|
|
|
{action: 'updateSearchQuery', params: {text, animate}},
|
|
|
|
{frameId: 0}
|
|
|
|
);
|
2020-07-11 19:20:00 +00:00
|
|
|
}
|
|
|
|
|
2020-06-28 18:59:01 +00:00
|
|
|
_sendMessageAllTabs(action, params={}) {
|
|
|
|
const callback = () => this._checkLastError(chrome.runtime.lastError);
|
|
|
|
chrome.tabs.query({}, (tabs) => {
|
|
|
|
for (const tab of tabs) {
|
|
|
|
chrome.tabs.sendMessage(tab.id, {action, params}, callback);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
_applyOptions(source) {
|
2020-07-11 19:20:51 +00:00
|
|
|
const options = this.getOptions({current: true});
|
2020-06-28 18:59:01 +00:00
|
|
|
this._updateBadge();
|
|
|
|
|
|
|
|
this._anki.setServer(options.anki.server);
|
|
|
|
this._anki.setEnabled(options.anki.enable);
|
|
|
|
|
|
|
|
if (options.parsing.enableMecabParser) {
|
|
|
|
this._mecab.startListener();
|
|
|
|
} else {
|
|
|
|
this._mecab.stopListener();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (options.general.enableClipboardPopups) {
|
|
|
|
this._clipboardMonitor.start();
|
|
|
|
} else {
|
|
|
|
this._clipboardMonitor.stop();
|
|
|
|
}
|
|
|
|
|
|
|
|
this._sendMessageAllTabs('optionsUpdated', {source});
|
|
|
|
}
|
|
|
|
|
|
|
|
_getProfile(optionsContext, useSchema=false) {
|
|
|
|
const options = this.getFullOptions(useSchema);
|
|
|
|
const profiles = options.profiles;
|
2020-07-11 19:20:51 +00:00
|
|
|
if (optionsContext.current) {
|
|
|
|
return profiles[options.profileCurrent];
|
|
|
|
}
|
2020-06-28 18:59:01 +00:00
|
|
|
if (typeof optionsContext.index === 'number') {
|
|
|
|
return profiles[optionsContext.index];
|
|
|
|
}
|
|
|
|
const profile = this._getProfileFromContext(options, optionsContext);
|
2020-07-11 19:20:51 +00:00
|
|
|
return profile !== null ? profile : profiles[options.profileCurrent];
|
2020-06-28 18:59:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
_getProfileFromContext(options, optionsContext) {
|
2020-09-04 21:44:00 +00:00
|
|
|
optionsContext = this._profileConditionsUtil.normalizeContext(optionsContext);
|
|
|
|
|
|
|
|
let index = 0;
|
2020-06-28 18:59:01 +00:00
|
|
|
for (const profile of options.profiles) {
|
|
|
|
const conditionGroups = profile.conditionGroups;
|
|
|
|
|
2020-09-04 21:44:00 +00:00
|
|
|
let schema;
|
|
|
|
if (index < this._profileConditionsSchemaCache.length) {
|
|
|
|
schema = this._profileConditionsSchemaCache[index];
|
|
|
|
} else {
|
|
|
|
schema = this._profileConditionsUtil.createSchema(conditionGroups);
|
|
|
|
this._profileConditionsSchemaCache.push(schema);
|
|
|
|
}
|
2020-06-28 18:59:01 +00:00
|
|
|
|
2020-09-04 21:44:00 +00:00
|
|
|
if (conditionGroups.length > 0 && this._optionsSchemaValidator.isValid(optionsContext, schema)) {
|
|
|
|
return profile;
|
2020-06-28 18:59:01 +00:00
|
|
|
}
|
2020-09-04 21:44:00 +00:00
|
|
|
++index;
|
2020-06-28 18:59:01 +00:00
|
|
|
}
|
|
|
|
|
2020-09-04 21:44:00 +00:00
|
|
|
return null;
|
2020-06-28 18:59:01 +00:00
|
|
|
}
|
|
|
|
|
2020-09-04 21:44:00 +00:00
|
|
|
_clearProfileConditionsSchemaCache() {
|
|
|
|
this._profileConditionsSchemaCache = [];
|
|
|
|
this._optionsSchemaValidator.clearCache();
|
2020-06-28 18:59:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
_checkLastError() {
|
|
|
|
// NOP
|
|
|
|
}
|
|
|
|
|
|
|
|
_runCommand(command, params) {
|
|
|
|
const handler = this._commandHandlers.get(command);
|
|
|
|
if (typeof handler !== 'function') { return false; }
|
|
|
|
|
|
|
|
handler(params);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
async _importDictionary(archiveSource, onProgress, details) {
|
2020-07-20 02:05:37 +00:00
|
|
|
return await this._dictionaryImporter.importDictionary(this._dictionaryDatabase, archiveSource, onProgress, details);
|
2020-06-28 18:59:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
async _textParseScanning(text, options) {
|
|
|
|
const results = [];
|
|
|
|
while (text.length > 0) {
|
|
|
|
const term = [];
|
|
|
|
const [definitions, sourceLength] = await this._translator.findTerms(
|
|
|
|
'simple',
|
|
|
|
text.substring(0, options.scanning.length),
|
|
|
|
{},
|
|
|
|
options
|
|
|
|
);
|
|
|
|
if (definitions.length > 0 && sourceLength > 0) {
|
|
|
|
const {expression, reading} = definitions[0];
|
|
|
|
const source = text.substring(0, sourceLength);
|
|
|
|
for (const {text: text2, furigana} of jp.distributeFuriganaInflected(expression, reading, source)) {
|
|
|
|
const reading2 = jp.convertReading(text2, furigana, options.parsing.readingMode);
|
|
|
|
term.push({text: text2, reading: reading2});
|
|
|
|
}
|
|
|
|
text = text.substring(source.length);
|
|
|
|
} else {
|
|
|
|
const reading = jp.convertReading(text[0], '', options.parsing.readingMode);
|
|
|
|
term.push({text: text[0], reading});
|
|
|
|
text = text.substring(1);
|
|
|
|
}
|
|
|
|
results.push(term);
|
|
|
|
}
|
|
|
|
return results;
|
|
|
|
}
|
|
|
|
|
|
|
|
async _textParseMecab(text, options) {
|
|
|
|
const results = [];
|
|
|
|
const rawResults = await this._mecab.parseText(text);
|
|
|
|
for (const [mecabName, parsedLines] of Object.entries(rawResults)) {
|
|
|
|
const result = [];
|
|
|
|
for (const parsedLine of parsedLines) {
|
|
|
|
for (const {expression, reading, source} of parsedLine) {
|
|
|
|
const term = [];
|
|
|
|
for (const {text: text2, furigana} of jp.distributeFuriganaInflected(
|
|
|
|
expression.length > 0 ? expression : source,
|
|
|
|
jp.convertKatakanaToHiragana(reading),
|
|
|
|
source
|
|
|
|
)) {
|
|
|
|
const reading2 = jp.convertReading(text2, furigana, options.parsing.readingMode);
|
|
|
|
term.push({text: text2, reading: reading2});
|
|
|
|
}
|
|
|
|
result.push(term);
|
|
|
|
}
|
|
|
|
result.push([{text: '\n', reading: ''}]);
|
|
|
|
}
|
|
|
|
results.push([mecabName, result]);
|
|
|
|
}
|
|
|
|
return results;
|
|
|
|
}
|
|
|
|
|
2020-05-02 16:57:13 +00:00
|
|
|
_createActionListenerPort(port, sender, handlers) {
|
|
|
|
let hasStarted = false;
|
2020-05-31 22:17:12 +00:00
|
|
|
let messageString = '';
|
2020-05-02 16:57:13 +00:00
|
|
|
|
2020-05-06 23:28:26 +00:00
|
|
|
const onProgress = (...data) => {
|
2020-05-02 16:57:13 +00:00
|
|
|
try {
|
|
|
|
if (port === null) { return; }
|
|
|
|
port.postMessage({type: 'progress', data});
|
|
|
|
} catch (e) {
|
|
|
|
// NOP
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2020-05-31 22:17:12 +00:00
|
|
|
const onMessage = (message) => {
|
2020-05-02 16:57:13 +00:00
|
|
|
if (hasStarted) { return; }
|
|
|
|
|
|
|
|
try {
|
2020-05-31 22:17:12 +00:00
|
|
|
const {action, data} = message;
|
|
|
|
switch (action) {
|
|
|
|
case 'fragment':
|
|
|
|
messageString += data;
|
|
|
|
break;
|
|
|
|
case 'invoke':
|
|
|
|
{
|
|
|
|
hasStarted = true;
|
|
|
|
port.onMessage.removeListener(onMessage);
|
|
|
|
|
|
|
|
const messageData = JSON.parse(messageString);
|
|
|
|
messageString = null;
|
|
|
|
onMessageComplete(messageData);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} catch (e) {
|
|
|
|
cleanup(e);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const onMessageComplete = async (message) => {
|
|
|
|
try {
|
|
|
|
const {action, params} = message;
|
2020-05-02 16:57:13 +00:00
|
|
|
port.postMessage({type: 'ack'});
|
|
|
|
|
|
|
|
const messageHandler = handlers.get(action);
|
|
|
|
if (typeof messageHandler === 'undefined') {
|
|
|
|
throw new Error('Invalid action');
|
|
|
|
}
|
2020-05-07 23:37:25 +00:00
|
|
|
const {handler, async, contentScript} = messageHandler;
|
|
|
|
|
|
|
|
if (!contentScript) {
|
|
|
|
this._validatePrivilegedMessageSender(sender);
|
|
|
|
}
|
2020-05-02 16:57:13 +00:00
|
|
|
|
|
|
|
const promiseOrResult = handler(params, sender, onProgress);
|
|
|
|
const result = async ? await promiseOrResult : promiseOrResult;
|
|
|
|
port.postMessage({type: 'complete', data: result});
|
|
|
|
} catch (e) {
|
2020-05-31 22:17:12 +00:00
|
|
|
cleanup(e);
|
2020-05-02 16:57:13 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2020-05-31 22:17:12 +00:00
|
|
|
const onDisconnect = () => {
|
|
|
|
cleanup(null);
|
|
|
|
};
|
|
|
|
|
|
|
|
const cleanup = (error) => {
|
2020-05-02 16:57:13 +00:00
|
|
|
if (port === null) { return; }
|
2020-05-31 22:17:12 +00:00
|
|
|
if (error !== null) {
|
|
|
|
port.postMessage({type: 'error', data: errorToJson(error)});
|
|
|
|
}
|
2020-05-02 16:57:13 +00:00
|
|
|
if (!hasStarted) {
|
|
|
|
port.onMessage.removeListener(onMessage);
|
|
|
|
}
|
2020-05-31 22:17:12 +00:00
|
|
|
port.onDisconnect.removeListener(onDisconnect);
|
2020-05-02 16:57:13 +00:00
|
|
|
port = null;
|
|
|
|
handlers = null;
|
|
|
|
};
|
|
|
|
|
|
|
|
port.onMessage.addListener(onMessage);
|
2020-05-31 22:17:12 +00:00
|
|
|
port.onDisconnect.addListener(onDisconnect);
|
2020-05-02 16:57:13 +00:00
|
|
|
}
|
|
|
|
|
2020-04-26 20:55:25 +00:00
|
|
|
_getErrorLevelValue(errorLevel) {
|
|
|
|
switch (errorLevel) {
|
|
|
|
case 'info': return 0;
|
|
|
|
case 'debug': return 0;
|
|
|
|
case 'warn': return 1;
|
|
|
|
case 'error': return 2;
|
|
|
|
default: return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-06 23:32:28 +00:00
|
|
|
_getModifySettingObject(target) {
|
|
|
|
const scope = target.scope;
|
|
|
|
switch (scope) {
|
|
|
|
case 'profile':
|
|
|
|
if (!isObject(target.optionsContext)) { throw new Error('Invalid optionsContext'); }
|
|
|
|
return this.getOptions(target.optionsContext, true);
|
|
|
|
case 'global':
|
|
|
|
return this.getFullOptions(true);
|
|
|
|
default:
|
|
|
|
throw new Error(`Invalid scope: ${scope}`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-24 17:50:34 +00:00
|
|
|
_getSetting(target) {
|
|
|
|
const options = this._getModifySettingObject(target);
|
|
|
|
const accessor = new ObjectPropertyAccessor(options);
|
|
|
|
const {path} = target;
|
|
|
|
if (typeof path !== 'string') { throw new Error('Invalid path'); }
|
|
|
|
return accessor.get(ObjectPropertyAccessor.getPathArray(path));
|
|
|
|
}
|
|
|
|
|
|
|
|
_modifySetting(target) {
|
2020-05-06 23:32:28 +00:00
|
|
|
const options = this._getModifySettingObject(target);
|
|
|
|
const accessor = new ObjectPropertyAccessor(options);
|
|
|
|
const action = target.action;
|
|
|
|
switch (action) {
|
|
|
|
case 'set':
|
2020-05-24 17:50:34 +00:00
|
|
|
{
|
|
|
|
const {path, value} = target;
|
|
|
|
if (typeof path !== 'string') { throw new Error('Invalid path'); }
|
|
|
|
const pathArray = ObjectPropertyAccessor.getPathArray(path);
|
|
|
|
accessor.set(pathArray, value);
|
|
|
|
return accessor.get(pathArray);
|
|
|
|
}
|
2020-05-06 23:32:28 +00:00
|
|
|
case 'delete':
|
2020-05-24 17:50:34 +00:00
|
|
|
{
|
|
|
|
const {path} = target;
|
|
|
|
if (typeof path !== 'string') { throw new Error('Invalid path'); }
|
|
|
|
accessor.delete(ObjectPropertyAccessor.getPathArray(path));
|
|
|
|
return true;
|
|
|
|
}
|
2020-05-06 23:32:28 +00:00
|
|
|
case 'swap':
|
2020-05-24 17:50:34 +00:00
|
|
|
{
|
|
|
|
const {path1, path2} = target;
|
|
|
|
if (typeof path1 !== 'string') { throw new Error('Invalid path1'); }
|
|
|
|
if (typeof path2 !== 'string') { throw new Error('Invalid path2'); }
|
|
|
|
accessor.swap(ObjectPropertyAccessor.getPathArray(path1), ObjectPropertyAccessor.getPathArray(path2));
|
|
|
|
return true;
|
|
|
|
}
|
2020-05-06 23:32:28 +00:00
|
|
|
case 'splice':
|
2020-05-24 17:50:34 +00:00
|
|
|
{
|
|
|
|
const {path, start, deleteCount, items} = target;
|
|
|
|
if (typeof path !== 'string') { throw new Error('Invalid path'); }
|
|
|
|
if (typeof start !== 'number' || Math.floor(start) !== start) { throw new Error('Invalid start'); }
|
|
|
|
if (typeof deleteCount !== 'number' || Math.floor(deleteCount) !== deleteCount) { throw new Error('Invalid deleteCount'); }
|
|
|
|
if (!Array.isArray(items)) { throw new Error('Invalid items'); }
|
|
|
|
const array = accessor.get(ObjectPropertyAccessor.getPathArray(path));
|
|
|
|
if (!Array.isArray(array)) { throw new Error('Invalid target type'); }
|
|
|
|
return array.splice(start, deleteCount, ...items);
|
|
|
|
}
|
2020-05-06 23:32:28 +00:00
|
|
|
default:
|
|
|
|
throw new Error(`Unknown action: ${action}`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-11 19:21:43 +00:00
|
|
|
_validatePrivilegedMessageSender(sender) {
|
|
|
|
const url = sender.url;
|
|
|
|
if (!(typeof url === 'string' && yomichan.isExtensionUrl(url))) {
|
|
|
|
throw new Error('Invalid message sender');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-12 00:46:11 +00:00
|
|
|
_getBrowserIconTitle() {
|
|
|
|
return (
|
2020-04-17 23:19:38 +00:00
|
|
|
isObject(chrome.browserAction) &&
|
2020-04-12 00:46:11 +00:00
|
|
|
typeof chrome.browserAction.getTitle === 'function' ?
|
2020-04-17 23:19:38 +00:00
|
|
|
new Promise((resolve) => chrome.browserAction.getTitle({}, resolve)) :
|
|
|
|
Promise.resolve('')
|
2020-04-12 00:46:11 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
_updateBadge() {
|
|
|
|
let title = this._defaultBrowserActionTitle;
|
2020-04-17 23:19:38 +00:00
|
|
|
if (title === null || !isObject(chrome.browserAction)) {
|
2020-04-12 00:46:11 +00:00
|
|
|
// Not ready or invalid
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
let text = '';
|
|
|
|
let color = null;
|
|
|
|
let status = null;
|
|
|
|
|
2020-04-26 20:55:25 +00:00
|
|
|
if (this._logErrorLevel !== null) {
|
|
|
|
switch (this._logErrorLevel) {
|
|
|
|
case 'error':
|
|
|
|
text = '!!';
|
|
|
|
color = '#f04e4e';
|
|
|
|
status = 'Error';
|
|
|
|
break;
|
|
|
|
default: // 'warn'
|
|
|
|
text = '!';
|
|
|
|
color = '#f0ad4e';
|
|
|
|
status = 'Warning';
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} else if (!this._isPrepared) {
|
2020-04-19 01:15:15 +00:00
|
|
|
if (this._prepareError) {
|
2020-04-12 00:58:52 +00:00
|
|
|
text = '!!';
|
|
|
|
color = '#f04e4e';
|
|
|
|
status = 'Error';
|
|
|
|
} else if (this._badgePrepareDelayTimer === null) {
|
2020-04-12 00:53:18 +00:00
|
|
|
text = '...';
|
|
|
|
color = '#f0ad4e';
|
|
|
|
status = 'Loading';
|
|
|
|
}
|
2020-04-12 00:46:11 +00:00
|
|
|
} else if (!this._anyOptionsMatches((options) => options.general.enable)) {
|
|
|
|
text = 'off';
|
|
|
|
color = '#555555';
|
|
|
|
status = 'Disabled';
|
|
|
|
} else if (!this._anyOptionsMatches((options) => this._isAnyDictionaryEnabled(options))) {
|
|
|
|
text = '!';
|
|
|
|
color = '#f0ad4e';
|
|
|
|
status = 'No dictionaries installed';
|
|
|
|
}
|
|
|
|
|
|
|
|
if (color !== null && typeof chrome.browserAction.setBadgeBackgroundColor === 'function') {
|
|
|
|
chrome.browserAction.setBadgeBackgroundColor({color});
|
|
|
|
}
|
|
|
|
if (text !== null && typeof chrome.browserAction.setBadgeText === 'function') {
|
|
|
|
chrome.browserAction.setBadgeText({text});
|
|
|
|
}
|
|
|
|
if (typeof chrome.browserAction.setTitle === 'function') {
|
|
|
|
if (status !== null) {
|
|
|
|
title = `${title} - ${status}`;
|
|
|
|
}
|
|
|
|
chrome.browserAction.setTitle({title});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
_isAnyDictionaryEnabled(options) {
|
|
|
|
for (const {enabled} of Object.values(options.dictionaries)) {
|
|
|
|
if (enabled) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
_anyOptionsMatches(predicate) {
|
2020-06-28 18:59:01 +00:00
|
|
|
for (const {options} of this._options.profiles) {
|
2020-04-12 00:46:11 +00:00
|
|
|
const value = predicate(options);
|
|
|
|
if (value) { return value; }
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2020-03-07 20:20:45 +00:00
|
|
|
async _renderTemplate(template, data) {
|
2020-06-16 00:11:54 +00:00
|
|
|
return await this._templateRenderer.render(template, data);
|
2020-03-07 20:20:45 +00:00
|
|
|
}
|
|
|
|
|
2020-04-23 16:58:31 +00:00
|
|
|
_getTemplates(options) {
|
|
|
|
const templates = options.anki.fieldTemplates;
|
2020-06-28 18:59:01 +00:00
|
|
|
return typeof templates === 'string' ? templates : this._defaultAnkiFieldTemplates;
|
2020-04-23 16:58:31 +00:00
|
|
|
}
|
|
|
|
|
2020-08-09 17:11:41 +00:00
|
|
|
async _getTabUrl(tabId) {
|
|
|
|
try {
|
|
|
|
const {url} = await this._sendMessageTab(
|
|
|
|
tabId,
|
|
|
|
{action: 'getUrl', params: {}},
|
|
|
|
{frameId: 0}
|
|
|
|
);
|
|
|
|
if (typeof url === 'string') {
|
|
|
|
return url;
|
|
|
|
}
|
|
|
|
} catch (e) {
|
|
|
|
// NOP
|
|
|
|
}
|
|
|
|
return null;
|
2019-12-10 02:41:24 +00:00
|
|
|
}
|
|
|
|
|
2020-06-28 18:59:01 +00:00
|
|
|
async _findTab(timeout, checkUrl) {
|
2019-12-10 02:41:24 +00:00
|
|
|
// This function works around the need to have the "tabs" permission to access tab.url.
|
|
|
|
const tabs = await new Promise((resolve) => chrome.tabs.query({}, resolve));
|
2020-06-28 18:39:43 +00:00
|
|
|
const {promise: matchPromise, resolve: matchPromiseResolve} = deferPromise();
|
2019-12-10 02:41:24 +00:00
|
|
|
|
|
|
|
const checkTabUrl = ({tab, url}) => {
|
|
|
|
if (checkUrl(url, tab)) {
|
|
|
|
matchPromiseResolve(tab);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const promises = [];
|
|
|
|
for (const tab of tabs) {
|
2020-06-28 18:59:01 +00:00
|
|
|
const promise = this._getTabUrl(tab);
|
2019-12-10 02:41:24 +00:00
|
|
|
promise.then(checkTabUrl);
|
|
|
|
promises.push(promise);
|
|
|
|
}
|
|
|
|
|
|
|
|
const racePromises = [
|
|
|
|
matchPromise,
|
|
|
|
Promise.all(promises).then(() => null)
|
|
|
|
];
|
|
|
|
if (typeof timeout === 'number') {
|
|
|
|
racePromises.push(new Promise((resolve) => setTimeout(() => resolve(null), timeout)));
|
|
|
|
}
|
|
|
|
|
|
|
|
return await Promise.race(racePromises);
|
|
|
|
}
|
|
|
|
|
2020-06-28 18:59:01 +00:00
|
|
|
async _focusTab(tab) {
|
2019-12-10 02:41:24 +00:00
|
|
|
await new Promise((resolve, reject) => {
|
|
|
|
chrome.tabs.update(tab.id, {active: true}, () => {
|
|
|
|
const e = chrome.runtime.lastError;
|
2020-02-16 00:48:02 +00:00
|
|
|
if (e) {
|
2020-02-23 16:59:57 +00:00
|
|
|
reject(new Error(e.message));
|
2020-02-16 00:48:02 +00:00
|
|
|
} else {
|
|
|
|
resolve();
|
|
|
|
}
|
2019-12-10 02:41:24 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
if (!(typeof chrome.windows === 'object' && chrome.windows !== null)) {
|
|
|
|
// Windows not supported (e.g. on Firefox mobile)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
2020-02-01 19:39:26 +00:00
|
|
|
const tabWindow = await new Promise((resolve, reject) => {
|
2020-02-17 20:21:30 +00:00
|
|
|
chrome.windows.get(tab.windowId, {}, (value) => {
|
2019-12-10 02:41:24 +00:00
|
|
|
const e = chrome.runtime.lastError;
|
2020-02-16 00:48:02 +00:00
|
|
|
if (e) {
|
2020-02-23 16:59:57 +00:00
|
|
|
reject(new Error(e.message));
|
2020-02-16 00:48:02 +00:00
|
|
|
} else {
|
2020-02-17 20:21:30 +00:00
|
|
|
resolve(value);
|
2020-02-16 00:48:02 +00:00
|
|
|
}
|
2019-12-10 02:41:24 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
if (!tabWindow.focused) {
|
|
|
|
await new Promise((resolve, reject) => {
|
|
|
|
chrome.windows.update(tab.windowId, {focused: true}, () => {
|
|
|
|
const e = chrome.runtime.lastError;
|
2020-02-16 00:48:02 +00:00
|
|
|
if (e) {
|
2020-02-23 16:59:57 +00:00
|
|
|
reject(new Error(e.message));
|
2020-02-16 00:48:02 +00:00
|
|
|
} else {
|
|
|
|
resolve();
|
|
|
|
}
|
2019-12-10 02:41:24 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
} catch (e) {
|
|
|
|
// Edge throws exception for no reason here.
|
|
|
|
}
|
|
|
|
}
|
2020-07-18 18:18:10 +00:00
|
|
|
|
|
|
|
_waitUntilTabFrameIsReady(tabId, frameId, timeout=null) {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
let timer = null;
|
|
|
|
let onMessage = (message, sender) => {
|
|
|
|
if (
|
|
|
|
!sender.tab ||
|
|
|
|
sender.tab.id !== tabId ||
|
|
|
|
sender.frameId !== frameId ||
|
|
|
|
!isObject(message) ||
|
2020-07-18 21:11:38 +00:00
|
|
|
message.action !== 'yomichanReady'
|
2020-07-18 18:18:10 +00:00
|
|
|
) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
cleanup();
|
|
|
|
resolve();
|
|
|
|
};
|
|
|
|
const cleanup = () => {
|
|
|
|
if (timer !== null) {
|
|
|
|
clearTimeout(timer);
|
|
|
|
timer = null;
|
|
|
|
}
|
|
|
|
if (onMessage !== null) {
|
|
|
|
chrome.runtime.onMessage.removeListener(onMessage);
|
|
|
|
onMessage = null;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
chrome.runtime.onMessage.addListener(onMessage);
|
|
|
|
|
|
|
|
chrome.tabs.sendMessage(tabId, {action: 'isReady'}, {frameId}, (response) => {
|
|
|
|
const error = chrome.runtime.lastError;
|
|
|
|
if (error) { return; }
|
|
|
|
|
|
|
|
try {
|
|
|
|
const value = yomichan.getMessageResponseResult(response);
|
|
|
|
if (!value) { return; }
|
|
|
|
|
|
|
|
cleanup();
|
|
|
|
resolve();
|
|
|
|
} catch (e) {
|
|
|
|
// NOP
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
if (timeout !== null) {
|
|
|
|
timer = setTimeout(() => {
|
|
|
|
timer = null;
|
|
|
|
cleanup();
|
|
|
|
reject(new Error('Timeout'));
|
|
|
|
}, timeout);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2020-08-02 17:30:55 +00:00
|
|
|
|
|
|
|
async _fetchAsset(url, json=false) {
|
|
|
|
const response = await fetch(chrome.runtime.getURL(url), {
|
|
|
|
method: 'GET',
|
|
|
|
mode: 'no-cors',
|
|
|
|
cache: 'default',
|
|
|
|
credentials: 'omit',
|
|
|
|
redirect: 'follow',
|
|
|
|
referrerPolicy: 'no-referrer'
|
|
|
|
});
|
|
|
|
if (!response.ok) {
|
|
|
|
throw new Error(`Failed to fetch ${url}: ${response.status}`);
|
|
|
|
}
|
|
|
|
return await (json ? response.json() : response.text());
|
|
|
|
}
|
2020-08-09 17:11:41 +00:00
|
|
|
|
|
|
|
_sendMessageTab(...args) {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
const callback = (response) => {
|
|
|
|
try {
|
|
|
|
resolve(yomichan.getMessageResponseResult(response));
|
|
|
|
} catch (error) {
|
|
|
|
reject(error);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
chrome.tabs.sendMessage(...args, callback);
|
|
|
|
});
|
|
|
|
}
|
2017-08-14 04:11:10 +00:00
|
|
|
}
|