site/node_modules/mathjax-full/ts/input/tex/newcommand/NewcommandMethods.ts

245 lines
9 KiB
TypeScript
Raw Normal View History

2024-10-14 08:09:33 +02:00
/*************************************************************
*
* Copyright (c) 2018-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 Mappings for TeX parsing for definitorial commands.
*
* @author v.sorge@mathjax.org (Volker Sorge)
*/
import {ParseMethod} from '../Types.js';
import TexError from '../TexError.js';
import TexParser from '../TexParser.js';
import * as sm from '../SymbolMap.js';
import {Symbol, Macro} from '../Symbol.js';
import BaseMethods from '../base/BaseMethods.js';
import ParseUtil from '../ParseUtil.js';
import {StackItem} from '../StackItem.js';
import NewcommandUtil from './NewcommandUtil.js';
// Namespace
let NewcommandMethods: Record<string, ParseMethod> = {};
/**
* Implements \newcommand{\name}[n][default]{...}
* @param {TexParser} parser The calling parser.
* @param {string} name The name of the calling command.
*/
NewcommandMethods.NewCommand = function(parser: TexParser, name: string) {
// @test Newcommand Simple
let cs = NewcommandUtil.GetCsNameArgument(parser, name);
let n = NewcommandUtil.GetArgCount(parser, name);
let opt = parser.GetBrackets(name);
let def = parser.GetArgument(name);
NewcommandUtil.addMacro(parser, cs, NewcommandMethods.Macro, [def, n, opt]);
};
/**
* Implements \newenvironment{name}[n][default]{begincmd}{endcmd}
* @param {TexParser} parser The calling parser.
* @param {string} name The name of the calling command.
*/
NewcommandMethods.NewEnvironment = function(parser: TexParser, name: string) {
// @test Newenvironment Empty, Newenvironment Content
let env = ParseUtil.trimSpaces(parser.GetArgument(name));
let n = NewcommandUtil.GetArgCount(parser, name);
let opt = parser.GetBrackets(name);
let bdef = parser.GetArgument(name);
let edef = parser.GetArgument(name);
NewcommandUtil.addEnvironment(parser, env, NewcommandMethods.BeginEnv, [true, bdef, edef, n, opt]);
};
/**
* Implements \def command.
* @param {TexParser} parser The calling parser.
* @param {string} name The name of the calling command.
*/
NewcommandMethods.MacroDef = function(parser: TexParser, name: string) {
// @test Def DoubleLet, DefReDef
let cs = NewcommandUtil.GetCSname(parser, name);
let params = NewcommandUtil.GetTemplate(parser, name, '\\' + cs);
let def = parser.GetArgument(name);
!(params instanceof Array) ?
// @test Def DoubleLet, DefReDef
NewcommandUtil.addMacro(parser, cs, NewcommandMethods.Macro, [def, params]) :
// @test Def Let
NewcommandUtil.addMacro(parser, cs, NewcommandMethods.MacroWithTemplate, [def].concat(params));
};
/**
* Implements the \let command.
*
* All \let commands create either new delimiters or macros in the extension
* maps. In the latter case if the let binds a symbol we have to generate a
* macro with the appropriate parse methods from the SymbolMap. Otherwise we
* simply copy the macro under a new name.
*
* Let does not always work on special characters as TeX does. For example
* "\let\car^ a\car b" will yield a superscript, on the otherhand
* \let\bgroup={ is possible and will work fine in \bgroup a } but will fail
* in \sqrt\bgroup a}.
*
* @param {TexParser} parser The calling parser.
* @param {string} name The name of the calling command.
*/
NewcommandMethods.Let = function(parser: TexParser, name: string) {
const cs = NewcommandUtil.GetCSname(parser, name);
let c = parser.GetNext();
// @test Let Bar, Let Caret
if (c === '=') {
// @test Let Brace Equal, Let Brace Equal Stretchy
parser.i++;
c = parser.GetNext();
}
const handlers = parser.configuration.handlers;
if (c === '\\') {
// @test Let Bar, Let Brace Equal Stretchy
name = NewcommandUtil.GetCSname(parser, name);
let macro = handlers.get('delimiter').lookup('\\' + name) as Symbol;
if (macro) {
// @test Let Bar, Let Brace Equal Stretchy
NewcommandUtil.addDelimiter(parser, '\\' + cs, macro.char, macro.attributes);
return;
}
const map = handlers.get('macro').applicable(name);
if (!map) {
// @test Let Undefined CS
return;
}
if (map instanceof sm.MacroMap) {
// @test Def Let, Newcommand Let
const macro = (map as sm.CommandMap).lookup(name) as Macro;
NewcommandUtil.addMacro(parser, cs, macro.func, macro.args, macro.symbol);
return;
}
macro = (map as sm.CharacterMap).lookup(name) as Symbol;
const newArgs = NewcommandUtil.disassembleSymbol(cs, macro);
const method = (p: TexParser, _cs: string, ...rest: any[]) => {
// @test Let Relet, Let Let, Let Circular Macro
const symb = NewcommandUtil.assembleSymbol(rest);
return map.parser(p, symb);
};
NewcommandUtil.addMacro(parser, cs, method, newArgs);
return;
}
// @test Let Brace Equal, Let Caret
parser.i++;
const macro = handlers.get('delimiter').lookup(c) as Symbol;
if (macro) {
// @test Let Paren Delim, Let Paren Stretchy
NewcommandUtil.addDelimiter(parser, '\\' + cs, macro.char, macro.attributes);
return;
}
// @test Let Brace Equal, Let Caret
NewcommandUtil.addMacro(parser, cs, NewcommandMethods.Macro, [c]);
};
/**
* Process a macro with a parameter template by replacing parameters in the
* parser's string.
* @param {TexParser} parser The calling parser.
* @param {string} name The name of the calling command.
* @param {string} text The text template of the macro.
* @param {string} n The number of parameters.
* @param {string[]} ...params The parameter values.
*/
NewcommandMethods.MacroWithTemplate = function (parser: TexParser, name: string,
text: string, n: string,
...params: string[]) {
const argCount = parseInt(n, 10);
// @test Def Let
if (argCount) {
// @test Def Let
let args = [];
parser.GetNext();
if (params[0] && !NewcommandUtil.MatchParam(parser, params[0])) {
// @test Missing Arguments
throw new TexError('MismatchUseDef',
'Use of %1 doesn\'t match its definition', name);
}
for (let i = 0; i < argCount; i++) {
// @test Def Let
args.push(NewcommandUtil.GetParameter(parser, name, params[i + 1]));
}
text = ParseUtil.substituteArgs(parser, args, text);
}
parser.string = ParseUtil.addArgs(parser, text,
parser.string.slice(parser.i));
parser.i = 0;
ParseUtil.checkMaxMacros(parser);
};
/**
* Process a user-defined environment.
* @param {TexParser} parser The calling parser.
* @param {StackItem} begin The begin stackitem.
* @param {string} bdef The begin definition in the newenvironment macro.
* @param {string} edef The end definition in the newenvironment macro.
* @param {number} n The number of parameters.
* @param {string} def Default for an optional parameter.
*/
NewcommandMethods.BeginEnv = function(parser: TexParser, begin: StackItem,
bdef: string, edef: string, n: number, def: string) {
// @test Newenvironment Empty, Newenvironment Content
// We have an end item, and we are supposed to close this environment.
if (begin.getProperty('end') && parser.stack.env['closing'] === begin.getName()) {
// @test Newenvironment Empty, Newenvironment Content
delete parser.stack.env['closing'];
// Parse the commands in the end environment definition.
let rest = parser.string.slice(parser.i);
parser.string = edef;
parser.i = 0;
parser.Parse();
// Reset to parsing the remainder of the expression.
parser.string = rest;
parser.i = 0;
// Close this environment.
return parser.itemFactory.create('end').setProperty('name', begin.getName());
}
if (n) {
// @test Newenvironment Optional, Newenvironment Arg Optional
let args: string[] = [];
if (def != null) {
// @test Newenvironment Optional, Newenvironment Arg Optional
let optional = parser.GetBrackets('\\begin{' + begin.getName() + '}');
args.push(optional == null ? def : optional);
}
for (let i = args.length; i < n; i++) {
// @test Newenvironment Arg Optional
args.push(parser.GetArgument('\\begin{' + begin.getName() + '}'));
}
bdef = ParseUtil.substituteArgs(parser, args, bdef);
edef = ParseUtil.substituteArgs(parser, [], edef); // no args, but get errors for #n in edef
}
parser.string = ParseUtil.addArgs(parser, bdef,
parser.string.slice(parser.i));
parser.i = 0;
return parser.itemFactory.create('beginEnv').setProperty('name', begin.getName());
};
NewcommandMethods.Macro = BaseMethods.Macro;
export default NewcommandMethods;