mirror of
https://github.com/ksyasuda/SubMiner.git
synced 2026-02-28 06:22:45 -08:00
209 lines
7.9 KiB
JavaScript
209 lines
7.9 KiB
JavaScript
/*
|
|
* 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 <https://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
import {Handlebars} from '../../lib/handlebars.js';
|
|
import {ExtensionError} from '../core/extension-error.js';
|
|
|
|
export class TemplateRenderer {
|
|
constructor() {
|
|
/** @type {Map<string, import('handlebars').TemplateDelegate<import('anki-templates').NoteData>>} */
|
|
this._cache = new Map();
|
|
/** @type {number} */
|
|
this._cacheMaxSize = 5;
|
|
/** @type {Map<import('anki-templates').RenderMode, import('template-renderer').DataType>} */
|
|
this._dataTypes = new Map();
|
|
/** @type {?((noteData: import('anki-templates').NoteData) => import('template-renderer').SetupCallbackResult)} */
|
|
this._renderSetup = null;
|
|
/** @type {?((noteData: import('anki-templates').NoteData) => import('template-renderer').CleanupCallbackResult)} */
|
|
this._renderCleanup = null;
|
|
}
|
|
|
|
/**
|
|
* @param {import('template-renderer').HelperFunctionsDescriptor} helpers
|
|
*/
|
|
registerHelpers(helpers) {
|
|
for (const [name, helper] of helpers) {
|
|
this._registerHelper(name, helper);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param {import('anki-templates').RenderMode} name
|
|
* @param {import('template-renderer').DataType} details
|
|
*/
|
|
registerDataType(name, {modifier, composeData}) {
|
|
this._dataTypes.set(name, {modifier, composeData});
|
|
}
|
|
|
|
/**
|
|
* @param {?((noteData: import('anki-templates').NoteData) => import('template-renderer').SetupCallbackResult)} setup
|
|
* @param {?((noteData: import('anki-templates').NoteData) => import('template-renderer').CleanupCallbackResult)} cleanup
|
|
*/
|
|
setRenderCallbacks(setup, cleanup) {
|
|
this._renderSetup = setup;
|
|
this._renderCleanup = cleanup;
|
|
}
|
|
|
|
/**
|
|
* @param {string} template
|
|
* @param {import('template-renderer').PartialOrCompositeRenderData} data
|
|
* @param {import('anki-templates').RenderMode} type
|
|
* @returns {import('template-renderer').RenderResult}
|
|
*/
|
|
render(template, data, type) {
|
|
const instance = this._getTemplateInstance(template);
|
|
const modifiedData = this._getModifiedData(data, void 0, type);
|
|
return this._renderTemplate(instance, modifiedData);
|
|
}
|
|
|
|
/**
|
|
* @param {import('template-renderer').RenderMultiItem[]} items
|
|
* @returns {import('core').Response<import('template-renderer').RenderResult>[]}
|
|
*/
|
|
renderMulti(items) {
|
|
/** @type {import('core').Response<import('template-renderer').RenderResult>[]} */
|
|
const results = [];
|
|
for (const {template, templateItems} of items) {
|
|
const instance = this._getTemplateInstance(template);
|
|
for (const {type, commonData, datas} of templateItems) {
|
|
for (const data of datas) {
|
|
let result;
|
|
try {
|
|
const data2 = this._getModifiedData(data, commonData, type);
|
|
const renderResult = this._renderTemplate(instance, data2);
|
|
result = {result: renderResult};
|
|
} catch (error) {
|
|
result = {error: ExtensionError.serialize(error)};
|
|
}
|
|
results.push(result);
|
|
}
|
|
}
|
|
}
|
|
return results;
|
|
}
|
|
|
|
/**
|
|
* @param {import('template-renderer').CompositeRenderData} data
|
|
* @param {import('anki-templates').RenderMode} type
|
|
* @returns {import('anki-templates').NoteData}
|
|
*/
|
|
getModifiedData(data, type) {
|
|
return this._getModifiedData(data, void 0, type);
|
|
}
|
|
|
|
// Private
|
|
|
|
/**
|
|
* @param {string} template
|
|
* @returns {import('handlebars').TemplateDelegate<import('anki-templates').NoteData>}
|
|
*/
|
|
_getTemplateInstance(template) {
|
|
const cache = this._cache;
|
|
let instance = cache.get(template);
|
|
if (typeof instance === 'undefined') {
|
|
this._updateCacheSize(this._cacheMaxSize - 1);
|
|
// 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
|
|
instance = /** @type {import('handlebars').TemplateDelegate<import('anki-templates').NoteData>} */ (Handlebars.compileAST(template));
|
|
cache.set(template, instance);
|
|
}
|
|
|
|
return instance;
|
|
}
|
|
|
|
/**
|
|
* @param {import('handlebars').TemplateDelegate<import('anki-templates').NoteData>} instance
|
|
* @param {import('anki-templates').NoteData} data
|
|
* @returns {import('template-renderer').RenderResult}
|
|
*/
|
|
_renderTemplate(instance, data) {
|
|
const renderSetup = this._renderSetup;
|
|
const renderCleanup = this._renderCleanup;
|
|
/** @type {string} */
|
|
let result;
|
|
/** @type {?import('template-renderer').SetupCallbackResult} */
|
|
let additions1;
|
|
/** @type {?import('template-renderer').CleanupCallbackResult} */
|
|
let additions2;
|
|
try {
|
|
additions1 = (typeof renderSetup === 'function' ? renderSetup(data) : null);
|
|
result = instance(data).replace(/^\n+|\n+$/g, '');
|
|
} finally {
|
|
additions2 = (typeof renderCleanup === 'function' ? renderCleanup(data) : null);
|
|
}
|
|
return /** @type {import('template-renderer').RenderResult} */ (Object.assign({result}, additions1, additions2));
|
|
}
|
|
|
|
/**
|
|
* @param {import('template-renderer').PartialOrCompositeRenderData} data
|
|
* @param {import('anki-note-builder').CommonData|undefined} commonData
|
|
* @param {import('anki-templates').RenderMode} type
|
|
* @returns {import('anki-templates').NoteData}
|
|
* @throws {Error}
|
|
*/
|
|
_getModifiedData(data, commonData, type) {
|
|
if (typeof type === 'string') {
|
|
const typeInfo = this._dataTypes.get(type);
|
|
if (typeof typeInfo !== 'undefined') {
|
|
if (typeof commonData !== 'undefined') {
|
|
const {composeData} = typeInfo;
|
|
data = composeData(data, commonData);
|
|
} else if (typeof data.commonData === 'undefined') {
|
|
throw new Error('Incomplete data');
|
|
}
|
|
const {modifier} = typeInfo;
|
|
return modifier(/** @type {import('template-renderer').CompositeRenderData} */ (data));
|
|
}
|
|
}
|
|
throw new Error(`Invalid type: ${type}`);
|
|
}
|
|
|
|
/**
|
|
* @param {number} maxSize
|
|
*/
|
|
_updateCacheSize(maxSize) {
|
|
const cache = this._cache;
|
|
let removeCount = cache.size - maxSize;
|
|
if (removeCount <= 0) { return; }
|
|
|
|
for (const key of cache.keys()) {
|
|
cache.delete(key);
|
|
if (--removeCount <= 0) { break; }
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param {string} name
|
|
* @param {import('template-renderer').HelperFunction} helper
|
|
*/
|
|
_registerHelper(name, helper) {
|
|
/**
|
|
* @this {unknown}
|
|
* @param {unknown[]} args
|
|
* @returns {unknown}
|
|
*/
|
|
function wrapper(...args) {
|
|
const argCountM1 = Math.max(0, args.length - 1);
|
|
const options = /** @type {import('handlebars').HelperOptions} */ (args[argCountM1]);
|
|
args.length = argCountM1;
|
|
return helper(args, this, options);
|
|
}
|
|
Handlebars.registerHelper(name, wrapper);
|
|
}
|
|
}
|