/************************************************************* * * Copyright (c) 2020-2025 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. */ /** * @file 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 = new (...args: any[]) => T; /*==========================================================================*/ /** * The properties needed in the MathDocument for sanitizing the internal MathML */ export interface SafeMathDocument extends MathDocument { /** * The Safe object for this document */ safe: Safe; } /** * The mixin for adding safe render action to MathDocuments * * @param {B} BaseDocument The MathDocument class to be extended * @returns {SafeMathDocument} The extended MathDocument class * * @template N The HTMLElement node class * @template T The Text node class * @template D The Document class * @template B The Base document */ export function SafeMathDocumentMixin< N, T, D, B extends MathDocumentConstructor>, >(BaseDocument: B): Constructor> & 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; /** * Extend the MathItem class used for this MathDocument * * @override * @class */ constructor(...args: any[]) { super(...args); this.safe = new this.options.SafeClass(this, this.options.safeOptions); 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 {object} data The argument containing math item and document * @param {MathItem} data.math The math item to sanitize * @param {SafeMathDocument} data.document The document to use for the * filter (note: this has been bound to the input jax) */ protected sanitize(data: { math: MathItem; document: SafeMathDocument; }) { 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 * @returns {Handler} The handler that was modified (for purposes of chaining extensions) */ export function SafeHandler( handler: Handler ): Handler { handler.documentClass = SafeMathDocumentMixin(handler.documentClass); return handler; }