/* * Copyright (C) 2023-2024 Yomitan Authors * Copyright (C) 2021-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 <https://www.gnu.org/licenses/>. */ import {Handlebars} from '../../lib/handlebars.js'; export class TemplateRendererMediaProvider { constructor() { /** @type {?import('anki-note-builder').Requirement[]} */ this._requirements = null; } /** @type {?import('anki-note-builder').Requirement[]} */ get requirements() { return this._requirements; } set requirements(value) { this._requirements = value; } /** * @param {import('anki-templates').NoteData} root * @param {unknown[]} args * @param {import('core').SerializableObject} namedArgs * @returns {boolean} */ hasMedia(root, args, namedArgs) { const {media} = root; const data = this._getMediaData(media, args, namedArgs); return (data !== null); } /** * @param {import('anki-templates').NoteData} root * @param {unknown[]} args * @param {import('core').SerializableObject} namedArgs * @returns {?string} */ getMedia(root, args, namedArgs) { const {media} = root; const data = this._getMediaData(media, args, namedArgs); if (data !== null) { const result = this._getFormattedValue(data, namedArgs); if (typeof result === 'string') { return result; } } const defaultValue = namedArgs.default; return defaultValue === null || typeof defaultValue === 'string' ? defaultValue : ''; } // Private /** * @param {import('anki-note-builder').Requirement} value */ _addRequirement(value) { if (this._requirements === null) { return; } this._requirements.push(value); } /** * @param {import('anki-templates').MediaObject} data * @param {import('core').SerializableObject} namedArgs * @returns {string} */ _getFormattedValue(data, namedArgs) { let {value} = data; const {escape = true} = namedArgs; if (escape) { // Handlebars is a custom version of the library without type information, so it's assumed to be "any". // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment value = Handlebars.Utils.escapeExpression(value); } return value; } /** * @param {import('anki-templates').Media} media * @param {unknown[]} args * @param {import('core').SerializableObject} namedArgs * @returns {?(import('anki-templates').MediaObject)} */ _getMediaData(media, args, namedArgs) { const type = args[0]; switch (type) { case 'audio': return this._getSimpleMediaData(media, 'audio'); case 'screenshot': return this._getSimpleMediaData(media, 'screenshot'); case 'clipboardImage': return this._getSimpleMediaData(media, 'clipboardImage'); case 'clipboardText': return this._getSimpleMediaData(media, 'clipboardText'); case 'selectionText': return this._getSimpleMediaData(media, 'selectionText'); case 'textFurigana': return this._getTextFurigana(media, args[1], namedArgs); case 'dictionaryMedia': return this._getDictionaryMedia(media, args[1], namedArgs); default: return null; } } /** * @param {import('anki-templates').Media} media * @param {import('anki-templates').MediaSimpleType} type * @returns {?import('anki-templates').MediaObject} */ _getSimpleMediaData(media, type) { const result = media[type]; if (typeof result === 'object' && result !== null) { return result; } this._addRequirement({type}); return null; } /** * @param {import('anki-templates').Media} media * @param {unknown} path * @param {import('core').SerializableObject} namedArgs * @returns {?import('anki-templates').MediaObject} */ _getDictionaryMedia(media, path, namedArgs) { if (typeof path !== 'string') { return null; } const {dictionaryMedia} = media; const {dictionary} = namedArgs; if (typeof dictionary !== 'string') { return null; } if ( typeof dictionaryMedia !== 'undefined' && Object.prototype.hasOwnProperty.call(dictionaryMedia, dictionary) ) { const dictionaryMedia2 = dictionaryMedia[dictionary]; if (Object.prototype.hasOwnProperty.call(dictionaryMedia2, path)) { const result = dictionaryMedia2[path]; if (typeof result === 'object' && result !== null) { return result; } } } this._addRequirement({ type: 'dictionaryMedia', dictionary, path }); return null; } /** * @param {import('anki-templates').Media} media * @param {unknown} text * @param {import('core').SerializableObject} namedArgs * @returns {?import('anki-templates').MediaObject} */ _getTextFurigana(media, text, namedArgs) { if (typeof text !== 'string') { return null; } const readingMode = this._normalizeReadingMode(namedArgs.readingMode); const {textFurigana} = media; if (Array.isArray(textFurigana)) { for (const entry of textFurigana) { if (entry.text !== text || entry.readingMode !== readingMode) { continue; } return entry.details; } } this._addRequirement({ type: 'textFurigana', text, readingMode }); return null; } /** * @param {unknown} value * @returns {?import('anki-templates').TextFuriganaReadingMode} */ _normalizeReadingMode(value) { switch (value) { case 'hiragana': case 'katakana': return value; default: return null; } } }