232 lines
5.3 KiB
JavaScript
Executable File
232 lines
5.3 KiB
JavaScript
Executable File
/**
|
||
* @import {Options} from 'micromark-extension-math'
|
||
* @import {Construct, Previous, Resolver, State, Token, TokenizeContext, Tokenizer} from 'micromark-util-types'
|
||
*/
|
||
|
||
// To do: next major: clean spaces in HTML compiler.
|
||
// This has to be coordinated together with `mdast-util-math`.
|
||
|
||
import { markdownLineEnding } from 'micromark-util-character';
|
||
/**
|
||
* @param {Options | null | undefined} [options={}]
|
||
* Configuration (default: `{}`).
|
||
* @returns {Construct}
|
||
* Construct.
|
||
*/
|
||
export function mathText(options) {
|
||
const options_ = options || {};
|
||
let single = options_.singleDollarTextMath;
|
||
if (single === null || single === undefined) {
|
||
single = true;
|
||
}
|
||
return {
|
||
tokenize: tokenizeMathText,
|
||
resolve: resolveMathText,
|
||
previous,
|
||
name: 'mathText'
|
||
};
|
||
|
||
/**
|
||
* @this {TokenizeContext}
|
||
* @type {Tokenizer}
|
||
*/
|
||
function tokenizeMathText(effects, ok, nok) {
|
||
const self = this;
|
||
let sizeOpen = 0;
|
||
/** @type {number} */
|
||
let size;
|
||
/** @type {Token} */
|
||
let token;
|
||
return start;
|
||
|
||
/**
|
||
* Start of math (text).
|
||
*
|
||
* ```markdown
|
||
* > | $a$
|
||
* ^
|
||
* > | \$a$
|
||
* ^
|
||
* ```
|
||
*
|
||
* @type {State}
|
||
*/
|
||
function start(code) {
|
||
effects.enter('mathText');
|
||
effects.enter('mathTextSequence');
|
||
return sequenceOpen(code);
|
||
}
|
||
|
||
/**
|
||
* In opening sequence.
|
||
*
|
||
* ```markdown
|
||
* > | $a$
|
||
* ^
|
||
* ```
|
||
*
|
||
* @type {State}
|
||
*/
|
||
|
||
function sequenceOpen(code) {
|
||
if (code === 36) {
|
||
effects.consume(code);
|
||
sizeOpen++;
|
||
return sequenceOpen;
|
||
}
|
||
|
||
// Not enough markers in the sequence.
|
||
if (sizeOpen < 2 && !single) {
|
||
return nok(code);
|
||
}
|
||
effects.exit('mathTextSequence');
|
||
return between(code);
|
||
}
|
||
|
||
/**
|
||
* Between something and something else.
|
||
*
|
||
* ```markdown
|
||
* > | $a$
|
||
* ^^
|
||
* ```
|
||
*
|
||
* @type {State}
|
||
*/
|
||
function between(code) {
|
||
if (code === null) {
|
||
return nok(code);
|
||
}
|
||
if (code === 36) {
|
||
token = effects.enter('mathTextSequence');
|
||
size = 0;
|
||
return sequenceClose(code);
|
||
}
|
||
|
||
// Tabs don’t work, and virtual spaces don’t make sense.
|
||
if (code === 32) {
|
||
effects.enter('space');
|
||
effects.consume(code);
|
||
effects.exit('space');
|
||
return between;
|
||
}
|
||
if (markdownLineEnding(code)) {
|
||
effects.enter("lineEnding");
|
||
effects.consume(code);
|
||
effects.exit("lineEnding");
|
||
return between;
|
||
}
|
||
|
||
// Data.
|
||
effects.enter('mathTextData');
|
||
return data(code);
|
||
}
|
||
|
||
/**
|
||
* In data.
|
||
*
|
||
* ```markdown
|
||
* > | $a$
|
||
* ^
|
||
* ```
|
||
*
|
||
* @type {State}
|
||
*/
|
||
function data(code) {
|
||
if (code === null || code === 32 || code === 36 || markdownLineEnding(code)) {
|
||
effects.exit('mathTextData');
|
||
return between(code);
|
||
}
|
||
effects.consume(code);
|
||
return data;
|
||
}
|
||
|
||
/**
|
||
* In closing sequence.
|
||
*
|
||
* ```markdown
|
||
* > | `a`
|
||
* ^
|
||
* ```
|
||
*
|
||
* @type {State}
|
||
*/
|
||
|
||
function sequenceClose(code) {
|
||
// More.
|
||
if (code === 36) {
|
||
effects.consume(code);
|
||
size++;
|
||
return sequenceClose;
|
||
}
|
||
|
||
// Done!
|
||
if (size === sizeOpen) {
|
||
effects.exit('mathTextSequence');
|
||
effects.exit('mathText');
|
||
return ok(code);
|
||
}
|
||
|
||
// More or less accents: mark as data.
|
||
token.type = 'mathTextData';
|
||
return data(code);
|
||
}
|
||
}
|
||
}
|
||
|
||
/** @type {Resolver} */
|
||
function resolveMathText(events) {
|
||
let tailExitIndex = events.length - 4;
|
||
let headEnterIndex = 3;
|
||
/** @type {number} */
|
||
let index;
|
||
/** @type {number | undefined} */
|
||
let enter;
|
||
|
||
// If we start and end with an EOL or a space.
|
||
if ((events[headEnterIndex][1].type === "lineEnding" || events[headEnterIndex][1].type === 'space') && (events[tailExitIndex][1].type === "lineEnding" || events[tailExitIndex][1].type === 'space')) {
|
||
index = headEnterIndex;
|
||
|
||
// And we have data.
|
||
while (++index < tailExitIndex) {
|
||
if (events[index][1].type === 'mathTextData') {
|
||
// Then we have padding.
|
||
events[tailExitIndex][1].type = 'mathTextPadding';
|
||
events[headEnterIndex][1].type = 'mathTextPadding';
|
||
headEnterIndex += 2;
|
||
tailExitIndex -= 2;
|
||
break;
|
||
}
|
||
}
|
||
}
|
||
|
||
// Merge adjacent spaces and data.
|
||
index = headEnterIndex - 1;
|
||
tailExitIndex++;
|
||
while (++index <= tailExitIndex) {
|
||
if (enter === undefined) {
|
||
if (index !== tailExitIndex && events[index][1].type !== "lineEnding") {
|
||
enter = index;
|
||
}
|
||
} else if (index === tailExitIndex || events[index][1].type === "lineEnding") {
|
||
events[enter][1].type = 'mathTextData';
|
||
if (index !== enter + 2) {
|
||
events[enter][1].end = events[index - 1][1].end;
|
||
events.splice(enter + 2, index - enter - 2);
|
||
tailExitIndex -= index - enter - 2;
|
||
index = enter + 2;
|
||
}
|
||
enter = undefined;
|
||
}
|
||
}
|
||
return events;
|
||
}
|
||
|
||
/**
|
||
* @this {TokenizeContext}
|
||
* @type {Previous}
|
||
*/
|
||
function previous(code) {
|
||
// If there is a previous code, there will always be a tail.
|
||
return code !== 36 || this.events[this.events.length - 1][1].type === "characterEscape";
|
||
} |