diff options
-rw-r--r-- | ext/js/background/offscreen-proxy.js | 2 | ||||
-rw-r--r-- | ext/js/background/offscreen.js | 38 | ||||
-rw-r--r-- | ext/js/core/api-map.js | 48 | ||||
-rw-r--r-- | types/ext/api-map.d.ts | 55 | ||||
-rw-r--r-- | types/ext/offscreen.d.ts | 116 |
5 files changed, 191 insertions, 68 deletions
diff --git a/ext/js/background/offscreen-proxy.js b/ext/js/background/offscreen-proxy.js index dfd342b4..99dc0741 100644 --- a/ext/js/background/offscreen-proxy.js +++ b/ext/js/background/offscreen-proxy.js @@ -74,7 +74,7 @@ export class OffscreenProxy { /** * @template {import('offscreen').MessageType} TMessageType * @param {import('offscreen').Message<TMessageType>} message - * @returns {Promise<import('offscreen').MessageReturn<TMessageType>>} + * @returns {Promise<import('offscreen').OffscreenApiReturn<TMessageType>>} */ sendMessagePromise(message) { return new Promise((resolve, reject) => { diff --git a/ext/js/background/offscreen.js b/ext/js/background/offscreen.js index d1cf3384..44b0af77 100644 --- a/ext/js/background/offscreen.js +++ b/ext/js/background/offscreen.js @@ -19,6 +19,7 @@ import * as wanakana from '../../lib/wanakana.js'; import {ClipboardReader} from '../comm/clipboard-reader.js'; import {invokeMessageHandler} from '../core.js'; +import {createApiMap, getApiMapHandler} from '../core/api-map.js'; import {ArrayBufferUtil} from '../data/sandbox/array-buffer-util.js'; import {DictionaryDatabase} from '../language/dictionary-database.js'; import {JapaneseUtil} from '../language/sandbox/japanese-util.js'; @@ -50,9 +51,10 @@ export class Offscreen { richContentPasteTargetSelector: '#clipboard-rich-content-paste-target' }); + /* eslint-disable no-multi-spaces */ - /** @type {import('offscreen').MessageHandlerMap} */ - this._messageHandlers = new Map(/** @type {import('offscreen').MessageHandlerMapInit} */ ([ + /** @type {import('offscreen').OffscreenApiMapInit} */ + const messageHandlersInit = [ ['clipboardGetTextOffscreen', this._getTextHandler.bind(this)], ['clipboardGetImageOffscreen', this._getImageHandler.bind(this)], ['clipboardSetBrowserOffscreen', this._setClipboardBrowser.bind(this)], @@ -65,8 +67,10 @@ export class Offscreen { ['findTermsOffscreen', this._findTermsHandler.bind(this)], ['getTermFrequenciesOffscreen', this._getTermFrequenciesHandler.bind(this)], ['clearDatabaseCachesOffscreen', this._clearDatabaseCachesHandler.bind(this)] - ])); - /* eslint-enable no-multi-spaces */ + ]; + + /** @type {import('offscreen').OffscreenApiMap} */ + this._messageHandlers = createApiMap(messageHandlersInit); /** @type {?Promise<void>} */ this._prepareDatabasePromise = null; @@ -77,22 +81,22 @@ export class Offscreen { chrome.runtime.onMessage.addListener(this._onMessage.bind(this)); } - /** @type {import('offscreen').MessageHandler<'clipboardGetTextOffscreen', true>} */ + /** @type {import('offscreen').OffscreenApiHandler<'clipboardGetTextOffscreen'>} */ async _getTextHandler({useRichText}) { return await this._clipboardReader.getText(useRichText); } - /** @type {import('offscreen').MessageHandler<'clipboardGetImageOffscreen', true>} */ + /** @type {import('offscreen').OffscreenApiHandler<'clipboardGetImageOffscreen'>} */ async _getImageHandler() { return await this._clipboardReader.getImage(); } - /** @type {import('offscreen').MessageHandler<'clipboardSetBrowserOffscreen', false>} */ + /** @type {import('offscreen').OffscreenApiHandler<'clipboardSetBrowserOffscreen'>} */ _setClipboardBrowser({value}) { this._clipboardReader.browser = value; } - /** @type {import('offscreen').MessageHandler<'databasePrepareOffscreen', true>} */ + /** @type {import('offscreen').OffscreenApiHandler<'databasePrepareOffscreen'>} */ _prepareDatabaseHandler() { if (this._prepareDatabasePromise !== null) { return this._prepareDatabasePromise; @@ -101,29 +105,29 @@ export class Offscreen { return this._prepareDatabasePromise; } - /** @type {import('offscreen').MessageHandler<'getDictionaryInfoOffscreen', true>} */ + /** @type {import('offscreen').OffscreenApiHandler<'getDictionaryInfoOffscreen'>} */ async _getDictionaryInfoHandler() { return await this._dictionaryDatabase.getDictionaryInfo(); } - /** @type {import('offscreen').MessageHandler<'databasePurgeOffscreen', true>} */ + /** @type {import('offscreen').OffscreenApiHandler<'databasePurgeOffscreen'>} */ async _purgeDatabaseHandler() { return await this._dictionaryDatabase.purge(); } - /** @type {import('offscreen').MessageHandler<'databaseGetMediaOffscreen', true>} */ + /** @type {import('offscreen').OffscreenApiHandler<'databaseGetMediaOffscreen'>} */ async _getMediaHandler({targets}) { const media = await this._dictionaryDatabase.getMedia(targets); const serializedMedia = media.map((m) => ({...m, content: ArrayBufferUtil.arrayBufferToBase64(m.content)})); return serializedMedia; } - /** @type {import('offscreen').MessageHandler<'translatorPrepareOffscreen', false>} */ + /** @type {import('offscreen').OffscreenApiHandler<'translatorPrepareOffscreen'>} */ _prepareTranslatorHandler({deinflectionReasons}) { this._translator.prepare(deinflectionReasons); } - /** @type {import('offscreen').MessageHandler<'findKanjiOffscreen', true>} */ + /** @type {import('offscreen').OffscreenApiHandler<'findKanjiOffscreen'>} */ async _findKanjiHandler({text, options}) { /** @type {import('translation').FindKanjiOptions} */ const modifiedOptions = { @@ -133,7 +137,7 @@ export class Offscreen { return await this._translator.findKanji(text, modifiedOptions); } - /** @type {import('offscreen').MessageHandler<'findTermsOffscreen', true>} */ + /** @type {import('offscreen').OffscreenApiHandler<'findTermsOffscreen'>} */ _findTermsHandler({mode, text, options}) { const enabledDictionaryMap = new Map(options.enabledDictionaryMap); const excludeDictionaryDefinitions = ( @@ -160,19 +164,19 @@ export class Offscreen { return this._translator.findTerms(mode, text, modifiedOptions); } - /** @type {import('offscreen').MessageHandler<'getTermFrequenciesOffscreen', true>} */ + /** @type {import('offscreen').OffscreenApiHandler<'getTermFrequenciesOffscreen'>} */ _getTermFrequenciesHandler({termReadingList, dictionaries}) { return this._translator.getTermFrequencies(termReadingList, dictionaries); } - /** @type {import('offscreen').MessageHandler<'clearDatabaseCachesOffscreen', false>} */ + /** @type {import('offscreen').OffscreenApiHandler<'clearDatabaseCachesOffscreen'>} */ _clearDatabaseCachesHandler() { this._translator.clearDatabaseCaches(); } /** @type {import('extension').ChromeRuntimeOnMessageCallback} */ _onMessage({action, params}, sender, callback) { - const messageHandler = this._messageHandlers.get(/** @type {import('offscreen').MessageType} */ (action)); + const messageHandler = getApiMapHandler(this._messageHandlers, action); if (typeof messageHandler === 'undefined') { return false; } return invokeMessageHandler(messageHandler, params, callback, sender); } diff --git a/ext/js/core/api-map.js b/ext/js/core/api-map.js new file mode 100644 index 00000000..eb4abeea --- /dev/null +++ b/ext/js/core/api-map.js @@ -0,0 +1,48 @@ +/* + * Copyright (C) 2023 Yomitan Authors + * + * 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 + * along with this program. If not, see <https://www.gnu.org/licenses/>. + */ + +/** + * @template {import('api-map').ApiSurface} [TApiSurface=never] + * @param {import('api-map').ApiMapInit<TApiSurface>} init + * @returns {import('api-map').ApiMap<TApiSurface>} + */ +export function createApiMap(init) { + return new Map(init); +} + +/** + * @template {import('api-map').ApiSurface} [TApiSurface=never] + * @param {import('api-map').ApiMap<TApiSurface>} map + * @param {import('api-map').ApiMapInit<TApiSurface>} init + * @throws {Error} + */ +export function extendApiMap(map, init) { + for (const [key, value] of init) { + if (map.has(key)) { throw new Error(`The handler for ${String(key)} has already been registered`); } + map.set(key, value); + } +} + +/** + * @template {import('api-map').ApiSurface} [TApiSurface=never] + * @param {import('api-map').ApiMap<TApiSurface>} map + * @param {string} name + * @returns {import('api-map').ApiHandlerAny<TApiSurface>|undefined} + */ +export function getApiMapHandler(map, name) { + return map.get(/** @type {import('api-map').ApiNames<TApiSurface>} */ (name)); +} diff --git a/types/ext/api-map.d.ts b/types/ext/api-map.d.ts new file mode 100644 index 00000000..eebc886a --- /dev/null +++ b/types/ext/api-map.d.ts @@ -0,0 +1,55 @@ +/* + * Copyright (C) 2023 Yomitan Authors + * + * 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 + * along with this program. If not, see <https://www.gnu.org/licenses/>. + */ + +type ApiSurface = { + [name: string]: ApiItem; +}; + +type ApiItem = { + params: void | {[name: string]: unknown}; + return: unknown; +}; + +export type ApiHandler<TApiItem extends ApiItem> = (params: TApiItem['params']) => TApiItem['return'] | Promise<TApiItem['return']>; + +type ApiHandlerSurface<TApiSurface extends ApiSurface> = {[name in ApiNames<TApiSurface>]: ApiHandler<TApiSurface[name]>}; + +export type ApiHandlerAny<TApiSurface extends ApiSurface> = ApiHandlerSurface<TApiSurface>[ApiNames<TApiSurface>]; + +export type ApiNames<TApiSurface extends ApiSurface> = keyof TApiSurface; + +export type ApiParams<TApiSurface extends ApiSurface, TName extends ApiNames<TApiSurface>> = TApiSurface[TName]['params']; + +export type ApiReturn<TApiSurface extends ApiSurface, TName extends ApiNames<TApiSurface>> = TApiSurface[TName]['return']; + +export type ApiMap<TApiSurface extends ApiSurface> = Map<ApiNames<TApiSurface>, ApiHandlerAny<TApiSurface>>; + +export type ApiMapInit<TApiSurface extends ApiSurface> = ApiMapInitItemAny<TApiSurface>[]; + +export type ApiMapInitLax<TApiSurface extends ApiSurface> = ApiMapInitLaxItem<TApiSurface>[]; + +export type ApiMapInitLaxItem<TApiSurface extends ApiSurface> = [ + name: ApiNames<TApiSurface>, + handler: ApiHandlerAny<TApiSurface>, +]; + +type ApiMapInitItem<TApiSurface extends ApiSurface, TName extends ApiNames<TApiSurface>> = [ + name: TName, + handler: ApiHandler<TApiSurface[TName]>, +]; + +type ApiMapInitItemAny<TApiSurface extends ApiSurface> = {[key in ApiNames<TApiSurface>]: ApiMapInitItem<TApiSurface, key>}[ApiNames<TApiSurface>]; diff --git a/types/ext/offscreen.d.ts b/types/ext/offscreen.d.ts index c741ac99..451f5f9e 100644 --- a/types/ext/offscreen.d.ts +++ b/types/ext/offscreen.d.ts @@ -15,7 +15,6 @@ * along with this program. If not, see <https://www.gnu.org/licenses/>. */ -import type * as Core from './core'; import type * as Deinflector from './deinflector'; import type * as Dictionary from './dictionary'; import type * as DictionaryDatabase from './dictionary-database'; @@ -23,64 +22,84 @@ import type * as DictionaryImporter from './dictionary-importer'; import type * as Environment from './environment'; import type * as Translation from './translation'; import type * as Translator from './translator'; +import type {ApiMap, ApiMapInit, ApiHandler, ApiParams, ApiReturn} from './api-map'; -export type Message<T extends MessageType> = ( - MessageDetailsMap[T] extends undefined ? - {action: T} : - {action: T, params: MessageDetailsMap[T]} -); - -export type MessageReturn<T extends MessageType> = MessageReturnMap[T]; - -type MessageDetailsMap = { - databasePrepareOffscreen: undefined; - getDictionaryInfoOffscreen: undefined; - databasePurgeOffscreen: undefined; +type OffscreenApiSurface = { + databasePrepareOffscreen: { + params: void; + return: void; + }; + getDictionaryInfoOffscreen: { + params: void; + return: DictionaryImporter.Summary[]; + }; + databasePurgeOffscreen: { + params: void; + return: boolean; + }; databaseGetMediaOffscreen: { - targets: DictionaryDatabase.MediaRequest[]; + params: { + targets: DictionaryDatabase.MediaRequest[]; + }; + return: DictionaryDatabase.Media<string>[]; }; translatorPrepareOffscreen: { - deinflectionReasons: Deinflector.ReasonsRaw; + params: { + deinflectionReasons: Deinflector.ReasonsRaw; + }; + return: void; }; findKanjiOffscreen: { - text: string; - options: FindKanjiOptionsOffscreen; + params: { + text: string; + options: FindKanjiOptionsOffscreen; + }; + return: Dictionary.KanjiDictionaryEntry[]; }; findTermsOffscreen: { - mode: Translator.FindTermsMode; - text: string; - options: FindTermsOptionsOffscreen; + params: { + mode: Translator.FindTermsMode; + text: string; + options: FindTermsOptionsOffscreen; + }; + return: Translator.FindTermsResult; }; getTermFrequenciesOffscreen: { - termReadingList: Translator.TermReadingList; - dictionaries: string[]; + params: { + termReadingList: Translator.TermReadingList; + dictionaries: string[]; + }; + return: Translator.TermFrequencySimple[]; + }; + clearDatabaseCachesOffscreen: { + params: void; + return: void; }; - clearDatabaseCachesOffscreen: undefined; clipboardSetBrowserOffscreen: { - value: Environment.Browser | null; + params: { + value: Environment.Browser | null; + }; + return: void; }; clipboardGetTextOffscreen: { - useRichText: boolean; + params: { + useRichText: boolean; + }; + return: string; + }; + clipboardGetImageOffscreen: { + params: void; + return: string | null; }; - clipboardGetImageOffscreen: undefined; }; -type MessageReturnMap = { - databasePrepareOffscreen: void; - getDictionaryInfoOffscreen: DictionaryImporter.Summary[]; - databasePurgeOffscreen: boolean; - databaseGetMediaOffscreen: DictionaryDatabase.Media<string>[]; - translatorPrepareOffscreen: void; - findKanjiOffscreen: Dictionary.KanjiDictionaryEntry[]; - findTermsOffscreen: Translator.FindTermsResult; - getTermFrequenciesOffscreen: Translator.TermFrequencySimple[]; - clearDatabaseCachesOffscreen: void; - clipboardSetBrowserOffscreen: void; - clipboardGetTextOffscreen: string; - clipboardGetImageOffscreen: string | null; -}; +export type Message<TName extends MessageType> = ( + OffscreenApiParams<TName> extends void ? + {action: TName} : + {action: TName, params: OffscreenApiParams<TName>} +); -export type MessageType = keyof MessageDetailsMap; +export type MessageType = keyof OffscreenApiSurface; export type FindKanjiOptionsOffscreen = Omit<Translation.FindKanjiOptions, 'enabledDictionaryMap'> & { enabledDictionaryMap: [ @@ -103,15 +122,12 @@ export type FindTermsTextReplacementOffscreen = Omit<Translation.FindTermsTextRe pattern: string; }; -export type MessageHandler< - TMessage extends MessageType, - TIsAsync extends boolean, -> = ( - details: MessageDetailsMap[TMessage], -) => (TIsAsync extends true ? Promise<MessageReturn<TMessage>> : MessageReturn<TMessage>); +export type OffscreenApiMap = ApiMap<OffscreenApiSurface>; + +export type OffscreenApiMapInit = ApiMapInit<OffscreenApiSurface>; -export type MessageHandlerMap = Map<MessageType, Core.MessageHandler>; +export type OffscreenApiHandler<TName extends keyof OffscreenApiSurface> = ApiHandler<OffscreenApiSurface[TName]>; -export type MessageHandlerMapInit = MessageHandlerMapInitItem[]; +export type OffscreenApiParams<TName extends keyof OffscreenApiSurface> = ApiParams<OffscreenApiSurface, TName>; -export type MessageHandlerMapInitItem = [messageType: MessageType, handler: Core.MessageHandler]; +export type OffscreenApiReturn<TName extends keyof OffscreenApiSurface> = ApiReturn<OffscreenApiSurface, TName>; |