2017-07-20 21:21:22 -07:00
|
|
|
/*
|
2020-04-10 11:06:55 -07:00
|
|
|
* Copyright (C) 2016-2020 Yomichan Authors
|
2017-07-20 21:21:22 -07: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 12:00:31 -05:00
|
|
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
2017-07-20 21:21:22 -07:00
|
|
|
*/
|
|
|
|
|
|
|
|
|
2019-12-14 16:40:05 -05:00
|
|
|
function apiOptionsSchemaGet() {
|
|
|
|
return _apiInvoke('optionsSchemaGet');
|
|
|
|
}
|
|
|
|
|
2019-09-07 13:58:19 -04:00
|
|
|
function apiOptionsGet(optionsContext) {
|
2019-11-26 17:23:55 -05:00
|
|
|
return _apiInvoke('optionsGet', {optionsContext});
|
2017-07-20 21:21:22 -07:00
|
|
|
}
|
|
|
|
|
2019-12-09 21:00:49 -05:00
|
|
|
function apiOptionsGetFull() {
|
|
|
|
return _apiInvoke('optionsGetFull');
|
|
|
|
}
|
|
|
|
|
|
|
|
function apiOptionsSave(source) {
|
|
|
|
return _apiInvoke('optionsSave', {source});
|
|
|
|
}
|
|
|
|
|
2019-11-04 20:43:40 -05:00
|
|
|
function apiTermsFind(text, details, optionsContext) {
|
2019-11-26 17:23:55 -05:00
|
|
|
return _apiInvoke('termsFind', {text, details, optionsContext});
|
2017-07-20 21:21:22 -07:00
|
|
|
}
|
|
|
|
|
2019-10-30 03:58:24 +02:00
|
|
|
function apiTextParse(text, optionsContext) {
|
2019-11-26 17:23:55 -05:00
|
|
|
return _apiInvoke('textParse', {text, optionsContext});
|
2019-10-30 03:58:24 +02:00
|
|
|
}
|
|
|
|
|
2019-09-07 13:58:19 -04:00
|
|
|
function apiKanjiFind(text, optionsContext) {
|
2019-11-26 17:23:55 -05:00
|
|
|
return _apiInvoke('kanjiFind', {text, optionsContext});
|
2017-07-20 21:21:22 -07:00
|
|
|
}
|
|
|
|
|
2020-03-15 17:13:00 -04:00
|
|
|
function apiDefinitionAdd(definition, mode, context, details, optionsContext) {
|
|
|
|
return _apiInvoke('definitionAdd', {definition, mode, context, details, optionsContext});
|
2017-07-20 21:21:22 -07:00
|
|
|
}
|
|
|
|
|
2020-03-15 17:13:00 -04:00
|
|
|
function apiDefinitionsAddable(definitions, modes, context, optionsContext) {
|
|
|
|
return _apiInvoke('definitionsAddable', {definitions, modes, context, optionsContext});
|
2017-07-20 21:21:22 -07:00
|
|
|
}
|
|
|
|
|
2017-07-22 23:19:38 -07:00
|
|
|
function apiNoteView(noteId) {
|
2019-11-26 17:23:55 -05:00
|
|
|
return _apiInvoke('noteView', {noteId});
|
2017-07-20 21:21:22 -07:00
|
|
|
}
|
2017-08-14 21:43:09 -07:00
|
|
|
|
2020-02-14 20:11:40 -05:00
|
|
|
function apiTemplateRender(template, data) {
|
|
|
|
return _apiInvoke('templateRender', {data, template});
|
|
|
|
}
|
|
|
|
|
2020-04-10 13:44:31 -04:00
|
|
|
function apiAudioGetUri(definition, source, details) {
|
|
|
|
return _apiInvoke('audioGetUri', {definition, source, details});
|
2019-08-17 15:42:36 -07:00
|
|
|
}
|
|
|
|
|
2019-10-19 22:30:16 -04:00
|
|
|
function apiCommandExec(command, params) {
|
2019-11-26 17:23:55 -05:00
|
|
|
return _apiInvoke('commandExec', {command, params});
|
2017-08-14 21:43:09 -07:00
|
|
|
}
|
2017-08-15 21:36:30 -07:00
|
|
|
|
2019-08-15 19:39:58 -04:00
|
|
|
function apiScreenshotGet(options) {
|
2019-11-26 17:23:55 -05:00
|
|
|
return _apiInvoke('screenshotGet', {options});
|
2019-08-15 19:39:58 -04:00
|
|
|
}
|
|
|
|
|
2020-05-06 19:27:21 -04:00
|
|
|
function apiSendMessageToFrame(frameId, action, params) {
|
|
|
|
return _apiInvoke('sendMessageToFrame', {frameId, action, params});
|
|
|
|
}
|
|
|
|
|
2020-04-10 20:00:18 -04:00
|
|
|
function apiBroadcastTab(action, params) {
|
|
|
|
return _apiInvoke('broadcastTab', {action, params});
|
2017-08-15 21:36:30 -07:00
|
|
|
}
|
2019-08-18 20:51:19 -04:00
|
|
|
|
|
|
|
function apiFrameInformationGet() {
|
2019-11-26 17:23:55 -05:00
|
|
|
return _apiInvoke('frameInformationGet');
|
2019-08-18 20:51:19 -04:00
|
|
|
}
|
2019-10-13 11:46:27 -04:00
|
|
|
|
2020-02-16 13:13:04 -05:00
|
|
|
function apiInjectStylesheet(type, value) {
|
|
|
|
return _apiInvoke('injectStylesheet', {type, value});
|
2019-10-13 11:46:27 -04:00
|
|
|
}
|
2019-10-13 17:20:55 -04:00
|
|
|
|
|
|
|
function apiGetEnvironmentInfo() {
|
2019-11-26 17:23:55 -05:00
|
|
|
return _apiInvoke('getEnvironmentInfo');
|
2019-10-13 17:20:55 -04:00
|
|
|
}
|
2019-10-27 15:46:27 +02:00
|
|
|
|
|
|
|
function apiClipboardGet() {
|
2019-11-26 17:23:55 -05:00
|
|
|
return _apiInvoke('clipboardGet');
|
|
|
|
}
|
|
|
|
|
2019-12-27 18:58:11 -05:00
|
|
|
function apiGetDisplayTemplatesHtml() {
|
|
|
|
return _apiInvoke('getDisplayTemplatesHtml');
|
|
|
|
}
|
|
|
|
|
2020-02-06 04:00:02 +02:00
|
|
|
function apiGetQueryParserTemplatesHtml() {
|
|
|
|
return _apiInvoke('getQueryParserTemplatesHtml');
|
|
|
|
}
|
|
|
|
|
2019-12-23 11:59:47 -05:00
|
|
|
function apiGetZoom() {
|
|
|
|
return _apiInvoke('getZoom');
|
|
|
|
}
|
|
|
|
|
2020-02-27 20:33:13 -05:00
|
|
|
function apiGetDefaultAnkiFieldTemplates() {
|
|
|
|
return _apiInvoke('getDefaultAnkiFieldTemplates');
|
|
|
|
}
|
|
|
|
|
2020-04-11 15:21:43 -04:00
|
|
|
function apiGetAnkiDeckNames() {
|
|
|
|
return _apiInvoke('getAnkiDeckNames');
|
|
|
|
}
|
|
|
|
|
|
|
|
function apiGetAnkiModelNames() {
|
|
|
|
return _apiInvoke('getAnkiModelNames');
|
|
|
|
}
|
|
|
|
|
|
|
|
function apiGetAnkiModelFieldNames(modelName) {
|
|
|
|
return _apiInvoke('getAnkiModelFieldNames', {modelName});
|
|
|
|
}
|
|
|
|
|
|
|
|
function apiGetDictionaryInfo() {
|
|
|
|
return _apiInvoke('getDictionaryInfo');
|
|
|
|
}
|
|
|
|
|
|
|
|
function apiGetDictionaryCounts(dictionaryNames, getTotal) {
|
|
|
|
return _apiInvoke('getDictionaryCounts', {dictionaryNames, getTotal});
|
|
|
|
}
|
|
|
|
|
|
|
|
function apiPurgeDatabase() {
|
|
|
|
return _apiInvoke('purgeDatabase');
|
|
|
|
}
|
|
|
|
|
2020-04-11 14:23:02 -04:00
|
|
|
function apiGetMedia(targets) {
|
|
|
|
return _apiInvoke('getMedia', {targets});
|
|
|
|
}
|
|
|
|
|
2020-04-26 16:55:25 -04:00
|
|
|
function apiLog(error, level, context) {
|
|
|
|
return _apiInvoke('log', {error, level, context});
|
|
|
|
}
|
|
|
|
|
|
|
|
function apiLogIndicatorClear() {
|
|
|
|
return _apiInvoke('logIndicatorClear');
|
|
|
|
}
|
|
|
|
|
2020-05-06 19:28:26 -04:00
|
|
|
function apiImportDictionaryArchive(archiveContent, details, onProgress) {
|
|
|
|
return _apiInvokeWithProgress('importDictionaryArchive', {archiveContent, details}, onProgress);
|
|
|
|
}
|
|
|
|
|
|
|
|
function apiDeleteDictionary(dictionaryName, onProgress) {
|
|
|
|
return _apiInvokeWithProgress('deleteDictionary', {dictionaryName}, onProgress);
|
|
|
|
}
|
|
|
|
|
2020-05-06 19:32:28 -04:00
|
|
|
function apiModifySettings(targets, source) {
|
|
|
|
return _apiInvoke('modifySettings', {targets, source});
|
|
|
|
}
|
|
|
|
|
2020-05-02 12:57:13 -04:00
|
|
|
function _apiCreateActionPort(timeout=5000) {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
let timer = null;
|
|
|
|
let portNameResolve;
|
|
|
|
let portNameReject;
|
|
|
|
const portNamePromise = new Promise((resolve2, reject2) => {
|
|
|
|
portNameResolve = resolve2;
|
|
|
|
portNameReject = reject2;
|
|
|
|
});
|
|
|
|
|
|
|
|
const onConnect = async (port) => {
|
|
|
|
try {
|
|
|
|
const portName = await portNamePromise;
|
|
|
|
if (port.name !== portName || timer === null) { return; }
|
|
|
|
} catch (e) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
clearTimeout(timer);
|
|
|
|
timer = null;
|
|
|
|
|
|
|
|
chrome.runtime.onConnect.removeListener(onConnect);
|
|
|
|
resolve(port);
|
|
|
|
};
|
|
|
|
|
|
|
|
const onError = (e) => {
|
|
|
|
if (timer !== null) {
|
|
|
|
clearTimeout(timer);
|
|
|
|
timer = null;
|
|
|
|
}
|
|
|
|
chrome.runtime.onConnect.removeListener(onConnect);
|
|
|
|
portNameReject(e);
|
|
|
|
reject(e);
|
|
|
|
};
|
|
|
|
|
|
|
|
timer = setTimeout(() => onError(new Error('Timeout')), timeout);
|
|
|
|
|
|
|
|
chrome.runtime.onConnect.addListener(onConnect);
|
|
|
|
_apiInvoke('createActionPort').then(portNameResolve, onError);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
function _apiInvokeWithProgress(action, params, onProgress, timeout=5000) {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
let timer = null;
|
|
|
|
let port = null;
|
|
|
|
|
|
|
|
if (typeof onProgress !== 'function') {
|
|
|
|
onProgress = () => {};
|
|
|
|
}
|
|
|
|
|
|
|
|
const onMessage = (message) => {
|
|
|
|
switch (message.type) {
|
|
|
|
case 'ack':
|
|
|
|
if (timer !== null) {
|
|
|
|
clearTimeout(timer);
|
|
|
|
timer = null;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'progress':
|
|
|
|
try {
|
2020-05-06 19:28:26 -04:00
|
|
|
onProgress(...message.data);
|
2020-05-02 12:57:13 -04:00
|
|
|
} catch (e) {
|
|
|
|
// NOP
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'complete':
|
|
|
|
cleanup();
|
|
|
|
resolve(message.data);
|
|
|
|
break;
|
|
|
|
case 'error':
|
|
|
|
cleanup();
|
|
|
|
reject(jsonToError(message.data));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const onDisconnect = () => {
|
|
|
|
cleanup();
|
|
|
|
reject(new Error('Disconnected'));
|
|
|
|
};
|
|
|
|
|
|
|
|
const cleanup = () => {
|
|
|
|
if (timer !== null) {
|
|
|
|
clearTimeout(timer);
|
|
|
|
timer = null;
|
|
|
|
}
|
|
|
|
if (port !== null) {
|
|
|
|
port.onMessage.removeListener(onMessage);
|
|
|
|
port.onDisconnect.removeListener(onDisconnect);
|
|
|
|
port.disconnect();
|
|
|
|
port = null;
|
|
|
|
}
|
|
|
|
onProgress = null;
|
|
|
|
};
|
|
|
|
|
|
|
|
timer = setTimeout(() => {
|
|
|
|
cleanup();
|
|
|
|
reject(new Error('Timeout'));
|
|
|
|
}, timeout);
|
|
|
|
|
|
|
|
(async () => {
|
|
|
|
try {
|
|
|
|
port = await _apiCreateActionPort(timeout);
|
|
|
|
port.onMessage.addListener(onMessage);
|
|
|
|
port.onDisconnect.addListener(onDisconnect);
|
|
|
|
port.postMessage({action, params});
|
|
|
|
} catch (e) {
|
|
|
|
cleanup();
|
|
|
|
reject(e);
|
|
|
|
} finally {
|
|
|
|
action = null;
|
|
|
|
params = null;
|
|
|
|
}
|
|
|
|
})();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2019-11-26 17:23:55 -05:00
|
|
|
function _apiInvoke(action, params={}) {
|
|
|
|
const data = {action, params};
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
try {
|
|
|
|
chrome.runtime.sendMessage(data, (response) => {
|
|
|
|
_apiCheckLastError(chrome.runtime.lastError);
|
|
|
|
if (response !== null && typeof response === 'object') {
|
|
|
|
if (typeof response.error !== 'undefined') {
|
|
|
|
reject(jsonToError(response.error));
|
|
|
|
} else {
|
|
|
|
resolve(response.result);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
const message = response === null ? 'Unexpected null response' : `Unexpected response of type ${typeof response}`;
|
|
|
|
reject(new Error(`${message} (${JSON.stringify(data)})`));
|
|
|
|
}
|
|
|
|
});
|
|
|
|
} catch (e) {
|
|
|
|
reject(e);
|
2019-12-20 13:44:33 -05:00
|
|
|
yomichan.triggerOrphaned(e);
|
2019-11-26 17:23:55 -05:00
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2019-11-25 14:35:53 -05:00
|
|
|
function _apiCheckLastError() {
|
2019-11-26 17:23:55 -05:00
|
|
|
// NOP
|
2019-10-27 15:46:27 +02:00
|
|
|
}
|
2020-04-26 16:55:25 -04:00
|
|
|
|
|
|
|
let _apiForwardLogsToBackendEnabled = false;
|
|
|
|
function apiForwardLogsToBackend() {
|
|
|
|
if (_apiForwardLogsToBackendEnabled) { return; }
|
|
|
|
_apiForwardLogsToBackendEnabled = true;
|
|
|
|
|
|
|
|
yomichan.on('log', async ({error, level, context}) => {
|
|
|
|
try {
|
|
|
|
await apiLog(errorToJson(error), level, context);
|
|
|
|
} catch (e) {
|
|
|
|
// NOP
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|