site/node_modules/mathjax-full/ts/ui/safe/SafeHandler.ts

128 lines
3.9 KiB
TypeScript
Raw Normal View History

2024-10-14 06:09:33 +00:00
/*************************************************************
*
* Copyright (c) 2020-2022 The MathJax Consortium
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* @fileoverview MathItem, MathDocument, and Handler for the safe extension
*
* @author dpvc@mathjax.org (Davide Cervone)
*/
import {MathItem} from '../../core/MathItem.js';
import {MathDocument, MathDocumentConstructor} from '../../core/MathDocument.js';
import {Handler} from '../../core/Handler.js';
import {Safe} from './safe.js';
/*==========================================================================*/
/**
* Generic constructor for Mixins
*/
export type Constructor<T> = new(...args: any[]) => T;
/*==========================================================================*/
/**
* The properties needed in the MathDocument for sanitizing the internal MathML
*/
export interface SafeMathDocument<N, T, D> extends MathDocument<N, T, D> {
/**
* The Safe object for this document
*/
safe: Safe<N, T, D>;
}
/**
* The mixin for adding safe render action to MathDocuments
*
* @param {B} BaseDocument The MathDocument class to be extended
* @return {SafeMathDocument<N,T,D>} The extended MathDocument class
*/
export function SafeMathDocumentMixin<N, T, D, B extends MathDocumentConstructor<MathDocument<N, T, D>>>(
BaseDocument: B
): Constructor<SafeMathDocument<N, T, D>> & B {
return class extends BaseDocument {
/**
* @override
*/
public static OPTIONS = {
...BaseDocument.OPTIONS,
safeOptions: {
...Safe.OPTIONS,
},
SafeClass: Safe
};
/**
* An instance of the Safe object
*/
public safe: Safe<N, T, D>;
/**
* Extend the MathItem class used for this MathDocument
*
* @override
* @constructor
*/
constructor(...args: any[]) {
super(...args);
this.safe = new this.options.SafeClass(this, this.options.safeOptions);
const ProcessBits = (this.constructor as typeof BaseDocument).ProcessBits;
if (!ProcessBits.has('safe')) {
ProcessBits.allocate('safe');
}
for (const jax of this.inputJax) {
if (jax.name.match(/MathML/)) {
(jax as any).mathml.filterAttribute = this.safe.mmlAttribute.bind(this.safe);
(jax as any).mathml.filterClassList = this.safe.mmlClassList.bind(this.safe);
} else if (jax.name.match(/TeX/)) {
jax.postFilters.add(this.sanitize.bind(jax), -5.5);
}
}
}
/**
* @param {{document:SafeDocument<N,T,D>}} data The document to use for the filter
* (note: this has been bound to the input jax)
*/
protected sanitize(data: {math: MathItem<N, T, D>, document: SafeMathDocument<N, T, D>}) {
data.math.root = (this as any).parseOptions.root;
data.document.safe.sanitize(data.math, data.document);
}
};
}
/*==========================================================================*/
/**
* Add context-menu support to a Handler instance
*
* @param {Handler} handler The Handler instance to enhance
* @return {Handler} The handler that was modified (for purposes of chaining extensions)
*/
export function SafeHandler<N, T, D>(handler: Handler<N, T, D>): Handler<N, T, D> {
handler.documentClass = SafeMathDocumentMixin(handler.documentClass);
return handler;
}