site/node_modules/mathjax-full/ts/input/tex/base/BaseMethods.ts
2024-10-14 08:09:33 +02:00

1618 lines
52 KiB
TypeScript

/*************************************************************
*
* Copyright (c) 2017-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 The Basic Parse methods.
*
* @author v.sorge@mathjax.org (Volker Sorge)
*/
import * as sitem from './BaseItems.js';
import {StackItem, EnvList} from '../StackItem.js';
import {Macro} from '../Symbol.js';
import {ParseMethod} from '../Types.js';
import NodeUtil from '../NodeUtil.js';
import TexError from '../TexError.js';
import TexParser from '../TexParser.js';
import {TexConstant} from '../TexConstants.js';
import ParseUtil from '../ParseUtil.js';
import {MmlNode, TEXCLASS} from '../../../core/MmlTree/MmlNode.js';
import {MmlMsubsup} from '../../../core/MmlTree/MmlNodes/msubsup.js';
import {MmlMunderover} from '../../../core/MmlTree/MmlNodes/munderover.js';
import {Label} from '../Tags.js';
import {em} from '../../../util/lengths.js';
import {entities} from '../../../util/Entities.js';
import {lookup} from '../../../util/Options.js';
// Namespace
let BaseMethods: Record<string, ParseMethod> = {};
const P_HEIGHT = 1.2 / .85; // cmex10 height plus depth over .85
const MmlTokenAllow: {[key: string]: number} = {
fontfamily: 1, fontsize: 1, fontweight: 1, fontstyle: 1,
color: 1, background: 1,
id: 1, 'class': 1, href: 1, style: 1
};
/**
* Handle LaTeX tokens.
*/
/**
* Handle {
* @param {TexParser} parser The calling parser.
* @param {string} c The parsed character.
*/
BaseMethods.Open = function(parser: TexParser, _c: string) {
// @test Identifier Font, Prime, Prime with subscript
parser.Push(parser.itemFactory.create('open'));
};
/**
* Handle }
* @param {TexParser} parser The calling parser.
* @param {string} c The parsed character.
*/
BaseMethods.Close = function(parser: TexParser, _c: string) {
// @test Identifier Font, Prime, Prime with subscript
parser.Push(parser.itemFactory.create('close'));
};
/**
* Handle tilde and spaces.
* @param {TexParser} parser The calling parser.
* @param {string} c The parsed character.
*/
BaseMethods.Tilde = function(parser: TexParser, _c: string) {
// @test Tilde, Tilde2
parser.Push(parser.create('token', 'mtext', {}, entities.nbsp));
};
/**
* Handling space, by doing nothing.
* @param {TexParser} parser The calling parser.
* @param {string} c The parsed character.
*/
BaseMethods.Space = function(_parser: TexParser, _c: string) {};
/**
* Handle ^
* @param {TexParser} parser The calling parser.
* @param {string} c The parsed character.
*/
BaseMethods.Superscript = function(parser: TexParser, _c: string) {
if (parser.GetNext().match(/\d/)) {
// don't treat numbers as a unit
parser.string = parser.string.substr(0, parser.i + 1) +
' ' + parser.string.substr(parser.i + 1);
}
let primes: MmlNode;
let base: MmlNode | void;
const top = parser.stack.Top();
if (top.isKind('prime')) {
// @test Prime on Prime
[base, primes] = top.Peek(2);
parser.stack.Pop();
} else {
// @test Empty base2, Square, Cube
base = parser.stack.Prev();
if (!base) {
// @test Empty base
base = parser.create('token', 'mi', {}, '');
}
}
const movesupsub = NodeUtil.getProperty(base, 'movesupsub');
let position = NodeUtil.isType(base, 'msubsup') ? (base as MmlMsubsup).sup :
(base as MmlMunderover).over;
if ((NodeUtil.isType(base, 'msubsup') && !NodeUtil.isType(base, 'msup') &&
NodeUtil.getChildAt(base, (base as MmlMsubsup).sup)) ||
(NodeUtil.isType(base, 'munderover') && !NodeUtil.isType(base, 'mover') &&
NodeUtil.getChildAt(base, (base as MmlMunderover).over) &&
!NodeUtil.getProperty(base, 'subsupOK'))) {
// @test Double-super-error, Double-over-error
throw new TexError('DoubleExponent', 'Double exponent: use braces to clarify');
}
if (!NodeUtil.isType(base, 'msubsup') || NodeUtil.isType(base, 'msup')) {
if (movesupsub) {
// @test Move Superscript, Large Operator
if (!NodeUtil.isType(base, 'munderover') || NodeUtil.isType(base, 'mover') ||
NodeUtil.getChildAt(base, (base as MmlMunderover).over)) {
// @test Large Operator
base = parser.create('node', 'munderover', [base], {movesupsub: true});
}
position = (base as MmlMunderover).over;
} else {
// @test Empty base, Empty base2, Square, Cube
base = parser.create('node', 'msubsup', [base]);
position = (base as MmlMsubsup).sup;
}
}
parser.Push(
parser.itemFactory.create('subsup', base).setProperties({
position: position, primes: primes, movesupsub: movesupsub
}) );
};
/**
* Handle _
* @param {TexParser} parser The calling parser.
* @param {string} c The parsed character.
*/
BaseMethods.Subscript = function(parser: TexParser, _c: string) {
if (parser.GetNext().match(/\d/)) {
// don't treat numbers as a unit
parser.string =
parser.string.substr(0, parser.i + 1) + ' ' +
parser.string.substr(parser.i + 1);
}
let primes, base;
const top = parser.stack.Top();
if (top.isKind('prime')) {
// @test Prime on Sub
[base, primes] = top.Peek(2);
parser.stack.Pop();
} else {
base = parser.stack.Prev();
if (!base) {
// @test Empty Base Index
base = parser.create('token', 'mi', {}, '');
}
}
const movesupsub = NodeUtil.getProperty(base, 'movesupsub');
let position = NodeUtil.isType(base, 'msubsup') ?
(base as MmlMsubsup).sub : (base as MmlMunderover).under;
if ((NodeUtil.isType(base, 'msubsup') && !NodeUtil.isType(base, 'msup') &&
NodeUtil.getChildAt(base, (base as MmlMsubsup).sub)) ||
(NodeUtil.isType(base, 'munderover') && !NodeUtil.isType(base, 'mover') &&
NodeUtil.getChildAt(base, (base as MmlMunderover).under) &&
!NodeUtil.getProperty(base, 'subsupOK'))) {
// @test Double-sub-error, Double-under-error
throw new TexError('DoubleSubscripts', 'Double subscripts: use braces to clarify');
}
if (!NodeUtil.isType(base, 'msubsup') || NodeUtil.isType(base, 'msup')) {
if (movesupsub) {
// @test Large Operator, Move Superscript
if (!NodeUtil.isType(base, 'munderover') || NodeUtil.isType(base, 'mover') ||
NodeUtil.getChildAt(base, (base as MmlMunderover).under)) {
// @test Move Superscript
base = parser.create('node', 'munderover', [base], {movesupsub: true});
}
position = (base as MmlMunderover).under;
} else {
// @test Empty Base Index, Empty Base Index2, Index
base = parser.create('node', 'msubsup', [base]);
position = (base as MmlMsubsup).sub;
}
}
parser.Push(
parser.itemFactory.create('subsup', base).setProperties({
position: position, primes: primes, movesupsub: movesupsub
}) );
};
/**
* Handle '
* @param {TexParser} parser The calling parser.
* @param {string} c The parsed character.
*/
BaseMethods.Prime = function(parser: TexParser, c: string) {
// @test Prime
let base = parser.stack.Prev();
if (!base) {
// @test PrimeSup, PrePrime, Prime on Sup
base = parser.create('node', 'mi');
}
if (NodeUtil.isType(base, 'msubsup') && !NodeUtil.isType(base, 'msup') &&
NodeUtil.getChildAt(base, (base as MmlMsubsup).sup)) {
// @test Double Prime Error
throw new TexError('DoubleExponentPrime',
'Prime causes double exponent: use braces to clarify');
}
let sup = '';
parser.i--;
do {
// @test Prime, PrimeSup, Double Prime, PrePrime
sup += entities.prime; parser.i++, c = parser.GetNext();
} while (c === '\'' || c === entities.rsquo);
sup = ['', '\u2032', '\u2033', '\u2034', '\u2057'][sup.length] || sup;
const node = parser.create('token', 'mo', {variantForm: true}, sup);
parser.Push(
parser.itemFactory.create('prime', base, node) );
};
/**
* Handle comments
* @param {TexParser} parser The calling parser.
* @param {string} c The parsed character.
*/
BaseMethods.Comment = function(parser: TexParser, _c: string) {
while (parser.i < parser.string.length && parser.string.charAt(parser.i) !== '\n') {
parser.i++;
}
};
/**
* Handle hash marks outside of definitions
* @param {TexParser} parser The calling parser.
* @param {string} c The parsed character.
*/
BaseMethods.Hash = function(_parser: TexParser, _c: string) {
// @test Hash Error
throw new TexError('CantUseHash1',
'You can\'t use \'macro parameter character #\' in math mode');
};
/**
*
* Handle LaTeX Macros
*
*/
/**
* Handle \mathrm, \mathbf, etc, allowing for multi-letter runs to be one <mi>.
*/
BaseMethods.MathFont = function(parser: TexParser, name: string, variant: string) {
const text = parser.GetArgument(name);
let mml = new TexParser(text, {
...parser.stack.env,
font: variant,
multiLetterIdentifiers: /^[a-zA-Z]+/ as any,
noAutoOP: true
}, parser.configuration).mml();
parser.Push(parser.create('node', 'TeXAtom', [mml]));
};
/**
* Setting font, e.g., via \\rm, \\bf etc.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
* @param {string} font The font name.
*/
BaseMethods.SetFont = function(parser: TexParser, _name: string, font: string) {
parser.stack.env['font'] = font;
};
/**
* Setting style, e.g., via \\displaystyle, \\textstyle, etc.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
* @param {string} texStyle The tex style name: D, T, S, SS
* @param {boolean} style True if we are in displaystyle.
* @param {string} level The nesting level for scripts.
*/
BaseMethods.SetStyle = function(parser: TexParser, _name: string,
texStyle: string, style: boolean,
level: string) {
parser.stack.env['style'] = texStyle;
parser.stack.env['level'] = level;
parser.Push(
parser.itemFactory.create('style').setProperty(
'styles', {displaystyle: style, scriptlevel: level}));
};
/**
* Setting size of an expression, e.g., \\small, \\huge.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
* @param {number} size The size value.
*/
BaseMethods.SetSize = function(parser: TexParser, _name: string, size: number) {
parser.stack.env['size'] = size;
parser.Push(
parser.itemFactory.create('style').setProperty('styles', {mathsize: em(size)}));
};
/**
* Setting explicit spaces, e.g., via commata or colons.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
* @param {string} space The space value.
*/
BaseMethods.Spacer = function(parser: TexParser, _name: string, space: number) {
// @test Positive Spacing, Negative Spacing
const node = parser.create('node', 'mspace', [], {width: em(space)});
const style = parser.create('node', 'mstyle', [node], {scriptlevel: 0});
parser.Push(style);
};
/**
* Parses left/right fenced expressions.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
*/
BaseMethods.LeftRight = function(parser: TexParser, name: string) {
// @test Fenced, Fenced3
const first = name.substr(1);
parser.Push(parser.itemFactory.create(first, parser.GetDelimiter(name), parser.stack.env.color));
};
/**
* Handle a named math function, e.g., \\sin, \\cos
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
* @param {string} id Alternative string representation of the function.
*/
BaseMethods.NamedFn = function(parser: TexParser, name: string, id: string) {
// @test Named Function
if (!id) {
id = name.substr(1);
}
const mml = parser.create('token', 'mi', {texClass: TEXCLASS.OP}, id);
parser.Push(parser.itemFactory.create('fn', mml));
};
/**
* Handle a named math operator, e.g., \\min, \\lim
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
* @param {string} id Alternative string representation of the operator.
*/
BaseMethods.NamedOp = function(parser: TexParser, name: string, id: string) {
// @test Limit
if (!id) {
id = name.substr(1);
}
id = id.replace(/&thinsp;/, '\u2006');
const mml = parser.create('token', 'mo', {
movablelimits: true,
movesupsub: true,
form: TexConstant.Form.PREFIX,
texClass: TEXCLASS.OP
}, id);
parser.Push(mml);
};
/**
* Handle a limits command for math operators.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
* @param {string} limits The limits arguments.
*/
BaseMethods.Limits = function(parser: TexParser, _name: string, limits: string) {
// @test Limits
let op = parser.stack.Prev(true);
// Get the texclass for the core operator.
if (!op || (NodeUtil.getTexClass(NodeUtil.getCoreMO(op)) !== TEXCLASS.OP &&
NodeUtil.getProperty(op, 'movesupsub') == null)) {
// @test Limits Error
throw new TexError('MisplacedLimits', '%1 is allowed only on operators', parser.currentCS);
}
const top = parser.stack.Top();
let node;
if (NodeUtil.isType(op, 'munderover') && !limits) {
// @test Limits UnderOver
node = parser.create('node', 'msubsup');
NodeUtil.copyChildren(op, node);
op = top.Last = node;
} else if (NodeUtil.isType(op, 'msubsup') && limits) {
// @test Limits SubSup
// node = parser.create('node', 'munderover', NodeUtil.getChildren(op), {});
// Needs to be copied, otherwise we get an error in MmlNode.appendChild!
node = parser.create('node', 'munderover');
NodeUtil.copyChildren(op, node);
op = top.Last = node;
}
NodeUtil.setProperty(op, 'movesupsub', limits ? true : false);
NodeUtil.setProperties(NodeUtil.getCoreMO(op), {'movablelimits': false});
if (NodeUtil.getAttribute(op, 'movablelimits') ||
NodeUtil.getProperty(op, 'movablelimits')) {
NodeUtil.setProperties(op, {'movablelimits': false});
}
};
/**
* Handle over commands.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
* @param {string} open The open delimiter in case of a "withdelim" version.
* @param {string} close The close delimiter.
*/
BaseMethods.Over = function(parser: TexParser, name: string, open: string, close: string) {
// @test Over
const mml = parser.itemFactory.create('over').setProperty('name', parser.currentCS) ;
if (open || close) {
// @test Choose
mml.setProperty('open', open);
mml.setProperty('close', close);
} else if (name.match(/withdelims$/)) {
// @test Over With Delims, Above With Delims
mml.setProperty('open', parser.GetDelimiter(name));
mml.setProperty('close', parser.GetDelimiter(name));
}
if (name.match(/^\\above/)) {
// @test Above, Above With Delims
mml.setProperty('thickness', parser.GetDimen(name));
}
else if (name.match(/^\\atop/) || open || close) {
// @test Choose
mml.setProperty('thickness', 0);
}
parser.Push(mml);
};
/**
* Parses a fraction.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
*/
BaseMethods.Frac = function(parser: TexParser, name: string) {
// @test Frac
const num = parser.ParseArg(name);
const den = parser.ParseArg(name);
const node = parser.create('node', 'mfrac', [num, den]);
parser.Push(node);
};
/**
* Parses a square root element.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
*/
BaseMethods.Sqrt = function(parser: TexParser, name: string) {
const n = parser.GetBrackets(name);
let arg = parser.GetArgument(name);
if (arg === '\\frac') {
arg += '{' + parser.GetArgument(arg) + '}{' + parser.GetArgument(arg) + '}';
}
let mml = new TexParser(arg, parser.stack.env, parser.configuration).mml();
if (!n) {
// @test Square Root
mml = parser.create('node', 'msqrt', [mml]);
} else {
// @test General Root
mml = parser.create('node', 'mroot', [mml, parseRoot(parser, n)]);
}
parser.Push(mml);
};
// Utility
/**
* Parse a general root.
* @param {TexParser} parser The calling parser.
* @param {string} n The index of the root.
*/
function parseRoot(parser: TexParser, n: string) {
// @test General Root, Explicit Root
const env = parser.stack.env;
const inRoot = env['inRoot'];
env['inRoot'] = true;
const newParser = new TexParser(n, env, parser.configuration);
let node = newParser.mml();
const global = newParser.stack.global;
if (global['leftRoot'] || global['upRoot']) {
// @test Tweaked Root
const def: EnvList = {};
if (global['leftRoot']) {
def['width'] = global['leftRoot'];
}
if (global['upRoot']) {
def['voffset'] = global['upRoot'];
def['height'] = global['upRoot'];
}
node = parser.create('node', 'mpadded', [node], def);
}
env['inRoot'] = inRoot;
return node;
}
/**
* Parse a general root.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
*/
BaseMethods.Root = function(parser: TexParser, name: string) {
const n = parser.GetUpTo(name, '\\of');
const arg = parser.ParseArg(name);
const node = parser.create('node', 'mroot', [arg, parseRoot(parser, n)]);
parser.Push(node);
};
/**
* Parses a movable index element in a root, e.g. \\uproot, \\leftroot
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
* @param {string} id Argument which should be a string representation of an integer.
*/
BaseMethods.MoveRoot = function(parser: TexParser, name: string, id: string) {
// @test Tweaked Root
if (!parser.stack.env['inRoot']) {
// @test Misplaced Move Root
throw new TexError('MisplacedMoveRoot', '%1 can appear only within a root', parser.currentCS);
}
if (parser.stack.global[id]) {
// @test Multiple Move Root
throw new TexError('MultipleMoveRoot', 'Multiple use of %1', parser.currentCS);
}
let n = parser.GetArgument(name);
if (!n.match(/-?[0-9]+/)) {
// @test Incorrect Move Root
throw new TexError('IntegerArg', 'The argument to %1 must be an integer', parser.currentCS);
}
n = (parseInt(n, 10) / 15) + 'em';
if (n.substr(0, 1) !== '-') {
n = '+' + n;
}
parser.stack.global[id] = n;
};
/**
* Handle accents.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
* @param {string} accent The accent.
* @param {boolean} stretchy True if accent is stretchy.
*/
BaseMethods.Accent = function(parser: TexParser, name: string, accent: string, stretchy: boolean) {
// @test Vector
const c = parser.ParseArg(name);
// @test Vector Font
const def = {...ParseUtil.getFontDef(parser), accent: true, mathaccent: true};
const entity = NodeUtil.createEntity(accent);
const moNode = parser.create('token', 'mo', def, entity);
const mml = moNode;
NodeUtil.setAttribute(mml, 'stretchy', stretchy ? true : false);
// @test Vector Op, Vector
const mo = (NodeUtil.isEmbellished(c) ? NodeUtil.getCoreMO(c) : c);
if (NodeUtil.isType(mo, 'mo') || NodeUtil.getProperty(mo, 'movablelimits')) {
// @test Vector Op
NodeUtil.setProperties(mo, {'movablelimits': false});
}
const muoNode = parser.create('node', 'munderover');
// This is necessary to get the empty element into the children.
NodeUtil.setChild(muoNode, 0, c);
NodeUtil.setChild(muoNode, 1, null);
NodeUtil.setChild(muoNode, 2, mml);
let texAtom = parser.create('node', 'TeXAtom', [muoNode]);
parser.Push(texAtom);
};
/**
* Handles stacked elements.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
* @param {string} c Character to stack.
* @param {boolean} stack True if stacked operator.
*/
BaseMethods.UnderOver = function(parser: TexParser, name: string, c: string, stack: boolean) {
const entity = NodeUtil.createEntity(c);
const mo = parser.create('token', 'mo', {stretchy: true, accent: true}, entity);
const pos = (name.charAt(1) === 'o' ? 'over' : 'under');
const base = parser.ParseArg(name);
parser.Push(ParseUtil.underOver(parser, base, mo, pos, stack));
};
/**
* Handles overset.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
*/
BaseMethods.Overset = function(parser: TexParser, name: string) {
// @test Overset
const top = parser.ParseArg(name);
const base = parser.ParseArg(name);
ParseUtil.checkMovableLimits(base);
if (top.isKind('mo')) {
NodeUtil.setAttribute(top, 'accent', false);
}
const node = parser.create('node', 'mover', [base, top]);
parser.Push(node);
};
/**
* Handles underset.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
*/
BaseMethods.Underset = function(parser: TexParser, name: string) {
// @test Underset
const bot = parser.ParseArg(name);
const base = parser.ParseArg(name);
ParseUtil.checkMovableLimits(base);
if (bot.isKind('mo')) {
NodeUtil.setAttribute(bot, 'accent', false);
}
const node = parser.create('node', 'munder', [base, bot], {accentunder: false});
parser.Push(node);
};
/**
* Handles overunderset.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
*/
BaseMethods.Overunderset = function(parser: TexParser, name: string) {
const top = parser.ParseArg(name);
const bot = parser.ParseArg(name);
const base = parser.ParseArg(name);
ParseUtil.checkMovableLimits(base);
if (top.isKind('mo')) {
NodeUtil.setAttribute(top, 'accent', false);
}
if (bot.isKind('mo')) {
NodeUtil.setAttribute(bot, 'accent', false);
}
const node = parser.create('node', 'munderover', [base, bot, top], {accent: false, accentunder: false});
parser.Push(node);
};
/**
* Creates TeXAtom, when class of element is changed explicitly.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
* @param {number} mclass The new TeX class.
*/
BaseMethods.TeXAtom = function(parser: TexParser, name: string, mclass: number) {
let def: EnvList = {texClass: mclass};
let mml: StackItem | MmlNode;
let node: MmlNode;
let parsed: MmlNode;
if (mclass === TEXCLASS.OP) {
def['movesupsub'] = def['movablelimits'] = true;
const arg = parser.GetArgument(name);
const match = arg.match(/^\s*\\rm\s+([a-zA-Z0-9 ]+)$/);
if (match) {
// @test Mathop
def['mathvariant'] = TexConstant.Variant.NORMAL;
node = parser.create('token', 'mi', def, match[1]);
} else {
// @test Mathop Cal
parsed = new TexParser(arg, parser.stack.env, parser.configuration).mml();
node = parser.create('node', 'TeXAtom', [parsed], def);
}
mml = parser.itemFactory.create('fn', node);
} else {
// @test Mathrel
parsed = parser.ParseArg(name);
mml = parser.create('node', 'TeXAtom', [parsed], def);
}
parser.Push(mml);
};
/**
* Creates mmltoken elements. Used in Macro substitutions.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
*/
BaseMethods.MmlToken = function(parser: TexParser, name: string) {
// @test Modulo
const kind = parser.GetArgument(name);
let attr = parser.GetBrackets(name, '').replace(/^\s+/, '');
const text = parser.GetArgument(name);
const def: EnvList = {};
const keep: string[] = [];
let node: MmlNode;
try {
node = parser.create('node', kind);
} catch (e) {
node = null;
}
if (!node || !node.isToken) {
// @test Token Illegal Type, Token Wrong Type
throw new TexError('NotMathMLToken', '%1 is not a token element', kind);
}
while (attr !== '') {
const match = attr.match(/^([a-z]+)\s*=\s*('[^']*'|"[^"]*"|[^ ,]*)\s*,?\s*/i);
if (!match) {
// @test Token Invalid Attribute
throw new TexError('InvalidMathMLAttr', 'Invalid MathML attribute: %1', attr);
}
if (!node.attributes.hasDefault(match[1]) && !MmlTokenAllow[match[1]]) {
// @test Token Unknown Attribute, Token Wrong Attribute
throw new TexError('UnknownAttrForElement',
'%1 is not a recognized attribute for %2',
match[1], kind);
}
let value: string | boolean = ParseUtil.MmlFilterAttribute(
parser, match[1], match[2].replace(/^(['"])(.*)\1$/, '$2'));
if (value) {
if (value.toLowerCase() === 'true') {
value = true;
}
else if (value.toLowerCase() === 'false') {
value = false;
}
def[match[1]] = value;
keep.push(match[1]);
}
attr = attr.substr(match[0].length);
}
if (keep.length) {
def['mjx-keep-attrs'] = keep.join(' ');
}
const textNode = parser.create('text', text);
node.appendChild(textNode);
NodeUtil.setProperties(node, def);
parser.Push(node);
};
/**
* Handle strut.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
*/
BaseMethods.Strut = function(parser: TexParser, _name: string) {
// @test Strut
const row = parser.create('node', 'mrow');
const padded = parser.create('node', 'mpadded', [row],
{height: '8.6pt', depth: '3pt', width: 0});
parser.Push(padded);
};
/**
* Handle phantom commands.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
* @param {string} v Vertical size.
* @param {string} h Horizontal size.
*/
BaseMethods.Phantom = function(parser: TexParser, name: string, v: string, h: string) {
// @test Phantom
let box = parser.create('node', 'mphantom', [parser.ParseArg(name)]);
if (v || h) {
// TEMP: Changes here
box = parser.create('node', 'mpadded', [box]);
if (h) {
// @test Horizontal Phantom
NodeUtil.setAttribute(box, 'height', 0);
NodeUtil.setAttribute(box, 'depth', 0);
}
if (v) {
// @test Vertical Phantom
NodeUtil.setAttribute(box, 'width', 0);
}
}
const atom = parser.create('node', 'TeXAtom', [box]);
parser.Push(atom);
};
/**
* Handle smash.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
*/
BaseMethods.Smash = function(parser: TexParser, name: string) {
// @test Smash, Smash Top, Smash Bottom
const bt = ParseUtil.trimSpaces(parser.GetBrackets(name, ''));
const smash = parser.create('node', 'mpadded', [parser.ParseArg(name)]);
// TEMP: Changes here:
switch (bt) {
case 'b': NodeUtil.setAttribute(smash, 'depth', 0); break;
case 't': NodeUtil.setAttribute(smash, 'height', 0); break;
default:
NodeUtil.setAttribute(smash, 'height', 0);
NodeUtil.setAttribute(smash, 'depth', 0);
}
const atom = parser.create('node', 'TeXAtom', [smash]);
parser.Push(atom);
};
/**
* Handle rlap and llap commands.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
*/
BaseMethods.Lap = function(parser: TexParser, name: string) {
// @test Llap, Rlap
const mml = parser.create('node', 'mpadded', [parser.ParseArg(name)], {width: 0});
if (name === '\\llap') {
// @test Llap
NodeUtil.setAttribute(mml, 'lspace', '-1width');
}
const atom = parser.create('node', 'TeXAtom', [mml]);
parser.Push(atom);
};
/**
* Handle raise and lower commands.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
*/
BaseMethods.RaiseLower = function(parser: TexParser, name: string) {
// @test Raise, Lower, Raise Negative, Lower Negative
let h = parser.GetDimen(name);
let item =
parser.itemFactory.create('position').setProperties({name: parser.currentCS, move: 'vertical'}) ;
// TEMP: Changes here:
if (h.charAt(0) === '-') {
// @test Raise Negative, Lower Negative
h = h.slice(1);
name = name.substr(1) === 'raise' ? '\\lower' : '\\raise';
}
if (name === '\\lower') {
// @test Raise, Raise Negative
item.setProperty('dh', '-' + h);
item.setProperty('dd', '+' + h);
} else {
// @test Lower, Lower Negative
item.setProperty('dh', '+' + h);
item.setProperty('dd', '-' + h);
}
parser.Push(item);
};
/**
* Handle moveleft, moveright commands
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
*/
BaseMethods.MoveLeftRight = function(parser: TexParser, name: string) {
// @test Move Left, Move Right, Move Left Negative, Move Right Negative
let h = parser.GetDimen(name);
let nh = (h.charAt(0) === '-' ? h.slice(1) : '-' + h);
if (name === '\\moveleft') {
let tmp = h;
h = nh;
nh = tmp;
}
parser.Push(
parser.itemFactory.create('position').setProperties({
name: parser.currentCS, move: 'horizontal',
left: parser.create('node', 'mspace', [], {width: h}),
right: parser.create('node', 'mspace', [], {width: nh})}) );
};
/**
* Handle horizontal spacing commands.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
*/
BaseMethods.Hskip = function(parser: TexParser, name: string) {
// @test Modulo
const node = parser.create('node', 'mspace', [],
{width: parser.GetDimen(name)});
parser.Push(node);
};
/**
* Handle removal of spaces in script modes
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
*/
BaseMethods.Nonscript = function(parser: TexParser, _name: string) {
parser.Push(parser.itemFactory.create('nonscript'));
};
/**
* Handle Rule and Space command
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
* @param {string} style The style of the rule spacer.
*/
BaseMethods.Rule = function(parser: TexParser, name: string, style: string) {
// @test Rule 3D, Space 3D
const w = parser.GetDimen(name),
h = parser.GetDimen(name),
d = parser.GetDimen(name);
let def: EnvList = {width: w, height: h, depth: d};
if (style !== 'blank') {
def['mathbackground'] = (parser.stack.env['color'] || 'black');
}
const node = parser.create('node', 'mspace', [], def);
parser.Push(node);
};
/**
* Handle rule command.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
*/
BaseMethods.rule = function(parser: TexParser, name: string) {
// @test Rule 2D
const v = parser.GetBrackets(name),
w = parser.GetDimen(name),
h = parser.GetDimen(name);
let mml = parser.create('node', 'mspace', [], {
width: w, height: h,
mathbackground: (parser.stack.env['color'] || 'black') });
if (v) {
mml = parser.create('node', 'mpadded', [mml], {voffset: v});
if (v.match(/^\-/)) {
NodeUtil.setAttribute(mml, 'height', v);
NodeUtil.setAttribute(mml, 'depth', '+' + v.substr(1));
} else {
NodeUtil.setAttribute(mml, 'height', '+' + v);
}
}
parser.Push(mml);
};
/**
* Handle big command sequences, e.g., \\big, \\Bigg.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
* @param {number} mclass The TeX class of the element.
* @param {number} size The em size.
*/
BaseMethods.MakeBig = function(parser: TexParser, name: string, mclass: number, size: number) {
// @test Choose, Over With Delims, Above With Delims
size *= P_HEIGHT;
let sizeStr = String(size).replace(/(\.\d\d\d).+/, '$1') + 'em';
const delim = parser.GetDelimiter(name, true);
const mo = parser.create('token', 'mo', {
minsize: sizeStr, maxsize: sizeStr,
fence: true, stretchy: true, symmetric: true
}, delim);
const node = parser.create('node', 'TeXAtom', [mo], {texClass: mclass});
parser.Push(node);
};
/**
* Handle buildrel command.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
*/
BaseMethods.BuildRel = function(parser: TexParser, name: string) {
// @test BuildRel, BuildRel Expression
const top = parser.ParseUpTo(name, '\\over');
const bot = parser.ParseArg(name);
const node = parser.create('node', 'munderover');
// This is necessary to get the empty element into the children.
NodeUtil.setChild(node, 0, bot);
NodeUtil.setChild(node, 1, null);
NodeUtil.setChild(node, 2, top);
const atom = parser.create('node', 'TeXAtom', [node], {texClass: TEXCLASS.REL});
parser.Push(atom);
};
/**
* Handle horizontal boxes.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
* @param {string} style Box style.
* @param {string} font The mathvariant to use
*/
BaseMethods.HBox = function(parser: TexParser, name: string, style: string, font?: string) {
// @test Hbox
parser.PushAll(ParseUtil.internalMath(parser, parser.GetArgument(name), style, font));
};
/**
* Handle framed boxes.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
*/
BaseMethods.FBox = function(parser: TexParser, name: string) {
// @test Fbox
const internal = ParseUtil.internalMath(parser, parser.GetArgument(name));
const node = parser.create('node', 'menclose', internal, {notation: 'box'});
parser.Push(node);
};
/**
* Handle framed boxes with options.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
*/
BaseMethods.FrameBox = function(parser: TexParser, name: string) {
const width = parser.GetBrackets(name);
const pos = parser.GetBrackets(name) || 'c';
let mml = ParseUtil.internalMath(parser, parser.GetArgument(name));
if (width) {
mml = [parser.create('node', 'mpadded', mml, {
width,
'data-align': lookup(pos, {l: 'left', r: 'right'}, 'center')
})];
}
const node = parser.create('node', 'TeXAtom',
[parser.create('node', 'menclose', mml, {notation: 'box'})],
{texClass: TEXCLASS.ORD});
parser.Push(node);
};
/**
* Handle \\not.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
*/
BaseMethods.Not = function(parser: TexParser, _name: string) {
// @test Negation Simple, Negation Complex, Negation Explicit,
// Negation Large
parser.Push(parser.itemFactory.create('not'));
};
/**
* Handle dots.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
*/
BaseMethods.Dots = function(parser: TexParser, _name: string) {
// @test Operator Dots
const ldotsEntity = NodeUtil.createEntity('2026');
const cdotsEntity = NodeUtil.createEntity('22EF');
const ldots = parser.create('token', 'mo', {stretchy: false}, ldotsEntity);
const cdots = parser.create('token', 'mo', {stretchy: false}, cdotsEntity);
parser.Push(
parser.itemFactory.create('dots').setProperties({
ldots: ldots,
cdots: cdots
}) );
};
/**
* Handle small matrix environments.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
* @param {string} open Opening fence.
* @param {string} close Closing fence.
* @param {string} align Column alignment.
* @param {string} spacing Column spacing.
* @param {string} vspacing Row spacing.
* @param {string} style Display or text style.
* @param {boolean} cases Is it a cases environment.
* @param {boolean} numbered Is it a numbered environment.
*/
BaseMethods.Matrix = function(parser: TexParser, _name: string,
open: string, close: string, align: string,
spacing: string, vspacing: string, style: string,
cases: boolean, numbered: boolean) {
const c = parser.GetNext();
if (c === '') {
// @test Matrix Error
throw new TexError('MissingArgFor', 'Missing argument for %1', parser.currentCS);
}
if (c === '{') {
// @test Matrix Braces, Matrix Columns, Matrix Rows.
parser.i++;
} else {
// @test Matrix Arg
parser.string = c + '}' + parser.string.slice(parser.i + 1);
parser.i = 0;
}
// @test Matrix Braces, Matrix Columns, Matrix Rows.
const array = parser.itemFactory.create('array').setProperty('requireClose', true) as sitem.ArrayItem;
array.arraydef = {
rowspacing: (vspacing || '4pt'),
columnspacing: (spacing || '1em')
};
if (cases) {
// @test Matrix Cases
array.setProperty('isCases', true);
}
if (numbered) {
// @test Matrix Numbered
array.setProperty('isNumbered', true);
array.arraydef.side = numbered;
}
if (open || close) {
// @test Matrix Parens, Matrix Parens Subscript, Matrix Cases
array.setProperty('open', open);
array.setProperty('close', close);
}
if (style === 'D') {
// @test Matrix Numbered
array.arraydef.displaystyle = true;
}
if (align != null) {
// @test Matrix Cases, Matrix Numbered
array.arraydef.columnalign = align;
}
parser.Push(array);
};
/**
* Handle array entry.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
*/
BaseMethods.Entry = function(parser: TexParser, name: string) {
// @test Label, Array, Cross Product Formula
parser.Push(parser.itemFactory.create('cell').setProperties({isEntry: true, name: name}));
const top = parser.stack.Top();
const env = top.getProperty('casesEnv') as string;
const cases = top.getProperty('isCases');
if (!cases && !env) return;
//
// Make second column be in \text{...} (unless it is already
// in a \text{...}, for backward compatibility).
//
const str = parser.string;
let braces = 0, close = -1, i = parser.i, m = str.length;
const end = (env ? new RegExp(`^\\\\end\\s*\\{${env.replace(/\*/, '\\*')}\\}`) : null);
//
// Look through the string character by character...
//
while (i < m) {
const c = str.charAt(i);
if (c === '{') {
//
// Increase the nested brace count and go on
//
braces++;
i++;
} else if (c === '}') {
//
// If there are too many close braces, just end (we will get an
// error message later when the rest of the string is parsed)
// Otherwise
// decrease the nested brace count,
// if it is now zero and we haven't already marked the end of the
// first brace group, record the position (use to check for \text{} later)
// go on to the next character.
//
if (braces === 0) {
m = 0;
} else {
braces--;
if (braces === 0 && close < 0) {
close = i - parser.i;
}
i++;
}
} else if (c === '&' && braces === 0) {
//
// Extra alignment tabs are not allowed in cases
//
// @test ExtraAlignTab
throw new TexError('ExtraAlignTab', 'Extra alignment tab in \\cases text');
} else if (c === '\\') {
//
// If the macro is \cr or \\, end the search, otherwise skip the macro
// (multi-letter names don't matter, as we will skip the rest of the
// characters in the main loop)
//
const rest = str.substr(i);
if (rest.match(/^((\\cr)[^a-zA-Z]|\\\\)/) || (end && rest.match(end))) {
m = 0;
} else {
i += 2;
}
} else {
//
// Go on to the next character
//
i++;
}
}
//
// Check if the second column text is already in \text{};
// If not, process the second column as text and continue parsing from there,
// (otherwise process the second column as normal, since it is in \text{}
//
const text = str.substr(parser.i, i - parser.i);
if (!text.match(/^\s*\\text[^a-zA-Z]/) || close !== text.replace(/\s+$/, '').length - 1) {
const internal = ParseUtil.internalMath(parser, ParseUtil.trimSpaces(text), 0);
parser.PushAll(internal);
parser.i = i;
}
};
/**
* Handle newline in array.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
*/
BaseMethods.Cr = function(parser: TexParser, name: string) {
// @test Cr Linebreak, Misplaced Cr
parser.Push(
parser.itemFactory.create('cell').setProperties({isCR: true, name: name}));
};
/**
* Handle newline outside array.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
* @param {boolean} nobrackets Flag indicating if newline is followed by
* brackets.
*/
BaseMethods.CrLaTeX = function(parser: TexParser, name: string, nobrackets: boolean = false) {
let n: string;
if (!nobrackets) {
// TODO: spaces before * and [ are not allowed in AMS environments like align, but
// should be allowed in array and eqnarray. This distinction should be honored here.
if (parser.string.charAt(parser.i) === '*') { // The * controls page breaking, so ignore it
parser.i++;
}
if (parser.string.charAt(parser.i) === '[') {
let dim = parser.GetBrackets(name, '');
let [value, unit, ] = ParseUtil.matchDimen(dim);
// @test Custom Linebreak
if (dim && !value) {
// @test Dimension Error
throw new TexError('BracketMustBeDimension',
'Bracket argument to %1 must be a dimension', parser.currentCS);
}
n = value + unit;
}
}
parser.Push(
parser.itemFactory.create('cell').setProperties({isCR: true, name: name, linebreak: true})
);
const top = parser.stack.Top();
let node: MmlNode;
if (top instanceof sitem.ArrayItem) {
// @test Array
if (n) {
top.addRowSpacing(n);
}
} else {
if (n) {
// @test Custom Linebreak
node = parser.create('node', 'mspace', [], {depth: n});
parser.Push(node);
}
// @test Linebreak
node = parser.create('node', 'mspace', [], {linebreak: TexConstant.LineBreak.NEWLINE});
parser.Push(node);
}
};
/**
* Handle horizontal lines in arrays.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
* @param {string} style Style of the line. E.g., dashed.
*/
BaseMethods.HLine = function(parser: TexParser, _name: string, style: string) {
if (style == null) {
style = 'solid';
}
const top = parser.stack.Top();
if (!(top instanceof sitem.ArrayItem) || top.Size()) {
// @test Misplaced hline
throw new TexError('Misplaced', 'Misplaced %1', parser.currentCS);
}
if (!top.table.length) {
// @test Enclosed top, Enclosed top bottom
top.frame.push('top');
} else {
// @test Enclosed bottom, Enclosed top bottom
const lines = (top.arraydef['rowlines'] ? (top.arraydef['rowlines'] as string).split(/ /) : []);
while (lines.length < top.table.length) {
lines.push('none');
}
lines[top.table.length - 1] = style;
top.arraydef['rowlines'] = lines.join(' ');
}
};
/**
* Handle hfill commands.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
*/
BaseMethods.HFill = function(parser: TexParser, _name: string) {
const top = parser.stack.Top();
if (top instanceof sitem.ArrayItem) {
// @test Hfill
top.hfill.push(top.Size());
} else {
// @test UnsupportedHFill
throw new TexError('UnsupportedHFill', 'Unsupported use of %1', parser.currentCS);
}
};
/**
* LaTeX environments
*/
/**
* Handle begin and end environments. This is a macro method.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
*/
BaseMethods.BeginEnd = function(parser: TexParser, name: string) {
// @test Array1, Array2, Array Test
let env = parser.GetArgument(name);
if (env.match(/\\/i)) {
// @test InvalidEnv
throw new TexError('InvalidEnv', 'Invalid environment name \'%1\'', env);
}
let macro = parser.configuration.handlers.get('environment').lookup(env) as Macro;
if (macro && name === '\\end') {
// If the first argument is true, we have some sort of user defined
// environment. Otherwise we have a standard LaTeX environment that is
// handled with begin and end items.
if (!macro.args[0]) {
const mml = parser.itemFactory.create('end').setProperty('name', env);
parser.Push(mml);
return;
}
// Remember the user defined environment we are closing.
parser.stack.env['closing'] = env;
}
ParseUtil.checkMaxMacros(parser, false);
parser.parse('environment', [parser, env]);
};
/**
* Handle array environment.
* @param {TexParser} parser The calling parser.
* @param {StackItem} begin The opening stackitem.
* @param {string} open Opening fence.
* @param {string} close Closing fence.
* @param {string} align Column alignment.
* @param {string} spacing Column spacing.
* @param {string} vspacing Row spacing.
* @param {string} style Display or text style.
* @param {boolean} raggedHeight Does the height need to be adjusted?
*/
BaseMethods.Array = function(parser: TexParser, begin: StackItem,
open: string, close: string, align: string,
spacing: string, vspacing: string, style: string,
raggedHeight: boolean) {
if (!align) {
// @test Array Single
align = parser.GetArgument('\\begin{' + begin.getName() + '}');
}
let lines = ('c' + align).replace(/[^clr|:]/g, '').replace(/[^|:]([|:])+/g, '$1');
align = align.replace(/[^clr]/g, '').split('').join(' ');
align = align.replace(/l/g, 'left').replace(/r/g, 'right').replace(/c/g, 'center');
const array = parser.itemFactory.create('array') as sitem.ArrayItem;
array.arraydef = {
columnalign: align,
columnspacing: (spacing || '1em'),
rowspacing: (vspacing || '4pt')
};
if (lines.match(/[|:]/)) {
// @test Enclosed left right
if (lines.charAt(0).match(/[|:]/)) {
// @test Enclosed left right, Enclosed left
array.frame.push('left');
array.dashed = lines.charAt(0) === ':';
}
if (lines.charAt(lines.length - 1).match(/[|:]/)) {
// @test Enclosed left right, Enclosed right
array.frame.push('right');
}
// @test Enclosed left right
lines = lines.substr(1, lines.length - 2);
array.arraydef.columnlines =
lines.split('').join(' ').replace(/[^|: ]/g, 'none').replace(/\|/g, 'solid').replace(/:/g, 'dashed');
}
if (open) {
// @test Cross Product
array.setProperty('open', parser.convertDelimiter(open));
}
if (close) {
// @test Cross Product
array.setProperty('close', parser.convertDelimiter(close));
}
if ((style || '').charAt(1) === '\'') {
array.arraydef['data-cramped'] = true;
style = style.charAt(0);
}
if (style === 'D') {
// TODO: This case never seems to occur! No test.
array.arraydef['displaystyle'] = true;
}
else if (style) {
// @test Subarray, Small Matrix
array.arraydef['displaystyle'] = false;
}
if (style === 'S') {
// @test Subarray, Small Matrix
array.arraydef['scriptlevel'] = 1;
}
if (raggedHeight) {
// @test Subarray, Small Matrix
array.arraydef['useHeight'] = false;
}
parser.Push(begin);
return array;
};
/**
* Handle aligned arrays.
* @param {TexParser} parser The calling parser.
* @param {StackItem} begin The opening stackitem.
*/
BaseMethods.AlignedArray = function(parser: TexParser, begin: StackItem) {
// @test Array1, Array2, Array Test
const align = parser.GetBrackets('\\begin{' + begin.getName() + '}');
let item = BaseMethods.Array(parser, begin);
return ParseUtil.setArrayAlign(item as sitem.ArrayItem, align);
};
/**
* Handle equation environment.
* @param {TexParser} parser The calling parser.
* @param {StackItem} begin The opening stackitem.
* @param {boolean} numbered True if environment is numbered.
*/
BaseMethods.Equation = function (parser: TexParser, begin: StackItem, numbered: boolean) {
parser.Push(begin);
ParseUtil.checkEqnEnv(parser);
return parser.itemFactory.create('equation', numbered).
setProperty('name', begin.getName());
};
/**
* Handle eqnarray.
* @param {TexParser} parser The calling parser.
* @param {StackItem} begin The opening stackitem.
* @param {boolean} numbered True if environment is numbered.
* @param {boolean} taggable True if taggable.
* @param {string} align Alignment string.
* @param {string} spacing Spacing between columns.
*/
BaseMethods.EqnArray = function(parser: TexParser, begin: StackItem,
numbered: boolean, taggable: boolean,
align: string, spacing: string) {
// @test The Lorenz Equations, Maxwell's Equations, Cubic Binomial
parser.Push(begin);
if (taggable) {
ParseUtil.checkEqnEnv(parser);
}
align = align.replace(/[^clr]/g, '').split('').join(' ');
align = align.replace(/l/g, 'left').replace(/r/g, 'right').replace(/c/g, 'center');
let newItem = parser.itemFactory.create('eqnarray', begin.getName(),
numbered, taggable, parser.stack.global) as sitem.ArrayItem;
newItem.arraydef = {
displaystyle: true,
columnalign: align,
columnspacing: (spacing || '1em'),
rowspacing: '3pt',
side: parser.options['tagSide'],
minlabelspacing: parser.options['tagIndent']
};
return newItem;
};
/**
* Handles no tag commands.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
*/
BaseMethods.HandleNoTag = function(parser: TexParser, _name: string) {
parser.tags.notag();
};
/**
* Record a label name for a tag
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
*/
BaseMethods.HandleLabel = function(parser: TexParser, name: string) {
// @test Label, Label Empty
let label = parser.GetArgument(name);
if (label === '') {
// @test Label Empty
return;
}
if (!parser.tags.refUpdate) {
// @test Label, Ref, Ref Unknown
if (parser.tags.label) {
// @test Double Label Error
throw new TexError('MultipleCommand', 'Multiple %1', parser.currentCS);
}
parser.tags.label = label;
if ((parser.tags.allLabels[label] || parser.tags.labels[label]) && !parser.options['ignoreDuplicateLabels']) {
// @ Duplicate Label Error
throw new TexError('MultipleLabel', 'Label \'%1\' multiply defined', label);
}
// TODO: This should be set in the tags structure!
parser.tags.labels[label] = new Label(); // will be replaced by tag value later
}
};
/**
* Handle a label reference.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
* @param {boolean} eqref True if formatted as eqref.
*/
BaseMethods.HandleRef = function(parser: TexParser, name: string, eqref: boolean) {
// @test Ref, Ref Unknown, Eqref, Ref Default, Ref Named
let label = parser.GetArgument(name);
let ref = parser.tags.allLabels[label] || parser.tags.labels[label];
if (!ref) {
// @test Ref Unknown
if (!parser.tags.refUpdate) {
parser.tags.redo = true;
}
ref = new Label();
}
let tag = ref.tag;
if (eqref) {
// @test Eqref
tag = parser.tags.formatTag(tag);
}
let node = parser.create('node', 'mrow', ParseUtil.internalMath(parser, tag), {
href: parser.tags.formatUrl(ref.id, parser.options.baseURL), 'class': 'MathJax_ref'
});
parser.Push(node);
};
/**
* Macros
*/
BaseMethods.Macro = function(parser: TexParser, name: string,
macro: string, argcount: number,
def?: string) {
if (argcount) {
const args: string[] = [];
if (def != null) {
const optional = parser.GetBrackets(name);
args.push(optional == null ? def : optional);
}
for (let i = args.length; i < argcount; i++) {
args.push(parser.GetArgument(name));
}
macro = ParseUtil.substituteArgs(parser, args, macro);
}
parser.string = ParseUtil.addArgs(parser, macro, parser.string.slice(parser.i));
parser.i = 0;
ParseUtil.checkMaxMacros(parser);
};
/**
* Handle MathChoice for elements whose exact size/style properties can only be
* determined after the expression has been parsed.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
*/
BaseMethods.MathChoice = function(parser: TexParser, name: string) {
const D = parser.ParseArg(name);
const T = parser.ParseArg(name);
const S = parser.ParseArg(name);
const SS = parser.ParseArg(name);
parser.Push(parser.create('node', 'MathChoice', [D, T, S, SS]));
};
export default BaseMethods;