site/node_modules/mathjax-full/ts/input/tex/amscd/AmsCdMethods.ts

202 lines
6.3 KiB
TypeScript
Raw Permalink Normal View History

2024-10-14 06:09:33 +00: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 Methods for the AMScd package.
*
* @author v.sorge@mathjax.org (Volker Sorge)
*/
import TexParser from '../TexParser.js';
import {ParseMethod} from '../Types.js';
import {StackItem, EnvList} from '../StackItem.js';
import {ArrayItem} from '../base/BaseItems.js';
import {Other} from '../base/BaseConfiguration.js';
import {MmlMunderover} from '../../../core/MmlTree/MmlNodes/munderover.js';
import {TEXCLASS} from '../../../core/MmlTree/MmlNode.js';
import NodeUtil from '../NodeUtil.js';
// Namespace
let AmsCdMethods: Record<string, ParseMethod> = {};
/**
* Handles CD environment for commutative diagrams.
* @param {TexParser} parser The calling parser.
* @param {StackItem} begin The opening stackitem.
*/
AmsCdMethods.CD = function(parser: TexParser, begin: StackItem) {
parser.Push(begin);
let item = parser.itemFactory.create('array') as ArrayItem;
let options = parser.configuration.options.amscd;
item.setProperties({
minw: parser.stack.env.CD_minw || options.harrowsize,
minh: parser.stack.env.CD_minh || options.varrowsize
});
item.arraydef = {
columnalign: 'center',
columnspacing: options.colspace,
rowspacing: options.rowspace,
displaystyle: true
};
return item;
};
/**
* Converts arrows.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
*/
AmsCdMethods.arrow = function(parser: TexParser, name: string) {
let c = parser.string.charAt(parser.i);
if (!c.match(/[><VA.|=]/)) {
return Other(parser, name);
} else {
parser.i++;
}
let first = parser.stack.Top();
if (!first.isKind('array') || first.Size()) {
AmsCdMethods.cell(parser, name);
first = parser.stack.Top();
}
let top = first as ArrayItem;
//
// Add enough cells to place the arrow correctly
//
let arrowRow = ((top.table.length % 2) === 1);
let n = (top.row.length + (arrowRow ? 0 : 1)) % 2;
while (n) {
AmsCdMethods.cell(parser, name);
n--;
}
let mml;
let hdef = {minsize: top.getProperty('minw'), stretchy: true},
vdef = {minsize: top.getProperty('minh'),
stretchy: true, symmetric: true, lspace: 0, rspace: 0};
if (c === '.') {
} else if (c === '|') {
mml = parser.create('token', 'mo', vdef, '\u2225');
} else if (c === '=') {
mml = parser.create('token', 'mo', hdef, '=');
} else {
//
// for @>>> @<<< @VVV and @AAA, get the arrow and labels
//
// TODO: cleanup!
let arrow: string = ({
'>': '\u2192', '<': '\u2190', 'V': '\u2193', 'A': '\u2191'} as {[key: string]: string}) [c];
let a = parser.GetUpTo(name + c, c);
let b = parser.GetUpTo(name + c, c);
if (c === '>' || c === '<') {
//
// Lay out horizontal arrows with munderover if it has labels
//
mml = parser.create('token', 'mo', hdef, arrow);
if (!a) {
a = '\\kern ' + top.getProperty('minw');
} // minsize needs work
if (a || b) {
let pad: EnvList = {width: '+.67em', lspace: '.33em'};
mml = parser.create('node', 'munderover', [mml]) as MmlMunderover;
if (a) {
let nodeA = new TexParser(a, parser.stack.env, parser.configuration).mml();
let mpadded = parser.create('node', 'mpadded', [nodeA], pad);
NodeUtil.setAttribute(mpadded, 'voffset', '.1em');
NodeUtil.setChild(mml, mml.over, mpadded);
}
if (b) {
let nodeB = new TexParser(b, parser.stack.env, parser.configuration).mml();
NodeUtil.setChild(mml, mml.under, parser.create('node', 'mpadded', [nodeB], pad));
}
if (parser.configuration.options.amscd.hideHorizontalLabels) {
mml = parser.create('node', 'mpadded', mml, {depth: 0, height: '.67em'});
}
}
} else {
//
// Lay out vertical arrows with mrow if there are labels
//
let arrowNode = parser.create('token', 'mo', vdef, arrow);
mml = arrowNode;
if (a || b) {
mml = parser.create('node', 'mrow');
if (a) {
NodeUtil.appendChildren(
mml, [new TexParser('\\scriptstyle\\llap{' + a + '}', parser.stack.env, parser.configuration).mml()]);
}
arrowNode.texClass = TEXCLASS.ORD;
NodeUtil.appendChildren(mml, [arrowNode]);
if (b) {
NodeUtil.appendChildren(mml, [new TexParser('\\scriptstyle\\rlap{' + b + '}',
parser.stack.env, parser.configuration).mml()]);
}
}
}
}
if (mml) {
parser.Push(mml);
}
AmsCdMethods.cell(parser, name);
};
/**
* Converts a cell in the diagram.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
*/
AmsCdMethods.cell = function(parser: TexParser, name: string) {
let top = parser.stack.Top() as ArrayItem;
if ((top.table || []).length % 2 === 0 && (top.row || []).length === 0) {
//
// Add a strut to the first cell in even rows to get
// better spacing of arrow rows.
//
parser.Push(parser.create('node', 'mpadded', [], {height: '8.5pt', depth: '2pt'}));
}
parser.Push(parser.itemFactory.create('cell').setProperties({isEntry: true, name: name}));
};
/**
* Sets minimal width for arrows.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
*/
AmsCdMethods.minCDarrowwidth = function(parser: TexParser, name: string) {
parser.stack.env.CD_minw = parser.GetDimen(name);
};
/**
* Sets minimal height for arrows.
* @param {TexParser} parser The calling parser.
* @param {string} name The macro name.
*/
AmsCdMethods.minCDarrowheight = function(parser: TexParser, name: string) {
parser.stack.env.CD_minh = parser.GetDimen(name);
};
export default AmsCdMethods;