/* * Copyright (C) 2023-2025 Yomitan Authors * Copyright (C) 2016-2022 Yomichan 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 . */ import {initWasm, Resvg} from '../../lib/resvg-wasm.js'; import {createApiMap, invokeApiMapHandler} from '../core/api-map.js'; import {ExtensionError} from '../core/extension-error.js'; import {log} from '../core/log.js'; import {safePerformance} from '../core/safe-performance.js'; import {stringReverse} from '../core/utilities.js'; import {Database} from '../data/database.js'; export class DictionaryDatabase { constructor() { /** @type {Database} */ this._db = new Database(); /** @type {string} */ this._dbName = 'dict'; /** @type {import('dictionary-database').CreateQuery} */ this._createOnlyQuery1 = (item) => IDBKeyRange.only(item); /** @type {import('dictionary-database').CreateQuery} */ this._createOnlyQuery2 = (item) => IDBKeyRange.only(item.query); /** @type {import('dictionary-database').CreateQuery} */ this._createOnlyQuery3 = (item) => IDBKeyRange.only(item.term); /** @type {import('dictionary-database').CreateQuery} */ this._createOnlyQuery4 = (item) => IDBKeyRange.only(item.path); /** @type {import('dictionary-database').CreateQuery} */ this._createOnlyQuery5 = (item) => IDBKeyRange.only(item.path); /** @type {import('dictionary-database').CreateQuery} */ this._createBoundQuery1 = (item) => IDBKeyRange.bound(item, `${item}\uffff`, false, false); /** @type {import('dictionary-database').CreateQuery} */ this._createBoundQuery2 = (item) => { item = stringReverse(item); return IDBKeyRange.bound(item, `${item}\uffff`, false, false); }; /** @type {import('dictionary-database').CreateResult} */ this._createTermBind1 = this._createTermExact.bind(this); /** @type {import('dictionary-database').CreateResult} */ this._createTermBind2 = this._createTermSequenceExact.bind(this); /** @type {import('dictionary-database').CreateResult} */ this._createTermMetaBind = this._createTermMeta.bind(this); /** @type {import('dictionary-database').CreateResult} */ this._createKanjiBind = this._createKanji.bind(this); /** @type {import('dictionary-database').CreateResult} */ this._createKanjiMetaBind = this._createKanjiMeta.bind(this); /** @type {import('dictionary-database').CreateResult} */ this._createMediaBind = this._createMedia.bind(this); /** @type {import('dictionary-database').CreateResult} */ this._createDrawMediaBind = this._createDrawMedia.bind(this); /** * @type {Worker?} */ this._worker = null; /** * @type {Uint8Array?} */ this._resvgFontBuffer = null; /** @type {import('dictionary-database').ApiMap} */ this._apiMap = createApiMap([ ['drawMedia', this._onDrawMedia.bind(this)], ]); } /** * do upgrades for the IndexedDB schema (basically limited to adding new stores when needed) */ async prepare() { // do not do upgrades in web workers as they are considered to be children of the main thread and are not responsible for database upgrades const isWorker = self.constructor.name !== 'Window'; const upgrade = /** @type {import('database').StructureDefinition[]?} */ ([ /** @type {import('database').StructureDefinition} */ ({ version: 20, stores: { terms: { primaryKey: {keyPath: 'id', autoIncrement: true}, indices: ['dictionary', 'expression', 'reading'], }, kanji: { primaryKey: {autoIncrement: true}, indices: ['dictionary', 'character'], }, tagMeta: { primaryKey: {autoIncrement: true}, indices: ['dictionary'], }, dictionaries: { primaryKey: {autoIncrement: true}, indices: ['title', 'version'], }, }, }), { version: 30, stores: { termMeta: { primaryKey: {autoIncrement: true}, indices: ['dictionary', 'expression'], }, kanjiMeta: { primaryKey: {autoIncrement: true}, indices: ['dictionary', 'character'], }, tagMeta: { primaryKey: {autoIncrement: true}, indices: ['dictionary', 'name'], }, }, }, { version: 40, stores: { terms: { primaryKey: {keyPath: 'id', autoIncrement: true}, indices: ['dictionary', 'expression', 'reading', 'sequence'], }, }, }, { version: 50, stores: { terms: { primaryKey: {keyPath: 'id', autoIncrement: true}, indices: ['dictionary', 'expression', 'reading', 'sequence', 'expressionReverse', 'readingReverse'], }, }, }, { version: 60, stores: { media: { primaryKey: {keyPath: 'id', autoIncrement: true}, indices: ['dictionary', 'path'], }, }, }, ]); await this._db.open( this._dbName, 60, isWorker ? null : upgrade, ); // when we are not a worker ourselves, create a worker which is basically just a wrapper around this class, which we can use to offload some functions to if (!isWorker) { this._worker = new Worker('/js/dictionary/dictionary-database-worker-main.js', {type: 'module'}); this._worker.addEventListener('error', (event) => { log.log('Worker terminated with error:', event); }); this._worker.addEventListener('unhandledrejection', (event) => { log.log('Unhandled promise rejection in worker:', event); }); } else { // when we are the worker, prepare to need to do some SVG work and load appropriate wasm & fonts await initWasm(fetch('/lib/resvg.wasm')); const font = await fetch('/fonts/NotoSansJP-Regular.ttf'); const fontData = await font.arrayBuffer(); this._resvgFontBuffer = new Uint8Array(fontData); } } /** */ async close() { this._db.close(); } /** * @returns {boolean} */ isPrepared() { return this._db.isOpen(); } /** * @returns {Promise} */ async purge() { if (this._db.isOpening()) { throw new Error('Cannot purge database while opening'); } if (this._db.isOpen()) { this._db.close(); } if (this._worker !== null) { this._worker.terminate(); this._worker = null; } let result = false; try { await Database.deleteDatabase(this._dbName); result = true; } catch (e) { log.error(e); } await this.prepare(); return result; } /** * @param {string} dictionaryName * @param {number} progressRate * @param {import('dictionary-database').DeleteDictionaryProgressCallback} onProgress */ async deleteDictionary(dictionaryName, progressRate, onProgress) { /** @type {[objectStoreName: import('dictionary-database').ObjectStoreName, key: string][][]} */ const targetGroups = [ [ ['kanji', 'dictionary'], ['kanjiMeta', 'dictionary'], ['terms', 'dictionary'], ['termMeta', 'dictionary'], ['tagMeta', 'dictionary'], ['media', 'dictionary'], ], [ ['dictionaries', 'title'], ], ]; let storeCount = 0; for (const targets of targetGroups) { storeCount += targets.length; } /** @type {import('dictionary-database').DeleteDictionaryProgressData} */ const progressData = { count: 0, processed: 0, storeCount, storesProcesed: 0, }; /** * @param {IDBValidKey[]} keys * @returns {IDBValidKey[]} */ const filterKeys = (keys) => { ++progressData.storesProcesed; progressData.count += keys.length; onProgress(progressData); return keys; }; const onProgressWrapper = () => { const processed = progressData.processed + 1; progressData.processed = processed; if ((processed % progressRate) === 0 || processed === progressData.count) { onProgress(progressData); } }; for (const targets of targetGroups) { const promises = []; for (const [objectStoreName, indexName] of targets) { const query = IDBKeyRange.only(dictionaryName); const promise = this._db.bulkDelete(objectStoreName, indexName, query, filterKeys, onProgressWrapper); promises.push(promise); } await Promise.all(promises); } } /** * @param {string[]} termList * @param {import('dictionary-database').DictionarySet} dictionaries * @param {import('dictionary-database').MatchType} matchType * @returns {Promise} */ findTermsBulk(termList, dictionaries, matchType) { const visited = new Set(); /** @type {import('dictionary-database').FindPredicate} */ const predicate = (row) => { if (!dictionaries.has(row.dictionary)) { return false; } const {id} = row; if (visited.has(id)) { return false; } visited.add(id); return true; }; const indexNames = (matchType === 'suffix') ? ['expressionReverse', 'readingReverse'] : ['expression', 'reading']; let createQuery = this._createOnlyQuery1; switch (matchType) { case 'prefix': createQuery = this._createBoundQuery1; break; case 'suffix': createQuery = this._createBoundQuery2; break; } const createResult = this._createTermGeneric.bind(this, matchType); return this._findMultiBulk('terms', indexNames, termList, createQuery, predicate, createResult); } /** * @param {import('dictionary-database').TermExactRequest[]} termList * @param {import('dictionary-database').DictionarySet} dictionaries * @returns {Promise} */ findTermsExactBulk(termList, dictionaries) { /** @type {import('dictionary-database').FindPredicate} */ const predicate = (row, item) => (row.reading === item.reading && dictionaries.has(row.dictionary)); return this._findMultiBulk('terms', ['expression'], termList, this._createOnlyQuery3, predicate, this._createTermBind1); } /** * @param {import('dictionary-database').DictionaryAndQueryRequest[]} items * @returns {Promise} */ findTermsBySequenceBulk(items) { /** @type {import('dictionary-database').FindPredicate} */ const predicate = (row, item) => (row.dictionary === item.dictionary); return this._findMultiBulk('terms', ['sequence'], items, this._createOnlyQuery2, predicate, this._createTermBind2); } /** * @param {string[]} termList * @param {import('dictionary-database').DictionarySet} dictionaries * @returns {Promise} */ findTermMetaBulk(termList, dictionaries) { /** @type {import('dictionary-database').FindPredicate} */ const predicate = (row) => dictionaries.has(row.dictionary); return this._findMultiBulk('termMeta', ['expression'], termList, this._createOnlyQuery1, predicate, this._createTermMetaBind); } /** * @param {string[]} kanjiList * @param {import('dictionary-database').DictionarySet} dictionaries * @returns {Promise} */ findKanjiBulk(kanjiList, dictionaries) { /** @type {import('dictionary-database').FindPredicate} */ const predicate = (row) => dictionaries.has(row.dictionary); return this._findMultiBulk('kanji', ['character'], kanjiList, this._createOnlyQuery1, predicate, this._createKanjiBind); } /** * @param {string[]} kanjiList * @param {import('dictionary-database').DictionarySet} dictionaries * @returns {Promise} */ findKanjiMetaBulk(kanjiList, dictionaries) { /** @type {import('dictionary-database').FindPredicate} */ const predicate = (row) => dictionaries.has(row.dictionary); return this._findMultiBulk('kanjiMeta', ['character'], kanjiList, this._createOnlyQuery1, predicate, this._createKanjiMetaBind); } /** * @param {import('dictionary-database').DictionaryAndQueryRequest[]} items * @returns {Promise<(import('dictionary-database').Tag|undefined)[]>} */ findTagMetaBulk(items) { /** @type {import('dictionary-database').FindPredicate} */ const predicate = (row, item) => (row.dictionary === item.dictionary); return this._findFirstBulk('tagMeta', 'name', items, this._createOnlyQuery2, predicate); } /** * @param {string} name * @param {string} dictionary * @returns {Promise} */ findTagForTitle(name, dictionary) { const query = IDBKeyRange.only(name); return this._db.find('tagMeta', 'name', query, (row) => (/** @type {import('dictionary-database').Tag} */ (row).dictionary === dictionary), null, null); } /** * @param {import('dictionary-database').MediaRequest[]} items * @returns {Promise} */ getMedia(items) { /** @type {import('dictionary-database').FindPredicate} */ const predicate = (row, item) => (row.dictionary === item.dictionary); return this._findMultiBulk('media', ['path'], items, this._createOnlyQuery4, predicate, this._createMediaBind); } /** * @param {import('dictionary-database').DrawMediaRequest[]} items * @param {MessagePort} source */ async drawMedia(items, source) { if (this._worker !== null) { // if a worker is available, offload the work to it this._worker.postMessage({action: 'drawMedia', params: {items}}, [source]); return; } // otherwise, you are the worker, so do the work safePerformance.mark('drawMedia:start'); // merge items with the same path to reduce the number of database queries. collects the canvases into a single array for each path. /** @type {Map} */ const groupedItems = new Map(); for (const item of items) { const {path, dictionary, canvasIndex, canvasWidth, canvasHeight, generation} = item; const key = `${path}:::${dictionary}`; if (!groupedItems.has(key)) { groupedItems.set(key, {path, dictionary, canvasIndexes: [], canvasWidth, canvasHeight, generation}); } groupedItems.get(key)?.canvasIndexes.push(canvasIndex); } const groupedItemsArray = [...groupedItems.values()]; /** @type {import('dictionary-database').FindPredicate} */ const predicate = (row, item) => (row.dictionary === item.dictionary); const results = await this._findMultiBulk('media', ['path'], groupedItemsArray, this._createOnlyQuery5, predicate, this._createDrawMediaBind); // move all svgs to front to have a hotter loop results.sort((a, _b) => (a.mediaType === 'image/svg+xml' ? -1 : 1)); safePerformance.mark('drawMedia:draw:start'); for (const m of results) { if (m.mediaType === 'image/svg+xml') { safePerformance.mark('drawMedia:draw:svg:start'); /** @type {import('@resvg/resvg-wasm').ResvgRenderOptions} */ const opts = { fitTo: { mode: 'width', value: m.canvasWidth, }, font: { fontBuffers: this._resvgFontBuffer !== null ? [this._resvgFontBuffer] : [], }, }; const resvgJS = new Resvg(new Uint8Array(m.content), opts); const render = resvgJS.render(); source.postMessage({action: 'drawBufferToCanvases', params: {buffer: render.pixels.buffer, width: render.width, height: render.height, canvasIndexes: m.canvasIndexes, generation: m.generation}}, [render.pixels.buffer]); safePerformance.mark('drawMedia:draw:svg:end'); safePerformance.measure('drawMedia:draw:svg', 'drawMedia:draw:svg:start', 'drawMedia:draw:svg:end'); } else { safePerformance.mark('drawMedia:draw:raster:start'); // ImageDecoder is slightly faster than Blob/createImageBitmap, but // 1) it is not available in Firefox <133 // 2) it is available in Firefox >=133, but it's not possible to transfer VideoFrames cross-process // // So the second branch is a fallback for all versions of Firefox and doesn't use ImageDecoder at all // The second branch can eventually be changed to use ImageDecoder when we are okay with dropping support for Firefox <133 // The branches can be unified entirely when Firefox implements support for transferring VideoFrames cross-process in postMessage if ('serviceWorker' in navigator) { // this is just a check for chrome, we don't actually use service worker functionality here const imageDecoder = new ImageDecoder({type: m.mediaType, data: m.content}); await imageDecoder.decode().then((decodedImageResult) => { source.postMessage({action: 'drawDecodedImageToCanvases', params: {decodedImage: decodedImageResult.image, canvasIndexes: m.canvasIndexes, generation: m.generation}}, [decodedImageResult.image]); }); } else { const image = new Blob([m.content], {type: m.mediaType}); await createImageBitmap(image, {resizeWidth: m.canvasWidth, resizeHeight: m.canvasHeight, resizeQuality: 'high'}).then((decodedImage) => { // we need to do a dumb hack where we convert this ImageBitmap to an ImageData by drawing it to a temporary canvas, because Firefox doesn't support transferring ImageBitmaps cross-process const canvas = new OffscreenCanvas(decodedImage.width, decodedImage.height); const ctx = canvas.getContext('2d'); if (ctx !== null) { ctx.drawImage(decodedImage, 0, 0); const imageData = ctx.getImageData(0, 0, decodedImage.width, decodedImage.height); source.postMessage({action: 'drawBufferToCanvases', params: {buffer: imageData.data.buffer, width: decodedImage.width, height: decodedImage.height, canvasIndexes: m.canvasIndexes, generation: m.generation}}, [imageData.data.buffer]); } }); } safePerformance.mark('drawMedia:draw:raster:end'); safePerformance.measure('drawMedia:draw:raster', 'drawMedia:draw:raster:start', 'drawMedia:draw:raster:end'); } } safePerformance.mark('drawMedia:draw:end'); safePerformance.measure('drawMedia:draw', 'drawMedia:draw:start', 'drawMedia:draw:end'); safePerformance.mark('drawMedia:end'); safePerformance.measure('drawMedia', 'drawMedia:start', 'drawMedia:end'); } /** * @returns {Promise} */ getDictionaryInfo() { return new Promise((resolve, reject) => { const transaction = this._db.transaction(['dictionaries'], 'readonly'); const objectStore = transaction.objectStore('dictionaries'); this._db.getAll(objectStore, null, resolve, reject, null); }); } /** * @param {string[]} dictionaryNames * @param {boolean} getTotal * @returns {Promise} */ getDictionaryCounts(dictionaryNames, getTotal) { return new Promise((resolve, reject) => { const targets = [ ['kanji', 'dictionary'], ['kanjiMeta', 'dictionary'], ['terms', 'dictionary'], ['termMeta', 'dictionary'], ['tagMeta', 'dictionary'], ['media', 'dictionary'], ]; const objectStoreNames = targets.map(([objectStoreName]) => objectStoreName); const transaction = this._db.transaction(objectStoreNames, 'readonly'); const databaseTargets = targets.map(([objectStoreName, indexName]) => { const objectStore = transaction.objectStore(objectStoreName); const index = objectStore.index(indexName); return {objectStore, index}; }); /** @type {import('database').CountTarget[]} */ const countTargets = []; if (getTotal) { for (const {objectStore} of databaseTargets) { countTargets.push([objectStore, void 0]); } } for (const dictionaryName of dictionaryNames) { const query = IDBKeyRange.only(dictionaryName); for (const {index} of databaseTargets) { countTargets.push([index, query]); } } /** * @param {number[]} results */ const onCountComplete = (results) => { const resultCount = results.length; const targetCount = targets.length; /** @type {import('dictionary-database').DictionaryCountGroup[]} */ const counts = []; for (let i = 0; i < resultCount; i += targetCount) { /** @type {import('dictionary-database').DictionaryCountGroup} */ const countGroup = {}; for (let j = 0; j < targetCount; ++j) { countGroup[targets[j][0]] = results[i + j]; } counts.push(countGroup); } const total = getTotal ? /** @type {import('dictionary-database').DictionaryCountGroup} */ (counts.shift()) : null; resolve({total, counts}); }; this._db.bulkCount(countTargets, onCountComplete, reject); }); } /** * @param {string} title * @returns {Promise} */ async dictionaryExists(title) { const query = IDBKeyRange.only(title); const result = await this._db.find('dictionaries', 'title', query, null, null, void 0); return typeof result !== 'undefined'; } /** * @template {import('dictionary-database').ObjectStoreName} T * @param {T} objectStoreName * @param {import('dictionary-database').ObjectStoreData[]} items * @param {number} start * @param {number} count * @returns {Promise} */ bulkAdd(objectStoreName, items, start, count) { return this._db.bulkAdd(objectStoreName, items, start, count); } /** * @template {import('dictionary-database').ObjectStoreName} T * @param {T} objectStoreName * @param {import('dictionary-database').ObjectStoreData} item * @returns {Promise>} */ addWithResult(objectStoreName, item) { return this._db.addWithResult(objectStoreName, item); } /** * @template {import('dictionary-database').ObjectStoreName} T * @param {T} objectStoreName * @param {import('dictionary-database').DatabaseUpdateItem[]} items * @param {number} start * @param {number} count * @returns {Promise} */ bulkUpdate(objectStoreName, items, start, count) { return this._db.bulkUpdate(objectStoreName, items, start, count); } // Private /** * @template [TRow=unknown] * @template [TItem=unknown] * @template [TResult=unknown] * @param {import('dictionary-database').ObjectStoreName} objectStoreName * @param {string[]} indexNames * @param {TItem[]} items * @param {import('dictionary-database').CreateQuery} createQuery * @param {import('dictionary-database').FindPredicate} predicate * @param {import('dictionary-database').CreateResult} createResult * @returns {Promise} */ _findMultiBulk(objectStoreName, indexNames, items, createQuery, predicate, createResult) { safePerformance.mark('findMultiBulk:start'); return new Promise((resolve, reject) => { const itemCount = items.length; const indexCount = indexNames.length; /** @type {TResult[]} */ const results = []; if (itemCount === 0 || indexCount === 0) { resolve(results); safePerformance.mark('findMultiBulk:end'); safePerformance.measure('findMultiBulk', 'findMultiBulk:start', 'findMultiBulk:end'); return; } const transaction = this._db.transaction([objectStoreName], 'readonly'); const objectStore = transaction.objectStore(objectStoreName); const indexList = []; for (const indexName of indexNames) { indexList.push(objectStore.index(indexName)); } let completeCount = 0; const requiredCompleteCount = itemCount * indexCount; /** * @param {TItem} item * @returns {(rows: TRow[], data: import('dictionary-database').FindMultiBulkData) => void} */ const onGetAll = (item) => (rows, data) => { if (typeof item === 'object' && item !== null && 'path' in item) { safePerformance.mark(`findMultiBulk:onGetAll:${item.path}:end`); safePerformance.measure(`findMultiBulk:onGetAll:${item.path}`, `findMultiBulk:onGetAll:${item.path}:start`, `findMultiBulk:onGetAll:${item.path}:end`); } for (const row of rows) { if (predicate(row, data.item)) { results.push(createResult(row, data)); } } if (++completeCount >= requiredCompleteCount) { resolve(results); safePerformance.mark('findMultiBulk:end'); safePerformance.measure('findMultiBulk', 'findMultiBulk:start', 'findMultiBulk:end'); } }; safePerformance.mark('findMultiBulk:getAll:start'); for (let i = 0; i < itemCount; ++i) { const item = items[i]; const query = createQuery(item); for (let j = 0; j < indexCount; ++j) { /** @type {import('dictionary-database').FindMultiBulkData} */ const data = {item, itemIndex: i, indexIndex: j}; if (typeof item === 'object' && item !== null && 'path' in item) { safePerformance.mark(`findMultiBulk:onGetAll:${item.path}:start`); } this._db.getAll(indexList[j], query, onGetAll(item), reject, data); } } safePerformance.mark('findMultiBulk:getAll:end'); safePerformance.measure('findMultiBulk:getAll', 'findMultiBulk:getAll:start', 'findMultiBulk:getAll:end'); }); } /** * @template [TRow=unknown] * @template [TItem=unknown] * @param {import('dictionary-database').ObjectStoreName} objectStoreName * @param {string} indexName * @param {TItem[]} items * @param {import('dictionary-database').CreateQuery} createQuery * @param {import('dictionary-database').FindPredicate} predicate * @returns {Promise<(TRow|undefined)[]>} */ _findFirstBulk(objectStoreName, indexName, items, createQuery, predicate) { return new Promise((resolve, reject) => { const itemCount = items.length; /** @type {(TRow|undefined)[]} */ const results = new Array(itemCount); if (itemCount === 0) { resolve(results); return; } const transaction = this._db.transaction([objectStoreName], 'readonly'); const objectStore = transaction.objectStore(objectStoreName); const index = objectStore.index(indexName); let completeCount = 0; /** * @param {TRow|undefined} row * @param {number} itemIndex */ const onFind = (row, itemIndex) => { results[itemIndex] = row; if (++completeCount >= itemCount) { resolve(results); } }; for (let i = 0; i < itemCount; ++i) { const item = items[i]; const query = createQuery(item); this._db.findFirst(index, query, onFind, reject, i, predicate, item, void 0); } }); } /** * @param {import('dictionary-database').MatchType} matchType * @param {import('dictionary-database').DatabaseTermEntryWithId} row * @param {import('dictionary-database').FindMultiBulkData} data * @returns {import('dictionary-database').TermEntry} */ _createTermGeneric(matchType, row, data) { const matchSourceIsTerm = (data.indexIndex === 0); const matchSource = (matchSourceIsTerm ? 'term' : 'reading'); if ((matchSourceIsTerm ? row.expression : row.reading) === data.item) { matchType = 'exact'; } return this._createTerm(matchSource, matchType, row, data.itemIndex); } /** * @param {import('dictionary-database').DatabaseTermEntryWithId} row * @param {import('dictionary-database').FindMultiBulkData} data * @returns {import('dictionary-database').TermEntry} */ _createTermExact(row, data) { return this._createTerm('term', 'exact', row, data.itemIndex); } /** * @param {import('dictionary-database').DatabaseTermEntryWithId} row * @param {import('dictionary-database').FindMultiBulkData} data * @returns {import('dictionary-database').TermEntry} */ _createTermSequenceExact(row, data) { return this._createTerm('sequence', 'exact', row, data.itemIndex); } /** * @param {import('dictionary-database').MatchSource} matchSource * @param {import('dictionary-database').MatchType} matchType * @param {import('dictionary-database').DatabaseTermEntryWithId} row * @param {number} index * @returns {import('dictionary-database').TermEntry} */ _createTerm(matchSource, matchType, row, index) { const {sequence} = row; return { index, matchType, matchSource, term: row.expression, reading: row.reading, definitionTags: this._splitField(row.definitionTags || row.tags), termTags: this._splitField(row.termTags), rules: this._splitField(row.rules), definitions: row.glossary, score: row.score, dictionary: row.dictionary, id: row.id, sequence: typeof sequence === 'number' ? sequence : -1, }; } /** * @param {import('dictionary-database').DatabaseKanjiEntry} row * @param {import('dictionary-database').FindMultiBulkData} data * @returns {import('dictionary-database').KanjiEntry} */ _createKanji(row, {itemIndex: index}) { const {stats} = row; return { index, character: row.character, onyomi: this._splitField(row.onyomi), kunyomi: this._splitField(row.kunyomi), tags: this._splitField(row.tags), definitions: row.meanings, stats: typeof stats === 'object' && stats !== null ? stats : {}, dictionary: row.dictionary, }; } /** * @param {import('dictionary-database').DatabaseTermMeta} row * @param {import('dictionary-database').FindMultiBulkData} data * @returns {import('dictionary-database').TermMeta} * @throws {Error} */ _createTermMeta({expression: term, mode, data, dictionary}, {itemIndex: index}) { switch (mode) { case 'freq': return {index, term, mode, data, dictionary}; case 'pitch': return {index, term, mode, data, dictionary}; case 'ipa': return {index, term, mode, data, dictionary}; default: throw new Error(`Unknown mode: ${mode}`); } } /** * @param {import('dictionary-database').DatabaseKanjiMeta} row * @param {import('dictionary-database').FindMultiBulkData} data * @returns {import('dictionary-database').KanjiMeta} */ _createKanjiMeta({character, mode, data, dictionary}, {itemIndex: index}) { return {index, character, mode, data, dictionary}; } /** * @param {import('dictionary-database').MediaDataArrayBufferContent} row * @param {import('dictionary-database').FindMultiBulkData} data * @returns {import('dictionary-database').Media} */ _createMedia(row, {itemIndex: index}) { const {dictionary, path, mediaType, width, height, content} = row; return {index, dictionary, path, mediaType, width, height, content}; } /** * @param {import('dictionary-database').MediaDataArrayBufferContent} row * @param {import('dictionary-database').FindMultiBulkData} data * @returns {import('dictionary-database').DrawMedia} */ _createDrawMedia(row, {itemIndex: index, item: {canvasIndexes, canvasWidth, canvasHeight, generation}}) { const {dictionary, path, mediaType, width, height, content} = row; return {index, dictionary, path, mediaType, width, height, content, canvasIndexes, canvasWidth, canvasHeight, generation}; } /** * @param {unknown} field * @returns {string[]} */ _splitField(field) { return typeof field === 'string' && field.length > 0 ? field.split(' ') : []; } // Parent-Worker API /** * @param {MessagePort} port */ async connectToDatabaseWorker(port) { if (this._worker !== null) { // executes outside of worker this._worker.postMessage({action: 'connectToDatabaseWorker'}, [port]); return; } // executes inside worker port.onmessage = (/** @type {MessageEvent} */event) => { const {action, params} = event.data; return invokeApiMapHandler(this._apiMap, action, params, [port], () => {}); }; port.onmessageerror = (event) => { const error = new ExtensionError('DictionaryDatabase: Error receiving message from main thread'); error.data = event; log.error(error); }; } /** @type {import('dictionary-database').ApiHandler<'drawMedia'>} */ _onDrawMedia(params, port) { void this.drawMedia(params.requests, port); } }